מבוא
במאמר הזה אסביר איך להשתמש ברכיב canvas ב-HTML5 כדי ליצור, לערוך, לפתוח ולייצא תמונות. בנוסף, אציג כמה כלים בקוד פתוח שרלוונטיים לטכנולוגיה הזו, ואספק כמה טיפים לשימוש בשיטות האלה באפליקציית אינטרנט קיימת.
בדיקת התמיכה בהדפסה על קנבס
הדבר הראשון שצריך לעשות הוא לבדוק שהדפדפן תומך באופן מלא ב-HTML5 canvas. דרך קלה לעשות זאת היא להשתמש ב-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 בינארי או כ-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 של נתונים במקום בכתובת ה-URL של התמונה. לכן, במקום כתובת ה-URL שלמעלה, אפשר לבצע את הפעולות הבאות:
img.src=""
יכול להיות שתתהו "למה כדאי להשתמש ב-URI של נתונים במקום בתמונה הבינארית?" יש לכך הרבה יתרונות. בהמשך המאמר נסביר איך אפשר לייצא בקלות תמונה על קנבס כ-URI של נתונים. כאן מופיע כלי להמרת קובץ תמונה בינארי ל-URI של נתונים.
מניפולציה של התמונה על קנבס
אם יצרתם פעם לוגו באמצעות תכנות, תהליך הציור על קנבס מבוסס על אותו עיקרון. בספר של 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 של נתונים
לרכיב הקנבס יש שיטה toDataURL(), שמקבלת סוג MIME כפרמטר. כך נוכל לייצא את תמונת הקנבס שבה השתמשנו למעלה. 
window.open(document.getElementById('ctx').toDataURL("image/png"));
הפעולה הזו מייצאת את לוח הציור כתמונה בפורמט PNG לחלון דפדפן חדש. עם זאת, התמונה היא לא תמונה רגילה בפורמט בינארי, אלא URI של נתונים שמקודד ב-base64 ואפשר להציג אותו בדפדפן. לכן, מנקודת המבט של המשתמש, אין הבדל בין הקוד הזה לבין המקבילה הבינארית שלו. 
שימו לב שצריך להריץ את שורת הקוד שלמעלה בשרת אינטרנט. הרצת הפקודה toDataURL() על קובץ מקומי תיכשל. בפנייה הזו אפשר לבדוק את הסטטוס של הבעיה הזו ב-Chrome. 
שילוב באפליקציית האינטרנט
Canvas יכול להיות תוסף יעיל מאוד לכל אפליקציית אינטרנט שמאחסנת תמונות שהמשתמשים העלו.
לדוגמה, יש לנו אפליקציה לאחסון קבצים באינטרנט שמשמרת תמונות שהמשתמשים העלו. אנחנו יכולים להוסיף לחצן עריכה כדי לפתוח את קובץ התמונה בעורך תמונות מבוסס-קנבס. 
אם אתם לא רוצים לכתוב עורך בד שלכם, Harmony הוא אחד מערוצי הבד הבודדים שזמינים באופן פתוח. אפשר להוסיף בקלות מברשות שיתאימו לטעם האמנותי שלכם. 
כשבוחרים באפשרות 'עריכת תמונה' בתפריט שמוצג למעלה, אמור להיפתח עורך בד, שיבצע קריאה לפונקציה read_file() בהתאמה אישית בפונקציה init() של העורך באופן הבא: 
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')
   }
}
הוספת LocalStorage ב-HTML5
אם אתם רוצים לשפר את חוויית המשתמש, כדאי לכם להשתמש ב-LocalStorage. לדוגמה, אם יש לכם אזור טקסט גדול שמחייב את המשתמש להזין הרבה מידע. כשהמשתמש עומד לשלוח את הטופס, הוא סוגר בטעות את הדפדפן (או שהדפדפן קורס). יכול להיות שהמשתמש יתרגז ולא ירצה לכתוב שוב את ההודעה. בדמו שבהמשך, במקום לשמור את הנתונים בשרת, פשוט שומרים את התמונה ב-LocalStorage כ-URI של נתונים:
// 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 הנתונים. לאחר מכן צריך לפענח את ה-URI של נתוני base64 בשרת. ב-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 של הנתונים. הטריק הוא שיש בעיות בפענוח מחרוזות ארוכות מ-5,000 תווים, והגישה הזו של 'חלוקה לגורמים וניצחון' תאפשר לפענח את המחרוזת. 
לבסוף, באמצעות fwrite(), אפשר לשמור את הקובץ הבינארי, $contents, בשרת. 
הפעלת האפשרות 'שמירה של תמונה' בדפדפן
Canvas הוא רכיב HTML. הוא נראה כמעט כמו תמונה, אבל בדפדפן לא מופיעה האפשרות 'שמירה כתמונה' כי הוא לא באמת רכיב תמונה. כדי להפעיל את האפשרות 'שמירה כתמונה', אפשר ליצור באופן דינמי אלמנט Img ולהגדיר את ה-src כ-URI של הנתונים של אלמנט הלוח. אפשר גם להשתמש בהשירות canvas2image.
עורך קנבס מתקדם יותר
אם אתם מחפשים עורך מתקדם יותר לקנבס, כדאי לנסות את PaintWeb. הוא נכתב על ידי Mihai Sucan, סטודנט רומני, במהלך Google Summer of Code 2009. הוא גם כתב כמה מדריכים לכתיבת אפליקציית ציור משלכם באינטרנט.
לספרייה מקצועית יותר, כדאי לבדוק את Pixati.
רוצים להוסיף עוד קצת כיף?
פול אירי (Paul Irish) שילב את Harmony עם הכלי לזיהוי כתב יד של Unistroke ב-1 $כדי ליצור ביצה של פסחא קטנה באתר שלו.
אפשר גם ללמוד איך לבדוק את לוח הציור באמצעות כלי הפיתוח ל-Chrome באמצעות תכונות הבדיקה החדשות שלנו.
מדריכים נוספים בנושא Canvas
- MDN: מדריך Canvas
 - הצגה מפורטת של HTML5: Canvas
 - במאמר HTML5 canvas – the basics ב-Dev.Opera מוסבר על הפרימטיביים הבסיסיים של ציור.
 - יצירת עותק כפול של Breakout ב-Canvas כולל תנועה, פיזיקה ואינטראקטיביות בסיסיים