using IPA.Config; using System; using System.Diagnostics; using System.IO; using System.Reflection; using System.Runtime.CompilerServices; using System.Threading; using UnityEngine; #if NET3 using Path = Net3_Proxy.Path; #endif namespace IPA.Utilities { /// /// Provides some basic utility methods and properties of Beat Saber /// public static class UnityGame { private static AlmostVersion _gameVersion; /// /// Provides the current game version. /// /// the SemVer version of the game public static AlmostVersion GameVersion => _gameVersion ?? (_gameVersion = new AlmostVersion(ApplicationVersionProxy)); internal static void SetEarlyGameVersion(AlmostVersion ver) { _gameVersion = ver; Logging.Logger.Default.Debug($"GameVersion set early to {ver}"); } private static string ApplicationVersionProxy { [MethodImpl(MethodImplOptions.NoInlining)] get { try { return Application.version; } catch(MissingMemberException ex) { Logging.Logger.Default.Error($"Tried to grab 'Application.version' too early, it's probably broken now."); if (SelfConfig.Debug_.ShowHandledErrorStackTraces_) Logging.Logger.Default.Error(ex); } catch (Exception ex) { Logging.Logger.Default.Error($"Error getting Application.version: {ex.Message}"); if (SelfConfig.Debug_.ShowHandledErrorStackTraces_) Logging.Logger.Default.Error(ex); } return string.Empty; } } internal static void EnsureRuntimeGameVersion() { try { var rtVer = new AlmostVersion(ApplicationVersionProxy); if (!rtVer.Equals(_gameVersion)) // this actually uses stricter equality than == for AlmostVersion { Logging.Logger.Default.Warn($"Early version {_gameVersion} parsed from game files doesn't match runtime version {rtVer}!"); _gameVersion = rtVer; } } catch (MissingMethodException e) { Logging.Logger.Default.Error("Application.version was not found! Cannot check early parsed version"); if (SelfConfig.Debug_.ShowHandledErrorStackTraces_) Logging.Logger.Default.Error(e); var st = new StackTrace(); Logging.Logger.Default.Notice($"{st}"); } } internal static bool IsGameVersionBoundary { get; private set; } internal static AlmostVersion OldVersion { get; private set; } internal static void CheckGameVersionBoundary() { var gameVer = GameVersion; var lastVerS = SelfConfig.LastGameVersion_; OldVersion = lastVerS != null ? new AlmostVersion(lastVerS, gameVer) : null; IsGameVersionBoundary = OldVersion != null && gameVer != OldVersion; SelfConfig.Instance.LastGameVersion = gameVer.ToString(); } private static Thread mainThread; /// /// Checks if the currently running code is running on the Unity main thread. /// /// if the curent thread is the Unity main thread, otherwise public static bool OnMainThread => Thread.CurrentThread.ManagedThreadId == mainThread?.ManagedThreadId; internal static void SetMainThread() => mainThread = Thread.CurrentThread; /// /// The different types of releases of the game. /// public enum Release { /// /// Indicates a Steam release. /// Steam, /// /// Indicates a non-Steam release. /// Other } private static Release? _releaseCache; /// /// Gets the type of this installation of Beat Saber /// /// /// This only gives a /// /// the type of release this is public static Release ReleaseType => (_releaseCache ?? (_releaseCache = CheckIsSteam() ? Release.Steam : Release.Other)).Value; private static string _installRoot; /// /// Gets the path to the game's install directory. /// /// the path of the game install directory public static string InstallPath { get { if (_installRoot == null) _installRoot = Path.GetFullPath( Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "..", "..")); return _installRoot; } } /// /// The path to the `Libs` folder. Use only if necessary. /// /// the path to the library directory public static string LibraryPath => Path.Combine(InstallPath, "Libs"); /// /// The path to the `Libs\Native` folder. Use only if necessary. /// /// the path to the native library directory public static string NativeLibraryPath => Path.Combine(LibraryPath, "Native"); /// /// The directory to load plugins from. /// /// the path to the plugin directory public static string PluginsPath => Path.Combine(InstallPath, "Plugins"); /// /// The path to the `UserData` folder. /// /// the path to the user data directory public static string UserDataPath => Path.Combine(InstallPath, "UserData"); private static bool CheckIsSteam() { var installDirInfo = new DirectoryInfo(InstallPath); return installDirInfo.Parent?.Name == "common" && installDirInfo.Parent?.Parent?.Name == "steamapps"; } } }