亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > HTML > 正文

Canvas高級路徑操作之拖拽對象的實現

2024-08-26 00:21:39
字體:
來源:轉載
供稿:網友

為了追蹤所畫內容,諸如畫圖應用程序、計算機輔助設計系統(computer-aided design system 簡稱CAD系統)以及游戲等許多應用程序,都會維護一份包含當前顯示對象的列表。通常來說,這些應用程序都允許用戶對當前顯示在屏幕上的物體進行操作。比方說,在CAD應用程序中,我們可以對設計中的元素進行選擇、移動、縮放等操作

——《HTML5 Canvas核心技術》

在Canvas中 實現拖拽 也同樣如此,Canvas提供了一個名為 isPointInPath(x, y) 的API,判斷 點(x, y) 是否在路徑之中。如果在路徑之中,則返回true。于是我們可以有如下思路:

維護一個可以描述各個路徑的 數組 ,通過 ispointInPath(x, y) 判斷點擊位置是否在某一個路徑之中,如果在此路徑之中,選中此路徑,進行操作(移動、縮放等),再繪制圖形

此文我以 多邊形拖拽為例進行說明 ,Demo如下(后面的印子是錄屏軟件的原因:japanese_ogre:):

 

 
Canvas,高級路徑,拖拽對象 

 

CodePen打開

Demo中的多邊形如何繪制之前做過總結,不再贅述:ghost::Canvas多邊形繪制

思路說明

下圖給了大致的說明及偽代碼,思路并不難,但有部分細節需要處理

 

 
Canvas,高級路徑,拖拽對象 

 

代碼結構說明

此處列舉代碼結構及標注其思路,更詳細的代碼注釋已在CodePen之中

因為本文重點在拖拽(drag),所以對繪圖部分描述會較少

//繪制多邊形路徑函數function drawPolygonPath//多邊形類定義class Polygon{    ...}//根據點擊事件返回在canvas中的位置function positoinInCanvas//獲取兩點間直線距離function getDistance//開始階段,記錄拖拽對象canvas.onmousedown//拖拽階段,畫路徑,描邊canvas.onmousemove//結束階段,更新拖拽對象位置canvas.onmouseup

關鍵部分說明

接下來開始代碼中的關鍵部分及細節處理

如何維護拖拽對象數組

在程序初始化時,我們定義一個polygonArray數組

polygonArray = []

在每次畫一個新的多邊形之后,都會new一個多邊形對象推入數組中進行維護

const polygon = new Polygon(mouseStart.get('x'), mouseStart.get('y'), sideNum, radius);polygonArray.push(polygon);//記錄路徑對象

在后續點擊操作時,需要根據對應信息確定點擊位置是否在路徑之中

點擊時,如何選取要拖拽的對象

首先獲取點擊時在 canvas中 的對應位置,我的代碼用 mouseStart 記錄 x 及 y 
接著遍歷 polygonArray 中的 polygon ,遍歷中調用 polygon.createPath() ,通過 isPointInPath() 判斷點擊位置是否有路徑,有的話 draggingPolygon = polygon 結束函數

const pos = positionInCanvas(e, canvasLeft, canvasTop);//獲取在canvas中的像素位置//記錄鼠標起始點smouseStart.set('x', pos.x);mouseStart.set('y', pos.y);...for (let polygon of polygonArray) {                polygon.createPath();                if (ctx.isPointInPath(mouseStart.get('x'), mouseStart.get('y'))) {                    draggingPolygon = polygon;                    return;                }            }

拖拽時的計算

這部分要完全理解推薦大家根據Demo中兩個 console.log(draggingPolygon) 及代碼進行調試,因為我們是在 mousemove 階段,這個階段觸發函數非常頻繁

我盡量用語言表達清楚

首先計算 move 時與 mouseStart 的距離,記為diff,有x軸上的 offsetX ,也有y軸上的 offsetY

