first implementation of ScreamAPI support

This commit is contained in:
pointfeev 2022-03-03 06:38:17 -05:00
parent d74d54b15e
commit 26bb1269dc
31 changed files with 1252 additions and 523 deletions

View file

@ -1,62 +0,0 @@
using System;
using System.Collections.Generic;
using System.Drawing;
using System.IO;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;
using HtmlAgilityPack;
namespace CreamInstaller.Classes;
internal static class HttpClientManager
{
private static HttpClient httpClient;
internal static void Setup()
{
httpClient = new();
httpClient.DefaultRequestHeaders.Add("user-agent", $"CreamInstaller-{Environment.MachineName}_{Environment.UserDomainName}_{Environment.UserName}");
}
internal static async Task<HtmlNodeCollection> GetDocumentNodes(string url, string xpath)
{
try
{
using HttpRequestMessage request = new(HttpMethod.Get, url);
using HttpResponseMessage response = await httpClient.SendAsync(request, HttpCompletionOption.ResponseHeadersRead);
response.EnsureSuccessStatusCode();
using Stream stream = await response.Content.ReadAsStreamAsync();
using StreamReader reader = new(stream, Encoding.UTF8);
HtmlDocument document = new();
document.LoadHtml(reader.ReadToEnd());
return document.DocumentNode.SelectNodes(xpath);
}
catch { return null; }
}
internal static async Task<Image> GetImageFromUrl(string url)
{
try
{
return new Bitmap(await httpClient.GetStreamAsync(url));
}
catch { }
return null;
}
internal static async Task ParseSteamStoreDlcAppIds(int appId, List<int> dlcIds)
{
// currently this is only really needed to get DLC that release without changing game buildid (very rare)
// it also finds things which aren't really connected to the game itself, and thus not needed (usually soundtracks, collections, packs, etc.)
HtmlNodeCollection nodes = await GetDocumentNodes(
$"https://store.steampowered.com/dlc/{appId}",
"//div[@class='recommendation']/div/a");
if (nodes is not null)
foreach (HtmlNode node in nodes)
if (int.TryParse(node.Attributes?["data-ds-appid"]?.Value, out int dlcAppId) && dlcAppId > 0 && !dlcIds.Contains(dlcAppId))
dlcIds.Add(dlcAppId);
}
internal static void Dispose() => httpClient.Dispose();
}

View file

@ -1,118 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Gameloop.Vdf.Linq;
namespace CreamInstaller.Classes;
internal class ProgramSelection
{
internal bool Enabled = false;
internal bool Usable = true;
internal int SteamAppId = 0;
internal string Name = "Program";
internal string IconStaticID = null;
internal string ClientIconStaticID = null;
internal string RootDirectory;
internal List<string> SteamApiDllDirectories;
internal VProperty AppInfo = null;
internal readonly SortedList<int, (string name, string iconStaticId)> AllSteamDlc = new();
internal readonly SortedList<int, (string name, string iconStaticId)> SelectedSteamDlc = new();
internal readonly List<Tuple<int, string, SortedList<int, (string name, string iconStaticId)>>> ExtraSteamAppIdDlc = new();
internal bool AreSteamApiDllsLocked
{
get
{
foreach (string directory in SteamApiDllDirectories)
{
directory.GetApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi);
if (api.IsFilePathLocked()
|| api_o.IsFilePathLocked()
|| api64.IsFilePathLocked()
|| api64_o.IsFilePathLocked()
|| cApi.IsFilePathLocked())
return true;
}
return false;
}
}
private void Toggle(int dlcAppId, (string name, string iconStaticId) dlcApp, bool enabled)
{
if (enabled) SelectedSteamDlc[dlcAppId] = dlcApp;
else SelectedSteamDlc.Remove(dlcAppId);
}
internal void ToggleDlc(int dlcAppId, bool enabled)
{
foreach (KeyValuePair<int, (string name, string iconStaticId)> pair in AllSteamDlc)
{
int appId = pair.Key;
(string name, string iconStaticId) dlcApp = pair.Value;
if (appId == dlcAppId)
{
Toggle(appId, dlcApp, enabled);
break;
}
}
Enabled = SelectedSteamDlc.Any();
}
internal void ToggleAllDlc(bool enabled)
{
if (!enabled) SelectedSteamDlc.Clear();
else foreach (KeyValuePair<int, (string name, string iconStaticId)> pair in AllSteamDlc)
{
int appId = pair.Key;
(string name, string iconStaticId) dlcApp = pair.Value;
Toggle(appId, dlcApp, enabled);
}
Enabled = SelectedSteamDlc.Any();
}
internal ProgramSelection() => All.Add(this);
internal void Validate()
{
if (Program.IsGameBlocked(Name, RootDirectory))
{
All.Remove(this);
return;
}
if (!Directory.Exists(RootDirectory))
{
All.Remove(this);
return;
}
SteamApiDllDirectories.RemoveAll(directory => !Directory.Exists(directory));
if (!SteamApiDllDirectories.Any()) All.Remove(this);
}
internal static void ValidateAll() => AllSafe.ForEach(selection => selection.Validate());
internal static List<ProgramSelection> All = new();
internal static List<ProgramSelection> AllSafe => All.ToList();
internal static List<ProgramSelection> AllUsable => All.FindAll(s => s.Usable);
internal static List<ProgramSelection> AllUsableEnabled => AllUsable.FindAll(s => s.Enabled);
internal static ProgramSelection FromAppId(int appId) => AllUsable.Find(s => s.SteamAppId == appId);
internal static (int gameAppId, (string name, string iconStaticId) app)? GetDlcFromAppId(int appId)
{
foreach (ProgramSelection selection in AllUsable)
foreach (KeyValuePair<int, (string name, string iconStaticId)> pair in selection.AllSteamDlc)
if (pair.Key == appId) return (selection.SteamAppId, pair.Value);
return null;
}
}

View file

@ -5,21 +5,21 @@
<UseWindowsForms>True</UseWindowsForms> <UseWindowsForms>True</UseWindowsForms>
<ApplicationIcon>Resources\ini.ico</ApplicationIcon> <ApplicationIcon>Resources\ini.ico</ApplicationIcon>
<IncludeAllContentForSelfExtract>true</IncludeAllContentForSelfExtract> <IncludeAllContentForSelfExtract>true</IncludeAllContentForSelfExtract>
<Version>2.5.0.1</Version> <Version>3.0.0.0</Version>
<PackageIcon>Resources\ini.ico</PackageIcon> <PackageIcon>Resources\ini.ico</PackageIcon>
<PackageIconUrl /> <PackageIconUrl />
<Description>Automatically generates and installs CreamAPI files for Steam games on the user's computer. It can also generate and install CreamAPI for the Paradox Launcher should the user select a Paradox Interactive game.</Description> <Description />
<PackageLicenseFile>LICENSE</PackageLicenseFile> <PackageLicenseFile>LICENSE</PackageLicenseFile>
<Copyright>2021, pointfeev (https://github.com/pointfeev)</Copyright> <Copyright>2021, pointfeev (https://github.com/pointfeev)</Copyright>
<PackageRequireLicenseAcceptance>true</PackageRequireLicenseAcceptance> <PackageRequireLicenseAcceptance>true</PackageRequireLicenseAcceptance>
<PackageProjectUrl>https://github.com/pointfeev/CreamInstaller</PackageProjectUrl> <PackageProjectUrl>https://github.com/pointfeev/CreamInstaller</PackageProjectUrl>
<RepositoryUrl>https://github.com/pointfeev/CreamInstaller</RepositoryUrl> <RepositoryUrl>https://github.com/pointfeev/CreamInstaller</RepositoryUrl>
<RepositoryType>git</RepositoryType> <RepositoryType>git</RepositoryType>
<PackageReleaseNotes>Automatically downloads and installs CreamAPI files for programs/games.</PackageReleaseNotes> <PackageReleaseNotes />
<PackageTags>steam, dlc</PackageTags> <PackageTags>steam, dlc</PackageTags>
<AssemblyName>CreamInstaller</AssemblyName> <AssemblyName>CreamInstaller</AssemblyName>
<Company>CreamInstaller</Company> <Company>CreamInstaller</Company>
<Product>CreamAPI Generator &amp; Installer</Product> <Product>CreamAPI/ScreamAPI Installer &amp; Configuration Generator</Product>
<Authors>pointfeev</Authors> <Authors>pointfeev</Authors>
<PackageId>pointfeev.creaminstaller</PackageId> <PackageId>pointfeev.creaminstaller</PackageId>
<StartupObject>CreamInstaller.Program</StartupObject> <StartupObject>CreamInstaller.Program</StartupObject>
@ -44,6 +44,7 @@
<ItemGroup> <ItemGroup>
<PackageReference Include="Gameloop.Vdf" Version="0.6.1" /> <PackageReference Include="Gameloop.Vdf" Version="0.6.1" />
<PackageReference Include="HtmlAgilityPack" Version="1.11.42" /> <PackageReference Include="HtmlAgilityPack" Version="1.11.42" />
<PackageReference Include="Newtonsoft.Json" Version="13.0.1" />
<PackageReference Include="Onova" Version="2.6.2" /> <PackageReference Include="Onova" Version="2.6.2" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>

View file

@ -0,0 +1,73 @@
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text.Json;
using System.Threading.Tasks;
using Microsoft.Win32;
namespace CreamInstaller.Epic;
internal static class EpicLibrary
{
private static string epicAppDataPath = null;
internal static string EpicAppDataPath
{
get
{
epicAppDataPath ??= Registry.GetValue(@"HKEY_LOCAL_MACHINE\SOFTWARE\Epic Games\EpicGamesLauncher", "AppDataPath", null) as string;
epicAppDataPath ??= Registry.GetValue(@"HKEY_LOCAL_MACHINE\SOFTWARE\WOW6432Node\Epic Games\EpicGamesLauncher", "AppDataPath", null) as string;
return epicAppDataPath;
}
}
internal static async Task<List<Manifest>> GetGames() => await Task.Run(() =>
{
List<Manifest> games = new();
if (!Directory.Exists(EpicAppDataPath)) return games;
string manifests = EpicAppDataPath + @"\Manifests";
if (!Directory.Exists(manifests)) return games;
string[] files = Directory.GetFiles(manifests);
foreach (string file in files)
{
if (Program.Canceled) return games;
if (Path.GetExtension(file) == ".item")
{
string json = File.ReadAllText(file);
try
{
Manifest manifest = JsonSerializer.Deserialize<Manifest>(json);
if (manifest is not null && manifest.CatalogItemId == manifest.MainGameCatalogItemId)
games.Add(manifest);
}
catch { };
}
}
return games;
});
internal static async Task<List<string>> GetDllDirectoriesFromGameDirectory(string gameDirectory) => await Task.Run(async () =>
{
List<string> dllDirectories = new();
if (Program.Canceled || !Directory.Exists(gameDirectory)) return null;
gameDirectory.GetScreamApiComponents(out string sdk, out string sdk_o, out string sdk64, out string sdk64_o, out string sApi);
if (File.Exists(sdk)
|| File.Exists(sdk_o)
|| File.Exists(sdk64)
|| File.Exists(sdk64_o)
|| File.Exists(sApi))
dllDirectories.Add(gameDirectory);
string[] directories = Directory.GetDirectories(gameDirectory);
foreach (string _directory in directories)
{
if (Program.Canceled) return null;
try
{
List<string> moreDllDirectories = await GetDllDirectoriesFromGameDirectory(_directory);
if (moreDllDirectories is not null) dllDirectories.AddRange(moreDllDirectories);
}
catch { }
}
return !dllDirectories.Any() ? null : dllDirectories;
});
}

View file

@ -0,0 +1,47 @@
using System.Collections.Generic;
using System.Net.Http;
using System.Net.Http.Headers;
using System.Threading.Tasks;
using System.Web;
using CreamInstaller.Epic.GraphQL;
using CreamInstaller.Utility;
using Newtonsoft.Json;
namespace CreamInstaller.Epic;
internal static class EpicStore
{
internal static async Task<List<(string id, string name)>> ParseDlcAppIds(string categoryNamespace)
{
List<(string id, string name)> dlcIds = new();
Response response = await QueryEpicGraphQL(categoryNamespace);
if (response is null)
return dlcIds;
List<Element> elements = new(response.Data.Catalog.CatalogOffers.Elements);
elements.AddRange(response.Data.Catalog.SearchStore.Elements);
foreach (Element element in elements)
{
(string id, string name) app = (element.Items[0].Id, element.Title);
if (!dlcIds.Contains(app))
dlcIds.Add(app);
}
return dlcIds;
}
internal static async Task<Response> QueryEpicGraphQL(string categoryNamespace)
{
string encoded = HttpUtility.UrlEncode(categoryNamespace);
Request request = new(encoded);
string payload = JsonConvert.SerializeObject(request);
HttpContent content = new StringContent(payload);
content.Headers.ContentType = new MediaTypeHeaderValue("application/json");
HttpClient client = HttpClientManager.HttpClient;
if (client is null) return null;
HttpResponseMessage httpResponse = await client.PostAsync("https://graphql.epicgames.com/graphql", content);
httpResponse.EnsureSuccessStatusCode();
string response = await httpResponse.Content.ReadAsStringAsync();
return JsonConvert.DeserializeObject<Response>(response);
}
}

View file

@ -0,0 +1,94 @@

using Newtonsoft.Json;
namespace CreamInstaller.Epic.GraphQL;
internal class Request
{
#pragma warning disable IDE0051 // Remove unused private members
#pragma warning disable CA1822 // Mark members as static
#pragma warning disable IDE0052 // Remove unread private members
#pragma warning disable IDE1006 // Naming Styles
[JsonProperty(PropertyName = "query")]
private string _gqlQuery => @"query searchOffers($namespace: String!) {
Catalog {
catalogOffers(
namespace: $namespace
params: {
count: 1000,
}
) {
elements {
id
title
offerType
items {
id
}
keyImages {
type
url
}
catalogNs {
mappings(pageType: ""productHome"") {
pageSlug
}
}
}
}
searchStore(category: ""games/edition/base"", namespace: $namespace) {
elements {
id
title
offerType
items {
id
}
keyImages {
type
url
}
catalogNs {
mappings(pageType: ""productHome"") {
pageSlug
}
}
}
}
}
}";
[JsonProperty(PropertyName = "variables")]
private Variables _variables { get; set; }
internal Request(string _namespace) => _variables = new Variables(_namespace);
private class Headers
{
[JsonProperty(PropertyName = "Content-Type")]
private string _contentType => "application/graphql";
}
private class Variables
{
[JsonProperty(PropertyName = "category")]
private string _category => "games/edition/base|bundles/games|editors|software/edition/base";
[JsonProperty(PropertyName = "count")]
private int _count => 1000;
[JsonProperty(PropertyName = "keywords")]
private string _keywords => "";
[JsonProperty(PropertyName = "namespace")]
private string _namespace { get; set; }
internal Variables(string _namespace) => this._namespace = _namespace;
}
#pragma warning restore IDE0051 // Remove unused private members
#pragma warning restore CA1822 // Mark members as static
#pragma warning restore IDE0052 // Remove unread private members
#pragma warning restore IDE1006 // Naming Styles
}

