2017-07-26 14:36:20 +02:00
|
|
|
|
/*
|
2019-01-09 11:25:21 +01:00
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
2017-07-26 14:36:20 +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/.
|
|
|
|
|
*/
|
|
|
|
|
|
2017-07-27 13:57:12 +02:00
|
|
|
|
using System;
|
2018-06-21 07:56:25 +02:00
|
|
|
|
using System.IO;
|
|
|
|
|
using SafeExamBrowser.Contracts.Browser;
|
2018-07-04 09:53:33 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.Communication.Data;
|
|
|
|
|
using SafeExamBrowser.Contracts.Communication.Events;
|
2018-03-15 14:32:07 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Communication.Hosts;
|
|
|
|
|
using SafeExamBrowser.Contracts.Communication.Proxies;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration;
|
2019-01-10 10:04:30 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.Cryptography;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.Configuration.Settings;
|
2018-08-31 10:06:27 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.Core;
|
|
|
|
|
using SafeExamBrowser.Contracts.Core.OperationModel;
|
2018-10-03 14:35:27 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.Core.OperationModel.Events;
|
2018-02-27 15:28:54 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.I18n;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.Logging;
|
|
|
|
|
using SafeExamBrowser.Contracts.Monitoring;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.UserInterface;
|
2018-03-14 12:07:20 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.UserInterface.MessageBox;
|
2017-08-15 15:30:31 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.UserInterface.Taskbar;
|
2018-03-14 12:07:20 +01:00
|
|
|
|
using SafeExamBrowser.Contracts.UserInterface.Windows;
|
2018-08-17 14:48:50 +02:00
|
|
|
|
using SafeExamBrowser.Contracts.WindowsApi;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
|
2018-08-31 10:06:27 +02:00
|
|
|
|
namespace SafeExamBrowser.Client
|
2017-07-26 14:36:20 +02:00
|
|
|
|
{
|
2018-01-18 08:16:20 +01:00
|
|
|
|
internal class ClientController : IClientController
|
2017-07-26 14:36:20 +02:00
|
|
|
|
{
|
2017-08-11 08:28:17 +02:00
|
|
|
|
private IDisplayMonitor displayMonitor;
|
2018-08-17 14:48:50 +02:00
|
|
|
|
private IExplorerShell explorerShell;
|
2019-01-10 10:04:30 +01:00
|
|
|
|
private IHashAlgorithm hashAlgorithm;
|
2017-07-27 11:46:31 +02:00
|
|
|
|
private ILogger logger;
|
2018-03-14 12:07:20 +01:00
|
|
|
|
private IMessageBox messageBox;
|
2018-02-12 12:21:55 +01:00
|
|
|
|
private IOperationSequence operations;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
private IProcessMonitor processMonitor;
|
2018-02-14 15:26:05 +01:00
|
|
|
|
private IRuntimeProxy runtime;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
private Action shutdown;
|
|
|
|
|
private ISplashScreen splashScreen;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
private ITaskbar taskbar;
|
2018-07-04 09:53:33 +02:00
|
|
|
|
private IText text;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
private IUserInterfaceFactory uiFactory;
|
2017-07-27 11:46:31 +02:00
|
|
|
|
private IWindowMonitor windowMonitor;
|
2018-06-29 09:50:20 +02:00
|
|
|
|
private AppConfig appConfig;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
|
2018-06-21 07:56:25 +02:00
|
|
|
|
public IBrowserApplicationController Browser { private get; set; }
|
2018-02-20 15:15:26 +01:00
|
|
|
|
public IClientHost ClientHost { private get; set; }
|
2018-02-21 14:01:21 +01:00
|
|
|
|
public Guid SessionId { private get; set; }
|
|
|
|
|
public Settings Settings { private get; set; }
|
|
|
|
|
|
2018-06-29 09:50:20 +02:00
|
|
|
|
public AppConfig AppConfig
|
2018-02-21 14:01:21 +01:00
|
|
|
|
{
|
|
|
|
|
set
|
|
|
|
|
{
|
2018-06-29 09:50:20 +02:00
|
|
|
|
appConfig = value;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
|
|
|
|
|
if (splashScreen != null)
|
|
|
|
|
{
|
2018-06-29 09:50:20 +02:00
|
|
|
|
splashScreen.AppConfig = value;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-02-20 15:15:26 +01:00
|
|
|
|
|
2018-01-17 08:26:44 +01:00
|
|
|
|
public ClientController(
|
2017-08-11 08:28:17 +02:00
|
|
|
|
IDisplayMonitor displayMonitor,
|
2018-08-17 14:48:50 +02:00
|
|
|
|
IExplorerShell explorerShell,
|
2019-01-10 10:04:30 +01:00
|
|
|
|
IHashAlgorithm hashAlgorithm,
|
2017-07-27 11:46:31 +02:00
|
|
|
|
ILogger logger,
|
2018-03-14 12:07:20 +01:00
|
|
|
|
IMessageBox messageBox,
|
2018-02-12 12:21:55 +01:00
|
|
|
|
IOperationSequence operations,
|
2017-07-27 11:46:31 +02:00
|
|
|
|
IProcessMonitor processMonitor,
|
2018-02-14 15:26:05 +01:00
|
|
|
|
IRuntimeProxy runtime,
|
2018-02-21 14:01:21 +01:00
|
|
|
|
Action shutdown,
|
2017-07-27 11:46:31 +02:00
|
|
|
|
ITaskbar taskbar,
|
2018-07-04 09:53:33 +02:00
|
|
|
|
IText text,
|
2018-02-21 14:01:21 +01:00
|
|
|
|
IUserInterfaceFactory uiFactory,
|
2017-08-11 08:28:17 +02:00
|
|
|
|
IWindowMonitor windowMonitor)
|
2017-07-26 14:36:20 +02:00
|
|
|
|
{
|
2017-08-11 08:28:17 +02:00
|
|
|
|
this.displayMonitor = displayMonitor;
|
2018-08-17 14:48:50 +02:00
|
|
|
|
this.explorerShell = explorerShell;
|
2019-01-10 10:04:30 +01:00
|
|
|
|
this.hashAlgorithm = hashAlgorithm;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
this.logger = logger;
|
2018-03-14 12:07:20 +01:00
|
|
|
|
this.messageBox = messageBox;
|
2018-02-12 12:21:55 +01:00
|
|
|
|
this.operations = operations;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
this.processMonitor = processMonitor;
|
2018-02-14 15:26:05 +01:00
|
|
|
|
this.runtime = runtime;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
this.shutdown = shutdown;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
this.taskbar = taskbar;
|
2018-07-04 09:53:33 +02:00
|
|
|
|
this.text = text;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
this.uiFactory = uiFactory;
|
2017-07-27 11:46:31 +02:00
|
|
|
|
this.windowMonitor = windowMonitor;
|
2017-07-26 14:36:20 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-02-20 15:15:26 +01:00
|
|
|
|
public bool TryStart()
|
2018-02-12 12:21:55 +01:00
|
|
|
|
{
|
2018-08-16 11:23:37 +02:00
|
|
|
|
logger.Info("Initiating startup procedure...");
|
2018-02-22 10:00:18 +01:00
|
|
|
|
|
2018-02-21 14:01:21 +01:00
|
|
|
|
splashScreen = uiFactory.CreateSplashScreen();
|
2018-10-03 14:35:27 +02:00
|
|
|
|
operations.ProgressChanged += Operations_ProgressChanged;
|
|
|
|
|
operations.StatusChanged += Operations_StatusChanged;
|
2018-02-21 14:01:21 +01:00
|
|
|
|
|
2018-02-28 15:49:06 +01:00
|
|
|
|
var success = operations.TryPerform() == OperationResult.Success;
|
2018-02-14 15:26:05 +01:00
|
|
|
|
|
2018-02-20 15:15:26 +01:00
|
|
|
|
if (success)
|
|
|
|
|
{
|
|
|
|
|
RegisterEvents();
|
2018-11-15 08:45:17 +01:00
|
|
|
|
StartBrowser();
|
2018-02-27 15:28:54 +01:00
|
|
|
|
|
2018-08-10 13:23:24 +02:00
|
|
|
|
var communication = runtime.InformClientReady();
|
|
|
|
|
|
|
|
|
|
if (communication.Success)
|
2018-02-27 15:28:54 +01:00
|
|
|
|
{
|
2018-10-04 11:24:16 +02:00
|
|
|
|
splashScreen.Close();
|
2018-08-10 13:23:24 +02:00
|
|
|
|
|
2018-08-16 11:23:37 +02:00
|
|
|
|
logger.Info("Application successfully initialized.");
|
2018-08-10 13:23:24 +02:00
|
|
|
|
logger.Log(string.Empty);
|
2018-02-27 15:28:54 +01:00
|
|
|
|
}
|
2018-08-10 13:23:24 +02:00
|
|
|
|
else
|
2018-02-27 15:28:54 +01:00
|
|
|
|
{
|
2018-08-10 13:23:24 +02:00
|
|
|
|
success = false;
|
|
|
|
|
logger.Error("Failed to inform runtime that client is ready!");
|
2018-02-27 15:28:54 +01:00
|
|
|
|
}
|
2018-02-22 10:00:18 +01:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2018-08-16 11:23:37 +02:00
|
|
|
|
logger.Info("Application startup aborted!");
|
2018-02-22 10:00:18 +01:00
|
|
|
|
logger.Log(string.Empty);
|
2018-02-20 15:15:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return success;
|
2018-02-12 12:21:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-02-20 15:15:26 +01:00
|
|
|
|
public void Terminate()
|
2017-07-26 14:36:20 +02:00
|
|
|
|
{
|
2018-02-22 10:00:18 +01:00
|
|
|
|
logger.Log(string.Empty);
|
2018-08-16 11:23:37 +02:00
|
|
|
|
logger.Info("Initiating shutdown procedure...");
|
2018-02-22 10:00:18 +01:00
|
|
|
|
|
2018-10-04 11:24:16 +02:00
|
|
|
|
splashScreen = uiFactory.CreateSplashScreen(appConfig);
|
2018-02-21 14:01:21 +01:00
|
|
|
|
splashScreen.Show();
|
2017-07-26 14:36:20 +02:00
|
|
|
|
|
2018-02-21 14:01:21 +01:00
|
|
|
|
DeregisterEvents();
|
2018-02-27 15:28:54 +01:00
|
|
|
|
|
2018-10-10 09:19:03 +02:00
|
|
|
|
var success = operations.TryRevert() == OperationResult.Success;
|
2018-02-22 10:00:18 +01:00
|
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
|
{
|
2018-08-16 11:23:37 +02:00
|
|
|
|
logger.Info("Application successfully finalized.");
|
2018-02-22 10:00:18 +01:00
|
|
|
|
logger.Log(string.Empty);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2018-08-16 11:23:37 +02:00
|
|
|
|
logger.Info("Shutdown procedure failed!");
|
2018-02-22 10:00:18 +01:00
|
|
|
|
logger.Log(string.Empty);
|
|
|
|
|
}
|
2018-02-21 14:01:21 +01:00
|
|
|
|
|
2018-10-04 11:24:16 +02:00
|
|
|
|
splashScreen.Close();
|
2018-02-20 15:15:26 +01:00
|
|
|
|
}
|
2018-02-14 15:26:05 +01:00
|
|
|
|
|
2018-02-20 15:15:26 +01:00
|
|
|
|
private void RegisterEvents()
|
|
|
|
|
{
|
2018-06-21 07:56:25 +02:00
|
|
|
|
Browser.ConfigurationDownloadRequested += Browser_ConfigurationDownloadRequested;
|
2018-12-14 12:31:31 +01:00
|
|
|
|
ClientHost.MessageBoxRequested += ClientHost_MessageBoxRequested;
|
2018-07-04 09:53:33 +02:00
|
|
|
|
ClientHost.PasswordRequested += ClientHost_PasswordRequested;
|
|
|
|
|
ClientHost.ReconfigurationDenied += ClientHost_ReconfigurationDenied;
|
2018-02-20 15:15:26 +01:00
|
|
|
|
ClientHost.Shutdown += ClientHost_Shutdown;
|
|
|
|
|
displayMonitor.DisplayChanged += DisplayMonitor_DisplaySettingsChanged;
|
|
|
|
|
processMonitor.ExplorerStarted += ProcessMonitor_ExplorerStarted;
|
2018-02-27 15:28:54 +01:00
|
|
|
|
runtime.ConnectionLost += Runtime_ConnectionLost;
|
2018-02-20 15:15:26 +01:00
|
|
|
|
taskbar.QuitButtonClicked += Taskbar_QuitButtonClicked;
|
|
|
|
|
windowMonitor.WindowChanged += WindowMonitor_WindowChanged;
|
|
|
|
|
}
|
2018-02-14 15:26:05 +01:00
|
|
|
|
|
2018-02-20 15:15:26 +01:00
|
|
|
|
private void DeregisterEvents()
|
|
|
|
|
{
|
|
|
|
|
displayMonitor.DisplayChanged -= DisplayMonitor_DisplaySettingsChanged;
|
|
|
|
|
processMonitor.ExplorerStarted -= ProcessMonitor_ExplorerStarted;
|
2018-02-27 15:28:54 +01:00
|
|
|
|
runtime.ConnectionLost -= Runtime_ConnectionLost;
|
2018-02-20 15:15:26 +01:00
|
|
|
|
taskbar.QuitButtonClicked -= Taskbar_QuitButtonClicked;
|
|
|
|
|
windowMonitor.WindowChanged -= WindowMonitor_WindowChanged;
|
2019-01-23 10:07:20 +01:00
|
|
|
|
|
|
|
|
|
if (Browser != null)
|
|
|
|
|
{
|
|
|
|
|
Browser.ConfigurationDownloadRequested -= Browser_ConfigurationDownloadRequested;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ClientHost != null)
|
|
|
|
|
{
|
|
|
|
|
ClientHost.MessageBoxRequested -= ClientHost_MessageBoxRequested;
|
|
|
|
|
ClientHost.PasswordRequested -= ClientHost_PasswordRequested;
|
|
|
|
|
ClientHost.ReconfigurationDenied -= ClientHost_ReconfigurationDenied;
|
|
|
|
|
ClientHost.Shutdown -= ClientHost_Shutdown;
|
|
|
|
|
}
|
2017-07-26 14:36:20 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-11-15 08:45:17 +01:00
|
|
|
|
private void StartBrowser()
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Starting browser application...");
|
|
|
|
|
Browser.Start();
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-11 08:28:17 +02:00
|
|
|
|
private void DisplayMonitor_DisplaySettingsChanged()
|
|
|
|
|
{
|
|
|
|
|
logger.Info("Reinitializing working area...");
|
|
|
|
|
displayMonitor.InitializePrimaryDisplay(taskbar.GetAbsoluteHeight());
|
|
|
|
|
logger.Info("Reinitializing taskbar bounds...");
|
|
|
|
|
taskbar.InitializeBounds();
|
|
|
|
|
logger.Info("Desktop successfully restored.");
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-26 14:36:20 +02:00
|
|
|
|
private void ProcessMonitor_ExplorerStarted()
|
|
|
|
|
{
|
2018-08-17 14:48:50 +02:00
|
|
|
|
logger.Info("Trying to terminate Windows explorer...");
|
|
|
|
|
explorerShell.Terminate();
|
2017-07-26 14:36:20 +02:00
|
|
|
|
logger.Info("Reinitializing working area...");
|
2017-08-11 08:28:17 +02:00
|
|
|
|
displayMonitor.InitializePrimaryDisplay(taskbar.GetAbsoluteHeight());
|
2017-07-26 14:36:20 +02:00
|
|
|
|
logger.Info("Reinitializing taskbar bounds...");
|
|
|
|
|
taskbar.InitializeBounds();
|
2017-07-27 13:57:12 +02:00
|
|
|
|
logger.Info("Desktop successfully restored.");
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-21 07:56:25 +02:00
|
|
|
|
private void Browser_ConfigurationDownloadRequested(string fileName, DownloadEventArgs args)
|
|
|
|
|
{
|
|
|
|
|
if (Settings.ConfigurationMode == ConfigurationMode.ConfigureClient)
|
|
|
|
|
{
|
2018-11-08 09:39:52 +01:00
|
|
|
|
logger.Info($"Received download request for configuration file '{fileName}'. Asking user to confirm the reconfiguration...");
|
2018-06-21 07:56:25 +02:00
|
|
|
|
|
2018-10-30 11:24:28 +01:00
|
|
|
|
var message = TextKey.MessageBox_ReconfigurationQuestion;
|
|
|
|
|
var title = TextKey.MessageBox_ReconfigurationQuestionTitle;
|
|
|
|
|
var result = messageBox.Show(message, title, MessageBoxAction.YesNo, MessageBoxIcon.Question, args.BrowserWindow);
|
2018-06-21 07:56:25 +02:00
|
|
|
|
var reconfigure = result == MessageBoxResult.Yes;
|
|
|
|
|
|
|
|
|
|
logger.Info($"The user chose to {(reconfigure ? "start" : "abort")} the reconfiguration.");
|
|
|
|
|
|
|
|
|
|
if (reconfigure)
|
|
|
|
|
{
|
|
|
|
|
args.AllowDownload = true;
|
|
|
|
|
args.Callback = Browser_ConfigurationDownloadFinished;
|
2018-06-29 09:50:20 +02:00
|
|
|
|
args.DownloadPath = Path.Combine(appConfig.DownloadDirectory, fileName);
|
2018-06-21 07:56:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Denied download request for configuration file '{fileName}' due to '{Settings.ConfigurationMode}' mode.");
|
2018-10-30 11:24:28 +01:00
|
|
|
|
messageBox.Show(TextKey.MessageBox_ReconfigurationDenied, TextKey.MessageBox_ReconfigurationDeniedTitle, parent: args.BrowserWindow);
|
2018-06-21 07:56:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Browser_ConfigurationDownloadFinished(bool success, string filePath = null)
|
|
|
|
|
{
|
|
|
|
|
if (success)
|
|
|
|
|
{
|
2018-08-10 13:23:24 +02:00
|
|
|
|
var communication = runtime.RequestReconfiguration(filePath);
|
|
|
|
|
|
|
|
|
|
if (communication.Success)
|
2018-06-21 07:56:25 +02:00
|
|
|
|
{
|
|
|
|
|
logger.Info($"Sent reconfiguration request for '{filePath}' to the runtime.");
|
|
|
|
|
}
|
2018-08-10 13:23:24 +02:00
|
|
|
|
else
|
2018-06-21 07:56:25 +02:00
|
|
|
|
{
|
2018-08-10 13:23:24 +02:00
|
|
|
|
logger.Error($"Failed to communicate reconfiguration request for '{filePath}'!");
|
2018-06-21 07:56:25 +02:00
|
|
|
|
messageBox.Show(TextKey.MessageBox_ReconfigurationError, TextKey.MessageBox_ReconfigurationErrorTitle, icon: MessageBoxIcon.Error);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Error($"Failed to download configuration file '{filePath}'!");
|
|
|
|
|
messageBox.Show(TextKey.MessageBox_ConfigurationDownloadError, TextKey.MessageBox_ConfigurationDownloadErrorTitle, icon: MessageBoxIcon.Error);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-14 12:31:31 +01:00
|
|
|
|
private void ClientHost_MessageBoxRequested(MessageBoxRequestEventArgs args)
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"Received message box request with id '{args.RequestId}'.");
|
|
|
|
|
|
|
|
|
|
var result = messageBox.Show(args.Message, args.Title, args.Action, args.Icon);
|
|
|
|
|
|
|
|
|
|
runtime.SubmitMessageBoxResult(args.RequestId, result);
|
|
|
|
|
logger.Info($"Message box request with id '{args.RequestId}' yielded result '{result}'.");
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-04 09:53:33 +02:00
|
|
|
|
private void ClientHost_PasswordRequested(PasswordRequestEventArgs args)
|
|
|
|
|
{
|
|
|
|
|
var isAdmin = args.Purpose == PasswordRequestPurpose.Administrator;
|
|
|
|
|
var message = isAdmin ? TextKey.PasswordDialog_AdminPasswordRequired : TextKey.PasswordDialog_SettingsPasswordRequired;
|
|
|
|
|
var title = isAdmin ? TextKey.PasswordDialog_AdminPasswordRequiredTitle : TextKey.PasswordDialog_SettingsPasswordRequiredTitle;
|
|
|
|
|
var dialog = uiFactory.CreatePasswordDialog(text.Get(message), text.Get(title));
|
|
|
|
|
|
|
|
|
|
logger.Info($"Received input request with id '{args.RequestId}' for the {args.Purpose.ToString().ToLower()} password.");
|
|
|
|
|
|
|
|
|
|
var result = dialog.Show();
|
|
|
|
|
|
|
|
|
|
runtime.SubmitPassword(args.RequestId, result.Success, result.Password);
|
|
|
|
|
logger.Info($"Password request with id '{args.RequestId}' was {(result.Success ? "successful" : "aborted by the user")}.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ClientHost_ReconfigurationDenied(ReconfigurationEventArgs args)
|
|
|
|
|
{
|
|
|
|
|
logger.Info($"The reconfiguration request for '{args.ConfigurationPath}' was denied by the runtime!");
|
|
|
|
|
messageBox.Show(TextKey.MessageBox_ReconfigurationDenied, TextKey.MessageBox_ReconfigurationDeniedTitle);
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-20 15:15:26 +01:00
|
|
|
|
private void ClientHost_Shutdown()
|
|
|
|
|
{
|
|
|
|
|
taskbar.Close();
|
2018-02-21 14:01:21 +01:00
|
|
|
|
shutdown.Invoke();
|
2018-02-20 15:15:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-03 14:35:27 +02:00
|
|
|
|
private void Operations_ProgressChanged(ProgressChangedEventArgs args)
|
|
|
|
|
{
|
|
|
|
|
if (args.CurrentValue.HasValue)
|
|
|
|
|
{
|
|
|
|
|
splashScreen?.SetValue(args.CurrentValue.Value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (args.IsIndeterminate == true)
|
|
|
|
|
{
|
|
|
|
|
splashScreen?.SetIndeterminate();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (args.MaxValue.HasValue)
|
|
|
|
|
{
|
|
|
|
|
splashScreen?.SetMaxValue(args.MaxValue.Value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (args.Progress == true)
|
|
|
|
|
{
|
|
|
|
|
splashScreen?.Progress();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (args.Regress == true)
|
|
|
|
|
{
|
|
|
|
|
splashScreen?.Regress();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Operations_StatusChanged(TextKey status)
|
|
|
|
|
{
|
2018-10-04 11:24:16 +02:00
|
|
|
|
splashScreen?.UpdateStatus(status, true);
|
2018-10-03 14:35:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-02-27 15:28:54 +01:00
|
|
|
|
private void Runtime_ConnectionLost()
|
|
|
|
|
{
|
|
|
|
|
logger.Error("Lost connection to the runtime!");
|
2018-03-14 12:07:20 +01:00
|
|
|
|
messageBox.Show(TextKey.MessageBox_ApplicationError, TextKey.MessageBox_ApplicationErrorTitle, icon: MessageBoxIcon.Error);
|
2018-02-27 15:28:54 +01:00
|
|
|
|
|
|
|
|
|
taskbar.Close();
|
|
|
|
|
shutdown.Invoke();
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-03 14:35:27 +02:00
|
|
|
|
private void Taskbar_QuitButtonClicked(System.ComponentModel.CancelEventArgs args)
|
2018-02-20 15:15:26 +01:00
|
|
|
|
{
|
2019-01-10 10:04:30 +01:00
|
|
|
|
var hasQuitPassword = !String.IsNullOrEmpty(Settings.QuitPasswordHash);
|
|
|
|
|
var requestShutdown = false;
|
|
|
|
|
|
|
|
|
|
if (hasQuitPassword)
|
|
|
|
|
{
|
|
|
|
|
requestShutdown = TryValidateQuitPassword();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
requestShutdown = TryConfirmShutdown();
|
|
|
|
|
}
|
2018-02-27 15:28:54 +01:00
|
|
|
|
|
2019-01-10 10:04:30 +01:00
|
|
|
|
if (requestShutdown)
|
2018-02-27 15:28:54 +01:00
|
|
|
|
{
|
2018-08-10 13:23:24 +02:00
|
|
|
|
var communication = runtime.RequestShutdown();
|
|
|
|
|
|
|
|
|
|
if (!communication.Success)
|
2018-02-27 15:28:54 +01:00
|
|
|
|
{
|
2018-08-10 13:23:24 +02:00
|
|
|
|
logger.Error("Failed to communicate shutdown request to the runtime!");
|
2018-03-14 12:07:20 +01:00
|
|
|
|
messageBox.Show(TextKey.MessageBox_QuitError, TextKey.MessageBox_QuitErrorTitle, icon: MessageBoxIcon.Error);
|
2018-02-27 15:28:54 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-06-21 08:54:43 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
args.Cancel = true;
|
|
|
|
|
}
|
2018-02-20 15:15:26 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-07-27 13:57:12 +02:00
|
|
|
|
private void WindowMonitor_WindowChanged(IntPtr window)
|
|
|
|
|
{
|
2018-10-02 15:45:45 +02:00
|
|
|
|
var allowed = processMonitor.BelongsToAllowedProcess(window);
|
2017-07-27 13:57:12 +02:00
|
|
|
|
|
2018-10-02 15:45:45 +02:00
|
|
|
|
if (!allowed)
|
|
|
|
|
{
|
|
|
|
|
var success = windowMonitor.Hide(window);
|
2018-09-27 11:24:13 +02:00
|
|
|
|
|
2018-10-02 15:45:45 +02:00
|
|
|
|
if (!success)
|
|
|
|
|
{
|
|
|
|
|
windowMonitor.Close(window);
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-07-26 14:36:20 +02:00
|
|
|
|
}
|
2019-01-10 10:04:30 +01:00
|
|
|
|
|
|
|
|
|
private bool TryConfirmShutdown()
|
|
|
|
|
{
|
|
|
|
|
var result = messageBox.Show(TextKey.MessageBox_Quit, TextKey.MessageBox_QuitTitle, MessageBoxAction.YesNo, MessageBoxIcon.Question);
|
|
|
|
|
var quit = result == MessageBoxResult.Yes;
|
|
|
|
|
|
|
|
|
|
if (quit)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("The user chose to terminate the application.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return quit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool TryValidateQuitPassword()
|
|
|
|
|
{
|
|
|
|
|
var dialog = uiFactory.CreatePasswordDialog(TextKey.PasswordDialog_QuitPasswordRequired, TextKey.PasswordDialog_QuitPasswordRequiredTitle);
|
|
|
|
|
var result = dialog.Show();
|
|
|
|
|
|
|
|
|
|
if (result.Success)
|
|
|
|
|
{
|
|
|
|
|
var passwordHash = hashAlgorithm.GenerateHashFor(result.Password);
|
|
|
|
|
var isCorrect = Settings.QuitPasswordHash.Equals(passwordHash, StringComparison.OrdinalIgnoreCase);
|
|
|
|
|
|
|
|
|
|
if (isCorrect)
|
|
|
|
|
{
|
|
|
|
|
logger.Info("The user entered the correct quit password, the application will now terminate.");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Info("The user entered the wrong quit password.");
|
|
|
|
|
messageBox.Show(TextKey.MessageBox_InvalidQuitPassword, TextKey.MessageBox_InvalidQuitPasswordTitle, icon: MessageBoxIcon.Warning);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return isCorrect;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2017-07-26 14:36:20 +02:00
|
|
|
|
}
|
|
|
|
|
}
|