const    pos = positionInCanvas(e, canvasLeft, canvasTop),    diff = new Map([      ['offsetX', pos.x - mouseStart.get('x')],      ['offsetY', pos.y - mouseStart.get('y')]    ]);

接著記錄當前拖拽對象的 centerX 及 centerY ,記為temp

let    tempCenterX = draggingPolygon.centerX,    tempCenterY = draggingPolygon.centerY;

這里就是難理解的點,為什么要記錄?繼續往下看,后面會使用到。

根據 diff 中的offset,設置draggingPolygon新的中心位置

draggingPolygon.centerX += diff.get('offsetX');draggingPolygon.centerY += diff.get('offsetY');

接著清空畫布進行繪制新的路徑和描邊

ctx.clearRect(0, 0, canvas.width, canvas.height);for (let polygon of polygonArray) {    drawPolygonPath(polygon.sideNum,         polygon.radius,         polygon.centerX,         polygon.centerY, ctx);    ctx.stroke();}

最后使用到上文中的 tempCenterX 與 tempCenterY :

draggingPolygon.centerX = tempCenterX;draggingPolygon.centerY = tempCenterY;

為什么需要這么做呢?

因為我們的拖拽是 基于多邊形的原位置 ,而 mousemove 階段并 不能確定函數的最終位置 ,如果這時沒有復原的話,會出現 "漂移" ,我把這兩行代碼注釋掉,效果如下:

 

 
Canvas,高級路徑,拖拽對象 

 

如果我沒說清楚,墻裂推薦大家對代碼進行修改和調試

拖拽后的處理

拖拽完成后是處于 mouseup 階段,此時我們已經確定dragginPolygon的最終位置,進行更新即可,最后置為null,排除 在沒有拖拽多邊形情況下,鼠標在畫布上移動觸發對應代碼

const    pos = positionInCanvas(e, canvasLeft, canvasTop),    offsetMap = new Map([        ['offsetX', pos.x - mouseStart.get('x')],        ['offsetY', pos.y - mouseStart.get('y')]    ]);draggingPolygon.centerX += offsetMap.get('offsetX');draggingPolygon.centerY += offsetMap.get('offsetY');draggingPolygon = null;

結語

其實這個功能實現并不難,關鍵是了解一個概念:通過維護當前顯示對象的列表及isPointInPath進行判斷來實現追蹤
最后歡迎大家交流學學習

參考資料

