描画キャンバスをウェブアプリに統合する

はじめに

この記事では、HTML5 の canvas 要素を使用して画像の作成、編集、表示、エクスポートを行う方法について説明します。また、この技術に関連するいくつかのオープンソース ツールを紹介し、これらの技術を既存のウェブ アプリケーションに適用する方法に関するヒントをいくつか紹介します。

キャンバスのサポートを確認する

まず、お使いのブラウザが 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);
}

このコードでは、最初のステップとして 2D コンテキストを取得します。これにより、すべての描画メソッドとプロパティを定義する API にアクセスできるようになります。次に、画像オブジェクトを作成し、src プロパティにバイナリ画像の場所を設定します。画像が読み込まれたら、drawImage() メソッドを使用してキャンバスの要素に画像をインポートします。 画像の URL の代わりにデータ URI を使用することもできます。そのため、上記の URL の代わりに、次のことができます。

img.src=""

「なぜバイナリ イメージではなくデータ URI を使用するのか」と尋ねられるかもしれません。多くのメリットがあります。この記事の後半では、キャンバス画像をデータ URI として簡単にエクスポートする方法を説明します。 バイナリ画像ファイルをデータ URI に変換するツールをご確認ください。

キャンバス画像の操作

なんらかのロゴ作成を行ったことがある方は、キャンバスに描画するのも同じ考え方です。 Mark Pilgrim の著書『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 としてエクスポートする

キャンバス要素には、パラメータとして MIME タイプを受け取る toDataURL() メソッドがあります。これで、先ほど使用したキャンバスをエクスポートできます。

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

キャンバスが PNG 画像として新しいブラウザ ウィンドウにエクスポートされます。ただし、この画像は通常のバイナリ画像ではなく、Base64 でエンコードされたデータ URI であり、ブラウザでレンダリングできます。したがって、ユーザーの観点からは、バイナリと同等のものの間に違いはありません。上記のコード行はウェブサーバーで実行する必要があることに注意してください。ローカル ファイルに対して toDataURL() を実行すると失敗します。Chrome におけるこの問題のステータスについては、こちらのチケットをご覧ください。

ウェブアプリへの統合

描画キャンバスは、ユーザーがアップロードした画像を保存するあらゆるウェブ アプリケーションの強力なアドオンです。

ボックス キャンバス

たとえば、ユーザーがアップロードした画像を保存するオンラインのファイル ストレージ アプリケーションがあります。編集ボタンを追加して、キャンバス ベースの画像エディタで画像ファイルを開くことができます。独自のキャンバス エディタを作成したくない場合は、公開されている数少ないキャンバス エディタの 1 つである Harmony を使用できます。簡単にブラシを着けられるので、芸術的な好みにもぴったりです。 上図のメニューで [画像を編集] を選択すると、キャンバス エディタが開き、エディタの init() 関数内で、次のようにカスタムの read_file() 関数が呼び出されます。

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')
   }
}
ハーモニー

HTML5 LocalStorage を追加する

ユーザー エクスペリエンスを重視する場合に常に考慮すべきちょっとした修正として、LocalStorage の適用があります。たとえば、ユーザーに多くの情報を入力する必要がある大きなテキスト領域がある場合です。ユーザーがフォームを送信しようとしたときに、誤ってブラウザを閉じてしまった(またはブラウザがクラッシュする)。ユーザーはイライラして、もう一度メッセージを書き直さなくても済む可能性があります。以下のデモでは、データをサーバーに保存するのではなく、単にデータを URI として LocalStorage に保存します。

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

これにより、データ URI を内容とする XHR 呼び出しが作成されます。次に、サーバーで base64 データ URI をデコードする必要があります。たとえば、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
}

最初の 2 行では、データ URI($contents)が 2 つの部分に分割されています。'data:image/png;base64''VBORw0KGgoAAAANSUhEUgAAAWwAAAB+CAIAAACPlLzKAAAACXBIWXMAAC4jAAAuIwF4pT92...' 次に、base64_decode() を使用してデータ URI 文字列をデコードします。ここでのコツは、5, 000 を超える文字列はデコードできず、この「除算統治」アプローチで文字列をデコードできることです。 最後に、fwrite() を使用して、バイナリ ファイル $contents をサーバーに保存できます。

ブラウザで [画像を保存] を有効にする

Canvas は HTML 要素です。これは画像によく似ていますが、お使いのブラウザには画像要素ではないため、[名前を付けて画像を保存] オプションはありません。 [名前を付けて画像を保存] を有効にするには、Img 要素を動的に作成し、src をキャンバス要素のデータ URI に設定します。 canvas2image ユーティリティを使用することもできます。

より高度なキャンバス エディタ

より高度なキャンバス エディタをお探しの場合は、PaintWeb をぜひお試しください。ルーマニア人学生の Mihai Sucan が Google Summer of Code 2009 で執筆しました。また、独自のオンライン ペイント アプリケーションを作成するためのチュートリアルもいくつか作成しました。

ペイントウェブ

専門的な図書館をお探しの場合は、Pixati をご覧ください。

キャンバスがもっと楽しくなりますか?

Paul Irish は Harmony と $1 の Uni ユーザーに Recognizer を組み込み、自身のウェブサイトに小さな Easter Egg を作成しました。

また、Chrome DevTools で最新の検査機能を使用してキャンバスを検査する方法もご確認ください。

キャンバスに関する他のチュートリアルで理解を深める