婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > HTML5 Canvas的事件處理介紹

HTML5 Canvas的事件處理介紹

熱門標簽:外呼系統電話怎么投訴 呼和浩特外呼電銷系統排名 申請400電話流程簡介 pageadm實現地圖標注 邢臺縣地圖標注app 南通數據外呼系統推廣 外呼線穩定線路 阜陽企業外呼系統 地圖標注位置能賺錢嗎

DOM是Web前端領域非常重要的組成部分,不僅在處理HTML元素時會用到DOM,圖形編程也同樣會用到。比如SVG繪圖,各種圖形都是以DOM節點的形式插入到頁面中,這就意味著可以使用DOM方法對圖形進行操作。比如有一個<path id="p1">元素,可以直接用jquery增加click事件$('#p1').click(function(){…})"。然而這種DOM處理方法在HTML5的Canvas里不再適用,Canvas使用的是另外一套機制,無論在Canvas上繪制多少圖形,Canvas都是一個整體,圖形本身實際都是Canvas的一部分,不可單獨獲取,所以也就無法直接給某個圖形增加JavaScript事件。

Canvas的限制

在Canvas里,所有圖形都繪制在幀上,繪制方法不會將繪制好的圖形元素作為一個返回值輸出,js也無法獲取到已經繪制好的圖形元素。比如:


復制代碼
代碼如下:

cvs = document.getElementById('mycanvas');
ctx = canvas.getContext('2d');
theRect = ctx.rect(10, 10, 100, 100);
ctx.stroke();
console.log(theRect); //undefined

這段代碼在canvas標簽里繪制了一個矩形,首先可以看到繪制圖形的rect方法沒有返回值。如果打開瀏覽器的開發者工具,還可以看到canvas標簽內部沒有增加任何內容,而在js里獲取到的canvas元素以及當前的上下文,也都沒有任何表示新增圖形的內容。

所以,前端常用的dom方法在canvas里是不適用的。比如點擊上面Canvas里的矩形,實際點擊的是整個Canvas元素。

給Canvas元素綁定事件

由于事件只能達到Canvas元素這一層,所以,如果想進一步深入,識別點擊發生在Canvas內部的哪一個圖形上,就需要增加代碼來進行處理。基本思路是:給Canvas元素綁定事件,當事件發生時,檢查事件對象的位置,然后檢查哪些圖形覆蓋了該位置。比如上面的例子里畫過一個矩形,該矩形覆蓋x軸10-110、y軸10-110的范圍。只要鼠標點擊在這個范圍里,就可以視為點擊了該矩形,也就可以手動觸發矩形需要處理的點擊事件。思路其實比較簡單,但是實現起來還是稍微有點復雜。不僅要考慮這個判斷過程的效率,有些地方還需要重新判斷事件類型,設置要重新定義一個Canvas內部的捕獲和冒泡機制。

首先要做的,是給Canvas元素綁定事件,比如Canvas內部某個圖形要綁定點擊事件,就需要通過Canvas元素代理該事件:


復制代碼
代碼如下:

cvs = document.getElementById('mycanvas');
cvs.addEventListener('click', function(e){
//...
}, false);

接下來需要判斷事件對象發生的位置,事件對象e的layerX和layerY屬性表示Canvas內部坐標系中的坐標。但是這個屬性Opera不支持,Safari也打算移除,所以要做一些兼容寫法:


復制代碼
代碼如下:

function getEventPosition(ev){
var x, y;
if (ev.layerX || ev.layerX == 0) {
x = ev.layerX;
y = ev.layerY;
} else if (ev.offsetX || ev.offsetX == 0) { // Opera
x = ev.offsetX;
y = ev.offsetY;
}
return {x: x, y: y};
}
//注:使用上面這個函數,需要給Canvas元素的position設為absolute。

現在有了事件對象的坐標位置,下面就要判斷Canvas里的圖形,有哪些覆蓋了這個坐標。

isPointInPath方法

