07-歌词滚动效果

news2024/11/19 23:20:09

现在学习的代码工作中不一定会需要,如果有,也已经做成了产品和库,

前端重点是创造,面试官考验你的能力是会提出最刁钻的问题给你的。

项目效果

 原生JS效率是最高的,框架只是提高代码的可读性

favico图标添加

两种方法

1、放到站点的根目录,但涉及网络和部署相关,不推荐

2、link引入

<link rel="shortcut icon" href="./assets//favicon.ico" type="image/x-icon">

音频控件

<audio controls src="./assets//music.mp3"></audio>

滚动原理

 快速创建lorem(乱数假文)

lorem在HTML中的应用可以生成随机文本

li*30>lorem4

给body设置text-align: center 

text-align: center是可继承的,body→div→ul→li

让它的行盒子元素内容水平居中

控制里层ul的位置

两种思路:

1、margin-top为负(渲染主线程)

但是margin改变会导致reflow(重排)性能低,是在主线程上进行的

 2、用JS控制css3的transform实现位移(合成线程)

css属性变化本身是没有动画效果的

transition过渡,是针对数值类的属性来说的

歌词滚动时,当前歌词高亮并放大,并且与进度条位置一致

JS控制哪一个特殊样式,CSS写样式

放大fontsize会影响布局树,用transform:scale 最好

transition写在li.active还是li?

li.active效果消失没有过渡效果

所以加在li里面

 实现界面交互效果,切入点是什么?

不是怎么设置元素的属性,怎么监控这个音乐播放器的进度

而是从数据切入,我的手上有什么数据,转换成数组或对象

var lrc = `[00:01.06]难念的经
[00:03.95]演唱:周华健
[00:06.78]
[00:30.96]笑你我枉花光心计
[00:34.15]爱竞逐镜花那美丽
[00:36.75]怕幸运会转眼远逝
[00:39.32]为贪嗔喜恶怒着迷`;

此案例中,给的是字符串,想办法变成数组

分割歌词字符串

 理想效果:

[{time: 1.06, words: '难念的经'}, {time: 3.95, words: '演唱:周华健'}]

思路:

  • 根据换行符 \n 分割,将每一行都包装成对象
  • 遍历每个对象,每个对象由时间和歌词键值对组成
  • 歌词时间格式化,统一转换成秒
/**
 * 解析歌词字符串
 * 得到一个歌词对象的数组
 * 每个歌词对象:
 * {time: 开始时间, words: 歌词内容}
 */
function parseLrc () {
	let lines = lrc.split('\n');
	const result = [];
	for (let i = 0; i < lines.length; i++) {
		let str = lines[i];
		let parts = str.split(']');
		let timeStr = parts[0].substring(1); //从第一项开始截取
		const obj = {
			time: parseTime(timeStr),
			words: parts[1]
		}
		result.push(obj);
	}
	return result;
}
/**
 * 时间字符串转换成秒
 * @param {*} timeStr
 * @returns {Number} 小数
 */
function parseTime (timeStr) {
	let parts = timeStr.split(':');
	return +parts[0] * 60 + +parts[1];
}

要高亮显示的歌词,决定了ul列表的偏移量

计算出,当前播放器播到第几秒,对应的高亮歌词下标

获取当前播放时间
audio.currentTime;
遍历歌词数组,判断找到比当前时间大的时间,拿到当前的下标 - 1,就是对应的高亮歌词下标

数据少时,直接在ul中appendChild() :

但是这样频繁改动dom树,70个数据就改动了70多次,数据越多性能越低

function createLrcElements () {
	for (let i = 0; i < lrcData.length; i++) {
		let li = document.createElement('li');
		li.textContent = lrcData[i].words;
		doms.ul.appendChild(li);
	}
}

永远不要率先优化

做代码顺序:数据逻辑→界面逻辑→事件

数据多时 :

利用文档片段过渡

先将li节点插到createDocumentFragment()

再把createDocumentFragment里的东西加到ul里面

function createLrcElements () {
	const frag = document.createDocumentFragment();
	for (let i = 0; i < lrcData.length; i++) {
		let li = document.createElement('li');
		li.textContent = lrcData[i].words;
		// doms.ul.appendChild(li);
		frag.appendChild(li);
	}
	doms.ul.appendChild(frag);
}

createLrcElements();

 

let maxOffset = doms.ul.clientHeight - containerHeight;

设置ul 元素的偏移量

  • 小于最小值时,让它等于0
  • 大于最大值时,它等于最大值
  • 高亮之前清除所有的高亮
