侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

使用Html5多媒体实现微信语音功能

微信(小程序)  /  管理员 发布于 7年前   184

随着微信等社交App的兴起,语音聊天成为很多App必备功能,大到将语音聊天作为主要功能的社交App,小到电商App的语音客服、店小二功能,语音聊天成为了必不可少的方式。

但是很多人感觉网页端语音离我们很遥远,这些更多是本地应用的工作,其实不然,随着Html5的发展,语音功能也渐渐成为前端必会的功能之一。

为什么要学会HTML5 的语音呢?

1.Html5 规范推进,手机的更新加速了操作系统更新,语音功能将会变成前端主要的工作之一,就像现在的canvas一样。前端实现语音功能开发速度更快,更节省人力(这意味着给老板省钱,给老板省钱就是在给自己涨工资)

2.即使是现在本地应用做语音功能,熟悉前端语音交互的各种坑能够让你们的同事关系更和谐,协作更顺畅,而不是互相掐架。

3.了解新的技术可以预防面试,二来可以预判技术潮流,不至于学了一堆屠龙之技或者墨守成规,更有利于让自己的知识和职业核心竞争力一直处在食物链的顶端。

4.前端大部分人对语音功能有误解,以为语音功能就是HTML5 audio标签而已,事实上真的不是那么简单的"而已"

不墨迹那么多,咱们直接开发一个小项目啥都明明白儿白儿了,先看效果图

clipboard.png

业务逻辑非常简单,

跟我们微信用法一模一样,手按下去字变成松开结束,同时说话被录下来,松手的时候,变成按下结束,同时发送语音给对方

我们一步一步一步来,首先我们先整一个html页面

<!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8">    <meta name="viewport" content="width=device-width, initial-scale=1.0">    <meta http-equiv="X-UA-Compatible" content="ie=edge">    <title>微信语音</title>    <link rel="stylesheet" href="css/record.css"></head><body>    <div id="wrap">        <header id="header"><div id="left">    <i class="material-icons">        chevron_left    </i>    微信(184)</div><div id="mid">艾达·王</div><div id="right">    <i class="material-icons">        more_horiz    </i></div>        </header>        <div id="contentWrap"><ul id="chatList">    <li class="item_me">        <div class="chatContent">我是不是你最疼爱的人?<span class="bot"></span><span class="top"></span>        </div>        <div class="avatar"><img src="images/ava1.png" alt="">        </div>    </li>    <li class="item_you">        <div class="avatar"><img src="images/ava2.jpg" alt="">        </div>        <div class="chatContent">奔跑吧,兄弟!(滚犊子)<span class="bot"></span><span class="top"></span>        </div>    </li>    <li class="item_me">        <div class="chatContent">这里我就不多说了,上来就是一梭子代码……<span class="bot"></span><span class="top"></span>        </div>        <div class="avatar"><img src="images/ava1.png" alt="">        </div>    </li>    <li class="item_you">        <div class="avatar"><img src="images/ava2.jpg" alt="">        </div>        <div class="chatContent">大彬哥,你说你咋这么优秀呢?看见你我有一种大海的感觉<span class="bot"></span><span class="top"></span>        </div>    </li>    <li class="item_me">        <div class="chatContent">老妹儿,你是不是喜欢上我了呢……<span class="bot"></span><span class="top"></span>        </div>        <div class="avatar"><img src="images/ava1.png" alt="">        </div>    </li>    <li class="item_you">        <div class="avatar"><img src="images/ava2.jpg" alt="">        </div>        <div class="chatContent">不是,我晕船,看见你想吐……<span class="bot"></span><span class="top"></span>        </div>    </li></ul>        </div>        <footer id="footer"><div id="keyboard">    <i class="material-icons">        keyboard    </i></div><div id="sayBtn">    <span id="sendBtn" class="sendBtn">按下 说话</span></div><div id="icon"><i class="material-icons">        sentiment_satisfied    </i></div><div id="add"><i class="material-icons">        add_circle_outline    </i></div>        </footer>    </div></body></html>

css部分,

