基于Vue2实现歌曲播放和歌词滚动效果

 更新时间:2024年09月11日 11:06:05   作者:愉快的小跳蛙  
这篇文章主要介绍了如何基于Vue2实现歌曲播放和歌词滚动效果,文中通过代码示例和图文讲解的非常详细,对大家的学习或工作有一定的帮助,感兴趣的小伙伴可以自己动手试一下

需求:需要实现歌词滚动效果。

思路:通过js+css的transform属性完成。

难点:主要就是需要知道正在播放的歌词是那句,然后对正在播放的歌词进行变色和放大,最难的就是让高亮歌词随着歌曲播放滚动。

1.先看效果图

2.处理歌词格式(项目中有后端兄弟实现转换的可以省略)

// 处理歌词格式
parseLrc(musicLrc) {
  const lines = musicLrc?.split(`\n`);
  let lrcList = lines.map((line) => {
    let [time, words] = line?.split("]") ?? [null, null];
    return {
      time: this.parseTime(time.substring(1)) || null,
      words: words || null,
    };
  });
  let lrcListRes = lrcList.filter((v) => {
    return v.name !== null && v.words !== null;
  });
  // this.computingTime();
  return lrcListRes;
},
 
parseTime(t) {
  const part = t?.split(":");
  return Number(part[0] * 60) + Number(part[1]);
},

歌词格式一般都是数组对象,对象的key各位可以自己根据需要命名,主要是思路。思路最重要!思路最重要!思路最重要!

我这里的数据格式如下图

3.利用audio的timeupdate的事件来进行计算歌词是否高亮以及偏移量

// 添加audio事件
autoDown() {
  const that = this;
  var audio = document.getElementById("myAudio");
  audio.addEventListener("ended", function () {
    that.switchingBtn("down");
  });
  audio.addEventListener("timeupdate", function () {
    that.computingTime();
  });
},
 
// 计算播放时间对应的下标
computingTime() {
  let arr = this.selectedFiles[this.playIndex].lrcList || [];
  let currentTime = document.getElementById("myAudio")?.currentTime || 0;
  let index = arr.findIndex((e) => currentTime < e.time) - 1;
  this.currentIndex = index >= 0 ? index : arr.length - 1;
},
// 计算偏移量(保证高亮的歌词在中间)
computingOffset(index) {
  // 外部大盒子高度
  let musicLrcBoxHeight = this.$refs.musicLrc?.clientHeight;
  // 歌词总高度
  let musicLrcHeight = this.$refs.musicLrc_bady?.clientHeight;
  // 每个li高度
  let musicLrcLiHeight = this.$refs.musicLrc_item?.clientHeight || 22;
  // 歌词偏移高度
  let offsetHenght =
    index * musicLrcLiHeight + musicLrcLiHeight / 2 - musicLrcBoxHeight / 2;
  // 最大偏移高度
  let offsetMax = musicLrcHeight - musicLrcBoxHeight + 10;
 
  if (offsetHenght < 0) {
    offsetHenght = 0;
  }
  // if (offsetHenght > offsetMax) {
  //   offsetHenght = offsetMax;
  // }
 
  this.$refs.musicLrc_bady.style.transform = `translateY(-${offsetHenght}px)`;
},

注意:这里的computingTime和computingOffset两个事件是核心代码!!!

4.整个demo源码

<template>
  <div class="h100 dis_sb bs">
    <div class="music bg-fff bs">
      <div class="p10 bs mb10" style="height: 40px">
        <el-button type="text" size="small" @click="triggerFileInput">
          选择歌曲
        </el-button>
        <span class="f12 ml10">请先选择本地音乐!!!</span>
        <input
          ref="audioInput"
          style="display: none; height: 10px"
          type="file"
          @change="handleFileSelect"
          multiple
          accept="audio/*"
        />
      </div>
 
      <div class="bs p10" v-if="selectedFiles?.length > 0" style="height: 60px">
        <audio
          id="myAudio"
          class="audio"
          controls
          :src="fileUrl || selectedFiles[0]?.url"
          autoplay
        />
        <div class="tac bs" style="line-height: 20px">
          <el-button type="text" size="small" @click="switchingBtn('up')">
            上一曲
          </el-button>
          <el-button type="text" size="small" @click="togglePlay">
            {{ playing ? "暂停" : "播放" }}
          </el-button>
          <el-button type="text" size="small" @click="switchingBtn('down')">
            下一曲
          </el-button>
        </div>
      </div>
 
      <div v-if="selectedFiles?.length > 0" class="p10 music_body">
        <ul class="main bs mt20">
          <li
            :class="[index == playIndex ? 'main_item_action li' : 'li']"
            v-for="(file, index) in selectedFiles"
            :key="index"
          >
            <p @click="choose(file, index)">{{ file.name }}</p>
          </li>
        </ul>
      </div>
    </div>
 
    <div class="bs h100 p10 musicLrc" ref="musicLrc">
      <ul class="musicLrc_bady tac f14" ref="musicLrc_bady">
        <li
          v-for="(v, i) in selectedFiles[playIndex]?.lrcList"
          :key="i"
          ref="musicLrc_item"
        >
          <p :class="[i == currentIndex ? 'musicLrc_action ' : '']">
            {{ v.words }}
          </p>
        </li>
        <div
          v-if="selectedFiles[playIndex]?.lrcList?.length == 0"
          class="tac bs"
          style="margin: auto; padding-top: 30px; color: #ccc"
        >
          暂无歌词
        </div>
      </ul>
    </div>
  </div>
