プログラミング

【PR】を含みます。

【JavaScript】Mutation ObserverでDOMの変化を監視する方法

JavaScript DOMの変化を監視する方法

JavaScriptのMutation ObserverでDOMの変化を監視する方法を紹介します。

Ajax通信で追加された要素に対して処理を実行、特定のclassが付与されたら処理を実行するなど、実務で時々使用するためコピペで使用できるようにまとめました。

Mutation Observerのオプション一覧

オプション
説明
childList
対象ノードの子ノードの追加と削除を監視します。
値:trueまたはfalse
attributes
対象ノードの属性の変更を監視します。
値:trueまたはfalse
characterData
対象ノードのテキストの変更を監視します。
値:trueまたはfalse
subtree
対象ノードおよびその子孫ノードの変化を監視します。
値:trueまたはfalse
attributeFilter
特定の属性の変更のみを監視します。
値:監視する属性名の配列
attributeOldValue
属性の変更前の値を記録します。
attributestrueの場合にのみ使用されます。
値:trueまたはfalse
characterDataOldValue
テキストの変更前の値を記録します。
characterDatatrueの場合にのみ使用されます。
値:trueまたはfalse

必須オプションについて

MutationObserverを正しく動作させるためには、以下のいずれか1つのオプションを指定する必要があります。

  • childList

  • attributes

  • characterData

【実装・テンプレートコード】Mutation ObserverでDOMの変化を監視

以下のコードはテンプレートのため複数のオプションを設定していますが、実務で使用する場合は必要なオプションのみを設定して使用することが多いです。

JavaSctipt
Copy
  1. // 監視対象の要素を取得
  2. const targetNode = document.querySelector('.watchElement');
  3. // オプション設定
  4. const config = {
  5.   childList: true,
  6.   attributes: true,
  7.   characterData: true,
  8.   subtree: true,
  9.   attributeFilter: ['class', 'style', 'data-demo'],
  10.   attributeOldValue: true,
  11.   characterDataOldValue: true,
  12. };
  13. // 変更が発生したときに呼び出されるコールバック関数
  14. const callback = function(mutationsList, observer) {
  15.   for (let mutation of mutationsList) {
  16.     // 変更があった箇所(要素)を取得
  17.     let targetElement = mutation.target;
  18.     if (mutation.type === 'childList') {
  19.       // 子ノードに追加または削除があった場合の処理
  20.       console.log(`子ノードが追加または削除されました。変更箇所:`, targetElement);
  21.     } else if (mutation.type === 'attributes') {
  22.       // 属性に変更があった場合の処理
  23.       console.log(`${mutation.attributeName} 属性が変更されました。変更箇所:`, targetElement, `古い値: ${mutation.oldValue}`);
  24.     } else if (mutation.type === 'characterData') {
  25.       // テキストが変更された場合の処理
  26.       console.log(`テキストが変更されました。変更箇所:`, targetElement, `古い値: ${mutation.oldValue}`);
  27.     }
  28.   }
  29. };
  30. // MutationObserverのインスタンスを作成
  31. const observer = new MutationObserver(callback);
  32. // 監視対象の要素にMutationObserverを設定
  33. observer.observe(targetNode, config);

【サンプル】Mutation ObserverでDOMの変化を監視

「クリック」ボタンを押下すると、上記実装コードで実装しているconsole.log()の内容をディベロッパツールのコンソールで確認できます。

監視対象要素

サンプルで使用しているコード

JavaSctipt
Copy
  1. <div class="watchElement">監視対象要素</div>
  2. <button id="btn">クリック</button>
JavaSctipt
Copy
  1. // 監視対象の要素を取得
  2. const targetNode = document.querySelector('.watchElement');
  3. // オプション設定
  4. const config = {
  5.   childList: true,
  6.   attributes: true,
  7.   characterData: true,
  8.   subtree: true,
  9.   attributeFilter: ['class', 'style', 'data-demo'],
  10.   attributeOldValue: true,
  11.   characterDataOldValue: true,
  12. };
  13. // 変更が発生したときに呼び出されるコールバック関数
  14. const callback = function(mutationsList, observer) {
  15.   for (let mutation of mutationsList) {
  16.     // 変更があった箇所(要素)を取得
  17.     let targetElement = mutation.target;
  18.     if (mutation.type === 'childList') {
  19.       // 子ノードに追加または削除があった場合の処理
  20.       console.log(`子ノードが追加または削除されました。変更箇所:`, targetElement);
  21.     } else if (mutation.type === 'attributes') {
  22.       // 属性に変更があった場合の処理
  23.       console.log(`${mutation.attributeName} 属性が変更されました。変更箇所:`, targetElement, `古い値: ${mutation.oldValue}`);
  24.     } else if (mutation.type === 'characterData') {
  25.       // テキストが変更された場合の処理
  26.       console.log(`テキストが変更されました。変更箇所:`, targetElement, `古い値: ${mutation.oldValue}`);
  27.     }
  28.   }
  29. };
  30. // MutationObserverのインスタンスを作成
  31. const observer = new MutationObserver(callback);
  32. // 監視対象の要素にMutationObserverを設定
  33. observer.observe(targetNode, config);
  34. const btn = document.getElementById('btn');
  35. btn.addEventListener('click', function () {
  36.   // 要素の追加
  37.   let createElement = document.createElement('div');
  38.   createElement.textContent = '要素を追加';
  39.   targetNode.appendChild(createElement);
  40.   // 属性の変更
  41.   targetNode.classList.add('demo');
  42.   targetNode.setAttribute('data-demo', 'demo');
  43.   // テキストの変更
  44.   targetNode.firstChild.data = '新しいテキスト';
  45. }, {once: true});

