2017-07-14 10:28:59 +02:00
|
|
|
|
/*
|
2019-01-09 11:25:21 +01:00
|
|
|
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
2017-07-14 10:28:59 +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/.
|
|
|
|
|
*/
|
|
|
|
|
|
2019-03-15 15:55:34 +01:00
|
|
|
|
using System;
|
2019-05-08 13:53:38 +02:00
|
|
|
|
using System.Net.Http;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Applications.Contracts;
|
|
|
|
|
using SafeExamBrowser.Applications.Contracts.Events;
|
2019-09-06 08:13:27 +02:00
|
|
|
|
using SafeExamBrowser.Browser.Contracts.Events;
|
2019-01-17 11:12:17 +01:00
|
|
|
|
using SafeExamBrowser.Browser.Events;
|
2018-03-08 07:35:58 +01:00
|
|
|
|
using SafeExamBrowser.Browser.Handlers;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.Configuration.Contracts;
|
2019-09-06 08:13:27 +02:00
|
|
|
|
using SafeExamBrowser.Configuration.Contracts.Settings.Browser;
|
2019-08-30 09:55:26 +02:00
|
|
|
|
using SafeExamBrowser.I18n.Contracts;
|
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts;
|
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts.Browser;
|
|
|
|
|
using SafeExamBrowser.UserInterface.Contracts.MessageBox;
|
2017-07-14 10:28:59 +02:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.Browser
|
|
|
|
|
{
|
2018-03-08 07:35:58 +01:00
|
|
|
|
internal class BrowserApplicationInstance : IApplicationInstance
|
2017-07-14 10:28:59 +02:00
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
private const double ZOOM_FACTOR = 0.2;
|
|
|
|
|
|
2018-06-29 09:50:20 +02:00
|
|
|
|
private AppConfig appConfig;
|
2017-07-31 20:22:53 +02:00
|
|
|
|
private IBrowserControl control;
|
|
|
|
|
private IBrowserWindow window;
|
2019-05-08 13:53:38 +02:00
|
|
|
|
private HttpClient httpClient;
|
2018-03-14 11:13:30 +01:00
|
|
|
|
private bool isMainInstance;
|
2019-01-17 11:12:17 +01:00
|
|
|
|
private IMessageBox messageBox;
|
2018-08-31 15:29:36 +02:00
|
|
|
|
private IModuleLogger logger;
|
2018-03-14 11:13:30 +01:00
|
|
|
|
private BrowserSettings settings;
|
|
|
|
|
private IText text;
|
|
|
|
|
private IUserInterfaceFactory uiFactory;
|
2019-01-23 09:37:47 +01:00
|
|
|
|
private string url;
|
2019-05-22 11:42:31 +02:00
|
|
|
|
private double zoomLevel;
|
2019-01-23 09:37:47 +01:00
|
|
|
|
|
|
|
|
|
private BrowserWindowSettings WindowSettings
|
|
|
|
|
{
|
|
|
|
|
get { return isMainInstance ? settings.MainWindowSettings : settings.AdditionalWindowSettings; }
|
|
|
|
|
}
|
2017-07-31 20:22:53 +02:00
|
|
|
|
|
2018-08-31 07:49:41 +02:00
|
|
|
|
public InstanceIdentifier Id { get; private set; }
|
2017-07-14 10:28:59 +02:00
|
|
|
|
public string Name { get; private set; }
|
|
|
|
|
|
2018-06-21 07:56:25 +02:00
|
|
|
|
public event DownloadRequestedEventHandler ConfigurationDownloadRequested;
|
2019-01-17 16:15:10 +01:00
|
|
|
|
public event IconChangedEventHandler IconChanged;
|
2018-07-06 15:57:38 +02:00
|
|
|
|
public event InstanceTerminatedEventHandler Terminated;
|
2019-01-17 11:12:17 +01:00
|
|
|
|
public event NameChangedEventHandler NameChanged;
|
|
|
|
|
public event PopupRequestedEventHandler PopupRequested;
|
2017-07-28 09:12:17 +02:00
|
|
|
|
|
2018-06-21 07:56:25 +02:00
|
|
|
|
public BrowserApplicationInstance(
|
2018-06-29 09:50:20 +02:00
|
|
|
|
AppConfig appConfig,
|
2018-06-21 07:56:25 +02:00
|
|
|
|
BrowserSettings settings,
|
2018-08-31 07:49:41 +02:00
|
|
|
|
InstanceIdentifier id,
|
|
|
|
|
bool isMainInstance,
|
2019-01-17 11:12:17 +01:00
|
|
|
|
IMessageBox messageBox,
|
2018-08-31 15:29:36 +02:00
|
|
|
|
IModuleLogger logger,
|
2018-06-21 07:56:25 +02:00
|
|
|
|
IText text,
|
2019-01-23 09:37:47 +01:00
|
|
|
|
IUserInterfaceFactory uiFactory,
|
|
|
|
|
string url)
|
2018-03-14 11:13:30 +01:00
|
|
|
|
{
|
2018-06-29 09:50:20 +02:00
|
|
|
|
this.appConfig = appConfig;
|
2018-08-31 07:49:41 +02:00
|
|
|
|
this.Id = id;
|
2019-05-08 13:53:38 +02:00
|
|
|
|
this.httpClient = new HttpClient();
|
2018-03-14 11:13:30 +01:00
|
|
|
|
this.isMainInstance = isMainInstance;
|
2019-01-17 11:12:17 +01:00
|
|
|
|
this.messageBox = messageBox;
|
2018-08-31 15:29:36 +02:00
|
|
|
|
this.logger = logger;
|
2018-03-14 11:13:30 +01:00
|
|
|
|
this.settings = settings;
|
|
|
|
|
this.text = text;
|
|
|
|
|
this.uiFactory = uiFactory;
|
2019-01-23 09:37:47 +01:00
|
|
|
|
this.url = url;
|
2018-03-14 11:13:30 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-08-30 12:30:00 +02:00
|
|
|
|
public void Activate()
|
|
|
|
|
{
|
|
|
|
|
window?.BringToForeground();
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-14 11:13:30 +01:00
|
|
|
|
internal void Initialize()
|
2019-09-06 08:13:27 +02:00
|
|
|
|
{
|
|
|
|
|
InitializeControl();
|
|
|
|
|
InitializeWindow();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal void Terminate()
|
|
|
|
|
{
|
|
|
|
|
window?.Close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void InitializeControl()
|
2017-07-14 10:28:59 +02:00
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
var contextMenuHandler = new ContextMenuHandler();
|
2019-01-17 16:15:10 +01:00
|
|
|
|
var displayHandler = new DisplayHandler();
|
2018-08-31 15:29:36 +02:00
|
|
|
|
var downloadLogger = logger.CloneFor($"{nameof(DownloadHandler)} {Id}");
|
|
|
|
|
var downloadHandler = new DownloadHandler(appConfig, settings, downloadLogger);
|
2019-01-17 11:12:17 +01:00
|
|
|
|
var keyboardHandler = new KeyboardHandler();
|
|
|
|
|
var lifeSpanHandler = new LifeSpanHandler();
|
2019-09-06 08:13:27 +02:00
|
|
|
|
var requestLogger = logger.CloneFor($"{nameof(RequestHandler)} {Id}");
|
|
|
|
|
var requestHandler = new RequestHandler(appConfig, settings, logger);
|
2018-06-21 07:56:25 +02:00
|
|
|
|
|
2019-01-17 16:15:10 +01:00
|
|
|
|
displayHandler.FaviconChanged += DisplayHandler_FaviconChanged;
|
2019-05-22 11:42:31 +02:00
|
|
|
|
displayHandler.ProgressChanged += DisplayHandler_ProgressChanged;
|
2018-10-30 11:24:28 +01:00
|
|
|
|
downloadHandler.ConfigurationDownloadRequested += DownloadHandler_ConfigurationDownloadRequested;
|
2019-01-18 16:11:33 +01:00
|
|
|
|
keyboardHandler.ReloadRequested += ReloadRequested;
|
|
|
|
|
keyboardHandler.ZoomInRequested += ZoomInRequested;
|
|
|
|
|
keyboardHandler.ZoomOutRequested += ZoomOutRequested;
|
|
|
|
|
keyboardHandler.ZoomResetRequested += ZoomResetRequested;
|
2019-01-17 11:12:17 +01:00
|
|
|
|
lifeSpanHandler.PopupRequested += LifeSpanHandler_PopupRequested;
|
2017-07-31 20:22:53 +02:00
|
|
|
|
|
2019-01-23 09:37:47 +01:00
|
|
|
|
control = new BrowserControl(contextMenuHandler, displayHandler, downloadHandler, keyboardHandler, lifeSpanHandler, requestHandler, url);
|
2017-07-31 20:22:53 +02:00
|
|
|
|
control.AddressChanged += Control_AddressChanged;
|
2018-03-01 08:50:08 +01:00
|
|
|
|
control.LoadingStateChanged += Control_LoadingStateChanged;
|
2017-07-31 20:22:53 +02:00
|
|
|
|
control.TitleChanged += Control_TitleChanged;
|
2019-01-11 08:25:40 +01:00
|
|
|
|
control.Initialize();
|
2017-07-31 20:22:53 +02:00
|
|
|
|
|
2018-08-31 15:29:36 +02:00
|
|
|
|
logger.Debug("Initialized browser control.");
|
2019-09-06 08:13:27 +02:00
|
|
|
|
}
|
2018-08-31 15:29:36 +02:00
|
|
|
|
|
2019-09-06 08:13:27 +02:00
|
|
|
|
private void InitializeWindow()
|
|
|
|
|
{
|
2019-01-23 09:37:47 +01:00
|
|
|
|
window = uiFactory.CreateBrowserWindow(control, settings, isMainInstance);
|
2017-07-31 20:22:53 +02:00
|
|
|
|
window.Closing += () => Terminated?.Invoke(Id);
|
|
|
|
|
window.AddressChanged += Window_AddressChanged;
|
|
|
|
|
window.BackwardNavigationRequested += Window_BackwardNavigationRequested;
|
2019-05-22 11:42:31 +02:00
|
|
|
|
window.DeveloperConsoleRequested += Window_DeveloperConsoleRequested;
|
2017-07-31 20:22:53 +02:00
|
|
|
|
window.ForwardNavigationRequested += Window_ForwardNavigationRequested;
|
2019-05-22 11:42:31 +02:00
|
|
|
|
window.ReloadRequested += ReloadRequested;
|
2019-01-18 16:11:33 +01:00
|
|
|
|
window.ZoomInRequested += ZoomInRequested;
|
|
|
|
|
window.ZoomOutRequested += ZoomOutRequested;
|
|
|
|
|
window.ZoomResetRequested += ZoomResetRequested;
|
2019-05-22 11:42:31 +02:00
|
|
|
|
window.UpdateZoomLevel(CalculateZoomPercentage());
|
2019-08-30 12:30:00 +02:00
|
|
|
|
window.Show();
|
2018-08-31 15:29:36 +02:00
|
|
|
|
|
|
|
|
|
logger.Debug("Initialized browser window.");
|
2017-07-31 20:22:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Control_AddressChanged(string address)
|
|
|
|
|
{
|
2019-01-17 11:12:17 +01:00
|
|
|
|
logger.Debug($"Navigated to '{address}'.");
|
2017-07-31 20:22:53 +02:00
|
|
|
|
window.UpdateAddress(address);
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-01 08:50:08 +01:00
|
|
|
|
private void Control_LoadingStateChanged(bool isLoading)
|
|
|
|
|
{
|
2019-01-23 09:37:47 +01:00
|
|
|
|
window.CanNavigateBackwards = WindowSettings.AllowBackwardNavigation && control.CanNavigateBackwards;
|
|
|
|
|
window.CanNavigateForwards = WindowSettings.AllowForwardNavigation && control.CanNavigateForwards;
|
2018-03-01 08:50:08 +01:00
|
|
|
|
window.UpdateLoadingState(isLoading);
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-31 20:22:53 +02:00
|
|
|
|
private void Control_TitleChanged(string title)
|
|
|
|
|
{
|
|
|
|
|
window.UpdateTitle(title);
|
|
|
|
|
NameChanged?.Invoke(title);
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-17 16:15:10 +01:00
|
|
|
|
private void DisplayHandler_FaviconChanged(string uri)
|
|
|
|
|
{
|
2019-05-08 13:53:38 +02:00
|
|
|
|
var request = new HttpRequestMessage(HttpMethod.Head, uri);
|
|
|
|
|
var response = httpClient.SendAsync(request).ContinueWith(task =>
|
|
|
|
|
{
|
|
|
|
|
if (task.IsCompleted && task.Result.IsSuccessStatusCode)
|
|
|
|
|
{
|
|
|
|
|
var icon = new BrowserIconResource(uri);
|
|
|
|
|
|
|
|
|
|
IconChanged?.Invoke(icon);
|
|
|
|
|
window.UpdateIcon(icon);
|
|
|
|
|
}
|
|
|
|
|
});
|
2019-01-17 16:15:10 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-05-22 11:42:31 +02:00
|
|
|
|
private void DisplayHandler_ProgressChanged(double value)
|
|
|
|
|
{
|
|
|
|
|
window.UpdateProgress(value);
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-30 11:24:28 +01:00
|
|
|
|
private void DownloadHandler_ConfigurationDownloadRequested(string fileName, DownloadEventArgs args)
|
|
|
|
|
{
|
2018-11-08 09:39:52 +01:00
|
|
|
|
if (settings.AllowConfigurationDownloads)
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Forwarding download request for configuration file '{fileName}'.");
|
|
|
|
|
ConfigurationDownloadRequested?.Invoke(fileName, args);
|
2019-09-04 15:12:59 +02:00
|
|
|
|
|
|
|
|
|
if (args.AllowDownload)
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Download request for configuration file '{fileName}' was granted. Asking user to confirm the reconfiguration...");
|
|
|
|
|
|
|
|
|
|
var message = TextKey.MessageBox_ReconfigurationQuestion;
|
|
|
|
|
var title = TextKey.MessageBox_ReconfigurationQuestionTitle;
|
|
|
|
|
var result = messageBox.Show(message, title, MessageBoxAction.YesNo, MessageBoxIcon.Question, window);
|
|
|
|
|
|
|
|
|
|
args.AllowDownload = result == MessageBoxResult.Yes;
|
|
|
|
|
logger.Info($"The user chose to {(args.AllowDownload ? "start" : "abort")} the reconfiguration.");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Download request for configuration file '{fileName}' was denied.");
|
|
|
|
|
messageBox.Show(TextKey.MessageBox_ReconfigurationDenied, TextKey.MessageBox_ReconfigurationDeniedTitle, parent: window);
|
|
|
|
|
}
|
2018-11-08 09:39:52 +01:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Discarded download request for configuration file '{fileName}'.");
|
|
|
|
|
}
|
2018-10-30 11:24:28 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-17 11:12:17 +01:00
|
|
|
|
private void LifeSpanHandler_PopupRequested(PopupRequestedEventArgs args)
|
|
|
|
|
{
|
|
|
|
|
if (settings.AllowPopups)
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Forwarding request to open new window for '{args.Url}'...");
|
|
|
|
|
PopupRequested?.Invoke(args);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"Blocked attempt to open new window for '{args.Url}'.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 16:11:33 +01:00
|
|
|
|
private void ReloadRequested()
|
2017-07-31 20:22:53 +02:00
|
|
|
|
{
|
2019-01-23 09:37:47 +01:00
|
|
|
|
if (WindowSettings.AllowReloading && WindowSettings.ShowReloadWarning)
|
2019-01-18 16:11:33 +01:00
|
|
|
|
{
|
|
|
|
|
var result = messageBox.Show(TextKey.MessageBox_ReloadConfirmation, TextKey.MessageBox_ReloadConfirmationTitle, MessageBoxAction.YesNo, MessageBoxIcon.Question, window);
|
|
|
|
|
|
|
|
|
|
if (result == MessageBoxResult.Yes)
|
|
|
|
|
{
|
|
|
|
|
logger.Debug("The user confirmed reloading the current page...");
|
|
|
|
|
control.Reload();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Debug("The user aborted reloading the current page.");
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-01-23 09:37:47 +01:00
|
|
|
|
else if (WindowSettings.AllowReloading)
|
2019-01-18 16:11:33 +01:00
|
|
|
|
{
|
|
|
|
|
logger.Debug("Reloading current page...");
|
|
|
|
|
control.Reload();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Debug("Blocked reload attempt, as the user is not allowed to reload web pages.");
|
|
|
|
|
}
|
2017-07-31 20:22:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 16:11:33 +01:00
|
|
|
|
private void Window_AddressChanged(string address)
|
2017-07-31 20:22:53 +02:00
|
|
|
|
{
|
2019-03-15 15:55:34 +01:00
|
|
|
|
var isValid = Uri.TryCreate(address, UriKind.Absolute, out _) || Uri.TryCreate($"https://{address}", UriKind.Absolute, out _);
|
|
|
|
|
|
|
|
|
|
if (isValid)
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"The user requested to navigate to '{address}', the URI is valid.");
|
|
|
|
|
control.NavigateTo(address);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Debug($"The user requested to navigate to '{address}', but the URI is not valid.");
|
|
|
|
|
window.UpdateAddress(string.Empty);
|
|
|
|
|
}
|
2017-07-31 20:22:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void Window_BackwardNavigationRequested()
|
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
logger.Debug("Navigating backwards...");
|
2017-07-31 20:22:53 +02:00
|
|
|
|
control.NavigateBackwards();
|
2017-07-14 10:28:59 +02:00
|
|
|
|
}
|
2017-07-24 17:31:28 +02:00
|
|
|
|
|
2019-05-22 11:42:31 +02:00
|
|
|
|
private void Window_DeveloperConsoleRequested()
|
|
|
|
|
{
|
|
|
|
|
logger.Debug("Showing developer console...");
|
|
|
|
|
control.ShowDeveloperConsole();
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-31 20:22:53 +02:00
|
|
|
|
private void Window_ForwardNavigationRequested()
|
2017-07-24 17:31:28 +02:00
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
logger.Debug("Navigating forwards...");
|
2017-07-31 20:22:53 +02:00
|
|
|
|
control.NavigateForwards();
|
2017-07-24 17:31:28 +02:00
|
|
|
|
}
|
2019-01-18 16:11:33 +01:00
|
|
|
|
|
|
|
|
|
private void ZoomInRequested()
|
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
if (settings.AllowPageZoom && CalculateZoomPercentage() < 300)
|
2019-01-18 16:11:33 +01:00
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
zoomLevel += ZOOM_FACTOR;
|
|
|
|
|
control.Zoom(zoomLevel);
|
|
|
|
|
window.UpdateZoomLevel(CalculateZoomPercentage());
|
|
|
|
|
logger.Debug($"Increased page zoom to {CalculateZoomPercentage()}%.");
|
2019-01-18 16:11:33 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ZoomOutRequested()
|
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
if (settings.AllowPageZoom && CalculateZoomPercentage() > 25)
|
2019-01-18 16:11:33 +01:00
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
zoomLevel -= ZOOM_FACTOR;
|
|
|
|
|
control.Zoom(zoomLevel);
|
|
|
|
|
window.UpdateZoomLevel(CalculateZoomPercentage());
|
|
|
|
|
logger.Debug($"Decreased page zoom to {CalculateZoomPercentage()}%.");
|
2019-01-18 16:11:33 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void ZoomResetRequested()
|
|
|
|
|
{
|
|
|
|
|
if (settings.AllowPageZoom)
|
|
|
|
|
{
|
2019-05-22 11:42:31 +02:00
|
|
|
|
zoomLevel = 0;
|
|
|
|
|
control.Zoom(0);
|
|
|
|
|
window.UpdateZoomLevel(CalculateZoomPercentage());
|
|
|
|
|
logger.Debug($"Reset page zoom to {CalculateZoomPercentage()}%.");
|
2019-01-18 16:11:33 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-05-22 11:42:31 +02:00
|
|
|
|
|
|
|
|
|
private double CalculateZoomPercentage()
|
|
|
|
|
{
|
|
|
|
|
return (zoomLevel * 25.0) + 100.0;
|
|
|
|
|
}
|
2017-07-14 10:28:59 +02:00
|
|
|
|
}
|
|
|
|
|
}
|