| 
									
										
										
										
											2016-10-19 21:41:04 +01: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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-20 16:38:14 +00:00
										 |  |  | import {Component, Directive, ElementRef, Injector, Input, NgModule, NgZone, SimpleChange, SimpleChanges, destroyPlatform} from '@angular/core'; | 
					
						
							| 
									
										
										
										
											2016-10-19 21:41:04 +01:00
										 |  |  | import {async} from '@angular/core/testing'; | 
					
						
							|  |  |  | import {BrowserModule} from '@angular/platform-browser'; | 
					
						
							|  |  |  | import {platformBrowserDynamic} from '@angular/platform-browser-dynamic'; | 
					
						
							| 
									
										
										
										
											2016-10-20 19:35:35 -07:00
										 |  |  | import {UpgradeComponent, UpgradeModule, downgradeComponent} from '@angular/upgrade/static'; | 
					
						
							| 
									
										
										
										
											2018-02-15 19:14:20 +02:00
										 |  |  | import * as angular from '@angular/upgrade/static/src/common/angular1'; | 
					
						
							| 
									
										
										
										
											2016-10-19 21:41:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-15 19:21:18 +02:00
										 |  |  | import {bootstrap, html, withEachNg1Version} from '../test_helpers'; | 
					
						
							| 
									
										
										
										
											2016-10-19 21:41:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-15 19:21:18 +02:00
										 |  |  | withEachNg1Version(() => { | 
					
						
							| 
									
										
										
										
											2016-10-19 21:41:04 +01:00
										 |  |  |   describe('scope/component change-detection', () => { | 
					
						
							|  |  |  |     beforeEach(() => destroyPlatform()); | 
					
						
							|  |  |  |     afterEach(() => destroyPlatform()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 16:45:23 -08:00
										 |  |  |     it('should interleave scope and component expressions', async(() => { | 
					
						
							|  |  |  |          const log: string[] = []; | 
					
						
							|  |  |  |          const l = (value: string) => { | 
					
						
							|  |  |  |            log.push(value); | 
					
						
							|  |  |  |            return value + ';'; | 
					
						
							|  |  |  |          }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          @Directive({selector: 'ng1a'}) | 
					
						
							|  |  |  |          class Ng1aComponent extends UpgradeComponent { | 
					
						
							|  |  |  |            constructor(elementRef: ElementRef, injector: Injector) { | 
					
						
							|  |  |  |              super('ng1a', elementRef, injector); | 
					
						
							|  |  |  |            } | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          @Directive({selector: 'ng1b'}) | 
					
						
							|  |  |  |          class Ng1bComponent extends UpgradeComponent { | 
					
						
							|  |  |  |            constructor(elementRef: ElementRef, injector: Injector) { | 
					
						
							|  |  |  |              super('ng1b', elementRef, injector); | 
					
						
							|  |  |  |            } | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          @Component({ | 
					
						
							|  |  |  |            selector: 'ng2', | 
					
						
							|  |  |  |            template: `{{l('2A')}}<ng1a></ng1a>{{l('2B')}}<ng1b></ng1b>{{l('2C')}}` | 
					
						
							|  |  |  |          }) | 
					
						
							|  |  |  |          class Ng2Component { | 
					
						
							|  |  |  |            l = l; | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          @NgModule({ | 
					
						
							|  |  |  |            declarations: [Ng1aComponent, Ng1bComponent, Ng2Component], | 
					
						
							|  |  |  |            entryComponents: [Ng2Component], | 
					
						
							|  |  |  |            imports: [BrowserModule, UpgradeModule] | 
					
						
							|  |  |  |          }) | 
					
						
							|  |  |  |          class Ng2Module { | 
					
						
							|  |  |  |            ngDoBootstrap() {} | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          const ng1Module = angular.module('ng1', []) | 
					
						
							|  |  |  |                                .directive('ng1a', () => ({template: '{{ l(\'ng1a\') }}'})) | 
					
						
							|  |  |  |                                .directive('ng1b', () => ({template: '{{ l(\'ng1b\') }}'})) | 
					
						
							|  |  |  |                                .directive('ng2', downgradeComponent({component: Ng2Component})) | 
					
						
							|  |  |  |                                .run(($rootScope: angular.IRootScopeService) => { | 
					
						
							|  |  |  |                                  $rootScope.l = l; | 
					
						
							|  |  |  |                                  $rootScope.reset = () => log.length = 0; | 
					
						
							|  |  |  |                                }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          const element = | 
					
						
							|  |  |  |              html('<div>{{reset(); l(\'1A\');}}<ng2>{{l(\'1B\')}}</ng2>{{l(\'1C\')}}</div>'); | 
					
						
							|  |  |  |          bootstrap(platformBrowserDynamic(), Ng2Module, element, ng1Module).then((upgrade) => { | 
					
						
							|  |  |  |            expect(document.body.textContent).toEqual('1A;2A;ng1a;2B;ng1b;2C;1C;'); | 
					
						
							|  |  |  |            expect(log).toEqual(['1A', '1C', '2A', '2B', '2C', 'ng1a', 'ng1b']); | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        })); | 
					
						
							| 
									
										
										
										
											2017-01-20 16:38:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-02 15:12:36 -08:00
										 |  |  |     it('should propagate changes to a downgraded component inside the ngZone', async(() => { | 
					
						
							|  |  |  |          const element = html('<my-app></my-app>'); | 
					
						
							|  |  |  |          let appComponent: AppComponent; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          @Component({selector: 'my-app', template: '<my-child [value]="value"></my-child>'}) | 
					
						
							|  |  |  |          class AppComponent { | 
					
						
							|  |  |  |            value?: number; | 
					
						
							|  |  |  |            constructor() { appComponent = this; } | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          @Component({ | 
					
						
							|  |  |  |            selector: 'my-child', | 
					
						
							|  |  |  |            template: '<div>{{ valueFromPromise }}</div>', | 
					
						
							|  |  |  |          }) | 
					
						
							|  |  |  |          class ChildComponent { | 
					
						
							|  |  |  |            valueFromPromise?: number; | 
					
						
							|  |  |  |            @Input() | 
					
						
							|  |  |  |            set value(v: number) { expect(NgZone.isInAngularZone()).toBe(true); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            constructor(private zone: NgZone) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            ngOnChanges(changes: SimpleChanges) { | 
					
						
							|  |  |  |              if (changes['value'].isFirstChange()) return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |              this.zone.onMicrotaskEmpty.subscribe( | 
					
						
							|  |  |  |                  () => { expect(element.textContent).toEqual('5'); }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |              // Create a micro-task to update the value to be rendered asynchronously.
 | 
					
						
							|  |  |  |              Promise.resolve().then(() => this.valueFromPromise = changes['value'].currentValue); | 
					
						
							|  |  |  |            } | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          @NgModule({ | 
					
						
							|  |  |  |            declarations: [AppComponent, ChildComponent], | 
					
						
							|  |  |  |            entryComponents: [AppComponent], | 
					
						
							|  |  |  |            imports: [BrowserModule, UpgradeModule] | 
					
						
							|  |  |  |          }) | 
					
						
							|  |  |  |          class Ng2Module { | 
					
						
							|  |  |  |            ngDoBootstrap() {} | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          const ng1Module = angular.module('ng1', []).directive( | 
					
						
							|  |  |  |              'myApp', downgradeComponent({component: AppComponent})); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |          bootstrap(platformBrowserDynamic(), Ng2Module, element, ng1Module).then((upgrade) => { | 
					
						
							|  |  |  |            appComponent.value = 5; | 
					
						
							|  |  |  |          }); | 
					
						
							|  |  |  |        })); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-20 16:38:14 +00:00
										 |  |  |     // This test demonstrates https://github.com/angular/angular/issues/6385
 | 
					
						
							|  |  |  |     // which was invalidly fixed by https://github.com/angular/angular/pull/6386
 | 
					
						
							|  |  |  |     // it('should not trigger $digest from an async operation in a watcher', async(() => {
 | 
					
						
							|  |  |  |     //      @Component({selector: 'my-app', template: ''})
 | 
					
						
							|  |  |  |     //      class AppComponent {
 | 
					
						
							|  |  |  |     //      }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //      @NgModule({declarations: [AppComponent], imports: [BrowserModule]})
 | 
					
						
							|  |  |  |     //      class Ng2Module {
 | 
					
						
							|  |  |  |     //      }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //      const adapter: UpgradeAdapter = new UpgradeAdapter(forwardRef(() => Ng2Module));
 | 
					
						
							|  |  |  |     //      const ng1Module = angular.module('ng1', []).directive(
 | 
					
						
							|  |  |  |     //          'myApp', adapter.downgradeNg2Component(AppComponent));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //      const element = html('<my-app></my-app>');
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //      adapter.bootstrap(element, ['ng1']).ready((ref) => {
 | 
					
						
							|  |  |  |     //        let doTimeout = false;
 | 
					
						
							|  |  |  |     //        let timeoutId: number;
 | 
					
						
							|  |  |  |     //        ref.ng1RootScope.$watch(() => {
 | 
					
						
							|  |  |  |     //          if (doTimeout && !timeoutId) {
 | 
					
						
							|  |  |  |     //            timeoutId = window.setTimeout(function() {
 | 
					
						
							|  |  |  |     //              timeoutId = null;
 | 
					
						
							|  |  |  |     //            }, 10);
 | 
					
						
							|  |  |  |     //          }
 | 
					
						
							|  |  |  |     //        });
 | 
					
						
							|  |  |  |     //        doTimeout = true;
 | 
					
						
							|  |  |  |     //      });
 | 
					
						
							|  |  |  |     //    }));
 | 
					
						
							| 
									
										
										
										
											2016-10-19 21:41:04 +01:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-02-15 19:21:18 +02:00
										 |  |  | }); |