function setOffset () {
	let index = findIndex();
	let offset = liHeight * index + liHeight / 2 - containerHeight / 2;
	/* 最小值 */
	if (offset < 0) {
		offset = 0;
	}
	/* 最大值 */
	if (offset > maxOffset) {
		offset = maxOffset;
	}
	doms.ul.style.transform = `translateY(-${offset}px)`;
	/* 去掉之前的样式 */
	let li = doms.ul.querySelector('.active');
	if (li) {
		li.classList.remove('active')
	}
	/* 拿到需要高亮的li标签 */
	li = doms.ul.children[index];
	if (li) {
		li.classList.add('active');
	}
}

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

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

相关文章

一个工具类让你彻底解决bean深拷贝

深拷贝是我们在代码开发当中经常需要使用到的&#xff0c;但是市面上的对象拷贝方法&#xff0c;比如spring自带的&#xff0c;或者其他工具类带的对象拷贝&#xff0c;大部分都是浅拷贝&#xff0c;根本无法满足咱们的业务需求&#xff0c;我们就只能对里面的引用对象进行专门…

guacamole 纯web rdp预研:web应用程序部分

文章目录 web rdp预研 web应用程序部分预研目的相关基础Web应用结构&#xff08;框架&#xff09;配置tomcat运行web项目与前端交互原理问题整理Java web基础知识Java web调试预研结论 web rdp预研 web应用程序部分 ⭐️来自很多年前的笔记&#xff0c;只是一个归档&#xff0…

面向对象【成员变量与局部变量、方法声明与作用】

文章目录 成员变量局部变量成员变量与局部变量的区别 方法方法的作用方法的声明 成员变量 Java中的成员变量是指类中声明的变量&#xff0c;也称为实例变量或属性。它们与方法一样属于类的成员&#xff0c;不同之处在于&#xff0c;它们存储在对象(堆)中而不是栈中&#xff0c;…

基于灰色预测模型的负荷预测(matlab程序)

0.代码链接 基于灰色预测模型的负荷预测&#xff08;matlab程序&#xff09;资源-CSDN文库 1.简述 灰色预测是一种对含有不确定因素的系统进行预测的方法。灰色预测通过鉴别系统因素之间发展趋势的相异程度&#xff0c;即进行关联分析&#xff0c;并对原始数据进行生成处理来…

python:并发编程(二十六)

前言 本文将和大家一起探讨python并发编程的实际项目&#xff1a;win图形界面应用&#xff08;篇八&#xff0c;共八篇&#xff09;&#xff0c;系列文章将会从零开始构建项目&#xff0c;并逐渐完善项目&#xff0c;最终将项目打造成适用于高并发场景的应用。 本文为python并…

ADAS(高级驾驶员辅助系统)

什么是 ADAS ADAS —— 高级驾驶员辅助系统&#xff0c;包含一系列硬件和软件组件&#xff0c;自动起到驾驶员的多项作用。目前&#xff0c;常见的车辆 ADAS 功能包括自适应巡航控制、盲点检测、变道检测(车道偏离警告系统)、自动车道跟随和自动紧急制动、泊车。 L0到L2称为AD…

测试用例详解(强,硬,牛)

目录&#xff1a; 测试用例价值与体系黑盒测试方法论-等价类黑盒测试方法论-边界值黑盒测试方法论-因果图黑盒测试方法论-判定表黑盒测试方法论-场景法测试用例基础概念测试用例设计与评审面试测试测试用例设计搜索功能测试用例设计 1.测试用例价值与体系 测试用例概念测试…

【MQTT】| 搭建——在云服务器上搭建MQTT服务器

系列文章目录 【MQTT】| 搭建——在云服务器上搭建MQTT服务 失败了也挺可爱&#xff0c;成功了就超帅。 文章目录 前言1. EMQX简介2. EMQX部署3. EMQX一些操作指令3.1 启动EMQX3.2 停止EMQX3.3 检查EMQX运行状态3.4 卸载EMQX 4. EMQX设置4.1 进入EMQX控制面板 5. EMQX测试5.1…

模型评估 (Model Assessment)

1.模型评估 (Model Assessment) 笔记来源于《白话机器学习的数学》 我们训练好模型后&#xff0c;要对知道这个模型到底好不好&#xff0c;定量描述这个模型好坏就是模型评估 把获取的全部训练数据分成两份&#xff1a;一份用于测试&#xff0c;一份用于训练。然后用前者来评估…

数据库实训报告3000字

