using System; using System.Collections.Generic; using System.Globalization; using System.IO; using System.Linq; using System.Security.AccessControl; using System.Security.Principal; using System.Threading.Tasks; using System.Windows; using System.Windows.Markup; using System.Windows.Threading; using Artemis.Core; using Artemis.Core.Ninject; using Artemis.Core.Services; using Artemis.UI.Ninject; using Artemis.UI.Screens; using Artemis.UI.Screens.Splash; using Artemis.UI.Shared.Ninject; using Artemis.UI.Shared.Services; using Artemis.UI.Stylet; using Ninject; using Serilog; using Stylet; namespace Artemis.UI { public class Bootstrapper : NinjectBootstrapper { private ICoreService _core; public static List StartupArguments { get; private set; } protected override void OnExit(ExitEventArgs e) { // Stop the Artemis core _core?.Dispose(); base.OnExit(e); } protected override void Launch() { Core.Utilities.ShutdownRequested += UtilitiesOnShutdownRequested; ILogger logger = Kernel.Get(); IViewManager viewManager = Kernel.Get(); StartupArguments = Args.ToList(); CreateDataDirectory(logger); // Create the Artemis core try { _core = Kernel.Get(); } catch (Exception e) { HandleFatalException(e, logger); throw; } // Ensure WPF uses the right culture in binding converters FrameworkElement.LanguageProperty.OverrideMetadata(typeof(FrameworkElement), new FrameworkPropertyMetadata(XmlLanguage.GetLanguage(CultureInfo.CurrentCulture.IetfLanguageTag))); // Create and bind the root view, this is a tray icon so don't show it with the window manager Execute.OnUIThread(() => viewManager.CreateAndBindViewForModelIfNecessary(RootViewModel)); // Initialize the core async so the UI can show the progress Task.Run(async () => { try { if (StartupArguments.Contains("--autorun")) { logger.Information("Sleeping for 15 seconds on auto run to allow applications like iCUE and LGS to start"); await Task.Delay(TimeSpan.FromSeconds(15)); } _core.StartupArguments = StartupArguments; _core.Initialize(); } catch (Exception e) { HandleFatalException(e, logger); throw; } }); } protected override void ConfigureIoC(IKernel kernel) { // kernel.Settings.InjectNonPublic = true; // Load the UI modules kernel.Load(); kernel.Load(); // Load the core assembly's module kernel.Load(); base.ConfigureIoC(kernel); } protected override void OnUnhandledException(DispatcherUnhandledExceptionEventArgs e) { ILogger logger = Kernel.Get(); logger.Fatal(e.Exception, "Unhandled exception"); IDialogService dialogService = Kernel.Get(); try { dialogService.ShowExceptionDialog("Artemis encountered an error", e.Exception); } catch (Exception) { // We did our best eh.. trap exceptions during exception display here to avoid an infinite loop } // Don't shut down, is that a good idea? Depends on the exception of course.. e.Handled = true; } private void UtilitiesOnShutdownRequested(object? sender, EventArgs e) { Execute.OnUIThread(() => Application.Current.Shutdown()); } private void CreateDataDirectory(ILogger logger) { // Ensure the data folder exists if (Directory.Exists(Constants.DataFolder)) return; logger.Information("Creating data directory at {dataDirectoryFolder}", Constants.DataFolder); Directory.CreateDirectory(Constants.DataFolder); // During creation ensure all local users can access the data folder // This is needed when later running Artemis as a different user or when Artemis is first run as admin DirectoryInfo directoryInfo = new DirectoryInfo(Constants.DataFolder); DirectorySecurity accessControl = directoryInfo.GetAccessControl(); accessControl.AddAccessRule(new FileSystemAccessRule( new SecurityIdentifier(WellKnownSidType.BuiltinUsersSid, null), FileSystemRights.FullControl, InheritanceFlags.ObjectInherit | InheritanceFlags.ContainerInherit, PropagationFlags.InheritOnly, AccessControlType.Allow) ); directoryInfo.SetAccessControl(accessControl); } private void HandleFatalException(Exception e, ILogger logger) { logger.Fatal(e, "Fatal exception during initialization, shutting down."); // Can't use a pretty exception dialog here since the UI might not even be visible Execute.OnUIThread(() => { Kernel.Get().ShowMessageBox(e.Message + "\n\n Please refer the log file for more details.", "Fatal exception during initialization", MessageBoxButton.OK, MessageBoxImage.Error ); Environment.Exit(1); }); } private void ShowMainWindow(IWindowManager windowManager, SplashViewModel splashViewModel) { Execute.OnUIThread(() => { windowManager.ShowWindow(RootViewModel); splashViewModel.RequestClose(); }); } } }