投稿者 tel | 2016年11月23日

UnityでMenuItemが動かない

MenuItemで追加したメニューから関数が呼ばれなかった場合があったのでメモ。(Unity 5.4.2)

以下のようにMenuItemの関数と同じ名前のオーバーロードされた関数を用意するとメニューは項目が追加されるが選択しても関数が実行されなかった。

public class MenuItemTest : Editor
{
    static void Test(int i)
    {
    }

    [MenuItem("Editor/Test")]
    static void Test()
    {
        Debug.Log("test");
    }
}

ちゃんとメニューには出てくる。
menu_item_ss

ただし以下のようにオーバーロードされた関数をあとに書いておくと実行される。

public class MenuItemTest : Editor
{
    [MenuItem("Editor/Test")]
    static void Test()
    {
        Debug.Log("test");
    }
    static void Test(int i)
    {
    }
}

Unity 5.5では直っていた。
https://issuetracker.unity3d.com/issues/override-existing-menu-items-does-not-work-until-reimporting-scripts

ParticleSystem.Pause を呼び出してパーティクルをポーズして、再度Playして一時停止を解除するとStart Delayの経過時間がリセットされている挙動をする。つまりPauseとPlayを交互に繰り返すと、いつまでたってもパーティクルが放出されない。

エフェクトの一時停止をしようとした場合、Start Delayが設定されているパーティクルは再生時間がずれてしまい困ったことになる。

https://docs.unity3d.com/ja/current/ScriptReference/ParticleSystem-startDelay.html

どうやらバグのようだが、Issue Trackerでは直ったことになっている。

https://issuetracker.unity3d.com/issues/shuriken-particlesystem-dot-pause-resets-start-delay
https://forum.unity3d.com/threads/5-3-5p5-754270-particles-start-delay-still-have-problem.413072/?_ga=1.122247371.1793905693.1359169028

続きを読む…

投稿者 tel | 2016年11月17日

油壺マリンパーク

油壺マリンパークに行ってきた。1700円。

犬と一緒に入場できるみたいで犬を連れているひとが結構いた。

img_20161104_144858

img_20161104_141323

img_20161104_151603
イワトビペンギン

img_20161104_145526
相模湾に面していて景色がいい。

img_20161104_154717
イルカ、アシカショー。演出が凝ってた。

img_20161104_162731

img_20161104_143545
メガマウスの剥製?

コツメカワウソがいっぱいいた。モモンガ探したけれど巣にはいっているようでみつからなかった。

Animatorコンポーネントのインスペクタにアニメーションのパラメータを表示するエディタ拡張を書いた。

動機

AnimatorControllerのパラメータを直接さわってアニメーションを確認していると、実行中でないときに編集してしまいそのままリポジトリにあげられてしまう場合がある。

parameter_ss

そのため、確認するときになるべくパラメータの部分は触らせたくない。

インスペクタの拡張

以下のようにAnimatorにのインスペクタにAnimatorControllerが持っているパラメータを表示した。これなら実行中でないときに編集しても元のAnimatorControllerには影響がない。

animator_inspector_ss

問題点

デフォルトのAnimatorのインスペクタが上書きされた状態なのでメッセージの表示とかがされない。継承して作ろうと思ったがinternalになっていた。

投稿者 tel | 2016年10月17日

すみだ水族館

9月の23日にすみだ水族館に行ってきた。2050円。

img_20160923_144540

img_20160923_123911

img_20160923_124213

img_20160923_125048

テヅルモヅルかなにか。

img_20160923_132147

チンアナゴ

img_20160923_140248

img_20160923_135931

img_20160923_135848

img_20160923_140026

ちょうど金魚展をやっていた。

新しいので内装がきれいでおしゃれだった。料金高めの割には魚の種類は他の水族館に比べると少ない。

img_20160923_150111

曇っていたのでスカイツリーの上の方は見えなかった。

投稿者 tel | 2016年10月12日

UnityのSphereのポリゴン

デフォルトのSphereのポリゴンがUnity4とUnity5で違っていることに最近気づいた。

sphere2 sphere

Unity 4.6.9とUnity 5.4.1のSphereのメッシュ。

Unity5のほうが頂点数が減って、ポリゴン数が増えている。また見た感じ三角形の面積が均一になっている。
UVも変わっているようで、テクスチャを貼り付けた結果が変わっていた。

tile2

上記のテクスチャを貼り付けてみる。

tex_4_ss tex_5_ss

Unity5のほうが極付近UVがきれいになっている。

投稿者 tel | 2016年10月11日

CEDEC2016

今年もCEDECに行ってきました。いろいろあって2日目の8/25、1日だけ。

とくに面白かったのは「ラピッドイテレーションを実現するゲームエンジンの設計」。カプコンのREエンジン開発でのセッション。REエンジンはバイオハザード7で使用しているエンジンらしい。

  • プログラミング言語としてC#を採用している。
  • C#のVMも自作している。
  • またリリース時にはIL2CPPでC++に変換している。(Unityの使ってるIL2CPPとは別物)
  • C#を使用するときに問題になるGCによるフリーズを解消するために専用のGCを作成している。

