HTML5 Canvas的事件处理介绍

   HTML5 Canvas的事件处理介绍

       DOM是Web前端领域非常重要的组成部分,不仅在处理HTML元素时会用到DOM,图形编程也同样会用到。比如SVG绘图,各种图形都是以DOM节点的形式插入到页面中,这就意味着可以使用DOM方法对图形进行操作。比如有一个
元素,可以直接用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,还需要记录鼠标移动的路线,给图形设置进出状态。由于处理的步骤变得复杂起来,必须对性能问题提高关注。

时间: 2024-08-04 07:12:26

HTML5 Canvas的事件处理介绍的相关文章

HTML5 Canvas绘图库介绍(Fabric.js)

HTML5 Canvas绘图库介绍 Fabric.js是一个用于简化HTML5 Canvas标签操作的JS框架. 这是一个canvas元素上的互动对象模型.它还是一个SVG-to-canvas解析器.支持以下浏览器: 1.Firefox 2+ 2.Safari 3+ 3.Opera 9.64+ 4.Chrome (all versions should work) 5.IE9+ 如果我们要在<canvas>上绘制复杂的图形,就需要学习各种几何知识. 好在网上有现成的绘图库供我们使用,不但可以轻

Day08 - HTML5 Canvas 实现彩虹画笔绘画板指南

Day08 - HTML5 Canvas 实现彩虹画笔绘画板指南 作者:liyuechun简介:JavaScript30 是 Wes Bos 推出的一个 30 天挑战.项目免费提供了 30 个视频教程.30 个挑战的起始文档和 30 个挑战解决方案源代码.目的是帮助人们用纯 JavaScript 来写东西,不借助框架和库,也不使用编译器和引用.现在你看到的是这系列指南的第 8 篇.完整中文版指南及视频教程在 从零到壹全栈部落. 项目效果 用 HTML5 中的 Canvas 的路径绘制实现一个绘画

HTML5 Canvas捕获用于游戏开发的键盘、鼠标和触摸事件

学习如何处理键盘和鼠标事件,如何阻止 Web 浏览器的默认事件行为,以及如何向游戏对象的某种逻辑表示传播事件.此外,还将学习如何处理 iPhone 和 iPad 等移动设备上与设备无关的(device-agnostic)输入. 令拥有 Flash 或 Silverlight 背景的开发人员感到惊讶的是,为 HTML5 Canvas 编写的应用程序在处理用户输入方面并没有什么特立独行之处.实质上,从启用了 JavaScript 的 Web 浏览器诞生之初开始,HTML 用户输入就涉及到使用内置于浏

8个经典炫酷的HTML5 Canvas动画欣赏

HTML5非常强大,尤其是Canvas技术的应用,让HTML5几乎可以完成所有Flash能完成的效果.本文精选了8个经典炫酷的HTML5 Canvas动画欣赏,每一个都提供全部的源代码,希望对你有所帮助. 1.HTML5 Canvas可拖动的弹性大树摇摆动画 今天让我们继续来分享一个炫酷的HTML5动画,它是一款基于HTML5 Canvas的大树摇摆动画,这款HTML5动画的特点是我们可以拖拽树枝,从而让整棵树摇摆起来,这样就真实地模拟了大树从摇摆到静止的整个过程,相当逼真. 在线演示     

Web Chart入门(1) Web端图形绘制SVG,VML, HTML5 Canvas技术比较

先介绍一下矢量图的概念: 矢量图使用直线和曲线来描述图形,这些图形的元素是一些点.线.矩形.多边形.圆和弧线等等,它们都是通过数学公式计算获得的.例如一幅花的矢量图形实际上是由线段形成外框轮廓,由外框的颜色以及外框所封闭的颜色决定花显示出的颜色. SVG,VML, HTML5 Canvas  这三个技术绘制的都是矢量图. 只是由不同的厂商开发出来的. 要达成的效果基本是一样的. 1. VML 全称Vector Markup Language(矢量可标记语言). 是微软1999年9月附带IE5.0

《HTML5 canvas开发详解(第2版)》——1.11 动画版本的Hello World

1.11 动画版本的Hello World "Hello World"和"猜字母"本身都是不错的示例,但是它们都没能回答出"为什么"--究竟为什么要使用HTML5 Canvas?自创立以来,静态的图像和文字就是HTML的领域,那么画布的不同之处在哪里呢?要回答这个问题,需要创建第二个"Hello World"示例.这个示例将介绍画布与HTML上的其他显示方式的最大不同之处:动画.在这个示例中,将为"Hello Wor

《HTML5 Canvas游戏开发实战》——导读

前言 为什么要写这本书 并非计算机专业的我,却最终走上了编程之路,并写了这样一本书,为什么呢?其实一切都是因为和游戏结了缘. 小时候我非常喜欢玩游戏,为了玩游戏和小伙伴们干过不少调皮捣蛋的事情.初中为了得到自己的第一台游戏机,和父亲打赌,破天荒拿了全班第一名.当然有了游戏机的相伴,从那以后就再也没有拿过第一名了.因为数学上较有优势,所以读大学时选择的是数学专业,没有选读计算机让后来做开发的我多少感到有些遗憾.和其他人一样,大学是真正改变我人生的时期,第一次有了电脑,第一次从室友嘴里得知QQ为何物

使用 HTML5 canvas 进行 Web 绘图

使用 HTML5 canvas 进行 Web 绘图 新的 HTML5 规范旨在帮助开发人员更轻松的编写出各类 Web 应用,以顺应当前 SaaS,云计算以及 RIA 等技术的最新趋势.在 HTML5 得以广泛推广之前,开发人员通常使用 SVG,VML 等技术进行 Web 绘图操作,但这些基于 XML 的绘图语言声明式的绘图方式并不能满足复杂绘图操作在性能上的需求,比如 Web 游戏所需要的像素级别的绘图能力.HTML5 canvas 元素的出现填补了这种不足,开发人员可以使用 JavaScrip

如何使用 HTML5 Canvas 制作水波纹效果

原文:如何使用 HTML5 Canvas 制作水波纹效果 今天,我们继续分享 JavaScript 实现的效果例子,这篇文章会介绍使用 JavaScript 实现水波纹效果.水波效果以图片为背景,点击图片任意位置都会触发.有时候,我们使用普通的 Javascript 就可以创建一个很有趣的解决功能.     在线演示      源码下载   Step 1. HTML 和以前一样,首先是 HTML 代码: <!DOCTYPE html> <html> <head> <