diff --git a/SafeExamBrowser.Configuration/Settings/SettingsImpl.cs b/SafeExamBrowser.Configuration/Settings/SettingsImpl.cs index acce0f86..f5ef343f 100644 --- a/SafeExamBrowser.Configuration/Settings/SettingsImpl.cs +++ b/SafeExamBrowser.Configuration/Settings/SettingsImpl.cs @@ -28,6 +28,8 @@ namespace SafeExamBrowser.Configuration Mouse = new MouseSettings(); } + public bool AllowApplicationLog => true; + public string AppDataFolderName => "SafeExamBrowser"; public string ApplicationLogFile diff --git a/SafeExamBrowser.Contracts/Configuration/Settings/ISettings.cs b/SafeExamBrowser.Contracts/Configuration/Settings/ISettings.cs index 855f291e..c777ec57 100644 --- a/SafeExamBrowser.Contracts/Configuration/Settings/ISettings.cs +++ b/SafeExamBrowser.Contracts/Configuration/Settings/ISettings.cs @@ -10,6 +10,11 @@ namespace SafeExamBrowser.Contracts.Configuration.Settings { public interface ISettings { + /// + /// Determines whether the user may access the application log during runtime. + /// + bool AllowApplicationLog { get; } + /// /// The name used for the application data folder. /// diff --git a/SafeExamBrowser.Contracts/I18n/TextKey.cs b/SafeExamBrowser.Contracts/I18n/TextKey.cs index 234b28fb..50547810 100644 --- a/SafeExamBrowser.Contracts/I18n/TextKey.cs +++ b/SafeExamBrowser.Contracts/I18n/TextKey.cs @@ -14,6 +14,7 @@ namespace SafeExamBrowser.Contracts.I18n public enum TextKey { Browser_ShowDeveloperConsole, + LogWindow_Title, MessageBox_ShutdownError, MessageBox_ShutdownErrorTitle, MessageBox_SingleInstance, @@ -21,6 +22,7 @@ namespace SafeExamBrowser.Contracts.I18n MessageBox_StartupError, MessageBox_StartupErrorTitle, Notification_AboutTooltip, + Notification_LogTooltip, SplashScreen_InitializeBrowser, SplashScreen_InitializeProcessMonitoring, SplashScreen_InitializeTaskbar, diff --git a/SafeExamBrowser.Contracts/Logging/ILogContentFormatter.cs b/SafeExamBrowser.Contracts/Logging/ILogContentFormatter.cs new file mode 100644 index 00000000..3516a317 --- /dev/null +++ b/SafeExamBrowser.Contracts/Logging/ILogContentFormatter.cs @@ -0,0 +1,18 @@ +/* + * Copyright (c) 2017 ETH Zürich, Educational Development and Technology (LET) + * + * 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/. + */ + +namespace SafeExamBrowser.Contracts.Logging +{ + public interface ILogContentFormatter + { + /// + /// Formats the given log content and returns it as a string. + /// + string Format(ILogContent content); + } +} diff --git a/SafeExamBrowser.Contracts/SafeExamBrowser.Contracts.csproj b/SafeExamBrowser.Contracts/SafeExamBrowser.Contracts.csproj index 87709b55..2bbdd10e 100644 --- a/SafeExamBrowser.Contracts/SafeExamBrowser.Contracts.csproj +++ b/SafeExamBrowser.Contracts/SafeExamBrowser.Contracts.csproj @@ -76,6 +76,7 @@ + diff --git a/SafeExamBrowser.Contracts/UserInterface/IUserInterfaceFactory.cs b/SafeExamBrowser.Contracts/UserInterface/IUserInterfaceFactory.cs index 1fad76ab..a9f70736 100644 --- a/SafeExamBrowser.Contracts/UserInterface/IUserInterfaceFactory.cs +++ b/SafeExamBrowser.Contracts/UserInterface/IUserInterfaceFactory.cs @@ -9,6 +9,7 @@ using SafeExamBrowser.Contracts.Configuration; using SafeExamBrowser.Contracts.Configuration.Settings; using SafeExamBrowser.Contracts.I18n; +using SafeExamBrowser.Contracts.Logging; namespace SafeExamBrowser.Contracts.UserInterface { @@ -29,6 +30,11 @@ namespace SafeExamBrowser.Contracts.UserInterface /// IBrowserWindow CreateBrowserWindow(IBrowserControl control, IBrowserSettings settings); + /// + /// Creates a new log window which runs on its own thread. + /// + IWindow CreateLogWindow(ILogger logger, ILogContentFormatter formatter, IText text); + /// /// Creates a taskbar notification, initialized with the given notification information. /// diff --git a/SafeExamBrowser.Core/Behaviour/Operations/TaskbarOperation.cs b/SafeExamBrowser.Core/Behaviour/Operations/TaskbarOperation.cs index 9dae6d3c..b4e074af 100644 --- a/SafeExamBrowser.Core/Behaviour/Operations/TaskbarOperation.cs +++ b/SafeExamBrowser.Core/Behaviour/Operations/TaskbarOperation.cs @@ -18,7 +18,8 @@ namespace SafeExamBrowser.Core.Behaviour.Operations public class TaskbarOperation : IOperation { private ILogger logger; - private INotificationController aboutController; + private ILogContentFormatter formatter; + private INotificationController aboutController, logController; private ITaskbar taskbar; private IUserInterfaceFactory uiFactory; private IText text; @@ -26,9 +27,16 @@ namespace SafeExamBrowser.Core.Behaviour.Operations public ISplashScreen SplashScreen { private get; set; } - public TaskbarOperation(ILogger logger, ISettings settings, ITaskbar taskbar, IText text, IUserInterfaceFactory uiFactory) + public TaskbarOperation( + ILogger logger, + ILogContentFormatter formatter, + ISettings settings, + ITaskbar taskbar, + IText text, + IUserInterfaceFactory uiFactory) { this.logger = logger; + this.formatter = formatter; this.settings = settings; this.taskbar = taskbar; this.text = text; @@ -40,6 +48,22 @@ namespace SafeExamBrowser.Core.Behaviour.Operations logger.Info("Initializing taskbar..."); SplashScreen.UpdateText(TextKey.SplashScreen_InitializeTaskbar); + if (settings.AllowApplicationLog) + { + CreateLogNotification(); + } + + CreateAboutNotification(); + } + + public void Revert() + { + logController?.Terminate(); + aboutController.Terminate(); + } + + private void CreateAboutNotification() + { var aboutInfo = new AboutNotificationInfo(text); var aboutNotification = uiFactory.CreateNotification(aboutInfo); @@ -49,9 +73,15 @@ namespace SafeExamBrowser.Core.Behaviour.Operations taskbar.AddNotification(aboutNotification); } - public void Revert() + private void CreateLogNotification() { - aboutController.Terminate(); + var logInfo = new LogNotificationInfo(text); + var logNotification = uiFactory.CreateNotification(logInfo); + + logController = new LogNotificationController(logger, formatter, text, uiFactory); + logController.RegisterNotification(logNotification); + + taskbar.AddNotification(logNotification); } } } diff --git a/SafeExamBrowser.Core/I18n/Text.xml b/SafeExamBrowser.Core/I18n/Text.xml index af3d7330..cad11af8 100644 --- a/SafeExamBrowser.Core/I18n/Text.xml +++ b/SafeExamBrowser.Core/I18n/Text.xml @@ -1,11 +1,13 @@  Open Console + Application Log An unexpected error occurred during the shutdown procedure! Please consult the application log for more information... Shutdown Error An unexpected error occurred during the startup procedure! Please consult the application log for more information... Startup Error About Safe Exam Browser + Application Log Initializing browser Initializing process monitoring Initializing taskbar diff --git a/SafeExamBrowser.Core/Logging/DefaultLogFormatter.cs b/SafeExamBrowser.Core/Logging/DefaultLogFormatter.cs new file mode 100644 index 00000000..7340daf0 --- /dev/null +++ b/SafeExamBrowser.Core/Logging/DefaultLogFormatter.cs @@ -0,0 +1,40 @@ +/* + * Copyright (c) 2017 ETH Zürich, Educational Development and Technology (LET) + * + * 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 SafeExamBrowser.Contracts.Logging; + +namespace SafeExamBrowser.Core.Logging +{ + public class DefaultLogFormatter : ILogContentFormatter + { + public string Format(ILogContent content) + { + if (content is ILogText) + { + return (content as ILogText).Text; + } + + if (content is ILogMessage) + { + return FormatLogMessage(content as ILogMessage); + } + + throw new NotImplementedException($"The default formatter is not yet implemented for log content of type {content.GetType()}!"); + } + + private string FormatLogMessage(ILogMessage message) + { + var date = message.DateTime.ToString("yyyy-MM-dd HH:mm:ss.fff"); + var severity = message.Severity.ToString().ToUpper(); + var threadInfo = $"{message.ThreadInfo.Id}{(message.ThreadInfo.HasName ? ": " + message.ThreadInfo.Name : string.Empty)}"; + + return $"{date} [{threadInfo}] - {severity}: {message.Message}"; + } + } +} diff --git a/SafeExamBrowser.Core/Logging/LogFileWriter.cs b/SafeExamBrowser.Core/Logging/LogFileWriter.cs index 9c5204ce..788b97c3 100644 --- a/SafeExamBrowser.Core/Logging/LogFileWriter.cs +++ b/SafeExamBrowser.Core/Logging/LogFileWriter.cs @@ -17,51 +17,28 @@ namespace SafeExamBrowser.Core.Logging { private static readonly object @lock = new object(); private readonly string filePath; + private readonly ILogContentFormatter formatter; - public LogFileWriter(ISettings settings) + public LogFileWriter(ILogContentFormatter formatter, ISettings settings) { if (!Directory.Exists(settings.LogFolderPath)) { Directory.CreateDirectory(settings.LogFolderPath); } - filePath = settings.ApplicationLogFile; + this.filePath = settings.ApplicationLogFile; + this.formatter = formatter; } public void Notify(ILogContent content) - { - if (content is ILogText) - { - WriteLogText(content as ILogText); - } - - if (content is ILogMessage) - { - WriteLogMessage(content as ILogMessage); - } - } - - private void WriteLogText(ILogText text) - { - Write(text.Text); - } - - private void WriteLogMessage(ILogMessage message) - { - var date = message.DateTime.ToString("yyyy-MM-dd HH:mm:ss.fff"); - var severity = message.Severity.ToString().ToUpper(); - var threadInfo = $"{message.ThreadInfo.Id}{(message.ThreadInfo.HasName ? ": " + message.ThreadInfo.Name : string.Empty)}"; - - Write($"{date} [{threadInfo}] - {severity}: {message.Message}"); - } - - private void Write(string content) { lock (@lock) { + var raw = formatter.Format(content); + using (var stream = new StreamWriter(filePath, true, Encoding.UTF8)) { - stream.WriteLine(content); + stream.WriteLine(raw); } } } diff --git a/SafeExamBrowser.Core/Notifications/AboutNotificationController.cs b/SafeExamBrowser.Core/Notifications/AboutNotificationController.cs index 6a12b470..92f24a1c 100644 --- a/SafeExamBrowser.Core/Notifications/AboutNotificationController.cs +++ b/SafeExamBrowser.Core/Notifications/AboutNotificationController.cs @@ -6,7 +6,6 @@ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ -using System; using SafeExamBrowser.Contracts.Behaviour; using SafeExamBrowser.Contracts.Configuration.Settings; using SafeExamBrowser.Contracts.I18n; diff --git a/SafeExamBrowser.Core/Notifications/LogNotificationController.cs b/SafeExamBrowser.Core/Notifications/LogNotificationController.cs new file mode 100644 index 00000000..d2d410c4 --- /dev/null +++ b/SafeExamBrowser.Core/Notifications/LogNotificationController.cs @@ -0,0 +1,60 @@ +/* + * Copyright (c) 2017 ETH Zürich, Educational Development and Technology (LET) + * + * 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 SafeExamBrowser.Contracts.Behaviour; +using SafeExamBrowser.Contracts.I18n; +using SafeExamBrowser.Contracts.Logging; +using SafeExamBrowser.Contracts.UserInterface; + +namespace SafeExamBrowser.Core.Notifications +{ + public class LogNotificationController : INotificationController + { + private ITaskbarNotification notification; + private ILogger logger; + private ILogContentFormatter formatter; + private IText text; + private IUserInterfaceFactory uiFactory; + private IWindow window; + + public LogNotificationController(ILogger logger, ILogContentFormatter formatter, IText text, IUserInterfaceFactory uiFactory) + { + this.logger = logger; + this.formatter = formatter; + this.text = text; + this.uiFactory = uiFactory; + } + + public void RegisterNotification(ITaskbarNotification notification) + { + this.notification = notification; + + notification.Clicked += Notification_Clicked; + } + + public void Terminate() + { + window?.Close(); + } + + private void Notification_Clicked() + { + if (window == null) + { + window = uiFactory.CreateLogWindow(logger, formatter, text); + + window.Closing += () => window = null; + window.Show(); + } + else + { + window.BringToForeground(); + } + } + } +} diff --git a/SafeExamBrowser.Core/Notifications/LogNotificationIconResource.cs b/SafeExamBrowser.Core/Notifications/LogNotificationIconResource.cs new file mode 100644 index 00000000..18027804 --- /dev/null +++ b/SafeExamBrowser.Core/Notifications/LogNotificationIconResource.cs @@ -0,0 +1,20 @@ +/* + * Copyright (c) 2017 ETH Zürich, Educational Development and Technology (LET) + * + * 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 SafeExamBrowser.Contracts.Configuration; + +namespace SafeExamBrowser.Core.Notifications +{ + class LogNotificationIconResource : IIconResource + { + public Uri Uri => new Uri("pack://application:,,,/SafeExamBrowser.UserInterface;component/Images/LogNotification.ico"); + public bool IsBitmapResource => true; + public bool IsXamlResource => false; + } +} diff --git a/SafeExamBrowser.Core/Notifications/LogNotificationInfo.cs b/SafeExamBrowser.Core/Notifications/LogNotificationInfo.cs new file mode 100644 index 00000000..b76778c5 --- /dev/null +++ b/SafeExamBrowser.Core/Notifications/LogNotificationInfo.cs @@ -0,0 +1,26 @@ +/* + * Copyright (c) 2017 ETH Zürich, Educational Development and Technology (LET) + * + * 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 SafeExamBrowser.Contracts.Configuration; +using SafeExamBrowser.Contracts.I18n; + +namespace SafeExamBrowser.Core.Notifications +{ + class LogNotificationInfo : INotificationInfo + { + private IText text; + + public string Tooltip => text.Get(TextKey.Notification_LogTooltip); + public IIconResource IconResource { get; } = new LogNotificationIconResource(); + + public LogNotificationInfo(IText text) + { + this.text = text; + } + } +} diff --git a/SafeExamBrowser.Core/SafeExamBrowser.Core.csproj b/SafeExamBrowser.Core/SafeExamBrowser.Core.csproj index 1525377d..7cd14296 100644 --- a/SafeExamBrowser.Core/SafeExamBrowser.Core.csproj +++ b/SafeExamBrowser.Core/SafeExamBrowser.Core.csproj @@ -69,6 +69,7 @@ + @@ -80,6 +81,9 @@ + + + diff --git a/SafeExamBrowser.UserInterface/Images/LogNotification.ico b/SafeExamBrowser.UserInterface/Images/LogNotification.ico new file mode 100644 index 00000000..267f0e7d Binary files /dev/null and b/SafeExamBrowser.UserInterface/Images/LogNotification.ico differ diff --git a/SafeExamBrowser.UserInterface/LogWindow.xaml b/SafeExamBrowser.UserInterface/LogWindow.xaml new file mode 100644 index 00000000..7e0350ae --- /dev/null +++ b/SafeExamBrowser.UserInterface/LogWindow.xaml @@ -0,0 +1,18 @@ + + + + + + + + diff --git a/SafeExamBrowser.UserInterface/LogWindow.xaml.cs b/SafeExamBrowser.UserInterface/LogWindow.xaml.cs new file mode 100644 index 00000000..dd50344c --- /dev/null +++ b/SafeExamBrowser.UserInterface/LogWindow.xaml.cs @@ -0,0 +1,66 @@ +/* + * Copyright (c) 2017 ETH Zürich, Educational Development and Technology (LET) + * + * 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.Windows; +using SafeExamBrowser.Contracts.I18n; +using SafeExamBrowser.Contracts.Logging; +using SafeExamBrowser.Contracts.UserInterface; +using SafeExamBrowser.UserInterface.ViewModels; + +namespace SafeExamBrowser.UserInterface +{ + public partial class LogWindow : Window, IWindow + { + private ILogger logger; + private LogViewModel model; + private WindowClosingEventHandler closing; + + event WindowClosingEventHandler IWindow.Closing + { + add { closing += value; } + remove { closing -= value; } + } + + public LogWindow(ILogger logger, ILogContentFormatter formatter, IText text) + { + InitializeComponent(); + + this.logger = logger; + this.model = new LogViewModel(logger.GetLog(), formatter, text); + + DataContext = model; + LogContent.DataContext = model; + + logger.Subscribe(model); + } + + public void BringToForeground() + { + Dispatcher.Invoke(Activate); + } + + public new void Close() + { + Dispatcher.Invoke(() => + { + logger.Unsubscribe(model); + base.Close(); + }); + } + + public new void Show() + { + Dispatcher.Invoke(base.Show); + } + + private void LogContent_TextChanged(object sender, System.Windows.Controls.TextChangedEventArgs e) + { + LogContent.ScrollToEnd(); + } + } +} diff --git a/SafeExamBrowser.UserInterface/SafeExamBrowser.UserInterface.csproj b/SafeExamBrowser.UserInterface/SafeExamBrowser.UserInterface.csproj index 6af6c63b..f8a97bc0 100644 --- a/SafeExamBrowser.UserInterface/SafeExamBrowser.UserInterface.csproj +++ b/SafeExamBrowser.UserInterface/SafeExamBrowser.UserInterface.csproj @@ -88,6 +88,9 @@ QuitButton.xaml + + LogWindow.xaml + Code @@ -110,6 +113,7 @@ + ResXFileCodeGenerator @@ -149,6 +153,10 @@ Designer MSBuild:Compile + + Designer + MSBuild:Compile + Designer MSBuild:Compile @@ -173,5 +181,8 @@ + + + \ No newline at end of file diff --git a/SafeExamBrowser.UserInterface/UserInterfaceFactory.cs b/SafeExamBrowser.UserInterface/UserInterfaceFactory.cs index fd9d3d47..51a297da 100644 --- a/SafeExamBrowser.UserInterface/UserInterfaceFactory.cs +++ b/SafeExamBrowser.UserInterface/UserInterfaceFactory.cs @@ -11,6 +11,7 @@ using System.Windows; using SafeExamBrowser.Contracts.Configuration; using SafeExamBrowser.Contracts.Configuration.Settings; using SafeExamBrowser.Contracts.I18n; +using SafeExamBrowser.Contracts.Logging; using SafeExamBrowser.Contracts.UserInterface; using SafeExamBrowser.UserInterface.Controls; @@ -33,6 +34,31 @@ namespace SafeExamBrowser.UserInterface return new BrowserWindow(control, settings); } + public IWindow CreateLogWindow(ILogger logger, ILogContentFormatter formatter, IText text) + { + LogWindow logWindow = null; + var logWindowReadyEvent = new AutoResetEvent(false); + var logWindowThread = new Thread(() => + { + logWindow = new LogWindow(logger, formatter, text); + logWindow.Closed += (o, args) => logWindow.Dispatcher.InvokeShutdown(); + logWindow.Show(); + + logWindowReadyEvent.Set(); + + System.Windows.Threading.Dispatcher.Run(); + }); + + logWindowThread.SetApartmentState(ApartmentState.STA); + logWindowThread.Name = "Log Window Thread"; + logWindowThread.IsBackground = true; + logWindowThread.Start(); + + logWindowReadyEvent.WaitOne(); + + return logWindow; + } + public ITaskbarNotification CreateNotification(INotificationInfo info) { return new NotificationIcon(info); diff --git a/SafeExamBrowser.UserInterface/ViewModels/LogViewModel.cs b/SafeExamBrowser.UserInterface/ViewModels/LogViewModel.cs new file mode 100644 index 00000000..b48970a4 --- /dev/null +++ b/SafeExamBrowser.UserInterface/ViewModels/LogViewModel.cs @@ -0,0 +1,49 @@ +/* + * Copyright (c) 2017 ETH Zürich, Educational Development and Technology (LET) + * + * 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.Collections.Generic; +using System.ComponentModel; +using System.Text; +using SafeExamBrowser.Contracts.I18n; +using SafeExamBrowser.Contracts.Logging; + +namespace SafeExamBrowser.UserInterface.ViewModels +{ + class LogViewModel : INotifyPropertyChanged, ILogObserver + { + private ILogContentFormatter formatter; + private IText text; + private StringBuilder builder = new StringBuilder(); + + public string Text + { + get { return builder.ToString(); } + } + + public string WindowTitle => text.Get(TextKey.LogWindow_Title); + + public event PropertyChangedEventHandler PropertyChanged; + + public LogViewModel(IList initial, ILogContentFormatter formatter, IText text) + { + this.formatter = formatter; + this.text = text; + + foreach (var content in initial) + { + Notify(content); + } + } + + public void Notify(ILogContent content) + { + builder.AppendLine(formatter.Format(content)); + PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Text))); + } + } +} diff --git a/SafeExamBrowser/CompositionRoot.cs b/SafeExamBrowser/CompositionRoot.cs index 08505f4f..33eeba07 100644 --- a/SafeExamBrowser/CompositionRoot.cs +++ b/SafeExamBrowser/CompositionRoot.cs @@ -36,6 +36,7 @@ namespace SafeExamBrowser private IApplicationInfo browserInfo; private IKeyboardInterceptor keyboardInterceptor; private ILogger logger; + private ILogContentFormatter logFormatter; private IMouseInterceptor mouseInterceptor; private INativeMethods nativeMethods; private IProcessMonitor processMonitor; @@ -56,13 +57,14 @@ namespace SafeExamBrowser { browserInfo = new BrowserApplicationInfo(); logger = new Logger(); + logFormatter = new DefaultLogFormatter(); nativeMethods = new NativeMethods(); settings = new SettingsImpl(); Taskbar = new Taskbar(); textResource = new XmlTextResource(); uiFactory = new UserInterfaceFactory(); - logger.Subscribe(new LogFileWriter(settings)); + logger.Subscribe(new LogFileWriter(logFormatter, settings)); text = new Text(textResource); browserController = new BrowserApplicationController(settings, text, uiFactory); @@ -81,7 +83,7 @@ namespace SafeExamBrowser StartupOperations.Enqueue(new WindowMonitorOperation(logger, windowMonitor)); StartupOperations.Enqueue(new ProcessMonitorOperation(logger, processMonitor)); StartupOperations.Enqueue(new WorkingAreaOperation(logger, Taskbar, workingArea)); - StartupOperations.Enqueue(new TaskbarOperation(logger, settings, Taskbar, text, uiFactory)); + StartupOperations.Enqueue(new TaskbarOperation(logger, logFormatter, settings, Taskbar, text, uiFactory)); StartupOperations.Enqueue(new BrowserOperation(browserController, browserInfo, logger, Taskbar, uiFactory)); StartupOperations.Enqueue(new RuntimeControllerOperation(runtimeController, logger)); StartupOperations.Enqueue(new MouseInterceptorOperation(logger, mouseInterceptor, nativeMethods));