web端 本地調(diào)用攝像頭述暂、前端攝像頭

最近公司項目涉及到一個需求娩鹉,調(diào)用電腦攝像頭功能,經(jīng)過在網(wǎng)上查找資料有以下三種方案實現(xiàn)元媚。

  1. 通過瀏覽器API來getUserMedia來實現(xiàn)調(diào)用用戶的攝像頭轧叽,但有兩點限制:部署到生產(chǎn)服務(wù)的時候要使用htts協(xié)議訪問該項目,第二點限制就是第一次獲取攝像頭時要點擊允許彈窗刊棕。
  2. 通過web端發(fā)送websocket請求給客戶端炭晒,然后客戶端調(diào)用攝像頭應(yīng)用程序,但每次初始打開攝像頭應(yīng)用時都有個初始化2~3秒左右甥角,用戶體驗不免友好网严。
  3. 通過客戶端獲取攝像頭的每一幀數(shù)據(jù),并通過websocket返回給web端嗤无,具體步驟下面有代碼講解震束。
getUserMedia方式實現(xiàn)代碼
<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <title>調(diào)用攝像頭錄像</title>
  </head>
  <body>
    <video id="video" width="640" height="480" autoplay></video>
    <button id="startRecord">開始錄制</button>
    <button id="stopRecord">停止錄制</button>
    <script>
      var video = document.querySelector('#video');
      var startRecord = document.querySelector('#startRecord');
      var stopRecord = document.querySelector('#stopRecord');
      var mediaRecorder;
      var chunks = [];

      navigator.mediaDevices.getUserMedia({ audio: false, video: true }).then(function (stream) {
        video.srcObject = stream;
        mediaRecorder = new MediaRecorder(stream);
        mediaRecorder.ondataavailable = function (e) {
          chunks.push(e.data);
        };
        mediaRecorder.onstop = function (e) {
          var blob = new Blob(chunks, { type: 'video/mp4' });
          chunks = [];
          var videoURL = window.URL.createObjectURL(blob);
          video.src = videoURL;
        };
      });
      startRecord.onclick = function () {
        mediaRecorder.start();
      };
      stopRecord.onclick = function () {
        mediaRecorder.stop();
      };
    </script>
  </body>
</html>

客戶端返回視頻流方式實現(xiàn)代碼

vue前端代碼
前端接收到H264視頻流,需要用JMuxer來轉(zhuǎn)碼才能在video標(biāo)簽里播放当犯,需要執(zhí)行npm install JMuxer來安裝這個庫垢村。詳細(xì)代碼如下:

<template>
  <video id="player" :controls="false" :muted="false" :autoplay="true"></video>
</template>
<script setup>
  import JMuxer from 'jmuxer';
  import { onMounted, onBeforeUnmount} from 'vue';
  var ws = null;
  var jmuxer = null;
  const list = reactive([]);
  const isDisable = ref(false);

  onMounted(() => {
    jmuxer = new JMuxer({
      node: 'player',
      // 可用值是:video、audio
      mode: 'video',
      // 最大延遲時間(毫秒), 默認(rèn)為值是500毫秒
      maxDelay: 100,
      // 緩沖區(qū)刷新時間,默認(rèn)為值是500毫秒
      flushingTime: 0,
      // 是否會自動清除播放的媒體緩沖區(qū)嚎卫。默認(rèn)為true
      clearBuffer: true,
      // 可選值嘉栓。視頻的幀率,如果它是已知的或固定值拓诸。如果所提供的媒體數(shù)據(jù)中沒有塊持續(xù)時間侵佃,它將用于查找?guī)掷m(xù)時間。
      fps: 30,
      // 將從MP4軌道數(shù)據(jù)讀取FPS奠支,而不是使用(以上)FPS值馋辈。默認(rèn)為false。
      readFpsFromTrack: false,
      // 將在瀏覽器控制臺打印調(diào)試日志胚宦。默認(rèn)為false
      debug: false,
      // 遇到任何丟失的視頻幀將會被觸發(fā)
      onMissingVideoFrames: function (data) {
        console.log('丟失的視頻幀');
      },
      onError: function (data) {
        if (/Safari/.test(navigator.userAgent) && /Apple Computer/.test(navigator.vendor)) {
          jmuxer.reset();
        }
      }
    });
    ws = new WebSocket('ws://127.0.0.1:12797');
    ws.binaryType = 'arraybuffer';
    ws.onmessage = function (event) {
      jmuxer.feed({ video: new Uint8Array(event.data) });
    };
    ws.onopen = function () {
      console.log('已連接');
      ws.send('{"interfaceId":"100002","command":"1"}');
    };
    ws.onerror = function (err) {
      console.log('出錯--Socket Error', err);
    };
    ws.onclose = function () {
      console.log('斷開');
    };
  });
