Tích hợp Canvas vào ứng dụng web

Giới thiệu

Trong bài viết này, tôi sẽ thảo luận về cách sử dụng phần tử canvas HTML5 để tạo, chỉnh sửa, mở và xuất hình ảnh. Tôi cũng sẽ giới thiệu một số công cụ nguồn mở có liên quan đến công nghệ này, đồng thời cung cấp một số mẹo về cách áp dụng những kỹ thuật này vào ứng dụng web hiện có.

Kiểm tra xem canvas có hỗ trợ không

Điều đầu tiên cần làm là kiểm tra xem trình duyệt của bạn có hỗ trợ đầy đủ canvas HTML5 hay không. Cách dễ dàng để thực hiện việc đó là sử dụng Hiện đại hoá để kiểm tra một tính năng nhất định:

if (Modernizr.canvas) {
  // Browser supports native HTML5 canvas.
} else {
  // Fallback to another solution, such as Flash, static image, download link, and so on.
}

Tạo phần tử canvas và nhập hình ảnh dưới dạng tệp nhị phân hoặc URI dữ liệu

Trước tiên, trang của bạn cần có một thành phần canvas. Khi sử dụng JavaScript, bạn sẽ thực hiện những việc sau:

var ctx = document.getElementById('new_canvas').getContext('2d');
var img = new Image();
img.src = "html5.gif"
img.onload = function () {
   ctx.drawImage(img,0,0);
}

Trong mã này, bước đầu tiên là tải ngữ cảnh 2D. Điều này cho phép chúng ta truy cập vào API xác định mọi phương thức và thuộc tính vẽ. Tiếp theo, chúng ta sẽ tạo một đối tượng hình ảnh và đặt thuộc tính src vào vị trí của hình ảnh nhị phân. Khi hình ảnh được tải, chúng ta sử dụng phương thức drawImage() để nhập hình ảnh vào phần tử canvas. Bạn cũng có thể sử dụng URI dữ liệu thay vì URL của hình ảnh. Vì vậy, thay vì URL ở trên, bạn có thể làm như sau:

img.src=""

Bạn có thể hỏi "Tại sao chúng tôi sử dụng URI dữ liệu thay vì hình ảnh nhị phân?" Có nhiều lợi thế. Ở phần sau của bài viết này, bạn sẽ thấy cách chúng ta có thể xuất hình ảnh canvas dưới dạng URI dữ liệu dễ dàng như thế nào. Dưới đây là công cụ giúp chuyển đổi tệp hình ảnh nhị phân thành URI dữ liệu.

Chỉnh sửa hình ảnh canvas

Nếu bạn đã từng thực hiện bất kỳ loại lập trình Biểu trưng nào, thì việc vẽ trên canvas cũng sử dụng cùng khái niệm đó. Mark Philippines có một chương trên canvas trong sách của mình, Tìm hiểu về HTML5. Dựa vào ví dụ trong chương này, chúng ta có thể thêm sơ đồ lưới vào hình ảnh mà chúng ta đã nhập ở trên bằng cách sử dụng đoạn mã sau:

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";

Bạn có thể sáng tạo hơn thế này, nhưng tôi sẽ để hướng dẫn đó là kết hợp với các hướng dẫn khác được liệt kê trong phụ lục của bài viết này để hướng dẫn thêm về chủ đề đó. Chúng ta chưa thấy có điều gì quá thú vị, nhưng phần tiếp theo sẽ thay đổi điều đó.

Xuất hình ảnh canvas dưới dạng URI dữ liệu

Phần tử canvas có phương thức toDataURL() lấy loại MIME làm tham số. Nhờ vậy, chúng ta có thể xuất ảnh in trên vải canvas đã sử dụng ở trên.

window.open(document.getElementById('ctx').toDataURL("image/png"));

Thao tác này sẽ xuất canvas dưới dạng hình ảnh PNG vào một cửa sổ trình duyệt mới. Tuy nhiên, hình ảnh này không phải là hình ảnh nhị phân thông thường mà là một URI dữ liệu được mã hoá base64 có thể hiển thị bằng trình duyệt. Do đó, từ quan điểm của người dùng, không có sự khác biệt giữa giá trị này và giá trị nhị phân tương đương. Lưu ý rằng dòng mã ở trên cần được chạy trên máy chủ web. Không chạy được toDataURL() trên một tệp cục bộ. Xem vé này để biết trạng thái của vấn đề này trong Chrome.

Tích hợp vào ứng dụng web

Canvas có thể là một tiện ích bổ sung rất hiệu quả cho bất kỳ ứng dụng web nào có lưu trữ hình ảnh do người dùng tải lên.

Canvas hộp

