多條語音消息合成一整條連續播放與進度條功能技術點!

多條語音匯成一整條開發

最近在作一個多條語音合成一整條語音而且結合進度條能夠快進或者後退功能,功能不復雜,可是所遇到的坑很多,因此我就想着把我遇到的坑寫下來,但願之後有用到的小夥伴們能夠少走點彎路:
項目原型圖片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();    
    }
});
代碼稍微有點亂,主要是供你們參考一下實現的思想,若是哪裏不足也請你們多多指教!

須要注意的地方:

  1. ios給audio設置currentTime須要等待語音資源加載完成,查了不少資料發現只有設置定時器判斷audio 的readyState方案兼容性相較好;
  2. audio在微信瀏覽器與ios或者一些安卓瀏覽器不會自動播放,涉及到流量問題,微信瀏覽器調用wx.ready方法達到自動播放效果;
  3. 連續播放時,播放完當前播放下一條時會有必定資源加載時間,致使最後時間會多出幾秒。解決方案:當播放下一條語音時獲取以前語音累計所加時間,從新設置定時器開始時間;

H5有些方法兼容性仍是有待提升!!!css

相關文章
相關標籤/搜索