Web前端—(原生JS)歌词滚动效果

歌词滚动效果实现

  • 歌词滚动效果
    • HTML部分
    • CSS部分
    • JS部分
      • 解析歌词字符串,得到歌词的对象数组
      • 计算在当前情况下,播放器播放到第几秒的情况
      • 创建歌词元素
      • 设置ul元素的偏移量
      • 最后对时间变化的事件进行监听
      • 完整JS代码

歌词滚动效果

  • 实现效果如图所示:

在这里插入图片描述

首先准备一个歌曲MP3文件,以及歌词,将歌词放入一个data.js文件中。
在这里插入图片描述

  • 歌词内容如下所示:
    在这里插入图片描述

HTML部分

  • 歌词采用ul-li的结构进行展示
  • 导入相应的css和js文件
  • 代码如下:
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><link rel="stylesheet" href="./css/index.css">
</head>
<body><audio src="./assets/薛之谦 - 狐狸.mp3" controls></audio><div class="container"><ul>           </ul></div><script src="./js/data.js"></script><script src="./js/index.js"></script>
</body>
</html>

CSS部分

  • 由于CSS属性变化本身没有动画效果,因此这里采用transition属性实现其动画效果
  • transition属性只针对于数值嘞的属性有效
  • 代码如下:(大家可以根据自己的喜好设置样式)
*{margin: 0;padding: 0;
}
body{background: #000;color: #666;text-align: center;
}
audio{width: 450px;margin: 30px 0;
}
.container{height: 420px;overflow: hidden;
}
.container ul{transition: 0.3s;}
.container li{height: 30px;line-height: 30px;transition: 0.3s;list-style: none;}
.container li.active{color: #fff;transform: scale(1.2);
}

JS部分

解析歌词字符串,得到歌词的对象数组

// 解析时间字符串
function parseTime(timeStr){let parts = timeStr.split(':');return +parts[0]*60 + +parts[1];
}
/*** 解析歌词字符串,得到歌词的对象数组* 每个歌词对象 * {time: 开始时间, words:歌词内容}*/
function parseLrc(){var result = [];var lines = lrc.split('\n'); // 字符串数组for(let i = 0; i < lines.length; i++){// 遍历lineslet str = lines[i]; //每一句歌词字符串let parts = str.split(']'); //分割let timeStr = parts[0].substring(1); //截取时间var obj ={time:parseTime(timeStr),words: parts[1]}result.push(obj);}return result;
}var lrcData = parseLrc();

计算在当前情况下,播放器播放到第几秒的情况

/*** 计算在当前情况下,播放器播放到第几秒的情况* lrcData数组中应该高亮显示的歌词下标*/
function findIndex(){// 获取播放器的当前时间let curTime = doms.audio.currentTime;for(let i = 0; i<lrcData.length; i++){if(curTime<lrcData[i].time){return i-1;}}// 如果没有任何一句歌词显示,则返回-1// 如果找遍了都没有找到,说明播放到最后一句return lrcData.length-1;
}

创建歌词元素

// 创建歌词元素
function createLrcElement(){for(let i=0; i<lrcData.length; i++){let li = document.createElement('li');li.textContent = lrcData[i].words;doms.ul.appendChild(li);}
}

设置ul元素的偏移量