Ví dụ: chúng tôi có một ứng dụng lưu trữ tệp trực tuyến dùng để lưu trữ hình ảnh do người dùng tải lên. Chúng ta có thể thêm nút chỉnh sửa để mở tệp hình ảnh trong trình chỉnh sửa ảnh dựa trên canvas. Nếu bạn không muốn tự viết trình chỉnh sửa ảnh in trên vải canvas, thì Harmony là một trong số ít trình chỉnh sửa ảnh in trên vải canvas hiện có. Tính năng này có tính năng dễ dàng bổ sung cọ vẽ, giúp bạn thoả mãn sở thích nghệ thuật. Khi bạn chọn "edit image" (chỉnh sửa hình ảnh) trong trình đơn minh hoạ ở trên, trình chỉnh sửa canvas sẽ mở ra và trình chỉnh sửa canvas sẽ thực hiện lệnh gọi đến hàm read_file() tuỳ chỉnh trong hàm init() của trình chỉnh sửa như sau:

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')
   }
}
Sự hoà hợp

Thêm HTML5 LocalStorage

Một chỉnh sửa nhỏ mà bạn phải luôn cân nhắc (nếu bạn quan tâm đến trải nghiệm người dùng) là áp dụng LocalStorage. Ví dụ: nếu bạn có vùng văn bản lớn yêu cầu người dùng nhập nhiều thông tin. Khi người dùng sắp gửi biểu mẫu, họ vô tình đóng trình duyệt (hoặc trình duyệt gặp sự cố). Người dùng có thể thấy thất vọng và không bận tâm đến việc viết lại thư nữa. Trong bản minh hoạ dưới đây, thay vì lưu dữ liệu vào máy chủ, bạn chỉ cần lưu hình ảnh vào LocalStorage dưới dạng URI dữ liệu:

// 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');
        }
//...
}

Lưu canvas dưới dạng tệp nhị phân vào máy chủ

Bạn có thể muốn lưu hình ảnh canvas dưới dạng tệp nhị phân. Có nhiều cách để thực hiện việc đó. Ví dụ: bạn có thể thực hiện thao tác POST để chuyển URI dữ liệu vào mã phụ trợ. Khi sử dụng jQuery, bạn sẽ thấy như sau:

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');

Thao tác này sẽ tạo một lệnh gọi XHR với nội dung là URI dữ liệu. Sau đó, bạn cần giải mã URI dữ liệu base64 trên máy chủ. Ví dụ: trong PHP, bạn có thể làm như sau:

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
}

Trong hai dòng đầu tiên, URI dữ liệu ($contents) được chia thành hai phần. 'data:image/png;base64''VBORw0KGgoAAAANSUhEUgAAAWwAAAB+CAIAAACPlLzKAAAACXBIWXMAAC4jAAAuIwF4pT92...' Sau đó, chúng ta sẽ sử dụng base64_decode() để giải mã chuỗi URI dữ liệu. Bí quyết ở đây là có các vấn đề khi giải mã chuỗi lớn hơn 5K và phương pháp "chia để chinh phục" này sẽ có thể giải mã chuỗi đó. Cuối cùng, bằng cách sử dụng fwrite(), bạn có thể lưu tệp nhị phân, $contents, vào máy chủ của mình.

Bật tính năng "lưu hình ảnh" trong trình duyệt

Canvas là một phần tử HTML. Hình ảnh này trông khá giống một hình ảnh, nhưng trình duyệt của bạn không cung cấp tuỳ chọn "Lưu hình ảnh dưới dạng" cho nó vì nó không thực sự là một thành phần hình ảnh. Để bật tính năng "Lưu hình ảnh dưới dạng", bạn có thể tự động tạo một phần tử Img và đặt src thành URI dữ liệu của phần tử canvas. Bạn cũng có thể sử dụng tiện ích canvass2image.

Trình chỉnh sửa canvas nâng cao hơn

Nếu đang tìm một trình chỉnh sửa canvas nâng cao hơn, bạn có thể dùng thử PaintWeb. Tác giả của cuốn sách này là Mihai Sucan, một sinh viên người Rumani, trong Mùa hè lập trình của Google năm 2009. Anh cũng là tác giả của một số hướng dẫn về cách tự viết ứng dụng sơn trực tuyến.

Tạo điểm ảnh trên web

Để dùng thư viện chuyên nghiệp hơn, hãy nhớ khám phá Pixati.

Có thêm nhiều điều thú vị với ảnh in trên vải canvas không?

Paul Ireland đã kết hợp Harmony và Unistroke Recognizer với $1 để tạo ra một Privacyer Egg nhỏ trên trang web của mình.

Bạn cũng có thể tìm hiểu cách kiểm tra canvas bằng Công cụ của Chrome cho nhà phát triển bằng cách sử dụng các tính năng kiểm tra gần đây của chúng tôi.

Tìm hiểu sâu hơn qua các hướng dẫn bổ sung về canvas