2016-10-19 16:41:04 -04:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2016-10-19 16:41:04 -04: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
|
|
|
|
*/
|
|
|
|
|
2020-04-13 19:40:21 -04:00
|
|
|
import {Component, destroyPlatform, Directive, ElementRef, Injector, Input, NgModule} from '@angular/core';
|
2020-07-31 15:43:18 -04:00
|
|
|
import {waitForAsync} from '@angular/core/testing';
|
2016-10-19 16:41:04 -04:00
|
|
|
import {BrowserModule} from '@angular/platform-browser';
|
|
|
|
import {platformBrowserDynamic} from '@angular/platform-browser-dynamic';
|
2020-04-13 19:40:21 -04:00
|
|
|
import {downgradeComponent, UpgradeComponent, UpgradeModule} from '@angular/upgrade/static';
|
2016-10-19 16:41:04 -04:00
|
|
|
|
2020-04-13 19:40:21 -04:00
|
|
|
import * as angular from '../../../src/common/src/angular1';
|
2019-03-22 05:42:52 -04:00
|
|
|
import {html, multiTrim, withEachNg1Version} from '../../../src/common/test/helpers/common_test_helpers';
|
2020-04-13 19:40:21 -04:00
|
|
|
|
2019-03-22 05:42:52 -04:00
|
|
|
import {bootstrap} from './static_test_helpers';
|
2016-10-19 16:41:04 -04:00
|
|
|
|
2018-02-15 12:21:18 -05:00
|
|
|
withEachNg1Version(() => {
|
2016-10-19 16:41:04 -04:00
|
|
|
describe('content projection', () => {
|
|
|
|
beforeEach(() => destroyPlatform());
|
|
|
|
afterEach(() => destroyPlatform());
|
|
|
|
|
2020-07-31 15:43:18 -04:00
|
|
|
it('should instantiate ng2 in ng1 template and project content', waitForAsync(() => {
|
2018-12-22 11:02:34 -05:00
|
|
|
// the ng2 component that will be used in ng1 (downgraded)
|
|
|
|
@Component({selector: 'ng2', template: `{{ prop }}(<ng-content></ng-content>)`})
|
|
|
|
class Ng2Component {
|
|
|
|
prop = 'NG2';
|
|
|
|
ngContent = 'ng2-content';
|
|
|
|
}
|
|
|
|
|
|
|
|
// our upgrade module to host the component to downgrade
|
|
|
|
@NgModule({
|
|
|
|
imports: [BrowserModule, UpgradeModule],
|
|
|
|
declarations: [Ng2Component],
|
|
|
|
entryComponents: [Ng2Component]
|
|
|
|
})
|
|
|
|
class Ng2Module {
|
|
|
|
ngDoBootstrap() {}
|
|
|
|
}
|
|
|
|
|
|
|
|
// the ng1 app module that will consume the downgraded component
|
|
|
|
const ng1Module = angular
|
2019-04-23 11:00:04 -04:00
|
|
|
.module_('ng1', [])
|
2018-12-22 11:02:34 -05:00
|
|
|
// create an ng1 facade of the ng2 component
|
|
|
|
.directive('ng2', downgradeComponent({component: Ng2Component}))
|
|
|
|
.run(($rootScope: angular.IRootScopeService) => {
|
|
|
|
$rootScope['prop'] = 'NG1';
|
|
|
|
$rootScope['ngContent'] = 'ng1-content';
|
|
|
|
});
|
|
|
|
|
|
|
|
const element = html('<div>{{ \'ng1[\' }}<ng2>~{{ ngContent }}~</ng2>{{ \']\' }}</div>');
|
|
|
|
|
|
|
|
bootstrap(platformBrowserDynamic(), Ng2Module, element, ng1Module).then((upgrade) => {
|
|
|
|
expect(document.body.textContent).toEqual('ng1[NG2(~ng1-content~)]');
|
|
|
|
});
|
|
|
|
}));
|
|
|
|
|
2020-07-31 15:43:18 -04:00
|
|
|
it('should correctly project structural directives', waitForAsync(() => {
|
2018-12-22 11:02:34 -05:00
|
|
|
@Component({selector: 'ng2', template: 'ng2-{{ itemId }}(<ng-content></ng-content>)'})
|
|
|
|
class Ng2Component {
|
|
|
|
// TODO(issue/24571): remove '!'.
|
2020-04-13 19:40:21 -04:00
|
|
|
@Input() itemId!: string;
|
2018-12-22 11:02:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
imports: [BrowserModule, UpgradeModule],
|
|
|
|
declarations: [Ng2Component],
|
|
|
|
entryComponents: [Ng2Component]
|
|
|
|
})
|
|
|
|
class Ng2Module {
|
|
|
|
ngDoBootstrap() {}
|
|
|
|
}
|
|
|
|
|
2019-04-23 11:00:04 -04:00
|
|
|
const ng1Module = angular.module_('ng1', [])
|
2018-12-22 11:02:34 -05:00
|
|
|
.directive('ng2', downgradeComponent({component: Ng2Component}))
|
|
|
|
.run(($rootScope: angular.IRootScopeService) => {
|
|
|
|
$rootScope['items'] = [
|
|
|
|
{id: 'a', subitems: [1, 2, 3]}, {id: 'b', subitems: [4, 5, 6]},
|
|
|
|
{id: 'c', subitems: [7, 8, 9]}
|
|
|
|
];
|
|
|
|
});
|
|
|
|
|
|
|
|
const element = html(`
|
2017-02-03 05:11:16 -05:00
|
|
|
<ng2 ng-repeat="item in items" [item-id]="item.id">
|
|
|
|
<div ng-repeat="subitem in item.subitems">{{ subitem }}</div>
|
|
|
|
</ng2>
|
|
|
|
`);
|
|
|
|
|
2018-12-22 11:02:34 -05:00
|
|
|
bootstrap(platformBrowserDynamic(), Ng2Module, element, ng1Module).then(upgrade => {
|
|
|
|
expect(multiTrim(document.body.textContent))
|
|
|
|
.toBe('ng2-a( 123 )ng2-b( 456 )ng2-c( 789 )');
|
|
|
|
});
|
|
|
|
}));
|
2017-02-03 05:11:16 -05:00
|
|
|
|
2020-07-31 15:43:18 -04:00
|
|
|
it('should instantiate ng1 in ng2 template and project content', waitForAsync(() => {
|
2016-10-19 16:41:04 -04:00
|
|
|
@Component({
|
|
|
|
selector: 'ng2',
|
2017-01-16 16:52:42 -05:00
|
|
|
template: `{{ 'ng2(' }}<ng1>{{ transclude }}</ng1>{{ ')' }}`,
|
2016-10-19 16:41:04 -04:00
|
|
|
})
|
|
|
|
class Ng2Component {
|
2017-01-16 16:52:42 -05:00
|
|
|
prop = 'ng2';
|
|
|
|
transclude = 'ng2-transclude';
|
2016-10-19 16:41:04 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
@Directive({selector: 'ng1'})
|
|
|
|
class Ng1WrapperComponent extends UpgradeComponent {
|
|
|
|
constructor(elementRef: ElementRef, injector: Injector) {
|
|
|
|
super('ng1', elementRef, injector);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [Ng1WrapperComponent, Ng2Component],
|
|
|
|
entryComponents: [Ng2Component],
|
|
|
|
imports: [BrowserModule, UpgradeModule]
|
|
|
|
})
|
|
|
|
class Ng2Module {
|
|
|
|
ngDoBootstrap() {}
|
|
|
|
}
|
|
|
|
|
2017-01-16 16:52:42 -05:00
|
|
|
const ng1Module =
|
2019-04-23 11:00:04 -04:00
|
|
|
angular.module_('ng1', [])
|
2017-01-16 16:52:42 -05:00
|
|
|
.directive('ng1', () => ({
|
|
|
|
transclude: true,
|
|
|
|
template: '{{ prop }}(<ng-transclude></ng-transclude>)'
|
|
|
|
}))
|
|
|
|
.directive('ng2', downgradeComponent({component: Ng2Component}))
|
|
|
|
.run(($rootScope: angular.IRootScopeService) => {
|
|
|
|
$rootScope['prop'] = 'ng1';
|
|
|
|
$rootScope['transclude'] = 'ng1-transclude';
|
|
|
|
});
|
2016-10-19 16:41:04 -04:00
|
|
|
|
2017-01-16 16:52:42 -05:00
|
|
|
const element = html('<div>{{ \'ng1(\' }}<ng2></ng2>{{ \')\' }}</div>');
|
2016-10-19 16:41:04 -04:00
|
|
|
|
|
|
|
bootstrap(platformBrowserDynamic(), Ng2Module, element, ng1Module).then((upgrade) => {
|
2017-01-16 16:52:42 -05:00
|
|
|
expect(document.body.textContent).toEqual('ng1(ng2(ng1(ng2-transclude)))');
|
2016-10-19 16:41:04 -04:00
|
|
|
});
|
|
|
|
}));
|
2017-03-13 20:34:53 -04:00
|
|
|
|
2020-07-31 15:43:18 -04:00
|
|
|
it('should support multi-slot projection', waitForAsync(() => {
|
2018-12-22 11:02:34 -05:00
|
|
|
@Component({
|
|
|
|
selector: 'ng2',
|
|
|
|
template: '2a(<ng-content select=".ng1a"></ng-content>)' +
|
|
|
|
'2b(<ng-content select=".ng1b"></ng-content>)'
|
|
|
|
})
|
|
|
|
class Ng2Component {
|
|
|
|
constructor() {}
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
declarations: [Ng2Component],
|
|
|
|
entryComponents: [Ng2Component],
|
|
|
|
imports: [BrowserModule, UpgradeModule]
|
|
|
|
})
|
|
|
|
class Ng2Module {
|
|
|
|
ngDoBootstrap() {}
|
|
|
|
}
|
|
|
|
|
2019-04-23 11:00:04 -04:00
|
|
|
const ng1Module = angular.module_('ng1', []).directive(
|
2018-12-22 11:02:34 -05:00
|
|
|
'ng2', downgradeComponent({component: Ng2Component}));
|
|
|
|
|
|
|
|
// 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>');
|
|
|
|
|
|
|
|
bootstrap(platformBrowserDynamic(), Ng2Module, element, ng1Module).then((upgrade) => {
|
|
|
|
expect(document.body.textContent).toEqual('2a(1a)2b(1b)');
|
|
|
|
});
|
|
|
|
}));
|
2016-10-19 16:41:04 -04:00
|
|
|
});
|
2018-02-15 12:21:18 -05:00
|
|
|
});
|