Rework how compiler settings are saved & loaded

This commit is contained in:
trawzified 2024-05-04 23:57:54 +02:00
parent 1e61bcaf45
commit 836f102ec6
6 changed files with 288 additions and 225 deletions

View File

@ -0,0 +1,143 @@
using Microsoft.Web.WebView2.Core;
using ReactiveUI.Fody.Helpers;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text.Json.Serialization;
using Wabbajack.Common;
using Wabbajack.Compiler;
using Wabbajack.DTOs;
using Wabbajack.Paths;
namespace Wabbajack;
public class CompilerSettingsVM : ViewModel
{
public CompilerSettingsVM() { }
public CompilerSettingsVM(CompilerSettings cs)
{
ModlistIsNSFW = cs.ModlistIsNSFW;
Source = cs.Source;
Downloads = cs.Downloads;
Game = cs.Game;
OutputFile = cs.OutputFile;
ModListImage = cs.ModListImage;
UseGamePaths = cs.UseGamePaths;
UseTextureRecompression = cs.UseTextureRecompression;
OtherGames = cs.OtherGames;
MaxVerificationTime = cs.MaxVerificationTime;
ModListName = cs.ModListName;
ModListAuthor = cs.ModListAuthor;
ModListDescription = cs.ModListDescription;
ModListReadme = cs.ModListReadme;
ModListWebsite = cs.ModListWebsite;
ModlistVersion = cs.ModlistVersion;
PublishUpdate = cs.PublishUpdate;
MachineUrl = cs.MachineUrl;
Profile = cs.Profile;
AdditionalProfiles = cs.AdditionalProfiles;
NoMatchInclude = cs.NoMatchInclude;
Include = cs.Include;
Ignore = cs.Ignore;
AlwaysEnabled = cs.AlwaysEnabled;
Version = cs.Version;
Description = cs.Description;
}
[Reactive] public bool ModlistIsNSFW { get; set; }
[Reactive] public AbsolutePath Source { get; set; }
[Reactive] public AbsolutePath Downloads { get; set; }
[Reactive] public Game Game { get; set; }
[Reactive] public AbsolutePath OutputFile { get; set; }
[Reactive] public AbsolutePath ModListImage { get; set; }
[Reactive] public bool UseGamePaths { get; set; }
[Reactive] public bool UseTextureRecompression { get; set; } = false;
[Reactive] public Game[] OtherGames { get; set; } = Array.Empty<Game>();
[Reactive] public TimeSpan MaxVerificationTime { get; set; } = TimeSpan.FromMinutes(1);
[Reactive] public string ModListName { get; set; } = "";
[Reactive] public string ModListAuthor { get; set; } = "";
[Reactive] public string ModListDescription { get; set; } = "";
[Reactive] public string ModListReadme { get; set; } = "";
[Reactive] public Uri? ModListWebsite { get; set; }
[Reactive] public Version ModlistVersion { get; set; } = Version.Parse("0.0.1.0");
[Reactive] public bool PublishUpdate { get; set; } = false;
[Reactive] public string MachineUrl { get; set; } = "";
/// <summary>
/// The main (default) profile
/// </summary>
[Reactive] public string Profile { get; set; } = "";
/// <summary>
/// Secondary profiles to include in the modlist
/// </summary>
[Reactive] public string[] AdditionalProfiles { get; set; } = Array.Empty<string>();
/// <summary>
/// All profiles to be added to the compiled modlist
/// </summary>
public IEnumerable<string> AllProfiles => AdditionalProfiles.Append(Profile);
public bool IsMO2Modlist => AllProfiles.Any(p => !string.IsNullOrWhiteSpace(p));
/// <summary>
/// This file, or files in these folders, are automatically included if they don't match
/// any other step
/// </summary>
[Reactive] public RelativePath[] NoMatchInclude { get; set; } = Array.Empty<RelativePath>();
/// <summary>
/// These files are inlined into the modlist
/// </summary>
[Reactive] public RelativePath[] Include { get; set; } = Array.Empty<RelativePath>();
/// <summary>
/// These files are ignored when compiling the modlist
/// </summary>
[Reactive] public RelativePath[] Ignore { get; set; } = Array.Empty<RelativePath>();
[Reactive] public RelativePath[] AlwaysEnabled { get; set; } = Array.Empty<RelativePath>();
[Reactive] public Version Version { get; set; }
[Reactive] public string Description { get; set; }
public CompilerSettings ToCompilerSettings()
{
return new CompilerSettings()
{
ModlistIsNSFW = ModlistIsNSFW,
Source = Source,
Downloads = Downloads,
Game = Game,
OutputFile = OutputFile,
ModListImage = ModListImage,
UseGamePaths = UseGamePaths,
UseTextureRecompression = UseTextureRecompression,
OtherGames = OtherGames,
MaxVerificationTime = MaxVerificationTime,
ModListName = ModListName,
ModListAuthor = ModListAuthor,
ModListDescription = ModListDescription,
ModListReadme = ModListReadme,
ModListWebsite = ModListWebsite,
ModlistVersion = ModlistVersion,
PublishUpdate = PublishUpdate,
MachineUrl = MachineUrl,
Profile = Profile,
AdditionalProfiles = AdditionalProfiles,
NoMatchInclude = NoMatchInclude,
Include = Include,
Ignore = Ignore,
AlwaysEnabled = AlwaysEnabled,
Version = Version,
Description = Description
};
}
public AbsolutePath CompilerSettingsPath => Source.Combine(ModListName).WithExtension(Ext.CompilerSettings);
public AbsolutePath ProfilePath => Source.Combine("profiles").Combine(Profile).Combine("modlist").WithExtension(Ext.Txt);
}

View File