// 容器高度
var containerHeight = doms.container.clientHeight;
// li高度
var liHeight = doms.ul.children[0].clientHeight;
// 最大高度
var maxOffset = doms.ul.clientHeight - containerHeight;// 设置ul元素的偏移量
function setOffset(){let index = findIndex();var offset = liHeight * index + liHeight/2 - containerHeight/2;if(offset < 0){offset = 0;}if(offset > maxOffset){offset = maxOffset;}doms.ul.style.transform = `translateY(-${offset}px)`;// 去掉之前的active样式let li1 = doms.ul.querySelector('.active');if(li1){li1.classList.remove('active');}let li2 = doms.ul.children[index];if(li2){li2.classList.add('active');}}

最后对时间变化的事件进行监听

doms.audio.addEventListener('timeupdate', setOffset);

如果大家没有MP3文件或者歌词的话,我会上传我的资源,可以自行下载。

完整JS代码

var doms = {audio:document.querySelector('audio'),ul:document.querySelector('.container ul'),container:document.querySelector('.container')
}// 解析时间字符串
function parseTime(timeStr){let parts = timeStr.split(':');return +parts[0]*60 + +parts[1];
}/*** 解析歌词字符串,得到歌词的对象数组* 每个歌词对象 * {time: 开始时间, words:歌词内容}*/
function parseLrc(){var result = [];var lines = lrc.split('\n'); // 字符串数组for(let i = 0; i < lines.length; i++){// 遍历lineslet str = lines[i]; //每一句歌词字符串let parts = str.split(']'); //分割let timeStr = parts[0].substring(1); //截取时间var obj ={time:parseTime(timeStr),words: parts[1]}result.push(obj);}return result;
}var lrcData = parseLrc();/*** 计算在当前情况下,播放器播放到第几秒的情况* lrcData数组中应该高亮显示的歌词下标*/
function findIndex(){// 获取播放器的当前时间let curTime = doms.audio.currentTime;for(let i = 0; i<lrcData.length; i++){if(curTime<lrcData[i].time){return i-1;}}// 如果没有任何一句歌词显示,则返回-1// 如果找遍了都没有找到,说明播放到最后一句return lrcData.length-1;
}// 创建歌词元素
function createLrcElement(){for(let i=0; i<lrcData.length; i++){let li = document.createElement('li');li.textContent = lrcData[i].words;doms.ul.appendChild(li);}
}
createLrcElement()// 容器高度
var containerHeight = doms.container.clientHeight;
// li高度
var liHeight = doms.ul.children[0].clientHeight;
// 最大高度
var maxOffset = doms.ul.clientHeight - containerHeight;// 设置ul元素的偏移量
function setOffset(){let index = findIndex();var offset = liHeight * index + liHeight/2 - containerHeight/2;if(offset < 0){offset = 0;}if(offset > maxOffset){offset = maxOffset;}doms.ul.style.transform = `translateY(-${offset}px)`;// 去掉之前的active样式let li1 = doms.ul.querySelector('.active');if(li1){li1.classList.remove('active');}let li2 = doms.ul.children[index];if(li2){li2.classList.add('active');}}// 事件监听
doms.audio.addEventListener('timeupdate', setOffset);```

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/781313.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

可重复不限数量结构数列的演化

有一个6*6的平面&#xff0c;这个平面的行和列可以自由的变换&#xff0c;在这个平面上有一个4点结构数列 按照8&#xff0c;13&#xff0c;5&#xff0c;8的顺序排列。让这个数列按照4-5-4的方式演化 这个数列很快收敛,收敛顺序为13&#xff0c;8&#xff0c;8&#xff0c;5 8…

前端小白如何理解mvc mvp mvvm

架构、框架、设计模式是都是啥&#xff1f; 架构&#xff1a;抽象出来不同组织或者对象亦或是简单组件&#xff0c;根据需求和各个单元的功能&#xff0c;进行组合排列。 从而完成系统的运行或者是实现目标。 框架&#xff1a;使用什么样的规则&#xff0c;什么样的开发语言&…

接口自动化测试问题汇总

本篇文章分享几个接口自动化用例编写过程遇到的问题总结&#xff0c;希望能对初次探索接口自动化测试的小伙伴们解决问题上提供一小部分思路。 sql语句内容出现错误 空格&#xff1a;由于有些字段判断是变量&#xff0c;需要将sql拼接起来&#xff0c;但是在拼接字符串时没有…

npm mongoose包下载冲突解决之道

我在新电脑下载完项目代码后,运行 npm install --registryhttps://registry.npm.taobao.org 1运行就报错&#xff1a; npm ERR! code ERESOLVE npm ERR! ERESOLVE unable to resolve dependency tree npm ERR! npm ERR! While resolving: lowcode-form-backend1.0.0 npm …

Python | 单变量时间序列分析与预测

时间序列数据是机器学习中最具挑战性的任务之一&#xff0c;也是与数据相关的现实问题之一&#xff0c;因为数据实体不仅取决于物理因素&#xff0c;而且主要取决于它们发生的时间顺序。我们可以基于一个单变量特征和两个双变量或多变量特征来预测时间序列中的目标值&#xff0…

合集:JS异步的六个解决方案详解。

Hello&#xff0c;各位老铁&#xff0c;最近发表了js异步的解决方案&#xff0c;是分开发的&#xff0c;这次我把他汇总起来&#xff0c;方便大家收藏、查看&#xff0c;欢迎点赞评论私信交流。 01.详解&#xff1a;JS异步解决方案之回调函数&#xff0c;及其弊端 02.详解&…

全套医院手术麻醉系统源码 人工智能麻醉系统源码 医疗管理系统源码

全套医院手术麻醉系统源码 人工智能麻醉系统源码 医疗管理系统源码 手术麻醉临床信息系统有着完善的临床业务功能&#xff0c;能够涵盖整个围术期的工作&#xff0c;能够采集、汇总、存储、处理、展现所有的临床诊疗资料。通过该系统的实施&#xff0c;能够规范麻醉科的工作流…

【LV16 day2 平台总线驱动开发---名称匹配】

一、总线、设备、驱动 硬编码式的驱动开发带来的问题&#xff1a; 垃圾代码太多结构不清晰一些统一设备功能难以支持开发效率低下 1.1 初期解决思路&#xff1a;设备和驱动分离 ​ struct device来表示一个具体设备&#xff0c;主要提供具体设备相关的资源&#xff08;如寄…

[深度学习]yolov8+pyqt5搭建精美界面GUI设计源码实现二

【简单介绍】 基于目标检测算法YOLOv8和灵活的PyQt5界面开发框架&#xff0c;我们精心打造了一款集直观性、易用性和功能性于一体的目标检测GUI界面。通过深度整合YOLOv8在目标识别上的卓越能力与PyQt5的精致界面设计&#xff0c;我们成功研发出一款既高效又稳定的软件GUI。 …

java全排列(力扣Leetcode46)

全排列 力扣原题链接 问题描述 给定一个不含重复数字的数组 nums&#xff0c;返回其所有可能的全排列。你可以按任意顺序返回答案。 示例 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3] 输出&#xff1a;[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]] 示例 2…

OpenHarmony error: signature verification failed due to not trusted app source

问题&#xff1a;error: signature verification failed due to not trusted app source 今天在做OpenHarmony App开发&#xff0c;之前一直用的设备A在测试开效果&#xff0c;今天换成了设备B&#xff0c;通过DevEco Studio安装应用程序的时候&#xff0c;就出现错误&#xf…

gpt 3d三角形 重心坐标填充 沿x轴炫赵师傅

go import pygame from pygame.locals import * import sys import math# 初始化Pygame pygame.init()# 设置窗口大小 width, height 800, 600 screen pygame.display.set_mode((width, height)) pygame.display.set_caption(3D Triangle Fill with Barycentric Coordinates)…

Windows 远程访问 Ubuntu Desktop - 虚拟网络控制台 (Virtual Network Console,VNC)

Windows 远程访问 Ubuntu Desktop - 虚拟网络控制台 [Virtual Network Console&#xff0c;VNC] References 1. Desktop Sharing 2. Desktop Sharing Preferences 勾选 允许其他人查看您的桌面 勾选 要求远程用户输入此密码 取消勾选 必须为对本机器的每次访问进行确定 3. 虚拟…

vue3+vite 模板vue3-element-admin框架如何关闭当前页面跳转 tabs

使用模版: 有来开源组织 / vue3-element-admin 需要关闭的.vue 页面增加以下方法 //setup 里import {LocationQuery, useRoute, useRouter} from "vue-router"; const router useRouter(); function close() {console.log(|--router.currentRoute.value, router.cur…

椋鸟数据结构笔记#3:链表

萌新的学习笔记&#xff0c;写错了恳请斧正。 目录 链表的定义 链表的分类 方向&#xff08;单向还是双向&#xff09; 头节点&#xff08;哨兵节点&#xff09;的有无 循环或不循环 8种分类 不带头单向不循环链表的实现 带头单向循环链表的实现 链表与顺序表的差异 链…

FPGA高端项目:解码索尼IMX390 MIPI相机转HDMI输出,提供FPGA开发板+2套工程源码+技术支持

目录 1、前言2、相关方案推荐本博主所有FPGA工程项目-->汇总目录我这里已有的 MIPI 编解码方案 3、本 MIPI CSI-RX IP 介绍4、个人 FPGA高端图像处理开发板简介5、详细设计方案设计原理框图IMX390 及其配置MIPI CSI RX图像 ISP 处理图像缓存HDMI输出工程源码架构 6、工程源码…

Mistral 7B v0.2 基础模型开源,大模型微调实践来了

Mistral AI在3月24日突然发布并开源了 Mistral 7B v0.2模型&#xff0c;有如下几个特点&#xff1a; 和上一代Mistral v0.1版本相比&#xff0c;上下文窗口长度从8k提升到32k&#xff0c;上下文窗口&#xff08;context window&#xff09;是指语言模型在进行预测或生成文本时&…

暴雨服务器X7740赋能大模型训练

数字经济浪潮愈演愈烈,大模型训练对服务器的要求也越来越高。在此背景下,暴雨信息发布专门为大规模模型训练而设计的全新旗舰GPU服务器—X7740,以卓越的计算性能、高速网络通信能力以及创新的能效表现,有效赋能大模型训练。 X7740 搭载了暴雨信息最新一代的英特尔至强可扩展处理…

银河麒麟安装回退至GCC5.4.0 添加镜像源地址并设置其优先级。问题已解决

问题&#xff1a; 因为软件依赖低版本的gcc&#xff0c;而新版银河麒麟gcc默认时高版本&#xff0c;考虑到软件适配应该考虑最低版本的麒麟系统&#xff0c;需要将gcc降级。 过程中遇到问题&#xff0c;无法直接找到gcc5.4.0 这是没有软件源&#xff0c;需要添加其他软件源&…

mysqldump备份数据库提示ERROR 1064 (42000): You have an error in your SQL syntax

在dos下备份数据库的时候提示上面的错误信息 1 错误详情 ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near mysql-v at line 1错误示例图 2 解决办法 通过查资料…