Canvas的isPointInPath方法可以判斷當前上下文的圖形是否覆蓋了某個坐標,比如:


復制代碼
代碼如下:

cvs = document.getElementById('mycanvas');
ctx = canvas.getContext('2d');
ctx.rect(10, 10, 100, 100);
ctx.stroke();
ctx.isPointInPath(50, 50); //true
ctx.isPointInPath(5, 5); //false

接下來增加一個事件判斷,就可以判斷一個點擊事件是否發生在矩形上:


復制代碼
代碼如下:

cvs.addEventListener('click', function(e){
p = getEventPosition(e);
if(ctx.isPointInPath(p.x, p.y)){
//點擊了矩形
}
}, false);

以上就是處理Canvas事件的基本方法,但是上面的代碼還有局限,由于isPointInPath方法僅判斷當前上下文環境中的路徑,所以當Canvas里已經繪制了多個圖形時,僅能以最后一個圖形的上下文環境來判斷事件,比如:

復制代碼
代碼如下:

cvs = document.getElementById('mycanvas');
ctx = canvas.getContext('2d');
ctx.beginPath();
ctx.rect(10, 10, 100, 100);
ctx.stroke();
ctx.isPointInPath(20, 20); //true
ctx.beginPath();
ctx.rect(110, 110, 100, 100);
ctx.stroke();
ctx.isPointInPath(150, 150); //true
ctx.isPointInPath(20, 20); //false

從上面這段代碼可以看到,isPointInPath方法僅能識別當前上下文環境里的圖形路徑,而之前繪制的路徑,無法回溯判斷。這種問題的解決方法是:當點擊事件發生時,重繪所有圖形,每繪制一個就使用isPointInPath方法,判斷事件坐標是否在該圖形覆蓋范圍內。

循環重繪和事件冒泡

為了實現循環重繪,所以就要將圖形的基本參數事先保存下來:

復制代碼
代碼如下:

arr = [
{x:10, y:10, width:100, height:100},
{x:110, y:110, width:100, height:100}
];

cvs = document.getElementById('mycanvas');
ctx = canvas.getContext('2d');

draw();

function draw(){
ctx.clearRech(0, 0, cvs.width, cvs.height);
arr.forEach(function(v){
ctx.beginPath();
ctx.rect(v.x, v.y, v.width, v.height);
ctx.stroke();
});
}

上面的代碼事先將兩個矩形的基本參數保存下來,每次調用draw方法,就會循環調用這些基本參數,用于繪制兩個矩形。這里還使用了clearRect方法,用于在重繪時清空畫布。接下來要做的是增加事件代理,以及在重繪時對每一個上下文環境使用isPointInPath方法:


復制代碼
代碼如下:

cvs.addEventListener('click', function(e){
p = getEventPosition(e);
draw(p);
}, false);

事件發生時,將事件對象的坐標傳給draw方法處理。這里還需要對draw方法做一些小改動:


復制代碼
代碼如下:

function draw(p){
var who = [];
ctx.clearRech(0, 0, cvs.width, cvs.height);
arr.forEach(function(v, i){
ctx.beginPath();
ctx.rect(v.x, v.y, v.width, v.height);
ctx.stroke();
if(p && ctx.isPointInPath(p.x, p.y)){
//如果傳入了事件坐標,就用isPointInPath判斷一下
//如果當前環境覆蓋了該坐標,就將當前環境的index值放到數組里
who.push(i);
}
});
//根據數組中的index值,可以到arr數組中找到相應的元素。
return who;
}

在上面代碼中,點擊事件發生時draw方法會執行一次重繪,并在重繪過程中檢查每一個圖形是否覆蓋了事件坐標,如果判斷為真,則視為點擊了該圖形,并將該圖形的index值放入數組,最后將數組作為draw方法的返回值。在這種處理機制下,如果Canvas里有N個圖形,它們有一部分是重疊的,而點擊事件恰巧發生在這個重疊區域上,那么draw方法的返回數組里會有N個成員。這時就有點類似事件冒泡的情況,數組的最后一個成員處于Canvas最上層,而第一個成員則在最下層,我們可以視為最上層的成員是e.target,而其他成員則是冒泡過程中傳遞到的節點。當然這只是最簡單的一種處理方法,如果真要模擬DOM處理,還要給圖形設置父子級關系。

