Canvas 动画: atan2 三角函数与鼠标跟随效果

9145201d6a64c39fc7fc31fbbda6be6c.gif

这个案例展示了如何使用HTML5的Canvas和JavaScript实现一个动态效果:在画布上绘制一个箭头,并让它实时跟随鼠标移动。这个小项目不仅有趣,还能帮助你理解编程和基本数学概念的实际应用。

项目需求

我们的目标是在一个画布上绘制一个箭头,并让这个箭头随着鼠标的移动自动旋转,始终指向鼠标的位置。

数学基础知识:atan2函数

在这个项目中,最关键的数学概念是atan2函数,它帮助我们计算箭头应该如何旋转才能指向鼠标的位置。

  • dxdy:这些是鼠标位置和箭头位置之间的水平和垂直距离。dx是x方向上的差值,dy是y方向上的差值。

  • atan2(dy, dx)是什么?:它是一个特殊的数学函数,用来计算给定的dxdy所对应的角度。这个角度表示从箭头到鼠标的方向。我们之所以使用atan2,是因为它能够处理所有可能的方向(上下左右斜角),并且它比普通的atan函数更为精确和稳定。

cf86be5a275bddc82c067f1ea927dd2b.jpeg

技术要点

让我们来逐步理解项目中涉及的技术要点。

  1. Canvas绘图

  • Canvas是HTML5提供的一个绘图环境。我们使用<canvas>标签来创建一个画布,之后在这个画布上绘制箭头。canvas.getContext('2d')提供了一个2D绘图上下文,通过这个上下文可以绘制图形、设置颜色、处理旋转等操作。

事件监听

  • 我们使用JavaScript的mousemove事件监听器,实时捕捉鼠标在画布上的位置。每次鼠标移动时,事件监听器都会记录鼠标的x和y坐标,这样我们就知道鼠标在哪里了。

图形旋转

  • 当我们知道鼠标的位置后,接下来要做的就是计算箭头应该朝向哪个方向。通过Math.atan2(dy, dx)计算出箭头的旋转角度,然后使用Canvas的rotate方法,让箭头旋转到正确的角度,指向鼠标。

代码展示

