感染シミュレーションの3

昨日の続き。

いきなりだが完成品の感染シミュレーション。 Chrome, FireFox, Safari, Edge のそこそこ新しいバージョンなら動くはず。 入力欄はいずれも正の整数のみで最大2桁。 入力チェックとか何もしていないので、無茶な値を入れないように。

シミュレーションとしてはかなり雑だが、感染拡大における密の恐ろしさは、眺めているとよく判る。 10人に1人で始めると、全員か感染するまでかなり時間がかかるのが、90人に1人で始めると、ほんの数秒で真っ赤だからね。

以下、ソースと若干の説明。

まずはHTMLのcanvasとコントローラー。

<canvas id="field" width="600" height="400"></canvas> <div class="ctrl"> <label> 全人数 <input type="number" min="1" max="99" id="numAll" value="10" /> </label> <label> 感染者 <input type="number" min="1" max="99" id="numBad" value="1" /> </label> <label> 感染率 <input type="number" min="1" max="99" id="numRto" value="10" /> </label> <input type="button" id="btnBgn" value="開始" /> <input type="button" id="btnEnd" value="終了" disabled="true" /> </div>

そしてJavaScript。

class Vct { constructor( x, y ) { this.x = x; this.y = y; } add( vct ) { this.x += vct.x; this.y += vct.y; } } class Rct { constructor( pt, sz ) { this.pt = pt; // ::Vct this.sz = sz; // ::Vct } get xl() { return this.pt.x } // X Left get yt() { return this.pt.y } // Y Top get xr() { return this.pt.x + this.sz.x } // X Right get yb() { return this.pt.y + this.sz.y } // Y Bottom isXOverflow( rct ) { // rctが左右方向で自分からはみ出しているか return ( rct.xl < this.xl || this.xr < rct.xr ); } isYOverflow( rct ) { // rctが上下方向で自分からはみ出しているか return ( rct.yt < this.yt || this.yb < rct.yb ); } isOverlap( rct ) { // rctが自分と重なっているか return !( this.xr < rct.xl || rct.xr < this.xl ) && !( this.yb < rct.yt || rct.yb < this.yt ) } } class Sprite extends Rct { constructor( pt, r, v, s ) { super( pt, new Vct( r * 2, r * 2 ) ); this.r = r; this.v = v; this.s = s; } move() { this.pt.add( this.v ); return this; } refrect( rct ) { if ( rct.isXOverflow( this ) ) this.v.x *= -1; if ( rct.isYOverflow( this ) ) this.v.y *= -1; return this; } overlap( ary, rto ) { const rtoTrue = ( Math.random() * 100 < rto ); this.s = ary.some( sp => this.s || this.isOverlap( sp ) && sp.s && rtoTrue ); return this; } update( ctx ) { ctx.beginPath(); ctx.fillStyle = this.s ? "red" : "blue"; ctx.arc( this.pt.x + this.r, this.pt.y + this.r, this.r, 0, Math.PI * 2, true ); ctx.fill(); ctx.closePath(); } } class Field extends Rct { constructor( cvs ) { super( new Vct( 0, 0), new Vct( cvs.width, cvs.height ) ); this.ctx = cvs.getContext( "2d" ); } spawn( allCnt, badCnt ) { const r = 8; this.sps = Array( allCnt ).fill().map( ( itm, idx ) => new Sprite( new Vct( Math.floor( Math.random() * ( this.xr - this.xl - r * 2 ) ), Math.floor( Math.random() * ( this.yb - this.yt - r * 2 ) ), ), r, new Vct( Math.floor( Math.random() * r - r / 2 ), Math.floor( Math.random() * r - r / 2 ), ), badCnt > idx, ) ); } start( rto ) { this.rto = rto; this.tid = requestAnimationFrame( this.update.bind( this ) ); } stop() { cancelAnimationFrame( this.tid ); } update() { this.ctx.clearRect( this.xl, this.yt, this.xr, this.yb ); this.sps.map( sp => sp.move() ) .map( sp => sp.refrect( this ) ) .map( ( sp, idx, ary ) => sp.overlap( ary, this.rto ) ) .forEach( sp => sp.update( this.ctx ) ); this.tid = requestAnimationFrame( this.update.bind( this ) ); } } window.onload = function() { const qs = ( slctr ) => document.querySelector( slctr ); const qi = ( slctr ) => parseInt( qs( slctr ).value ); const btnBgn = qs( "#btnBgn" ); const btnEnd = qs( "#btnEnd" ); const fld = new Field( qs( "#field") ); btnBgn.addEventListener( "click", () => { fld.spawn( qi( "#numAll" ), qi( "#numBad" ) ); fld.start( qi( "#numRto" ) ); btnBgn.disabled = true; btnEnd.disabled = false; }, false ); btnEnd.addEventListener( "click", () => { fld.stop(); btnBgn.disabled = false; btnEnd.disabled = true; }, false ); };

HTMLの方は、特に言うことは無い。

JavaScriptは、VctとRctは昨日のまま。 新たにRctを継承した Sprite と Field を追加している。 あとは、ボタンにイベントリスナーを設定する初期処理。

Sprite は丸のクラス。 移動(move)と、壁での反射(refrect)と、重なり判定(overlap)と、自分の描画(update)のメソッドを持つ。 重なり判定のメソッド名が overlap でいいのかという疑問はあるが、気にしない。

Field は丸が動ける領域のクラス。 丸の生成(spawn)と、シミュレーション開始(start)と終了(stop)と、描画更新(update)のメソッドを持つ。 spawn で生成した丸の管理もこいつがやる。

余談だけど、わらわら生み出すのが spawn なんだね。 メソッドの名前をどうしようかとちょっと悩んで、 make とか create とか produce とか、その手の単語を久しぶりに辞書で調べた。 結局 「カエルの卵のように大量に」 なんて説明されていた spawn に決めたのだが、この単語を使うのは、俺の人生で今日が初めてかも。

その Field.spawn だが、まず人数分の要素を持つ配列を作り、これを一旦 undefined で埋めて、それから map で Sprite の配列に変換している。 fill しなくてもいけそうなものだが、何故かやらないとうまくいかない。

生成する Sprite の位置と速度は乱数で決めている。 当然だが、初期位置は領域をはみ出さないように。 速度は、あまり速いと壁際の挙動が不自然になるので、半径の値を上限にしている。

描画更新の Field.update がやってるのは、以下の通り。

  1. 一旦canvasをクリアする。

  2. 配列として保持している丸のそれぞれを動かす。

  3. 動かした結果で、壁で反射するものがあれば反射させる。

  4. 反射した結果で重なり合いを判定し、感染した丸と重なっていた場合は感染率に応じて色を変える。

  5. ここまでの結果を描画する。

  6. 次の周期用に requestAnimationFrame に update を登録する。

requestAnimationFrame を使うときによくやるフレームレート対策は、今回は無し。 120fpsのディスプレイだと倍速で表示されてしまうが、倍速だから見えないってこともないだろう。 若干、味気なくなるかもしれないが。 自宅にそんなディスプレイがないので、少なくとも俺は困らない。

シミュレーションとは全然関係無いけど、シミュレーションしてて気付いたことがある。

俺は、最後に残った青を、なんか祈るような気持ちで見てるんだよな。 なんとか無事に逃げてくれと。 まあ、願いは叶わず、すぐに赤くなってしまうのだが。