以上就是Canvas事件處理的基本方法。在實際運用時,如何緩存圖形參數,如何進行循環重繪,以及如何處理事件冒泡,都還需要根據實際情況花一些心思去處理。另外,click是一個比較好處理的事件,相對麻煩的是mouseover、mouseout和mousemove這些事件,由于鼠標一旦進入Canvas元素,始終發生的都是mousemove事件,所以如果要給某個圖形單獨設置mouseover或mouseout,還需要記錄鼠標移動的路線,給圖形設置進出狀態。由于處理的步驟變得復雜起來,必須對性能問題提高關注。

標簽:楊凌 蚌埠 辛集 撫順 內蒙古 鶴崗 黃山 德州

巨人網絡通訊聲明:本文標題《HTML5 Canvas的事件處理介紹》,本文關鍵詞  HTML5,Canvas,的,事件,處理,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《HTML5 Canvas的事件處理介紹》相關的同類信息!
  • 本頁收集關于HTML5 Canvas的事件處理介紹的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    国产精品91一区二区| 国产精品美女久久久久久| 在线亚洲欧美专区二区| 欧美性猛交xxxx黑人交| 国产三级精品三级| 国产精品亚洲专一区二区三区| 欧美日韩日日夜夜| 亚洲电影第三页| 日本高清成人免费播放| 亚洲日本护士毛茸茸| 91影院在线免费观看| 中文字幕一区av| 91福利社在线观看| 亚洲电影视频在线| 欧美一级搡bbbb搡bbbb| 奇米综合一区二区三区精品视频| 欧美性受xxxx| 国产三级精品在线| 成人av动漫网站| 亚洲人成精品久久久久久| 在线观看精品一区| 美女一区二区视频| 久久婷婷久久一区二区三区| 成人妖精视频yjsp地址| 午夜av一区二区三区| 欧美一区二区成人| 粉嫩av一区二区三区粉嫩| 亚洲私人黄色宅男| 欧美一级二级三级乱码| 国产成人精品影视| 美国av一区二区| 国产精品久久久久国产精品日日| 91久久久免费一区二区| 九九国产精品视频| 亚洲欧美二区三区| 欧美丰满高潮xxxx喷水动漫 | 免费欧美日韩国产三级电影| 久久久噜噜噜久久人人看| a亚洲天堂av| 毛片av中文字幕一区二区| 中文字幕五月欧美| 日本一区二区三区电影| 2021国产精品久久精品| av电影在线观看一区| 国产凹凸在线观看一区二区| 日韩va亚洲va欧美va久久| 1024成人网| 精品日韩一区二区三区| 日本欧美肥老太交大片| 久久er99精品| 午夜精品福利视频网站| 亚洲电影一区二区| 亚洲午夜国产一区99re久久| 中文字幕一区不卡| 日韩一区在线播放| 亚洲免费在线看| 国产日韩欧美精品一区| 欧美亚洲国产一区在线观看网站| 国产成人久久精品77777最新版本| 亚洲五月六月丁香激情| 1024亚洲合集| 婷婷一区二区三区| 国产一区二区h| 91久久精品日日躁夜夜躁欧美| 777久久久精品| 国产精品久久网站| 美女诱惑一区二区| 一本久道久久综合中文字幕| 欧美一级片免费看| 国产精品久久久久久福利一牛影视| 亚洲美女偷拍久久| 国产成人精品综合在线观看| 欧美日韩精品一区二区三区蜜桃| 久久久亚洲欧洲日产国码αv| 国产精品欧美精品| 国产一区二区在线电影| 欧美性色黄大片| 日韩毛片高清在线播放| 成人av网址在线| 日韩av不卡在线观看| 成人亚洲精品久久久久软件| 国产精品麻豆网站| 亚洲成人动漫在线观看| 色综合久久天天| 亚洲欧美日韩国产中文在线| 成人av集中营| 日日骚欧美日韩| 欧美一级日韩免费不卡| 国产一区二区三区黄视频| 久久久久久久免费视频了| 国产一区二区三区在线观看免费 | 91精品国产手机| 国产一区在线观看视频| 中文字幕精品在线不卡| 不卡视频在线观看| 亚洲最新视频在线观看| 7777精品伊人久久久大香线蕉最新版| 一区二区在线看| 日韩欧美国产综合在线一区二区三区| 麻豆免费精品视频| 亚洲激情六月丁香| 国产偷v国产偷v亚洲高清| 在线观看亚洲一区| 国产不卡在线播放| 久久99国产乱子伦精品免费| 亚洲欧美激情一区二区| 国产亚洲美州欧州综合国| 精品福利在线导航| 91极品视觉盛宴| 国产精品12区| 老司机免费视频一区二区| 亚洲一区在线观看免费 | 高清国产午夜精品久久久久久| 一区二区三区在线视频免费 | 亚洲国产精品一区二区尤物区| 蜜乳av一区二区| 亚洲欧洲综合另类| 久久久久一区二区三区四区| 日韩午夜激情av| 91精品国产欧美日韩| 91精品国产品国语在线不卡| 日韩三级精品电影久久久 | 成人性视频网站| 国产99久久久国产精品潘金网站| 日韩国产欧美在线播放| 午夜精品在线视频一区| 亚洲精品第1页| 亚洲成a人v欧美综合天堂| 午夜久久久久久久久| 美国十次了思思久久精品导航| 麻豆一区二区三| 北条麻妃一区二区三区| 99免费精品在线观看| 国产美女精品在线| 一本到不卡免费一区二区| 在线一区二区三区做爰视频网站| 国产99久久精品| 7777精品伊人久久久大香线蕉的 | 成人福利视频网站| 欧美日韩在线直播| 欧美成人video| 亚洲成人福利片| 波多野结衣精品在线| 欧美久久久一区| 亚洲欧美激情在线| 国产乱子伦视频一区二区三区| 欧美自拍偷拍一区| 久久亚洲二区三区| 日韩在线一二三区| 色av成人天堂桃色av| 中文字幕免费不卡在线| 激情成人午夜视频| 久久一留热品黄| 久久精品国产亚洲一区二区三区| 在线亚洲一区二区| 亚洲欧美影音先锋| 成人深夜视频在线观看| 中文字幕五月欧美| 欧美在线视频日韩| 三级成人在线视频| 2欧美一区二区三区在线观看视频| 亚洲va韩国va欧美va| 欧美中文字幕一区二区三区亚洲| 亚洲视频免费看| 欧美日韩大陆在线| 男女男精品网站| 欧美国产97人人爽人人喊| 成人免费三级在线| 亚洲美女屁股眼交| 欧美一区二区精品| 成人午夜免费视频| 亚洲精品精品亚洲| 蜜桃视频在线观看一区二区| 91精品婷婷国产综合久久竹菊| 人人狠狠综合久久亚洲| 国产日韩精品一区| 欧美亚洲一区二区在线观看| 看片的网站亚洲| 欧美成人r级一区二区三区| 五月天视频一区| 亚洲一区在线观看网站| 99精品黄色片免费大全| 亚洲男人电影天堂| 欧美日韩黄视频| 成人午夜在线播放| 日韩黄色小视频| 最新成人av在线| 久久久久久97三级| 欧美不卡在线视频| 91黄色免费观看| 91福利在线免费观看| 国产黄人亚洲片| 精品一区二区三区香蕉蜜桃| 国产精品不卡在线| 欧美日韩国产片| 色综合色综合色综合| 91浏览器打开| 色婷婷av一区二区三区大白胸| 91小宝寻花一区二区三区| 99这里只有久久精品视频|