<!doctype html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>跟随鼠标移动的箭头</title><style>body {font-family: Arial, sans-serif;display: flex;flex-direction: column;align-items: center;justify-content: center;height: 100vh;margin: 0;background-color: #f0f0f0;}canvas {border: 1px solid #000;}header {margin-bottom: 20px;}aside {margin-top: 10px;font-size: 14px;}</style>
</head>
<body><header>Example from <a href="https://qianduandaren.com"><em>前端达人</em></a></header><canvas id="canvas" width="400" height="400"></canvas><aside>请移动鼠标</aside><script>class Utils {static captureMouse(element) {const mouse = { x: 0, y: 0, event: null };element.addEventListener('mousemove', event => {const rect = element.getBoundingClientRect();mouse.x = event.clientX - rect.left;mouse.y = event.clientY - rect.top;mouse.event = event;});return mouse;}}class Arrow {constructor() {this.x = 0;this.y = 0;this.color = "#ffff00";this.rotation = 0;}draw(context) {context.save();context.translate(this.x, this.y);context.rotate(this.rotation);context.lineWidth = 2;context.fillStyle = this.color;context.beginPath();context.moveTo(-50, -25);context.lineTo(0, -25);context.lineTo(0, -50);context.lineTo(50, 0);context.lineTo(0, 50);context.lineTo(0, 25);context.lineTo(-50, 25);context.lineTo(-50, -25);context.closePath();context.fill();context.stroke();context.restore();}}window.onload = () => {const canvas = document.getElementById('canvas');const context = canvas.getContext('2d');const mouse = Utils.captureMouse(canvas);const arrow = new Arrow();arrow.x = canvas.width / 2;arrow.y = canvas.height / 2;const drawFrame = () => {requestAnimationFrame(drawFrame);context.clearRect(0, 0, canvas.width, canvas.height);const dx = mouse.x - arrow.x;const dy = mouse.y - arrow.y;arrow.rotation = Math.atan2(dy, dx);arrow.draw(context);};drawFrame();};</script>
</body>
</html>

分步解释

1. Canvas设置与初始化

const canvas = document.getElementById('canvas');
const context = canvas.getContext('2d');

获取Canvas元素和上下文:首先,在onload函数中,我们通过getElementById获取HTML中的canvas元素,并通过getContext('2d')获取2D绘图上下文,这个上下文是我们用来绘制图形的基础。

2. 工具类Utils:捕捉鼠标位置

class Utils {static captureMouse(element) {const mouse = { x: 0, y: 0, event: null };element.addEventListener('mousemove', event => {const rect = element.getBoundingClientRect();mouse.x = event.clientX - rect.left;mouse.y = event.clientY - rect.top;mouse.event = event;});return mouse;}
}

Utils类简介:

Utils类提供了一个静态方法captureMouse,这个方法的作用是帮助我们实时捕捉鼠标在画布中的位置。

捕捉鼠标位置:

  • getBoundingClientRect():用于获取画布相对于浏览器窗口的位置和大小。

  • 监听mousemove事件:每次鼠标在画布上移动时,都会触发mousemove事件,这时我们计算鼠标相对于画布的x、y坐标,并存储在mouse对象中。

  • 最后返回这个mouse对象,以便我们在后续的代码中随时获取鼠标的位置。

3、定义箭头类Arrow

class Arrow {constructor() {this.x = 0;this.y = 0;this.color = "#ffff00";this.rotation = 0;}draw(context) {context.save();context.translate(this.x, this.y);context.rotate(this.rotation);context.lineWidth = 2;context.fillStyle = this.color;context.beginPath();context.moveTo(-50, -25);context.lineTo(0, -25);context.lineTo(0, -50);context.lineTo(50, 0);context.lineTo(0, 50);context.lineTo(0, 25);context.lineTo(-50, 25);context.lineTo(-50, -25);context.closePath();context.fill();context.stroke();context.restore();}
}
  • Arrow类用于定义箭头的初始属性,包括位置(x、y)、颜色和旋转角度。

  • draw方法:用于在画布上绘制箭头。这个方法使用了Canvas的绘图API,首先保存当前绘图状态(context.save()),然后移动并旋转画布(translate和rotate),根据预定的路径绘制出一个箭头形状,最后填充颜色和描边(fill和stroke),并恢复画布状态(context.restore())。

4.绘制与旋转箭头

const drawFrame = () => {requestAnimationFrame(drawFrame);context.clearRect(0, 0, canvas.width, canvas.height);const dx = mouse.x - arrow.x;const dy = mouse.y - arrow.y;arrow.rotation = Math.atan2(dy, dx);arrow.draw(context);};drawFrame();
  • 动画帧更新:使用requestAnimationFrame(drawFrame)来实现平滑的动画效果。这个方法让浏览器在每次重绘时调用drawFrame,从而以高效的方式不断更新箭头的位置和方向。

  • 清除画布:每一帧开始时,我们使用context.clearRect(0, 0, canvas.width, canvas.height)清空画布,这样就不会看到之前绘制的内容残留。这样做可以确保每次重绘都是干净的。

  • 计算方向

    • dxdy:计算鼠标相对于箭头的水平和垂直距离。

    • 旋转角度:通过Math.atan2(dy, dx)计算出箭头需要旋转的角度。atan2函数根据这两个差值返回一个介于π之间的角度值,表示从箭头位置到鼠标位置的方向。

  • 绘制箭头:在计算完旋转角度后,我们调用arrow.draw(context),根据新的角度在画布上绘制箭头。这使得箭头能够实时指向鼠标的位置。

结束

这个项目演示了如何使用HTML5的Canvas和JavaScript来创建一个动态的跟随鼠标移动的箭头效果。我们通过atan2函数计算出箭头旋转的角度,并使用Canvas的绘图功能将其实时显示在网页上。通过这个案例,您不仅学会了如何使用Canvas绘图和JavaScript事件监听,还掌握了如何将数学函数应用于实际的编程问题中。

希望这个案例能帮助您更好地理解这些技术,并激发您进一步学习和探索的兴趣!

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

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

相关文章

HTML+CSS浮动和清除浮动的效果及其应用场景举例

一、清除浮动的效果 解释 .container&#xff1a;用于展示浮动和清除浮动效果的容器&#xff0c;具有边框和背景色以便于区分。 .float-box&#xff1a;浮动元素&#xff0c;用不同的背景色标识。 .clearfix&#xff1a;使用伪元素清除浮动的类&#xff0c;应用于第二个容器。 …

电脑浏览器打不开部分网页

电脑浏览器打不开部分网页 时间: 2024-08-25 问题描述: 电脑突然打不开部分网页 例如腾讯文档 夸克网盘 但其他网页能够正常打开 原因 可能为域名解析问题 更改DNS即可解决 解决办法 控制面板–> 网络和Internet—>网络连接—> WLAN----> 属性 —> Interne…

已解决:`javax.security.auth.RefreshFailedException` 刷新失败的正确解决方法,亲测有效!!!

在 Java 开发中&#xff0c;javax.security.auth.RefreshFailedException 异常通常涉及到安全认证和授权机制。这个异常通常在尝试刷新或更新认证凭证时出现问题&#xff0c;可能会影响到应用的安全性和稳定性。本文将详细分析该异常的原因&#xff0c;并提供有效的解决方法。 …

Kotlin 学习手册01 高阶函数

Kotlin 学习手册01 高阶函数 1 函数作为参数2 函数作为返回值3 测试代码 在Kotlin中&#xff0c;高阶函数&#xff08;Higher-Order Functions&#xff09;是指接受函数作为参数或返回一个函数的函数。这种特性允许你将函数像变量一样传递&#xff0c;从而使代码更加灵活和可重…

一元四次方程求解-【附MATLAB代码】

目录 前言 求解方法 MATLAB验证 附&#xff1a;一元四次方程的故事 前言 最近在研究机器人的干涉&#xff08;碰撞&#xff09;检测&#xff0c;遇到了一个问题&#xff0c;就是在求椭圆到原点的最短距离时&#xff0c;构建的方程是一个一元四次方程。无论是高中的初等数学…

文件包含漏洞案例

一、PHP://INPUT Example 1&#xff1a;造成任意代码执行 源代码&#xff1a; <meta charset"utf8"> <?php error_reporting(0); $file $_GET["file"]; if(stristr($file,"php://filter") || stristr($file,"zip://") |…

linux安装python3.11.9

下载 wget https://www.python.org/ftp/python/3.11.9/Python-3.11.9.tgz解压 tar -zxf Python.tgz准备编译环境 yum -y install gcc make zlib zlib-devel libffi libffi-devel readline-devel openssl-devel openssl11 openssl11-devel安装openssl11&#xff0c;后期的pip…

批量整理文件夹的得力助手:让杂乱无章变得井井有条

在日常工作和生活中&#xff0c;我们经常会遇到这样一个问题&#xff1a;电脑或移动硬盘里的文件夹越积越多&#xff0c;文件散落各处&#xff0c;找起来既费时又费力。 有时候&#xff0c;为了找一个文件&#xff0c;你可能得在无数个文件夹中来回穿梭&#xff0c;仿佛进行了…

C++ 设计模式——命令模式

C 设计模式——命令模式 C 设计模式——命令模式主要组成部分构建过程命令模式 UML 图UML 图解析 命令模式的优点命令模式的缺点命令模式适用场景总结完整代码 C 设计模式——命令模式 命令(Command)模式是一种行为型模式&#xff0c;它将请求封装为对象&#xff0c;从而使您能…

Centos7整合fail2ban配置ssh和pgsql以及vault

配置Fail2Ban 配置环境 useradd fail2ban visudo#添加 fail2ban ALL(ALL) ALL配置conda源 conda config --add channels https://mirrors.tuna.tsinghua.edu.cn/anaconda/pkgs/free/ conda config --add channels https://mirrors.tuna.tsinghua.edu.cn/anaconda/pkgs/main/…

数据中台架构设计

由于当前项目需要对接多个不同的数据源&#xff0c;同时涉及到多端处理&#xff0c;而且需要考虑海量数据处理&#xff0c;还有总部与分部架构部署问题&#xff0c;因而整体技术栈倾向于大数据和分表分库式处理数据层接入问题。 简单讲&#xff0c;项目分为数据中台和业务中台…

【MySQL数据库管理问答题】第3章 理解 MySQL 体系

目录 1. MySQL 服务器进程 mysqld 由哪三层组成&#xff0c;从各自所实现的功能角度分别加以说明。 2. MySQL 支持的通讯协议都有哪些&#xff1f;其连接方式和所支持的操作系统有何不同。 3. 请说明 SQL 层的组件名称和各自的作用。 4. MySQL 8.0 除了默认的存储引擎 InnoD…

数据库之权限与安全

目录 一、MySQL权限表 1.MySQL权限系统的工作原理 2.权限表 二、账号管理 1.创建账号 2.删除账号 3.root用户修改自身密码 4.root用户修改普通用户密码 5.root用户密码丢失的解决方法 三、MySQL权限管理 1.查看账号权限 2.给账号授权 3.收回权限 一、MySQL权限表 …

[新手入门]1台电脑+1个电视+2个软件(sunshine+moonlight) 解决黑神话悟空没有hdmi线的痛...

sunshinemoonlight 解决黑神话悟空 本地串流投屏 背景:偶然间在B站发现了sunshinemoonlight方案,替代hdmi线,做本地串流...于是心灵手巧的我开始尝试踩坑之路:1.准备安装包2.开始安装2.1 笔记本windows安装sunshine2.2 遇到了第一个坑.Fatal: ViGEmBus is not installed or run…

【ansible】ansible roles

ansible roles 简介 Ansible Roles是一种组织和管理Ansible Playbooks的方法。它们允许将相关的配置和任务分组到一个可重用的单元中&#xff0c;使得代码更加模块化和可维护。 一个Ansible Role包含了一组预定义的变量、任务和文件结构。它可以被其他Playbooks调用和使用&am…

docke进阶---镜像迁移、容器的ip地址、端口映射和持久化

1.镜像的迁移 1.镜像打包 #查看镜像有一个centos的镜像 [rootdocker0 ~]# docker images REPOSITORY TAG IMAGE ID CREATED SIZE centos latest 5d0da3dc9764 2 years ago 231MB 3查看帮助文件 docker --help save Save one or more…

【Unity】移动端草海解决方案

草海是开放大世界渲染的必不可少的因素&#xff0c;Unity 原生的 Terrain 草海效率较低&#xff0c;而且无法与 RVT 结合起来&#xff0c;无法在移动端上实现。因此我们自己搓出来一套草海系统&#xff0c;使用 C# 多线程辅助运算&#xff0c;并能支持割草、烧草等进阶玩法。草…

突破编程:深入理解C++中的组合模式

突破编程&#xff1a;深入理解C中的组合模式 在C及众多面向对象编程语言中&#xff0c;设计模式是解决问题的经典方案&#xff0c;它们帮助开发者在面对复杂系统设计时&#xff0c;能够遵循一套经过验证的最佳实践。组合模式&#xff08;Composite Pattern&#xff09;是这些设…

数据库内容保密检查系统:及时发现“潜在”安全威胁

日前&#xff0c;国内专注于保密与非密领域的分级保护、等级保护、业务连续性安全和大数据安全产品解决方案与相关技术研究开发的领军企业——国联易安自主研发的国联数据库内容保密检查系统V1.0通过国保局涉密检测&#xff0c;获得涉密信息系统产品检测证书。其主要具备以下主…

JS数据类型——【Map】精讲

JavaScript 中的 Map 是一种用于存储键值对的集合&#xff0c;它与传统的对象&#xff08;Object&#xff09;类似&#xff0c;但提供了更丰富的功能和灵活性。以下是 Map 相关的主要知识点&#xff1a; 1. 基础概念 键值对存储&#xff1a;Map 使用键值对的形式存储数据&…