| 
									
										
										
										
											2016-07-25 15:57:51 -07: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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import {NgFor, NgIf} from '@angular/common'; | 
					
						
							|  |  |  | import {Component} from '@angular/core'; | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  | import {TestBed, fakeAsync, tick} from '@angular/core/testing'; | 
					
						
							|  |  |  | import {beforeEach, ddescribe, describe, expect, iit, inject, it, xdescribe, xit} from '@angular/core/testing/testing_internal'; | 
					
						
							|  |  |  | import {FormsModule, NgForm} from '@angular/forms'; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | import {By} from '@angular/platform-browser/src/dom/debug/by'; | 
					
						
							|  |  |  | import {getDOM} from '@angular/platform-browser/src/dom/dom_adapter'; | 
					
						
							|  |  |  | import {dispatchEvent} from '@angular/platform-browser/testing/browser_util'; | 
					
						
							|  |  |  | import {ListWrapper} from '../src/facade/collection'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function main() { | 
					
						
							|  |  |  |   describe('template-driven forms integration tests', () => { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |     beforeEach(() => { | 
					
						
							|  |  |  |       TestBed.configureTestingModule({ | 
					
						
							|  |  |  |         declarations: [ | 
					
						
							|  |  |  |           StandaloneNgModel, NgModelForm, NgModelGroupForm, NgModelValidBinding, NgModelNgIfForm, | 
					
						
							|  |  |  |           NgModelRadioForm, NgModelSelectForm, NgNoFormComp, InvalidNgModelNoName, | 
					
						
							|  |  |  |           NgModelOptionsStandalone | 
					
						
							|  |  |  |         ], | 
					
						
							|  |  |  |         imports: [FormsModule] | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |       TestBed.compileComponents(); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |     describe('basic functionality', () => { | 
					
						
							|  |  |  |       it('should support ngModel for standalone fields', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(StandaloneNgModel); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.name = 'oldValue'; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            // model -> view
 | 
					
						
							|  |  |  |            const input = fixture.debugElement.query(By.css('input')).nativeElement; | 
					
						
							|  |  |  |            expect(input.value).toEqual('oldValue'); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            input.value = 'updatedValue'; | 
					
						
							|  |  |  |            dispatchEvent(input, 'input'); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            // view -> model
 | 
					
						
							|  |  |  |            expect(fixture.debugElement.componentInstance.name).toEqual('updatedValue'); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should support ngModel registration with a parent form', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.name = 'Nancy'; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            expect(form.value).toEqual({name: 'Nancy'}); | 
					
						
							|  |  |  |            expect(form.valid).toBe(false); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should support ngModelGroup', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelGroupForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.first = 'Nancy'; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.last = 'Drew'; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.email = 'some email'; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            // model -> view
 | 
					
						
							|  |  |  |            const inputs = fixture.debugElement.queryAll(By.css('input')); | 
					
						
							|  |  |  |            expect(inputs[0].nativeElement.value).toEqual('Nancy'); | 
					
						
							|  |  |  |            expect(inputs[1].nativeElement.value).toEqual('Drew'); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            inputs[0].nativeElement.value = 'Carson'; | 
					
						
							|  |  |  |            dispatchEvent(inputs[0].nativeElement, 'input'); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            // view -> model
 | 
					
						
							|  |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            expect(form.value).toEqual({name: {first: 'Carson', last: 'Drew'}, email: 'some email'}); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should add controls and control groups to form control model', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelGroupForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.first = 'Nancy'; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.last = 'Drew'; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.email = 'some email'; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            expect(form.control.get('name').value).toEqual({first: 'Nancy', last: 'Drew'}); | 
					
						
							|  |  |  |            expect(form.control.get('name.first').value).toEqual('Nancy'); | 
					
						
							|  |  |  |            expect(form.control.get('email').value).toEqual('some email'); | 
					
						
							|  |  |  |          })); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should remove controls and control groups from form control model', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelNgIfForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.emailShowing = true; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.first = 'Nancy'; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.email = 'some email'; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            expect(form.control.get('email').value).toEqual('some email'); | 
					
						
							|  |  |  |            expect(form.value).toEqual({name: {first: 'Nancy'}, email: 'some email'}); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            // should remove individual control successfully
 | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.emailShowing = false; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(form.control.get('email')).toBe(null); | 
					
						
							|  |  |  |            expect(form.value).toEqual({name: {first: 'Nancy'}}); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(form.control.get('name').value).toEqual({first: 'Nancy'}); | 
					
						
							|  |  |  |            expect(form.control.get('name.first').value).toEqual('Nancy'); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            // should remove form group successfully
 | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.groupShowing = false; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(form.control.get('name')).toBe(null); | 
					
						
							|  |  |  |            expect(form.control.get('name.first')).toBe(null); | 
					
						
							|  |  |  |            expect(form.value).toEqual({}); | 
					
						
							|  |  |  |          })); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should set status classes with ngModel', () => { | 
					
						
							|  |  |  |         const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							|  |  |  |         fixture.debugElement.componentInstance.name = 'aa'; | 
					
						
							|  |  |  |         fixture.detectChanges(); | 
					
						
							|  |  |  |         fixture.whenStable().then(() => { | 
					
						
							|  |  |  |           fixture.detectChanges(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           const input = fixture.debugElement.query(By.css('input')).nativeElement; | 
					
						
							|  |  |  |           const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |           expect(sortedClassList(input)).toEqual(['ng-invalid', 'ng-pristine', 'ng-untouched']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           dispatchEvent(input, 'blur'); | 
					
						
							|  |  |  |           fixture.detectChanges(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           expect(sortedClassList(input)).toEqual(['ng-invalid', 'ng-pristine', 'ng-touched']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           input.value = 'updatedValue'; | 
					
						
							|  |  |  |           dispatchEvent(input, 'input'); | 
					
						
							|  |  |  |           fixture.detectChanges(); | 
					
						
							|  |  |  |           expect(sortedClassList(input)).toEqual(['ng-dirty', 'ng-touched', 'ng-valid']); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should set status classes with ngModelGroup and ngForm', () => { | 
					
						
							|  |  |  |         const fixture = TestBed.createComponent(NgModelGroupForm); | 
					
						
							|  |  |  |         fixture.debugElement.componentInstance.first = ''; | 
					
						
							|  |  |  |         fixture.detectChanges(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const form = fixture.debugElement.query(By.css('form')).nativeElement; | 
					
						
							|  |  |  |         const modelGroup = fixture.debugElement.query(By.css('[ngModelGroup]')).nativeElement; | 
					
						
							|  |  |  |         const input = fixture.debugElement.query(By.css('input')).nativeElement; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // ngModelGroup creates its control asynchronously
 | 
					
						
							|  |  |  |         fixture.whenStable().then(() => { | 
					
						
							|  |  |  |           fixture.detectChanges(); | 
					
						
							|  |  |  |           expect(sortedClassList(modelGroup)).toEqual([ | 
					
						
							|  |  |  |             'ng-invalid', 'ng-pristine', 'ng-untouched' | 
					
						
							|  |  |  |           ]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           expect(sortedClassList(form)).toEqual(['ng-invalid', 'ng-pristine', 'ng-untouched']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           dispatchEvent(input, 'blur'); | 
					
						
							|  |  |  |           fixture.detectChanges(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           expect(sortedClassList(modelGroup)).toEqual(['ng-invalid', 'ng-pristine', 'ng-touched']); | 
					
						
							|  |  |  |           expect(sortedClassList(form)).toEqual(['ng-invalid', 'ng-pristine', 'ng-touched']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           input.value = 'updatedValue'; | 
					
						
							|  |  |  |           dispatchEvent(input, 'input'); | 
					
						
							|  |  |  |           fixture.detectChanges(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           expect(sortedClassList(modelGroup)).toEqual(['ng-dirty', 'ng-touched', 'ng-valid']); | 
					
						
							|  |  |  |           expect(sortedClassList(form)).toEqual(['ng-dirty', 'ng-touched', 'ng-valid']); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should not create a template-driven form when ngNoForm is used', () => { | 
					
						
							|  |  |  |         const fixture = TestBed.createComponent(NgNoFormComp); | 
					
						
							|  |  |  |         fixture.detectChanges(); | 
					
						
							|  |  |  |         expect(fixture.debugElement.children[0].providerTokens.length).toEqual(0); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |     describe('name and ngModelOptions', () => { | 
					
						
							|  |  |  |       it('should throw if ngModel has a parent form but no name attr or standalone label', () => { | 
					
						
							|  |  |  |         const fixture = TestBed.createComponent(InvalidNgModelNoName); | 
					
						
							|  |  |  |         expect(() => fixture.detectChanges()) | 
					
						
							|  |  |  |             .toThrowError(new RegExp(`name attribute must be set`)); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should not throw if ngModel has a parent form, no name attr, and a standalone label', | 
					
						
							|  |  |  |          () => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelOptionsStandalone); | 
					
						
							|  |  |  |            expect(() => fixture.detectChanges()).not.toThrow(); | 
					
						
							|  |  |  |          }); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should not register standalone ngModels with parent form', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelOptionsStandalone); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.one = 'some data'; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.two = 'should not show'; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            const inputs = fixture.debugElement.queryAll(By.css('input')); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(form.value).toEqual({one: 'some data'}); | 
					
						
							|  |  |  |            expect(inputs[1].nativeElement.value).toEqual('should not show'); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should override name attribute with ngModelOptions name if provided', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.options = {name: 'override'}; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.name = 'some data'; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            expect(form.value).toEqual({override: 'some data'}); | 
					
						
							|  |  |  |          })); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |     describe('submit and reset events', () => { | 
					
						
							|  |  |  |       it('should emit ngSubmit event on submit', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.name = 'old'; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const form = fixture.debugElement.query(By.css('form')); | 
					
						
							|  |  |  |            dispatchEvent(form.nativeElement, 'submit'); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(fixture.debugElement.componentInstance.name).toEqual('submitted'); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should mark NgForm as submitted on submit event', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            tick(); | 
					
						
							|  |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            expect(form.submitted).toBe(false); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const formEl = fixture.debugElement.query(By.css('form')).nativeElement; | 
					
						
							|  |  |  |            dispatchEvent(formEl, 'submit'); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(form.submitted).toBe(true); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-08-11 23:27:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should reset the form to empty when reset event is fired', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.name = 'should be cleared'; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-08-11 23:27:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            const formEl = fixture.debugElement.query(By.css('form')); | 
					
						
							|  |  |  |            const input = fixture.debugElement.query(By.css('input')); | 
					
						
							| 
									
										
										
										
											2016-08-11 23:27:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(input.nativeElement.value).toBe('should be cleared');  // view value
 | 
					
						
							|  |  |  |            expect(fixture.debugElement.componentInstance.name) | 
					
						
							|  |  |  |                .toBe('should be cleared');                        // ngModel value
 | 
					
						
							|  |  |  |            expect(form.value.name).toEqual('should be cleared');  // control value
 | 
					
						
							| 
									
										
										
										
											2016-08-11 23:27:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            dispatchEvent(formEl.nativeElement, 'reset'); | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-08-11 23:27:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(input.nativeElement.value).toBe('');                      // view value
 | 
					
						
							|  |  |  |            expect(fixture.debugElement.componentInstance.name).toBe(null);  // ngModel value
 | 
					
						
							|  |  |  |            expect(form.value.name).toEqual(null);                           // control value
 | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should reset the form submit state when reset button is clicked', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							|  |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            const formEl = fixture.debugElement.query(By.css('form')); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            dispatchEvent(formEl.nativeElement, 'submit'); | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  |            expect(form.submitted).toBe(true); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            dispatchEvent(formEl.nativeElement, 'reset'); | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  |            expect(form.submitted).toBe(false); | 
					
						
							|  |  |  |          })); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |     describe('valueChange and statusChange events', () => { | 
					
						
							|  |  |  |       it('should emit valueChanges and statusChanges on init', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							|  |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.name = 'aa'; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(form.valid).toEqual(true); | 
					
						
							|  |  |  |            expect(form.value).toEqual({}); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            let formValidity: string; | 
					
						
							|  |  |  |            let formValue: Object; | 
					
						
							| 
									
										
										
										
											2016-08-02 15:53:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            form.statusChanges.subscribe((status: string) => formValidity = status); | 
					
						
							|  |  |  |            form.valueChanges.subscribe((value: string) => formValue = value); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(formValidity).toEqual('INVALID'); | 
					
						
							|  |  |  |            expect(formValue).toEqual({name: 'aa'}); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should mark controls dirty before emitting the value change event', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm).form; | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            form.get('name').valueChanges.subscribe( | 
					
						
							|  |  |  |                () => { expect(form.get('name').dirty).toBe(true); }); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const inputEl = fixture.debugElement.query(By.css('input')).nativeElement; | 
					
						
							|  |  |  |            inputEl.value = 'newValue'; | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            dispatchEvent(inputEl, 'input'); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should mark controls pristine before emitting the value change event when resetting ', | 
					
						
							|  |  |  |          fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelForm); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  |            fixture.detectChanges(); | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |            const form = fixture.debugElement.children[0].injector.get(NgForm).form; | 
					
						
							|  |  |  |            const formEl = fixture.debugElement.query(By.css('form')).nativeElement; | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const inputEl = fixture.debugElement.query(By.css('input')).nativeElement; | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            inputEl.value = 'newValue'; | 
					
						
							|  |  |  |            dispatchEvent(inputEl, 'input'); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(form.get('name').pristine).toBe(false); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            form.get('name').valueChanges.subscribe( | 
					
						
							|  |  |  |                () => { expect(form.get('name').pristine).toBe(true); }); | 
					
						
							| 
									
										
										
										
											2016-07-28 14:25:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |            dispatchEvent(formEl, 'reset'); | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |          })); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |     describe('radio controls', () => { | 
					
						
							|  |  |  |       it('should support <type=radio>', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelRadioForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.food = 'fish'; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            // model -> view
 | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |            const inputs = fixture.debugElement.queryAll(By.css('input')); | 
					
						
							|  |  |  |            expect(inputs[0].nativeElement.checked).toEqual(false); | 
					
						
							|  |  |  |            expect(inputs[1].nativeElement.checked).toEqual(true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            dispatchEvent(inputs[0].nativeElement, 'change'); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            // view -> model
 | 
					
						
							|  |  |  |            expect(fixture.debugElement.componentInstance.food).toEqual('chicken'); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |            expect(inputs[1].nativeElement.checked).toEqual(false); | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should support multiple named <type=radio> groups', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelRadioForm); | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.food = 'fish'; | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.drink = 'sprite'; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            const inputs = fixture.debugElement.queryAll(By.css('input')); | 
					
						
							|  |  |  |            expect(inputs[0].nativeElement.checked).toEqual(false); | 
					
						
							|  |  |  |            expect(inputs[1].nativeElement.checked).toEqual(true); | 
					
						
							|  |  |  |            expect(inputs[2].nativeElement.checked).toEqual(false); | 
					
						
							|  |  |  |            expect(inputs[3].nativeElement.checked).toEqual(true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            dispatchEvent(inputs[0].nativeElement, 'change'); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(fixture.debugElement.componentInstance.food).toEqual('chicken'); | 
					
						
							|  |  |  |            expect(fixture.debugElement.componentInstance.drink).toEqual('sprite'); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |            expect(inputs[1].nativeElement.checked).toEqual(false); | 
					
						
							|  |  |  |            expect(inputs[2].nativeElement.checked).toEqual(false); | 
					
						
							|  |  |  |            expect(inputs[3].nativeElement.checked).toEqual(true); | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |     describe('select controls', () => { | 
					
						
							|  |  |  |       it('with option values that are objects', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelSelectForm); | 
					
						
							|  |  |  |            const comp = fixture.debugElement.componentInstance; | 
					
						
							|  |  |  |            comp.cities = [{'name': 'SF'}, {'name': 'NYC'}, {'name': 'Buffalo'}]; | 
					
						
							|  |  |  |            comp.selectedCity = comp.cities[1]; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            const select = fixture.debugElement.query(By.css('select')); | 
					
						
							|  |  |  |            const nycOption = fixture.debugElement.queryAll(By.css('option'))[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            // model -> view
 | 
					
						
							|  |  |  |            expect(select.nativeElement.value).toEqual('1: Object'); | 
					
						
							|  |  |  |            expect(nycOption.nativeElement.selected).toBe(true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            select.nativeElement.value = '2: Object'; | 
					
						
							|  |  |  |            dispatchEvent(select.nativeElement, 'change'); | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            // view -> model
 | 
					
						
							|  |  |  |            expect(comp.selectedCity.name).toEqual('Buffalo'); | 
					
						
							|  |  |  |          })); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('when new options are added', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelSelectForm); | 
					
						
							|  |  |  |            const comp = fixture.debugElement.componentInstance; | 
					
						
							|  |  |  |            comp.cities = [{'name': 'SF'}, {'name': 'NYC'}]; | 
					
						
							|  |  |  |            comp.selectedCity = comp.cities[1]; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            comp.cities.push({'name': 'Buffalo'}); | 
					
						
							|  |  |  |            comp.selectedCity = comp.cities[2]; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            const select = fixture.debugElement.query(By.css('select')); | 
					
						
							|  |  |  |            const buffalo = fixture.debugElement.queryAll(By.css('option'))[2]; | 
					
						
							|  |  |  |            expect(select.nativeElement.value).toEqual('2: Object'); | 
					
						
							|  |  |  |            expect(buffalo.nativeElement.selected).toBe(true); | 
					
						
							|  |  |  |          })); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('when options are removed', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelSelectForm); | 
					
						
							|  |  |  |            const comp = fixture.debugElement.componentInstance; | 
					
						
							|  |  |  |            comp.cities = [{'name': 'SF'}, {'name': 'NYC'}]; | 
					
						
							|  |  |  |            comp.selectedCity = comp.cities[1]; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            const select = fixture.debugElement.query(By.css('select')); | 
					
						
							|  |  |  |            expect(select.nativeElement.value).toEqual('1: Object'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            comp.cities.pop(); | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            expect(select.nativeElement.value).not.toEqual('1: Object'); | 
					
						
							|  |  |  |          })); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('when option values have same content, but different identities', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelSelectForm); | 
					
						
							|  |  |  |            const comp = fixture.debugElement.componentInstance; | 
					
						
							|  |  |  |            comp.cities = [{'name': 'SF'}, {'name': 'NYC'}, {'name': 'NYC'}]; | 
					
						
							|  |  |  |            comp.selectedCity = comp.cities[0]; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            comp.selectedCity = comp.cities[2]; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            const select = fixture.debugElement.query(By.css('select')); | 
					
						
							|  |  |  |            const secondNYC = fixture.debugElement.queryAll(By.css('option'))[2]; | 
					
						
							|  |  |  |            expect(select.nativeElement.value).toEqual('2: Object'); | 
					
						
							|  |  |  |            expect(secondNYC.nativeElement.selected).toBe(true); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     describe('ngModel corner cases', () => { | 
					
						
							|  |  |  |       it('should update the view when the model is set back to what used to be in the view', | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |          fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(StandaloneNgModel); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |            fixture.debugElement.componentInstance.name = ''; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            const input = fixture.debugElement.query(By.css('input')).nativeElement; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |            input.value = 'aa'; | 
					
						
							|  |  |  |            input.selectionStart = 1; | 
					
						
							|  |  |  |            dispatchEvent(input, 'input'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  |            expect(fixture.debugElement.componentInstance.name).toEqual('aa'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            // Programmatically update the input value to be "bb".
 | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.name = 'bb'; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  |            expect(input.value).toEqual('bb'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            // Programatically set it back to "aa".
 | 
					
						
							|  |  |  |            fixture.debugElement.componentInstance.name = 'aa'; | 
					
						
							|  |  |  |            fixture.detectChanges(); | 
					
						
							|  |  |  |            tick(); | 
					
						
							|  |  |  |            expect(input.value).toEqual('aa'); | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |       it('should not crash when validity is checked from a binding', fakeAsync(() => { | 
					
						
							|  |  |  |            const fixture = TestBed.createComponent(NgModelValidBinding); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |            tick(); | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |            expect(() => fixture.detectChanges()).not.toThrowError(); | 
					
						
							|  |  |  |          })); | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'standalone-ng-model', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <input type="text" [(ngModel)]="name"> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class StandaloneNgModel { | 
					
						
							|  |  |  |   name: string; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'ng-model-form', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <form (ngSubmit)="name='submitted'"> | 
					
						
							|  |  |  |       <input name="name" [(ngModel)]="name" minlength="10" [ngModelOptions]="options"> | 
					
						
							|  |  |  |     </form> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class NgModelForm { | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |   name: string; | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  |   options = {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'ng-model-group-form', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <form> | 
					
						
							|  |  |  |       <div ngModelGroup="name"> | 
					
						
							|  |  |  |         <input name="first" [(ngModel)]="first" required> | 
					
						
							|  |  |  |         <input name="last" [(ngModel)]="last"> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |       <input name="email" [(ngModel)]="email"> | 
					
						
							|  |  |  |     </form> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class NgModelGroupForm { | 
					
						
							|  |  |  |   first: string; | 
					
						
							|  |  |  |   last: string; | 
					
						
							|  |  |  |   email: string; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'ng-model-group-form', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <form> | 
					
						
							|  |  |  |       <div ngModelGroup="name" #group="ngModelGroup"> | 
					
						
							|  |  |  |         <input name="first" [(ngModel)]="first" required> | 
					
						
							|  |  |  |         {{ group.valid }} | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     </form> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class NgModelValidBinding { | 
					
						
							|  |  |  |   first: string; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'ng-model-ngif-form', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <form> | 
					
						
							|  |  |  |       <div ngModelGroup="name" *ngIf="groupShowing"> | 
					
						
							|  |  |  |         <input name="first" [(ngModel)]="first"> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |       <input name="email" [(ngModel)]="email" *ngIf="emailShowing"> | 
					
						
							|  |  |  |     </form> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class NgModelNgIfForm { | 
					
						
							|  |  |  |   first: string; | 
					
						
							|  |  |  |   groupShowing = true; | 
					
						
							|  |  |  |   emailShowing = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'ng-no-form', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <form ngNoForm> | 
					
						
							|  |  |  |       <input name="name"> | 
					
						
							|  |  |  |     </form> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class NgNoFormComp { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'invalid-ng-model-noname', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <form> | 
					
						
							|  |  |  |       <input [(ngModel)]="name"> | 
					
						
							|  |  |  |     </form> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class InvalidNgModelNoName { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'ng-model-options-standalone', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <form> | 
					
						
							|  |  |  |       <input name="one" [(ngModel)]="one"> | 
					
						
							|  |  |  |       <input [(ngModel)]="two" [ngModelOptions]="{standalone: true}"> | 
					
						
							|  |  |  |     </form> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class NgModelOptionsStandalone { | 
					
						
							|  |  |  |   one: string; | 
					
						
							|  |  |  |   two: string; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'ng-model-radio-form', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <form> | 
					
						
							|  |  |  |       <input type="radio" name="food" [(ngModel)]="food" value="chicken"> | 
					
						
							|  |  |  |       <input type="radio" name="food"  [(ngModel)]="food" value="fish"> | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       <input type="radio" name="drink" [(ngModel)]="drink" value="cola"> | 
					
						
							|  |  |  |       <input type="radio" name="drink" [(ngModel)]="drink" value="sprite"> | 
					
						
							|  |  |  |     </form> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class NgModelRadioForm { | 
					
						
							|  |  |  |   food: string; | 
					
						
							|  |  |  |   drink: string; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'ng-model-select-form', | 
					
						
							|  |  |  |   template: `
 | 
					
						
							|  |  |  |     <select [(ngModel)]="selectedCity"> | 
					
						
							|  |  |  |       <option *ngFor="let c of cities" [ngValue]="c"> {{c.name}} </option> | 
					
						
							|  |  |  |     </select> | 
					
						
							|  |  |  |   `
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class NgModelSelectForm { | 
					
						
							|  |  |  |   selectedCity: Object = {}; | 
					
						
							|  |  |  |   cities: any[] = []; | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 16:37:59 -07:00
										 |  |  | function sortedClassList(el: HTMLElement) { | 
					
						
							| 
									
										
										
										
											2016-07-25 15:57:51 -07:00
										 |  |  |   var l = getDOM().classList(el); | 
					
						
							|  |  |  |   ListWrapper.sort(l); | 
					
						
							|  |  |  |   return l; | 
					
						
							|  |  |  | } |