当前位置:   article > 正文

基于vue2.0实现音乐/视频播放进度条组件的思路及具体实现方法+代码解释_vue实现播放音乐左边图标中间进度条右边时间

vue实现播放音乐左边图标中间进度条右边时间

基于vue2.0实现音乐/视频播放进度条组件的方法及代码解释

需求分析:

①:进度条随着歌曲的播放延长,歌曲播放完时长度等于黑色总进度条长度;时间实时更新。

②:当滑动按钮时,实时更新播放时间,橙色进度条长度也会随着按钮的滑动而改变,当滑动结束时,橙色区域停留在滑动结束的位置,歌曲从当前进度开始播放。

③:点击进度条,橙色进度条长度变为点击处至起点的长度,并从当前点开始播放歌曲。

大概思路:

①:左边的时间可以通过audio播放时派发的timeupdate事件获取,右边的时间为接口获取的当前歌曲的总时间。

②:进度条子组件的长度通过父组件传入一个percent值计算,percent值为播放进度与总进度的比值。

③:进度条的滑动及点击结束后,需要向父组件传递一个percent值,使用this.$emit()像父组件派发事件,父组件中设置事件响应函数,接收percent参数值,用于改变audio中当前播放的音乐进度。

详细实现,关键代码已经注释:

先上组件源码:

  1. <template>
  2. <div class="progress-bar" ref="progressBar" @click="progressClick">
  3. <div class="bar-inner">
  4. <div class="progress" ref="progress"></div>
  5. <div class="progress-btn-wrapper"ref="progressBtn"
  6. @touchstart.prevent = "progressTouchStart"
  7. @touchmove.prevent = "progressTouchMove"
  8. @touchend = "progressTouchEnd"
  9. >
  10. <div class="progress-btn"></div>
  11. </div>
  12. </div>
  13. </div>
  14. </template>
  15. <script type="text/ecmascript-6">
  16. // 进度条按钮宽度,由于style中没有设置width,因此只能用clientWidth获取
  17. export default {
  18. data() {
  19. return {
  20. btnWidth: {
  21. type: Number,
  22. default: 0
  23. },
  24. touchInfo: {
  25. initiated: false
  26. }
  27. }
  28. },
  29. props: {
  30. percent: {
  31. type: Number,
  32. default: 0
  33. }
  34. },
  35. mounted() {
  36. this.btnWidth = document.getElementsByClassName('progress-btn')[0].clientWidth
  37. },
  38. methods: {
  39. // 点击按钮
  40. progressTouchStart(e) {
  41. // 记录touch事件已经初始化
  42. this.touchInfo.initiated = true
  43. // 点击位置
  44. this.touchInfo.startX = e.touches[0].pageX
  45. // 点击时进度条长度
  46. this.touchInfo.left = this.$refs.progress.clientWidth
  47. },
  48. // 开始移动
  49. progressTouchMove(e) {
  50. if (!this.touchInfo.initiated) {
  51. return
  52. }
  53. // 计算移动距离
  54. const moveX = e.touches[0].pageX - this.touchInfo.startX
  55. // 设置偏移值
  56. const offsetWidth = Math.min(Math.max(0, this.touchInfo.left + moveX),
  57. this.$refs.progressBar.clientWidth - this.btnWidth)
  58. this._setOffset(offsetWidth)
  59. },
  60. // 移动结束
  61. progressTouchEnd(e) {
  62. this.touchInfo.initiated = false
  63. // 向父组件派发事件,传递当前百分比值
  64. this._triggerPercent()
  65. },
  66. // 进度条点击事件
  67. progressClick(e) {
  68. console.log('clikc')
  69. // 设置进度条及按钮偏移
  70. this._setOffset(e.offsetX)
  71. // 通知父组件播放进度变化
  72. this._triggerPercent()
  73. },
  74. _triggerPercent() {
  75. const barWidth = this.$refs.progressBar.clientWidth - this.btnWidth
  76. const percent = Math.min(1, this.$refs.progress.clientWidth / barWidth)
  77. this.$emit('percentChange', percent)
  78. },
  79. // 设置偏移
  80. _setOffset(offsetWidth) {
  81. // 设置进度长度随着百分比变化
  82. this.$refs.progress.style.width = `${offsetWidth}px`
  83. // 设置按钮随着百分比偏移
  84. this.$refs.progressBtn.style.transform = `translate3d(${offsetWidth}px, 0, 0)`
  85. }
  86. },
  87. watch: {
  88. // 监听歌曲播放百分比变化
  89. percent(newPercent, oldPercent) {
  90. if (newPercent > 0 && !this.touchInfo.initiated) {
  91. // 进度条总长度
  92. const barWidth = this.$refs.progressBar.clientWidth - this.btnWidth
  93. const offsetWidth = barWidth * newPercent
  94. // 设置进度条及按钮偏移
  95. this._setOffset(offsetWidth)
  96. }
  97. }
  98. }
  99. }
  100. </script>
  101. <style lang="stylus" rel="stylesheet/stylus">
  102. @import "~common/stylus/variable.styl"
  103. .progress-bar
  104. height 0.5rem
  105. .bar-inner
  106. position relative
  107. top 0.2rem
  108. height 0.08rem
  109. background rgba(0, 0, 0, 0.3)
  110. .progress
  111. position absolute
  112. height 100%
  113. background $color-theme
  114. .progress-btn-wrapper
  115. position absolute
  116. left -0.25rem
  117. top -0.25rem
  118. width 0.5rem
  119. height 0.5rem
  120. .progress-btn
  121. position relative
  122. top 0.12rem
  123. left 0.12rem
  124. box-sizing border-box
  125. width 0.32rem
  126. height 0.32rem
  127. border 0.06rem solid $color-text
  128. border-radius 50%
  129. background $color-theme
  130. </style>

