/**
 * @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, Directive, EventEmitter, Input, Output, forwardRef} from '@angular/core';
import {ComponentFixture, TestBed, fakeAsync, tick} from '@angular/core/testing';
import {ControlValueAccessor, FormArray, FormControl, FormGroup, FormGroupDirective, FormsModule, NG_ASYNC_VALIDATORS, NG_VALIDATORS, NgControl, ReactiveFormsModule, Validator, Validators} from '@angular/forms';
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';
import {AbstractControl} from '../src/model';
export function main() {
  describe('reactive forms integration tests', () => {
    beforeEach(() => {
      TestBed.configureTestingModule({
        imports: [FormsModule, ReactiveFormsModule],
        declarations: [
          FormControlComp, FormGroupComp, FormArrayComp, FormArrayNestedGroup,
          FormControlNameSelect, FormControlNumberInput, FormControlRadioButtons, WrappedValue,
          WrappedValueForm, MyInput, MyInputForm, FormGroupNgModel, FormControlNgModel,
          LoginIsEmptyValidator, LoginIsEmptyWrapper, UniqLoginValidator, UniqLoginWrapper
        ]
      });
      TestBed.compileComponents();
    });
    describe('basic functionality', () => {
      it('should work with single controls', () => {
        const fixture = TestBed.createComponent(FormControlComp);
        const control = new FormControl('old value');
        fixture.debugElement.componentInstance.control = control;
        fixture.detectChanges();
        // model -> view
        const input = fixture.debugElement.query(By.css('input'));
        expect(input.nativeElement.value).toEqual('old value');
        input.nativeElement.value = 'updated value';
        dispatchEvent(input.nativeElement, 'input');
        // view -> model
        expect(control.value).toEqual('updated value');
      });
      it('should work with formGroups (model -> view)', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        fixture.debugElement.componentInstance.form =
            new FormGroup({'login': new FormControl('loginValue')});
        fixture.detectChanges();
        const input = fixture.debugElement.query(By.css('input'));
        expect(input.nativeElement.value).toEqual('loginValue');
      });
      it('work with formGroups (view -> model)', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const form = new FormGroup({'login': new FormControl('oldValue')});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        const input = fixture.debugElement.query(By.css('input'));
        input.nativeElement.value = 'updatedValue';
        dispatchEvent(input.nativeElement, 'input');
        expect(form.value).toEqual({'login': 'updatedValue'});
      });
      it('should update DOM elements when rebinding the form group', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        fixture.debugElement.componentInstance.form =
            new FormGroup({'login': new FormControl('oldValue')});
        fixture.detectChanges();
        fixture.debugElement.componentInstance.form =
            new FormGroup({'login': new FormControl('newValue')});
        fixture.detectChanges();
        const input = fixture.debugElement.query(By.css('input'));
        expect(input.nativeElement.value).toEqual('newValue');
      });
    });
    describe('form arrays', () => {
      it('should support form arrays', () => {
        const fixture = TestBed.createComponent(FormArrayComp);
        const cityArray = new FormArray([new FormControl('SF'), new FormControl('NY')]);
        const form = new FormGroup({cities: cityArray});
        fixture.debugElement.componentInstance.form = form;
        fixture.debugElement.componentInstance.cityArray = cityArray;
        fixture.detectChanges();
        const inputs = fixture.debugElement.queryAll(By.css('input'));
        // model -> view
        expect(inputs[0].nativeElement.value).toEqual('SF');
        expect(inputs[1].nativeElement.value).toEqual('NY');
        expect(form.value).toEqual({cities: ['SF', 'NY']});
        inputs[0].nativeElement.value = 'LA';
        dispatchEvent(inputs[0].nativeElement, 'input');
        fixture.detectChanges();
        //  view -> model
        expect(form.value).toEqual({cities: ['LA', 'NY']});
      });
      it('should support pushing new controls to form arrays', () => {
        const fixture = TestBed.createComponent(FormArrayComp);
        const cityArray = new FormArray([new FormControl('SF'), new FormControl('NY')]);
        const form = new FormGroup({cities: cityArray});
        fixture.debugElement.componentInstance.form = form;
        fixture.debugElement.componentInstance.cityArray = cityArray;
        fixture.detectChanges();
        cityArray.push(new FormControl('LA'));
        fixture.detectChanges();
        const inputs = fixture.debugElement.queryAll(By.css('input'));
        expect(inputs[2].nativeElement.value).toEqual('LA');
        expect(form.value).toEqual({cities: ['SF', 'NY', 'LA']});
      });
      it('should support form groups nested in form arrays', () => {
        const fixture = TestBed.createComponent(FormArrayNestedGroup);
        const cityArray = new FormArray([
          new FormGroup({town: new FormControl('SF'), state: new FormControl('CA')}),
          new FormGroup({town: new FormControl('NY'), state: new FormControl('NY')})
        ]);
        const form = new FormGroup({cities: cityArray});
        fixture.debugElement.componentInstance.form = form;
        fixture.debugElement.componentInstance.cityArray = cityArray;
        fixture.detectChanges();
        const inputs = fixture.debugElement.queryAll(By.css('input'));
        expect(inputs[0].nativeElement.value).toEqual('SF');
        expect(inputs[1].nativeElement.value).toEqual('CA');
        expect(inputs[2].nativeElement.value).toEqual('NY');
        expect(inputs[3].nativeElement.value).toEqual('NY');
        expect(form.value).toEqual({
          cities: [{town: 'SF', state: 'CA'}, {town: 'NY', state: 'NY'}]
        });
        inputs[0].nativeElement.value = 'LA';
        dispatchEvent(inputs[0].nativeElement, 'input');
        fixture.detectChanges();
        expect(form.value).toEqual({
          cities: [{town: 'LA', state: 'CA'}, {town: 'NY', state: 'NY'}]
        });
      });
    });
    describe('programmatic changes', () => {
      it('should update the value in the DOM when setValue is called', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const login = new FormControl('oldValue');
        const form = new FormGroup({'login': login});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        login.setValue('newValue');
        fixture.detectChanges();
        const input = fixture.debugElement.query(By.css('input'));
        expect(input.nativeElement.value).toEqual('newValue');
      });
    });
    describe('user input', () => {
      it('should mark controls as touched after interacting with the DOM control', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const login = new FormControl('oldValue');
        const form = new FormGroup({'login': login});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        const loginEl = fixture.debugElement.query(By.css('input'));
        expect(login.touched).toBe(false);
        dispatchEvent(loginEl.nativeElement, 'blur');
        expect(login.touched).toBe(true);
      });
    });
    describe('submit and reset events', () => {
      it('should emit ngSubmit event on submit', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        fixture.debugElement.componentInstance.form =
            new FormGroup({'login': new FormControl('loginValue')});
        fixture.debugElement.componentInstance.data = 'should be changed';
        fixture.detectChanges();
        const formEl = fixture.debugElement.query(By.css('form')).nativeElement;
        dispatchEvent(formEl, 'submit');
        fixture.detectChanges();
        expect(fixture.debugElement.componentInstance.data).toEqual('submitted');
      });
      it('should mark formGroup as submitted on submit event', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        fixture.debugElement.componentInstance.form =
            new FormGroup({'login': new FormControl('loginValue')});
        fixture.detectChanges();
        const formGroupDir = fixture.debugElement.children[0].injector.get(FormGroupDirective);
        expect(formGroupDir.submitted).toBe(false);
        const formEl = fixture.debugElement.query(By.css('form')).nativeElement;
        dispatchEvent(formEl, 'submit');
        fixture.detectChanges();
        expect(formGroupDir.submitted).toEqual(true);
      });
      it('should set value in UI when form resets to that value programmatically', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const login = new FormControl('some value');
        const form = new FormGroup({'login': login});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        const loginEl = fixture.debugElement.query(By.css('input')).nativeElement;
        expect(loginEl.value).toBe('some value');
        form.reset({'login': 'reset value'});
        expect(loginEl.value).toBe('reset value');
      });
      it('should clear value in UI when form resets programmatically', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const login = new FormControl('some value');
        const form = new FormGroup({'login': login});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        const loginEl = fixture.debugElement.query(By.css('input')).nativeElement;
        expect(loginEl.value).toBe('some value');
        form.reset();
        expect(loginEl.value).toBe('');
      });
    });
    describe('value changes and status changes', () => {
      it('should mark controls as dirty before emitting a value change event', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const login = new FormControl('oldValue');
        fixture.debugElement.componentInstance.form = new FormGroup({'login': login});
        fixture.detectChanges();
        login.valueChanges.subscribe(() => { expect(login.dirty).toBe(true); });
        const loginEl = fixture.debugElement.query(By.css('input')).nativeElement;
        loginEl.value = 'newValue';
        dispatchEvent(loginEl, 'input');
      });
      it('should mark control as pristine before emitting a value change event when resetting ',
         () => {
           const fixture = TestBed.createComponent(FormGroupComp);
           const login = new FormControl('oldValue');
           const form = new FormGroup({'login': login});
           fixture.debugElement.componentInstance.form = form;
           fixture.detectChanges();
           const loginEl = fixture.debugElement.query(By.css('input')).nativeElement;
           loginEl.value = 'newValue';
           dispatchEvent(loginEl, 'input');
           expect(login.pristine).toBe(false);
           login.valueChanges.subscribe(() => { expect(login.pristine).toBe(true); });
           form.reset();
         });
    });
    describe('setting status classes', () => {
      it('should work with single fields', () => {
        const fixture = TestBed.createComponent(FormControlComp);
        const control = new FormControl('', Validators.required);
        fixture.debugElement.componentInstance.control = control;
        fixture.detectChanges();
        const input = fixture.debugElement.query(By.css('input')).nativeElement;
        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 work with single fields in parent forms', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const form = new FormGroup({'login': new FormControl('', Validators.required)});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        const input = fixture.debugElement.query(By.css('input')).nativeElement;
        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 work with formGroup', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const form = new FormGroup({'login': new FormControl('', Validators.required)});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        const input = fixture.debugElement.query(By.css('input')).nativeElement;
        const formEl = fixture.debugElement.query(By.css('form')).nativeElement;
        expect(sortedClassList(formEl)).toEqual(['ng-invalid', 'ng-pristine', 'ng-untouched']);
        dispatchEvent(input, 'blur');
        fixture.detectChanges();
        expect(sortedClassList(formEl)).toEqual(['ng-invalid', 'ng-pristine', 'ng-touched']);
        input.value = 'updatedValue';
        dispatchEvent(input, 'input');
        fixture.detectChanges();
        expect(sortedClassList(formEl)).toEqual(['ng-dirty', 'ng-touched', 'ng-valid']);
      });
    });
    describe('value accessors', () => {
      it('should support  without type', () => {
        TestBed.overrideComponent(
            FormControlComp, {set: {template: ``}});
        const fixture = TestBed.createComponent(FormControlComp);
        const control = new FormControl('old');
        fixture.debugElement.componentInstance.control = control;
        fixture.detectChanges();
        // model -> view
        const input = fixture.debugElement.query(By.css('input'));
        expect(input.nativeElement.value).toEqual('old');
        input.nativeElement.value = 'new';
        dispatchEvent(input.nativeElement, 'input');
        // view -> model
        expect(control.value).toEqual('new');
      });
      it('should support ', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const form = new FormGroup({'login': new FormControl('old')});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        // model -> view
        const input = fixture.debugElement.query(By.css('input'));
        expect(input.nativeElement.value).toEqual('old');
        input.nativeElement.value = 'new';
        dispatchEvent(input.nativeElement, 'input');
        // view -> model
        expect(form.value).toEqual({'login': 'new'});
      });
      it('should ignore the change event for ', () => {
        const fixture = TestBed.createComponent(FormGroupComp);
        const form = new FormGroup({'login': new FormControl('oldValue')});
        fixture.debugElement.componentInstance.form = form;
        fixture.detectChanges();
        const input = fixture.debugElement.query(By.css('input'));
        form.valueChanges.subscribe({next: (value) => { throw 'Should not happen'; }});
        input.nativeElement.value = 'updatedValue';
        dispatchEvent(input.nativeElement, 'change');
      });
      it('should support