template 代码
<view class="player">
<view class="progress">
<view class="current-time">{{ slider_time }}</view>
<view class="slider"><slider :value="slider_index" :max="song_time" activeColor="#0190ff" backgroundColor="#dddddd" block-color="#0190ff" block-size="12" /></view>
<view class="max-time">{{ slider_maxTime }}</view>
</view>
<view class="control">
<view class="col">
<view class="previous-btn" @click="changePrevious()"><view class="icon-previous"></view></view>
</view>
<view class="col">
<view class="outer" v-if="song_loading"></view>
<view class="inner" v-if="song_loading"></view>
<view class="play-btn" v-else @click="changePlayState()"><view :class="[song_playState ? 'icon-pause' : 'icon-play']"></view></view>
</view>
<view class="col">
<view class="next-btn" @click="changeNext()"><view class="icon-next"></view></view>
</view>
</view>
</view>
script 代码
var innerAudioContext;
export default {
data() {
return {
musicList: [
{ song: '扬州三月', url: 'https://static.cdn.ejcms.com/1.mp3' },
{ song: 'The Right Path', url: 'https://static.cdn.ejcms.com/2.mp3' },
{ song: '故乡的原风景', url: 'https://static.cdn.ejcms.com/3.mp3' },
{ song: '夏夜', url: 'https://static.cdn.ejcms.com/4.mp3' },
{ song: 'Going Home', url: 'https://static.cdn.ejcms.com/5.mp3' }
],
song_index: 0,
song_url: '',
song_loading: false,
song_playState: false,
song_time: 0,
slider_index: 0,
slider_time: '',
slider_maxTime: ''
};
},
onLoad() {
this.playAudio(this.musicList[0]);
},
methods: {
initPlay() {
let _this = this;
// 销毁当前实例
if (innerAudioContext) {
innerAudioContext.destroy();
}
// 创建音频
innerAudioContext = uni.createInnerAudioContext();
// 音频的数据链接,用于直接播放。
innerAudioContext.src = _this.song_url;
// 开始播放的位置(单位:s),默认 0
innerAudioContext.startTime = 0;
// 是否自动开始播放,默认 false
innerAudioContext.autoplay = true;
// 音频进入可以播放状态,但不保证后面可以流畅播放
innerAudioContext.onCanplay(res => {
_this.song_loading = false;
});
// 音频播放事件
innerAudioContext.onPlay(() => {
_this.song_playState = true;
});
// 音频暂停事件
innerAudioContext.onPause(() => {
_this.song_playState = false;
});
// 音频停止事件
innerAudioContext.onStop(res => {
innerAudioContext.destroy();
// console.log(res);
});
// 音频自然播放结束事件
innerAudioContext.onEnded(res => {
innerAudioContext.destroy();
_this.song_playState = false;
_this.changeNext();
});
// 音频播放进度更新事件
innerAudioContext.onTimeUpdate(() => {
let { currentTime, duration } = innerAudioContext;
_this.slider_index = currentTime;
_this.song_time = duration;
_this.slider_time = _this.formatTime(currentTime);
_this.slider_maxTime = _this.formatTime(duration);
});
// 音频播放错误事件
innerAudioContext.onError(res => {
_this.song_playState = false;
console.log('onError', res);
});
// 音频加载中事件,当音频因为数据不足,需要停下来加载时会触发
innerAudioContext.onWaiting(res => {
_this.song_loading = true;
});
},
// 上一曲
changePrevious() {
if (this.song_index > 0) {
this.song_index--;
} else {
this.song_index = 0;
}
this.playAudio(this.musicList[this.song_index]);
},
// 播放状态
changePlayState() {
if (this.song_playState) {
innerAudioContext.pause();
} else {
innerAudioContext.play();
}
},
// 下一曲
changeNext() {
if (this.musicList.length > this.song_index + 1) {
this.song_index++;
} else {
this.song_index = 0;
}
this.playAudio(this.musicList[this.song_index]);
},
playAudio(data) {
this.slider_index = 0;
this.song_url = data.url;
this.initPlay();
},
formatTime(num) {
num = num.toFixed(0);
let second = num % 60;
if (second < 10) {
second = '0' + second;
}
let min = Math.floor(num / 60);
if (min < 10) {
min = '0' + min;
}
return min + ':' + second;
}
}
};
style 代码
uni-app,
page {
background-image: linear-gradient(35deg, #b6e8fc, #ecccfb, #b6e8fc);
}
.player {
margin-top: calc(100vh - 150px);
width: 100%;
height: 150px;
background-image: linear-gradient(150deg, #e7f5ff, #ffffff, #f1eeff);
border-top-right-radius: 30px;
border-top-left-radius: 30px;
position: relative;
.progress {
height: 50px;
display: flex;
justify-content: space-between;
align-items: center;
.current-time {
width: 60px;
text-align: right;
color: #898989;
}
.slider {
width: calc(100% - 120px);
}
.max-time {
width: 60px;
text-align: left;
color: #898989;
}
}
.control {
margin: 0 auto;
width: 80%;
height: 80px;
.col {
float: left;
position: relative;
display: block;
box-sizing: border-box;
width: 33.3333336%;
height: 100px;
line-height: 100px;
display: flex;
justify-content: center;
.outer {
border: 5px solid rgba(0, 148, 255, 0.9);
opacity: 0.9;
width: 68px;
height: 68px;
border-top-color: transparent;
border-bottom-color: transparent;
border-radius: 50%;
animation: spinRight 0.5s linear infinite normal;
animation-delay: 0;
margin: 0 auto;
}
.inner {
border: 5px solid rgba(0, 148, 255, 0.9);
opacity: 0.9;
width: 38px;
height: 38px;
border-left-color: transparent;
border-bottom-color: transparent;
border-radius: 50%;
animation: spinLeft 0.5s linear infinite normal;
animation-delay: 0;
margin: 0 auto;
top: -63px;
display: block;
position: relative;
}
.play-btn,
.previous-btn,
.next-btn {
width: 68px;
height: 68px;
line-height: 68px;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
}
.play-btn {
background-color: #0190ff;
}
}
}
}
.icon-play {
width: 26px;
height: 26px;
background-image: url('');
}
.icon-pause {
width: 26px;
height: 26px;
background-image: url('');
}
.icon-previous {
width: 22px;
height: 22px;
background-image: url('');
}
.icon-next {
width: 22px;
height: 22px;
background-image: url('');
}
Tenglin (作者)
没有,最后放弃了
2024-12-02 10:03