// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Diagnostics; using System.IO; using System.Reflection; using System.Runtime.Versioning; using Microsoft.Win32; using osu.Desktop.Security; using osu.Framework.Platform; using osu.Game; using osu.Desktop.Updater; using osu.Framework; using osu.Framework.Logging; using osu.Game.Updater; using osu.Desktop.Windows; using osu.Game.IO; using osu.Game.IPC; using osu.Game.Online.Multiplayer; using osu.Game.Utils; using SDL2; namespace osu.Desktop { internal partial class OsuGameDesktop : OsuGame { private OsuSchemeLinkIPCChannel? osuSchemeLinkIPCChannel; private ArchiveImportIPCChannel? archiveImportIPCChannel; public OsuGameDesktop(string[]? args = null) : base(args) { } public override StableStorage? GetStorageForStableInstall() { try { if (Host is DesktopGameHost desktopHost) { string? stablePath = getStableInstallPath(); if (!string.IsNullOrEmpty(stablePath)) return new StableStorage(stablePath, desktopHost); } } catch (Exception) { Logger.Log("Could not find a stable install", LoggingTarget.Runtime, LogLevel.Important); } return null; } private string? getStableInstallPath() { static bool checkExists(string p) => Directory.Exists(Path.Combine(p, "Songs")) || File.Exists(Path.Combine(p, "osu!.cfg")); string? stableInstallPath; if (OperatingSystem.IsWindows()) { try { stableInstallPath = getStableInstallPathFromRegistry(); if (!string.IsNullOrEmpty(stableInstallPath) && checkExists(stableInstallPath)) return stableInstallPath; } catch { } } stableInstallPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), @"osu!"); if (checkExists(stableInstallPath)) return stableInstallPath; stableInstallPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".osu"); if (checkExists(stableInstallPath)) return stableInstallPath; return null; } [SupportedOSPlatform("windows")] private string? getStableInstallPathFromRegistry() { using (RegistryKey? key = Registry.ClassesRoot.OpenSubKey("osu")) return key?.OpenSubKey(@"shell\open\command")?.GetValue(string.Empty)?.ToString()?.Split('"')[1].Replace("osu!.exe", ""); } protected override UpdateManager CreateUpdateManager() { string? packageManaged = Environment.GetEnvironmentVariable("OSU_EXTERNAL_UPDATE_PROVIDER"); if (!string.IsNullOrEmpty(packageManaged)) return new NoActionUpdateManager(); switch (RuntimeInfo.OS) { case RuntimeInfo.Platform.Windows: Debug.Assert(OperatingSystem.IsWindows()); return new SquirrelUpdateManager(); default: return new SimpleUpdateManager(); } } public override bool RestartAppWhenExited() { switch (RuntimeInfo.OS) { case RuntimeInfo.Platform.Windows: Debug.Assert(OperatingSystem.IsWindows()); // Of note, this is an async method in squirrel that adds an arbitrary delay before returning // likely to ensure the external process is in a good state. // // We're not waiting on that here, but the outro playing before the actual exit should be enough // to cover this. Squirrel.UpdateManager.RestartAppWhenExited().FireAndForget(); return true; } return base.RestartAppWhenExited(); } protected override void LoadComplete() { base.LoadComplete(); LoadComponentAsync(new DiscordRichPresence(), Add); if (RuntimeInfo.OS == RuntimeInfo.Platform.Windows) LoadComponentAsync(new GameplayWinKeyBlocker(), Add); LoadComponentAsync(new ElevatedPrivilegesChecker(), Add); osuSchemeLinkIPCChannel = new OsuSchemeLinkIPCChannel(Host, this); archiveImportIPCChannel = new ArchiveImportIPCChannel(Host, this); } public override void SetHost(GameHost host) { base.SetHost(host); var iconStream = Assembly.GetExecutingAssembly().GetManifestResourceStream(GetType(), "lazer.ico"); if (iconStream != null) host.Window.SetIconFromStream(iconStream); host.Window.CursorState |= CursorState.Hidden; host.Window.Title = Name; } protected override BatteryInfo CreateBatteryInfo() => new SDL2BatteryInfo(); protected override void Dispose(bool isDisposing) { base.Dispose(isDisposing); osuSchemeLinkIPCChannel?.Dispose(); archiveImportIPCChannel?.Dispose(); } private class SDL2BatteryInfo : BatteryInfo { public override double? ChargeLevel { get { SDL.SDL_GetPowerInfo(out _, out int percentage); if (percentage == -1) return null; return percentage / 100.0; } } public override bool OnBattery => SDL.SDL_GetPowerInfo(out _, out _) == SDL.SDL_PowerState.SDL_POWERSTATE_ON_BATTERY; } } }