fix(upgrade): fix downgrade content projection and injector inheritance
- Full support for content projection in downgraded Angular 2 components. In particular, this enables multi-slot projection and other features on <ng-content>. - Correctly wire up hierarchical injectors for downgraded Angular 2 components: downgraded components inherit the injector of the first other downgraded Angular 2 component they find up the DOM tree. Closes #6629, #7727, #8729, #9643, #9649, #12675
This commit is contained in:
parent
d6e5e9283c
commit
d91a86aac6
|
@ -70,6 +70,14 @@ export class JitCompiler implements Compiler {
|
||||||
return this._compileModuleAndAllComponents(moduleType, false).asyncResult;
|
return this._compileModuleAndAllComponents(moduleType, false).asyncResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getNgContentSelectors(component: Type<any>): string[] {
|
||||||
|
const template = this._compiledTemplateCache.get(component);
|
||||||
|
if (!template) {
|
||||||
|
throw new Error(`The component ${stringify(component)} is not yet compiled!`);
|
||||||
|
}
|
||||||
|
return template.compMeta.template.ngContentSelectors;
|
||||||
|
}
|
||||||
|
|
||||||
private _compileModuleAndComponents<T>(moduleType: Type<T>, isSync: boolean):
|
private _compileModuleAndComponents<T>(moduleType: Type<T>, isSync: boolean):
|
||||||
SyncAsyncResult<NgModuleFactory<T>> {
|
SyncAsyncResult<NgModuleFactory<T>> {
|
||||||
const loadingPromise = this._loadModules(moduleType, isSync);
|
const loadingPromise = this._loadModules(moduleType, isSync);
|
||||||
|
@ -408,6 +416,11 @@ class ModuleBoundCompiler implements Compiler {
|
||||||
return this._delegate.compileModuleAndAllComponentsAsync(moduleType);
|
return this._delegate.compileModuleAndAllComponentsAsync(moduleType);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getNgContentSelectors(component: Type<any>): string[] {
|
||||||
|
return this._delegate.getNgContentSelectors(component);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Clears all caches
|
* Clears all caches
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -70,6 +70,10 @@ export class TestingCompilerImpl implements TestingCompiler {
|
||||||
return this._compiler.compileModuleAndAllComponentsAsync(moduleType);
|
return this._compiler.compileModuleAndAllComponentsAsync(moduleType);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getNgContentSelectors(component: Type<any>): string[] {
|
||||||
|
return this._compiler.getNgContentSelectors(component);
|
||||||
|
}
|
||||||
|
|
||||||
overrideModule(ngModule: Type<any>, override: MetadataOverride<NgModule>): void {
|
overrideModule(ngModule: Type<any>, override: MetadataOverride<NgModule>): void {
|
||||||
const oldMetadata = this._moduleResolver.resolve(ngModule, false);
|
const oldMetadata = this._moduleResolver.resolve(ngModule, false);
|
||||||
this._moduleResolver.setNgModule(
|
this._moduleResolver.setNgModule(
|
||||||
|
|
|
@ -82,6 +82,14 @@ export class Compiler {
|
||||||
throw _throwError();
|
throw _throwError();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exposes the CSS-style selectors that have been used in `ngContent` directives within
|
||||||
|
* the template of the given component.
|
||||||
|
* This is used by the `upgrade` library to compile the appropriate transclude content
|
||||||
|
* in the Angular 1 wrapper component.
|
||||||
|
*/
|
||||||
|
getNgContentSelectors(component: Type<any>): string[] { throw _throwError(); }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Clears all caches.
|
* Clears all caches.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -27,7 +27,7 @@ export interface IModule {
|
||||||
run(a: IInjectable): IModule;
|
run(a: IInjectable): IModule;
|
||||||
}
|
}
|
||||||
export interface ICompileService {
|
export interface ICompileService {
|
||||||
(element: Element|NodeList|string, transclude?: Function): ILinkFn;
|
(element: Element|NodeList|Node[]|string, transclude?: Function): ILinkFn;
|
||||||
}
|
}
|
||||||
export interface ILinkFn {
|
export interface ILinkFn {
|
||||||
(scope: IScope, cloneAttachFn?: ICloneAttachFunction, options?: ILinkFnOptions): IAugmentedJQuery;
|
(scope: IScope, cloneAttachFn?: ICloneAttachFunction, options?: ILinkFnOptions): IAugmentedJQuery;
|
||||||
|
|
|
@ -21,4 +21,4 @@ export const NG1_INJECTOR = '$injector';
|
||||||
export const NG1_PARSE = '$parse';
|
export const NG1_PARSE = '$parse';
|
||||||
export const NG1_TEMPLATE_CACHE = '$templateCache';
|
export const NG1_TEMPLATE_CACHE = '$templateCache';
|
||||||
export const NG1_TESTABILITY = '$$testability';
|
export const NG1_TESTABILITY = '$$testability';
|
||||||
export const REQUIRE_INJECTOR = '?^' + NG2_INJECTOR;
|
export const REQUIRE_INJECTOR = '?^^' + NG2_INJECTOR;
|
||||||
|
|
|
@ -23,26 +23,21 @@ export class DowngradeNg2ComponentAdapter {
|
||||||
componentRef: ComponentRef<any> = null;
|
componentRef: ComponentRef<any> = null;
|
||||||
changeDetector: ChangeDetectorRef = null;
|
changeDetector: ChangeDetectorRef = null;
|
||||||
componentScope: angular.IScope;
|
componentScope: angular.IScope;
|
||||||
childNodes: Node[];
|
|
||||||
contentInsertionPoint: Node = null;
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private id: string, private info: ComponentInfo, private element: angular.IAugmentedJQuery,
|
private info: ComponentInfo, private element: angular.IAugmentedJQuery,
|
||||||
private attrs: angular.IAttributes, private scope: angular.IScope,
|
private attrs: angular.IAttributes, private scope: angular.IScope,
|
||||||
private parentInjector: Injector, private parse: angular.IParseService,
|
private parentInjector: Injector, private parse: angular.IParseService,
|
||||||
private componentFactory: ComponentFactory<any>) {
|
private componentFactory: ComponentFactory<any>) {
|
||||||
(<any>this.element[0]).id = id;
|
|
||||||
this.componentScope = scope.$new();
|
this.componentScope = scope.$new();
|
||||||
this.childNodes = <Node[]><any>element.contents();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bootstrapNg2() {
|
bootstrapNg2(projectableNodes: Node[][]) {
|
||||||
const childInjector = ReflectiveInjector.resolveAndCreate(
|
const childInjector = ReflectiveInjector.resolveAndCreate(
|
||||||
[{provide: NG1_SCOPE, useValue: this.componentScope}], this.parentInjector);
|
[{provide: NG1_SCOPE, useValue: this.componentScope}], this.parentInjector);
|
||||||
this.contentInsertionPoint = document.createComment('ng1 insertion point');
|
|
||||||
|
|
||||||
this.componentRef = this.componentFactory.create(
|
this.componentRef =
|
||||||
childInjector, [[this.contentInsertionPoint]], this.element[0]);
|
this.componentFactory.create(childInjector, projectableNodes, this.element[0]);
|
||||||
this.changeDetector = this.componentRef.changeDetectorRef;
|
this.changeDetector = this.componentRef.changeDetectorRef;
|
||||||
this.component = this.componentRef.instance;
|
this.component = this.componentRef.instance;
|
||||||
}
|
}
|
||||||
|
@ -103,16 +98,6 @@ export class DowngradeNg2ComponentAdapter {
|
||||||
this.componentScope.$watch(() => this.changeDetector && this.changeDetector.detectChanges());
|
this.componentScope.$watch(() => this.changeDetector && this.changeDetector.detectChanges());
|
||||||
}
|
}
|
||||||
|
|
||||||
projectContent() {
|
|
||||||
const childNodes = this.childNodes;
|
|
||||||
const parent = this.contentInsertionPoint.parentNode;
|
|
||||||
if (parent) {
|
|
||||||
for (let i = 0, ii = childNodes.length; i < ii; i++) {
|
|
||||||
parent.insertBefore(childNodes[i], this.contentInsertionPoint);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
setupOutputs() {
|
setupOutputs() {
|
||||||
const attrs = this.attrs;
|
const attrs = this.attrs;
|
||||||
const outputs = this.info.outputs || [];
|
const outputs = this.info.outputs || [];
|
||||||
|
|
|
@ -6,15 +6,17 @@
|
||||||
* found in the LICENSE file at https://angular.io/license
|
* found in the LICENSE file at https://angular.io/license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import {CssSelector, SelectorMatcher, createElementCssSelector} from '@angular/compiler';
|
||||||
import {Compiler, CompilerOptions, ComponentFactory, Injector, NgModule, NgModuleRef, NgZone, Provider, Testability, Type} from '@angular/core';
|
import {Compiler, CompilerOptions, ComponentFactory, Injector, NgModule, NgModuleRef, NgZone, Provider, Testability, Type} from '@angular/core';
|
||||||
import {platformBrowserDynamic} from '@angular/platform-browser-dynamic';
|
import {platformBrowserDynamic} from '@angular/platform-browser-dynamic';
|
||||||
|
|
||||||
import * as angular from './angular_js';
|
import * as angular from './angular_js';
|
||||||
import {NG1_COMPILE, NG1_INJECTOR, NG1_PARSE, NG1_ROOT_SCOPE, NG1_TESTABILITY, NG2_COMPILER, NG2_COMPONENT_FACTORY_REF_MAP, NG2_INJECTOR, NG2_ZONE, REQUIRE_INJECTOR} from './constants';
|
import {NG1_COMPILE, NG1_INJECTOR, NG1_PARSE, NG1_ROOT_SCOPE, NG1_TESTABILITY, NG2_COMPILER, NG2_COMPONENT_FACTORY_REF_MAP, NG2_INJECTOR, NG2_ZONE, REQUIRE_INJECTOR} from './constants';
|
||||||
import {DowngradeNg2ComponentAdapter} from './downgrade_ng2_adapter';
|
import {DowngradeNg2ComponentAdapter} from './downgrade_ng2_adapter';
|
||||||
|
import {isPresent} from './facade/lang';
|
||||||
import {ComponentInfo, getComponentInfo} from './metadata';
|
import {ComponentInfo, getComponentInfo} from './metadata';
|
||||||
import {UpgradeNg1ComponentAdapterBuilder} from './upgrade_ng1_adapter';
|
import {UpgradeNg1ComponentAdapterBuilder} from './upgrade_ng1_adapter';
|
||||||
import {controllerKey, onError, Deferred} from './util';
|
import {Deferred, controllerKey, getAttributesAsArray, onError} from './util';
|
||||||
|
|
||||||
let upgradeCount: number = 0;
|
let upgradeCount: number = 0;
|
||||||
|
|
||||||
|
@ -58,8 +60,8 @@ let upgradeCount: number = 0;
|
||||||
* ### Example
|
* ### Example
|
||||||
*
|
*
|
||||||
* ```
|
* ```
|
||||||
* var adapter = new UpgradeAdapter(forwardRef(() => MyNg2Module), myCompilerOptions);
|
* const adapter = new UpgradeAdapter(forwardRef(() => MyNg2Module), myCompilerOptions);
|
||||||
* var module = angular.module('myExample', []);
|
* const module = angular.module('myExample', []);
|
||||||
* module.directive('ng2Comp', adapter.downgradeNg2Component(Ng2Component));
|
* module.directive('ng2Comp', adapter.downgradeNg2Component(Ng2Component));
|
||||||
*
|
*
|
||||||
* module.directive('ng1Hello', function() {
|
* module.directive('ng1Hello', function() {
|
||||||
|
@ -98,9 +100,7 @@ let upgradeCount: number = 0;
|
||||||
* @stable
|
* @stable
|
||||||
*/
|
*/
|
||||||
export class UpgradeAdapter {
|
export class UpgradeAdapter {
|
||||||
/* @internal */
|
|
||||||
private idPrefix: string = `NG2_UPGRADE_${upgradeCount++}_`;
|
private idPrefix: string = `NG2_UPGRADE_${upgradeCount++}_`;
|
||||||
/* @internal */
|
|
||||||
private upgradedComponents: Type<any>[] = [];
|
private upgradedComponents: Type<any>[] = [];
|
||||||
/**
|
/**
|
||||||
* An internal map of ng1 components which need to up upgraded to ng2.
|
* An internal map of ng1 components which need to up upgraded to ng2.
|
||||||
|
@ -111,8 +111,11 @@ export class UpgradeAdapter {
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
private ng1ComponentsToBeUpgraded: {[name: string]: UpgradeNg1ComponentAdapterBuilder} = {};
|
private ng1ComponentsToBeUpgraded: {[name: string]: UpgradeNg1ComponentAdapterBuilder} = {};
|
||||||
/* @internal */
|
|
||||||
private providers: Provider[] = [];
|
private providers: Provider[] = [];
|
||||||
|
private ngZone: NgZone;
|
||||||
|
private ng1Module: angular.IModule;
|
||||||
|
private moduleRef: NgModuleRef<any> = null;
|
||||||
|
private ng2BootstrapDeferred: Deferred<angular.IInjectorService>;
|
||||||
|
|
||||||
constructor(private ng2AppModule: Type<any>, private compilerOptions?: CompilerOptions) {
|
constructor(private ng2AppModule: Type<any>, private compilerOptions?: CompilerOptions) {
|
||||||
if (!ng2AppModule) {
|
if (!ng2AppModule) {
|
||||||
|
@ -149,8 +152,8 @@ export class UpgradeAdapter {
|
||||||
* ### Example
|
* ### Example
|
||||||
*
|
*
|
||||||
* ```
|
* ```
|
||||||
* var adapter = new UpgradeAdapter(forwardRef(() => MyNg2Module));
|
* const adapter = new UpgradeAdapter(forwardRef(() => MyNg2Module));
|
||||||
* var module = angular.module('myExample', []);
|
* const module = angular.module('myExample', []);
|
||||||
* module.directive('greet', adapter.downgradeNg2Component(Greeter));
|
* module.directive('greet', adapter.downgradeNg2Component(Greeter));
|
||||||
*
|
*
|
||||||
* @Component({
|
* @Component({
|
||||||
|
@ -227,8 +230,8 @@ export class UpgradeAdapter {
|
||||||
* ### Example
|
* ### Example
|
||||||
*
|
*
|
||||||
* ```
|
* ```
|
||||||
* var adapter = new UpgradeAdapter(forwardRef(() => MyNg2Module));
|
* const adapter = new UpgradeAdapter(forwardRef(() => MyNg2Module));
|
||||||
* var module = angular.module('myExample', []);
|
* const module = angular.module('myExample', []);
|
||||||
*
|
*
|
||||||
* module.directive('greet', function() {
|
* module.directive('greet', function() {
|
||||||
* return {
|
* return {
|
||||||
|
@ -278,8 +281,8 @@ export class UpgradeAdapter {
|
||||||
* ### Example
|
* ### Example
|
||||||
*
|
*
|
||||||
* ```
|
* ```
|
||||||
* var adapter = new UpgradeAdapter();
|
* const adapter = new UpgradeAdapter();
|
||||||
* var module = angular.module('myExample', []);
|
* const module = angular.module('myExample', []);
|
||||||
* module.directive('ng2', adapter.downgradeNg2Component(Ng2));
|
* module.directive('ng2', adapter.downgradeNg2Component(Ng2));
|
||||||
*
|
*
|
||||||
* module.directive('ng1', function() {
|
* module.directive('ng1', function() {
|
||||||
|
@ -314,126 +317,21 @@ export class UpgradeAdapter {
|
||||||
*/
|
*/
|
||||||
bootstrap(element: Element, modules?: any[], config?: angular.IAngularBootstrapConfig):
|
bootstrap(element: Element, modules?: any[], config?: angular.IAngularBootstrapConfig):
|
||||||
UpgradeAdapterRef {
|
UpgradeAdapterRef {
|
||||||
const ngZone =
|
this.declareNg1Module(modules);
|
||||||
new NgZone({enableLongStackTrace: Zone.hasOwnProperty('longStackTraceZoneSpec')});
|
|
||||||
const upgrade = new UpgradeAdapterRef();
|
const upgrade = new UpgradeAdapterRef();
|
||||||
let ng1Injector: angular.IInjectorService = null;
|
|
||||||
let moduleRef: NgModuleRef<any> = null;
|
|
||||||
const delayApplyExps: Function[] = [];
|
|
||||||
let original$applyFn: Function;
|
|
||||||
let rootScopePrototype: any;
|
|
||||||
let rootScope: angular.IRootScopeService;
|
|
||||||
const componentFactoryRefMap: ComponentFactoryRefMap = {};
|
|
||||||
const ng1Module = angular.module(this.idPrefix, modules);
|
|
||||||
let ng1BootstrapPromise: Promise<any>;
|
|
||||||
const ng2BootstrapDeferred = new Deferred();
|
|
||||||
ng1Module.factory(NG2_INJECTOR, () => moduleRef.injector.get(Injector))
|
|
||||||
.value(NG2_ZONE, ngZone)
|
|
||||||
.factory(NG2_COMPILER, () => moduleRef.injector.get(Compiler))
|
|
||||||
.value(NG2_COMPONENT_FACTORY_REF_MAP, componentFactoryRefMap)
|
|
||||||
.config([
|
|
||||||
'$provide', '$injector',
|
|
||||||
(provide: angular.IProvideService, ng1Injector: angular.IInjectorService) => {
|
|
||||||
provide.decorator(NG1_ROOT_SCOPE, [
|
|
||||||
'$delegate',
|
|
||||||
function(rootScopeDelegate: angular.IRootScopeService) {
|
|
||||||
// Capture the root apply so that we can delay first call to $apply until we
|
|
||||||
// bootstrap Angular 2 and then we replay and restore the $apply.
|
|
||||||
rootScopePrototype = rootScopeDelegate.constructor.prototype;
|
|
||||||
if (rootScopePrototype.hasOwnProperty('$apply')) {
|
|
||||||
original$applyFn = rootScopePrototype.$apply;
|
|
||||||
rootScopePrototype.$apply = (exp: any) => delayApplyExps.push(exp);
|
|
||||||
} else {
|
|
||||||
throw new Error('Failed to find \'$apply\' on \'$rootScope\'!');
|
|
||||||
}
|
|
||||||
return rootScope = rootScopeDelegate;
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
if (ng1Injector.has(NG1_TESTABILITY)) {
|
|
||||||
provide.decorator(NG1_TESTABILITY, [
|
|
||||||
'$delegate',
|
|
||||||
function(testabilityDelegate: angular.ITestabilityService) {
|
|
||||||
|
|
||||||
const originalWhenStable: Function = testabilityDelegate.whenStable;
|
|
||||||
// Cannot use arrow function below because we need the context
|
|
||||||
const newWhenStable = function(callback: Function) {
|
|
||||||
originalWhenStable.call(this, function() {
|
|
||||||
const ng2Testability: Testability = moduleRef.injector.get(Testability);
|
|
||||||
if (ng2Testability.isStable()) {
|
|
||||||
callback.apply(this, arguments);
|
|
||||||
} else {
|
|
||||||
ng2Testability.whenStable(newWhenStable.bind(this, callback));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
testabilityDelegate.whenStable = newWhenStable;
|
|
||||||
return testabilityDelegate;
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
|
|
||||||
ng1Module.run([
|
|
||||||
'$injector', '$rootScope',
|
|
||||||
(injector: angular.IInjectorService, rootScope: angular.IRootScopeService) => {
|
|
||||||
ng1Injector = injector;
|
|
||||||
UpgradeNg1ComponentAdapterBuilder.resolve(this.ng1ComponentsToBeUpgraded, injector)
|
|
||||||
.then(() => {
|
|
||||||
// At this point we have ng1 injector and we have lifted ng1 components into ng2, we
|
|
||||||
// now can bootstrap ng2.
|
|
||||||
const DynamicNgUpgradeModule =
|
|
||||||
NgModule({
|
|
||||||
providers: [
|
|
||||||
{provide: NG1_INJECTOR, useFactory: () => ng1Injector},
|
|
||||||
{provide: NG1_COMPILE, useFactory: () => ng1Injector.get(NG1_COMPILE)},
|
|
||||||
this.providers
|
|
||||||
],
|
|
||||||
imports: [this.ng2AppModule]
|
|
||||||
}).Class({
|
|
||||||
constructor: function DynamicNgUpgradeModule() {},
|
|
||||||
ngDoBootstrap: function() {}
|
|
||||||
});
|
|
||||||
|
|
||||||
(platformBrowserDynamic() as any)
|
|
||||||
._bootstrapModuleWithZone(
|
|
||||||
DynamicNgUpgradeModule, this.compilerOptions, ngZone,
|
|
||||||
(componentFactories: ComponentFactory<any>[]) => {
|
|
||||||
componentFactories.forEach((componentFactory: ComponentFactory<any>) => {
|
|
||||||
const type: Type<any> = componentFactory.componentType;
|
|
||||||
if (this.upgradedComponents.indexOf(type) !== -1) {
|
|
||||||
componentFactoryRefMap[getComponentInfo(type).selector] =
|
|
||||||
componentFactory;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.then((ref: NgModuleRef<any>) => {
|
|
||||||
moduleRef = ref;
|
|
||||||
angular.element(element).data(
|
|
||||||
controllerKey(NG2_INJECTOR), moduleRef.injector);
|
|
||||||
ngZone.onMicrotaskEmpty.subscribe({
|
|
||||||
next: (_: any) => ngZone.runOutsideAngular(() => rootScope.$evalAsync())
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.then(ng2BootstrapDeferred.resolve, ng2BootstrapDeferred.reject);
|
|
||||||
})
|
|
||||||
.catch(ng2BootstrapDeferred.reject);
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Make sure resumeBootstrap() only exists if the current bootstrap is deferred
|
// Make sure resumeBootstrap() only exists if the current bootstrap is deferred
|
||||||
const windowAngular = (window as any /** TODO #???? */)['angular'];
|
const windowAngular = (window as any /** TODO #???? */)['angular'];
|
||||||
windowAngular.resumeBootstrap = undefined;
|
windowAngular.resumeBootstrap = undefined;
|
||||||
|
|
||||||
ngZone.run(() => { angular.bootstrap(element, [this.idPrefix], config); });
|
this.ngZone.run(() => { angular.bootstrap(element, [this.ng1Module.name], config); });
|
||||||
ng1BootstrapPromise = new Promise((resolve) => {
|
const ng1BootstrapPromise = new Promise((resolve) => {
|
||||||
if (windowAngular.resumeBootstrap) {
|
if (windowAngular.resumeBootstrap) {
|
||||||
const originalResumeBootstrap: () => void = windowAngular.resumeBootstrap;
|
const originalResumeBootstrap: () => void = windowAngular.resumeBootstrap;
|
||||||
windowAngular.resumeBootstrap = function() {
|
windowAngular.resumeBootstrap = function() {
|
||||||
let args = arguments;
|
|
||||||
windowAngular.resumeBootstrap = originalResumeBootstrap;
|
windowAngular.resumeBootstrap = originalResumeBootstrap;
|
||||||
ngZone.run(() => { windowAngular.resumeBootstrap.apply(this, args); });
|
windowAngular.resumeBootstrap.apply(this, arguments);
|
||||||
resolve();
|
resolve();
|
||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
|
@ -441,17 +339,10 @@ export class UpgradeAdapter {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
Promise.all([ng1BootstrapPromise, ng2BootstrapDeferred.promise]).then(() => {
|
Promise.all([this.ng2BootstrapDeferred.promise, ng1BootstrapPromise]).then(([ng1Injector]) => {
|
||||||
moduleRef.injector.get(NgZone).run(() => {
|
angular.element(element).data(controllerKey(NG2_INJECTOR), this.moduleRef.injector);
|
||||||
if (rootScopePrototype) {
|
this.moduleRef.injector.get(NgZone).run(
|
||||||
rootScopePrototype.$apply = original$applyFn; // restore original $apply
|
() => { (<any>upgrade)._bootstrapDone(this.moduleRef, ng1Injector); });
|
||||||
while (delayApplyExps.length) {
|
|
||||||
rootScope.$apply(delayApplyExps.shift());
|
|
||||||
}
|
|
||||||
(<any>upgrade)._bootstrapDone(moduleRef, ng1Injector);
|
|
||||||
rootScopePrototype = null;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}, onError);
|
}, onError);
|
||||||
return upgrade;
|
return upgrade;
|
||||||
}
|
}
|
||||||
|
@ -473,16 +364,16 @@ export class UpgradeAdapter {
|
||||||
* }
|
* }
|
||||||
* }
|
* }
|
||||||
*
|
*
|
||||||
* var module = angular.module('myExample', []);
|
* const module = angular.module('myExample', []);
|
||||||
* module.service('server', Server);
|
* module.service('server', Server);
|
||||||
* module.service('login', Login);
|
* module.service('login', Login);
|
||||||
*
|
*
|
||||||
* var adapter = new UpgradeAdapter();
|
* const adapter = new UpgradeAdapter();
|
||||||
* adapter.upgradeNg1Provider('server');
|
* adapter.upgradeNg1Provider('server');
|
||||||
* adapter.upgradeNg1Provider('login', {asToken: Login});
|
* adapter.upgradeNg1Provider('login', {asToken: Login});
|
||||||
*
|
*
|
||||||
* adapter.bootstrap(document.body, ['myExample']).ready((ref) => {
|
* adapter.bootstrap(document.body, ['myExample']).ready((ref) => {
|
||||||
* var example: Example = ref.ng2Injector.get(Example);
|
* const example: Example = ref.ng2Injector.get(Example);
|
||||||
* });
|
* });
|
||||||
*
|
*
|
||||||
* ```
|
* ```
|
||||||
|
@ -506,13 +397,13 @@ export class UpgradeAdapter {
|
||||||
* class Example {
|
* class Example {
|
||||||
* }
|
* }
|
||||||
*
|
*
|
||||||
* var adapter = new UpgradeAdapter();
|
* const adapter = new UpgradeAdapter();
|
||||||
*
|
*
|
||||||
* var module = angular.module('myExample', []);
|
* const module = angular.module('myExample', []);
|
||||||
* module.factory('example', adapter.downgradeNg2Provider(Example));
|
* module.factory('example', adapter.downgradeNg2Provider(Example));
|
||||||
*
|
*
|
||||||
* adapter.bootstrap(document.body, ['myExample']).ready((ref) => {
|
* adapter.bootstrap(document.body, ['myExample']).ready((ref) => {
|
||||||
* var example: Example = ref.ng1Injector.get('example');
|
* const example: Example = ref.ng1Injector.get('example');
|
||||||
* });
|
* });
|
||||||
*
|
*
|
||||||
* ```
|
* ```
|
||||||
|
@ -522,42 +413,258 @@ export class UpgradeAdapter {
|
||||||
(<any>factory).$inject = [NG2_INJECTOR];
|
(<any>factory).$inject = [NG2_INJECTOR];
|
||||||
return factory;
|
return factory;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Declare the Angular 1 upgrade module for this adapter without bootstrapping the whole
|
||||||
|
* hybrid application.
|
||||||
|
*
|
||||||
|
* This method is automatically called by `bootstrap()`.
|
||||||
|
*
|
||||||
|
* @param modules The Angular 1 modules that this upgrade module should depend upon.
|
||||||
|
* @returns The Angular 1 upgrade module that is declared by this method
|
||||||
|
*
|
||||||
|
* ### Example
|
||||||
|
*
|
||||||
|
* ```
|
||||||
|
* const upgradeAdapter = new UpgradeAdapter();
|
||||||
|
* upgradeAdapter.declareNg1Module(['heroApp']);
|
||||||
|
* ```
|
||||||
|
*/
|
||||||
|
private declareNg1Module(modules: string[] = []): angular.IModule {
|
||||||
|
const delayApplyExps: Function[] = [];
|
||||||
|
let original$applyFn: Function;
|
||||||
|
let rootScopePrototype: any;
|
||||||
|
let rootScope: angular.IRootScopeService;
|
||||||
|
const componentFactoryRefMap: ComponentFactoryRefMap = {};
|
||||||
|
const upgradeAdapter = this;
|
||||||
|
const ng1Module = this.ng1Module = angular.module(this.idPrefix, modules);
|
||||||
|
const platformRef = platformBrowserDynamic();
|
||||||
|
|
||||||
|
this.ngZone = new NgZone({enableLongStackTrace: Zone.hasOwnProperty('longStackTraceZoneSpec')});
|
||||||
|
this.ng2BootstrapDeferred = new Deferred();
|
||||||
|
ng1Module.factory(NG2_INJECTOR, () => this.moduleRef.injector.get(Injector))
|
||||||
|
.constant(NG2_ZONE, this.ngZone)
|
||||||
|
.constant(NG2_COMPONENT_FACTORY_REF_MAP, componentFactoryRefMap)
|
||||||
|
.factory(NG2_COMPILER, () => this.moduleRef.injector.get(Compiler))
|
||||||
|
.config([
|
||||||
|
'$provide', '$injector',
|
||||||
|
(provide: angular.IProvideService, ng1Injector: angular.IInjectorService) => {
|
||||||
|
provide.decorator(NG1_ROOT_SCOPE, [
|
||||||
|
'$delegate',
|
||||||
|
function(rootScopeDelegate: angular.IRootScopeService) {
|
||||||
|
// Capture the root apply so that we can delay first call to $apply until we
|
||||||
|
// bootstrap Angular 2 and then we replay and restore the $apply.
|
||||||
|
rootScopePrototype = rootScopeDelegate.constructor.prototype;
|
||||||
|
if (rootScopePrototype.hasOwnProperty('$apply')) {
|
||||||
|
original$applyFn = rootScopePrototype.$apply;
|
||||||
|
rootScopePrototype.$apply = (exp: any) => delayApplyExps.push(exp);
|
||||||
|
} else {
|
||||||
|
throw new Error('Failed to find \'$apply\' on \'$rootScope\'!');
|
||||||
|
}
|
||||||
|
return rootScope = rootScopeDelegate;
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
if (ng1Injector.has(NG1_TESTABILITY)) {
|
||||||
|
provide.decorator(NG1_TESTABILITY, [
|
||||||
|
'$delegate',
|
||||||
|
function(testabilityDelegate: angular.ITestabilityService) {
|
||||||
|
const originalWhenStable: Function = testabilityDelegate.whenStable;
|
||||||
|
// Cannot use arrow function below because we need the context
|
||||||
|
const newWhenStable = function(callback: Function) {
|
||||||
|
originalWhenStable.call(this, function() {
|
||||||
|
const ng2Testability: Testability =
|
||||||
|
upgradeAdapter.moduleRef.injector.get(Testability);
|
||||||
|
if (ng2Testability.isStable()) {
|
||||||
|
callback.apply(this, arguments);
|
||||||
|
} else {
|
||||||
|
ng2Testability.whenStable(newWhenStable.bind(this, callback));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
testabilityDelegate.whenStable = newWhenStable;
|
||||||
|
return testabilityDelegate;
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
|
||||||
|
ng1Module.run([
|
||||||
|
'$injector', '$rootScope',
|
||||||
|
(ng1Injector: angular.IInjectorService, rootScope: angular.IRootScopeService) => {
|
||||||
|
UpgradeNg1ComponentAdapterBuilder.resolve(this.ng1ComponentsToBeUpgraded, ng1Injector)
|
||||||
|
.then(() => {
|
||||||
|
// At this point we have ng1 injector and we have lifted ng1 components into ng2, we
|
||||||
|
// now can bootstrap ng2.
|
||||||
|
const DynamicNgUpgradeModule =
|
||||||
|
NgModule({
|
||||||
|
providers: [
|
||||||
|
{provide: NG1_INJECTOR, useFactory: () => ng1Injector},
|
||||||
|
{provide: NG1_COMPILE, useFactory: () => ng1Injector.get(NG1_COMPILE)},
|
||||||
|
this.providers
|
||||||
|
],
|
||||||
|
imports: [this.ng2AppModule]
|
||||||
|
}).Class({
|
||||||
|
constructor: function DynamicNgUpgradeModule() {},
|
||||||
|
ngDoBootstrap: function() {}
|
||||||
|
});
|
||||||
|
(platformRef as any)
|
||||||
|
._bootstrapModuleWithZone(
|
||||||
|
DynamicNgUpgradeModule, this.compilerOptions, this.ngZone,
|
||||||
|
(componentFactories: ComponentFactory<any>[]) => {
|
||||||
|
componentFactories.forEach((componentFactory) => {
|
||||||
|
const type: Type<any> = componentFactory.componentType;
|
||||||
|
if (this.upgradedComponents.indexOf(type) !== -1) {
|
||||||
|
componentFactoryRefMap[getComponentInfo(type).selector] =
|
||||||
|
componentFactory;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.then((ref: NgModuleRef<any>) => {
|
||||||
|
this.moduleRef = ref;
|
||||||
|
let subscription = this.ngZone.onMicrotaskEmpty.subscribe({
|
||||||
|
next: (_: any) => this.ngZone.runOutsideAngular(() => rootScope.$evalAsync())
|
||||||
|
});
|
||||||
|
rootScope.$on('$destroy', () => { subscription.unsubscribe(); });
|
||||||
|
this.ngZone.run(() => {
|
||||||
|
if (rootScopePrototype) {
|
||||||
|
rootScopePrototype.$apply = original$applyFn; // restore original $apply
|
||||||
|
while (delayApplyExps.length) {
|
||||||
|
rootScope.$apply(delayApplyExps.shift());
|
||||||
|
}
|
||||||
|
rootScopePrototype = null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.then(() => this.ng2BootstrapDeferred.resolve(ng1Injector), onError);
|
||||||
|
})
|
||||||
|
.catch((e) => this.ng2BootstrapDeferred.reject(e));
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
|
||||||
|
return ng1Module;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
interface ComponentFactoryRefMap {
|
interface ComponentFactoryRefMap {
|
||||||
[selector: string]: ComponentFactory<any>;
|
[selector: string]: ComponentFactory<any>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Synchronous promise-like object to wrap parent injectors,
|
||||||
|
* to preserve the synchronous nature of AngularJS v1's $compile.
|
||||||
|
*/
|
||||||
|
class ParentInjectorPromise {
|
||||||
|
private injector: Injector;
|
||||||
|
private callbacks: ((injector: Injector) => any)[] = [];
|
||||||
|
|
||||||
|
constructor(private element: angular.IAugmentedJQuery) {
|
||||||
|
// store the promise on the element
|
||||||
|
element.data(controllerKey(NG2_INJECTOR), this);
|
||||||
|
}
|
||||||
|
|
||||||
|
then(callback: (injector: Injector) => any) {
|
||||||
|
if (this.injector) {
|
||||||
|
callback(this.injector);
|
||||||
|
} else {
|
||||||
|
this.callbacks.push(callback);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
resolve(injector: Injector) {
|
||||||
|
this.injector = injector;
|
||||||
|
|
||||||
|
// reset the element data to point to the real injector
|
||||||
|
this.element.data(controllerKey(NG2_INJECTOR), injector);
|
||||||
|
|
||||||
|
// clean out the element to prevent memory leaks
|
||||||
|
this.element = null;
|
||||||
|
|
||||||
|
// run all the queued callbacks
|
||||||
|
this.callbacks.forEach((callback) => callback(injector));
|
||||||
|
this.callbacks.length = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
function ng1ComponentDirective(info: ComponentInfo, idPrefix: string): Function {
|
function ng1ComponentDirective(info: ComponentInfo, idPrefix: string): Function {
|
||||||
(<any>directiveFactory).$inject = [NG1_INJECTOR, NG2_COMPONENT_FACTORY_REF_MAP, NG1_PARSE];
|
(<any>directiveFactory).$inject =
|
||||||
|
[NG1_INJECTOR, NG1_COMPILE, NG2_COMPONENT_FACTORY_REF_MAP, NG1_PARSE];
|
||||||
function directiveFactory(
|
function directiveFactory(
|
||||||
ng1Injector: angular.IInjectorService, componentFactoryRefMap: ComponentFactoryRefMap,
|
ng1Injector: angular.IInjectorService, ng1Compile: angular.ICompileService,
|
||||||
|
componentFactoryRefMap: ComponentFactoryRefMap,
|
||||||
parse: angular.IParseService): angular.IDirective {
|
parse: angular.IParseService): angular.IDirective {
|
||||||
let idCount = 0;
|
let idCount = 0;
|
||||||
|
let dashSelector = info.selector.replace(/[A-Z]/g, char => '-' + char.toLowerCase());
|
||||||
return {
|
return {
|
||||||
restrict: 'E',
|
restrict: 'E',
|
||||||
|
terminal: true,
|
||||||
require: REQUIRE_INJECTOR,
|
require: REQUIRE_INJECTOR,
|
||||||
link: {
|
compile: (templateElement: angular.IAugmentedJQuery, templateAttributes: angular.IAttributes,
|
||||||
post: (scope: angular.IScope, element: angular.IAugmentedJQuery, attrs: angular.IAttributes,
|
transclude: angular.ITranscludeFunction) => {
|
||||||
parentInjector: any, transclude: angular.ITranscludeFunction): void => {
|
// We might have compile the contents lazily, because this might have been triggered by the
|
||||||
const componentFactory: ComponentFactory<any> = componentFactoryRefMap[info.selector];
|
// UpgradeNg1ComponentAdapterBuilder, when the ng2 templates have not been compiled yet
|
||||||
if (!componentFactory)
|
return {
|
||||||
throw new Error('Expecting ComponentFactory for: ' + info.selector);
|
post: (scope: angular.IScope, element: angular.IAugmentedJQuery,
|
||||||
|
attrs: angular.IAttributes, parentInjector: Injector | ParentInjectorPromise,
|
||||||
|
transclude: angular.ITranscludeFunction): void => {
|
||||||
|
let id = idPrefix + (idCount++);
|
||||||
|
(<any>element[0]).id = id;
|
||||||
|
|
||||||
if (parentInjector === null) {
|
let injectorPromise = new ParentInjectorPromise(element);
|
||||||
parentInjector = ng1Injector.get(NG2_INJECTOR);
|
|
||||||
|
const ng2Compiler = ng1Injector.get(NG2_COMPILER) as Compiler;
|
||||||
|
const ngContentSelectors = ng2Compiler.getNgContentSelectors(info.type);
|
||||||
|
const linkFns = compileProjectedNodes(templateElement, ngContentSelectors);
|
||||||
|
|
||||||
|
const componentFactory: ComponentFactory<any> = componentFactoryRefMap[info.selector];
|
||||||
|
if (!componentFactory)
|
||||||
|
throw new Error('Expecting ComponentFactory for: ' + info.selector);
|
||||||
|
|
||||||
|
element.empty();
|
||||||
|
let projectableNodes = linkFns.map(link => {
|
||||||
|
let projectedClone: Node[];
|
||||||
|
link(scope, (clone: Node[]) => {
|
||||||
|
projectedClone = clone;
|
||||||
|
element.append(clone);
|
||||||
|
});
|
||||||
|
return projectedClone;
|
||||||
|
});
|
||||||
|
|
||||||
|
parentInjector = parentInjector || ng1Injector.get(NG2_INJECTOR);
|
||||||
|
|
||||||
|
if (parentInjector instanceof ParentInjectorPromise) {
|
||||||
|
parentInjector.then((resolvedInjector: Injector) => downgrade(resolvedInjector));
|
||||||
|
} else {
|
||||||
|
downgrade(parentInjector);
|
||||||
|
}
|
||||||
|
|
||||||
|
function downgrade(injector: Injector) {
|
||||||
|
const facade = new DowngradeNg2ComponentAdapter(
|
||||||
|
info, element, attrs, scope, injector, parse, componentFactory);
|
||||||
|
facade.setupInputs();
|
||||||
|
facade.bootstrapNg2(projectableNodes);
|
||||||
|
facade.setupOutputs();
|
||||||
|
facade.registerCleanup();
|
||||||
|
injectorPromise.resolve(facade.componentRef.injector);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
const facade = new DowngradeNg2ComponentAdapter(
|
};
|
||||||
idPrefix + (idCount++), info, element, attrs, scope, <Injector>parentInjector, parse,
|
|
||||||
componentFactory);
|
|
||||||
facade.setupInputs();
|
|
||||||
facade.bootstrapNg2();
|
|
||||||
facade.projectContent();
|
|
||||||
facade.setupOutputs();
|
|
||||||
facade.registerCleanup();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function compileProjectedNodes(
|
||||||
|
templateElement: angular.IAugmentedJQuery,
|
||||||
|
ngContentSelectors: string[]): angular.ILinkFn[] {
|
||||||
|
if (!ngContentSelectors)
|
||||||
|
throw new Error('Expecting ngContentSelectors for: ' + info.selector);
|
||||||
|
// We have to sort the projected content before we compile it, hence the terminal: true
|
||||||
|
let projectableTemplateNodes =
|
||||||
|
sortProjectableNodes(ngContentSelectors, templateElement.contents());
|
||||||
|
return projectableTemplateNodes.map(nodes => ng1Compile(nodes));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return directiveFactory;
|
return directiveFactory;
|
||||||
}
|
}
|
||||||
|
@ -602,3 +709,36 @@ export class UpgradeAdapterRef {
|
||||||
this.ng2ModuleRef.destroy();
|
this.ng2ModuleRef.destroy();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sort a set of DOM nodes that into groups based on the given content selectors
|
||||||
|
*/
|
||||||
|
export function sortProjectableNodes(ngContentSelectors: string[], childNodes: Node[]): Node[][] {
|
||||||
|
let projectableNodes: Node[][] = [];
|
||||||
|
let matcher = new SelectorMatcher();
|
||||||
|
let wildcardNgContentIndex: number;
|
||||||
|
for (let i = 0, ii = ngContentSelectors.length; i < ii; i++) {
|
||||||
|
projectableNodes[i] = [];
|
||||||
|
if (ngContentSelectors[i] === '*') {
|
||||||
|
wildcardNgContentIndex = i;
|
||||||
|
} else {
|
||||||
|
matcher.addSelectables(CssSelector.parse(ngContentSelectors[i]), i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (let node of childNodes) {
|
||||||
|
let ngContentIndices: number[] = [];
|
||||||
|
let selector =
|
||||||
|
createElementCssSelector(node.nodeName.toLowerCase(), getAttributesAsArray(node));
|
||||||
|
matcher.match(
|
||||||
|
selector, (selector, ngContentIndex) => { ngContentIndices.push(ngContentIndex); });
|
||||||
|
ngContentIndices.sort();
|
||||||
|
if (wildcardNgContentIndex !== undefined) {
|
||||||
|
ngContentIndices.push(wildcardNgContentIndex);
|
||||||
|
}
|
||||||
|
if (ngContentIndices.length > 0) {
|
||||||
|
projectableNodes[ngContentIndices[0]].push(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return projectableNodes;
|
||||||
|
}
|
||||||
|
|
|
@ -6,8 +6,6 @@
|
||||||
* found in the LICENSE file at https://angular.io/license
|
* found in the LICENSE file at https://angular.io/license
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
export function onError(e: any) {
|
export function onError(e: any) {
|
||||||
// TODO: (misko): We seem to not have a stack trace here!
|
// TODO: (misko): We seem to not have a stack trace here!
|
||||||
if (console.error) {
|
if (console.error) {
|
||||||
|
@ -23,6 +21,19 @@ export function controllerKey(name: string): string {
|
||||||
return '$' + name + 'Controller';
|
return '$' + name + 'Controller';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getAttributesAsArray(node: Node): string[][] {
|
||||||
|
const attributes = node.attributes;
|
||||||
|
let asArray: string[][];
|
||||||
|
if (attributes) {
|
||||||
|
let attrLen = attributes.length;
|
||||||
|
asArray = new Array(attrLen);
|
||||||
|
for (let i = 0; i < attrLen; i++) {
|
||||||
|
asArray[i] = [attributes[i].nodeName, attributes[i].nodeValue];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return asArray || [];
|
||||||
|
}
|
||||||
|
|
||||||
export class Deferred<R> {
|
export class Deferred<R> {
|
||||||
promise: Promise<R>;
|
promise: Promise<R>;
|
||||||
resolve: (value?: R|PromiseLike<R>) => void;
|
resolve: (value?: R|PromiseLike<R>) => void;
|
||||||
|
@ -34,4 +45,4 @@ export class Deferred<R> {
|
||||||
this.reject = rej;
|
this.reject = rej;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,8 +10,8 @@ import {ChangeDetectorRef, Class, Component, EventEmitter, NO_ERRORS_SCHEMA, NgM
|
||||||
import {async, fakeAsync, flushMicrotasks, tick} from '@angular/core/testing';
|
import {async, fakeAsync, flushMicrotasks, tick} from '@angular/core/testing';
|
||||||
import {BrowserModule} from '@angular/platform-browser';
|
import {BrowserModule} from '@angular/platform-browser';
|
||||||
import {platformBrowserDynamic} from '@angular/platform-browser-dynamic';
|
import {platformBrowserDynamic} from '@angular/platform-browser-dynamic';
|
||||||
import {UpgradeAdapter} from '@angular/upgrade';
|
|
||||||
import * as angular from '@angular/upgrade/src/angular_js';
|
import * as angular from '@angular/upgrade/src/angular_js';
|
||||||
|
import {UpgradeAdapter, sortProjectableNodes} from '@angular/upgrade/src/upgrade_adapter';
|
||||||
|
|
||||||
export function main() {
|
export function main() {
|
||||||
describe('adapter: ng1 to ng2', () => {
|
describe('adapter: ng1 to ng2', () => {
|
||||||
|
@ -178,7 +178,7 @@ export function main() {
|
||||||
adapter.bootstrap(element, ['ng1']).ready((ref) => {
|
adapter.bootstrap(element, ['ng1']).ready((ref) => {
|
||||||
expect(document.body.textContent).toEqual('1A;2A;ng1a;2B;ng1b;2C;1C;');
|
expect(document.body.textContent).toEqual('1A;2A;ng1a;2B;ng1b;2C;1C;');
|
||||||
// https://github.com/angular/angular.js/issues/12983
|
// https://github.com/angular/angular.js/issues/12983
|
||||||
expect(log).toEqual(['1A', '1B', '1C', '2A', '2B', '2C', 'ng1a', 'ng1b']);
|
expect(log).toEqual(['1A', '1C', '2A', '2B', '2C', 'ng1a', 'ng1b']);
|
||||||
ref.dispose();
|
ref.dispose();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
@ -359,6 +359,33 @@ export function main() {
|
||||||
ref.dispose();
|
ref.dispose();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
it('should support multi-slot projection', async(() => {
|
||||||
|
const ng1Module = angular.module('ng1', []);
|
||||||
|
|
||||||
|
const Ng2 = Component({
|
||||||
|
selector: 'ng2',
|
||||||
|
template: '2a(<ng-content select=".ng1a"></ng-content>)' +
|
||||||
|
'2b(<ng-content select=".ng1b"></ng-content>)'
|
||||||
|
}).Class({constructor: function() {}});
|
||||||
|
|
||||||
|
const Ng2Module = NgModule({declarations: [Ng2], imports: [BrowserModule]}).Class({
|
||||||
|
constructor: function() {}
|
||||||
|
});
|
||||||
|
|
||||||
|
// The ng-if on one of the projected children is here to make sure
|
||||||
|
// the correct slot is targeted even with structural directives in play.
|
||||||
|
const element = html(
|
||||||
|
'<ng2><div ng-if="true" class="ng1a">1a</div><div' +
|
||||||
|
' class="ng1b">1b</div></ng2>');
|
||||||
|
|
||||||
|
const adapter: UpgradeAdapter = new UpgradeAdapter(Ng2Module);
|
||||||
|
ng1Module.directive('ng2', adapter.downgradeNg2Component(Ng2));
|
||||||
|
adapter.bootstrap(element, ['ng1']).ready((ref) => {
|
||||||
|
expect(document.body.textContent).toEqual('2a(1a)2b(1b)');
|
||||||
|
ref.dispose();
|
||||||
|
});
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('upgrade ng1 component', () => {
|
describe('upgrade ng1 component', () => {
|
||||||
|
@ -1128,6 +1155,33 @@ export function main() {
|
||||||
ref.dispose();
|
ref.dispose();
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
it('should respect hierarchical dependency injection for ng2', async(() => {
|
||||||
|
const ng1Module = angular.module('ng1', []);
|
||||||
|
|
||||||
|
const Ng2Parent = Component({
|
||||||
|
selector: 'ng2-parent',
|
||||||
|
template: `ng2-parent(<ng-content></ng-content>)`
|
||||||
|
}).Class({constructor: function() {}});
|
||||||
|
const Ng2Child = Component({selector: 'ng2-child', template: `ng2-child`}).Class({
|
||||||
|
constructor: [Ng2Parent, function(parent: any) {}]
|
||||||
|
});
|
||||||
|
|
||||||
|
const Ng2Module =
|
||||||
|
NgModule({declarations: [Ng2Parent, Ng2Child], imports: [BrowserModule]}).Class({
|
||||||
|
constructor: function() {}
|
||||||
|
});
|
||||||
|
|
||||||
|
const element = html('<ng2-parent><ng2-child></ng2-child></ng2-parent>');
|
||||||
|
|
||||||
|
const adapter: UpgradeAdapter = new UpgradeAdapter(Ng2Module);
|
||||||
|
ng1Module.directive('ng2Parent', adapter.downgradeNg2Component(Ng2Parent))
|
||||||
|
.directive('ng2Child', adapter.downgradeNg2Component(Ng2Child));
|
||||||
|
adapter.bootstrap(element, ['ng1']).ready((ref) => {
|
||||||
|
expect(document.body.textContent).toEqual('ng2-parent(ng2-child)');
|
||||||
|
ref.dispose();
|
||||||
|
});
|
||||||
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('testability', () => {
|
describe('testability', () => {
|
||||||
|
@ -1241,6 +1295,70 @@ export function main() {
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('sortProjectableNodes', () => {
|
||||||
|
it('should return an array of node collections for each selector', () => {
|
||||||
|
const contentNodes = nodes(
|
||||||
|
'<div class="x"><span>div-1 content</span></div>' +
|
||||||
|
'<input type="number" name="myNum">' +
|
||||||
|
'<input type="date" name="myDate">' +
|
||||||
|
'<span>span content</span>' +
|
||||||
|
'<div class="x"><span>div-2 content</span></div>');
|
||||||
|
|
||||||
|
const selectors = ['input[type=date]', 'span', '.x'];
|
||||||
|
const projectableNodes = sortProjectableNodes(selectors, contentNodes);
|
||||||
|
|
||||||
|
expect(projectableNodes[0]).toEqual(nodes('<input type="date" name="myDate">'));
|
||||||
|
expect(projectableNodes[1]).toEqual(nodes('<span>span content</span>'));
|
||||||
|
expect(projectableNodes[2])
|
||||||
|
.toEqual(nodes(
|
||||||
|
'<div class="x"><span>div-1 content</span></div>' +
|
||||||
|
'<div class="x"><span>div-2 content</span></div>'));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should collect up unmatched nodes for the wildcard selector', () => {
|
||||||
|
const contentNodes = nodes(
|
||||||
|
'<div class="x"><span>div-1 content</span></div>' +
|
||||||
|
'<input type="number" name="myNum">' +
|
||||||
|
'<input type="date" name="myDate">' +
|
||||||
|
'<span>span content</span>' +
|
||||||
|
'<div class="x"><span>div-2 content</span></div>');
|
||||||
|
|
||||||
|
const selectors = ['.x', '*', 'input[type=date]'];
|
||||||
|
const projectableNodes = sortProjectableNodes(selectors, contentNodes);
|
||||||
|
|
||||||
|
expect(projectableNodes[0])
|
||||||
|
.toEqual(nodes(
|
||||||
|
'<div class="x"><span>div-1 content</span></div>' +
|
||||||
|
'<div class="x"><span>div-2 content</span></div>'));
|
||||||
|
expect(projectableNodes[1])
|
||||||
|
.toEqual(nodes(
|
||||||
|
'<input type="number" name="myNum">' +
|
||||||
|
'<span>span content</span>'));
|
||||||
|
expect(projectableNodes[2]).toEqual(nodes('<input type="date" name="myDate">'));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return an array of empty arrays if there are no nodes passed in', () => {
|
||||||
|
const selectors = ['.x', '*', 'input[type=date]'];
|
||||||
|
const projectableNodes = sortProjectableNodes(selectors, []);
|
||||||
|
expect(projectableNodes).toEqual([[], [], []]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return an empty array for each selector that does not match', () => {
|
||||||
|
const contentNodes = nodes(
|
||||||
|
'<div class="x"><span>div-1 content</span></div>' +
|
||||||
|
'<input type="number" name="myNum">' +
|
||||||
|
'<input type="date" name="myDate">' +
|
||||||
|
'<span>span content</span>' +
|
||||||
|
'<div class="x"><span>div-2 content</span></div>');
|
||||||
|
|
||||||
|
const noSelectorNodes = sortProjectableNodes([], contentNodes);
|
||||||
|
expect(noSelectorNodes).toEqual([]);
|
||||||
|
|
||||||
|
const noMatchSelectorNodes = sortProjectableNodes(['.not-there'], contentNodes);
|
||||||
|
expect(noMatchSelectorNodes).toEqual([[]]);
|
||||||
|
});
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function multiTrim(text: string): string {
|
function multiTrim(text: string): string {
|
||||||
|
@ -1257,3 +1375,9 @@ function html(html: string): Element {
|
||||||
|
|
||||||
return body;
|
return body;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function nodes(html: string) {
|
||||||
|
const element = document.createElement('div');
|
||||||
|
element.innerHTML = html;
|
||||||
|
return Array.prototype.slice.call(element.childNodes);
|
||||||
|
}
|
||||||
|
|
|
@ -219,6 +219,7 @@ export declare class Compiler {
|
||||||
compileModuleAndAllComponentsSync<T>(moduleType: Type<T>): ModuleWithComponentFactories<T>;
|
compileModuleAndAllComponentsSync<T>(moduleType: Type<T>): ModuleWithComponentFactories<T>;
|
||||||
compileModuleAsync<T>(moduleType: Type<T>): Promise<NgModuleFactory<T>>;
|
compileModuleAsync<T>(moduleType: Type<T>): Promise<NgModuleFactory<T>>;
|
||||||
compileModuleSync<T>(moduleType: Type<T>): NgModuleFactory<T>;
|
compileModuleSync<T>(moduleType: Type<T>): NgModuleFactory<T>;
|
||||||
|
getNgContentSelectors(component: Type<any>): string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @experimental */
|
/** @experimental */
|
||||||
|
|
Loading…
Reference in New Issue