|
■No101806 (MASA さん) に返信 > using (Bitmap bitmap = (Bitmap)eventArgs.Frame.Clone()) > { > pictureBox.Image?.Dispose(); > pictureBox.Image = (Bitmap)eventArgs.Frame.Clone(); > }
(1) この書き換えは無意味に見えます。 Clone による複製を 2 回行っていますが、変数 bitmap の方は、 一度も使われることなく捨てられていますからね。
> private void button2_Click(object sender, EventArgs e) > { > VideoDevice.Stop();
(2) VideoResolution の設定は Start 前に行う必要がある、という認識は正しいです。
(3) 停止処理に Stop メソッドを使うのはやめましょう。解説文にも控えるように書かれていますよね。 Since the method aborts background thread, its usage is highly not preferred and should be done only if there are no other options. The correct way of stopping camera is signaling it stop and then waiting for background thread's completion.
UI スレッドからは、SignalToStop メソッドを呼び出して停止信号を出すだけにして、 撮影スレッドが停止するまで待機する方が望ましいようです。
(4) 繰り返しになりますが、スレッド セーフを目指しましょう。 元ライブラリ自体が Task ベースでは無いこともあり、使いにくい面はあるのですけれど。
以下サンプル。PictureBox (pictureBox1) と ComboBox (comboCamera, comboVideoResolution) を貼っています。
public partial class Form1 : Form { private FilterInfoCollection captureDevices; private VideoCaptureDevice videoDevice;
public Form1() { InitializeComponent(); }
/// <summary>撮影開始</summary> private void StartCamera() { // カメラのインスタンスを取得 string moniker = comboCamera.SelectedValue.ToString(); videoDevice = new VideoCaptureDevice(moniker);
// そのカメラの解像度情報を列挙 var resolutions = videoDevice.VideoCapabilities.ToDictionary( k => k, v => $"{v.FrameSize.Width}×{v.FrameSize.Height}, {v.BitCount}bit").ToArray(); comboVideoResolution.DataSource = resolutions;
// 撮影開始 videoDevice.NewFrame += VideoDevice_NewFrame; videoDevice.Start(); }
protected override void OnLoad(EventArgs e) { base.OnLoad(e);
#region このあたりはデザイン時に設定しておけばよい comboCamera.DropDownStyle = ComboBoxStyle.DropDownList; comboVideoResolution.DropDownStyle = ComboBoxStyle.DropDownList; comboCamera.DisplayMember = "Name"; comboCamera.ValueMember = "MonikerString"; comboVideoResolution.DisplayMember = "Value"; comboVideoResolution.ValueMember = "Key"; #endregion
captureDevices = new FilterInfoCollection(FilterCategory.VideoInputDevice);
// カメラを列挙 comboCamera.DataSource = captureDevices; }
protected override void OnFormClosing(FormClosingEventArgs e) { // 撮影中のままフォームが破棄されると都合が悪いので、先にカメラを止めるようにする if (videoDevice?.IsRunning ?? false) { e.Cancel = true; // 撮影中なら、画面の終了を一時中断する videoDevice.PlayingFinished += delegate { videoDevice = null; Invoke(new MethodInvoker(Close)); // カメラが停止したので、改めて閉じなおす }; videoDevice.SignalToStop(); // 停止信号を発布 } base.OnFormClosing(e); }
/// <summary>カメラ選択ドロップダウンが変更されたとき</summary> private async void comboCamera_SelectedIndexChanged(object sender, EventArgs e) { // 撮影中なら止める if (videoDevice?.IsRunning ?? false) { videoDevice.SignalToStop(); try { await Task.Run(videoDevice.WaitForStop); } catch (Exception) { } videoDevice.NewFrame -= VideoDevice_NewFrame; }
// カメラのインスタンスを取得 string moniker = comboCamera.SelectedValue.ToString(); videoDevice = new VideoCaptureDevice(moniker);
// そのカメラの解像度情報を列挙 var resolutions = videoDevice.VideoCapabilities; comboVideoResolution.DataSource = resolutions.ToDictionary( k => k, v => $"{v.FrameSize.Width}×{v.FrameSize.Height}, {v.BitCount}bit").ToArray(); ;
// 1920x1080 解像度を仮選択 Size defaultSize = new Size(1920, 1080); VideoCapabilities resolution = resolutions.FirstOrDefault(v => v.FrameSize == defaultSize);
// 選択肢に無い場合は null がセットされ、そのデバイスの既定の解像度が使われる videoDevice.VideoResolution = resolution;
// 撮影開始 videoDevice.NewFrame += VideoDevice_NewFrame; videoDevice.Start(); }
/// <summary>解像度一覧ドロップダウンが変更されたとき</summary> private async void comboVideoResolution_SelectedIndexChanged(object sender, EventArgs e) { // 選択されている解像度を取得 VideoCapabilities resolution = comboVideoResolution.SelectedValue as VideoCapabilities; if (resolution != null && (videoDevice?.IsRunning ?? false)) { // カメラを一時停止して、解像度を設定してから撮影再開 videoDevice.SignalToStop(); try { await Task.Run(videoDevice.WaitForStop); } catch (Exception) { } videoDevice.VideoResolution = resolution; videoDevice.Start(); } }
/// <summary>動画撮影中に呼ばれるイベント</summary> private void VideoDevice_NewFrame(object sender, NewFrameEventArgs eventArgs) { ChangeFrame((Image)eventArgs.Frame.Clone()); }
/// <summary>フレーム差し替え</summary> private void ChangeFrame(Image newImage) { if (Disposing || IsDisposed) { return; } if (!InvokeRequired) { Text = $"{newImage.Size}"; pictureBox1.Image?.Dispose(); pictureBox1.Image = newImage; } else { // 別スレッドからの呼び出しったので、UI スレッドに依頼しなおす Invoke(new MethodInvoker(() => ChangeFrame(newImage))); } } }
AForge.NET でカメラを止める場合、どのスレッドで待ち合わせるべきなのか、 その停止の待ち合わせに PlayingFinished イベントを使うのが良いのか、それとも WaitForStop メソッドを使うのか、IsRunning プロパティ監視とするのかなど、 Web 上のサンプルでも意見が分かれていて、いったい何が正解なのやら。
|