</template>
 
<script>
import musicList from "./musicList.js";
export default {
  data() {
    return {
      selectedFiles: [],
      lrcList: [],
      fileUrl: null,
      playing: false,
      isPlay: false,
      playIndex: 0,
      currentTime: 0,
      currentIndex: null,
    };
  },
 
  created() {
    musicList.forEach((e) => {
      if (e.lrc) {
        e.lrcList = this.parseLrc(e.lrc);
      } else {
        e.lrcList = [];
      }
    });
    this.selectedFiles = JSON.parse(JSON.stringify(musicList));
  },
  watch: {
    selectedFiles: {
      handler(newVal, oldVal) {
        if (newVal?.length > 4) {
          this.$nextTick(() => {
            this.fileUrl = newVal[0]?.url;
            this.playing = true;
            this.playIndex = 0;
          });
        }
      },
      deep: true,
    },
    $route: {
      handler(newVal, oldVal) {
        if (newVal?.path !== "/music") {
          this.$nextTick(() => {
            this.playing = false;
          });
        }
      },
      deep: true,
    },
    currentIndex: {
      handler(newVal, oldVal) {
        if (newVal > 0) {
          setTimeout(() => {
            this.computingOffset(newVal);
          }, 150);
        } else {
          this.computingOffset(0);
        }
      },
      deep: true,
    },
  },
 
  mounted() {
    this.$nextTick(() => {
      this.autoDown();
    });
  },
 
  methods: {
    // 添加本地音乐
    handleFileSelect(event) {
      const files = event.target.files;
      this.selectedFiles = JSON.parse(JSON.stringify(musicList));
      this.fileUrl = null;
      for (let i = 0; i < files?.length; i++) {
        const file = files[i];
        const reader = new FileReader();
        reader.onload = (e) => {
          const fileObj = {
            name: file.name,
            url: e.target.result,
            // lrc: null,
            lrcList: [],
          };
          this.selectedFiles.push(fileObj);
        };
        reader.readAsDataURL(file);
      }
    },
 
    triggerFileInput() {
      this.$refs.audioInput.click();
    },
    // 选择播放歌曲
    choose(v, i) {
      const that = this;
      that.playing = true;
      that.fileUrl = that.selectedFiles[i]?.url;
      that.playIndex = i;
    },
    // 判断播放状态
    togglePlay() {
      const that = this;
      var audio = document.getElementById("myAudio");
      if (audio.paused) {
        audio.play();
        that.playing = true;
      } else {
        audio.pause();
        that.playing = false;
      }
    },
    // 添加audio事件
    autoDown() {
      const that = this;
      var audio = document.getElementById("myAudio");
      audio.addEventListener("ended", function () {
        that.switchingBtn("down");
      });
      audio.addEventListener("timeupdate", function () {
        that.computingTime();
      });
    },
    // 播放歌曲
    playAudio() {
      var audio = document.getElementById("myAudio");
      audio.play();
      this.currentIndex = 0;
    },
    // 切换歌曲
    switchingBtn(v) {
      const that = this;
      that.playing = true;
      const length = this.selectedFiles?.length || 0;
 
      if (v === "down") {
        this.playIndex = (this.playIndex + 1) % length;
      } else {
        this.playIndex = (this.playIndex - 1 + length) % length;
      }
      that.fileUrl = that.selectedFiles[that.playIndex]?.url;
 
      setTimeout(() => {
        that.playAudio();
      }, 150);
    },
    // 处理歌词格式
    parseLrc(musicLrc) {
      const lines = musicLrc?.split(`\n`);
      let lrcList = lines.map((line) => {
        let [time, words] = line?.split("]") ?? [null, null];
        return {
          time: this.parseTime(time.substring(1)) || null,
          words: words || null,
        };
      });
      let lrcListRes = lrcList.filter((v) => {
        return v.name !== null && v.words !== null;
      });
      // this.computingTime();
      return lrcListRes;
    },
 
    parseTime(t) {
      const part = t?.split(":");
      return Number(part[0] * 60) + Number(part[1]);
    },
    // 计算播放时间对应的下标
    computingTime() {
      let arr = this.selectedFiles[this.playIndex].lrcList || [];
      let currentTime = document.getElementById("myAudio")?.currentTime || 0;
      let index = arr.findIndex((e) => currentTime < e.time) - 1;
      this.currentIndex = index >= 0 ? index : arr.length - 1;
    },
    // 计算偏移量(保证高亮的歌词在中间)
    computingOffset(index) {
      // 外部大盒子高度
      let musicLrcBoxHeight = this.$refs.musicLrc?.clientHeight;
      // 歌词总高度
      let musicLrcHeight = this.$refs.musicLrc_bady?.clientHeight;
      // 每个li高度
      let musicLrcLiHeight = this.$refs.musicLrc_item?.clientHeight || 22;
      // 歌词偏移高度
      let offsetHenght =
        index * musicLrcLiHeight + musicLrcLiHeight / 2 - musicLrcBoxHeight / 2;
      // 最大偏移高度
      let offsetMax = musicLrcHeight - musicLrcBoxHeight + 10;
 
      if (offsetHenght < 0) {
        offsetHenght = 0;
      }
      // if (offsetHenght > offsetMax) {
      //   offsetHenght = offsetMax;
      // }
 
      this.$refs.musicLrc_bady.style.transform = `translateY(-${offsetHenght}px)`;
    },
  },
};
</script>
 
