HTML5技术实现的小钢琴

HTML5技术实现的小钢琴

用HTML5实现的小钢琴,按下钢琴键上的相应字母用或用鼠标点击钢琴键发声,源码如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>用HTML5实现的小钢琴</title><style>.key {width: 50px;height: 150px;border: 1px solid black;display: inline-block;margin: 5px;text-align: center;line-height: 150px;cursor: pointer;position: relative; /* 为了绝对定位伪元素而需要 */transition: transform 0.1s; /* 按键按下时平滑过渡效果 */}/* 当按键被按下时的样式 */.key.active {transform: translateY(4px); /* 将按键向下移动 */}</style>
</head>
<body><div id="A" class="key">A</div><div id="S" class="key">S</div><div id="D" class="key">D</div><div id="F" class="key">F</div><div id="G" class="key">G</div><div id="H" class="key">H</div><div id="J" class="key">J</div><div id="K" class="key">K</div><script>// 创建一个新的AudioContext实例,用于处理音频const audioContext = new (window.AudioContext || window.webkitAudioContext)();// 播放特定频率的声音function playSound(frequency) {const oscillator = audioContext.createOscillator();oscillator.type = 'sine'; // 选择波形类型,这里是正弦波oscillator.frequency.setValueAtTime(frequency, audioContext.currentTime); // 设置频率oscillator.connect(audioContext.destination); // 连接到音频输出oscillator.start(); // 开始播放oscillator.stop(audioContext.currentTime + 0.3); // 0.3秒后停止播放}// 获取所有的琴键元素const keys = document.querySelectorAll('.key');keys.forEach(key => {// 为每个琴键添加鼠标点击事件key.addEventListener('click', () => {playNote(key.textContent); // 播放琴键对应的音符animatePress(key); // 触发按键动画效果});});// 监听键盘按下事件document.addEventListener('keydown', (event) => {const keyName = event.key.toUpperCase(); // 获取按键名并转为大写const keyElement = document.getElementById(keyName); // 获取对应的琴键元素if (keyElement) {playNote(keyName); // 播放琴键对应的音符animatePress(keyElement); // 触发按键动画效果}});// 触发琴键按下的动画效果function animatePress(keyElement) {keyElement.classList.add('active'); // 添加按下状态的样式类setTimeout(() => {keyElement.classList.remove('active'); // 在100毫秒后移除按下状态的样式类}, 100); // 这个时间应该与CSS中的过渡时间一致}// 根据音符播放声音function playNote(note) {let frequency;switch (note) {case 'A':frequency = 261.63;break;case 'S':frequency = 293.66;break;case 'D':frequency = 329.63;break;case 'F':frequency = 349.23;break;case 'G':frequency = 392.00;break;case 'H':frequency = 440.00;break;case 'J':frequency = 493.88;break;case 'K':frequency = 523.25;break;default:return; // 如果不是琴键对应的字母,则不执行任何操作}playSound(frequency); // 调用函数播放声音}</script>
</body>
</html>

代码中使用了JavaScript中的箭头函数(Arrow Function)语法。箭头函数是ES6中的一种函数声明方式,它的语法比较简洁。箭头函数的基本格式是:(参数) => {函数体}。

keys.forEach(key => { })

箭头函数的参数是key,表示forEach遍历的每个元素。

key.addEventListener('click', () => { })

它是给每个琴键添加鼠标点击事件的代码。addEventListener 方法用于给元素添加事件监听器。这里添加了一个点击事件的监听器,当用户点击琴键时,箭头函数会被执行。

我对乐理了解很少,相关音高(pitch)请参考这里https://www.autopiano.cn/toolbox/pitch

用浏览器运行效果如下:

你可以用英文状态下按下琴键上的字符,或用鼠标单击琴键试试了。

钢琴演奏中,滑音是通过在音符之间平滑地滑动手指,使音符之间产生连贯的过渡效果。

