質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.92%

WebRTCにおけるカメラの切り替え

解決済

回答 1

投稿

  • 評価
  • クリップ 2
  • VIEW 10K+

yokatone

score 43

こんにちは。
現在ToughPad(Windows10)を用いたWebRTCによるカメラの撮影機能を作成しております。

 目的

この機器には前面と背面両方にカメラが備わっており、
それを切り替えて撮影し、サーバーにアップロードできるようなプログラムを作成しようとしています。

 経過

ブラウザでWebカメラの映像を画像化【javasript】を参考に
カメラの表示、画像のキャプチャは作成できましたので、

HTML5で背面カメラを取得する
Android Chrome でテストすると【今は】カメラの front と back を切り替えできます
javascriptで内カメラと外カメラの切り替え
を参考に、入力ソース(前面カメラ/背面カメラ)を切替える機能を作成しようとしましたが、
これがうまく動作しません。

 詳細

まず、 MediaStreamTrack.getSources がis not a functionと表示されたので、 mediaDevices.enumerateDevices で置き換えました。

これで背面/前面のsourceIDを取得することができましたので、
取得したIDをselectboxにappendし、onイベントで変更時にアクションを取る、という風に作りましたが
これでセレクトボックスを変更してもvideoinputが切り替わりません。
どうやらChromeで許可したカメラデバイスを基準としているようで、

navigator.mediaDevices.getUserMedia({
    video:{
        optional: [{sourceId: source}]
    }
})

のsourceIdを背面カメラ、Chromeの許可カメラを前面カメラに設定し、ページにアクセスしてみましたが
表示されるのは前面カメラのみでした。

videoinputのソースに設定する方法にnavigator.getUserMediaを使用していますが、
他に適切な方法があるのでしょうか?
以下にソース全文を記載しておきます。
(※logical errorさんのソースをlist()以外ほぼそのまま使用しています)
なにとぞお力添えをよろしくお願いいたします。

<html>
    <head>
        <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.2.1/jquery.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/toastr.min.js"></script>
<link href="//cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/toastr.min.css" rel="stylesheet" type="text/css">
    </head>
    <body>


<video id="camera" autoplay></video>
<br>
<select name="dlist">
</select>

<script>
jQuery.isMobile = (/android|webos|iphone|ipad|ipod|blackberry|iemobile|opera mini/i.test(navigator.userAgent.toLowerCase()));
toastr.options.positionClass = "toast-top-center";

if ( $.isMobile ) {
    $("#camera").css("width","100%");
}

// カメラ用 video 要素(DOM オブジェクト)
var camera;
// スマホ用コンボボックス(jQuery オブジェクト)
var select;
// カメラの一覧
var camera_data;
// 現在のカメラの ID
var camera_id = "default";

check();

// *************************************
// navigator.getUserMedia チェック
// *************************************
function check() {

    var api = [
        "webkitGetUserMedia", "mozGetUserMedia","msGetUserMedia"
    ]

    $.each(api,function(idx){
        if (navigator.getUserMedia = navigator.getUserMedia || navigator[api[idx]]) {
            return false;
        }
    });

    if ( !navigator.getUserMedia ) {
        error("WebRTC を使用できません");
        return;
    }

    // WEBカメラの初期化
    init();
}

// *************************************
// カメラストリーム
// *************************************
function init() {

    camera = $("#camera").get(0);

    if ( location.hash != "" ) {
        camera_id = location.hash;
        camera_id = camera_id.replace("#","");
        console.log(camera_id);
        $("<div>"+camera_id+" を選択しました</div>").appendTo("body");
    }

    navigator.getUserMedia(
        {video: {
            optional: [{sourceId: camera_id }]
        }}, 
        function(stream) {
            camera.src = window.URL.createObjectURL(stream);
        },
        function(err){
            error(err.name);
        }
    );

    list();

}

// *************************************
// MediaStreamTrack が使える場合
// カメラ一覧をコンボボックス化
// *************************************
function list() {

    navigator.mediaDevices.enumerateDevices()
    .then(function(devices) {
        devices.forEach(function(device) {
            if(device.kind == "videoinput"){
                $('[name="dlist"]').append($('<option>').html(device.label).val(device.deviceId));
            }
        });
    })
    .catch(function(err) {
        options.error(err.name + ": " + err.message);
    });

    $('[name="dlist"]').on("change",function(){
        var val = $('[name="dlist"]').val();
        console.log(val);

        var txt = $('[name="dlist"] option:selected').text();
        console.log(txt);

        location.hash=val;
        location.reload(true);
    });
}

// *************************************
// エラー処理
// ※ 動画で代替
// *************************************
function error( message ) {
        toastr.error( message + "<br>代替として動画を表示します" );
        $("#camera")
        .prop({ 
            "loop" : true, "muted" : true, "controls" : true,
            "src" : "https://lightbox.sakura.ne.jp/demo/mp4/waterfall-free-video9.mp4"
        })
        .css("border", "solid 1px #000");
}
</script>
    </body>
</html>
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

check解決した方法

+2

 解決

Streamを切っていないのが問題でした。
WebRTCのGitHub
のサンプルデモだと動作したので参考にしたところ、

function setCameraObject(){

        //videoElement -> videoタグ, videoSelect -> ソースが書かれたセレクトボックス
        var videoElement = document.querySelector('video');
        var videoSelect = document.querySelector('select#videoSource');

        //streamの停止
        if (window.stream) {
            window.stream.getTracks().forEach(function(track) {
                track.stop();
            });
        }

        var videoSource = videoSelect.value;
        var constraints = {
            video: {deviceId: videoSource ? {exact: videoSource} : undefined}
        };

        //getUserMediaで割り当てたvideoSourceを用いてstreamを開始
        navigator.mediaDevices.getUserMedia(constraints).then(function(stream){
            window.stream = stream;
            videoElement.srcObject = stream;
        }).catch(function(err){
            console.log(err);
        });
    }

としたところ動作しました。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.92%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る