View file

@ -0,0 +1,95 @@
using System;
using Newtonsoft.Json;
namespace CreamInstaller.Epic.GraphQL;
public class Response
{
[JsonProperty(PropertyName = "data")]
public Data Data { get; protected set; }
}
public class Data
{
[JsonProperty(PropertyName = "Catalog")]
public Catalog Catalog { get; protected set; }
}
public class Catalog
{
[JsonProperty(PropertyName = "catalogOffers")]
public SearchStore CatalogOffers { get; protected set; }
[JsonProperty(PropertyName = "searchStore")]
public SearchStore SearchStore { get; protected set; }
}
public class CatalogOffers
{
[JsonProperty(PropertyName = "namespace")]
public string Namespace { get; protected set; }
[JsonProperty(PropertyName = "params")]
public Parameters Parameters { get; protected set; }
}
public class Parameters
{
[JsonProperty(PropertyName = "count")]
public int Count { get; protected set; }
}
public class SearchStore
{
[JsonProperty(PropertyName = "elements")]
public Element[] Elements { get; protected set; }
}
public class Element
{
[JsonProperty(PropertyName = "id")]
public string Id { get; protected set; }
[JsonProperty(PropertyName = "title")]
public string Title { get; protected set; }
[JsonProperty(PropertyName = "offerType")]
public string OfferType { get; protected set; }
[JsonProperty(PropertyName = "items")]
public Item[] Items { get; protected set; }
[JsonProperty(PropertyName = "keyImages")]
public KeyImage[] KeyImages { get; protected set; }
[JsonProperty(PropertyName = "catalogNs")]
public CatalogNs CatalogNs { get; protected set; }
}
public class Item
{
[JsonProperty(PropertyName = "id")]
public string Id { get; protected set; }
}
public class KeyImage
{
[JsonProperty(PropertyName = "type")]
public string Type { get; protected set; }
[JsonProperty(PropertyName = "url")]
public string Url { get; protected set; }
}
public class CatalogNs
{
[JsonProperty(PropertyName = "mappings")]
public Mapping[] Mappings { get; protected set; }
}
public class Mapping
{
[JsonProperty(PropertyName = "pageSlug")]
public string PageSlug { get; protected set; }
}

View file

@ -0,0 +1,53 @@
using System.Collections.Generic;
namespace CreamInstaller.Epic;
public class Manifest
{
#pragma warning disable IDE1006 // Naming Styles
public int FormatVersion { get; set; }
public bool bIsIncompleteInstall { get; set; }
public string LaunchCommand { get; set; }
public string LaunchExecutable { get; set; }
public string ManifestLocation { get; set; }
public bool bIsApplication { get; set; }
public bool bIsExecutable { get; set; }
public bool bIsManaged { get; set; }
public bool bNeedsValidation { get; set; }
public bool bRequiresAuth { get; set; }
public bool bAllowMultipleInstances { get; set; }
public bool bCanRunOffline { get; set; }
public bool bAllowUriCmdArgs { get; set; }
public List<string> BaseURLs { get; set; }
public string BuildLabel { get; set; }
public List<string> AppCategories { get; set; }
public List<object> ChunkDbs { get; set; }
public List<object> CompatibleApps { get; set; }
public string DisplayName { get; set; }
public string InstallationGuid { get; set; }
public string InstallLocation { get; set; }
public string InstallSessionId { get; set; }
public List<object> InstallTags { get; set; }
public List<object> InstallComponents { get; set; }
public string HostInstallationGuid { get; set; }
public List<string> PrereqIds { get; set; }
public string StagingLocation { get; set; }
public string TechnicalType { get; set; }
public string VaultThumbnailUrl { get; set; }
public string VaultTitleText { get; set; }
public long InstallSize { get; set; }
public string MainWindowProcessName { get; set; }
public List<object> ProcessNames { get; set; }
public List<object> BackgroundProcessNames { get; set; }
public string MandatoryAppFolderName { get; set; }
public string OwnershipToken { get; set; }
public string CatalogNamespace { get; set; }
public string CatalogItemId { get; set; }
public string AppName { get; set; }
public string AppVersionString { get; set; }
public string MainGameCatalogNamespace { get; set; }
public string MainGameCatalogItemId { get; set; }
public string MainGameAppName { get; set; }
public List<object> AllowedUriEnvVars { get; set; }
#pragma warning restore IDE1006 // Naming Styles
}

View file

@ -34,7 +34,7 @@ internal class CustomTreeView : TreeView
Font subFont = new(font.FontFamily, font.SizeInPoints, FontStyle.Regular, font.Unit, font.GdiCharSet, font.GdiVerticalFont); Font subFont = new(font.FontFamily, font.SizeInPoints, FontStyle.Regular, font.Unit, font.GdiCharSet, font.GdiVerticalFont);
string subText = node.Name; string subText = node.Name;
if (subText is null || !int.TryParse(subText, out int subInt) || subInt <= 0) if (subText == "ParadoxLauncher")
return; return;
Size subSize = TextRenderer.MeasureText(graphics, subText, subFont); Size subSize = TextRenderer.MeasureText(graphics, subText, subFont);

View file

@ -1,4 +1,5 @@
using System.Drawing; using System;
using System.Drawing;
using System.Windows.Forms; using System.Windows.Forms;
using CreamInstaller.Forms.Components; using CreamInstaller.Forms.Components;
@ -9,12 +10,11 @@ internal partial class DialogForm : CustomForm
{ {
internal DialogForm(IWin32Window owner) : base(owner) => InitializeComponent(); internal DialogForm(IWin32Window owner) : base(owner) => InitializeComponent();
internal DialogResult Show(string formName, Icon descriptionIcon, string descriptionText, string acceptButtonText, string cancelButtonText = null, Icon customFormIcon = null) internal DialogResult Show(Icon descriptionIcon, string descriptionText, string acceptButtonText, string cancelButtonText = null, Icon customFormIcon = null)
{ {
if (customFormIcon is not null) if (customFormIcon is not null)
Icon = customFormIcon; Icon = customFormIcon;
icon.Image = descriptionIcon.ToBitmap(); icon.Image = descriptionIcon.ToBitmap();
Text = formName;
descriptionLabel.Text = descriptionText; descriptionLabel.Text = descriptionText;
acceptButton.Text = acceptButtonText; acceptButton.Text = acceptButtonText;
if (cancelButtonText is null) if (cancelButtonText is null)
@ -23,6 +23,13 @@ internal partial class DialogForm : CustomForm
cancelButton.Visible = false; cancelButton.Visible = false;
} }
else cancelButton.Text = cancelButtonText; else cancelButton.Text = cancelButtonText;
OnResize(null, null);
Resize += OnResize;
return ShowDialog(); return ShowDialog();
} }
internal void OnResize(object s, EventArgs e) =>
Text = TextRenderer.MeasureText(Program.ApplicationName, Font).Width > Size.Width - 100
? Program.ApplicationNameShort
: Program.ApplicationName;
} }

View file

