Integra Canvas en tu app web

Introducción

En este artículo, analizaremos cómo utilizar el elemento lienzo de HTML5 para crear, editar, abrir y exportar imágenes. También presentaré varias herramientas de código abierto relevantes para esta tecnología y proporcionaré algunas sugerencias sobre cómo estas técnicas se pueden aplicar a una aplicación web existente.

Verifica la compatibilidad con lienzos

Lo primero que debes hacer es comprobar que tu navegador sea completamente compatible con el lienzo HTML5. Una forma sencilla de hacerlo es usar Modernizr para buscar una función determinada:

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

Crea un elemento de lienzo y, luego, importa una imagen como URI binario o de datos

Primero, deberás tener un elemento lienzo en tu página. Con JavaScript, puedes hacer lo siguiente:

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

En este código, el primer paso es obtener el contexto 2D, lo que nos da acceso a la API que define todos los métodos y propiedades de dibujo. A continuación, crearemos un objeto de imagen y configuraremos la propiedad src en la ubicación de la imagen binaria. Cuando se carga la imagen, usamos el método drawImage() para importar la imagen al elemento lienzo. También puedes usar un URI de datos en lugar de la URL de una imagen. Por lo tanto, en lugar de la URL anterior, puedes hacer lo siguiente:

img.src=""

Es posible que te preguntes: “¿Por qué deberíamos usar el URI de datos en lugar de la imagen binaria?”. Existen muchas ventajas. Más adelante en este artículo, verás lo fácil que es exportar una imagen de lienzo como URI de datos. Esta es una herramienta para convertir un archivo de imagen binario en un URI de datos.

Manipula la imagen del lienzo

Si alguna vez realizaste algún tipo de programación de logotipos, dibujar en un lienzo usa el mismo concepto. Mark Pilgrim tiene un capítulo sobre lienzo en su libro "Dive Into HTML5". Basándonos en un ejemplo del capítulo, podemos agregar un diagrama de cuadrícula a la imagen que importamos anteriormente usando lo siguiente:

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

Puedes ser más creativo que esto, pero se lo dejo a los demás tutoriales que aparecen en el apéndice de este artículo para obtener instrucciones adicionales sobre ese tema. Aún no vimos nada interesante, pero la próxima sección cambiará eso.

Exporta la imagen de lienzo como URI de datos

El elemento de lienzo tiene un método toDataURL(), que toma un tipo de MIME como parámetro. Con esto, podemos exportar el lienzo que usamos anteriormente.

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

Esto exportará el lienzo como una imagen PNG a una nueva ventana del navegador. Sin embargo, la imagen no es una imagen binaria común, sino un URI de datos codificado en base64 que un navegador puede procesar. Por lo tanto, desde el punto de vista del usuario, no hay diferencia entre eso y el equivalente binario. Ten en cuenta que la línea de código anterior debe ejecutarse en un servidor web. La ejecución de toDataURL() en un archivo local fallaría. Consulta este ticket para ver el estado del problema en Chrome.

Realiza la integración en tu app web

Canvas puede ser un complemento muy potente para cualquier aplicación web que almacene imágenes subidas por los usuarios.

Lienzo en caja

Por ejemplo, tenemos una aplicación de almacenamiento de archivos en línea que almacena imágenes subidas por los usuarios. Podemos agregar un botón de edición para abrir el archivo de imagen en un editor de imágenes basado en lienzo. Si no quieres escribir tu propio editor de lienzos, Harmony es uno de los pocos editores de lienzos disponibles. Además, incluye también pinceles que pueden satisfacer tus gustos artísticos. Cuando eliges "editar imagen" en el menú ilustrado anteriormente, se abre un editor de lienzo y se realiza una llamada a una función read_file() personalizada en la función init() del editor de la siguiente manera:

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')
   }
}
Armonía

Cómo agregar LocalStorage de HTML5

Un pequeño retoque que deberías considerar siempre (si te interesa la experiencia del usuario) es aplicar LocalStorage. Por ejemplo, si tienes un área de texto grande que requiere que el usuario ingrese mucha información. Cuando el usuario está a punto de enviar el formulario, cierra el navegador por accidente (o este falla). El usuario podría sentirse frustrado y no molestarse a volver a escribir el mensaje. En la siguiente demostración, en lugar de guardar los datos en el servidor, simplemente guarda la imagen en LocalStorage como URI de datos:

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

Guardar el lienzo como un archivo binario en el servidor

Es posible que quieras guardar la imagen del lienzo como un archivo binario. Hay muchas maneras de hacerlo. Por ejemplo, puedes realizar una acción POST para pasar el URI de datos a tu código de backend. Con jQuery, se vería así:

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

Esto crea una llamada XHR con el contenido como el URI de datos. Luego, debes decodificar el URI de datos en base64 en el servidor. Por ejemplo, en PHP puedes hacer lo siguiente:

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
}

En las primeras dos líneas, el URI de datos ($contents) se divide en dos partes. 'data:image/png;base64' y 'VBORw0KGgoAAAANSUhEUgAAAWwAAAB+CAIAAACPlLzKAAAACXBIWXMAAC4jAAAuIwF4pT92...' Luego, usaremos base64_decode() para decodificar la cadena de URI de datos. El truco aquí es que hay problemas de decodificación de cadenas superiores a 5K, y este enfoque de "dividir y vencer" podrá decodificar la cadena. Por último, puedes usar fwrite() para guardar el archivo binario, $contents, en tu servidor.

Habilitar la opción "Guardar imagen" en el navegador

Canvas es un elemento HTML. Se parece bastante a una imagen, pero tu navegador no ofrece la opción "Guardar imagen como" porque, en realidad, no es un elemento de imagen. Para habilitar "Guardar imagen como", puedes crear un elemento Img de forma dinámica y establecer src en el URI de datos del elemento de lienzo. También puedes usar la utilidad canvas2image.

Un editor de lienzo más avanzado

Si buscas un editor de lienzos más avanzado, quizás te convenga probar PaintWeb. Fue escrita por Mihai Sucan, un estudiante rumano, durante el Google Summer of Code 2009. También creó algunos instructivos para crear tu propia aplicación de pintura en línea.

Paint Web

Si buscas una biblioteca más profesional, visita Pixati.

¿Te diviertes más con el lienzo?

Paul Irlandés combinó Harmony y USD 1 Recognizer para crear un pequeño Huevo de Pascua en su sitio web.

También puedes aprender a inspeccionar lienzos con las Herramientas para desarrolladores de Chrome usando nuestras funciones de inspección recientes.

Profundiza con los instructivos adicionales sobre lienzo