寫在前面css
1.入門幾天小白的做品,但願爲您有幫助,有好的意見或簡易煩請賜教html
2.微信小程序審覈音樂類別已經下架,想要發佈選題需慎重。附一個參考連接,感謝https://www.hishop.com.cn/xiaocx/show_53774.htmlgit
3.寫的過程當中參考了前輩們的方法,借過幾位博客園、CSDN、簡書前輩的路,這裏表示感謝。github
4.官方API很重要數據庫
寫在第二json
資深大牛在地鐵上問我有沒有玩過微信小程序,自覺慚愧。因而萌發了寫個小程序長長見識的想法,畢竟,誰都想要作一位行業大牛嘛。小程序
寫這個小程序花了4天,中間第一天無從下手,次日開悟,到後兩天的優化。這中間我收穫極大,感謝生活。OK,廢話很少說。進入正文微信小程序
正文在這裏api
先看效果吧 求求你點開我吧數組
小程序有兩個頁面,主頁與播放頁,由於採用了leanCloud做爲後臺數據開發,因此有一個lib包
樹結構,上圖 附微信小程序使用leanCloud連接
index頁的功能描述:提供音樂查找與選擇,搜索框不輸入點擊搜索獲得數據庫中因此音樂文件(.mp3格式),支持對歌名或歌手的模糊查詢;
點擊列表中的某一首便可跳轉至播放頁進行播放,從播放頁清單回退至index頁時,index頁底部有播放小窗,點擊可回到播放頁
看這裏:
代碼送上:index.wxml
<view class='theMain' style='background-image:url({{bgimage}});'> <view> <image class='image1'src="{{imageUrl}}" mode='aspectFit' wx:if="{{!searchTop}}"> </image> </view> <view class='page_row'> <view class="search"> <view class="df search_arr"> <icon class="searchcion" size='20' type='search'></icon> <input class="searchInput" placeholder="請輸入歌名,歌手" value="{{searchValue}}" bindinput='searchValueInput'/> </view> </view> <view class='sousuo' bindtap='sousuoButton'>搜索</view> </view> <view class='listBar'> <text class="search_top" style='width:80%;' wx:if="{{searchTop}}">搜索結果</text> </view> <view wx:for='{{json}}' class='music_list' bindtap='playTheMusic'data-name="{{item.name}}" data-url="{{item.url}}"> <text class='musictext'>{{item.name}}</text> <view class='url'>{{item.url}}</view> </view> <view class="search_no" wx:if="{{!centent_Show}}"> <text>暫時沒有庫存,聯繫馮大神上傳哈</text> </view> <view class='littlebar' bindtap='littlebar'> <view class='littleImage' style='background-image:url({{imageUrl}});'></view> <view class='littleName'> <text class='songNameText'>{{songName}}</text> </view> </view> </view>
index.wxss這個css樣式代碼我以爲就沒啥可看的啦,這個仍是看本身喜愛調,若是須要的話,文章最下方附有Github連接,
重要的仍是在index.js上啦,一塊兒來看看(我以爲不錯的地方直接在代碼上標註了,能夠參考一下)
const AV = require("../../libs/av-weapp-min");//這裏是對LeanCloud的引用,你們須要的話可問度娘,不少詳細教程,我仍是蠻喜歡這個工具的 var app=getApp() Page({ /** * 頁面的初始數據 */ data: { imageUrl: "http://lc-9qxpppvr.cn-n1.lcfile.com/9c1af72ea0506789a9b9.jpg", searchValue:'', //搜索值 centent_Show: true, //這個可看wxml中的wx:if屬性,用來動態顯示與隱藏 searchTop:false, //同上 bgimage: '', //背景圖片,在獲得搜索結果的時候顯示 toName:'', songName:'' }, searchValueInput: function (e) {//獲得搜索框的內容並渲染到data下的searchValue中 var value = e.detail.value; this.setData({ searchValue: value, }); }, sousuoButton:function(){//對LeanCloud數據庫的查詢操做
var that=this; //這裏很是重要,注意,此函數中then方法內還有一個函數,而該函數須要用到setData渲染數據。通過兩層不能夠直接用this.setData var nameQuery1 = new AV.Query('_File'); nameQuery1.contains('name', that.data.searchValue); var nameQuery2 = new AV.Query('_File'); nameQuery2.contains('name', '.mp3'); var query = new AV.Query.and(nameQuery1, nameQuery2);//這裏至關於數據庫where語句下的and操做 query.find().then(function (results) { // results is an array of AV.Object. //將data轉成json格式 //轉爲數組 var jsonObj = JSON.parse(JSON.stringify(results)); app.globalData.musicList=jsonObj.concat(); //設置了全局變量,concat()方法爲數組的複製,playing頁面須要。具體用法可問度娘,app對象爲頂部建立的對象 if (jsonObj.length == 0) { that.setData({ centent_Show: false, }); return; } that.setData({ json: app.globalData.musicList, searchTop:true, bgimage: "http://lc-9qxpppvr.cn-n1.lcfile.com/9c1af72ea0506789a9b9.jpg", centent_Show:true }); console.log(that.data.c) }, function (error) { console.log(error); // error is an instance of AVError. }); }, //點擊底部音樂bar進入play界面 littlebar: function () { //這裏有一個較難解決的問題,我會在下面單獨寫出,大神要是有思路請賜教,畢竟我才入門4天,不少都不懂 var pages=getCurrentPages(); var playingPage=pages[pages.length-2]; playingPage.setData({ angle:app.globalData.angle, }) wx.navigateBack(); }, //點擊清單跳轉到播放界面
//data-name="{{item.name}}" data-url="{{item.url}}"
//只有在列表渲染的view控件中設置這些屬性,該函數纔可獲得點擊後對應的屬性(可見上面的index.wxml)
playTheMusic:function(e){
console.log(e.currentTarget.dataset.name); //一個調試方法,調試器輸出點擊的歌曲名 this.setData({ toName: e.currentTarget.dataset.name }); var songUrl = e.currentTarget.dataset.url; var songName = e.currentTarget.dataset.name; app.globalData.songName = songName; var theUrl = "../playing/playing?songUrl=" + songUrl + "&songName=" + songName //url攜帶參數 wx.redirectTo({ //此種跳轉當前頁面數據會保存在頁面棧中,能夠回退,可問度娘
url: theUrl,
})
}, /** * 生命週期函數--監聽頁面加載 */ onLoad: function (options) { /** * 監聽音樂播放 */ wx.onBackgroundAudioPlay(function () { // callback console.log('onBackgroundAudioPlay') }) }, /** * 生命週期函數--監聽頁面顯示 */ onShow: functionthis.setData({ songName: app.globalData.songName , json: app.globalData.musicList, searchTop: true, }); } })
好了,這裏我想談一下在index.js中我提到的單獨講的內容。情景在於微信小程序界面之間的跳轉,使用wx.navigateTo()方法跳轉到另外一個頁面時能夠將當前頁面存入頁面棧,再經過wx.navigateBack()方法對頁面棧出棧操做,可回退到當前頁面,使用wx.redirectTo()直接跳轉,不存入頁面棧。在這個程序中,index->playing->index->playing重複屢次沒法直接實現。這個問題我想了好久仍是沒有完美的將其解決。
個人暫時邏輯是:點擊對應音樂使用wx.redirectTo()方法,不保存頁面棧,跳轉至playing頁;在playing頁點擊list圖標使用wx.navigateTo(),保存頁面棧,同時跳到index頁,這時候會調用onShow()函數,將保存的list信息渲染到index頁面上;點擊index底部的播放小框調用wx.navigateBack()方法回到playing頁(playing頁的數據較多,因此採用了這種邏輯)
哪位大哥有更好解決方法煩請賜教哈
接下來是playing頁:(推薦使用wx.getBackgroundAudioManager()接口對音樂行爲進行操做)
此頁面須要完成播放、暫停、上一曲、下一曲、回到index頁、動態顯示播放時間和總長度(部分實現)、快進(還未實現)
看界面:
而後上代碼了:
playing.wxml:圖標來源於百度圖片,簡單ps摳圖後傳至雲端使用;中間專輯圖旋轉感謝這篇博客,其中的旋轉快慢,每次角度可經過調節常量值實現,下面代碼有標註
<view class='Main'> <view class='songNameView'> <text></text> <text class='songName'>{{name}}</text> </view> <view class='imageView' style="background-image: url({{imageUrl}});" animation="{{animationData}}"> </view> <view class="backIndex" bindtap='backIndex' style='background-image:url({{homeImage}})'></view> <view class='line'> <view class='nowView'> <text class='now'>{{cur}}</text> </view> <view class='theLine'></view> <view class='allTimeView'> <text class='allTime'>{{duration}}</text> </view> </view> <view class="button"> <view class='back MusicIcon' style="background-image: url({{backUrl}});" bindtap='theBack'></view> <view class='center MusicIcon' style="background-image: url({{playOrStopUrl}});" bindtap='play'></view> <view class='next MusicIcon' style="background-image: url({{nextUrl}});" bindtap='theNext'></view> </view> </view>
一樣,playing.wxss可在個人Github中查看
下面是展現playing.js的時刻,各類邏輯在代碼中直接標註了
// pages/playing/playing.js const AV = require("../../libs/av-weapp-min"); var app = getApp() Page({ /** * 頁面的初始數據 */ data: { name:'測試', url:'', imageUrl:'http://lc-9qxpppvr.cn-n1.lcfile.com/9c1af72ea0506789a9b9.jpg', homeImage:"http://lc-9qxpppvr.cn-n1.lcfile.com/a9603240aab63a7950b0.png", animationData: {}, isPlay:false, //播放標誌 thePosition:0, //用來保存暫停時播放位置 angle:0, //用來不斷保存旋轉次數,用於解決界面屢次跳轉後旋轉失速問題 cur:'--:--', //當前時間 duration:'--:--' //總時長 }, //返回到清單頁
//此處在上面有說起,點擊list圖標到index頁面,並將跳轉前最後一個旋轉角度渲染給全局變量 backIndex: function () { wx.navigateTo({ url: '../index/index', }); app.globalData.angle=this.data.angle; }, /** * 生命週期函數--監聽頁面加載 */ onLoad: function (options) { //加載傳遞過來的參數
//options爲隨url傳遞過來的參數
this.setData({ name: options.songName, url: options.songUrl, imageUrl:"http://lc-9qxpppvr.cn-n1.lcfile.com/9c1af72ea0506789a9b9.jpg", backUrl:"http://lc-9qxpppvr.cn-n1.lcfile.com/2574f1888750f8eaea88.png", nextUrl:"http://lc-9qxpppvr.cn-n1.lcfile.com/299a6353324cb312b00e.png", playOrStopUrl:"http://lc-9qxpppvr.cn-n1.lcfile.com/56885a2b7b3e478dd0c6.png", isPlay:true }) //加載頁面時執行播放動做 wx.playBackgroundAudio({ dataUrl: this.data.url, }) }, //播放/暫停 play:function(){ const backgroundAudioManager = wx.getBackgroundAudioManager(); var theTime; var allTime; if(this.data.isPlay){ wx.pauseBackgroundAudio(); //推薦都使用這個API,我以前不知道這個API,致使在後來的開發中沒法實現上面列舉的所有功能 theTime = backgroundAudioManager.currentTime;//不甘心,因此在點擊播放或暫停時可對頁面進行時間的動態渲染,也算是完成了一點吧 allTime = backgroundAudioManager.duration; var theString1 = theTime.toFixed(0); var theInt1 = parseInt(theString1); var m1 = theInt1 / 60; var mString1 = m1.toFixed(0); //截取小數點後0位數字,結果爲String類型 var mInt1 = parseInt(mString1); //轉number var s1 = theInt1 % 60 / 100; var cur = mInt1 + s1; var theString = allTime.toFixed(0); var theInt = parseInt(theString); var m = theInt/60; var mString = m.toFixed(0); var mInt = parseInt(mString); var s = theInt%60/100; var all = mInt+s; this.setData({ playOrStopUrl:"http://lc-9qxpppvr.cn-n1.lcfile.com/22a26757fca8c46a2940.png",//替換爲暫停圖標 isPlay: false, //渲染一些須要的數據 thePosition: theTime, duration:all, cur: cur }); }else{ backgroundAudioManager.seek(this.data.thePosition); backgroundAudioManager.play(); this.setData({ playOrStopUrl: "http://lc-9qxpppvr.cn-n1.lcfile.com/56885a2b7b3e478dd0c6.png", isPlay:true }); } }, //下一首 theNext:function(){//對全局變量下的查找清單進行操做,若是當前歌曲爲最後一首,跳轉到第一首 var j; var musicList = getApp().globalData.musicList.concat(); for (var i = 0; i < musicList.length;i++){ if(musicList[i].name==this.data.name){ j=i; break; }else{ j=-1; } } if (musicList.length-1==j){ this.setData({ name: musicList[0].name, url: musicList[0].url }); }else{ this.setData({ name: musicList[j+1].name, url: musicList[j + 1].url }); } wx.playBackgroundAudio({ dataUrl: this.data.url, }) }, //上一首 theBack:function(){//若爲第一首,跳轉到最後一首 var j; var theLength=0; var musicList = getApp().globalData.musicList.concat(); theLength=musicList.length; for (var i = 0; i < musicList.length; i++) { if (musicList[i].name == this.data.name) { j = i; break; } else { j = 1; } } if (j==0) { this.setData({ name: musicList[theLength-1].name, url: musicList[theLength-1].url }); } else { this.setData({ name: musicList[j-1].name, url: musicList[j-1].url }); } wx.playBackgroundAudio({ dataUrl: this.data.url, }) },/** * 生命週期函數--監聽頁面顯示 */ onShow: function () {//這裏爲專輯圖旋轉函數,調用wx.createAnimation()接口,更多參數詳情可看官方API var animation = wx.createAnimation({ duration: 1000, timingFunction: 'ease', }) this.animation = animation // animation.scale(2, 2).rotate(45).step() this.setData({ animationData: animation.export() }) var n = 0; //連續動畫須要添加定時器,所傳參數每次+1就行 setInterval(function () { n=this.data.angle; if(this.data.isPlay){//暫停時中止旋轉,播放旋轉的邏輯 n = n + 1; }else{ n=n; } this.setData({ angle: n, }) this.animation.rotate(8 * n).step()//8爲每次轉8°,應該是,我的喜愛設置 this.setData({ animationData: this.animation.export() }) }.bind(this), 360)//360ms轉一個角度,我的喜愛設置 } })
OK,到這裏就能夠實現一個演示視頻中全部功能的簡易播放器了,是否是以爲很簡單。
相對於網上不少前輩的功能完善的音樂播放器來講,我這個真的是可望不可即,將來還有很長的路要走。可是東西出來了嘛, 仍是很開心的;固然,我仍是會不斷的去完善它的,這之中的playing頁面的專輯圖浮動旋轉效果是我最驚喜的了,以前只是想加個陰影,沒想到陰影能夠跟隨旋轉,賊帥。
感謝可愛的你看了這篇博客