diff --git a/.gitignore b/.gitignore
index 0e223e7..0f34f01 100644
--- a/.gitignore
+++ b/.gitignore
@@ -553,6 +553,5 @@ MigrationBackup/
# End of https://www.toptal.com/developers/gitignore/api/visualstudio,jetbrains,windows,visualstudiocode,rider,dotnetcore
-**/Secrets.cs
-/MainWindow.xaml.BACKUP
-/MainWindow.xaml.cs.BACKUP
+ConsoleApp1/
+/Utils/Secrets.cs
diff --git a/.gitmodules b/.gitmodules
deleted file mode 100644
index bd149f0..0000000
--- a/.gitmodules
+++ /dev/null
@@ -1,3 +0,0 @@
-[submodule "SteamStorefrontAPI"]
- path = SteamStorefrontAPI
- url = https://git.jeddunk.xyz/jeddunk/SteamStorefrontAPI.git
diff --git a/App.xaml b/App.xaml
new file mode 100644
index 0000000..cf27f85
--- /dev/null
+++ b/App.xaml
@@ -0,0 +1,8 @@
+
+
+
+
+
diff --git a/App.xaml.cs b/App.xaml.cs
new file mode 100644
index 0000000..0c71a85
--- /dev/null
+++ b/App.xaml.cs
@@ -0,0 +1,9 @@
+namespace auto_creamapi
+{
+ ///
+ /// Interaction logic for App.xaml
+ ///
+ public partial class App
+ {
+ }
+}
diff --git a/DownloadWindow.xaml b/DownloadWindow.xaml
new file mode 100644
index 0000000..aff7206
--- /dev/null
+++ b/DownloadWindow.xaml
@@ -0,0 +1,21 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/DownloadWindow.xaml.cs b/DownloadWindow.xaml.cs
new file mode 100644
index 0000000..4e90436
--- /dev/null
+++ b/DownloadWindow.xaml.cs
@@ -0,0 +1,32 @@
+using System;
+using System.Collections.Generic;
+using System.Text;
+using System.Windows;
+using System.Windows.Controls;
+using System.Windows.Data;
+using System.Windows.Documents;
+using System.Windows.Input;
+using System.Windows.Media;
+using System.Windows.Media.Imaging;
+using System.Windows.Shapes;
+using auto_creamapi.Utils;
+
+namespace auto_creamapi
+{
+ ///
+ /// Interaction logic for DownloadWindow.xaml
+ ///
+ public partial class DownloadWindow
+ {
+ public DownloadWindow()
+ {
+ WindowStartupLocation = WindowStartupLocation.CenterScreen;
+ InitializeComponent();
+ }
+
+ private void ProgressBar_OnValueChanged(object sender, RoutedPropertyChangedEventArgs e)
+ {
+ MyLogger.Log.Information(ProgressBar.Value.ToString("N"));
+ }
+ }
+}
diff --git a/MainWindow.xaml b/MainWindow.xaml
new file mode 100644
index 0000000..3ca06e5
--- /dev/null
+++ b/MainWindow.xaml
@@ -0,0 +1,54 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/MainWindow.xaml.cs b/MainWindow.xaml.cs
new file mode 100644
index 0000000..0049424
--- /dev/null
+++ b/MainWindow.xaml.cs
@@ -0,0 +1,212 @@
+using System;
+using System.IO;
+using System.Linq;
+using System.Windows;
+using System.Windows.Input;
+using auto_creamapi.Model;
+using auto_creamapi.Utils;
+using Microsoft.Win32;
+
+namespace auto_creamapi
+{
+ ///
+ /// Interaction logic for MainWindow.xaml
+ ///
+ public partial class MainWindow
+ {
+ private const string DefaultLangSelection = "english";
+ private static CacheModel _cacheModel;
+ private static CreamConfigModel _configModel;
+ private static CreamDllModel _dllModel;
+
+ public MainWindow()
+ {
+ _cacheModel = CacheModel.Instance;
+ _configModel = CreamConfigModel.Instance;
+ _dllModel = CreamDllModel.Instance;
+ InitializeComponent();
+ _cacheModel.Languages.ForEach(x => Lang.Items.Add(x));
+ Lang.SelectedItem = DefaultLangSelection;
+ SteamDb.IsChecked = true;
+ }
+
+ ///
+ /// Looks up the AppID of the specified game.
+ ///
+ ///
+ ///
+ private void Search_Click(object sender, RoutedEventArgs e)
+ {
+ var app = _cacheModel.GetAppByName(Game.Text);
+ if (app != null)
+ {
+ AppId.Text = app.AppId.ToString();
+ }
+ else
+ {
+ var listOfAppsByName = _cacheModel.GetListOfAppsByName(Game.Text);
+ var searchWindow = new SearchResultWindow(listOfAppsByName);
+ searchWindow.Show();
+ }
+ }
+
+ private void DllPath_MouseDoubleClick(object sender, MouseButtonEventArgs e)
+ {
+ MyOpenFile();
+ }
+
+ ///
+ /// Opens a file chooser to select the path to steam_api(64).dll.
+ ///
+ ///
+ ///
+ private void OpenFile_Click(object sender, RoutedEventArgs e)
+ {
+ MyOpenFile();
+ }
+
+ private void MyOpenFile()
+ {
+ var dialog = new OpenFileDialog
+ {
+ Filter = "SteamAPI DLL|steam_api.dll;steam_api64.dll|" +
+ "All files (*.*)|*.*",
+ Multiselect = false,
+ Title = "Select SteamAPI DLL..."
+ };
+ if (dialog.ShowDialog() == true)
+ {
+ //Console.WriteLine(dialog.FileName);
+ var filePath = dialog.FileName;
+ DllPath.Text = filePath;
+ var dirPath = Path.GetDirectoryName(filePath);
+ if (dirPath != null)
+ {
+ _configModel.ReadFile(Path.Combine(dirPath, "cream_api.ini"));
+ ResetFormData();
+ _dllModel.TargetPath = dirPath;
+ _dllModel.CheckExistence();
+ CheckExistance();
+ }
+ }
+ }
+
+ ///
+ /// Gets a list of DLCs for the specified AppID.
+ ///
+ ///
+ ///
+ private async void GetListOfDlc_Click(object sender, RoutedEventArgs e)
+ {
+ if (int.TryParse(AppId.Text, out var appId))
+ {
+ if (appId > 0)
+ {
+ var app = new POCOs.App() {AppId = appId, Name = Game.Text};
+ var listOfDlc = await _cacheModel.GetListOfDlc(app,
+ SteamDb.IsChecked != null && (bool) SteamDb.IsChecked);
+ var result = "";
+ listOfDlc.Sort((app1, app2) => app1.AppId.CompareTo(app2.AppId));
+ listOfDlc.ForEach(x => result += $"{x.AppId}={x.Name}\n");
+ ListOfDlcs.Text = result;
+ }
+ else
+ {
+ MyLogger.Log.Error($"GetListOfDlc: Invalid AppID {appId}");
+ }
+ }
+ }
+
+ ///
+ /// Saves form data to cream_api.ini.
+ ///
+ ///
+ ///
+ private void Save_Click(object sender, RoutedEventArgs e)
+ {
+ _configModel.SetConfigData(
+ Convert.ToInt32(AppId.Text),
+ Lang.SelectedItem.ToString(),
+ UnlockAll.IsChecked != null && (bool) UnlockAll.IsChecked,
+ ExtraProtection.IsChecked != null && (bool) ExtraProtection.IsChecked,
+ ForceOffline.IsChecked != null && (bool) ForceOffline.IsChecked,
+ ListOfDlcs.Text
+ );
+ _configModel.SaveFile();
+ _dllModel.Save();
+ CheckExistance();
+ }
+
+ ///
+ /// Resets form data.
+ ///
+ ///
+ ///
+ private void Reset_Click(object sender, RoutedEventArgs e)
+ {
+ ResetFormData();
+ CheckExistance();
+ }
+
+ ///
+ /// Gets app name on id change
+ ///
+ ///
+ ///
+ private void AppId_OnTextChanged(object sender, KeyEventArgs keyEventArgs)
+ {
+ SetNameById();
+ }
+
+ private void SetNameById()
+ {
+ if (int.TryParse(AppId.Text, out var appId))
+ {
+ if (appId > 0)
+ {
+ var app = _cacheModel.GetAppById(appId);
+ if (app != null)
+ {
+ Game.Text = app.Name;
+ }
+ else
+ {
+ MyLogger.Log.Error($"No app found for ID {appId}");
+ }
+ }
+ else
+ {
+ MyLogger.Log.Error($"SetNameById: Invalid AppID {appId}");
+ }
+ }
+ }
+
+ private void ResetFormData()
+ {
+ AppId.Text = _configModel.Config.AppId.ToString();
+ Lang.SelectedItem = _configModel.Config.Language;
+ UnlockAll.IsChecked = _configModel.Config.UnlockAll; // public bool UnlockAll;
+ ExtraProtection.IsChecked = _configModel.Config.ExtraProtection; // public bool ExtraProtection;
+ ForceOffline.IsChecked = _configModel.Config.ForceOffline; // public bool ForceOffline;
+ // public Dictionary DlcList;
+ var dlcListString = "";
+ if (_configModel.Config.DlcList.Count > 0)
+ {
+ foreach (var (id, name) in _configModel.Config.DlcList)
+ {
+ dlcListString += $"{id}={name},\n";
+ }
+ }
+
+ ListOfDlcs.Text = dlcListString;
+ SetNameById();
+ }
+
+ private void CheckExistance()
+ {
+
+ creamApiApplied.IsChecked = _dllModel.CreamApiApplied();
+ configExists.IsChecked = _configModel.ConfigExists();
+ }
+ }
+}
diff --git a/Model/CacheModel.cs b/Model/CacheModel.cs
new file mode 100644
index 0000000..b29febb
--- /dev/null
+++ b/Model/CacheModel.cs
@@ -0,0 +1,216 @@
+using System;
+using System.Collections.Generic;
+using System.IO;
+using System.Linq;
+using System.Net.Http;
+using System.Text;
+using System.Text.Json;
+using System.Threading.Tasks;
+using AngleSharp.Dom;
+using AngleSharp.Html.Parser;
+using auto_creamapi.POCOs;
+using auto_creamapi.Utils;
+using NinjaNye.SearchExtensions;
+using NinjaNye.SearchExtensions.Models;
+using SteamStorefrontAPI;
+
+namespace auto_creamapi.Model
+{
+ public class CacheModel
+ {
+ private const string CachePath = "steamapps.json";
+ private const string SteamUri = "https://api.steampowered.com/ISteamApps/GetAppList/v2/";
+ private const string UserAgent =
+ "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) " +
+ "Chrome/87.0.4280.88 Safari/537.36";
+ private static List _cache = new List();
+
+ public readonly List Languages = new List(new[]
+ {
+ "arabic",
+ "bulgarian",
+ "schinese",
+ "tchinese",
+ "czech",
+ "danish",
+ "dutch",
+ "english",
+ "finnish",
+ "french",
+ "german",
+ "greek",
+ "hungarian",
+ "italian",
+ "japanese",
+ "koreana",
+ "norwegian",
+ "polish",
+ "portuguese",
+ "brazilian",
+ "romanian",
+ "russian",
+ "spanish",
+ "latam",
+ "swedish",
+ "thai",
+ "turkish",
+ "ukrainian",
+ "vietnamese"
+ });
+
+ private static readonly Lazy Lazy =
+ new Lazy(() => new CacheModel());
+
+ public static CacheModel Instance => Lazy.Value;
+
+ private CacheModel()
+ {
+ UpdateCache();
+ }
+
+ private static void UpdateCache()
+ {
+ var updateNeeded = DateTime.Now.Subtract(File.GetCreationTimeUtc(CachePath)).TotalDays >= 1;
+ string cacheString;
+ if (updateNeeded)
+ {
+ MyLogger.Log.Information("Updating cache...");
+ var client = new HttpClient();
+ var httpCall = client.GetAsync(SteamUri);
+ var response = httpCall.Result;
+ var readAsStringAsync = response.Content.ReadAsStringAsync();
+ var responseBody = readAsStringAsync.Result;
+
+ /*var writeAllTextAsync = File.WriteAllTextAsync(CachePath, responseBody, Encoding.UTF8);
+ writeAllTextAsync.RunSynchronously();*/
+ File.WriteAllText(CachePath, responseBody, Encoding.UTF8);
+ cacheString = responseBody;
+ }
+ else
+ {
+ MyLogger.Log.Information("Cache already up to date!");
+ cacheString = File.ReadAllText(CachePath);
+ }
+
+ var steamApps = JsonSerializer.Deserialize(cacheString);
+ _cache = steamApps.AppList.Apps;
+ MyLogger.Log.Information("Loaded cache into memory!");
+ }
+
+ public EnumerableStringSearch GetListOfAppsByName(string name)
+ {
+ var listOfAppsByName = _cache.Search(x => x.Name)
+ .SetCulture(StringComparison.OrdinalIgnoreCase)
+ .ContainingAll(name.Split(' '));
+ return listOfAppsByName;
+ }
+
+ public POCOs.App GetAppByName(string name)
+ {
+ MyLogger.Log.Information($"Trying to get app {name}");
+ var app = _cache.Find(x => x.Name.ToLower().Equals(name.ToLower()));
+ if (app != null) MyLogger.Log.Information($"Successfully got app {app}");
+ return app;
+ }
+
+ public POCOs.App GetAppById(int appid)
+ {
+ MyLogger.Log.Information($"Trying to get app with ID {appid}");
+ var app = _cache.Find(x => x.AppId.Equals(appid));
+ if (app != null) MyLogger.Log.Information($"Successfully got app {app}");
+ return app;
+ }
+
+ public async Task> GetListOfDlc(POCOs.App app, bool useSteamDb)
+ {
+ MyLogger.Log.Information("Get DLC");
+ var dlcList = new List();
+ if (app != null)
+ {
+ var task = AppDetails.GetAsync(app.AppId);
+ var steamApp = await task;
+ steamApp?.DLC.ForEach(x =>
+ {
+ var result = _cache.Find(y => y.AppId.Equals(x)) ??
+ new POCOs.App {AppId = x, Name = $"Unknown DLC {x}"};
+ dlcList.Add(result);
+ });
+
+ dlcList.ForEach(x => MyLogger.Log.Debug($"{x.AppId}={x.Name}"));
+ MyLogger.Log.Information("Got DLC successfully...");
+
+ // Get DLC from SteamDB
+ // Get Cloudflare cookie
+ // Scrape and parse HTML page
+ // Add missing to DLC list
+ if (useSteamDb)
+ {
+ var steamDbUri = new Uri($"https://steamdb.info/app/{app.AppId}/dlc/");
+
+ /* var handler = new ClearanceHandler();
+
+ var client = new HttpClient(handler);
+
+ var content = client.GetStringAsync(steamDbUri).Result;
+ MyLogger.Log.Debug(content); */
+
+ var client = new HttpClient();
+ client.DefaultRequestHeaders.UserAgent.ParseAdd(UserAgent);
+
+ MyLogger.Log.Information("Get SteamDB App");
+ var httpCall = client.GetAsync(steamDbUri);
+ var response = await httpCall;
+ MyLogger.Log.Debug(httpCall.Status.ToString());
+ MyLogger.Log.Debug(response.EnsureSuccessStatusCode().ToString());
+
+ var readAsStringAsync = response.Content.ReadAsStringAsync();
+ var responseBody = await readAsStringAsync;
+ MyLogger.Log.Debug(readAsStringAsync.Status.ToString());
+
+ var parser = new HtmlParser();
+ var doc = parser.ParseDocument(responseBody);
+ // Console.WriteLine(doc.DocumentElement.OuterHtml);
+
+ var query1 = doc.QuerySelector("#dlc");
+ if (query1 != null)
+ {
+ var query2 = query1.QuerySelectorAll(".app");
+ foreach (var element in query2)
+ {
+ var dlcId = element.GetAttribute("data-appid");
+ var dlcName = $"Unknown DLC {dlcId}";
+ var query3 = element.QuerySelectorAll("td");
+ if (query3 != null)
+ {
+ dlcName = query3[1].Text().Replace("\n", "").Trim();
+ }
+
+ var dlcApp = new POCOs.App {AppId = Convert.ToInt32(dlcId), Name = dlcName};
+ var i = dlcList.FindIndex(x => x.CompareId(dlcApp));
+ if (i > -1)
+ {
+ if (dlcList[i].Name.Contains("Unknown DLC")) dlcList[i] = dlcApp;
+ }
+ else
+ {
+ dlcList.Add(dlcApp);
+ }
+ }
+ dlcList.ForEach(x => MyLogger.Log.Debug($"{x.AppId}={x.Name}"));
+ MyLogger.Log.Information("Got DLC from SteamDB successfully...");
+ }
+ else
+ {
+ MyLogger.Log.Error("Could not get DLC from SteamDB1");
+ }
+ }
+ }
+ else
+ {
+ MyLogger.Log.Error($"Could not find game: {app}");
+ }
+
+ return dlcList;
+ }
+ }
+}
\ No newline at end of file
diff --git a/Model/CreamConfigModel.cs b/Model/CreamConfigModel.cs
new file mode 100644
index 0000000..d326574
--- /dev/null
+++ b/Model/CreamConfigModel.cs
@@ -0,0 +1,184 @@
+using System;
+using System.Collections.Generic;
+using System.IO;
+using System.Text;
+using System.Text.RegularExpressions;
+using auto_creamapi.Utils;
+using IniParser;
+using IniParser.Model;
+
+namespace auto_creamapi.Model
+{
+ public sealed class CreamConfigModel
+ {
+
+ // ReSharper disable once MemberCanBePrivate.Global
+ public struct CreamConfig
+ {
+ public int AppId;
+ public string Language;
+ public bool UnlockAll;
+ public bool ExtraProtection;
+ public bool ForceOffline;
+ public Dictionary DlcList;
+ }
+
+ private CreamConfig _config;
+
+ public CreamConfig Config => _config;
+
+ private static readonly Lazy Lazy =
+ new Lazy(() => new CreamConfigModel());
+
+ public static CreamConfigModel Instance => Lazy.Value;
+
+ // ReSharper disable once MemberCanBePrivate.Global
+ // ReSharper disable once UnusedAutoPropertyAccessor.Global
+ private string _configFilePath;
+
+ private CreamConfigModel()
+ {
+ _config.DlcList = new Dictionary();
+ SetConfigData();
+ }
+
+ public void ReadFile(string configFilePath)
+ {
+ _configFilePath = configFilePath;
+ if (File.Exists(configFilePath)) {
+ MyLogger.Log.Information($"Config file found @ {configFilePath}, parsing...");
+ var parser = new FileIniDataParser();
+ var data = parser.ReadFile(_configFilePath, Encoding.UTF8);
+
+ SetConfigData(); // clear previous config data
+ _config.AppId = Convert.ToInt32(data["steam"]["appid"]);
+ _config.Language = data["steam"]["language"];
+ _config.UnlockAll = Convert.ToBoolean(data["steam"]["unlockall"]);
+ _config.ExtraProtection = Convert.ToBoolean(data["steam"]["extraprotection"]);
+ _config.ForceOffline = Convert.ToBoolean(data["steam"]["forceoffline"]);
+
+ var dlcCollection = data["dlc"];
+ foreach (var item in dlcCollection)
+ {
+ _config.DlcList.Add(int.Parse(item.KeyName), item.Value);
+ }
+ }
+ else
+ {
+ MyLogger.Log.Information($"Config file does not exist @ {configFilePath}, skipping...");
+ SetConfigData();
+ }
+ }
+
+ public void SaveFile()
+ {
+ var parser = new FileIniDataParser();
+ var data = new IniData();
+
+ data["steam"]["appid"] = _config.AppId.ToString();
+ data["steam"]["language"] = _config.Language;
+ data["steam"]["unlockall"] = _config.UnlockAll.ToString();
+ data["steam"]["extraprotection"] = _config.ExtraProtection.ToString();
+ data["steam"]["forceoffline"] = _config.ForceOffline.ToString();
+
+ data.Sections.AddSection("dlc");
+ foreach (var (key, value) in _config.DlcList)
+ {
+ data["dlc"].AddKey(key.ToString(), value);
+ }
+
+ parser.WriteFile(_configFilePath, data, Encoding.UTF8);
+ }
+
+ public void SetConfigData()
+ {
+ _config.AppId = 0;
+ _config.Language = "";
+ _config.UnlockAll = false;
+ _config.ExtraProtection = false;
+ _config.ForceOffline = false;
+ _config.DlcList.Clear();
+ }
+
+ public void SetConfigData(int appId,
+ string language,
+ bool unlockAll,
+ bool extraProtection,
+ bool forceOffline,
+ string dlcList)
+ {
+ _config.AppId = appId;
+ _config.Language = language;
+ _config.UnlockAll = unlockAll;
+ _config.ExtraProtection = extraProtection;
+ _config.ForceOffline = forceOffline;
+
+ SetDlcFromString(dlcList);
+ }
+
+ /*private void SetConfigData(int appId,
+ string language,
+ bool unlockAll,
+ bool extraProtection,
+ bool forceOffline,
+ List dlcList)
+ {
+ _config.AppId = appId;
+ _config.Language = language;
+ _config.UnlockAll = unlockAll;
+ _config.ExtraProtection = extraProtection;
+ _config.ForceOffline = forceOffline;
+
+ SetDlcFromAppList(dlcList);
+ }*/
+
+ private void SetDlcFromString(string dlcList)
+ {
+ _config.DlcList.Clear();
+ var expression = new Regex(@"(?.*) *= *(?.*)");
+ using var reader = new StringReader(dlcList);
+ string line;
+ while ((line = reader.ReadLine()) != null)
+ {
+ var match = expression.Match(line);
+ if (match.Success)
+ {
+ _config.DlcList.Add(int.Parse(match.Groups["id"].Value), match.Groups["name"].Value);
+ }
+ }
+ }
+
+ /*private void SetDlcFromAppList(List dlcList)
+ {
+ _config.DlcList.Clear();
+ dlcList.ForEach(x => _config.DlcList.Add(x.AppId, x.Name));
+ }*/
+
+ public override string ToString()
+ {
+ var str = $"INI file: {_configFilePath}, " +
+ $"AppID: {_config.AppId}, " +
+ $"Language: {_config.Language}, " +
+ $"UnlockAll: {_config.UnlockAll}, " +
+ $"ExtraProtection: {_config.ExtraProtection}, " +
+ $"ForceOffline: {_config.ForceOffline}, " +
+ $"DLC ({_config.DlcList.Count}):\n[\n";
+ if (_config.DlcList.Count > 0)
+ {
+ foreach (var (key, value) in _config.DlcList)
+ {
+ str += $" {key}={value},\n";
+ }
+
+ }
+ str += "]";
+
+ return str;
+ }
+
+ public bool ConfigExists()
+ {
+ return File.Exists(_configFilePath);
+ }
+ }
+}
diff --git a/Model/CreamDllModel.cs b/Model/CreamDllModel.cs
new file mode 100644
index 0000000..2d38cfe
--- /dev/null
+++ b/Model/CreamDllModel.cs
@@ -0,0 +1,244 @@
+using System;
+using System.Collections.Generic;
+using System.IO;
+using System.Linq;
+using System.Net;
+using System.Net.Http;
+using System.Security.Cryptography;
+using System.Text.RegularExpressions;
+using System.Threading.Tasks;
+using System.Windows.Threading;
+using auto_creamapi.Utils;
+using SharpCompress.Archives;
+using SharpCompress.Common;
+using SharpCompress.Readers;
+using HttpProgress;
+
+namespace auto_creamapi.Model
+{
+ public class CreamDllModel
+ {
+ private struct CreamDll
+ {
+ public readonly string Filename;
+ public readonly string OrigFilename;
+ public readonly string Hash;
+
+ public CreamDll(string filename, string origFilename)
+ {
+ Filename = filename;
+ OrigFilename = origFilename;
+ Hash = "";
+
+ using var md5 = MD5.Create();
+ using var stream = File.OpenRead(Filename);
+ Hash = BitConverter
+ .ToString(md5.ComputeHash(stream))
+ .Replace("-", string.Empty);
+ }
+ }
+
+ private static readonly Lazy Lazy =
+ new Lazy(() => new CreamDllModel());
+
+ public static CreamDllModel Instance => Lazy.Value;
+ public string TargetPath { get; set; }
+
+ private readonly Dictionary _creamDlls = new Dictionary();
+ private static readonly string HashPath = Path.Combine(Directory.GetCurrentDirectory(), "cream_api.md5");
+ private const string X86Arch = "x86";
+ private const string X64Arch = "x64";
+
+ private bool _x86Exists;
+ private bool _x64Exists;
+
+ private CreamDllModel()
+ {
+ if (!(File.Exists("steam_api.dll") && File.Exists("steam_api64.dll")))
+ {
+ MyLogger.Log.Information("Missing files, trying to download...");
+ new Action(async() => await DownloadDll(Secrets.ForumUsername, Secrets.ForumPassword))();
+ }
+ else
+ {
+ Init();
+ }
+ }
+
+ private void Init()
+ {
+ _creamDlls.Add(X86Arch, new CreamDll("steam_api.dll", "steam_api_o.dll"));
+ _creamDlls.Add(X64Arch, new CreamDll("steam_api64.dll", "steam_api64_o.dll"));
+
+ if (!File.Exists(HashPath))
+ {
+ MyLogger.Log.Information("Writing md5sum file...");
+ File.WriteAllLines(HashPath, new[]
+ {
+ $"{_creamDlls[X86Arch].Hash} {_creamDlls[X86Arch].Filename}",
+ $"{_creamDlls[X64Arch].Hash} {_creamDlls[X64Arch].Filename}"
+ });
+ }
+ }
+
+ private async Task DownloadDll(string username, string password)
+ {
+ var wnd = new DownloadWindow();
+ wnd.Show();
+ var container = new CookieContainer();
+ var handler = new HttpClientHandler {CookieContainer = container};
+ var client = new HttpClient(handler);
+ var formContent = new FormUrlEncodedContent(new[]
+ {
+ new KeyValuePair("username", username),
+ new KeyValuePair("password", password),
+ new KeyValuePair("redirect", "./ucp.php?mode=login"),
+ new KeyValuePair("login", "login")
+ });
+ var response1 = await client.PostAsync("https://cs.rin.ru/forum/ucp.php?mode=login", formContent);
+ MyLogger.Log.Debug($"Login Status Code: {response1.EnsureSuccessStatusCode().StatusCode.ToString()}");
+ var cookie = container.GetCookies(new Uri("https://cs.rin.ru/forum/ucp.php?mode=login"))
+ .FirstOrDefault(c => c.Name.Contains("_sid"));
+ MyLogger.Log.Debug($"Login Cookie: {cookie}");
+ var response2 = await client.GetAsync("https://cs.rin.ru/forum/viewtopic.php?t=70576");
+ MyLogger.Log.Debug(
+ $"Download Page Status Code: {response2.EnsureSuccessStatusCode().StatusCode.ToString()}");
+ var content = response2.Content.ReadAsStringAsync();
+ var contentResult = await content;
+
+ var expression =
+ new Regex(".*\\/download\\/file\\.php\\?id=.*)\">(?.*)<\\/a>.*");
+ using var reader = new StringReader(contentResult);
+ string line;
+ var archiveFileList = new Dictionary();
+ while ((line = await reader.ReadLineAsync()) != null)
+ {
+ var match = expression.Match(line);
+ // ReSharper disable once InvertIf
+ if (match.Success)
+ {
+ archiveFileList.Add(match.Groups["filename"].Value,
+ $"https://cs.rin.ru/forum{match.Groups["url"].Value}");
+ MyLogger.Log.Debug(archiveFileList.LastOrDefault().Key);
+ }
+ }
+
+ /*foreach (var (filename, url) in archiveFileList)
+ {
+ MyLogger.Log.Information($"Downloading file: {filename}");
+ var fileResponse = await client.GetAsync(url);
+ var download = fileResponse.Content.ReadAsByteArrayAsync();
+ await File.WriteAllBytesAsync(filename, await download);
+ }*/
+ MyLogger.Log.Debug("Choosing first element from list...");
+ var (filename, url) = archiveFileList.FirstOrDefault();
+ MyLogger.Log.Information("Start download...");
+ wnd.FilenameLabel.Content = filename;
+ /*var fileResponse = await client.GetAsync(url);
+ var download = fileResponse.Content.ReadAsByteArrayAsync();
+ await File.WriteAllBytesAsync(filename, await download);
+ MyLogger.Log.Information($"Download success? {download.IsCompletedSuccessfully}");*/
+ var progress = new Progress(
+ x =>
+ {
+ wnd.PercentLabel.Content = x.PercentComplete.ToString("P");
+ wnd.ProgressBar.Dispatcher.Invoke(() => wnd.ProgressBar.Value = x.PercentComplete,
+ DispatcherPriority.Background);
+ });
+ await using (var fileStream = File.OpenWrite(filename))
+ {
+ var task = client.GetAsync(url, fileStream, progress);
+ var response = await task;
+ /*if (task.IsCompletedSuccessfully)
+ {
+ wnd.PercentLabel.Content = "100,00%";
+ wnd.ProgressBar.Value = 1;
+ }*/
+ }
+
+ MyLogger.Log.Information("Start extraction...");
+ var options = new ReaderOptions {Password = "cs.rin.ru"};
+ var archive = ArchiveFactory.Open(filename, options);
+ var expression1 = new Regex(@"nonlog_build\\steam_api(?:64)?\.dll");
+ foreach (var entry in archive.Entries)
+ {
+ // ReSharper disable once InvertIf
+ if (!entry.IsDirectory && expression1.IsMatch(entry.Key))
+ {
+ MyLogger.Log.Debug(entry.Key);
+ entry.WriteToDirectory(Directory.GetCurrentDirectory(), new ExtractionOptions
+ {
+ ExtractFullPath = false,
+ Overwrite = true
+ });
+ }
+ }
+ MyLogger.Log.Information("Extraction done!");
+ wnd.Close();
+ Init();
+ }
+
+ public void Save()
+ {
+ if (_x86Exists) CopyDll(X86Arch);
+ if (_x64Exists) CopyDll(X64Arch);
+ }
+
+ private void CopyDll(string arch)
+ {
+ var sourceSteamApiDll = _creamDlls[arch].Filename;
+ var targetSteamApiDll = Path.Combine(TargetPath, _creamDlls[arch].Filename);
+ var targetSteamApiOrigDll = Path.Combine(TargetPath, _creamDlls[arch].OrigFilename);
+ var targetSteamApiDllBackup = Path.Combine(TargetPath, $"{_creamDlls[arch].Filename}.backup");
+ MyLogger.Log.Information($"Creating CreamAPI DLL @ {targetSteamApiDll}");
+ // Create backup of steam_api.dll
+ File.Copy(targetSteamApiDll, targetSteamApiDllBackup, true);
+ // Check if steam_api_o.dll already exists
+ // If missing rename original file
+ if (!File.Exists(targetSteamApiOrigDll))
+ File.Move(targetSteamApiDll, targetSteamApiOrigDll, true);
+ // Copy creamapi dll
+ File.Copy(sourceSteamApiDll, targetSteamApiDll, true);
+ }
+
+ public void CheckExistence()
+ {
+ var x86file = Path.Combine(TargetPath, "steam_api.dll");
+ var x64file = Path.Combine(TargetPath, "steam_api64.dll");
+ _x86Exists = File.Exists(x86file);
+ _x64Exists = File.Exists(x64file);
+ if (_x86Exists) MyLogger.Log.Information($"x86 SteamAPI DLL found: {x86file}");
+ if (_x64Exists) MyLogger.Log.Information($"x64 SteamAPI DLL found: {x64file}");
+ }
+
+ public bool CreamApiApplied(string arch)
+ {
+ bool a = File.Exists(Path.Combine(TargetPath, _creamDlls[arch].OrigFilename));
+ bool b = GetHash(Path.Combine(TargetPath, _creamDlls[arch].Filename)).Equals(_creamDlls[arch].Hash);
+ return a & b;
+ }
+
+ public bool CreamApiApplied()
+ {
+ bool a = CreamApiApplied("x86");
+ bool b = CreamApiApplied("x64");
+ return a | b;
+ }
+
+ private string GetHash(string filename)
+ {
+ if (File.Exists(filename))
+ {
+ using var md5 = MD5.Create();
+ using var stream = File.OpenRead(filename);
+ return BitConverter
+ .ToString(md5.ComputeHash(stream))
+ .Replace("-", string.Empty);
+ }
+ else
+ {
+ return "";
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/POCOs/SteamAppPOCOs.cs b/POCOs/SteamAppPOCOs.cs
new file mode 100644
index 0000000..ed02c37
--- /dev/null
+++ b/POCOs/SteamAppPOCOs.cs
@@ -0,0 +1,36 @@
+using System.Collections.Generic;
+using System.Text.Json.Serialization;
+
+namespace auto_creamapi.POCOs
+{
+ public class App
+ {
+ [JsonPropertyName("appid")]
+ public int AppId { get; set; }
+
+ [JsonPropertyName("name")]
+ public string Name { get; set; }
+
+ public override string ToString()
+ {
+ return $"AppId: {AppId}, Name: {Name}";
+ }
+
+ public bool CompareId(App app)
+ {
+ return AppId.Equals(app.AppId);
+ }
+ }
+
+ public class Applist
+ {
+ [JsonPropertyName("apps")]
+ public List Apps { get; set; }
+ }
+
+ public class SteamApps
+ {
+ [JsonPropertyName("applist")]
+ public Applist AppList { get; set; }
+ }
+}
\ No newline at end of file
diff --git a/README.md b/README.md
index ae07184..0ec37d3 100644
--- a/README.md
+++ b/README.md
@@ -1,8 +1,10 @@
# Auto-CreamAPI 2
-[](https://ko-fi.com/H2H4330U3)
Set your game automatically up for use with CreamAPI.
+[](https://www.jetbrains.com/?from=Auto-CreamAPI)
+[Made with software provided by JetBrains s.r.o.](https://www.jetbrains.com/?from=Auto-CreamAPI)
+
## Features
* Setup CreamAPI’s DLLs and configuration file automatically.
* Find the AppID by providing the game’s name without having to look it up manually.
@@ -10,9 +12,11 @@ Set your game automatically up for use with CreamAPI.
* Set flags like "offline mode" and "extra protection" and select a language from a list.
## Installation
+
Download the latest release and extract it into any folder (e.g. `%USERPROFILE%\Desktop\auto-creamapi`).
## Usage
+
* Double-click `auto-creamapi.exe` to open the application. (When starting it for the first time, it might take a few
seconds since it needs to cache a list of games available on the Steam Store and download the latest CreamAPI DLL files.)
* Click on the *Open File* button on the top right and select the *steam_api.dll* or *steam_api64.dll*
@@ -25,13 +29,9 @@ Download the latest release and extract it into any folder (e.g. `%USERPROFILE%\
* Click on *"Save"*.
## License
+
Auto-CreamAPI itself is licensed under the GNU General Public License v3.0
-CreamAPI © 2016-2020, deadmau5. All Rights Reserved.
+CreamAPI © 2016-2019, deadmau5. All Rights Reserved.
*Dependencies will be listed ASAP.*
-
-## Software used
-
-[](https://www.jetbrains.com/?from=Auto-CreamAPI)
-[Made with software provided by JetBrains s.r.o.](https://www.jetbrains.com/?from=Auto-CreamAPI)
diff --git a/SearchResultWindow.xaml b/SearchResultWindow.xaml
new file mode 100644
index 0000000..b0fa782
--- /dev/null
+++ b/SearchResultWindow.xaml
@@ -0,0 +1,26 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/SearchResultWindow.xaml.cs b/SearchResultWindow.xaml.cs
new file mode 100644
index 0000000..11f030f
--- /dev/null
+++ b/SearchResultWindow.xaml.cs
@@ -0,0 +1,59 @@
+using System;
+using System.Collections.Generic;
+using System.Collections.ObjectModel;
+using System.Linq;
+using System.Text;
+using System.Windows;
+using System.Windows.Controls;
+using System.Windows.Data;
+using System.Windows.Documents;
+using System.Windows.Input;
+using System.Windows.Media;
+using System.Windows.Media.Imaging;
+using System.Windows.Shapes;
+using auto_creamapi.Utils;
+using NinjaNye.SearchExtensions;
+using NinjaNye.SearchExtensions.Models;
+
+namespace auto_creamapi
+{
+ ///
+ /// Interaction logic for SearchResultWindow.xaml
+ ///
+ public partial class SearchResultWindow
+ {
+ public SearchResultWindow(IEnumerable list)
+ {
+ InitializeComponent();
+ DgApps.ItemsSource = list;
+ }
+
+ private void OK_OnClick(object sender, RoutedEventArgs e)
+ {
+ GetSelectedApp();
+ }
+
+ private void DgApps_OnMouseDoubleClick(object sender, MouseButtonEventArgs e)
+ {
+ GetSelectedApp();
+ }
+
+ private void Cancel_OnClick(object sender, RoutedEventArgs e)
+ {
+ Close();
+ }
+
+ private void GetSelectedApp()
+ {
+ if (Application.Current.MainWindow is MainWindow currentMainWindow)
+ {
+ var app = (POCOs.App) DgApps.SelectedItem;
+ MyLogger.Log.Information($"Successfully got app {app}");
+ currentMainWindow.Game.Text = app.Name;
+ currentMainWindow.AppId.Text = app.AppId.ToString();
+ }
+
+ Close();
+ }
+ }
+}
diff --git a/auto-creamapi/Utils/MyLogger.cs b/Utils/MyLogger.cs
similarity index 76%
rename from auto-creamapi/Utils/MyLogger.cs
rename to Utils/MyLogger.cs
index 903e5a9..9e1b19a 100644
--- a/auto-creamapi/Utils/MyLogger.cs
+++ b/Utils/MyLogger.cs
@@ -1,14 +1,12 @@
using Serilog;
using Serilog.Core;
-using Serilog.Exceptions;
namespace auto_creamapi.Utils
{
- public static class MyLogger
+ public class MyLogger
{
public static readonly Logger Log = new LoggerConfiguration()
.MinimumLevel.Debug()
- .Enrich.WithExceptionDetails()
.WriteTo.Console()
.WriteTo.File("autocreamapi.log", rollingInterval: RollingInterval.Day)
.CreateLogger();
diff --git a/Utils/Secrets.EXAMPLE.cs b/Utils/Secrets.EXAMPLE.cs
new file mode 100644
index 0000000..06fcee6
--- /dev/null
+++ b/Utils/Secrets.EXAMPLE.cs
@@ -0,0 +1,14 @@
+namespace auto_creamapi.Utils
+{
+ ///
+ /// To use this:
+ /// Rename file Secrets.EXAMPLE.cs to Secrets.cs
+ /// Rename class Secrets_REMOVETHIS to Secrets
+ /// Enter the relevant info below
+ ///
+ public class Secrets_REMOVETHIS
+ {
+ public const string Username = "Enter username here";
+ public const string Password = "Enter password here";
+ }
+}
\ No newline at end of file
diff --git a/auto-creamapi.csproj b/auto-creamapi.csproj
new file mode 100644
index 0000000..805f150
--- /dev/null
+++ b/auto-creamapi.csproj
@@ -0,0 +1,40 @@
+
+
+
+ WinExe
+ netcoreapp3.1
+ auto_creamapi
+ true
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/auto-creamapi.sln b/auto-creamapi.sln
index 78be129..133faf0 100644
--- a/auto-creamapi.sln
+++ b/auto-creamapi.sln
@@ -1,9 +1,9 @@
Microsoft Visual Studio Solution File, Format Version 12.00
-# Visual Studio Version 17
-VisualStudioVersion = 17.8.34330.188
+# Visual Studio Version 16
+VisualStudioVersion = 16.0.30413.136
MinimumVisualStudioVersion = 10.0.40219.1
-Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "auto-creamapi", "auto-creamapi\auto-creamapi.csproj", "{26060B32-199E-4366-8FDE-6B1E10E0EF62}"
+Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "auto-creamapi", "auto-creamapi.csproj", "{F49CEA2D-367E-4F57-A4A1-DA3D8F1D97EC}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
@@ -11,10 +11,10 @@ Global
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
- {26060B32-199E-4366-8FDE-6B1E10E0EF62}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
- {26060B32-199E-4366-8FDE-6B1E10E0EF62}.Debug|Any CPU.Build.0 = Debug|Any CPU
- {26060B32-199E-4366-8FDE-6B1E10E0EF62}.Release|Any CPU.ActiveCfg = Release|Any CPU
- {26060B32-199E-4366-8FDE-6B1E10E0EF62}.Release|Any CPU.Build.0 = Release|Any CPU
+ {F49CEA2D-367E-4F57-A4A1-DA3D8F1D97EC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {F49CEA2D-367E-4F57-A4A1-DA3D8F1D97EC}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {F49CEA2D-367E-4F57-A4A1-DA3D8F1D97EC}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {F49CEA2D-367E-4F57-A4A1-DA3D8F1D97EC}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
diff --git a/auto-creamapi/App.xaml b/auto-creamapi/App.xaml
deleted file mode 100644
index 112bf05..0000000
--- a/auto-creamapi/App.xaml
+++ /dev/null
@@ -1,10 +0,0 @@
-
-
-
\ No newline at end of file
diff --git a/auto-creamapi/App.xaml.cs b/auto-creamapi/App.xaml.cs
deleted file mode 100644
index d78cbe5..0000000
--- a/auto-creamapi/App.xaml.cs
+++ /dev/null
@@ -1,16 +0,0 @@
-using MvvmCross.Core;
-using MvvmCross.Platforms.Wpf.Core;
-
-namespace auto_creamapi
-{
- ///
- /// Interaction logic for App.xaml
- ///
- public partial class App
- {
- protected override void RegisterSetup()
- {
- this.RegisterSetupType();
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/COPYING b/auto-creamapi/COPYING
deleted file mode 100644
index f288702..0000000
--- a/auto-creamapi/COPYING
+++ /dev/null
@@ -1,674 +0,0 @@
- GNU GENERAL PUBLIC LICENSE
- Version 3, 29 June 2007
-
- Copyright (C) 2007 Free Software Foundation, Inc.
- Everyone is permitted to copy and distribute verbatim copies
- of this license document, but changing it is not allowed.
-
- Preamble
-
- The GNU General Public License is a free, copyleft license for
-software and other kinds of works.
-
- The licenses for most software and other practical works are designed
-to take away your freedom to share and change the works. By contrast,
-the GNU General Public License is intended to guarantee your freedom to
-share and change all versions of a program--to make sure it remains free
-software for all its users. We, the Free Software Foundation, use the
-GNU General Public License for most of our software; it applies also to
-any other work released this way by its authors. You can apply it to
-your programs, too.
-
- When we speak of free software, we are referring to freedom, not
-price. Our General Public Licenses are designed to make sure that you
-have the freedom to distribute copies of free software (and charge for
-them if you wish), that you receive source code or can get it if you
-want it, that you can change the software or use pieces of it in new
-free programs, and that you know you can do these things.
-
- To protect your rights, we need to prevent others from denying you
-these rights or asking you to surrender the rights. Therefore, you have
-certain responsibilities if you distribute copies of the software, or if
-you modify it: responsibilities to respect the freedom of others.
-
- For example, if you distribute copies of such a program, whether
-gratis or for a fee, you must pass on to the recipients the same
-freedoms that you received. You must make sure that they, too, receive
-or can get the source code. And you must show them these terms so they
-know their rights.
-
- Developers that use the GNU GPL protect your rights with two steps:
-(1) assert copyright on the software, and (2) offer you this License
-giving you legal permission to copy, distribute and/or modify it.
-
- For the developers' and authors' protection, the GPL clearly explains
-that there is no warranty for this free software. For both users' and
-authors' sake, the GPL requires that modified versions be marked as
-changed, so that their problems will not be attributed erroneously to
-authors of previous versions.
-
- Some devices are designed to deny users access to install or run
-modified versions of the software inside them, although the manufacturer
-can do so. This is fundamentally incompatible with the aim of
-protecting users' freedom to change the software. The systematic
-pattern of such abuse occurs in the area of products for individuals to
-use, which is precisely where it is most unacceptable. Therefore, we
-have designed this version of the GPL to prohibit the practice for those
-products. If such problems arise substantially in other domains, we
-stand ready to extend this provision to those domains in future versions
-of the GPL, as needed to protect the freedom of users.
-
- Finally, every program is threatened constantly by software patents.
-States should not allow patents to restrict development and use of
-software on general-purpose computers, but in those that do, we wish to
-avoid the special danger that patents applied to a free program could
-make it effectively proprietary. To prevent this, the GPL assures that
-patents cannot be used to render the program non-free.
-
- The precise terms and conditions for copying, distribution and
-modification follow.
-
- TERMS AND CONDITIONS
-
- 0. Definitions.
-
- "This License" refers to version 3 of the GNU General Public License.
-
- "Copyright" also means copyright-like laws that apply to other kinds of
-works, such as semiconductor masks.
-
- "The Program" refers to any copyrightable work licensed under this
-License. Each licensee is addressed as "you". "Licensees" and
-"recipients" may be individuals or organizations.
-
- To "modify" a work means to copy from or adapt all or part of the work
-in a fashion requiring copyright permission, other than the making of an
-exact copy. The resulting work is called a "modified version" of the
-earlier work or a work "based on" the earlier work.
-
- A "covered work" means either the unmodified Program or a work based
-on the Program.
-
- To "propagate" a work means to do anything with it that, without
-permission, would make you directly or secondarily liable for
-infringement under applicable copyright law, except executing it on a
-computer or modifying a private copy. Propagation includes copying,
-distribution (with or without modification), making available to the
-public, and in some countries other activities as well.
-
- To "convey" a work means any kind of propagation that enables other
-parties to make or receive copies. Mere interaction with a user through
-a computer network, with no transfer of a copy, is not conveying.
-
- An interactive user interface displays "Appropriate Legal Notices"
-to the extent that it includes a convenient and prominently visible
-feature that (1) displays an appropriate copyright notice, and (2)
-tells the user that there is no warranty for the work (except to the
-extent that warranties are provided), that licensees may convey the
-work under this License, and how to view a copy of this License. If
-the interface presents a list of user commands or options, such as a
-menu, a prominent item in the list meets this criterion.
-
- 1. Source Code.
-
- The "source code" for a work means the preferred form of the work
-for making modifications to it. "Object code" means any non-source
-form of a work.
-
- A "Standard Interface" means an interface that either is an official
-standard defined by a recognized standards body, or, in the case of
-interfaces specified for a particular programming language, one that
-is widely used among developers working in that language.
-
- The "System Libraries" of an executable work include anything, other
-than the work as a whole, that (a) is included in the normal form of
-packaging a Major Component, but which is not part of that Major
-Component, and (b) serves only to enable use of the work with that
-Major Component, or to implement a Standard Interface for which an
-implementation is available to the public in source code form. A
-"Major Component", in this context, means a major essential component
-(kernel, window system, and so on) of the specific operating system
-(if any) on which the executable work runs, or a compiler used to
-produce the work, or an object code interpreter used to run it.
-
- The "Corresponding Source" for a work in object code form means all
-the source code needed to generate, install, and (for an executable
-work) run the object code and to modify the work, including scripts to
-control those activities. However, it does not include the work's
-System Libraries, or general-purpose tools or generally available free
-programs which are used unmodified in performing those activities but
-which are not part of the work. For example, Corresponding Source
-includes interface definition files associated with source files for
-the work, and the source code for shared libraries and dynamically
-linked subprograms that the work is specifically designed to require,
-such as by intimate data communication or control flow between those
-subprograms and other parts of the work.
-
- The Corresponding Source need not include anything that users
-can regenerate automatically from other parts of the Corresponding
-Source.
-
- The Corresponding Source for a work in source code form is that
-same work.
-
- 2. Basic Permissions.
-
- All rights granted under this License are granted for the term of
-copyright on the Program, and are irrevocable provided the stated
-conditions are met. This License explicitly affirms your unlimited
-permission to run the unmodified Program. The output from running a
-covered work is covered by this License only if the output, given its
-content, constitutes a covered work. This License acknowledges your
-rights of fair use or other equivalent, as provided by copyright law.
-
- You may make, run and propagate covered works that you do not
-convey, without conditions so long as your license otherwise remains
-in force. You may convey covered works to others for the sole purpose
-of having them make modifications exclusively for you, or provide you
-with facilities for running those works, provided that you comply with
-the terms of this License in conveying all material for which you do
-not control copyright. Those thus making or running the covered works
-for you must do so exclusively on your behalf, under your direction
-and control, on terms that prohibit them from making any copies of
-your copyrighted material outside their relationship with you.
-
- Conveying under any other circumstances is permitted solely under
-the conditions stated below. Sublicensing is not allowed; section 10
-makes it unnecessary.
-
- 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
-
- No covered work shall be deemed part of an effective technological
-measure under any applicable law fulfilling obligations under article
-11 of the WIPO copyright treaty adopted on 20 December 1996, or
-similar laws prohibiting or restricting circumvention of such
-measures.
-
- When you convey a covered work, you waive any legal power to forbid
-circumvention of technological measures to the extent such circumvention
-is effected by exercising rights under this License with respect to
-the covered work, and you disclaim any intention to limit operation or
-modification of the work as a means of enforcing, against the work's
-users, your or third parties' legal rights to forbid circumvention of
-technological measures.
-
- 4. Conveying Verbatim Copies.
-
- You may convey verbatim copies of the Program's source code as you
-receive it, in any medium, provided that you conspicuously and
-appropriately publish on each copy an appropriate copyright notice;
-keep intact all notices stating that this License and any
-non-permissive terms added in accord with section 7 apply to the code;
-keep intact all notices of the absence of any warranty; and give all
-recipients a copy of this License along with the Program.
-
- You may charge any price or no price for each copy that you convey,
-and you may offer support or warranty protection for a fee.
-
- 5. Conveying Modified Source Versions.
-
- You may convey a work based on the Program, or the modifications to
-produce it from the Program, in the form of source code under the
-terms of section 4, provided that you also meet all of these conditions:
-
- a) The work must carry prominent notices stating that you modified
- it, and giving a relevant date.
-
- b) The work must carry prominent notices stating that it is
- released under this License and any conditions added under section
- 7. This requirement modifies the requirement in section 4 to
- "keep intact all notices".
-
- c) You must license the entire work, as a whole, under this
- License to anyone who comes into possession of a copy. This
- License will therefore apply, along with any applicable section 7
- additional terms, to the whole of the work, and all its parts,
- regardless of how they are packaged. This License gives no
- permission to license the work in any other way, but it does not
- invalidate such permission if you have separately received it.
-
- d) If the work has interactive user interfaces, each must display
- Appropriate Legal Notices; however, if the Program has interactive
- interfaces that do not display Appropriate Legal Notices, your
- work need not make them do so.
-
- A compilation of a covered work with other separate and independent
-works, which are not by their nature extensions of the covered work,
-and which are not combined with it such as to form a larger program,
-in or on a volume of a storage or distribution medium, is called an
-"aggregate" if the compilation and its resulting copyright are not
-used to limit the access or legal rights of the compilation's users
-beyond what the individual works permit. Inclusion of a covered work
-in an aggregate does not cause this License to apply to the other
-parts of the aggregate.
-
- 6. Conveying Non-Source Forms.
-
- You may convey a covered work in object code form under the terms
-of sections 4 and 5, provided that you also convey the
-machine-readable Corresponding Source under the terms of this License,
-in one of these ways:
-
- a) Convey the object code in, or embodied in, a physical product
- (including a physical distribution medium), accompanied by the
- Corresponding Source fixed on a durable physical medium
- customarily used for software interchange.
-
- b) Convey the object code in, or embodied in, a physical product
- (including a physical distribution medium), accompanied by a
- written offer, valid for at least three years and valid for as
- long as you offer spare parts or customer support for that product
- model, to give anyone who possesses the object code either (1) a
- copy of the Corresponding Source for all the software in the
- product that is covered by this License, on a durable physical
- medium customarily used for software interchange, for a price no
- more than your reasonable cost of physically performing this
- conveying of source, or (2) access to copy the
- Corresponding Source from a network server at no charge.
-
- c) Convey individual copies of the object code with a copy of the
- written offer to provide the Corresponding Source. This
- alternative is allowed only occasionally and noncommercially, and
- only if you received the object code with such an offer, in accord
- with subsection 6b.
-
- d) Convey the object code by offering access from a designated
- place (gratis or for a charge), and offer equivalent access to the
- Corresponding Source in the same way through the same place at no
- further charge. You need not require recipients to copy the
- Corresponding Source along with the object code. If the place to
- copy the object code is a network server, the Corresponding Source
- may be on a different server (operated by you or a third party)
- that supports equivalent copying facilities, provided you maintain
- clear directions next to the object code saying where to find the
- Corresponding Source. Regardless of what server hosts the
- Corresponding Source, you remain obligated to ensure that it is
- available for as long as needed to satisfy these requirements.
-
- e) Convey the object code using peer-to-peer transmission, provided
- you inform other peers where the object code and Corresponding
- Source of the work are being offered to the general public at no
- charge under subsection 6d.
-
- A separable portion of the object code, whose source code is excluded
-from the Corresponding Source as a System Library, need not be
-included in conveying the object code work.
-
- A "User Product" is either (1) a "consumer product", which means any
-tangible personal property which is normally used for personal, family,
-or household purposes, or (2) anything designed or sold for incorporation
-into a dwelling. In determining whether a product is a consumer product,
-doubtful cases shall be resolved in favor of coverage. For a particular
-product received by a particular user, "normally used" refers to a
-typical or common use of that class of product, regardless of the status
-of the particular user or of the way in which the particular user
-actually uses, or expects or is expected to use, the product. A product
-is a consumer product regardless of whether the product has substantial
-commercial, industrial or non-consumer uses, unless such uses represent
-the only significant mode of use of the product.
-
- "Installation Information" for a User Product means any methods,
-procedures, authorization keys, or other information required to install
-and execute modified versions of a covered work in that User Product from
-a modified version of its Corresponding Source. The information must
-suffice to ensure that the continued functioning of the modified object
-code is in no case prevented or interfered with solely because
-modification has been made.
-
- If you convey an object code work under this section in, or with, or
-specifically for use in, a User Product, and the conveying occurs as
-part of a transaction in which the right of possession and use of the
-User Product is transferred to the recipient in perpetuity or for a
-fixed term (regardless of how the transaction is characterized), the
-Corresponding Source conveyed under this section must be accompanied
-by the Installation Information. But this requirement does not apply
-if neither you nor any third party retains the ability to install
-modified object code on the User Product (for example, the work has
-been installed in ROM).
-
- The requirement to provide Installation Information does not include a
-requirement to continue to provide support service, warranty, or updates
-for a work that has been modified or installed by the recipient, or for
-the User Product in which it has been modified or installed. Access to a
-network may be denied when the modification itself materially and
-adversely affects the operation of the network or violates the rules and
-protocols for communication across the network.
-
- Corresponding Source conveyed, and Installation Information provided,
-in accord with this section must be in a format that is publicly
-documented (and with an implementation available to the public in
-source code form), and must require no special password or key for
-unpacking, reading or copying.
-
- 7. Additional Terms.
-
- "Additional permissions" are terms that supplement the terms of this
-License by making exceptions from one or more of its conditions.
-Additional permissions that are applicable to the entire Program shall
-be treated as though they were included in this License, to the extent
-that they are valid under applicable law. If additional permissions
-apply only to part of the Program, that part may be used separately
-under those permissions, but the entire Program remains governed by
-this License without regard to the additional permissions.
-
- When you convey a copy of a covered work, you may at your option
-remove any additional permissions from that copy, or from any part of
-it. (Additional permissions may be written to require their own
-removal in certain cases when you modify the work.) You may place
-additional permissions on material, added by you to a covered work,
-for which you have or can give appropriate copyright permission.
-
- Notwithstanding any other provision of this License, for material you
-add to a covered work, you may (if authorized by the copyright holders of
-that material) supplement the terms of this License with terms:
-
- a) Disclaiming warranty or limiting liability differently from the
- terms of sections 15 and 16 of this License; or
-
- b) Requiring preservation of specified reasonable legal notices or
- author attributions in that material or in the Appropriate Legal
- Notices displayed by works containing it; or
-
- c) Prohibiting misrepresentation of the origin of that material, or
- requiring that modified versions of such material be marked in
- reasonable ways as different from the original version; or
-
- d) Limiting the use for publicity purposes of names of licensors or
- authors of the material; or
-
- e) Declining to grant rights under trademark law for use of some
- trade names, trademarks, or service marks; or
-
- f) Requiring indemnification of licensors and authors of that
- material by anyone who conveys the material (or modified versions of
- it) with contractual assumptions of liability to the recipient, for
- any liability that these contractual assumptions directly impose on
- those licensors and authors.
-
- All other non-permissive additional terms are considered "further
-restrictions" within the meaning of section 10. If the Program as you
-received it, or any part of it, contains a notice stating that it is
-governed by this License along with a term that is a further
-restriction, you may remove that term. If a license document contains
-a further restriction but permits relicensing or conveying under this
-License, you may add to a covered work material governed by the terms
-of that license document, provided that the further restriction does
-not survive such relicensing or conveying.
-
- If you add terms to a covered work in accord with this section, you
-must place, in the relevant source files, a statement of the
-additional terms that apply to those files, or a notice indicating
-where to find the applicable terms.
-
- Additional terms, permissive or non-permissive, may be stated in the
-form of a separately written license, or stated as exceptions;
-the above requirements apply either way.
-
- 8. Termination.
-
- You may not propagate or modify a covered work except as expressly
-provided under this License. Any attempt otherwise to propagate or
-modify it is void, and will automatically terminate your rights under
-this License (including any patent licenses granted under the third
-paragraph of section 11).
-
- However, if you cease all violation of this License, then your
-license from a particular copyright holder is reinstated (a)
-provisionally, unless and until the copyright holder explicitly and
-finally terminates your license, and (b) permanently, if the copyright
-holder fails to notify you of the violation by some reasonable means
-prior to 60 days after the cessation.
-
- Moreover, your license from a particular copyright holder is
-reinstated permanently if the copyright holder notifies you of the
-violation by some reasonable means, this is the first time you have
-received notice of violation of this License (for any work) from that
-copyright holder, and you cure the violation prior to 30 days after
-your receipt of the notice.
-
- Termination of your rights under this section does not terminate the
-licenses of parties who have received copies or rights from you under
-this License. If your rights have been terminated and not permanently
-reinstated, you do not qualify to receive new licenses for the same
-material under section 10.
-
- 9. Acceptance Not Required for Having Copies.
-
- You are not required to accept this License in order to receive or
-run a copy of the Program. Ancillary propagation of a covered work
-occurring solely as a consequence of using peer-to-peer transmission
-to receive a copy likewise does not require acceptance. However,
-nothing other than this License grants you permission to propagate or
-modify any covered work. These actions infringe copyright if you do
-not accept this License. Therefore, by modifying or propagating a
-covered work, you indicate your acceptance of this License to do so.
-
- 10. Automatic Licensing of Downstream Recipients.
-
- Each time you convey a covered work, the recipient automatically
-receives a license from the original licensors, to run, modify and
-propagate that work, subject to this License. You are not responsible
-for enforcing compliance by third parties with this License.
-
- An "entity transaction" is a transaction transferring control of an
-organization, or substantially all assets of one, or subdividing an
-organization, or merging organizations. If propagation of a covered
-work results from an entity transaction, each party to that
-transaction who receives a copy of the work also receives whatever
-licenses to the work the party's predecessor in interest had or could
-give under the previous paragraph, plus a right to possession of the
-Corresponding Source of the work from the predecessor in interest, if
-the predecessor has it or can get it with reasonable efforts.
-
- You may not impose any further restrictions on the exercise of the
-rights granted or affirmed under this License. For example, you may
-not impose a license fee, royalty, or other charge for exercise of
-rights granted under this License, and you may not initiate litigation
-(including a cross-claim or counterclaim in a lawsuit) alleging that
-any patent claim is infringed by making, using, selling, offering for
-sale, or importing the Program or any portion of it.
-
- 11. Patents.
-
- A "contributor" is a copyright holder who authorizes use under this
-License of the Program or a work on which the Program is based. The
-work thus licensed is called the contributor's "contributor version".
-
- A contributor's "essential patent claims" are all patent claims
-owned or controlled by the contributor, whether already acquired or
-hereafter acquired, that would be infringed by some manner, permitted
-by this License, of making, using, or selling its contributor version,
-but do not include claims that would be infringed only as a
-consequence of further modification of the contributor version. For
-purposes of this definition, "control" includes the right to grant
-patent sublicenses in a manner consistent with the requirements of
-this License.
-
- Each contributor grants you a non-exclusive, worldwide, royalty-free
-patent license under the contributor's essential patent claims, to
-make, use, sell, offer for sale, import and otherwise run, modify and
-propagate the contents of its contributor version.
-
- In the following three paragraphs, a "patent license" is any express
-agreement or commitment, however denominated, not to enforce a patent
-(such as an express permission to practice a patent or covenant not to
-sue for patent infringement). To "grant" such a patent license to a
-party means to make such an agreement or commitment not to enforce a
-patent against the party.
-
- If you convey a covered work, knowingly relying on a patent license,
-and the Corresponding Source of the work is not available for anyone
-to copy, free of charge and under the terms of this License, through a
-publicly available network server or other readily accessible means,
-then you must either (1) cause the Corresponding Source to be so
-available, or (2) arrange to deprive yourself of the benefit of the
-patent license for this particular work, or (3) arrange, in a manner
-consistent with the requirements of this License, to extend the patent
-license to downstream recipients. "Knowingly relying" means you have
-actual knowledge that, but for the patent license, your conveying the
-covered work in a country, or your recipient's use of the covered work
-in a country, would infringe one or more identifiable patents in that
-country that you have reason to believe are valid.
-
- If, pursuant to or in connection with a single transaction or
-arrangement, you convey, or propagate by procuring conveyance of, a
-covered work, and grant a patent license to some of the parties
-receiving the covered work authorizing them to use, propagate, modify
-or convey a specific copy of the covered work, then the patent license
-you grant is automatically extended to all recipients of the covered
-work and works based on it.
-
- A patent license is "discriminatory" if it does not include within
-the scope of its coverage, prohibits the exercise of, or is
-conditioned on the non-exercise of one or more of the rights that are
-specifically granted under this License. You may not convey a covered
-work if you are a party to an arrangement with a third party that is
-in the business of distributing software, under which you make payment
-to the third party based on the extent of your activity of conveying
-the work, and under which the third party grants, to any of the
-parties who would receive the covered work from you, a discriminatory
-patent license (a) in connection with copies of the covered work
-conveyed by you (or copies made from those copies), or (b) primarily
-for and in connection with specific products or compilations that
-contain the covered work, unless you entered into that arrangement,
-or that patent license was granted, prior to 28 March 2007.
-
- Nothing in this License shall be construed as excluding or limiting
-any implied license or other defenses to infringement that may
-otherwise be available to you under applicable patent law.
-
- 12. No Surrender of Others' Freedom.
-
- If conditions are imposed on you (whether by court order, agreement or
-otherwise) that contradict the conditions of this License, they do not
-excuse you from the conditions of this License. If you cannot convey a
-covered work so as to satisfy simultaneously your obligations under this
-License and any other pertinent obligations, then as a consequence you may
-not convey it at all. For example, if you agree to terms that obligate you
-to collect a royalty for further conveying from those to whom you convey
-the Program, the only way you could satisfy both those terms and this
-License would be to refrain entirely from conveying the Program.
-
- 13. Use with the GNU Affero General Public License.
-
- Notwithstanding any other provision of this License, you have
-permission to link or combine any covered work with a work licensed
-under version 3 of the GNU Affero General Public License into a single
-combined work, and to convey the resulting work. The terms of this
-License will continue to apply to the part which is the covered work,
-but the special requirements of the GNU Affero General Public License,
-section 13, concerning interaction through a network will apply to the
-combination as such.
-
- 14. Revised Versions of this License.
-
- The Free Software Foundation may publish revised and/or new versions of
-the GNU General Public License from time to time. Such new versions will
-be similar in spirit to the present version, but may differ in detail to
-address new problems or concerns.
-
- Each version is given a distinguishing version number. If the
-Program specifies that a certain numbered version of the GNU General
-Public License "or any later version" applies to it, you have the
-option of following the terms and conditions either of that numbered
-version or of any later version published by the Free Software
-Foundation. If the Program does not specify a version number of the
-GNU General Public License, you may choose any version ever published
-by the Free Software Foundation.
-
- If the Program specifies that a proxy can decide which future
-versions of the GNU General Public License can be used, that proxy's
-public statement of acceptance of a version permanently authorizes you
-to choose that version for the Program.
-
- Later license versions may give you additional or different
-permissions. However, no additional obligations are imposed on any
-author or copyright holder as a result of your choosing to follow a
-later version.
-
- 15. Disclaimer of Warranty.
-
- THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
-APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
-HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
-OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
-THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
-PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
-IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
-ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
-
- 16. Limitation of Liability.
-
- IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
-WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
-THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
-GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
-USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
-DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
-PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
-EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
-SUCH DAMAGES.
-
- 17. Interpretation of Sections 15 and 16.
-
- If the disclaimer of warranty and limitation of liability provided
-above cannot be given local legal effect according to their terms,
-reviewing courts shall apply local law that most closely approximates
-an absolute waiver of all civil liability in connection with the
-Program, unless a warranty or assumption of liability accompanies a
-copy of the Program in return for a fee.
-
- END OF TERMS AND CONDITIONS
-
- How to Apply These Terms to Your New Programs
-
- If you develop a new program, and you want it to be of the greatest
-possible use to the public, the best way to achieve this is to make it
-free software which everyone can redistribute and change under these terms.
-
- To do so, attach the following notices to the program. It is safest
-to attach them to the start of each source file to most effectively
-state the exclusion of warranty; and each file should have at least
-the "copyright" line and a pointer to where the full notice is found.
-
-
- Copyright (C)
-
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
-
- You should have received a copy of the GNU General Public License
- along with this program. If not, see .
-
-Also add information on how to contact you by electronic and paper mail.
-
- If the program does terminal interaction, make it output a short
-notice like this when it starts in an interactive mode:
-
- Copyright (C)
- This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
- This is free software, and you are welcome to redistribute it
- under certain conditions; type `show c' for details.
-
-The hypothetical commands `show w' and `show c' should show the appropriate
-parts of the General Public License. Of course, your program's commands
-might be different; for a GUI interface, you would use an "about box".
-
- You should also get your employer (if you work as a programmer) or school,
-if any, to sign a "copyright disclaimer" for the program, if necessary.
-For more information on this, and how to apply and follow the GNU GPL, see
-.
-
- The GNU General Public License does not permit incorporating your program
-into proprietary programs. If your program is a subroutine library, you
-may consider it more useful to permit linking proprietary applications with
-the library. If this is what you want to do, use the GNU Lesser General
-Public License instead of this License. But first, please read
-.
diff --git a/auto-creamapi/Converters/ListOfDLcToStringConverter.cs b/auto-creamapi/Converters/ListOfDLcToStringConverter.cs
deleted file mode 100644
index 5691ee6..0000000
--- a/auto-creamapi/Converters/ListOfDLcToStringConverter.cs
+++ /dev/null
@@ -1,67 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.Collections.ObjectModel;
-using System.Globalization;
-using System.IO;
-using System.Text.RegularExpressions;
-using auto_creamapi.Models;
-using auto_creamapi.Utils;
-using MvvmCross.Converters;
-using MvvmCross.Platforms.Wpf.Converters;
-
-namespace auto_creamapi.Converters
-{
- public class ListOfDLcToStringNativeConverter : MvxNativeValueConverter
- {
- }
-
- public class ListOfDLcToStringConverter : MvxValueConverter, string>
- {
- protected override string Convert(ObservableCollection value, Type targetType, object parameter,
- CultureInfo culture)
- {
- if (value == null) return "";
- MyLogger.Log.Debug("ListOfDLcToStringConverter: Convert");
- var dlcListToString = DlcListToString(value);
- return dlcListToString.GetType() == targetType ? dlcListToString : "";
- }
-
- protected override ObservableCollection ConvertBack(string value, Type targetType, object parameter,
- CultureInfo culture)
- {
- MyLogger.Log.Debug("ListOfDLcToStringConverter: ConvertBack");
- var stringToDlcList = StringToDlcList(value);
- return stringToDlcList.GetType() == targetType ? stringToDlcList : [];
- }
-
- private static ObservableCollection StringToDlcList(string value)
- {
- var result = new ObservableCollection();
- var expression = new Regex("(?.*) *= *(?.*)");
- using var reader = new StringReader(value);
- string line;
- while ((line = reader.ReadLine()) != null)
- {
- var match = expression.Match(line);
- if (match.Success)
- {
- result.Add(new SteamApp
- {
- AppId = int.Parse(match.Groups["id"].Value),
- Name = match.Groups["name"].Value
- });
- }
- }
-
- return result;
- }
-
- private static string DlcListToString(IEnumerable value)
- {
- var result = "";
- //value.ForEach(x => result += $"{x}\n");
- foreach (var steamApp in value) result += $"{steamApp}\n";
- return result;
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Core/MainApplication.cs b/auto-creamapi/Core/MainApplication.cs
deleted file mode 100644
index 2201b8a..0000000
--- a/auto-creamapi/Core/MainApplication.cs
+++ /dev/null
@@ -1,20 +0,0 @@
-using auto_creamapi.ViewModels;
-using MvvmCross.IoC;
-using MvvmCross.ViewModels;
-
-namespace auto_creamapi.Core
-{
- public class MainApplication : MvxApplication
- {
- public override void Initialize()
- {
- //Mvx.IoCProvider.RegisterType();
- CreatableTypes()
- .EndingWith("Service")
- .AsInterfaces()
- .RegisterAsLazySingleton();
-
- RegisterAppStart();
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/MainWindow.xaml b/auto-creamapi/MainWindow.xaml
deleted file mode 100644
index 7212a0c..0000000
--- a/auto-creamapi/MainWindow.xaml
+++ /dev/null
@@ -1,11 +0,0 @@
-
-
-
\ No newline at end of file
diff --git a/auto-creamapi/MainWindow.xaml.cs b/auto-creamapi/MainWindow.xaml.cs
deleted file mode 100644
index 2e09a9c..0000000
--- a/auto-creamapi/MainWindow.xaml.cs
+++ /dev/null
@@ -1,13 +0,0 @@
-using MvvmCross.Platforms.Wpf.Presenters.Attributes;
-
-namespace auto_creamapi
-{
- [MvxWindowPresentation(Identifier = nameof(MainWindow), Modal = false)]
- public partial class MainWindow
- {
- public MainWindow()
- {
- InitializeComponent();
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Messenger/ProgressMessage.cs b/auto-creamapi/Messenger/ProgressMessage.cs
deleted file mode 100644
index a83606b..0000000
--- a/auto-creamapi/Messenger/ProgressMessage.cs
+++ /dev/null
@@ -1,40 +0,0 @@
-using HttpProgress;
-using MvvmCross.Plugin.Messenger;
-
-namespace auto_creamapi.Messenger
-{
- public class ProgressMessage : MvxMessage
- {
- private readonly ICopyProgress _progress;
- private double _percentProgress;
-
- public ProgressMessage(object sender, string info, string filename, ICopyProgress progress) : base(sender)
- {
- Info = info;
- Filename = filename;
- _progress = progress;
- }
-
- public ProgressMessage(object sender, string info, string filename, double progress) : base(sender)
- {
- _progress = null;
- Info = info;
- Filename = filename;
- PercentComplete = progress;
- }
-
- public string Info { get; }
- public string Filename { get; }
-
- public double PercentComplete
- {
- get => _progress?.PercentComplete ?? _percentProgress;
- private set => _percentProgress = value;
- }
-
- // public long BytesTransferred => _progress.BytesTransferred;
- // public long ExpectedBytes => _progress.ExpectedBytes;
- // public long BytesPerSecond => _progress.BytesPerSecond;
- // public TimeSpan TransferTime => _progress.TransferTime;
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Models/CreamConfigModel.cs b/auto-creamapi/Models/CreamConfigModel.cs
deleted file mode 100644
index b48634b..0000000
--- a/auto-creamapi/Models/CreamConfigModel.cs
+++ /dev/null
@@ -1,38 +0,0 @@
-using System.Collections.Generic;
-using System.Linq;
-
-namespace auto_creamapi.Models
-{
- public class CreamConfig
- {
- public CreamConfig()
- {
- DlcList = new List();
- }
-
- public int AppId { get; set; }
- public string Language { get; set; }
- public bool UnlockAll { get; set; }
- public bool ExtraProtection { get; set; }
- public bool ForceOffline { get; set; }
- public List DlcList { get; set; }
-
- public override string ToString()
- {
- var value = $"AppID: {AppId}\n" +
- $"Language: {Language}\n" +
- $"UnlockAll: {UnlockAll}\n" +
- $"ExtraProtection: {ExtraProtection}\n" +
- $"ForceOffline: {ForceOffline}\n" +
- $"DLC ({DlcList.Count}):\n[\n";
- if (DlcList.Count > 0)
- value = DlcList.Aggregate(value, (current, x) => current + $" {x.AppId}={x.Name},\n");
- value += "]";
- return value;
- }
- }
-
- public sealed class CreamConfigModel
- {
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Models/CreamDllModel.cs b/auto-creamapi/Models/CreamDllModel.cs
deleted file mode 100644
index 6415459..0000000
--- a/auto-creamapi/Models/CreamDllModel.cs
+++ /dev/null
@@ -1,33 +0,0 @@
-using System;
-using System.IO;
-using System.Security.Cryptography;
-
-namespace auto_creamapi.Models
-{
- internal class CreamDll
- {
- public readonly string Filename;
- public readonly string Hash;
- public readonly string OrigFilename;
-
- public CreamDll(string filename, string origFilename)
- {
- Filename = filename;
- OrigFilename = origFilename;
- Hash = "";
-
- using var md5 = MD5.Create();
- if (File.Exists(Filename))
- {
- using var stream = File.OpenRead(Filename);
- Hash = BitConverter
- .ToString(md5.ComputeHash(stream))
- .Replace("-", string.Empty);
- }
- }
- }
-
- public class CreamDllModel
- {
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Models/SteamAppModel.cs b/auto-creamapi/Models/SteamAppModel.cs
deleted file mode 100644
index 3f74258..0000000
--- a/auto-creamapi/Models/SteamAppModel.cs
+++ /dev/null
@@ -1,45 +0,0 @@
-using System.Collections.Generic;
-using System.Text.Json.Serialization;
-using System.Text.RegularExpressions;
-using auto_creamapi.Utils;
-
-namespace auto_creamapi.Models
-{
- public class SteamApp
- {
- private string _name;
- private string _comparableName;
- [JsonPropertyName("appid")] public int AppId { get; set; }
-
- [JsonPropertyName("name")]
- public string Name
- {
- get => _name;
- set
- {
- _name = value;
- _comparableName = Regex.Replace(value, Misc.SpecialCharsRegex, "").ToLower();
- }
- }
-
- public bool CompareName(string value)
- {
- return _comparableName.Equals(value);
- }
-
- public override string ToString()
- {
- return $"{AppId}={Name}";
- }
- }
-
- public class AppList
- {
- [JsonPropertyName("apps")] public List Apps { get; set; }
- }
-
- public class SteamApps
- {
- [JsonPropertyName("applist")] public AppList AppList { get; set; }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/README.md b/auto-creamapi/README.md
deleted file mode 100644
index ae07184..0000000
--- a/auto-creamapi/README.md
+++ /dev/null
@@ -1,37 +0,0 @@
-# Auto-CreamAPI 2
-[](https://ko-fi.com/H2H4330U3)
-
-Set your game automatically up for use with CreamAPI.
-
-## Features
-* Setup CreamAPI’s DLLs and configuration file automatically.
-* Find the AppID by providing the game’s name without having to look it up manually.
-* Fetch a list of DLCs for an AppID from both the Steam Store and SteamDB.
-* Set flags like "offline mode" and "extra protection" and select a language from a list.
-
-## Installation
-Download the latest release and extract it into any folder (e.g. `%USERPROFILE%\Desktop\auto-creamapi`).
-
-## Usage
-* Double-click `auto-creamapi.exe` to open the application. (When starting it for the first time, it might take a few
- seconds since it needs to cache a list of games available on the Steam Store and download the latest CreamAPI DLL files.)
-* Click on the *Open File* button on the top right and select the *steam_api.dll* or *steam_api64.dll*
- **in the game folder**.
-* Enter the name of the game and click on the *Search* button.
- * If it did not find the right game, either try again, or copy the app ID from the Steam Store to field to the right
- of the *Search* button.
-* Click the lower right *"Get DLCs for AppID"* button to fetch all available DLCs for the game.
-* Select a language and tick the options if needed.
-* Click on *"Save"*.
-
-## License
-Auto-CreamAPI itself is licensed under the GNU General Public License v3.0
-
-CreamAPI © 2016-2020, deadmau5. All Rights Reserved.
-
-*Dependencies will be listed ASAP.*
-
-## Software used
-
-[](https://www.jetbrains.com/?from=Auto-CreamAPI)
-[Made with software provided by JetBrains s.r.o.](https://www.jetbrains.com/?from=Auto-CreamAPI)
diff --git a/auto-creamapi/Services/CacheService.cs b/auto-creamapi/Services/CacheService.cs
deleted file mode 100644
index 223bbe5..0000000
--- a/auto-creamapi/Services/CacheService.cs
+++ /dev/null
@@ -1,232 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.Diagnostics;
-using System.IO;
-using System.Linq;
-using System.Net.Http;
-using System.Text;
-using System.Text.Json;
-using System.Text.RegularExpressions;
-using System.Threading.Tasks;
-using AngleSharp.Dom;
-using AngleSharp.Html.Parser;
-using auto_creamapi.Models;
-using auto_creamapi.Utils;
-using NinjaNye.SearchExtensions;
-using SteamStorefrontAPI;
-
-namespace auto_creamapi.Services
-{
- public interface ICacheService
- {
- public Task Initialize();
-
- //public Task UpdateCache();
- public IEnumerable GetListOfAppsByName(string name);
- public SteamApp GetAppByName(string name);
- public SteamApp GetAppById(int appid);
- public Task> GetListOfDlc(SteamApp steamApp, bool useSteamDb, bool ignoreUnknown);
- }
-
- public class CacheService : ICacheService
- {
- private const string CachePath = "steamapps.json";
- private const string SteamUri = "https://api.steampowered.com/ISteamApps/GetAppList/v2/";
-
- private HashSet _cache = [];
-
- public async Task Initialize()
- {
- MyLogger.Log.Information("Updating cache...");
- var updateNeeded = DateTime.Now.Subtract(File.GetLastWriteTimeUtc(CachePath)).TotalDays >= 1;
- string cacheString;
- if (updateNeeded)
- {
- cacheString = await UpdateCache().ConfigureAwait(false);
- }
- else
- {
- MyLogger.Log.Information("Cache already up to date!");
- // ReSharper disable once MethodHasAsyncOverload
- cacheString = File.ReadAllText(CachePath);
- }
- var steamApps = JsonSerializer.Deserialize(cacheString);
- _cache = new HashSet(steamApps.AppList.Apps);
- MyLogger.Log.Information("Loaded cache into memory!");
- }
-
- private static async Task UpdateCache()
- {
- MyLogger.Log.Information("Getting content from API...");
- var client = new HttpClient();
- var httpCall = client.GetAsync(SteamUri);
- var response = await httpCall.ConfigureAwait(false);
- var readAsStringAsync = response.Content.ReadAsStringAsync();
- var responseBody = await readAsStringAsync.ConfigureAwait(false);
- MyLogger.Log.Information("Got content from API successfully. Writing to file...");
-
- await File.WriteAllTextAsync(CachePath, responseBody, Encoding.UTF8).ConfigureAwait(false);
- var cacheString = responseBody;
- MyLogger.Log.Information("Cache written to file successfully.");
- return cacheString;
- }
-
- public IEnumerable GetListOfAppsByName(string name)
- {
- var listOfAppsByName = _cache.Search(x => x.Name)
- .SetCulture(StringComparison.OrdinalIgnoreCase)
- .ContainingAll(name.Split(' '));
- return listOfAppsByName;
- }
-
- public SteamApp GetAppByName(string name)
- {
- MyLogger.Log.Information("Trying to get app {Name}", name);
- var comparableName = Regex.Replace(name, Misc.SpecialCharsRegex, "").ToLower();
- var app = _cache.FirstOrDefault(x => x.CompareName(comparableName));
- if (app != null) MyLogger.Log.Information("Successfully got app {App}", app);
- return app;
- }
-
- public SteamApp GetAppById(int appid)
- {
- MyLogger.Log.Information("Trying to get app with ID {AppId}", appid);
- var app = _cache.FirstOrDefault(x => x.AppId.Equals(appid));
- if (app != null) MyLogger.Log.Information("Successfully got app {App}", app);
- return app;
- }
-
- public async Task> GetListOfDlc(SteamApp steamApp, bool useSteamDb, bool ignoreUnknown)
- {
- MyLogger.Log.Debug("Start: GetListOfDlc");
- var dlcList = new List();
- try
- {
- if (steamApp != null)
- {
- var steamAppDetails = await AppDetails.GetAsync(steamApp.AppId).ConfigureAwait(false);
- if (steamAppDetails != null)
- {
- MyLogger.Log.Debug("Type for Steam App {Name}: \"{Type}\"", steamApp.Name,
- steamAppDetails.Type);
- if (steamAppDetails.Type == "game" || steamAppDetails.Type == "demo")
- {
- steamAppDetails.DLC.ForEach(x =>
- {
- var result = _cache.FirstOrDefault(y => y.AppId.Equals(x));
- if (result == null) return;
- var dlcDetails = AppDetails.GetAsync(x).Result;
- dlcList.Add(dlcDetails != null
- ? new SteamApp { AppId = dlcDetails.SteamAppId, Name = dlcDetails.Name }
- : new SteamApp { AppId = x, Name = $"Unknown DLC {x}" });
- });
-
- dlcList.ForEach(x => MyLogger.Log.Debug("{AppId}={Name}", x.AppId, x.Name));
- MyLogger.Log.Information("Got DLC successfully...");
-
- // Return if Steam DB is deactivated
- if (!useSteamDb) return dlcList;
-
- string steamDbUrl = $"https://steamdb.info/app/{steamApp.AppId}/dlc/";
-
- var client = new HttpClient();
- string archiveJson = await client.GetStringAsync($"https://archive.org/wayback/available?url={steamDbUrl}");
- var archiveResult = JsonSerializer.Deserialize(archiveJson);
-
- if (archiveResult == null || archiveResult.ArchivedSnapshots.Closest?.Status != "200")
- {
- return dlcList;
- }
-
- //language=regex
- const string pattern = @"^(https?:\/\/web\.archive\.org\/web\/\d+)(\/.+)$";
- const string substitution = "$1id_$2";
- const RegexOptions options = RegexOptions.Multiline;
-
- Regex regex = new(pattern, options);
- string newUrl = regex.Replace(archiveResult.ArchivedSnapshots.Closest.Url, substitution);
-
- //client.DefaultRequestHeaders.UserAgent.ParseAdd(UserAgent);
-
- MyLogger.Log.Information("Get SteamDB App");
- var httpCall = client.GetAsync(newUrl);
- var response = await httpCall.ConfigureAwait(false);
- MyLogger.Log.Debug("{Status}", httpCall.Status.ToString());
- MyLogger.Log.Debug("{Boolean}", response.IsSuccessStatusCode.ToString());
-
- response.EnsureSuccessStatusCode();
-
- var readAsStringAsync = response.Content.ReadAsStringAsync();
- var responseBody = await readAsStringAsync.ConfigureAwait(false);
- MyLogger.Log.Debug("{Status}", readAsStringAsync.Status.ToString());
-
- var parser = new HtmlParser();
- var doc = parser.ParseDocument(responseBody);
- // Console.WriteLine(doc.DocumentElement.OuterHtml);
-
- var query1 = doc.QuerySelector("#dlc");
- if (query1 != null)
- {
- var query2 = query1.QuerySelectorAll(".app");
- foreach (var element in query2)
- {
- var dlcId = element.GetAttribute("data-appid");
- var query3 = element.QuerySelectorAll("td");
- var dlcName = query3 == null
- ? $"Unknown DLC {dlcId}"
- : query3[1].Text().Replace("\n", "").Trim();
-
- if (ignoreUnknown && dlcName.Contains("SteamDB Unknown App"))
- {
- MyLogger.Log.Information("Skipping SteamDB Unknown App {DlcId}", dlcId);
- }
- else
- {
- var dlcApp = new SteamApp { AppId = Convert.ToInt32(dlcId), Name = dlcName };
- var i = dlcList.FindIndex(x => x.AppId.Equals(dlcApp.AppId));
- if (i > -1)
- {
- if (dlcList[i].Name.Contains("Unknown DLC")) dlcList[i] = dlcApp;
- }
- else
- {
- dlcList.Add(dlcApp);
- }
- }
- }
-
- dlcList.ForEach(x => MyLogger.Log.Debug("{AppId}={Name}", x.AppId, x.Name));
- MyLogger.Log.Information("Got DLC from SteamDB successfully...");
- }
- else
- {
- MyLogger.Log.Error("Could not get DLC from SteamDB!");
- }
- }
- else
- {
- MyLogger.Log.Error("Could not get DLC: Steam App is not of type: \"Game\"");
- }
- }
- else
- {
- MyLogger.Log.Error("Could not get DLC: Could not get Steam App details");
- }
- }
- else
- {
- MyLogger.Log.Error("Could not get DLC: Invalid Steam App");
- }
-
- //return dlcList;
- }
- catch (Exception e)
- {
- MyLogger.Log.Error("Could not get DLC!");
- MyLogger.Log.Debug(e.Demystify(), "Exception thrown!");
- }
-
- return dlcList;
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Services/CreamConfigService.cs b/auto-creamapi/Services/CreamConfigService.cs
deleted file mode 100644
index d21f13d..0000000
--- a/auto-creamapi/Services/CreamConfigService.cs
+++ /dev/null
@@ -1,199 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.Collections.ObjectModel;
-using System.IO;
-using System.Linq;
-using System.Text;
-using System.Text.RegularExpressions;
-using auto_creamapi.Models;
-using auto_creamapi.Utils;
-using IniParser;
-using IniParser.Model;
-
-namespace auto_creamapi.Services
-{
- public interface ICreamConfigService
- {
- public CreamConfig Config { get; }
- public void Initialize();
- public void ReadFile(string configFilePath);
- public void SaveFile();
-
- public void SetConfigData(int appId,
- string language,
- bool unlockAll,
- bool extraProtection,
- bool forceOffline,
- string dlcList);
-
- public void SetConfigData(int appId,
- string language,
- bool unlockAll,
- bool extraProtection,
- bool forceOffline,
- List dlcList);
-
- public void SetConfigData(int appId,
- string language,
- bool unlockAll,
- bool extraProtection,
- bool forceOffline,
- IEnumerable dlcList);
-
- public bool ConfigExists();
- }
-
- public class CreamConfigService : ICreamConfigService
- {
- private string _configFilePath;
-
- public CreamConfig Config { get; private set; }
-
- public void Initialize()
- {
- //await Task.Run(() =>
- //{
- //MyLogger.Log.Debug("CreamConfigService: init start");
- Config = new CreamConfig();
- ResetConfigData();
- //MyLogger.Log.Debug("CreamConfigService: init end");
- //});
- }
-
- public void ReadFile(string configFilePath)
- {
- _configFilePath = configFilePath;
- if (File.Exists(configFilePath))
- {
- MyLogger.Log.Information("Config file found @ {ConfigFilePath}, parsing...", configFilePath);
- var parser = new FileIniDataParser();
- var data = parser.ReadFile(_configFilePath, Encoding.UTF8);
-
- ResetConfigData(); // clear previous config data
- Config.AppId = Convert.ToInt32(data["steam"]["appid"]);
- Config.Language = data["steam"]["language"];
- Config.UnlockAll = Convert.ToBoolean(data["steam"]["unlockall"]);
- Config.ExtraProtection = Convert.ToBoolean(data["steam"]["extraprotection"]);
- Config.ForceOffline = Convert.ToBoolean(data["steam"]["forceoffline"]);
-
- var dlcCollection = data["dlc"];
- foreach (var item in dlcCollection)
- //Config.DlcList.Add(int.Parse(item.KeyName), item.Value);
- Config.DlcList.Add(new SteamApp {AppId = int.Parse(item.KeyName), Name = item.Value});
- }
- else
- {
- MyLogger.Log.Information("Config file does not exist @ {ConfigFilePath}, skipping...", configFilePath);
- ResetConfigData();
- }
- }
-
- public void SaveFile()
- {
- var parser = new FileIniDataParser();
- var data = new IniData();
-
- data["steam"]["appid"] = Config.AppId.ToString();
- data["steam"]["language"] = Config.Language;
- data["steam"]["unlockall"] = Config.UnlockAll.ToString();
- data["steam"]["extraprotection"] = Config.ExtraProtection.ToString();
- data["steam"]["forceoffline"] = Config.ForceOffline.ToString();
-
- data.Sections.AddSection("dlc");
- Config.DlcList.ForEach(x => data["dlc"].AddKey(x.AppId.ToString(), x.Name));
- /*foreach (var steamApp in Config.DlcList)
- {
- data["dlc"].AddKey(key.ToString(), value);
- }*/
-
- parser.WriteFile(_configFilePath, data, Encoding.UTF8);
- }
-
- public void SetConfigData(int appId,
- string language,
- bool unlockAll,
- bool extraProtection,
- bool forceOffline,
- string dlcList)
- {
- Config.AppId = appId;
- Config.Language = language;
- Config.UnlockAll = unlockAll;
- Config.ExtraProtection = extraProtection;
- Config.ForceOffline = forceOffline;
- SetDlcFromString(dlcList);
- }
-
- public void SetConfigData(int appId,
- string language,
- bool unlockAll,
- bool extraProtection,
- bool forceOffline,
- List dlcList)
- {
- Config.AppId = appId;
- Config.Language = language;
- Config.UnlockAll = unlockAll;
- Config.ExtraProtection = extraProtection;
- Config.ForceOffline = forceOffline;
- Config.DlcList = dlcList;
- }
-
- public void SetConfigData(int appId,
- string language,
- bool unlockAll,
- bool extraProtection,
- bool forceOffline,
- IEnumerable dlcList)
- {
- Config.AppId = appId;
- Config.Language = language;
- Config.UnlockAll = unlockAll;
- Config.ExtraProtection = extraProtection;
- Config.ForceOffline = forceOffline;
- Config.DlcList = new List(dlcList);
- }
-
- public bool ConfigExists()
- {
- return File.Exists(_configFilePath);
- }
-
- private void ResetConfigData()
- {
- Config.AppId = -1;
- Config.Language = Misc.DefaultLanguageSelection;
- Config.UnlockAll = false;
- Config.ExtraProtection = false;
- Config.ForceOffline = false;
- Config.DlcList.Clear();
- }
-
- private void SetDlcFromString(string dlcList)
- {
- Config.DlcList.Clear();
- var expression = new Regex(@"(?.*) *= *(?.*)");
- using var reader = new StringReader(dlcList);
- string line;
- while ((line = reader.ReadLine()) != null)
- {
- var match = expression.Match(line);
- if (match.Success)
- Config.DlcList.Add(
- new SteamApp {AppId = int.Parse(match.Groups["id"].Value), Name = match.Groups["name"].Value});
- }
- }
- /*private void SetDlcFromAppList(List dlcList)
- {
- Config.DlcList.Clear();
- dlcList.ForEach(x => Config.DlcList.Add(x.AppId, x.Name));
- }*/
-
- public override string ToString()
- {
- var str = $"INI file: {_configFilePath}\n{Config}";
-
- return str;
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Services/CreamDllService.cs b/auto-creamapi/Services/CreamDllService.cs
deleted file mode 100644
index de3c9b7..0000000
--- a/auto-creamapi/Services/CreamDllService.cs
+++ /dev/null
@@ -1,115 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.IO;
-using System.Security.Cryptography;
-using System.Threading.Tasks;
-using auto_creamapi.Models;
-using auto_creamapi.Utils;
-
-namespace auto_creamapi.Services
-{
- public interface ICreamDllService
- {
- public string TargetPath { get; set; }
- public Task Initialize();
- public void Save();
- public void CheckIfDllExistsAtTarget();
- public bool CreamApiApplied();
- public bool CreamApiApplied(string arch);
- }
-
- public class CreamDllService : ICreamDllService
- {
- private const string X86Arch = "x86";
- private const string X64Arch = "x64";
- private static readonly string HashPath = Path.Combine(Directory.GetCurrentDirectory(), "cream_api.md5");
-
- private Dictionary _creamDlls;
- private bool _x64Exists;
-
- private bool _x86Exists;
-
- public string TargetPath { get; set; }
-
- public async Task Initialize()
- {
- MyLogger.Log.Debug("CreamDllService: Initialize begin");
- _creamDlls = new Dictionary
- {
- {X86Arch, new CreamDll("steam_api.dll", "steam_api_o.dll")},
- {X64Arch, new CreamDll("steam_api64.dll", "steam_api64_o.dll")}
- };
-
- if (!File.Exists(HashPath))
- {
- MyLogger.Log.Information("Writing md5sum file...");
- await File.WriteAllLinesAsync(HashPath,
- new[]
- {
- $"{_creamDlls[X86Arch].Hash} {_creamDlls[X86Arch].Filename}",
- $"{_creamDlls[X64Arch].Hash} {_creamDlls[X64Arch].Filename}"
- }).ConfigureAwait(false);
- }
-
- MyLogger.Log.Debug("CreamDllService: Initialize end");
- }
-
- public void Save()
- {
- if (_x86Exists) CopyDll(X86Arch);
- if (_x64Exists) CopyDll(X64Arch);
- }
-
- public void CheckIfDllExistsAtTarget()
- {
- var x86File = Path.Combine(TargetPath, "steam_api.dll");
- var x64File = Path.Combine(TargetPath, "steam_api64.dll");
- _x86Exists = File.Exists(x86File);
- _x64Exists = File.Exists(x64File);
- if (_x86Exists) MyLogger.Log.Information("x86 SteamAPI DLL found: {X}", x86File);
- if (_x64Exists) MyLogger.Log.Information("x64 SteamAPI DLL found: {X}", x64File);
- }
-
- public bool CreamApiApplied()
- {
- var a = CreamApiApplied("x86");
- var b = CreamApiApplied("x64");
- return a | b;
- }
-
- private void CopyDll(string arch)
- {
- var sourceSteamApiDll = _creamDlls[arch].Filename;
- var targetSteamApiDll = Path.Combine(TargetPath, _creamDlls[arch].Filename);
- var targetSteamApiOrigDll = Path.Combine(TargetPath, _creamDlls[arch].OrigFilename);
- var targetSteamApiDllBackup = Path.Combine(TargetPath, $"{_creamDlls[arch].Filename}.backup");
- MyLogger.Log.Information("Setting up CreamAPI DLL @ {TargetPath} (arch :{Arch})", TargetPath, arch);
- // Create backup of steam_api.dll
- File.Copy(targetSteamApiDll, targetSteamApiDllBackup, true);
- // Check if steam_api_o.dll already exists
- // If missing rename original file
- if (!File.Exists(targetSteamApiOrigDll))
- File.Move(targetSteamApiDll, targetSteamApiOrigDll, true);
- // Copy creamapi dll
- File.Copy(sourceSteamApiDll, targetSteamApiDll, true);
- }
-
- public bool CreamApiApplied(string arch)
- {
- var a = File.Exists(Path.Combine(TargetPath, _creamDlls[arch].OrigFilename));
- var b = GetHash(Path.Combine(TargetPath, _creamDlls[arch].Filename)).Equals(_creamDlls[arch].Hash);
- return a & b;
- }
-
- private static string GetHash(string filename)
- {
- if (!File.Exists(filename)) return "";
- using var md5 = MD5.Create();
- using var stream = File.OpenRead(filename);
- return BitConverter
- .ToString(md5.ComputeHash(stream))
- .Replace("-", string.Empty);
-
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Services/DownloadCreamApiService.cs b/auto-creamapi/Services/DownloadCreamApiService.cs
deleted file mode 100644
index ba9f75c..0000000
--- a/auto-creamapi/Services/DownloadCreamApiService.cs
+++ /dev/null
@@ -1,143 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.IO;
-using System.Linq;
-using System.Net;
-using System.Net.Http;
-using System.Text.RegularExpressions;
-using System.Threading.Tasks;
-using auto_creamapi.Messenger;
-using auto_creamapi.Utils;
-using HttpProgress;
-using MvvmCross.Plugin.Messenger;
-using SevenZip;
-
-namespace auto_creamapi.Services
-{
- public interface IDownloadCreamApiService
- {
- public Task Download(string username, string password);
- public Task Extract(string filename);
- }
-
- public class DownloadCreamApiService : IDownloadCreamApiService
- {
- private const string ArchivePassword = "cs.rin.ru";
- private readonly IMvxMessenger _messenger;
-
- public DownloadCreamApiService(IMvxMessenger messenger)
- {
- _messenger = messenger;
- }
-
- public async Task Download(string username, string password)
- {
- MyLogger.Log.Debug("Download");
- var container = new CookieContainer();
- var handler = new HttpClientHandler {CookieContainer = container};
- var client = new HttpClient(handler);
- client.DefaultRequestHeaders.UserAgent.ParseAdd("Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:86.0) " +
- "Gecko/20100101 Firefox/86.0");
- var formContent = new FormUrlEncodedContent(new[]
- {
- new KeyValuePair("username", username),
- new KeyValuePair("password", password),
- new KeyValuePair("redirect", "./ucp.php?mode=login"),
- new KeyValuePair("login", "login")
- });
- MyLogger.Log.Debug("Download: post login");
- var response1 = await client.PostAsync("https://cs.rin.ru/forum/ucp.php?mode=login", formContent)
- .ConfigureAwait(false);
- MyLogger.Log.Debug("Login Status Code: {StatusCode}",
- response1.EnsureSuccessStatusCode().StatusCode);
- var cookie = container.GetCookies(new Uri("https://cs.rin.ru/forum/ucp.php?mode=login"))
- .FirstOrDefault(c => c.Name.Contains("_sid"));
- MyLogger.Log.Debug("Login Cookie: {Cookie}", cookie);
- var response2 = await client.GetAsync("https://cs.rin.ru/forum/viewtopic.php?t=70576")
- .ConfigureAwait(false);
- MyLogger.Log.Debug("Download Page Status Code: {StatusCode}",
- response2.EnsureSuccessStatusCode().StatusCode);
- var content = response2.Content.ReadAsStringAsync();
- var contentResult = await content.ConfigureAwait(false);
-
- var expression =
- new Regex(".*\\/download\\/file\\.php\\?id=.*)\">(?.*)<\\/a>.*");
- using var reader = new StringReader(contentResult);
- string line;
- var archiveFileList = new Dictionary();
- while ((line = await reader.ReadLineAsync().ConfigureAwait(false)) != null)
- {
- var match = expression.Match(line);
- // ReSharper disable once InvertIf
- if (match.Success)
- {
- archiveFileList.Add(match.Groups["filename"].Value,
- $"https://cs.rin.ru/forum{match.Groups["url"].Value}");
- MyLogger.Log.Debug("{X}", archiveFileList.LastOrDefault().Key);
- }
- }
-
- MyLogger.Log.Debug("Choosing first element from list...");
- var (filename, url) = archiveFileList.FirstOrDefault();
- if (File.Exists(filename))
- {
- MyLogger.Log.Information("{Filename} already exists, skipping download...", filename);
- return filename;
- }
-
- MyLogger.Log.Information("Start download...");
- var progress = new Progress(
- x => _messenger.Publish(new ProgressMessage(this, "Downloading...", filename, x)));
- await using var fileStream = File.OpenWrite(filename);
- var task = client.GetAsync(url, fileStream, progress);
- await task.ConfigureAwait(false);
- if (task.IsCompletedSuccessfully)
- _messenger.Publish(new ProgressMessage(this, "Downloading...", filename, 1.0));
- MyLogger.Log.Information("Download done.");
- return filename;
- }
-
- public async Task Extract(string filename)
- {
- MyLogger.Log.Debug("Extract");
- var cwd = Directory.GetCurrentDirectory();
- const string nonlogBuild = "nonlog_build";
- const string steamApi64Dll = "steam_api64.dll";
- const string steamApiDll = "steam_api.dll";
- MyLogger.Log.Information(@"Start extraction of ""{Filename}""...", filename);
- var nonlogBuildPath = Path.Combine(cwd, nonlogBuild);
- if (Directory.Exists(nonlogBuildPath))
- Directory.Delete(nonlogBuildPath, true);
- _messenger.Publish(new ProgressMessage(this, "Extracting...", filename, 1.0));
- SevenZipBase.SetLibraryPath(Path.Combine(cwd, "resources/7z.dll"));
- using (var extractor =
- new SevenZipExtractor(filename, ArchivePassword, InArchiveFormat.Rar)
- {PreserveDirectoryStructure = false})
- {
- await extractor.ExtractFilesAsync(
- cwd,
- $@"{nonlogBuild}\{steamApi64Dll}",
- $@"{nonlogBuild}\{steamApiDll}"
- ).ConfigureAwait(false);
- }
-
- if (File.Exists(Path.Combine(nonlogBuildPath, steamApi64Dll)))
- File.Move(
- Path.Combine(cwd, nonlogBuild, steamApi64Dll),
- Path.Combine(cwd, steamApi64Dll),
- true
- );
-
- if (File.Exists(Path.Combine(nonlogBuildPath, steamApiDll)))
- File.Move(
- Path.Combine(nonlogBuildPath, steamApiDll),
- Path.Combine(cwd, steamApiDll),
- true
- );
-
- if (Directory.Exists(nonlogBuildPath))
- Directory.Delete(nonlogBuildPath, true);
- MyLogger.Log.Information("Extraction done!");
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Setup.cs b/auto-creamapi/Setup.cs
deleted file mode 100644
index 9a5682a..0000000
--- a/auto-creamapi/Setup.cs
+++ /dev/null
@@ -1,29 +0,0 @@
-using auto_creamapi.Core;
-using auto_creamapi.Utils;
-using Microsoft.Extensions.Logging;
-using MvvmCross.Platforms.Wpf.Core;
-using Serilog;
-using Serilog.Extensions.Logging;
-using System;
-using System.Collections.Generic;
-using System.Linq;
-using System.Text;
-using System.Threading.Tasks;
-
-namespace auto_creamapi
-{
- public class Setup : MvxWpfSetup
- {
- protected override ILoggerFactory CreateLogFactory()
- {
- Log.Logger = MyLogger.Log;
-
- return new SerilogLoggerFactory();
- }
-
- protected override ILoggerProvider CreateLogProvider()
- {
- return new SerilogLoggerProvider();
- }
- }
-}
diff --git a/auto-creamapi/Utils/AvailabeArchive.cs b/auto-creamapi/Utils/AvailabeArchive.cs
deleted file mode 100644
index f7561b1..0000000
--- a/auto-creamapi/Utils/AvailabeArchive.cs
+++ /dev/null
@@ -1,35 +0,0 @@
-using System.Text.Json.Serialization;
-
-namespace auto_creamapi.Utils
-{
-
- public class AvailableArchive
- {
- [JsonPropertyName("url")]
- public string Url { get; set; }
-
- [JsonPropertyName("archived_snapshots")]
- public ArchivedSnapshot ArchivedSnapshots { get; set; }
- }
-
- public class ArchivedSnapshot
- {
- [JsonPropertyName("closest")]
- public Closest Closest { get; set; }
- }
-
- public class Closest
- {
- [JsonPropertyName("status")]
- public string Status { get; set; }
-
- [JsonPropertyName("available")]
- public bool Available { get; set; }
-
- [JsonPropertyName("url")]
- public string Url { get; set; }
-
- [JsonPropertyName("timestamp")]
- public string Timestamp { get; set; }
- }
-}
diff --git a/auto-creamapi/Utils/ISecrets.cs b/auto-creamapi/Utils/ISecrets.cs
deleted file mode 100644
index 01840ac..0000000
--- a/auto-creamapi/Utils/ISecrets.cs
+++ /dev/null
@@ -1,8 +0,0 @@
-namespace auto_creamapi.Utils
-{
- public interface ISecrets
- {
- public string ForumUsername();
- public string ForumPassword();
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/Utils/Misc.cs b/auto-creamapi/Utils/Misc.cs
deleted file mode 100644
index 5e0883c..0000000
--- a/auto-creamapi/Utils/Misc.cs
+++ /dev/null
@@ -1,43 +0,0 @@
-using System.Collections.Generic;
-using System.Collections.ObjectModel;
-
-namespace auto_creamapi.Utils
-{
- public static class Misc
- {
- public const string SpecialCharsRegex = "[^0-9a-zA-Z]+";
- public const string DefaultLanguageSelection = "english";
- public static readonly ObservableCollection DefaultLanguages = new(new[]
- {
- "arabic",
- "bulgarian",
- "schinese",
- "tchinese",
- "czech",
- "danish",
- "dutch",
- "english",
- "finnish",
- "french",
- "german",
- "greek",
- "hungarian",
- "italian",
- "japanese",
- "koreana",
- "norwegian",
- "polish",
- "portuguese",
- "brazilian",
- "romanian",
- "russian",
- "spanish",
- "latam",
- "swedish",
- "thai",
- "turkish",
- "ukrainian",
- "vietnamese"
- });
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/ViewModels/DownloadViewModel.cs b/auto-creamapi/ViewModels/DownloadViewModel.cs
deleted file mode 100644
index 238509c..0000000
--- a/auto-creamapi/ViewModels/DownloadViewModel.cs
+++ /dev/null
@@ -1,107 +0,0 @@
-using System;
-using System.Threading.Tasks;
-using System.Windows;
-using auto_creamapi.Messenger;
-using auto_creamapi.Services;
-using auto_creamapi.Utils;
-using Microsoft.Extensions.Logging;
-using MvvmCross.Navigation;
-using MvvmCross.Plugin.Messenger;
-using MvvmCross.ViewModels;
-
-namespace auto_creamapi.ViewModels
-{
- public class DownloadViewModel : MvxNavigationViewModel
- {
- private readonly IDownloadCreamApiService _download;
- private readonly IMvxNavigationService _navigationService;
- private readonly MvxSubscriptionToken _token;
- private readonly ILogger _logger;
- private string _filename;
-
- private string _info;
- private double _progress;
-
- private readonly Secrets _secrets = new();
-
- public DownloadViewModel(ILoggerFactory loggerFactory, IMvxNavigationService navigationService,
- IDownloadCreamApiService download, IMvxMessenger messenger) : base(loggerFactory, navigationService)
- {
- _navigationService = navigationService;
- _logger = loggerFactory.CreateLogger();
- _download = download;
- _token = messenger.Subscribe(OnProgressMessage);
- _logger.LogDebug("{Count}", messenger.CountSubscriptionsFor());
- }
-
- public string InfoLabel
- {
- get => _info;
- set
- {
- _info = value;
- RaisePropertyChanged(() => InfoLabel);
- }
- }
-
- public string FilenameLabel
- {
- get => _filename;
- set
- {
- _filename = value;
- RaisePropertyChanged(() => FilenameLabel);
- }
- }
-
- public double Progress
- {
- get => _progress;
- set
- {
- _progress = value;
- RaisePropertyChanged(() => Progress);
- RaisePropertyChanged(() => ProgressPercent);
- }
- }
-
- public string ProgressPercent => _progress.ToString("P2");
-
- public override void Prepare()
- {
- InfoLabel = "Please wait...";
- FilenameLabel = "";
- Progress = 0.0;
- }
-
- public override async Task Initialize()
- {
- try
- {
- await base.Initialize().ConfigureAwait(false);
- var download = _download.Download(_secrets.ForumUsername(), _secrets.ForumPassword());
- var filename = await download.ConfigureAwait(false);
- var extract = _download.Extract(filename);
- await extract.ConfigureAwait(false);
- _token.Dispose();
- await _navigationService.Close(this).ConfigureAwait(false);
- }
- catch (Exception e)
- {
- MessageBox.Show("Could not download CreamAPI!\nPlease add CreamAPI DLLs manually!\nShutting down...",
- "Error", MessageBoxButton.OK, MessageBoxImage.Error);
- _token.Dispose();
- await _navigationService.Close(this).ConfigureAwait(false);
- Console.WriteLine(e);
- throw;
- }
- }
-
- private void OnProgressMessage(ProgressMessage obj)
- {
- InfoLabel = obj.Info;
- FilenameLabel = obj.Filename;
- Progress = obj.PercentComplete;
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/ViewModels/MainViewModel.cs b/auto-creamapi/ViewModels/MainViewModel.cs
deleted file mode 100644
index 296164d..0000000
--- a/auto-creamapi/ViewModels/MainViewModel.cs
+++ /dev/null
@@ -1,441 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.Collections.ObjectModel;
-using System.Diagnostics;
-using System.IO;
-using System.Threading.Tasks;
-using auto_creamapi.Models;
-using auto_creamapi.Services;
-using auto_creamapi.Utils;
-using Microsoft.Extensions.Logging;
-using Microsoft.Win32;
-using MvvmCross.Commands;
-using MvvmCross.Navigation;
-using MvvmCross.ViewModels;
-
-namespace auto_creamapi.ViewModels
-{
- public class MainViewModel : MvxViewModel
- {
- private readonly ICacheService _cache;
- private readonly ICreamConfigService _config;
-
- private readonly ILogger _logger;
- private readonly ICreamDllService _dll;
- private readonly IMvxNavigationService _navigationService;
- private int _appId;
- private bool _configExists;
- private ObservableCollection _dlcs;
- private bool _dllApplied;
- private string _dllPath;
- private bool _extraProtection;
- private string _gameName;
- private string _lang;
- private ObservableCollection _languages;
-
- //private readonly IDownloadCreamApiService _download;
- private bool _mainWindowEnabled;
- private bool _offline;
- private string _status;
- private bool _unlockAll;
-
- private bool _useSteamDb;
-
- private bool _ignoreUnknown;
- //private const string DlcRegexPattern = @"(?.*) *= *(?.*)";
-
- public MainViewModel(ICacheService cache, ICreamConfigService config, ICreamDllService dll,
- IMvxNavigationService navigationService, ILoggerFactory loggerFactory)
- {
- _navigationService = navigationService;
- _logger = loggerFactory.CreateLogger();
- _cache = cache;
- _config = config;
- _dll = dll;
- //_download = download;
- }
-
- public override async void Prepare()
- {
- base.Prepare();
- _config.Initialize();
- var tasks = new List { _cache.Initialize() };
- if (!File.Exists("steam_api.dll") | !File.Exists("steam_api64.dll"))
- tasks.Add(_navigationService.Navigate());
- //tasks.Add(_navigationService.Navigate());
- tasks.Add(_dll.Initialize());
- await Task.WhenAll(tasks).ConfigureAwait(false);
- Languages = new ObservableCollection(Misc.DefaultLanguages);
- ResetForm();
- UseSteamDb = true;
- MainWindowEnabled = true;
- Status = "Ready.";
- }
-
- // // COMMANDS // //
-
- public IMvxCommand OpenFileCommand => new MvxAsyncCommand(OpenFile);
-
- public IMvxCommand SearchCommand => new MvxAsyncCommand(async () => await Search().ConfigureAwait(false)); //Command(Search);
-
- public IMvxCommand GetListOfDlcCommand => new MvxAsyncCommand(GetListOfDlc);
-
- public IMvxCommand SaveCommand => new MvxCommand(Save);
-
- public IMvxCommand ResetFormCommand => new MvxCommand(ResetForm);
-
- public IMvxCommand GoToForumThreadCommand => new MvxCommand(GoToForumThread);
-
- public IMvxCommand GoToSteamdbCommand => new MvxCommand(GoToSteamdb);
-
- // // ATTRIBUTES // //
-
- public bool MainWindowEnabled
- {
- get => _mainWindowEnabled;
- set
- {
- _mainWindowEnabled = value;
- RaisePropertyChanged(() => MainWindowEnabled);
- }
- }
-
- public string DllPath
- {
- get => _dllPath;
- set
- {
- _dllPath = value;
- RaisePropertyChanged(() => DllPath);
- }
- }
-
- public string GameName
- {
- get => _gameName;
- set
- {
- _gameName = value;
- RaisePropertyChanged(() => GameName);
- }
- }
-
- public int AppId
- {
- get => _appId;
- set
- {
- _appId = value;
- RaisePropertyChanged(() => AppId);
- SetNameById();
- }
- }
-
- public string Lang
- {
- get => _lang;
- set
- {
- _lang = value;
- RaisePropertyChanged(() => Lang);
- }
- }
-
- public bool Offline
- {
- get => _offline;
- set
- {
- _offline = value;
- RaisePropertyChanged(() => Offline);
- }
- }
-
- public bool ExtraProtection
- {
- get => _extraProtection;
- set
- {
- _extraProtection = value;
- RaisePropertyChanged(() => ExtraProtection);
- }
- }
-
- public bool UnlockAll
- {
- get => _unlockAll;
- set
- {
- _unlockAll = value;
- RaisePropertyChanged(() => UnlockAll);
- }
- }
-
- public bool UseSteamDb
- {
- get => _useSteamDb;
- set
- {
- _useSteamDb = value;
- RaisePropertyChanged(() => UseSteamDb);
- }
- }
-
- public ObservableCollection Dlcs
- {
- get => _dlcs;
- set
- {
- _dlcs = value;
- RaisePropertyChanged(() => Dlcs);
- }
- }
-
- public bool DllApplied
- {
- get => _dllApplied;
- set
- {
- _dllApplied = value;
- RaisePropertyChanged(() => DllApplied);
- }
- }
-
- public bool ConfigExists
- {
- get => _configExists;
- set
- {
- _configExists = value;
- RaisePropertyChanged(() => ConfigExists);
- }
- }
-
- public string Status
- {
- get => _status;
- set
- {
- _status = value;
- RaisePropertyChanged(() => Status);
- }
- }
-
- public ObservableCollection Languages
- {
- get => _languages;
- set
- {
- _languages = value;
- RaisePropertyChanged(() => Languages);
- }
- }
-
- public bool IgnoreUnknown
- {
- get => _ignoreUnknown;
- set
- {
- _ignoreUnknown = value;
- RaisePropertyChanged(() => IgnoreUnknown);
- }
- }
-
- private async Task OpenFile()
- {
- Status = "Waiting for file...";
- var dialog = new OpenFileDialog
- {
- Filter = "SteamAPI DLL|steam_api.dll;steam_api64.dll|" +
- "All files (*.*)|*.*",
- Multiselect = false,
- Title = "Select SteamAPI DLL..."
- };
- if (dialog.ShowDialog() == true)
- {
- var filePath = dialog.FileName;
- DllPath = filePath;
- var dirPath = Path.GetDirectoryName(filePath);
- if (dirPath != null)
- {
- _config.ReadFile(Path.Combine(dirPath, "cream_api.ini"));
- ResetForm();
- _dll.TargetPath = dirPath;
- _dll.CheckIfDllExistsAtTarget();
- CheckSetupStatus();
- if (!ConfigExists)
- {
- var separator = Path.DirectorySeparatorChar;
- var strings = new List(dirPath.Split(separator));
- var index = strings.Contains("common") ? strings.FindIndex(x => x.Equals("common")) + 1 : -1;
- if (index == -1)
- index = strings.Contains("steamapps")
- ? strings.FindIndex(x => x.Equals("steamapps")) + 2
- : -1;
- var s = index > -1 ? strings[index] : null;
- if (s != null) GameName = s;
- await Search().ConfigureAwait(false);
- // await GetListOfDlc().ConfigureAwait(false);
- }
-
- Status = "Ready.";
- }
- }
- else
- {
- Status = "File selection canceled.";
- }
- }
-
- private async Task Search()
- {
- if (!string.IsNullOrEmpty(GameName))
- {
- var app = _cache.GetAppByName(GameName);
- if (app != null)
- {
- GameName = app.Name;
- AppId = app.AppId;
- }
- else
- {
- MainWindowEnabled = false;
- var navigate = _navigationService.Navigate, SteamApp>(
- _cache.GetListOfAppsByName(GameName));
- await navigate.ConfigureAwait(false);
- var navigateResult = navigate.Result;
- if (navigateResult != null)
- {
- GameName = navigateResult.Name;
- AppId = navigateResult.AppId;
- }
- }
-
- // await GetListOfDlc().ConfigureAwait(false);
- }
- else
- {
- _logger.LogWarning("Empty game name, cannot initiate search!");
- }
-
- MainWindowEnabled = true;
- }
-
- private async Task GetListOfDlc()
- {
- Status = "Trying to get DLC, please wait...";
- if (AppId > 0)
- {
- var app = new SteamApp { AppId = AppId, Name = GameName };
- var task = _cache.GetListOfDlc(app, UseSteamDb, IgnoreUnknown);
- MainWindowEnabled = false;
- var listOfDlc = await task.ConfigureAwait(false);
- if (task.IsCompletedSuccessfully)
- {
- listOfDlc.Sort((app1, app2) => app1.AppId.CompareTo(app2.AppId));
- Dlcs = new ObservableCollection(listOfDlc);
- Status = $"Got DLC for AppID {AppId} (Count: {Dlcs.Count})";
- }
- else
- {
- Status = $"Could not get DLC for AppID {AppId}";
- }
-
- MainWindowEnabled = true;
- }
- else
- {
- Status = $"Could not get DLC for AppID {AppId}";
- _logger.LogError("GetListOfDlc: Invalid AppID {AppId}", AppId);
- }
- }
-
- private void Save()
- {
- Status = "Saving...";
- _config.SetConfigData(
- AppId,
- Lang,
- UnlockAll,
- ExtraProtection,
- Offline,
- Dlcs
- );
- _config.SaveFile();
- _dll.Save();
- CheckSetupStatus();
- Status = "Saving successful.";
- }
-
- private void ResetForm()
- {
- AppId = _config.Config.AppId;
- Lang = _config.Config.Language;
- UnlockAll = _config.Config.UnlockAll;
- ExtraProtection = _config.Config.ExtraProtection;
- Offline = _config.Config.ForceOffline;
- Dlcs = new ObservableCollection(_config.Config.DlcList);
- Status = "Changes have been reset.";
- }
-
- private void GoToForumThread()
- {
- Status = "Opening URL...";
- if (AppId > 0)
- {
- var searchTerm = AppId; //$"{GameName.Replace(" ", "+")}+{appId}";
- var destinationUrl =
- $"https://cs.rin.ru/forum/search.php?keywords={searchTerm}&terms=any&fid[]=10&sf=firstpost&sr=topics&submit=Search";
- var uri = new Uri(destinationUrl);
- var process = new ProcessStartInfo(uri.AbsoluteUri)
- {
- UseShellExecute = true
- };
- Process.Start(process);
- }
- else
- {
- _logger.LogError("OpenURL: Invalid AppID {AppId}", AppId);
- Status = $"Could not open URL: Invalid AppID {AppId}";
- }
- }
-
- private void GoToSteamdb()
- {
- Status = "Opening URL...";
- if (AppId > 0)
- {
- var searchTerm = AppId; //$"{GameName.Replace(" ", "+")}+{appId}";
- var destinationUrl =
- $"https://steamdb.info/app/{searchTerm}/dlc/";
- var uri = new Uri(destinationUrl);
- var process = new ProcessStartInfo(uri.AbsoluteUri)
- {
- UseShellExecute = true
- };
- Process.Start(process);
- }
- else
- {
- _logger.LogError("OpenURL: Invalid AppID {AppId}", AppId);
- Status = $"Could not open URL: Invalid AppID {AppId}";
- }
- }
-
- private void CheckSetupStatus()
- {
- DllApplied = _dll.CreamApiApplied();
- ConfigExists = _config.ConfigExists();
- }
-
- private void SetNameById()
- {
- if (_appId > 0)
- {
- var appById = _cache.GetAppById(_appId);
- GameName = appById != null ? appById.Name : "";
- }
- else GameName = "";
- }
- }
-}
\ No newline at end of file
diff --git a/auto-creamapi/ViewModels/SearchResultViewModel.cs b/auto-creamapi/ViewModels/SearchResultViewModel.cs
deleted file mode 100644
index 82e401c..0000000
--- a/auto-creamapi/ViewModels/SearchResultViewModel.cs
+++ /dev/null
@@ -1,85 +0,0 @@
-using System.Collections.Generic;
-using System.Threading.Tasks;
-using auto_creamapi.Models;
-using auto_creamapi.Utils;
-using Microsoft.Extensions.Logging;
-using MvvmCross.Commands;
-using MvvmCross.Logging;
-using MvvmCross.Navigation;
-using MvvmCross.ViewModels;
-
-namespace auto_creamapi.ViewModels
-{
- public class SearchResultViewModel : MvxNavigationViewModel>,
- IMvxViewModel, SteamApp>
- {
- private readonly IMvxNavigationService _navigationService;
- private readonly ILogger _logger;
- private IEnumerable _steamApps;
-
- /*public override async Task Initialize()
- {
- await base.Initialize();
- }*/
- public SearchResultViewModel(ILoggerFactory loggerFactory, IMvxNavigationService navigationService) : base(
- loggerFactory, navigationService)
- {
- _navigationService = navigationService;
- _logger = loggerFactory.CreateLogger();
- }
-
- public IEnumerable Apps
- {
- get => _steamApps;
- set
- {
- _steamApps = value;
- RaisePropertyChanged(() => Apps);
- }
- }
-
- public SteamApp Selected
- {
- get;
- set;
- //RaisePropertyChanged(Selected);
- }
-
- public IMvxCommand SaveCommand => new MvxAsyncCommand(Save);
-
- public IMvxCommand CloseCommand => new MvxCommand(Close);
-
- public override void Prepare(IEnumerable parameter)
- {
- Apps = parameter;
- }
-
- public TaskCompletionSource