Event.eventPhase

eventPhase`Event インターフェイスの読み取り専用プロパティは、イベントフローのどの段階が現在評価されているかを示します。

整数値で、イベントフローの現在の評価段階を指定する値を返します。使用可能な値は以下のとおりです。

Event.NONE (0)

現在、このイベントは処理されていません。

Event.CAPTURING_PHASE (1)

イベントはターゲットの祖先オブジェクトを通じて伝播されます。 この処理は Window から始まり、 DocumentHTMLHtmlElement と、ターゲットの親に到達するまで要素を介して行われます。 キャプチャモード向けに EventTarget.addEventListener() が呼ばれた時に登録されたイベントリスナーは、この段階で起動されます。

Event.AT_TARGET (2)

このイベントがイベントのターゲットに到達しています。 この段階向けに登録されたイベントリスナーは、この時点で呼び出されます。もし Event.bubblesfalse ならば、この段階の終了後にイベントの処理を終了します。

Event.BUBBLING_PHASE (3)

イベントはターゲットの祖先を逆順に伝播していきます。親から始まり、最終的に Window を含む先祖に到達します。 これはバブリングと呼ばれ、 Event.bubblestrue のときのみ発生します。この段階向けに登録されたイベントリスナーは、この処理中に起動されます。

HTML

<h4>Event Propagation Chain</h4>
<ul>
  <li>Click 'd1'</li>
  <li>Analyze event propagation chain</li>
  <li>Click next div and repeat the experience</li>
  <li>Change Capturing mode</li>
  <li>Repeat the experience</li>
</ul>
<input type="checkbox" id="chCapture" />
<label for="chCapture">Use Capturing</label>
<div id="d1">d1
  <div id="d2">d2
    <div id="d3">d3
      <div id="d4">d4</div>
    </div>
  </div>
</div>
<div id="divInfo"></div>

CSS

div {
  margin: 20px;
  padding: 4px;
  border: thin black solid;
}

#divInfo {
  margin: 18px;
  padding: 8px;
  background-color: white;
  font-size: 80%;
}

JavaScript

let clear = false,
    divInfo = null,
    divs = null,
    chCapture = null;

window.onload = function () {
  divInfo = document.getElementById('divInfo');
  divs = document.getElementsByTagName('div');
  chCapture = document.getElementById('chCapture');
  chCapture.onclick = function () {
    removeListeners();
    addListeners();
    clearDivs();
  };
  clearDivs();
  addListeners();
}

function removeListeners() {
  for (const div of divs) {
    if (div.id != 'divInfo') {
      div.removeEventListener('click', onDivClick, true);
      div.removeEventListener('click', onDivClick, false);
    }
  }
}

function addListeners() {
  for (const div of divs) {
    if (div.id != 'divInfo') {
        if (chCapture.checked) {
            div.addEventListener('click', onDivClick, true);
        }
        else {
            div.addEventListener('click', onDivClick, false);
            div.onmousemove = function () { clear = true };
        }
    }
  }
}

function onDivClick(e) {
  if (clear) {
    clearDivs();
    clear = false;
  }
  if (e.eventPhase == 2) {
    e.currentTarget.style.backgroundColor = 'red';
  }
  const level =
      e.eventPhase == 0 ? 'none' :
      e.eventPhase == 1 ? 'capturing' :
      e.eventPhase == 2 ? 'target' :
      e.eventPhase == 3 ? 'bubbling' : 'error';
  const para = document.createElement('p');
  para.textContent = `${e.currentTarget.id}; eventPhase: ${level}`;
  divInfo.appendChild(para);
}

function clearDivs() {
  for (let i = 0; i < divs.length; i++) {
    if (divs[i].id != 'divInfo') {
      divs[i].style.backgroundColor = (i & 1) ? '#f6eedb' : '#cceeff';
    }
  }
  divInfo.textContent = '';
}

結果

仕様書

Specification
DOM Standard
# ref-for-dom-event-eventphase③

ブラウザーの互換性

BCD tables only load in the browser