はじめに
Unity1Weekという1週間ゲームジャムに参加し、
カメラを使ったゲームを開発しました。
一応、遊べるレベルまで完成しましたが、他の環境でもちゃんと動くのかが心配です😂#unityroom #unity1week pic.twitter.com/1gwXC7Bf2M
— KENTO⚽️XRエンジニア😎Shader100記事マラソン挑戦中36/100 (@okprogramming) May 4, 2021
口を開けるとオジサンが笑いながらジャンプします。
前例が無い中、いろいろと問題を解決してようやく公開できたので
同じ苦労する人が減ることを願いメモします。
カメラへのアクセス
unityroomでカメラを使ったゲームを公開できるかどうかの調査から行いました。
カメラのアクセスを試みるにあたって、
jsのプラグインを作ったりといろいろ検証してました。
結論、iframeで画面にWebGLを埋め込んでいる現時点(2021/06)の形式では
こちらから何かしら手を加えてカメラにアクセスすることはできなかったです。
なので、運営の方に直接問い合わせて許可をいただき、
埋め込まれたリンクを直指定してWebGLを開く形式で対応しました。
概要欄に「ここから遊んでね!」みたいなのを書くというやり方です。
jsプラグイン
カメラへのパーミッション要求はWebCamTextureを呼び出すなど、
カメラを必要とする関数に紐づいて勝手に要求処理が走るのですが、
せっかくなので勉強がてらjsのプラグインを作成してパーミッション要求を行ってみました。
以下C#側からjsを呼び出すサンプルです。
まず.jslib
という拡張子のファイルを作成し、そこにjsの処理を記述します。
保存先ディレクトリはAssets/Plugins
配下です。
mergeInto(LibraryManager.library, {
CameraRequest: function () {
var constraints = { audio: false, video: { width: 640, height: 480 } };
navigator.mediaDevices.getUserMedia(constraints)
.then(function (mediaStream) {
var video = document.querySelector('video');
video.srcObject = mediaStream;
video.onloadedmetadata = function (e) {
video.play();
};
})
.catch(function (err) { console.log(err.name + ": " + err.message); });
},
});
次にC#側のコードです。クラス名がjsと一致していればOKです。
任意の箇所で呼び出し可能です。私は CameraRequest()
をボタン押下時に呼び出すよう実装しました。
[DllImport("__Internal")]
private static extern void CameraRequest();
【参考リンク】:Unity2020でWebGLを試してみた! ~基本編~
OpenCV
カメラの問題は片付きました。
次に、OpenCVで顔の認識を行うサンプルがWebGLで問題なく動作するか検証しました。
下記のサンプルをほぼそのまま流用しました。
【参考リンク】:FaceTracker Example
ただ、結論から言うとFPSが安定しませんでした。
何か削れる処理や軽量化に繋がる画期的方法が無いか模索しましたが、
サンプルそのままでもさほど安定していなかったので限界と判断し早急に諦めました。
model及びcascadeの読み込み
OpenCVではモデルデータと識別器(cascade)の読み込みが必要で、
WebGLの場合はStreamingAssetsとしての利用が可能な実装になっていました。
現時点でunityroomにおいてStreamingAssetsは使用できないため、
Gistにプライベート(リンクを知っている人のみアクセス可)で公開し、参照するようにしました。
【参考リンク】:UnityroomではStreaming Assetsは使えないのでしょうか?
工夫した点
顔の認識が確認できた以降の実装では
キャラクターがジャンプする処理などを追加して挙動を確認していきました。
ステージがクリア不可能でないか等を動かしてデバッグする際に
カメラを付けて口をパクパクして...というのが面倒だったので
入力機能をモジュール化してキー入力と顔入力を切り替えれるようにしました。
DIライブラリにはVContainerを使用しました。
おわりに
unity1weekはエンジニアになって以来、
毎回新しいことに取り組む場として活用させていただいています。
1週間の中盤~終盤にかけて参加したことを後悔する(己の実装力に心が折れる)瞬間もありますが、
他の方の頑張りをSNS等で拝見して自分に鞭を打ちながら毎回走り切れています。(ほぼ遅刻)
運営の方はもちろんのこと、参加者の方、遊んでくれた方、みなさんありがとうございます。
次回も楽しみにしています。