MediaDevices: devicechange イベント

devicechange はイベントで、カメラ、マイク、スピーカーなどのメディア機器がシステムに接続・切断されるたびに MediaDevices へ送られます。

このイベントはキャンセル不可で、バブリングしません。

構文

このイベント名を addEventListener() のようなメソッドで使用するか、イベントハンドラープロパティを設定するかしてください。

js
addEventListener("devicechange", (event) => {});

ondevicechange = (event) => {};

イベント型

一般的な Event です。

この例では、 updateDeviceList() という関数を作成します。この関数は MediaDevices.getUserMedia() がストリームの取得に成功したときに一度呼び出され、その後で機器一覧が変更されるたびに呼び出されます。ブラウザーのウィンドウに、オーディオ機器と映像機器の 2 つの一覧が表示され、機器のラベル(名前)と入力機器か出力機器かが表示されます。この例では devicechange イベントに対するハンドラーを提供しているので、メディア機器がサンプルを実行している機器に接続されたり取り外されたりするたびに、一覧が更新されます。

オーディオ機器と映像機器の一覧に使用される <ul> 要素への参照を保持するためのグローバル変数を用意します。

js
const audioList = document.getElementById("audioList");
const videoList = document.getElementById("videoList");

機器一覧の取得と描画

updateDeviceList() メソッドでは、現在のメディア機器の一覧を取得した後、先ほど用意したグローバル変数を用いて、表示されているオーディオ機器と映像機器の一覧を更新します。

js
function updateDeviceList() {
  navigator.mediaDevices.enumerateDevices().then((devices) => {
    audioList.innerHTML = "";
    videoList.innerHTML = "";

    devices.forEach((device) => {
      const elem = document.createElement("li");
      const [kind, type, direction] = device.kind.match(/(\w+)(input|output)/i);

      elem.innerHTML = `<strong>${device.label}</strong> (${direction})`;
      if (type === "audio") {
        audioList.appendChild(elem);
      } else if (type === "video") {
        videoList.appendChild(elem);
      }
    });
  });
}

updateDeviceList()navigator.mediaDevices プロパティで参照している MediaDevices オブジェクトの enumerateDevices() の関数呼び出しが全体を占め、 enumerateDevices() から返されるプロミス (promise) が履行されたときに実行されるコードもあります。 履行ハンドラーは、機器一覧の準備ができたときに呼び出されます。 この一覧は履行ハンドラーへ MediaDeviceInfo オブジェクトの配列として渡され、それぞれがメディアの入力機器や出力機器を表しています。

すべての機器の情報を出力するために、 forEach() ループを使用しています。それぞれの機器で、ユーザーへこの機器情報を見せるために新しい <li> オブジェクトを作成します。

let [kind, type, direction] = device.kind.match(/(\w+)(input|output)/i); の行について詳しく説明します。ここでは分割代入を使用しており、 String.match() によって返された配列の値を kindtypedirection の変数へ代入しています。なぜこのようなことをするのかというと、 MediaDeviceInfo.kind の文字列は、 "audioinput" や "videooutput" のように、メディア種別とメディアフローの向きの 2 つの情報を含んでいるためです。この行で、種別("audio" または "video")と方向("input" と "output")を取り出すことで、一覧に表示する文字列を作成することができます。

太字のデバイス名と括弧で囲まれた方向を含む文字列が作成されると、デバイス種別に基づいてaudioListまたはvideoList対応する一覧へappendChild()の呼び出しによって追加されます。

機器一覧の変更の扱い

updateDeviceList() は 2 か所呼び出されます。 1 か所目は getUserMedia() のプロミスの履行ハンドラーであり、ストリームが開かれた時に最初に一覧を埋めます。 2 か所目は devicechange のイベントハンドラーからです。

js
navigator.mediaDevices.ondevicechange = (event) => {
  updateDeviceList();
};

このコードでは、ユーザがカメラ、マイク、他のメディア機器を接続したり、電源を入れたり、電源を切ったりする度に、 updateDeviceList() を呼び出し接続された機器一覧を再描画します。

結果

仕様書

Specification
Media Capture and Streams
# event-mediadevices-devicechange
Media Capture and Streams
# dom-mediadevices-ondevicechange

ブラウザーの互換性

BCD tables only load in the browser