简介
在本文中,我将介绍如何使用 HTML5 canvas 元素创建、编辑、打开和导出图片。我还将介绍几种与此技术相关的开源工具,并提供一些有关如何将这些技术应用于现有 Web 应用的技巧。
检查画布支持情况
首先,请检查您的浏览器是否完全支持 HTML5 画布。一种简单的方法是使用 Modernizr 检查是否支持某项功能:
if (Modernizr.canvas) {
  // Browser supports native HTML5 canvas.
} else {
  // Fallback to another solution, such as Flash, static image, download link, and so on.
}
创建画布元素并以二进制或数据 URI 格式导入图片
首先,您需要在网页中添加一个画布元素。 使用 JavaScript,您可以执行以下操作:
var ctx = document.getElementById('new_canvas').getContext('2d');
var img = new Image();
img.src = "html5.gif"
img.onload = function () {
   ctx.drawImage(img,0,0);
}
在此代码中,第一步是获取 2D 上下文,以便我们访问用于定义所有绘制方法和属性的 API。接下来,我们创建一个 Image 对象,并将 src 属性设置为二进制图片的位置。图片加载后,我们使用 drawImage() 方法将图片导入画布元素。您还可以使用数据 URI 来代替图片的网址。因此,您可以改为使用以下网址,而不是上述网址:
img.src=""
您可能会问:“为什么要使用数据 URI 而非二进制图片?”这样做有很多优势。在本文的后面部分,您将了解如何轻松地将画布图片导出为数据 URI。 以下是用于将二进制图片文件转换为数据 URI 的工具。
操纵画布图片
如果您曾进行过任何类型的 Logo 编程,那么在画布上绘制图形也是采用相同的概念。Mark Pilgrim 在其《Dive Into HTML5》一书中有一个关于画布的章节。根据本章中的示例,我们可以使用以下代码为上面导入的图片添加网格图表:
var img2 = new Image();
img2.onload = function () {
  var context2 = document.getElementById('new_canvas2').getContext('2d');
  /* vertical lines then horizontal ones */
  for (var x = 0.5; x < 800; x += 10) { context2.moveTo(x, 0); context2.lineTo(x, 500); } 
  for (var y = 0.5; y < 500; y += 10) { context2.moveTo(0, y); context2.lineTo(800, y); }
  context2.strokeStyle = "#bbb";
  context2.stroke();
  context2.drawImage(img2,0,0);
}
img2.src = "html5.gif";
您可以发挥更多创意,但请参阅本文附录中列出的其他教程,详细了解该主题。 我们还没有看到任何非常令人兴奋的内容,但下一部分将会改变这一点。
将画布图片导出为数据 URI
Canvas 元素有一个 toDataURL() 方法,该方法将 MIME 类型作为参数。这样,我们就可以导出上面使用的画布。
window.open(document.getElementById('ctx').toDataURL("image/png"));
这会将画布作为 PNG 图片导出到新的浏览器窗口中。不过,该图片不是普通的二进制图片,而是可以由浏览器呈现的 base64 编码数据 URI。因此,从用户的角度来看,这与二进制等效项没有区别。请注意,上述代码行需要在 Web 服务器上运行。对本地文件运行 toDataURL() 将会失败。如需了解此问题在 Chrome 中的状态,请参阅此工单。
集成到 Web 应用中
Canvas 是一款功能强大的插件,适用于存储用户上传的图片的任何 Web 应用。
例如,我们有一个在线文件存储应用,用于存储用户上传的图片。我们可以添加一个“修改”按钮,以便在基于画布的图片编辑器中打开图片文件。
如果您不想编写自己的画布编辑器,Harmony 是少数几个公开提供的画布编辑器之一。它支持轻松添加画笔,可满足您的艺术品味。
当您在上述菜单中选择“编辑图片”时,系统应会打开画布编辑器,并在编辑器的 init() 函数中调用自定义 read_file() 函数,如下所示:
function read_file() {
   var url = file_id;
   // hide a copy of the original image if it is needed to load
   document.getElementById('editableImage').src = url; 
   image = new Image();
   image.src = url;
   image.onload = function() {
      context.drawImage(image,0,0); // context, defined above, as canvas.getContext('2d')
   }
}
添加 HTML5 LocalStorage
如果您注重用户体验,则应始终考虑进行一些小修饰,例如应用 LocalStorage。例如,如果您有一个大文本区域,需要用户输入大量信息。当用户准备提交表单时,不小心关闭了浏览器(或浏览器崩溃)。用户可能会感到沮丧,不愿再重新撰写消息。在以下演示中,只需将图片作为数据 URI 保存到 LocalStorage,而无需将数据保存到服务器:
// Save Image
function saveToLocalStorage() {
    localStorage.setItem('canvas', canvas.toDataURL('image/png'));
}
// Load Image
function init() {
        // for demo purpose, all variables are declared in the parent scope
        canvas = document.createElement('canvas');
        context = canvas.getContext('2d');
        // Use Modernizr to detect whether localstorage is supported by the browser
        if (Modernizr.localstorage && localStorage.getItem('canvas'))
        {
            localStorageImage = new Image();
            localStorageImage.addEventListener("load", function (event) {
                //...
                context.drawImage(localStorageImage, 0, 0);
            }, false);
            localStorageImage.src = localStorage.getItem('canvas');
        }
//...
}
将画布保存为二进制文件到服务器
您可能需要将画布图片另存为二进制文件。您可以通过多种方式来实现这一点。例如,您可以执行 POST 操作,将数据 URI 传递给后端代码。使用 jQuery 时,代码如下所示:
var url = '/api/write/' + file_id + '?data_url_to_binary=1';
var data_url = flattenCanvas.toDataURL('image/png');
var params = { contents: data_url };
$j.post(url, params, function(json){
   if (json.status == 'upload_ok')
   {
      //ok
   }
}, 'json');
这会创建一个 XHR 调用,其中内容为数据 URI。然后,您需要在服务器上解码 base64 数据 URI。例如,在 PHP 中,您可以执行以下操作:
if ($_GET['data_url_to_binary'])
{
   $contents_split = explode(',', $contents);
   $encoded = $contents_split[count($contents_split)-1];
   $decoded = "";
   for ($i=0; $i < ceil(strlen($encoded)/256); $i++) {
      $decoded = $decoded . base64_decode(substr($encoded,$i*256,256)); 
   }
   $contents = $decoded; // output
}
在前两行中,数据 URI ($contents) 被拆分为两部分。'data:image/png;base64' 和 'VBORw0KGgoAAAANSUhEUgAAAWwAAAB+CAIAAACPlLzKAAAACXBIWXMAAC4jAAAuIwF4pT92...'。然后,我们将使用 base64_decode() 解码数据 URI 字符串。这里的诀窍在于,解码大于 5K 的字符串会出现问题,而这种“分而治之”的方法能够解码字符串。
最后,您可以使用 fwrite() 将二进制文件 $contents 保存到服务器上。
在浏览器中启用“保存图片”
Canvas 是一种 HTML 元素。它看起来很像图片,但您的浏览器不提供“另存为图片”选项,因为它实际上不是图片元素。 如需启用“另存为”,您可以动态创建一个 Img 元素,并将 src 设置为画布元素的数据 URI。 您还可以使用 canvas2image 实用程序。
更高级的画布编辑器
如果您想要使用更高级的画布编辑器,不妨试试 PaintWeb。该扩展程序由罗马尼亚学生 Mihai Sucan 在 2009 年 Google 编程之夏期间编写。他还撰写了一些有关编写自己的在线绘图应用的教程。
如需使用更专业的库,请务必查看 Pixati。
想让画布更有趣?
Paul Irish 将 Harmony 和 $1 Unistroke Recognizer 结合使用,在自己的网站上创建了一个小小的彩蛋。
您还可以了解如何使用 Chrome 开发者工具通过我们近期推出的检查功能检查画布。
通过 Canvas 的其他教程深入学习
- MDN:Canvas 教程
 - 深入了解 HTML5:Canvas
 - Dev.Opera 上的 HTML5 画布 - 基础知识介绍了基本绘制基元
 - 在画布中构建 Breakout 克隆,包括基本动作、物理特性和互动