数据库实训报告1 一、实习目的 认识实习是本科教学计划中非常重要的实践性教学环节&#xff0c;其目的是使学生了解和掌握电力生产知识、印证、巩固和丰富已学过的计算机专业课程内容&#xff0c;培养学生理论联系实际&#xff0c;提高其在生产实践中调查研究、观察问题、分析问…

银行账户管理系统

1. 目的与要求 1、目的: (1)熟练掌握 C语言的基本知识和技能: (2)掌握面向对象程序设计的基本思想和方法;(3)能够利用所学的面向对象基本知识和技能&#xff0c;解决简单应用的程序设计 2、基本要求:(1)利用面向对象的方法以及 C的编程思想来完成系统的设计;(2)在设计的过程中…

抽象工厂模式(Abstract Factory)

定义 抽象工厂是一种创建型设计模式&#xff0c;它能创建一系列相关的对象&#xff0c;而无需指定其具体类。 前言 1. 问题 假设你正在开发一款家具商店模拟器。你的代码中包括一些类&#xff0c;用于表示&#xff1a; 一系列相关产品&#xff0c;例如椅子&#xff08;Chai…

091基于深度学习的手写汉字数字识别含10多种模型

emo仓库和视频演示找091期&#xff1a; 银色子弹zg的个人空间-银色子弹zg个人主页-哔哩哔哩视频 效果展示图如下&#xff1a; 代码文件展示如下&#xff1a; 运行01数据集文本生成制作.py可以读取图片路径保存再txt文本中&#xff0c; 运行02train.py可以对txt文本中的图片路…

同程数科基于 Apache Doris 构建统一实时数仓,查询提速数十倍!

本文导读&#xff1a; 同程数科是同程集团旗下的旅游产业金融科技服务平台&#xff0c;为上下游企业和个人消费者提供数字金融科技服务。近年来&#xff0c;随着同程数科业务的不断拓展和用户量的增加&#xff0c;高效可靠的一站式数据中心建设已成为必不可少的需求。为帮助业…

团体程序设计天梯赛-练习集L2篇④

&#x1f680;欢迎来到本文&#x1f680; &#x1f349;个人简介&#xff1a;Hello大家好呀&#xff0c;我是陈童学&#xff0c;一个与你一样正在慢慢前行的普通人。 &#x1f3c0;个人主页&#xff1a;陈童学哦CSDN &#x1f4a1;所属专栏&#xff1a;PTA &#x1f381;希望各…

WPF 零基础入门笔记(1):WPF静态页面,布局+样式+触发器

文章目录 官方文档往期回顾零基础笔记项目实战&#xff08;已完结&#xff09; WPF项目创建为什么选net core版本 WPF 静态页面WPF 页面布局WPF样式Style样式行内样式行外样式如果是简单样式&#xff0c;可以这么写如果是复杂样式 WPF样式继承WPF触发器单条件触发器多条件触发 …

LLDP(链路层发现协议)详解及C/C++代码实现

LLDP&#xff08;链路层发现协议&#xff09;是一种IEEE标准协议&#xff08;IEEE 802.1AB&#xff09;&#xff0c;它定义了封装在以太网帧中的消息&#xff0c;目的是通过默认情况下每30秒从每个端口定期重传一次&#xff0c;为设备提供一种向LAN&#xff08;局域网&#xff…

20个Java编程技巧

1. 把字符串常量放在前面 通过把字符串常量放在比较函数equals()比较项的左侧来防止偶然的 NullPointerException 从来都不是一个坏主意&#xff0c;就像这样&#xff1a; 这是毫无疑问的&#xff0c;把一种表达式转换成另一种更好的表达式&#xff0c;并不会失去什么。只要我…

mysql锁机制及MVCC底层原理

一、锁介绍 按性能可分为乐观锁&#xff08;适用于读多写少的情况下&#xff0c;如果是写多&#xff0c;导致过多cpu空转&#xff0c;影响性能&#xff09;和悲观锁&#xff08;适用于写多的情况&#xff09;按数据库操作粒度可分为表锁、页锁、行锁按数据库操作类型可分为读锁…

UE4/5动画系列(1.模板制作)

目录 动画模板制作 同步模板组制作 有模板做什么都方便&#xff0c;所以这里我们做一个动画蓝图的模板&#xff08;动物专用&#xff09; 动画模板制作 第一步创建一个动画蓝图的模板 然后找到第三人称的模板&#xff0c;将其模板的蓝图改名&#xff1a; 在动画蓝图的模板里…