上面的代码滑动按键能产生了滑音效果,但滑动鼠标不能,现在修改上面程序,使用鼠标都能产生了滑音效果。

修改后源码如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>用HTML5实现的小钢琴2</title><style>.key {width: 50px;height: 150px;border: 1px solid black;display: inline-block;margin: 5px;text-align: center;line-height: 150px;cursor: pointer;position: relative;transition: transform 0.1s;}.key.active {transform: translateY(4px);}</style>
</head>
<body><div id="A" class="key">A</div><div id="S" class="key">S</div><div id="D" class="key">D</div><div id="F" class="key">F</div><div id="G" class="key">G</div><div id="H" class="key">H</div><div id="J" class="key">J</div><div id="K" class="key">K</div><script>const audioContext = new (window.AudioContext || window.webkitAudioContext)();let isMouseDown = false; // 记录鼠标是否被按下function playSound(frequency) {const oscillator = audioContext.createOscillator();oscillator.type = 'sine';oscillator.frequency.setValueAtTime(frequency, audioContext.currentTime);oscillator.connect(audioContext.destination);oscillator.start();oscillator.stop(audioContext.currentTime + 0.3);}const keys = document.querySelectorAll('.key');keys.forEach(key => {key.addEventListener('mousedown', (event) => {isMouseDown = true; // 设置鼠标按下状态为trueplayNoteFromElement(event.target);});key.addEventListener('mouseenter', (event) => {if (isMouseDown) { // 如果鼠标处于按下状态,则播放音符playNoteFromElement(event.target);}});// 鼠标按下并移动到其他琴键时,无需额外mouseup事件监听});// 监听文档的mouseup事件,以处理鼠标在琴键外松开的情况document.addEventListener('mouseup', () => {isMouseDown = false;});// 监听键盘按下事件document.addEventListener('keydown', (event) => {const keyName = event.key.toUpperCase(); // 获取按键名并转为大写const keyElement = document.getElementById(keyName); // 获取对应的琴键元素if (keyElement && !keyElement.classList.contains('active')) {playNoteFromElement(keyElement);}});function playNoteFromElement(element) {const note = element.textContent;playNote(note);animatePress(element);}function animatePress(keyElement) {keyElement.classList.add('active');setTimeout(() => {keyElement.classList.remove('active');}, 100);}function playNote(note) {let frequency;switch (note) {case 'A':frequency = 261.63;break;case 'S':frequency = 293.66;break;case 'D':frequency = 329.63;break;case 'F':frequency = 349.23;break;case 'G':frequency = 392.00;break;case 'H':frequency = 440.00;break;case 'J':frequency = 493.88;break;case 'K':frequency = 523.25;break;default:return;}playSound(frequency);}</script>
</body>
</html>

请注意,这段代码包含了对鼠标按下、移动和松开,以及键盘按下的事件监听。

你可以在此基础上继续改进优化。

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

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

相关文章

原型设计工具Axure RP

Axure RP是一款专业的快速原型设计工具。Axure&#xff08;发音&#xff1a;Ack-sure&#xff09;&#xff0c;代表美国Axure公司&#xff1b;RP则是Rapid Prototyping&#xff08;快速原型&#xff09;的缩写。 下载链接&#xff1a;https://www.axure.com/ 下载 可以免费试用…

钧达股份:光伏跨界新贵只身赴港股,光伏“秩序重塑”?

2月21日&#xff0c;钧达股份终是在“千呼万唤”之中披露最新业绩快报。 快报显示&#xff0c;钧达股份预计2023年经调整后营业收入183.97亿元&#xff0c;同比增长58.65%&#xff0c;归母净利润8.32亿元&#xff0c;同比增长16.00%。 其中&#xff0c;由于Q4完整计提了9.5GW…

Microsoft的PromptBench可以做啥?

目录 PromptBench简介 PromptBench的快速模型性能评估 PromptBench数据集介绍 PromptBench模型介绍 PromptBench模型加载遇到的问题 第一次在M1 Mac上加载模型 vicuna和llama系列模型 PromptBench各个模型加载情况总结 PromptBench的Prompt快速工程 chain of thought…

