##記事の概要
OpenCV for Unityという95ドルのアセットに、Unityの画面を録画する「VideoWriter」というシーンがあります。このシーンをWebGLにビルドし、ブラウザ上で録画した動画をパソコンのローカル上に保存する方法を説明します。
##作ったもの
https://aguroshou.github.io/OpenCvForUnityWebGlMovieDownload/
Recボタンを押して録画を開始したあとに、もう一度Recボタンを押して録画を中断します。
スペースキーを押すと録画した動画がローカル上に保存されると思います。
##作り方
OpenCV for Unityをインポートし、「VideoWriter」というシーンを開きます。
MainCameraに付いている、「VideoWriterExample.cs」を以下のプログラムに書き換えます。
追加した部分に「//追加」と書いています。
using UnityEngine;
using UnityEngine.SceneManagement;
using UnityEngine.UI;
using OpenCVForUnity.CoreModule;
using OpenCVForUnity.VideoioModule;
using OpenCVForUnity.ImgprocModule;
using OpenCVForUnity.UnityUtils;
//追加
using System.IO;
#if UNITY_WEBGL
using System.Runtime.InteropServices;
#endif
namespace OpenCVForUnitySample
{
/// <summary>
/// VideoWriter Example
/// An example of saving a video file using the VideoWriter class.
/// http://docs.opencv.org/3.2.0/dd/d43/tutorial_py_video_display.html
/// </summary>
public class VideoWriterExample : MonoBehaviour
{
/// <summary>
/// The cube.
/// </summary>
public GameObject cube;
/// <summary>
/// The preview panel.
/// </summary>
public RawImage previewPanel;
/// <summary>
/// The rec button.
/// </summary>
public Button RecButton;
/// <summary>
/// The play button.
/// </summary>
public Button PlayButton;
/// <summary>
/// The save path input field.
/// </summary>
public InputField savePathInputField;
/// <summary>
/// The max frame count.
/// </summary>
const int maxframeCount = 300;
/// <summary>
/// The frame count.
/// </summary>
int frameCount;
/// <summary>
/// The videowriter.
/// </summary>
VideoWriter writer;
/// <summary>
/// The videocapture.
/// </summary>
VideoCapture capture;
/// <summary>
/// The screen capture.
/// </summary>
Texture2D screenCapture;
/// <summary>
/// The recording frame rgb mat.
/// </summary>
Mat recordingFrameRgbMat;
/// <summary>
/// The preview rgb mat.
/// </summary>
Mat previewRgbMat;
/// <summary>
/// The preview texture.
/// </summary>
Texture2D previrwTexture;
/// <summary>
/// Indicates whether videowriter is recording.
/// </summary>
bool isRecording;
/// <summary>
/// Indicates whether videocapture is playing.
/// </summary>
bool isPlaying;
/// <summary>
/// The save path.
/// </summary>
string savePath;
//追加
#if UNITY_WEBGL
[DllImport("__Internal")]
private static extern void FileDownLoad(byte[] bytes, int size, string filename);
#endif
// Use this for initialization
void Start()
{
PlayButton.interactable = false;
previewPanel.gameObject.SetActive(false);
Initialize();
}
private void Initialize()
{
Texture2D imgTexture = Resources.Load("face") as Texture2D;
Mat imgMat = new Mat(imgTexture.height, imgTexture.width, CvType.CV_8UC4);
Utils.texture2DToMat(imgTexture, imgMat);
Texture2D texture = new Texture2D(imgMat.cols(), imgMat.rows(), TextureFormat.RGBA32, false);
Utils.matToTexture2D(imgMat, texture);
cube.GetComponent<Renderer>().material.mainTexture = texture;
}
// Update is called once per frame
void Update()
{
if (!isPlaying)
{
cube.transform.Rotate(new Vector3(90, 90, 0) * Time.deltaTime, Space.Self);
}
if (isPlaying)
{
//Loop play
if (capture.get(Videoio.CAP_PROP_POS_FRAMES) >= capture.get(Videoio.CAP_PROP_FRAME_COUNT))
capture.set(Videoio.CAP_PROP_POS_FRAMES, 0);
if (capture.grab())
{
capture.retrieve(previewRgbMat, 0);
Imgproc.rectangle(previewRgbMat, new Point(0, 0), new Point(previewRgbMat.cols(), previewRgbMat.rows()), new Scalar(0, 0, 255), 3);
Imgproc.cvtColor(previewRgbMat, previewRgbMat, Imgproc.COLOR_BGR2RGB);
Utils.fastMatToTexture2D(previewRgbMat, previrwTexture);
}
}
//追加
if (Input.GetKeyDown(KeyCode.Space))
{
string filename = "test.avi";
byte[] bytes = System.IO.File.ReadAllBytes(savePath);
if (bytes != null)
{
#if UNITY_EDITOR
string path = Path.Combine(Application.persistentDataPath, filename);
File.WriteAllBytes(path, bytes);
Debug.Log(filename + " has been saved into " + path);
#elif UNITY_WEBGL
FileDownLoad(bytes, bytes.Length, filename);
#endif
}
}
}
void OnPostRender()
{
if (isRecording)
{
if (frameCount >= maxframeCount ||
recordingFrameRgbMat.width() != Screen.width || recordingFrameRgbMat.height() != Screen.height)
{
OnRecButtonClick();
return;
}
frameCount++;
// Take screen shot.
screenCapture.ReadPixels(new UnityEngine.Rect(0, 0, Screen.width, Screen.height), 0, 0);
screenCapture.Apply();
Utils.texture2DToMat(screenCapture, recordingFrameRgbMat);
Imgproc.cvtColor(recordingFrameRgbMat, recordingFrameRgbMat, Imgproc.COLOR_RGB2BGR);
Imgproc.putText(recordingFrameRgbMat, frameCount.ToString(), new Point(recordingFrameRgbMat.cols() - 70, 30), Imgproc.FONT_HERSHEY_SIMPLEX, 1.0, new Scalar(255, 255, 255), 2, Imgproc.LINE_AA, false);
Imgproc.putText(recordingFrameRgbMat, "SavePath:", new Point(5, recordingFrameRgbMat.rows() - 30), Imgproc.FONT_HERSHEY_SIMPLEX, 0.8, new Scalar(0, 0, 255), 2, Imgproc.LINE_AA, false);
Imgproc.putText(recordingFrameRgbMat, savePath, new Point(5, recordingFrameRgbMat.rows() - 8), Imgproc.FONT_HERSHEY_SIMPLEX, 0.5, new Scalar(255, 255, 255), 0, Imgproc.LINE_AA, false);
writer.write(recordingFrameRgbMat);
}
}
private void StartRecording(string savePath)
{
if (isRecording || isPlaying)
return;
this.savePath = savePath;
writer = new VideoWriter();
#if !UNITY_IOS
writer.open(savePath, VideoWriter.fourcc('M', 'J', 'P', 'G'), 30, new Size(Screen.width, Screen.height));
#else
writer.open(savePath, VideoWriter.fourcc('D', 'V', 'I', 'X'), 30, new Size(Screen.width, Screen.height));
#endif
if (!writer.isOpened())
{
Debug.LogError("writer.isOpened() false");
writer.release();
return;
}
screenCapture = new Texture2D(Screen.width, Screen.height, TextureFormat.RGB24, false);
recordingFrameRgbMat = new Mat(Screen.height, Screen.width, CvType.CV_8UC3);
frameCount = 0;
isRecording = true;
}
private void StopRecording()
{
if (!isRecording || isPlaying)
return;
if (writer != null && !writer.IsDisposed)
writer.release();
if (recordingFrameRgbMat != null && !recordingFrameRgbMat.IsDisposed)
recordingFrameRgbMat.Dispose();
savePathInputField.text = savePath;
isRecording = false;
}
private void PlayVideo(string filePath)
{
if (isPlaying || isRecording)
return;
capture = new VideoCapture();
capture.open(filePath);
if (!capture.isOpened())
{
Debug.LogError("capture.isOpened() is false. ");
capture.release();
return;
}
Debug.Log("CAP_PROP_FORMAT: " + capture.get(Videoio.CAP_PROP_FORMAT));
Debug.Log("CAP_PROP_POS_MSEC: " + capture.get(Videoio.CAP_PROP_POS_MSEC));
Debug.Log("CAP_PROP_POS_FRAMES: " + capture.get(Videoio.CAP_PROP_POS_FRAMES));
Debug.Log("CAP_PROP_POS_AVI_RATIO: " + capture.get(Videoio.CAP_PROP_POS_AVI_RATIO));
Debug.Log("CAP_PROP_FRAME_COUNT: " + capture.get(Videoio.CAP_PROP_FRAME_COUNT));
Debug.Log("CAP_PROP_FPS: " + capture.get(Videoio.CAP_PROP_FPS));
Debug.Log("CAP_PROP_FRAME_WIDTH: " + capture.get(Videoio.CAP_PROP_FRAME_WIDTH));
Debug.Log("CAP_PROP_FRAME_HEIGHT: " + capture.get(Videoio.CAP_PROP_FRAME_HEIGHT));
double ext = capture.get(Videoio.CAP_PROP_FOURCC);
Debug.Log("CAP_PROP_FOURCC: " + (char)((int)ext & 0XFF) + (char)(((int)ext & 0XFF00) >> 8) + (char)(((int)ext & 0XFF0000) >> 16) + (char)(((int)ext & 0XFF000000) >> 24));
previewRgbMat = new Mat();
capture.grab();
capture.retrieve(previewRgbMat, 0);
int frameWidth = previewRgbMat.cols();
int frameHeight = previewRgbMat.rows();
previrwTexture = new Texture2D(frameWidth, frameHeight, TextureFormat.RGB24, false);
capture.set(Videoio.CAP_PROP_POS_FRAMES, 0);
previewPanel.texture = previrwTexture;
isPlaying = true;
}
private void StopVideo()
{
if (!isPlaying || isRecording)
return;
if (capture != null && !capture.IsDisposed)
capture.release();
if (previewRgbMat != null && !previewRgbMat.IsDisposed)
previewRgbMat.Dispose();
isPlaying = false;
}
/// <summary>
/// Raises the destroy event.
/// </summary>
void OnDestroy()
{
StopRecording();
StopVideo();
}
/// <summary>
/// Raises the back button click event.
/// </summary>
public void OnBackButtonClick()
{
SceneManager.LoadScene("OpenCVForUnityExample");
}
/// <summary>
/// Raises the rec button click event.
/// </summary>
public void OnRecButtonClick()
{
if (isRecording)
{
RecButton.GetComponentInChildren<UnityEngine.UI.Text>().color = Color.black;
StopRecording();
PlayButton.interactable = true;
previewPanel.gameObject.SetActive(false);
}
else
{
#if !UNITY_IOS
StartRecording(Application.persistentDataPath + "/VideoWriterExample_output.avi");
#else
StartRecording(Application.persistentDataPath + "/VideoWriterExample_output.m4v");
#endif
if (isRecording)
{
RecButton.GetComponentInChildren<UnityEngine.UI.Text>().color = Color.red;
PlayButton.interactable = false;
}
}
}
/// <summary>
/// Raises the play button click event.
/// </summary>
public void OnPlayButtonClick()
{
if (isPlaying)
{
StopVideo();
PlayButton.GetComponentInChildren<UnityEngine.UI.Text>().text = "Play";
RecButton.interactable = true;
previewPanel.gameObject.SetActive(false);
}
else
{
if (string.IsNullOrEmpty(savePath))
return;
PlayVideo(savePath);
PlayButton.GetComponentInChildren<UnityEngine.UI.Text>().text = "Stop";
RecButton.interactable = false;
previewPanel.gameObject.SetActive(true);
}
}
}
}
https://www.fast-system.jp/unity-file-to-bytes-howto/
動画ファイルをBytesに変換する方法はこのサイトを参考にしました。
https://junk-box.net/kuyo/index.php/2019/11/24/panorama-dl/
こちらのサイトのPanorama.csを適当な箇所に追加しただけです。
また、同じサイトに書いてある手順のとおりに、Assets/Plugins/WebGLに“FileDownload.jslib”を作成する必要があります。
https://www.fast-system.jp/unity-webgl-multi-threads/
こちらのサイトを参考にマルチスレッドの設定をすることができます。
この設定でアプリの動作が少し軽くなると思います。(それでも重いですが)
##感想
OpenCV for Unityを使うことで、動画作成ブラウザアプリを作れる可能性が出てきたと思います。
録画機能がかなり重いのですが、うまくプログラムを編集すれば、使えるものになるのかもしれません。
(アプリ内のブロックが回転する速度に、現実の時間に基づかせるTime.deltaTimeを使用していたため、これを変えるだけで録画する動画の質が良くなると思います。)
WebGLからファイルをローカルに保存する方法として有名なアセットは「StandaloneFileBrowser」なのですが、これをOpenCV for Unityと一緒にWebGL向けにビルドして実行しようとすると原因不明のエラーとなって動作しませんでした…。
このアセットには画像ファイルなどをWebGLのUnityアプリにロードする機能があるので、これも追加できればもっといろいろなことができそうです。
#補足
OpenCV for UnityのReadMe.pdfに記載されていることですが、UnityEditorの [Tools/OpenCV for Unity/Set Plugin Import Settings]の設定が必要かもしれません。
MacでWebGLにビルドするとアプリ起動時にエラーが発生し、Windowsではエラーが発生しませんでした。原因は特定できていません。