diff --git a/GCMM/AutoPatchingState.cs b/GCMM/AutoPatchingState.cs new file mode 100644 index 0000000..42e7c88 --- /dev/null +++ b/GCMM/AutoPatchingState.cs @@ -0,0 +1,9 @@ +namespace GCMM +{ + public enum AutoPatchingState + { + Unspecified, + Disabled, + Enabled + } +} \ No newline at end of file diff --git a/GCMM/DialogUtils.cs b/GCMM/DialogUtils.cs new file mode 100644 index 0000000..d2b056a --- /dev/null +++ b/GCMM/DialogUtils.cs @@ -0,0 +1,19 @@ +using System.Windows.Forms; + +namespace GCMM +{ + public static class DialogUtils + { + public static void ShowInfo(string message, string title) => + MessageBox.Show(message, title, MessageBoxButtons.OK, MessageBoxIcon.Information); + + public static void ShowError(string message, string title) => + MessageBox.Show(message, title, MessageBoxButtons.OK, MessageBoxIcon.Error); + + public static void ShowWarning(string message, string title) => + MessageBox.Show(message, title, MessageBoxButtons.OK, MessageBoxIcon.Warning); + + public static bool ShowYesNoQuestion(string message, string title) => + MessageBox.Show(message, title, MessageBoxButtons.YesNo, MessageBoxIcon.Warning) == DialogResult.Yes; + } +} \ No newline at end of file diff --git a/GCMM/MainForm.cs b/GCMM/MainForm.cs index 19b0c9d..ff092d8 100644 --- a/GCMM/MainForm.cs +++ b/GCMM/MainForm.cs @@ -2,6 +2,7 @@ using Newtonsoft.Json.Linq; using System; using System.Collections.Generic; +using System.ComponentModel; using System.Diagnostics; using System.Drawing; using System.IO; @@ -20,34 +21,33 @@ namespace GCMM public MainForm() { InitializeComponent(); + resources = new ComponentResourceManager(typeof(MainForm)); } + private readonly ComponentResourceManager resources; private readonly Dictionary mods = new Dictionary(); private readonly ModInfo gcipa = new ModInfo { Author = "modtainers", Name = "GCIPA" }; private readonly ModInfo gcmm = new ModInfo { Author = "NorbiPeti", Name = "GCMM" }; private DateTime lastGameUpdateTime; -#if USING_STEAM - private string steamPath; -#endif private const string defaultInfo = @" Techblox Mod Manager If you click on a mod it will show some info about it. The install instructions there are usually for manual installs. To get started, click on a mod and select Install mod. Most mods need TechbloxModdingAPI as well so it'll be installed. -Then launch Techblox: if you enabled auto-patching then you can use Steam but if you didn't then you must use the Play button. +Then launch Techblox: if you enabled auto-patching then you can use the launcher but if you didn't then you must use the Play button. This will first download and run the patcher (GCIPA) if needed. If all goes well, after some time a modded Techblox should launch. After a Techblox update there's a good chance that mods will break. If this happens you may get errors when trying to start Techblox. Until updated versions are released, use the ""Disable mods"" checkbox at the bottom to launch the game without mods. If you enabled auto-patching you will get a warning about this. -If you don't have auto-patching enabled then you will need to run the mod manager each time Techblox updates and click ""Patch & Play"". +If you don't have auto-patching enabled then you will need to run the mod manager each time Techblox updates and click ""Patch & Play"" or the game may not function properly. Disclaimer: This mod manager and the mods in the list are made by the ExMods developers. We are not associated with Freejam or Techblox. Modify Techblox at your own risk. If you encounter an issue while any mods are installed, report it to us. If you think it's an issue with the game, test again with the ""Disable mods"" option checked before reporting to Freejam. -You may also want to verify the game's files by clicking on the Validate game button. +You may also want to verify the game's files in the launcher. "; private async void Form1_Load(object sender, EventArgs e) @@ -68,41 +68,28 @@ You may also want to verify the game's files by clicking on the Validate game bu mods.Clear(); //This method may get called twice when ran from the command line UpdateButton(installbtn, false); modinfobox.Text = defaultInfo; -#if USING_STEAM - var (steamPath, user) = GetSteamLocationAndUser(); - if (steamPath != null) - this.steamPath = steamPath; - else - { - MessageBox.Show("Steam not found! If you have Steam installed, please report this to ExMods.\n\nThe Steam install is checked to autodetect where the game is installed and to optionally configure auto-patching.", "Steam not found"); - status.Text = "Status: Steam not found"; - return; - } - if (Settings.Default.SteamUserID == 0 && Settings.Default.AutoLaunch) - { - if (MessageBox.Show("Do you want GCMM to change the game's launch settings so it can ensure the game is patched?\n\n" + - "If you say yes, GCMM will do a quick check before the game is launched and patches if necessary. " + - "This way you (hopefully) won't see crashes after a Techblox update.\n\n" + - "Note that this also means that if you (re)move GCMM without disabling this in the settings then you won't be able to launch Techblox until you change the launch options in Steam.", - "GCMM auto-patching", MessageBoxButtons.YesNo) == DialogResult.Yes) - DetectConfigLocationAndAutoStart(steamPath, ref user); - else - Settings.Default.AutoLaunch = false; - Settings.Default.Save(); - } -#endif if (string.IsNullOrWhiteSpace(Settings.Default.GamePath) || GetExe() == null) { Settings.Default.GamePath = GetGameFolder(); if (string.IsNullOrWhiteSpace(Settings.Default.GamePath)) + { + DialogUtils.ShowWarning(resources.GetString("Game_not_found"), ""); Settings.Default.GamePath = SelectGameFolder(); + } else - MessageBox.Show("Found game at " + Settings.Default.GamePath); + DialogUtils.ShowInfo(string.Format(resources.GetString("Found_game_at"), Settings.Default.GamePath), ""); Settings.Default.Save(); } + if (Settings.Default.AutoLaunch == AutoPatchingState.Unspecified) + { + Settings.Default.AutoLaunch = AutoPatchingState.Disabled; + if (DialogUtils.ShowYesNoQuestion(resources.GetString("Change_launch_settings_question"), + resources.GetString("Change_launch_settings_title"))) + EnableDisableAutoPatchingWithDialog(true); + } if(string.IsNullOrWhiteSpace(Settings.Default.GamePath)) { - status.Text = "Status: Game not found"; + status.Text = resources.GetString("Status_Game_not_found"); return; } DeleteEmptyPluginsDir(out bool pexists, out bool dexists); diff --git a/GCMM/MainForm.resx b/GCMM/MainForm.resx index 292cae0..5c60fc3 100644 --- a/GCMM/MainForm.resx +++ b/GCMM/MainForm.resx @@ -1867,4 +1867,29 @@ APAIAADhCAAAw48AAMf/AAD//wAA//8AAA== + + Found game at {0} + + + Game not found. Is it installed correctly? Anyway, please locate the game. + + + Status: Game not found + + + Do you want TBMM to change the game's launch settings so it can ensure the game is patched? + +If you say yes, TBMM will do a quick check before the game is launched and patches if necessary. This way you (hopefully) won't see crashes after a Techblox update. + +Note that this also means that if you (re)move TBMM without disabling this in the settings then you won't be able to launch Techblox until you reinstall TBMM or fix the launcher configuration. + + + TBMM auto-patching + + + Launcher settings updated. + + + Failed to update launcher settings! You can still run TBMM manually or retry in settings. + \ No newline at end of file diff --git a/GCMM/MainUtils.cs b/GCMM/MainUtils.cs index 220860f..18bc29d 100644 --- a/GCMM/MainUtils.cs +++ b/GCMM/MainUtils.cs @@ -32,157 +32,64 @@ namespace GCMM } } - public string GetGameFolder() + private bool EnableDisableAutoPatching(bool enable) { -#if USING_STEAM - string libs; - if (Environment.OSVersion.Platform == PlatformID.Win32NT) - libs = steamPath + @"\steamapps\libraryfolders.vdf"; - else - return null; - foreach (var line in File.ReadAllLines(libs).Concat(new[] { "\t\"19\"\t\t\"" + steamPath + "\"" })) + string launcherConfig = + Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), + "Techblox Launcher", "launcher_settings.ini"); + bool gamePathSet = false; + if (File.Exists(launcherConfig)) { - var regex = new Regex("\\t\"\\d+\"\\t\\t\"(.+)\""); - var match = regex.Match(line); - if (!match.Success) - continue; - string library = match.Groups[1].Value.Replace("\\\\", "\\"); - library += @"\steamapps\common\"; - if (GetExe(library + "Techblox") != null) - return library + "Techblox"; - } - return null; -#endif - return null; //TODO - } - - public string SelectGameFolder() - { - var ofd = new OpenFileDialog(); - ofd.Filter = "Techblox executable|Techblox.exe|Techblox Preview executable|TechbloxPreview.exe"; - ofd.Title = "Game location"; -#if USING_STEAM - ofd.InitialDirectory = steamPath + @"\steamapps\common\"; -#endif //TODO - ofd.CheckFileExists = true; - ofd.ShowDialog(); - if (string.IsNullOrWhiteSpace(ofd.FileName)) - return null; - return Directory.GetParent(ofd.FileName).FullName; - } + string[] lines = File.ReadLines(launcherConfig).Select(line => + { + if (line.StartsWith("133062..GAME_PATH=") && line.Contains('/')) + { + gamePathSet = true; + return + $"{line.Substring(0, Math.Max(line.LastIndexOf("TBMM", StringComparison.Ordinal), line.Length))}{(enable ? "TBMM/" : "")}"; + } -#if USING_STEAM - public (string, int) AskForSteamLogin() - { - while (MessageBox.Show("Couid not find your Steam configuration to set launch options.\n\n" + - "Please make sure you are logged into Steam and click Retry or click Cancel to skip setting this up.", - "Steam config not found", MessageBoxButtons.RetryCancel) != DialogResult.Cancel) - { - var ret = GetSteamLocationAndUser(); - if (ret != (null, 0)) - return ret; + return line; + }).ToArray(); //Need to close the file + File.WriteAllLines(launcherConfig, lines); } - return (null, 0); + + return gamePathSet; } - /// - /// Does not return the current value if setting it is not possible because Steam is running. - /// - /// The value to set or null to keep as is - /// The current value, unless setting it while Steam is running - public bool UpdateOrGetSteamConfigToAutoStart(bool? autoLaunch) + public void EnableDisableAutoPatchingWithDialog(bool enable) { - string commandToUse = Application.ExecutablePath + " -start %command%"; - if (autoLaunch.HasValue && Process.GetProcessesByName("steam").Length > 0) - { //Setting it while Steam is running - if (MessageBox.Show("Cannot set launch options while Steam is running." + - (autoLaunch.Value - ? " Do you want to do it manually? If so, it will be copied on your clipboard." + - " Right click the game, select Properties and paste it in the launch options field." - : " Do you want to do it manually?" + - " If so, right click the game, select Properties and remove the text from the launch options field."), - "Launch options in Steam", MessageBoxButtons.YesNo) == DialogResult.Yes && autoLaunch.Value) - Clipboard.SetText(commandToUse); - return false; - } - var regex = new Regex(@"(\t{6}""LaunchOptions""\t+"")(.*)("")"); - string path = steamPath + @"\userdata\" + Settings.Default.SteamUserID + @"\config\localconfig.vdf"; - var lines = File.ReadAllLines(path); - bool shouldMatch = false; - bool ret = false; - for (int i = 0; i < lines.Length; i++) + if (EnableDisableAutoPatching(enable)) { - if (lines[i] == "\t\t\t\t\t\"1078000\"") - shouldMatch = true; //Found the game - else if(shouldMatch) - { - var match = regex.Match(lines[i]); - if (!match.Success) - continue; - ret = match.Groups[2].Value.Contains("GCMM.exe"); - string enabledCommand = match.Groups[1].Value + commandToUse.Replace("\\", "\\\\") + match.Groups[3].Value; - if (autoLaunch.HasValue) - { - if (autoLaunch.Value) - lines[i] = enabledCommand; - else - lines[i] = match.Groups[1].Value + match.Groups[3].Value; - File.WriteAllLines(path, lines); - } - else if (ret && lines[i] != enabledCommand) //GCMM got moved or something and it's only queried, not set - { - lines[i] = enabledCommand; - File.WriteAllLines(path, lines); - } - break; - } + DialogUtils.ShowInfo(resources.GetString("Change_launch_settings_done"), + resources.GetString("Change_launch_settings_title")); + Settings.Default.AutoLaunch = enable ? AutoPatchingState.Enabled : AutoPatchingState.Disabled; } - return ret; + else + DialogUtils.ShowError(resources.GetString("Change_launch_settings_error"), + resources.GetString("Change_launch_settings_title")); } - public (string, int) GetSteamLocationAndUser() + public string GetGameFolder() { - if (Environment.OSVersion.Platform != PlatformID.Win32NT) return (null, 0); - using (var key = Registry.CurrentUser.OpenSubKey(@"Software\Valve\Steam\ActiveProcess")) - { - string path = (string)key?.GetValue("SteamClientDll"); - path = path != null ? Directory.GetParent(path).FullName : null; - return (path, (int)(key?.GetValue("ActiveUser") ?? 0)); - } + string launcherConfig = + Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), + "Techblox Launcher", "launcher_settings.ini"); + if (File.Exists(launcherConfig)) + return File.ReadLines(launcherConfig).FirstOrDefault(line => line.StartsWith("133062..GAME_PATH=")) + ?.Substring("133062..GAME_PATH=".Length); + return null; } - public void DetectConfigLocationAndAutoStart(string steamPath, ref int user) + public string SelectGameFolder() { - string path = steamPath + @"\userdata"; - if (user == 0) - { - var dirs = Directory.GetDirectories(path); - var goodPaths = (from dir in dirs - where File.Exists(dir + @"\config\localconfig.vdf") - select dir).ToArray(); - if (goodPaths.Length != 1) - { - (_, user) = AskForSteamLogin(); - path += user; - } - else - { - path = goodPaths[0]; - user = int.Parse(Path.GetFileName(path)); - } - } - else - path += "\\" + user; - path += @"\config\localconfig.vdf"; - if (path != null && user != 0 && File.Exists(path)) - { - Settings.Default.SteamUserID = user; - UpdateOrGetSteamConfigToAutoStart(true); - } - else - Settings.Default.AutoLaunch = false; + var ofd = new OpenFileDialog(); + ofd.Filter = "Techblox executable|Techblox.exe|Techblox Preview executable|TechbloxPreview.exe"; + ofd.Title = "Game location"; + ofd.CheckFileExists = true; + ofd.ShowDialog(); + return string.IsNullOrWhiteSpace(ofd.FileName) ? null : Directory.GetParent(ofd.FileName)?.FullName; } -#endif private (EventHandler, Task) CheckStartGame(string command) { @@ -204,15 +111,11 @@ namespace GCMM await CheckModUpdatesAsync(); Process.Start(command); } -#if USING_STEAM - else if (Environment.OSVersion.Platform == PlatformID.Win32NT) - Process.Start("steam://run/1078000/"); - else - Process.Start("xdg-open", "steam://run/1078000/"); -#else if (Environment.OSVersion.Platform == PlatformID.Win32NT) - Process.Start(GamePath("\\" + GetExe())); -#endif + Process.Start(new ProcessStartInfo(GamePath("\\" + GetExe())) + { + WorkingDirectory = GamePath("\\") //Mods are only loaded if the working directory is correct + }); EndWork(false); tcs.SetResult(null); }; diff --git a/GCMM/Properties/Settings.Designer.cs b/GCMM/Properties/Settings.Designer.cs index 5f9ddfc..eb611aa 100644 --- a/GCMM/Properties/Settings.Designer.cs +++ b/GCMM/Properties/Settings.Designer.cs @@ -85,10 +85,10 @@ namespace GCMM.Properties { [global::System.Configuration.UserScopedSettingAttribute()] [global::System.Diagnostics.DebuggerNonUserCodeAttribute()] - [global::System.Configuration.DefaultSettingValueAttribute("True")] - public bool AutoLaunch { + [global::System.Configuration.DefaultSettingValueAttribute("0")] + public AutoPatchingState AutoLaunch { get { - return ((bool)(this["AutoLaunch"])); + return ((AutoPatchingState)(this["AutoLaunch"])); } set { this["AutoLaunch"] = value; diff --git a/GCMM/Properties/Settings.settings b/GCMM/Properties/Settings.settings index 266a9e4..b88984e 100644 --- a/GCMM/Properties/Settings.settings +++ b/GCMM/Properties/Settings.settings @@ -1,4 +1,4 @@ - + @@ -17,8 +17,9 @@ 0 - - True + + 0 - \ No newline at end of file + + diff --git a/GCMM/SettingsForm.cs b/GCMM/SettingsForm.cs index 939080a..ac3a5e9 100644 --- a/GCMM/SettingsForm.cs +++ b/GCMM/SettingsForm.cs @@ -16,9 +16,7 @@ namespace GCMM public partial class SettingsForm : Form { private MainForm mainForm; -#if USING_STEAM private bool autopatchingEnabled; -#endif public SettingsForm() { InitializeComponent(); @@ -29,10 +27,8 @@ namespace GCMM gamelocation.Text = Settings.Default.GamePath; useProxy.Checked = Settings.Default.UseProxy; mainForm = Owner as MainForm; -#if USING_STEAM - autopatchingEnabled = Settings.Default.SteamUserID != 0 && mainForm.UpdateOrGetSteamConfigToAutoStart(null); + autopatchingEnabled = Settings.Default.AutoLaunch == AutoPatchingState.Enabled; autopatching.Checked = autopatchingEnabled; -#endif } private void browsebtn_Click(object sender, EventArgs e) @@ -44,20 +40,8 @@ namespace GCMM { Settings.Default.GamePath = gamelocation.Text; Settings.Default.UseProxy = useProxy.Checked; -#if USING_STEAM if (autopatching.Checked != autopatchingEnabled) - { - if (autopatching.Checked && Settings.Default.SteamUserID == 0) - { - var (steamPath, user) = mainForm.GetSteamLocationAndUser(); - if (user != 0) - mainForm.DetectConfigLocationAndAutoStart(steamPath, ref user); - Settings.Default.SteamUserID = user; //If it's 0 then it's no change - } - else - mainForm.UpdateOrGetSteamConfigToAutoStart(autopatching.Checked); - } -#endif + mainForm.EnableDisableAutoPatchingWithDialog(autopatching.Checked); Settings.Default.Save(); Close(); }