برخلاف AngularJS، در برنامههای Angular امکانات two way data binding به صورت پیشفرض ارائه نمیشوند تا از تمام مشکلات آن مانند digest cycle ،watchers و غیره خبری نباشد. اما گاهی از اوقات نیاز است انقیاد دو طرفهی سفارشی را بین دو کامپوننت ایجاد کنیم. در این مطلب روش ایجاد یک چنین انقیادهایی را بررسی خواهیم کرد و در اینجا در ابتدا نیاز است دو پیشنیاز Property Binding و Event Binding را بررسی کنیم که از جمع آنها two way data binding حاصل میشود:
البته Angular به همراه دایرکتیو ویژهای به نام ngModel است که two-way data binding را با import ماژول ویژهی فرمها میسر میکند:
که آن نیز در اصل از جمع Property Binding و Event Binding تشکیل شدهاست:
<input [ngModel]="username" (ngModelChange)="username = $event">
و یا به صورت خلاصه:
<input [(ngModel)]='username' />
در اینجا میخواهیم یک چنین امکانی را بدون استفاده از ngModel و ماژول فرمها پیاده سازی کنیم.
انقیاد به خواص یا Property binding
فرض کنید دو کامپوننت والد و فرزند را ایجاد کردهایم:
در کامپوننت والد، مقداری را توسط متد deposit هربار 100 آیتم افزایش میدهیم:
import { Component, OnInit } from "@angular/core";
@Component({
selector: "app-parent",
templateUrl: "./parent.component.html",
styleUrls: ["./parent.component.css"]
})
export class ParentComponent implements OnInit {
amount = 500;
constructor() { }
ngOnInit() {
}
deposit() {
this.amount += 100;
}
}
با این قالب:
<h2>Custom two way data binding</h2>
<div class="panel panel-primary">
<div class="panel-heading">
<h2 class="panel-title">Parnet Component</h2>
</div>
<div class="panel-body">
<label>Available amount:</label> {{amount}}
<button (click)="deposit()" class="btn btn-success">Deposit 100</button>
<div>
<app-child [amount]="amount"> </app-child>
</div>
</div>
</div>
که در آن مقدار amount کامپوننت والد نمایش داده شدهاست و همچنین این مقدار به خاصیت ورودی کامپوننتی به نام app-child نیز نسبت داده شدهاست.
کامپوننت فرزند به صورت ذیل تعریف میشود:
import { Component, OnInit, Input } from "@angular/core";
@Component({
selector: "app-child",
templateUrl: "./child.component.html",
styleUrls: ["./child.component.css"]
})
export class ChildComponent implements OnInit {
@Input() amount: number;
constructor() { }
ngOnInit() {
}
withdraw() {
this.amount -= 100;
}
}
که در آن خاصیت amount، از والد آن، توسط ویژگی Input دریافت میشود. سپس در متد withdraw هربار میتوان 100 آیتم را از آن کسر کرد.
با این قالب:
<div class="panel panel-default">
<div class="panel-heading">
<h2 class="panel-title">Child Component</h2>
</div>
<div class="panel-body">
<label>Amount available: </label> {{amount}}
<button (click)="withdraw()" class="btn btn-danger">Withdraw 100</button>
</div>
</div>
که در آن مقدار amount فرزند نمایش داده شدهاست و همچنین امکان فراخوانی متد withdraw وجود دارد.
در اینجا زمانیکه data binding را به صورت ذیل تعریف میکنیم:
<app-child [amount]="amount"> </app-child>
روش مقدار دهی خاصیت amount داخل [] ، انقیاد به خواص نامیده میشود و سمت راست آن نیز یک خاصیت درنظر گرفته میشود. یعنی مقدار خاصیت amount والد (درون "") به مقدار خاصیت amount فرزند (درون []) نسبت داده خواهد شد.
این ارتباط نیز یک طرفهاست. برای مثال اگر بر روی دکمهی Deposit والد کلیک کنیم:
مقدار افزایش یافتهی در والد، به فرزند نیز منتقل میشود و نمایش داده خواهد شد. اما اگر بر روی دکمهی withdraw فرزند کلیک کنیم:
تغییر صورت گرفته، به والد انعکاس پیدا نمیکند. برای اطلاع رسانی به والد، به انقیاد به رخدادها نیاز داریم.
انقیاد به رخدادها یا Event binding
یک کامپوننت میتواند به رخدادهای صادر شدهی توسط کامپوننتی دیگر گوش فرا دهد:
import { Component, OnInit, Input, Output, EventEmitter } from "@angular/core";
@Component({
selector: "app-child",
templateUrl: "./child.component.html",
styleUrls: ["./child.component.css"]
})
export class ChildComponent implements OnInit {
@Input() amount: number;
@Output() amountChange = new EventEmitter();
constructor() { }
ngOnInit() {
}
withdraw() {
this.amount -= 100;
this.amountChange.emit(this.amount);
}
}
برای این منظور در کامپوننت فرزند، یک خاصیت Output را به نام amountChange از نوع EventEmitter تعریف میکنیم. سپس جایی که قرار است کار کاهش amount صورت گیرد، با صدور رخدادی (this.amountChange.emit)، این مقدار را به والد اعلام میکنیم.
اکنون در قالب کامپوننت والد، این رخداد را درون یک () معرفی خواهیم کرد:
<app-child [amount]="amount" (amountChange)="this.amount= $event"> </app-child>
به این ترتیب زمانیکه کامپوننت فرزند، مقدار amount را تغییر میدهد، این مقدار توسط this.amountChange.emit به والد منتشر خواهد شد و میتوان در سمت والد توسط event$ به آن دسترسی یافته و آنرا به خاصیت this.amount کامپوننت والد نسبت دهیم.
اکنون اگر برنامه را آزمایش کنیم، با کلیک بر روی دکمهی withdraw فرزند، مقدار کاهش یافته به والد نیز منعکس میشود:
پیاده سازی syntax ویژهی Banana in a box
تا اینجا پیاده سازی two way data-binding سفارشی به پایان میرسد. اما تعریف طولانی:
<app-child [amount]="amount" (amountChange)="this.amount= $event"> </app-child>
به صورت ذیل هم قابل نوشتن و ساده سازی است:
<app-child [(amount)]="amount"> </app-child>
که به آن syntax ویژه Banana in a box نیز گفته میشود.
نکتهی ویژهی آن، وجود پسوند Change در نام رخداد تعریف شدهاست:
@Input() amount: number;
@Output() amountChange = new EventEmitter();
اگر نام خاصیت Input مساوی x باشد، باید جهت فعالسازی syntax ویژه Banana in a box، نام رخداد متناظر با آن دقیقا مساوی x
Change انتخاب شود. مانند amount ورودی در اینجا و amount
Change خروجی تعریف شده.
بنابراین به صورت خلاصه جهت تعریف یک انقیاد دو طرفه سفارشی:
- ابتدا باید انقیاد به یک خاصیت ورودی x را تعریف کرد.
- سپس نیاز است انقیاد به یک رخداد خروجی همنام، که نام آن، پسوند Change را اضافهتر دارد، یعنی xChange را تعریف کرد.
- اکنون میتوان two-way data binding syntax ویژهای را به نام banana in a box بر روی ایندو تعریف کرد[(x)].
کدهای کامل این مطلب را از اینجا میتوانید دریافت کنید.