diff --git a/Wabbajack.Common/GOGHandler.cs b/Wabbajack.Common/GOGHandler.cs deleted file mode 100644 index 4a3a4f57..00000000 --- a/Wabbajack.Common/GOGHandler.cs +++ /dev/null @@ -1,124 +0,0 @@ -using System; -using System.Collections.Generic; -using System.IO; -using System.Linq; -using Microsoft.Win32; - -namespace Wabbajack.Common -{ - public class GOGGame - { - public int GameID; - public string Path; - public string GameName; - public Game? Game; - } - - /// - /// Class for all GOG operations - /// - public class GOGHandler - { - private static readonly Lazy instance = new Lazy( - () => new GOGHandler(true), - isThreadSafe: true); - public static GOGHandler Instance => instance.Value; - - private const string GOGRegKey = @"Software\GOG.com\Games"; - private const string GOG64RegKey = @"Software\WOW6432Node\GOG.com\Games"; - - public HashSet Games { get; internal set; } - public RegistryKey GOGKey { get; internal set; } - - public GOGHandler(bool init) - { - var gogKey = Registry.LocalMachine.OpenSubKey(GOGRegKey) ?? Registry.LocalMachine.OpenSubKey(GOG64RegKey); - if (gogKey == null) - { - Utils.ErrorThrow(new Exception("Could not find GOG in registry!")); - } - else - { - Utils.Log($"GOG registry key is ${gogKey}"); - GOGKey = gogKey; - if (!init) return; - LoadAllGames(); - } - } - - /// - /// Finds the installation path of a GOG game by ID - /// - /// ID of the GOG game - /// - public string GetGamePathById(int id) - { - return Games.FirstOrDefault(f => f.GameID == id)?.Path; - } - - /// - /// Enumerates through all subkeys found in the GOG registry entry to get all - /// GOG games - /// - public void LoadAllGames() - { - Games = new HashSet(); - if (GOGKey == null) return; - string[] keys = GOGKey.GetSubKeyNames(); - Utils.Log($"Found {keys.Length} SubKeys for GOG"); - foreach (var key in keys) - { - if (!int.TryParse(key, out var gameID)) - { - Utils.ErrorThrow(new Exception($"Could not read gameID for key {key}")); - } - - var subKey = GOGKey.OpenSubKey(key); - if (subKey == null) - { - Utils.ErrorThrow(new Exception($"Could not open SubKey for {key}")); - return; - } - - var gameNameValue = subKey.GetValue("GAMENAME"); - if (gameNameValue == null) - { - Utils.ErrorThrow(new Exception($"Could not get GAMENAME for {gameID} at {key}")); - return; - } - - var gameName = gameNameValue.ToString(); - - var pathValue = subKey.GetValue("PATH"); - if (pathValue == null) - { - Utils.ErrorThrow(new Exception($"Could not get PATH for {gameID} at {key}")); - return; - } - - var path = pathValue.ToString(); - - - var game = new GOGGame - { - GameID = gameID, - GameName = gameName, - Path = path - }; - - Utils.Log($"Found GOG Game: {gameName}({gameID}) at {path}"); - - game.Game = GameRegistry.Games.Values - .FirstOrDefault(g => g.GOGIDs != null && g.GOGIDs.Contains(game.GameID) - && - g.RequiredFiles.TrueForAll(s => File.Exists(Path.Combine(game.Path, s))))?.Game; - - if (game.Game == null) - { - Utils.Log("Found no matching game, continuing"); - } - Games.Add(game); - } - } - } -} diff --git a/Wabbajack.Common/GameMetaData.cs b/Wabbajack.Common/GameMetaData.cs index b89f50ac..fb09fb3e 100644 --- a/Wabbajack.Common/GameMetaData.cs +++ b/Wabbajack.Common/GameMetaData.cs @@ -3,6 +3,7 @@ using System.ComponentModel; using System.Linq; using Alphaleonis.Win32.Filesystem; using Microsoft.Win32; +using Wabbajack.Common.StoreHandlers; namespace Wabbajack.Common { @@ -71,11 +72,7 @@ namespace Wabbajack.Common public string GameLocation() { - if (Consts.TestMode) - return Directory.GetCurrentDirectory(); - - return SteamHandler.Instance.Games.FirstOrDefault(g => g.Game == Game)?.InstallDir ?? - GOGHandler.Instance.Games.FirstOrDefault(g => g.Game == Game)?.Path; + return Consts.TestMode ? Directory.GetCurrentDirectory() : StoreHandler.Instance.GetGamePath(Game); } } diff --git a/Wabbajack.Common/SteamHandler.cs b/Wabbajack.Common/SteamHandler.cs deleted file mode 100644 index 86c60e31..00000000 --- a/Wabbajack.Common/SteamHandler.cs +++ /dev/null @@ -1,278 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Diagnostics; -using System.IO; -using System.Linq; -using System.Text; -using DynamicData; -using Microsoft.Win32; - -namespace Wabbajack.Common -{ - [DebuggerDisplay("{" + nameof(Name) + "}")] - public class SteamGame - { - public int AppId; - public string Name; - public string InstallDir; - public Game? Game; - - public HashSet WorkshopItems; - /// - /// Combined size of all installed workshop items on disk in bytes - /// - public int WorkshopItemsSize; - } - - public class SteamWorkshopItem - { - public SteamGame Game; - public int ItemID; - /// - /// Size is in bytes - /// - public int Size; - } - - /// - /// Class for all Steam operations - /// - public class SteamHandler - { - private static readonly Lazy instance = new Lazy( - () => new SteamHandler(true), - isThreadSafe: true); - public static SteamHandler Instance => instance.Value; - - private const string SteamRegKey = @"Software\Valve\Steam"; - - /// - /// Path to the Steam folder - /// - public string SteamPath { get; internal set; } - /// - /// HashSet of all known Steam Libraries - /// - public HashSet InstallFolders { get; internal set; } - /// - /// HashSet of all known SteamGames - /// - public HashSet Games { get; internal set; } - - private string SteamConfig => Path.Combine(SteamPath, "config", "config.vdf"); - - public SteamHandler(bool init) - { - var steamKey = Registry.CurrentUser.OpenSubKey(SteamRegKey); - SteamPath = steamKey?.GetValue("SteamPath").ToString(); - if(string.IsNullOrWhiteSpace(SteamPath) || steamKey == null || !Directory.Exists(SteamPath)) - Utils.ErrorThrow(new Exception("Could not find the Steam folder!")); - if(!init) return; - LoadInstallFolders(); - LoadAllSteamGames(); - } - - /// - /// Finds the installation path of a Steam game by ID - /// - /// ID of the Steam game - /// - public string GetGamePathById(int id) - { - return Games.FirstOrDefault(f => f.AppId == id)?.InstallDir; - } - - /// - /// Reads the config file and adds all found installation folders to the HashSet - /// - public void LoadInstallFolders() - { - var paths = new HashSet(); - - File.ReadLines(SteamConfig, Encoding.UTF8).Do(l => - { - if (!l.Contains("BaseInstallFolder_")) return; - var s = GetVdfValue(l); - s = Path.Combine(s, "steamapps"); - if (!Directory.Exists(s)) - return; - - paths.Add(s); - Utils.Log($"Steam Library found at {s}"); - }); - - Utils.Log($"Total number of Steam Libraries found: {paths.Count}"); - - // Default path in the Steam folder isn't in the configs - if(Directory.Exists(Path.Combine(SteamPath, "steamapps"))) - paths.Add(Path.Combine(SteamPath, "steamapps")); - - InstallFolders = paths; - } - - /// - /// Enumerates through all Steam Libraries to find and read all .afc files, adding the found game - /// to the HashSet - /// - public void LoadAllSteamGames() - { - var games = new HashSet(); - - InstallFolders.Do(p => - { - Directory.EnumerateFiles(p, "*.acf", SearchOption.TopDirectoryOnly).Where(File.Exists).Do(f => - { - var steamGame = new SteamGame(); - var valid = false; - File.ReadAllLines(f, Encoding.UTF8).Do(l => - { - if(l.Contains("\"appid\"")) - if (!int.TryParse(GetVdfValue(l), out steamGame.AppId)) - return; - if(l.Contains("\"name\"")) - steamGame.Name = GetVdfValue(l); - if (l.Contains("\"installdir\"")) - { - var path = Path.Combine(p, "common", GetVdfValue(l)); - steamGame.InstallDir = Directory.Exists(path) ? path : null; - } - - if (steamGame.AppId != 0 && !string.IsNullOrWhiteSpace(steamGame.Name) && - !string.IsNullOrWhiteSpace(steamGame.InstallDir)) - valid = true; - }); - - if (!valid) - return; - - steamGame.Game = GameRegistry.Games.Values - .FirstOrDefault(g => - g.SteamIDs.Contains(steamGame.AppId) - && - g.RequiredFiles.TrueForAll(s => File.Exists(Path.Combine(steamGame.InstallDir, s))) - )?.Game; - games.Add(steamGame); - - Utils.Log($"Found Game: {steamGame.Name} ({steamGame.AppId}) at {steamGame.InstallDir}"); - }); - }); - - Utils.Log($"Total number of Steam Games found: {games.Count}"); - - Games = games; - } - - public void LoadWorkshopItems(SteamGame game) - { - if(game.WorkshopItems == null) - game.WorkshopItems = new HashSet(); - - InstallFolders.Do(p => - { - var workshop = Path.Combine(p, "workshop"); - if(!Directory.Exists(workshop)) - return; - - Directory.EnumerateFiles(workshop, "*.acf", SearchOption.TopDirectoryOnly).Where(File.Exists).Do(f => - { - if (Path.GetFileName(f) != $"appworkshop_{game.AppId}.acf") - return; - - var foundAppID = false; - var workshopItemsInstalled = 0; - var workshopItemDetails = 0; - var currentItem = new SteamWorkshopItem(); - var bracketStart = 0; - var bracketEnd = 0; - var lines = File.ReadAllLines(f, Encoding.UTF8); - var end = false; - lines.Do(l => - { - if (end) - return; - if(currentItem == null) - currentItem = new SteamWorkshopItem(); - - var currentLine = lines.IndexOf(l); - if (l.Contains("\"appid\"") && !foundAppID) - { - if (!int.TryParse(GetVdfValue(l), out var appID)) - return; - - foundAppID = true; - - if (appID != game.AppId) - return; - } - - if (!foundAppID) - return; - - if (l.Contains("\"SizeOnDisk\"")) - { - if (!int.TryParse(GetVdfValue(l), out var sizeOnDisk)) - return; - - game.WorkshopItemsSize = sizeOnDisk; - } - - if (l.Contains("\"WorkshopItemsInstalled\"")) - workshopItemsInstalled = currentLine; - - if (l.Contains("\"WorkshopItemDetails\"")) - workshopItemDetails = currentLine; - - if (workshopItemsInstalled == 0) - return; - - if (currentLine <= workshopItemsInstalled + 1 && currentLine >= workshopItemDetails - 1) - return; - - if (currentItem.ItemID == 0) - if (!int.TryParse(GetSingleVdfValue(l), out currentItem.ItemID)) - return; - - if (currentItem.ItemID == 0) - return; - - if (bracketStart == 0 && l.Contains("{")) - bracketStart = currentLine; - - if (bracketEnd == 0 && l.Contains("}")) - bracketEnd = currentLine; - - if (bracketStart == 0) - return; - - if (currentLine == bracketStart + 1) - if (!int.TryParse(GetVdfValue(l), out currentItem.Size)) - return; - - if (bracketStart == 0 || bracketEnd == 0 || currentItem.ItemID == 0 || currentItem.Size == 0) - return; - - bracketStart = 0; - bracketEnd = 0; - currentItem.Game = game; - game.WorkshopItems.Add(currentItem); - currentItem = null; - end = true; - }); - }); - }); - } - - private static string GetVdfValue(string line) - { - var trim = line.Trim('\t').Replace("\t", ""); - string[] s = trim.Split('\"'); - return s[3].Replace("\\\\", "\\"); - } - - private static string GetSingleVdfValue(string line) - { - var trim = line.Trim('\t').Replace("\t", ""); - return trim.Split('\"')[1]; - } - } -} diff --git a/Wabbajack.Lib/CompilationSteps/IncludeSteamWorkshopItems.cs b/Wabbajack.Lib/CompilationSteps/IncludeSteamWorkshopItems.cs index 1e891328..5b2a15c4 100644 --- a/Wabbajack.Lib/CompilationSteps/IncludeSteamWorkshopItems.cs +++ b/Wabbajack.Lib/CompilationSteps/IncludeSteamWorkshopItems.cs @@ -4,6 +4,7 @@ using System.Text.RegularExpressions; using System.Threading.Tasks; using Alphaleonis.Win32.Filesystem; using Wabbajack.Common; +using Wabbajack.Common.StoreHandlers; namespace Wabbajack.Lib.CompilationSteps { diff --git a/Wabbajack.Lib/Downloaders/SteamWorkshopDownloader.cs b/Wabbajack.Lib/Downloaders/SteamWorkshopDownloader.cs index dc871383..73ebf912 100644 --- a/Wabbajack.Lib/Downloaders/SteamWorkshopDownloader.cs +++ b/Wabbajack.Lib/Downloaders/SteamWorkshopDownloader.cs @@ -7,6 +7,7 @@ using System.Text; using System.Threading; using System.Threading.Tasks; using Wabbajack.Common; +using Wabbajack.Common.StoreHandlers; using Wabbajack.Lib.Validation; namespace Wabbajack.Lib.Downloaders @@ -50,18 +51,17 @@ namespace Wabbajack.Lib.Downloaders public override async Task Download(Archive a, string destination) { - var currentLib = ""; - SteamHandler.Instance.InstallFolders.Where(f => f.Contains(Item.Game.InstallDir)).Do(s => currentLib = s); + var currentLib = Item.Game.Universe; - var downloadFolder = Path.Combine(currentLib, "workshop", "downloads", Item.Game.AppId.ToString()); - var contentFolder = Path.Combine(currentLib, "workshop", "content", Item.Game.AppId.ToString()); + var downloadFolder = Path.Combine(currentLib, "workshop", "downloads", Item.Game.ID.ToString()); + var contentFolder = Path.Combine(currentLib, "workshop", "content", Item.Game.ID.ToString()); var p = new Process { StartInfo = new ProcessStartInfo { - FileName = Path.Combine(SteamHandler.Instance.SteamPath, "steam.exe"), + FileName = Path.Combine(StoreHandler.Instance.SteamHandler.SteamPath, "steam.exe"), CreateNoWindow = true, - Arguments = $"console +workshop_download_item {Item.Game.AppId} {Item.ItemID}" + Arguments = $"console +workshop_download_item {Item.Game.ID} {Item.ItemID}" } }; diff --git a/Wabbajack.Lib/VortexCompiler.cs b/Wabbajack.Lib/VortexCompiler.cs index ac7e9f6a..833510e2 100644 --- a/Wabbajack.Lib/VortexCompiler.cs +++ b/Wabbajack.Lib/VortexCompiler.cs @@ -9,6 +9,7 @@ using System.Threading; using Microsoft.WindowsAPICodePack.Shell; using Newtonsoft.Json; using Wabbajack.Common; +using Wabbajack.Common.StoreHandlers; using Wabbajack.Lib.CompilationSteps; using Wabbajack.Lib.NexusApi; using Wabbajack.Lib.Validation; @@ -70,11 +71,10 @@ namespace Wabbajack.Lib ActiveArchives = new List(); // there can be max one game after filtering - SteamHandler.Instance.Games.Where(g => g.Game != null && g.Game == game).Do(g => + StoreHandler.Instance.StoreGames.Where(g => g.Game == game && g.Type == StoreType.STEAM).Do(g => { _isSteamGame = true; - _steamGame = g; - SteamHandler.Instance.LoadWorkshopItems(_steamGame); + _steamGame = (SteamGame)g; _hasSteamWorkshopItems = _steamGame.WorkshopItems.Count > 0; }); } @@ -431,7 +431,7 @@ namespace Wabbajack.Lib var metaString = "[General]\n" + "repository=Steam\n" + $"gameName={GameName}\n" + - $"steamID={_steamGame.AppId}\n" + + $"steamID={_steamGame.ID}\n" + $"itemID={item.ItemID}\n" + $"itemSize={item.Size}\n"; try diff --git a/Wabbajack.Lib/VortexInstaller.cs b/Wabbajack.Lib/VortexInstaller.cs index 438bd44a..fad6aeeb 100644 --- a/Wabbajack.Lib/VortexInstaller.cs +++ b/Wabbajack.Lib/VortexInstaller.cs @@ -6,6 +6,7 @@ using System.Threading.Tasks; using System.Threading; using System.Windows; using Wabbajack.Common; +using Wabbajack.Common.StoreHandlers; using Directory = Alphaleonis.Win32.Filesystem.Directory; using DirectoryInfo = Alphaleonis.Win32.Filesystem.DirectoryInfo; using File = Alphaleonis.Win32.Filesystem.File; @@ -157,7 +158,7 @@ namespace Wabbajack.Lib { //var currentLib = ""; SteamGame currentSteamGame = null; - SteamHandler.Instance.Games.Where(g => g.Game == GameInfo.Game).Do(s => currentSteamGame = s); + StoreHandler.Instance.SteamHandler.Games.Where(g => g.Game == GameInfo.Game).Do(s => currentSteamGame = (SteamGame)s); /*SteamHandler.Instance.InstallFolders.Where(f => f.Contains(currentSteamGame.InstallDir)).Do(s => currentLib = s); var downloadFolder = Path.Combine(currentLib, "workshop", "downloads", currentSteamGame.AppId.ToString()); @@ -188,9 +189,9 @@ namespace Wabbajack.Lib { StartInfo = new ProcessStartInfo { - FileName = System.IO.Path.Combine(SteamHandler.Instance.SteamPath, "steam.exe"), + FileName = Path.Combine(StoreHandler.Instance.SteamHandler.SteamPath, "steam.exe"), CreateNoWindow = true, - Arguments = $"console +workshop_download_item {currentSteamGame.AppId} {item.ItemID}" + Arguments = $"console +workshop_download_item {currentSteamGame.ID} {currentSteamGame.ID}" } }; diff --git a/Wabbajack/View Models/Compilers/VortexCompilerVM.cs b/Wabbajack/View Models/Compilers/VortexCompilerVM.cs index 2e9136d2..848706c5 100644 --- a/Wabbajack/View Models/Compilers/VortexCompilerVM.cs +++ b/Wabbajack/View Models/Compilers/VortexCompilerVM.cs @@ -9,7 +9,9 @@ using DynamicData.Binding; using ReactiveUI; using ReactiveUI.Fody.Helpers; using Wabbajack.Common; +using Wabbajack.Common.StoreHandlers; using Wabbajack.Lib; +using GOGHandler = Wabbajack.Common.GOGHandler; namespace Wabbajack { @@ -172,14 +174,12 @@ namespace Wabbajack private void SetGameToSteamLocation() { - var steamGame = SteamHandler.Instance.Games.FirstOrDefault(g => g.Game.HasValue && g.Game == SelectedGame.Game); - GameLocation.TargetPath = steamGame?.InstallDir; + GameLocation.TargetPath = StoreHandler.Instance.GetGamePath(SelectedGame.Game, StoreType.STEAM); } private void SetGameToGogLocation() { - var gogGame = GOGHandler.Instance.Games.FirstOrDefault(g => g.Game.HasValue && g.Game == SelectedGame.Game); - GameLocation.TargetPath = gogGame?.Path; + GameLocation.TargetPath = StoreHandler.Instance.GetGamePath(SelectedGame.Game, StoreType.GOG); } public async Task Compile()