用VUEJS(2.x)做一個網(wǎng)易云音樂

前言:自己學(xué)習(xí)VUEJS也一段時間,但一直沒有做出來一東西励背。我自己一直喜歡用網(wǎng)易云音樂app乾蛤,于是乎就做了這個app。

技術(shù)棧

  • vue全家桶 (vue vue-router vuex)
  • axios
  • Muse-UI(一個基于Vue2.x的material design 風(fēng)格UI框架)

功能與思路分析

我之前學(xué)習(xí)JS的時候?qū)tml5 audio研究過捷泞,也寫過一些例子,那時的功能并不是很全面寿谴。在寫這個程序之前锁右,我好好的查閱了當前的HTML5中的audio標簽,發(fā)現(xiàn)園子上一位園友總結(jié)的很不錯(這里)讶泰。于是就先把網(wǎng)易云音樂最基本的功能實現(xiàn)咏瑟,歌單部分(這也是我喜歡網(wǎng)易云音樂的原因這一),然后實現(xiàn)音樂的上一曲痪署、下一曲码泞,播放、暫停狼犯。列表功能余寥。

后臺

后臺采用.net做為后臺提供系統(tǒng)請求所用的API(源碼),原理很簡單就是用.net偽裝成一個客戶端去訪問網(wǎng)易云音樂的API然后悯森,把返回的json數(shù)據(jù)轉(zhuǎn)發(fā)出來宋舷。同時服務(wù)端做下跨域處理。

核心代碼:

/// <summary>
/// 請求網(wǎng)易云音樂接口
/// </summary>
/// <typeparam name="T">要請求的接口類型</typeparam>
/// <param name="config">要請求的接口類型的對象</param>
/// <returns>請求結(jié)果(JSON)</returns>
public static string Request<T>(T config) where T : RequestData, new()
{
    // 請求URL
    string requestURL = config.Url;
    // 將數(shù)據(jù)包對象轉(zhuǎn)換成QueryString形式的字符串
    string @params = config.FormData.ParseQueryString();
    bool isPost = config.Method.Equals("post", StringComparison.CurrentCultureIgnoreCase);

    if (!isPost)
    {
        // get方式 拼接請求url
        string sep = requestURL.Contains('?') ? "&" : "?";
        requestURL += sep + @params;
    }

    HttpWebRequest req = (HttpWebRequest)WebRequest.Create(requestURL);
    req.Accept = "*/*";
    req.Headers.Add("Accept-Language", "zh-CN,zh;q=0.8,gl;q=0.6,zh-TW;q=0.4");
    // 如果服務(wù)端啟用了GZIP瓢姻,那么下面必須解壓祝蝠,否則一直亂碼。
    // 參見:http://www.crifan.com/set_accept_encoding_header_to_gzip_deflate_return_messy_code/
    req.Headers.Add("Accept-Encoding", "gzip,deflate,sdch");
    req.ContentType = "application/x-www-form-urlencoded";
    req.KeepAlive = true;
    req.Host = "music.163.com";
    req.Referer = "http://music.163.com/search/";
    req.UserAgent = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_9_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/33.0.1750.152 Safari/537";            
    // 設(shè)置cookies
    req.Headers["Cookie"] = "appver=1.5.2";
    req.Method = config.Method;
    req.AutomaticDecompression = DecompressionMethods.GZip;
    if (isPost)
    {
        // 寫入post請求包
        byte[] formData = Encoding.UTF8.GetBytes(@params);
        // 設(shè)置HTTP請求頭  參考:https://github.com/darknessomi/musicbox/blob/master/NEMbox/api.py          
        req.GetRequestStream().Write(formData, 0, formData.Length);
    }            
    // 發(fā)送http請求 并讀取響應(yīng)內(nèi)容返回
    return new StreamReader(req.GetResponse().GetResponseStream(), Encoding.GetEncoding("UTF-8")).ReadToEnd();
}

vuejs部分

項目結(jié)構(gòu)

├── index.html
├── main.js
├── api
│   └── ... # 抽取出API請求
├── components
│   ├── playBar.vue
│   └── ...
└── store
│    └── index.js        # 整個項目的vuex部分
└── router
│   └── router.js        # 整個項目的路由
└── utils                # 一些工具類模塊
│
└── views                # 項目中的一些route-view

說項目的路由之前幻碱,先來看一張效果圖

對于整個項目來說:視圖區(qū)別在于頂部導(dǎo)航绎狭,下面的bar的是否出來取決于,當前系統(tǒng)列表中是否有歌曲褥傍,如果有就會出現(xiàn)儡嘶。

router.js核心部分