レッドブルは1年でCEDECで配っているのくらいしか飲まないが今年は飲み忘れた。

img_20160825_175440

CEDEC AWARDはスプラトゥーンが3冠とってた。スプラトゥーンは技術的な話も聞いてみたいのでなにかセッションがあればよかったのに。

テクスチャのインポート設定からMax Sizeで縮小した場合の表示がUnity4とUnity5で違っているみたいなので調べてみた。

元画像(256×256)

tile

インポートの設定でMax Sizeを128、FormatをTrue Colorにする。

import_ss

Unity5(5.4.0)での結果

tile_unity5_ss

Unity4(4.6.9p4)での結果

tile_unity4_ss

Unity4は白と黒の色しか出ていないが、Unity5の方は灰色になってしまっている。

元画像の線やタイルのパターンは2ドットのサイズで描いているので単純に半分に縮小したらUnity4の結果のように周囲の色が混ざらない方が自然だと思う。

ピクセルの配置によっては意図しないアーティファクトが出てしまう場合があるので、Max Sizeよりも大きいテクスチャはあらかじめ縮小しておいたほうがよさそう。

投稿者 tel | 2016年9月10日

一定周期の三角波

前にノコギリ波を作ったので今度は三角波を作る。

一定の周期のノコギリ波

まず高さが1のノコギリ波を作る。

float x = fract(t)

tr4

0.5引く。

tr3

絶対値を取る。

tr2

2倍すると高さが1の三角波になる。

tr
float tt = abs(fract(t) – 0.5) * 2.0

WWWに代わるAPIとしてUnityWebRequestがUnity5.4から正式に使えるようになったのでファイルをダウンロードして保存する処理を作ってみる。

従来の問題点

WWWを使ってファイルをダウンロードして保存しようとした場合、ダウンロードが完了してからWWW.bytesのデータをファイルとして保存する。WWW.bytesにアクセスするとMonoメモリ上にダウンロードしたデータがまるごと確保されてしまい、大きなファイルをダウンロードしようとしたときに一気にMonoのピークが上がってしまう。

特にモバイル系だとメモリが厳しいのでファイルのダウンロード後にゲームを続けるとすぐに落ちてしまう事態に陥る。

DownloadHandlerScript

UnityWebRequestではダウンロードの挙動をdownloadHandlerで変えることができる。

DownloadHandlerScriptを継承したクラスを用意することで、ダウンロードデータの受信時に直接ファイルに書き込むことができる。つまり今までの全部ダウンロードしてからファイル保存から、少しずつダウロードして都度ファイルに書き込むようにできるためメモリの消費が減らせる。
またコンストラクタにあらかじめ確保したバッファを渡すことで、受信時にそのバッファが使われるようになる。毎回バッファが生成されなくなるため断片化しづらくなる。

class FileDownloadHandler : DownloadHandlerScript
{
    FileStream fs;
    int offset = 0;
    int length = 0;

    public FileDownloadHandler(string path, byte[] buffer)
        : base(buffer)
    {
        fs = new FileStream(path, FileMode.Create, FileAccess.Write);
    }
    // データを受信すると呼び出される
    protected override bool ReceiveData(byte[] data, int dataLength)
    {
        fs.Write(data, 0, dataLength);
        offset += dataLength;
        return true;
    }
    // ダウンロードが終わった時に呼び出される
    protected override void CompleteContent()
    {
        fs.Flush();
        fs.Close();
    }
    // ダウンロードするサイズ
    protected override void ReceiveContentLength(int contentLength)
    {
        length = contentLength;
    }
    // downloadProgressの値
    protected override float GetProgress()
    {
        if (length == 0)
            return 0.0f;

        return (float)offset / length;
    }
}

上記のようなファイル保存用のクラスを用意して、インスタンスをUnityWebRequest.downloadHandlerに設定する。Sendを呼び出すとダウンロードが開始される。

実験

それぞれの場合でダウンロードしてファイル保存するのにMonoのメモリ(Profiler.GetMonoHeapSize)がどれだけ増えるか調べた。WWWとUnityWebRequest.Getはダウンロードした後でFile.WriteAllBytesでファイルに保存している。DownloadHandlerScriptには256KBのバッファを渡している。

  • WWW
  • UnityWebRequest.Get
  • DownloadHandlerScript

計測環境

  • Nexus7 2012
  • 13,981,285MBのファイル

結果

  • ダウンロード前: 684,032 (byte)
  • WWW: 14,798,848 (byte)
  • UnityWebRequest.Get: 14,798,848 (byte)
  • DownloadHandlerScript: 1,175,552 (byte)

まとめ

  • メモリのピークが1桁くらい違っているのでUnityWebRequestを使うのは非常に有効
  • 途中で通信が切れた場合とかのエラー処理をちゃんと書かないといけない

参考

Older Posts »

カテゴリー