此为progerss-bar.vue组件源码,组件所需要父组件传入的值只有一个“percent”,为父组件中audio当前播放时间与总时间的比值,用于计算此组件中橙色进度条的长度。

组件的使用:

首先导入并注册组件(在此不做解释),随后使用此组件,dom:

  1. <div class="progress-wrapper">
  2. <span class="time time-l">{{formatTime(currentTime)}}</span>
  3. <div class="progress-bar-wrapper">
  4. <progress-bar :percent="percent" @percentChange="setProgress"></progress-bar>
  5. </div>
  6. <span class="time time-r">{{formatTime(currentSong.duration)}}</span>
  7. </div>
解释:两个span为左右两个时间值,progress-bar为调用的组件,需要传入percent值,用于子组件设置进度条长度

percent值来自于audio的currenTime与歌曲总长度的比值:

  1. // 计算百分比
  2. percent() {
  3. return Math.min(1, this.currentTime / this.currentSong.duration)
  4. }

@percentChange为子组件中派发过来的事件,详细请看子组件中源码及注释“_triggerPercent()”部分,此事件调用的方法用于接收子组件传过来的拖动按钮、点击进度条改变歌曲播放进度后的播放百分比,用于改变父组件中audio标签的currentTime,进而将歌曲播放进度设置为当前时间。

以下为父组件中,接收到子组件派发过来的事件后调用的函数。

  1. // 设置进度
  2. setProgress(percent) {
  3. // 根据子组件传过来的百分比设置播放进度
  4. this.$refs.audio.currentTime = this.currentSong.duration * percent
  5. // 拖动后设置歌曲播放
  6. if (!this.playing) {
  7. this.togglePlaying()
  8. }
  9. },

样式(本人使用stylus):

  1. .progress-wrapper
  2. display flex
  3. .time
  4. font-size 0.24rem
  5. &.time-l
  6. position absolute
  7. bottom 1.62rem
  8. left 1rem
  9. &.time-r
  10. position absolute
  11. bottom 1.62rem
  12. right 1rem
  13. .progress-bar-wrapper
  14. position absolute
  15. bottom 1.5rem
  16. left 1.7rem
  17. width 4.2rem

至此,进度条组件的实现及使用方法均介绍完毕。



声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Cpp五条/article/detail/130267
推荐阅读
相关标签
  

闽ICP备14008679号