Intégrer Canvas dans votre application Web

Introduction

Dans cet article, nous allons vous expliquer comment utiliser l'élément canvas HTML5 pour créer, modifier, ouvrir et exporter des images. Je vous présenterai également plusieurs outils Open Source pertinents pour cette technologie, ainsi que quelques conseils sur la façon de les appliquer à une application Web existante.

Vérifier la compatibilité avec les canevas

Commencez par vérifier que votre navigateur est entièrement compatible avec le canevas HTML5. Pour ce faire, il vous suffit d'utiliser Modernizr pour vérifier une certaine fonctionnalité:

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

Créer un élément de canevas et importer une image en tant qu'URI binaire ou de données

Tout d'abord, vous devez disposer d'un élément canevas sur votre page. En utilisant JavaScript, vous pouvez:

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

Dans ce code, la première étape consiste à obtenir le contexte 2D, ce qui nous donne accès à l'API qui définit toutes les méthodes et propriétés de dessin. Ensuite, nous créons un objet "image" et définissons la propriété "src" sur l'emplacement de l'image binaire. Une fois l'image chargée, nous utilisons la méthode drawImage() pour l'importer dans l'élément canevas. Vous pouvez également utiliser un URI de données au lieu de l'URL d'une image. Ainsi, au lieu d'utiliser l'URL ci-dessus, vous pouvez procéder comme suit:

img.src=""

Vous pouvez vous demander : "Pourquoi utiliserions-nous l'URI de données au lieu de l'image binaire ?". Cette approche présente de nombreux avantages. Plus loin dans cet article, vous verrez avec quelle facilité il est possible d'exporter une image de canevas en tant qu'URI de données. Voici un outil permettant de convertir un fichier image binaire en URI de données.

Manipuler l'image du canevas

Si vous avez déjà réalisé n'importe quelle programmation de logo, dessiner sur un canevas repose sur le même concept. Mark Pilgrim a créé un chapitre sur les toiles dans son livre "Plongez dans le langage HTML5". Sur la base d'un exemple du chapitre, nous pouvons ajouter un diagramme de grille à l'image que nous avons importée ci-dessus en utilisant le code suivant :

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

Vous pouvez être plus créatif que cela, mais je laisse les autres didacticiels listés dans l'annexe de cet article pour des instructions supplémentaires sur ce sujet. Nous n'avons encore rien vu de très excitant, mais la section suivante va changer la donne.

Exporter l'image de canevas en tant qu'URI de données

L'élément canevas comporte une méthode toDataURL(), qui utilise un type MIME comme paramètre. À présent, nous pouvons exporter le canevas que nous avons utilisé ci-dessus.

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

Le canevas est alors exporté sous forme d'image PNG dans une nouvelle fenêtre du navigateur. Toutefois, l'image n'est pas une image binaire ordinaire, mais un URI de données encodé en base64 qui peut être affiché par un navigateur. Ainsi, du point de vue de l'utilisateur, il n'existe aucune différence entre cela et l'équivalent binaire. Notez que la ligne de code ci-dessus doit être exécutée sur un serveur Web. L'exécution de toDataURL() sur un fichier local échouerait. Consultez cette demande pour connaître l'état du problème dans Chrome.

Intégration dans votre application Web

Dessin peut être un module complémentaire très puissant à toute application Web qui stocke des images importées par les utilisateurs.

Toile de la boîte

Prenons l'exemple d'une application de stockage de fichiers en ligne qui stocke les images importées par les utilisateurs. Nous pouvons ajouter un bouton de modification pour ouvrir le fichier image dans un éditeur de photos basé sur un canevas. Si vous ne voulez pas écrire votre propre éditeur de canevas, Harmony est l'un des seuls éditeurs de canevas disponibles publiquement. Elle permet d'ajouter facilement des pinceaux pour satisfaire vos goûts artistiques. Lorsque vous choisissez "modifier l'image" dans le menu illustré ci-dessus, un éditeur de canevas doit s'ouvrir et appeler une fonction read_file() personnalisée dans la fonction init() de l'éditeur, comme suit:

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

Ajouter un stockage local HTML5

Si vous vous souciez de l'expérience utilisateur, vous devez toujours prendre en compte les modifications apportées à l'application LocalStorage. C'est le cas, par exemple, si vous disposez d'une grande zone de texte qui oblige l'utilisateur à saisir beaucoup d'informations. Lorsque l'utilisateur est sur le point d'envoyer le formulaire, il ferme accidentellement le navigateur (ou celui-ci se bloque). L'utilisateur peut être frustré et ne pas prendre la peine de réécrire le message. Dans la démonstration ci-dessous, au lieu d'enregistrer les données sur le serveur, enregistrez simplement l'image sur LocalStorage en tant qu'URI de données:

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

Enregistrer le canevas en tant que fichier binaire sur le serveur

Vous pouvez enregistrer l'image de canevas en tant que fichier binaire. Il existe de nombreuses façons de le faire. Par exemple, vous pouvez effectuer une action POST pour transmettre l'URI de données au code de votre backend. En utilisant jQuery, cela ressemblerait à ceci:

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

Cela crée un appel XHR dont le contenu correspond à l'URI de données. Vous devez ensuite décoder l'URI de données base64 sur le serveur. En PHP, par exemple, vous pouvez effectuer les opérations suivantes:

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
}

Dans les deux premières lignes, l'URI de données ($contents) est divisé en deux parties. 'data:image/png;base64' et 'VBORw0KGgoAAAANSUhEUgAAAWwAAAB+CAIAAACPlLzKAAAACXBIWXMAAC4jAAAuIwF4pT92...'. Nous utiliserons ensuite base64_decode() pour décoder la chaîne d'URI de données. L'astuce est qu'il existe des problèmes de décodage d'une chaîne supérieure à 5K, et que cette approche de type "diviser pour régner" permettra de décoder la chaîne. Enfin, à l'aide de fwrite(), vous pouvez enregistrer le fichier binaire $contents sur votre serveur.

Activer l'option "Enregistrer l'image" dans le navigateur

Le canevas est un élément HTML. Elle ressemble à une image, mais votre navigateur ne propose pas l'option "Enregistrer l'image sous", car il ne s'agit pas vraiment d'un élément image. Pour activer l'option "Enregistrer l'image sous", vous pouvez créer un élément Img de manière dynamique et définir src sur l'URI de données de l'élément canevas. Vous pouvez également utiliser l'utilitaire canvas2image.

Un éditeur de canevas plus avancé

Si vous recherchez un éditeur de canevas plus avancé, PaintWeb peut probablement s'avérer utile. Il a été écrit par Mihai Sucan, une étudiante roumaine, lors de l'événement Google Summer of Code 2009 de Google. Il est également l'auteur de quelques tutoriels sur la création de votre propre application de peinture en ligne.

Peinture sur le Web

Pour une bibliothèque plus professionnelle, consultez Pixati.

Les toiles vous plaisent plus ?

Paul Irish a combiné Harmony et $1 Unistroke Recognizer pour créer un petit Easter Egg sur son site Web.

Vous pouvez également découvrir comment inspecter le canevas avec les outils pour les développeurs Chrome grâce à nos fonctionnalités d'inspection récentes.

Approfondissez vos connaissances avec des tutoriels supplémentaires sur les canevas