成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

HTML5移動端音樂播放器(啟蒙篇)

Lin_R / 2932人閱讀

摘要:前些天偷臺風(fēng)的閑暇時寫了一個移動端音樂播放器,作為練手項(xiàng)目。有的歌詞周杰倫算什么男人格式是時間點(diǎn)時間歌詞創(chuàng)建映射首先以將歌詞字符串分割成以時間點(diǎn)文字的數(shù)組,但由于這樣分割之后最后一個元素是空的,所以用刪除最后一個元素。

這段時間公司一直在做一個PC的教育類單頁應(yīng)用,龐大復(fù)雜,涉及非常多H5的知識,音頻就是其中的一部分。前些天偷臺風(fēng)的閑暇時寫了一個移動端音樂播放器,作為練手項(xiàng)目。

在線地址:請猛擊這里
源碼:請猛擊這里

注意:使用PC瀏覽最好打開移動設(shè)備模式,使用移動設(shè)備瀏覽需要關(guān)閉無痕瀏覽模式(否則無法使用本地存儲,一般瀏覽器都是默認(rèn)不開啟),項(xiàng)目需要在本地服務(wù)器線上服務(wù)器運(yùn)行,以file:///形式的地址打開是無法進(jìn)行ajax請求的,從而無法看到音樂數(shù)據(jù)。

項(xiàng)目實(shí)現(xiàn)的功能及所用知識

播放器的基礎(chǔ)操作,上一首,下一首(順序播放、隨機(jī)播放、單曲循環(huán)),播放暫停,滑動時間軸的歌詞定位

初始handlebar模板渲染音樂列表數(shù)據(jù),下拉滾動加載音樂列表數(shù)據(jù)。

歌曲列表可添加喜愛音樂,于下次刷新時更新喜愛音樂列表,基于HTML5本地存儲。

布局采用rem布局,自適應(yīng)移動端手機(jī)設(shè)備。

iconfont在線圖標(biāo)應(yīng)用的使用

項(xiàng)目目錄文件結(jié)構(gòu)

css:存放樣式文件
lib:?存放公共腳本庫
js:?存放項(xiàng)目腳本文件
img:?存放圖片
fonts:?項(xiàng)目字體文件
res:?項(xiàng)目音樂資源
ui:項(xiàng)目ui文件(psd)

