mirror of
https://github.com/Artemis-RGB/Artemis
synced 2025-12-13 05:48:35 +00:00
197 lines
7.6 KiB
C#
197 lines
7.6 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Net.Http;
|
|
using System.Reflection;
|
|
using System.Security.Principal;
|
|
using System.Threading;
|
|
using System.Windows;
|
|
using Artemis.Core;
|
|
using Artemis.UI.Utilities;
|
|
using Ninject;
|
|
using Ookii.Dialogs.Wpf;
|
|
using Stylet;
|
|
|
|
namespace Artemis.UI
|
|
{
|
|
public class ApplicationStateManager
|
|
{
|
|
// ReSharper disable once NotAccessedField.Local - Kept in scope to ensure it does not get released
|
|
private Mutex _artemisMutex;
|
|
|
|
public ApplicationStateManager(IKernel kernel, string[] startupArguments)
|
|
{
|
|
StartupArguments = startupArguments;
|
|
IsElevated = new WindowsPrincipal(WindowsIdentity.GetCurrent()).IsInRole(WindowsBuiltInRole.Administrator);
|
|
|
|
Core.Utilities.ShutdownRequested += UtilitiesOnShutdownRequested;
|
|
Core.Utilities.RestartRequested += UtilitiesOnRestartRequested;
|
|
|
|
// On Windows shutdown dispose the kernel just so device providers get a chance to clean up
|
|
Application.Current.SessionEnding += (_, _) => kernel.Dispose();
|
|
}
|
|
|
|
public string[] StartupArguments { get; }
|
|
public bool IsElevated { get; }
|
|
|
|
public bool FocusExistingInstance()
|
|
{
|
|
_artemisMutex = new Mutex(true, "Artemis-3c24b502-64e6-4587-84bf-9072970e535d", out bool createdNew);
|
|
if (createdNew)
|
|
return false;
|
|
|
|
return RemoteFocus();
|
|
}
|
|
|
|
public void DisplayException(Exception e)
|
|
{
|
|
using TaskDialog dialog = new();
|
|
AssemblyInformationalVersionAttribute versionAttribute = typeof(ApplicationStateManager).Assembly.GetCustomAttribute<AssemblyInformationalVersionAttribute>();
|
|
dialog.WindowTitle = $"Artemis {versionAttribute?.InformationalVersion} build {Constants.BuildInfo.BuildNumberDisplay}";
|
|
dialog.MainInstruction = "Unfortunately Artemis ran into an unhandled exception and cannot continue.";
|
|
dialog.Content = e.Message;
|
|
dialog.ExpandedInformation = e.StackTrace.Trim();
|
|
|
|
dialog.CollapsedControlText = "Show stack trace";
|
|
dialog.ExpandedControlText = "Hide stack trace";
|
|
|
|
dialog.Footer = "If this keeps happening check out the <a href=\"https://wiki.artemis-rgb.com\">wiki</a> or hit us up on <a href=\"https://discord.gg/S3MVaC9\">Discord</a>.";
|
|
dialog.FooterIcon = TaskDialogIcon.Error;
|
|
dialog.EnableHyperlinks = true;
|
|
dialog.HyperlinkClicked += OpenHyperlink;
|
|
|
|
TaskDialogButton copyButton = new("Copy stack trace");
|
|
TaskDialogButton closeButton = new("Close") {Default = true};
|
|
dialog.Buttons.Add(copyButton);
|
|
dialog.Buttons.Add(closeButton);
|
|
dialog.ButtonClicked += (_, args) =>
|
|
{
|
|
if (args.Item == copyButton)
|
|
{
|
|
Clipboard.SetText(e.ToString());
|
|
args.Cancel = true;
|
|
}
|
|
};
|
|
|
|
dialog.ShowDialog(Application.Current.MainWindow);
|
|
}
|
|
|
|
private bool RemoteFocus()
|
|
{
|
|
// At this point we cannot read the database yet to retrieve the web server port.
|
|
// Instead use the method external applications should use as well.
|
|
if (!File.Exists(Path.Combine(Constants.DataFolder, "webserver.txt")))
|
|
{
|
|
KillOtherInstances();
|
|
return false;
|
|
}
|
|
|
|
string url = File.ReadAllText(Path.Combine(Constants.DataFolder, "webserver.txt"));
|
|
using HttpClient client = new();
|
|
try
|
|
{
|
|
HttpResponseMessage httpResponseMessage = client.Send(new HttpRequestMessage(HttpMethod.Post, url + "remote/bring-to-foreground"));
|
|
httpResponseMessage.EnsureSuccessStatusCode();
|
|
return true;
|
|
}
|
|
catch (Exception)
|
|
{
|
|
KillOtherInstances();
|
|
return false;
|
|
}
|
|
}
|
|
|
|
private void KillOtherInstances()
|
|
{
|
|
// Kill everything else heh
|
|
List<Process> processes = Process.GetProcessesByName("Artemis.UI").Where(p => p.Id != Process.GetCurrentProcess().Id).ToList();
|
|
foreach (Process process in processes)
|
|
{
|
|
try
|
|
{
|
|
process.Kill(true);
|
|
}
|
|
catch (Exception)
|
|
{
|
|
// ignored
|
|
}
|
|
}
|
|
}
|
|
|
|
private void UtilitiesOnRestartRequested(object sender, RestartEventArgs e)
|
|
{
|
|
List<string> argsList = new();
|
|
argsList.AddRange(StartupArguments);
|
|
if (e.ExtraArgs != null)
|
|
argsList.AddRange(e.ExtraArgs.Except(argsList));
|
|
string args = argsList.Any() ? "-ArgumentList " + string.Join(',', argsList) : "";
|
|
string command =
|
|
$"-Command \"& {{Start-Sleep -Milliseconds {(int) e.Delay.TotalMilliseconds}; " +
|
|
"(Get-Process 'Artemis.UI').kill(); " +
|
|
$"Start-Process -FilePath '{Constants.ExecutablePath}' -WorkingDirectory '{Constants.ApplicationFolder}' {args}}}\"";
|
|
// Elevated always runs with RunAs
|
|
if (e.Elevate)
|
|
{
|
|
ProcessStartInfo info = new()
|
|
{
|
|
Arguments = command.Replace("}\"", " -Verb RunAs}\""),
|
|
WindowStyle = ProcessWindowStyle.Hidden,
|
|
CreateNoWindow = true,
|
|
FileName = "PowerShell.exe"
|
|
};
|
|
Process.Start(info);
|
|
}
|
|
// Non-elevated runs regularly if currently not elevated
|
|
else if (!IsElevated)
|
|
{
|
|
ProcessStartInfo info = new()
|
|
{
|
|
Arguments = command,
|
|
WindowStyle = ProcessWindowStyle.Hidden,
|
|
CreateNoWindow = true,
|
|
FileName = "PowerShell.exe"
|
|
};
|
|
Process.Start(info);
|
|
}
|
|
// Non-elevated runs via a utility method is currently elevated (de-elevating is hacky)
|
|
else
|
|
{
|
|
string powerShell = Path.Combine(Environment.SystemDirectory, "WindowsPowerShell", "v1.0", "powershell.exe");
|
|
ProcessUtilities.RunAsDesktopUser(powerShell, command, true);
|
|
}
|
|
|
|
// Lets try a graceful shutdown, PowerShell will kill if needed
|
|
Execute.OnUIThread(() => Application.Current.Shutdown());
|
|
}
|
|
|
|
private void UtilitiesOnShutdownRequested(object sender, EventArgs e)
|
|
{
|
|
// Use PowerShell to kill the process after 8 sec just in case
|
|
if (!StartupArguments.Contains("--disable-forced-shutdown"))
|
|
{
|
|
ProcessStartInfo info = new()
|
|
{
|
|
Arguments = "-Command \"& {Start-Sleep -s 8; (Get-Process -Id " + Process.GetCurrentProcess().Id + ").kill()}",
|
|
WindowStyle = ProcessWindowStyle.Hidden,
|
|
CreateNoWindow = true,
|
|
FileName = "PowerShell.exe"
|
|
};
|
|
Process.Start(info);
|
|
}
|
|
|
|
Execute.OnUIThread(() => Application.Current.Shutdown());
|
|
}
|
|
|
|
private void OpenHyperlink(object sender, HyperlinkClickedEventArgs e)
|
|
{
|
|
ProcessStartInfo processInfo = new()
|
|
{
|
|
FileName = e.Href,
|
|
UseShellExecute = true
|
|
};
|
|
Process.Start(processInfo);
|
|
}
|
|
}
|
|
} |