【実装コード】子ノードの追加と削除を監視

JavaSctipt
Copy
  1. // 監視対象の要素を取得
  2. const targetNode = document.querySelector('.watchElement');
  3. // オプション設定
  4. const config = {
  5.   childList: true,
  6. };
  7. // 変更が発生したときに呼び出されるコールバック関数
  8. const callback = function(mutationsList, observer) {
  9.   for (let mutation of mutationsList) {
  10.     if (mutation.type === 'childList') {
  11.       // 子ノードに追加または削除があった場合の処理
  12.       console.log('子ノードが追加または削除されました。');
  13.     }
  14.   }
  15. };
  16. // MutationObserverのインスタンスを作成
  17. const observer = new MutationObserver(callback);
  18. // 監視対象の要素にMutationObserverを設定
  19. observer.observe(targetNode, config);

【実装コード】属性の変更を監視

class等の属性の変更を監視することができます。

JavaSctipt
Copy
  1. // 監視対象の要素を取得
  2. const targetNode = document.querySelector('.watchElement');
  3. // オプション設定
  4. const config = {
  5.   attributes: true,
  6. };
  7. // 変更が発生したときに呼び出されるコールバック関数
  8. const callback = function(mutationsList, observer) {
  9.   for (let mutation of mutationsList) {
  10.     if (mutation.type === 'attributes') {
  11.       // 属性に変更があった場合の処理
  12.       console.log('属性が変更されました。');
  13.     }
  14.   }
  15. };
  16. // MutationObserverのインスタンスを作成
  17. const observer = new MutationObserver(callback);
  18. // 監視対象の要素にMutationObserverを設定
  19. observer.observe(targetNode, config);

【実装コード】テキストの変更を監視

JavaSctipt
Copy
  1. // 監視対象の要素を取得
  2. const targetNode = document.querySelector('.watchElement');
  3. // オプション設定
  4. const config = {
  5.   characterData: true,
  6.   subtree: true,
  7. };
  8. // 変更が発生したときに呼び出されるコールバック関数
  9. const callback = function(mutationsList, observer) {
  10.   for (let mutation of mutationsList) {
  11.     if (mutation.type === 'characterData') {
  12.       // テキストが変更された場合の処理
  13.       console.log('テキストが変更されました。');
  14.     }
  15.   }
  16. };
  17. // MutationObserverのインスタンスを作成
  18. const observer = new MutationObserver(callback);
  19. // 監視対象の要素にMutationObserverを設定
  20. observer.observe(targetNode, config);

テキストの変更を監視する場合の注意点

対象ノードのテキストの変更をcharacterDataとして検知したい場合、オプションにsubtree: trueを指定します。

テキストの変更は、dataプロパティを使用してテキストノードを直接変更する必要があります。

例:targetNode.firstChild.data = '新しいテキスト';

textContent等でテキスト変更が行われている場合、characterDataで検知することはできません。

textContentでテキストが変更されている場合、childListを使用して監視する必要があります。

【実装コード】Mutation Observerの監視を停止する方法

disconnect()メソッドを呼び出すことで、Mutation Observerの監視を停止できます。

以下のコードは、子ノードに追加または削除があった場合に監視を停止します。

JavaSctipt
Copy
  1. // 監視対象の要素を取得
  2. const targetNode = document.querySelector('.watchElement');
  3. // オプション設定
  4. const config = {
  5.   childList: true,
  6. };
  7. // 変更が発生したときに呼び出されるコールバック関数
  8. const callback = function(mutationsList, observer) {
  9.   for (let mutation of mutationsList) {
  10.     if (mutation.type === 'childList') {
  11.       // 子ノードに追加または削除があった場合の処理
  12.       console.log('子ノードが追加または削除されました。');
  13.       observer.disconnect(); // 監視を停止
  14.       break; // ループを抜ける
  15.     }
  16.   }
  17. };
  18. // MutationObserverのインスタンスを作成
  19. const observer = new MutationObserver(callback);
  20. // 監視対象の要素にMutationObserverを設定
  21. observer.observe(targetNode, config);

HTMLとCSSの知識がありJavaScriptを学びたいという方におすすめの1冊

アイコン画像

もみじ

HTMLとCSSの知識がありJavaScriptを学びたいという方に入門書としておすすめの書籍です。

実務で役立つサンプルを手を動かしながら学ぶことができ、実践的なスキルを身に付けることができます。

JavaScriptのフロントエンドエンジニアを目指している方におすすめの1冊

アイコン画像

もみじ

この本はJavaScript初心者からフロントエンドエンジニアを目指す方にぴったりの1冊です。

非同期処理やAJAXなど、現代の開発で必要なスキルがわかりやすく解説されており、実務で役立つスキルを学ぶことができます。

-プログラミング
-