// ============================配置變量================================
var rootPath = window.location.href.replace(//w+.w+/, "/");
var Settings = {
playmode: 0, //0列表循環(huán),1隨機(jī),2為單曲循環(huán)
volume: 0.5, //音量
initNum: 10, //列表初始化歌曲數(shù)
reqNum: 10 //后續(xù)請求歌曲數(shù)
};

// ============================工具函數(shù)================================
var Util = (function() {
return {

}
})()
// ============================Dom選擇器================================
var Dom = {

}

// ============================全局變量================================
var winH = $(window).height();

var songNum = 0; //當(dāng)前列表歌曲數(shù)目
var lrcHighIndex = 0; // 歌詞高亮索引
var lrcMoveIndex = 0; // 歌詞移動單位索引
var moveDis = 0; // 單句歌詞每次移動距離

var duration = 0; // 當(dāng)前歌曲的時間
var index = 0; //當(dāng)前播放歌曲的索引
var songInfo = null; // 當(dāng)前歌曲信息
var songModelUI = null; // 當(dāng)前歌曲UI模型
var timeArr = []; //當(dāng)前歌曲時間數(shù)組
var formatTimeArr = []; //當(dāng)前歌曲時間數(shù)組(格式化為秒數(shù))

// ============================入口函數(shù)================================
function main() {
initUIFrame();
var initModel = PlayerModel();

var songListUI = ModelUIFrame(Dom.songListContainer);
var lsongListUI = ModelUIFrame(Dom.lSongListContainer);
initModel.getSongList("data/data.json", function(data) {
// 生成所有歌曲列表
songListUI.renderList(data, 0, null, function() {
songListUI.updateList();
});
// 生成喜愛歌曲列表
initModel.getLoveSongArr(function(lSongArr) {
lsongListUI.renderList(data, 1, lSongArr);
});
// 添加動畫
Util.addAnimationDelay(Dom.song);
// 保存歌詞數(shù)據(jù)
initModel.saveLyric(data);

});
EventHandler();
}
// ============================初始化UI函數(shù)================================
function initUIFrame() {

}
// ============================實(shí)現(xiàn)數(shù)據(jù)交互方法================================
function PlayerModel() {

}
// ============================模型動態(tài)UI模塊================================
function ModelUIFrame(container) {

}
// ============================事件綁定模塊================================
function EventHandler() {

}
// 調(diào)用入口函數(shù)
main();
功能點(diǎn)詳解 Handlebar.js初次渲染及滾動加載

使用前端模板優(yōu)點(diǎn)是把數(shù)據(jù)和結(jié)構(gòu)分離出來,代碼更清晰。但后來發(fā)現(xiàn)handlerbar.js似乎無法在js中示例模板對象,而html中的handlebar在初次進(jìn)入頁面便會被編譯了,因此后續(xù)添加音樂還是采用傳統(tǒng)的拼接字符串的方式,如果你有更優(yōu)雅的動態(tài)加載方式,歡迎討論交流。

html:handlebars模板包含在script標(biāo)簽之中并且type類型為”text/x-handlebars-template”,在初始化頁面的時候根據(jù)js獲取數(shù)據(jù)植入后就渲染出相應(yīng)的html。

js:

function renderAllList(data) {
    var preTpl;
    var lsongArr = Util.getItem("lsonglist") === null ? [] : JSON.parse(Util.getItem("lsonglist"));
    // 生成列表
    if (!sListTpl) {
        // 后續(xù)動態(tài)生成歌曲
        var tpl = "";
        var songIndex = songNum;
        $.each(data, function(index, el) {
            if (index >= songIndex && index < songIndex + Settings.reqNum) {
                tpl += "
  • [站外圖片上傳中……(1)]

    " + el.songName + "

    " + el.singer + "
  • "; songNum++; } }); $(container).append($(tpl)); } else { // 首次生成歌曲 preTpl = Handlebars.compile(sListTpl); $(container).html(preTpl(data)); } // 更新喜愛圖標(biāo) if (lsongArr.length !== 0) { $.each(lsongArr, function(index, val) { Dom.songListContainer.find(".song").eq(val).find(".icon-love").addClass("active"); }); } }
    rem布局自適應(yīng)方案

    大體上指的是html根元素上定義一個字體大小,然后css樣式定義時使用rem作為單位,包括margin、paffffding、用于絕對定位的單位等等。然后js根據(jù)手機(jī)設(shè)備的屏幕大小,改變根字體的大小,這樣整個頁面也會跟著相應(yīng)的縮小或放大。
    更多詳解,請看這一篇文章《移動端自適應(yīng)布局解決方案——rem》,您可以猛擊這里跳轉(zhuǎn)。

    關(guān)于歌詞的同步方案實(shí)現(xiàn)

    目前音樂播放器的歌詞同步顯示大概有兩種,一種是精確到單個文字,一種是精確到單行歌詞。本文實(shí)現(xiàn)的是第二種。

    整體實(shí)現(xiàn)思路

    頁面初始化時,請求歌曲數(shù)據(jù)json(本地json文件模擬),其中歌名、歌手、圖片等按需渲染到html中,將歌詞存儲到localStorage中。此時,F(xiàn)12打開chrome調(diào)試器,進(jìn)入Application-LocalStorage可以看到:

    點(diǎn)擊一首歌進(jìn)入播放頁面后,歌詞就會從本地存儲中讀取,此時你會看到生成這樣的歌詞結(jié)構(gòu):

    每一行歌詞都將要將歌詞時間綁定在data-point上,監(jiān)聽歌曲播放的timeupdate事件,當(dāng)歌曲的時間(經(jīng)過取整處理)與當(dāng)前data-point值相等時,就為當(dāng)前歌詞高亮(相當(dāng)于給p添加current類名),并且根據(jù)當(dāng)前高亮歌詞的index索引將整個歌詞盒子向上移動p標(biāo)簽的高度+margin-top的高度

    lrc歌詞的結(jié)構(gòu)

    來自網(wǎng)易云音樂的歌詞數(shù)據(jù):

    [00:14.64]如果不是那鏡子
    [00:16.73]不像你不藏秘密
    [00:21.26]我還不肯相信
    [00:23.02]沒有你我的笑更美麗
    [00:28.99]那天聽你在電話里略帶抱歉的關(guān)心
    
    
    [00:16.959]摘一顆蘋果
    [00:19.800]等你從門前經(jīng)過
    [00:22.700]送到你的手中幫你解渴
    [00:25.570]像夏天的可樂
    
    
    [00:00.00] 作曲 : 周杰倫
    [00:01.00] 作詞 : 周杰倫
    [00:05.620]
    [00:37.980]親吻你的手
    

    可以看到格式 = [時間點(diǎn)] + 要顯示的文字 +
    這里有兩個坑需要注意:
    有的歌詞秒數(shù)是精確到小數(shù)點(diǎn)后兩位,有的是三位
    有的歌詞(周杰倫《算什么男人》)格式是[時間點(diǎn)]+

    時間歌詞創(chuàng)建映射

    首先以n將歌詞字符串分割成以[時間點(diǎn)]文字的數(shù)組,但由于這樣分割之后最后一個元素是空的,所以用tempArr.splice(-1, 1)刪除最后一個元素。
    接下來循環(huán)遍歷這個臨時數(shù)組,由于上面提到的秒數(shù)精確度的問題,所以判斷一下index為9是否為數(shù)字,若為數(shù)字則將該位數(shù)字刪除。(采用字符串截取方式,若你對js字符串方法不熟悉,可以猛擊這里)
    經(jīng)過這樣的處理之后,臨時數(shù)組的元素格式不再有區(qū)別了,此時再進(jìn)行字符串截取,將截取到的時間點(diǎn)放入timeArr,將截取的歌詞放入lyricArr,并以返回保存著這兩個變量的對象。

    function createArrMap(lyric) {
        var timeArr = [],
            lyricArr = [];
        var tempArr = lyric.split("
    ");
        tempArr.splice(-1, 1);
        var tempStr = "";
        $(tempArr).each(function(index) {
            tempStr = this;
            if (tempStr.charAt(9).match(/d/) !== null) {
                tempStr = tempStr.substring(0, 9) + tempStr.substring(10);
            }
            timeArr.push(tempStr.substring(0, 10));
            lyricArr.push(tempStr.substring(10));
        });
        return {
            timeArr: timeArr,
            lyricArr: lyricArr
        };
    }
    生成歌詞

    由于上面歌詞格式造成時間點(diǎn)對應(yīng)的歌詞為空,此時如果渲染出一個
    標(biāo)簽的高度將為0,這會影響歌詞向上移動距離的不統(tǒng)一。因此下面作出個判斷如果為空,則替換為“————–”。(為空的時候大多數(shù)是歌曲中間停頓或過渡的時候)

    function renderLyric(songinfo) {
        var arrMap = Util.createArrMap(songinfo.lyric);
        var tpl = "";
        $.each(arrMap.lyricArr, function(index, lyric) {
            var lyricContent = lyric === "" ? "--------------" : lyric;
            tpl += "

    " + lyricContent + "

    "; }); Dom.lrcwrap.html(tpl); }
    歌詞同步

    歌詞同步我寫在了syncLyric方法中,監(jiān)聽audio元素的timeupdate事件調(diào)用。
    這個方法接收兩個參數(shù),第一個是當(dāng)前播放歌曲時間(秒),第二個是轉(zhuǎn)化為秒數(shù)的時間點(diǎn)數(shù)組。
    如果當(dāng)前時間>=時間點(diǎn),那么高亮當(dāng)前歌詞(以lrcHighIndex)存儲,并且lrcHighIndex自增1。
    當(dāng)歌詞高亮索引lrcHighIndex>=1即歌詞高亮不為第一句時,計(jì)算索引并讓歌詞盒子向上移動。

    function syncLyric(curS, formatTimeArr) {
        if (Math.floor(curS) >= formatTimeArr[lrcHighIndex]) {
            Dom.lrc.eq(lrcHighIndex).addClass("current").siblings().removeClass("current");
            if (lrcHighIndex >= 1) {
                lrcMoveIndex = lrcHighIndex - 2;
                moveDis += Util.getMoveDis(lrcMoveIndex);
                Dom.lrcwrap.animate({
                    "top": "-" + moveDis + "px"
                }, 100);
                lrcMoveIndex++;
            }
            lrcHighIndex++;
        }
    }

    文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

    轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/80730.html

    相關(guān)文章

    • Vue 實(shí)現(xiàn)網(wǎng)易云音樂 WebApp

      摘要:基于等開發(fā)一款移動端音樂,界面參考了安卓版的網(wǎng)易云音樂布局適配常見移動端。圖標(biāo)使用阿里巴巴圖標(biāo)庫,中間的唱片旋轉(zhuǎn)動畫使用了實(shí)現(xiàn)。搜索功能實(shí)現(xiàn)功能搜索歌手歌單歌曲熱門搜索數(shù)據(jù)節(jié)流上拉刷新保存搜索記錄。 基于 Vue(2.5) + vuex + vue-router + vue-axios +better-scroll + Scss + ES6 等開發(fā)一款移動端音樂 WebApp,UI ...

      Karuru 評論0 收藏0

    發(fā)表評論

    0條評論

    最新活動
    閱讀需要支付1元查看
    <