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()