js實現(xiàn)調用網絡攝像頭及常見錯誤處理
最近由于業(yè)務的原因,需要在Web端頁面接入調試各類的網絡攝像頭,遇到了很多匪夷所思的問題(說的就是讀得出攝像頭的品牌,讀不出攝像頭的分辨率)。于是整理了這篇文章作為備忘錄,也希望能幫到有類似的小伙伴們。
基礎代碼
navigator.mediaDevices.getUserMedia({ audio: false, video: true }).then(async (stream) => { let video = document.getElementById('#video') // 兼容性監(jiān)測 if( 'srcObject' in video ) { video.srcObject = stream } else { // 在支持srcObject的瀏覽器上,不再支持使用這種方式 video.src = URL.createObjectURL(stream) } await video.play() })
兼容性
從caniuse的兼容性來看,整體兼容性一般,IE系列瀏覽器完全不支持,iOS不僅需要iOS 11以上的版本,而且在APP的嵌入式頁面也無法通過api進行調用。
開發(fā)遇到的各種問題
瀏覽器控制臺提示mediaDevices.getUserMedia is not a function
由于受瀏覽器的限制,navigator.mediaDevices.getUserMedia在https協(xié)議下是可以正常使用的,而在http協(xié)議下只允許localhost/127.0.0.1這兩個域名訪問,因此在開發(fā)時應做好容災處理,上線時則需要確認生產環(huán)境是否處于https協(xié)議下。
let mediaDevices = navigator.mediaDevices || null if( mediaDevices === null ) { console.warn(`請確定是否處于https協(xié)議環(huán)境下`) return } mediaDevices.getUserMedia({ audio: false, video: true }).then(async (stream) => {})
獲取攝像頭的硬件參數(shù)
我在項目開發(fā)中需要用到的硬件參數(shù)主要有兩種:品牌,分辨率。獲取攝像頭的品牌名稱相對來說比較簡單,可直接通過mediaDevices.enumerateDevices()獲取電腦上可使用的外設列表,通過kind字段過濾出攝像頭。
if (!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices) { console.log("瀏覽器不支持enumerateDevices屬性") return } navigator.mediaDevices.enumerateDevices().then((devices) => { let devicesList = devices.filter((device) => device.kind === 'videoinput') // devicesList -> [{ kind: 'videoinput', name: 'FaceTime HD Camera (Built-in)', deviceId: xxx }] // 在devicesList獲取到的deviceId可以用于切換攝像頭 // 具體方法:mediaDevices.getUserMedia({ audio: false, video: { deviceId } }) })
分辨率則不能直接通過官方的api獲取到,從MDN上查到的理由是為了保護用戶的個人隱私,而分辨率就在保護的范疇內。(個人非常好奇分辨率為啥是隱私?)
MDN原文(鏈接):
由于隱私保護的原因,無法訪問用戶的攝像頭和麥克風信息
但也并不是完全無法獲取到,由于可以通過video標簽在網頁上播放攝像頭中所錄取到的內容,而video標簽會默認將大小設置為與攝像頭相同的大小,因此通過獲取video的大小來獲取攝像頭的分辨率。
經過測試,獲取到的值不受樣式的影響,所以可以通過樣式控制video的大小,但是不會影響到分辨率。
let mediaDevices = navigator.mediaDevices || null if( mediaDevices === null ) { console.warn(`請確定是否處于https協(xié)議環(huán)境下`) return } mediaDevices.getUserMedia({ audio: false, video: true }).then(async (stream) => { let video = document.getElementById('#video') video.srcObject = stream await video.play() // 1280,720 console.log(video.videoWidth, video.videoHeight) })
無攝像頭/無使用權限等錯誤的處理
getUserMedia本身集成了幾個比較常見的錯誤提示,比如常見的無攝像頭、無使用權限等,通過catch能處理大部分類似的錯誤。
let mediaDevices = navigator.mediaDevices || null if( mediaDevices === null ) { console.warn(`請確定是否處于https協(xié)議環(huán)境下`) return } mediaDevices.getUserMedia({ audio: false, video: true }).then(async (stream) => { let video = document.getElementById('#video') video.srcObject = stream await video.play() }).catch((error) => { let message = error.message || error, response = { 'permission denied': '瀏覽器禁止本頁面使用攝像頭,請開啟相關的權限', 'requested device not found': '未檢測到攝像頭' } alert(response[ message.toLowerCase() ] || '未知錯誤') })
攝像頭拔出檢查
手機端由于攝像頭是手機自帶的,所以一般不需要對攝像頭是否拔出進行檢查。但在PC上有拔出攝像頭數(shù)據(jù)線的情況發(fā)生,這種時候就需要對攝像頭的狀態(tài)進行監(jiān)控。
最開始想到的是,getUserMedia在攝像頭拔出時可能會通過catch報錯。然而經過多次的實驗,getUserMedia在攝像頭拔出時,不會響應找不到攝像頭的錯誤,想通過catch直接監(jiān)控這種方法并不可行。
在幾乎沒有思路的時候,在getUserMedia文檔上看到了這么一句話:
getUserMedia返回一個 Promise , 這個Promise成功后的回調函數(shù)帶一個 MediaStream 對象作為其參數(shù)。
MediaStream是接收多媒體(包括音頻、視頻)內容流的一個對象,在谷歌瀏覽器(其他瀏覽器未測試)的控制臺上打印之后,其屬性值如下:
id是MediaStream對象的唯一標識符,active是當前內容流是否處于活動狀態(tài),下面幾個字段則是谷歌瀏覽器提供的鉤子。
在攝像頭拔出的一瞬間,active會從true變更為false,同時觸發(fā)oninactive鉤子,有了狀態(tài)監(jiān)聽之后事情就簡單了許多。代碼經過測試后發(fā)現(xiàn),對用戶變更攝像頭權限也有效。
// 判斷攝像頭是否在線 let cameraIsOnline = false const loadWebCamera = () => { let mediaDevices = navigator.mediaDevices || null if( mediaDevices === null ) { console.warn(`請確定是否處于https協(xié)議環(huán)境下`) return } mediaDevices.getUserMedia({ audio: false, video: true }).then(async (stream) => { let video = document.getElementById('#video') video.srcObject = stream // 兼容性處理 if( stream.oninactive === null ) { // 監(jiān)聽流中斷,流中斷后將重新進行調用自身進行狀態(tài)監(jiān)測 stream.oninactive = () => loadWebCamera() } await video.play() cameraIsOnline = true }).catch((error) => { let message = error.message || error, response = { 'permission denied': '瀏覽器禁止本頁面使用攝像頭,請開啟相關的權限', 'requested device not found': '未檢測到攝像頭', 'could not start video source': '無法訪問到攝像頭,請重新插拔后重試' } cameraIsOnline = false alert(response[ message.toLowerCase() ] || '未知錯誤') }) }
不過,兼容性也非常地捉急,也有很多字段都是提案階段,開發(fā)階段建議做好兼容性處理,防止生產環(huán)境出現(xiàn)問題。
到此這篇關于js實現(xiàn)調用網絡攝像頭及常見錯誤處理的文章就介紹到這了,更多相關js 調用網絡攝像頭內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
JavaScript知識點總結(六)之JavaScript判斷變量數(shù)據(jù)類型
這篇文章主要介紹了JavaScript知識點總結(六)之JavaScript判斷變量數(shù)據(jù)類型的相關資料,非常不錯具有參考借鑒價值,需要的朋友可以參考下2016-05-05JavaScript 一行代碼,輕松搞定浮動快捷留言-V2升級版
前天熬了大半宿發(fā)了一篇[一行代碼輕松搞定快捷留言功能],同時發(fā)布了V1.0beta版的快捷留言功能和源代碼,之所以是beta版,就是當時感覺雖然基本功能有了,但是還不夠完善,特性也不一定合理2010-04-04微信小程序wxss如何引用外部CSS文件以及iconfont
這篇文章主要給大家介紹了關于微信小程序wxss如何引用外部CSS文件以及iconfont的相關資料,文中通過圖文介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-03-03JS實現(xiàn)簡單的選擇題測評系統(tǒng)代碼思路詳解(demo)
本文給大家分享js實現(xiàn)簡單的選擇題測評系統(tǒng)實例代碼,非常不錯,具有參考借鑒價值,需要的朋友參考下吧2017-09-09Moment.js 不容錯過的超棒Javascript日期處理類庫
moment.js是一個輕量級并且健壯的js日期處理類庫,相信大家在javascript開發(fā)過程中,都自己動手寫過,或者使用google和百度搜索過相關的實現(xiàn)函數(shù)2012-04-04用Golang運行JavaScript的實現(xiàn)示例
這篇文章主要介紹了用Golang運行JavaScript的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-11-11