@ -61,36 +61,10 @@ namespace Wabbajack
public FilePickerVM ModlistLocation { get; } public FilePickerVM ModlistLocation { get; }
public FilePickerVM DownloadLocation { get; } public FilePickerVM DownloadLocation { get; }
public FilePickerVM OutputLocation { get; } public FilePickerVM OutputLocation { get; }
// Modlist Settings
[Reactive] public string ModListName { get; set; }
[Reactive] public string Version { get; set; }
[Reactive] public string Author { get; set; }
[Reactive] public string Description { get; set; }
public FilePickerVM ModListImagePath { get; } = new();
[Reactive] public ImageSource ModListImage { get; set; }
[Reactive] public string Website { get; set; }
[Reactive] public string Readme { get; set; }
[Reactive] public bool IsNSFW { get; set; }
[Reactive] public bool PublishUpdate { get; set; }
[Reactive] public string MachineUrl { get; set; }
[Reactive] public Game BaseGame { get; set; }
[Reactive] public string SelectedProfile { get; set; }
[Reactive] public AbsolutePath GamePath { get; set; }
[Reactive] public bool IsMO2Compilation { get; set; }
[Reactive] public RelativePath[] AlwaysEnabled { get; set; } = Array.Empty<RelativePath>(); [Reactive] public CompilerSettingsVM Settings { get; set; } = new();
[Reactive] public RelativePath[] NoMatchInclude { get; set; } = Array.Empty<RelativePath>();
[Reactive] public RelativePath[] Include { get; set; } = Array.Empty<RelativePath>(); public FilePickerVM ModListImageLocation { get; } = new();
[Reactive] public RelativePath[] Ignore { get; set; } = Array.Empty<RelativePath>();
[Reactive] public string[] OtherProfiles { get; set; } = Array.Empty<string>();
[Reactive] public AbsolutePath Source { get; set; }
public AbsolutePath SettingsOutputLocation => Source.Combine(ModListName).WithExtension(Ext.CompilerSettings);
public ReactiveCommand<Unit, Unit> ExecuteCommand { get; } public ReactiveCommand<Unit, Unit> ExecuteCommand { get; }
public ReactiveCommand<Unit, Unit> ReInferSettingsCommand { get; set; } public ReactiveCommand<Unit, Unit> ReInferSettingsCommand { get; set; }
@ -115,27 +89,33 @@ namespace Wabbajack
_wjClient = wjClient; _wjClient = wjClient;
MessageBus.Current.Listen<LoadModlistForCompiling>() MessageBus.Current.Listen<LoadModlistForCompiling>()
.Subscribe(msg => LoadCompilerSettings(msg.CompilerSettings)) .Subscribe(msg => {
var csVm = new CompilerSettingsVM(msg.CompilerSettings);
ModlistLocation.TargetPath = csVm.ProfilePath;
ModListImageLocation.TargetPath = csVm.ModListImage;
DownloadLocation.TargetPath = csVm.Downloads;
OutputLocation.TargetPath = csVm.OutputFile;
Settings = csVm;
})
.DisposeWith(CompositeDisposable); .DisposeWith(CompositeDisposable);
StatusText = "Compiler Settings"; StatusText = "Compiler Settings";
StatusProgress = Percent.Zero; StatusProgress = Percent.Zero;
BackCommand = BackCommand = ReactiveCommand.CreateFromTask(async () =>
ReactiveCommand.CreateFromTask(async () => {
{ await SaveSettingsFile();
await SaveSettingsFile(); NavigateToGlobal.Send(ScreenType.Home);
NavigateToGlobal.Send(ScreenType.Home); });
});
SubCompilerVM = new MO2CompilerVM(this); SubCompilerVM = new MO2CompilerVM(this);
ExecuteCommand = ReactiveCommand.CreateFromTask(async () => await StartCompilation()); ExecuteCommand = ReactiveCommand.CreateFromTask(async () => await StartCompilation());
ReInferSettingsCommand = ReactiveCommand.CreateFromTask(async () => await ReInferSettings(), ReInferSettingsCommand = ReactiveCommand.CreateFromTask(async () => await ReInferSettings(),
this.WhenAnyValue(vm => vm.Source) this.WhenAnyValue(vm => vm.Settings.Source)
.ObserveOnGuiThread() .ObserveOnGuiThread()
.Select(v => v != default) .Select(v => v != default)
.CombineLatest(this.WhenAnyValue(vm => vm.ModListName) .CombineLatest(this.WhenAnyValue(vm => vm.Settings.ModListName)
.ObserveOnGuiThread() .ObserveOnGuiThread()
.Select(p => !string.IsNullOrWhiteSpace(p))) .Select(p => !string.IsNullOrWhiteSpace(p)))
.Select(v => v.First && v.Second)); .Select(v => v.First && v.Second));
@ -174,23 +154,37 @@ namespace Wabbajack
Disposable.Empty.DisposeWith(disposables); Disposable.Empty.DisposeWith(disposables);
ModlistLocation.WhenAnyValue(vm => vm.TargetPath) ModlistLocation.WhenAnyValue(vm => vm.TargetPath)
.Subscribe(p => InferModListFromLocation(p).FireAndForget()) .Subscribe(async p => {
if (string.IsNullOrEmpty(Settings.ModListName))
{
Settings = new CompilerSettingsVM(await InferModListFromLocation(p));
}
else await ReInferSettings();
})
.DisposeWith(disposables); .DisposeWith(disposables);
this.WhenAnyValue(x => x.DownloadLocation.TargetPath) this.WhenAnyValue(x => x.DownloadLocation.TargetPath)
.CombineLatest(this.WhenAnyValue(x => x.ModlistLocation.TargetPath), .CombineLatest(this.WhenAnyValue(x => x.ModlistLocation.TargetPath),
this.WhenAnyValue(x => x.OutputLocation.TargetPath), this.WhenAnyValue(x => x.OutputLocation.TargetPath),
this.WhenAnyValue(x => x.DownloadLocation.ErrorState), this.WhenAnyValue(x => x.DownloadLocation.ErrorState),
this.WhenAnyValue(x => x.ModlistLocation.ErrorState), this.WhenAnyValue(x => x.ModlistLocation.ErrorState),
this.WhenAnyValue(x => x.OutputLocation.ErrorState), this.WhenAnyValue(x => x.OutputLocation.ErrorState))
this.WhenAnyValue(x => x.ModListName),
this.WhenAnyValue(x => x.Version))
.Select(_ => Validate()) .Select(_ => Validate())
.BindToStrict(this, vm => vm.ErrorState) .BindToStrict(this, vm => vm.ErrorState)
.DisposeWith(disposables); .DisposeWith(disposables);
//LoadLastSavedSettings().FireAndForget(); this.WhenAnyValue(x => x.Settings)
.Throttle(TimeSpan.FromSeconds(2))
.Subscribe(_ => SaveSettingsFile().FireAndForget())
.DisposeWith(disposables);
this.WhenAnyValue(x => x.ModListImageLocation.TargetPath)
.BindToStrict(this, vm => vm.Settings.ModListImage)
.DisposeWith(disposables);
this.WhenAnyValue(x => x.DownloadLocation.TargetPath)
.BindToStrict(this, vm => vm.Settings.Downloads)
.DisposeWith(disposables);
}); });
} }
@ -199,7 +193,7 @@ namespace Wabbajack
private async Task ReInferSettings() private async Task ReInferSettings()
{ {
var newSettings = await _inferencer.InferModListFromLocation( var newSettings = await _inferencer.InferModListFromLocation(
Source.Combine("profiles", SelectedProfile, "modlist.txt")); Settings.Source.Combine("profiles", Settings.Profile, "modlist.txt"));
if (newSettings == null) if (newSettings == null)
{ {
@ -207,28 +201,30 @@ namespace Wabbajack
return; return;
} }
Include = newSettings.Include; Settings.Include = newSettings.Include;
Ignore = newSettings.Ignore; Settings.Ignore = newSettings.Ignore;
AlwaysEnabled = newSettings.AlwaysEnabled; Settings.AlwaysEnabled = newSettings.AlwaysEnabled;
NoMatchInclude = newSettings.NoMatchInclude; Settings.NoMatchInclude = newSettings.NoMatchInclude;
OtherProfiles = newSettings.AdditionalProfiles; Settings.AdditionalProfiles = newSettings.AdditionalProfiles;
} }
private ErrorResponse Validate() private ErrorResponse Validate()
{ {
var errors = new List<ErrorResponse>(); var errors = new List<ErrorResponse>
errors.Add(DownloadLocation.ErrorState); {
errors.Add(ModlistLocation.ErrorState); DownloadLocation.ErrorState,
errors.Add(OutputLocation.ErrorState); ModlistLocation.ErrorState,
OutputLocation.ErrorState
};
return ErrorResponse.Combine(errors); return ErrorResponse.Combine(errors);
} }
private async Task InferModListFromLocation(AbsolutePath path) private async Task<CompilerSettings> InferModListFromLocation(AbsolutePath path)
{ {
using var _ = LoadingLock.WithLoading(); using var _ = LoadingLock.WithLoading();
CompilerSettings settings; CompilerSettings settings;
if (path == default) return; if (path == default) return new();
if (path.FileName.Extension == Ext.CompilerSettings) if (path.FileName.Extension == Ext.CompilerSettings)
{ {
await using var fs = path.Open(FileMode.Open, FileAccess.Read, FileShare.Read); await using var fs = path.Open(FileMode.Open, FileAccess.Read, FileShare.Read);
@ -237,46 +233,14 @@ namespace Wabbajack
else if (path.FileName == "modlist.txt".ToRelativePath()) else if (path.FileName == "modlist.txt".ToRelativePath())
{ {
settings = await _inferencer.InferModListFromLocation(path); settings = await _inferencer.InferModListFromLocation(path);
if (settings == null) return; if (settings == null) return new();
} }
else else
{ {
return; return new();
} }
LoadCompilerSettings(settings); return settings;
if (path.FileName == "modlist.txt".ToRelativePath())
{
await LoadLastSavedSettings();
}
}
private void LoadCompilerSettings(CompilerSettings settings)
{
BaseGame = settings.Game;
ModListName = settings.ModListName;
Version = settings.Version?.ToString() ?? "";
Author = settings.ModListAuthor;
Description = settings.Description;
ModListImagePath.TargetPath = settings.ModListImage;
Website = settings.ModListWebsite?.ToString() ?? "";
Readme = settings.ModListReadme?.ToString() ?? "";
IsNSFW = settings.ModlistIsNSFW;
Source = settings.Source;
DownloadLocation.TargetPath = settings.Downloads;
if (settings.OutputFile.Extension == Ext.Wabbajack)
settings.OutputFile = settings.OutputFile.Parent;
OutputLocation.TargetPath = settings.OutputFile;
SelectedProfile = settings.Profile;
PublishUpdate = settings.PublishUpdate;
MachineUrl = settings.MachineUrl;
OtherProfiles = settings.AdditionalProfiles;
AlwaysEnabled = settings.AlwaysEnabled;
NoMatchInclude = settings.NoMatchInclude;
Include = settings.Include;
Ignore = settings.Ignore;
} }
private async Task StartCompilation() private async Task StartCompilation()
@ -285,24 +249,22 @@ namespace Wabbajack
{ {
try try
{ {
await SaveSettingsFile(); await SaveSettingsFile();
var token = CancellationToken.None; var token = CancellationToken.None;
State = CompilerState.Compiling; State = CompilerState.Compiling;
var mo2Settings = GetSettings(); Settings.UseGamePaths = true;
mo2Settings.UseGamePaths = true; if (Settings.OutputFile.DirectoryExists())
if (mo2Settings.OutputFile.DirectoryExists()) Settings.OutputFile = Settings.OutputFile.Combine(Settings.ModListName.ToRelativePath()
mo2Settings.OutputFile = mo2Settings.OutputFile.Combine(mo2Settings.ModListName.ToRelativePath()
.WithExtension(Ext.Wabbajack)); .WithExtension(Ext.Wabbajack));
if (PublishUpdate && !await RunPreflightChecks(token)) if (Settings.PublishUpdate && !await RunPreflightChecks(token))
{ {
State = CompilerState.Errored; State = CompilerState.Errored;
return; return;
} }
var compiler = MO2Compiler.Create(_serviceProvider, mo2Settings); var compiler = MO2Compiler.Create(_serviceProvider, Settings.ToCompilerSettings());
var events = Observable.FromEventPattern<StatusUpdate>(h => compiler.OnStatusUpdate += h, var events = Observable.FromEventPattern<StatusUpdate>(h => compiler.OnStatusUpdate += h,
h => compiler.OnStatusUpdate -= h) h => compiler.OnStatusUpdate -= h)
@ -327,12 +289,12 @@ namespace Wabbajack
events.Dispose(); events.Dispose();
} }
if (PublishUpdate) if (Settings.PublishUpdate)
{ {
_logger.LogInformation("Publishing List"); _logger.LogInformation("Publishing List");
var downloadMetadata = _dtos.Deserialize<DownloadMetadata>( var downloadMetadata = _dtos.Deserialize<DownloadMetadata>(
await mo2Settings.OutputFile.WithExtension(Ext.Meta).WithExtension(Ext.Json).ReadAllTextAsync())!; await Settings.OutputFile.WithExtension(Ext.Meta).WithExtension(Ext.Json).ReadAllTextAsync())!;
await _wjClient.PublishModlist(MachineUrl, System.Version.Parse(Version), mo2Settings.OutputFile, downloadMetadata); await _wjClient.PublishModlist(Settings.MachineUrl, Settings.Version, Settings.OutputFile, downloadMetadata);
} }
_logger.LogInformation("Compiler Finished"); _logger.LogInformation("Compiler Finished");
@ -366,15 +328,9 @@ namespace Wabbajack
private async Task<bool> RunPreflightChecks(CancellationToken token) private async Task<bool> RunPreflightChecks(CancellationToken token)
{ {
var lists = await _wjClient.GetMyModlists(token); var lists = await _wjClient.GetMyModlists(token);
if (!lists.Any(x => x.Equals(MachineUrl, StringComparison.InvariantCultureIgnoreCase))) if (!lists.Any(x => x.Equals(Settings.MachineUrl, StringComparison.InvariantCultureIgnoreCase)))
{ {
_logger.LogError("Preflight Check failed, list {MachineUrl} not found in any repository", MachineUrl); _logger.LogError("Preflight Check failed, list {MachineUrl} not found in any repository", Settings.MachineUrl);
return false;
}
if (!System.Version.TryParse(Version, out var v))
{
_logger.LogError("Bad Version Number {Version}", Version);
return false; return false;
} }
@ -383,16 +339,14 @@ namespace Wabbajack
private async Task SaveSettingsFile() private async Task SaveSettingsFile()
{ {
if (Source == default) return; if (Settings.Source == default) return;
var settings = GetSettings(); await using var st = Settings.CompilerSettingsPath.Open(FileMode.Create, FileAccess.Write, FileShare.None);
await JsonSerializer.SerializeAsync(st, Settings.ToCompilerSettings(), _dtos.Options);
await using var st = SettingsOutputLocation.Open(FileMode.Create, FileAccess.Write, FileShare.None);
await JsonSerializer.SerializeAsync(st, settings, _dtos.Options);
var allSavedCompilerSettings = await _settingsManager.Load<List<AbsolutePath>>(Consts.AllSavedCompilerSettingsPaths); var allSavedCompilerSettings = await _settingsManager.Load<List<AbsolutePath>>(Consts.AllSavedCompilerSettingsPaths);
allSavedCompilerSettings.Remove(SettingsOutputLocation); allSavedCompilerSettings.Remove(Settings.CompilerSettingsPath);
allSavedCompilerSettings.Insert(0, SettingsOutputLocation); allSavedCompilerSettings.Insert(0, Settings.CompilerSettingsPath);
await _settingsManager.Save(Consts.AllSavedCompilerSettingsPaths, allSavedCompilerSettings); await _settingsManager.Save(Consts.AllSavedCompilerSettingsPaths, allSavedCompilerSettings);
} }
@ -408,89 +362,57 @@ namespace Wabbajack
ModlistLocation.TargetPath = lastPath; ModlistLocation.TargetPath = lastPath;
} }
private CompilerSettings GetSettings()
{
System.Version.TryParse(Version, out var pversion);
Uri.TryCreate(Website, UriKind.Absolute, out var websiteUri);
return new CompilerSettings
{
ModListName = ModListName,
ModListAuthor = Author,
Version = pversion ?? new Version(),
Description = Description,
ModListReadme = Readme,
ModListImage = ModListImagePath.TargetPath,
ModlistIsNSFW = IsNSFW,
ModListWebsite = websiteUri ?? new Uri("http://www.wabbajack.org"),
Downloads = DownloadLocation.TargetPath,
Source = Source,
Game = BaseGame,
PublishUpdate = PublishUpdate,
MachineUrl = MachineUrl,
Profile = SelectedProfile,
UseGamePaths = true,
OutputFile = OutputLocation.TargetPath,
AlwaysEnabled = AlwaysEnabled,
AdditionalProfiles = OtherProfiles,
NoMatchInclude = NoMatchInclude,
Include = Include,
Ignore = Ignore
};
}
#region ListOps #region ListOps
public void AddOtherProfile(string profile) public void AddOtherProfile(string profile)
{ {
OtherProfiles = (OtherProfiles ?? Array.Empty<string>()).Append(profile).Distinct().ToArray(); Settings.AdditionalProfiles = (Settings.AdditionalProfiles ?? Array.Empty<string>()).Append(profile).Distinct().ToArray();
} }
public void RemoveProfile(string profile) public void RemoveProfile(string profile)
{ {
OtherProfiles = OtherProfiles.Where(p => p != profile).ToArray(); Settings.AdditionalProfiles = Settings.AdditionalProfiles.Where(p => p != profile).ToArray();
} }
public void AddAlwaysEnabled(RelativePath path) public void AddAlwaysEnabled(RelativePath path)
{ {
AlwaysEnabled = (AlwaysEnabled ?? Array.Empty<RelativePath>()).Append(path).Distinct().ToArray(); Settings.AlwaysEnabled = (Settings.AlwaysEnabled ?? Array.Empty<RelativePath>()).Append(path).Distinct().ToArray();
} }
public void RemoveAlwaysEnabled(RelativePath path) public void RemoveAlwaysEnabled(RelativePath path)
{ {
AlwaysEnabled = AlwaysEnabled.Where(p => p != path).ToArray(); Settings.AlwaysEnabled = Settings.AlwaysEnabled.Where(p => p != path).ToArray();
} }
public void AddNoMatchInclude(RelativePath path) public void AddNoMatchInclude(RelativePath path)
{ {
NoMatchInclude = (NoMatchInclude ?? Array.Empty<RelativePath>()).Append(path).Distinct().ToArray(); Settings.NoMatchInclude = (Settings.NoMatchInclude ?? Array.Empty<RelativePath>()).Append(path).Distinct().ToArray();
} }
public void RemoveNoMatchInclude(RelativePath path) public void RemoveNoMatchInclude(RelativePath path)
{ {
NoMatchInclude = NoMatchInclude.Where(p => p != path).ToArray(); Settings.NoMatchInclude = Settings.NoMatchInclude.Where(p => p != path).ToArray();
} }
public void AddInclude(RelativePath path) public void AddInclude(RelativePath path)
{ {
Include = (Include ?? Array.Empty<RelativePath>()).Append(path).Distinct().ToArray(); Settings.Include = (Settings.Include ?? Array.Empty<RelativePath>()).Append(path).Distinct().ToArray();
} }
public void RemoveInclude(RelativePath path) public void RemoveInclude(RelativePath path)
{ {
Include = Include.Where(p => p != path).ToArray(); Settings.Include = Settings.Include.Where(p => p != path).ToArray();
} }
public void AddIgnore(RelativePath path) public void AddIgnore(RelativePath path)
{ {
Ignore = (Ignore ?? Array.Empty<RelativePath>()).Append(path).Distinct().ToArray(); Settings.Ignore = (Settings.Ignore ?? Array.Empty<RelativePath>()).Append(path).Distinct().ToArray();
} }
public void RemoveIgnore(RelativePath path) public void RemoveIgnore(RelativePath path)
{ {
Ignore = Ignore.Where(p => p != path).ToArray(); Settings.Ignore = Settings.Ignore.Where(p => p != path).ToArray();
} }
#endregion #endregion

View File

@ -49,7 +49,10 @@ namespace Wabbajack
_serviceProvider = serviceProvider; _serviceProvider = serviceProvider;
_dtos = dtos; _dtos = dtos;
CompileModListCommand = ReactiveCommand.Create(() => NavigateToGlobal.Send(ScreenType.Compiler)); CompileModListCommand = ReactiveCommand.Create(() => NavigateToGlobal.Send(ScreenType.Compiler));
LoadAllCompilerSettings().FireAndForget(); this.WhenActivated(disposables =>
{
LoadAllCompilerSettings().DisposeWith(disposables);
});
} }
private async Task LoadAllCompilerSettings() private async Task LoadAllCompilerSettings()

View File

@ -28,7 +28,7 @@
<ColumnDefinition Width="5*" /> <ColumnDefinition Width="5*" />
<ColumnDefinition Width="5" /> <ColumnDefinition Width="5" />
</Grid.ColumnDefinitions> </Grid.ColumnDefinitions>
<Grid Grid.Row="1" Grid.Column="3"> <Grid Grid.Row="0" Grid.Column="3">
<local:DetailImageView x:Name="DetailImage" BorderThickness="0" /> <local:DetailImageView x:Name="DetailImage" BorderThickness="0" />
</Grid> </Grid>

View File

@ -40,20 +40,20 @@ namespace Wabbajack
.BindToStrict(this, x => x.CompilationComplete.TitleText.Text) .BindToStrict(this, x => x.CompilationComplete.TitleText.Text)
.DisposeWith(disposables); .DisposeWith(disposables);
ViewModel.WhenAny(vm => vm.ModListImagePath.TargetPath) ViewModel.WhenAny(vm => vm.ModListImageLocation.TargetPath)
.Where(i => i.FileExists()) .Where(i => i.FileExists())
.Select(i => (UIUtils.TryGetBitmapImageFromFile(i, out var img), img)) .Select(i => (UIUtils.TryGetBitmapImageFromFile(i, out var img), img))
.Where(i => i.Item1) .Where(i => i.Item1)
.Select(i => i.img) .Select(i => i.img)
.BindToStrict(this, view => view.DetailImage.Image); .BindToStrict(this, view => view.DetailImage.Image);
ViewModel.WhenAny(vm => vm.ModListName) ViewModel.WhenAny(vm => vm.Settings.ModListName)
.BindToStrict(this, view => view.DetailImage.Title); .BindToStrict(this, view => view.DetailImage.Title);
ViewModel.WhenAny(vm => vm.Author) ViewModel.WhenAny(vm => vm.Settings.ModListAuthor)
.BindToStrict(this, view => view.DetailImage.Author); .BindToStrict(this, view => view.DetailImage.Author);
ViewModel.WhenAny(vm => vm.Description) ViewModel.WhenAny(vm => vm.Settings.ModListDescription)
.BindToStrict(this, view => view.DetailImage.Description); .BindToStrict(this, view => view.DetailImage.Description);
CompilationComplete.GoToModlistButton.Command = ReactiveCommand.Create(() => CompilationComplete.GoToModlistButton.Command = ReactiveCommand.Create(() =>
@ -75,18 +75,6 @@ namespace Wabbajack
.BindToStrict(this, view => view.BeginButton.Command) .BindToStrict(this, view => view.BeginButton.Command)
.DisposeWith(disposables); .DisposeWith(disposables);
/*
ViewModel.WhenAnyValue(vm => vm.BackCommand)
.BindToStrict(this, view => view.BackButton.Command)
.DisposeWith(disposables);
*/
/*
ViewModel.WhenAnyValue(vm => vm.ReInferSettingsCommand)
.BindToStrict(this, view => view.ReInferSettings.Command)
.DisposeWith(disposables);
*/
ViewModel.WhenAnyValue(vm => vm.State) ViewModel.WhenAnyValue(vm => vm.State)
.Select(v => v == CompilerState.Configuration ? Visibility.Visible : Visibility.Collapsed) .Select(v => v == CompilerState.Configuration ? Visibility.Visible : Visibility.Collapsed)
.BindToStrict(this, view => view.BottomCompilerSettingsGrid.Visibility) .BindToStrict(this, view => view.BottomCompilerSettingsGrid.Visibility)
@ -148,38 +136,45 @@ namespace Wabbajack
// Settings // Settings
this.Bind(ViewModel, vm => vm.ModListName, view => view.ModListNameSetting.Text) this.Bind(ViewModel, vm => vm.Settings.ModListName, view => view.ModListNameSetting.Text)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.SelectedProfile, view => view.SelectedProfile.Text) this.Bind(ViewModel, vm => vm.Settings.Profile, view => view.SelectedProfile.Text)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.Author, view => view.AuthorNameSetting.Text) this.Bind(ViewModel, vm => vm.Settings.ModListAuthor, view => view.AuthorNameSetting.Text)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.Version, view => view.VersionSetting.Text) this.Bind(ViewModel,
vm => vm.Settings.Version,
view => view.VersionSetting.Text,
vmVersion => vmVersion?.ToString() ?? "",
viewVersion => {
Version.TryParse(viewVersion, out var version);
return version ?? Version.Parse("1.0.0");
})
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.Description, view => view.DescriptionSetting.Text) this.Bind(ViewModel, vm => vm.Settings.ModListDescription, view => view.DescriptionSetting.Text)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.ModListImagePath, view => view.ImageFilePicker.PickerVM) this.Bind(ViewModel, vm => vm.ModListImageLocation, view => view.ImageFilePicker.PickerVM)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.Website, view => view.WebsiteSetting.Text) this.Bind(ViewModel, vm => vm.Settings.ModListWebsite, view => view.WebsiteSetting.Text)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.Readme, view => view.ReadmeSetting.Text) this.Bind(ViewModel, vm => vm.Settings.ModListReadme, view => view.ReadmeSetting.Text)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.IsNSFW, view => view.NSFWSetting.IsChecked) this.Bind(ViewModel, vm => vm.Settings.ModlistIsNSFW, view => view.NSFWSetting.IsChecked)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.PublishUpdate, view => view.PublishUpdate.IsChecked) this.Bind(ViewModel, vm => vm.Settings.PublishUpdate, view => view.PublishUpdate.IsChecked)
.DisposeWith(disposables); .DisposeWith(disposables);
this.Bind(ViewModel, vm => vm.MachineUrl, view => view.MachineUrl.Text) this.Bind(ViewModel, vm => vm.Settings.MachineUrl, view => view.MachineUrl.Text)
.DisposeWith(disposables); .DisposeWith(disposables);
@ -194,7 +189,7 @@ namespace Wabbajack
.DisposeWith(disposables); .DisposeWith(disposables);
*/ */
ViewModel.WhenAnyValue(vm => vm.AlwaysEnabled) ViewModel.WhenAnyValue(vm => vm.Settings.AlwaysEnabled)
.WhereNotNull() .WhereNotNull()
.Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveAlwaysEnabled(itm))).ToArray()) .Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveAlwaysEnabled(itm))).ToArray())
.BindToStrict(this, view => view.AlwaysEnabled.ItemsSource) .BindToStrict(this, view => view.AlwaysEnabled.ItemsSource)
@ -203,7 +198,7 @@ namespace Wabbajack
AddAlwaysEnabled.Command = ReactiveCommand.CreateFromTask(async () => await AddAlwaysEnabledCommand()); AddAlwaysEnabled.Command = ReactiveCommand.CreateFromTask(async () => await AddAlwaysEnabledCommand());
ViewModel.WhenAnyValue(vm => vm.OtherProfiles) ViewModel.WhenAnyValue(vm => vm.Settings.AdditionalProfiles)
.WhereNotNull() .WhereNotNull()
.Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveProfile(itm))).ToArray()) .Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveProfile(itm))).ToArray())
.BindToStrict(this, view => view.OtherProfiles.ItemsSource) .BindToStrict(this, view => view.OtherProfiles.ItemsSource)
@ -211,7 +206,7 @@ namespace Wabbajack
AddOtherProfile.Command = ReactiveCommand.CreateFromTask(async () => await AddOtherProfileCommand()); AddOtherProfile.Command = ReactiveCommand.CreateFromTask(async () => await AddOtherProfileCommand());
ViewModel.WhenAnyValue(vm => vm.NoMatchInclude) ViewModel.WhenAnyValue(vm => vm.Settings.NoMatchInclude)
.WhereNotNull() .WhereNotNull()
.Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveNoMatchInclude(itm))).ToArray()) .Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveNoMatchInclude(itm))).ToArray())
.BindToStrict(this, view => view.NoMatchInclude.ItemsSource) .BindToStrict(this, view => view.NoMatchInclude.ItemsSource)
@ -219,7 +214,7 @@ namespace Wabbajack
AddNoMatchInclude.Command = ReactiveCommand.CreateFromTask(async () => await AddNoMatchIncludeCommand()); AddNoMatchInclude.Command = ReactiveCommand.CreateFromTask(async () => await AddNoMatchIncludeCommand());
ViewModel.WhenAnyValue(vm => vm.Include) ViewModel.WhenAnyValue(vm => vm.Settings.Include)
.WhereNotNull() .WhereNotNull()
.Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveInclude(itm))).ToArray()) .Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveInclude(itm))).ToArray())
.BindToStrict(this, view => view.Include.ItemsSource) .BindToStrict(this, view => view.Include.ItemsSource)
@ -228,7 +223,7 @@ namespace Wabbajack
AddInclude.Command = ReactiveCommand.CreateFromTask(async () => await AddIncludeCommand()); AddInclude.Command = ReactiveCommand.CreateFromTask(async () => await AddIncludeCommand());
AddIncludeFiles.Command = ReactiveCommand.CreateFromTask(async () => await AddIncludeFilesCommand()); AddIncludeFiles.Command = ReactiveCommand.CreateFromTask(async () => await AddIncludeFilesCommand());
ViewModel.WhenAnyValue(vm => vm.Ignore) ViewModel.WhenAnyValue(vm => vm.Settings.Ignore)
.WhereNotNull() .WhereNotNull()
.Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveIgnore(itm))).ToArray()) .Select(itms => itms.Select(itm => new RemovableItemViewModel(itm.ToString(), () => ViewModel.RemoveIgnore(itm))).ToArray())
.BindToStrict(this, view => view.Ignore.ItemsSource) .BindToStrict(this, view => view.Ignore.ItemsSource)
@ -246,13 +241,13 @@ namespace Wabbajack
{ {
AbsolutePath dirPath; AbsolutePath dirPath;
if (ViewModel!.Source != default && ViewModel.Source.Combine("mods").DirectoryExists()) if (ViewModel!.Settings.Source != default && ViewModel.Settings.Source.Combine("mods").DirectoryExists())
{ {
dirPath = ViewModel.Source.Combine("mods"); dirPath = ViewModel.Settings.Source.Combine("mods");
} }
else else
{ {
dirPath = ViewModel.Source; dirPath = ViewModel.Settings.Source;
} }
var dlg = new CommonOpenFileDialog var dlg = new CommonOpenFileDialog
@ -276,9 +271,9 @@ namespace Wabbajack
{ {
var selectedPath = fileName.ToAbsolutePath(); var selectedPath = fileName.ToAbsolutePath();
if (!selectedPath.InFolder(ViewModel.Source)) continue; if (!selectedPath.InFolder(ViewModel.Settings.Source)) continue;
ViewModel.AddAlwaysEnabled(selectedPath.RelativeTo(ViewModel.Source)); ViewModel.AddAlwaysEnabled(selectedPath.RelativeTo(ViewModel.Settings.Source));
} }
} }
@ -286,13 +281,13 @@ namespace Wabbajack
{ {
AbsolutePath dirPath; AbsolutePath dirPath;
if (ViewModel!.Source != default && ViewModel.Source.Combine("mods").DirectoryExists()) if (ViewModel!.Settings.Source != default && ViewModel.Settings.Source.Combine("mods").DirectoryExists())
{ {
dirPath = ViewModel.Source.Combine("mods"); dirPath = ViewModel.Settings.Source.Combine("mods");
} }
else else
{ {
dirPath = ViewModel.Source; dirPath = ViewModel.Settings.Source;
} }
var dlg = new CommonOpenFileDialog var dlg = new CommonOpenFileDialog
@ -316,7 +311,7 @@ namespace Wabbajack
{ {
var selectedPath = filename.ToAbsolutePath(); var selectedPath = filename.ToAbsolutePath();
if (!selectedPath.InFolder(ViewModel.Source.Combine("profiles"))) continue; if (!selectedPath.InFolder(ViewModel.Settings.Source.Combine("profiles"))) continue;
ViewModel.AddOtherProfile(selectedPath.FileName.ToString()); ViewModel.AddOtherProfile(selectedPath.FileName.ToString());
} }
@ -328,10 +323,10 @@ namespace Wabbajack
{ {
Title = "Please select a folder", Title = "Please select a folder",
IsFolderPicker = true, IsFolderPicker = true,
InitialDirectory = ViewModel!.Source.ToString(), InitialDirectory = ViewModel!.Settings.Source.ToString(),
AddToMostRecentlyUsedList = false, AddToMostRecentlyUsedList = false,
AllowNonFileSystemItems = false, AllowNonFileSystemItems = false,
DefaultDirectory = ViewModel!.Source.ToString(), DefaultDirectory = ViewModel!.Settings.Source.ToString(),
EnsureFileExists = true, EnsureFileExists = true,
EnsurePathExists = true, EnsurePathExists = true,
EnsureReadOnly = false, EnsureReadOnly = false,
@ -345,9 +340,9 @@ namespace Wabbajack
{ {
var selectedPath = filename.ToAbsolutePath(); var selectedPath = filename.ToAbsolutePath();
if (!selectedPath.InFolder(ViewModel.Source)) continue; if (!selectedPath.InFolder(ViewModel.Settings.Source)) continue;
ViewModel.AddNoMatchInclude(selectedPath.RelativeTo(ViewModel!.Source)); ViewModel.AddNoMatchInclude(selectedPath.RelativeTo(ViewModel!.Settings.Source));
} }
return Task.CompletedTask; return Task.CompletedTask;
@ -359,10 +354,10 @@ namespace Wabbajack
{ {
Title = "Please select folders to include", Title = "Please select folders to include",
IsFolderPicker = true, IsFolderPicker = true,
InitialDirectory = ViewModel!.Source.ToString(), InitialDirectory = ViewModel!.Settings.Source.ToString(),
AddToMostRecentlyUsedList = false, AddToMostRecentlyUsedList = false,
AllowNonFileSystemItems = false, AllowNonFileSystemItems = false,
DefaultDirectory = ViewModel!.Source.ToString(), DefaultDirectory = ViewModel!.Settings.Source.ToString(),
EnsureFileExists = true, EnsureFileExists = true,
EnsurePathExists = true, EnsurePathExists = true,
EnsureReadOnly = false, EnsureReadOnly = false,
@ -376,9 +371,9 @@ namespace Wabbajack
{ {
var selectedPath = filename.ToAbsolutePath(); var selectedPath = filename.ToAbsolutePath();
if (!selectedPath.InFolder(ViewModel.Source)) continue; if (!selectedPath.InFolder(ViewModel.Settings.Source)) continue;
ViewModel.AddInclude(selectedPath.RelativeTo(ViewModel!.Source)); ViewModel.AddInclude(selectedPath.RelativeTo(ViewModel!.Settings.Source));
} }
} }
@ -388,10 +383,10 @@ namespace Wabbajack
{ {
Title = "Please select files to include", Title = "Please select files to include",
IsFolderPicker = false, IsFolderPicker = false,
InitialDirectory = ViewModel!.Source.ToString(), InitialDirectory = ViewModel!.Settings.Source.ToString(),
AddToMostRecentlyUsedList = false, AddToMostRecentlyUsedList = false,
AllowNonFileSystemItems = false, AllowNonFileSystemItems = false,
DefaultDirectory = ViewModel!.Source.ToString(), DefaultDirectory = ViewModel!.Settings.Source.ToString(),
EnsureFileExists = true, EnsureFileExists = true,
EnsurePathExists = true, EnsurePathExists = true,
EnsureReadOnly = false, EnsureReadOnly = false,
@ -405,9 +400,9 @@ namespace Wabbajack
{ {
var selectedPath = filename.ToAbsolutePath(); var selectedPath = filename.ToAbsolutePath();
if (!selectedPath.InFolder(ViewModel.Source)) continue; if (!selectedPath.InFolder(ViewModel.Settings.Source)) continue;
ViewModel.AddInclude(selectedPath.RelativeTo(ViewModel!.Source)); ViewModel.AddInclude(selectedPath.RelativeTo(ViewModel!.Settings.Source));
} }
} }
@ -417,10 +412,10 @@ namespace Wabbajack
{ {
Title = "Please select folders to ignore", Title = "Please select folders to ignore",
IsFolderPicker = true, IsFolderPicker = true,
InitialDirectory = ViewModel!.Source.ToString(), InitialDirectory = ViewModel!.Settings.Source.ToString(),
AddToMostRecentlyUsedList = false, AddToMostRecentlyUsedList = false,
AllowNonFileSystemItems = false, AllowNonFileSystemItems = false,
DefaultDirectory = ViewModel!.Source.ToString(), DefaultDirectory = ViewModel!.Settings.Source.ToString(),
EnsureFileExists = true, EnsureFileExists = true,
EnsurePathExists = true, EnsurePathExists = true,
EnsureReadOnly = false, EnsureReadOnly = false,
@ -434,9 +429,9 @@ namespace Wabbajack
{ {
var selectedPath = filename.ToAbsolutePath(); var selectedPath = filename.ToAbsolutePath();
if (!selectedPath.InFolder(ViewModel.Source)) continue; if (!selectedPath.InFolder(ViewModel.Settings.Source)) continue;
ViewModel.AddIgnore(selectedPath.RelativeTo(ViewModel!.Source)); ViewModel.AddIgnore(selectedPath.RelativeTo(ViewModel!.Settings.Source));
} }
} }
@ -446,10 +441,10 @@ namespace Wabbajack
{ {
Title = "Please select files to ignore", Title = "Please select files to ignore",
IsFolderPicker = false, IsFolderPicker = false,
InitialDirectory = ViewModel!.Source.ToString(), InitialDirectory = ViewModel!.Settings.Source.ToString(),
AddToMostRecentlyUsedList = false, AddToMostRecentlyUsedList = false,
AllowNonFileSystemItems = false, AllowNonFileSystemItems = false,
DefaultDirectory = ViewModel!.Source.ToString(), DefaultDirectory = ViewModel!.Settings.Source.ToString(),
EnsureFileExists = true, EnsureFileExists = true,
EnsurePathExists = true, EnsurePathExists = true,
EnsureReadOnly = false, EnsureReadOnly = false,
@ -463,9 +458,9 @@ namespace Wabbajack
{ {
var selectedPath = filename.ToAbsolutePath(); var selectedPath = filename.ToAbsolutePath();
if (!selectedPath.InFolder(ViewModel.Source)) continue; if (!selectedPath.InFolder(ViewModel.Settings.Source)) continue;
ViewModel.AddIgnore(selectedPath.RelativeTo(ViewModel!.Source)); ViewModel.AddIgnore(selectedPath.RelativeTo(ViewModel!.Settings.Source));
} }
} }
} }

View File

@ -50,7 +50,7 @@
Grid.Row="0" Grid.Row="0"
Grid.Column="1" Grid.Column="1"
VerticalAlignment="Top" VerticalAlignment="Top"
Symbol="AppsAddIn" Symbol="Add"
IsFilled="True" IsFilled="True"
FontSize="28" FontSize="28"
/> />
@ -81,7 +81,7 @@
Grid.Row="0" Grid.Row="0"
Grid.Column="1" Grid.Column="1"
VerticalAlignment="Top" VerticalAlignment="Top"
Symbol="AppsAddIn" Symbol="ArrowImport"
IsFilled="True" IsFilled="True"
FontSize="28" FontSize="28"
/> />