vue——一個頁面實現音樂播放器

羅羅諾亞丶索隆發表於2018-02-02

vue——一個頁面實現音樂播放器

vue——一個頁面實現音樂播放器

專案地址:github.com/ermu5922752…

演示地址:ermu592275254.github.io/MiniMusicPl…(歌曲連結已失效)

開發前構思

介面

做音樂播放器,介面一定要炫酷。太low了聽歌沒感覺。本身是為了在上班的時候用,於是做成了一個類似網易雲音樂的介面,大小合適。不用相容手機端。

用css做圖示

本懷著簡單實用的需求去考慮,圖示可用SVG、url或者css。相對於url,SVG和css更好一些。為了修煉,最終選擇的css。利用好after和before,能減少很多dom巢狀。

 .next {
    position: relative;
    display: inline-block;
    height: 36px;
    width: 36px;
    border: 2px solid #fff;
    border-radius: 20px;
    -webkit-border-radius: 20px;
    -moz-border-radius: 20px;
}
        
.next:before {
    content: '';
    height: 0;
    width: 0;
    display: block;
    border: 10px transparent solid;
    border-right-width: 0;
    border-left-color: #fff;
    position: absolute;
    top: 8px;
    left: 10px;
}

.next:after {
    content: '';
    height: 20px;
    width: 4px;
    display: block;
    background: #fff;
    position: absolute;
    top: 8px;
    left: 22px;
}

複製程式碼

畫一個唱片

網易雲的唱片很好看,我也要弄一個唱片! 用好box-shadow,一個元素便可以做成很好看的唱片效果。

