async関数とawaitを用いて画像ファイルを読み込んだあとにゲーム画面を描画するようにしている。
クリックしたセル位置に読み込んだ画像をランダムで表示させている。
ソースコード
index.html
1 | <!DOCTYPE html> |
2 | < html lang = "ja" > |
3 | < head > |
4 | < meta charset = "UTF-8" > |
5 | < meta name = "viewport" content = "width=device-width, initial-scale=1.0" > |
6 | < title >OTIMONO(04スプライトをステージに描画する)</ title > |
7 | < style > |
8 | *{ |
9 | margin: 0; |
10 | padding: 0; |
11 | } |
12 | body{ |
13 | margin: 0 50px; |
14 | } |
15 | </ style > |
16 | < script src = "main.js" type = "text/javascript" ></ script > |
17 | </ head > |
18 | < body > |
19 | < canvas id = "canvas" width = "512" height = "512" ></ canvas > |
20 | </ body > |
21 | </ html > |
main.js
1 | // キャンバスでのマウスイベントの状態を表示する |
2 | let canvas = null ; |
3 | let g = null ; |
4 | let pos = {x:0, y:0}; // キャンバス上のマウス座標 |
5 | let cpos = {x:0, y:0}; // マウス座標から得られるセル位置 |
6 | let isMouseDown = false ; // マウスを押しているか? |
7 |
8 | const CELL_SIZE = 64; // パズルキャラのサイズ(64x64ピクセル) |
9 | const ROW_SIZE = COLUMN_SIZE = 512 / CELL_SIZE; // 行と列に何個並べるか? |
10 | const CELL_LENGTH = ROW_SIZE * COLUMN_SIZE; // パズルキャラの数 |
11 |
12 | // ゲームに使う画像ファイル |
13 | const image_files = [ "null.png" , "alien.png" , "apple.png" , "dog.png" , "obake.png" , "ultra.png" , "usi.png" ]; |
14 |
15 | // 画像オブジェクト(画像ファイルをゲームで利用するため) |
16 | let images = []; |
17 |
18 | // ステージ情報配列 |
19 | let stage = new Array(CELL_LENGTH); |
20 |
21 | /* |
22 | * マウスを動かしている時の処理 |
23 | */ |
24 | const mouseMoveListener = (e) => { |
25 | // オフセット位置:キャンバスがブラウザの左上からどれくらいの位置にあるか?(rect.left, rect.top) |
26 | const rect = e.target.getBoundingClientRect(); |
27 |
28 | // e.clientXとe.clientYはブラウザ左上からのキャンバスクリック位置なのでオフセット分を引くとキャンバス上の座標が分かる |
29 | pos.x = e.clientX - rect.left; |
30 | pos.y = e.clientY - rect.top; |
31 |
32 | // マウス座標からセル位置を求める |
33 | cpos.x = Math.floor(pos.x / CELL_SIZE); |
34 | cpos.y = Math.floor(pos.y / CELL_SIZE); |
35 | }; |
36 |
37 | /* |
38 | * マウスボタンを押した時の処理 |
39 | */ |
40 | const mouseDownListener = () => { |
41 | isMouseDown = true ; |
42 | // ステージのクリック位置に画像を設定 |
43 | const index = Math.floor(Math.random() * (images.length-1)) + 1; // ランダム |
44 | stage[cpos.x + cpos.y * COLUMN_SIZE] = index; |
45 | console.log(cpos); |
46 | }; |
47 |
48 | /* |
49 | * マウスボタンを離した時の処理 |
50 | */ |
51 | const mouseUpListener = () => { |
52 | isMouseDown = false ; |
53 | }; |
54 |
55 | /* |
56 | * キャンバスにマウス情報を表示(利用していない) |
57 | */ |
58 | const drawMouseInfo = ()=> { |
59 | // 表示する文字列を設定 |
60 | const mouseDownString = isMouseDown ? "押した!" : "離した!" ; |
61 | const txtPos = `x: ${pos.x} y:${pos.y} ${mouseDownString}`; |
62 |
63 | // マウス情報を表示する |
64 | g.fillStyle = "#555" ; |
65 | g.fillText(txtPos, 150, canvas.height/2); |
66 | } |
67 |
68 | /* |
69 | * 背景を表示 |
70 | */ |
71 | const drawBackground = ()=> { |
72 | let x = 0, y = 0; // 描画するセル位置 |
73 | let color1 = "pink" , color2 = "skyblue" ; // 市松模様のステージ色 |
74 |
75 | for (i=0; i<CELL_LENGTH; i++){ |
76 | // 1段下がると先頭の色を入れ替える |
77 | if (i % 8 == 0) [color1, color2] = [color2, color1]; |
78 | // 格子状にブロックを表示 |
79 | x = i % COLUMN_SIZE; // x方向セル位置 |
80 | y = Math.floor(i / ROW_SIZE); // Y方向セル位置 |
81 |
82 | // 色を交互に設定 |
83 | bgColor = (i % 2 == 0) ? color1: color2; |
84 | g.fillStyle = bgColor; |
85 | g.fillRect(x * CELL_SIZE, y * CELL_SIZE, CELL_SIZE, CELL_SIZE); |
86 | } |
87 | } |
88 |
89 | /* |
90 | * カーソルを表示 |
91 | */ |
92 | const drawCursor = ()=> { |
93 | g.fillStyle = "rgba(128, 128, 128, 0.5)" ; // グレー色で透過 |
94 | g.fillRect(cpos.x * CELL_SIZE, cpos.y * CELL_SIZE, CELL_SIZE, CELL_SIZE); |
95 | } |
96 |
97 | /* |
98 | * ステージを表示 |
99 | */ |
100 | const drawStage = ()=> { |
101 | for (let i=0; i<CELL_LENGTH; i++){ |
102 | x = i % COLUMN_SIZE; // x方向セル位置 |
103 | y = Math.floor(i / ROW_SIZE); // Y方向セル位置 |
104 |
105 | g.drawImage(images[stage[i]], x*CELL_SIZE, y*CELL_SIZE); |
106 | } |
107 | } |
108 |
109 | /* |
110 | * 画像ファイル読み込み |
111 | */ |
112 | const loadImages = async ()=> { |
113 | const images = []; |
114 | const promises = []; |
115 |
116 | image_files.forEach((url) =>{ |
117 | const promise = new Promise((resolve)=>{ |
118 | const img = new Image(); |
119 | img.addEventListener( "load" , ()=>{ |
120 | console.log( "loaded " + url); |
121 | resolve(); // 読み込み完了通知 |
122 | }); |
123 | img.src = "./asset/" + url; // 画像ファイル読み込み開始 |
124 | images.push(img); // 画像オブジェクト配列に格納 |
125 | }); |
126 | promises.push(promise); |
127 | }); |
128 | await Promise.all(promises); |
129 | return images; |
130 | } |
131 |
132 | /* |
133 | * ゲームのメイン処理(まだゲームではない) |
134 | */ |
135 | const GameMain = () =>{ |
136 | // キャンバスクリア |
137 | g.clearRect(0, 0, canvas.width, canvas.height); |
138 | // ステージ描画 |
139 | drawBackground(); |
140 | // カーソル描画 |
141 | drawCursor(); |
142 | // ステージ表示 |
143 | drawStage(); |
144 | // マウス情報表示 |
145 | //drawMouseInfo(); |
146 | // フレーム再描画 |
147 | requestAnimationFrame(GameMain); |
148 | } |
149 |
150 | /* |
151 | * 起動時の処理 |
152 | */ |
153 | window.addEventListener( "load" , async ()=>{ |
154 | canvas = document.getElementById( "canvas" ); |
155 | g = canvas.getContext( "2d" ); |
156 | g.font = "bold 24px System" ; // フォントサイズ・フォント種類 設定 |
157 |
158 | // マウスイベント設定 |
159 | canvas.addEventListener( "mousemove" , mouseMoveListener, false ); |
160 | canvas.addEventListener( "mousedown" , mouseDownListener, false ); |
161 | canvas.addEventListener( "mouseup" , mouseUpListener, false ); |
162 |
163 | // 画像ファイルを読み込む |
164 | images = await loadImages(image_files); |
165 | console.log(images); |
166 |
167 | // ステージ情報を空にする |
168 | stage.fill(0); |
169 |
170 | // ゲームのメイン処理呼び出し(ゲーム開始) |
171 | GameMain(); |
172 | }); |
コメント