Comments
Description
Transcript
Chapter 07 キャンバスに描画する
7 Chapter キャンバスに描画する [Canvas] この章ではCanvasの使い方について説明します。また、 Canvasを使ってリアルタイムゲームと簡単なノベルシステム を作成します。 なお、Canvasのピクセル単位での処理に関し ては別途、11章のWeb Workersで説明しています。 7.1 7.2 7.3 7.4 7.5 7.6 7.7 7.8 Canvas とは ……………………………………………… 170 四角形の描画 …………………………………………… 174 パスの作成と描画 ……………………………………… 177 画像の描画 ……………………………………………… 182 回転、 移動など ………………………………………… 188 文字の表示 ……………………………………………… 194 Canvas を使ったリアルタイムゲームの作成 …… 200 Canvas を使ったノベルシステムの作成 ………… 208 [編集部註]本章では、カラー表現に関する記述が含まれています。本文では モノクロ画像で掲載しておりますが、下記 URL のサポートサイトにカラー画像は 掲載しておりますので、詳しくはこちらをご覧下さい。 http://www.ric.co.jp/book/contents/pdfs/897_support.pdf 一目でわかる「Canvas」のしくみ ❶ getContext(contextID) コンテキストを取得 contextID Canvas 2d clearRect(x,y,width,height) 四角形の範囲を消去 webgl fillRect(x,y,width,height) 塗り潰された四角形を描画 rect(x,y,width,height) いずれか 1 つを指定 四角形のパスを作成 moveTo(x,y) lineTo(x,y) strokeRect(x,y,width,height) 指定座標にペンを移動 ペン位置から指定座標まで直線のパスを作成 枠だけの四角形を描画 (x,y) (x,y) (x,y) width height arc(x,y,r,startA,endA,anticlockwise) beginPath() closePath() isPointInPath(x,y) 正円/円弧のパスを作成 パスの新規作成 パスを閉じる 座標点がパス内にあるか調べる anticlockwise (x,y) (x,y) r startA endA arcTo(x1,y1,x2,y2,r) bezierCurveTo(cp1x,cp1y,cp2x,cp2y,x,y) quadraticCurveTo(cpx,cpy,x,y) 円弧のパスを作成 3 次ベジエ曲線のパスを作成 2 次ベジエ曲線のパスを作成 (cp1x,cpy1) (cp2x,cpy2) (cpx,cpy) r (x,y) (x,y) (x2,y2) 166 (x1,y1) stroke() strokeStyle fill() fillStyle clip() パスの枠を描画 線のスタイル パスを塗り潰す 塗りのスタイル パスをクリップ領域にする lineWidth 線幅を指定(単位はピクセル [px]) lineCap lineJoin miterLimit 線端形状を指定 連結方法を指定 角のとがり具合を指定 butt round square bevel round miter shadowBlur shadowColor shadowOffsetX shadowOffsetY createPattern(image,repetition) 影のぼかしを指定 影の色を指定 影のずれ具合(オフセット)を指定 パターンを作成 no-repeat shadowOffsetX repeat-x repeat-y repeat shadowOffsetY createLinearGradient(x0,y0,x1,y1) createRadialGradient(x0,y0,r0,x1,y1,r1) addColorStop(offset,color) 直線的なグラデーションを作成 円形グラデーションを作成 中間点の位置と色を指定 (x0,y0) (x0,y0) offset r0 r1 color 一目でわかる 「Canvas」のしくみ ① color (x1,y1) (x1,y1) translate(x,y) rotate(rad) scale(x,y) 移動 回転 拡大縮小/スケーリング rad (x,y) + 167 Chapter 7 globalAlpha 不透明度を指定 キャンバスに描画する [Canvas] globalCompositeOperation 描画モードを指定 一目でわかる「Canvas」のしくみ ❷ fillText(text,x,y,maxWidth) strokeText(text,x,y,maxWidth) mtx = measureText(text) mtx.width 指定座標に通常の文字を描画 指定座標に袋文字を描画 文字幅を持つオブジェクトを返す maxWidth width maxWidth Sample (x,y) Sample (x,y) font textAlign textBaseline 文字のフォントを CSS 形式で指定 行揃えを指定 ベースラインを指定 Sample italic bold 24px Times Sample Sample Sample Sample Sample Abcdfg 描画 Abcdfg 描画 Abcdfg 描画 Abcdfg 描画 Abcdfg 描画 Abcdfg 描画 left center right start end transform(m11,m12,m21,m22,dx,dy) drawImage(image, dx, dy) drawImage(image, dx, dy, dw, dh) drawImage(image, sx, sy, sw, sh, dx, dy, dw, dh) 変形行列を追加 画像を描画 setTransform(m11,m12,m21,m22,dx,dy) 変形行列を新規に設定 m11 m21 dx m12 m22 dy 0 0 1 top hanging middle alphabetic ideographic bottom image (sx,sy) (dx,dy) sh dh sw dw save() restore() Canvas 情報を保存 Canvas 情報を復元 Canvas as strokeStyle,fillStyle,globalAlpha, lineWidth,lineCap,lineJoin,miter Limit,shadowOffsetX,shadowO ffsetY,shadowBlur,shadowColor, globalCompositeOperation,font ,textAlign,textBaseline Canvas as 保存情報 保存情報 保存情報 strokeStyle,fillStyle,globalAlpha, lineWidth,lineCap,lineJoin,miter Limit,shadowOffsetX,shadowO ffsetY,shadowBlur,shadowColor, globalCompositeOperation,font ,textAlign,textBaseline スタック 薄い文字は省略可能なパラメータ 168 (x,y) image (sx,sy) (x,y) h h w w createImageData(width,height) toDataURL(type) 空のピクセルデータを作成 Canvas 内容を URL 形式 (data:∼) に変換 image width Canvas as ... height toDataURL(type) getContext(contextId) lineTo(x,y) lineWidth measureText(text) miterLimit moveTo(x,y) putImageData(image,x,y,sx,sy,w,h) quadraticCurveTo(cpx,cpy,x,y) rect(x,y,width,height) restore() rotate(rad) save() scale(x,y) setTransform(m11,m12,m21,m22,dx,dy) shadowBlur shadowColor shadowOffsetX shadowOffsetY stroke() strokeRect(x,y,width,height) strokeStyle strokeText(text,x,y,maxWidth) textAlign textBaseline transform(m11,m12,m21,m22,dx,dy) translate(x,y) 一目でわかる 「Canvas」のしくみ ② addColorStop(offset,color) arc(x,y,r,startA,endA,anticlockwise) arcTo(x1,y1,x2,y2,r) beginPath() bezierCurveTo(cp1x,cp1y,cp2x,cp2y,x,y) clearRect(x,y,width,height) clip() closePath() createImageData(width,height) createLinearGradient(x0,y0,x1,y1) createPattern(image,repetition) createRadialGradient(x0,y0,r0,x1,y1,r1) drawImage(image, dx, dy, dw, dh) drawImage(image, dx, dy) drawImage(image, sx, sy, sw, sh, dx, dy, dw, dh) fill() fillRect(x,y,width,height) fillStyle fillText(text,x,y,maxWidth) font getImageData(x,y,w,h) globalAlpha globalCompositeOperation isPointInPath(x,y) lineCap lineJoin 薄い文字は省略可能なパラメータ 169 Chapter 7 putImageData(image,x,y,sx,sy,w,h) ピクセルデータを描画 キャンバスに描画する [Canvas] getImageData(x,y,w,h) ピクセルデータを読み出し 7.1 Canvas とは Section この章ではCanvasを使って隕石を破壊するリアルタイムゲームと、画像と文字を表示するノベルシス テムを作成します。 隕石を破壊するゲームは上から落下してくる隕石をタップしてレーザーで破壊するものです。隕石を 破壊すると爆風が広がります。ノベルゲームはあらかじめ用意されたシナリオファイルを読み込みシーン に合わせて画像と文字を表示します。 作成するリアルタイムゲームとノベルシステムの画面は図 7.1.1 のようになります。 実際にこれらのプログラムを作成する前にCanvasについて説明します。 Canvasは2D(平面)に自由にグラフィックを描くことができるものと、WebGLを使って3D(立体)を 描画するものがあります。Android 標準ブラウザではWeb GL(3D)はサポートされていないので、こ こでは2D(平面、二次元描画) のグラフィックスを使ってゲームを作成します(*1)。 2D 描画に関してはW3C のページに仕様が公開されています。 ● HTML Canvas 2D Context ● http://www.w3.org/TR/2dcontext/ Android 2.x ∼ 4で使えるメソッドとプロパティを表 7.1.1 に示します。また、パソコン版 Google ChromeとAndroid 2.x ∼ 4 および Firefoxでの Canvas 関連のメソッド/プロパティ対応に関しては 「付録 4 canvas 要素のメソッド/プロパティ比較表」 を参照してください。 *1 Firefox、Google ChromeではWebGLによる3D 描画が可能です。 170 図 7.1.1 本章で作成するリアルタイムゲームとノベルシステム スコア表示 隕石。上から回転しながら落下。 隕石を破壊 した時の爆風 砲台から発射されたレーザー タップした位置まで描画される ■ノベルシステム senario.txt #img images/waterfall.jpg ♪♬♪ 音楽 シーンに応じて 対応する画像を 表示 #bgm bgm/star.mp3 「ねじだる」の次に現れるのが「霧ヶ滝」です。 エメラルドグリーンの滝壺は非常にきれいで 濁りがありません。 #wait 7.1 Canvasとは シーンに応じて 対応する文字を 表示 (26 文字 ×5 行 ) 文字表示用に 黒の半透明の 四角形を表示 シナリオファイル 171 Chapter 7 キャンバスに描画する [Canvas] ■隕石破壊ゲーム 表 7.1.1 Android の標準ブラウザで使えるメソッドとプロパティ arc 円を描く。 arcTo 円弧を描く。 beginPath パスの作成 bezierCurveTo 3 次ベジエ曲線 clearRect 四角形の範囲を消去 clearShadow 影を消去 clip パスをクリッピング範囲にする。 closePath パスを閉じる。 createImageData ピクセルデータ領域を作成 createLinearGradient 直線的なグラデーションを作成 createPattern パターンを作成 createRadialGradient 円形グラデーションを作成 drawImage 画像を描画 fill パスを塗り潰し fillRect 四角形の範囲を塗り潰し fillStyle 塗り潰しスタイルを設定 fillText 塗り潰した文字を描画 font 文字のサイズやフォントを一括指定(CSS の font と同じ) getImageData Canvas 内のピクセルデータの取得 globalAlpha 不透明度(0 が完全な透明で 1 が完全な不透明) globalCompositeOperation 合成モード(指定可能なモードについては表 7.1.2 を参照) isPointInPath 指定した点がパス内にあるか調べる。 lineCap 線端形状 lineJoin 線端結合方法 lineTo 現在の座標から指定した座標まで直線のパスを作成 lineWidth パスの線幅 measureText 文字の横幅をピクセル数で返す。 miterLimit 線端結合時の限界値 moveTo 指定座標に移動 putImageData ピクセルデータを描画 quadraticCurveTo 2 次ベジエ曲線 rect 四角形のパスを作成 restore Canvas の状態を戻す。 rotate 回転 save Canvas の状態を保存 scale 拡大縮小/スケール設定 setTransform 新規に変形マトリクスを設定 172 shadowColor 影の色 shadowOffsetX 影の横のずれ具合(オフセット) shadowOffsetY 影の縦のずれ具合(オフセット) stroke パスを線で描画 strokeRect 線だけの四角形を描画 strokeStyle 線のスタイル strokeText 線だけの文字を描画(アウトライン文字) textAlign 行揃え textBaseline 文字のベースライン transform 現在の変形マトリクスに新たな変形を追加 translate 移動 表 7.1.2 合成モード 合成モード source-atop destination-out source-in* destination-over source-out* lighter source-over copy* destination-atop* xor destination-in* *仕様とは異なる描画結果になる (Android 2.x /標準ブラウザの場合) 本章で作成するリアルタイムゲームはCanvas の以下の機能を使って処理しています。また、ノベル システムでは以下のうち (1) (4) (7) を使用しています。 (1)画面の消去(clearRect) 7.1 Canvasとは (2)直線の描画(beginPath,strokeStyle,moveTo, lineTo) (3)円の描画(arc) (4)画像の描画(drawImage) (5)回転機能(rotate, translate) (6)文字描画(font, fillText) (7)半透明処理(globalAlpha) 以後のセクションで、これらの機能について説明していきます。 173 Chapter 7 影のぼかし具合 キャンバスに描画する [Canvas] shadowBlur 7.2 四角形の描画 Section このセクションではCanvasと描画機能について説明します。なお、Canvas 機能の説明に関する以 後のセクションではjQuery Mobileは使用していません。純粋にJavaScriptだけを使用しています。 まず、四角形を描く方法について説明します。CanvasはHTML 要素だけでは何も描画することが できません。同じグラフィック機能を扱うSVG(Scalable Vector Graphics)は要素を記述すれば描く ことができますが、CanvasではJavaScript が必須となります。 ● canvas 要素を用意する 描画するまでには手順が必要になります。最初にHTMLでcanvas 要素を用意します。widthで横 幅、heightで縦幅を指定します。CSSで指定することもできますが表示結果が異なる場合があるので HTML の width、height 属性で指定する方が確実です。また、以下の例ではJavaScript からアク セスしやすくするためにID 名 myCanvasを指定しています。 <canvas id="myCanvas" width="300" height="400"></canvas> ● Java Script 側の処理 HTML の準備はこれで終わりです(*1)。次にJavaScript 側の処理です。まず、canvas 要素にアク セスしやすくするために変数にcanvas への参照を入れておきます。 var canvasObj = document.getElementById("myCanvas"); *1 Androidでは、どの機種であってもCanvasを利用できますが、もしIE8などパソコンの古いブラウザを対象にする場合には以下 のようにフォールバック機構を利用してメッセージ等を表示することができます。 <canvas id="myCanvas" width="300" height="400"> お使いのブラウザは対象外です。 </canvas> 174 次に、使用するCanvas の 2D 機能のコンテキスト(オブジェクト)を指定します。これは以下のように ここまで来たら、ようやくCanvasに描画できるようになります。赤色の四角形を描いてみましょう。 fillStyleプロパティに塗り潰す色を指定します。色指定はCSS3と同じものを利用できます。 context.fillStyle = "red"; 塗り潰された四角形の場合は以下のようにfillRect()を使います。 context.fillRect(20, 30, 200, 250); パラメータはX 座標、Y 座標、横幅、縦幅の順番になります。四角形を描くメソッドは表 7.2.1 に示 すものがあります。いずれもパラメータは同じです。実際のプログラムはサンプル 7.2.1 になります。 表 7.2.1 四角形を描くメソッド 図 7.2.1 clearRect(x, y, 横幅 , 縦幅) fillRect(x, y, 横幅 , 縦幅) strokeRect(x, y, 横幅 , 縦幅) rect(x, y, 横幅 , 縦幅) は、実行しても何も描画されず「パス」だけが生成されます。Canvasには「直接描画する」 ものと 「パス を作成した後で描画する」タイプの 2 つがあります。前者は四角形といったシンプルな図形だけが対応 しています。後者は複雑な図形を描画するのに利用されます。 次のセクションでは「パスを作成した後で描画する」方法について説明します。 175 7.2 四角形の描画 四角形を描くメソッドはrect 以外はそのままcanvasに描画 / 消去の処理が行われます。rect の場合 Chapter 7 var context = canvasObj.getContext("2d"); キャンバスに描画する [Canvas] getContext("2d")とすることでCanvas の機能にアクセスできるオブジェクトが変数に入ります。 ●サンプル 7.2.1 ● <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content= "initial-scale=1"> <title>Canvas サンプル </title> </head> <body> <canvas id="myCanvas" width="300" height="400"></canvas> <script src="js/draw.js"></script> </body> </html> ●サンプル 7.2.1 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.fillStyle = "red"; // 赤色 context.fillRect(20, 30, 200, 250); // 塗り潰した四角形 176 パスの作成と描画 Section このセクションではパスの作成と描画について説明します。パスという用語はディレクトリパスのようにも 使われることがありますが、Canvasでのパスは「複数の座標点で構成された図形」を示します。複数 の座標点で構築することができるため、より複雑な図形を描くことができます。パスを構築するメソッドに は表 7.3.1 のものがあります。 表 7.3.1 パスを構築するメソッド arc(x, y, 半径 , 開始角度 , 終了角度 , 描画方向) arcTo(x1, y1, x2, y2, 半径) bezierCurveTo(制御点座標 1x, 制御点座標 1y, 制御点座標 2x, 制御点座標 2y, x, y) lineTo(x, y) moveTo(x, y) quadraticCurveTo(制御点座標 x, 制御点座標 y, x, y) rect(x, y, 横幅 , 縦幅) パスを使う場合にはあらかじめ beginPath()を呼び出します。このメソッドにより以前のパスは消去さ れ新たなパスが生成されます。beginPath()を実行した時点では座標点は何もありません。 簡単なところで直線を2 本描画してみます。直線を描画するには開始点と終了点の 2 つが必要にな 実際のプログラムはサンプル 7.3.1 になります。山形になった赤い直線が描画されます。なお、線の 色はstrokeStyleプロパティに設定します。CSS3 のカラー指定と同じものを使うことができます。 177 7.3 stroke()メソッドを使います。 パスの作成と描画 ります。まず、開始点はmoveTo()で指定し、終了点はlineTo()で指定します。線を描画するには Chapter 7 キャンバスに描画する [Canvas] 7.3 図 7.3.1 山形の直線が描画される。 ●サンプル 7.3.1 ● <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content= "initial-scale=1"> <title>Canvas サンプル </title> </head> <body> <canvas id="myCanvas" width="400" height="400"></canvas> <script src="js/draw.js"></script> </body> </html> ●サンプル 7.3.1 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.strokeStyle = "red"; // 赤色 context.beginPath(); context.moveTo(10, 350); context.lineTo(150, 10); context.lineTo(290, 350); context.stroke(); 178 ●オープンパスとクローズパス いるものです。サンプル7.3.1は始点と終点がつながっていないのでオープンパスになります。 図形をクローズパスにするにはclosePath()を使います。このメソッドを実行すると始点と終点が自動 的に結ばれパスが閉じられます。サンプル7.3.1を修正してクローズパスにしたものがサンプル 7.3.2 で す。クローズパスにしたので三角形が描かれています。 また、線が太くなっています。線の太さはlineWidthで指定することができます。4を入れると4ピク セル幅の太さになります。なお、CSSとは異なり単位の指定はできません。 図 7.3.2 クローズパスにしたので三角形が描かれている。 HTML 文は、サンプル7.3.1と同じコードなので省略します。 ●サンプル 7.3.2 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.strokeStyle = "red"; // 赤色 context.lineWidth = 4; context.beginPath(); context.moveTo(10, 350); context.lineTo(150, 10); 次頁へ続く 179 7.3 パスの作成と描画 ●サンプル 7.3.2 ● Chapter 7 と終点がつながっていないものです。もう1 つがクローズパスと呼ばれるもので始点と終点がつながって キャンバスに描画する [Canvas] beginPath()によって作成されたパスには2 種類あります。1 つはオープンパスと呼ばれるもので始点 context.lineTo(290, 350); context.closePath(); context.stroke(); ●より複雑な図形を作成するには 他のメソッドと組み合わせることで、より複雑な図形も作成することができます。サンプル 7.3.3 は円弧 と直線を組み合わせて、ソフトクリームのような形をした図形を描画しています。 図 7.3.3 円弧と直線を組み合わせた図形 ●サンプル 7.3.3 ● HTML 文は、サンプル7.3.1と同じコードなので省略します。 ●サンプル 7.3.3 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.strokeStyle = "red"; // 赤色 context.lineWidth = 20; context.beginPath(); context.arc(120, 200, 80, 0, Math.PI, true); context.lineTo(120, 300); context.lineTo(200, 200); context.closePath(); context.stroke(); 180 ●重なった部分をくり抜くには サンプル 7.3.4 ではサンプル7.3.2で描画した三角形を、もう1 つ描いています。ただし、パスの作成 方向は逆にしてあります。このため、重なった部分はくり抜かれて表示されます。 図 7.3.4 重なった部分はくり抜かれる (くり抜かれた部分は透明になっている)。 ●サンプル 7.3.4 ● HTML 文は、サンプル7.3.1と同じコードなので省略します。 ●サンプル 7.3.4 draw.js ● Android 1.6 ∼ 4まではarc() のバグでドーナツ型を表示できません。このためドーナツ型にするには、自前で円を描くプログラム を作成する必要があります。 181 7.3 *1 パスの作成と描画 var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.fillStyle = "red"; // 赤色 context.beginPath(); context.moveTo(10, 350); // 上向きの三角のパスを作成 context.lineTo(150, 10); // 時計回りにパスを作成 context.lineTo(290, 350); context.closePath(); // パスを閉じる(重要) context.moveTo(70, 350); // 位置を少し右にして上向きの三角のパスを作成 context.lineTo(350, 350); // 反時計回りにパスを作成 context.lineTo(210, 10); context.closePath(); context.fill(); Chapter 7 れを利用するとドーナツのような形も簡単にできます(*1)。 キャンバスに描画する [Canvas] Canvasではパスの作成した向き(ベクトル方向)が異なる場合、重なった部分はくり抜かれます。こ 7.4 画像の描画 Section このセクションでは画像の描画について説明します。 Canvasに画像を描画するにはdrawImage()を使います。この drawImage()はパラメータの数によっ て、描画結果が変わります。描画パターンは表 7.4.1 に示すように3 つあります。 表 7.4.1 drawImage のパラメータ (1)drawImage(image, dx, dy) 指定座標に元画像のサイズで描画 (2)drawImage(image, dx, dy, dw, dh) 指定座標に指定サイズで描画 (3)drawImage(image, sx, sy, sw, sh, dx, dy, dw, dh) 元画像の一部を指定した座標に指定したサイズで描画 image : 画像オブジェクト sx : 描画元の X 座標 dx : 描画先の X 座標 sy : 描画元の Y 座標 dy : 描画先の Y 座標 sw : 描画元の横幅 dw : 描画先の横幅 sh : 描画元の縦幅 dh : 描画先の縦幅 ●指定座標に元画像のサイズで描画 まず、表 7.4.1(1)drawImage(image, dx, dy) の場合から説明します。Canvasに画像を描画す るにはdrawImage()を使うのですが、その際描画する画像データは読み込みが完了している必要が あります。画像データが読み込まれていない場合には何も描画されません。このため、Imageオブジェ クトが読み込まれたら発生するloadイベントを捕捉します。手軽なのはonloadプロパティにイベントハン ドラを設定しておくことです。 イベントハンドラ内でdrawImage()を使ってCanvasに描画を行います。実際のプログラムはサン プル 7.4.1 になります。読み込んだ画像を座標 (0, 0)に元画像と同じサイズで描画しています。なお、 context.drawImage(imageObj, 0, 0);とある部 分 はcontext.drawImage(this, 0, 0);としても同じ です。 182 図 7.4.1 drawImage() による描画 元画像 (image) Canvas (dx, dy) (2)drawImage(image, dx, dy, dw, dh) 元画像 (image) Canvas (dx, dy) dh dw (3)drawImage(image, sx, sy, sw, sh, dx, dy, dw, dh) 元画像 (image) Canvas sh dh sw dw 183 7.4 画像の描画 (dx, dy) (sx, sy) Chapter 7 キャンバスに描画する [Canvas] (1)drawImage(image, dx, dy) ●サンプル 7.4.1 ● <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content= "initial-scale=1"> <title>Canvas サンプル </title> </head> <body> <canvas id="myCanvas" width="300" height="400"></canvas> <script src="js/draw.js"></script> </body> </html> ●サンプル 7.4.1 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); var imageObj = new Image(); imageObj.src = "images/waterfall.jpg"; // 画像の URL imageObj.onload = function(){ context.drawImage(imageObj, 0, 0); // (0,0) に描画 } もしかしたらサンプル7.4.1 のように処理するのは面倒だと 感じる人がいるかもしれません。もし、HTMLファイル内に Canvasに描画する画像データを用意しておくことができるので あれば別の方法もあります。それはdrawImage() の最初のパ ラメータにimg 要素を指定するという方法です。 実際のプログラムはサンプル 7.4.2 のようになります。ページ のレイアウトや状況によりけりですが、この方がシンプルでわか りやすい上に、画像を変更する場合にスクリプトを操作しなくて もよいというメリットもあります。 184 図 7.4.2 HTML に img 要素を指定してあれば、 その要素を指定して描画することができる。 ●サンプル 7.4.2 ● ●サンプル 7.4.2 draw.js ● window.onload = function(){ var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); var imageObj = document.getElementById("photo"); context.drawImage(imageObj, 0, 0); // (0,0) に描画 } ●指定座標に指定サイズで描画 次に表 7.4.1(2)drawImage(image, dx, dy, dw, dh) の 図 7.4.3 画像が指定した幅で描画されている (元サイズの半分)。 場合ですが、パラメータに横幅と縦幅を指定することができま す。サンプル 7.4.3 では座標(0, 0)から横幅 150ピクセル、縦 幅 200ピクセルで画像を描画します。 7.4 画像の描画 185 Chapter 7 キャンバスに描画する [Canvas] <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content= "initial-scale=1"> <title>Canvas サンプル </title> </head> <body> <canvas id="myCanvas" width="300" height="400"></canvas> <img src="images/waterfall.jpg" width="1" height="1" id="photo"> <script src="js/draw.js"></script> </body> </html> ●サンプル 7.4.3 ● HTML 文は、サンプル7.4.1と同じコードなので省略します。 ●サンプル 7.4.3 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); var imageObj = new Image(); imageObj.src = "images/waterfall.jpg"; // 画像の URL context.drawImage(imageObj, 0, 0, 150, 200); } ●元画像の一部を指定した座標に指定したサイズで描画 最後の表 7.4.1(3)drawImage(image, sx, sy, sw, sh, dx, dy, dw, dh)の場合は、やや複雑 でパラメータがたくさんあります。元画像の転送先座標とサイズ、そしてCanvas の描画先の座標とサ イズを一括して指定します。サンプル 7.4.4 の場合は、元画像の(70, 260)から横幅 40ピクセル、縦幅 60ピクセルの範囲をCanvas の座標(50, 70) に横幅 160ピクセル、縦幅 240ピクセルを描画します。 サンプル7.4.4では元画像の一部を拡大して描画していることになります。 図 7.4.4 元画像の一部が拡大されて描画される。 186 ●サンプル 7.4.4 ● ●サンプル 6.4.4 JavaScript(storage.js) ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); var imageObj = new Image(); imageObj.src = "images/waterfall.jpg"; // 画像の URL imageObj.onload = function(){ context.drawImage(imageObj, 70, 260, 40, 60, 50, 70, 160, 240); } 7.4 画像の描画 187 Chapter 7 キャンバスに描画する [Canvas] HTML 文は、サンプル7.4.1と同じコードなので省略します。 7.5 回転、移動など Section このセクションではCanvas の回転と移動、拡大縮小について説明します。 Canvasでは表 7.5.1 に示す変形機能が用意されています。transform() および setTransform()は rotate()、translate()、scale()による変形処理を一括して指定できます。 Canvasでは特定の図形や画像を回転させる機能はなく、表 7.5.1に示すメソッドを使うと以後に描画 される全ての図形や文字に対して変形が適用されます。 表 7.5.1 Canvas の変形機能 rotate(角度) 回転 translate(横の移動量 , 縦の移動量) 移動 scale(横の倍率 , 楯の倍率) 拡大縮小 setTransfrom(m11, m12, m21, m22, 横の移動量 , 縦の移動量) 変形マトリックスを設定(以前の設定は消去される) transfrom(m11, m12, m21, m22, 横の移動量 , 縦の移動量) 現在の変形マトリクスに新たな変形を追加 • m11 ∼ m22は以下の変形マトリクス m11 m21 dx m12 m22 dy 0 0 1 http://himaxoff.blog111.fc2.com/blog-entry-86.html http://www.w3.org/TR/2dcontext/#dom-context-2d-transform ●画像を回転する まず、画像を回転させてみます。回転はrotate()でパラメータには回転角度をラジアンで指定します。 ラジアンなのでMath.PI*2とするとちょうど一回転することになります。角度からラジアンに変換したい場 合には以下の計算式を使います。 角度 × π ÷ 180 rotate() のパラメータが正数の場合は時計回りに、負数の場合は反時計回りに回転します。サンプ ル 7.5.1 では時計回りに20 度画像を回転させています。 188 図 7.5.1 画像が Canvas の左上を基準にして時計回りに 20 度回転した。 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content= "initial-scale=1"> <title>Canvas サンプル </title> </head> <body> <canvas id="myCanvas" width="300" height="400" style="border:1px solid black;"></canvas> <script src="js/draw.js"></script> </body> </html> var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); var imageObj = new Image(); imageObj.src = "images/waterfall.jpg"; // 画像の URL imageObj.onload = function(){ context.rotate(20 * Math.PI / 180); context.drawImage(imageObj, 0, 0); } 189 7.5 回転、移動など ●サンプル 7.5.1 draw.js ● Chapter 7 キャンバスに描画する [Canvas] ●サンプル 7.5.1 ● サンプル7.5.1を実行すると画像は回転して表示されますが、回転の中心が Canvas の左上になって います(原点)。回転の中心の位置を変更したい場合にはtranslate()を使います。translate()は横と 縦の移動量を指定します。これにより回転の中心が指定した座標になります。 に原点を移動させた後、rotate()を使って回転させて サンプル 7.5.2 ではCanvas の座標(150, 200) います。注意しないといけないのは、移動と回転の中心の順番が変われば結果が異なるという点です。 図 7.5.2 Canvas の中心 (150, 200)を基準にして画像が回転する。 ●サンプル 7.5.2 ● HTML 文は、サンプル7.5.1と同じコードなので省略します。 ●サンプル 7.5.2 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); var imageObj = new Image(); imageObj.src = "images/waterfall.jpg"; // 画像の URL imageObj.onload = function(){ context.translate(150, 200); context.rotate(20 * Math.PI / 180); context.drawImage(imageObj, 0, 0); } 190 ● Canvas の状態を保存、 復元するメソッド です。原点の移動量や回転、拡大縮小率などを操作していくと、元に戻す場合には逆の処理を行わ なければいけません。しかし、変形処理が複雑になった場合には簡単に元の状態に戻すのは難しいで しょう。 そこで、便利なのが以下の表 7.5.2 に示す Canvas の状態を保存、復元するメソッドです。 表 7.5.2 Canvas の状態を処理するメソッド save() Canvas の状態を保存 restore() Canvas の状態を復元 save()はCanvas の状態を保存すると書きましたが、これはCanvasで描画される表 7.5.3 に示すプ ロパティ値と変形マトリクスを保存するものです。Canvasに描かれているピクセルデータを保存するわ けではありません(*1)。また、パスなどはリセットされず残ったままになります。 表 7.5.3 保存されるプロパティ strokeStyle fillStyle globalAlpha lineWidth lineCap lineJoin miterLimit shadowOffsetX shadowOffsetY shadowColor globalCompositeOperation font textAlign textBaseline *1 Android 4 以降ではCanvas の内容をtoDataURL()メソッドを使って読み出すことができます。これによりdataURL 形式として ピクセルデータを取得できます。dataURL 形式はテキストなのでCanvas 内容をLocal Storageに保存することもできます。 191 7.5 回転、移動など shadowBlur Chapter 7 に戻さないと以後に描画される図形や画像、文字の描画結果が意図しないものになる場合があるため キャンバスに描画する [Canvas] translate()によって原点を移動させ回転させた後には、元の状態に戻しておく必要があります。元 save()により保存された内容を元に戻すにはrestore()を使います。なお、save()は入れ子(ネスト) に して使うことができます。Canvasを使ったライブラリなどを開発する際には必須の機能です。 サンプル 7.5.3 が save()、restore()を使ったプログラムです。基本的には変形前にsave()で状態保 存し、処理が終わったらrestore()で戻します。save()、restore() がなければ 2 番目に描画される画像 は回転して描画されてしまいます。 図 7.5.3 1 枚目の画像は回転して描画されるが、 2 枚目は元の状態になっているため回転せずに描画されている。 ●サンプル 7.5.3 ● HTML 文は、サンプル7.5.1と同じコードなので省略します。 ●サンプル 7.5.3 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); var imageObj = new Image(); imageObj.src = "images/waterfall.jpg"; // 画像の URL imageObj.onload = function(){ context.save(); context.translate(75, 100); context.rotate(45 * Math.PI / 180); context.drawImage(imageObj, -75, -100); context.restore(); context.drawImage(imageObj, 150, 200); } 192 ●画像を拡大・縮小する サンプル 7.5.4 では図形を回転させた後に横を200%、縦を50% の縮尺にしてから画像を描画してい ます。 図 7.5.4 画像が回転、変形して描画される。 ●サンプル 7.5.4 ● HTML 文は、サンプル7.5.1と同じコードなので省略します。 ●サンプル 7.5.4 draw.js ● 193 7.5 回転、移動など var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); var imageObj = new Image(); imageObj.src = "images/waterfall.jpg"; // 画像の URL imageObj.onload = function(){ context.save(); context.translate(75, 100); context.rotate(45 * Math.PI / 180); context.scale(2, 0.5); context.drawImage(imageObj, -75, -100); context.restore(); } Chapter 7 します。1.0を指定すると100%となります。0.5なら50%、2.0なら200%になります。 キャンバスに描画する [Canvas] 最後に拡大縮小です。拡大縮小はscale()メソッドを使います。パラメータには横と縦の倍率を指定 7.6 文字の表示 Section このセクションでは文字の描画について説明します。 Canvasでは以下の表 7.6.1 に示す文字描画に関する機能が用意されています。なお、Canvasに 描画される文字の方向はcanvas 要素に指定されているスタイルシートの directionとunicode-bidiに 依存します。 表 7.6.1 Canvas の文字描画機能 font フォントを設定 fillText(表示文字 , x, y) 塗り潰された文字を描画 strokeText(表示文字 , x, y) アウトライン文字を描画 measureText(表示文字) 文字の横幅を求める textAlign 行揃え textBaseline ベースライン ●文字描画の基本 描画する文字のフォントやサイズなどはfontプロパティに一括して設定します。fontプロパティには CSS の fontプロパティと同じ内容を指定することができます。例えば「italic bold 64px Times」と設定 すれば Timesフォントで64ピクセルの太字の斜体で文字が描画されます。 また、描画する文字の色はfillText() の場合はfillStyleプロパティに設定されたカラーになります。 strokeText() の場合はstrokeStyleプロパティに設定されたカラーになります。 fillText()、strokeText()とも最初のパラメータに描画する文字列を指定します。なお、画面からは み出しても自動的に改行はされません。また、描画位置を示す座標は文字のベースラインが基準になり ます。 2 番目と3 番目には文字を描画する座標を指定します。行揃えはtextAlignで「start, end, left, right, center」のいずれかを指定します。 サンプル 7.6.1 が塗り潰された文字を描画、サンプル 7.6.2 がアウトライン文字を描画するプログラム になります。 194 図 7.6.1 赤色で文字が描画される。 図 7.6.2 アウトライン文字が描画される。 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content= "initial-scale=1"> <title>Canvas サンプル </title> </head> <body> <canvas id="myCanvas" width="300" height="400"></canvas> <script src="js/draw.js"></script> </body> </html> var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.fillStyle = "red"; context.font = "italic bold 64px Times" context.fillText("Android", 20, 100); 195 7.6 文字の表示 ●サンプル 7.6.1 draw.js ● Chapter 7 キャンバスに描画する [Canvas] ●サンプル 7.6.1 ● ●サンプル 7.6.2 ● HTML 文は、サンプル7.6.1と同じコードなので省略します。 ●サンプル 7.6.2 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.strokeStyle = "red"; context.font = "italic bold 64px Times" context.strokeText("Android", 20, 100); ●文字にアウトラインを付ける 塗り潰された文字にアウトラインを描画する場合にはサンプル 7.6.3 のように別々に描画します。 図 7.6.3 黄色い文字に赤いアウトラインが描画される。 ●サンプル 7.6.3 ● HTML 文は、サンプル7.6.1と同じコードなので省略します。 ●サンプル 7.6.3 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.fillStyle = "yellow"; context.strokeStyle = "red"; context.font = "italic bold 64px Times" context.fillText("Android", 20, 100); context.strokeText("Android", 20, 100); 196 ●影を付ける は正常な動作になっています。 影は以下の表 7.6.2 に示す機能があります。clearShadow()はCanvasに描画されている影を消す のではなく、影を描画しないようにするメソッドです。 表 7.6.2 影に関する機能 shadowBlur 影のぼかし具合 shadowColor 影の色 shadowOffsetX 影の横方向のずれ(オフセット) shadowOffsetY 影の縦方向のずれ(オフセット) clearShadow() 影を消去 実際に文字に影を描画するプログラムがサンプル 7.6.4 です。塗り潰された黄色い文字に青い影が 描画されます。その後、clearShadow()を使って影の描画を行わないようにしているため、アウトライン 文字には影は描画されていません。 図 7.6.4 青い影が描画される。 7.6 文字の表示 197 Chapter 7 3ではshadowOffsetYプロパティに描画方向が逆になるという不具合があります。Android 4 以降で キャンバスに描画する [Canvas] Canvasでは描画する図形や文字に影(シャドウ)を付けることができます。ただし、Android 2.x ∼ ●サンプル 7.6.4 ● HTML 文は、サンプル7.6.1と同じコードなので省略します。 ●サンプル 7.6.4 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.shadowBlur = 5; context.shadowColor = "blue"; context.fillStyle = "yellow"; context.strokeStyle = "red"; context.font = "italic bold 64px Times" context.fillText("Android", 20, 100); context.clearShadow(); context.strokeText("Android", 20, 100); ●不透明度を指定する 文字の不透明度を指定して描画することもできます。Canvasでは不透明度を設定するには2 つの 方法があります。1 つはglobalAlphaプロパティに不透明度 0.0 ∼ 1.0を設定する方法です。これは全 ての描画機能に対して不透明度が設定されます。 もう1 つの方法が fillStyle や strokeStyleプロパティにCSS の rgba() 形式で不透明度含めたカラー 値を設定するものです。この場合、カラーまたは枠の色に関してのみ不透明度が反映され、他には影 響を与えません。 サンプル 7.6.5、サンプル 7.6.6とも文字を半透明にして描画しますが、結果は同じになります。 図 7.6.5 黄色い文字の不透明度を 70% にしているため、 重なった部分は半透明になっている。 198 ●サンプル 7.6.5 ● ●サンプル 7.6.5 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.font = "italic bold 64px Times" context.fillStyle = "red"; context.fillText("Android", 20, 100); context.globalAlpha = 0.7; context.fillStyle = "yellow"; context.fillText("Android", 17, 97); ●サンプル 7.6.6 ● HTML 文は、サンプル7.6.1と同じコードなので省略します。 ●サンプル 7.6.6 draw.js ● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.font = "italic bold 64px Times" context.fillStyle = "red"; context.fillText("Android", 20, 100); context.fillStyle = "rgba(255, 255, 0, 0.7)"; context.fillText("Android", 17, 97); 7.6 文字の表示 199 Chapter 7 キャンバスに描画する [Canvas] HTML 文は、サンプル7.6.1と同じコードなので省略します。 7.7 Canvas を使った リアルタイムゲームの作成 Section このセクションではCanvasを使ったリアルタイムゲームの作成について説明します。 ●ゲームの概要 章の始めにも少し説明しましたが、ここで作成するゲームは上空から回転しながら落下してくる隕石を タップして破壊するものです。タップすると中央の砲台からタップした位置までレーザービームが瞬時に 発射されます。隕石を破壊すると半透明の爆風が広がっていき、やがて消えます。隕石をタップする 際に画面下であれば、より高得点になります。隕石が完全に落下するとゲームオーバーになります。 まとめると以下のようになります。 (1)隕石は上から回転しつつ落下 (2)隕石が画面下まで落下するとゲームオーバー (3)画面をタップすると中央の砲台からレーザービームが出る。 (4)隕石をタップしたら爆風を出す。その後、再度隕石を上から落下させる。 図 7.7.1 実際のゲーム画面 200 ●プログラムの構成と処理の流れ (1)初期設定 (2)ページ読み込み後のゲーム開始処理 (3)ビームの発射処理 (4)隕石の移動や爆風などの処理 これらの処理の説明の前にHTMLとCSS 部分について説明しておきます。この Canvas の章では Canvasに画像を描画する時にはdrawImage()を使うと説明しました。今回のゲームでも図 7.7.1 を見 ればわかるように背景を描画しています。 普通に考えると背景もCanvasに描画することになります。一般的には、そのようになりますが、ここ ではプログラムの負荷を下げて処理速度を向上させるため、Canvas の背景はスタイルシートを使って 表示しています。HTMLファイルの以下の部分が Canvasに背景を設定している部分です。 canvas { position: absolute; top:10px; left:10px; border:1px solid black; background-image: url(images/bg.png); } Canvas自体は何も描画しなければ「透明」ですので、このようにすると背景が表示されることになりま す。この CSS 部分の処理はブラウザが行いますので、下手にJavaScriptを使って自前で処理するより も高速ですし、そもそも背景を描画するプログラムが不要になります。HTML5でゲーム等を作成する のであれば、このような背景との合成方法、利用方法も覚えておくとよいでしょう。 7.7 Canvasを使ったリアルタイムゲームの作成 ●サンプル 7.7.1 ● <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content= "initial-scale=1, user-scalable=no"> <title> 隕石落下防止ゲーム </title> <link rel="stylesheet" href="css/jquery.mobile.css"> <style> canvas { position: absolute; top:10px; left:10px; border:1px solid black; 次頁へ続く 201 Chapter 7 このプログラムは大きく分けると以下の 4 つになっています。 キャンバスに描画する [Canvas] 今回は最初にゲームのプログラムを見てもらいましょう。以下が実際のゲームのプログラムになります。 background-image: url(images/bg.png); } #main { height: 350px; } </style> <script src="js/jquery.js"></script> <script src="js/shoot.js"></script> <script src="js/jquery.mobile.js"></script> </head> <body> <div data-role="page" id="content"> <div data-role="content" id="main"> <canvas id="myCanvas" width="300" height="350"></canvas> </div> </div> </body> </html> ●サンプル 7.7.1 shoot.js ● var g = { // ゲーム関係の情報を入れるグローバルオブジェクト canvasW : 0, // Canvas の横幅 canvasH : 0, // Canvas の縦幅 stoneImg : new Image(), // 隕石の画像オブジェクト stoneSize : 32, // 隕石の幅 context : null, // Canvas の 2D コンテキスト timerID : null, // タイマー変数 flag : false, // タップしたかどうかのフラグ。true= タップした tapx : 0, // タップした X 座標 tapy : 0, // タップした Y 座標 ix : 0, // 隕石の X 座標 iy : 0, // 隕石の Y 座標 dy : 0, // 隕石の移動量(落下量) bx : 0, // 爆風の X 座標 by : 0, // 爆風の Y 座標 bCounter : 1000, // 爆風のカウンタ変数 score : 0, // スコア rad : 0 // 隕石の回転角度(ラジアン) }; $(function(){ // Canvas 関係の読み込みとイベント、タイマー設定 var canvasObj = document.getElementById("myCanvas"); g.canvasW = canvasObj.width; g.canvasH = canvasObj.height; g.context = canvasObj.getContext("2d"); g.stoneImg.src = "images/stone.png"; g.stoneImg.onload = function(){ // 隕石の画像データを読み込み完了後にゲーム開始 g.ix = Math.random() * 200 + 50; // 隕石の出現範囲 g.iy = -g.stoneSize; 202 7.7 Canvasを使ったリアルタイムゲームの作成 }); // タップしたらビームを発射 function startBeam(evt){ // タップされた座標を求める g.tapx = evt.touches[0].pageX; g.tapy = evt.touches[0].pageY; g.flag = true; // 判定。2 点間の距離で判定する var absx = Math.abs(g.tapx - g.ix); var absy = Math.abs(g.tapy - g.iy); var d = Math.sqrt(Math.pow(absx, 2) + Math.pow(absy, 2)); if (d > 50){ return; } // 50px より大きい場合は何もしない g.score = g.score + Math.floor(g.iy); g.bx = g.tapx; // 爆風の座標を設定 g.by = g.tapy; g.bCounter = 0; g.ix = Math.random() * 200+50; g.iy = -g.stoneSize; g.dy = Math.random() * 5 + 2 + g.score/1000; // 落下速度を計算 } // 隕石移動&描画 function moveStone(){ g.context.clearRect(0,0, g.canvasW, g.canvasH); // Canvas 内容を消去 g.iy = g.iy + g.dy; // 隕石を移動(落下) g.context.save(); // コンテキストを保存しておく var centerX = g.ix + g.stoneSize/2; var centerY = g.iy + g.stoneSize/2; g.context.translate(centerX, centerY); // 隕石の中心に移動 g.context.rotate(g.rad); // 回転させる g.context.translate(-centerX, -centerY); // 元に戻す g.rad = g.rad + Math.PI / 9; g.context.drawImage(g.stoneImg, g.ix, g.iy); g.context.restore(); // 保存したコンテキストを元に戻す // 隕石が完全に落下したか調べる if (g.iy > g.canvasH){ clearInterval(g.timerID); alert("Game Over"); // 隕石が落下したのでゲームオーバー } // タップされた場合は一度だけレーザービームを描画する if (g.flag){ g.context.lineWidth = 3; g.context.strokeStyle = "yellow"; g.context.beginPath(); g.context.moveTo(150, 327); 次頁へ続く 203 Chapter 7 キャンバスに描画する [Canvas] g.dy = Math.random() * 5 + 2; g.timerID = setInterval("moveStone()", 100); } canvasObj.addEventListener("touchstart", startBeam, false); g.context.lineTo(g.tapx, g.tapy); g.context.stroke(); g.flag = false; } // 爆風があるなら処理する if (g.bCounter < 10){ g.context.beginPath(); g.context.fillStyle = "yellow"; g.context.globalAlpha = 0.7; // 不透明度を 70% にする g.context.arc(g.bx, g.by, g.bCounter*10, 0, Math.PI*2, false); // 円を描く g.context.fill(); g.context.globalAlpha = 1.0; g.bCounter = g.bCounter + 1; } // スコアを表示する g.context.font = "normal bold 22px Times"; g.context.fillStyle = "red"; g.context.fillText("SCORE "+g.score, 10, 20); } 7.7.1 初期設定とゲーム開始処理 ●初期設定 まず、 (1) の初期設定部分です。これはグローバル変数(オブジェクト)gを用意し、その中にゲーム で使われる情報を格納するようにしています。どのプロパティが何を示しているかはプログラム内のコメン トに記載してあります。 ●ゲーム開始処理 次に(2)のページが読み込まれた後、ゲームを開始する部分です。$(function(){ ∼で示される部分 が該当します。ここではCanvas の横幅や縦幅、コンテキストを読み出してします。その後、隕石の画 像を読み込みます。隕石の画像が完全に読み込まれたら隕石の落下開始位置を設定します。設定が 終わったらインターバルタイマーを使って隕石の移動処理を行うmoveStone() 関数を0.1 秒ごとに呼び出 します。 Android 端末では処理速度にばらつきがあるため、インターバルタイマーの間隔を短くしても効果が ないことがあります(*1)。 最後にCanvasにタッチされた場合に呼び出す関数(イベントリスナー)を設定します。タッチされた場 合にはstartBeam() 関数が呼び出されます。 *1 作成するゲームによりけりですが別解としてsetRequestAnimationFrame()を使う方法もあります。 204 7.7.2 レーザービームの処理 央の砲台からタップした位置までレーザービームを描画します。レーザービームは直線ですのでlineTo() を使って描画することになります。が、実際にはstartBeam() 関数内にはレーザービームを描画する処 理がありません。 レーザービームの描画処理はmoveStone() 関数内で行うようにしています。なぜ、このようになって いるのでしょうか? それはAndroid 2.xとAndroid 3 以降では画面の書き換えタイミングが異なってい るためです。Android 2.xであれば startBeam() 関数内でlineTo()を使って描画しても問題なく画面 に表示されます。 ところが、Android 3 以降では画面の描画タイミングが異なるため、startBeam() 関数内で処理して も表示される場合もあれば、されない場合もあります。そこで、レーザービームが発射された事だけを 示すフラグ (g.flag) をtrueにしています。 次にタップした場所が隕石と同じ場所かどうか調べます。正確にタップするというのはAndroidでは 難しいため、隕石から50ピクセル以内であればヒットしたことにしています。これが以下の部分になりま す。2 点間の距離で判別しています。より正確に判定するのでれば、隕石の中心座標から行うとよいで しょう。 var absx = Math.abs(g.tapx - g.ix); var absy = Math.abs(g.tapy - g.iy); var d = Math.sqrt(Math.pow(absx, 2) + Math.pow(absy, 2)); if (d > 50){ return; } // 50px より大きい場合は何もしない 隕石がタップされたらスコアを加算します。スコアは隕石を下でタップすれば高得点になるのでY 座標 を使って整数値にしています。 g.score = g.score + Math.floor(g.iy); 最後に隕石をタップした場合に広がる爆風の処理です。レーザービームと同様にstartBeam() 関数 内では処理せず、以下のように爆風のサイズを示すカウンタを設定するようにしています。 g.bCounter = 0; 205 7.7 Canvasを使ったリアルタイムゲームの作成 を加算するスコアにします。なお、このゲームではY 座標は小数値になることがあるのでMath.floor() Chapter 7 次に(3)のレーザービームの処理です。レーザービームはタップした位置の座標を取得し、画面下中 キャンバスに描画する [Canvas] ●ビームの発射処理 7.7.3 隕石の移動と描画処理 ●隕石の移動・爆風ほかの処理 最後に (4)の隕石の移動と爆風の処理です。まず、Canvas の内容を消去します。これは、以前の セクションでも解説したclearRect()メソッドを使います。Canvas の内容は消去されますが、CSSで設 定した背景画像はそのままです。もし、CSSで背景画像を設定していない場合は、この後背景画像の 描画処理を行うことになります。リアルタイムゲームでは、 そのような処理を追加するだけで速度が遅くなっ てしまいますので、そのような方法は避けるのがよいでしょう。 g.context.clearRect(0,0, g.canvasW, g.canvasH); // Canvas 内容を消去 隕石の落下処理は簡単で隕石の Y 座標に移動量を加算するだけです。 g.iy = g.iy + g.dy; // 隕石を移動(落下) この隕石は単純に落下するのではなく回転しながら落下してきます。回転は以前のセクションで説 明したようにrotate()メソッドを使います。translate()を使って隕石の中心を回転するようにします。 Canvas の状態を保存、復元するsave()、restore()を忘れないようにしておきます。忘れると描画され る画像などがおかしくなってしまいます。 g.context.save(); // コンテキストを保存しておく var centerX = g.ix + g.stoneSize/2; var centerY = g.iy + g.stoneSize/2; g.context.translate(centerX, centerY); // 隕石の中心に移動 g.context.rotate(g.rad); // 回転させる g.context.translate(-centerX, -centerY); // 元に戻す g.rad = g.rad + Math.PI / 9; g.context.drawImage(g.stoneImg, g.ix, g.iy); g.context.restore(); // 保存したコンテキストを元に戻す 隕石が完全に下まで落下したかどうかはCanvas の高さを超えたかどうかを調べます。落下していた らインターバルタイマーを止めた後にゲームオーバーのメッセージを出します。今回のゲームではアラート ダイアログを使っていますが、Canvas 内にGame Over の文字を描画してもよいでしょう。 if (g.iy > g.canvasH){ clearInterval(g.timerID); alert("Game Over"); // 隕石が落下したのでゲームオーバー } 206 ここまで来れば残っている処理はレーザービームの描画、爆風の描画、スコアの表示だけです。こ なお、Android 2.x 以降では爆風は半透明になりますが、Android 1.6 端末では爆風は半透明にな りません。 if (g.bCounter < 10){ g.context.beginPath(); g.context.fillStyle = "yellow"; g.context.globalAlpha = 0.7; // 不透明度を 70% にする g.context.arc(g.bx, g.by, g.bCounter*10, 0, Math.PI*2, false); // 円を描く g.context.fill(); g.context.globalAlpha = 1.0; g.bCounter = g.bCounter + 1; } 隕石の数を増やしてみる、爆風を複数表示するように改良してみるのもよいでしょう。 7.7 Canvasを使ったリアルタイムゲームの作成 207 Chapter 7 ウンタを1 つずつ増やしていき10 未満の場合のみ arc()を使って黄色い円を描画しています。 キャンバスに描画する [Canvas] れまでのセクションでの説明を理解していれば難しい部分はありません。爆風が広がる処理ですが、カ 7.8 Canvas を使った ノベルシステムの作成 Section このセクションではCanvasを使ったノベルシステムの作成について説明します。ノベルシステムは Canvasを2 枚重ねて表示します。一枚が画像を表示するCanvasで、もう1 つが文字を表示する Canvasです。Canvasを2 枚重ねることで、文字を書き換える際に背景の画像の書き換えを考慮しなく てもよいというメリットがあります。 ●ノベルシステムとシナリオファイル ノベルシステムは、あらかじめ用意されたシナリオファイルに従って画像と文字を表示していくものです。 ノベルゲームであれば条件分岐なども必要になりますが、ここで作成するシステムはそのような機能はあ りません。単純に画像とテキストを表示しBGMを演奏するだけのものです。 シナリオファイルはテキストで構成されており、何もしなければ文字がそのままCanvas 上に描画されま す。ただし、Canvas 上に表示される文字は26 文字×5 行です。この文字数を超えても自動的にスクロー ルして表示する機能はありません。このため、一度に表示される文字は26 文字×5 行に収める必要が あります。 一定数文字を表示したらタップされるのを待つ、画像を切り換える、といった処理はシナリオファイル 内に専用のコマンドを埋め込むことで実現します。ここで作成するノベルシステムでは表 7.8.1 に示すコ マンドを用意しています。今回のシステムでは行頭に# があるものをコマンドとして処理します。 表 7.8.1 使用できるコマンド img URL URL にある画像を表示。ネットワーク上にあるものであれば何でも使用できる。なお、画像は合成で きるので PNG 形式で透明情報が含まれていれば背景に人物を重ねて表示することもできる。 wait 画面がタップされるのを待つ。 bgm URL URL にある音楽を演奏する。Android 2.x ∼ 3 では多重演奏できるが Android 4 では単独演奏のみ。 end 処理を終了。最後の画面が表示されたままになる。 以下の例ではimagesフォルダにあるschool.jpgを表示しbgmフォルダにあるopening.mp3を演奏し ます。その後「ようこそ!」 という文字を表示して画面がタップされるのを待ちます。 208 #wait ノベルシステムでは、このようなシナリオデータを読み込み処理します。実際のプログラムがサンプル 7.8.1 になります。Canvas 部分に関しては、これまでの説明で十分理解できるでしょう。また、BGM 演奏 もAudioオブジェクトを生成した後、play()メソッドで再生しているだけです。 図 7.8.1 ノベルシステムの画面 ●シナリオファイルの読み込みと解析処理 ここではシナリオファイルを読み込み処理する部分について説明します。 まず、シナリオファイルは以下のようにして非同期通信を利用して読み込ます。シナリオファイル名は senario.txtとしています。シナリオデータはsplit()を使って一行単位で分けて配列変数 storyに入れて 想定しています(String.fromCharCode(10) の部分)。 シナリオデータを読み込んだら解析開始の行番号を0にします(pointer = 0 の部分)。その後、実際 に解析を行うメソッドを呼び出します(nsFunc.main() の部分)。 $.get("./senario.txt", null, function(data){ story = data.split(String.fromCharCode(10)); pointer = 0; nsFunc.main(); }); 209 7.8 Canvasを使ったノベルシステムの作成 おきます。これは解析処理を簡単にするためです。なお、改行コードはLF(アスキーコード10 番)を Chapter 7 ようこそ! キャンバスに描画する [Canvas] #img images/school.jpg #bgm bgm/opening.mp3 解析を行う部分は以下のブロックです。 while(true){ var text = story[pointer]; if (text.indexOf("#end") > -1){ endFlag = true; return; } pointer++; if (text.charCodeAt(0) < 32){ continue; } if (text.indexOf("#bgm") > -1){ nsFunc.bgm(text); continue; } if (text.indexOf("#wait") > -1){ return; } if (text.indexOf("#img") > -1){ nsFunc.image(text); continue; } contextText.fillStyle = "white"; contextText.font = "11px bold"; contextText.fillText(text, 10, linePointer, 300); linePointer = linePointer + 18; // 行間隔は 18 ピクセル } 最初にendコマンドを処理します。endコマンドであれば何もする必要がないため returnで関数から 抜けます。 endコマンド以外であれば解析する行を示すポインタ(pointer++)を進めます。もし、空行であれば 特に処理せずにポインタを進めます。 次にbgmコマンドであればコマンドの後に続くURL の曲を演奏します。解析処理は以後に説明する 画像表示と同じなので省略します。 次にタップを待つ waitコマンドです。この場合は何もせず関数から抜けます。タップしたら次に進ま せるには以下のようにCanvasでtapイベントが発生したら解析処理を呼び出すようにします。 $("#textCanvas").bind("tap", nsFunc.main); 次にimgコマンドです。imgコマンドの場合は半角空白で区切られた後に続く文字をURLとして処 理する必要があります。これは一行のデータをsplit(" ")とし空白単位で切り分けます。配列の 2 番目に 格納されるのが URLになるので、これを画像オブジェクトの srcに入れます。後はCanvasに描画すれ ばできあがりです。 image : function(text){ var data = text.split(" "); // 空白区切り var imageObj = new Image(); imageObj.src = data[1]; // 2 番目が画像の URL imageObj.onload = function(){ contextBG.drawImage(this, 0, 0); } }, 210 最後に文字表示です。いずれのコマンドでもない場合はCanvasに文字を表示します。そして、い #evalコマンドを追加し、以後に続く文字列をJavaScriptコードとして実行させるという手法もあります。 この方法だとシステムに変更を加えることなく複雑な処理も実現できます。 図 7.8.2 最初の画面 図 7.8.3 図 7.8.4 画面をタップすると画像と文字、または文字だけが切り替わる (①∼④)。 ① ② 7.8 Canvasを使ったノベルシステムの作成 211 Chapter 7 他にコマンドを追加して条件分岐やローカルストレージへの保存を行うのも面白いでしょう。ちなみに キャンバスに描画する [Canvas] ずれかのコマンドが見つかるまで解析処理を行います。 図 7.8.5 図 7.8.6 ③ ④ 図 7.8.7 bgmコマンドにより音楽の演奏が開始される。 図 7.8.8 タップすることで次々と画像と文字が切り替わる (①∼④)。 ① 212 図 7.8.9 図 7.8.12 endコマンドを処理すると終了する。 なお、 End の文字は透明の PNG 画像になっており、 描画済みの背景画像に重ねて表示している。 ④ 7.8 Canvasを使ったノベルシステムの作成 213 Chapter 7 図 7.8.11 ③ キャンバスに描画する [Canvas] ② 図 7.8.10 ●サンプル 7.8.1 ● <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content= "initial-scale=1, user-scalable=no"> <title> ノベルシステム </title> <link rel="stylesheet" href="css/jquery.mobile.css"> <style> canvas { position: absolute; top:50px; left:10px; } #main { height: 400px; } </style> <script src="js/jquery.js"></script> <script src="js/novel.js"></script> <script src="js/jquery.mobile.js"></script> </head> <body> <div data-role="page" id="content"> <div data-role="header"> <h1> ノベルシステム </h1> </div> <div data-role="content" id="main"> <canvas id="bgCanvas" width="300" height="400"></canvas> <canvas id="textCanvas" width="300" height="400"></canvas> </div> </div> </body> </html> ●サンプル 7.8.1 novel.js ● var story, contextBG, contextText, canvasW, canvasH, pointer; var endFlag = false; $(function(){ // シナリオデータを読み込む $.get("./senario.txt", null, function(data){ story = data.split(String.fromCharCode(10)); pointer = 0; nsFunc.main(); }); // Canvas を初期化し塗り潰す var canvasBgObj = document.getElementById("bgCanvas"); canvasW = canvasBgObj.width; canvasH = canvasBgObj.height; contextBG = canvasBgObj.getContext("2d"); var canvasTextObj = document.getElementById("textCanvas"); contextText = canvasTextObj.getContext("2d"); 214 7.8 Canvasを使ったノベルシステムの作成 }); var nsFunc = { // 解析する main : function(){ if (endFlag == true){ return; } var linePointer = 320; // 文字を表示する開始 Y 座標 nsFunc.clearText(); while(true){ var text = story[pointer]; if (text.indexOf("#end") > -1){ endFlag = true; return; } pointer++; if (text.charCodeAt(0) < 32){ continue; } if (text.indexOf("#bgm") > -1){ nsFunc.bgm(text); continue; } if (text.indexOf("#wait") > -1){ return; } if (text.indexOf("#img") > -1){ nsFunc.image(text); continue; } contextText.fillStyle = "white"; contextText.font = "11px bold"; contextText.fillText(text, 10, linePointer, 300); linePointer = linePointer + 18; // 行間隔は 18 ピクセル } }, // 文字表示用の Canvas をクリア clearText : function(){ contextText.save(); contextText.clearRect(0,0, canvasW, canvasH); contextText.fillStyle = "black"; contextText.globalAlpha = 0.65; contextText.fillRect(0, 300, canvasW, 100); contextText.restore(); }, // 指定された画像を表示 image : function(text){ var data = text.split(" "); // 空白区切り var imageObj = new Image(); imageObj.src = data[1]; // 2 番目が画像の URL imageObj.onload = function(){ contextBG.drawImage(this, 0, 0); } }, // 指定された曲を演奏 bgm : function(text){ var data = text.split(" "); // 空白区切り var audioObj = new Audio(data[1]); // 2 番目が BGM の URL audioObj.play(); } } 215 Chapter 7 キャンバスに描画する [Canvas] nsFunc.clearText(); $("#textCanvas").bind("tap", nsFunc.main); ●サンプル 7.8.1 senario.txt ● #img images/title.jpg ここは木曽谷。 木曽の山奥には、いろいろな滝があります。 今回は南木曽町にある柿其渓谷の滝を紹介しましょう。 #wait 【………… 省 略 …………】 #img images/waterfall.jpg #bgm bgm/star.mp3 「ねじだる」の次に現れるのが「霧ヶ滝」です。 エメラルドグリーンの滝壺は非常にきれいで濁りが ありません。 #wait 【………… 省 略 …………】 #img images/end.png 以上で終わりです。 #end 216 Column Canvas 関係のバグ このバグは Android 4 では修正されています。 ●コード● var canvasObj = document.getElementById("myCanvas"); context = canvasObj.getContext("2d"); context.fillStyle = "red"; context.shadowBlur = 5; context.shadowOffsetX = 4; context.shadowOffsetY = 8; context.shadowColor = "orange"; context.font = "32px Times"; context.fillText("Android", 10, 60); 図 7.8.13 Android 2.x では影が上方向に描画がされている。 図 7.8.14 Android 4 では影が正しく下方向に描画がされている。 本来は下方向に描画されなければならない。 7.8 Canvasを使ったノベルシステムの作成 次頁へ続く 217 Chapter 7 shadowOffsetY プロパティは Android 2.x, 3 では、通常とは逆方向に影が描画されてしまうバ グがあります。正数を指定すると下方向に影が描画されるはずが、上方向に描画されてしまいます。 キャンバスに描画する [Canvas] 【バグ】shadowOffsetY プロパティ 【バグ】arc メソッド Canvas では描画する方向(ベクトル)が異なっている場合、重なったパスの内部はくり抜かれます。 このため、arc() メソッドの最後のパラメータで時計回りに描画するか、反時計回りに描画するかを指 定するフラグがあります。しかし、Android 1.6 ∼ 4.0 の全てにおいて最後のパラメータが正しく反 映されません。 このため、ドーナツ型の図形を描こうとしても、ただの塗り潰された丸になってしまいます。通常の パスを構築した場合は正常に処理されるので arc() メソッドのバグでしょう。 図 7.8.15 Android 4 (Galaxy Nexus) での実行結果。 図 7.8.16 iOS での実行結果。これが正しい結果。 重なった部分がくり抜かれない。 パソコンのブラウザでも同様の結果になる。 ●コード● var canvasObj = document.getElementById("myCanvas"); var context = canvasObj.getContext("2d"); context.fillStyle = "red"; // 赤色 context.beginPath(); context.arc(120, 200, 60, 0, Math.PI*2, true); context.closePath(); context.arc(180, 200, 60, 0, Math.PI*2, false); context.closePath(); context.fill(); 218