国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > HTML > 正文

Canvas高級(jí)路徑操作之拖拽對(duì)象的實(shí)現(xiàn)

2024-08-26 00:21:39
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

為了追蹤所畫內(nèi)容,諸如畫圖應(yīng)用程序、計(jì)算機(jī)輔助設(shè)計(jì)系統(tǒng)(computer-aided design system 簡(jiǎn)稱CAD系統(tǒng))以及游戲等許多應(yīng)用程序,都會(huì)維護(hù)一份包含當(dāng)前顯示對(duì)象的列表。通常來(lái)說(shuō),這些應(yīng)用程序都允許用戶對(duì)當(dāng)前顯示在屏幕上的物體進(jìn)行操作。比方說(shuō),在CAD應(yīng)用程序中,我們可以對(duì)設(shè)計(jì)中的元素進(jìn)行選擇、移動(dòng)、縮放等操作

——《HTML5 Canvas核心技術(shù)》

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

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

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

 

 
Canvas,高級(jí)路徑,拖拽對(duì)象 

 

CodePen打開

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

思路說(shuō)明

下圖給了大致的說(shuō)明及偽代碼,思路并不難,但有部分細(xì)節(jié)需要處理

 

 
Canvas,高級(jí)路徑,拖拽對(duì)象 

 

代碼結(jié)構(gòu)說(shuō)明

此處列舉代碼結(jié)構(gòu)及標(biāo)注其思路,更詳細(xì)的代碼注釋已在CodePen之中

因?yàn)楸疚闹攸c(diǎn)在拖拽(drag),所以對(duì)繪圖部分描述會(huì)較少

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

關(guān)鍵部分說(shuō)明

接下來(lái)開始代碼中的關(guān)鍵部分及細(xì)節(jié)處理

如何維護(hù)拖拽對(duì)象數(shù)組

在程序初始化時(shí),我們定義一個(gè)polygonArray數(shù)組

polygonArray = []

在每次畫一個(gè)新的多邊形之后,都會(huì)new一個(gè)多邊形對(duì)象推入數(shù)組中進(jìn)行維護(hù)

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

在后續(xù)點(diǎn)擊操作時(shí),需要根據(jù)對(duì)應(yīng)信息確定點(diǎn)擊位置是否在路徑之中

點(diǎn)擊時(shí),如何選取要拖拽的對(duì)象

首先獲取點(diǎn)擊時(shí)在 canvas中 的對(duì)應(yīng)位置,我的代碼用 mouseStart 記錄 x 及 y 
接著遍歷 polygonArray 中的 polygon ,遍歷中調(diào)用 polygon.createPath() ,通過 isPointInPath() 判斷點(diǎn)擊位置是否有路徑,有的話 draggingPolygon = polygon 結(jié)束函數(shù)

const pos = positionInCanvas(e, canvasLeft, canvasTop);//獲取在canvas中的像素位置//記錄鼠標(biāo)起始點(diǎn)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;                }            }

拖拽時(shí)的計(jì)算

這部分要完全理解推薦大家根據(jù)Demo中兩個(gè) console.log(draggingPolygon) 及代碼進(jìn)行調(diào)試,因?yàn)槲覀兪窃?nbsp;mousemove 階段,這個(gè)階段觸發(fā)函數(shù)非常頻繁

我盡量用語(yǔ)言表達(dá)清楚

首先計(jì)算 move 時(shí)與 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')]    ]);

接著記錄當(dāng)前拖拽對(duì)象的 centerX 及 centerY ,記為temp

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

這里就是難理解的點(diǎn),為什么要記錄?繼續(xù)往下看,后面會(huì)使用到。

根據(jù) diff 中的offset,設(shè)置draggingPolygon新的中心位置

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

接著清空畫布進(jìn)行繪制新的路徑和描邊

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;

為什么需要這么做呢?

因?yàn)槲覀兊耐献?nbsp;基于多邊形的原位置 ,而 mousemove 階段并 不能確定函數(shù)的最終位置 ,如果這時(shí)沒有復(fù)原的話,會(huì)出現(xiàn) "漂移" ,我把這兩行代碼注釋掉,效果如下:

 

 
Canvas,高級(jí)路徑,拖拽對(duì)象 

 

如果我沒說(shuō)清楚,墻裂推薦大家對(duì)代碼進(jìn)行修改和調(diào)試

拖拽后的處理

拖拽完成后是處于 mouseup 階段,此時(shí)我們已經(jīng)確定dragginPolygon的最終位置,進(jìn)行更新即可,最后置為null,排除 在沒有拖拽多邊形情況下,鼠標(biāo)在畫布上移動(dòng)觸發(fā)對(duì)應(yīng)代碼

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;

結(jié)語(yǔ)

其實(shí)這個(gè)功能實(shí)現(xiàn)并不難,關(guān)鍵是了解一個(gè)概念:通過維護(hù)當(dāng)前顯示對(duì)象的列表及isPointInPath進(jìn)行判斷來(lái)實(shí)現(xiàn)追蹤
最后歡迎大家交流學(xué)學(xué)習(xí)

參考資料

《HTML5 Canvas核心技術(shù)》

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持VeVb武林網(wǎng)。


注:相關(guān)教程知識(shí)閱讀請(qǐng)移步到HTML教程頻道。
發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 西乡县| 玛沁县| 桓仁| 都匀市| 永仁县| 崇州市| 原阳县| 安吉县| 扎兰屯市| 马公市| 峨眉山市| 罗定市| 忻州市| 密山市| 彰化县| 综艺| 东明县| 墨脱县| 岫岩| 九寨沟县| 壤塘县| 涡阳县| 吉林市| 北宁市| 乌恰县| 武安市| 临泽县| 日照市| 手游| 石门县| 称多县| 上高县| 牟定县| 德化县| 普定县| 息烽县| 长垣县| 沁源县| 伊宁县| 玉林市| 巴彦淖尔市|