前端实现一个绕圆心转动的功能

news2025/1/24 8:48:55

前言: 今天遇到了一个有意思的需求,如何实现一个元素绕某一个点来进行圆周运动,用到了一些初高中的数学知识,实现起来还是挺有趣的,特来分享🎁。

一. 效果展示

  1. 我们先展示效果,如下图所示,放大镜会绕着我们设定好的一个圆心来进行圆周运动。相信这个需求还是比较常见的,接下来我会一步一步讲解我的实现过程。
    saomiao.gif

  2. tisp: 注意,放大镜是一张 png 的图片,并不是我们手动画出来的

二. 回忆画圆的步骤

  1. 不要着急想代码如何实现,我们先思考如何手动画一个圆?在这里我们很容易想到一个初中学习数学用到的工具----圆规,如下图所示。
    IMB_DQj3Fs.gif

  2. 从上图可以得知,手动画一个圆其实很简单,大致分为三个步骤。

    1. 确定圆心
    2. 确定半径
    3. 旋转圆规
  3. 前两个我们可以很轻松的根据我们的需求设定好,圆心就是确定页面上的一个点坐标而已,半径也是我们手动写死的一个变量值而已,关键点其实就在于第三步我们如何实现。

  4. 在实现第三步之前,我会提前预热一下弧度的知识,因为等会我们需要用到 sincos 的相关知识,而在 js 中的相关三角函数的参数必须为弧度,所以我们需要通过角度来计算得出弧度
    image.png

  5. 我们大家都知道,一个圆有360度,且度数弧度之间有一个转换公式,如下。
    image.png

  6. 得知了转换关系,我们就可以定义一个变量 angle 来表示我们这个 div 做圆周运动时绕圆心转过的角度,则弧度(radian)radian =(angle*π)/180

三. 回忆 sin 和 cos

  1. 我们先在草稿纸上演练一遍我们的逻辑是否可行。让我们先准备一个矩形来代表我们的页面,然后确定一个点来作为圆心
    image.png

  2. 圆心的位置坐标其实很简单,不要想复杂了,就是相对于页面的位置而已,我们记住这个 (100,100) 的值,等会它会作为我们实现圆周运动的圆心位置。
    image.png

  3. 接下来确定半径,这里我就随便取一个 50 作为半径值。

  4. 假设这是我们的 div 已经绕圆心转了一些弧度以后的情景。
    image.png

  5. tips: 在这里有一个十分重要的概念,div 滑过的路径其实是由无数个不同坐标位置的点构成的。
    image.png

  6. 接下来就是本文的关键部分,请大家喝口水认真听讲。我们取一个中间时刻,假设 div 此时做圆周运动到了点 B ,那么我们的问题就转化为了如何求点 B 的坐标信息。(不要忘了,坐标信息其实就是相对于页面上的 left 和 top 而已。)
    image.png

  7. 一步一步来,我们先求 X 坐标的值,换算下来其实就是要求 a 的长度。
    image.png

  8. 此时我们准备好拿出已有的数据,来套公式即可。度数我们是有的,因为我们已经用变量 angle 来假设我们绕过的度数,则弧度radian =(angle*π)/180 ,然后根据三角函数的正弦定理可得 sin(rad)=a/radius

  9. 此时半径已知,sin(rad) 已知,则 a=sin(rad) X 50,然后加上圆心的 x 坐标值 100,即可得出此时 B 点相对于页面的 left 值。

  10. Y 坐标同理,只不过公式换为 cos 即可,换算过程不再重复,但是需要注意的点是,我们计算 Y 坐标 的目的其实在求的是 B 点的 top 值,又因为我们前端的坐标Y轴其实和数学的坐标Y轴的正负极是相反的,所以我们其实要算的值是这一段的距离。如下图:
    image.png

  11. 即 B 点的 top 值为圆心的 Y 坐标值100 - 距离b,至此我们所有需要的数据都已经获得,接下来就是用代码验证我们思路的可行性。

四. 实现圆周运动函数

  1. 经过上面的解释,我们接下来就可以动手写代码了。

  2. 我们先简单设计一个画面,内容很简单,一个普通的蓝底页面中间有一个居中的红色 div,要想脱离文档流自行移动,那么这个 div 肯定是 absolute 绝对定位。
    image.png

  3. 然后提前定义好我们需要用到的变量,为接下来的工作做准备。
    image.png

  4. 因为我们是需要不断通过改变 divX 偏移值和 Y 偏移值来实现圆周运动的,(先不考虑性能和优化)所以很容易想到 setInterval,所以现在你的代码应该是这个样子。

    // 1. 确定圆心的位置
    const centerPointer = { x: 100, y: 100 };
    // 2. 确定圆周运动的半径
    const radius = 50;
    // 3. div 运动时的角度
    let angle = 0;
    
    function run() {
      setInterval(() => {}, 16);
    }
    
  5. 首先我们需要让我们定义好的 angle 自增。

    function run() {
      setInterval(() => {
        angle += 1;
      }, 16);
    }
    
  6. 接下来计算 ab 的值,根据我们上面的出的公式 radian =(angle*π)/180sin(rad)*radius=a,可以写出下面的代码。

    function run() {
      setInterval(() => {
        angle += 1;
        const radian = (angle * Math.PI) / 180;
        const a = Math.sin(radian) * radius;
        const b = Math.cos(radian) * radius;
      }, 16);
    }
    
  7. 根据我们在上面的结论,divtop 值为 半径 + a left 值为 半径 - b,你现在的 run函数 代码应该是这样的。

    
    function run() {
      setInterval(() => {
        if (!box.value) return;
        angle += 1;
        const radian = (angle * Math.PI) / 180;
        const a = Math.sin(radian) * radius;
        const b = Math.cos(radian) * radius;
        const x = centerPointer.x + a;
        const y = centerPointer.y - b;
    
        box.value.style.left = x + "px";
        box.value.style.top = y + "px";
      }, 16);
    }
    
  8. 然后给我们的 div 打上 ref 那到这个 dom 元素。
    image.png

  9. onMounted 里执行我们的 run 函数,接下来就是见证奇迹的时刻。
    yuanzhou.gif

