303 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			303 lines
		
	
	
		
			9.0 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 {DoCheck, ViewEncapsulation} from '../../src/core';
 | |
| import {getRenderedText} from '../../src/render3/component';
 | |
| import {LifecycleHooksFeature, defineComponent, markDirty} from '../../src/render3/index';
 | |
| import {bind, container, containerRefreshEnd, containerRefreshStart, elementEnd, elementProperty, elementStart, embeddedViewEnd, embeddedViewStart, text, textBinding, tick} from '../../src/render3/instructions';
 | |
| import {createRendererType2} from '../../src/view/index';
 | |
| 
 | |
| import {getRendererFactory2} from './imported_renderer2';
 | |
| import {containerEl, renderComponent, renderToHtml, requestAnimationFrame, toHtml} from './render_util';
 | |
| 
 | |
| describe('component', () => {
 | |
|   class CounterComponent {
 | |
|     count = 0;
 | |
| 
 | |
|     increment() { this.count++; }
 | |
| 
 | |
|     static ngComponentDef = defineComponent({
 | |
|       type: CounterComponent,
 | |
|       tag: 'counter',
 | |
|       template: function(ctx: CounterComponent, cm: boolean) {
 | |
|         if (cm) {
 | |
|           text(0);
 | |
|         }
 | |
|         textBinding(0, bind(ctx.count));
 | |
|       },
 | |
|       factory: () => new CounterComponent,
 | |
|       inputs: {count: 'count'},
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   describe('renderComponent', () => {
 | |
|     it('should render on initial call', () => {
 | |
|       renderComponent(CounterComponent);
 | |
|       expect(toHtml(containerEl)).toEqual('0');
 | |
|     });
 | |
| 
 | |
|     it('should re-render on input change or method invocation', () => {
 | |
|       const component = renderComponent(CounterComponent);
 | |
|       expect(toHtml(containerEl)).toEqual('0');
 | |
|       component.count = 123;
 | |
|       markDirty(component);
 | |
|       expect(toHtml(containerEl)).toEqual('0');
 | |
|       requestAnimationFrame.flush();
 | |
|       expect(toHtml(containerEl)).toEqual('123');
 | |
|       component.increment();
 | |
|       markDirty(component);
 | |
|       expect(toHtml(containerEl)).toEqual('123');
 | |
|       requestAnimationFrame.flush();
 | |
|       expect(toHtml(containerEl)).toEqual('124');
 | |
|     });
 | |
| 
 | |
|   });
 | |
| 
 | |
| });
 | |
| 
 | |
| describe('component with a container', () => {
 | |
| 
 | |
|   function showItems(ctx: {items: string[]}, cm: boolean) {
 | |
|     if (cm) {
 | |
|       container(0);
 | |
|     }
 | |
|     containerRefreshStart(0);
 | |
|     {
 | |
|       for (const item of ctx.items) {
 | |
|         const cm0 = embeddedViewStart(0);
 | |
|         {
 | |
|           if (cm0) {
 | |
|             text(0);
 | |
|           }
 | |
|           textBinding(0, bind(item));
 | |
|         }
 | |
|         embeddedViewEnd();
 | |
|       }
 | |
|     }
 | |
|     containerRefreshEnd();
 | |
|   }
 | |
| 
 | |
|   class WrapperComponent {
 | |
|     items: string[];
 | |
|     static ngComponentDef = defineComponent({
 | |
|       type: WrapperComponent,
 | |
|       tag: 'wrapper',
 | |
|       template: function ChildComponentTemplate(ctx: {items: string[]}, cm: boolean) {
 | |
|         if (cm) {
 | |
|           container(0);
 | |
|         }
 | |
|         containerRefreshStart(0);
 | |
|         {
 | |
|           const cm0 = embeddedViewStart(0);
 | |
|           { showItems({items: ctx.items}, cm0); }
 | |
|           embeddedViewEnd();
 | |
|         }
 | |
|         containerRefreshEnd();
 | |
|       },
 | |
|       factory: () => new WrapperComponent,
 | |
|       inputs: {items: 'items'}
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   function template(ctx: {items: string[]}, cm: boolean) {
 | |
|     if (cm) {
 | |
|       elementStart(0, WrapperComponent);
 | |
|       elementEnd();
 | |
|     }
 | |
|     elementProperty(0, 'items', bind(ctx.items));
 | |
|   }
 | |
| 
 | |
|   it('should re-render on input change', () => {
 | |
|     const ctx: {items: string[]} = {items: ['a']};
 | |
|     expect(renderToHtml(template, ctx)).toEqual('<wrapper>a</wrapper>');
 | |
| 
 | |
|     ctx.items = [...ctx.items, 'b'];
 | |
|     expect(renderToHtml(template, ctx)).toEqual('<wrapper>ab</wrapper>');
 | |
|   });
 | |
| 
 | |
| });
 | |
| 
 | |
| // TODO: add tests with Native once tests are run in real browser (domino doesn't support shadow
 | |
| // root)
 | |
| describe('encapsulation', () => {
 | |
|   class WrapperComponent {
 | |
|     static ngComponentDef = defineComponent({
 | |
|       type: WrapperComponent,
 | |
|       tag: 'wrapper',
 | |
|       template: function(ctx: WrapperComponent, cm: boolean) {
 | |
|         if (cm) {
 | |
|           elementStart(0, EncapsulatedComponent);
 | |
|           elementEnd();
 | |
|         }
 | |
|       },
 | |
|       factory: () => new WrapperComponent,
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   class EncapsulatedComponent {
 | |
|     static ngComponentDef = defineComponent({
 | |
|       type: EncapsulatedComponent,
 | |
|       tag: 'encapsulated',
 | |
|       template: function(ctx: EncapsulatedComponent, cm: boolean) {
 | |
|         if (cm) {
 | |
|           text(0, 'foo');
 | |
|           elementStart(1, LeafComponent);
 | |
|           elementEnd();
 | |
|         }
 | |
|       },
 | |
|       factory: () => new EncapsulatedComponent,
 | |
|       rendererType:
 | |
|           createRendererType2({encapsulation: ViewEncapsulation.Emulated, styles: [], data: {}}),
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   class LeafComponent {
 | |
|     static ngComponentDef = defineComponent({
 | |
|       type: LeafComponent,
 | |
|       tag: 'leaf',
 | |
|       template: function(ctx: LeafComponent, cm: boolean) {
 | |
|         if (cm) {
 | |
|           elementStart(0, 'span');
 | |
|           { text(1, 'bar'); }
 | |
|           elementEnd();
 | |
|         }
 | |
|       },
 | |
|       factory: () => new LeafComponent,
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   it('should encapsulate children, but not host nor grand children', () => {
 | |
|     renderComponent(WrapperComponent, {rendererFactory: getRendererFactory2(document)});
 | |
|     expect(containerEl.outerHTML)
 | |
|         .toMatch(
 | |
|             /<div host=""><encapsulated _nghost-c(\d+)="">foo<leaf _ngcontent-c\1=""><span>bar<\/span><\/leaf><\/encapsulated><\/div>/);
 | |
|   });
 | |
| 
 | |
|   it('should encapsulate host', () => {
 | |
|     renderComponent(EncapsulatedComponent, {rendererFactory: getRendererFactory2(document)});
 | |
|     expect(containerEl.outerHTML)
 | |
|         .toMatch(
 | |
|             /<div host="" _nghost-c(\d+)="">foo<leaf _ngcontent-c\1=""><span>bar<\/span><\/leaf><\/div>/);
 | |
|   });
 | |
| 
 | |
|   it('should encapsulate host and children with different attributes', () => {
 | |
|     class WrapperComponentWith {
 | |
|       static ngComponentDef = defineComponent({
 | |
|         type: WrapperComponentWith,
 | |
|         tag: 'wrapper',
 | |
|         template: function(ctx: WrapperComponentWith, cm: boolean) {
 | |
|           if (cm) {
 | |
|             elementStart(0, LeafComponentwith);
 | |
|             elementEnd();
 | |
|           }
 | |
|         },
 | |
|         factory: () => new WrapperComponentWith,
 | |
|         rendererType:
 | |
|             createRendererType2({encapsulation: ViewEncapsulation.Emulated, styles: [], data: {}}),
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     class LeafComponentwith {
 | |
|       static ngComponentDef = defineComponent({
 | |
|         type: LeafComponentwith,
 | |
|         tag: 'leaf',
 | |
|         template: function(ctx: LeafComponentwith, cm: boolean) {
 | |
|           if (cm) {
 | |
|             elementStart(0, 'span');
 | |
|             { text(1, 'bar'); }
 | |
|             elementEnd();
 | |
|           }
 | |
|         },
 | |
|         factory: () => new LeafComponentwith,
 | |
|         rendererType:
 | |
|             createRendererType2({encapsulation: ViewEncapsulation.Emulated, styles: [], data: {}}),
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     renderComponent(WrapperComponentWith, {rendererFactory: getRendererFactory2(document)});
 | |
|     expect(containerEl.outerHTML)
 | |
|         .toMatch(
 | |
|             /<div host="" _nghost-c(\d+)=""><leaf _ngcontent-c\1="" _nghost-c(\d+)=""><span _ngcontent-c\2="">bar<\/span><\/leaf><\/div>/);
 | |
|   });
 | |
| 
 | |
| });
 | |
| 
 | |
| describe('recursive components', () => {
 | |
|   let events: string[] = [];
 | |
|   let count = 0;
 | |
| 
 | |
|   class TreeNode {
 | |
|     constructor(
 | |
|         public value: number, public depth: number, public left: TreeNode|null,
 | |
|         public right: TreeNode|null) {}
 | |
|   }
 | |
| 
 | |
|   class TreeComponent {
 | |
|     data: TreeNode = _buildTree(0);
 | |
| 
 | |
|     ngDoCheck() { events.push('check' + this.data.value); }
 | |
| 
 | |
|     static ngComponentDef = defineComponent({
 | |
|       type: TreeComponent,
 | |
|       tag: 'tree-comp',
 | |
|       factory: () => new TreeComponent(),
 | |
|       template: (ctx: TreeComponent, cm: boolean) => {
 | |
|         if (cm) {
 | |
|           text(0);
 | |
|           container(1);
 | |
|           container(2);
 | |
|         }
 | |
|         textBinding(0, bind(ctx.data.value));
 | |
|         containerRefreshStart(1);
 | |
|         {
 | |
|           if (ctx.data.left != null) {
 | |
|             if (embeddedViewStart(0)) {
 | |
|               elementStart(0, TreeComponent);
 | |
|               elementEnd();
 | |
|             }
 | |
|             elementProperty(0, 'data', bind(ctx.data.left));
 | |
|             embeddedViewEnd();
 | |
|           }
 | |
|         }
 | |
|         containerRefreshEnd();
 | |
|         containerRefreshStart(2);
 | |
|         {
 | |
|           if (ctx.data.right != null) {
 | |
|             if (embeddedViewStart(0)) {
 | |
|               elementStart(0, TreeComponent);
 | |
|               elementEnd();
 | |
|             }
 | |
|             elementProperty(0, 'data', bind(ctx.data.right));
 | |
|             embeddedViewEnd();
 | |
|           }
 | |
|         }
 | |
|         containerRefreshEnd();
 | |
|       },
 | |
|       inputs: {data: 'data'}
 | |
|     });
 | |
|   }
 | |
| 
 | |
| 
 | |
|   function _buildTree(currDepth: number): TreeNode {
 | |
|     const children = currDepth < 2 ? _buildTree(currDepth + 1) : null;
 | |
|     const children2 = currDepth < 2 ? _buildTree(currDepth + 1) : null;
 | |
|     return new TreeNode(count++, currDepth, children, children2);
 | |
|   }
 | |
| 
 | |
|   it('should check each component just once', () => {
 | |
|     const comp = renderComponent(TreeComponent, {hostFeatures: [LifecycleHooksFeature]});
 | |
|     expect(getRenderedText(comp)).toEqual('6201534');
 | |
|     expect(events).toEqual(['check6', 'check2', 'check0', 'check1', 'check5', 'check3', 'check4']);
 | |
| 
 | |
|     events = [];
 | |
|     tick(comp);
 | |
|     expect(events).toEqual(['check6', 'check2', 'check0', 'check1', 'check5', 'check3', 'check4']);
 | |
|   });
 | |
| });
 |