4
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

横国ゲーム制作部Advent Calendar 2022

Day 1

Unityエディタ上でスクリーンショットを撮る

Last updated at Posted at 2022-12-01

はじめに

自分がUnityでゲームを作っている時には、ゲーム画面のスクリーンショットを撮りたい場面が結構あります。
ぱっと思いつくようなことでも、面白かった場面やバグが起きているところ、ストアにアップする用の写真、プロモーション用の絵など、撮影したいシチュエーション・用途は様々に考えられます。
画面全体を撮影したい場合はOSに組み込まれている機能(WindowsならWindows+PrintScreen、macOSならcommand+shift+3などで撮影できる)を使えばいいし、ゲーム画面を動画として撮影したい場合はUnity公式が提供しているパッケージRecorderをインストールすれば簡単にできるのですが、ゲーム画面を写真として撮影するという比較的単純でニーズも多いと思われる機能は、意外とUnityデフォルト状態では提供されていなかったりします。
この記事では、このスクリーンショットを撮るという要求に対して取り得るアプローチを2つ紹介します。

方法1:スクリーンショット撮影用のエディタ拡張スクリプトを書く

1つ目は、自分で撮影用スクリプトを書いてしまう方法です。事前準備に多少時間はかかりますが、拡張性も高くプロジェクトを跨いで使い回せるのでベターな選択肢かなと思います。
一般的にはUnityEngine名前空間で提供されているScreenCapture.CaptureScreenshotを使って実装することになります。以下にはそのスクリプトの実装例をScreenshotUtilとして記載しました。

ScreenshotUtil.cs
#if UNITY_EDITOR
using System;
using System.IO;
using UnityEditor;
using UnityEngine;

public static class ScreenshotUtil
{
    private const string outputDirectory = "Screenshots/";
    private const string fileNamePrefix = "screenshot";

    // ctrl(macならcommand)+alt+Sをショートカットに設定
    [MenuItem("Tools/Screenshot %&S", false)]
    public static void CaptureScreenshot()
    {
        // 出力先ディレクトリがなければ作成する
        if (!Directory.Exists(outputDirectory))
        {
            Directory.CreateDirectory(outputDirectory);
        }

        var now = DateTime.Now;
        // 出力ファイル名を指定
        var fileName = $"{fileNamePrefix}_{now.Year}_{now.Month}_{now.Day}_{now.Hour}_{now.Minute}_{now.Second}.png";
        // 出力パスを指定
        var outputPath = Path.Combine(outputDirectory, fileName);
        ScreenCapture.CaptureScreenshot(outputPath);
        // 撮影したことをログで通知
        Debug.Log($"Captured a screenshot: {fileName}");
    }
}
#endif

このスクリプトをプロジェクト内の任意の場所に置くことで動作します。スクリプト全体を囲う条件付きコンパイルのシンボルにUNITY_EDITORを指定しているので実際のビルドには含まれず、よくエディタ拡張を置くのに使われるEditorディレクトリ下に置かなくても問題ありません。

いくつか注意点があるので少し解説します。

出力先のディレクトリ、Assets以下に出力する場合

private const string outputDirectory = "Screenshots/";

この部分で出力先の指定を行なっていますが、これはプロジェクトのルートディレクトリから見たパスです。上のコードではScreenshots/を指定していますが、下の画像のようにAssets下に置いてUnityエディタ上から見たい場合はAssets/Screenshotsなどのように変更する必要があります。
また、Assets以下のディレクトリを指定した場合、上記のコードだけではスクリーンショット後自動でエディタ上に変更が反映されません。これはAssetDatabaseのリフレッシュ処理などをコード上で行なっていないからですが、このままでも手動でリフレッシュを実行すると下の画像のようにファイルが見えるようになります。コード上でリフレッシュを行いたい場合はやや面倒で、自分が試したところスクリーンショットの撮影タイミングとファイルの保存タイミングにラグがあり、非同期的に保存を待ってからリフレッシュを行う必要がありそうです。
スクリーンショット 2022-12-01 23.21.10.png

出力される画像のサイズ

出力される画像のサイズは、ゲームビューでアスペクトを相対値やフリーに設定していた場合は実際今表示されている解像度になります。アスペクトを1920x1080などの固定値に設定していた場合はそのサイズで出力されます。

ショートカットの指定

[MenuItem("Tools/Screenshot %&S", false)]

MenuItem属性でこのメソッドを実行するメニューの名前とショートカットを指定しています。上記の例ではメニュー名としてTools/Screenshots、ショートカットとしてctrl(macならcommand)+alt+Sを設定しました。ctrlまたはcommandは%、shiftは#、alt(option)は&などのように、文字列で使いたいショートカットを指定できます。
ショートカットの設定はUnityのShortcuts設定からも可能ですが、プロジェクトによって導入するかどうかが変わるようなエディタ拡張に割り振るショートカットは、全体設定では行わない方が無難だと思います(ただしMenuItemで指定する方法も、ショートカット設定が局在化して見づらいと言う問題はあるかもしれません)。
スクリーンショット 2022-12-01 23.33.53.png

方法2:Unityの開発者モードを利用する

自分が最近知ったのが、こちらの開発者モードを利用してスクリーンショットを撮る方法です。
最近アップロードされていたUnity公式のYoutube動画で紹介されていて知りました。

Unityの開発者モードには、About Unityのパネルを表示した状態でinternalと入力することで入ることができます。
スクリーンショット 2022-12-01 23.42.06.png
スクリーンショット 2022-12-01 23.42.16.png
入るとこのように各タブの左上に虹色に光る丸が追加されます。
開発者モードに入っている状態だと、WindowタブにInternalメニューが追加され、その中のScreenshotからスクリーンショットを撮影できるようになります。
これで撮影できるならこれでいいと言う気もしますが、常時開発者モードで作業することは推奨されておらず、また保存場所も選べないので、事前に準備する時間があるならやはり自分でスクリプトを書いた方がいいと個人的には思います。
この方法でスクリーンショットを取った場合は、もう一度About Unityのパネルを表示した状態でinternalと入力して開発者モードから抜けることを忘れないようにしましょう。
スクリーンショット 2022-12-01 23.50.10.png

さいごに

デフォルトでスクリーンショットを撮る機能はあってもいいような気がしますが、何か導入するに際して弊害があるのでしょうか。

4
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?