2017-08-11 08:28:17 +02:00
|
|
|
|
/*
|
2024-03-05 18:37:42 +01:00
|
|
|
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
2017-08-11 08:28:17 +02: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;
|
2021-05-30 20:04:44 +02:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Management;
|
2020-09-02 16:21:52 +02:00
|
|
|
|
using System.Threading.Tasks;
|
2017-08-11 08:28:17 +02:00
|
|
|
|
using System.Windows.Forms;
|
|
|
|
|
using Microsoft.Win32;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
2019-09-05 09:00:41 +02:00
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Display;
|
|
|
|
|
using SafeExamBrowser.Monitoring.Contracts.Display.Events;
|
2021-05-30 20:04:44 +02:00
|
|
|
|
using SafeExamBrowser.Settings.Monitoring;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.SystemComponents.Contracts;
|
|
|
|
|
using SafeExamBrowser.WindowsApi.Contracts;
|
|
|
|
|
using OperatingSystem = SafeExamBrowser.SystemComponents.Contracts.OperatingSystem;
|
2017-08-11 08:28:17 +02:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Monitoring.Display
|
|
|
|
|
{
|
|
|
|
|
public class DisplayMonitor : IDisplayMonitor
|
|
|
|
|
{
|
|
|
|
|
private IBounds originalWorkingArea;
|
2024-01-11 12:02:01 +01:00
|
|
|
|
private readonly ILogger logger;
|
|
|
|
|
private readonly INativeMethods nativeMethods;
|
|
|
|
|
private readonly ISystemInfo systemInfo;
|
2017-08-11 08:28:17 +02:00
|
|
|
|
private string wallpaper;
|
|
|
|
|
|
|
|
|
|
public event DisplayChangedEventHandler DisplayChanged;
|
|
|
|
|
|
2019-03-19 16:09:07 +01:00
|
|
|
|
public DisplayMonitor(ILogger logger, INativeMethods nativeMethods, ISystemInfo systemInfo)
|
2017-08-11 08:28:17 +02:00
|
|
|
|
{
|
|
|
|
|
this.logger = logger;
|
|
|
|
|
this.nativeMethods = nativeMethods;
|
2019-03-19 16:09:07 +01:00
|
|
|
|
this.systemInfo = systemInfo;
|
2017-08-11 08:28:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void InitializePrimaryDisplay(int taskbarHeight)
|
|
|
|
|
{
|
|
|
|
|
InitializeWorkingArea(taskbarHeight);
|
|
|
|
|
InitializeWallpaper();
|
2017-08-11 11:16:44 +02:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-29 17:34:05 +02:00
|
|
|
|
public void ResetPrimaryDisplay()
|
|
|
|
|
{
|
|
|
|
|
ResetWorkingArea();
|
|
|
|
|
ResetWallpaper();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void StartMonitoringDisplayChanges()
|
|
|
|
|
{
|
|
|
|
|
SystemEvents.DisplaySettingsChanged += SystemEvents_DisplaySettingsChanged;
|
|
|
|
|
logger.Info("Started monitoring display changes.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void StopMonitoringDisplayChanges()
|
|
|
|
|
{
|
|
|
|
|
SystemEvents.DisplaySettingsChanged -= SystemEvents_DisplaySettingsChanged;
|
|
|
|
|
logger.Info("Stopped monitoring display changes.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public ValidationResult ValidateConfiguration(DisplaySettings settings)
|
2021-05-30 20:04:44 +02:00
|
|
|
|
{
|
2021-06-29 17:34:05 +02:00
|
|
|
|
var result = new ValidationResult();
|
2021-05-30 20:04:44 +02:00
|
|
|
|
|
|
|
|
|
if (TryLoadDisplays(out var displays))
|
|
|
|
|
{
|
|
|
|
|
var active = displays.Where(d => d.IsActive);
|
|
|
|
|
var count = active.Count();
|
|
|
|
|
|
2021-06-29 17:34:05 +02:00
|
|
|
|
result.ExternalDisplays = active.Count(d => !d.IsInternal);
|
|
|
|
|
result.InternalDisplays = active.Count(d => d.IsInternal);
|
|
|
|
|
result.IsAllowed = count <= settings.AllowedDisplays;
|
2021-05-30 20:04:44 +02:00
|
|
|
|
|
2021-06-29 17:34:05 +02:00
|
|
|
|
if (result.IsAllowed)
|
2021-05-30 20:04:44 +02:00
|
|
|
|
{
|
|
|
|
|
logger.Info($"Detected {count} active displays, {settings.AllowedDisplays} are allowed.");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Warn($"Detected {count} active displays but only {settings.AllowedDisplays} are allowed!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (settings.InternalDisplayOnly && active.Any(d => !d.IsInternal))
|
|
|
|
|
{
|
2021-06-29 17:34:05 +02:00
|
|
|
|
result.IsAllowed = false;
|
2021-05-30 20:04:44 +02:00
|
|
|
|
logger.Warn("Detected external display but only internal displays are allowed!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-06-29 17:34:05 +02:00
|
|
|
|
result.IsAllowed = settings.IgnoreError;
|
|
|
|
|
logger.Warn($"Failed to validate display configuration, {(result.IsAllowed ? "ignoring error" : "active configuration is not allowed")}.");
|
2021-05-30 20:04:44 +02:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-29 17:34:05 +02:00
|
|
|
|
return result;
|
2017-08-11 08:28:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void SystemEvents_DisplaySettingsChanged(object sender, EventArgs e)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Display change detected!");
|
2020-09-02 16:21:52 +02:00
|
|
|
|
Task.Run(() => DisplayChanged?.Invoke());
|
2017-08-11 08:28:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void InitializeWorkingArea(int taskbarHeight)
|
|
|
|
|
{
|
|
|
|
|
var identifier = GetIdentifierForPrimaryDisplay();
|
|
|
|
|
|
2019-03-26 10:56:09 +01:00
|
|
|
|
if (originalWorkingArea == null)
|
|
|
|
|
{
|
|
|
|
|
originalWorkingArea = nativeMethods.GetWorkingArea();
|
|
|
|
|
LogWorkingArea($"Saved original working area for {identifier}", originalWorkingArea);
|
|
|
|
|
}
|
2017-08-11 08:28:17 +02:00
|
|
|
|
|
|
|
|
|
var area = new Bounds
|
|
|
|
|
{
|
|
|
|
|
Left = 0,
|
|
|
|
|
Top = 0,
|
|
|
|
|
Right = Screen.PrimaryScreen.Bounds.Width,
|
|
|
|
|
Bottom = Screen.PrimaryScreen.Bounds.Height - taskbarHeight
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
LogWorkingArea($"Trying to set new working area for {identifier}", area);
|
|
|
|
|
nativeMethods.SetWorkingArea(area);
|
|
|
|
|
LogWorkingArea($"Working area of {identifier} is now set to", nativeMethods.GetWorkingArea());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void InitializeWallpaper()
|
|
|
|
|
{
|
2019-03-19 16:09:07 +01:00
|
|
|
|
if (systemInfo.OperatingSystem == OperatingSystem.Windows7)
|
2017-08-11 08:28:17 +02:00
|
|
|
|
{
|
2019-03-19 16:09:07 +01:00
|
|
|
|
var path = nativeMethods.GetWallpaperPath();
|
|
|
|
|
|
2021-05-30 20:04:44 +02:00
|
|
|
|
if (!string.IsNullOrEmpty(path))
|
2019-03-19 16:09:07 +01:00
|
|
|
|
{
|
|
|
|
|
wallpaper = path;
|
|
|
|
|
logger.Info($"Saved wallpaper image: {wallpaper}.");
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-11 08:28:17 +02:00
|
|
|
|
nativeMethods.RemoveWallpaper();
|
|
|
|
|
logger.Info("Removed current wallpaper.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-30 20:04:44 +02:00
|
|
|
|
private bool TryLoadDisplays(out IList<Display> displays)
|
|
|
|
|
{
|
|
|
|
|
var success = true;
|
|
|
|
|
|
|
|
|
|
displays = new List<Display>();
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
using (var searcher = new ManagementObjectSearcher(@"Root\WMI", "SELECT * FROM WmiMonitorBasicDisplayParams"))
|
|
|
|
|
using (var results = searcher.Get())
|
|
|
|
|
{
|
|
|
|
|
var displayParameters = results.Cast<ManagementObject>();
|
|
|
|
|
|
|
|
|
|
foreach (var display in displayParameters)
|
|
|
|
|
{
|
|
|
|
|
displays.Add(new Display
|
|
|
|
|
{
|
|
|
|
|
Identifier = Convert.ToString(display["InstanceName"]),
|
|
|
|
|
IsActive = Convert.ToBoolean(display["Active"])
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using (var searcher = new ManagementObjectSearcher(@"Root\WMI", "SELECT * FROM WmiMonitorConnectionParams"))
|
|
|
|
|
using (var results = searcher.Get())
|
|
|
|
|
{
|
|
|
|
|
var connectionParameters = results.Cast<ManagementObject>();
|
|
|
|
|
|
|
|
|
|
foreach (var connection in connectionParameters)
|
|
|
|
|
{
|
|
|
|
|
var identifier = Convert.ToString(connection["InstanceName"]);
|
|
|
|
|
var isActive = Convert.ToBoolean(connection["Active"]);
|
|
|
|
|
var technologyValue = Convert.ToInt64(connection["VideoOutputTechnology"]);
|
|
|
|
|
var technology = (VideoOutputTechnology) technologyValue;
|
|
|
|
|
var display = displays.FirstOrDefault(d => d.Identifier?.Equals(identifier, StringComparison.OrdinalIgnoreCase) == true);
|
|
|
|
|
|
|
|
|
|
if (!Enum.IsDefined(typeof(VideoOutputTechnology), technology))
|
|
|
|
|
{
|
|
|
|
|
logger.Warn($"Detected undefined video output technology '{technologyValue}' for display '{identifier}'!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (display != default(Display))
|
|
|
|
|
{
|
|
|
|
|
display.IsActive &= isActive;
|
|
|
|
|
display.Technology = technology;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
success = false;
|
|
|
|
|
logger.Error("Failed to query displays!", e);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (var display in displays)
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Detected {(display.IsActive ? "active" : "inactive")}, {(display.IsInternal ? "internal" : "external")} display '{display.Identifier}' connected via '{display.Technology}'.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return success;
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-11 08:28:17 +02:00
|
|
|
|
private void ResetWorkingArea()
|
|
|
|
|
{
|
|
|
|
|
var identifier = GetIdentifierForPrimaryDisplay();
|
|
|
|
|
|
|
|
|
|
if (originalWorkingArea != null)
|
|
|
|
|
{
|
|
|
|
|
nativeMethods.SetWorkingArea(originalWorkingArea);
|
|
|
|
|
LogWorkingArea($"Restored original working area for {identifier}", originalWorkingArea);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Warn($"Could not restore original working area for {identifier}!");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ResetWallpaper()
|
|
|
|
|
{
|
2019-03-19 16:09:07 +01:00
|
|
|
|
if (systemInfo.OperatingSystem == OperatingSystem.Windows7 && !String.IsNullOrEmpty(wallpaper))
|
2017-08-11 08:28:17 +02:00
|
|
|
|
{
|
|
|
|
|
nativeMethods.SetWallpaper(wallpaper);
|
|
|
|
|
logger.Info($"Restored wallpaper image: {wallpaper}.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetIdentifierForPrimaryDisplay()
|
|
|
|
|
{
|
2021-05-30 20:04:44 +02:00
|
|
|
|
var name = Screen.PrimaryScreen.DeviceName?.Replace(@"\\.\", string.Empty);
|
|
|
|
|
var resolution = $"{Screen.PrimaryScreen.Bounds.Width}x{Screen.PrimaryScreen.Bounds.Height}";
|
|
|
|
|
var identifier = $"{name} ({resolution})";
|
2017-08-11 08:28:17 +02:00
|
|
|
|
|
2021-05-30 20:04:44 +02:00
|
|
|
|
return identifier;
|
2017-08-11 08:28:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void LogWorkingArea(string message, IBounds area)
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"{message}: Left = {area.Left}, Top = {area.Top}, Right = {area.Right}, Bottom = {area.Bottom}.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|