はじめに
こんにちは、Unityエンジニアをやっている竹内です。
今回は、「立ち絵の差分絵でアセットの容量が増える」という問題に対して、
「差分絵を1枚の画像に変換する事によって容量を抑える」という方法について
検討・実装したことについてまとめました。
立ち絵を実装する場合下記のような方法がよく使われます
- Live2D
- Spine
- 表情などの差分部位だけ別部品にして実行時に重ねて表示
- 差分の画像をすべて用意する
各手法のメリット・デメリット
Live2D、Spine
メリット
- リッチな表現が可能 (モーションなどで見栄えがとても良くなる)
- アセットの容量が小さい
デメリット
パーツ分け、モーション付けなど行わないといけない為制作コストが高い
表情などの差分部位だけ別部品にして実行時に重ねて表示
メリット
- 表情などの差分絵を用意するだけな為制作コストが低い
・アセットの容量が小さい
デメリット
- 差異部品の位置調整がおかしいとズレる
- 動作差分には不向き
差分の画像をすべて入れる
メリット
- 差分絵を用意するだけな為制作コストが低い
デメリット
- アセットの容量が非常に大きくなる
今回紹介する方法
メリット
- 差分絵を用意し変換処理を行うだけな為制作コストが低い
- ある程度の動作差分は表現できる
- アセットの容量が小さい
デメリット
- 素材がちゃんと差分画像になっていないと成果物の容量が大きくなる
立ち絵差分の画像を今回の手法でアセット作成すると元データの約20%ほどの容量で差分絵表現ができるようになります
成果物
左が上記の画像から生成した画像、右が元の1枚絵の画像
実装の話
実装の大まかな内容は下記になります
- テクスチャを32×32のブロックの集合とみなして1ブロック内のピクセル情報からハッシュ値を生成
- 差分絵全てのブロックハッシュを生成し同じハッシュ値のブロックはスキップ
- テクスチャにブロックのピクセルを書き込み
- 全ての元画像のブロックの位置、ハッシュ値と3で生成したテクスチャのブロックの位置、ハッシュ値を記録したScriptableObjectを生成
- uGUIのGraphicクラスを継承したクラスのOnPopulateMesh内で4の情報を元にUV値をポリゴンに指定して描画
ピクセル補完問題
何もせず上記の実装を行うとピクセル補完をバイリニアなどにした時に問題が発生します
ブロックとブロックの境界(パディングを入れている場合透明なピクセル)が隣のピクセルと補完される事で下記の画像のような境界線がくっきり出てしまう問題が発生します
この問題を軽減する為に行ったのが外周に同じ色のピクセルを置けば隣のピクセルを巻き込んでも違和感のない色になるという解決方法です
外周に同じ色のピクセルを置いた状態
同じ色を置く位置に色を付けた状態
拡大(最も外側のピクセルが内側のピクセルと同じ色になっています)
またImageコンポーネントを継承して実装する事でマスクを掛けたりHierarchyの順番を入れ替えるだけで前後関係を制御できるようになります
予算がないけどアセット容量を抑えつつ差分表現をしたい!という時に非常に有用な手法なので是非参考にしてください
この機能の実装のサンプルとソースコードは下記URLから参照する事ができます
- https://github.com/oTAMAKOo/UniModules/tree/master/Scripts/Modules/PatternTexture
- https://github.com/oTAMAKOo/Example-PatternImage
使用方法
- メニューのOpen PatternTexturePackerから下記ダイアログを開きます
- パックしたいテクスチャを選択した状態でCreateを押下します
- 出力先を指定するとパックされたテクスチャが出力されます
BlockSizeは1ブロックのサイズになります
ブロックサイズが小さいと出力データが大きくなる事があります
AlphaMapはピクセルがあるブロックに当たり判定を持たせることができる機能です
出力データが少し大きくなりますがキャラの形に当たり判定を設定したい時などに活用できます
まとめ
テクスチャ容量を抑えつつ差分表現を行う方法を 検討・実装したことについてまとめました 。 結果として画像容量を80%ほど軽減して差分表現を行うことができるようになりました。 最後まで読んでくださりありがとうございます。