שילוב בד ציור באפליקציית האינטרנט שלך

דייוויד טונג
דיוויד טונג

מבוא

במאמר הזה אני אדבר על אופן השימוש בלוח הציור של HTML5 כדי ליצור, לערוך, לפתוח ולייצא תמונות. כמו כן, הצגת מספר כלים בקוד פתוח שרלוונטיים לטכנולוגיה הזו, ואתן כמה טיפים איך ליישם את הטכניקות האלה ביישומי אינטרנט קיימים.

בדיקה אם יש תמיכה בהדפסה על קנבס

קודם כל, צריך לוודא שהדפדפן תומך באופן מלא בהדפסה על קנבס בפורמט 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);
}

השלב הראשון בקוד הזה הוא לקבל את ההקשר הדו-ממדי, שמעניק לנו גישה ל-API שמגדיר את כל שיטות השרטוט והמאפיינים. בשלב הבא יוצרים אובייקט תמונה ומגדירים את המאפיין src למיקום של התמונה הבינארית. לאחר טעינת התמונה, אנחנו משתמשים בשיטת PaintImage() כדי לייבא את התמונה לרכיב הקנבס. אפשר גם להשתמש ב-URI של נתונים במקום בכתובת URL של תמונה. במקום כתובת האתר שלמעלה, תוכלו לבצע את הפעולות הבאות:

img.src=""

ייתכן שתשאלו את השאלה "למה נשתמש ב-URI של נתונים במקום בתמונה הבינארית?" יש הרבה יתרונות. בהמשך המאמר תראו כמה קל לייצא תמונה מבד קנבס כ-URI של נתונים. כאן יש כלי להמרת קובץ תמונה בינארית ל-URI של נתונים.

ביצוע מניפולציות על תמונת הקנבס

אם יצרתם אי פעם תכנות סמלי לוגו, תוכלו לצייר על קנבס באמצעות אותו קונספט. למארק פילגרים יש פרק על בד קנבס בספר שלו, 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.

שילוב באפליקציית האינטרנט

לוח הציור יכול להיות תוסף חזק מאוד לכל אפליקציית האינטרנט שמאחסנת תמונות שהועלו על ידי משתמשים.

קנבס בקופסה

לדוגמה, יש לנו יישום מקוון לאחסון קבצים שמאחסן תמונות שהועלו על ידי משתמשים. נוכל להוסיף לחצן עריכה לפתיחת קובץ התמונה בעורך תמונות שמבוסס על בד ציור. אם אתם לא רוצים לכתוב בעצמכם כלי עריכה על קנבס, 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')
   }
}
Harmony

הוספת HTML5 LocalStorage

שיפור קטן שכדאי תמיד לקחת בחשבון - אם חשוב לכם חוויית המשתמש - הוא החלת 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 של הנתונים. הטריק הוא שיש בעיות בפענוח מחרוזת שגדולה מ-5K, ושיטת ה "חלוקה והמנצח" תוכל לפענח את המחרוזת. לבסוף, באמצעות fwrite(), ניתן לשמור בשרת את הקובץ הבינארי, $contents.

הפעלת האפשרות 'שמירת תמונה' בדפדפן

לוח הציור הוא רכיב HTML. התמונה נראית כמעט כמו תמונה, אבל הדפדפן לא מספק אפשרות של 'שמירת תמונה בשם' מפני שהוא לא באמת רכיב תמונה. כדי להפעיל את האפשרות 'שמירת תמונה בשם', אפשר ליצור באופן דינמי רכיב Img ולהגדיר את ה-src ל-URI של הנתונים של רכיב הקנבס. ניתן גם להשתמש בכלי העזר לתמונה Canvas2image.

גרסה מתקדמת יותר של עורך ההדפסות על קנבס

אם אתם מחפשים כלי מתקדם יותר לעריכת תמונות על קנבס, כדאי לכם לנסות את PaintWeb. הוא נכתב על ידי מיהאי סוקן, סטודנט רומני, במהלך קיץ של Google בשנת 2009. הוא גם כתב כמה מדריכים בנושא כתיבת אפליקציית צבע באינטרנט.

המרת תמונה וקטורית למפת סיביות (painting)

לקבלת ספרייה מקצועית יותר, כדאי לבדוק את Pixati.

יותר כיף להשתמש על קנבס?

פול אירן שילב את Harmony עם Recognizer בסך $1 UniStrike כדי ליצור Easter Egg באתר שלו.

בנוסף, אפשר ללמוד איך לבדוק את לוח הציור באמצעות כלי הפיתוח ל-Chrome באמצעות תכונות הבדיקה האחרונות שלנו.

מתעמקים יותר עם מדריכים נוספים על קנבס