将ngModel中的对象传递给自定义控件Angular2

时间:2017-03-07 13:45:41

标签: angular angular2-forms angular2-ngmodel

我想创建包含其他自定义控件的自定义控件,并使用连接所有自定义控件的ngModel。 喜欢:

PersonSearchComponent:

  • DeveloperSearchControl
    • 名称 - 一些字符串输入
    • 姓氏
    • ProffesionSelect - 期望ProffesionModel的自定义控件
  • BuildingSearchControl
    • 此处有一些自定义控件
  • CountryCustomControl - 期望CountryModel的自定义控件
  • PersonListComponent: - 由某些服务提供有关PersonSearchComponent中项目的数据
  • SomeOtherSearchComponent
    • DeveloperSearchControl - 可重复使用

所以现在我有工作版本,但我觉得我做错了(也许我应该使用FormBuilder):

PersonSearch Template:

    <div>
            <developer-search-control [(ngModel)]="searchModel.developerSearchModel"></developer-search-control>
            <building-search-control [(ngModel)]="searchModel.buildingSearchModel"></building-search-control>
            <country-select [(ngModel)]="searchModel.countryModel"><country-select>
    </div>

ParentSearch组件

...
export class PersonSearch {
  @Output() searchDataEmitter= new EventEmitter();//Exports data to above component which contains this, and listComponent
searchModel : PersonSearchModel : new PersonSearchModel();

    performSearch()
    {
        //gets data from service with searchModel and searchDataEmitter transpors it to above component
    }
}

型号: PersonSearchModel:

developerSearchModel : DeveloperSearchModel = new DeveloperSearchModel();
buildingSearchModel: BuildingSearchModel = new BuildingSearchModel();
countryModel : CountryModel;

DeveloperSearchModel:

name : string
surname : string
proffesion : ProfessionModel

模板 developerSearchControl.component.html:

<div *ngIf="value">//This is the problem
     <input [(ngModel)]="value.name"></input>
     <input [(ngModel)]="value.surname"></input>
     <profesion-select [(ngModel)]="value.ProffesionSelect">
</div>

developerSearchControl.component:

...
@Component({
  selector: 'developer-search-control',
  templateUrl: './developerSearchControl.component.html',
  providers: [{
    provide: NG_VALUE_ACCESSOR,
    useExisting: forwardRef(() => DeveloperSearchControlComponent),
    multi: true,
  }],
})

export class DeveloperSearchControlComponent extends ElementBase<DeveloperSearchModel > {
  protected model: NgModel;

  constructor(
    @Optional() @Inject(NG_VALIDATORS) validators: Array<any>,
    @Optional() @Inject(NG_ASYNC_VALIDATORS) asyncValidators: Array<any>,
  ) {
    super(validators, asyncValidators);
  }
}

profesionSelect.component

    ...
@Component({
    selector: 'profesion-select',
    template: `
     <div>
        <label *ngIf="label">{{label}}</label>
        <dx-select-box 
            [dataSource]="data" 
            [(ngModel)]="value" 
            displayExpr="proffesionName" 
            [placeholder]="placeholder"
            [searchEnabled]="true"
            >
        </dx-select-box>
    </div>`,
        providers: [{
        provide: NG_VALUE_ACCESSOR,
        useExisting: ProfessionComponent,
        multi: true,
    }],
})

export class ProfessionComponent extends ElementBase<string> implements OnInit {
    private data: ProfessionModel[];
    private label: string = 'proffesion :';
    private placeholder: string = 'Select profession';
    @ViewChild(NgModel) model: NgModel;

    constructor(private proffesionService: ProfessionDataService,
        @Optional() @Inject(NG_VALIDATORS) validators: Array<any>,
        @Optional() @Inject(NG_ASYNC_VALIDATORS) asyncValidators: Array<any>,
    ) {
        super(validators, asyncValidators);
    }

    ngOnInit() {
        this.professionService.getDevelopersProfessions().subscribe(
            data => {
                this.data = data;
            }
        );
    }
}

ElementBase是一个通用的ControlValueAccessor,其验证来自:http://blog.rangle.io/angular-2-ngmodel-and-custom-form-components/

所以我的问题是,当我为子项创建模板时(developerSearch,buildingSearch),使用ngModel传递的值没有为它们初始化,我得到:

EXCEPTION: Uncaught (in promise): Error: Error in ./DeveloperSearchControlComponent class DeveloperSearchControlComponent - inline template:2:33 caused by: Cannot read property 'name' of null
Error: Error in ./DeveloperSearchControlComponent class DeveloperSearchControlComponent - inline template:2:33 caused by: Cannot read property 'name' of null

因为ngModel的值在开始时为null。所以我必须在看起来很糟糕的子组件模板中使用* ngFor =“value”。 是否有任何解决方案在模板验证之前初始化对象?或者我这样做是非常错误的?

1 个答案:

答案 0 :(得分:5)

有一种方法可以使用双向绑定和安全导航操作符:

<input [ngModel]="value?.name" (ngModelChange)="value?.name ? value.name = $event : null"> 

道具:https://stackoverflow.com/a/36016472/5706293