<style scoped>
.music {
  width: calc(50% - 5px);
  height: 100%;
  border-radius: 10px;
}
 
#myAudio {
  width: 100%;
  height: 30px;
}
 
.music_body {
  height: calc(100% - 126px);
  overflow-x: hidden;
  overflow-y: auto;
}
 
.main {
  .li {
    border: 1px solid #eee;
    border-radius: 5px;
    padding: 0 10px;
    margin-bottom: 10px;
    box-sizing: border-box;
    line-height: 30px;
    font-size: 12px;
    cursor: grab;
    color: #666;
  }
  .main_item_action {
    border: 1px solid #409eff;
    color: #409eff;
  }
}
 
.musicLrc {
  width: calc(50% - 5px);
  height: 100%;
  background-color: rgb(0, 0, 0);
  border-radius: 10px;
  color: #ccc;
  line-height: 22px;
  /* overflow: hidden; */
  transform: translateY();
  overflow-x: hidden;
  overflow-y: auto;
}
 
.musicLrc_bady {
  li {
    transition: 0.8s;
  }
}
 
.musicLrc_action {
  transform: scale(1.5);
  color: #409eff;
}
</style>

以上就是基于Vue2实现歌曲播放和歌词滚动效果的详细内容,更多关于Vue2歌曲播放和歌词滚动的资料请关注脚本之家其它相关文章!

相关文章

  • vue Element-ui input 远程搜索与修改建议显示模版的示例代码

    vue Element-ui input 远程搜索与修改建议显示模版的示例代码

    本文分为html,js和css代码给大家详细介绍了vue Element-ui input 远程搜索与修改建议显示模版功能,感兴趣的朋友一起看看吧
    2017-10-10
  • vue项目实现会议预约功能(包含某天的某个时间段和某月的某几天)

    vue项目实现会议预约功能(包含某天的某个时间段和某月的某几天)

    这篇文章主要介绍了vue项目实现会议预约功能(包含某天的某个时间段和某月的某几天),本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-02-02
  • vue 修改vant自带的样式过程

    vue 修改vant自带的样式过程

    这篇文章主要介绍了vue 修改vant自带的样式过程,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-04-04
  • vue如何使用html2canvas和JsPDF导出pdf组件

    vue如何使用html2canvas和JsPDF导出pdf组件

    这篇文章主要介绍了vue如何使用html2canvas和JsPDF导出pdf组件问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-09-09
  • vue中如何动态绑定图片,vue中通过data返回图片路径的方法

    vue中如何动态绑定图片,vue中通过data返回图片路径的方法

    下面小编就为大家分享一篇vue中如何动态绑定图片,vue中通过data返回图片路径的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-02-02
  • 前端vuex中dispatch的使用方法总结

    前端vuex中dispatch的使用方法总结

    这篇文章主要给大家介绍了关于前端vuex中dispatch使用方法的相关资料,vuex的dispatch方法用于触发一个action,以便更新state,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-04-04
  • el-input设置后缀显示单位并阻止滚轮微调的解决方法

    el-input设置后缀显示单位并阻止滚轮微调的解决方法

    在Element UI或Element Plus中,使用el-input组件时,可以通过suffix插槽添加单位显示,如果设置了type为'number',滚轮滚动可能会导致数值微调,解决方法是阻止滚轮事件的默认行为,并用CSS隐藏掉输入框的上下箭头,确保数值输入的准确性,这样既美观又提升了用户体验
    2024-09-09
  • 如何一步步基于element-ui封装查询组件

    如何一步步基于element-ui封装查询组件

    这篇文章主要给大家介绍了关于如何一步步基于element-ui封装查询组件的相关资料,本文通过示例代码介绍的非常详细,对大家学习或者使用vue.js具有一定的参考学习价值,需要的朋友可以参考下
    2021-10-10
  • vue项目中使用TDesign的方法

    vue项目中使用TDesign的方法

    tdesign-vue是TDesign 适配桌面端的组件库,适合在 vue 2 技术栈项目中使用,这篇文章主要介绍了vue项目中使用TDesign ,需要的朋友可以参考下
    2023-04-04
  • 详解el-table表头文字换行的三种方式

    详解el-table表头文字换行的三种方式

    本文主要介绍了el-table表头文字换行的三种方式,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11

最新评论