この記事は Riot.js Advent Calendar 2016 の最終日の記事です。ギリギリの更新ですみません;
ゲームにも使えるRiot.js
Riot.js、いいですよね。
実用的なアプリで利用されている印象ですが、ゲーム開発にも便利に使えそうだと感じたので、試しに使ってみました。
とりあえず3Dっぽい迷路をうろうろするゲームを作ってみましたが、1フレーム単位でシビアなタイミングが要求されるゲームでなければ非常に実用的だと思います。
デモは↓こちら。
Plunkerでのデモ
簡単な解説
- 迷路データ(0:通路、1:壁、など)を配列に格納
- 移動用のボタンを用意し、押下時に現在地データ(座標と方角)を書き換え
- 現在地データと迷路データをもとに、迷路画像などの表示を切り替え
- Plunker便利(Riot.js関係ない)
ソース
やっつけ感漂うソースですみません。特にsetToView
ファンクションのベタベタ感が……。
index.html
<html>
<body>
<mz-main></mz-main>
<script type="riot/tag" src="mz-main.html"></script>
<script src="//cdnjs.cloudflare.com/ajax/libs/riot/3.0.2/riot+compiler.min.js
"></script>
<script>
riot.mount('mz-main',{
current: {
x:2,
y:2,
direction:'E'
},
maze: [
// 0:passageway
// 1:wall
// 9:goal
/*N*/
[1,1,1,1,1,1,1,1,1],
[1,1,1,1,1,1,1,1,1],
[1,1,0,0,0,0,0,1,1],
[1,1,1,0,1,0,1,1,1],
/*W*/ [1,1,1,1,1,0,1,1,1], /*E*/
[1,1,1,0,0,0,0,1,1],
[1,1,1,1,1,1,9,1,1],
[1,1,1,1,1,1,1,1,1],
[1,1,1,1,1,1,1,1,1]
/*S*/
],
view: [
[0,0,0],
[0,0,0],
[0,0,0]
],
message: ""
});
</script>
</body>
</html>
mz-main.html
<mz-main>
<img style="position:absolute;top:4px;left:4px;" src="">
<!--20--><img if="{ opts.view[0][2] == 1 }" style="position:absolute;top:4px;left:4px;" src="">
<!--22--><img if="{ opts.view[2][2] == 1 }" style="position:absolute;top:4px;left:4px;" src="">
<!--21--><img if="{ opts.view[1][2] == 1 }" style="position:absolute;top:4px;left:4px;" src="">
<!--10--><img if="{ opts.view[0][1] == 1 }" style="position:absolute;top:4px;left:4px;" src="">
<!--12--><img if="{ opts.view[2][1] == 1 }" style="position:absolute;top:4px;left:4px;" src="">
<!--11--><img if="{ opts.view[1][1] == 1 }" style="position:absolute;top:4px;left:4px;" src="">
<!--00--><img if="{ opts.view[0][0] == 1 }" style="position:absolute;top:4px;left:4px;" src="">
<!--02--><img if="{ opts.view[2][0] == 1 }" style="position:absolute;top:4px;left:4px;" src="">
<br><br><br><br><br><br><br><br><br><br><br><br><br><br><br><br>
<div>
({ opts.current.x },{ opts.current.y })
{ opts.current.direction }
{ opts.message }
</div>
<table>
<thead></thead>
<tbody>
<tr>
<td></td>
<td><button onclick={ moveToFront }>↑</button></td>
<td></td>
</tr>
<tr>
<td><button onclick={ turnToLeft }>←</button></td>
<td><button onclick={ turnToBack }>↓</button></td>
<td><button onclick={ turnToRight }>→</button></td>
</tr>
</tbody>
</table>
<script>
this.on('mount', function(){
console.log(this);
this.setToView();
this.update();
});
this.moveToFront = function(e) {
var d = opts.current.direction;
var x = opts.current.x;
var y = opts.current.y;
if (d == 'N') y--;
if (d == 'S') y++;
if (d == 'E') x++;
if (d == 'W') x--;
if (opts.maze[y][x] != null && opts.maze[y][x] != 1) {
opts.current.x = x;
opts.current.y = y;
if (opts.maze[y][x] == 9) {
opts.message = "*GOAL*";
} else {
opts.message = "";
}
this.setToView();
} else {
opts.message = "*OUCH*";
}
}
this.turnToLeft = function(e) {
var d = opts.current.direction;
if (d == 'N') opts.current.direction = 'W';
if (d == 'S') opts.current.direction = 'E';
if (d == 'E') opts.current.direction = 'N';
if (d == 'W') opts.current.direction = 'S';
opts.message = "";
this.setToView();
}
this.turnToRight = function(e) {
var d = opts.current.direction;
if (d == 'N') opts.current.direction = 'E';
if (d == 'S') opts.current.direction = 'W';
if (d == 'E') opts.current.direction = 'S';
if (d == 'W') opts.current.direction = 'N';
opts.message = "";
this.setToView();
}
this.turnToBack = function(e) {
var d = opts.current.direction;
if (d == 'N') opts.current.direction = 'S';
if (d == 'S') opts.current.direction = 'N';
if (d == 'E') opts.current.direction = 'W';
if (d == 'W') opts.current.direction = 'E';
opts.message = "";
this.setToView();
}
this.setToView = function() {
opts.view[1][0] = opts.maze[opts.current.y][opts.current.x];
var d = opts.current.direction;
if (d == 'N') {
opts.view[0][2] = opts.maze[opts.current.y-2][opts.current.x-1];
opts.view[1][2] = opts.maze[opts.current.y-2][opts.current.x];
opts.view[2][2] = opts.maze[opts.current.y-2][opts.current.x+1];
opts.view[0][1] = opts.maze[opts.current.y-1][opts.current.x-1];
opts.view[1][1] = opts.maze[opts.current.y-1][opts.current.x];
opts.view[2][1] = opts.maze[opts.current.y-1][opts.current.x+1];
opts.view[0][0] = opts.maze[opts.current.y][opts.current.x-1];
opts.view[2][0] = opts.maze[opts.current.y][opts.current.x+1];
}
if (d == 'S') {
opts.view[0][2] = opts.maze[opts.current.y+2][opts.current.x+1];
opts.view[1][2] = opts.maze[opts.current.y+2][opts.current.x];
opts.view[2][2] = opts.maze[opts.current.y+2][opts.current.x-1];
opts.view[0][1] = opts.maze[opts.current.y+1][opts.current.x+1];
opts.view[1][1] = opts.maze[opts.current.y+1][opts.current.x];
opts.view[2][1] = opts.maze[opts.current.y+1][opts.current.x-1];
opts.view[0][0] = opts.maze[opts.current.y][opts.current.x+1];
opts.view[2][0] = opts.maze[opts.current.y][opts.current.x-1];
}
if (d == 'E') {
opts.view[0][2] = opts.maze[opts.current.y-1][opts.current.x+2];
opts.view[1][2] = opts.maze[opts.current.y][opts.current.x+2];
opts.view[2][2] = opts.maze[opts.current.y+1][opts.current.x+2];
opts.view[0][1] = opts.maze[opts.current.y-1][opts.current.x+1];
opts.view[1][1] = opts.maze[opts.current.y][opts.current.x+1];
opts.view[2][1] = opts.maze[opts.current.y+1][opts.current.x+1];
opts.view[0][0] = opts.maze[opts.current.y-1][opts.current.x];
opts.view[2][0] = opts.maze[opts.current.y+1][opts.current.x];
}
if (d == 'W') {
opts.view[0][2] = opts.maze[opts.current.y+1][opts.current.x-2];
opts.view[1][2] = opts.maze[opts.current.y][opts.current.x-2];
opts.view[2][2] = opts.maze[opts.current.y-1][opts.current.x-2];
opts.view[0][1] = opts.maze[opts.current.y+1][opts.current.x-1];
opts.view[1][1] = opts.maze[opts.current.y][opts.current.x-1];
opts.view[2][1] = opts.maze[opts.current.y-1][opts.current.x-1];
opts.view[0][0] = opts.maze[opts.current.y+1][opts.current.x];
opts.view[2][0] = opts.maze[opts.current.y-1][opts.current.x];
}
}
</script>
</mz-main>