Introdução
Neste artigo, vou discutir como usar o elemento canvas (tela) do HTML5 para criar, editar, abrir e exportar imagens. Também vou apresentar várias ferramentas de código aberto relevantes para essa tecnologia e dar algumas dicas sobre como essas técnicas podem ser aplicadas a um aplicativo da Web existente.
Verificar o suporte a tela
A primeira coisa a fazer é verificar se o navegador oferece suporte total à tela HTML5. Uma maneira fácil de fazer isso é usar o Modernizr para verificar 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 URI de dados ou binário
Primeiro, você precisa ter um elemento de tela na página. Com o 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 é receber o contexto 2D, que dá acesso à API que define todos os métodos e propriedades de exibição. 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 no elemento canvas. Você também pode usar um URI de dados em vez do URL de uma imagem. Em vez do URL acima, faça o seguinte:
img.src=""
Você pode se perguntar: "Por que usar o URI de dados em vez da imagem binária?" Há muitas vantagens. Mais adiante neste artigo, você vai conferir como é fácil exportar uma imagem de tela como URI de dados. Confira uma ferramenta para converter um arquivo de imagem binário em um URI de dados.
Como manipular a imagem da tela
Se você já fez algum tipo de programação de logotipos, desenhar em uma tela usa o mesmo conceito. Mark Pilgrim tem um capítulo sobre canvas 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:
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 que isso, mas deixo para os outros tutoriais listados no apêndice deste artigo instruções adicionais sobre esse assunto. Ainda não vimos nada muito interessante, mas isso vai mudar na próxima seção.
Exportar 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 usada acima.
window.open(document.getElementById('ctx').toDataURL("image/png"));
Isso exporta a tela como uma imagem PNG em uma nova janela do navegador. No entanto, a imagem não é uma imagem binária comum, mas 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 isso 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 do problema no Chrome.
Como integrar ao seu app da Web
O Canvas pode ser um complemento muito poderoso para qualquer aplicativo da Web que armazene imagens enviadas pelo usuário.

Por exemplo, temos um aplicativo de armazenamento de arquivos on-line que armazena imagens enviadas pelo usuário. 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 quiser escrever seu próprio editor de tela, a Harmony é um dos poucos editores de tela disponíveis. Ele tem adição fácil de pincéis, 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')
}
}

Como adicionar o LocalStorage do HTML5
Uma pequena correção que você sempre deve considerar, se se preocupa com a experiência do usuário, é aplicar o LocalStorage. Por exemplo, se você tiver 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 o navegador acidentalmente (ou o navegador trava). O usuário pode ficar frustrado e não se preocupar em 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');
}
//...
}
Salvar a tela como um arquivo binário no servidor
Talvez você queira salvar a imagem da tela como um arquivo binário. Há muitas maneiras de fazer isso. Por exemplo, é possível realizar uma ação POST para transmitir o URI de dados ao código de back-end. Usando o jQuery, 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 base64 no servidor. No 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 duas primeiras linhas, o URI de dados ($contents) é dividido em duas partes. 'data:image/png;base64'
e 'VBORw0KGgoAAAANSUhEUgAAAWwAAAB+CAIAAACPlLzKAAAACXBIWXMAAC4jAAAuIwF4pT92...'
. Em seguida, vamos usar base64_decode()
para decodificar a string de URI de dados. O truque aqui é que há problemas ao decodificar strings maiores que 5K, e essa abordagem de "dividir para conquistar" vai conseguir decodificar a string.
Por fim, usando fwrite(), você pode salvar o arquivo binário, $contents, no servidor.
Como ativar a opção "salvar imagem" no navegador
A tela é um elemento HTML. Ele se parece muito com uma imagem, mas o navegador não oferece a opção "Salvar imagem como" porque ele não é um elemento de imagem. Para ativar a opção "Salvar imagem como", crie dinamicamente um elemento Img e defina o src como 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ê está procurando um editor de tela mais avançado, talvez valha a pena tentar o PaintWeb. Ele foi escrito por Mihai Sucan, um estudante romeno, durante o Google Summer of Code 2009. Ele também escreveu alguns tutoriais sobre como criar seu próprio aplicativo de pintura on-line.

Para uma biblioteca mais profissional, confira o Pixati.
Mais diversão com a tela?
Paul Irish combinou o Harmony e o Recognizer do Unistroke para criar um pequeno 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.
Aprofunde-se com outros tutoriais na tela
- MDN: tutorial de tela
- Introdução ao HTML5: tela
- HTML5 canvas: os conceitos básicos no Dev.Opera aborda as primitivas de desenho básicas.
- Criar um clone do Breakout na tela inclui movimento, física e interatividade básicos