グラフィックスとアニメーション(Canvas とその仲間たち)
核心の問い
昔のウェブページは、文字と画像しか表示できませんでした。しかし、ブロック崩しゲームや华丽な動的エフェクト、自由にドラッグできるデータレポートを作りたい場合、<div> だけでは全く足りません。それが Canvas(キャンバス) が生まれた理由です。
このガイドでは、最初の一本の線を描くところから始め、レベルアップしながら、最終的にブラウザで 60 FPS でスムーズに動くパーティクルエンジンを自らの手で書き上げるまでを案内します。
1. Canvas とは何か?
もし初期のウェブページがレゴブロック(HTML タグ)で組み立てられた静的な模型だとすれば、HTML5 の <canvas> タグは、あなたに巨大なデジタルキャンバスを手渡し、コードで制御される筆を渡して、あとは自由に創造させるようなものです。
ここでの絵にはタグ構造がありません。筆で塗ったものは、一度描かれると純粋な「ピクセルの絵の具」になります。
1.1 Canvas と SVG:二つの異なる流派のアーティスト
フロントエンドの描画世界では、Canvas には SVG というライバルがいます。これらは全く異なる二つの描画観念を代表しています:
- Canvas(ビットマップ描画ボード):
- 原理:実際に紙の上に塗るようなもので、数筆描くと颜料の塊(ピクセル)になります。
- 利点:コンピュータは画面に「絵の具を飛ばす」だけで済むため、パフォーマンスは爆発的!数千の弾むフラッシュパーティクルを同時に描画できます。
- 欠点:一度描いたものは個別に元に戻せず(DOM ノードで選択不可)、拡大するとモザイクでぼやけます。
- SVG(ベクターグラフィックの組み立て):
- 原理:PowerPoint のようなものです。円を描くと、画面上に独立したタグの「円エンティティ」が生成されます。
- 利点:100 倍でも 10 万倍でも拡大しても、常に鮮明。各図形は独立した DOM ノードであり、CSS や JS でいつでも色を変えたり、クリックイベントをバインドしたりできます。
- 欠点:数万個のオブジェクトを飛ばそうとすると、重い DOM ツリーとレイアウトエンジンがブラウザをフリーズさせます。
簡単なまとめ:動的ゲームやクールなパーティクルエフェクトには Canvas を、精密なロゴやインタラクティブな小さなチャートには SVG を使いましょう。
2. 最初の一筆:直感に反する座標系を理解する
2.1 この紙の上下がなぜ逆さまなのか?
描き始める準備ができたら、まず Canvas の定規が逆向きであることを理解する必要があります。伝統的な数学の座標系では、中心のゼロ点が中央にあり、上に行くほど大きくなります。しかし、コンピュータの画面表示分野では、ほぼすべてのデバイスで「原点(0, 0)」が画面の左上隅に定められています。右に進むと X 軸が大きくなるのは問題ありませんが、下に進むと Y 軸が大きくなります。
Canvas 座標系の核心原則:
- ネイティブ単位:ピクセル (px)、画面の物理ピクセルと 1:1 対応。
- X 軸:右が正方向、
0からcanvas.widthまで。 - Y 軸:下が正方向、
0からcanvas.heightまで。
👇 下の小さな点をドラッグして、コンピュータグラフィックスにおける座標の原点と方向を直感的に感じてください:
2.2 魔法の筆に調味料を加える
座標系が整ったら、筆(コードでは Context、略して ctx と呼ばれます)を呼び出せます。実際のパレットで絵を描くのと同じように、Canvas の API 設計は物理的な絵画の三つのステップを完璧に踏襲しています:
- 色の調合(State):
fillStyleで塗りつぶし色、strokeStyleで線の色を設定。 - 形の構築(Path):線(
lineTo)、円(arc)、矩形(rect)のどれを描くかを考えます。 - 最小限の筆運び(Render):内部を塗りつぶす(
fill())か、輪郭を描く(stroke())かを決定。
Canvas は純粋なビットマップキャンバスであり、「一度描いたら戻れない」ため、描いた瞬間にピクセルとして乾き、独立したオブジェクトとして取り消すことはできません。
👇 下のデモで異なる形状と色を選んで、背後のコードが上記の「三つのステップ」をどのように実行しているかを見てみましょう:
3. パラパラ漫画:画面を極めて滑らかに動かす方法
Canvas は一度塗りつぶすと永久的なピクセルになるため、HTML5 ゲームで画面中を走り回るキャラクターはどうやって作られているのでしょうか?
答えは「あなたの目を欺くこと」です。これはパラパラ漫画や映画のフィルムの原理と全く同じです。
- 黒板を消す(Clear):
clearRect()でキャンバス全体の内容を容赦なくクリアする。 - 新しい位置を計算する(Update):キャラクターの X 座標にこっそり 2 ピクセルを加える。
- 描き直す(Render):キャラクターを新しい位置に描き直す。
- 猛烈なループ(Loop):ブラウザ内蔵の極めて正確なメトロノーム
requestAnimationFrameと組み合わせる。ディスプレイのリフレッシュレート(通常は毎秒 60 回、つまり 60 FPS)でこの三つの動作を繰り返します。
人間の目には「視覚の残像」があるため、毎秒 60 回の【消去 → 更新 → 再描画】の中で、あなたが見るのは点滅する黒板ではなく、シルクのように滑らかなアニメーションです。
👇 下のデモで再生速度を調整して、各フレームの変位がどのように滑らかな動きにつながるかを観察してください:
4. 目隠し状態:Canvas 内でクリックインタラクションをどう実現するか?
Canvas はブラウザから見ると、構造のない「絵の具布」にすぎません。キャンバス上に arc() でモンスターを描いたとします。「モンスターをクリックしてダメージを与える」を実装したい場合、従来の document.getElementById を使ってこのモンスターを取得することは不可能です。HTML 構造の中には、幅 600 ピクセルの死んだ <canvas> タグしかないからです。
これがグラフィックスプログラミングにおける最も古典的な問題です:衝突検出(Collision Detection)とイベント委任。
ブラウザはマウスが Canvas の画面座標 (x, y) をクリックしたことしか知らないため、中学の幾何学を使って自分で逆算する必要があります:
- 円の場合:三平方の定理を使って、
マウスクリック位置から円の中心位置までの距離を計算し、距離が半径より小さければ「ヒットした」ことになります。 - 矩形の場合:クリックされた
xが矩形の左右の境界内にあり、同時にyが上下の境界内にあるかどうかを判断します。
キャンバス上にいくつの要素があろうと、マウスのホバーやクリックイベントは常に Canvas という唯一のコンテナにバインドされています。これが究極の「イベント委任」です。
👇 下でマウス(クリック、ドラッグ、ホバー)やキーボード(矢印キーで移動)を使って、「手動で距離を計算する」という低レベルのインタラクションロジックを体験してください:
Instructions / 操作说明
- Click Mode:点击画布创建圆形,按住 Shift 可创建不同颜色
Event Log / 事件日志
5. 計算力の解放:パーティクルシステムとビジュアルマジック
この段階で、「座標系」、「アニメーションループ」、「色と形状」をすべて融合させ、その数を数百や数千の微小な断片にまで増殖させると、視覚を爆発させる究極の武器を手に入れたことになります:パーティクルシステム(Particle System)。
その核心となるアプローチは極めてシンプルかつ効果的です:
- 数百の独立した「パーティクルオブジェクト」を詰め込んだ巨大な配列を作成する。
- 各オブジェクトは独自のライフサイクル(
life)、加速度(vx/vy)、重力ダンピング(gravity)を持つ。 requestAnimationFrameが発火するたびに、これら数百のパーティクルを反復更新し、レンダリングし、最後に「死亡した」(寿命が尽きた/画面外に落ちた)パーティクルを静かにクリーンアップする。
あなたのブラウザは一瞬にして、花火、大雪、爆発を製造するドリームファクトリーになります。
👇 異なるエフェクトをクリックし、重力とパーティクル数を調整して、最もシンプルな物理と数学の公式がどのように複雑な群衆のビジュアルを提示するかを観察してください:
6. FPS の栄光を守る:過熱する CPU にどう対処するか?
数千のオブジェクトを 1 秒間に 60 回計算して再描画することは、非常にパフォーマンスを消費します。無計画に行うと、コンピュータのファンがすぐに飛び立ちます。
以下は、真のエンジンの専門家がフレームレートを救うために使う「護身術」です:
部分消去(ダーティレクタングル Dirty Rect): キャラクターが広大な草原を走っている場合、毎フレーム
clearRectで草原全体を消去しないでください!キャラクターが通過した小さな部分だけを「小消しゴム」で消去して再描画すれば、パフォーマンスは指数関数的に向上します。舞台裏のスタントダブル(オフスクリーン Canvas): 背景が満点の星空で、複雑で絢爛な山脈がある場合、毎回リアルタイムでレンダリングするのは愚かです。通常、メモリ内に見えない
<canvas>を密かに作成し、美しい背景を一度だけ描画します。その後のフレーム更新では、drawImage()を使ってこの合成された「静的ネガ」を直接貼り付けるだけで、膨大な基本計算を省けます。バッチ筆洗い(Batching): パレットで赤から青に切り替えるのは、低レベルではコストがかかります。キャンバスに 1000 個の赤い円と 1000 個の青い円が交互に散らばっている場合、最速の方法は:まず赤い絵の具を準備し、すべての赤い円を描き終えてから、青い絵の具に切り替えてすべての青い円を描くことです。これが有名なバッチレンダリング(Batch Rendering)の考え方です。
👇 オブジェクト数を 3000 以上に引き上げ、ウェブページがカクつきの深淵に落ちるのを見てから、右下の「最適化技術」スイッチを順番にオンにして、実際のフレームレートの救出を目撃してください:
7. 専門用語のまとめ
| 用語 | わかりやすい説明 |
|---|---|
| Canvas | HTML5 が提供する 2D キャンバス。描画は極めて高速だが、描き終わるとピクセルの絵の具になり、DOM 操作で内容を操作することはできない。 |
| SVG | ベクターグラフィックス。拡大しても決してぼやけず、各図形が独立したタグ要素であり、CSS スタイルやインタラクションを簡単にバインドできる。 |
| Context (ctx) | あなたがリクエストした「2D 魔法の筆」で、色の調合、形状の設定、各種特殊効果の描画に使用する。 |
| requestAnimationFrame | ブラウザ内蔵の神レベルのメトロノーム。ディスプレイのリフレッシュレートに厳密に従ってコールバックを実行し、シルキーなアニメーションを作るための唯一の選択肢。 |
| FPS (Frame Rate) | フレームレート。60 FPS は、ブラウザが 1 秒間に 60 回キャンバスをシームレスに消去し、60 枚の新しい絵を再描画していることを意味する。 |
| ダーティレクタングル (Dirty Rect) | 変化があった微小な領域でのみ正確に消去と再描画を行い、パフォーマンスを強力に維持する。 |
| オフスクリーン Canvas | メモリに隠された「影のキャンバス」。極めて複雑だが動かない風景を事前に描画し、後は固定テクスチャとして繰り返し使用する。 |
単純な直線から、壮大で絢爛なパーティクルシステムエンジンまで——魔法のように見えるすべてのエフェクトは、毎秒 60 回の座標計算と再描画のサイクルに過ぎないのです。