const router = new VueRouter({
  mode: 'history',
  routes: [{
    path: '/index',
    component: require('../views/index'),
    children: [
      {
        path: 'rage',
        component: require('../views/rage')
      },
      {
        path: 'songList',
        component: require('../views/songList')
      },
      {
        path: 'leaderBoard',
        component: require('../views/leaderBoard')
      },
      {
        path: 'hotSinger',
        component: require('../views/hotSinger')
      }
    ]
  }, {
    name: 'playerDetail',
    path: '/playerDetail/:id',
    component: require('../views/playerDetail')
  }, {
    path: '/playListDetail/:id',
    name: 'playListDetail',
    component: require('../views/playListDetail')
  }, {
    path: '*', redirect: '/index/rage'
  }],
  // 讓每個頁面都滾動到頂部,改變模式為mode: history
  scrollBehavior (to, from, savedPosition) {
    if (savedPosition) {
      return savedPosition
    } else {
      return { x: 0, y: 0 }
    }
  }
})

vuex部分

這部分摔桦,主要是歌曲這一塊社付,因為不同的頁面有不同的使用到了歌曲信息,把把這部分數(shù)據(jù)放到vuex中做統(tǒng)一的數(shù)據(jù)處理邻耕!
sotre/index.js

const store = new Vuex.Store({
  state: {
    audio: {
      'id': 0,
      'name': '歌曲名稱',
      'singer': '演唱者',
      'albumPic': '/static/player-bar.png',
      'location': '',
      'album': ''
    },
    lyric: '正在加載中鸥咖。。',
    currentIndex: 0, // 當前播放的歌曲位置
    playing: false, // 是否正在播放
    loading: false, // 是否正在加載中
    showDetail: false,
    songList: [],    // 播放列表
    currentTime: 0,
    tmpCurrentTime: 0,
    durationTime: 0,
    bufferedTime: 0,
    change: false   // 判斷是更改的時間還是播放的時間
  },
  getters: {
    audio: state => state.audio,
    playing: state => state.playing,
    loading: state => state.loading,
    showDetail: state => state.showDetail,
    durationTime: state => state.durationTime,
    currentIndex: state => state.currentIndex,
    bufferedTime: state => state.bufferedTime,
    tmpCurrentTime: state => state.tmpCurrentTime,
    songList: state => state.songList,
    change: state => state.change,
    currentTime: state => state.currentTime,
    prCurrentTime: state => {
      return state.currentTime / state.durationTime * 100
    },
    prBufferedTime: state => {
      return state.bufferedTime / state.durationTime * 100
    }
  },
  mutations: {
    play (state) {
      state.playing = true
    },
    pause (state) {
      state.playing = false
    },
    toggleDetail (state) {
      state.showDetail = !state.showDetail
    },
    setAudio (state) {
      state.audio = state.songList[state.currentIndex - 1]
    },
    setAudioIndex (state, index) {
      state.audio = state.songList[index]
      state.currentIndex = index + 1
    },
    removeAudio (state, index) {
      state.songList.splice(index, 1)
      state.audio = state.songList[index - 1]
      state.currentIndex = state.currentIndex - 1
      if (state.songList.length === 0) {
        state.audio = {
          'id': 0,
          'name': '歌曲名稱',
          'singer': '演唱者',
          'albumPic': '/static/player-bar.png',
          'location': '',
          'album': ''
        }
        state.playing = false
      }
    },
    setChange (state, flag) {
      state.change = flag
    },
    setLocation (state, location) {
      state.audio.location = location
    },
    updateCurrentTime (state, time) {
      state.currentTime = time
    },
    updateDurationTime (state, time) {
      state.durationTime = time
    },
    updateBufferedTime (state, time) {
      state.bufferedTime = time
    },
    changeTime (state, time) {
      state.tmpCurrentTime = time
    },
    openLoading (state) {
      state.loading = true
    },
    closeLoading (state) {
      state.loading = false
    },
    resetAudio (state) {
      state.currentTime = 0
    },
    playNext (state) { // 播放下一曲
      state.currentIndex++
      if (state.currentIndex > state.songList.length) {
        state.currentIndex = 1
      }
      state.audio = state.songList[state.currentIndex - 1]
    },
    playPrev (state) { // 播放上一曲
      state.currentIndex--
      if (state.currentIndex < 1) {
        state.currentIndex = state.songList.length
      }
      state.audio = state.songList[state.currentIndex - 1]
    },
    addToList (state, item) {
      var flag = false
      state.songList.forEach(function (element, index) { // 檢測歌曲重復(fù)
        if (element.id === item.id) {
          flag = true
          state.currentIndex = index + 1
        }
      })
      if (!flag) {
        state.songList.push(item)
        state.currentIndex = state.songList.length
      }
    },
    setLrc (state, lrc) {
      state.lyric = lrc
    }
  },
  // 異步的數(shù)據(jù)操作
  actions: {
    getSong ({commit, state}, id) {
      commit('openLoading')
      Axios.get(api.getSong(id)).then(res => {
        // 統(tǒng)一數(shù)據(jù)模型兄世,方便后臺接口的改變
        var url = res.data.data[0].url
        commit('setAudio')
        commit('setLocation', url)
      })
    },
    getLrc ({commit, state}, id) {
      commit('setLrc', '[txt](加載中啼辣。。御滩。')
      Axios.get(api.getLrc(id)).then(res => {
        // 1鸥拧、先判斷是否有歌詞
        if (res.data.nolyric) {
          commit('setLrc', '[txt](⊙0⊙) 暫無歌詞')
        } else {
          console.log(res.data.lrc.lyric)
          commit('setLrc', res.data.lrc.lyric)
        }
      })
    }
  }
})

