using System;
using System.Diagnostics;
using System.Drawing;
using System.Windows.Forms;
using Quasar.Server.Utilities;
namespace Quasar.Server.Controls
{
public interface IRapidPictureBox
{
bool Running { get; set; }
Image GetImageSafe { get; set; }
void Start();
void Stop();
void UpdateImage(Bitmap bmp, bool cloneBitmap = false);
}
///
/// Custom PictureBox Control designed for rapidly-changing images.
///
public class RapidPictureBox : PictureBox, IRapidPictureBox
{
///
/// True if the PictureBox is currently streaming images, else False.
///
public bool Running { get; set; }
///
/// Returns the width of the original screen.
///
public int ScreenWidth { get; private set; }
///
/// Returns the height of the original screen.
///
public int ScreenHeight { get; private set; }
///
/// Provides thread-safe access to the Image of this Picturebox.
///
public Image GetImageSafe
{
get
{
return Image;
}
set
{
lock (_imageLock)
{
Image = value;
}
}
}
///
/// The lock object for the Picturebox's image.
///
private readonly object _imageLock = new object();
///
/// The Stopwatch for internal FPS measuring.
///
private Stopwatch _sWatch;
///
/// The internal class for FPS measuring.
///
private FrameCounter _frameCounter;
///
/// Subscribes an Eventhandler to the FrameUpdated event.
///
/// The Eventhandler to set.
public void SetFrameUpdatedEvent(FrameUpdatedEventHandler e)
{
_frameCounter.FrameUpdated += e;
}
///
/// Unsubscribes an Eventhandler from the FrameUpdated event.
///
/// The Eventhandler to remove.
public void UnsetFrameUpdatedEvent(FrameUpdatedEventHandler e)
{
_frameCounter.FrameUpdated -= e;
}
///
/// Starts the internal FPS measuring.
///
public void Start()
{
_frameCounter = new FrameCounter();
_sWatch = Stopwatch.StartNew();
Running = true;
}
///
/// Stops the internal FPS measuring.
///
public void Stop()
{
_sWatch?.Stop();
Running = false;
}
///
/// Updates the Image of this Picturebox.
///
/// The new bitmap to use.
/// If True the bitmap will be cloned, else it uses the original bitmap.
public void UpdateImage(Bitmap bmp, bool cloneBitmap)
{
try
{
CountFps();
if ((ScreenWidth != bmp.Width) && (ScreenHeight != bmp.Height))
UpdateScreenSize(bmp.Width, bmp.Height);
lock (_imageLock)
{
// get old image to dispose it correctly
var oldImage = GetImageSafe;
SuspendLayout();
GetImageSafe = cloneBitmap ? new Bitmap(bmp, Width, Height) /*resize bitmap*/ : bmp;
ResumeLayout();
oldImage?.Dispose();
}
}
catch (InvalidOperationException)
{
}
catch (Exception)
{
}
}
///
/// Constructor, sets Picturebox double-buffered and initializes the Framecounter.
///
public RapidPictureBox()
{
this.SetStyle(ControlStyles.UserPaint | ControlStyles.AllPaintingInWmPaint | ControlStyles.OptimizedDoubleBuffer, true);
}
protected override CreateParams CreateParams
{
get
{
CreateParams cp = base.CreateParams;
cp.ExStyle |= 0x02000000; // Turn on WS_EX_COMPOSITED
return cp;
}
}
protected override void OnPaint(PaintEventArgs pe)
{
lock (_imageLock)
{
if (GetImageSafe != null)
{
pe.Graphics.DrawImage(GetImageSafe, Location);
}
}
}
private void UpdateScreenSize(int newWidth, int newHeight)
{
ScreenWidth = newWidth;
ScreenHeight = newHeight;
}
private void CountFps()
{
var deltaTime = (float)_sWatch.Elapsed.TotalSeconds;
_sWatch = Stopwatch.StartNew();
_frameCounter.Update(deltaTime);
}
}
}