// 實現(xiàn)拍照功能
  const toScan = () => {
      const videoEl = document.getElementById('player');
      const canvas = document.createElement('canvas');
      canvas.width = videoEl.videoWidth;
      canvas.height = videoEl.videoHeight;
      const ctx = canvas.getContext('2d');
      ctx.drawImage(videoEl, 0, 0, videoEl.videoWidth, videoEl.videoHeight);
      console.log(canvas.toDataURL('image/png'));
  };

  onBeforeUnmount(() => {
    console.info('離開頁面,關(guān)閉高拍儀');
    ws.close();
    jmuxer.destroy();
  });
</script>
C++客戶端

https://github.com/qiyingcong/vue-c-websocket點擊下載客戶端包
雙擊打開這個應(yīng)用程序就可以訪問本地的websocket的視頻流了

image.png

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末首有,一起剝皮案震驚了整個濱河市燕垃,隨后出現(xiàn)的幾起案子枢劝,更是在濱河造成了極大的恐慌井联,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,561評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件您旁,死亡現(xiàn)場離奇詭異烙常,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)鹤盒,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,218評論 3 385
  • 文/潘曉璐 我一進(jìn)店門蚕脏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人侦锯,你說我怎么就攤上這事驼鞭。” “怎么了尺碰?”我有些...
    開封第一講書人閱讀 157,162評論 0 348
  • 文/不壞的土叔 我叫張陵挣棕,是天一觀的道長。 經(jīng)常有香客問我亲桥,道長洛心,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,470評論 1 283
  • 正文 為了忘掉前任题篷,我火速辦了婚禮词身,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘番枚。我一直安慰自己法严,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,550評論 6 385
  • 文/花漫 我一把揭開白布葫笼。 她就那樣靜靜地躺著深啤,像睡著了一般。 火紅的嫁衣襯著肌膚如雪渔欢。 梳的紋絲不亂的頭發(fā)上墓塌,一...
    開封第一講書人閱讀 49,806評論 1 290
  • 那天,我揣著相機(jī)與錄音奥额,去河邊找鬼苫幢。 笑死,一個胖子當(dāng)著我的面吹牛垫挨,可吹牛的內(nèi)容都是我干的韩肝。 我是一名探鬼主播,決...
    沈念sama閱讀 38,951評論 3 407
  • 文/蒼蘭香墨 我猛地睜開眼九榔,長吁一口氣:“原來是場噩夢啊……” “哼哀峻!你這毒婦竟也來了涡相?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,712評論 0 266
  • 序言:老撾萬榮一對情侶失蹤剩蟀,失蹤者是張志新(化名)和其女友劉穎催蝗,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體育特,經(jīng)...
    沈念sama閱讀 44,166評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡丙号,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,510評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了缰冤。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片犬缨。...
    茶點故事閱讀 38,643評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖棉浸,靈堂內(nèi)的尸體忽然破棺而出怀薛,到底是詐尸還是另有隱情,我是刑警寧澤迷郑,帶...
    沈念sama閱讀 34,306評論 4 330
  • 正文 年R本政府宣布枝恋,位于F島的核電站,受9級特大地震影響三热,放射性物質(zhì)發(fā)生泄漏鼓择。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,930評論 3 313
  • 文/蒙蒙 一就漾、第九天 我趴在偏房一處隱蔽的房頂上張望呐能。 院中可真熱鬧,春花似錦抑堡、人聲如沸摆出。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,745評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽偎漫。三九已至,卻和暖如春有缆,著一層夾襖步出監(jiān)牢的瞬間象踊,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,983評論 1 266
  • 我被黑心中介騙來泰國打工棚壁, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留杯矩,地道東北人。 一個月前我還...
    沈念sama閱讀 46,351評論 2 360
  • 正文 我出身青樓袖外,卻偏偏與公主長得像史隆,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子曼验,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,509評論 2 348

推薦閱讀更多精彩內(nèi)容