mirror of
https://github.com/angular/angular
synced 2026-05-24 09:28:37 +00:00
The view data store operation is not an instruction and shouldn't be located in the instructions folder. PR Close #61425
320 lines
11 KiB
TypeScript
320 lines
11 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright Google LLC All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
* found in the LICENSE file at https://angular.dev/license
|
|
*/
|
|
|
|
import {PipeTransform} from '../change_detection/pipe_transform';
|
|
import {setInjectImplementation} from '../di/inject_switch';
|
|
import {formatRuntimeError, RuntimeError, RuntimeErrorCode} from '../errors';
|
|
import {Type} from '../interface/type';
|
|
|
|
import {InjectorProfilerContext, setInjectorProfilerContext} from './debug/injector_profiler';
|
|
import {getFactoryDef} from './definition_factory';
|
|
import {NodeInjector, setIncludeViewProviders} from './di';
|
|
import {ɵɵdirectiveInject} from './instructions/all';
|
|
import {isHostComponentStandalone} from './instructions/element_validation';
|
|
import {PipeDef, PipeDefList} from './interfaces/definition';
|
|
import {TTextNode} from './interfaces/node';
|
|
import {CONTEXT, DECLARATION_COMPONENT_VIEW, HEADER_OFFSET, LView, TVIEW} from './interfaces/view';
|
|
import {
|
|
pureFunction1Internal,
|
|
pureFunction2Internal,
|
|
pureFunction3Internal,
|
|
pureFunction4Internal,
|
|
pureFunctionVInternal,
|
|
} from './pure_function';
|
|
import {getBindingRoot, getCurrentTNode, getLView, getTView} from './state';
|
|
import {load, store} from './util/view_utils';
|
|
|
|
/**
|
|
* Create a pipe.
|
|
*
|
|
* @param index Pipe index where the pipe will be stored.
|
|
* @param pipeName The name of the pipe
|
|
* @returns T the instance of the pipe.
|
|
*
|
|
* @codeGenApi
|
|
*/
|
|
export function ɵɵpipe(index: number, pipeName: string): any {
|
|
const tView = getTView();
|
|
let pipeDef: PipeDef<any>;
|
|
const adjustedIndex = index + HEADER_OFFSET;
|
|
|
|
if (tView.firstCreatePass) {
|
|
// The `getPipeDef` throws if a pipe with a given name is not found
|
|
// (so we use non-null assertion below).
|
|
pipeDef = getPipeDef(pipeName, tView.pipeRegistry)!;
|
|
tView.data[adjustedIndex] = pipeDef;
|
|
if (pipeDef.onDestroy) {
|
|
(tView.destroyHooks ??= []).push(adjustedIndex, pipeDef.onDestroy);
|
|
}
|
|
} else {
|
|
pipeDef = tView.data[adjustedIndex] as PipeDef<any>;
|
|
}
|
|
|
|
const pipeFactory = pipeDef.factory || (pipeDef.factory = getFactoryDef(pipeDef.type, true));
|
|
|
|
let previousInjectorProfilerContext: InjectorProfilerContext;
|
|
if (ngDevMode) {
|
|
previousInjectorProfilerContext = setInjectorProfilerContext({
|
|
injector: new NodeInjector(getCurrentTNode() as TTextNode, getLView()),
|
|
token: pipeDef.type,
|
|
});
|
|
}
|
|
const previousInjectImplementation = setInjectImplementation(ɵɵdirectiveInject);
|
|
try {
|
|
// DI for pipes is supposed to behave like directives when placed on a component
|
|
// host node, which means that we have to disable access to `viewProviders`.
|
|
const previousIncludeViewProviders = setIncludeViewProviders(false);
|
|
const pipeInstance = pipeFactory();
|
|
setIncludeViewProviders(previousIncludeViewProviders);
|
|
store(tView, getLView(), adjustedIndex, pipeInstance);
|
|
return pipeInstance;
|
|
} finally {
|
|
// we have to restore the injector implementation in finally, just in case the creation of the
|
|
// pipe throws an error.
|
|
setInjectImplementation(previousInjectImplementation);
|
|
ngDevMode && setInjectorProfilerContext(previousInjectorProfilerContext!);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Searches the pipe registry for a pipe with the given name. If one is found,
|
|
* returns the pipe. Otherwise, an error is thrown because the pipe cannot be resolved.
|
|
*
|
|
* @param name Name of pipe to resolve
|
|
* @param registry Full list of available pipes
|
|
* @returns Matching PipeDef
|
|
*/
|
|
function getPipeDef(name: string, registry: PipeDefList | null): PipeDef<any> | undefined {
|
|
if (registry) {
|
|
if (ngDevMode) {
|
|
const pipes = registry.filter((pipe) => pipe.name === name);
|
|
// TODO: Throw an error in the next major
|
|
if (pipes.length > 1) {
|
|
console.warn(
|
|
formatRuntimeError(
|
|
RuntimeErrorCode.MULTIPLE_MATCHING_PIPES,
|
|
getMultipleMatchingPipesMessage(name),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
for (let i = registry.length - 1; i >= 0; i--) {
|
|
const pipeDef = registry[i];
|
|
if (name === pipeDef.name) {
|
|
return pipeDef;
|
|
}
|
|
}
|
|
}
|
|
if (ngDevMode) {
|
|
throw new RuntimeError(RuntimeErrorCode.PIPE_NOT_FOUND, getPipeNotFoundErrorMessage(name));
|
|
}
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Generates a helpful error message for the user when multiple pipes match the name.
|
|
*
|
|
* @param name Name of the pipe
|
|
* @returns The error message
|
|
*/
|
|
function getMultipleMatchingPipesMessage(name: string) {
|
|
const lView = getLView();
|
|
const declarationLView = lView[DECLARATION_COMPONENT_VIEW] as LView<Type<unknown>>;
|
|
const context = declarationLView[CONTEXT];
|
|
const hostIsStandalone = isHostComponentStandalone(lView);
|
|
const componentInfoMessage = context ? ` in the '${context.constructor.name}' component` : '';
|
|
const verifyMessage = `check ${
|
|
hostIsStandalone ? "'@Component.imports' of this component" : 'the imports of this module'
|
|
}`;
|
|
const errorMessage = `Multiple pipes match the name \`${name}\`${componentInfoMessage}. ${verifyMessage}`;
|
|
return errorMessage;
|
|
}
|
|
|
|
/**
|
|
* Generates a helpful error message for the user when a pipe is not found.
|
|
*
|
|
* @param name Name of the missing pipe
|
|
* @returns The error message
|
|
*/
|
|
function getPipeNotFoundErrorMessage(name: string) {
|
|
const lView = getLView();
|
|
const declarationLView = lView[DECLARATION_COMPONENT_VIEW] as LView<Type<unknown>>;
|
|
const context = declarationLView[CONTEXT];
|
|
const hostIsStandalone = isHostComponentStandalone(lView);
|
|
const componentInfoMessage = context ? ` in the '${context.constructor.name}' component` : '';
|
|
const verifyMessage = `Verify that it is ${
|
|
hostIsStandalone
|
|
? "included in the '@Component.imports' of this component"
|
|
: 'declared or imported in this module'
|
|
}`;
|
|
const errorMessage = `The pipe '${name}' could not be found${componentInfoMessage}. ${verifyMessage}`;
|
|
return errorMessage;
|
|
}
|
|
|
|
/**
|
|
* Invokes a pipe with 1 arguments.
|
|
*
|
|
* This instruction acts as a guard to {@link PipeTransform#transform} invoking
|
|
* the pipe only when an input to the pipe changes.
|
|
*
|
|
* @param index Pipe index where the pipe was stored on creation.
|
|
* @param offset the binding offset
|
|
* @param v1 1st argument to {@link PipeTransform#transform}.
|
|
*
|
|
* @codeGenApi
|
|
*/
|
|
export function ɵɵpipeBind1(index: number, offset: number, v1: any): any {
|
|
const adjustedIndex = index + HEADER_OFFSET;
|
|
const lView = getLView();
|
|
const pipeInstance = load<PipeTransform>(lView, adjustedIndex);
|
|
return isPure(lView, adjustedIndex)
|
|
? pureFunction1Internal(
|
|
lView,
|
|
getBindingRoot(),
|
|
offset,
|
|
pipeInstance.transform,
|
|
v1,
|
|
pipeInstance,
|
|
)
|
|
: pipeInstance.transform(v1);
|
|
}
|
|
|
|
/**
|
|
* Invokes a pipe with 2 arguments.
|
|
*
|
|
* This instruction acts as a guard to {@link PipeTransform#transform} invoking
|
|
* the pipe only when an input to the pipe changes.
|
|
*
|
|
* @param index Pipe index where the pipe was stored on creation.
|
|
* @param slotOffset the offset in the reserved slot space
|
|
* @param v1 1st argument to {@link PipeTransform#transform}.
|
|
* @param v2 2nd argument to {@link PipeTransform#transform}.
|
|
*
|
|
* @codeGenApi
|
|
*/
|
|
export function ɵɵpipeBind2(index: number, slotOffset: number, v1: any, v2: any): any {
|
|
const adjustedIndex = index + HEADER_OFFSET;
|
|
const lView = getLView();
|
|
const pipeInstance = load<PipeTransform>(lView, adjustedIndex);
|
|
return isPure(lView, adjustedIndex)
|
|
? pureFunction2Internal(
|
|
lView,
|
|
getBindingRoot(),
|
|
slotOffset,
|
|
pipeInstance.transform,
|
|
v1,
|
|
v2,
|
|
pipeInstance,
|
|
)
|
|
: pipeInstance.transform(v1, v2);
|
|
}
|
|
|
|
/**
|
|
* Invokes a pipe with 3 arguments.
|
|
*
|
|
* This instruction acts as a guard to {@link PipeTransform#transform} invoking
|
|
* the pipe only when an input to the pipe changes.
|
|
*
|
|
* @param index Pipe index where the pipe was stored on creation.
|
|
* @param slotOffset the offset in the reserved slot space
|
|
* @param v1 1st argument to {@link PipeTransform#transform}.
|
|
* @param v2 2nd argument to {@link PipeTransform#transform}.
|
|
* @param v3 4rd argument to {@link PipeTransform#transform}.
|
|
*
|
|
* @codeGenApi
|
|
*/
|
|
export function ɵɵpipeBind3(index: number, slotOffset: number, v1: any, v2: any, v3: any): any {
|
|
const adjustedIndex = index + HEADER_OFFSET;
|
|
const lView = getLView();
|
|
const pipeInstance = load<PipeTransform>(lView, adjustedIndex);
|
|
return isPure(lView, adjustedIndex)
|
|
? pureFunction3Internal(
|
|
lView,
|
|
getBindingRoot(),
|
|
slotOffset,
|
|
pipeInstance.transform,
|
|
v1,
|
|
v2,
|
|
v3,
|
|
pipeInstance,
|
|
)
|
|
: pipeInstance.transform(v1, v2, v3);
|
|
}
|
|
|
|
/**
|
|
* Invokes a pipe with 4 arguments.
|
|
*
|
|
* This instruction acts as a guard to {@link PipeTransform#transform} invoking
|
|
* the pipe only when an input to the pipe changes.
|
|
*
|
|
* @param index Pipe index where the pipe was stored on creation.
|
|
* @param slotOffset the offset in the reserved slot space
|
|
* @param v1 1st argument to {@link PipeTransform#transform}.
|
|
* @param v2 2nd argument to {@link PipeTransform#transform}.
|
|
* @param v3 3rd argument to {@link PipeTransform#transform}.
|
|
* @param v4 4th argument to {@link PipeTransform#transform}.
|
|
*
|
|
* @codeGenApi
|
|
*/
|
|
export function ɵɵpipeBind4(
|
|
index: number,
|
|
slotOffset: number,
|
|
v1: any,
|
|
v2: any,
|
|
v3: any,
|
|
v4: any,
|
|
): any {
|
|
const adjustedIndex = index + HEADER_OFFSET;
|
|
const lView = getLView();
|
|
const pipeInstance = load<PipeTransform>(lView, adjustedIndex);
|
|
return isPure(lView, adjustedIndex)
|
|
? pureFunction4Internal(
|
|
lView,
|
|
getBindingRoot(),
|
|
slotOffset,
|
|
pipeInstance.transform,
|
|
v1,
|
|
v2,
|
|
v3,
|
|
v4,
|
|
pipeInstance,
|
|
)
|
|
: pipeInstance.transform(v1, v2, v3, v4);
|
|
}
|
|
|
|
/**
|
|
* Invokes a pipe with variable number of arguments.
|
|
*
|
|
* This instruction acts as a guard to {@link PipeTransform#transform} invoking
|
|
* the pipe only when an input to the pipe changes.
|
|
*
|
|
* @param index Pipe index where the pipe was stored on creation.
|
|
* @param slotOffset the offset in the reserved slot space
|
|
* @param values Array of arguments to pass to {@link PipeTransform#transform} method.
|
|
*
|
|
* @codeGenApi
|
|
*/
|
|
export function ɵɵpipeBindV(index: number, slotOffset: number, values: [any, ...any[]]): any {
|
|
const adjustedIndex = index + HEADER_OFFSET;
|
|
const lView = getLView();
|
|
const pipeInstance = load<PipeTransform>(lView, adjustedIndex);
|
|
return isPure(lView, adjustedIndex)
|
|
? pureFunctionVInternal(
|
|
lView,
|
|
getBindingRoot(),
|
|
slotOffset,
|
|
pipeInstance.transform,
|
|
values,
|
|
pipeInstance,
|
|
)
|
|
: pipeInstance.transform.apply(pipeInstance, values);
|
|
}
|
|
|
|
function isPure(lView: LView, index: number): boolean {
|
|
return (<PipeDef<any>>lView[TVIEW].data[index]).pure;
|
|
}
|