2019-11-13 00:32:58 -05:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2019-11-13 00:32:58 -05: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 {COMPILER_OPTIONS, Component, destroyPlatform, NgModule, ViewEncapsulation} from '@angular/core';
|
2019-11-13 00:32:58 -05:00
|
|
|
import {BrowserModule} from '@angular/platform-browser';
|
|
|
|
import {platformBrowserDynamic} from '@angular/platform-browser-dynamic';
|
2020-02-18 13:55:55 -05:00
|
|
|
import {onlyInIvy, withBody} from '@angular/private/testing';
|
2019-11-13 00:32:58 -05:00
|
|
|
|
|
|
|
describe('bootstrap', () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
beforeEach(destroyPlatform);
|
|
|
|
afterEach(destroyPlatform);
|
|
|
|
|
2019-12-18 08:35:22 -05:00
|
|
|
it('should bootstrap using #id selector',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<div>before|</div><button id="my-app"></button>', async () => {
|
2019-11-13 00:32:58 -05:00
|
|
|
try {
|
2019-12-18 08:35:22 -05:00
|
|
|
const ngModuleRef = await platformBrowserDynamic().bootstrapModule(IdSelectorAppModule);
|
|
|
|
expect(document.body.textContent).toEqual('before|works!');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
} catch (err) {
|
|
|
|
console.error(err);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should bootstrap using one of selectors from the list',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<div>before|</div><div class="bar"></div>', async () => {
|
2019-12-18 08:35:22 -05:00
|
|
|
try {
|
|
|
|
const ngModuleRef =
|
|
|
|
await platformBrowserDynamic().bootstrapModule(MultipleSelectorsAppModule);
|
|
|
|
expect(document.body.textContent).toEqual('before|works!');
|
2019-11-13 00:32:58 -05:00
|
|
|
ngModuleRef.destroy();
|
|
|
|
} catch (err) {
|
|
|
|
console.error(err);
|
|
|
|
}
|
|
|
|
}));
|
2020-02-18 13:55:55 -05:00
|
|
|
|
|
|
|
describe('options', () => {
|
|
|
|
function createComponentAndModule(
|
|
|
|
options: {encapsulation?: ViewEncapsulation; preserveWhitespaces?: boolean} = {}) {
|
|
|
|
@Component({
|
|
|
|
selector: 'my-app',
|
|
|
|
styles: [''],
|
|
|
|
template: '<span>a b</span>',
|
|
|
|
encapsulation: options.encapsulation,
|
|
|
|
preserveWhitespaces: options.preserveWhitespaces,
|
|
|
|
jit: true,
|
|
|
|
})
|
|
|
|
class TestComponent {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
imports: [BrowserModule],
|
|
|
|
declarations: [TestComponent],
|
|
|
|
bootstrap: [TestComponent],
|
|
|
|
jit: true,
|
|
|
|
})
|
|
|
|
class TestModule {
|
|
|
|
}
|
|
|
|
|
|
|
|
return TestModule;
|
|
|
|
}
|
|
|
|
|
|
|
|
it('should use ViewEncapsulation.Emulated as default',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
|
|
|
|
const ngModuleRef = await platformBrowserDynamic().bootstrapModule(TestModule);
|
|
|
|
expect(document.body.innerHTML).toContain('<span _ngcontent-');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should allow setting defaultEncapsulation using bootstrap option',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
|
|
|
|
const ngModuleRef = await platformBrowserDynamic().bootstrapModule(
|
|
|
|
TestModule, {defaultEncapsulation: ViewEncapsulation.None});
|
|
|
|
expect(document.body.innerHTML).toContain('<span>');
|
|
|
|
expect(document.body.innerHTML).not.toContain('_ngcontent-');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should allow setting defaultEncapsulation using compiler option',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
|
|
|
|
const ngModuleRef = await platformBrowserDynamic([{
|
|
|
|
provide: COMPILER_OPTIONS,
|
|
|
|
useValue: {defaultEncapsulation: ViewEncapsulation.None},
|
|
|
|
multi: true
|
|
|
|
}]).bootstrapModule(TestModule);
|
|
|
|
expect(document.body.innerHTML).toContain('<span>');
|
|
|
|
expect(document.body.innerHTML).not.toContain('_ngcontent-');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should prefer encapsulation on component over bootstrap option',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule({encapsulation: ViewEncapsulation.Emulated});
|
|
|
|
|
|
|
|
const ngModuleRef = await platformBrowserDynamic().bootstrapModule(
|
|
|
|
TestModule, {defaultEncapsulation: ViewEncapsulation.None});
|
|
|
|
expect(document.body.innerHTML).toContain('<span _ngcontent-');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should use preserveWhitespaces: false as default',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
|
|
|
|
const ngModuleRef = await platformBrowserDynamic().bootstrapModule(TestModule);
|
|
|
|
expect(document.body.innerHTML).toContain('a b');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should allow setting preserveWhitespaces using bootstrap option',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
|
|
|
|
const ngModuleRef = await platformBrowserDynamic().bootstrapModule(
|
|
|
|
TestModule, {preserveWhitespaces: true});
|
|
|
|
expect(document.body.innerHTML).toContain('a b');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should allow setting preserveWhitespaces using compiler option',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
|
|
|
|
const ngModuleRef =
|
|
|
|
await platformBrowserDynamic([
|
|
|
|
{provide: COMPILER_OPTIONS, useValue: {preserveWhitespaces: true}, multi: true}
|
|
|
|
]).bootstrapModule(TestModule);
|
|
|
|
expect(document.body.innerHTML).toContain('a b');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should prefer preserveWhitespaces on component over bootstrap option',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule({preserveWhitespaces: false});
|
|
|
|
|
|
|
|
const ngModuleRef = await platformBrowserDynamic().bootstrapModule(
|
|
|
|
TestModule, {preserveWhitespaces: true});
|
|
|
|
expect(document.body.innerHTML).toContain('a b');
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
onlyInIvy('options cannot be changed in Ivy').describe('changing bootstrap options', () => {
|
2020-04-13 19:40:21 -04:00
|
|
|
beforeEach(() => {
|
|
|
|
spyOn(console, 'error');
|
|
|
|
});
|
2020-02-18 13:55:55 -05:00
|
|
|
|
|
|
|
it('should log an error when changing defaultEncapsulation bootstrap options',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
const platformRef = platformBrowserDynamic();
|
|
|
|
|
|
|
|
const ngModuleRef = await platformRef.bootstrapModule(
|
|
|
|
TestModule, {defaultEncapsulation: ViewEncapsulation.None});
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
|
|
|
|
const ngModuleRef2 = await platformRef.bootstrapModule(
|
|
|
|
TestModule, {defaultEncapsulation: ViewEncapsulation.ShadowDom});
|
|
|
|
expect(console.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
'Provided value for `defaultEncapsulation` can not be changed once it has been set.');
|
|
|
|
|
|
|
|
// The options should not have been changed
|
|
|
|
expect(document.body.innerHTML).not.toContain('_ngcontent-');
|
|
|
|
|
|
|
|
ngModuleRef2.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should log an error when changing preserveWhitespaces bootstrap options',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
const platformRef = platformBrowserDynamic();
|
|
|
|
|
|
|
|
const ngModuleRef =
|
|
|
|
await platformRef.bootstrapModule(TestModule, {preserveWhitespaces: true});
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
|
|
|
|
const ngModuleRef2 =
|
|
|
|
await platformRef.bootstrapModule(TestModule, {preserveWhitespaces: false});
|
|
|
|
expect(console.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
'Provided value for `preserveWhitespaces` can not be changed once it has been set.');
|
|
|
|
|
|
|
|
// The options should not have been changed
|
|
|
|
expect(document.body.innerHTML).toContain('a b');
|
|
|
|
|
|
|
|
ngModuleRef2.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should log an error when changing defaultEncapsulation to its default',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
const platformRef = platformBrowserDynamic();
|
|
|
|
|
|
|
|
const ngModuleRef = await platformRef.bootstrapModule(TestModule);
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
|
|
|
|
const ngModuleRef2 = await platformRef.bootstrapModule(
|
|
|
|
TestModule, {defaultEncapsulation: ViewEncapsulation.Emulated});
|
|
|
|
// Although the configured value may be identical to the default, the provided set of
|
|
|
|
// options has still been changed compared to the previously provided options.
|
|
|
|
expect(console.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
'Provided value for `defaultEncapsulation` can not be changed once it has been set.');
|
|
|
|
|
|
|
|
ngModuleRef2.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should log an error when changing preserveWhitespaces to its default',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
const platformRef = platformBrowserDynamic();
|
|
|
|
|
|
|
|
const ngModuleRef = await platformRef.bootstrapModule(TestModule);
|
|
|
|
ngModuleRef.destroy();
|
|
|
|
|
|
|
|
const ngModuleRef2 =
|
|
|
|
await platformRef.bootstrapModule(TestModule, {preserveWhitespaces: false});
|
|
|
|
// Although the configured value may be identical to the default, the provided set of
|
|
|
|
// options has still been changed compared to the previously provided options.
|
|
|
|
expect(console.error)
|
|
|
|
.toHaveBeenCalledWith(
|
|
|
|
'Provided value for `preserveWhitespaces` can not be changed once it has been set.');
|
|
|
|
|
|
|
|
ngModuleRef2.destroy();
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should not log an error when passing identical bootstrap options',
|
2020-04-13 19:40:21 -04:00
|
|
|
withBody('<my-app></my-app>', async () => {
|
2020-02-18 13:55:55 -05:00
|
|
|
const TestModule = createComponentAndModule();
|
|
|
|
const platformRef = platformBrowserDynamic();
|
|
|
|
|
|
|
|
const ngModuleRef1 = await platformRef.bootstrapModule(
|
|
|
|
TestModule,
|
|
|
|
{defaultEncapsulation: ViewEncapsulation.None, preserveWhitespaces: true});
|
|
|
|
ngModuleRef1.destroy();
|
|
|
|
|
|
|
|
// Bootstrapping multiple modules using the exact same options should be allowed.
|
|
|
|
const ngModuleRef2 = await platformRef.bootstrapModule(
|
|
|
|
TestModule,
|
|
|
|
{defaultEncapsulation: ViewEncapsulation.None, preserveWhitespaces: true});
|
|
|
|
ngModuleRef2.destroy();
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
});
|
2019-11-13 00:32:58 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
selector: '#my-app',
|
|
|
|
template: 'works!',
|
|
|
|
})
|
2019-12-18 08:35:22 -05:00
|
|
|
export class IdSelectorAppComponent {
|
2019-11-13 00:32:58 -05:00
|
|
|
}
|
|
|
|
|
2019-12-18 08:35:22 -05:00
|
|
|
@NgModule({
|
|
|
|
imports: [BrowserModule],
|
|
|
|
declarations: [IdSelectorAppComponent],
|
|
|
|
bootstrap: [IdSelectorAppComponent],
|
|
|
|
})
|
|
|
|
export class IdSelectorAppModule {
|
2019-11-13 00:32:58 -05:00
|
|
|
}
|
2019-12-18 08:35:22 -05:00
|
|
|
|
|
|
|
@Component({
|
|
|
|
selector: '[foo],span,.bar',
|
|
|
|
template: 'works!',
|
|
|
|
})
|
|
|
|
export class MultipleSelectorsAppComponent {
|
|
|
|
}
|
|
|
|
|
|
|
|
@NgModule({
|
|
|
|
imports: [BrowserModule],
|
|
|
|
declarations: [MultipleSelectorsAppComponent],
|
|
|
|
bootstrap: [MultipleSelectorsAppComponent],
|
|
|
|
})
|
|
|
|
export class MultipleSelectorsAppModule {
|
2020-02-18 13:55:55 -05:00
|
|
|
}
|