五. 源码

<script setup lang="ts">
import { ref, onMounted,  } from "vue";

const box = ref<HTMLDivElement>();
// 1. 确定圆心的位置
const centerPointer = { x: 100, y: 100 };
// 2. 确定圆周运动的半径
const radius = 50;
// 3. div 运动时的角度
let angle = 0;

function run() {
  setInterval(() => {
    if (!box.value) return;
    angle += 1;
    const radian = (angle * Math.PI) / 180;
    const a = Math.sin(radian) * radius;
    const b = Math.cos(radian) * radius;
    const x = centerPointer.x + a;
    const y = centerPointer.y - b;

    box.value.style.left = x + "px";
    box.value.style.top = y + "px";
  }, 16);
}

onMounted(() => {
  run();
});
</script>
<template>
  <div class="w-full h-full centered bg-blue relative flex flex-col centered">
    <div ref="box" class="absolute w-50px h-50px bg-red"></div>
  </div>
</template>

六. 思考题

现在我们的 div顺时针运动,我们该如何让它逆时针运动呢?🤔

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

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

相关文章

C#知识点-22(ADO.NET五个对象,SQL漏洞注入攻击)

ADO.NET 概念&#xff1a;ADO.NET就是一组类库&#xff0c;这组类库可以让我们通过程序的方式访问数据库&#xff0c;就像System.IO的类用类操作文件一样&#xff0c;System.Data这组类是用来操作数据库的&#xff08;不光是MSSql Server&#xff09;&#xff0c;它提供了统一…

骨传导蓝牙耳机哪款好?掌握几大选购技巧不再踩坑

骨传导蓝牙耳机&#xff0c;顾名思义就是依靠骨头传递声音的耳机&#xff0c;它们的佩戴方式与市面上常见的入耳式耳机显著不同。在骨传导技术领域&#xff0c;它的地位犹如智能手机界的苹果&#xff0c;无论是用户体验还是整体声誉都相当出色&#xff0c;赢得了广泛的消费者认…

【Web安全】SQL各类注入与绕过

【Web安全】SQL各类注入与绕过 【Web安全靶场】sqli-labs-master 1-20 BASIC-Injection 【Web安全靶场】sqli-labs-master 21-37 Advanced-Injection 【Web安全靶场】sqli-labs-master 38-53 Stacked-Injections 【Web安全靶场】sqli-labs-master 54-65 Challenges 与62关二…

删除指定的数

删除指定的数 题目描述&#xff1a;解法思路&#xff1a;解法代码&#xff1a;运行结果&#xff1a; 题目描述&#xff1a; 先输入10个整数存放在数组中&#xff0c;再输入⼀个整数n&#xff0c;删除数组中所有等于n的数字&#xff0c;数组中剩余的数组保证数组的最前面&#…

python跨文件夹调用

如图所示&#xff0c;我们要在bin文件夹下的run_patchcore.py文件中调用src/patchcore文件夹下的backbone.py, common.py等文件。如第13行 方法1&#xff1a;将patchcore的上一层目录src添加到环境变量中 run_patchcore.py 文件中写入import sys sys.path.append(/home/rui/P…

【QA-SYSTEMS】CANTATA-解决Jenkins中build Cantata报错

【更多软件使用问题请点击亿道电子官方网站查询】 1、 文档目标 解决Jenkins中build Cantata测试项目报找不到license server的错误。 2、 问题场景 在Jenkins中build Cantata测试项目&#xff0c;报错“Failed to figure out the license server correctly”。 3、软硬件环…

Java精品项目--第7期基于SpringBoot的驾校后台管理系统的设计分析与实现

项目使用技术栈 JavaSpringBootMavenMySQLMyBatisShiroHTMLVue.js&#xff08;非前后端分离&#xff09;… 项目截图

开发一个小程序需要多少钱

【腾讯云】 爆款2核2G3M云服务器首年 61元&#xff0c;叠加红包再享折上折 要做小程序的小伙伴们福利来啦&#xff01; 抢购方式&#xff1a; 方式一&#xff1a;直达链接》点我直接抢购 方式二&#xff1a;长按住识别以下官方二维码直接抢购 01 小程序认证 根据微信公众平台…

