diff --git a/Wabbajack/View Models/InstallerVM.cs b/Wabbajack/View Models/InstallerVM.cs index 6e668a41..956e8886 100644 --- a/Wabbajack/View Models/InstallerVM.cs +++ b/Wabbajack/View Models/InstallerVM.cs @@ -26,21 +26,23 @@ using DynamicData.Binding; using System.Reactive; using System.Text; using Wabbajack.Lib; +using Splat; namespace Wabbajack { - public class InstallerVM : ViewModel, IDataErrorInfo + public class InstallerVM : ViewModel { public SlideShow Slideshow { get; } + public MainWindowVM MWVM { get; } + public BitmapImage WabbajackLogo { get; } = UIUtils.BitmapImageFromResource("Wabbajack.Resources.Banner_Dark.png"); + private readonly ObservableAsPropertyHelper _ModList; public ModList ModList => _ModList.Value; private string _ModListPath; - public string ModListPath { get => _ModListPath; private set => this.RaiseAndSetIfChanged(ref _ModListPath, value); } - - public RunMode Mode => RunMode.Install; + public string ModListPath { get => _ModListPath; set => this.RaiseAndSetIfChanged(ref _ModListPath, value); } private readonly ObservableAsPropertyHelper _ModListName; public string ModListName => _ModListName.Value; @@ -60,9 +62,10 @@ namespace Wabbajack private string _DownloadLocation; public string DownloadLocation { get => _DownloadLocation; set => this.RaiseAndSetIfChanged(ref _DownloadLocation, value); } + private readonly ObservableAsPropertyHelper _Image; + public BitmapImage Image => _Image.Value; + // Command properties - public IReactiveCommand ChangePathCommand { get; } - public IReactiveCommand ChangeDownloadPathCommand { get; } public IReactiveCommand BeginCommand { get; } public IReactiveCommand ShowReportCommand { get; } public IReactiveCommand OpenReadmeCommand { get; } @@ -83,6 +86,7 @@ namespace Wabbajack this.MWVM = mainWindowVM; this._ModList = this.WhenAny(x => x.ModListPath) + .ObserveOn(RxApp.TaskpoolScheduler) .Select(source => { if (source == null) return default; @@ -106,6 +110,7 @@ namespace Wabbajack return modlist; }) .ObserveOnGuiThread() + .StartWith(default(ModList)) .ToProperty(this, nameof(this.ModList)); this._HTMLReport = this.WhenAny(x => x.ModList) .Select(modList => modList?.ReportHTML) @@ -114,9 +119,69 @@ namespace Wabbajack .Select(modList => modList?.Name) .ToProperty(this, nameof(this.ModListName)); + this.Slideshow = new SlideShow(this); + + // Locate and create modlist image if it exists + var modListImage = Observable.CombineLatest( + this.WhenAny(x => x.ModList), + this.WhenAny(x => x.ModListPath), + (modList, modListPath) => (modList, modListPath)) + .ObserveOn(RxApp.TaskpoolScheduler) + .Select(u => + { + if (u.modList == null + || u.modListPath == null + || !File.Exists(u.modListPath) + || string.IsNullOrEmpty(u.modList.Image) + || u.modList.Image.Length != 36) + { + return WabbajackLogo; + } + try + { + using (var fs = new FileStream(u.modListPath, FileMode.Open, FileAccess.Read, FileShare.Read)) + using (var ar = new ZipArchive(fs, ZipArchiveMode.Read)) + using (var ms = new MemoryStream()) + { + var entry = ar.GetEntry(u.modList.Image); + using (var e = entry.Open()) + e.CopyTo(ms); + var image = new BitmapImage(); + image.BeginInit(); + image.CacheOption = BitmapCacheOption.OnLoad; + image.StreamSource = ms; + image.EndInit(); + image.Freeze(); + + return image; + } + } + catch (Exception ex) + { + this.Log().Warn(ex, "Error loading modlist splash image."); + return WabbajackLogo; + } + }) + .ObserveOnGuiThread() + .StartWith(default(BitmapImage)) + .Replay(1) + .RefCount(); + + // Set displayed image to modlist image if configuring, or to the current slideshow image if installing + this._Image = Observable.CombineLatest( + modListImage + .StartWith(default(BitmapImage)), + this.WhenAny(x => x.Slideshow.Image) + .StartWith(default(BitmapImage)), + this.WhenAny(x => x.Installing) + .StartWith(false), + resultSelector: (modList, slideshow, installing) => + { + return installing ? slideshow : modList; + }) + .ToProperty(this, nameof(this.Image)); + // Define commands - this.ChangePathCommand = ReactiveCommand.Create(ExecuteChangePath); - this.ChangeDownloadPathCommand = ReactiveCommand.Create(ExecuteChangeDownloadPath); this.ShowReportCommand = ReactiveCommand.Create(ShowReport); this.OpenReadmeCommand = ReactiveCommand.Create( execute: this.OpenReadmeWindow, @@ -125,27 +190,20 @@ namespace Wabbajack .ObserveOnGuiThread()); this.BeginCommand = ReactiveCommand.Create( execute: this.ExecuteBegin, - canExecute: this.WhenAny(x => x.UIReady) + canExecute: this.WhenAny(x => x.Installing) + .Select(installing => !installing) .ObserveOnGuiThread()); - this.Slideshow = new SlideShow(this); - } - - private void ExecuteChangePath() - { - var folder = UIUtils.ShowFolderSelectionDialog("Select Installation directory"); - if (folder == null) return; - Location = folder; - if (DownloadLocation == null) - { - DownloadLocation = Path.Combine(Location, "downloads"); - } - } - - private void ExecuteChangeDownloadPath() - { - var folder = UIUtils.ShowFolderSelectionDialog("Select a location for MO2 downloads"); - if (folder != null) DownloadLocation = folder; + // Have Installation location updates modify the downloads location if empty + this.WhenAny(x => x.Location) + .Subscribe(installPath => + { + if (string.IsNullOrWhiteSpace(this.DownloadLocation)) + { + this.DownloadLocation = Path.Combine(installPath, "downloads"); + } + }) + .DisposeWith(this.CompositeDisposable); } private void ShowReport() @@ -163,6 +221,11 @@ namespace Wabbajack using (var ms = new MemoryStream()) { var entry = ar.GetEntry(this.ModList.Readme); + if (entry == null) + { + Utils.Log($"Tried to open a non-existant readme: {this.ModList.Readme}"); + return; + } using (var e = entry.Open()) { e.CopyTo(ms); @@ -176,78 +239,31 @@ namespace Wabbajack } } - public string Error => "Error"; - - public string this[string columnName] => Validate(columnName); - - private string Validate(string columnName) - { - string validationMessage = null; - switch (columnName) - { - case "Location": - if (Location == null) - { - validationMessage = null; - } - else switch (Mode) - { - case RunMode.Install when Location != null && Directory.Exists(Location) && !Directory.EnumerateFileSystemEntries(Location).Any(): - validationMessage = null; - break; - case RunMode.Install when Location != null && Directory.Exists(Location) && Directory.EnumerateFileSystemEntries(Location).Any(): - validationMessage = "You have selected a non-empty directory. Installing the modlist here might result in a broken install!"; - break; - default: - validationMessage = "Invalid Mod Organizer profile directory"; - break; - } - break; - } - return validationMessage; - } - private void ExecuteBegin() { - UIReady = false; - if (this.Mode == RunMode.Install) + this.Installing = true; + var installer = new Installer(this.ModListPath, this.ModList, Location) { - this.Installing = true; - var installer = new Installer(this.ModListPath, this.ModList, Location) + DownloadFolder = DownloadLocation + }; + var th = new Thread(() => + { + try { - DownloadFolder = DownloadLocation - }; - var th = new Thread(() => + installer.Install(); + } + catch (Exception ex) { - UIReady = false; - try - { - installer.Install(); - } - catch (Exception ex) - { - while (ex.InnerException != null) ex = ex.InnerException; - Utils.Log(ex.StackTrace); - Utils.Log(ex.ToString()); - Utils.Log($"{ex.Message} - Can't continue"); - } - finally - { - UIReady = true; - this.Installing = false; - } - }) - { - Priority = ThreadPriority.BelowNormal - }; - th.Start(); - } - } - - public void Init(string source) - { - this.ModListPath = source; - this.UIReady = true; + while (ex.InnerException != null) ex = ex.InnerException; + Utils.Log(ex.StackTrace); + Utils.Log(ex.ToString()); + Utils.Log($"{ex.Message} - Can't continue"); + } + }) + { + Priority = ThreadPriority.BelowNormal + }; + th.Start(); } } } \ No newline at end of file diff --git a/Wabbajack/View Models/MainWindowVM.cs b/Wabbajack/View Models/MainWindowVM.cs index 1fb26139..866d71b2 100644 --- a/Wabbajack/View Models/MainWindowVM.cs +++ b/Wabbajack/View Models/MainWindowVM.cs @@ -60,23 +60,6 @@ namespace Wabbajack Utils.SetLoggerFn(s => _logSubj.OnNext(s)); Utils.SetStatusFn((msg, progress) => WorkQueue.Report(msg, progress)); - // Initialize work queue - WorkQueue.Init( - report_function: (id, msg, progress) => this._statusSubject.OnNext(new CPUStatus() { ID = id, Msg = msg, Progress = progress }), - report_queue_size: (max, current) => this.SetQueueSize(max, current)); - - // Compile progress updates and populate ObservableCollection - this._statusSubject - .ObserveOn(RxApp.TaskpoolScheduler) - .ToObservableChangeSet(x => x.ID) - .Batch(TimeSpan.FromMilliseconds(250)) - .EnsureUniqueChanges() - .ObserveOn(RxApp.MainThreadScheduler) - .Sort(SortExpressionComparer.Ascending(s => s.ID), SortOptimisations.ComparesImmutableValuesOnly) - .Bind(this.StatusList) - .Subscribe() - .DisposeWith(this.CompositeDisposable); - // Wire mode to drive the active pane this._ActivePane = this.WhenAny(x => x.Mode) .ObserveOn(RxApp.MainThreadScheduler) @@ -96,9 +79,25 @@ namespace Wabbajack this.WhenAny(x => x.ActivePane) .ObserveOn(RxApp.TaskpoolScheduler) .WhereCastable() - .Subscribe(vm => vm.Init(source)) + .Subscribe(vm => vm.ModListPath = source) .DisposeWith(this.CompositeDisposable); + // Initialize work queue + WorkQueue.Init( + report_function: (id, msg, progress) => this._statusSubject.OnNext(new CPUStatus() { ID = id, Msg = msg, Progress = progress }), + report_queue_size: (max, current) => this.SetQueueSize(max, current)); + + // Compile progress updates and populate ObservableCollection + this._statusSubject + .ObserveOn(RxApp.TaskpoolScheduler) + .ToObservableChangeSet(x => x.ID) + .Batch(TimeSpan.FromMilliseconds(250)) + .EnsureUniqueChanges() + .ObserveOn(RxApp.MainThreadScheduler) + .Sort(SortExpressionComparer.Ascending(s => s.ID), SortOptimisations.ComparesImmutableValuesOnly) + .Bind(this.StatusList) + .Subscribe() + .DisposeWith(this.CompositeDisposable); } private void SetQueueSize(int max, int current) diff --git a/Wabbajack/View Models/SlideShow.cs b/Wabbajack/View Models/SlideShow.cs index c3b6759f..c54c1abb 100644 --- a/Wabbajack/View Models/SlideShow.cs +++ b/Wabbajack/View Models/SlideShow.cs @@ -35,7 +35,6 @@ namespace Wabbajack public InstallerVM Installer { get; } public BitmapImage NextIcon { get; } = UIUtils.BitmapImageFromResource("Wabbajack.Resources.Icons.next.png"); - public BitmapImage WabbajackLogo { get; } = UIUtils.BitmapImageFromResource("Wabbajack.Resources.Banner_Dark.png"); private bool _ShowNSFW; public bool ShowNSFW { get => _ShowNSFW; set => this.RaiseAndSetIfChanged(ref _ShowNSFW, value); } @@ -49,9 +48,6 @@ namespace Wabbajack private BitmapImage _Image; public BitmapImage Image { get => _Image; set => this.RaiseAndSetIfChanged(ref _Image, value); } - private readonly ObservableAsPropertyHelper _ModlistImage; - public BitmapImage ModlistImage => _ModlistImage.Value; - private string _ModName = "Wabbajack"; public string ModName { get => _ModName; set => this.RaiseAndSetIfChanged(ref _ModName, value); } @@ -104,49 +100,6 @@ namespace Wabbajack }) .DisposeWith(this.CompositeDisposable); - this._ModlistImage = Observable.CombineLatest( - this.WhenAny(x => x.Installer.ModList), - this.WhenAny(x => x.Installer.ModListPath), - (modList, modListPath) => (modList, modListPath)) - .ObserveOn(RxApp.TaskpoolScheduler) - .Select(u => - { - if (u.modList == null - || u.modListPath == null - || !File.Exists(u.modListPath) - || string.IsNullOrEmpty(u.modList.Image) - || u.modList.Image.Length != 36) - { - return default(BitmapImage); - } - try - { - using (var fs = new FileStream(u.modListPath, FileMode.Open, FileAccess.Read, FileShare.Read)) - using (var ar = new ZipArchive(fs, ZipArchiveMode.Read)) - using (var ms = new MemoryStream()) - { - var entry = ar.GetEntry(u.modList.Image); - using (var e = entry.Open()) - e.CopyTo(ms); - var image = new BitmapImage(); - image.BeginInit(); - image.CacheOption = BitmapCacheOption.OnLoad; - image.StreamSource = ms; - image.EndInit(); - image.Freeze(); - - return image; - } - } - catch (Exception ex) - { - this.Log().Warn(ex, "Error loading modlist splash image."); - return default(BitmapImage); - } - }) - .ObserveOnGuiThread() - .ToProperty(this, nameof(this.ModlistImage)); - /// Wire slideshow updates // Merge all the sources that trigger a slideshow update Observable.Merge( diff --git a/Wabbajack/Views/InstallationView.xaml b/Wabbajack/Views/InstallationView.xaml index 089ffa51..9103c20d 100644 --- a/Wabbajack/Views/InstallationView.xaml +++ b/Wabbajack/Views/InstallationView.xaml @@ -3,13 +3,12 @@ xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:d="http://schemas.microsoft.com/expression/blend/2008" - xmlns:icon="http://metro.mahapps.com/winfx/xaml/iconpacks" xmlns:local="clr-namespace:Wabbajack" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" xmlns:mahapps="clr-namespace:MahApps.Metro.Controls;assembly=MahApps.Metro" + xmlns:icon="http://metro.mahapps.com/winfx/xaml/iconpacks" d:DesignHeight="500" d:DesignWidth="800" - Background="#443700B3" mc:Ignorable="d"> #92000000 @@ -69,7 +68,7 @@ - + + Source="{Binding Installer.ModlistImage}" /> @@ -140,8 +139,12 @@ @@ -249,7 +252,7 @@ - + + TextAlignment="Right" + FontFamily="Lucida Sans" > + + + + - - + + + + + + + + + - + diff --git a/Wabbajack/Views/MainWindow.xaml b/Wabbajack/Views/MainWindow.xaml index d9a70a45..25d21ff3 100644 --- a/Wabbajack/Views/MainWindow.xaml +++ b/Wabbajack/Views/MainWindow.xaml @@ -8,8 +8,8 @@ Title="Wabbajack" Width="1280" Height="960" - MinWidth="1024" - MinHeight="768" + MinWidth="850" + MinHeight="650" Closing="Window_Closing" Icon="../Resources/Icons/wabbajack.ico" ResizeMode="CanResize" diff --git a/Wabbajack/Views/SlideshowView.xaml b/Wabbajack/Views/SlideshowView.xaml deleted file mode 100644 index f8707337..00000000 --- a/Wabbajack/Views/SlideshowView.xaml +++ /dev/null @@ -1,122 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Enable the Slideshow - - - Show NSFW Mods in the Slideshow - - - - - - - diff --git a/Wabbajack/Views/SlideshowView.xaml.cs b/Wabbajack/Views/SlideshowView.xaml.cs deleted file mode 100644 index b183908a..00000000 --- a/Wabbajack/Views/SlideshowView.xaml.cs +++ /dev/null @@ -1,28 +0,0 @@ -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 SlideshowView.xaml - /// - public partial class SlideshowView : UserControl - { - public SlideshowView() - { - InitializeComponent(); - } - } -} diff --git a/Wabbajack/Wabbajack.csproj b/Wabbajack/Wabbajack.csproj index 23544f7e..cc1eb73d 100644 --- a/Wabbajack/Wabbajack.csproj +++ b/Wabbajack/Wabbajack.csproj @@ -136,10 +136,6 @@ - - MSBuild:Compile - Designer - FilePicker.xaml @@ -149,9 +145,6 @@ - - SlideshowView.xaml - ModeSelectionWindow.xaml @@ -166,11 +159,11 @@ Designer MSBuild:Compile - - Designer + MSBuild:Compile + Designer - + Designer MSBuild:Compile