命令执行 [WUSTCTF2020]朴实无华1

做题&#xff1a; 打开题目 我们用dirsearch扫描一下看看 扫描到有robots.txt&#xff0c;访问一下看看 提示我们 /fAke_f1agggg.php 那就访问一下&#xff0c;不是真的flag bp抓包一下 得到提示&#xff0c; /fl4g.php&#xff0c;访问一下看看 按alt&#xff0c;点击修复文…

品优购项目

一、品优购项目规划 1.1 网站制作流程 1.2 品优购项目整体介绍 项目名称&#xff1a;品优购 项目描述&#xff1a;品优购是一个电商网站&#xff0c;我们要完成PC端首页、列表页、注册页面的制作 1.3 品优购项目的学习目的 1.电商类网站比较综合&#xff0c;里面需要大量的…

红队评估四靶场

文章目录 环境搭建1.设置所需网卡2.更改win7设置3.DC设置4.web设置开启docker服务5.kali网段`渗透启动`1.确认对方靶机的IP地址2.端口探测3.web探测`2001端口``2002端口`Tomcat/8.5.19漏洞复现`2003端口`4.docker逃逸5.ssh密钥爆破`域渗透启动`1.提权2.隧道搭建各项配置文件内容…

远程连接 vscode 出错 “远程主机可能不符合 glibc 和 libstdc++ VS Code 服务器的先决条件”

原因&#xff1a; vscode 版本是 1.86&#xff0c;服务器上的 glibc 和 libstdc 版本不满足 要求(2.28 和 3.4.25)。 解决&#xff1a; 1、下载 1.85.2&#xff0c;解压直接运行 Code.exe。 2、回退 Remote-ssh 到 0.107.1。 参考&#xff1a; vscode 1.86版本远程ssh不兼容旧…

C语言----字符数组指针

1.char arr[] {a,b,c,d,e,f}; sizeof分析类型就可以计算所占的内存空间的大小&#xff1b; &#xff08;1&#xff09;printf("%d\n", sizeof(arr)); 数组名单独放进里面&#xff0c;计算整个数组大小&#xff0c;所以是6字节&#xff1b; &#xff08;2&#xff…

Java的String类

目录 String类的常用方法 1.1 字符串构造 1.2 String对象的比较 1.3 字符串查找 1.4 转换 1.5 字符串替换 1.6字符串拆分 1.7 字符串截取 1.8 其他操作方法 1.9 字符串的不可变性 1.10 字符串修改 String类的常用方法 1.1 字符串构造 String类常用的构造方法有很多…

__proto__和protype的区别

概述&#xff1a; prototype 函数静态属性&#xff0c;非实例属性,所有实例都可以继承它 __proto__ 实例属性&#xff0c;指向实例的原型对象&#xff0c;原型对象包括构造函数和protype属性 替代 现代浏览器中可以使用Object.getPrototypeOf()来替代__proto__来获取原型对象 …

srs集群下行edge处理逻辑

官方关于源站集群的介绍&#xff1a; Origin Cluster | SRS 下行边缘是指观众端从边缘edge拉流&#xff0c;边缘edge回源到源站origin节点拉流&#xff0c;然后再 把流转给客户端 边缘处理类SrsPlayEdge 当服务器收到播放请求时&#xff0c;创建对应的consumer消费者。在创…

【Algorithms 4】算法(第4版)学习笔记 10 - 3.3 平衡查找树(上篇)

文章目录 前言参考目录学习笔记0&#xff1a;符号表 ST 的回顾1&#xff1a;2-3 查找树1.1&#xff1a;定义1.2&#xff1a;2-3 树 demo 演示1.2.1&#xff1a;搜索&#xff1a;成功命中1.2.2&#xff1a;搜索&#xff1a;未命中1.2.3&#xff1a;插入&#xff1a;2-节点1.2.4&…