.disc {
    position: relative;
    margin-top: 10%;
    margin-left: 10%;
    width: 300px;
    height: 300px;
    border-radius: 300px;
    transform: rotate(45deg);
    background-image: radial-gradient(5em 30em ellipse, #fff, #000);
    border: 2px solid #131313;
    box-shadow: 0 0 0 10px #343935;
    opacity: 0.7;
}
複製程式碼

用range做進度條

audio本身的樣式很難看,並且不同的瀏覽器呈現出來的效果也不一樣。當然你可以修改audio的樣式,傳統的做法是通過controls屬性來隱藏audio,然後用div來代替。現在是html5時代了,當然要用更符合場景的新元素————range。

 input[type=range] {
    -webkit-appearance: none;
    width: 80%;
    height: 8px;
    border-radius: 10px;
    background-color: #fff;
}
input[type=range]::-webkit-slider-thumb{
    -webkit-appearance: none;
} 
input[type=range]::-webkit-slider-runnable-track {
    height: 8px;
    border-radius: 20px;
}
input[type=range]:focus {
    outline: none;
}

input[type=range]::-webkit-slider-thumb {
    -webkit-appearance: none;
    margin-top: -3px;
    height: 14px;
    width: 14px;
    background: #eb7470;
    border-radius: 50%;
    border: solid 3px #fff;
    box-shadow: 0 0 0 3px rgba(255, 255, 255, 0.5);
}
複製程式碼

背景濾鏡模糊

將圖片設定為背景的感覺很棒,可以說整個播放器的顏值這背景提供了一半。設定也非常簡單,用到了css3的濾鏡。

.bg-blur {
    position: absolute;
    top: 50%;
    left: 50%;
    transform: translate(-50%, -50%);
    width: 100%;
    height: 100%;
    background-position: center;
    background-repeat: no-repeat;
    background-size: cover;
    filter: blur(20px);
    z-index: -1;
}
複製程式碼

背景圖片通過js控制。

 <div class="bg-blur" :style="`background-image:url(${currentSong.album_logo})`"></div>
複製程式碼

歌曲資源

爬下介面

直接去蝦米官網開啟network,將url複製到postman裡面去做請求。通過修改headers發現,會校驗Referer。也就是說只有蝦米允許的域名可以訪問此介面。 http://api.xiami.com/web?v=2.0&app_key=1&key=aliez&page=1&limit=5&callback=jsonp154&r=search/songs

解決跨域問題

因為介面支援jsonp。起初嘗試將chrome瀏覽器設定跨域,然後通過$.ajax去做一個jsonp的請求。可以正常訪問。

之後突然不行了,是不是蝦米做了限制?

遂改用node啟動一個服務,去偽造referer發起請求,再將請求轉發到頁面。無意中寫了一個代理。

...
case '/song':
    let songOptions = {
        url: 'http://api.xiami.com/web?'+ urlArr[1],
        headers: {
            'Referer': 'http://m.xiami.com/'
        }
    };
    function callback1(error, response, body) {
        if (!error && response.statusCode == 200) {
            res.end(body);
        }
    }
    request(songOptions, callback1);
    break;
...
複製程式碼

歌詞滾動

作為一款逼格比較高的播放器,歌詞滾動是必須的。

原理

將每一句歌詞儲存為一個物件,有對應的時間。當歌曲播放的當前時長大於或等於歌詞的時間,小於此歌詞的下一句歌詞的時間,那麼就將此歌詞滾動到可視區域。並且修改字型顏色。

格式化歌詞

介面返回的歌詞一臉懵逼,仔細研究一下,發現是有規律的。

[ti:aLIEz]
[ar:SawanoHiroyuki[nZk]:mizuki]
[al:o1]
[ly:澤野弘之]
[mu:澤野弘之]
[ma:]
[pu:]
[by:ttpod]
[total:268512]
[offset:0]
[00:00.000]<195>aLIEz <199>- <451>SawanoHiroyuki[nZk]:mizuki
[x-trans]徹頭徹尾的謊言 - SawanoHiroyuki[nZk]:mizuki
[00:01.095]<201>作<250>詞<200>:<201>澤<200>野<199>弘<300>之
[x-trans]
[00:02.846]<200>作<150>曲<150>:<200>澤<200>野<351>弘<349>之
[x-trans]
[00:20.828]<200>決<250>め<200>つ<201>け<149>ば<201>か<349>り
[x-trans]一直獨斷專權
[00:23.279]<200>自<200>惚<200>れ<200>を<200>著<400>た
[x-trans]總是自負逞強
[00:24.979]<200>チ<200>ー<200>プ<450>な<550>hokori<350>で
[x-trans]明明只是一文不值的驕傲
......
   refactoringLyrics(lyric){
    let text = lyric.split('[offset:0]')[1];
    let textArr = text.split('\n');
    let lyricsArr = [], translate = [];
    textArr.forEach((item, index) => {
        let time = 0, text = '';
        if (item.indexOf('[x-trans]') > -1) {
            translate.push(item.split('[x-trans]')[1])
        } else if (item.trim() != '') {
            time = item.slice(1, 6).split(':');
            time = parseInt(time[0]) * 60 + parseInt(time[1]);
            text = item.slice(11);
            let arr = text.split('>');
            let str = arr.reduce((a, b) => {
                return a.split('<')[0] + b.split('<')[0]
            });
            let obj = {
                time: time,
                text: str
            };
            lyricsArr.push(obj);
        }
    });
    for (let i in translate) {
        lyricsArr[i].text = lyricsArr[i].text + '\n' + translate[i];
    }
    this.currentLyrics = lyricsArr;
},
複製程式碼

搜尋欄實現

同檔案下子元件掛載

為了遵循模組化開發,決定將搜尋欄寫成一個子元件。在同一頁面下寫子元件,子元件掛載到對應的template就有講究了。此template不能被父元件的掛載元素包含,否則父元件渲染時會因為無法渲染子元件中的資料而報undefined。

<div id="app" class="main">
...
</div>
<template id="search-box">
...
</template>

var searchBox = {
        template: '#search-box',
        props: {
            isShow: Boolean,
            openFun: Function
        },
        data(){
            return {
                resultList: [],
                searchValue: '',
            }
        },
        methods: {
        }
    };
 new Vue({
    el: '#app',
    components: {
        'com-tip': tip,
        'search-box': searchBox
    },
    ...
})
複製程式碼

eventBus解決資料傳輸

通過jsonp去請求資料,需要設定一個callback函式,此callback寫成一個全域性函式,如果不這樣寫,而是通過 searchBox.methods.callback的形式,this指向將為methods。而無法直接給searchBox的data賦值。 於是通過eventBus來處理,這樣更易維護。

var EventBus = new Vue();
var callBack = function(result) {
    console.log(result);
    EventBus.$emit('callBack', result);
};
...
mounted(){
    let self = this;
    EventBus.$on('callBack', function(res) {
        if (res && res.data) {
            self.resultList = res.data.songs;
        }
    })
}
...
複製程式碼

localStrong儲存歌曲資訊

下次再開啟,應該播放列表應該保留上一次的資料,這個可直接用localstrong實現

技術補充

濾鏡模糊 http://www.zhangxinxu.com/wordpress/2013/11/css-svg-image-blur/

css圖示 www.uiplayground.in/css3-icons/

flex 相容寫法: www.cnblogs.com/iriszhang/p…

flex 佈局教程 www.ruanyifeng.com/blog/2015/0…

滑動條樣式自定義 blog.csdn.net/u013347241/…

隱藏滾動條相容 segmentfault.com/q/101000000…

audio屬性 developer.mozilla.org/zh-CN/docs/…

audio事件 developer.mozilla.org/zh-CN/docs/…

踩了坑

prop傳遞資料

使用cdn,vue的prop只支援中線格式,駝峰格式不生效

ps: 在用webpack打包的專案中用駝峰是可以,在打包過程中,會做處理。

// 正確寫法
<search-box :is-show="showSearch" :open-fun="openSearch" @push-song="pushNewSong"
                @play-song="playSong"></search-box>
// 錯誤寫法
<search-box :isShow="showSearch" :openFun="openSearch" @pushSong="pushNewSong"
                @playSong="playSong"></search-box>
複製程式碼

待優化

手動修改進度,偶爾會不生效。

搜尋暫不支援分頁

不支援建歌單

背景顏色與進度條顏色相近需修改進度條顏色

不支援播放模式選擇-單曲迴圈-隨機播放

相關文章