WebRTC设备管理 五、第一节 WebRTC获取音视频设备

    技术2022-07-10  354

    今天我们开始学习WebRTC内容了,首先我们看一下WebRTC如何进行设备管理。

    在 WebRTC 的规范中,给我们提供一个重要的API,叫

    enumerateDevices

    通过这个API 我们就可以获取到电脑中的音频和视频设备,首先我们来看一下它的基本格式

    /** * 基本格式 通过navigator.mediaDevices下的enumerateDevices方法获取所有的音频和视频设备 * 最后它返回的值是一个Promise,这是JavaScript中特有的一个对象 */ var ePromise = navigator.mediaDevices.enumerateDevices();

    在Promise里面有个重要的结构体

    MediaDevicesInfo

    在这个结构体里面存放几个非常重要的信息

    deviceID就是这个设备的唯一标识符

    label就是设备的名字,就是我们人可读的,比如说内置音频设备,内置音频输入设备、内置音频输出设备或者是耳机等等一些人 可读的设备名字。

    kind表示设备的种类,比如音频输入设备、视频输出设备、还有视频输入设备

    groupID表示组ID,就是两个设备GroupID相同的组ID,说明是同一个物理设备,以音频 为例,同一个设备里面包含两种类型,一种是输入一种是输出

    以上就是enumerateDevices的基本信息,通过调用这个API就可以拿到音频和视频设备了,就非常简单。

    ——————————————————————————————————————————————————————

    下面我们简单介绍一些JavaScript中的Promise

    上图很好的解释了JavaScript中的Promise是什么东西,我们都知道JavaScript是使用单线程去处理整个逻辑的,所以为了防止它被阻塞,大量使用了这个异步调用。这个Promise就是异步调用其中的一种方式。也是现在比较流行,也是大家比较认可的一种方式,下面介绍一下它的基本思想。

    首先你在创建Promise的时候要传给它一个handle的处理函数,这个handle处理函数是用来处理你的主要逻辑,在它处理之后,如果成功了会 调用 这个处理函数中的resolve函数,如果失败了它就会调用reject这个函数,这样就创建好了一个Promise.

    Promise可以通过两个方法,一个是then一个是catch, then就是当我整个逻辑处理成功之后会收到这个on_resolve事件,收到这个事件之后就处理一些逻辑,这就是它这个then。catch就是失败了,当失败的时候处理一些失败的逻辑,then这个 如果 成功之后,它是一个链式的,可以继续写一个then。

    它有几个状态,第一个是未执行的一个状态,第二个当你执行handle的逻辑的时候赋予这个运行状态,当这个处理成功之后进入这个resolve改为成功状态,如果失败了进入 reject里面设置为出错状态。当你注册这个函数之后你 就可以根据Promise里面这个状态机如果成功了就调用这个then方法 ,如果失败了就 调用这个catch方法 ,这就是它的基本逻辑 。

    ——————————————————————————————————————————————————————

    我们回到上面enumerateDevices的函数,实际上在这个 函数里面它就new 了一个Promise,并且给它注册了一个handle,所以当它这个 函数执行的时候它 就返回一个Promise,在我们用的时候拿到这个Promise之后,我们就给他注册两个函数,一个是then的方法一个是catch的方法,如果成功了他就会调用then我们成功的一些逻辑,如果失败了就会调用失败那些处理逻辑,这就是javaScript中的Promise的基本思想。

    那么接下来我们就通过一个实际的例子,如何通过WebRTC的API来获取到我们的音视频设备,我们先新建一个文件夹devide,在文件夹里面建一个index.html

    <html> <head> <title> WebRTC get audio and video devices</title> </head> <body> <div> <label>audio input device:</label> <select id="audioSource"></select> </div> <div> <label>audio output device:</label> <select id="audioOutput"></select> </div> <div> <label>video input device:</label> <select id="videoSource"></select> </div> <script src="./js/client.js"></script> </body> </html>

    我在里面引入我们编写的脚本client.js,这样当我们打开页面的时候,这个JavaScript代码就会执行 ,就是chrome浏览器会给他交到底层的V8引擎去解析然后去渲染,这样我们的Html就算写完了。

    下面我们来实现client.js

    // 1、使用严格语法 'use strict' var audioSource = document.querySelector("select#audioSource"); var audioOutput = document.querySelector("select#audioOutput"); var videoSource = document.querySelector("select#videoSource"); // 2、首先判断浏览器是否支持此方法,在浏览器支持的情况下才调用 if(!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices){ console.log('enumerateDevices is not supported!'); }else { navigator.mediaDevices.enumerateDevices() .then(gotDevices) .catch(handleError); } // 3、实现gotDevices,在gotDevices就可以得到我们刚说的 deviceInfos 了 function gotDevices(deviceInfos){ // 5、当我们拿到deviceInfos这个数组之后我们就开始遍历这个数组 // 在这个每一项里面实际上我们可以注册一个匿名函数去处理每一项的内容,它的参数就是每一项的值 deviceInfos.forEach( function(deviceInfo){ console.log(deviceInfo.kind + ": label = " + deviceInfo.label + ": id = " + deviceInfo.deviceId + ": groupId = " + deviceInfo.groupId); var option = document.createElement('option'); option.text = deviceInfo.label; option.value = deviceInfo.deviceId; if(deviceInfo.kind === 'audioinput'){ audioSource.appendChild(option); }else if(deviceInfo.kind === 'audiooutput'){ audioOutput.appendChild(option); }else if(deviceInfo.kind === 'videoinput'){ videoSource.appendChild(option); } }); } // 4、在handleError里面会返回一个错误 function handleError(err){ console.log(err.name + " : " + err.message); }

    保存之后,我们运行node server.js

    我们可以看到有输入设备,label没有显示,id是default,我们看到前两个groupId是一样的,第三个是视频的输入设备,proupId不一样了,第四个 是音频的输入设备,这个输入和输出设备物理设备是一样的,所以他们有同一个groupId;

    大家在url前面输入https,这样设备名称 就显示出来了

    这样我们就将这个例子演示完了,通过这个例子大家可以很清楚的知道去获取一个音视频设备是多么方便。

    Processed: 0.008, SQL: 9