2018-05-09 08:35:25 -07:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. 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.io/license
|
|
|
|
*/
|
|
|
|
|
2019-01-11 16:07:01 -08:00
|
|
|
import {R3DirectiveMetadataFacade, getCompilerFacade} from '../../compiler/compiler_facade';
|
2019-10-25 19:45:08 +02:00
|
|
|
import {R3ComponentMetadataFacade, R3QueryMetadataFacade} from '../../compiler/compiler_facade_interface';
|
2018-12-18 13:36:19 -08:00
|
|
|
import {resolveForwardRef} from '../../di/forward_ref';
|
2019-01-11 16:07:01 -08:00
|
|
|
import {getReflect, reflectDependencies} from '../../di/jit/util';
|
2019-01-09 13:49:16 -08:00
|
|
|
import {Type} from '../../interface/type';
|
2018-10-29 10:07:40 +01:00
|
|
|
import {Query} from '../../metadata/di';
|
2019-01-28 20:45:41 -08:00
|
|
|
import {Component, Directive, Input} from '../../metadata/directives';
|
2018-06-22 19:05:31 -07:00
|
|
|
import {componentNeedsResolution, maybeQueueResolutionOfComponentResources} from '../../metadata/resource_loading';
|
2018-07-31 11:14:06 -07:00
|
|
|
import {ViewEncapsulation} from '../../metadata/view';
|
2019-08-15 14:14:25 +01:00
|
|
|
import {initNgDevMode} from '../../util/ng_dev_mode';
|
2019-10-25 19:45:08 +02:00
|
|
|
import {getComponentDef, getDirectiveDef} from '../definition';
|
2018-12-13 15:51:47 -08:00
|
|
|
import {EMPTY_ARRAY, EMPTY_OBJ} from '../empty';
|
2019-10-25 19:45:08 +02:00
|
|
|
import {NG_COMP_DEF, NG_DIR_DEF, NG_FACTORY_DEF} from '../fields';
|
2019-04-01 15:36:43 -07:00
|
|
|
import {ComponentType} from '../interfaces/definition';
|
2019-04-24 14:50:01 +02:00
|
|
|
import {stringifyForError} from '../util/misc_utils';
|
2018-05-09 08:35:25 -07:00
|
|
|
|
|
|
|
import {angularCoreEnv} from './environment';
|
2020-02-18 19:55:55 +01:00
|
|
|
import {getJitOptions} from './jit_options';
|
2018-12-03 13:58:07 -08:00
|
|
|
import {flushModuleScopingQueueAsMuchAsPossible, patchComponentDefWithScope, transitiveScopesFor} from './module';
|
2018-05-09 08:35:25 -07:00
|
|
|
|
2018-10-24 16:02:25 -07:00
|
|
|
|
2018-06-12 16:58:09 -07:00
|
|
|
|
2018-05-09 08:35:25 -07:00
|
|
|
/**
|
|
|
|
* Compile an Angular component according to its decorator metadata, and patch the resulting
|
2019-10-10 14:57:15 -07:00
|
|
|
* component def (ɵcmp) onto the component type.
|
2018-05-09 08:35:25 -07:00
|
|
|
*
|
|
|
|
* Compilation may be asynchronous (due to the need to resolve URLs for the component template or
|
|
|
|
* other resources, for example). In the event that compilation is not immediate, `compileComponent`
|
2019-10-10 14:57:15 -07:00
|
|
|
* will enqueue resource resolution into a global queue and will fail to return the `ɵcmp`
|
2018-06-22 19:05:31 -07:00
|
|
|
* until the global queue has been resolved with a call to `resolveComponentResources`.
|
2018-05-09 08:35:25 -07:00
|
|
|
*/
|
2018-06-22 19:05:31 -07:00
|
|
|
export function compileComponent(type: Type<any>, metadata: Component): void {
|
2019-08-15 14:14:25 +01:00
|
|
|
// Initialize ngDevMode. This must be the first statement in compileComponent.
|
|
|
|
// See the `initNgDevMode` docstring for more information.
|
|
|
|
(typeof ngDevMode === 'undefined' || ngDevMode) && initNgDevMode();
|
|
|
|
|
2018-08-06 14:09:38 -07:00
|
|
|
let ngComponentDef: any = null;
|
2019-08-12 09:26:20 +03:00
|
|
|
|
2018-06-22 19:05:31 -07:00
|
|
|
// Metadata may have resources which need to be resolved.
|
2019-03-11 10:35:25 -07:00
|
|
|
maybeQueueResolutionOfComponentResources(type, metadata);
|
2019-08-12 09:26:20 +03:00
|
|
|
|
2019-09-01 12:26:04 +02:00
|
|
|
// Note that we're using the same function as `Directive`, because that's only subset of metadata
|
|
|
|
// that we need to create the ngFactoryDef. We're avoiding using the component metadata
|
|
|
|
// because we'd have to resolve the asynchronous templates.
|
|
|
|
addDirectiveFactoryDef(type, metadata);
|
2019-08-12 09:26:20 +03:00
|
|
|
|
2019-10-10 14:57:15 -07:00
|
|
|
Object.defineProperty(type, NG_COMP_DEF, {
|
2018-05-21 08:15:19 -07:00
|
|
|
get: () => {
|
2018-08-06 14:09:38 -07:00
|
|
|
if (ngComponentDef === null) {
|
2019-08-12 09:26:20 +03:00
|
|
|
const compiler = getCompilerFacade();
|
2019-09-01 12:26:04 +02:00
|
|
|
|
|
|
|
if (componentNeedsResolution(metadata)) {
|
|
|
|
const error = [`Component '${type.name}' is not resolved:`];
|
|
|
|
if (metadata.templateUrl) {
|
|
|
|
error.push(` - templateUrl: ${metadata.templateUrl}`);
|
|
|
|
}
|
|
|
|
if (metadata.styleUrls && metadata.styleUrls.length) {
|
|
|
|
error.push(` - styleUrls: ${JSON.stringify(metadata.styleUrls)}`);
|
|
|
|
}
|
|
|
|
error.push(`Did you run and wait for 'resolveComponentResources()'?`);
|
|
|
|
throw new Error(error.join('\n'));
|
|
|
|
}
|
|
|
|
|
2020-02-18 19:55:55 +01:00
|
|
|
const jitOptions = getJitOptions();
|
|
|
|
let preserveWhitespaces = metadata.preserveWhitespaces;
|
|
|
|
if (preserveWhitespaces === undefined) {
|
|
|
|
if (jitOptions !== null && jitOptions.preserveWhitespaces !== undefined) {
|
|
|
|
preserveWhitespaces = jitOptions.preserveWhitespaces;
|
|
|
|
} else {
|
|
|
|
preserveWhitespaces = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let encapsulation = metadata.encapsulation;
|
|
|
|
if (encapsulation === undefined) {
|
|
|
|
if (jitOptions !== null && jitOptions.defaultEncapsulation !== undefined) {
|
|
|
|
encapsulation = jitOptions.defaultEncapsulation;
|
|
|
|
} else {
|
|
|
|
encapsulation = ViewEncapsulation.Emulated;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-01 12:26:04 +02:00
|
|
|
const templateUrl = metadata.templateUrl || `ng:///${type.name}/template.html`;
|
|
|
|
const meta: R3ComponentMetadataFacade = {
|
|
|
|
...directiveMetadata(type, metadata),
|
|
|
|
typeSourceSpan: compiler.createParseSourceSpan('Component', type.name, templateUrl),
|
2020-02-18 19:55:55 +01:00
|
|
|
template: metadata.template || '', preserveWhitespaces,
|
2019-09-01 12:26:04 +02:00
|
|
|
styles: metadata.styles || EMPTY_ARRAY,
|
|
|
|
animations: metadata.animations,
|
|
|
|
directives: [],
|
|
|
|
changeDetection: metadata.changeDetection,
|
2020-02-18 19:55:55 +01:00
|
|
|
pipes: new Map(), encapsulation,
|
2019-09-01 12:26:04 +02:00
|
|
|
interpolation: metadata.interpolation,
|
|
|
|
viewProviders: metadata.viewProviders || null,
|
|
|
|
};
|
|
|
|
if (meta.usesInheritance) {
|
2019-10-25 19:45:08 +02:00
|
|
|
addDirectiveDefToUndecoratedParents(type);
|
2019-09-01 12:26:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ngComponentDef = compiler.compileComponent(angularCoreEnv, templateUrl, meta);
|
2018-06-06 11:23:38 -07:00
|
|
|
|
2018-12-03 13:58:07 -08:00
|
|
|
// When NgModule decorator executed, we enqueued the module definition such that
|
|
|
|
// it would only dequeue and add itself as module scope to all of its declarations,
|
|
|
|
// but only if if all of its declarations had resolved. This call runs the check
|
|
|
|
// to see if any modules that are in the queue can be dequeued and add scope to
|
|
|
|
// their declarations.
|
|
|
|
flushModuleScopingQueueAsMuchAsPossible();
|
|
|
|
|
2018-06-06 11:23:38 -07:00
|
|
|
// If component compilation is async, then the @NgModule annotation which declares the
|
|
|
|
// component may execute and set an ngSelectorScope property on the component type. This
|
2018-08-20 15:20:12 +02:00
|
|
|
// allows the component to patch itself with directiveDefs from the module after it
|
|
|
|
// finishes compiling.
|
2018-06-06 11:23:38 -07:00
|
|
|
if (hasSelectorScope(type)) {
|
2018-08-06 14:09:38 -07:00
|
|
|
const scopes = transitiveScopesFor(type.ngSelectorScope);
|
|
|
|
patchComponentDefWithScope(ngComponentDef, scopes);
|
2018-06-06 11:23:38 -07:00
|
|
|
}
|
2018-05-21 08:15:19 -07:00
|
|
|
}
|
2018-08-06 14:09:38 -07:00
|
|
|
return ngComponentDef;
|
2018-05-21 08:15:19 -07:00
|
|
|
},
|
2018-08-06 14:09:38 -07:00
|
|
|
// Make the property configurable in dev mode to allow overriding in tests
|
|
|
|
configurable: !!ngDevMode,
|
2018-05-21 08:15:19 -07:00
|
|
|
});
|
2019-08-12 09:26:20 +03:00
|
|
|
}
|
|
|
|
|
2018-06-06 11:23:38 -07:00
|
|
|
function hasSelectorScope<T>(component: Type<T>): component is Type<T>&
|
|
|
|
{ngSelectorScope: Type<any>} {
|
|
|
|
return (component as{ngSelectorScope?: any}).ngSelectorScope !== undefined;
|
|
|
|
}
|
|
|
|
|
2018-05-21 08:15:19 -07:00
|
|
|
/**
|
|
|
|
* Compile an Angular directive according to its decorator metadata, and patch the resulting
|
2019-10-11 12:28:12 -07:00
|
|
|
* directive def onto the component type.
|
2018-05-21 08:15:19 -07:00
|
|
|
*
|
|
|
|
* In the event that compilation is not immediate, `compileDirective` will return a `Promise` which
|
|
|
|
* will resolve when compilation completes and the directive becomes usable.
|
|
|
|
*/
|
2019-08-12 14:56:30 -07:00
|
|
|
export function compileDirective(type: Type<any>, directive: Directive | null): void {
|
2018-08-06 14:09:38 -07:00
|
|
|
let ngDirectiveDef: any = null;
|
2019-08-12 09:26:20 +03:00
|
|
|
|
2019-09-01 12:26:04 +02:00
|
|
|
addDirectiveFactoryDef(type, directive || {});
|
2019-08-12 09:26:20 +03:00
|
|
|
|
2019-10-11 12:28:12 -07:00
|
|
|
Object.defineProperty(type, NG_DIR_DEF, {
|
2018-05-21 08:15:19 -07:00
|
|
|
get: () => {
|
2018-08-06 14:09:38 -07:00
|
|
|
if (ngDirectiveDef === null) {
|
2019-08-12 14:56:30 -07:00
|
|
|
// `directive` can be null in the case of abstract directives as a base class
|
|
|
|
// that use `@Directive()` with no selector. In that case, pass empty object to the
|
|
|
|
// `directiveMetadata` function instead of null.
|
2019-08-12 09:26:20 +03:00
|
|
|
const meta = getDirectiveMetadata(type, directive || {});
|
|
|
|
ngDirectiveDef =
|
|
|
|
getCompilerFacade().compileDirective(angularCoreEnv, meta.sourceMapUrl, meta.metadata);
|
2018-05-21 08:15:19 -07:00
|
|
|
}
|
2018-08-06 14:09:38 -07:00
|
|
|
return ngDirectiveDef;
|
2018-05-21 08:15:19 -07:00
|
|
|
},
|
2018-08-06 14:09:38 -07:00
|
|
|
// Make the property configurable in dev mode to allow overriding in tests
|
|
|
|
configurable: !!ngDevMode,
|
2018-05-21 08:15:19 -07:00
|
|
|
});
|
2018-05-09 08:35:25 -07:00
|
|
|
}
|
|
|
|
|
2019-08-12 09:26:20 +03:00
|
|
|
function getDirectiveMetadata(type: Type<any>, metadata: Directive) {
|
|
|
|
const name = type && type.name;
|
2019-10-11 12:28:12 -07:00
|
|
|
const sourceMapUrl = `ng:///${name}/ɵdir.js`;
|
2019-08-12 09:26:20 +03:00
|
|
|
const compiler = getCompilerFacade();
|
|
|
|
const facade = directiveMetadata(type as ComponentType<any>, metadata);
|
|
|
|
facade.typeSourceSpan = compiler.createParseSourceSpan('Directive', name, sourceMapUrl);
|
|
|
|
if (facade.usesInheritance) {
|
2019-10-25 19:45:08 +02:00
|
|
|
addDirectiveDefToUndecoratedParents(type);
|
2019-08-12 09:26:20 +03:00
|
|
|
}
|
|
|
|
return {metadata: facade, sourceMapUrl};
|
|
|
|
}
|
|
|
|
|
2019-09-01 12:26:04 +02:00
|
|
|
function addDirectiveFactoryDef(type: Type<any>, metadata: Directive | Component) {
|
|
|
|
let ngFactoryDef: any = null;
|
|
|
|
|
|
|
|
Object.defineProperty(type, NG_FACTORY_DEF, {
|
|
|
|
get: () => {
|
|
|
|
if (ngFactoryDef === null) {
|
|
|
|
const meta = getDirectiveMetadata(type, metadata);
|
2019-10-03 21:54:49 +02:00
|
|
|
const compiler = getCompilerFacade();
|
|
|
|
ngFactoryDef = compiler.compileFactory(angularCoreEnv, `ng:///${type.name}/ɵfac.js`, {
|
|
|
|
...meta.metadata,
|
|
|
|
injectFn: 'directiveInject',
|
|
|
|
target: compiler.R3FactoryTarget.Directive
|
|
|
|
});
|
2019-09-01 12:26:04 +02:00
|
|
|
}
|
|
|
|
return ngFactoryDef;
|
|
|
|
},
|
|
|
|
// Make the property configurable in dev mode to allow overriding in tests
|
|
|
|
configurable: !!ngDevMode,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-06-18 08:05:06 -07:00
|
|
|
export function extendsDirectlyFromObject(type: Type<any>): boolean {
|
|
|
|
return Object.getPrototypeOf(type.prototype) === Object.prototype;
|
|
|
|
}
|
|
|
|
|
2018-05-21 08:15:19 -07:00
|
|
|
/**
|
|
|
|
* Extract the `R3DirectiveMetadata` for a particular directive (either a `Directive` or a
|
|
|
|
* `Component`).
|
|
|
|
*/
|
2019-03-08 18:21:15 +01:00
|
|
|
export function directiveMetadata(type: Type<any>, metadata: Directive): R3DirectiveMetadataFacade {
|
2018-05-21 08:15:19 -07:00
|
|
|
// Reflect inputs and outputs.
|
2019-12-11 22:08:56 +01:00
|
|
|
const reflect = getReflect();
|
|
|
|
const propMetadata = reflect.ownPropMetadata(type);
|
2018-06-12 16:58:09 -07:00
|
|
|
|
2018-05-21 08:15:19 -07:00
|
|
|
return {
|
|
|
|
name: type.name,
|
2018-10-24 16:02:25 -07:00
|
|
|
type: type,
|
2018-07-13 14:49:01 -07:00
|
|
|
typeArgumentCount: 0,
|
fix(ivy): support abstract directives in template type checking (#33131)
Recently it was made possible to have a directive without selector,
which are referred to as abstract directives. Such directives should not
be registered in an NgModule, but can still contain decorators for
inputs, outputs, queries, etc. The information from these decorators and
the `@Directive()` decorator itself needs to be registered with the
central `MetadataRegistry` so that other areas of the compiler can
request information about a given directive, an example of which is the
template type checker that needs to know about the inputs and outputs of
directives.
Prior to this change, however, abstract directives would only register
themselves with the `MetadataRegistry` as being an abstract directive,
without all of its other metadata like inputs and outputs. This meant
that the template type checker was unable to resolve the inputs and
outputs of these abstract directives, therefore failing to check them
correctly. The typical error would be that some property does not exist
on a DOM element, whereas said property should have been bound to the
abstract directive's input.
This commit fixes the problem by always registering the metadata of a
directive or component with the `MetadataRegistry`. Tests have been
added to ensure abstract directives are handled correctly in the
template type checker, together with tests to verify the form of
abstract directives in declaration files.
Fixes #30080
PR Close #33131
2019-10-13 17:00:13 +02:00
|
|
|
selector: metadata.selector !== undefined ? metadata.selector : null,
|
2018-10-24 16:02:25 -07:00
|
|
|
deps: reflectDependencies(type),
|
|
|
|
host: metadata.host || EMPTY_OBJ,
|
|
|
|
propMetadata: propMetadata,
|
|
|
|
inputs: metadata.inputs || EMPTY_ARRAY,
|
|
|
|
outputs: metadata.outputs || EMPTY_ARRAY,
|
2018-12-11 10:43:02 -08:00
|
|
|
queries: extractQueriesMetadata(type, propMetadata, isContentQuery),
|
2019-12-11 22:08:56 +01:00
|
|
|
lifecycle: {usesOnChanges: reflect.hasLifecycleHook(type, 'ngOnChanges')},
|
2018-05-21 08:15:19 -07:00
|
|
|
typeSourceSpan: null !,
|
2018-06-18 08:05:06 -07:00
|
|
|
usesInheritance: !extendsDirectlyFromObject(type),
|
2019-01-08 16:30:57 -08:00
|
|
|
exportAs: extractExportAs(metadata.exportAs),
|
2018-10-24 16:02:25 -07:00
|
|
|
providers: metadata.providers || null,
|
2019-09-01 12:26:04 +02:00
|
|
|
viewQueries: extractQueriesMetadata(type, propMetadata, isViewQuery)
|
2018-05-21 08:15:19 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-04-21 17:37:15 +02:00
|
|
|
/**
|
2019-10-25 19:45:08 +02:00
|
|
|
* Adds a directive definition to all parent classes of a type that don't have an Angular decorator.
|
2019-04-21 17:37:15 +02:00
|
|
|
*/
|
2019-10-25 19:45:08 +02:00
|
|
|
function addDirectiveDefToUndecoratedParents(type: Type<any>) {
|
2019-04-21 17:37:15 +02:00
|
|
|
const objPrototype = Object.prototype;
|
2019-12-11 22:08:56 +01:00
|
|
|
let parent = Object.getPrototypeOf(type.prototype).constructor;
|
2019-04-21 17:37:15 +02:00
|
|
|
|
|
|
|
// Go up the prototype until we hit `Object`.
|
|
|
|
while (parent && parent !== objPrototype) {
|
|
|
|
// Since inheritance works if the class was annotated already, we only need to add
|
2019-10-25 19:45:08 +02:00
|
|
|
// the def if there are no annotations and the def hasn't been created already.
|
|
|
|
if (!getDirectiveDef(parent) && !getComponentDef(parent) &&
|
|
|
|
shouldAddAbstractDirective(parent)) {
|
|
|
|
compileDirective(parent, null);
|
2019-04-21 17:37:15 +02:00
|
|
|
}
|
|
|
|
parent = Object.getPrototypeOf(parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-24 16:02:25 -07:00
|
|
|
function convertToR3QueryPredicate(selector: any): any|string[] {
|
2018-12-18 13:36:19 -08:00
|
|
|
return typeof selector === 'string' ? splitByComma(selector) : resolveForwardRef(selector);
|
2018-10-29 10:07:40 +01:00
|
|
|
}
|
|
|
|
|
2018-10-24 16:02:25 -07:00
|
|
|
export function convertToR3QueryMetadata(propertyName: string, ann: Query): R3QueryMetadataFacade {
|
2018-10-29 10:07:40 +01:00
|
|
|
return {
|
|
|
|
propertyName: propertyName,
|
|
|
|
predicate: convertToR3QueryPredicate(ann.selector),
|
|
|
|
descendants: ann.descendants,
|
|
|
|
first: ann.first,
|
2019-02-18 17:33:59 -08:00
|
|
|
read: ann.read ? ann.read : null,
|
|
|
|
static: !!ann.static
|
2018-10-29 10:07:40 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
function extractQueriesMetadata(
|
2018-12-11 10:43:02 -08:00
|
|
|
type: Type<any>, propMetadata: {[key: string]: any[]},
|
2018-10-24 16:02:25 -07:00
|
|
|
isQueryAnn: (ann: any) => ann is Query): R3QueryMetadataFacade[] {
|
|
|
|
const queriesMeta: R3QueryMetadataFacade[] = [];
|
2018-10-29 10:07:40 +01:00
|
|
|
for (const field in propMetadata) {
|
|
|
|
if (propMetadata.hasOwnProperty(field)) {
|
2019-01-28 20:45:41 -08:00
|
|
|
const annotations = propMetadata[field];
|
|
|
|
annotations.forEach(ann => {
|
2018-10-29 10:07:40 +01:00
|
|
|
if (isQueryAnn(ann)) {
|
2018-12-11 10:43:02 -08:00
|
|
|
if (!ann.selector) {
|
|
|
|
throw new Error(
|
|
|
|
`Can't construct a query for the property "${field}" of ` +
|
2019-04-24 14:50:01 +02:00
|
|
|
`"${stringifyForError(type)}" since the query selector wasn't defined.`);
|
2018-12-11 10:43:02 -08:00
|
|
|
}
|
2019-10-25 19:45:08 +02:00
|
|
|
if (annotations.some(isInputAnnotation)) {
|
2019-01-28 20:45:41 -08:00
|
|
|
throw new Error(`Cannot combine @Input decorators with query decorators`);
|
|
|
|
}
|
2018-10-29 10:07:40 +01:00
|
|
|
queriesMeta.push(convertToR3QueryMetadata(field, ann));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return queriesMeta;
|
|
|
|
}
|
|
|
|
|
2019-01-08 16:30:57 -08:00
|
|
|
function extractExportAs(exportAs: string | undefined): string[]|null {
|
2019-10-25 19:45:08 +02:00
|
|
|
return exportAs === undefined ? null : splitByComma(exportAs);
|
2019-01-08 16:30:57 -08:00
|
|
|
}
|
|
|
|
|
2018-10-29 10:07:40 +01:00
|
|
|
function isContentQuery(value: any): value is Query {
|
|
|
|
const name = value.ngMetadataName;
|
|
|
|
return name === 'ContentChild' || name === 'ContentChildren';
|
|
|
|
}
|
|
|
|
|
|
|
|
function isViewQuery(value: any): value is Query {
|
|
|
|
const name = value.ngMetadataName;
|
|
|
|
return name === 'ViewChild' || name === 'ViewChildren';
|
|
|
|
}
|
|
|
|
|
2019-10-25 19:45:08 +02:00
|
|
|
function isInputAnnotation(value: any): value is Input {
|
2019-01-28 20:45:41 -08:00
|
|
|
return value.ngMetadataName === 'Input';
|
|
|
|
}
|
|
|
|
|
2018-10-29 10:07:40 +01:00
|
|
|
function splitByComma(value: string): string[] {
|
|
|
|
return value.split(',').map(piece => piece.trim());
|
|
|
|
}
|
2019-10-25 19:45:08 +02:00
|
|
|
|
|
|
|
const LIFECYCLE_HOOKS = [
|
|
|
|
'ngOnChanges', 'ngOnInit', 'ngOnDestroy', 'ngDoCheck', 'ngAfterViewInit', 'ngAfterViewChecked',
|
|
|
|
'ngAfterContentInit', 'ngAfterContentChecked'
|
|
|
|
];
|
|
|
|
|
|
|
|
function shouldAddAbstractDirective(type: Type<any>): boolean {
|
2019-12-11 22:08:56 +01:00
|
|
|
const reflect = getReflect();
|
|
|
|
|
|
|
|
if (LIFECYCLE_HOOKS.some(hookName => reflect.hasLifecycleHook(type, hookName))) {
|
2019-10-25 19:45:08 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-12-11 22:08:56 +01:00
|
|
|
const propMetadata = reflect.propMetadata(type);
|
2019-10-25 19:45:08 +02:00
|
|
|
|
|
|
|
for (const field in propMetadata) {
|
|
|
|
const annotations = propMetadata[field];
|
|
|
|
|
|
|
|
for (let i = 0; i < annotations.length; i++) {
|
|
|
|
const current = annotations[i];
|
|
|
|
const metadataName = current.ngMetadataName;
|
|
|
|
|
|
|
|
if (isInputAnnotation(current) || isContentQuery(current) || isViewQuery(current) ||
|
|
|
|
metadataName === 'Output' || metadataName === 'HostBinding' ||
|
|
|
|
metadataName === 'HostListener') {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|