記錄--怎麼寫一個可以滑鼠控制旋轉的div?

来源:https://www.cnblogs.com/smileZAZ/archive/2023/10/17/17770431.html
-Advertisement-
Play Games

這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 說在前面 滑鼠控制元素旋轉在現在也是一個很常見的功能,讓我們從實現div元素的旋轉控制開始來瞭解元素旋轉的具體原理和實現方法吧。 效果展示 體驗地址 code.juejin.cn/pen/7290719… 實現步驟 畫一個div 首先我們 ...


這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助

說在前面

滑鼠控制元素旋轉在現在也是一個很常見的功能,讓我們從實現div元素的旋轉控制開始來瞭解元素旋轉的具體原理和實現方法吧。

效果展示

體驗地址

code.juejin.cn/pen/7290719…

實現步驟

畫一個div

首先我們需要先畫一個div,併在它上方加上旋轉圖標,我們可以通過這個圖標來對div進行旋轉,具體代碼如下:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
    <link rel="stylesheet" href="./index.css" />
  </head>
  <body>
    <div class="container">
      <div class="rotate-div">
        <div class="rotate-icon">↻</div>
      </div>
    </div>
  </body>
  <script src="./index.js"></script>
</html>

加點css

將div設置為紅色背景的方塊,調整旋轉圖標的位置,具體代碼如下:

.container {
  display: flex;
  justify-content: center;
  align-items: center;
  height: 100vh;
}

.rotate-div {
  position: relative;
  width: 200px;
  height: 200px;
  background-color: red;
  transform-origin: center center;
}

.rotate-icon {
  position: absolute;
  top: -50px; /* 調整圖標的位置 */
  left: 50%;
  transform: translateX(-50%);
  font-size: 20px;
  cursor: pointer;
}

效果如下:

完成滑鼠拖拽旋轉功能

滑鼠在旋轉圖標按下的時候,我們需要監聽滑鼠移動事件,根據滑鼠移動位置和初始點擊位置的相對角度來計算方塊旋轉的角度。

1、獲取方塊和旋轉圖標元素對象

首先我們要先獲取方塊和旋轉圖標元素對象,便於後續事件監聽和元素操作。

const rotateDiv = document.querySelector(".rotate-div");
const rotateIcon = document.querySelector(".rotate-icon");

返回值類型:TextRectangle對象,每個矩形具有四個整數性質( 上, 右 , 下,和左 )表示的坐標的矩形,以像素為單位。

 rectObject.top:元素上邊到視窗上邊的距離;

 rectObject.right:元素右邊到視窗左邊的距離;

 rectObject.bottom:元素下邊到視窗上邊的距離;

 rectObject.left:元素左邊到視窗左邊的距離;

我們記錄下方塊的初始中心點:

 const centerX = rect.left + rect.width / 2;
 const centerY = rect.top + rect.height / 2;
(2)計算旋轉角度

Math.atan2()

Math.atan2()  返回從原點 (0,0) 到 (x,y) 點的線段與 x 軸正方向之間的平面角度 (弧度值),也就是 Math.atan2(y,x)

Math.atan2(y, x)

y, x

atan2 方法返回一個 -pi 到 pi 之間的數值,表示點 (x, y) 對應的偏移角度。這是一個逆時針角度,以弧度為單位,正 X 軸和點 (x, y) 與原點連線 之間。註意此函數接受的參數:先傳遞 y 坐標,然後是 x 坐標。

atan2 接受單獨的 x 和 y 參數,而 atan 接受兩個參數的比值。

由於 atan2 是 Math 的靜態方法,所以應該像這樣使用:Math.atan2(),而不是作為你創建的 Math 實例的方法。

function getAngle(centerX, centerY, mouseX, mouseY) {
  return Math.atan2(mouseY - centerY, mouseX - centerX) * (180 / Math.PI);
}

使用當前滑鼠位置相對角度減去滑鼠初始點擊點的相對角度即可得到滑鼠旋轉的角度。

startingMouseAngle = getAngle(centerX, centerY, event.clientX, event.clientY);
const deltaMouseAngle = currentMouseAngle - startingMouseAngle;
(3)旋轉角度簡化

方塊的最大旋轉角度為360度,所以我們對角度進行取模,保持旋轉角度在360度以內即可。

function normalizeRotation(rotation) {
  if (rotation >= 0) {
    return rotation % 360;
  } else {
    return (rotation % 360) + 360;
  }
}
(4)給方塊設置旋轉角度
rotateDiv.style.transform = `rotate(${newRotation}deg)`;

3、移除旋轉邏輯

滑鼠抬起的時候我們應該將旋轉邏輯給移除,及將滑鼠移動和抬起事件移除。

function stopSpin() {
  window.removeEventListener("mousemove", spin);
  window.removeEventListener("mouseup", stopSpin);
}

4、完整代碼

完整的JavaScrip代碼如下:

const rotateDiv = document.querySelector(".rotate-div");
const rotateIcon = document.querySelector(".rotate-icon");

