diff --git a/Wabbajack.Common/Utils.cs b/Wabbajack.Common/Utils.cs index 73d2e57d..0e3836f8 100644 --- a/Wabbajack.Common/Utils.cs +++ b/Wabbajack.Common/Utils.cs @@ -5,6 +5,7 @@ using System.Diagnostics; using System.IO; using System.Linq; using System.Net.Http; +using System.Reactive.Linq; using System.Reactive.Subjects; using System.Reflection; using System.Runtime.InteropServices; @@ -39,6 +40,14 @@ namespace Wabbajack.Common } public static string LogFile { get; private set; } + + public enum FileEventType + { + Created, + Changed, + Deleted + } + static Utils() { var programName = Assembly.GetEntryAssembly()?.Location ?? "Wabbajack"; @@ -47,6 +56,13 @@ namespace Wabbajack.Common if (LogFile.FileExists()) File.Delete(LogFile); + + var watcher = new FileSystemWatcher(Consts.LocalAppDataPath); + AppLocalEvents = Observable.Merge(Observable.FromEventPattern(h => watcher.Changed += h, h => watcher.Changed -= h).Select(e => (FileEventType.Changed, e.EventArgs)), + Observable.FromEventPattern(h => watcher.Created += h, h => watcher.Created -= h).Select(e => (FileEventType.Created, e.EventArgs)), + Observable.FromEventPattern(h => watcher.Deleted += h, h => watcher.Deleted -= h).Select(e => (FileEventType.Deleted, e.EventArgs))) + .ObserveOn(RxApp.TaskpoolScheduler); + watcher.EnableRaisingEvents = true; } private static readonly Subject LoggerSubj = new Subject(); @@ -961,12 +977,37 @@ namespace Wabbajack.Common public static T FromEncryptedJson(string key) { - var path = Path.Combine(KnownFolders.LocalAppData.Path, "Wabbajack", key); + var path = Path.Combine(Consts.LocalAppDataPath, key); var bytes = File.ReadAllBytes(path); var decoded = ProtectedData.Unprotect(bytes, Encoding.UTF8.GetBytes(key), DataProtectionScope.LocalMachine); return Encoding.UTF8.GetString(decoded).FromJSONString(); } + public static bool HaveEncryptedJson(string key) + { + var path = Path.Combine(Consts.LocalAppDataPath, key); + return File.Exists(path); + } + + public static IObservable<(FileEventType, FileSystemEventArgs)> AppLocalEvents { get; } + + public static IObservable HaveEncryptedJsonObservable(string key) + { + var path = Path.Combine(Consts.LocalAppDataPath, key).ToLower(); + return AppLocalEvents.Where(t => t.Item2.FullPath.ToLower() == path) + .Select(_ => File.Exists(path)) + .StartWith(File.Exists(path)) + .DistinctUntilChanged(); + } + + public static void DeleteEncryptedJson(string key) + { + var path = Path.Combine(Consts.LocalAppDataPath, key); + if (File.Exists(path)) + File.Delete(path); + } + + public static bool IsInPath(this string path, string parent) { return path.ToLower().TrimEnd('\\').StartsWith(parent.ToLower().TrimEnd('\\') + "\\"); diff --git a/Wabbajack.Lib/Downloaders/INeedsLogin.cs b/Wabbajack.Lib/Downloaders/INeedsLogin.cs new file mode 100644 index 00000000..68ab9d09 --- /dev/null +++ b/Wabbajack.Lib/Downloaders/INeedsLogin.cs @@ -0,0 +1,22 @@ +using System; +using System.Collections.Generic; +using System.ComponentModel; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using System.Windows.Input; + +namespace Wabbajack.Lib.Downloaders +{ + public interface INeedsLogin : INotifyPropertyChanged + { + ICommand TriggerLogin { get; } + ICommand ClearLogin { get; } + IObservable IsLoggedIn { get; } + string SiteName { get; } + string MetaInfo { get; } + Uri SiteURL { get; } + Uri IconUri { get; } + + } +} diff --git a/Wabbajack.Lib/Downloaders/LoversLabDownloader.cs b/Wabbajack.Lib/Downloaders/LoversLabDownloader.cs index d6891951..56027549 100644 --- a/Wabbajack.Lib/Downloaders/LoversLabDownloader.cs +++ b/Wabbajack.Lib/Downloaders/LoversLabDownloader.cs @@ -4,22 +4,47 @@ using System.IO; using System.Linq; using System.Net; using System.Net.Http; +using System.Reactive.Linq; using System.Text.RegularExpressions; using System.Threading; using System.Threading.Tasks; using System.Web; +using System.Windows.Input; +using ReactiveUI; using Wabbajack.Common; using Wabbajack.Lib.LibCefHelpers; +using Wabbajack.Lib.NexusApi; using Wabbajack.Lib.Validation; using Xilium.CefGlue.Common; using File = Alphaleonis.Win32.Filesystem.File; namespace Wabbajack.Lib.Downloaders { - public class LoversLabDownloader : IDownloader + public class LoversLabDownloader : ViewModel, IDownloader, INeedsLogin { internal HttpClient _authedClient; + + #region INeedsDownload + + public ICommand TriggerLogin { get; } + public ICommand ClearLogin { get; } + public IObservable IsLoggedIn => Utils.HaveEncryptedJsonObservable("loverslabcookies"); + public string SiteName => "Lovers Lab"; + public string MetaInfo => ""; + public Uri SiteURL => new Uri("https://loverslab.com"); + public Uri IconUri => new Uri("https://www.loverslab.com/favicon.ico"); + + + #endregion + + public LoversLabDownloader() + { + TriggerLogin = ReactiveCommand.Create(async () => await Utils.Log(new RequestLoversLabLogin()).Task, IsLoggedIn.Select(b => !b).ObserveOn(RxApp.MainThreadScheduler)); + ClearLogin = ReactiveCommand.Create(() => Utils.DeleteEncryptedJson("loverslabcookies"), IsLoggedIn.ObserveOn(RxApp.MainThreadScheduler)); + } + + public async Task GetDownloaderState(dynamic archive_ini) { Uri url = DownloaderUtils.GetDirectURL(archive_ini); @@ -84,7 +109,7 @@ namespace Wabbajack.Lib.Downloaders } catch (FileNotFoundException) { } - cookies = Utils.Log(new RequestLoversLabLogin()).Task.Result; + cookies = await Utils.Log(new RequestLoversLabLogin()).Task; return Helpers.GetClient(cookies, "https://www.loverslab.com"); } @@ -176,6 +201,7 @@ namespace Wabbajack.Lib.Downloaders return $"* Lovers Lab - [{a.Name}](https://www.loverslab.com/files/file/{FileName}/?do=download&r={FileID})"; } } + } public class RequestLoversLabLogin : AUserIntervention diff --git a/Wabbajack.Lib/Downloaders/NexusDownloader.cs b/Wabbajack.Lib/Downloaders/NexusDownloader.cs index f94550f0..b589cf47 100644 --- a/Wabbajack.Lib/Downloaders/NexusDownloader.cs +++ b/Wabbajack.Lib/Downloaders/NexusDownloader.cs @@ -1,7 +1,11 @@ using System; +using System.ComponentModel; using System.Linq; +using System.Reactive.Linq; using System.Threading; using System.Threading.Tasks; +using System.Windows.Input; +using ReactiveUI; using Wabbajack.Common; using Wabbajack.Common.StatusFeed.Errors; using Wabbajack.Lib.NexusApi; @@ -9,13 +13,39 @@ using Wabbajack.Lib.Validation; namespace Wabbajack.Lib.Downloaders { - public class NexusDownloader : IDownloader + public class NexusDownloader : ViewModel, IDownloader, INeedsLogin { private bool _prepared; private SemaphoreSlim _lock = new SemaphoreSlim(1); private UserStatus _status; private NexusApiClient _client; + public NexusDownloader() + { + TriggerLogin = ReactiveCommand.Create(async () => await NexusApiClient.RequestAndCacheAPIKey(), IsLoggedIn.Select(b => !b).ObserveOn(RxApp.MainThreadScheduler)); + ClearLogin = ReactiveCommand.Create(() => Utils.DeleteEncryptedJson("nexusapikey"), IsLoggedIn.ObserveOn(RxApp.MainThreadScheduler)); + } + + public IObservable IsLoggedIn => Utils.HaveEncryptedJsonObservable("nexusapikey"); + + public string SiteName => "Nexus Mods"; + + public string MetaInfo + { + get + { + return ""; + } + } + + + public Uri SiteURL => new Uri("https://www.nexusmods.com"); + + public Uri IconUri => new Uri("https://www.nexusmods.com/favicon.ico"); + + public ICommand TriggerLogin { get; } + public ICommand ClearLogin { get; } + public async Task GetDownloaderState(dynamic archiveINI) { var general = archiveINI?.General; diff --git a/Wabbajack.Lib/NexusApi/NexusApi.cs b/Wabbajack.Lib/NexusApi/NexusApi.cs index 2c9bada0..512b1f31 100644 --- a/Wabbajack.Lib/NexusApi/NexusApi.cs +++ b/Wabbajack.Lib/NexusApi/NexusApi.cs @@ -89,9 +89,7 @@ namespace Wabbajack.Lib.NexusApi return env_key; } - var result = await Utils.Log(new RequestNexusAuthorization()).Task; - result.ToEcryptedJson("nexusapikey"); - return result; + return await RequestAndCacheAPIKey(); } finally { @@ -99,6 +97,13 @@ namespace Wabbajack.Lib.NexusApi } } + public static async Task RequestAndCacheAPIKey() + { + var result = await Utils.Log(new RequestNexusAuthorization()).Task; + result.ToEcryptedJson("nexusapikey"); + return result; + } + class RefererHandler : RequestHandler { private string _referer; diff --git a/Wabbajack.Lib/Wabbajack.Lib.csproj b/Wabbajack.Lib/Wabbajack.Lib.csproj index 512c3af5..07d87416 100644 --- a/Wabbajack.Lib/Wabbajack.Lib.csproj +++ b/Wabbajack.Lib/Wabbajack.Lib.csproj @@ -116,6 +116,7 @@ + diff --git a/Wabbajack/UserInterventions/ShowLoginManager.cs b/Wabbajack/UserInterventions/ShowLoginManager.cs new file mode 100644 index 00000000..672f2bb3 --- /dev/null +++ b/Wabbajack/UserInterventions/ShowLoginManager.cs @@ -0,0 +1,21 @@ +using System; +using System.Collections.Generic; +using System.ComponentModel; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using Wabbajack.Common; + +namespace Wabbajack.UserInterventions +{ + public class ShowLoginManager : AUserIntervention + { + public override string ShortDescription => "User requested to show the login manager"; + + public override string ExtendedDescription => "User requested to show the UI for managing all the logins supported by Wabbajack"; + + public override void Cancel() + { + } + } +} diff --git a/Wabbajack/View Models/LoginManagerVM.cs b/Wabbajack/View Models/LoginManagerVM.cs new file mode 100644 index 00000000..7c37cbf3 --- /dev/null +++ b/Wabbajack/View Models/LoginManagerVM.cs @@ -0,0 +1,25 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using System.Windows.Input; +using ReactiveUI; +using Wabbajack.Lib; +using Wabbajack.Lib.Downloaders; + +namespace Wabbajack +{ + public class LoginManagerVM : ViewModel + { + private MainWindowVM mainWindowVM; + public ICommand BackCommand { get; } + public List Downloaders { get; } + + public LoginManagerVM(MainWindowVM mainWindowVM) + { + BackCommand = ReactiveCommand.Create(() => mainWindowVM.NavigateBack()); + Downloaders = DownloadDispatcher.Downloaders.OfType().ToList(); + } + } +} diff --git a/Wabbajack/View Models/MainWindowVM.cs b/Wabbajack/View Models/MainWindowVM.cs index 0f70c6a4..ff51475f 100644 --- a/Wabbajack/View Models/MainWindowVM.cs +++ b/Wabbajack/View Models/MainWindowVM.cs @@ -3,6 +3,7 @@ using DynamicData.Binding; using ReactiveUI; using ReactiveUI.Fody.Helpers; using System; +using System.Collections.Generic; using System.Diagnostics; using System.Linq; using System.Reactive.Disposables; @@ -38,9 +39,17 @@ namespace Wabbajack public readonly ModeSelectionVM ModeSelectionVM; public readonly WebBrowserVM WebBrowserVM; public readonly UserInterventionHandlers UserInterventionHandlers; + public readonly LoginManagerVM LoginManagerVM; + + + public readonly List NavigationTrail = new List(); + public Dispatcher ViewDispatcher { get; set; } public ICommand CopyVersionCommand { get; } + + public ICommand ShowLoginManagerVM { get; } + public ICommand GoBackCommand { get; } public string VersionDisplay { get; } public MainWindowVM(MainWindow mainWindow, MainSettings settings) @@ -54,6 +63,7 @@ namespace Wabbajack ModeSelectionVM = new ModeSelectionVM(this); WebBrowserVM = new WebBrowserVM(); UserInterventionHandlers = new UserInterventionHandlers(this); + LoginManagerVM = new LoginManagerVM(this); // Set up logging Utils.LogMessages @@ -101,7 +111,6 @@ namespace Wabbajack Clipboard.SetText($"Wabbajack {VersionDisplay}\n{ThisAssembly.Git.Sha}"); }); } - private static bool IsStartingFromModlist(out string modlistPath) { string[] args = Environment.GetCommandLineArgs(); @@ -115,6 +124,7 @@ namespace Wabbajack return true; } + public void OpenInstaller(string path) { if (path == null) return; @@ -124,6 +134,19 @@ namespace Wabbajack installer.ModListLocation.TargetPath = path; } + public void NavigateBack() + { + var prev = NavigationTrail.Last(); + NavigationTrail.RemoveAt(NavigationTrail.Count - 1); + ActivePane = prev; + } + + public void NavigateTo(ViewModel vm) + { + NavigationTrail.Add(ActivePane); + ActivePane = vm; + } + public void ShutdownApplication() { Dispose(); diff --git a/Wabbajack/View Models/UserInterventionHandlers.cs b/Wabbajack/View Models/UserInterventionHandlers.cs index bcc5d941..e401cd75 100644 --- a/Wabbajack/View Models/UserInterventionHandlers.cs +++ b/Wabbajack/View Models/UserInterventionHandlers.cs @@ -10,6 +10,7 @@ using ReactiveUI; using Wabbajack.Common; using Wabbajack.Lib.Downloaders; using Wabbajack.Lib.NexusApi; +using Wabbajack.UserInterventions; namespace Wabbajack { @@ -72,9 +73,14 @@ namespace Wabbajack break; case ConfirmationIntervention c: break; + case ShowLoginManager c: + MainWindow.NavigateTo(MainWindow.LoginManagerVM); + break; default: throw new NotImplementedException($"No handler for {msg}"); } } + + } } diff --git a/Wabbajack/Views/LinksView.xaml b/Wabbajack/Views/LinksView.xaml index 9945fc2a..0a4c867c 100644 --- a/Wabbajack/Views/LinksView.xaml +++ b/Wabbajack/Views/LinksView.xaml @@ -12,11 +12,23 @@ + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Wabbajack/Views/LoginManagerView.xaml.cs b/Wabbajack/Views/LoginManagerView.xaml.cs new file mode 100644 index 00000000..81f601ce --- /dev/null +++ b/Wabbajack/Views/LoginManagerView.xaml.cs @@ -0,0 +1,28 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +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.Navigation; +using System.Windows.Shapes; + +namespace Wabbajack +{ + /// + /// Interaction logic for LoginManager.xaml + /// + public partial class LoginManagerView : UserControl + { + public LoginManagerView() + { + InitializeComponent(); + } + } +} diff --git a/Wabbajack/Views/MainWindow.xaml b/Wabbajack/Views/MainWindow.xaml index 29531bcb..a2cedc17 100644 --- a/Wabbajack/Views/MainWindow.xaml +++ b/Wabbajack/Views/MainWindow.xaml @@ -27,6 +27,9 @@ + + + diff --git a/Wabbajack/Wabbajack.csproj b/Wabbajack/Wabbajack.csproj index 440111ee..c8cba25e 100644 --- a/Wabbajack/Wabbajack.csproj +++ b/Wabbajack/Wabbajack.csproj @@ -176,7 +176,9 @@ UnderMaintenanceOverlay.xaml + + CompilationCompleteView.xaml @@ -212,6 +214,9 @@ LinksView.xaml + + LoginManagerView.xaml + ModeSelectionView.xaml @@ -313,6 +318,10 @@ Designer MSBuild:Compile + + Designer + MSBuild:Compile + Designer MSBuild:Compile