@ -7,9 +7,9 @@ using System.Text;
using System.Threading.Tasks; using System.Threading.Tasks;
using System.Windows.Forms; using System.Windows.Forms;
using CreamInstaller.Classes;
using CreamInstaller.Forms.Components; using CreamInstaller.Forms.Components;
using CreamInstaller.Resources; using CreamInstaller.Resources;
using CreamInstaller.Utility;
namespace CreamInstaller; namespace CreamInstaller;
@ -53,9 +53,8 @@ internal partial class InstallForm : CustomForm
} }
} }
internal static void WriteConfiguration(StreamWriter writer, int steamAppId, string name, SortedList<int, (string name, string iconStaticId)> steamDlcApps, InstallForm installForm = null) internal static void WriteCreamConfiguration(StreamWriter writer, string steamAppId, string name, SortedList<string, (string name, string iconStaticId)> steamDlcApps, InstallForm installForm = null)
{ {
writer.WriteLine();
writer.WriteLine($"; {name}"); writer.WriteLine($"; {name}");
writer.WriteLine("[steam]"); writer.WriteLine("[steam]");
writer.WriteLine($"appid = {steamAppId}"); writer.WriteLine($"appid = {steamAppId}");
@ -63,19 +62,19 @@ internal partial class InstallForm : CustomForm
writer.WriteLine("[dlc]"); writer.WriteLine("[dlc]");
if (installForm is not null) if (installForm is not null)
installForm.UpdateUser($"Added game to cream_api.ini with appid {steamAppId} ({name})", InstallationLog.Resource, info: false); installForm.UpdateUser($"Added game to cream_api.ini with appid {steamAppId} ({name})", InstallationLog.Resource, info: false);
foreach (KeyValuePair<int, (string name, string iconStaticId)> pair in steamDlcApps) foreach (KeyValuePair<string, (string name, string iconStaticId)> pair in steamDlcApps)
{ {
int appId = pair.Key; string appId = pair.Key;
(string name, string iconStaticId) dlcApp = pair.Value; (string dlcName, _) = pair.Value;
writer.WriteLine($"{appId} = {dlcApp.name}"); writer.WriteLine($"{appId} = {dlcName}");
if (installForm is not null) if (installForm is not null)
installForm.UpdateUser($"Added DLC to cream_api.ini with appid {appId} ({dlcApp.name})", InstallationLog.Resource, info: false); installForm.UpdateUser($"Added DLC to cream_api.ini with appid {appId} ({dlcName})", InstallationLog.Resource, info: false);
} }
} }
internal static async Task UninstallCreamAPI(string directory, InstallForm installForm = null) => await Task.Run(() => internal static async Task UninstallCreamAPI(string directory, InstallForm installForm = null) => await Task.Run(() =>
{ {
directory.GetApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi); directory.GetCreamApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi);
if (File.Exists(api_o)) if (File.Exists(api_o))
{ {
if (File.Exists(api)) if (File.Exists(api))
@ -110,7 +109,7 @@ internal partial class InstallForm : CustomForm
internal static async Task InstallCreamAPI(string directory, ProgramSelection selection, InstallForm installForm = null) => await Task.Run(() => internal static async Task InstallCreamAPI(string directory, ProgramSelection selection, InstallForm installForm = null) => await Task.Run(() =>
{ {
directory.GetApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi); directory.GetCreamApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi);
if (File.Exists(api) && !File.Exists(api_o)) if (File.Exists(api) && !File.Exists(api_o))
{ {
File.Move(api, api_o); File.Move(api, api_o);
@ -136,14 +135,125 @@ internal partial class InstallForm : CustomForm
installForm.UpdateUser($"Wrote resource to file: {Path.GetFileName(api64)}", InstallationLog.Resource, info: false); installForm.UpdateUser($"Wrote resource to file: {Path.GetFileName(api64)}", InstallationLog.Resource, info: false);
} }
if (installForm is not null) if (installForm is not null)
installForm.UpdateUser("Generating CreamAPI for " + selection.Name + $" in directory \"{directory}\" . . . ", InstallationLog.Operation); installForm.UpdateUser("Generating CreamAPI configuration for " + selection.Name + $" in directory \"{directory}\" . . . ", InstallationLog.Operation);
File.Create(cApi).Close(); File.Create(cApi).Close();
StreamWriter writer = new(cApi, true, Encoding.UTF8); StreamWriter writer = new(cApi, true, Encoding.UTF8);
writer.WriteLine("; " + Application.CompanyName + " v" + Application.ProductVersion); if (selection.Id != "ParadoxLauncher")
if (selection.SteamAppId > 0) WriteCreamConfiguration(writer, selection.Id, selection.Name, selection.SelectedDlc, installForm);
WriteConfiguration(writer, selection.SteamAppId, selection.Name, selection.SelectedSteamDlc, installForm); foreach (Tuple<string, string, SortedList<string, (string name, string iconStaticId)>> extraAppDlc in selection.ExtraDlc)
foreach (Tuple<int, string, SortedList<int, (string name, string iconStaticId)>> extraAppDlc in selection.ExtraSteamAppIdDlc) WriteCreamConfiguration(writer, extraAppDlc.Item1, extraAppDlc.Item2, extraAppDlc.Item3, installForm);
WriteConfiguration(writer, extraAppDlc.Item1, extraAppDlc.Item2, extraAppDlc.Item3, installForm); writer.Flush();
writer.Close();
});
internal static void WriteScreamConfiguration(StreamWriter writer, SortedList<string, (string name, string iconStaticId)> steamDlcApps, InstallForm installForm = null)
{
writer.WriteLine("{");
writer.WriteLine(" \"version\": 2,");
writer.WriteLine(" \"logging\": false,");
writer.WriteLine(" \"eos_logging\": false,");
writer.WriteLine(" \"block_metrics\": false,");
writer.WriteLine(" \"catalog_items\": {");
writer.WriteLine(" \"unlock_all\": false,");
writer.WriteLine(" \"override\": [");
KeyValuePair<string, (string name, string iconStaticId)> last = steamDlcApps.Last();
foreach (KeyValuePair<string, (string name, string iconStaticId)> pair in steamDlcApps)
{
string id = pair.Key;
(string name, _) = pair.Value;
writer.WriteLine($" \"{id}\"{(pair.Equals(last) ? "" : ",")}");
if (installForm is not null)
installForm.UpdateUser($"Added DLC to ScreamAPI.json with id {id} ({name})", InstallationLog.Resource, info: false);
}
writer.WriteLine(" ]");
writer.WriteLine(" },");
writer.WriteLine(" \"entitlements\": {");
writer.WriteLine(" \"unlock_all\": false,");
writer.WriteLine(" \"auto_inject\": false,");
writer.WriteLine(" \"inject\": [");
foreach (KeyValuePair<string, (string name, string iconStaticId)> pair in steamDlcApps)
{
string id = pair.Key;
(string name, _) = pair.Value;
writer.WriteLine($" \"{id}\"{(pair.Equals(last) ? "" : ",")}");
if (installForm is not null)
installForm.UpdateUser($"Added DLC to ScreamAPI.json with id {id} ({name})", InstallationLog.Resource, info: false);
}
writer.WriteLine(" ]");
writer.WriteLine(" }");
writer.WriteLine("}");
}
internal static async Task UninstallScreamAPI(string directory, InstallForm installForm = null) => await Task.Run(() =>
{
directory.GetScreamApiComponents(out string sdk, out string sdk_o, out string sdk64, out string sdk64_o, out string sApi);
if (File.Exists(sdk_o))
{
if (File.Exists(sdk))
{
File.Delete(sdk);
if (installForm is not null)
installForm.UpdateUser($"Deleted file: {Path.GetFileName(sdk)}", InstallationLog.Resource, info: false);
}
File.Move(sdk_o, sdk);
if (installForm is not null)
installForm.UpdateUser($"Renamed file: {Path.GetFileName(sdk_o)} -> {Path.GetFileName(sdk)}", InstallationLog.Resource, info: false);
}
if (File.Exists(sdk64_o))
{
if (File.Exists(sdk64))
{
File.Delete(sdk64);
if (installForm is not null)
installForm.UpdateUser($"Deleted file: {Path.GetFileName(sdk64)}", InstallationLog.Resource, info: false);
}
File.Move(sdk64_o, sdk64);
if (installForm is not null)
installForm.UpdateUser($"Renamed file: {Path.GetFileName(sdk64_o)} -> {Path.GetFileName(sdk64)}", InstallationLog.Resource, info: false);
}
if (File.Exists(sApi))
{
File.Delete(sApi);
if (installForm is not null)
installForm.UpdateUser($"Deleted file: {Path.GetFileName(sApi)}", InstallationLog.Resource, info: false);
}
});
internal static async Task InstallScreamAPI(string directory, ProgramSelection selection, InstallForm installForm = null) => await Task.Run(() =>
{
directory.GetScreamApiComponents(out string sdk, out string sdk_o, out string sdk64, out string sdk64_o, out string sApi);
if (File.Exists(sdk) && !File.Exists(sdk_o))
{
File.Move(sdk, sdk_o);
if (installForm is not null)
installForm.UpdateUser($"Renamed file: {Path.GetFileName(sdk)} -> {Path.GetFileName(sdk_o)}", InstallationLog.Resource, info: false);
}
if (File.Exists(sdk_o))
{
Properties.Resources.SDK.Write(sdk);
if (installForm is not null)
installForm.UpdateUser($"Wrote resource to file: {Path.GetFileName(sdk)}", InstallationLog.Resource, info: false);
}
if (File.Exists(sdk64) && !File.Exists(sdk64_o))
{
File.Move(sdk64, sdk64_o);
if (installForm is not null)
installForm.UpdateUser($"Renamed file: {Path.GetFileName(sdk64)} -> {Path.GetFileName(sdk64_o)}", InstallationLog.Resource, info: false);
}
if (File.Exists(sdk64_o))
{
Properties.Resources.SDK64.Write(sdk64);
if (installForm is not null)
installForm.UpdateUser($"Wrote resource to file: {Path.GetFileName(sdk64)}", InstallationLog.Resource, info: false);
}
if (installForm is not null)
installForm.UpdateUser("Generating ScreamAPI configuration for " + selection.Name + $" in directory \"{directory}\" . . . ", InstallationLog.Operation);
File.Create(sApi).Close();
StreamWriter writer = new(sApi, true, Encoding.UTF8);
if (selection.Id != "ParadoxLauncher")
WriteScreamConfiguration(writer, selection.SelectedDlc, installForm);
foreach (Tuple<string, string, SortedList<string, (string name, string iconStaticId)>> extraAppDlc in selection.ExtraDlc)
WriteScreamConfiguration(writer, extraAppDlc.Item3, installForm);
writer.Flush(); writer.Flush();
writer.Close(); writer.Close();
}); });
@ -151,16 +261,27 @@ internal partial class InstallForm : CustomForm
private async Task OperateFor(ProgramSelection selection) private async Task OperateFor(ProgramSelection selection)
{ {
UpdateProgress(0); UpdateProgress(0);
int count = selection.SteamApiDllDirectories.Count; int count = selection.DllDirectories.Count;
int cur = 0; int cur = 0;
foreach (string directory in selection.SteamApiDllDirectories) foreach (string directory in selection.DllDirectories)
{ {
UpdateUser($"{(Uninstalling ? "Uninstalling" : "Installing")} CreamAPI for " + selection.Name + $" in directory \"{directory}\" . . . ", InstallationLog.Operation); UpdateUser($"{(Uninstalling ? "Uninstalling" : "Installing")} {(selection.IsSteam ? "CreamAPI" : "ScreamAPI")}" +
$" {(Uninstalling ? "from" : "for")} " + selection.Name + $" in directory \"{directory}\" . . . ", InstallationLog.Operation);
if (!Program.IsProgramRunningDialog(this, selection)) throw new OperationCanceledException(); if (!Program.IsProgramRunningDialog(this, selection)) throw new OperationCanceledException();
if (selection.IsSteam)
{
if (Uninstalling) if (Uninstalling)
await UninstallCreamAPI(directory, this); await UninstallCreamAPI(directory, this);
else else
await InstallCreamAPI(directory, selection, this); await InstallCreamAPI(directory, selection, this);
}
else
{
if (Uninstalling)
await UninstallScreamAPI(directory, this);
else
await InstallScreamAPI(directory, selection, this);
}
UpdateProgress(++cur / count * 100); UpdateProgress(++cur / count * 100);
} }
UpdateProgress(100); UpdateProgress(100);
@ -208,11 +329,11 @@ internal partial class InstallForm : CustomForm
try try
{ {
await Operate(); await Operate();
UpdateUser($"CreamAPI successfully {(Uninstalling ? "uninstalled" : "installed and generated")} for " + ProgramCount + " program(s).", InstallationLog.Success); UpdateUser($"CreamAPI/ScreamAPI successfully {(Uninstalling ? "uninstalled" : "installed and generated")} for " + ProgramCount + " program(s).", InstallationLog.Success);
} }
catch (Exception exception) catch (Exception exception)
{ {
UpdateUser($"CreamAPI {(Uninstalling ? "uninstallation" : "installation and/or generation")} failed: " + exception.ToString(), InstallationLog.Error); UpdateUser($"CreamAPI/ScreamAPI {(Uninstalling ? "uninstallation" : "installation and/or generation")} failed: " + exception.ToString(), InstallationLog.Error);
retryButton.Enabled = true; retryButton.Enabled = true;
} }
userProgressBar.Value = userProgressBar.Maximum; userProgressBar.Value = userProgressBar.Maximum;

View file

@ -7,8 +7,8 @@ using System.Threading.Tasks;
using System.Web; using System.Web;
using System.Windows.Forms; using System.Windows.Forms;
using CreamInstaller.Classes;
using CreamInstaller.Forms.Components; using CreamInstaller.Forms.Components;
using CreamInstaller.Utility;
using HtmlAgilityPack; using HtmlAgilityPack;
@ -23,7 +23,7 @@ internal partial class MainForm : CustomForm
internal MainForm() : base() internal MainForm() : base()
{ {
InitializeComponent(); InitializeComponent();
Text = Program.ApplicationName; Text = Program.ApplicationNameShort;
} }
private static CancellationTokenSource cancellationTokenSource; private static CancellationTokenSource cancellationTokenSource;
@ -127,7 +127,7 @@ internal partial class MainForm : CustomForm
{ {
string FileName = Path.GetFileName(Program.CurrentProcessFilePath); string FileName = Path.GetFileName(Program.CurrentProcessFilePath);
if (FileName != "CreamInstaller.exe") if (FileName != "CreamInstaller.exe")
if (new DialogForm(this).Show(Program.ApplicationName, SystemIcons.Warning, if (new DialogForm(this).Show(SystemIcons.Warning,
"WARNING: CreamInstaller.exe was renamed!" + "WARNING: CreamInstaller.exe was renamed!" +
"\n\nThis will cause unwanted behavior when updating the program!", "\n\nThis will cause unwanted behavior when updating the program!",
"Ignore", "Abort") == DialogResult.Cancel) "Ignore", "Abort") == DialogResult.Cancel)

View file

@ -113,7 +113,7 @@ namespace CreamInstaller
this.noneFoundLabel.Name = "noneFoundLabel"; this.noneFoundLabel.Name = "noneFoundLabel";
this.noneFoundLabel.Size = new System.Drawing.Size(554, 218); this.noneFoundLabel.Size = new System.Drawing.Size(554, 218);
this.noneFoundLabel.TabIndex = 1002; this.noneFoundLabel.TabIndex = 1002;
this.noneFoundLabel.Text = "No CreamAPI-applicable programs were found on your computer!"; this.noneFoundLabel.Text = "No CreamAPI-applicable or ScreamAPI-applicable programs were found on your computer!";
this.noneFoundLabel.TextAlign = System.Drawing.ContentAlignment.MiddleCenter; this.noneFoundLabel.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
this.noneFoundLabel.Visible = false; this.noneFoundLabel.Visible = false;
// //

View file

@ -9,9 +9,12 @@ using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using System.Windows.Forms; using System.Windows.Forms;
using CreamInstaller.Classes; using CreamInstaller.Epic;
using CreamInstaller.Forms.Components; using CreamInstaller.Forms.Components;
using CreamInstaller.Paradox;
using CreamInstaller.Resources; using CreamInstaller.Resources;
using CreamInstaller.Steam;
using CreamInstaller.Utility;
using Gameloop.Vdf.Linq; using Gameloop.Vdf.Linq;
@ -79,35 +82,45 @@ internal partial class SelectForm : CustomForm
} }
internal readonly List<Task> RunningTasks = new(); internal readonly List<Task> RunningTasks = new();
private async Task GetCreamApiApplicablePrograms(IProgress<int> progress) private async Task GetApplicablePrograms(IProgress<int> progress)
{ {
if (Program.Canceled) return; if (Program.Canceled) return;
List<Tuple<int, string, string, int, string>> applicablePrograms = new();
if (Directory.Exists(SteamLibrary.ParadoxLauncherInstallPath))
applicablePrograms.Add(new(0, "Paradox Launcher", "", 0, SteamLibrary.ParadoxLauncherInstallPath));
List<string> gameLibraryDirectories = await SteamLibrary.GetLibraryDirectories();
foreach (string libraryDirectory in gameLibraryDirectories)
{
List<Tuple<int, string, string, int, string>> games = await SteamLibrary.GetGamesFromLibraryDirectory(libraryDirectory);
if (games is not null)
foreach (Tuple<int, string, string, int, string> game in games)
if (!applicablePrograms.Any(_game => _game.Item1 == game.Item1))
applicablePrograms.Add(game);
}
int CompleteTasks = 0; int CompleteTasks = 0;
RunningTasks.Clear(); // contains all running tasks including games AND their dlc RunningTasks.Clear(); // contains all running tasks including games AND their dlc
RemainingGames.Clear(); // for display purposes only, otherwise ignorable RemainingGames.Clear(); // for display purposes only, otherwise ignorable
RemainingDLCs.Clear(); // for display purposes only, otherwise ignorable RemainingDLCs.Clear(); // for display purposes only, otherwise ignorable
List<Task> appTasks = new(); List<Task> appTasks = new();
foreach (Tuple<int, string, string, int, string> program in applicablePrograms) if (Directory.Exists(ParadoxLauncher.InstallPath))
{ {
int appId = program.Item1; ProgramSelection selection = ProgramSelection.FromId("ParadoxLauncher");
selection ??= new();
if (allCheckBox.Checked) selection.Enabled = true;
selection.Usable = true;
selection.Id = "ParadoxLauncher";
selection.Name = "Paradox Launcher";
selection.RootDirectory = ParadoxLauncher.InstallPath;
List<string> steamDllDirectories = await SteamLibrary.GetDllDirectoriesFromGameDirectory(selection.RootDirectory);
selection.DllDirectories = steamDllDirectories ?? await EpicLibrary.GetDllDirectoriesFromGameDirectory(selection.RootDirectory);
selection.IsSteam = steamDllDirectories is not null;
TreeNode programNode = TreeNodes.Find(s => s.Name == selection.Id) ?? new();
programNode.Name = selection.Id;
programNode.Text = selection.Name;
programNode.Checked = selection.Enabled;
programNode.Remove();
selectionTreeView.Nodes.Add(programNode);
}
if (Directory.Exists(SteamLibrary.InstallPath))
{
List<Tuple<string, string, string, int, string>> steamGames = await SteamLibrary.GetGames();
foreach (Tuple<string, string, string, int, string> program in steamGames)
{
string appId = program.Item1;
string name = program.Item2; string name = program.Item2;
string branch = program.Item3; string branch = program.Item3;
int buildId = program.Item4; int buildId = program.Item4;
string directory = program.Item5; string directory = program.Item5;
ProgramSelection selection = ProgramSelection.FromAppId(appId); ProgramSelection selection = ProgramSelection.FromId(appId);
if (Program.Canceled) return; if (Program.Canceled) return;
if (Program.IsGameBlocked(name, directory)) continue; if (Program.IsGameBlocked(name, directory)) continue;
AddToRemainingGames(name); AddToRemainingGames(name);
@ -120,24 +133,23 @@ internal partial class SelectForm : CustomForm
RemoveFromRemainingGames(name); RemoveFromRemainingGames(name);
return; return;
} }
VProperty appInfo = null; VProperty appInfo = appInfo = await SteamCMD.GetAppInfo(appId, branch, buildId);
if (appId > 0) appInfo = await SteamCMD.GetAppInfo(appId, branch, buildId); if (appInfo is null)
if (appId > 0 && appInfo is null)
{ {
RemoveFromRemainingGames(name); RemoveFromRemainingGames(name);
return; return;
} }
if (Program.Canceled) return; if (Program.Canceled) return;
ConcurrentDictionary<int, (string name, string iconStaticId)> dlc = new(); ConcurrentDictionary<string, (string name, string iconStaticId)> dlc = new();
List<Task> dlcTasks = new(); List<Task> dlcTasks = new();
List<int> dlcIds = await SteamCMD.ParseDlcAppIds(appInfo); List<string> dlcIds = await SteamCMD.ParseDlcAppIds(appInfo);
await HttpClientManager.ParseSteamStoreDlcAppIds(appId, dlcIds); await SteamStore.ParseDlcAppIds(appId, dlcIds);
if (dlcIds.Count > 0) if (dlcIds.Count > 0)
{ {
foreach (int dlcAppId in dlcIds) foreach (string dlcAppId in dlcIds)
{ {
if (Program.Canceled) return; if (Program.Canceled) return;
AddToRemainingDLCs(dlcAppId.ToString()); AddToRemainingDLCs(dlcAppId);
Task task = Task.Run(async () => Task task = Task.Run(async () =>
{ {
if (Program.Canceled) return; if (Program.Canceled) return;
@ -154,7 +166,7 @@ internal partial class SelectForm : CustomForm
if (Program.Canceled) return; if (Program.Canceled) return;
if (!string.IsNullOrWhiteSpace(dlcName)) if (!string.IsNullOrWhiteSpace(dlcName))
dlc[dlcAppId] = (dlcName, dlcIconStaticId); dlc[dlcAppId] = (dlcName, dlcIconStaticId);
RemoveFromRemainingDLCs(dlcAppId.ToString()); RemoveFromRemainingDLCs(dlcAppId);
progress.Report(++CompleteTasks); progress.Report(++CompleteTasks);
}); });
dlcTasks.Add(task); dlcTasks.Add(task);
@ -163,7 +175,7 @@ internal partial class SelectForm : CustomForm
Thread.Sleep(10); // to reduce control & window freezing Thread.Sleep(10); // to reduce control & window freezing
} }
} }
else if (appId > 0) else
{ {
RemoveFromRemainingGames(name); RemoveFromRemainingGames(name);
return; return;
@ -176,47 +188,41 @@ internal partial class SelectForm : CustomForm
} }
selection ??= new(); selection ??= new();
if (allCheckBox.Checked) selection.Enabled = true;
selection.Usable = true; selection.Usable = true;
selection.SteamAppId = appId; selection.Id = appId;
selection.Name = name; selection.Name = name;
selection.RootDirectory = directory; selection.RootDirectory = directory;
selection.SteamApiDllDirectories = dllDirectories; selection.DllDirectories = dllDirectories;
selection.IsSteam = true;
selection.AppInfo = appInfo; selection.AppInfo = appInfo;
selection.IconStaticID = appInfo?.Value?.GetChild("common")?.GetChild("icon")?.ToString(); selection.IconStaticID = appInfo?.Value?.GetChild("common")?.GetChild("icon")?.ToString();
selection.ClientIconStaticID = appInfo?.Value?.GetChild("common")?.GetChild("clienticon")?.ToString(); selection.ClientIconStaticID = appInfo?.Value?.GetChild("common")?.GetChild("clienticon")?.ToString();
if (allCheckBox.Checked) selection.Enabled = true;
if (Program.Canceled) return; if (Program.Canceled) return;
Program.Invoke(selectionTreeView, delegate Program.Invoke(selectionTreeView, delegate
{ {
if (Program.Canceled) return; if (Program.Canceled) return;
TreeNode programNode = TreeNodes.Find(s => s.Name == "" + appId) ?? new(); TreeNode programNode = TreeNodes.Find(s => s.Name == appId) ?? new();
programNode.Name = "" + appId; programNode.Name = appId;
programNode.Text = /*(appId > 0 ? $"[{appId}] " : "") +*/ name; programNode.Text = name;
programNode.Checked = selection.Enabled; programNode.Checked = selection.Enabled;
programNode.Remove(); programNode.Remove();
selectionTreeView.Nodes.Add(programNode); selectionTreeView.Nodes.Add(programNode);
if (appId == 0) // paradox launcher foreach (KeyValuePair<string, (string name, string iconStaticId)> pair in dlc)
{
// maybe add game and/or dlc choice here?
}
else
{
foreach (KeyValuePair<int, (string name, string iconStaticId)> pair in dlc)
{ {
if (Program.Canceled || programNode is null) return; if (Program.Canceled || programNode is null) return;
int appId = pair.Key; string appId = pair.Key;
(string name, string iconStaticId) dlcApp = pair.Value; (string name, string iconStaticId) dlcApp = pair.Value;
selection.AllSteamDlc[appId] = dlcApp; selection.AllDlc[appId] = dlcApp;
if (allCheckBox.Checked) selection.SelectedSteamDlc[appId] = dlcApp; if (allCheckBox.Checked) selection.SelectedDlc[appId] = dlcApp;
TreeNode dlcNode = TreeNodes.Find(s => s.Name == "" + appId) ?? new(); TreeNode dlcNode = TreeNodes.Find(s => s.Name == appId) ?? new();
dlcNode.Name = appId.ToString(); dlcNode.Name = appId;
dlcNode.Text = dlcApp.name; dlcNode.Text = dlcApp.name;
dlcNode.Checked = selection.SelectedSteamDlc.ContainsKey(appId); dlcNode.Checked = selection.SelectedDlc.ContainsKey(appId);
dlcNode.Remove(); dlcNode.Remove();
programNode.Nodes.Add(dlcNode); programNode.Nodes.Add(dlcNode);
} }
}
}); });
if (Program.Canceled) return; if (Program.Canceled) return;
RemoveFromRemainingGames(name); RemoveFromRemainingGames(name);
@ -226,6 +232,107 @@ internal partial class SelectForm : CustomForm
RunningTasks.Add(task); RunningTasks.Add(task);
progress.Report(-RunningTasks.Count); progress.Report(-RunningTasks.Count);
} }
}
if (Directory.Exists(EpicLibrary.EpicAppDataPath))
{
List<Manifest> epicGames = await EpicLibrary.GetGames();
Dictionary<string, List<string>> games = new();
foreach (Manifest manifest in epicGames)
{
string id = manifest.CatalogNamespace;
string name = manifest.DisplayName;
string directory = manifest.InstallLocation;
ProgramSelection selection = ProgramSelection.FromId(id);
if (Program.Canceled) return;
if (Program.IsGameBlocked(name, directory)) continue;
AddToRemainingGames(name);
Task task = Task.Run(async () =>
{
if (Program.Canceled) return;
List<string> dllDirectories = await EpicLibrary.GetDllDirectoriesFromGameDirectory(directory);
if (dllDirectories is null)
{
RemoveFromRemainingGames(name);
return;
}
if (Program.Canceled) return;
ConcurrentDictionary<string, string> dlc = new();
List<Task> dlcTasks = new();
List<(string id, string name)> dlcIds = await EpicStore.ParseDlcAppIds(id);
if (dlcIds.Count > 0)
{
foreach ((string id, string name) in dlcIds)
{
if (Program.Canceled) return;
AddToRemainingDLCs(id);
Task task = Task.Run(() =>
{
if (Program.Canceled) return;
dlc[id] = name;
RemoveFromRemainingDLCs(id);
progress.Report(++CompleteTasks);
});
dlcTasks.Add(task);
RunningTasks.Add(task);
progress.Report(-RunningTasks.Count);
Thread.Sleep(10); // to reduce control & window freezing
}
}
else
{
RemoveFromRemainingGames(name);
return;
}
if (Program.Canceled) return;
foreach (Task task in dlcTasks)
{
if (Program.Canceled) return;
await task;
}
selection ??= new();
if (allCheckBox.Checked) selection.Enabled = true;
selection.Usable = true;
selection.Id = id;
selection.Name = name;
selection.RootDirectory = directory;
selection.DllDirectories = dllDirectories;
if (Program.Canceled) return;
Program.Invoke(selectionTreeView, delegate
{
if (Program.Canceled) return;
TreeNode programNode = TreeNodes.Find(s => s.Name == id) ?? new();
programNode.Name = id;
programNode.Text = name;
programNode.Checked = selection.Enabled;
programNode.Remove();
selectionTreeView.Nodes.Add(programNode);
foreach (KeyValuePair<string, string> pair in dlc)
{
if (Program.Canceled || programNode is null) return;
string dlcId = pair.Key;
string dlcName = pair.Value;
(string name, string iconStaticId) dlcApp = (dlcName, null); // temporary?
selection.AllDlc[dlcId] = dlcApp;
if (allCheckBox.Checked) selection.SelectedDlc[dlcId] = dlcApp;
TreeNode dlcNode = TreeNodes.Find(s => s.Name == dlcId) ?? new();
dlcNode.Name = dlcId;
dlcNode.Text = dlcName;
dlcNode.Checked = selection.SelectedDlc.ContainsKey(dlcId);
dlcNode.Remove();
programNode.Nodes.Add(dlcNode);
}
});
if (Program.Canceled) return;
RemoveFromRemainingGames(name);
progress.Report(++CompleteTasks);
});
appTasks.Add(task);
RunningTasks.Add(task);
progress.Report(-RunningTasks.Count);
}
}
foreach (Task task in appTasks) foreach (Task task in appTasks)
{ {
if (Program.Canceled) return; if (Program.Canceled) return;
@ -267,17 +374,19 @@ internal partial class SelectForm : CustomForm
progressBar.Value = p; progressBar.Value = p;
}; };
if (Directory.Exists(SteamLibrary.InstallPath))
{
progressLabel.Text = $"Setting up SteamCMD . . . "; progressLabel.Text = $"Setting up SteamCMD . . . ";
await SteamCMD.Setup(iProgress); await SteamCMD.Setup(iProgress);
}
setup = false; setup = false;
progressLabel.Text = "Gathering and caching your applicable games and their DLCs . . . "; progressLabel.Text = "Gathering and caching your applicable games and their DLCs . . . ";
ProgramSelection.ValidateAll(); ProgramSelection.ValidateAll();
TreeNodes.ForEach(node => TreeNodes.ForEach(node =>
{ {
if (!int.TryParse(node.Name, out int appId) || node.Parent is null && ProgramSelection.FromAppId(appId) is null) node.Remove(); if (!int.TryParse(node.Name, out int appId) || node.Parent is null && ProgramSelection.FromId(node.Name) is null) node.Remove();
}); });
await GetCreamApiApplicablePrograms(iProgress); await GetApplicablePrograms(iProgress);
await SteamCMD.Cleanup(); await SteamCMD.Cleanup();
HideProgressBar(); HideProgressBar();
@ -302,21 +411,23 @@ internal partial class SelectForm : CustomForm
{ {
if (e.Action == TreeViewAction.Unknown) return; if (e.Action == TreeViewAction.Unknown) return;
TreeNode node = e.Node; TreeNode node = e.Node;
if (node is not null && int.TryParse(node.Name, out int appId)) if (node is not null)
{ {
ProgramSelection selection = ProgramSelection.FromAppId(appId); string appId = node.Name;
ProgramSelection selection = ProgramSelection.FromId(appId);
if (selection is null) if (selection is null)
{ {
TreeNode parent = node.Parent; TreeNode parent = node.Parent;
if (parent is not null && int.TryParse(parent.Name, out int gameAppId)) if (parent is not null)
{ {
ProgramSelection.FromAppId(gameAppId).ToggleDlc(appId, node.Checked); string gameAppId = parent.Name;
ProgramSelection.FromId(gameAppId).ToggleDlc(appId, node.Checked);
parent.Checked = parent.Nodes.Cast<TreeNode>().ToList().Any(treeNode => treeNode.Checked); parent.Checked = parent.Nodes.Cast<TreeNode>().ToList().Any(treeNode => treeNode.Checked);
} }
} }
else else
{ {
if (selection.AllSteamDlc.Any()) if (selection.AllDlc.Any())
{ {
selection.ToggleAllDlc(node.Checked); selection.ToggleAllDlc(node.Checked);
node.Nodes.Cast<TreeNode>().ToList().ForEach(treeNode => treeNode.Checked = node.Checked); node.Nodes.Cast<TreeNode>().ToList().ForEach(treeNode => treeNode.Checked = node.Checked);
@ -345,10 +456,19 @@ internal partial class SelectForm : CustomForm
private class TreeNodeSorter : IComparer private class TreeNodeSorter : IComparer
{ {
public int Compare(object a, object b) => public int Compare(object a, object b)
!int.TryParse((a as TreeNode).Name, out int A) ? 1 {
: !int.TryParse((b as TreeNode).Name, out int B) ? 0 string aId = (a as TreeNode).Name;
: A > B ? 1 : 0; string bId = (b as TreeNode).Name;
return aId == "ParadoxLauncher" ? -1
: bId == "ParadoxLauncher" ? 1
: !int.TryParse(aId, out _) && !int.TryParse(bId, out _) ? string.Compare(aId, bId)
: !int.TryParse(aId, out int A) ? 1
: !int.TryParse(bId, out int B) ? -1
: A > B ? 1
: A < B ? -1
: 0;
}
} }
private void ShowProgressBar() private void ShowProgressBar()
@ -389,11 +509,11 @@ internal partial class SelectForm : CustomForm
Dictionary<string, Image> images = new(); Dictionary<string, Image> images = new();
Task.Run(async () => Task.Run(async () =>
{ {
if (Directory.Exists(SteamLibrary.ParadoxLauncherInstallPath)) if (Directory.Exists(ParadoxLauncher.InstallPath))
{ {
foreach (string file in Directory.GetFiles(SteamLibrary.ParadoxLauncherInstallPath, "*.exe")) foreach (string file in Directory.GetFiles(ParadoxLauncher.InstallPath, "*.exe"))
{ {
images["Paradox Launcher"] = IconGrabber.GetFileIconImage(file); images["Icon_ParadoxLauncher"] = IconGrabber.GetFileIconImage(file);
break; break;
} }
} }
@ -405,7 +525,7 @@ internal partial class SelectForm : CustomForm
images["Steam Community"] = await HttpClientManager.GetImageFromUrl("https://steamcommunity.com/favicon.ico"); images["Steam Community"] = await HttpClientManager.GetImageFromUrl("https://steamcommunity.com/favicon.ico");
}); });
Image Image(string identifier) => images.GetValueOrDefault(identifier, null); Image Image(string identifier) => images.GetValueOrDefault(identifier, null);
void TrySetImageAsync(ToolStripMenuItem menuItem, int appId, string iconStaticId, bool client = false) => void TrySetImageAsync(ToolStripMenuItem menuItem, string appId, string iconStaticId, bool client = false) =>
Task.Run(async () => Task.Run(async () =>
{ {
menuItem.Image = client ? await IconGrabber.GetSteamClientIcon(appId, iconStaticId) : await IconGrabber.GetSteamIcon(appId, iconStaticId); menuItem.Image = client ? await IconGrabber.GetSteamClientIcon(appId, iconStaticId) : await IconGrabber.GetSteamIcon(appId, iconStaticId);
@ -415,33 +535,33 @@ internal partial class SelectForm : CustomForm
{ {
TreeNode node = e.Node; TreeNode node = e.Node;
TreeNode parentNode = node.Parent; TreeNode parentNode = node.Parent;
if (!int.TryParse(node.Name, out int appId)) return; string id = node.Name;
ProgramSelection selection = ProgramSelection.FromAppId(appId); ProgramSelection selection = ProgramSelection.FromId(id);
(int gameAppId, (string name, string iconStaticId) app)? dlc = null; (string gameAppId, (string name, string iconStaticId) app)? dlc = null;
if (selection is null) dlc = ProgramSelection.GetDlcFromAppId(appId); if (selection is null) dlc = ProgramSelection.GetDlcFromId(id);
if (e.Button == MouseButtons.Right && node.Bounds.Contains(e.Location)) if (e.Button == MouseButtons.Right && node.Bounds.Contains(e.Location))
{ {
selectionTreeView.SelectedNode = node; selectionTreeView.SelectedNode = node;
nodeContextMenu.Items.Clear(); nodeContextMenu.Items.Clear();
ToolStripMenuItem header = new(selection?.Name ?? node.Text, Image(appId == 0 ? "Paradox Launcher" : "Icon_" + node.Name)); ToolStripMenuItem header = new(selection?.Name ?? node.Text, Image("Icon_" + id));
if (header.Image is null) if (header.Image is null)
{ {
string iconStaticId = dlc?.app.iconStaticId ?? selection?.IconStaticID; string iconStaticId = dlc?.app.iconStaticId ?? selection?.IconStaticID;
if (iconStaticId is not null) if (iconStaticId is not null)
TrySetImageAsync(header, appId, iconStaticId); TrySetImageAsync(header, id, iconStaticId);
else if (dlc is not null) else if (dlc is not null)
{ {
int gameAppId = dlc.Value.gameAppId; string gameAppId = dlc.Value.gameAppId;
header.Image = Image("Icon_" + gameAppId); header.Image = Image("Icon_" + gameAppId);
ProgramSelection gameSelection = ProgramSelection.FromAppId(gameAppId); ProgramSelection gameSelection = ProgramSelection.FromId(gameAppId);
iconStaticId = gameSelection?.IconStaticID; iconStaticId = gameSelection?.IconStaticID;
if (header.Image is null && iconStaticId is not null) if (header.Image is null && iconStaticId is not null)
TrySetImageAsync(header, gameAppId, iconStaticId); TrySetImageAsync(header, gameAppId, iconStaticId);
} }
} }
nodeContextMenu.Items.Add(header); nodeContextMenu.Items.Add(header);
string appInfo = $@"{SteamCMD.AppInfoPath}\{appId}.vdf"; string appInfo = $@"{SteamCMD.AppInfoPath}\{id}.vdf";
if (appId != 0 && Directory.Exists(Directory.GetDirectoryRoot(appInfo)) && File.Exists(appInfo)) if (Directory.Exists(Directory.GetDirectoryRoot(appInfo)) && File.Exists(appInfo))
{ {
nodeContextMenu.Items.Add(new ToolStripSeparator()); nodeContextMenu.Items.Add(new ToolStripSeparator());
nodeContextMenu.Items.Add(new ToolStripMenuItem("Open AppInfo", Image("Notepad"), nodeContextMenu.Items.Add(new ToolStripMenuItem("Open AppInfo", Image("Notepad"),
@ -459,39 +579,54 @@ internal partial class SelectForm : CustomForm
} }
if (selection is not null) if (selection is not null)
{ {
if (appId == 0) if (id == "ParadoxLauncher")
{ {
nodeContextMenu.Items.Add(new ToolStripSeparator()); nodeContextMenu.Items.Add(new ToolStripSeparator());
nodeContextMenu.Items.Add(new ToolStripMenuItem("Repair", Image("Command Prompt"), nodeContextMenu.Items.Add(new ToolStripMenuItem("Repair", Image("Command Prompt"),
new EventHandler(async (sender, e) => new EventHandler(async (sender, e) =>
{ {
if (!Program.IsProgramRunningDialog(this, selection)) return; if (!Program.IsProgramRunningDialog(this, selection)) return;
byte[] cApiIni = null; byte[] cApiIni = null;
byte[] properApi = null; byte[] properApi = null;
byte[] properApi64 = null; byte[] properApi64 = null;
foreach (string directory in selection.SteamApiDllDirectories)
byte[] sApiJson = null;
byte[] properSdk = null;
byte[] properSdk64 = null;
foreach (string directory in selection.DllDirectories)
{ {
directory.GetApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi); directory.GetCreamApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi);
if (cApiIni is null && File.Exists(cApi)) if (cApiIni is null && File.Exists(cApi))
cApiIni = File.ReadAllBytes(cApi); cApiIni = File.ReadAllBytes(cApi);
await InstallForm.UninstallCreamAPI(directory); await InstallForm.UninstallCreamAPI(directory);
if (properApi is null && File.Exists(api) && !FileResourceExtensions.Equals(Properties.Resources.API, api)) if (properApi is null && File.Exists(api) && !Properties.Resources.API.EqualsFile(api))
properApi = File.ReadAllBytes(api); properApi = File.ReadAllBytes(api);
if (properApi64 is null && File.Exists(api64) && !FileResourceExtensions.Equals(Properties.Resources.API64, api64)) if (properApi64 is null && File.Exists(api64) && !Properties.Resources.API64.EqualsFile(api64))
properApi64 = File.ReadAllBytes(api64); properApi64 = File.ReadAllBytes(api64);
directory.GetScreamApiComponents(out string sdk, out string sdk_o, out string sdk64, out string sdk64_o, out string sApi);
if (sApiJson is null && File.Exists(sApi))
sApiJson = File.ReadAllBytes(sApi);
await InstallForm.UninstallCreamAPI(directory);
if (properSdk is null && File.Exists(sdk) && !Properties.Resources.SDK.EqualsFile(sdk))
properSdk = File.ReadAllBytes(sdk);
if (properSdk64 is null && File.Exists(sdk64) && !Properties.Resources.SDK64.EqualsFile(sdk64))
properSdk64 = File.ReadAllBytes(sdk64);
} }
if (properApi is not null || properApi64 is not null) if (properApi is not null || properApi64 is not null || properSdk is not null || properSdk64 is not null)
{ {
bool neededRepair = false; bool neededRepair = false;
foreach (string directory in selection.SteamApiDllDirectories) foreach (string directory in selection.DllDirectories)
{ {
directory.GetApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi); directory.GetCreamApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi);
if (properApi is not null && FileResourceExtensions.Equals(Properties.Resources.API, api)) if (properApi is not null && Properties.Resources.API.EqualsFile(api))
{ {
properApi.Write(api); properApi.Write(api);
neededRepair = true; neededRepair = true;
} }
if (properApi64 is not null && FileResourceExtensions.Equals(Properties.Resources.API64, api64)) if (properApi64 is not null && Properties.Resources.API64.EqualsFile(api64))
{ {
properApi64.Write(api64); properApi64.Write(api64);
neededRepair = true; neededRepair = true;
@ -501,46 +636,67 @@ internal partial class SelectForm : CustomForm
await InstallForm.InstallCreamAPI(directory, selection); await InstallForm.InstallCreamAPI(directory, selection);
cApiIni.Write(cApi); cApiIni.Write(cApi);
} }
directory.GetScreamApiComponents(out string sdk, out string sdk_o, out string sdk64, out string sdk64_o, out string sApi);
if (properSdk is not null && Properties.Resources.SDK.EqualsFile(sdk))
{
properSdk.Write(sdk);
neededRepair = true;
}
if (properSdk64 is not null && Properties.Resources.SDK64.EqualsFile(sdk64))
{
properSdk64.Write(sdk64);
neededRepair = true;
}
if (sApiJson is not null)
{
await InstallForm.InstallScreamAPI(directory, selection);
sApiJson.Write(sApi);
}
} }
if (neededRepair) if (neededRepair)
new DialogForm(this).Show("Paradox Launcher Repair", Icon, "Paradox Launcher successfully repaired!", "OK"); new DialogForm(this).Show(Icon, "Paradox Launcher successfully repaired!", "OK");
else else
new DialogForm(this).Show("Paradox Launcher Repair", SystemIcons.Information, "Paradox Launcher does not need to be repaired.", "OK"); new DialogForm(this).Show(SystemIcons.Information, "Paradox Launcher does not need to be repaired.", "OK");
} }
else else
new DialogForm(this).Show("Paradox Launcher Repair", SystemIcons.Error, "Paradox Launcher repair failed!" new DialogForm(this).Show(SystemIcons.Error, "Paradox Launcher repair failed!"
+ "\n\nAn original Steamworks API file could not be found." + "\n\nAn original Steamworks API or EOS SDK file could not be found."
+ "\nYou must reinstall Paradox Launcher to fix this issue.", "OK"); + "\nYou must reinstall Paradox Launcher to fix this issue.", "OK");
}))); })));
} }
nodeContextMenu.Items.Add(new ToolStripSeparator()); nodeContextMenu.Items.Add(new ToolStripSeparator());
nodeContextMenu.Items.Add(new ToolStripMenuItem("Open Root Directory", Image("File Explorer"), nodeContextMenu.Items.Add(new ToolStripMenuItem("Open Root Directory", Image("File Explorer"),
new EventHandler((sender, e) => Diagnostics.OpenDirectoryInFileExplorer(selection.RootDirectory)))); new EventHandler((sender, e) => Diagnostics.OpenDirectoryInFileExplorer(selection.RootDirectory))));
for (int i = 0; i < selection.SteamApiDllDirectories.Count; i++) for (int i = 0; i < selection.DllDirectories.Count; i++)
{ {
string directory = selection.SteamApiDllDirectories[i]; string directory = selection.DllDirectories[i];
nodeContextMenu.Items.Add(new ToolStripMenuItem($"Open Steamworks Directory ({i + 1})", Image("File Explorer"), nodeContextMenu.Items.Add(new ToolStripMenuItem($"Open {(selection.IsSteam ? "Steamworks API" : "EOS SDK")} Directory ({i + 1})", Image("File Explorer"),
new EventHandler((sender, e) => Diagnostics.OpenDirectoryInFileExplorer(directory)))); new EventHandler((sender, e) => Diagnostics.OpenDirectoryInFileExplorer(directory))));
} }
} }
if (appId != 0) if (id != "ParadoxLauncher" && selection is not null)
{
if (selection.IsSteam)
{ {
nodeContextMenu.Items.Add(new ToolStripSeparator()); nodeContextMenu.Items.Add(new ToolStripSeparator());
nodeContextMenu.Items.Add(new ToolStripMenuItem("Open SteamDB", Image("SteamDB"), nodeContextMenu.Items.Add(new ToolStripMenuItem("Open SteamDB", Image("SteamDB"),
new EventHandler((sender, e) => Diagnostics.OpenUrlInInternetBrowser("https://steamdb.info/app/" + appId)))); new EventHandler((sender, e) => Diagnostics.OpenUrlInInternetBrowser("https://steamdb.info/app/" + id))));
nodeContextMenu.Items.Add(new ToolStripMenuItem("Open Steam Store", Image("Steam Store"), nodeContextMenu.Items.Add(new ToolStripMenuItem("Open Steam Store", Image("Steam Store"),
new EventHandler((sender, e) => Diagnostics.OpenUrlInInternetBrowser("https://store.steampowered.com/app/" + appId)))); new EventHandler((sender, e) => Diagnostics.OpenUrlInInternetBrowser("https://store.steampowered.com/app/" + id))));
if (selection is not null) ToolStripMenuItem steamCommunity = new("Open Steam Community", Image("ClientIcon_" + id),
{ new EventHandler((sender, e) => Diagnostics.OpenUrlInInternetBrowser("https://steamcommunity.com/app/" + id)));
ToolStripMenuItem steamCommunity = new("Open Steam Community", Image("ClientIcon_" + node.Name),
new EventHandler((sender, e) => Diagnostics.OpenUrlInInternetBrowser("https://steamcommunity.com/app/" + appId)));
nodeContextMenu.Items.Add(steamCommunity); nodeContextMenu.Items.Add(steamCommunity);
if (steamCommunity.Image is null) if (steamCommunity.Image is null)
{ {
steamCommunity.Image = Image("Steam Community"); steamCommunity.Image = Image("Steam Community");
TrySetImageAsync(steamCommunity, appId, selection.ClientIconStaticID, true); TrySetImageAsync(steamCommunity, id, selection.ClientIconStaticID, true);
} }
} }
else
{
// Epic Games links?
}
} }
nodeContextMenu.Show(selectionTreeView, e.Location); nodeContextMenu.Show(selectionTreeView, e.Location);
} }
@ -548,52 +704,13 @@ internal partial class SelectForm : CustomForm
OnLoad(); OnLoad();
} }
private static void PopulateParadoxLauncherDlc(ProgramSelection paradoxLauncher = null)
{
paradoxLauncher ??= ProgramSelection.FromAppId(0);
if (paradoxLauncher is not null)
{
paradoxLauncher.ExtraSteamAppIdDlc.Clear();
foreach (ProgramSelection selection in ProgramSelection.AllUsableEnabled)
{
if (selection.Name == paradoxLauncher.Name) continue;
if (selection.AppInfo.Value?.GetChild("extended")?.GetChild("publisher")?.ToString() != "Paradox Interactive") continue;
paradoxLauncher.ExtraSteamAppIdDlc.Add(new(selection.SteamAppId, selection.Name, selection.SelectedSteamDlc));
}
if (!paradoxLauncher.ExtraSteamAppIdDlc.Any())
foreach (ProgramSelection selection in ProgramSelection.AllUsable)
{
if (selection.Name == paradoxLauncher.Name) continue;
if (selection.AppInfo.Value?.GetChild("extended")?.GetChild("publisher")?.ToString() != "Paradox Interactive") continue;
paradoxLauncher.ExtraSteamAppIdDlc.Add(new(selection.SteamAppId, selection.Name, selection.AllSteamDlc));
}
}
}
private static bool ParadoxLauncherDlcDialog(Form form)
{
ProgramSelection paradoxLauncher = ProgramSelection.FromAppId(0);
if (paradoxLauncher is not null && paradoxLauncher.Enabled)
{
PopulateParadoxLauncherDlc(paradoxLauncher);
if (!paradoxLauncher.ExtraSteamAppIdDlc.Any())
{
return new DialogForm(form).Show(Program.ApplicationName, SystemIcons.Warning,
$"WARNING: There are no installed games with DLC that can be added to the Paradox Launcher!" +
"\n\nInstalling CreamAPI for the Paradox Launcher is pointless, since no DLC will be added to the configuration!",
"Ignore", "Cancel") != DialogResult.OK;
}
}
return false;
}
private void OnAccept(bool uninstall = false) private void OnAccept(bool uninstall = false)
{ {
if (ProgramSelection.All.Any()) if (ProgramSelection.All.Any())
{ {
foreach (ProgramSelection selection in ProgramSelection.AllUsableEnabled) foreach (ProgramSelection selection in ProgramSelection.AllUsableEnabled)
if (!Program.IsProgramRunningDialog(this, selection)) return; if (!Program.IsProgramRunningDialog(this, selection)) return;
if (ParadoxLauncherDlcDialog(this)) return; if (ParadoxLauncher.DlcDialog(this)) return;
Hide(); Hide();
InstallForm installForm = new(this, uninstall); InstallForm installForm = new(this, uninstall);
installForm.ShowDialog(); installForm.ShowDialog();
@ -655,9 +772,9 @@ internal partial class SelectForm : CustomForm
string blockedDirectoryExceptions = ""; string blockedDirectoryExceptions = "";
foreach (string name in Program.ProtectedGameDirectoryExceptions) foreach (string name in Program.ProtectedGameDirectoryExceptions)
blockedDirectoryExceptions += helpButtonListPrefix + name; blockedDirectoryExceptions += helpButtonListPrefix + name;
new DialogForm(this).Show(blockedGamesCheckBox.Text, SystemIcons.Information, new DialogForm(this).Show(SystemIcons.Information,
"Blocks the program from caching and displaying games protected by DLL checks," + "Blocks the program from caching and displaying games protected by DLL checks," +
"\nanti-cheats, or that are confirmed not to be working with CreamAPI." + "\nanti-cheats, or that are confirmed not to be working with CreamAPI or ScreamAPI." +
"\n\nBlocked game names:" + blockedGames + "\n\nBlocked game names:" + blockedGames +
"\n\nBlocked game sub-directories:" + blockedDirectories + "\n\nBlocked game sub-directories:" + blockedDirectories +
"\n\nBlocked game sub-directory exceptions (not blocked):" + blockedDirectoryExceptions, "\n\nBlocked game sub-directory exceptions (not blocked):" + blockedDirectoryExceptions,

View file

@ -0,0 +1,62 @@
using System.Drawing;
using System.Linq;
using System.Windows.Forms;
using CreamInstaller.Steam;
using Microsoft.Win32;
namespace CreamInstaller.Paradox;
internal static class ParadoxLauncher
{
private static string installPath = null;
internal static string InstallPath
{
get
{
installPath ??= Registry.GetValue(@"HKEY_CURRENT_USER\Software\Paradox Interactive\Paradox Launcher v2", "LauncherInstallation", null) as string;
installPath ??= Registry.GetValue(@"HKEY_CURRENT_USER\Software\Wow6432Node\Paradox Interactive\Paradox Launcher v2", "LauncherInstallation", null) as string;
return installPath;
}
}
private static void PopulateDlc(ProgramSelection paradoxLauncher = null)
{
paradoxLauncher ??= ProgramSelection.FromId("ParadoxLauncher");
if (paradoxLauncher is not null)
{
paradoxLauncher.ExtraDlc.Clear();
foreach (ProgramSelection selection in ProgramSelection.AllUsableEnabled)
{
if (selection == paradoxLauncher) continue;
if (selection.AppInfo is null || selection.AppInfo.Value?.GetChild("extended")?.GetChild("publisher")?.ToString() != "Paradox Interactive") continue;
paradoxLauncher.ExtraDlc.Add(new(selection.Id, selection.Name, selection.SelectedDlc));
}
if (!paradoxLauncher.ExtraDlc.Any())
foreach (ProgramSelection selection in ProgramSelection.AllUsable)
{
if (selection == paradoxLauncher) continue;
if (selection.AppInfo is null || selection.AppInfo.Value?.GetChild("extended")?.GetChild("publisher")?.ToString() != "Paradox Interactive") continue;
paradoxLauncher.ExtraDlc.Add(new(selection.Id, selection.Name, selection.AllDlc));
}
}
}
internal static bool DlcDialog(Form form)
{
ProgramSelection paradoxLauncher = ProgramSelection.FromId("ParadoxLauncher");
if (paradoxLauncher is not null && paradoxLauncher.Enabled)
{
PopulateDlc(paradoxLauncher);
if (!paradoxLauncher.ExtraDlc.Any())
{
return new DialogForm(form).Show(SystemIcons.Warning,
$"WARNING: There are no installed games with DLC that can be added to the Paradox Launcher!" +
"\n\nInstalling CreamAPI/ScreamAPI for the Paradox Launcher is pointless, since no DLC will be added to the configuration!",
"Ignore", "Cancel") != DialogResult.OK;
}
}
return false;
}
}

View file

@ -7,20 +7,22 @@ using System.Reflection;
using System.Threading; using System.Threading;
using System.Windows.Forms; using System.Windows.Forms;
using CreamInstaller.Classes; using CreamInstaller.Steam;
using CreamInstaller.Utility;
namespace CreamInstaller; namespace CreamInstaller;
internal static class Program internal static class Program
{ {
internal static readonly string ApplicationName = Application.CompanyName + " v" + Application.ProductVersion + ": " + Application.ProductName; internal static readonly string ApplicationName = Application.CompanyName + " v" + Application.ProductVersion + ": " + Application.ProductName;
internal static readonly string ApplicationNameShort = Application.CompanyName + " v" + Application.ProductVersion;
internal static readonly Assembly EntryAssembly = Assembly.GetEntryAssembly(); internal static readonly Assembly EntryAssembly = Assembly.GetEntryAssembly();
internal static readonly Process CurrentProcess = Process.GetCurrentProcess(); internal static readonly Process CurrentProcess = Process.GetCurrentProcess();
internal static readonly string CurrentProcessFilePath = CurrentProcess.MainModule.FileName; internal static readonly string CurrentProcessFilePath = CurrentProcess.MainModule.FileName;
internal static bool BlockProtectedGames = true; internal static bool BlockProtectedGames = true;
internal static readonly string[] ProtectedGameNames = { "PAYDAY 2", "Call to Arms" }; // non-functioning CreamAPI or DLL detections internal static readonly string[] ProtectedGameNames = { "PAYDAY 2", "Call to Arms" }; // non-functioning CreamAPI/ScreamAPI or DLL detections
internal static readonly string[] ProtectedGameDirectories = { @"\EasyAntiCheat", @"\BattlEye" }; // DLL detections internal static readonly string[] ProtectedGameDirectories = { @"\EasyAntiCheat", @"\BattlEye" }; // DLL detections
internal static readonly string[] ProtectedGameDirectoryExceptions = { "Arma 3" }; // Arma 3's BattlEye doesn't detect DLL changes? internal static readonly string[] ProtectedGameDirectoryExceptions = { "Arma 3" }; // Arma 3's BattlEye doesn't detect DLL changes?
@ -44,9 +46,9 @@ internal static class Program
internal static bool IsProgramRunningDialog(Form form, ProgramSelection selection) internal static bool IsProgramRunningDialog(Form form, ProgramSelection selection)
{ {
if (selection.AreSteamApiDllsLocked) if (selection.AreDllsLocked)
{ {
if (new DialogForm(form).Show(ApplicationName, SystemIcons.Error, if (new DialogForm(form).Show(SystemIcons.Error,
$"ERROR: {selection.Name} is currently running!" + $"ERROR: {selection.Name} is currently running!" +
"\n\nPlease close the program/game to continue . . . ", "\n\nPlease close the program/game to continue . . . ",
"Retry", "Cancel") == DialogResult.OK) "Retry", "Cancel") == DialogResult.OK)
@ -56,7 +58,7 @@ internal static class Program
return false; return false;
} }
internal static void GetApiComponents(this string directory, out string api, out string api_o, out string api64, out string api64_o, out string cApi) internal static void GetCreamApiComponents(this string directory, out string api, out string api_o, out string api64, out string api64_o, out string cApi)
{ {
api = directory + @"\steam_api.dll"; api = directory + @"\steam_api.dll";
api_o = directory + @"\steam_api_o.dll"; api_o = directory + @"\steam_api_o.dll";
@ -65,6 +67,15 @@ internal static class Program
cApi = directory + @"\cream_api.ini"; cApi = directory + @"\cream_api.ini";
} }
internal static void GetScreamApiComponents(this string directory, out string sdk, out string sdk_o, out string sdk64, out string sdk64_o, out string sApi)
{
sdk = directory + @"\EOSSDK-Win32-Shipping.dll";
sdk_o = directory + @"\EOSSDK-Win32-Shipping_o.dll";
sdk64 = directory + @"\EOSSDK-Win64-Shipping.dll";
sdk64_o = directory + @"\EOSSDK-Win64-Shipping_o.dll";
sApi = directory + @"\ScreamAPI.json";
}
[STAThread] [STAThread]
private static void Main() private static void Main()
{ {

View file

@ -0,0 +1,126 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Gameloop.Vdf.Linq;
namespace CreamInstaller;
internal class ProgramSelection
{
internal bool Enabled = false;
internal bool Usable = true;
internal string Id = "0";
internal string Name = "Program";
internal string IconStaticID = null;
internal string ClientIconStaticID = null;
internal string RootDirectory = null;
internal List<string> DllDirectories = null;
internal bool IsSteam = false;
internal VProperty AppInfo = null;
internal readonly SortedList<string, (string name, string iconStaticId)> AllDlc = new();
internal readonly SortedList<string, (string name, string iconStaticId)> SelectedDlc = new();
internal readonly List<Tuple<string, string, SortedList<string, (string name, string iconStaticId)>>> ExtraDlc = new();
internal bool AreDllsLocked
{
get
{
foreach (string directory in DllDirectories)
{
directory.GetCreamApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi);
if (api.IsFilePathLocked()
|| api_o.IsFilePathLocked()
|| api64.IsFilePathLocked()
|| api64_o.IsFilePathLocked()
|| cApi.IsFilePathLocked())
return true;
directory.GetScreamApiComponents(out string sdk, out string sdk_o, out string sdk64, out string sdk64_o, out string sApi);
if (sdk.IsFilePathLocked()
|| sdk_o.IsFilePathLocked()
|| sdk64.IsFilePathLocked()
|| sdk64_o.IsFilePathLocked()
|| sApi.IsFilePathLocked())
return true;
}
return false;
}
}
private void Toggle(string dlcAppId, (string name, string iconStaticId) dlcApp, bool enabled)
{
if (enabled) SelectedDlc[dlcAppId] = dlcApp;
else SelectedDlc.Remove(dlcAppId);
}
internal void ToggleDlc(string dlcAppId, bool enabled)
{
foreach (KeyValuePair<string, (string name, string iconStaticId)> pair in AllDlc)
{
string appId = pair.Key;
(string name, string iconStaticId) dlcApp = pair.Value;
if (appId == dlcAppId)
{
Toggle(appId, dlcApp, enabled);
break;
}
}
Enabled = SelectedDlc.Any();
}
internal void ToggleAllDlc(bool enabled)
{
if (!enabled) SelectedDlc.Clear();
else foreach (KeyValuePair<string, (string name, string iconStaticId)> pair in AllDlc)
{
string appId = pair.Key;
(string name, string iconStaticId) dlcApp = pair.Value;
Toggle(appId, dlcApp, enabled);
}
Enabled = SelectedDlc.Any();
}
internal ProgramSelection() => All.Add(this);
internal void Validate()
{
if (Program.IsGameBlocked(Name, RootDirectory))
{
All.Remove(this);
return;
}
if (!Directory.Exists(RootDirectory))
{
All.Remove(this);
return;
}
DllDirectories.RemoveAll(directory => !Directory.Exists(directory));
if (!DllDirectories.Any()) All.Remove(this);
}
internal static void ValidateAll() => AllSafe.ForEach(selection => selection.Validate());
internal static List<ProgramSelection> All = new();
internal static List<ProgramSelection> AllSafe => All.ToList();
internal static List<ProgramSelection> AllUsable => All.FindAll(s => s.Usable);
internal static List<ProgramSelection> AllUsableEnabled => AllUsable.FindAll(s => s.Enabled);
internal static ProgramSelection FromId(string id) => AllUsable.Find(s => s.Id == id);
internal static (string gameAppId, (string name, string iconStaticId) app)? GetDlcFromId(string appId)
{
foreach (ProgramSelection selection in AllUsable)
foreach (KeyValuePair<string, (string name, string iconStaticId)> pair in selection.AllDlc)
if (pair.Key == appId) return (selection.Id, pair.Value);
return null;
}
}

View file

@ -19,7 +19,7 @@ namespace CreamInstaller.Properties {
// class via a tool like ResGen or Visual Studio. // class via a tool like ResGen or Visual Studio.
// To add or remove a member, edit your .ResX file then rerun ResGen // To add or remove a member, edit your .ResX file then rerun ResGen
// with the /str option, or rebuild your VS project. // with the /str option, or rebuild your VS project.
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "16.0.0.0")] [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "17.0.0.0")]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()] [global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
internal class Resources { internal class Resources {
@ -89,5 +89,25 @@ namespace CreamInstaller.Properties {
return ((System.Drawing.Icon)(obj)); return ((System.Drawing.Icon)(obj));
} }
} }
/// <summary>
/// Looks up a localized resource of type System.Byte[].
/// </summary>
internal static byte[] SDK {
get {
object obj = ResourceManager.GetObject("SDK", resourceCulture);
return ((byte[])(obj));
}
}
/// <summary>
/// Looks up a localized resource of type System.Byte[].
/// </summary>
internal static byte[] SDK64 {
get {
object obj = ResourceManager.GetObject("SDK64", resourceCulture);
return ((byte[])(obj));
}
}
} }
} }

View file

@ -112,12 +112,12 @@
<value>2.0</value> <value>2.0</value>
</resheader> </resheader>
<resheader name="reader"> <resheader name="reader">
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=5.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> <value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</resheader> </resheader>
<resheader name="writer"> <resheader name="writer">
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=5.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> <value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</resheader> </resheader>
<assembly alias="System.Windows.Forms" name="System.Windows.Forms, Version=5.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" /> <assembly alias="System.Windows.Forms" name="System.Windows.Forms, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" />
<data name="API" type="System.Resources.ResXFileRef, System.Windows.Forms"> <data name="API" type="System.Resources.ResXFileRef, System.Windows.Forms">
<value>..\resources\steam_api.dll;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> <value>..\resources\steam_api.dll;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</data> </data>
@ -127,4 +127,10 @@
<data name="Icon" type="System.Resources.ResXFileRef, System.Windows.Forms"> <data name="Icon" type="System.Resources.ResXFileRef, System.Windows.Forms">
<value>..\resources\ini.ico;System.Drawing.Icon, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a</value> <value>..\resources\ini.ico;System.Drawing.Icon, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a</value>
</data> </data>
<data name="SDK" type="System.Resources.ResXFileRef, System.Windows.Forms">
<value>..\Resources\EOSSDK-Win32-Shipping.dll;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</data>
<data name="SDK64" type="System.Resources.ResXFileRef, System.Windows.Forms">
<value>..\Resources\EOSSDK-Win64-Shipping.dll;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</data>
</root> </root>

Binary file not shown.

Binary file not shown.

View file

@ -10,7 +10,7 @@ internal static class FileResourceExtensions
file.Write(resource); file.Write(resource);
} }
internal static bool Equals(byte[] resource, string filePath) internal static bool EqualsFile(this byte[] resource, string filePath)
{ {
byte[] file = File.ReadAllBytes(filePath); byte[] file = File.ReadAllBytes(filePath);
if (resource.Length != file.Length) if (resource.Length != file.Length)

View file

@ -14,7 +14,7 @@ using CreamInstaller.Resources;
using Gameloop.Vdf.Linq; using Gameloop.Vdf.Linq;
namespace CreamInstaller.Classes; namespace CreamInstaller.Steam;
internal static class SteamCMD internal static class SteamCMD
{ {
@ -120,6 +120,7 @@ internal static class SteamCMD
internal static async Task Cleanup() => await Task.Run(async () => internal static async Task Cleanup() => await Task.Run(async () =>
{ {
if (!Directory.Exists(DirectoryPath)) return;
await Kill(); await Kill();
try try
{ {
@ -171,7 +172,7 @@ internal static class SteamCMD
catch { } catch { }
}); });
internal static async Task<VProperty> GetAppInfo(int appId, string branch = "public", int buildId = 0) internal static async Task<VProperty> GetAppInfo(string appId, string branch = "public", int buildId = 0)
{ {
if (Program.Canceled) return null; if (Program.Canceled) return null;
string output; string output;
@ -200,10 +201,8 @@ internal static class SteamCMD
goto restart; goto restart;
} }
if (appInfo.Value is VValue) if (appInfo.Value is VValue)
{
//new DialogForm(null).Show("GetAppInfo", SystemIcons.Information, "VValue exception:\n\n" + output, "OK"); //new DialogForm(null).Show("GetAppInfo", SystemIcons.Information, "VValue exception:\n\n" + output, "OK");
goto restart; goto restart;
}
if (appInfo is null || appInfo.Value?.Children()?.ToList()?.Count == 0) return appInfo; if (appInfo is null || appInfo.Value?.Children()?.ToList()?.Count == 0) return appInfo;
VToken type = appInfo.Value?.GetChild("common")?.GetChild("type"); VToken type = appInfo.Value?.GetChild("common")?.GetChild("type");
if (type is null || type.ToString() == "Game") if (type is null || type.ToString() == "Game")
@ -212,8 +211,8 @@ internal static class SteamCMD
if (buildid is null && type is not null) return appInfo; if (buildid is null && type is not null) return appInfo;
if (type is null || int.TryParse(buildid, out int gamebuildId) && gamebuildId < buildId) if (type is null || int.TryParse(buildid, out int gamebuildId) && gamebuildId < buildId)
{ {
List<int> dlcAppIds = await ParseDlcAppIds(appInfo); List<string> dlcAppIds = await ParseDlcAppIds(appInfo);
foreach (int id in dlcAppIds) foreach (string id in dlcAppIds)
{ {
string dlcAppUpdateFile = $@"{AppInfoPath}\{id}.vdf"; string dlcAppUpdateFile = $@"{AppInfoPath}\{id}.vdf";
if (File.Exists(dlcAppUpdateFile)) File.Delete(dlcAppUpdateFile); if (File.Exists(dlcAppUpdateFile)) File.Delete(dlcAppUpdateFile);
@ -225,9 +224,9 @@ internal static class SteamCMD
return appInfo; return appInfo;
} }
internal static async Task<List<int>> ParseDlcAppIds(VProperty appInfo) => await Task.Run(() => internal static async Task<List<string>> ParseDlcAppIds(VProperty appInfo) => await Task.Run(() =>
{ {
List<int> dlcIds = new(); List<string> dlcIds = new();
#pragma warning disable IDE0150 // Prefer 'null' check over type check #pragma warning disable IDE0150 // Prefer 'null' check over type check
if (Program.Canceled || appInfo is not VProperty) return dlcIds; if (Program.Canceled || appInfo is not VProperty) return dlcIds;
#pragma warning restore IDE0150 // Prefer 'null' check over type check #pragma warning restore IDE0150 // Prefer 'null' check over type check
@ -236,13 +235,15 @@ internal static class SteamCMD
foreach (VProperty property in extended) foreach (VProperty property in extended)
if (property.Key == "listofdlc") if (property.Key == "listofdlc")
foreach (string id in property.Value.ToString().Split(",")) foreach (string id in property.Value.ToString().Split(","))
if (int.TryParse(id, out int appId) && !dlcIds.Contains(appId)) dlcIds.Add(appId); if (int.TryParse(id, out int appId)
&& !dlcIds.Contains("" + appId))
dlcIds.Add("" + appId);
VToken depots = appInfo.Value.GetChild("depots"); VToken depots = appInfo.Value.GetChild("depots");
if (depots is not null) foreach (VProperty property in depots) if (depots is not null) foreach (VProperty property in depots)
if (int.TryParse(property.Key, out int _) if (int.TryParse(property.Key, out int _)
&& int.TryParse(property.Value.GetChild("dlcappid")?.ToString(), out int appid) && int.TryParse(property.Value.GetChild("dlcappid")?.ToString(), out int appid)
&& !dlcIds.Contains(appid)) && !dlcIds.Contains("" + appid))
dlcIds.Add(appid); dlcIds.Add("" + appid);
return dlcIds; return dlcIds;
}); });
@ -250,7 +251,6 @@ internal static class SteamCMD
{ {
List<Task> tasks = new(); List<Task> tasks = new();
foreach (Process process in Process.GetProcessesByName("steamcmd")) foreach (Process process in Process.GetProcessesByName("steamcmd"))
{
tasks.Add(Task.Run(() => tasks.Add(Task.Run(() =>
{ {
try try
@ -260,7 +260,6 @@ internal static class SteamCMD
} }
catch { } catch { }
})); }));
}
foreach (Task task in tasks) await task; foreach (Task task in tasks) await task;
} }

View file

@ -9,62 +9,64 @@ using Gameloop.Vdf.Linq;
using Microsoft.Win32; using Microsoft.Win32;
namespace CreamInstaller.Classes; namespace CreamInstaller.Steam;
internal static class SteamLibrary internal static class SteamLibrary
{ {
internal static string paradoxLauncherInstallPath = null; private static string installPath = null;
internal static string ParadoxLauncherInstallPath internal static string InstallPath
{ {
get get
{ {
paradoxLauncherInstallPath ??= Registry.GetValue(@"HKEY_CURRENT_USER\Software\Paradox Interactive\Paradox Launcher v2", "LauncherInstallation", null) as string; installPath ??= Registry.GetValue(@"HKEY_LOCAL_MACHINE\SOFTWARE\Valve\Steam", "InstallPath", null) as string;
return paradoxLauncherInstallPath; installPath ??= Registry.GetValue(@"HKEY_LOCAL_MACHINE\SOFTWARE\WOW6432Node\Valve\Steam", "InstallPath", null) as string;
return installPath;
} }
} }
internal static string steamInstallPath = null; internal static async Task<List<Tuple<string, string, string, int, string>>> GetGames() => await Task.Run(async () =>
internal static string SteamInstallPath
{ {
get List<Tuple<string, string, string, int, string>> games = new();
List<string> gameLibraryDirectories = await GetLibraryDirectories();
foreach (string libraryDirectory in gameLibraryDirectories)
{ {
steamInstallPath ??= Registry.GetValue(@"HKEY_LOCAL_MACHINE\Software\Valve\Steam", "InstallPath", null) as string; List<Tuple<string, string, string, int, string>> directoryGames = await GetGamesFromLibraryDirectory(libraryDirectory);
steamInstallPath ??= Registry.GetValue(@"HKEY_LOCAL_MACHINE\Software\Wow6432Node\Valve\Steam", "InstallPath", null) as string; if (directoryGames is not null)
return steamInstallPath; foreach (Tuple<string, string, string, int, string> game in directoryGames)
if (!games.Any(_game => _game.Item1 == game.Item1))
games.Add(game);
} }
} return games;
internal static async Task<List<string>> GetLibraryDirectories() => await Task.Run(() =>
{
List<string> gameDirectories = new();
if (Program.Canceled) return gameDirectories;
string steamInstallPath = SteamInstallPath;
if (steamInstallPath != null && Directory.Exists(steamInstallPath))
{
string libraryFolder = steamInstallPath + @"\steamapps";
if (Directory.Exists(libraryFolder))
{
gameDirectories.Add(libraryFolder);
string libraryFolders = libraryFolder + @"\libraryfolders.vdf";
if (File.Exists(libraryFolders) && ValveDataFile.TryDeserialize(File.ReadAllText(libraryFolders, Encoding.UTF8), out VProperty result))
{
foreach (VProperty property in result.Value)
if (int.TryParse(property.Key, out int _))
{
string path = property.Value.GetChild("path")?.ToString();
if (string.IsNullOrWhiteSpace(path)) continue;
path += @"\steamapps";
if (Directory.Exists(path) && !gameDirectories.Contains(path)) gameDirectories.Add(path);
}
}
}
}
return gameDirectories;
}); });
internal static async Task<List<Tuple<int, string, string, int, string>>> GetGamesFromLibraryDirectory(string libraryDirectory) => await Task.Run(() => internal static async Task<List<string>> GetDllDirectoriesFromGameDirectory(string gameDirectory) => await Task.Run(async () =>
{ {
List<Tuple<int, string, string, int, string>> games = new(); List<string> dllDirectories = new();
if (Program.Canceled || !Directory.Exists(gameDirectory)) return null;
gameDirectory.GetCreamApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi);
if (File.Exists(api)
|| File.Exists(api_o)
|| File.Exists(api64)
|| File.Exists(api64_o)
|| File.Exists(cApi))
dllDirectories.Add(gameDirectory);
string[] directories = Directory.GetDirectories(gameDirectory);
foreach (string _directory in directories)
{
if (Program.Canceled) return null;
try
{
List<string> moreDllDirectories = await GetDllDirectoriesFromGameDirectory(_directory);
if (moreDllDirectories is not null) dllDirectories.AddRange(moreDllDirectories);
}
catch { }
}
return !dllDirectories.Any() ? null : dllDirectories;
});
internal static async Task<List<Tuple<string, string, string, int, string>>> GetGamesFromLibraryDirectory(string libraryDirectory) => await Task.Run(() =>
{
List<Tuple<string, string, string, int, string>> games = new();
if (Program.Canceled || !Directory.Exists(libraryDirectory)) return null; if (Program.Canceled || !Directory.Exists(libraryDirectory)) return null;
string[] files = Directory.GetFiles(libraryDirectory); string[] files = Directory.GetFiles(libraryDirectory);
foreach (string file in files) foreach (string file in files)
@ -86,34 +88,35 @@ internal static class SteamLibrary
string gameDirectory = libraryDirectory + @"\common\" + installdir; string gameDirectory = libraryDirectory + @"\common\" + installdir;
if (!int.TryParse(appId, out int appIdInt)) continue; if (!int.TryParse(appId, out int appIdInt)) continue;
if (!int.TryParse(buildId, out int buildIdInt)) continue; if (!int.TryParse(buildId, out int buildIdInt)) continue;
games.Add(new(appIdInt, name, branch, buildIdInt, gameDirectory)); games.Add(new(appId, name, branch, buildIdInt, gameDirectory));
} }
} }
return !games.Any() ? null : games; return !games.Any() ? null : games;
}); });
internal static async Task<List<string>> GetDllDirectoriesFromGameDirectory(string gameDirectory) => await Task.Run(async () => internal static async Task<List<string>> GetLibraryDirectories() => await Task.Run(() =>
{ {
List<string> dllDirectories = new(); List<string> gameDirectories = new();
if (Program.Canceled || !Directory.Exists(gameDirectory)) return null; if (Program.Canceled) return gameDirectories;
gameDirectory.GetApiComponents(out string api, out string api_o, out string api64, out string api64_o, out string cApi); string steamInstallPath = InstallPath;
if (File.Exists(api) if (steamInstallPath != null && Directory.Exists(steamInstallPath))
|| File.Exists(api_o)
|| File.Exists(api64)
|| File.Exists(api64_o)
|| File.Exists(cApi))
dllDirectories.Add(gameDirectory);
string[] directories = Directory.GetDirectories(gameDirectory);
foreach (string _directory in directories)
{ {
if (Program.Canceled) return null; string libraryFolder = steamInstallPath + @"\steamapps";
try if (Directory.Exists(libraryFolder))
{ {
List<string> moreDllDirectories = await GetDllDirectoriesFromGameDirectory(_directory); gameDirectories.Add(libraryFolder);
if (moreDllDirectories is not null) dllDirectories.AddRange(moreDllDirectories); string libraryFolders = libraryFolder + @"\libraryfolders.vdf";
if (File.Exists(libraryFolders) && ValveDataFile.TryDeserialize(File.ReadAllText(libraryFolders, Encoding.UTF8), out VProperty result))
foreach (VProperty property in result.Value)
if (int.TryParse(property.Key, out int _))
{
string path = property.Value.GetChild("path")?.ToString();
if (string.IsNullOrWhiteSpace(path)) continue;
path += @"\steamapps";
if (Directory.Exists(path) && !gameDirectories.Contains(path)) gameDirectories.Add(path);
} }
catch { }
} }
return !dllDirectories.Any() ? null : dllDirectories; }
return gameDirectories;
}); });
} }

View file

@ -0,0 +1,24 @@
using System.Collections.Generic;
using System.Threading.Tasks;
using CreamInstaller.Utility;
using HtmlAgilityPack;
namespace CreamInstaller.Steam;
internal static class SteamStore
{
internal static async Task ParseDlcAppIds(string appId, List<string> dlcIds)
{
// currently this is only really needed to get DLC that release without changing game buildid (very rare)
// it also finds things which aren't really connected to the game itself, and thus not needed (usually soundtracks, collections, packs, etc.)
HtmlNodeCollection nodes = await HttpClientManager.GetDocumentNodes(
$"https://store.steampowered.com/dlc/{appId}",
"//div[@class='recommendation']/div/a");
if (nodes is not null)
foreach (HtmlNode node in nodes)
if (int.TryParse(node.Attributes?["data-ds-appid"]?.Value, out int dlcAppId) && dlcAppId > 0 && !dlcIds.Contains("" + dlcAppId))
dlcIds.Add("" + dlcAppId);
}
}

View file

@ -2,7 +2,7 @@
using Gameloop.Vdf; using Gameloop.Vdf;
using Gameloop.Vdf.Linq; using Gameloop.Vdf.Linq;
namespace CreamInstaller.Classes; namespace CreamInstaller.Steam;
internal static class ValveDataFile internal static class ValveDataFile
{ {

View file

@ -1,6 +1,6 @@
using System.Diagnostics; using System.Diagnostics;
namespace CreamInstaller.Classes; namespace CreamInstaller.Utility;
internal static class Diagnostics internal static class Diagnostics
{ {

View file

@ -1,7 +1,7 @@
using System; using System;
using System.Windows.Forms; using System.Windows.Forms;
namespace CreamInstaller.Classes; namespace CreamInstaller.Utility;
internal static class ExceptionHandler internal static class ExceptionHandler
{ {

View file

@ -0,0 +1,50 @@
using System;
using System.Drawing;
using System.IO;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;
using HtmlAgilityPack;
namespace CreamInstaller.Utility;
internal static class HttpClientManager
{
internal static HttpClient HttpClient;
internal static void Setup()
{
HttpClient = new();
HttpClient.DefaultRequestHeaders.Add("user-agent", $"CreamInstaller-{Environment.MachineName}_{Environment.UserDomainName}_{Environment.UserName}");
}
internal static async Task<HtmlDocument> Get(string url)
{
try
{
using HttpRequestMessage request = new(HttpMethod.Get, url);
using HttpResponseMessage response = await HttpClient.SendAsync(request, HttpCompletionOption.ResponseHeadersRead);
response.EnsureSuccessStatusCode();
using Stream stream = await response.Content.ReadAsStreamAsync();
using StreamReader reader = new(stream, Encoding.UTF8);
HtmlDocument document = new();
document.LoadHtml(reader.ReadToEnd());
return document;
}
catch { return null; }
}
internal static async Task<HtmlNodeCollection> GetDocumentNodes(string url, string xpath) => (await Get(url))?.DocumentNode?.SelectNodes(xpath);
internal static async Task<Image> GetImageFromUrl(string url)
{
try
{
return new Bitmap(await HttpClient.GetStreamAsync(url));
}
catch { }
return null;
}
internal static void Dispose() => HttpClient.Dispose();
}

View file

@ -3,7 +3,7 @@ using System.Drawing;
using System.IO; using System.IO;
using System.Threading.Tasks; using System.Threading.Tasks;
namespace CreamInstaller.Classes; namespace CreamInstaller.Utility;
internal static class IconGrabber internal static class IconGrabber
{ {
@ -14,8 +14,8 @@ internal static class IconGrabber
} }
private static readonly string SteamAppImagesPath = "https://cdn.cloudflare.steamstatic.com/steamcommunity/public/images/apps/"; private static readonly string SteamAppImagesPath = "https://cdn.cloudflare.steamstatic.com/steamcommunity/public/images/apps/";
internal static async Task<Image> GetSteamIcon(int steamAppId, string iconStaticId) => await HttpClientManager.GetImageFromUrl(SteamAppImagesPath + $"/{steamAppId}/{iconStaticId}.jpg"); internal static async Task<Image> GetSteamIcon(string steamAppId, string iconStaticId) => await HttpClientManager.GetImageFromUrl(SteamAppImagesPath + $"/{steamAppId}/{iconStaticId}.jpg");
internal static async Task<Image> GetSteamClientIcon(int steamAppId, string clientIconStaticId) => await HttpClientManager.GetImageFromUrl(SteamAppImagesPath + $"/{steamAppId}/{clientIconStaticId}.ico"); internal static async Task<Image> GetSteamClientIcon(string steamAppId, string clientIconStaticId) => await HttpClientManager.GetImageFromUrl(SteamAppImagesPath + $"/{steamAppId}/{clientIconStaticId}.ico");
internal static Image GetFileIconImage(string path) => File.Exists(path) ? Icon.ExtractAssociatedIcon(path).ToBitmap() : null; internal static Image GetFileIconImage(string path) => File.Exists(path) ? Icon.ExtractAssociatedIcon(path).ToBitmap() : null;

View file

@ -1,7 +1,7 @@
using System.Drawing; using System.Drawing;
using System.Windows.Forms; using System.Windows.Forms;
namespace CreamInstaller.Classes; namespace CreamInstaller.Utility;
internal static class InstallationLog internal static class InstallationLog
{ {