let startingMouseAngle = 0;
let startingRotation = 0;

rotateIcon.addEventListener("selectstart", function (event) {
  event.preventDefault();
});
rotateIcon.addEventListener("mousedown", function (event) {
  const rect = rotateDiv.getBoundingClientRect();
  const centerX = rect.left + rect.width / 2;
  const centerY = rect.top + rect.height / 2;
  startingMouseAngle = getAngle(centerX, centerY, event.clientX, event.clientY);
  startingRotation = getCurrentRotation();

  window.addEventListener("mousemove", spin);
  window.addEventListener("mouseup", stopSpin);
});

function stopSpin() {
  window.removeEventListener("mousemove", spin);
  window.removeEventListener("mouseup", stopSpin);
}

function spin(event) {
  const rect = rotateDiv.getBoundingClientRect();
  const centerX = rect.left + rect.width / 2;
  const centerY = rect.top + rect.height / 2;
  const currentMouseAngle = getAngle(
    centerX,
    centerY,
    event.clientX,
    event.clientY
  );
  const deltaMouseAngle = currentMouseAngle - startingMouseAngle;
  let newRotation = startingRotation + deltaMouseAngle;
  newRotation = normalizeRotation(newRotation);
  rotateDiv.style.transform = `rotate(${newRotation}deg)`;
}

function normalizeRotation(rotation) {
  if (rotation >= 0) {
    return rotation % 360;
  } else {
    return (rotation % 360) + 360;
  }
}

function getAngle(centerX, centerY, mouseX, mouseY) {
  return Math.atan2(mouseY - centerY, mouseX - centerX) * (180 / Math.PI);
}

function getCurrentRotation() {
  const transformStyle = window
    .getComputedStyle(rotateDiv)
    .getPropertyValue("transform");
  const matrix = new DOMMatrixReadOnly(transformStyle);
  const angle = Math.acos(matrix.a) * (180 / Math.PI);
  return matrix.b < 0 ? -angle : angle;
}

本文轉載於:

https://juejin.cn/post/7290410631655292969

如果對您有所幫助,歡迎您點個關註,我會定時更新技術文檔,大家一起討論學習,一起進步。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 【學習課程】:【【小白入門 通俗易懂】2021韓順平 一周學會Linux】 https://www.bilibili.com/video/BV1Sv411r7vd/?p=14&share_source=copy_web&vd_source=2c07d62293f5003c919b2df9b2e054 ...
  • 在Linux系統中,測試硬碟的性能一般使用fio工具實現,fio是Flexible I/O Tester的縮寫。是一個常受歡迎的、用於測試存儲性能的工具,而且還可以模擬多種不同的I/O模式和工作負載。 一般我們要測試一塊硬碟的性能,一般需要進行隨機寫入測試、隨機讀取測試、順序寫入測試、順序讀取測試和 ...
  • 本章介紹了MongoDB複製集的配置和使用方法,如何初始化和添加節點到複製集,驗證主節點的寫入和從節點的讀取功能。瞭解如何查詢複製集的狀態,包括成員的健康狀況、同步信息和角色等。最後,我們介紹瞭如何配置複製集的安全認證,包括創建用戶和生成keyFile文件,並演示了使用認證信息連接複製集的方式。通過... ...
  • 1 背景 在講述分散式事務的概念之前,我們先來回顧下事務相關的一些概念。 1.1 事務的基本概念 就是一個程式執行單元,裡面的操作要麼全部執行成功,要麼全部執行失敗,不允許只成功一半另外一半執行失敗的事情發生。例如一段事務代碼做了兩次資料庫更新操作,那麼這兩次資料庫操作要麼全部執行成功,要麼全部回滾 ...
  • GaussDB(DWS) 從8.2.1版本後支持三種形式的臨時表:本地臨時表、Volatile臨時表、全局臨時表。本地臨時表特點:表定義和數據都是會話相關,其他會話看不到本會話創建的本地臨時表。 ...
  • 在Oracle資料庫中,一般我們使用sys.user或dba_users去監控/檢查用戶密碼是否快過期,另外,它還能監控新用戶的創建時間、賬號密碼過期、賬號修改時間等,dba_user其實是一個系統視圖,它的數據來源於sys.user$等基礎表。dba_user的定義如下所示: CREATE FOR ...
  • 康師傅YYDS MySQL中只有InnoDB支持事務 1 SHOW ENGINES; 事務基礎知識 事務的ACID特性 原子性(atomicity): 原子性是指事務是一個不可分割的工作單位,要麼全部提交,要麼全部失敗回滾。 一致性(consistency): 根據定義,一致性是指事務執行前後,數據 ...
  • 由於測試環境JED申請比較繁瑣,所以Eone提供了單機版Mysql供用戶使用,近期Eone搭建Mysql5的時候發現莫名被kill了,容器規格是4C8G,磁碟30G。這不科學,之前都是可以的,鏡像沒變,配置沒變,咋就不行了呢,一定不是我的問題,是機器的問題 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...