2021-12-26 21:56:44 +00:00
|
|
|
|
using System;
|
2022-01-01 16:18:08 +00:00
|
|
|
|
using System.Collections.ObjectModel;
|
2021-12-26 21:56:44 +00:00
|
|
|
|
using ReactiveUI;
|
|
|
|
|
using System.Reactive.Disposables;
|
|
|
|
|
using System.Reactive.Linq;
|
|
|
|
|
using System.Windows.Media.Imaging;
|
|
|
|
|
using ReactiveUI.Fody.Helpers;
|
|
|
|
|
using System.Windows.Media;
|
|
|
|
|
using DynamicData;
|
|
|
|
|
using DynamicData.Binding;
|
|
|
|
|
using System.Reactive;
|
2021-12-31 00:50:38 +00:00
|
|
|
|
using System.Text;
|
2021-12-31 22:00:03 +00:00
|
|
|
|
using System.Threading;
|
2021-12-30 23:55:41 +00:00
|
|
|
|
using System.Threading.Tasks;
|
2021-12-31 23:58:09 +00:00
|
|
|
|
using System.Windows.Shell;
|
2021-12-28 00:53:14 +00:00
|
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
2021-12-28 00:24:53 +00:00
|
|
|
|
using Microsoft.Extensions.Logging;
|
2021-12-26 21:56:44 +00:00
|
|
|
|
using Microsoft.WindowsAPICodePack.Dialogs;
|
2021-12-27 00:38:44 +00:00
|
|
|
|
using Microsoft.WindowsAPICodePack.Shell;
|
2021-12-30 23:55:41 +00:00
|
|
|
|
using Wabbajack.Common;
|
2021-12-31 22:00:03 +00:00
|
|
|
|
using Wabbajack.Downloaders.GameFile;
|
2021-12-30 23:55:41 +00:00
|
|
|
|
using Wabbajack.DTOs;
|
2021-12-28 00:53:14 +00:00
|
|
|
|
using Wabbajack.DTOs.JsonConverters;
|
2021-12-30 00:15:37 +00:00
|
|
|
|
using Wabbajack.Extensions;
|
2021-12-31 00:50:38 +00:00
|
|
|
|
using Wabbajack.Hashing.xxHash64;
|
2021-12-30 23:55:41 +00:00
|
|
|
|
using Wabbajack.Installer;
|
2021-12-30 00:15:37 +00:00
|
|
|
|
using Wabbajack.Interventions;
|
2021-12-29 17:26:12 +00:00
|
|
|
|
using Wabbajack.Messages;
|
2021-12-31 22:00:03 +00:00
|
|
|
|
using Wabbajack.Models;
|
2021-12-30 23:55:41 +00:00
|
|
|
|
using Wabbajack.Paths;
|
2021-12-27 00:38:44 +00:00
|
|
|
|
using Wabbajack.RateLimiter;
|
2021-12-27 05:13:28 +00:00
|
|
|
|
using Wabbajack.View_Models;
|
2021-12-28 06:11:39 +00:00
|
|
|
|
using Wabbajack.Paths.IO;
|
2021-12-31 00:18:20 +00:00
|
|
|
|
using Wabbajack.Services.OSIntegrated;
|
2021-12-31 22:00:03 +00:00
|
|
|
|
using Wabbajack.Util;
|
2022-01-01 16:18:08 +00:00
|
|
|
|
using Configuration = Wabbajack.Networking.WabbajackClientApi.Configuration;
|
2021-12-30 00:15:37 +00:00
|
|
|
|
using Consts = Wabbajack.Consts;
|
2021-12-28 06:11:39 +00:00
|
|
|
|
using KnownFolders = Wabbajack.Paths.IO.KnownFolders;
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
2021-12-28 00:24:53 +00:00
|
|
|
|
namespace Wabbajack;
|
|
|
|
|
|
2021-12-28 00:54:56 +00:00
|
|
|
|
public enum ModManager
|
2021-12-26 21:56:44 +00:00
|
|
|
|
{
|
2021-12-28 00:24:53 +00:00
|
|
|
|
Standard
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 22:00:03 +00:00
|
|
|
|
public enum InstallState
|
|
|
|
|
{
|
|
|
|
|
Configuration,
|
|
|
|
|
Installing,
|
|
|
|
|
Success,
|
|
|
|
|
Failure
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-01 16:18:08 +00:00
|
|
|
|
public class InstallerVM : BackNavigatingVM, IBackNavigatingVM, ICpuStatusVM
|
2021-12-28 00:24:53 +00:00
|
|
|
|
{
|
2021-12-31 00:18:20 +00:00
|
|
|
|
private const string LastLoadedModlist = "last-loaded-modlist";
|
2021-12-31 00:50:38 +00:00
|
|
|
|
private const string InstallSettingsPrefix = "install-settings-";
|
2021-12-31 00:18:20 +00:00
|
|
|
|
|
2021-12-31 23:44:36 +00:00
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public Percent StatusProgress { get; set; }
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public string StatusText { get; set; }
|
|
|
|
|
|
2021-12-28 00:24:53 +00:00
|
|
|
|
[Reactive]
|
2021-12-30 23:55:41 +00:00
|
|
|
|
public ModList ModList { get; set; }
|
|
|
|
|
|
2021-12-31 22:00:03 +00:00
|
|
|
|
[Reactive]
|
|
|
|
|
public ModlistMetadata ModlistMetadata { get; set; }
|
|
|
|
|
|
2021-12-28 00:24:53 +00:00
|
|
|
|
[Reactive]
|
|
|
|
|
public ErrorResponse? Completed { get; set; }
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
[Reactive]
|
|
|
|
|
public FilePickerVM ModListLocation { get; set; }
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public MO2InstallerVM Installer { get; set; }
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public BitmapFrame ModListImage { get; set; }
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
|
|
|
|
|
public BitmapFrame SlideShowImage { get; set; }
|
2021-12-31 22:00:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public InstallState InstallState { get; set; }
|
2021-12-30 23:55:41 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Slideshow Data
|
|
|
|
|
/// </summary>
|
|
|
|
|
[Reactive]
|
|
|
|
|
public string SlideShowTitle { get; set; }
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public string SlideShowAuthor { get; set; }
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public string SlideShowDescription { get; set; }
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
private readonly ObservableAsPropertyHelper<bool> _installing;
|
|
|
|
|
private readonly DTOSerializer _dtos;
|
2021-12-28 00:24:53 +00:00
|
|
|
|
private readonly ILogger<InstallerVM> _logger;
|
2021-12-31 00:18:20 +00:00
|
|
|
|
private readonly SettingsManager _settingsManager;
|
2021-12-31 22:00:03 +00:00
|
|
|
|
private readonly IServiceProvider _serviceProvider;
|
|
|
|
|
private readonly SystemParametersConstructor _parametersConstructor;
|
|
|
|
|
private readonly IGameLocator _gameLocator;
|
|
|
|
|
private readonly LoggerProvider _loggerProvider;
|
2022-01-01 16:18:08 +00:00
|
|
|
|
private readonly ResourceMonitor _resourceMonitor;
|
|
|
|
|
private readonly Services.OSIntegrated.Configuration _configuration;
|
|
|
|
|
public ReadOnlyObservableCollection<CPUDisplayVM> StatusList => _resourceMonitor.Tasks;
|
2021-12-31 00:18:20 +00:00
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
[Reactive]
|
|
|
|
|
public bool Installing { get; set; }
|
|
|
|
|
|
2021-12-31 22:00:03 +00:00
|
|
|
|
[Reactive]
|
|
|
|
|
public LoggerProvider LoggerProvider { get; set; }
|
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
|
2021-12-28 00:24:53 +00:00
|
|
|
|
// Command properties
|
|
|
|
|
public ReactiveCommand<Unit, Unit> ShowManifestCommand { get; }
|
|
|
|
|
public ReactiveCommand<Unit, Unit> OpenReadmeCommand { get; }
|
|
|
|
|
public ReactiveCommand<Unit, Unit> VisitModListWebsiteCommand { get; }
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
2021-12-28 00:24:53 +00:00
|
|
|
|
public ReactiveCommand<Unit, Unit> CloseWhenCompleteCommand { get; }
|
|
|
|
|
public ReactiveCommand<Unit, Unit> OpenLogsCommand { get; }
|
|
|
|
|
public ReactiveCommand<Unit, Unit> GoToInstallCommand { get; }
|
|
|
|
|
public ReactiveCommand<Unit, Unit> BeginCommand { get; }
|
2021-12-30 23:55:41 +00:00
|
|
|
|
|
|
|
|
|
public ReactiveCommand<Unit, Unit> BackCommand { get; }
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
2021-12-31 22:00:03 +00:00
|
|
|
|
public InstallerVM(ILogger<InstallerVM> logger, DTOSerializer dtos, SettingsManager settingsManager, IServiceProvider serviceProvider,
|
2022-01-01 16:18:08 +00:00
|
|
|
|
SystemParametersConstructor parametersConstructor, IGameLocator gameLocator, LoggerProvider loggerProvider, ResourceMonitor resourceMonitor,
|
|
|
|
|
Wabbajack.Services.OSIntegrated.Configuration configuration) : base(logger)
|
2021-12-28 00:24:53 +00:00
|
|
|
|
{
|
|
|
|
|
_logger = logger;
|
2022-01-01 16:18:08 +00:00
|
|
|
|
_configuration = configuration;
|
2021-12-31 22:00:03 +00:00
|
|
|
|
LoggerProvider = loggerProvider;
|
2021-12-31 00:18:20 +00:00
|
|
|
|
_settingsManager = settingsManager;
|
2021-12-30 23:55:41 +00:00
|
|
|
|
_dtos = dtos;
|
2021-12-31 22:00:03 +00:00
|
|
|
|
_serviceProvider = serviceProvider;
|
|
|
|
|
_parametersConstructor = parametersConstructor;
|
|
|
|
|
_gameLocator = gameLocator;
|
2022-01-01 16:18:08 +00:00
|
|
|
|
_resourceMonitor = resourceMonitor;
|
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
Installer = new MO2InstallerVM(this);
|
|
|
|
|
|
|
|
|
|
BackCommand = ReactiveCommand.Create(() => NavigateToGlobal.Send(NavigateToGlobal.ScreenType.ModeSelectionView));
|
2021-12-31 00:50:38 +00:00
|
|
|
|
|
|
|
|
|
BeginCommand = ReactiveCommand.Create(() => BeginInstall().FireAndForget());
|
2021-12-30 23:55:41 +00:00
|
|
|
|
|
|
|
|
|
OpenReadmeCommand = ReactiveCommand.Create(() =>
|
2021-12-28 00:24:53 +00:00
|
|
|
|
{
|
2021-12-30 23:55:41 +00:00
|
|
|
|
UIUtils.OpenWebsite(new Uri(ModList!.Readme));
|
|
|
|
|
}, LoadingLock.IsNotLoadingObservable);
|
|
|
|
|
|
|
|
|
|
VisitModListWebsiteCommand = ReactiveCommand.Create(() =>
|
2021-12-28 00:24:53 +00:00
|
|
|
|
{
|
2021-12-30 23:55:41 +00:00
|
|
|
|
UIUtils.OpenWebsite(ModList!.Website);
|
|
|
|
|
}, LoadingLock.IsNotLoadingObservable);
|
|
|
|
|
|
2021-12-28 00:24:53 +00:00
|
|
|
|
ModListLocation = new FilePickerVM
|
|
|
|
|
{
|
|
|
|
|
ExistCheckOption = FilePickerVM.CheckOptions.On,
|
|
|
|
|
PathType = FilePickerVM.PathTypeOptions.File,
|
|
|
|
|
PromptTitle = "Select a ModList to install"
|
|
|
|
|
};
|
|
|
|
|
ModListLocation.Filters.Add(new CommonFileDialogFilter("Wabbajack Modlist", "*.wabbajack"));
|
2021-12-28 00:53:14 +00:00
|
|
|
|
|
2022-01-01 16:18:08 +00:00
|
|
|
|
OpenLogsCommand = ReactiveCommand.Create(() =>
|
|
|
|
|
{
|
|
|
|
|
UIUtils.OpenFolder(_configuration.LogLocation);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
GoToInstallCommand = ReactiveCommand.Create(() =>
|
|
|
|
|
{
|
|
|
|
|
UIUtils.OpenFolder(Installer.Location.TargetPath);
|
|
|
|
|
});
|
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
MessageBus.Current.Listen<LoadModlistForInstalling>()
|
2021-12-31 22:00:03 +00:00
|
|
|
|
.Subscribe(msg => LoadModlist(msg.Path, msg.Metadata).FireAndForget())
|
2021-12-30 23:55:41 +00:00
|
|
|
|
.DisposeWith(CompositeDisposable);
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
2021-12-31 00:18:20 +00:00
|
|
|
|
MessageBus.Current.Listen<LoadLastLoadedModlist>()
|
|
|
|
|
.Subscribe(msg =>
|
|
|
|
|
{
|
|
|
|
|
LoadLastModlist().FireAndForget();
|
|
|
|
|
});
|
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
this.WhenActivated(disposables =>
|
|
|
|
|
{
|
2021-12-31 22:00:03 +00:00
|
|
|
|
|
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
ModListLocation.WhenAnyValue(l => l.TargetPath)
|
2021-12-31 22:00:03 +00:00
|
|
|
|
.Subscribe(p => LoadModlist(p, null).FireAndForget())
|
2021-12-30 23:55:41 +00:00
|
|
|
|
.DisposeWith(disposables);
|
2021-12-28 00:24:53 +00:00
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
});
|
2021-12-28 00:24:53 +00:00
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
}
|
2021-12-28 00:24:53 +00:00
|
|
|
|
|
2021-12-31 00:18:20 +00:00
|
|
|
|
private async Task LoadLastModlist()
|
|
|
|
|
{
|
|
|
|
|
var lst = await _settingsManager.Load<AbsolutePath>(LastLoadedModlist);
|
|
|
|
|
if (lst.FileExists())
|
2021-12-31 22:00:03 +00:00
|
|
|
|
await LoadModlist(lst, null);
|
2021-12-31 00:18:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 22:00:03 +00:00
|
|
|
|
private async Task LoadModlist(AbsolutePath path, ModlistMetadata? metadata)
|
2021-12-30 23:55:41 +00:00
|
|
|
|
{
|
|
|
|
|
using var ll = LoadingLock.WithLoading();
|
2021-12-31 22:00:03 +00:00
|
|
|
|
InstallState = InstallState.Configuration;
|
2021-12-30 23:55:41 +00:00
|
|
|
|
ModListLocation.TargetPath = path;
|
|
|
|
|
try
|
2021-12-28 00:24:53 +00:00
|
|
|
|
{
|
2021-12-30 23:55:41 +00:00
|
|
|
|
ModList = await StandardInstaller.LoadFromFile(_dtos, path);
|
|
|
|
|
ModListImage = BitmapFrame.Create(await StandardInstaller.ModListImageStream(path));
|
2021-12-31 00:50:38 +00:00
|
|
|
|
|
2021-12-31 23:44:36 +00:00
|
|
|
|
StatusText = $"Install configuration for {ModList.Name}";
|
2021-12-31 23:58:09 +00:00
|
|
|
|
TaskBarUpdate.Send($"Loaded {ModList.Name}", TaskbarItemProgressState.Normal);
|
2021-12-31 23:44:36 +00:00
|
|
|
|
|
2021-12-31 00:50:38 +00:00
|
|
|
|
var hex = (await ModListLocation.TargetPath.ToString().Hash()).ToHex();
|
|
|
|
|
var prevSettings = await _settingsManager.Load<SavedInstallSettings>(InstallSettingsPrefix + hex);
|
|
|
|
|
|
|
|
|
|
if (prevSettings.ModListLocation == path)
|
|
|
|
|
{
|
|
|
|
|
ModListLocation.TargetPath = prevSettings.ModListLocation;
|
|
|
|
|
Installer.Location.TargetPath = prevSettings.InstallLocation;
|
|
|
|
|
Installer.DownloadLocation.TargetPath = prevSettings.DownloadLoadction;
|
2021-12-31 22:00:03 +00:00
|
|
|
|
ModlistMetadata = metadata ?? prevSettings.Metadata;
|
2021-12-31 00:50:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
PopulateSlideShow(ModList);
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
ll.Succeed();
|
2021-12-31 00:18:20 +00:00
|
|
|
|
await _settingsManager.Save(LastLoadedModlist, path);
|
2021-12-30 23:55:41 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError(ex, "While loading modlist");
|
|
|
|
|
ll.Fail();
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
2021-12-31 22:00:03 +00:00
|
|
|
|
private async Task BeginInstall()
|
2021-12-31 00:50:38 +00:00
|
|
|
|
{
|
2021-12-31 22:00:03 +00:00
|
|
|
|
InstallState = InstallState.Installing;
|
2021-12-31 00:50:38 +00:00
|
|
|
|
var postfix = (await ModListLocation.TargetPath.ToString().Hash()).ToHex();
|
|
|
|
|
await _settingsManager.Save(InstallSettingsPrefix + postfix, new SavedInstallSettings
|
|
|
|
|
{
|
|
|
|
|
ModListLocation = ModListLocation.TargetPath,
|
|
|
|
|
InstallLocation = Installer.Location.TargetPath,
|
2021-12-31 22:00:03 +00:00
|
|
|
|
DownloadLoadction = Installer.DownloadLocation.TargetPath,
|
|
|
|
|
Metadata = ModlistMetadata
|
2021-12-31 00:50:38 +00:00
|
|
|
|
});
|
|
|
|
|
|
2021-12-31 22:00:03 +00:00
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var installer = StandardInstaller.Create(_serviceProvider, new InstallerConfiguration
|
|
|
|
|
{
|
|
|
|
|
Game = ModList.GameType,
|
|
|
|
|
Downloads = Installer.DownloadLocation.TargetPath,
|
|
|
|
|
Install = Installer.Location.TargetPath,
|
|
|
|
|
ModList = ModList,
|
|
|
|
|
ModlistArchive = ModListLocation.TargetPath,
|
|
|
|
|
SystemParameters = _parametersConstructor.Create(),
|
|
|
|
|
GameFolder = _gameLocator.GameLocation(ModList.GameType)
|
|
|
|
|
});
|
|
|
|
|
|
2021-12-31 23:44:36 +00:00
|
|
|
|
|
|
|
|
|
installer.OnStatusUpdate = update =>
|
|
|
|
|
{
|
|
|
|
|
StatusText = update.StatusText;
|
|
|
|
|
StatusProgress = update.StepsProgress;
|
|
|
|
|
|
2021-12-31 23:58:09 +00:00
|
|
|
|
TaskBarUpdate.Send(update.StatusText, TaskbarItemProgressState.Indeterminate, update.StepsProgress.Value);
|
2021-12-31 23:44:36 +00:00
|
|
|
|
};
|
2021-12-31 22:00:03 +00:00
|
|
|
|
await installer.Begin(CancellationToken.None);
|
2021-12-31 23:58:09 +00:00
|
|
|
|
|
|
|
|
|
TaskBarUpdate.Send($"Finished install of {ModList.Name}", TaskbarItemProgressState.Normal);
|
2021-12-31 22:00:03 +00:00
|
|
|
|
|
|
|
|
|
InstallState = InstallState.Success;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
2021-12-31 23:58:09 +00:00
|
|
|
|
TaskBarUpdate.Send($"Error during install of {ModList.Name}", TaskbarItemProgressState.Error);
|
2021-12-31 22:00:03 +00:00
|
|
|
|
InstallState = InstallState.Failure;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 00:50:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-31 23:44:36 +00:00
|
|
|
|
|
2021-12-31 00:50:38 +00:00
|
|
|
|
class SavedInstallSettings
|
|
|
|
|
{
|
|
|
|
|
public AbsolutePath ModListLocation { get; set; }
|
|
|
|
|
public AbsolutePath InstallLocation { get; set; }
|
|
|
|
|
public AbsolutePath DownloadLoadction { get; set; }
|
2021-12-31 22:00:03 +00:00
|
|
|
|
|
|
|
|
|
public ModlistMetadata Metadata { get; set; }
|
2021-12-31 00:50:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-30 23:55:41 +00:00
|
|
|
|
private void PopulateSlideShow(ModList modList)
|
|
|
|
|
{
|
|
|
|
|
SlideShowTitle = modList.Name;
|
|
|
|
|
SlideShowAuthor = modList.Author;
|
|
|
|
|
SlideShowDescription = modList.Description;
|
|
|
|
|
SlideShowImage = ModListImage;
|
2021-12-26 21:56:44 +00:00
|
|
|
|
}
|
2021-12-30 23:55:41 +00:00
|
|
|
|
|
2021-12-28 00:24:53 +00:00
|
|
|
|
}
|