| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 23:40:19 -08:00
										 |  |  | import {RendererType2} from '../../src/render/api'; | 
					
						
							| 
									
										
										
										
											2019-04-04 11:41:52 -07:00
										 |  |  | import {getLContext} from '../../src/render3/context_discovery'; | 
					
						
							| 
									
										
										
										
											2019-09-06 23:43:16 +02:00
										 |  |  | import {AttributeMarker, ɵɵadvance, ɵɵattribute, ɵɵdefineComponent, ɵɵdefineDirective, ɵɵhostProperty, ɵɵproperty} from '../../src/render3/index'; | 
					
						
							| 
									
										
										
										
											2020-01-09 12:05:40 -08:00
										 |  |  | import {ɵɵcontainer, ɵɵcontainerRefreshEnd, ɵɵcontainerRefreshStart, ɵɵelement, ɵɵelementEnd, ɵɵelementStart, ɵɵembeddedViewEnd, ɵɵembeddedViewStart, ɵɵprojection, ɵɵprojectionDef, ɵɵtemplate, ɵɵtext, ɵɵtextInterpolate} from '../../src/render3/instructions/all'; | 
					
						
							| 
									
										
										
										
											2019-04-04 11:41:52 -07:00
										 |  |  | import {MONKEY_PATCH_KEY_NAME} from '../../src/render3/interfaces/context'; | 
					
						
							| 
									
										
										
										
											2018-12-13 15:51:47 -08:00
										 |  |  | import {RenderFlags} from '../../src/render3/interfaces/definition'; | 
					
						
							| 
									
										
										
										
											2018-12-10 23:40:19 -08:00
										 |  |  | import {RElement, Renderer3, RendererFactory3, domRendererFactory3} from '../../src/render3/interfaces/renderer'; | 
					
						
							| 
									
										
										
										
											2019-04-04 11:41:52 -07:00
										 |  |  | import {CONTEXT, HEADER_OFFSET} from '../../src/render3/interfaces/view'; | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  | import {ɵɵsanitizeUrl} from '../../src/sanitization/sanitization'; | 
					
						
							| 
									
										
										
										
											2019-07-31 13:15:50 -07:00
										 |  |  | import {Sanitizer} from '../../src/sanitization/sanitizer'; | 
					
						
							|  |  |  | import {SecurityContext} from '../../src/sanitization/security'; | 
					
						
							| 
									
										
										
										
											2019-07-14 11:11:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-01 15:19:27 +02:00
										 |  |  | import {NgIf} from './common_with_def'; | 
					
						
							| 
									
										
										
										
											2019-05-15 20:55:33 +02:00
										 |  |  | import {ComponentFixture, MockRendererFactory, renderToHtml} from './render_util'; | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-03 11:42:48 +01:00
										 |  |  | describe('render3 integration test', () => { | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   describe('render', () => { | 
					
						
							| 
									
										
										
										
											2019-05-15 20:55:33 +02:00
										 |  |  |     describe('text bindings', () => { | 
					
						
							|  |  |  |       it('should support creation-time values in text nodes', () => { | 
					
						
							|  |  |  |         function Template(rf: RenderFlags, value: string) { | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-31 14:41:07 -07:00
										 |  |  |             ɵɵtext(0, value); | 
					
						
							| 
									
										
										
										
											2018-07-26 17:22:41 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-10-22 14:10:42 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-15 20:55:33 +02:00
										 |  |  |         expect(renderToHtml(Template, 'once', 1, 1)).toEqual('once'); | 
					
						
							|  |  |  |         expect(renderToHtml(Template, 'twice', 1, 1)).toEqual('once'); | 
					
						
							|  |  |  |         expect(ngDevMode).toHaveProperties({ | 
					
						
							| 
									
										
										
										
											2019-11-01 13:06:17 -07:00
										 |  |  |           firstCreatePass: 0, | 
					
						
							| 
									
										
										
										
											2019-05-15 20:55:33 +02:00
										 |  |  |           tNode: 2, | 
					
						
							|  |  |  |           tView: 2,  // 1 for root view, 1 for template
 | 
					
						
							|  |  |  |           rendererSetText: 1, | 
					
						
							| 
									
										
										
										
											2018-10-22 14:10:42 +02:00
										 |  |  |         }); | 
					
						
							| 
									
										
										
										
											2019-05-15 20:55:33 +02:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-07-26 17:22:41 +02:00
										 |  |  |     }); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |   describe('tree', () => { | 
					
						
							|  |  |  |     interface Tree { | 
					
						
							|  |  |  |       beforeLabel?: string; | 
					
						
							|  |  |  |       subTrees?: Tree[]; | 
					
						
							|  |  |  |       afterLabel?: string; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     interface ParentCtx { | 
					
						
							|  |  |  |       beforeTree: Tree; | 
					
						
							|  |  |  |       projectedTree: Tree; | 
					
						
							|  |  |  |       afterTree: Tree; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |     function showLabel(rf: RenderFlags, ctx: {label: string | undefined}) { | 
					
						
							|  |  |  |       if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainer(0); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |       if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainerRefreshStart(0); | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |         { | 
					
						
							|  |  |  |           if (ctx.label != null) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             let rf1 = ɵɵembeddedViewStart(0, 1, 1); | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |             if (rf1 & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |               ɵɵtext(0); | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |             } | 
					
						
							|  |  |  |             if (rf1 & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-08-27 16:44:11 +02:00
										 |  |  |               ɵɵtextInterpolate(ctx.label); | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵembeddedViewEnd(); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainerRefreshEnd(); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |     function showTree(rf: RenderFlags, ctx: {tree: Tree}) { | 
					
						
							|  |  |  |       if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainer(0); | 
					
						
							|  |  |  |         ɵɵcontainer(1); | 
					
						
							|  |  |  |         ɵɵcontainer(2); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |       if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainerRefreshStart(0); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |           const rf0 = ɵɵembeddedViewStart(0, 1, 0); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |           { showLabel(rf0, {label: ctx.tree.beforeLabel}); } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |           ɵɵembeddedViewEnd(); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainerRefreshEnd(); | 
					
						
							|  |  |  |         ɵɵcontainerRefreshStart(1); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |         { | 
					
						
							|  |  |  |           for (let subTree of ctx.tree.subTrees || []) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             const rf0 = ɵɵembeddedViewStart(0, 3, 0); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |             { showTree(rf0, {tree: subTree}); } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵembeddedViewEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainerRefreshEnd(); | 
					
						
							|  |  |  |         ɵɵcontainerRefreshStart(2); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |           const rf0 = ɵɵembeddedViewStart(0, 1, 0); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |           { showLabel(rf0, {label: ctx.tree.afterLabel}); } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |           ɵɵembeddedViewEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainerRefreshEnd(); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class ChildComponent { | 
					
						
							| 
									
										
										
										
											2018-06-18 16:38:33 -07:00
										 |  |  |       // TODO(issue/24571): remove '!'.
 | 
					
						
							|  |  |  |       beforeTree !: Tree; | 
					
						
							|  |  |  |       // TODO(issue/24571): remove '!'.
 | 
					
						
							|  |  |  |       afterTree !: Tree; | 
					
						
							| 
									
										
										
										
											2019-08-12 09:26:20 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new ChildComponent; | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2018-03-29 16:41:45 -07:00
										 |  |  |         selectors: [['child']], | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |         type: ChildComponent, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 3, | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |         vars: 0, | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |         template: function ChildComponentTemplate( | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |             rf: RenderFlags, ctx: {beforeTree: Tree, afterTree: Tree}) { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵprojectionDef(); | 
					
						
							|  |  |  |             ɵɵcontainer(0); | 
					
						
							|  |  |  |             ɵɵprojection(1); | 
					
						
							|  |  |  |             ɵɵcontainer(2); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |           if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵcontainerRefreshStart(0); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |               const rf0 = ɵɵembeddedViewStart(0, 3, 0); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |               { showTree(rf0, {tree: ctx.beforeTree}); } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |               ɵɵembeddedViewEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵcontainerRefreshEnd(); | 
					
						
							|  |  |  |             ɵɵcontainerRefreshStart(2); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |               const rf0 = ɵɵembeddedViewStart(0, 3, 0); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |               { showTree(rf0, {tree: ctx.afterTree}); } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |               ɵɵembeddedViewEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵcontainerRefreshEnd(); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |           } | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         inputs: {beforeTree: 'beforeTree', afterTree: 'afterTree'} | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |     function parentTemplate(rf: RenderFlags, ctx: ParentCtx) { | 
					
						
							|  |  |  |       if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵelementStart(0, 'child'); | 
					
						
							|  |  |  |         { ɵɵcontainer(1); } | 
					
						
							|  |  |  |         ɵɵelementEnd(); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |       if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-05-24 14:20:41 -07:00
										 |  |  |         ɵɵproperty('beforeTree', ctx.beforeTree); | 
					
						
							|  |  |  |         ɵɵproperty('afterTree', ctx.afterTree); | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainerRefreshStart(1); | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |           const rf0 = ɵɵembeddedViewStart(0, 3, 0); | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |           { showTree(rf0, {tree: ctx.projectedTree}); } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |           ɵɵembeddedViewEnd(); | 
					
						
							| 
									
										
										
										
											2018-04-10 20:57:09 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |         ɵɵcontainerRefreshEnd(); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should work with a tree', () => { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const ctx: ParentCtx = { | 
					
						
							|  |  |  |         beforeTree: {subTrees: [{beforeLabel: 'a'}]}, | 
					
						
							|  |  |  |         projectedTree: {beforeLabel: 'p'}, | 
					
						
							|  |  |  |         afterTree: {afterLabel: 'z'} | 
					
						
							|  |  |  |       }; | 
					
						
							| 
									
										
										
										
											2018-03-29 12:58:41 -07:00
										 |  |  |       const defs = [ChildComponent]; | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |       expect(renderToHtml(parentTemplate, ctx, 2, 2, defs)).toEqual('<child>apz</child>'); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       ctx.projectedTree = {subTrees: [{}, {}, {subTrees: [{}, {}]}, {}]}; | 
					
						
							|  |  |  |       ctx.beforeTree.subTrees !.push({afterLabel: 'b'}); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |       expect(renderToHtml(parentTemplate, ctx, 2, 2, defs)).toEqual('<child>abz</child>'); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       ctx.projectedTree.subTrees ![1].afterLabel = 'h'; | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |       expect(renderToHtml(parentTemplate, ctx, 2, 2, defs)).toEqual('<child>abhz</child>'); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  |       ctx.beforeTree.subTrees !.push({beforeLabel: 'c'}); | 
					
						
							| 
									
										
										
										
											2018-08-18 11:14:50 -07:00
										 |  |  |       expect(renderToHtml(parentTemplate, ctx, 2, 2, defs)).toEqual('<child>abchz</child>'); | 
					
						
							| 
									
										
										
										
											2018-01-25 15:32:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       // To check the context easily:
 | 
					
						
							|  |  |  |       // console.log(JSON.stringify(ctx));
 | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  | }); | 
					
						
							| 
									
										
										
										
											2017-12-08 11:48:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  | describe('component styles', () => { | 
					
						
							|  |  |  |   it('should pass in the component styles directly into the underlying renderer', () => { | 
					
						
							|  |  |  |     class StyledComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new StyledComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: StyledComp, | 
					
						
							|  |  |  |         styles: ['div { color: red; }'], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         encapsulation: 100, | 
					
						
							|  |  |  |         selectors: [['foo']], | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: StyledComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelement(0, 'div'); | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const rendererFactory = new ProxyRenderer3Factory(); | 
					
						
							|  |  |  |     new ComponentFixture(StyledComp, {rendererFactory}); | 
					
						
							|  |  |  |     expect(rendererFactory.lastCapturedType !.styles).toEqual(['div { color: red; }']); | 
					
						
							|  |  |  |     expect(rendererFactory.lastCapturedType !.encapsulation).toEqual(100); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  | describe('component animations', () => { | 
					
						
							|  |  |  |   it('should pass in the component styles directly into the underlying renderer', () => { | 
					
						
							|  |  |  |     const animA = {name: 'a'}; | 
					
						
							|  |  |  |     const animB = {name: 'b'}; | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     class AnimComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new AnimComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: AnimComp, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 0, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         data: { | 
					
						
							|  |  |  |           animation: [ | 
					
						
							|  |  |  |             animA, | 
					
						
							|  |  |  |             animB, | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         selectors: [['foo']], | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: AnimComp) => {} | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const rendererFactory = new ProxyRenderer3Factory(); | 
					
						
							|  |  |  |     new ComponentFixture(AnimComp, {rendererFactory}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const capturedAnimations = rendererFactory.lastCapturedType !.data !['animation']; | 
					
						
							|  |  |  |     expect(Array.isArray(capturedAnimations)).toBeTruthy(); | 
					
						
							|  |  |  |     expect(capturedAnimations.length).toEqual(2); | 
					
						
							|  |  |  |     expect(capturedAnimations).toContain(animA); | 
					
						
							|  |  |  |     expect(capturedAnimations).toContain(animB); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should include animations in the renderType data array even if the array is empty', () => { | 
					
						
							|  |  |  |     class AnimComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new AnimComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: AnimComp, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 0, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         data: { | 
					
						
							|  |  |  |           animation: [], | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         selectors: [['foo']], | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: AnimComp) => {} | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const rendererFactory = new ProxyRenderer3Factory(); | 
					
						
							|  |  |  |     new ComponentFixture(AnimComp, {rendererFactory}); | 
					
						
							|  |  |  |     const data = rendererFactory.lastCapturedType !.data; | 
					
						
							|  |  |  |     expect(data.animation).toEqual([]); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should allow [@trigger] bindings to be picked up by the underlying renderer', () => { | 
					
						
							|  |  |  |     class AnimComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new AnimComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: AnimComp, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 1, | 
					
						
							|  |  |  |         selectors: [['foo']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         consts: [[AttributeMarker.Bindings, '@fooAnimation']], | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         template: (rf: RenderFlags, ctx: AnimComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |             ɵɵelement(0, 'div', 0); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           } | 
					
						
							|  |  |  |           if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-05-31 10:39:14 -07:00
										 |  |  |             ɵɵattribute('@fooAnimation', ctx.animationValue); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |       animationValue = '123'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const rendererFactory = new MockRendererFactory(['setAttribute']); | 
					
						
							|  |  |  |     const fixture = new ComponentFixture(AnimComp, {rendererFactory}); | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const renderer = rendererFactory.lastRenderer !; | 
					
						
							|  |  |  |     fixture.component.animationValue = '456'; | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const spy = renderer.spies['setAttribute']; | 
					
						
							|  |  |  |     const [elm, attr, value] = spy.calls.mostRecent().args; | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(attr).toEqual('@fooAnimation'); | 
					
						
							|  |  |  |     expect(value).toEqual('456'); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-01-03 18:24:21 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should allow creation-level [@trigger] properties to be picked up by the underlying renderer', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        class AnimComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new AnimComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: AnimComp, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 1, | 
					
						
							|  |  |  |            selectors: [['foo']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            consts: [['@fooAnimation', '']], | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            template: (rf: RenderFlags, ctx: AnimComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |                ɵɵelement(0, 'div', 0); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |              } | 
					
						
							|  |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const rendererFactory = new MockRendererFactory(['setProperty']); | 
					
						
							|  |  |  |        const fixture = new ComponentFixture(AnimComp, {rendererFactory}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const renderer = rendererFactory.lastRenderer !; | 
					
						
							|  |  |  |        fixture.update(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const spy = renderer.spies['setProperty']; | 
					
						
							|  |  |  |        const [elm, attr, value] = spy.calls.mostRecent().args; | 
					
						
							|  |  |  |        expect(attr).toEqual('@fooAnimation'); | 
					
						
							|  |  |  |      }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-24 14:20:41 -07:00
										 |  |  |   // TODO(benlesh): this test does not seem to be testing anything we could actually generate with
 | 
					
						
							|  |  |  |   // these instructions. ɵɵbind should be present in the ɵɵelementProperty call in the hostBindings,
 | 
					
						
							|  |  |  |   // however adding that causes an error because the slot has not been allocated. There is a
 | 
					
						
							|  |  |  |   // directive called `comp-with-anim`, that seems to want to be a component, but is defined as a
 | 
					
						
							|  |  |  |   // directive that is looking for a property `@fooAnim` to update.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //   it('should allow host binding animations to be picked up and rendered', () => {
 | 
					
						
							|  |  |  |   //     class ChildCompWithAnim {
 | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |   //       static ɵfac = () => new ChildCompWithAnim();
 | 
					
						
							| 
									
										
										
										
											2019-10-11 12:28:12 -07:00
										 |  |  |   //       static ɵdir = ɵɵdefineDirective({
 | 
					
						
							| 
									
										
										
										
											2019-05-24 14:20:41 -07:00
										 |  |  |   //         type: ChildCompWithAnim,
 | 
					
						
							|  |  |  |   //         selectors: [['child-comp-with-anim']],
 | 
					
						
							|  |  |  |   //         hostBindings: function(rf: RenderFlags, ctx: any, elementIndex: number): void {
 | 
					
						
							|  |  |  |   //           if (rf & RenderFlags.Update) {
 | 
					
						
							|  |  |  |   //             ɵɵelementProperty(0, '@fooAnim', ctx.exp);
 | 
					
						
							|  |  |  |   //           }
 | 
					
						
							|  |  |  |   //         },
 | 
					
						
							|  |  |  |   //       });
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //       exp = 'go';
 | 
					
						
							|  |  |  |   //     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //     class ParentComp {
 | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |   //       static ɵfac = () => new ParentComp();
 | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |   //       static ɵcmp = ɵɵdefineComponent({
 | 
					
						
							| 
									
										
										
										
											2019-05-24 14:20:41 -07:00
										 |  |  |   //         type: ParentComp,
 | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |   //         decls: 1,
 | 
					
						
							| 
									
										
										
										
											2019-05-24 14:20:41 -07:00
										 |  |  |   //         vars: 1,
 | 
					
						
							|  |  |  |   //         selectors: [['foo']],
 | 
					
						
							|  |  |  |   //         template: (rf: RenderFlags, ctx: ParentComp) => {
 | 
					
						
							|  |  |  |   //           if (rf & RenderFlags.Create) {
 | 
					
						
							|  |  |  |   //             ɵɵelement(0, 'child-comp-with-anim');
 | 
					
						
							|  |  |  |   //           }
 | 
					
						
							|  |  |  |   //         },
 | 
					
						
							|  |  |  |   //         directives: [ChildCompWithAnim]
 | 
					
						
							|  |  |  |   //       });
 | 
					
						
							|  |  |  |   //     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //     const rendererFactory = new MockRendererFactory(['setProperty']);
 | 
					
						
							|  |  |  |   //     const fixture = new ComponentFixture(ParentComp, {rendererFactory});
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //     const renderer = rendererFactory.lastRenderer !;
 | 
					
						
							|  |  |  |   //     fixture.update();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //     const spy = renderer.spies['setProperty'];
 | 
					
						
							|  |  |  |   //     const [elm, attr, value] = spy.calls.mostRecent().args;
 | 
					
						
							|  |  |  |   //     expect(attr).toEqual('@fooAnim');
 | 
					
						
							|  |  |  |   //   });
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  | }); | 
					
						
							| 
									
										
										
										
											2018-09-05 15:23:59 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  | describe('element discovery', () => { | 
					
						
							|  |  |  |   it('should only monkey-patch immediate child nodes in a component', () => { | 
					
						
							|  |  |  |     class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: StructuredComp, | 
					
						
							|  |  |  |         selectors: [['structured-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 2, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: StructuredComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelementStart(0, 'div'); | 
					
						
							|  |  |  |             ɵɵelementStart(1, 'p'); | 
					
						
							|  |  |  |             ɵɵelementEnd(); | 
					
						
							|  |  |  |             ɵɵelementEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           if (rf & RenderFlags.Update) { | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const host = fixture.hostElement; | 
					
						
							|  |  |  |     const parent = host.querySelector('div') as any; | 
					
						
							|  |  |  |     const child = host.querySelector('p') as any; | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(parent[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |     expect(child[MONKEY_PATCH_KEY_NAME]).toBeFalsy(); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should only monkey-patch immediate child nodes in a sub component', () => { | 
					
						
							|  |  |  |     class ChildComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new ChildComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: ChildComp, | 
					
						
							|  |  |  |         selectors: [['child-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 3, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: ChildComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelement(0, 'div'); | 
					
						
							|  |  |  |             ɵɵelement(1, 'div'); | 
					
						
							|  |  |  |             ɵɵelement(2, 'div'); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     class ParentComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new ParentComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: ParentComp, | 
					
						
							|  |  |  |         selectors: [['parent-comp']], | 
					
						
							|  |  |  |         directives: [ChildComp], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 2, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: ParentComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelementStart(0, 'section'); | 
					
						
							|  |  |  |             ɵɵelementStart(1, 'child-comp'); | 
					
						
							|  |  |  |             ɵɵelementEnd(); | 
					
						
							|  |  |  |             ɵɵelementEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const fixture = new ComponentFixture(ParentComp); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const host = fixture.hostElement; | 
					
						
							|  |  |  |     const child = host.querySelector('child-comp') as any; | 
					
						
							|  |  |  |     expect(child[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const [kid1, kid2, kid3] = Array.from(host.querySelectorAll('child-comp > *')); | 
					
						
							|  |  |  |     expect(kid1[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |     expect(kid2[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |     expect(kid3[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should only monkey-patch immediate child nodes in an embedded template container', () => { | 
					
						
							|  |  |  |     class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: StructuredComp, | 
					
						
							|  |  |  |         selectors: [['structured-comp']], | 
					
						
							|  |  |  |         directives: [NgIf], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 2, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 1, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         consts: [['ngIf', '']], | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         template: (rf: RenderFlags, ctx: StructuredComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelementStart(0, 'section'); | 
					
						
							|  |  |  |             ɵɵtemplate(1, (rf, ctx) => { | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |               if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |                 ɵɵelementStart(0, 'div'); | 
					
						
							|  |  |  |                 ɵɵelement(1, 'p'); | 
					
						
							|  |  |  |                 ɵɵelementEnd(); | 
					
						
							|  |  |  |                 ɵɵelement(2, 'div'); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |               } | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |             }, 3, 0, 'ng-template', 0); | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelementEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-09-06 23:43:16 +02:00
										 |  |  |             ɵɵadvance(1); | 
					
						
							| 
									
										
										
										
											2019-05-24 14:20:41 -07:00
										 |  |  |             ɵɵproperty('ngIf', true); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const host = fixture.hostElement; | 
					
						
							|  |  |  |     const [section, div1, p, div2] = Array.from(host.querySelectorAll('section, div, p')); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(section.nodeName.toLowerCase()).toBe('section'); | 
					
						
							|  |  |  |     expect(section[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(div1.nodeName.toLowerCase()).toBe('div'); | 
					
						
							|  |  |  |     expect(div1[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(p.nodeName.toLowerCase()).toBe('p'); | 
					
						
							|  |  |  |     expect(p[MONKEY_PATCH_KEY_NAME]).toBeFalsy(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(div2.nodeName.toLowerCase()).toBe('div'); | 
					
						
							|  |  |  |     expect(div2[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should return a context object from a given dom node', () => { | 
					
						
							|  |  |  |     class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: StructuredComp, | 
					
						
							|  |  |  |         selectors: [['structured-comp']], | 
					
						
							|  |  |  |         directives: [NgIf], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 2, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: StructuredComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelement(0, 'section'); | 
					
						
							|  |  |  |             ɵɵelement(1, 'div'); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const section = fixture.hostElement.querySelector('section') !; | 
					
						
							|  |  |  |     const sectionContext = getLContext(section) !; | 
					
						
							|  |  |  |     const sectionLView = sectionContext.lView !; | 
					
						
							|  |  |  |     expect(sectionContext.nodeIndex).toEqual(HEADER_OFFSET); | 
					
						
							|  |  |  |     expect(sectionLView.length).toBeGreaterThan(HEADER_OFFSET); | 
					
						
							|  |  |  |     expect(sectionContext.native).toBe(section); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const div = fixture.hostElement.querySelector('div') !; | 
					
						
							|  |  |  |     const divContext = getLContext(div) !; | 
					
						
							|  |  |  |     const divLView = divContext.lView !; | 
					
						
							|  |  |  |     expect(divContext.nodeIndex).toEqual(HEADER_OFFSET + 1); | 
					
						
							|  |  |  |     expect(divLView.length).toBeGreaterThan(HEADER_OFFSET); | 
					
						
							|  |  |  |     expect(divContext.native).toBe(div); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(divLView).toBe(sectionLView); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should cache the element context on a element was pre-emptively monkey-patched', () => { | 
					
						
							|  |  |  |     class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: StructuredComp, | 
					
						
							|  |  |  |         selectors: [['structured-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: StructuredComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelement(0, 'section'); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const section = fixture.hostElement.querySelector('section') !as any; | 
					
						
							|  |  |  |     const result1 = section[MONKEY_PATCH_KEY_NAME]; | 
					
						
							|  |  |  |     expect(Array.isArray(result1)).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const context = getLContext(section) !; | 
					
						
							|  |  |  |     const result2 = section[MONKEY_PATCH_KEY_NAME]; | 
					
						
							|  |  |  |     expect(Array.isArray(result2)).toBeFalsy(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(result2).toBe(context); | 
					
						
							|  |  |  |     expect(result2.lView).toBe(result1); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should cache the element context on an intermediate element that isn\'t pre-emptively monkey-patched', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: StructuredComp, | 
					
						
							|  |  |  |            selectors: [['structured-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 2, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							|  |  |  |            template: (rf: RenderFlags, ctx: StructuredComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |                ɵɵelementStart(0, 'section'); | 
					
						
							|  |  |  |                ɵɵelement(1, 'p'); | 
					
						
							|  |  |  |                ɵɵelementEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  |              } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |        fixture.update(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const section = fixture.hostElement.querySelector('section') !as any; | 
					
						
							|  |  |  |        expect(section[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const p = fixture.hostElement.querySelector('p') !as any; | 
					
						
							|  |  |  |        expect(p[MONKEY_PATCH_KEY_NAME]).toBeFalsy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const pContext = getLContext(p) !; | 
					
						
							|  |  |  |        expect(pContext.native).toBe(p); | 
					
						
							|  |  |  |        expect(p[MONKEY_PATCH_KEY_NAME]).toBe(pContext); | 
					
						
							|  |  |  |      }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should be able to pull in element context data even if the element is decorated using styling', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: StructuredComp, | 
					
						
							|  |  |  |            selectors: [['structured-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							|  |  |  |            template: (rf: RenderFlags, ctx: StructuredComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-09-09 13:14:26 -07:00
										 |  |  |                ɵɵelement(0, 'section'); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  |              } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |        fixture.update(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const section = fixture.hostElement.querySelector('section') !as any; | 
					
						
							|  |  |  |        const result1 = section[MONKEY_PATCH_KEY_NAME]; | 
					
						
							|  |  |  |        expect(Array.isArray(result1)).toBeTruthy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const elementResult = result1[HEADER_OFFSET];  // first element
 | 
					
						
							| 
									
										
										
										
											2019-05-28 10:31:01 -07:00
										 |  |  |        expect(elementResult).toBe(section); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |        const context = getLContext(section) !; | 
					
						
							|  |  |  |        const result2 = section[MONKEY_PATCH_KEY_NAME]; | 
					
						
							|  |  |  |        expect(Array.isArray(result2)).toBeFalsy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(context.native).toBe(section); | 
					
						
							|  |  |  |      }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should monkey-patch immediate child nodes in a content-projected region with a reference to the parent component', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        /* | 
					
						
							|  |  |  |          <!-- DOM view --> | 
					
						
							|  |  |  |          <section> | 
					
						
							|  |  |  |            <projection-comp> | 
					
						
							|  |  |  |              welcome | 
					
						
							|  |  |  |              <header> | 
					
						
							|  |  |  |                <h1> | 
					
						
							|  |  |  |                  <p>this content is projected</p> | 
					
						
							|  |  |  |                  this content is projected also | 
					
						
							|  |  |  |                </h1> | 
					
						
							|  |  |  |              </header> | 
					
						
							|  |  |  |            </projection-comp> | 
					
						
							|  |  |  |          </section> | 
					
						
							|  |  |  |        */ | 
					
						
							|  |  |  |        class ProjectorComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new ProjectorComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: ProjectorComp, | 
					
						
							|  |  |  |            selectors: [['projector-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 4, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							|  |  |  |            template: (rf: RenderFlags, ctx: ProjectorComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |                ɵɵprojectionDef(); | 
					
						
							|  |  |  |                ɵɵtext(0, 'welcome'); | 
					
						
							|  |  |  |                ɵɵelementStart(1, 'header'); | 
					
						
							|  |  |  |                ɵɵelementStart(2, 'h1'); | 
					
						
							|  |  |  |                ɵɵprojection(3); | 
					
						
							|  |  |  |                ɵɵelementEnd(); | 
					
						
							|  |  |  |                ɵɵelementEnd(); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  |              } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |              if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  |              } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        class ParentComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new ParentComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: ParentComp, | 
					
						
							|  |  |  |            selectors: [['parent-comp']], | 
					
						
							|  |  |  |            directives: [ProjectorComp], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 5, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							|  |  |  |            template: (rf: RenderFlags, ctx: ParentComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |                ɵɵelementStart(0, 'section'); | 
					
						
							|  |  |  |                ɵɵelementStart(1, 'projector-comp'); | 
					
						
							|  |  |  |                ɵɵelementStart(2, 'p'); | 
					
						
							|  |  |  |                ɵɵtext(3, 'this content is projected'); | 
					
						
							|  |  |  |                ɵɵelementEnd(); | 
					
						
							|  |  |  |                ɵɵtext(4, 'this content is projected also'); | 
					
						
							|  |  |  |                ɵɵelementEnd(); | 
					
						
							|  |  |  |                ɵɵelementEnd(); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |              } | 
					
						
							|  |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const fixture = new ComponentFixture(ParentComp); | 
					
						
							|  |  |  |        fixture.update(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const host = fixture.hostElement; | 
					
						
							|  |  |  |        const textNode = host.firstChild as any; | 
					
						
							|  |  |  |        const section = host.querySelector('section') !as any; | 
					
						
							|  |  |  |        const projectorComp = host.querySelector('projector-comp') !as any; | 
					
						
							|  |  |  |        const header = host.querySelector('header') !as any; | 
					
						
							|  |  |  |        const h1 = host.querySelector('h1') !as any; | 
					
						
							|  |  |  |        const p = host.querySelector('p') !as any; | 
					
						
							|  |  |  |        const pText = p.firstChild as any; | 
					
						
							|  |  |  |        const projectedTextNode = p.nextSibling; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(projectorComp.children).toContain(header); | 
					
						
							|  |  |  |        expect(h1.children).toContain(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(textNode[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |        expect(section[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |        expect(projectorComp[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |        expect(header[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |        expect(h1[MONKEY_PATCH_KEY_NAME]).toBeFalsy(); | 
					
						
							|  |  |  |        expect(p[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  |        expect(pText[MONKEY_PATCH_KEY_NAME]).toBeFalsy(); | 
					
						
							|  |  |  |        expect(projectedTextNode[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const parentContext = getLContext(section) !; | 
					
						
							|  |  |  |        const shadowContext = getLContext(header) !; | 
					
						
							|  |  |  |        const projectedContext = getLContext(p) !; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const parentComponentData = parentContext.lView; | 
					
						
							|  |  |  |        const shadowComponentData = shadowContext.lView; | 
					
						
							|  |  |  |        const projectedComponentData = projectedContext.lView; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(projectedComponentData).toBe(parentComponentData); | 
					
						
							|  |  |  |        expect(shadowComponentData).not.toBe(parentComponentData); | 
					
						
							|  |  |  |      }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should return `null` when an element context is retrieved that isn\'t situated in Angular', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        const elm1 = document.createElement('div'); | 
					
						
							|  |  |  |        const context1 = getLContext(elm1); | 
					
						
							|  |  |  |        expect(context1).toBeFalsy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const elm2 = document.createElement('div'); | 
					
						
							|  |  |  |        document.body.appendChild(elm2); | 
					
						
							|  |  |  |        const context2 = getLContext(elm2); | 
					
						
							|  |  |  |        expect(context2).toBeFalsy(); | 
					
						
							|  |  |  |      }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should return `null` when an element context is retrieved that is a DOM node that was not created by Angular', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: StructuredComp, | 
					
						
							|  |  |  |            selectors: [['structured-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							|  |  |  |            template: (rf: RenderFlags, ctx: StructuredComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |                ɵɵelement(0, 'section'); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |              } | 
					
						
							|  |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |        fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        const section = fixture.hostElement.querySelector('section') !as any; | 
					
						
							|  |  |  |        const manuallyCreatedElement = document.createElement('div'); | 
					
						
							|  |  |  |        section.appendChild(manuallyCreatedElement); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        const context = getLContext(manuallyCreatedElement); | 
					
						
							|  |  |  |        expect(context).toBeFalsy(); | 
					
						
							|  |  |  |      }); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should by default monkey-patch the bootstrap component with context details', () => { | 
					
						
							|  |  |  |     class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: StructuredComp, | 
					
						
							|  |  |  |         selectors: [['structured-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 0, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: StructuredComp) => {} | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const hostElm = fixture.hostElement; | 
					
						
							|  |  |  |     const component = fixture.component; | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const componentLView = (component as any)[MONKEY_PATCH_KEY_NAME]; | 
					
						
							|  |  |  |     expect(Array.isArray(componentLView)).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const hostLView = (hostElm as any)[MONKEY_PATCH_KEY_NAME]; | 
					
						
							|  |  |  |     expect(hostLView).toBe(componentLView); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const context1 = getLContext(hostElm) !; | 
					
						
							|  |  |  |     expect(context1.lView).toBe(hostLView); | 
					
						
							|  |  |  |     expect(context1.native).toEqual(hostElm); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const context2 = getLContext(component) !; | 
					
						
							|  |  |  |     expect(context2).toBe(context1); | 
					
						
							|  |  |  |     expect(context2.lView).toBe(hostLView); | 
					
						
							|  |  |  |     expect(context2.native).toEqual(hostElm); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should by default monkey-patch the directives with LView so that they can be examined', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        let myDir1Instance: MyDir1|null = null; | 
					
						
							|  |  |  |        let myDir2Instance: MyDir2|null = null; | 
					
						
							|  |  |  |        let myDir3Instance: MyDir2|null = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        class MyDir1 { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => myDir1Instance = new MyDir1(); | 
					
						
							| 
									
										
										
										
											2019-10-11 12:28:12 -07:00
										 |  |  |          static ɵdir = ɵɵdefineDirective({type: MyDir1, selectors: [['', 'my-dir-1', '']]}); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        } | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        class MyDir2 { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => myDir2Instance = new MyDir2(); | 
					
						
							| 
									
										
										
										
											2019-10-11 12:28:12 -07:00
										 |  |  |          static ɵdir = ɵɵdefineDirective({type: MyDir2, selectors: [['', 'my-dir-2', '']]}); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        } | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        class MyDir3 { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => myDir3Instance = new MyDir2(); | 
					
						
							| 
									
										
										
										
											2019-10-11 12:28:12 -07:00
										 |  |  |          static ɵdir = ɵɵdefineDirective({type: MyDir3, selectors: [['', 'my-dir-3', '']]}); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        class StructuredComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new StructuredComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: StructuredComp, | 
					
						
							|  |  |  |            selectors: [['structured-comp']], | 
					
						
							|  |  |  |            directives: [MyDir1, MyDir2, MyDir3], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 2, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            consts: [['my-dir-1', '', 'my-dir-2', ''], ['my-dir-3']], | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            template: (rf: RenderFlags, ctx: StructuredComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |                ɵɵelement(0, 'div', 0); | 
					
						
							|  |  |  |                ɵɵelement(1, 'div', 1); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  |              } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const fixture = new ComponentFixture(StructuredComp); | 
					
						
							|  |  |  |        fixture.update(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const hostElm = fixture.hostElement; | 
					
						
							|  |  |  |        const div1 = hostElm.querySelector('div:first-child') !as any; | 
					
						
							|  |  |  |        const div2 = hostElm.querySelector('div:last-child') !as any; | 
					
						
							|  |  |  |        const context = getLContext(hostElm) !; | 
					
						
							|  |  |  |        const componentView = context.lView[context.nodeIndex]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(componentView).toContain(myDir1Instance); | 
					
						
							|  |  |  |        expect(componentView).toContain(myDir2Instance); | 
					
						
							|  |  |  |        expect(componentView).toContain(myDir3Instance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(Array.isArray((myDir1Instance as any)[MONKEY_PATCH_KEY_NAME])).toBeTruthy(); | 
					
						
							|  |  |  |        expect(Array.isArray((myDir2Instance as any)[MONKEY_PATCH_KEY_NAME])).toBeTruthy(); | 
					
						
							|  |  |  |        expect(Array.isArray((myDir3Instance as any)[MONKEY_PATCH_KEY_NAME])).toBeTruthy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const d1Context = getLContext(myDir1Instance) !; | 
					
						
							|  |  |  |        const d2Context = getLContext(myDir2Instance) !; | 
					
						
							|  |  |  |        const d3Context = getLContext(myDir3Instance) !; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(d1Context.lView).toEqual(componentView); | 
					
						
							|  |  |  |        expect(d2Context.lView).toEqual(componentView); | 
					
						
							|  |  |  |        expect(d3Context.lView).toEqual(componentView); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect((myDir1Instance as any)[MONKEY_PATCH_KEY_NAME]).toBe(d1Context); | 
					
						
							|  |  |  |        expect((myDir2Instance as any)[MONKEY_PATCH_KEY_NAME]).toBe(d2Context); | 
					
						
							|  |  |  |        expect((myDir3Instance as any)[MONKEY_PATCH_KEY_NAME]).toBe(d3Context); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(d1Context.nodeIndex).toEqual(HEADER_OFFSET); | 
					
						
							|  |  |  |        expect(d1Context.native).toBe(div1); | 
					
						
							|  |  |  |        expect(d1Context.directives as any[]).toEqual([myDir1Instance, myDir2Instance]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(d2Context.nodeIndex).toEqual(HEADER_OFFSET); | 
					
						
							|  |  |  |        expect(d2Context.native).toBe(div1); | 
					
						
							|  |  |  |        expect(d2Context.directives as any[]).toEqual([myDir1Instance, myDir2Instance]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(d3Context.nodeIndex).toEqual(HEADER_OFFSET + 1); | 
					
						
							|  |  |  |        expect(d3Context.native).toBe(div2); | 
					
						
							|  |  |  |        expect(d3Context.directives as any[]).toEqual([myDir3Instance]); | 
					
						
							|  |  |  |      }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should monkey-patch the exact same context instance of the DOM node, component and any directives on the same element', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        let myDir1Instance: MyDir1|null = null; | 
					
						
							|  |  |  |        let myDir2Instance: MyDir2|null = null; | 
					
						
							|  |  |  |        let childComponentInstance: ChildComp|null = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        class MyDir1 { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => myDir1Instance = new MyDir1(); | 
					
						
							| 
									
										
										
										
											2019-10-11 12:28:12 -07:00
										 |  |  |          static ɵdir = ɵɵdefineDirective({type: MyDir1, selectors: [['', 'my-dir-1', '']]}); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        } | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        class MyDir2 { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => myDir2Instance = new MyDir2(); | 
					
						
							| 
									
										
										
										
											2019-10-11 12:28:12 -07:00
										 |  |  |          static ɵdir = ɵɵdefineDirective({type: MyDir2, selectors: [['', 'my-dir-2', '']]}); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        class ChildComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => childComponentInstance = new ChildComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: ChildComp, | 
					
						
							|  |  |  |            selectors: [['child-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							|  |  |  |            template: (rf: RenderFlags, ctx: ChildComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |                ɵɵelement(0, 'div'); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  |              } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        class ParentComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new ParentComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: ParentComp, | 
					
						
							|  |  |  |            selectors: [['parent-comp']], | 
					
						
							|  |  |  |            directives: [ChildComp, MyDir1, MyDir2], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            consts: [['my-dir-1', '', 'my-dir-2', '']], | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            template: (rf: RenderFlags, ctx: ParentComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |                ɵɵelement(0, 'child-comp', 0); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  |              } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const fixture = new ComponentFixture(ParentComp); | 
					
						
							|  |  |  |        fixture.update(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const childCompHostElm = fixture.hostElement.querySelector('child-comp') !as any; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const lView = childCompHostElm[MONKEY_PATCH_KEY_NAME]; | 
					
						
							|  |  |  |        expect(Array.isArray(lView)).toBeTruthy(); | 
					
						
							|  |  |  |        expect((myDir1Instance as any)[MONKEY_PATCH_KEY_NAME]).toBe(lView); | 
					
						
							|  |  |  |        expect((myDir2Instance as any)[MONKEY_PATCH_KEY_NAME]).toBe(lView); | 
					
						
							|  |  |  |        expect((childComponentInstance as any)[MONKEY_PATCH_KEY_NAME]).toBe(lView); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        const childNodeContext = getLContext(childCompHostElm) !; | 
					
						
							|  |  |  |        expect(childNodeContext.component).toBeFalsy(); | 
					
						
							|  |  |  |        expect(childNodeContext.directives).toBeFalsy(); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(myDir1Instance); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(myDir2Instance); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(childComponentInstance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(getLContext(myDir1Instance)).toBe(childNodeContext); | 
					
						
							|  |  |  |        expect(childNodeContext.component).toBeFalsy(); | 
					
						
							|  |  |  |        expect(childNodeContext.directives !.length).toEqual(2); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(myDir1Instance, false); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(myDir2Instance, false); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(childComponentInstance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(getLContext(myDir2Instance)).toBe(childNodeContext); | 
					
						
							|  |  |  |        expect(childNodeContext.component).toBeFalsy(); | 
					
						
							|  |  |  |        expect(childNodeContext.directives !.length).toEqual(2); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(myDir1Instance, false); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(myDir2Instance, false); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(childComponentInstance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        expect(getLContext(childComponentInstance)).toBe(childNodeContext); | 
					
						
							|  |  |  |        expect(childNodeContext.component).toBeTruthy(); | 
					
						
							|  |  |  |        expect(childNodeContext.directives !.length).toEqual(2); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(myDir1Instance, false); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(myDir2Instance, false); | 
					
						
							|  |  |  |        assertMonkeyPatchValueIsLView(childComponentInstance, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        function assertMonkeyPatchValueIsLView(value: any, yesOrNo = true) { | 
					
						
							|  |  |  |          expect(Array.isArray((value as any)[MONKEY_PATCH_KEY_NAME])).toBe(yesOrNo); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  |      }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   it('should monkey-patch sub components with the view data and then replace them with the context result once a lookup occurs', | 
					
						
							|  |  |  |      () => { | 
					
						
							|  |  |  |        class ChildComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new ChildComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: ChildComp, | 
					
						
							|  |  |  |            selectors: [['child-comp']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 3, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							|  |  |  |            template: (rf: RenderFlags, ctx: ChildComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |                ɵɵelement(0, 'div'); | 
					
						
							|  |  |  |                ɵɵelement(1, 'div'); | 
					
						
							|  |  |  |                ɵɵelement(2, 'div'); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  |              } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        class ParentComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |          static ɵfac = () => new ParentComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |          static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            type: ParentComp, | 
					
						
							|  |  |  |            selectors: [['parent-comp']], | 
					
						
							|  |  |  |            directives: [ChildComp], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |            decls: 2, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |            vars: 0, | 
					
						
							|  |  |  |            template: (rf: RenderFlags, ctx: ParentComp) => { | 
					
						
							|  |  |  |              if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |                ɵɵelementStart(0, 'section'); | 
					
						
							|  |  |  |                ɵɵelementStart(1, 'child-comp'); | 
					
						
							|  |  |  |                ɵɵelementEnd(); | 
					
						
							|  |  |  |                ɵɵelementEnd(); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |              } | 
					
						
							|  |  |  |            } | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        } | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        const fixture = new ComponentFixture(ParentComp); | 
					
						
							|  |  |  |        fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        const host = fixture.hostElement; | 
					
						
							|  |  |  |        const child = host.querySelector('child-comp') as any; | 
					
						
							|  |  |  |        expect(child[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        const context = getLContext(child) !; | 
					
						
							|  |  |  |        expect(child[MONKEY_PATCH_KEY_NAME]).toBeTruthy(); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        const componentData = context.lView[context.nodeIndex]; | 
					
						
							|  |  |  |        const component = componentData[CONTEXT]; | 
					
						
							|  |  |  |        expect(component instanceof ChildComp).toBeTruthy(); | 
					
						
							|  |  |  |        expect(component[MONKEY_PATCH_KEY_NAME]).toBe(context.lView); | 
					
						
							| 
									
										
										
										
											2018-08-29 13:52:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |        const componentContext = getLContext(component) !; | 
					
						
							|  |  |  |        expect(component[MONKEY_PATCH_KEY_NAME]).toBe(componentContext); | 
					
						
							|  |  |  |        expect(componentContext.nodeIndex).toEqual(context.nodeIndex); | 
					
						
							|  |  |  |        expect(componentContext.native).toEqual(context.native); | 
					
						
							|  |  |  |        expect(componentContext.lView).toEqual(context.lView); | 
					
						
							|  |  |  |      }); | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
										
										
											2018-08-22 16:57:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  | describe('sanitization', () => { | 
					
						
							|  |  |  |   it('should sanitize data using the provided sanitization interface', () => { | 
					
						
							|  |  |  |     class SanitizationComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new SanitizationComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: SanitizationComp, | 
					
						
							|  |  |  |         selectors: [['sanitize-this']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 1, | 
					
						
							|  |  |  |         template: (rf: RenderFlags, ctx: SanitizationComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-05-17 18:49:21 -07:00
										 |  |  |             ɵɵelement(0, 'a'); | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-05-24 14:20:41 -07:00
										 |  |  |             ɵɵproperty('href', ctx.href, ɵɵsanitizeUrl); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |       private href = ''; | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |       updateLink(href: any) { this.href = href; } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const sanitizer = new LocalSanitizer((value) => { return 'http://bar'; }); | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const fixture = new ComponentFixture(SanitizationComp, {sanitizer}); | 
					
						
							|  |  |  |     fixture.component.updateLink('http://foo'); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const anchor = fixture.hostElement.querySelector('a') !; | 
					
						
							|  |  |  |     expect(anchor.getAttribute('href')).toEqual('http://bar'); | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     fixture.component.updateLink(sanitizer.bypassSecurityTrustUrl('http://foo')); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2019-01-03 10:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(anchor.getAttribute('href')).toEqual('http://foo'); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2019-01-03 10:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |   it('should sanitize HostBindings data using provided sanitization interface', () => { | 
					
						
							|  |  |  |     let hostBindingDir: UnsafeUrlHostBindingDir; | 
					
						
							|  |  |  |     class UnsafeUrlHostBindingDir { | 
					
						
							|  |  |  |       // @HostBinding()
 | 
					
						
							|  |  |  |       cite: any = 'http://cite-dir-value'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => hostBindingDir = new UnsafeUrlHostBindingDir(); | 
					
						
							| 
									
										
										
										
											2019-10-11 12:28:12 -07:00
										 |  |  |       static ɵdir = ɵɵdefineDirective({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: UnsafeUrlHostBindingDir, | 
					
						
							|  |  |  |         selectors: [['', 'unsafeUrlHostBindingDir', '']], | 
					
						
							| 
									
										
										
										
											2020-01-09 12:05:40 -08:00
										 |  |  |         hostVars: 1, | 
					
						
							| 
									
										
										
										
											2019-09-06 23:43:16 +02:00
										 |  |  |         hostBindings: (rf: RenderFlags, ctx: any) => { | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           if (rf & RenderFlags.Update) { | 
					
						
							| 
									
										
										
										
											2019-07-14 11:11:10 +02:00
										 |  |  |             ɵɵhostProperty('cite', ctx.cite, ɵɵsanitizeUrl); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-01-03 10:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     class SimpleComp { | 
					
						
							| 
									
										
										
										
											2019-10-11 14:18:45 -07:00
										 |  |  |       static ɵfac = () => new SimpleComp(); | 
					
						
							| 
									
										
										
										
											2019-10-10 14:57:15 -07:00
										 |  |  |       static ɵcmp = ɵɵdefineComponent({ | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         type: SimpleComp, | 
					
						
							|  |  |  |         selectors: [['sanitize-this']], | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         decls: 1, | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         vars: 0, | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |         consts: [['unsafeUrlHostBindingDir', '']], | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |         template: (rf: RenderFlags, ctx: SimpleComp) => { | 
					
						
							|  |  |  |           if (rf & RenderFlags.Create) { | 
					
						
							| 
									
										
										
										
											2019-09-23 20:08:51 +02:00
										 |  |  |             ɵɵelement(0, 'blockquote', 0); | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |           } | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         directives: [UnsafeUrlHostBindingDir] | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-01-03 10:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const sanitizer = new LocalSanitizer((value) => 'http://bar'); | 
					
						
							| 
									
										
										
										
											2019-01-03 10:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const fixture = new ComponentFixture(SimpleComp, {sanitizer}); | 
					
						
							|  |  |  |     hostBindingDir !.cite = 'http://foo'; | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2019-01-03 10:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     const anchor = fixture.hostElement.querySelector('blockquote') !; | 
					
						
							|  |  |  |     expect(anchor.getAttribute('cite')).toEqual('http://bar'); | 
					
						
							| 
									
										
										
										
											2019-01-03 10:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     hostBindingDir !.cite = sanitizer.bypassSecurityTrustUrl('http://foo'); | 
					
						
							|  |  |  |     fixture.update(); | 
					
						
							| 
									
										
										
										
											2019-01-03 10:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-08 15:03:54 -08:00
										 |  |  |     expect(anchor.getAttribute('cite')).toEqual('http://foo'); | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2017-12-01 14:23:03 -08:00
										 |  |  | }); | 
					
						
							| 
									
										
										
										
											2018-05-09 15:30:16 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | class LocalSanitizedValue { | 
					
						
							|  |  |  |   constructor(public value: any) {} | 
					
						
							|  |  |  |   toString() { return this.value; } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class LocalSanitizer implements Sanitizer { | 
					
						
							|  |  |  |   constructor(private _interceptor: (value: string|null|any) => string) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   sanitize(context: SecurityContext, value: LocalSanitizedValue|string|null): string|null { | 
					
						
							|  |  |  |     if (value instanceof LocalSanitizedValue) { | 
					
						
							|  |  |  |       return value.toString(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return this._interceptor(value); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bypassSecurityTrustHtml(value: string) {} | 
					
						
							|  |  |  |   bypassSecurityTrustStyle(value: string) {} | 
					
						
							|  |  |  |   bypassSecurityTrustScript(value: string) {} | 
					
						
							|  |  |  |   bypassSecurityTrustResourceUrl(value: string) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bypassSecurityTrustUrl(value: string) { return new LocalSanitizedValue(value); } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-31 11:14:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-06 18:50:57 -07:00
										 |  |  | class ProxyRenderer3Factory implements RendererFactory3 { | 
					
						
							| 
									
										
										
										
											2018-07-31 11:14:06 -07:00
										 |  |  |   lastCapturedType: RendererType2|null = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   createRenderer(hostElement: RElement|null, rendererType: RendererType2|null): Renderer3 { | 
					
						
							|  |  |  |     this.lastCapturedType = rendererType; | 
					
						
							|  |  |  |     return domRendererFactory3.createRenderer(hostElement, rendererType); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-12-13 15:51:47 -08:00
										 |  |  | } |