Como integrar Canvas ao seu app da Web

Introdução

Neste artigo, vou mostrar como usar o elemento canvas do HTML5 para criar, editar, abrir e exportar imagens. Também apresentarei diversas ferramentas de código aberto relevantes para essa tecnologia e fornecerei algumas dicas sobre como essas técnicas podem ser aplicadas a um aplicativo da web existente.

Verificar o suporte à tela

A primeira coisa a fazer é verificar se seu navegador é totalmente compatível com canvas em HTML5. Uma maneira fácil de fazer isso é usar o Modernizr para verificar se há um determinado recurso:

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

Criar um elemento de tela e importar uma imagem como binário ou URI de dados

Primeiro, você precisa ter um elemento de tela em sua página. Usando JavaScript, você faz o seguinte:

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

Nesse código, a primeira etapa é acessar o contexto 2D, que dá acesso à API que define todos os métodos e propriedades de desenho. Em seguida, criamos um objeto de imagem e definimos a propriedade src para o local da imagem binária. Quando a imagem é carregada, usamos o método drawImage() para importá-la para o elemento canvas. Também é possível usar um URI de dados em vez do URL de uma imagem. Então, em vez do URL acima, você pode fazer o seguinte:

img.src=""

Você pode perguntar: "Por que usaríamos o URI de dados em vez da imagem binária?" Há muitas vantagens. Mais adiante neste artigo, você vai entender como é fácil exportar uma imagem de tela como URI de dados. Esta é uma ferramenta para converter um arquivo de imagem binário em um URI de dados.

Manipulando a imagem da tela

Se você já fez qualquer tipo de programação de logotipos, o desenho em uma tela usa o mesmo conceito. Mark Pilgrim tem um capítulo sobre tela no livro Dive Into HTML5. Com base em um exemplo do capítulo, podemos adicionar um diagrama de grade à imagem importada acima usando o seguinte código:

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

Você pode ser mais criativo do que isso, mas deixo para os outros tutoriais listados no apêndice deste artigo para obter instruções adicionais sobre esse assunto. Ainda não vimos nada muito interessante, mas a próxima seção mudará isso.

Exportando a imagem da tela como URI de dados

O elemento de tela tem um método toDataURL(), que usa um tipo MIME como parâmetro. Com isso, podemos exportar a tela que usamos acima.

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

Isso exporta a tela como uma imagem PNG para uma nova janela do navegador. No entanto, a imagem não é uma imagem binária comum, mas sim um URI de dados codificado em base64 que pode ser renderizado por um navegador. Portanto, do ponto de vista do usuário, não há diferença entre ele e o equivalente binário. A linha de código acima precisa ser executada em um servidor da Web. A execução de toDataURL() em um arquivo local falharia. Consulte este tíquete para saber o status desse problema no Chrome.

Integrar ao seu app da Web

O Canvas pode ser um complemento muito útil para qualquer aplicativo da Web que armazena imagens enviadas pelo usuário.

Tela de caixa

Por exemplo, temos um aplicativo de armazenamento de arquivos on-line que armazena imagens enviadas por usuários. Podemos adicionar um botão de edição para abrir o arquivo de imagem em um editor de imagens baseado em tela. Se você não quer criar seu próprio editor de tela, o Harmony é um dos poucos editores de tela disponíveis. É fácil adicionar pincéis, o que pode satisfazer seus gostos artísticos. Quando você escolhe "editar imagem" no menu ilustrado acima, um editor de tela é aberto e faz uma chamada para uma função read_file() personalizada na função init() do editor da seguinte maneira:

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

Como adicionar armazenamento local HTML5

Se você se importa com a experiência do usuário, considere a aplicação do LocalStorage. Por exemplo, se você tem uma área de texto grande que exige que o usuário insira muitas informações. Quando o usuário está prestes a enviar o formulário, ele fecha acidentalmente o navegador (ou o navegador trava). O usuário pode ficar frustrado e não ter que reescrever a mensagem novamente. Na demonstração abaixo, em vez de salvar os dados no servidor, basta salvar a imagem no LocalStorage como URI de dados:

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

Salvando a tela como um arquivo binário no servidor

Você pode salvar a imagem da tela como um arquivo binário. Há muitas maneiras de fazer isso. Por exemplo, você pode executar uma ação POST para transmitir o URI de dados ao código de back-end. Usando jQuery, a aparência ficaria assim:

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

Isso cria uma chamada XHR com o conteúdo sendo o URI de dados. Em seguida, você precisa decodificar o URI de dados em base64 no servidor. Em PHP, por exemplo, é possível fazer o seguinte:

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
}

Nas primeiras duas linhas, o URI de dados ($contents) é dividido em duas partes. 'data:image/png;base64' e 'VBORw0KGgoAAAANSUhEUgAAAWwAAAB+CAIAAACPlLzKAAAACXBIWXMAAC4jAAAuIwF4pT92...' Em seguida, usaremos base64_decode() para decodificar a string de URI de dados. O problema é que há problemas na decodificação de strings maiores que 5K, e essa abordagem de "dividir e conquistar" será capaz de decodificar a string. Por fim, use fwrite() para salvar o arquivo binário $contents no seu servidor.

Ativando a opção "salvar imagem" no navegador

Canvas é um elemento HTML. Ele se parece com uma imagem, mas seu navegador não fornece a opção "Salvar imagem como" porque ela não é realmente um elemento de imagem. Para ativar "Salvar imagem como", é possível criar dinamicamente um elemento Img e definir o src para o URI de dados do elemento da tela. Também é possível usar o utilitário canvas2image.

Um editor de tela mais avançado

Se você procura um editor de tela mais avançado, o PaintWeb provavelmente vale a pena. Ele foi escrito por Mihai Sucan, um estudante romeno, durante o Google Summer of Code 2009. Ele também é autor de alguns tutoriais sobre como escrever seu próprio aplicativo de pintura on-line.

Pintura Web

Para acessar uma biblioteca mais profissional, confira o Pixati.

Mais diversão com o canvas?

Paul Irlanda combinou Harmony e US $1 de Reconhecedor de Unistroke para criar um Easter Egg no site dele.

Você também pode aprender a inspecionar a tela com o Chrome DevTools usando nossos recursos de inspeção recentes.

Saiba mais sobre outros tutoriais sobre o canvas