每日一题 — 有效三角形的个数

611. 有效三角形的个数 - 力扣&#xff08;LeetCode&#xff09; 思路&#xff1a; 双指针思想&#xff0c;先将数据排序 然后先固定最大的数Max&#xff0c;也就是数组最后的数再定义一左一右两个下标 left 、 right&#xff0c;当这个值相加大于最大的数&#xff0c;那么他两…

软件设计师12--进程的概念概述

软件设计师12--进程的概念概述 考点1&#xff1a;线程的概念进程的概念进程与程序进程与线程例题&#xff1a; 考点2&#xff1a;进程的状态例题&#xff1a; 考点1&#xff1a;线程的概念 进程的概念 进程与程序 进程与线程 进程的2个基本属性&#xff1a;可拥有资源的独立单…

二分查找算法:高效搜索有序数据的利器

二分查找算法&#xff1a;高效搜索有序数据的利器 在计算机科学中&#xff0c;搜索是一项基本而重要的操作。对于有序数据&#xff0c;二分查找算法是一种高效的搜索方法。本文将介绍二分查找算法的原理、实现以及其在实际应用中的优势&#xff0c;帮助读者理解和应用这一常用的…

当前爆火的:ChatGPT4、Claude3、Gemini、Sora、GPTs及AI领域中的集中大模型的最新技术

原文链接&#xff1a;当前爆火的&#xff1a;ChatGPT4、Claude3、Gemini、Sora、GPTs及AI领域中的集中大模型的最新技术 第一&#xff1a;2024年AI领域最新技术 1.最新超强模型Claude3使用 2.OpenAI新模型-GPT-5 3.谷歌新模型-Gemini使用 4.Meta新模型-LLama3 5.阿里巴巴…

【教育部白名单赛事】C语言编程题解析--软件编程邀请赛(决赛)

文章目录 1、保留12位小数的浮点数2、气温统计3.大写字母的判断4、【递归】母鸡的故事5、小白免再排队 1、保留12位小数的浮点数 输入一个双精度浮点数&#xff0c;保留12位小数&#xff0c;输出这个浮点数。 时间限制&#xff1a;1000 内存限制&#xff1a;65536 【输入】 只…

24计算机考研调剂 | 北京语言大学

北京语言大学 刘忠宝教授课题组招收计算机学硕调剂生2名 考研调剂招生信息 学校:北京语言大学 专业:工学->计算机科学与技术->计算机应用技术 年级:2023 招生人数:2 招生状态:正在招生中 联系方式:********* (为保护个人隐私,联系方式仅限APP查看) 补充内容 一、…

2024护网面试题精选(一)

0x00.基础漏洞篇 00-TOP10漏洞 1.SQL注入 2.失效的身份认证和会话管理 3.跨站脚本攻击XSS 4.直接引用不安全的对象 5.安全配置错误 6.敏感信息泄露 7.缺少功能级的访问控制 8.跨站请求伪造CSRF 9.实验含有已知漏洞的组件 10.未验证的重定向和转发 01-SQL注入漏洞 …

git unable to access的方式,【有代理】

git unable to access的方式&#xff0c;【有代理】 Git命令取消proxy代理 git config --global --unset http.proxy git config --global --unset https.proxy查看自身代理&#xff0c;这边小打个码。 然后切换 git config git config --global http.proxy http://127.0.0.…

Redis实战—商户查询缓存

本博客为个人学习笔记&#xff0c;学习网站&#xff1a;黑马程序员Redis入门到实战 实战篇之商户查询缓存 目录 什么是缓存 添加Redis缓存 缓存更新策略 数据库缓存不一致解决方案 案例&#xff1a;给查询商铺的缓存添加超时剔除和主动更新策略 缓存穿透 案例&#xff1…

【书生·浦语大模型实战营】第6节 OpenCompass 大模型评测 课后作业

OpenCompass 大模型评测 0. 课程链接1. 课后作业1.1 基础作业1.2 进阶作业 2. 结业总结 0. 课程链接 链接&#xff1a;https://github.com/InternLM/tutorial/blob/main/opencompass/opencompass_tutorial.md 1. 课后作业 1.1 基础作业 使用 OpenCompass 评测 InternLM2-Cha…

Map集合体系——遍历,HashMap,TreeMap,LikedHashMap

认识Map集合 Map集合体系特点 方法 代码示例 package com.zz.Map;import java.util.*;public class Test {public static void main(String args[]){Map<String, Integer> map new HashMap <>();//经典代码&#xff0c;按照键 无序 不重复 无索引map.put("…

Android岗大厂面试官常问的那些问题,2024年Android者未来的出路在哪里

前言 伟人曾经说过&#xff1a; 书是人类进步的阶梯 书中自有黄金屋&#xff0c;书中自有颜如玉 读书破万卷&#xff0c;下笔如有神 书是唯一不死的东西。 书籍是伟大的天才留给人类的遗产。 最近有很多朋友在我的公众号上提问“Android开发的经典入门教材和学习路线&#xff…