2017-08-16 09:00:03 -07:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 12:08:49 -07:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2017-08-16 09:00:03 -07:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Attention:
|
|
|
|
// This file duplicates types and values from @angular/core
|
|
|
|
// so that we are able to make @angular/compiler independent of @angular/core.
|
|
|
|
// This is important to prevent a build cycle, as @angular/core needs to
|
|
|
|
// be compiled with the compiler.
|
|
|
|
|
2018-04-20 08:52:07 -07:00
|
|
|
import {CssSelector} from './selector';
|
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export interface Inject {
|
|
|
|
token: any;
|
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
export const createInject = makeMetadataFactory<Inject>('Inject', (token: any) => ({token}));
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
export const createInjectionToken = makeMetadataFactory<object>(
|
2019-10-15 12:41:30 -07:00
|
|
|
'InjectionToken', (desc: string) => ({_desc: desc, ɵprov: undefined}));
|
2017-08-16 09:00:03 -07:00
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export interface Attribute {
|
2020-07-18 18:35:28 +05:30
|
|
|
attributeName: string;
|
2020-04-08 10:14:18 -07:00
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
export const createAttribute =
|
2020-07-18 18:35:28 +05:30
|
|
|
makeMetadataFactory<Attribute>('Attribute', (attributeName: string) => ({attributeName}));
|
2017-08-16 09:00:03 -07:00
|
|
|
|
|
|
|
export interface Query {
|
|
|
|
descendants: boolean;
|
|
|
|
first: boolean;
|
|
|
|
read: any;
|
|
|
|
isViewQuery: boolean;
|
|
|
|
selector: any;
|
2019-10-03 21:54:07 +02:00
|
|
|
static?: boolean;
|
2017-08-16 09:00:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
export const createContentChildren = makeMetadataFactory<Query>(
|
|
|
|
'ContentChildren',
|
|
|
|
(selector?: any, data: any = {}) =>
|
|
|
|
({selector, first: false, isViewQuery: false, descendants: false, ...data}));
|
|
|
|
export const createContentChild = makeMetadataFactory<Query>(
|
2020-04-08 10:14:18 -07:00
|
|
|
'ContentChild',
|
|
|
|
(selector?: any, data: any = {}) =>
|
|
|
|
({selector, first: true, isViewQuery: false, descendants: true, ...data}));
|
2017-08-16 09:00:03 -07:00
|
|
|
export const createViewChildren = makeMetadataFactory<Query>(
|
2020-04-08 10:14:18 -07:00
|
|
|
'ViewChildren',
|
|
|
|
(selector?: any, data: any = {}) =>
|
|
|
|
({selector, first: false, isViewQuery: true, descendants: true, ...data}));
|
2017-08-16 09:00:03 -07:00
|
|
|
export const createViewChild = makeMetadataFactory<Query>(
|
2020-04-08 10:14:18 -07:00
|
|
|
'ViewChild',
|
|
|
|
(selector: any, data: any) =>
|
|
|
|
({selector, first: true, isViewQuery: true, descendants: true, ...data}));
|
2017-08-16 09:00:03 -07:00
|
|
|
|
|
|
|
export interface Directive {
|
|
|
|
selector?: string;
|
|
|
|
inputs?: string[];
|
|
|
|
outputs?: string[];
|
|
|
|
host?: {[key: string]: string};
|
|
|
|
providers?: Provider[];
|
|
|
|
exportAs?: string;
|
|
|
|
queries?: {[key: string]: any};
|
2017-11-29 16:29:05 -08:00
|
|
|
guards?: {[key: string]: any};
|
2017-08-16 09:00:03 -07:00
|
|
|
}
|
|
|
|
export const createDirective =
|
|
|
|
makeMetadataFactory<Directive>('Directive', (dir: Directive = {}) => dir);
|
|
|
|
|
|
|
|
export interface Component extends Directive {
|
|
|
|
changeDetection?: ChangeDetectionStrategy;
|
|
|
|
viewProviders?: Provider[];
|
|
|
|
moduleId?: string;
|
|
|
|
templateUrl?: string;
|
|
|
|
template?: string;
|
|
|
|
styleUrls?: string[];
|
|
|
|
styles?: string[];
|
|
|
|
animations?: any[];
|
|
|
|
encapsulation?: ViewEncapsulation;
|
|
|
|
interpolation?: [string, string];
|
|
|
|
entryComponents?: Array<Type|any[]>;
|
|
|
|
preserveWhitespaces?: boolean;
|
|
|
|
}
|
|
|
|
export enum ViewEncapsulation {
|
|
|
|
Emulated = 0,
|
2020-10-08 16:59:29 +02:00
|
|
|
// Historically the 1 value was for `Native` encapsulation which has been removed as of v11.
|
2018-06-28 12:42:04 -07:00
|
|
|
None = 2,
|
|
|
|
ShadowDom = 3
|
2017-08-16 09:00:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
export enum ChangeDetectionStrategy {
|
|
|
|
OnPush = 0,
|
|
|
|
Default = 1
|
|
|
|
}
|
|
|
|
|
|
|
|
export const createComponent = makeMetadataFactory<Component>(
|
|
|
|
'Component', (c: Component = {}) => ({changeDetection: ChangeDetectionStrategy.Default, ...c}));
|
|
|
|
|
|
|
|
export interface Pipe {
|
|
|
|
name: string;
|
|
|
|
pure?: boolean;
|
|
|
|
}
|
|
|
|
export const createPipe = makeMetadataFactory<Pipe>('Pipe', (p: Pipe) => ({pure: true, ...p}));
|
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export interface Input {
|
|
|
|
bindingPropertyName?: string;
|
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
export const createInput =
|
|
|
|
makeMetadataFactory<Input>('Input', (bindingPropertyName?: string) => ({bindingPropertyName}));
|
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export interface Output {
|
|
|
|
bindingPropertyName?: string;
|
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
export const createOutput = makeMetadataFactory<Output>(
|
|
|
|
'Output', (bindingPropertyName?: string) => ({bindingPropertyName}));
|
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export interface HostBinding {
|
|
|
|
hostPropertyName?: string;
|
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
export const createHostBinding = makeMetadataFactory<HostBinding>(
|
|
|
|
'HostBinding', (hostPropertyName?: string) => ({hostPropertyName}));
|
|
|
|
|
|
|
|
export interface HostListener {
|
|
|
|
eventName?: string;
|
|
|
|
args?: string[];
|
|
|
|
}
|
|
|
|
export const createHostListener = makeMetadataFactory<HostListener>(
|
|
|
|
'HostListener', (eventName?: string, args?: string[]) => ({eventName, args}));
|
|
|
|
|
|
|
|
export interface NgModule {
|
|
|
|
providers?: Provider[];
|
|
|
|
declarations?: Array<Type|any[]>;
|
|
|
|
imports?: Array<Type|ModuleWithProviders|any[]>;
|
|
|
|
exports?: Array<Type|any[]>;
|
|
|
|
entryComponents?: Array<Type|any[]>;
|
|
|
|
bootstrap?: Array<Type|any[]>;
|
|
|
|
schemas?: Array<SchemaMetadata|any[]>;
|
|
|
|
id?: string;
|
|
|
|
}
|
|
|
|
export const createNgModule =
|
|
|
|
makeMetadataFactory<NgModule>('NgModule', (ngModule: NgModule) => ngModule);
|
|
|
|
|
|
|
|
export interface ModuleWithProviders {
|
|
|
|
ngModule: Type;
|
|
|
|
providers?: Provider[];
|
|
|
|
}
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
export interface Injectable {
|
2018-03-07 15:10:38 -08:00
|
|
|
providedIn?: Type|'root'|any;
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
useClass?: Type|any;
|
|
|
|
useExisting?: Type|any;
|
|
|
|
useValue?: any;
|
|
|
|
useFactory?: Type|any;
|
|
|
|
deps?: Array<Type|any[]>;
|
|
|
|
}
|
|
|
|
export const createInjectable =
|
|
|
|
makeMetadataFactory('Injectable', (injectable: Injectable = {}) => injectable);
|
2020-04-08 10:14:18 -07:00
|
|
|
export interface SchemaMetadata {
|
|
|
|
name: string;
|
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
|
|
|
|
export const CUSTOM_ELEMENTS_SCHEMA: SchemaMetadata = {
|
|
|
|
name: 'custom-elements'
|
|
|
|
};
|
|
|
|
|
|
|
|
export const NO_ERRORS_SCHEMA: SchemaMetadata = {
|
|
|
|
name: 'no-errors-schema'
|
|
|
|
};
|
|
|
|
|
|
|
|
export const createOptional = makeMetadataFactory('Optional');
|
|
|
|
export const createSelf = makeMetadataFactory('Self');
|
|
|
|
export const createSkipSelf = makeMetadataFactory('SkipSelf');
|
|
|
|
export const createHost = makeMetadataFactory('Host');
|
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export interface Type extends Function {
|
|
|
|
new(...args: any[]): any;
|
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
export const Type = Function;
|
|
|
|
|
|
|
|
export enum SecurityContext {
|
|
|
|
NONE = 0,
|
|
|
|
HTML = 1,
|
|
|
|
STYLE = 2,
|
|
|
|
SCRIPT = 3,
|
|
|
|
URL = 4,
|
|
|
|
RESOURCE_URL = 5,
|
|
|
|
}
|
|
|
|
|
|
|
|
export type Provider = any;
|
|
|
|
|
|
|
|
export const enum NodeFlags {
|
|
|
|
None = 0,
|
|
|
|
TypeElement = 1 << 0,
|
|
|
|
TypeText = 1 << 1,
|
|
|
|
ProjectedTemplate = 1 << 2,
|
|
|
|
CatRenderNode = TypeElement | TypeText,
|
|
|
|
TypeNgContent = 1 << 3,
|
|
|
|
TypePipe = 1 << 4,
|
|
|
|
TypePureArray = 1 << 5,
|
|
|
|
TypePureObject = 1 << 6,
|
|
|
|
TypePurePipe = 1 << 7,
|
|
|
|
CatPureExpression = TypePureArray | TypePureObject | TypePurePipe,
|
|
|
|
TypeValueProvider = 1 << 8,
|
|
|
|
TypeClassProvider = 1 << 9,
|
|
|
|
TypeFactoryProvider = 1 << 10,
|
|
|
|
TypeUseExistingProvider = 1 << 11,
|
|
|
|
LazyProvider = 1 << 12,
|
|
|
|
PrivateProvider = 1 << 13,
|
|
|
|
TypeDirective = 1 << 14,
|
|
|
|
Component = 1 << 15,
|
|
|
|
CatProviderNoDirective =
|
|
|
|
TypeValueProvider | TypeClassProvider | TypeFactoryProvider | TypeUseExistingProvider,
|
|
|
|
CatProvider = CatProviderNoDirective | TypeDirective,
|
|
|
|
OnInit = 1 << 16,
|
|
|
|
OnDestroy = 1 << 17,
|
|
|
|
DoCheck = 1 << 18,
|
|
|
|
OnChanges = 1 << 19,
|
|
|
|
AfterContentInit = 1 << 20,
|
|
|
|
AfterContentChecked = 1 << 21,
|
|
|
|
AfterViewInit = 1 << 22,
|
|
|
|
AfterViewChecked = 1 << 23,
|
|
|
|
EmbeddedViews = 1 << 24,
|
|
|
|
ComponentView = 1 << 25,
|
|
|
|
TypeContentQuery = 1 << 26,
|
|
|
|
TypeViewQuery = 1 << 27,
|
|
|
|
StaticQuery = 1 << 28,
|
|
|
|
DynamicQuery = 1 << 29,
|
2018-02-02 09:31:58 -08:00
|
|
|
TypeModuleProvider = 1 << 30,
|
2017-08-16 09:00:03 -07:00
|
|
|
CatQuery = TypeContentQuery | TypeViewQuery,
|
|
|
|
|
|
|
|
// mutually exclusive values...
|
|
|
|
Types = CatRenderNode | TypeNgContent | TypePipe | CatPureExpression | CatProvider | CatQuery
|
|
|
|
}
|
|
|
|
|
|
|
|
export const enum DepFlags {
|
|
|
|
None = 0,
|
|
|
|
SkipSelf = 1 << 0,
|
|
|
|
Optional = 1 << 1,
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
Self = 1 << 2,
|
|
|
|
Value = 1 << 3,
|
|
|
|
}
|
|
|
|
|
2018-04-12 12:30:21 -07:00
|
|
|
/**
|
|
|
|
* Injection flags for DI.
|
|
|
|
*/
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
export const enum InjectFlags {
|
|
|
|
Default = 0,
|
|
|
|
|
2018-04-12 12:30:21 -07:00
|
|
|
/**
|
|
|
|
* Specifies that an injector should retrieve a dependency from any injector until reaching the
|
|
|
|
* host element of the current component. (Only used with Element Injector)
|
|
|
|
*/
|
|
|
|
Host = 1 << 0,
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
/** Don't descend into ancestors of the node requesting injection. */
|
|
|
|
Self = 1 << 1,
|
2018-04-12 12:30:21 -07:00
|
|
|
/** Skip the node that is requesting injection. */
|
|
|
|
SkipSelf = 1 << 2,
|
|
|
|
/** Inject `defaultValue` instead if token not found. */
|
|
|
|
Optional = 1 << 3,
|
2017-08-16 09:00:03 -07:00
|
|
|
}
|
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export const enum ArgumentType {
|
|
|
|
Inline = 0,
|
|
|
|
Dynamic = 1
|
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
|
|
|
|
export const enum BindingFlags {
|
|
|
|
TypeElementAttribute = 1 << 0,
|
|
|
|
TypeElementClass = 1 << 1,
|
|
|
|
TypeElementStyle = 1 << 2,
|
|
|
|
TypeProperty = 1 << 3,
|
|
|
|
SyntheticProperty = 1 << 4,
|
|
|
|
SyntheticHostProperty = 1 << 5,
|
|
|
|
CatSyntheticProperty = SyntheticProperty | SyntheticHostProperty,
|
|
|
|
|
|
|
|
// mutually exclusive values...
|
|
|
|
Types = TypeElementAttribute | TypeElementClass | TypeElementStyle | TypeProperty
|
|
|
|
}
|
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export const enum QueryBindingType {
|
|
|
|
First = 0,
|
|
|
|
All = 1
|
|
|
|
}
|
2017-08-16 09:00:03 -07:00
|
|
|
|
|
|
|
export const enum QueryValueType {
|
|
|
|
ElementRef = 0,
|
|
|
|
RenderElement = 1,
|
|
|
|
TemplateRef = 2,
|
|
|
|
ViewContainerRef = 3,
|
|
|
|
Provider = 4
|
|
|
|
}
|
|
|
|
|
|
|
|
export const enum ViewFlags {
|
|
|
|
None = 0,
|
|
|
|
OnPush = 1 << 1,
|
|
|
|
}
|
|
|
|
|
|
|
|
export enum MissingTranslationStrategy {
|
|
|
|
Error = 0,
|
|
|
|
Warning = 1,
|
|
|
|
Ignore = 2,
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface MetadataFactory<T> {
|
|
|
|
(...args: any[]): T;
|
|
|
|
isTypeOf(obj: any): obj is T;
|
|
|
|
ngMetadataName: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
function makeMetadataFactory<T>(name: string, props?: (...args: any[]) => T): MetadataFactory<T> {
|
2019-02-08 14:01:51 -08:00
|
|
|
// This must be declared as a function, not a fat arrow, so that ES2015 devmode produces code
|
|
|
|
// that works with the static_reflector.ts in the ViewEngine compiler.
|
|
|
|
// In particular, `_registerDecoratorOrConstructor` assumes that the value returned here can be
|
|
|
|
// new'ed.
|
|
|
|
function factory(...args: any[]) {
|
2017-08-16 09:00:03 -07:00
|
|
|
const values = props ? props(...args) : {};
|
|
|
|
return {
|
|
|
|
ngMetadataName: name,
|
|
|
|
...values,
|
|
|
|
};
|
2019-02-08 14:01:51 -08:00
|
|
|
}
|
|
|
|
(factory as any).isTypeOf = (obj: any) => obj && obj.ngMetadataName === name;
|
|
|
|
(factory as any).ngMetadataName = name;
|
|
|
|
return factory as any;
|
2017-08-16 09:00:03 -07:00
|
|
|
}
|
2017-10-20 09:46:41 -07:00
|
|
|
|
|
|
|
export interface Route {
|
|
|
|
children?: Route[];
|
|
|
|
loadChildren?: string|Type|any;
|
|
|
|
}
|
2018-04-20 08:52:07 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Flags used to generate R3-style CSS Selectors. They are pasted from
|
|
|
|
* core/src/render3/projection.ts because they cannot be referenced directly.
|
|
|
|
*/
|
|
|
|
export const enum SelectorFlags {
|
|
|
|
/** Indicates this is the beginning of a new negative selector */
|
|
|
|
NOT = 0b0001,
|
|
|
|
|
|
|
|
/** Mode for matching attributes */
|
|
|
|
ATTRIBUTE = 0b0010,
|
|
|
|
|
|
|
|
/** Mode for matching tag names */
|
|
|
|
ELEMENT = 0b0100,
|
|
|
|
|
|
|
|
/** Mode for matching class names */
|
|
|
|
CLASS = 0b1000,
|
|
|
|
}
|
|
|
|
|
|
|
|
// These are a copy the CSS types from core/src/render3/interfaces/projection.ts
|
|
|
|
// They are duplicated here as they cannot be directly referenced from core.
|
2020-04-08 10:14:18 -07:00
|
|
|
export type R3CssSelector = (string|SelectorFlags)[];
|
2018-04-20 08:52:07 -07:00
|
|
|
export type R3CssSelectorList = R3CssSelector[];
|
|
|
|
|
|
|
|
function parserSelectorToSimpleSelector(selector: CssSelector): R3CssSelector {
|
|
|
|
const classes = selector.classNames && selector.classNames.length ?
|
|
|
|
[SelectorFlags.CLASS, ...selector.classNames] :
|
|
|
|
[];
|
|
|
|
const elementName = selector.element && selector.element !== '*' ? selector.element : '';
|
|
|
|
return [elementName, ...selector.attrs, ...classes];
|
|
|
|
}
|
|
|
|
|
|
|
|
function parserSelectorToNegativeSelector(selector: CssSelector): R3CssSelector {
|
|
|
|
const classes = selector.classNames && selector.classNames.length ?
|
|
|
|
[SelectorFlags.CLASS, ...selector.classNames] :
|
|
|
|
[];
|
|
|
|
|
|
|
|
if (selector.element) {
|
|
|
|
return [
|
|
|
|
SelectorFlags.NOT | SelectorFlags.ELEMENT, selector.element, ...selector.attrs, ...classes
|
|
|
|
];
|
|
|
|
} else if (selector.attrs.length) {
|
|
|
|
return [SelectorFlags.NOT | SelectorFlags.ATTRIBUTE, ...selector.attrs, ...classes];
|
|
|
|
} else {
|
|
|
|
return selector.classNames && selector.classNames.length ?
|
|
|
|
[SelectorFlags.NOT | SelectorFlags.CLASS, ...selector.classNames] :
|
|
|
|
[];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function parserSelectorToR3Selector(selector: CssSelector): R3CssSelector {
|
|
|
|
const positive = parserSelectorToSimpleSelector(selector);
|
|
|
|
|
|
|
|
const negative: R3CssSelectorList = selector.notSelectors && selector.notSelectors.length ?
|
|
|
|
selector.notSelectors.map(notSelector => parserSelectorToNegativeSelector(notSelector)) :
|
|
|
|
[];
|
|
|
|
|
|
|
|
return positive.concat(...negative);
|
|
|
|
}
|
|
|
|
|
2020-04-08 10:14:18 -07:00
|
|
|
export function parseSelectorToR3Selector(selector: string|null): R3CssSelectorList {
|
2018-11-22 15:38:28 +01:00
|
|
|
return selector ? CssSelector.parse(selector).map(parserSelectorToR3Selector) : [];
|
2018-04-20 08:52:07 -07:00
|
|
|
}
|
2018-04-24 11:34:11 -07:00
|
|
|
|
|
|
|
// Pasted from render3/interfaces/definition since it cannot be referenced directly
|
|
|
|
/**
|
|
|
|
* Flags passed into template functions to determine which blocks (i.e. creation, update)
|
|
|
|
* should be executed.
|
|
|
|
*
|
|
|
|
* Typically, a template runs both the creation block and the update block on initialization and
|
|
|
|
* subsequent runs only execute the update block. However, dynamically created views require that
|
|
|
|
* the creation block be executed separately from the update block (for backwards compat).
|
|
|
|
*/
|
|
|
|
export const enum RenderFlags {
|
|
|
|
/* Whether to run the creation block (e.g. create elements and directives) */
|
|
|
|
Create = 0b01,
|
|
|
|
|
|
|
|
/* Whether to run the update block (e.g. refresh bindings) */
|
|
|
|
Update = 0b10
|
|
|
|
}
|
2018-06-19 12:45:00 -07:00
|
|
|
|
2018-08-20 15:20:12 +02:00
|
|
|
// Pasted from render3/interfaces/node.ts
|
|
|
|
/**
|
2018-12-13 15:51:47 -08:00
|
|
|
* A set of marker values to be used in the attributes arrays. These markers indicate that some
|
2018-08-20 15:20:12 +02:00
|
|
|
* items are not regular attributes and the processing should be adapted accordingly.
|
|
|
|
*/
|
|
|
|
export const enum AttributeMarker {
|
|
|
|
/**
|
|
|
|
* Marker indicates that the following 3 values in the attributes array are:
|
|
|
|
* namespaceUri, attributeName, attributeValue
|
|
|
|
* in that order.
|
|
|
|
*/
|
|
|
|
NamespaceURI = 0,
|
|
|
|
|
2018-12-13 15:51:47 -08:00
|
|
|
/**
|
|
|
|
* Signals class declaration.
|
|
|
|
*
|
|
|
|
* Each value following `Classes` designates a class name to include on the element.
|
|
|
|
* ## Example:
|
|
|
|
*
|
|
|
|
* Given:
|
|
|
|
* ```
|
|
|
|
* <div class="foo bar baz">...<d/vi>
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* the generated code is:
|
|
|
|
* ```
|
|
|
|
* var _c1 = [AttributeMarker.Classes, 'foo', 'bar', 'baz'];
|
|
|
|
* ```
|
|
|
|
*/
|
|
|
|
Classes = 1,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals style declaration.
|
|
|
|
*
|
|
|
|
* Each pair of values following `Styles` designates a style name and value to include on the
|
|
|
|
* element.
|
|
|
|
* ## Example:
|
|
|
|
*
|
|
|
|
* Given:
|
|
|
|
* ```
|
|
|
|
* <div style="width:100px; height:200px; color:red">...</div>
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* the generated code is:
|
|
|
|
* ```
|
|
|
|
* var _c1 = [AttributeMarker.Styles, 'width', '100px', 'height'. '200px', 'color', 'red'];
|
|
|
|
* ```
|
|
|
|
*/
|
|
|
|
Styles = 2,
|
|
|
|
|
2018-08-20 15:20:12 +02:00
|
|
|
/**
|
2019-03-07 08:31:31 +00:00
|
|
|
* Signals that the following attribute names were extracted from input or output bindings.
|
|
|
|
*
|
|
|
|
* For example, given the following HTML:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <div moo="car" [foo]="exp" (bar)="doSth()">
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* the generated code is:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* var _c1 = ['moo', 'car', AttributeMarker.Bindings, 'foo', 'bar'];
|
|
|
|
* ```
|
2018-08-20 15:20:12 +02:00
|
|
|
*/
|
2019-03-07 08:31:31 +00:00
|
|
|
Bindings = 3,
|
2019-03-07 08:31:31 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals that the following attribute names were hoisted from an inline-template declaration.
|
|
|
|
*
|
|
|
|
* For example, given the following HTML:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <div *ngFor="let value of values; trackBy:trackBy" dirA [dirB]="value">
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* the generated code for the `template()` instruction would include:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* ['dirA', '', AttributeMarker.Bindings, 'dirB', AttributeMarker.Template, 'ngFor', 'ngForOf',
|
|
|
|
* 'ngForTrackBy', 'let-value']
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* while the generated code for the `element()` instruction inside the template function would
|
|
|
|
* include:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* ['dirA', '', AttributeMarker.Bindings, 'dirB']
|
|
|
|
* ```
|
|
|
|
*/
|
|
|
|
Template = 4,
|
2019-04-08 22:47:23 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals that the following attribute is `ngProjectAs` and its value is a parsed `CssSelector`.
|
|
|
|
*
|
|
|
|
* For example, given the following HTML:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <h1 attr="value" ngProjectAs="[title]">
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* the generated code for the `element()` instruction would include:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* ['attr', 'value', AttributeMarker.ProjectAs, ['', 'title', '']]
|
|
|
|
* ```
|
|
|
|
*/
|
2019-05-15 17:08:50 +02:00
|
|
|
ProjectAs = 5,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Signals that the following attribute will be translated by runtime i18n
|
|
|
|
*
|
|
|
|
* For example, given the following HTML:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <div moo="car" foo="value" i18n-foo [bar]="binding" i18n-bar>
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* the generated code is:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* var _c1 = ['moo', 'car', AttributeMarker.I18n, 'foo', 'bar'];
|
|
|
|
*/
|
|
|
|
I18n = 6,
|
2018-12-13 15:51:47 -08:00
|
|
|
}
|