2024-02-21 18:37:23 +01:00
|
|
|
|
/*
|
2024-03-05 18:37:42 +01:00
|
|
|
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
2024-02-21 18:37:23 +01:00
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Drawing;
|
|
|
|
|
using System.Drawing.Imaging;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Windows.Forms;
|
|
|
|
|
using KGySoft.Drawing;
|
|
|
|
|
using KGySoft.Drawing.Imaging;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
|
|
|
|
using SafeExamBrowser.Settings.Proctoring;
|
|
|
|
|
using ImageFormat = SafeExamBrowser.Settings.Proctoring.ImageFormat;
|
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Proctoring.ScreenProctoring.Imaging
|
|
|
|
|
{
|
|
|
|
|
internal class ScreenShotProcessor : IDisposable
|
|
|
|
|
{
|
|
|
|
|
private readonly ILogger logger;
|
|
|
|
|
private readonly ScreenProctoringSettings settings;
|
|
|
|
|
|
|
|
|
|
private Bitmap bitmap;
|
|
|
|
|
private DateTime captureTime;
|
|
|
|
|
private byte[] data;
|
|
|
|
|
private ImageFormat format;
|
|
|
|
|
private int height;
|
|
|
|
|
private int width;
|
|
|
|
|
|
|
|
|
|
internal ScreenShot Data => new ScreenShot
|
|
|
|
|
{
|
|
|
|
|
CaptureTime = captureTime,
|
|
|
|
|
Data = data,
|
|
|
|
|
Format = format,
|
|
|
|
|
Height = height,
|
|
|
|
|
Width = width
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
public ScreenShotProcessor(ILogger logger, ScreenProctoringSettings settings)
|
|
|
|
|
{
|
|
|
|
|
this.logger = logger;
|
|
|
|
|
this.settings = settings;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
|
{
|
|
|
|
|
bitmap?.Dispose();
|
|
|
|
|
bitmap = default;
|
|
|
|
|
data = default;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal void Compress()
|
|
|
|
|
{
|
|
|
|
|
var order = ProcessingOrder.QuantizingDownscaling;
|
|
|
|
|
var original = ToReducedString();
|
|
|
|
|
var parameters = $"{order}, {settings.ImageQuantization}, 1:{settings.ImageDownscaling}";
|
|
|
|
|
|
|
|
|
|
switch (order)
|
|
|
|
|
{
|
|
|
|
|
case ProcessingOrder.DownscalingQuantizing:
|
|
|
|
|
Downscale();
|
|
|
|
|
Quantize();
|
|
|
|
|
Serialize();
|
|
|
|
|
break;
|
|
|
|
|
case ProcessingOrder.QuantizingDownscaling:
|
|
|
|
|
Quantize();
|
|
|
|
|
Downscale();
|
|
|
|
|
Serialize();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logger.Debug($"Compressed from '{original}' to '{ToReducedString()}' ({parameters}).");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal void Take()
|
|
|
|
|
{
|
|
|
|
|
var x = Screen.AllScreens.Min(s => s.Bounds.X);
|
|
|
|
|
var y = Screen.AllScreens.Min(s => s.Bounds.Y);
|
|
|
|
|
var width = Screen.AllScreens.Max(s => s.Bounds.X + s.Bounds.Width) - x;
|
|
|
|
|
var height = Screen.AllScreens.Max(s => s.Bounds.Y + s.Bounds.Height) - y;
|
|
|
|
|
|
|
|
|
|
bitmap = new Bitmap(width, height, PixelFormat.Format24bppRgb);
|
|
|
|
|
captureTime = DateTime.Now;
|
|
|
|
|
format = settings.ImageFormat;
|
|
|
|
|
this.height = height;
|
|
|
|
|
this.width = width;
|
|
|
|
|
|
|
|
|
|
using (var graphics = Graphics.FromImage(bitmap))
|
|
|
|
|
{
|
|
|
|
|
graphics.CopyFromScreen(x, y, 0, 0, new Size(width, height), CopyPixelOperation.SourceCopy);
|
|
|
|
|
graphics.DrawCursorPosition();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Serialize();
|
|
|
|
|
|
|
|
|
|
logger.Debug($"Captured '{ToReducedString()}' at {captureTime}.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Downscale()
|
|
|
|
|
{
|
|
|
|
|
if (settings.ImageDownscaling > 1)
|
|
|
|
|
{
|
|
|
|
|
height = Convert.ToInt32(height / settings.ImageDownscaling);
|
|
|
|
|
width = Convert.ToInt32(width / settings.ImageDownscaling);
|
|
|
|
|
|
|
|
|
|
var downscaled = new Bitmap(width, height, bitmap.PixelFormat);
|
|
|
|
|
|
|
|
|
|
bitmap.DrawInto(downscaled, new Rectangle(0, 0, width, height), ScalingMode.NearestNeighbor);
|
|
|
|
|
bitmap.Dispose();
|
|
|
|
|
bitmap = downscaled;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Quantize()
|
|
|
|
|
{
|
|
|
|
|
var ditherer = settings.ImageDownscaling > 1 ? OrderedDitherer.Bayer2x2 : default;
|
|
|
|
|
var pixelFormat = settings.ImageQuantization.ToPixelFormat();
|
|
|
|
|
var quantizer = settings.ImageQuantization.ToQuantizer();
|
|
|
|
|
|
|
|
|
|
bitmap = bitmap.ConvertPixelFormat(pixelFormat, quantizer, ditherer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Serialize()
|
|
|
|
|
{
|
|
|
|
|
using (var memoryStream = new MemoryStream())
|
|
|
|
|
{
|
|
|
|
|
if (format == ImageFormat.Jpg)
|
|
|
|
|
{
|
|
|
|
|
SerializeJpg(memoryStream);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
bitmap.Save(memoryStream, format.ToSystemFormat());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data = memoryStream.ToArray();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void SerializeJpg(MemoryStream memoryStream)
|
|
|
|
|
{
|
|
|
|
|
var codec = ImageCodecInfo.GetImageEncoders().First(c => c.FormatID == System.Drawing.Imaging.ImageFormat.Jpeg.Guid);
|
|
|
|
|
var parameters = new EncoderParameters(1);
|
|
|
|
|
var quality = 100;
|
|
|
|
|
|
|
|
|
|
parameters.Param[0] = new EncoderParameter(Encoder.Quality, quality);
|
|
|
|
|
bitmap.Save(memoryStream, codec, parameters);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string ToReducedString()
|
|
|
|
|
{
|
|
|
|
|
return $"{width}x{height}, {data.Length / 1000:N0}kB, {format.ToString().ToUpper()}";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|