要素はイベントを使用して状態の変化をDOMツリーを通じて親要素に伝達します。Polymer要素は、標準のDOM APIを使ってイベントの生成、ディスパッチ、監視(listen)を行うことができます。
Polymerにはアノテーション付イベントリスナーも用意されています。このリスナーを使用すると、要素のDOMテンプレートの一部としてイベントリスナーを宣言的に記述できます。
アノテーション付イベントリスナーを追加
ローカルDOMの子にイベントリスナーを追加するには、テンプレート内でon-event
アノテーションを使用します。これにより、イベントリスナーをバインドするためだけに要素にid
を付与するといった処理が不要になります。
例:
<dom-module id="x-custom">
<template>
<button on-click="handleClick">Kick Me</button>
</template>
<script>
class XCustom extends Polymer.Element {
static get is() {return 'x-custom'}
handleClick() {
console.log('Ow!');
}
}
customElements.define(XCustom.is, XCustom);
</script>
</dom-module>
ヒント:タッチ(モバイル)デバイスとクリック(デスクトップ)デバイスの両方で一貫したイベントを発生させるため、
on-click
代わりにon-tap
を使用してください。tap
イベントは、ジェスチャーイベントとして提供されるオプションのひとつです。要素でジェスチャーイベントが有効にサポートされるようにする方法は、[ジェスチャーイベント](https://github.com/jtakiguchi/polymer-docs-japanese-translation
/blob/master/docs/polymer2/events/gesture-events.md)を参照してください。
イベント名はHTML属性を利用して指定されるので、**常に小文字に変換されます。**これは、HTML属性名が大文字と小文字を区別しないためです。したがって、on-myEvent
を指定した場合、イベントmyEvent
に対してリスナーが追加されます。イベントハンドラの名前(例えば、handleClick
)は大文字と小文字を区別します。混乱を避けるため、常に小文字のイベント名を使用するようにして下さい。
リスナーを命令的に追加/削除
標準のaddEventListener
及びremoveEventListener
メソッドを使用して、イベントリスナーを命令的に追加または削除することができます。
ready() {
super.ready();
this.addEventListener('click', e => this._myClickListener(e));
}
上の例では、(ES6から導入された)アロー関数を使用して、カスタム要素をthis
の値としてリスナーが呼び出されるようにしています。またbind
を使って、リスナー関数がバインドされたインスタンスを作成することもできます。この方法は、リスナーを削除する必要がある場合に役立ちます。
constructor() {
super();
this._boundListener = this._myLocationListener.bind(this);
}
connectedCallback() {
super.connectedCallback();
window.addEventListener('hashchange', this._boundListener);
}
disconnectedCallback() {
super.disconnectedCallback();
window.removeEventListener('hashchange', this._boundListener);
}
自身またはShadow DOMの子のどこかにイベントリスナーを追加した要素は、その要素がガベージコレクトされるのを禁止すべきではありません。しかしながら、ウィンドウまたはドキュメントレベルのような外部の要素に追加されたイベントリスナーによって、要素がガベージコレクションされないことがあります。メモリリークを防止するためdisconnectedCallback
コールバック内でイベントリスナーを削除するようにしてください。
カスタムイベント
ホスト要素からカスタムイベント(独自に作成したイベント)を発火するには、標準のCustomEvent
コンストラクタとdispatchEvent
メソッドを使用します。
例:
<dom-module id="x-custom">
<template>
<button on-click="handleClick">Kick Me</button>
</template>
<script>
class XCustom extends Polymer.Element {
static get is() {return 'x-custom'}
handleClick(e) {
this.dispatchEvent(new CustomEvent('kick', {detail: {kicked: true}}));
}
}
customElements.define(XCustom.is, XCustom);
</script>
</dom-module>
<x-custom></x-custom>
<script>
document.querySelector('x-custom').addEventListener('kick', function (e) {
console.log(e.detail.kicked); // true
})
</script>
デフォルトでは、カスタムイベントはShadow DOMの境界で停止します。カスタムイベントがShadow DOMの境界を越えて伝播するようにするには、イベントを作成する際にcomposed
フラグをtrueに設定します。:
var event = new CustomEvent('my-event', {bubbles: true, composed: true});
リターゲティングされたイベントの処理
Shadow DOMには、イベントがバブルアップする際に、ターゲットを変更する「イベントリターゲッティング(event retargetting)」という機能があり、そのターゲットは常にイベントを受け取る要素と同じスコープになります。(例えば、メインドキュメントのリスナーの場合、ターゲットはShadow Tree内ではなくメインドキュメント内の要素になります。)
イベントのcomposedPath()
メソッドは、イベントが通過するノードを配列で返します。そのためevent.composedPath()[0]
は、イベントの原初のターゲットを表します(ただし、ターゲットが閉じられたShadow Root内に隠れていない場合に限ります)。
例:
<!-- event-retargeting.html -->
...
<dom-module id="event-retargeting">
<template>
<button id="myButton">Click Me</button>
</template>
<script>
class EventRetargeting extends Polymer.Element {
static get is() {return 'event-retargeting'}
ready() {
super.ready();
this.$.myButton.addEventListener('click', e => {this._handleClick(e)});
}
_handleClick(e) {
console.info(e.target.id + ' was clicked.');
}
}
customElements.define(EventRetargeting.is, EventRetargeting);
</script>
</dom-module>
<!-- index.html -->
...
<event-retargeting></event-retargeting>
<script>
var el = document.querySelector('event-retargeting');
el.addEventListener('click', function(e){
// logs the instance of event-targeting that hosts #myButton
console.info('target is:', e.target);
// logs [#myButton, ShadowRoot, event-retargeting,
// body, html, document, Window]
console.info('composedPath is:', e.composedPath());
});
</script>
この例では、原初のイベントが<event-retargeting>
要素のローカルDOMツリー内の<button>
でトリガーされています。リスナーは、メインドキュメント上の<event-retargeting>
要素に対して設定されています。イベントはリターゲティングされるので、要素の実装を隠してしまえばクリックイベントは<button>
要素からというよりむしろ<event-retargeting>
要素から生じているようにみえます。
Shadow Rootはdocument-fragment
としてコンソールに表示されるかもしれません。Shady DOMでは、DocumentFragment
のインスタンスになるためです。ネイティブのShadow DOMでは、ShadowRoot
(DocumentFragment
を拡張するDOMインターフェース)のインスタンスとして表示されます。
プロパティ変更イベント
特定のプロパティ値が変更された際に、ノンバブリングなDOMイベントを発生する要素を構築することもできます。詳細については、[変更通知イベント](https://github.com/jtakiguchi/polymer-docs-japanese-translation
/blob/master/docs/polymer2/data-system/data-system-concepts.md#change-notification-events)を参照してください。