angular-docs-cn/packages/core/test/acceptance/integration_spec.ts

1976 lines
70 KiB
TypeScript
Raw Normal View History

/**
* @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
*/
import {CommonModule} from '@angular/common';
import {Component, ContentChild, Directive, ElementRef, EventEmitter, HostBinding, HostListener, Input, NgModule, OnInit, Output, Pipe, QueryList, TemplateRef, ViewChild, ViewChildren, ViewContainerRef} from '@angular/core';
import {TVIEW} from '@angular/core/src/render3/interfaces/view';
import {getLView} from '@angular/core/src/render3/state';
import {ngDevModeResetPerfCounters} from '@angular/core/src/util/ng_dev_mode';
import {TestBed} from '@angular/core/testing';
import {By} from '@angular/platform-browser';
import {expect} from '@angular/platform-browser/testing/src/matchers';
import {ivyEnabled, onlyInIvy} from '@angular/private/testing';
describe('acceptance integration tests', () => {
function stripHtmlComments(str: string) { return str.replace(/<!--[\s\S]*?-->/g, ''); }
describe('render', () => {
it('should render basic template', () => {
@Component({template: '<span title="Hello">Greetings</span>'})
class App {
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
expect(fixture.nativeElement.innerHTML).toEqual('<span title="Hello">Greetings</span>');
});
it('should render and update basic "Hello, World" template', () => {
ngDevModeResetPerfCounters();
@Component({template: '<h1>Hello, {{name}}!</h1>'})
class App {
name = '';
}
onlyInIvy('perf counters').expectPerfCounters({
tView: 0,
tNode: 0,
});
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.name = 'World';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<h1>Hello, World!</h1>');
onlyInIvy('perf counters').expectPerfCounters({
tView: 2, // Host view + App
tNode: 4, // Host Node + App Node + <span> + #text
});
fixture.componentInstance.name = 'New World';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<h1>Hello, New World!</h1>');
// Assert that the tView/tNode count does not increase (they are correctly cached)
onlyInIvy('perf counters').expectPerfCounters({
tView: 2,
tNode: 4,
});
});
});
describe('ng-container', () => {
it('should insert as a child of a regular element', () => {
@Component(
{template: '<div>before|<ng-container>Greetings<span></span></ng-container>|after</div>'})
class App {
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
// Strip comments since VE and Ivy put them in different places.
expect(stripHtmlComments(fixture.nativeElement.innerHTML))
.toBe('<div>before|Greetings<span></span>|after</div>');
});
it('should add and remove DOM nodes when ng-container is a child of a regular element', () => {
@Component({
template:
'<ng-template [ngIf]="render"><div><ng-container>content</ng-container></div></ng-template>'
})
class App {
render = false;
}
TestBed.configureTestingModule({declarations: [App], imports: [CommonModule]});
const fixture = TestBed.createComponent(App);
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('');
fixture.componentInstance.render = true;
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('<div>content</div>');
fixture.componentInstance.render = false;
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('');
});
it('should add and remove DOM nodes when ng-container is a child of an embedded view', () => {
@Component({template: '<ng-container *ngIf="render">content</ng-container>'})
class App {
render = false;
}
TestBed.configureTestingModule({declarations: [App], imports: [CommonModule]});
const fixture = TestBed.createComponent(App);
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('');
fixture.componentInstance.render = true;
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('content');
fixture.componentInstance.render = false;
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('');
});
// https://stackblitz.com/edit/angular-tfhcz1?file=src%2Fapp%2Fapp.component.ts
it('should add and remove DOM nodes when ng-container is a child of a delayed embedded view',
() => {
@Directive({selector: '[testDirective]'})
class TestDirective {
constructor(private _tplRef: TemplateRef<any>, private _vcRef: ViewContainerRef) {}
createAndInsert() { this._vcRef.insert(this._tplRef.createEmbeddedView({})); }
clear() { this._vcRef.clear(); }
}
@Component({
template: '<ng-template testDirective><ng-container>content</ng-container></ng-template>'
})
class App {
@ViewChild(TestDirective, {static: true}) testDirective !: TestDirective;
}
TestBed.configureTestingModule({declarations: [App, TestDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toBe('');
fixture.componentInstance.testDirective.createAndInsert();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toBe('content');
fixture.componentInstance.testDirective.clear();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toBe('');
});
it('should render at the component view root', () => {
@Component(
{selector: 'test-cmpt', template: '<ng-container>component template</ng-container>'})
class TestCmpt {
}
@Component({template: '<test-cmpt></test-cmpt>'})
class App {
}
TestBed.configureTestingModule({declarations: [App, TestCmpt]});
const fixture = TestBed.createComponent(App);
expect(stripHtmlComments(fixture.nativeElement.innerHTML))
.toBe('<test-cmpt>component template</test-cmpt>');
});
it('should render inside another ng-container', () => {
@Component({
selector: 'test-cmpt',
template:
'<ng-container><ng-container><ng-container>content</ng-container></ng-container></ng-container>'
})
class TestCmpt {
}
@Component({template: '<test-cmpt></test-cmpt>'})
class App {
}
TestBed.configureTestingModule({declarations: [App, TestCmpt]});
const fixture = TestBed.createComponent(App);
expect(stripHtmlComments(fixture.nativeElement.innerHTML))
.toBe('<test-cmpt>content</test-cmpt>');
});
it('should render inside another ng-container at the root of a delayed view', () => {
@Directive({selector: '[testDirective]'})
class TestDirective {
constructor(private _tplRef: TemplateRef<any>, private _vcRef: ViewContainerRef) {}
createAndInsert() { this._vcRef.insert(this._tplRef.createEmbeddedView({})); }
clear() { this._vcRef.clear(); }
}
@Component({
template:
'<ng-template testDirective><ng-container><ng-container><ng-container>content</ng-container></ng-container></ng-container></ng-template>'
})
class App {
@ViewChild(TestDirective, {static: true}) testDirective !: TestDirective;
}
TestBed.configureTestingModule({declarations: [App, TestDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toBe('');
fixture.componentInstance.testDirective.createAndInsert();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toBe('content');
fixture.componentInstance.testDirective.createAndInsert();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toBe('contentcontent');
fixture.componentInstance.testDirective.clear();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toBe('');
});
it('should support directives and inject ElementRef', () => {
@Directive({selector: '[dir]'})
class TestDirective {
constructor(public elRef: ElementRef) {}
}
@Component({template: '<div><ng-container dir></ng-container></div>'})
class App {
@ViewChild(TestDirective) testDirective !: TestDirective;
}
TestBed.configureTestingModule({declarations: [App, TestDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('<div></div>');
expect(fixture.componentInstance.testDirective.elRef.nativeElement.nodeType)
.toBe(Node.COMMENT_NODE);
});
it('should support ViewContainerRef when ng-container is at the root of a view', () => {
@Directive({selector: '[dir]'})
class TestDirective {
@Input()
contentTpl: TemplateRef<{}>|null = null;
constructor(private _vcRef: ViewContainerRef) {}
insertView() { this._vcRef.createEmbeddedView(this.contentTpl as TemplateRef<{}>); }
clear() { this._vcRef.clear(); }
}
@Component({
template:
'<ng-container dir [contentTpl]="content"><ng-template #content>Content</ng-template></ng-container>'
})
class App {
@ViewChild(TestDirective) testDirective !: TestDirective;
}
TestBed.configureTestingModule({declarations: [App, TestDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('');
fixture.componentInstance.testDirective.insertView();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('Content');
fixture.componentInstance.testDirective.clear();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('');
});
it('should support ViewContainerRef on <ng-template> inside <ng-container>', () => {
@Directive({selector: '[dir]'})
class TestDirective {
constructor(private _tplRef: TemplateRef<{}>, private _vcRef: ViewContainerRef) {}
insertView() { this._vcRef.createEmbeddedView(this._tplRef); }
clear() { this._vcRef.clear(); }
}
@Component({template: '<ng-container><ng-template dir>Content</ng-template></ng-container>'})
class App {
@ViewChild(TestDirective) testDirective !: TestDirective;
}
TestBed.configureTestingModule({declarations: [App, TestDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('');
fixture.componentInstance.testDirective.insertView();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('Content');
fixture.componentInstance.testDirective.clear();
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('');
});
it('should not set any attributes', () => {
@Component({template: '<div><ng-container id="foo"></ng-container></div>'})
class App {
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripHtmlComments(fixture.nativeElement.innerHTML)).toEqual('<div></div>');
});
});
describe('text bindings', () => {
it('should render "undefined" as ""', () => {
@Component({template: '{{name}}'})
class App {
name: string|undefined = 'benoit';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('benoit');
fixture.componentInstance.name = undefined;
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('');
});
it('should render "null" as ""', () => {
@Component({template: '{{name}}'})
class App {
name: string|null = 'benoit';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('benoit');
fixture.componentInstance.name = null;
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('');
});
});
describe('ngNonBindable handling', () => {
function stripNgNonBindable(str: string) { return str.replace(/ ngnonbindable=""/i, ''); }
it('should keep local ref for host element', () => {
@Component({
template: `
<b ngNonBindable #myRef id="my-id">
<i>Hello {{ name }}!</i>
</b>
{{ myRef.id }}
`
})
class App {
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripNgNonBindable(fixture.nativeElement.innerHTML))
.toEqual('<b id="my-id"><i>Hello {{ name }}!</i></b> my-id ');
});
it('should invoke directives for host element', () => {
let directiveInvoked: boolean = false;
@Directive({selector: '[directive]'})
class TestDirective implements OnInit {
ngOnInit() { directiveInvoked = true; }
}
@Component({
template: `
<b ngNonBindable directive>
<i>Hello {{ name }}!</i>
</b>
`
})
class App {
name = 'World';
}
TestBed.configureTestingModule({declarations: [App, TestDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripNgNonBindable(fixture.nativeElement.innerHTML))
.toEqual('<b directive=""><i>Hello {{ name }}!</i></b>');
expect(directiveInvoked).toEqual(true);
});
it('should not invoke directives for nested elements', () => {
let directiveInvoked: boolean = false;
@Directive({selector: '[directive]'})
class TestDirective implements OnInit {
ngOnInit() { directiveInvoked = true; }
}
@Component({
template: `
<b ngNonBindable>
<i directive>Hello {{ name }}!</i>
</b>
`
})
class App {
name = 'World';
}
TestBed.configureTestingModule({declarations: [App, TestDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(stripNgNonBindable(fixture.nativeElement.innerHTML))
.toEqual('<b><i directive="">Hello {{ name }}!</i></b>');
expect(directiveInvoked).toEqual(false);
});
});
describe('Siblings update', () => {
it('should handle a flat list of static/bound text nodes', () => {
@Component({template: 'Hello {{name}}!'})
class App {
name = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.name = 'world';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('Hello world!');
fixture.componentInstance.name = 'monde';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('Hello monde!');
});
it('should handle a list of static/bound text nodes as element children', () => {
@Component({template: '<b>Hello {{name}}!</b>'})
class App {
name = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.name = 'world';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<b>Hello world!</b>');
fixture.componentInstance.name = 'mundo';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<b>Hello mundo!</b>');
});
it('should render/update text node as a child of a deep list of elements', () => {
@Component({template: '<b><b><b><b>Hello {{name}}!</b></b></b></b>'})
class App {
name = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.name = 'world';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<b><b><b><b>Hello world!</b></b></b></b>');
fixture.componentInstance.name = 'mundo';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<b><b><b><b>Hello mundo!</b></b></b></b>');
});
it('should update 2 sibling elements', () => {
@Component({template: '<b><span></span><span class="foo" [id]="id"></span></b>'})
class App {
id = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.id = 'foo';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML)
.toEqual('<b><span></span><span class="foo" id="foo"></span></b>');
fixture.componentInstance.id = 'bar';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML)
.toEqual('<b><span></span><span class="foo" id="bar"></span></b>');
});
it('should handle sibling text node after element with child text node', () => {
@Component({template: '<p>hello</p>{{name}}'})
class App {
name = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.name = 'world';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<p>hello</p>world');
fixture.componentInstance.name = 'mundo';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<p>hello</p>mundo');
});
});
describe('basic components', () => {
@Component({selector: 'todo', template: '<p>Todo{{value}}</p>'})
class TodoComponent {
value = ' one';
}
it('should support a basic component template', () => {
@Component({template: '<todo></todo>'})
class App {
}
TestBed.configureTestingModule({declarations: [App, TodoComponent]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<todo><p>Todo one</p></todo>');
});
it('should support a component template with sibling', () => {
@Component({template: '<todo></todo>two'})
class App {
}
TestBed.configureTestingModule({declarations: [App, TodoComponent]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<todo><p>Todo one</p></todo>two');
});
it('should support a component template with component sibling', () => {
@Component({template: '<todo></todo><todo></todo>'})
class App {
}
TestBed.configureTestingModule({declarations: [App, TodoComponent]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML)
.toEqual('<todo><p>Todo one</p></todo><todo><p>Todo one</p></todo>');
});
it('should support a component with binding on host element', () => {
@Component({selector: 'todo', template: '{{title}}'})
class TodoComponentHostBinding {
@HostBinding()
title = 'one';
}
@Component({template: '<todo></todo>'})
class App {
@ViewChild(TodoComponentHostBinding)
todoComponentHostBinding !: TodoComponentHostBinding;
}
TestBed.configureTestingModule({declarations: [App, TodoComponentHostBinding]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<todo title="one">one</todo>');
fixture.componentInstance.todoComponentHostBinding.title = 'two';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<todo title="two">two</todo>');
});
it('should support root component with host attribute', () => {
@Component({selector: 'host-attr-comp', template: '', host: {'role': 'button'}})
class HostAttributeComp {
}
TestBed.configureTestingModule({declarations: [HostAttributeComp]});
const fixture = TestBed.createComponent(HostAttributeComp);
fixture.detectChanges();
expect(fixture.nativeElement.getAttribute('role')).toEqual('button');
});
it('should support component with bindings in template', () => {
@Component({selector: 'comp', template: '<p>{{ name }}</p>'})
class MyComp {
name = 'Bess';
}
@Component({template: '<comp></comp>'})
class App {
}
TestBed.configureTestingModule({declarations: [App, MyComp]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<comp><p>Bess</p></comp>');
});
it('should support a component with sub-views', () => {
@Component({selector: 'comp', template: '<div *ngIf="condition">text</div>'})
class MyComp {
@Input()
condition !: boolean;
}
@Component({template: '<comp [condition]="condition"></comp>'})
class App {
condition = false;
}
TestBed.configureTestingModule({declarations: [App, MyComp], imports: [CommonModule]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
const compElement = fixture.nativeElement.querySelector('comp');
fixture.componentInstance.condition = true;
fixture.detectChanges();
expect(stripHtmlComments(compElement.innerHTML)).toEqual('<div>text</div>');
fixture.componentInstance.condition = false;
fixture.detectChanges();
expect(stripHtmlComments(compElement.innerHTML)).toEqual('');
});
});
describe('element bindings', () => {
describe('elementAttribute', () => {
it('should support attribute bindings', () => {
@Component({template: '<button [attr.title]="title"></button>'})
class App {
title: string|null = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.title = 'Hello';
fixture.detectChanges();
// initial binding
expect(fixture.nativeElement.innerHTML).toEqual('<button title="Hello"></button>');
// update binding
fixture.componentInstance.title = 'Hi!';
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<button title="Hi!"></button>');
// remove attribute
fixture.componentInstance.title = null;
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<button></button>');
});
it('should stringify values used attribute bindings', () => {
@Component({template: '<button [attr.title]="title"></button>'})
class App {
title: any;
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.title = NaN;
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<button title="NaN"></button>');
fixture.componentInstance.title = {toString: () => 'Custom toString'};
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML)
.toEqual('<button title="Custom toString"></button>');
});
it('should update bindings', () => {
@Component({
template: [
'a:{{c[0]}}{{c[1]}}{{c[2]}}{{c[3]}}{{c[4]}}{{c[5]}}{{c[6]}}{{c[7]}}{{c[8]}}{{c[9]}}{{c[10]}}{{c[11]}}{{c[12]}}{{c[13]}}{{c[14]}}{{c[15]}}{{c[16]}}',
'a0:{{c[1]}}',
'a1:{{c[0]}}{{c[1]}}{{c[16]}}',
'a2:{{c[0]}}{{c[1]}}{{c[2]}}{{c[3]}}{{c[16]}}',
'a3:{{c[0]}}{{c[1]}}{{c[2]}}{{c[3]}}{{c[4]}}{{c[5]}}{{c[16]}}',
'a4:{{c[0]}}{{c[1]}}{{c[2]}}{{c[3]}}{{c[4]}}{{c[5]}}{{c[6]}}{{c[7]}}{{c[16]}}',
'a5:{{c[0]}}{{c[1]}}{{c[2]}}{{c[3]}}{{c[4]}}{{c[5]}}{{c[6]}}{{c[7]}}{{c[8]}}{{c[9]}}{{c[16]}}',
'a6:{{c[0]}}{{c[1]}}{{c[2]}}{{c[3]}}{{c[4]}}{{c[5]}}{{c[6]}}{{c[7]}}{{c[8]}}{{c[9]}}{{c[10]}}{{c[11]}}{{c[16]}}',
'a7:{{c[0]}}{{c[1]}}{{c[2]}}{{c[3]}}{{c[4]}}{{c[5]}}{{c[6]}}{{c[7]}}{{c[8]}}{{c[9]}}{{c[10]}}{{c[11]}}{{c[12]}}{{c[13]}}{{c[16]}}',
'a8:{{c[0]}}{{c[1]}}{{c[2]}}{{c[3]}}{{c[4]}}{{c[5]}}{{c[6]}}{{c[7]}}{{c[8]}}{{c[9]}}{{c[10]}}{{c[11]}}{{c[12]}}{{c[13]}}{{c[14]}}{{c[15]}}{{c[16]}}',
].join('\n')
})
class App {
c = ['(', 0, 'a', 1, 'b', 2, 'c', 3, 'd', 4, 'e', 5, 'f', 6, 'g', 7, ')'];
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
expect(fixture.nativeElement.textContent.trim()).toEqual([
'a:(0a1b2c3d4e5f6g7)',
'a0:0',
'a1:(0)',
'a2:(0a1)',
'a3:(0a1b2)',
'a4:(0a1b2c3)',
'a5:(0a1b2c3d4)',
'a6:(0a1b2c3d4e5)',
'a7:(0a1b2c3d4e5f6)',
'a8:(0a1b2c3d4e5f6g7)',
].join('\n'));
fixture.componentInstance.c.reverse();
fixture.detectChanges();
expect(fixture.nativeElement.textContent.trim()).toEqual([
'a:)7g6f5e4d3c2b1a0(',
'a0:7',
'a1:)7(',
'a2:)7g6(',
'a3:)7g6f5(',
'a4:)7g6f5e4(',
'a5:)7g6f5e4d3(',
'a6:)7g6f5e4d3c2(',
'a7:)7g6f5e4d3c2b1(',
'a8:)7g6f5e4d3c2b1a0(',
].join('\n'));
fixture.componentInstance.c.reverse();
fixture.detectChanges();
expect(fixture.nativeElement.textContent.trim()).toEqual([
'a:(0a1b2c3d4e5f6g7)',
'a0:0',
'a1:(0)',
'a2:(0a1)',
'a3:(0a1b2)',
'a4:(0a1b2c3)',
'a5:(0a1b2c3d4)',
'a6:(0a1b2c3d4e5)',
'a7:(0a1b2c3d4e5f6)',
'a8:(0a1b2c3d4e5f6g7)',
].join('\n'));
});
it('should not update DOM if context has not changed', () => {
@Component({
template: `
<span [attr.title]="title">
<b [attr.title]="title" *ngIf="shouldRender"></b>
</span>
`
})
class App {
title: string|null = '';
shouldRender = true;
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
const span: HTMLSpanElement = fixture.nativeElement.querySelector('span');
const bold: HTMLElement = span.querySelector('b') !;
fixture.componentInstance.title = 'Hello';
fixture.detectChanges();
// initial binding
expect(span.getAttribute('title')).toBe('Hello');
expect(bold.getAttribute('title')).toBe('Hello');
// update DOM manually
bold.setAttribute('title', 'Goodbye');
// refresh with same binding
fixture.detectChanges();
expect(span.getAttribute('title')).toBe('Hello');
expect(bold.getAttribute('title')).toBe('Goodbye');
// refresh again with same binding
fixture.detectChanges();
expect(span.getAttribute('title')).toBe('Hello');
expect(bold.getAttribute('title')).toBe('Goodbye');
});
it('should support host attribute bindings', () => {
@Directive({selector: '[hostBindingDir]'})
class HostBindingDir {
@HostBinding('attr.aria-label')
label = 'some label';
}
@Component({template: '<div hostBindingDir></div>'})
class App {
@ViewChild(HostBindingDir) hostBindingDir !: HostBindingDir;
}
TestBed.configureTestingModule({declarations: [App, HostBindingDir]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
const hostBindingEl = fixture.nativeElement.querySelector('div');
// Needs `toLowerCase`, because different browsers produce
// attributes either in camel case or lower case.
expect(hostBindingEl.getAttribute('aria-label')).toBe('some label');
fixture.componentInstance.hostBindingDir.label = 'other label';
fixture.detectChanges();
expect(hostBindingEl.getAttribute('aria-label')).toBe('other label');
});
});
describe('elementStyle', () => {
it('should support binding to styles', () => {
@Component({template: '<span [style.font-size]="size"></span>'})
class App {
size: string|null = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.size = '10px';
fixture.detectChanges();
const span: HTMLElement = fixture.nativeElement.querySelector('span');
expect(span.style.fontSize).toBe('10px');
fixture.componentInstance.size = '16px';
fixture.detectChanges();
expect(span.style.fontSize).toBe('16px');
fixture.componentInstance.size = null;
fixture.detectChanges();
expect(span.style.fontSize).toBeFalsy();
});
it('should support binding to styles with suffix', () => {
@Component({template: '<span [style.font-size.px]="size"></span>'})
class App {
size: string|number|null = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.size = '100';
fixture.detectChanges();
const span: HTMLElement = fixture.nativeElement.querySelector('span');
expect(span.style.fontSize).toEqual('100px');
fixture.componentInstance.size = 200;
fixture.detectChanges();
expect(span.style.fontSize).toEqual('200px');
fixture.componentInstance.size = 0;
fixture.detectChanges();
expect(span.style.fontSize).toEqual('0px');
fixture.componentInstance.size = null;
fixture.detectChanges();
expect(span.style.fontSize).toBeFalsy();
});
});
describe('class-based styling', () => {
it('should support CSS class toggle', () => {
@Component({template: '<span [class.active]="value"></span>'})
class App {
value: any;
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.value = true;
fixture.detectChanges();
const span = fixture.nativeElement.querySelector('span');
expect(span.getAttribute('class')).toEqual('active');
fixture.componentInstance.value = false;
fixture.detectChanges();
expect(span.getAttribute('class')).toBeFalsy();
// truthy values
fixture.componentInstance.value = 'a_string';
fixture.detectChanges();
expect(span.getAttribute('class')).toEqual('active');
fixture.componentInstance.value = 10;
fixture.detectChanges();
expect(span.getAttribute('class')).toEqual('active');
// falsy values
fixture.componentInstance.value = '';
fixture.detectChanges();
expect(span.getAttribute('class')).toBeFalsy();
fixture.componentInstance.value = 0;
fixture.detectChanges();
expect(span.getAttribute('class')).toBeFalsy();
});
it('should work correctly with existing static classes', () => {
@Component({template: '<span class="existing" [class.active]="value"></span>'})
class App {
value: any;
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.value = true;
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<span class="existing active"></span>');
fixture.componentInstance.value = false;
fixture.detectChanges();
expect(fixture.nativeElement.innerHTML).toEqual('<span class="existing"></span>');
});
it('should apply classes properly when nodes are components', () => {
@Component({selector: 'my-comp', template: 'Comp Content'})
class MyComp {
}
@Component({template: '<my-comp [class.active]="value"></my-comp>'})
class App {
value: any;
}
TestBed.configureTestingModule({declarations: [App, MyComp]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.value = true;
fixture.detectChanges();
const compElement = fixture.nativeElement.querySelector('my-comp');
expect(fixture.nativeElement.textContent).toContain('Comp Content');
expect(compElement.getAttribute('class')).toBe('active');
fixture.componentInstance.value = false;
fixture.detectChanges();
expect(compElement.getAttribute('class')).toBeFalsy();
});
it('should apply classes properly when nodes have containers', () => {
@Component({selector: 'structural-comp', template: 'Comp Content'})
class StructuralComp {
@Input()
tmp !: TemplateRef<any>;
constructor(public vcr: ViewContainerRef) {}
create() { this.vcr.createEmbeddedView(this.tmp); }
}
@Component({
template: `
<ng-template #foo>Temp Content</ng-template>
<structural-comp [class.active]="value" [tmp]="foo"></structural-comp>
`
})
class App {
@ViewChild(StructuralComp) structuralComp !: StructuralComp;
value: any;
}
TestBed.configureTestingModule({declarations: [App, StructuralComp]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.value = true;
fixture.detectChanges();
const structuralCompEl = fixture.nativeElement.querySelector('structural-comp');
expect(structuralCompEl.getAttribute('class')).toEqual('active');
fixture.componentInstance.structuralComp.create();
fixture.detectChanges();
expect(structuralCompEl.getAttribute('class')).toEqual('active');
fixture.componentInstance.value = false;
fixture.detectChanges();
expect(structuralCompEl.getAttribute('class')).toBeFalsy();
});
@Directive({selector: '[DirWithClass]'})
class DirWithClassDirective {
public classesVal: string = '';
@Input('class')
set klass(value: string) { this.classesVal = value; }
}
@Directive({selector: '[DirWithStyle]'})
class DirWithStyleDirective {
public stylesVal: any = '';
@Input()
set style(value: any) { this.stylesVal = value; }
}
it('should delegate initial classes to a [class] input binding if present on a directive on the same element',
() => {
@Component({template: '<div class="apple orange banana" DirWithClass></div>'})
class App {
@ViewChild(DirWithClassDirective)
mockClassDirective !: DirWithClassDirective;
}
TestBed.configureTestingModule({declarations: [App, DirWithClassDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
// the initial values always get sorted in non VE code
// but there is no sorting guarantee within VE code
expect(fixture.componentInstance.mockClassDirective.classesVal.split(/\s+/).sort())
.toEqual(['apple', 'banana', 'orange']);
});
it('should delegate initial styles to a [style] input binding if present on a directive on the same element',
() => {
@Component({template: '<div style="width: 100px; height: 200px" DirWithStyle></div>'})
class App {
@ViewChild(DirWithStyleDirective)
mockStyleDirective !: DirWithStyleDirective;
}
TestBed.configureTestingModule({declarations: [App, DirWithStyleDirective]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
const styles = fixture.componentInstance.mockStyleDirective.stylesVal;
// Use `toContain` since Ivy and ViewEngine have some slight differences in formatting.
expect(styles).toContain('width: 100px');
expect(styles).toContain('height: 200px');
});
it('should update `[class]` and bindings in the provided directive if the input is matched',
() => {
@Component({template: '<div DirWithClass [class]="value"></div>'})
class App {
@ViewChild(DirWithClassDirective)
mockClassDirective !: DirWithClassDirective;
value = '';
}
TestBed.configureTestingModule({declarations: [App, DirWithClassDirective]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.value = 'cucumber grape';
fixture.detectChanges();
expect(fixture.componentInstance.mockClassDirective.classesVal)
.toEqual('cucumber grape');
});
onlyInIvy('Passing an object into [style] works differently')
.it('should update `[style]` and bindings in the provided directive if the input is matched',
() => {
@Component({template: '<div DirWithStyle [style]="value"></div>'})
class App {
@ViewChild(DirWithStyleDirective)
mockStyleDirective !: DirWithStyleDirective;
value !: {[key: string]: string};
}
TestBed.configureTestingModule({declarations: [App, DirWithStyleDirective]});
const fixture = TestBed.createComponent(App);
fixture.componentInstance.value = {width: '200px', height: '500px'};
fixture.detectChanges();
expect(fixture.componentInstance.mockStyleDirective.stylesVal)
.toEqual({width: '200px', height: '500px'});
});
onlyInIvy('Style binding merging works differently in Ivy')
.it('should apply initial styling to the element that contains the directive with host styling',
() => {
@Directive({
selector: '[DirWithInitialStyling]',
host: {
'title': 'foo',
'class': 'heavy golden',
'style': 'color: purple',
'[style.font-weight]': '"bold"'
}
})
class DirWithInitialStyling {
}
@Component({
template: `
<div DirWithInitialStyling
class="big"
style="color:black; font-size:200px"></div>
`
})
class App {
}
TestBed.configureTestingModule({declarations: [App, DirWithInitialStyling]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
const target: HTMLDivElement = fixture.nativeElement.querySelector('div');
const classes = target.getAttribute('class') !.split(/\s+/).sort();
expect(classes).toEqual(['big', 'golden', 'heavy']);
expect(target.getAttribute('title')).toEqual('foo');
expect(target.style.getPropertyValue('color')).toEqual('black');
expect(target.style.getPropertyValue('font-size')).toEqual('200px');
expect(target.style.getPropertyValue('font-weight')).toEqual('bold');
});
onlyInIvy('Style binding merging works differently in Ivy')
.it('should apply single styling bindings present within a directive onto the same element and defer the element\'s initial styling values when missing',
() => {
@Directive({
selector: '[DirWithSingleStylingBindings]',
host: {
'class': 'def',
'[class.xyz]': 'activateXYZClass',
'[style.width]': 'width',
'[style.height]': 'height'
}
})
class DirWithSingleStylingBindings {
width: string|null|undefined = undefined;
height: string|null|undefined = undefined;
activateXYZClass: boolean = false;
}
@Component({
template: `
<div DirWithSingleStylingBindings class="abc" style="width:100px;"></div>
`
})
class App {
@ViewChild(DirWithSingleStylingBindings)
dirInstance !: DirWithSingleStylingBindings;
}
TestBed.configureTestingModule({declarations: [App, DirWithSingleStylingBindings]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
const dirInstance = fixture.componentInstance.dirInstance;
const target: HTMLDivElement = fixture.nativeElement.querySelector('div');
expect(target.style.getPropertyValue('width')).toEqual('100px');
expect(target.style.getPropertyValue('height')).toEqual('');
expect(target.classList.contains('abc')).toBeTruthy();
expect(target.classList.contains('def')).toBeTruthy();
expect(target.classList.contains('xyz')).toBeFalsy();
dirInstance.width = '444px';
dirInstance.height = '999px';
dirInstance.activateXYZClass = true;
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('100px');
expect(target.style.getPropertyValue('height')).toEqual('999px');
expect(target.classList.contains('abc')).toBeTruthy();
expect(target.classList.contains('def')).toBeTruthy();
expect(target.classList.contains('xyz')).toBeTruthy();
dirInstance.width = undefined;
dirInstance.height = undefined;
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('100px');
expect(target.style.getPropertyValue('height')).toEqual('');
expect(target.classList.contains('abc')).toBeTruthy();
expect(target.classList.contains('def')).toBeTruthy();
expect(target.classList.contains('xyz')).toBeTruthy();
});
onlyInIvy('Style binding merging works differently in Ivy')
.it('should properly prioritize single style binding collisions when they exist on multiple directives',
() => {
@Directive({selector: '[Dir1WithStyle]', host: {'[style.width]': 'width'}})
class Dir1WithStyle {
width: null|string|undefined = undefined;
}
@Directive({
selector: '[Dir2WithStyle]',
host: {'style': 'width: 111px', '[style.width]': 'width'}
})
class Dir2WithStyle {
width: null|string|undefined = undefined;
}
@Component(
{template: '<div Dir1WithStyle Dir2WithStyle [style.width]="width"></div>'})
class App {
@ViewChild(Dir1WithStyle) dir1Instance !: Dir1WithStyle;
@ViewChild(Dir2WithStyle) dir2Instance !: Dir2WithStyle;
width: string|null|undefined = undefined;
}
TestBed.configureTestingModule({declarations: [App, Dir2WithStyle, Dir1WithStyle]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
const {dir1Instance, dir2Instance} = fixture.componentInstance;
const target: HTMLDivElement = fixture.nativeElement.querySelector('div');
expect(target.style.getPropertyValue('width')).toEqual('111px');
fixture.componentInstance.width = '999px';
dir1Instance.width = '222px';
dir2Instance.width = '333px';
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('999px');
fixture.componentInstance.width = undefined;
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('222px');
dir1Instance.width = undefined;
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('333px');
dir2Instance.width = undefined;
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('111px');
dir1Instance.width = '666px';
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('666px');
fixture.componentInstance.width = '777px';
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('777px');
});
onlyInIvy('Style binding merging works differently in Ivy')
.it('should properly prioritize multi style binding collisions when they exist on multiple directives',
() => {
@Directive({
selector: '[Dir1WithStyling]',
host: {'[style]': 'stylesExp', '[class]': 'classesExp'}
})
class Dir1WithStyling {
classesExp: any = {};
stylesExp: any = {};
}
@Directive({
selector: '[Dir2WithStyling]',
host: {'style': 'width: 111px', '[style]': 'stylesExp'}
})
class Dir2WithStyling {
stylesExp: any = {};
}
@Component({
template:
'<div Dir1WithStyling Dir2WithStyling [style]="stylesExp" [class]="classesExp"></div>'
})
class App {
@ViewChild(Dir1WithStyling) dir1Instance !: Dir1WithStyling;
@ViewChild(Dir2WithStyling) dir2Instance !: Dir2WithStyling;
stylesExp: any = {};
classesExp: any = {};
}
TestBed.configureTestingModule(
{declarations: [App, Dir2WithStyling, Dir1WithStyling]});
const fixture = TestBed.createComponent(App);
fixture.detectChanges();
const {dir1Instance, dir2Instance} = fixture.componentInstance;
const target = fixture.nativeElement.querySelector('div') !;
expect(target.style.getPropertyValue('width')).toEqual('111px');
const compInstance = fixture.componentInstance;
compInstance.stylesExp = {width: '999px', height: undefined};
compInstance.classesExp = {one: true, two: false};
dir1Instance.stylesExp = {width: '222px'};
dir1Instance.classesExp = {two: true, three: false};
dir2Instance.stylesExp = {width: '333px', height: '100px'};
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('999px');
expect(target.style.getPropertyValue('height')).toEqual('100px');
expect(target.classList.contains('one')).toBeTruthy();
expect(target.classList.contains('two')).toBeFalsy();
expect(target.classList.contains('three')).toBeFalsy();
compInstance.stylesExp = {};
compInstance.classesExp = {};
dir1Instance.stylesExp = {width: '222px', height: '200px'};
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('222px');
expect(target.style.getPropertyValue('height')).toEqual('200px');
expect(target.classList.contains('one')).toBeFalsy();
expect(target.classList.contains('two')).toBeTruthy();
expect(target.classList.contains('three')).toBeFalsy();
dir1Instance.stylesExp = {};
dir1Instance.classesExp = {};
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('333px');
expect(target.style.getPropertyValue('height')).toEqual('100px');
expect(target.classList.contains('one')).toBeFalsy();
expect(target.classList.contains('two')).toBeFalsy();
expect(target.classList.contains('three')).toBeFalsy();
dir2Instance.stylesExp = {};
compInstance.stylesExp = {height: '900px'};
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('111px');
expect(target.style.getPropertyValue('height')).toEqual('900px');
dir1Instance.stylesExp = {width: '666px', height: '600px'};
dir1Instance.classesExp = {four: true, one: true};
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('666px');
expect(target.style.getPropertyValue('height')).toEqual('900px');
expect(target.classList.contains('one')).toBeTruthy();
expect(target.classList.contains('two')).toBeFalsy();
expect(target.classList.contains('three')).toBeFalsy();
expect(target.classList.contains('four')).toBeTruthy();
compInstance.stylesExp = {width: '777px'};
compInstance.classesExp = {four: false};
fixture.detectChanges();
expect(target.style.getPropertyValue('width')).toEqual('777px');
expect(target.style.getPropertyValue('height')).toEqual('600px');
expect(target.classList.contains('one')).toBeTruthy();
expect(target.classList.contains('two')).toBeFalsy();
expect(target.classList.contains('three')).toBeFalsy();
expect(target.classList.contains('four')).toBeFalsy();
});
});
it('should properly handle and render interpolation for class attribute bindings', () => {
@Component({template: '<div class="-{{name}}-{{age}}-"></div>'})
class App {
name = '';
age = '';
}
TestBed.configureTestingModule({declarations: [App]});
const fixture = TestBed.createComponent(App);
const target = fixture.nativeElement.querySelector('div') !;
expect(target.classList.contains('-fred-36-')).toBeFalsy();
fixture.componentInstance.name = 'fred';
fixture.componentInstance.age = '36';
fixture.detectChanges();
expect(target.classList.contains('-fred-36-')).toBeTruthy();
});
});
describe('NgModule assertions', () => {
it('should throw with descriptive error message when a module imports itself', () => {
@Component({template: ''})
class FixtureComponent {
}
@NgModule({imports: [SomeModule], declarations: [FixtureComponent]})
class SomeModule {
}
expect(() => {
TestBed.configureTestingModule({imports: [SomeModule]}).createComponent(FixtureComponent);
}).toThrowError(`'SomeModule' module can't import itself`);
});
it('should throw with descriptive error message when a directive is passed to imports', () => {
@Component({template: ''})
class SomeComponent {
}
@NgModule({imports: [SomeComponent]})
class ModuleWithImportedComponent {
}
expect(() => {
TestBed.configureTestingModule({imports: [ModuleWithImportedComponent]})
.createComponent(SomeComponent);
})
.toThrowError(
// The ViewEngine error has a typo, whereas the Ivy one fixes it.
/^Unexpected directive 'SomeComponent' imported by the module 'ModuleWithImportedComponent'\. Please add (a|an) @NgModule annotation\.$/);
});
it('should throw with descriptive error message when a pipe is passed to imports', () => {
@Component({template: ''})
class FixtureComponent {
}
@Pipe({name: 'somePipe'})
class SomePipe {
}
@NgModule({imports: [SomePipe], declarations: [FixtureComponent]})
class ModuleWithImportedPipe {
}
expect(() => {
TestBed.configureTestingModule({imports: [ModuleWithImportedPipe]})
.createComponent(FixtureComponent);
})
.toThrowError(
// The ViewEngine error has a typo, whereas the Ivy one fixes it.
/^Unexpected pipe 'SomePipe' imported by the module 'ModuleWithImportedPipe'\. Please add (a|an) @NgModule annotation\.$/);
});
it('should throw with descriptive error message when a module is passed to declarations', () => {
@Component({template: ''})
class FixtureComponent {
}
@NgModule({})
class SomeModule {
}
@NgModule({declarations: [SomeModule, FixtureComponent]})
class ModuleWithDeclaredModule {
}
// The error is almost the same in Ivy and ViewEngine, however since Ivy's
// message is more correct it doesn't make sense to align it ViewEngine.
const expectedErrorMessage = ivyEnabled ?
`Unexpected value 'SomeModule' declared by the module 'ModuleWithDeclaredModule'. Please add a @Pipe/@Directive/@Component annotation.` :
`Unexpected module 'SomeModule' declared by the module 'ModuleWithDeclaredModule'. Please add a @Pipe/@Directive/@Component annotation.`;
expect(() => {
TestBed.configureTestingModule({imports: [ModuleWithDeclaredModule]})
.createComponent(FixtureComponent);
}).toThrowError(expectedErrorMessage);
});
it('should throw with descriptive error message when a declaration is missing annotation', () => {
@Component({template: ''})
class FixtureComponent {
}
class SomeClass {}
@NgModule({declarations: [SomeClass, FixtureComponent]})
class SomeModule {
}
expect(() => {
TestBed.configureTestingModule({imports: [SomeModule]}).createComponent(FixtureComponent);
})
.toThrowError(
`Unexpected value 'SomeClass' declared by the module 'SomeModule'. Please add a @Pipe/@Directive/@Component annotation.`);
});
it('should throw with descriptive error message when an imported module is missing annotation',
() => {
@Component({template: ''})
class FixtureComponent {
}
class SomeModule {}
@NgModule({imports: [SomeModule], declarations: [FixtureComponent]})
class ModuleWithImportedModule {
}
expect(() => {
TestBed.configureTestingModule({imports: [ModuleWithImportedModule]})
.createComponent(FixtureComponent);
})
.toThrowError(
// The ViewEngine error has a typo, whereas the Ivy one fixes it.
/^Unexpected value 'SomeModule' imported by the module 'ModuleWithImportedModule'\. Please add (a|an) @NgModule annotation\.$/);
});
});
it('should only call inherited host listeners once', () => {
let clicks = 0;
@Component({template: ''})
class ButtonSuperClass {
@HostListener('click')
clicked() { clicks++; }
}
@Component({selector: 'button[custom-button]', template: ''})
class ButtonSubClass extends ButtonSuperClass {
}
@Component({template: '<button custom-button></button>'})
class MyApp {
}
TestBed.configureTestingModule({declarations: [MyApp, ButtonSuperClass, ButtonSubClass]});
const fixture = TestBed.createComponent(MyApp);
const button = fixture.debugElement.query(By.directive(ButtonSubClass));
fixture.detectChanges();
button.nativeElement.click();
fixture.detectChanges();
expect(clicks).toBe(1);
});
it('should support inherited view queries', () => {
@Directive({selector: '[someDir]'})
class SomeDir {
}
@Component({template: '<div someDir></div>'})
class SuperComp {
@ViewChildren(SomeDir) dirs !: QueryList<SomeDir>;
}
@Component({selector: 'button[custom-button]', template: '<div someDir></div>'})
class SubComp extends SuperComp {
}
@Component({template: '<button custom-button></button>'})
class MyApp {
}
TestBed.configureTestingModule({declarations: [MyApp, SuperComp, SubComp, SomeDir]});
const fixture = TestBed.createComponent(MyApp);
const subInstance = fixture.debugElement.query(By.directive(SubComp)).componentInstance;
fixture.detectChanges();
expect(subInstance.dirs.length).toBe(1);
expect(subInstance.dirs.first).toBeAnInstanceOf(SomeDir);
});
it('should not set inputs after destroy', () => {
@Directive({
selector: '[no-assign-after-destroy]',
})
class NoAssignAfterDestroy {
private _isDestroyed = false;
@Input()
get value() { return this._value; }
set value(newValue: any) {
if (this._isDestroyed) {
throw Error('Cannot assign to value after destroy.');
}
this._value = newValue;
}
private _value: any;
ngOnDestroy() { this._isDestroyed = true; }
}
@Component({template: '<div no-assign-after-destroy [value]="directiveValue"></div>'})
class App {
directiveValue = 'initial-value';
}
TestBed.configureTestingModule({declarations: [NoAssignAfterDestroy, App]});
let fixture = TestBed.createComponent(App);
fixture.destroy();
expect(() => {
fixture = TestBed.createComponent(App);
fixture.detectChanges();
}).not.toThrow();
});
it('should support host attribute and @ContentChild on the same component', () => {
@Component(
{selector: 'test-component', template: `foo`, host: {'[attr.aria-disabled]': 'true'}})
class TestComponent {
@ContentChild(TemplateRef, {static: true}) tpl !: TemplateRef<any>;
}
TestBed.configureTestingModule({declarations: [TestComponent]});
const fixture = TestBed.createComponent(TestComponent);
fixture.detectChanges();
expect(fixture.componentInstance.tpl).not.toBeNull();
expect(fixture.debugElement.nativeElement.getAttribute('aria-disabled')).toBe('true');
});
it('should inherit inputs from undecorated superclasses', () => {
class ButtonSuperClass {
@Input() isDisabled !: boolean;
}
@Component({selector: 'button[custom-button]', template: ''})
class ButtonSubClass extends ButtonSuperClass {
}
@Component({template: '<button custom-button [isDisabled]="disableButton"></button>'})
class MyApp {
disableButton = false;
}
TestBed.configureTestingModule({declarations: [MyApp, ButtonSubClass]});
const fixture = TestBed.createComponent(MyApp);
const button = fixture.debugElement.query(By.directive(ButtonSubClass)).componentInstance;
fixture.detectChanges();
expect(button.isDisabled).toBe(false);
fixture.componentInstance.disableButton = true;
fixture.detectChanges();
expect(button.isDisabled).toBe(true);
});
it('should inherit outputs from undecorated superclasses', () => {
let clicks = 0;
class ButtonSuperClass {
@Output() clicked = new EventEmitter<void>();
emitClick() { this.clicked.emit(); }
}
@Component({selector: 'button[custom-button]', template: ''})
class ButtonSubClass extends ButtonSuperClass {
}
@Component({template: '<button custom-button (clicked)="handleClick()"></button>'})
class MyApp {
handleClick() { clicks++; }
}
TestBed.configureTestingModule({declarations: [MyApp, ButtonSubClass]});
const fixture = TestBed.createComponent(MyApp);
const button = fixture.debugElement.query(By.directive(ButtonSubClass)).componentInstance;
button.emitClick();
fixture.detectChanges();
expect(clicks).toBe(1);
});
it('should inherit host bindings from undecorated superclasses', () => {
class BaseButton {
@HostBinding('attr.tabindex')
tabindex = -1;
}
@Component({selector: '[sub-button]', template: '<ng-content></ng-content>'})
class SubButton extends BaseButton {
}
@Component({template: '<button sub-button>Click me</button>'})
class App {
}
TestBed.configureTestingModule({declarations: [SubButton, App]});
const fixture = TestBed.createComponent(App);
const button = fixture.debugElement.query(By.directive(SubButton));
fixture.detectChanges();
expect(button.nativeElement.getAttribute('tabindex')).toBe('-1');
button.componentInstance.tabindex = 2;
fixture.detectChanges();
expect(button.nativeElement.getAttribute('tabindex')).toBe('2');
});
it('should inherit host bindings from undecorated grand superclasses', () => {
class SuperBaseButton {
@HostBinding('attr.tabindex')
tabindex = -1;
}
class BaseButton extends SuperBaseButton {}
@Component({selector: '[sub-button]', template: '<ng-content></ng-content>'})
class SubButton extends BaseButton {
}
@Component({template: '<button sub-button>Click me</button>'})
class App {
}
TestBed.configureTestingModule({declarations: [SubButton, App]});
const fixture = TestBed.createComponent(App);
const button = fixture.debugElement.query(By.directive(SubButton));
fixture.detectChanges();
expect(button.nativeElement.getAttribute('tabindex')).toBe('-1');
button.componentInstance.tabindex = 2;
fixture.detectChanges();
expect(button.nativeElement.getAttribute('tabindex')).toBe('2');
});
it('should inherit host listeners from undecorated superclasses', () => {
let clicks = 0;
class BaseButton {
@HostListener('click')
handleClick() { clicks++; }
}
@Component({selector: '[sub-button]', template: '<ng-content></ng-content>'})
class SubButton extends BaseButton {
}
@Component({template: '<button sub-button>Click me</button>'})
class App {
}
TestBed.configureTestingModule({declarations: [SubButton, App]});
const fixture = TestBed.createComponent(App);
const button = fixture.debugElement.query(By.directive(SubButton)).nativeElement;
button.click();
fixture.detectChanges();
expect(clicks).toBe(1);
});
it('should inherit host listeners from superclasses once', () => {
let clicks = 0;
@Directive({selector: '[baseButton]'})
class BaseButton {
@HostListener('click')
handleClick() { clicks++; }
}
@Component({selector: '[subButton]', template: '<ng-content></ng-content>'})
class SubButton extends BaseButton {
}
@Component({template: '<button subButton>Click me</button>'})
class App {
}
TestBed.configureTestingModule({declarations: [SubButton, BaseButton, App]});
const fixture = TestBed.createComponent(App);
const button = fixture.debugElement.query(By.directive(SubButton)).nativeElement;
button.click();
fixture.detectChanges();
expect(clicks).toBe(1);
});
it('should inherit host listeners from grand superclasses once', () => {
let clicks = 0;
@Directive({selector: '[superBaseButton]'})
class SuperBaseButton {
@HostListener('click')
handleClick() { clicks++; }
}
@Directive({selector: '[baseButton]'})
class BaseButton extends SuperBaseButton {
}
@Component({selector: '[subButton]', template: '<ng-content></ng-content>'})
class SubButton extends BaseButton {
}
@Component({template: '<button subButton>Click me</button>'})
class App {
}
TestBed.configureTestingModule({declarations: [SubButton, SuperBaseButton, BaseButton, App]});
const fixture = TestBed.createComponent(App);
const button = fixture.debugElement.query(By.directive(SubButton)).nativeElement;
button.click();
fixture.detectChanges();
expect(clicks).toBe(1);
});
it('should inherit host listeners from grand grand superclasses once', () => {
let clicks = 0;
@Directive({selector: '[superSuperBaseButton]'})
class SuperSuperBaseButton {
@HostListener('click')
handleClick() { clicks++; }
}
@Directive({selector: '[superBaseButton]'})
class SuperBaseButton extends SuperSuperBaseButton {
}
@Directive({selector: '[baseButton]'})
class BaseButton extends SuperBaseButton {
}
@Component({selector: '[subButton]', template: '<ng-content></ng-content>'})
class SubButton extends BaseButton {
}
@Component({template: '<button subButton>Click me</button>'})
class App {
}
TestBed.configureTestingModule(
{declarations: [SubButton, SuperBaseButton, SuperSuperBaseButton, BaseButton, App]});
const fixture = TestBed.createComponent(App);
const button = fixture.debugElement.query(By.directive(SubButton)).nativeElement;
button.click();
fixture.detectChanges();
expect(clicks).toBe(1);
});
fix(ivy): don't mask errors by calling lifecycle hooks after a crash (#31244) The Angular runtime frequently calls into user code (for example, when writing to a property binding). Since user code can throw errors, calls to it are frequently wrapped in a try-finally block. In Ivy, the following pattern is common: ```typescript enterView(); try { callUserCode(); } finally { leaveView(); } ``` This has a significant problem, however: `leaveView` has a side effect: it calls any pending lifecycle hooks that might've been scheduled during the current round of change detection. Generally it's a bad idea to run lifecycle hooks after the application has crashed. The application is in an inconsistent state - directives may not be instantiated fully, queries may not be resolved, bindings may not have been applied, etc. Invariants that the app code relies upon may not hold. Further crashes or broken behavior are likely. Frequently, lifecycle hooks are used to make assertions about these invariants. When these assertions fail, they will throw and "swallow" the original error, making debugging of the problem much more difficult. This commit modifies `leaveView` to understand whether the application is currently crashing, via a parameter `safeToRunHooks`. This parameter is set by modifying the above pattern: ```typescript enterView(); let safeToRunHooks = false; try { callUserCode(); safeToRunHooks = true; } finally { leaveView(..., safeToRunHooks); } ``` If `callUserCode` crashes, then `safeToRunHooks` will never be set to `true` and `leaveView` won't call any further user code. The original error will then propagate back up the stack and be reported correctly. A test is added to verify this behavior. PR Close #31244
2019-06-24 16:14:05 -04:00
it('should not mask errors thrown during lifecycle hooks', () => {
@Directive({
selector: '[dir]',
inputs: ['dir'],
})
class Dir {
get dir(): any { return null; }
set dir(value: any) { throw new Error('this error is expected'); }
}
@Component({
template: '<div [dir]="3"></div>',
})
class Cmp {
ngAfterViewInit(): void {
// This lifecycle hook should never run, since attempting to bind to Dir's input will throw
// an error. If the runtime continues to run lifecycle hooks after that error, then it will
// execute this hook and throw this error, which will mask the real problem. This test
// verifies this don't happen.
throw new Error('this error is unexpected');
}
}
TestBed.configureTestingModule({
declarations: [Cmp, Dir],
});
const fixture = TestBed.createComponent(Cmp);
expect(() => fixture.detectChanges()).toThrowError('this error is expected');
});
describe('tView.firstUpdatePass', () => {
function isFirstUpdatePass() {
const lView = getLView();
const tView = lView[TVIEW];
return tView.firstUpdatePass;
}
function assertAttrValues(element: Element, value: string) {
expect(element.getAttribute('data-comp')).toEqual(value);
expect(element.getAttribute('data-dir')).toEqual(value);
}
onlyInIvy('tView instances are ivy-specific')
.it('should be marked with `firstUpdatePass` up until the template and host bindings are evaluated',
() => {
@Directive({
selector: '[dir]',
})
class Dir {
@HostBinding('attr.data-dir')
get text() {
return isFirstUpdatePass() ? 'first-update-pass' : 'post-update-pass';
}
}
@Component({
template: '<div [attr.data-comp]="text" dir></div>',
})
class Cmp {
get text() {
return isFirstUpdatePass() ? 'first-update-pass' : 'post-update-pass';
}
}
TestBed.configureTestingModule({
declarations: [Cmp, Dir],
});
const fixture = TestBed.createComponent(Cmp);
fixture.detectChanges(false);
const element = fixture.nativeElement.querySelector('div') !;
assertAttrValues(element, 'first-update-pass');
fixture.detectChanges(false);
assertAttrValues(element, 'post-update-pass');
});
onlyInIvy('tView instances are ivy-specific')
.it('tView.firstUpdatePass should be applied immediately after the first embedded view is processed',
() => {
@Directive({
selector: '[dir]',
})
class Dir {
@HostBinding('attr.data-dir')
get text() {
return isFirstUpdatePass() ? 'first-update-pass' : 'post-update-pass';
}
}
@Component({
template: `
<div *ngFor="let item of items" dir [attr.data-comp]="text">
...
</div>
`
})
class Cmp {
items = [1, 2, 3];
get text() {
return isFirstUpdatePass() ? 'first-update-pass' : 'post-update-pass';
}
}
TestBed.configureTestingModule({
declarations: [Cmp, Dir],
});
const fixture = TestBed.createComponent(Cmp);
fixture.detectChanges(false);
const elements = fixture.nativeElement.querySelectorAll('div');
assertAttrValues(elements[0], 'first-update-pass');
assertAttrValues(elements[1], 'post-update-pass');
assertAttrValues(elements[2], 'post-update-pass');
fixture.detectChanges(false);
assertAttrValues(elements[0], 'post-update-pass');
assertAttrValues(elements[1], 'post-update-pass');
assertAttrValues(elements[2], 'post-update-pass');
});
});
});