《HTML5 Canvas核心技術》

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到HTML教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美精品少妇videofree| 91成人免费观看网站| 久久99热精品| 成人妇女淫片aaaa视频| 97超碰蝌蚪网人人做人人爽| 国产亚洲欧洲高清一区| 亚洲第一福利在线观看| 97久久久免费福利网址| 亚洲欧洲日产国码av系列天堂| 17婷婷久久www| 福利一区福利二区微拍刺激| 成人h猎奇视频网站| 国产不卡一区二区在线播放| 国产v综合v亚洲欧美久久| 国产免费一区二区三区香蕉精| 日韩av影视综合网| 性色av一区二区三区免费| 国产精品福利在线观看| 美日韩丰满少妇在线观看| 亚洲国产婷婷香蕉久久久久久| 91av在线免费观看| 日韩最新在线视频| 日韩av网站大全| 91成品人片a无限观看| 国产视频福利一区| 久久精品国产视频| 亚洲国产日韩欧美在线图片| 色青青草原桃花久久综合| 97国产在线视频| 97人人模人人爽人人喊中文字| 日本亚洲欧美成人| www.国产一区| 成人免费看片视频| 欧美黄色成人网| 色黄久久久久久| 国产精品成人av在线| 亚洲天堂av在线免费观看| 国产亚洲欧美视频| 日韩成人激情在线| 在线成人激情黄色| 奇米四色中文综合久久| 亚洲精品美女视频| 日韩免费观看av| 亚洲欧美国产精品va在线观看| 国产一区二区丝袜高跟鞋图片| 国产精品久久久久久久av大片| 国产精品免费小视频| 国产精品看片资源| 欧美一区二区.| 欧美富婆性猛交| 色婷婷av一区二区三区久久| 欧美日韩裸体免费视频| 在线视频欧美日韩精品| 国外日韩电影在线观看| 日本中文字幕不卡免费| 4438全国成人免费| 亚洲电影在线看| 久久久亚洲天堂| 国产成人精品a视频一区www| 欧美一级大片在线观看| 欧美午夜精品久久久久久浪潮| 国产精品色婷婷视频| 夜夜嗨av色综合久久久综合网| 欧美亚洲一区在线| 亚洲精品久久久久久久久久久久| 77777亚洲午夜久久多人| 色999日韩欧美国产| 成人欧美一区二区三区黑人| 国产精品综合网站| 亚洲黄在线观看| 亚洲理论在线a中文字幕| 亚洲电影免费观看高清完整版在线观看| 亚洲天堂久久av| 久久亚洲春色中文字幕| 欧美视频在线观看 亚洲欧| 美日韩精品视频免费看| 国产91精品在线播放| 热门国产精品亚洲第一区在线| 国产精品入口免费视频一| 国产精品1234| 亲子乱一区二区三区电影| 国产欧美亚洲视频| 性色av一区二区三区在线观看| 欧美成人精品三级在线观看| 91在线免费观看网站| 亚洲影院色在线观看免费| 色婷婷综合久久久久| 亚洲美女喷白浆| 国产精品视频精品| 色www亚洲国产张柏芝| 精品五月天久久| 亚洲区bt下载| 欧美日韩国产精品一区二区不卡中文| 久久精品99久久久香蕉| 中国日韩欧美久久久久久久久| 国产精品偷伦免费视频观看的| 日韩中文在线中文网在线观看| 国a精品视频大全| 欧美激情中文网| 久国内精品在线| 国产精品一二三在线| 久久久久久久影院| 欧美黄网免费在线观看| 色偷偷噜噜噜亚洲男人| 久久久噜久噜久久综合| 亚洲第一精品夜夜躁人人躁| 色哟哟亚洲精品一区二区| xxxx欧美18另类的高清| 日韩电影在线观看免费| 欧美精品videofree1080p| 中文字幕欧美视频在线| 精品国产31久久久久久| 色悠悠久久88| 欧美成人黑人xx视频免费观看| 亚洲欧美日韩视频一区| 欧美一级电影久久| 在线国产精品视频| 黑人巨大精品欧美一区二区一视频| 92福利视频午夜1000合集在线观看| 亚洲国产欧美精品| 最近2019好看的中文字幕免费| 亚洲国产美女久久久久| 欧美国产视频一区二区| 欧美专区日韩视频| 深夜福利一区二区| 欧美成人国产va精品日本一级| 久久99久久久久久久噜噜| 亚洲色图18p| 日韩精品视频在线播放| 亚洲性av在线| 国产国语刺激对白av不卡| 成人日韩在线电影| 欧美成年人网站| 亚洲欧美日韩区| 在线日韩av观看| 中文国产亚洲喷潮| 国产有码在线一区二区视频| 精品久久久香蕉免费精品视频| 亚洲一级黄色片| 国产一区av在线| 在线日韩日本国产亚洲| 欧美一二三视频| 深夜福利亚洲导航| 久久精品亚洲国产| 97热在线精品视频在线观看| 7m第一福利500精品视频| 久久国产精品亚洲| 91精品视频大全| 成人国产亚洲精品a区天堂华泰| 亚洲欧美三级在线| 精品人伦一区二区三区蜜桃网站| 日本在线观看天堂男亚洲| 尤物yw午夜国产精品视频明星| 青青草原一区二区| 欧美日本亚洲视频| 国产精品激情自拍| 热草久综合在线| 欧美午夜精品久久久久久久| 日韩欧美999| 欧美放荡办公室videos4k| 国产成人一区二区三区| 精品一区二区三区三区| 久久精品国产电影| 一区二区三区四区精品|