最后上點項目截圖

github項目地址:https://github.com/javaSwing/NeteaseCloudWebApp

目前只完成app歌單部分党远,也是最核心的部分。這個項目會一直更新!如果覺的不錯就給個star吧

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末富弦,一起剝皮案震驚了整個濱河市沟娱,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌腕柜,老刑警劉巖济似,帶你破解...
    沈念sama閱讀 218,122評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異盏缤,居然都是意外死亡砰蠢,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評論 3 395
  • 文/潘曉璐 我一進店門唉铜,熙熙樓的掌柜王于貴愁眉苦臉地迎上來台舱,“玉大人,你說我怎么就攤上這事潭流【和铮” “怎么了?”我有些...
    開封第一講書人閱讀 164,491評論 0 354
  • 文/不壞的土叔 我叫張陵幻枉,是天一觀的道長碰声。 經(jīng)常有香客問我,道長熬甫,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,636評論 1 293
  • 正文 為了忘掉前任蔓罚,我火速辦了婚禮椿肩,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘豺谈。我一直安慰自己郑象,他們只是感情好,可當我...
    茶點故事閱讀 67,676評論 6 392
  • 文/花漫 我一把揭開白布茬末。 她就那樣靜靜地躺著厂榛,像睡著了一般。 火紅的嫁衣襯著肌膚如雪丽惭。 梳的紋絲不亂的頭發(fā)上击奶,一...
    開封第一講書人閱讀 51,541評論 1 305
  • 那天,我揣著相機與錄音责掏,去河邊找鬼柜砾。 笑死,一個胖子當著我的面吹牛换衬,可吹牛的內(nèi)容都是我干的痰驱。 我是一名探鬼主播证芭,決...
    沈念sama閱讀 40,292評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼担映!你這毒婦竟也來了废士?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,211評論 0 276
  • 序言:老撾萬榮一對情侶失蹤蝇完,失蹤者是張志新(化名)和其女友劉穎湃密,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體四敞,經(jīng)...
    沈念sama閱讀 45,655評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡泛源,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,846評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了忿危。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片达箍。...
    茶點故事閱讀 39,965評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖铺厨,靈堂內(nèi)的尸體忽然破棺而出缎玫,到底是詐尸還是另有隱情,我是刑警寧澤解滓,帶...
    沈念sama閱讀 35,684評論 5 347
  • 正文 年R本政府宣布赃磨,位于F島的核電站,受9級特大地震影響洼裤,放射性物質(zhì)發(fā)生泄漏邻辉。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,295評論 3 329
  • 文/蒙蒙 一腮鞍、第九天 我趴在偏房一處隱蔽的房頂上張望值骇。 院中可真熱鬧,春花似錦移国、人聲如沸吱瘩。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,894評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽使碾。三九已至,卻和暖如春祝懂,著一層夾襖步出監(jiān)牢的瞬間票摇,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,012評論 1 269
  • 我被黑心中介騙來泰國打工嫂易, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留兄朋,地道東北人。 一個月前我還...
    沈念sama閱讀 48,126評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像颅和,于是被迫代替她去往敵國和親傅事。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,914評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 至尊寶之所以變成孫悟空峡扩,是為了救紫霞蹭越!以為生離已夠殘忍!卻怎么也沒想到要面對死別教届!當終于要永遠失去紫霞响鹃,至...
    語默若語閱讀 253評論 0 0
  • 開篇第一個內(nèi)容首先是對九型人格的簡易測試。通過108題的選擇案训,得出了我自己屬于雙重人格:悲情浪漫者和保護者买置。帶...
    14廣告馮秋潔閱讀 897評論 0 5
  • 一 小馬:“洛落!哥强霎,別睡了忿项,起來嗨!”城舞。 小馬是我大學(xué)好友轩触,曾和我一起泡妹子、賺錢家夺。畢業(yè)后脱柱,小馬去了昆明的工地工...
    張大意_閱讀 352評論 0 1
  • 犯罪份子會事先在自己的耳朵周圍涂好血紅色物質(zhì),然后用一只手擋住拉馋,假裝自己在掏耳朵榨为,并且伺機尋找作案目標,一旦發(fā)現(xiàn)合...
    是不是因為我愛你閱讀 216評論 0 0