最近在作一個多條語音合成一整條語音而且結合進度條能夠快進或者後退功能,功能不復雜,可是所遇到的坑很多,因此我就想着把我遇到的坑寫下來,但願之後有用到的小夥伴們能夠少走點彎路:
javascript
html+css代碼: <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title></title> <link rel="stylesheet" type="text/css" href="resets.css"/> <script src="jquery-3.1.1.min.js" type="text/javascript" charset="utf-8"></script> <script src="flexible.js" type="text/javascript" charset="utf-8"></script> //第一步:首先加載一個微信JS-SDK <script src="http://res.wx.qq.com/open/js/jweixin-1.0.0.js"></script> <style type="text/css"> .duration-time-container{ height: 3rem; width: 100%; padding-top: 3.666rem; } .control-bar{ position: relative; height: 4px; background: #ebebeb; border-radius: .133333rem; width: 70%; margin: 0 auto; } #slider{ position: absolute; display: block; top: 50%; transform: translate(-50%,-50%); cursor: pointer; width: 0.4rem; height: 0.4rem; left: 0; -webkit-border-radius: .4rem; border-radius: .4rem; background: #fff; border: 1px solid #e6e6e6; -webkit-box-shadow: 0 0.04rem 0.066667rem 0 hsla(0,0%,90%,.8); box-shadow: 0 0.04rem 0.066667rem 0 hsla(0,0%,90%,.8); } #slider::after{ display: block; content: ""; width: 1.6rem; height: .8rem; opacity: 0; position: absolute; display: block; top: 50%; left: 50%; transform: translate(-50%,-50%); } #playing-bar{ display: block; position: absolute; width: 0; height: 100%; -webkit-border-radius: .266667rem; border-radius: .266667rem; background: #ef5670; } .play-cont{ margin-top: 0.8rem; text-align: center; } .on-log{ width: 1.653333rem; height: 1.653333rem; background-size: 100%; display: inline-block; cursor :pointer; } .btn-play{ background: url() 50%; } .btn-pause{ background: url() 50%; } </style> </head> <body> <p class="cont"></p> <div class="duration-time-container"> <div class="control-bar"> <span id="playing-bar" ></span> <span id="slider"></span> <div class="time"> <div></div> <div></div> </div> </div> </div> <div class="play-cont"> <span class="btn-play on-log" data-log = "播放"></span> </div> <!--語音標籤--> <audio id="first-audio" preload="auto"></audio> <script src="allVoice.js" type="text/javascript" charset="utf-8"></script> <script type="text/javascript"> $(function(){ getDistance("slider","playing-bar"); //用於微信瀏覽器自動播放 if(isWeiXinClient()){//檢測微信環境 wx.ready(function() { playAudio(startSeconds,currtIndex,result); }); } }); </script> </body> </html>
js代碼 數組結構: { "id":299, "userId":22, "fileUrl":url, "seconds":13//每條語音的時間長度 } //請求語音數組 var result = res.result; //語音總長度 var voiceLength = 0; //語音對象的秒數 var voiceSecond = 0; //進度條動畫 var progress; //進度條長度 var width = $(".control-bar").width(); //拖拽時須要減去左邊距,因爲個人進度條長度爲70%,因此marginleft值爲body寬的百分15 var marLeft = $("body").width()*0.15; //語音播放到多少秒用於尋找對應的語音消息 var currentVoiceLeg = 0; //語音位置,單位百分比 var voicePosition; //定義移動值 var moveleft = 0; //語音開始播放位置 var startSeconds = 0; //語音是否在播放 var isPlay = false; //是否結束 var isOver = true; for(var i=0;i<result.length;i++){ result[i]["startLength"] = voiceSecond; voiceSecond+=result[i].seconds; result[i]["endLength"] = voiceSecond; voiceLength += result[i].seconds; }; //聲音百分比 var voicePercent = Number((voiceLength/100).toFixed(2)); //百分比/聲音秒 var secondNum = width/voiceLength; //當前播放audio var currut_audio = $("#first-audio")[0]; //初始化秒數 $(".voice-length").text(getVoiceTime(voiceLength)); $(".voice-currtTime").text(getVoiceTime(currentVoiceLeg)); /* * 格式化秒數 * voiceLength 秒數 */ function getVoiceTime(voiceLength){ var leng = parseInt(voiceLength); if(leng < 60){ if(leng<10){ return "00:"+"0"+leng; }else{ return "00:"+leng; } }else{ var mint = parseInt(leng/60),second = leng-mint*60; if(mint<10){ mint = "0"+mint; }; if(second<10){ second = "0"+second; }; return mint+':'+second; } }; /*拖拽效果*/ /* * 獲取拖拽距離 * @param string btnId拖拽元素id值 * @param string progId進度條id值 */ function getDistance(btnId,progId){ var btn = document.getElementById(btnId); btn.addEventListener("touchmove",function(evt){ evt.preventDefault(); if (evt.targetTouches.length == 1) { var touch = event.targetTouches[0]; //移動值應該減去拖拽塊1/2寬度與距離最左邊的值 moveleft = touch.pageX-marLeft; voicePosition = (moveleft/width)*100; if(moveleft<=0){//拖拽塊的left值最小是0; voicePosition = 0; moveleft = 0; }; if(moveleft>=parseInt(width)-10){//拖拽塊的left值最大是拖拽區的width減去拖拽塊的width; voicePosition = 100; moveleft = parseInt(width)-10; } setProgressBar(btnId,progId,voicePosition+"%"); }; },false); btn.addEventListener("touchend",function(evt){ isPlay = false; currentVoiceLeg = Number(voicePercent*voicePosition); findCurrentVoice(result); $(".voice-currtTime").text(getVoiceTime(currentVoiceLeg)); },false); }; /* * 動態設置拖拽元素left 和進度條長度 * @param string btnId拖拽元素id值 * @param string progId進度條id值 * @param string value須要設置的值 */ function setProgressBar(btnId,progId,value){ $("#"+btnId).css('left',value); $("#"+progId).width(value); }; /* * 匹配對應的語音消息而且快進 */ //當前語音消息下標 var currtIndex = 0; function findCurrentVoice(arry){ for(var i =0;i<arry.length;i++){ var num = Number(arry[i].endLength); if(num >= currentVoiceLeg){ startSeconds = Number(currentVoiceLeg - arry[i].startLength); var src = arry[i].fileUrl; currtIndex =i; playAudio(startSeconds,currtIndex,arry); break; }; }; }; /* * 播放動畫 * btnId 動畫按鈕id * progId 進度條動畫id */ function setIntVoice(btnId,progId){ if(!isPlay){ progress = setInterval(function(){ moveleft+=secondNum; currentVoiceLeg+=1; voicePosition = moveleft/width*100; if(currentVoiceLeg>=voiceLength&&isOver){ voicePosition = 0; moveleft = 0; currentVoiceLeg =0; controlAudio(); }; $(".voice-currtTime").text(getVoiceTime(currentVoiceLeg)); setProgressBar(btnId,progId,voicePosition+"%"); },1000); } }; /* * 播放聲音函數 * @param string src 播放地址 * @param string num 播放聲音的位置(秒) */ //當前語音下標 var currntNum; function playAudio(num,index,arry){ isOver = false; var src = arry[index].fileUrl; currut_audio.src = src; currut_audio.load(); currntNum = index; var audioState = setInterval(function(){ if(currut_audio.readyState>1){ currut_audio.currentTime = num; currut_audio.play(); controlAudio(); clearInterval(audioState); }; },100) currut_audio.addEventListener("error", function (e) { alert('親,網絡有延遲,請稍後重試!'); }); }; //監聽語音播放事件 currut_audio.addEventListener("play", function () { isPlay = true; }, false); //暫停 currut_audio.addEventListener("pause", function () { isPlay = false; }, false); //中止 currut_audio.addEventListener("ended", function () { startSeconds = 0; //播放完成後切換音頻 currtIndex+=1; if(currtIndex<=result.length-1){ currentVoiceLeg = result[currtIndex].startLength; playAudio(startSeconds,currtIndex,result); }else{ startSeconds = 0; currtIndex = 0; isOver = true; isPlay = false; } }, false); //控制界面交互,定時器 function controlAudio(){ clearInterval(progress); if(!isPlay&&!isOver){ $(".on-log").addClass("btn-pause").removeClass("btn-play").attr("data-log","暫停"); if(currentVoiceLeg != 0){ currut_audio.play(); }; setIntVoice("slider","playing-bar"); }else{ $(".on-log").addClass("btn-play").removeClass("btn-pause").attr("data-log","播放"); clearInterval(progress); currut_audio.pause(); } }; //點擊播放按鈕 $(".on-log").on("click",function(){ if(isOver){ isOver = false; playAudio(startSeconds,currtIndex,result); }else{ controlAudio(); } });
代碼稍微有點亂,主要是供你們參考一下實現的思想,若是哪裏不足也請你們多多指教!
H5有些方法兼容性仍是有待提升!!!css