*{    margin: 0;    padding: 0;}ul li{ list-style: none;}html,body{    height: 100%;    width: 100%;    overflow: hidden;}body{    background: #ebebeb;}@font-face {    font-family: 'Material Icons';    font-style: normal;    font-weight: 400;    src: url(../css/iconfont/MaterialIcons-Regular.eot); /* For IE6-8 */    src: local('Material Icons'),      local('MaterialIcons-Regular'),      url(../css/iconfont/MaterialIcons-Regular.woff) format('woff2'),      url(../css/iconfont/MaterialIcons-Regular.woff2) format('woff'),      url(../css/iconfont/MaterialIcons-Regular.ttf) format('truetype');  }  .material-icons {    font-family: 'Material Icons';    font-weight: normal;    font-style: normal;    font-size: 32px;  /* Preferred icon size */    display: inline-block;    /* line-height: 0.01rem; */    text-transform: none;    letter-spacing: normal;    word-wrap: normal;    white-space: nowrap;    direction: ltr;      /* Support for all WebKit browsers. */    -webkit-font-smoothing: antialiased;    /* Support for Safari and Chrome. */    text-rendering: optimizeLegibility;      /* Support for Firefox. */    -moz-osx-font-smoothing: grayscale;      /* Support for IE. */    font-feature-settings: 'liga';  }#wrap{    display: flex;    flex-direction: column;    justify-content: space-between;    height: 100%;}#header{    height: 46px;    line-height: 46px;    background: #363539;    display: flex;    align-items: center;    color: #fff;    justify-content: space-between;}#header #left{    display: flex;    align-items: center;    font-size: 14px;    width: 100px;}#header #right{    display: flex;    align-items: center;    width: 100px;    justify-content: flex-end;}#header #right i{    padding-right: 6px;}#header #mid{    text-align: center;    flex: 1;}#contentWrap{    flex: 1;    overflow-y:auto;}.item_me,.item_audio{    display: flex;    align-items: flex-start;    justify-content:flex-end;    padding: 8px;}.item_you{    display: flex;    align-items: flex-start;    justify-content:flex-start;    padding: 8px;}.avatar{    width: 40px;    height: 40px;}.avatar img{width: 100%;}.item_me .chatContent{    padding: 10px;    background: #a0e75a;    border: 1px solid #6fb44d;    margin-right: 15px;    border-radius: 5px;    position: relative;}.chatContent span{width:0; height:0; font-size:0; overflow:hidden; position:absolute;}.item_me .chatContent span.bot{    border-width:8px;     border-style:solid dashed dashed;     border-color: transparent transparent transparent #6fb44d;     right:-17px;     top:10px;}.item_me .chatContent span.top{    border-width:8px;     border-style:solid dashed dashed;     border-color:transparent transparent transparent #a0e75a ;      right:-15px;     top:10px;} .item_you .chatContent{    padding: 10px;    background: #a0e75a;    border: 1px solid #6fb44d;    margin-left: 15px;    border-radius: 5px;    position: relative;} .item_you .chatContent span.bot{    border-width:8px;     border-style:solid dashed dashed;     border-color: transparent #6fb44d transparent transparent ;     left:-17px;     top:10px;}.item_you .chatContent span.top{    border-width:8px;     border-style:solid dashed dashed;     border-color:transparent #a0e75a transparent transparent  ;      left:-15px;     top:10px;}        #footer{    height: 46px;    padding: 0 4px;    background: #f4f5f6;    border-top: 1px solid #d7d7d8;    display: flex;    align-items: center;    color: #7f8389;    justify-content: space-around;}#sayBtn{    flex: 1;    display: flex;    margin: 0 5px;    color:#565656;    font-weight: bold;}.sendBtn{    display: block;    flex: 1;    padding: 8px;    background: #f4f5f6;    border:1px solid #bec2c1;    border-radius: 5px;    text-align: center;}.activeBtn{    display: block;    flex: 1;    padding: 8px;    background: #c6c7ca;    border:1px solid #bec2c1;    border-radius: 5px;    text-align: center;}.item_audio .chatContent{    padding: 6px;    background: #fff;    border: 1px solid #999;    border-radius: 5px;    margin-right: 15px;    position: relative;    width:120px;    min-height: 20px;}.item_audio .chatContent span.bot{    border-width:8px;     border-style:solid dashed dashed;     border-color: transparent transparent transparent #999;     right:-17px;     top:10px;}.item_audio .chatContent span.top{    border-width:8px;     border-style:solid dashed dashed;     border-color:transparent transparent transparent #fff ;      right:-15px;     top:10px;} .material-icons_wifi{    transform: rotate(90deg);    color: #a5a5a5;    font-size: 22px;}.redDot{    background: #f45454;    border-radius: 50%;    width: 8px;    height: 8px;    margin-right: 10px;}

这里我说两个注意点,

1.html部分:

图省事我并没有像素级切图,图省事我也直接用了svg图标,具体库我使用的是

https://material.io/tools/icons/?style=outline

2.css部分:使用flex布局。我只是为了讲解Html5功能,所以flex并没有写兼容性写法,另外App头部部分写法大家注意一下,那里是非常常用的。

下面说重点js部分。

 

<!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8">    <meta name="viewport" content="width=device-width, initial-scale=1.0">    <meta http-equiv="X-UA-Compatible" content="ie=edge">    <title>微信语音</title>    <link rel="stylesheet" href="css/record.css">    <script>        document.addEventListener('DOMContentLoaded', function () {var oSendBtn = document.getElementById('sendBtn');var soundClips = document.querySelector('.sound-clips');var mediaRecorder;var oChatList = document.getElementById('chatList');navigator.getUserMedia = (navigator.getUserMedia ||    navigator.webkitGetUserMedia ||    navigator.mozGetUserMedia ||    navigator.msGetUserMedia);if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {    navigator.mediaDevices.getUserMedia(        // constraints - only audio needed for this app        {audio: true        })        // Success callback        .then(function (stream) {rec(stream);        })        // Error callback        .catch(function (err) {        }        );} else {} function rec(stream) {    mediaRecorder = new MediaRecorder(stream);    oSendBtn.addEventListener('touchstart', function (ev) {        ev.preventDefault();        this.innerHTML = '松开 结束';        this.classList.add('activeBtn');        mediaRecorder.start();    }, false);    oSendBtn.addEventListener('touchend', function (ev) {        ev.preventDefault();        this.innerHTML = '按下 说话';        this.classList.remove('activeBtn');        mediaRecorder.stop();    }, false);    mediaRecorder.ondataavailable = function (e) {        var clipContainer = document.createElement('li');        var audio = document.createElement('audio');        clipContainer.classList.add('item_audio');        clipContainer.innerHTML = `        <div class = "redDot"></div>        <div class="chatContent"><i class="material-icons material-icons_wifi">wifi</i><span class="bot"></span><span class="top"></span>        </div>        <div class="avatar"><img src="images/ava1.png" alt="">        </div>`;        audio.setAttribute('controls', '');        oChatList.appendChild(clipContainer);        var audioURL = window.URL.createObjectURL(e.data);        audio.src = audioURL;        oChatList.addEventListener('touchstart', function (ev) {if (ev.srcElement.parentNode.className!== 'item_audio') return;audio.play();ev.srcElement.parentNode.removeChild(ev.srcElement.parentNode.children[0])        }, false);    };}        }, false);    </script></head><body>    <div id="wrap">        <header id="header"><div id="left">    <i class="material-icons">        chevron_left    </i>    微信(184)</div><div id="mid">艾达·王</div><div id="right">    <i class="material-icons">        more_horiz    </i></div>        </header>        <div id="contentWrap"><ul id="chatList">    <li class="item_me">        <div class="chatContent">我是不是你最疼爱的人?<span class="bot"></span><span class="top"></span>        </div>        <div class="avatar"><img src="images/ava1.png" alt="">        </div>    </li>    <li class="item_you">        <div class="avatar"><img src="images/ava2.jpg" alt="">        </div>        <div class="chatContent">奔跑吧,兄弟!(滚犊子)<span class="bot"></span><span class="top"></span>        </div>    </li>    <li class="item_me">        <div class="chatContent">这里我就不多说了,上来就是一梭子代码……<span class="bot"></span><span class="top"></span>        </div>        <div class="avatar"><img src="images/ava1.png" alt="">        </div>    </li>    <li class="item_you">        <div class="avatar"><img src="images/ava2.jpg" alt="">        </div>        <div class="chatContent">大彬哥,你说你咋这么优秀呢?看见你我有一种大海的感觉<span class="bot"></span><span class="top"></span>        </div>    </li>    <li class="item_me">        <div class="chatContent">老妹儿,你是不是喜欢上我了呢……<span class="bot"></span><span class="top"></span>        </div>        <div class="avatar"><img src="images/ava1.png" alt="">        </div>    </li>    <li class="item_you">        <div class="avatar"><img src="images/ava2.jpg" alt="">        </div>        <div class="chatContent">不是,我晕船,看见你想吐……<span class="bot"></span><span class="top"></span>        </div>    </li></ul>        </div>        <footer id="footer"><div id="keyboard">    <i class="material-icons">        keyboard    </i></div><div id="sayBtn">    <span id="sendBtn" class="sendBtn">按下 说话</span></div><div id="icon"><i class="material-icons">        sentiment_satisfied    </i></div><div id="add"><i class="material-icons">        add_circle_outline    </i></div>        </footer>    </div></body></html>

这里实现的录影功能要注意的点很多,我们一个个说,

第一个东西,

 

navigator.getUserMedia = (navigator.getUserMedia ||    navigator.webkitGetUserMedia ||    navigator.mozGetUserMedia ||    navigator.msGetUserMedia);if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {    navigator.mediaDevices.getUserMedia(        {audio: true        })        // Success callback        .then(function (stream) {rec(stream);        })        // Error callback        .catch(function (err) {        }        );} else {}

 

当大家看一些html5关于录音的接口的时候,你看到这个

Navigator.getUserMedia()

就要小心了,这个是老规范的东西了,被废了,新的是

navigator.mediaDevices.getUserMedia

html5 多媒体里面的语音这块换了好几茬规范,很乱,有些标签甚至一个浏览器都没实现过,未曾绽放就枯萎了,你也不用关心也没必要浪费那个时间知道,你只要知道我说这些就够了,因为你知道那些被废掉的过往没啥用,有那个时间还不如来一局LOL或者王者荣耀(虽然我并不懂二者的区别,不过这两个游戏应该都挺好玩吧,没玩过不懂)。

里面的东西大家也不需要看懂,什么promise了,什么媒体流了,你不用知道,你就知道这样一件事就行了,

上面的代码就相当于打开了水龙头(或者说按下的录音机的录音键),那么我们得有东西接着水啊,我们可以用电饭锅(录音机的话就是磁带)放水龙头下面看着它往里面ci(我们老家话,射的意思),如下代码

mediaRecorder = new MediaRecorder(stream);

接下来就是,一按按钮就生米煮成熟饭了,对应录音机就是录完了按按钮就播放了,但是在我们程序里面要想播放你不仅要有磁带,还得有录音机,录音机就是audio标签,没有好办,我们new一个。这个世界上没有什么对象是程序员不敢new的,new一个不行,就new两个。剩下的代码除了吓人之外,没啥缺点,简单的令人发指。

mediaRecorder.ondataavailable = function (e) {        var clipContainer = document.createElement('li');        var audio = document.createElement('audio');        clipContainer.classList.add('item_audio');        clipContainer.innerHTML = `        <div class = "redDot"></div>        <div class="chatContent"><i class="material-icons material-icons_wifi">wifi</i><span class="bot"></span><span class="top"></span>        </div>        <div class="avatar"><img src="images/ava1.png" alt="">        </div>`;        audio.setAttribute('controls', '');        oChatList.appendChild(clipContainer);        var audioURL = window.URL.createObjectURL(e.data);        audio.src = audioURL;        oChatList.addEventListener('touchstart', function (ev) {if (ev.srcElement.parentNode.className!== 'item_audio') return;audio.play();ev.srcElement.parentNode.removeChild(ev.srcElement.parentNode.children[0])        }, false);    };

 

其实就是录好了就播。

OK,是不是很简单 ,整个项目我说几个点吧:

1.切图结构合理是你后面做功能的前提,结构做的好,后面就省事,想想诸葛不亮吧,未出茅庐人家就把html5结构搭好了,有三个section.

2.原生js和ES6的基础打牢可以为你提供不同的思路,比如我这里就使用了事件委托,还有ES6模板引擎。尤其是事件委托,不用的话查找节点很麻烦,另外代码套来套去也容易乱。

3.新的 知识和技术其实并不复杂,其实很简单,你想如果新技术不是为了让功能更好实现,更能解决我们的问题,那开发新技术干嘛?因为那帮大胡子的大牛们没事干怕被领导说工作量不饱和?技术是为了解决问题和让我们生活更美好服务的。

4.这个项目IOS 11以下跑不通,因为IOS 11.2之前不支持这个方法,需要IOS本地应用开发人员给你提供支援,但是在android下面是很OK的。而且可以预见,再过几年IOS 原生也不用给你支援都支持了,那你开发效率得多高。不要以为这些技术很遥远,html5真正商用也不过15年左右(vue 、react、angular大规模使用才几年?),机会留给有准备的人。

整个项目细节和要注意的点还是很多的,希望大家真正自己敲一遍,因为你看懂了我的文章跟你会用这个技术两码事,祝大家在前端的路上越走越远(记得常回来看看^_^)。

总结

以上所述是小编给大家介绍的使用Html5多媒体实现微信语音功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!


  • 上一条:
    MAC系统中怎么查找微信小视频和图片保存的文件夹地址?
    下一条:
    微信小程序之html5 canvas绘图并保存到系统相册
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 微信模板消息改版后发送规则记录(微信订阅消息参数值内容限制说明)(1个评论)
    • 微信支付v3对接所需工具及命令(0个评论)
    • 2023年9月1日起:微信小程序必须备案才能上线运营(0个评论)
    • 腾讯官方客服回应了:微信好友上限约10000个!(1个评论)
    • 2023年做微信小程序的老铁注意:新增收费项、微信小程序获取手机号也收费了(2个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2017-10
    • 2018-01
    • 2020-03
    • 2021-06
    • 2021-10
    • 2022-03
    • 2023-02
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-10
    • 2023-11
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客