304 lines
13 KiB
TypeScript
304 lines
13 KiB
TypeScript
/**
|
|
* @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 {ErrorHandler, Injector, RenderComponentType, RootRenderer, Sanitizer, SecurityContext, ViewEncapsulation, WrappedValue, getDebugNode} from '@angular/core';
|
|
import {getDebugContext} from '@angular/core/src/errors';
|
|
import {ArgumentType, BindingFlags, DebugContext, NodeDef, NodeFlags, OutputType, RootData, Services, ViewData, ViewDefinition, ViewFlags, ViewHandleEventFn, ViewUpdateFn, anchorDef, asElementData, elementDef, rootRenderNodes, textDef, viewDef} from '@angular/core/src/view/index';
|
|
import {TestBed} from '@angular/core/testing';
|
|
import {getDOM} from '@angular/platform-browser/src/dom/dom_adapter';
|
|
|
|
import {ARG_TYPE_VALUES, checkNodeInlineOrDynamic, createRootView, isBrowser, recordNodeToRemove} from './helper';
|
|
|
|
export function main() {
|
|
describe(`View Elements`, () => {
|
|
function compViewDef(
|
|
nodes: NodeDef[], updateDirectives?: ViewUpdateFn, updateRenderer?: ViewUpdateFn,
|
|
viewFlags: ViewFlags = ViewFlags.None): ViewDefinition {
|
|
return viewDef(viewFlags, nodes, updateDirectives, updateRenderer);
|
|
}
|
|
|
|
function createAndGetRootNodes(
|
|
viewDef: ViewDefinition, context?: any): {rootNodes: any[], view: ViewData} {
|
|
const view = createRootView(viewDef, context);
|
|
const rootNodes = rootRenderNodes(view);
|
|
return {rootNodes, view};
|
|
}
|
|
|
|
describe('create', () => {
|
|
it('should create elements without parents', () => {
|
|
const rootNodes = createAndGetRootNodes(compViewDef([
|
|
elementDef(NodeFlags.None, null !, null !, 0, 'span')
|
|
])).rootNodes;
|
|
expect(rootNodes.length).toBe(1);
|
|
expect(getDOM().nodeName(rootNodes[0]).toLowerCase()).toBe('span');
|
|
});
|
|
|
|
it('should create views with multiple root elements', () => {
|
|
const rootNodes = createAndGetRootNodes(compViewDef([
|
|
elementDef(NodeFlags.None, null !, null !, 0, 'span'),
|
|
elementDef(NodeFlags.None, null !, null !, 0, 'span')
|
|
])).rootNodes;
|
|
expect(rootNodes.length).toBe(2);
|
|
});
|
|
|
|
it('should create elements with parents', () => {
|
|
const rootNodes = createAndGetRootNodes(compViewDef([
|
|
elementDef(NodeFlags.None, null !, null !, 1, 'div'),
|
|
elementDef(NodeFlags.None, null !, null !, 0, 'span'),
|
|
])).rootNodes;
|
|
expect(rootNodes.length).toBe(1);
|
|
const spanEl = getDOM().childNodes(rootNodes[0])[0];
|
|
expect(getDOM().nodeName(spanEl).toLowerCase()).toBe('span');
|
|
});
|
|
|
|
it('should set fixed attributes', () => {
|
|
const rootNodes = createAndGetRootNodes(compViewDef([
|
|
elementDef(NodeFlags.None, null !, null !, 0, 'div', [['title', 'a']]),
|
|
])).rootNodes;
|
|
expect(rootNodes.length).toBe(1);
|
|
expect(getDOM().getAttribute(rootNodes[0], 'title')).toBe('a');
|
|
});
|
|
|
|
it('should add debug information to the renderer', () => {
|
|
const someContext = new Object();
|
|
const {view, rootNodes} = createAndGetRootNodes(
|
|
compViewDef([elementDef(NodeFlags.None, null !, null !, 0, 'div')]), someContext);
|
|
expect(getDebugNode(rootNodes[0]) !.nativeNode).toBe(asElementData(view, 0).renderElement);
|
|
});
|
|
});
|
|
|
|
describe('change properties', () => {
|
|
ARG_TYPE_VALUES.forEach((inlineDynamic) => {
|
|
it(`should update via strategy ${inlineDynamic}`, () => {
|
|
|
|
const {view, rootNodes} = createAndGetRootNodes(compViewDef(
|
|
[
|
|
elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'input', null !,
|
|
[
|
|
[BindingFlags.TypeProperty, 'title', SecurityContext.NONE],
|
|
[BindingFlags.TypeProperty, 'value', SecurityContext.NONE]
|
|
]),
|
|
],
|
|
null !, (check, view) => {
|
|
checkNodeInlineOrDynamic(check, view, 0, inlineDynamic, ['v1', 'v2']);
|
|
}));
|
|
|
|
Services.checkAndUpdateView(view);
|
|
|
|
const el = rootNodes[0];
|
|
expect(getDOM().getProperty(el, 'title')).toBe('v1');
|
|
expect(getDOM().getProperty(el, 'value')).toBe('v2');
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('change attributes', () => {
|
|
ARG_TYPE_VALUES.forEach((inlineDynamic) => {
|
|
it(`should update via strategy ${inlineDynamic}`, () => {
|
|
const {view, rootNodes} = createAndGetRootNodes(compViewDef(
|
|
[
|
|
elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'div', null !,
|
|
[
|
|
[BindingFlags.TypeElementAttribute, 'a1', SecurityContext.NONE],
|
|
[BindingFlags.TypeElementAttribute, 'a2', SecurityContext.NONE]
|
|
]),
|
|
],
|
|
null !, (check, view) => {
|
|
checkNodeInlineOrDynamic(check, view, 0, inlineDynamic, ['v1', 'v2']);
|
|
}));
|
|
|
|
Services.checkAndUpdateView(view);
|
|
|
|
const el = rootNodes[0];
|
|
expect(getDOM().getAttribute(el, 'a1')).toBe('v1');
|
|
expect(getDOM().getAttribute(el, 'a2')).toBe('v2');
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('change classes', () => {
|
|
ARG_TYPE_VALUES.forEach((inlineDynamic) => {
|
|
it(`should update via strategy ${inlineDynamic}`, () => {
|
|
const {view, rootNodes} = createAndGetRootNodes(compViewDef(
|
|
[
|
|
elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'div', null !,
|
|
[
|
|
[BindingFlags.TypeElementClass, 'c1', null !],
|
|
[BindingFlags.TypeElementClass, 'c2', null !]
|
|
]),
|
|
],
|
|
(check, view) => {
|
|
checkNodeInlineOrDynamic(check, view, 0, inlineDynamic, [true, true]);
|
|
}));
|
|
|
|
Services.checkAndUpdateView(view);
|
|
|
|
const el = rootNodes[0];
|
|
expect(getDOM().hasClass(el, 'c1')).toBeTruthy();
|
|
expect(getDOM().hasClass(el, 'c2')).toBeTruthy();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('change styles', () => {
|
|
ARG_TYPE_VALUES.forEach((inlineDynamic) => {
|
|
it(`should update via strategy ${inlineDynamic}`, () => {
|
|
const {view, rootNodes} = createAndGetRootNodes(compViewDef(
|
|
[
|
|
elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'div', null !,
|
|
[
|
|
[BindingFlags.TypeElementStyle, 'width', 'px'],
|
|
[BindingFlags.TypeElementStyle, 'color', null !]
|
|
]),
|
|
],
|
|
null !, (check, view) => {
|
|
checkNodeInlineOrDynamic(check, view, 0, inlineDynamic, [10, 'red']);
|
|
}));
|
|
|
|
Services.checkAndUpdateView(view);
|
|
|
|
const el = rootNodes[0];
|
|
expect(getDOM().getStyle(el, 'width')).toBe('10px');
|
|
expect(getDOM().getStyle(el, 'color')).toBe('red');
|
|
});
|
|
});
|
|
});
|
|
|
|
if (isBrowser()) {
|
|
describe('listen to DOM events', () => {
|
|
function createAndAttachAndGetRootNodes(viewDef: ViewDefinition):
|
|
{rootNodes: any[], view: ViewData} {
|
|
const result = createAndGetRootNodes(viewDef);
|
|
// Note: We need to append the node to the document.body, otherwise `click` events
|
|
// won't work in IE.
|
|
result.rootNodes.forEach((node) => {
|
|
document.body.appendChild(node);
|
|
recordNodeToRemove(node);
|
|
});
|
|
return result;
|
|
}
|
|
|
|
it('should listen to DOM events', () => {
|
|
const handleEventSpy = jasmine.createSpy('handleEvent');
|
|
const removeListenerSpy =
|
|
spyOn(HTMLElement.prototype, 'removeEventListener').and.callThrough();
|
|
const {view, rootNodes} = createAndAttachAndGetRootNodes(compViewDef([elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'button', null !, null !, [[null !, 'click']],
|
|
handleEventSpy)]));
|
|
|
|
rootNodes[0].click();
|
|
|
|
expect(handleEventSpy).toHaveBeenCalled();
|
|
let handleEventArgs = handleEventSpy.calls.mostRecent().args;
|
|
expect(handleEventArgs[0]).toBe(view);
|
|
expect(handleEventArgs[1]).toBe('click');
|
|
expect(handleEventArgs[2]).toBeTruthy();
|
|
|
|
Services.destroyView(view);
|
|
|
|
expect(removeListenerSpy).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should listen to window events', () => {
|
|
const handleEventSpy = jasmine.createSpy('handleEvent');
|
|
const addListenerSpy = spyOn(window, 'addEventListener');
|
|
const removeListenerSpy = spyOn(window, 'removeEventListener');
|
|
const {view, rootNodes} = createAndAttachAndGetRootNodes(compViewDef([elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'button', null !, null !,
|
|
[['window', 'windowClick']], handleEventSpy)]));
|
|
|
|
expect(addListenerSpy).toHaveBeenCalled();
|
|
expect(addListenerSpy.calls.mostRecent().args[0]).toBe('windowClick');
|
|
addListenerSpy.calls.mostRecent().args[1]({name: 'windowClick'});
|
|
|
|
expect(handleEventSpy).toHaveBeenCalled();
|
|
const handleEventArgs = handleEventSpy.calls.mostRecent().args;
|
|
expect(handleEventArgs[0]).toBe(view);
|
|
expect(handleEventArgs[1]).toBe('window:windowClick');
|
|
expect(handleEventArgs[2]).toBeTruthy();
|
|
|
|
Services.destroyView(view);
|
|
|
|
expect(removeListenerSpy).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should listen to document events', () => {
|
|
const handleEventSpy = jasmine.createSpy('handleEvent');
|
|
const addListenerSpy = spyOn(document, 'addEventListener');
|
|
const removeListenerSpy = spyOn(document, 'removeEventListener');
|
|
const {view, rootNodes} = createAndAttachAndGetRootNodes(compViewDef([elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'button', null !, null !,
|
|
[['document', 'documentClick']], handleEventSpy)]));
|
|
|
|
expect(addListenerSpy).toHaveBeenCalled();
|
|
expect(addListenerSpy.calls.mostRecent().args[0]).toBe('documentClick');
|
|
addListenerSpy.calls.mostRecent().args[1]({name: 'documentClick'});
|
|
|
|
expect(handleEventSpy).toHaveBeenCalled();
|
|
const handleEventArgs = handleEventSpy.calls.mostRecent().args;
|
|
expect(handleEventArgs[0]).toBe(view);
|
|
expect(handleEventArgs[1]).toBe('document:documentClick');
|
|
expect(handleEventArgs[2]).toBeTruthy();
|
|
|
|
Services.destroyView(view);
|
|
|
|
expect(removeListenerSpy).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should preventDefault only if the handler returns false', () => {
|
|
let eventHandlerResult: any;
|
|
let preventDefaultSpy: jasmine.Spy = undefined !;
|
|
|
|
const {view, rootNodes} = createAndAttachAndGetRootNodes(compViewDef([elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'button', null !, null !, [[null !, 'click']],
|
|
(view, eventName, event) => {
|
|
preventDefaultSpy = spyOn(event, 'preventDefault').and.callThrough();
|
|
return eventHandlerResult;
|
|
})]));
|
|
|
|
eventHandlerResult = undefined;
|
|
rootNodes[0].click();
|
|
expect(preventDefaultSpy).not.toHaveBeenCalled();
|
|
|
|
eventHandlerResult = true;
|
|
rootNodes[0].click();
|
|
expect(preventDefaultSpy).not.toHaveBeenCalled();
|
|
|
|
eventHandlerResult = 'someString';
|
|
rootNodes[0].click();
|
|
expect(preventDefaultSpy).not.toHaveBeenCalled();
|
|
|
|
eventHandlerResult = false;
|
|
rootNodes[0].click();
|
|
expect(preventDefaultSpy).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should report debug info on event errors', () => {
|
|
const handleErrorSpy = spyOn(TestBed.get(ErrorHandler), 'handleError');
|
|
const addListenerSpy = spyOn(HTMLElement.prototype, 'addEventListener').and.callThrough();
|
|
const {view, rootNodes} = createAndAttachAndGetRootNodes(compViewDef([elementDef(
|
|
NodeFlags.None, null !, null !, 0, 'button', null !, null !, [[null !, 'click']],
|
|
() => { throw new Error('Test'); })]));
|
|
|
|
addListenerSpy.calls.mostRecent().args[1]('SomeEvent');
|
|
const err = handleErrorSpy.calls.mostRecent().args[0];
|
|
expect(err).toBeTruthy();
|
|
expect(err.message).toBe('Test');
|
|
const debugCtx = getDebugContext(err);
|
|
expect(debugCtx.view).toBe(view);
|
|
expect(debugCtx.nodeIndex).toBe(0);
|
|
});
|
|
});
|
|
}
|
|
});
|
|
}
|