wabbajack/Wabbajack/View Models/ModListMetadataVM.cs

176 lines
6.3 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
2019-11-30 09:08:04 +00:00
using System.Diagnostics;
using System.Linq;
using System.Net;
2019-11-30 09:08:04 +00:00
using System.Reactive;
using System.Reactive.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Input;
using System.Windows.Media.Imaging;
using Alphaleonis.Win32.Filesystem;
using ReactiveUI;
2019-11-30 09:08:04 +00:00
using ReactiveUI.Fody.Helpers;
using Wabbajack.Common;
using Wabbajack.Lib;
using Wabbajack.Lib.Downloaders;
using Wabbajack.Lib.ModListRegistry;
2019-11-30 09:08:04 +00:00
namespace Wabbajack
{
public class ModListMetadataVM : ViewModel
{
2019-11-30 09:08:04 +00:00
public ModlistMetadata Metadata { get; }
private ModListGalleryVM _parent;
2019-11-30 09:08:04 +00:00
public ICommand OpenWebsiteCommand { get; }
public ICommand ExecuteCommand { get; }
2019-11-30 09:08:04 +00:00
private readonly ObservableAsPropertyHelper<bool> _Exists;
public bool Exists => _Exists.Value;
public string Location { get; }
2019-11-30 09:08:04 +00:00
[Reactive]
public double ProgressPercent { get; private set; }
2019-12-17 04:06:30 +00:00
[Reactive]
public bool IsBroken { get; private set; }
[Reactive]
public IErrorResponse Error { get; private set; }
private readonly ObservableAsPropertyHelper<BitmapImage> _Image;
public BitmapImage Image => _Image.Value;
private readonly ObservableAsPropertyHelper<bool> _LoadingImage;
public bool LoadingImage => _LoadingImage.Value;
2019-11-30 09:08:04 +00:00
public ModListMetadataVM(ModListGalleryVM parent, ModlistMetadata metadata)
{
_parent = parent;
Metadata = metadata;
Location = Path.Combine(Consts.ModListDownloadFolder, Metadata.Links.MachineURL + ExtensionManager.Extension);
2019-12-17 04:06:30 +00:00
IsBroken = metadata.ValidationSummary.HasFailures;
2019-11-30 09:08:04 +00:00
OpenWebsiteCommand = ReactiveCommand.Create(() => Process.Start($"https://www.wabbajack.org/modlist/{Metadata.Links.MachineURL}"));
ExecuteCommand = ReactiveCommand.CreateFromObservable<Unit, Unit>(
2019-12-17 04:06:30 +00:00
canExecute: this.WhenAny(x => x.IsBroken).Select(x => !x),
execute: (unit) =>
2019-11-30 09:08:04 +00:00
Observable.Return(unit)
.WithLatestFrom(
this.WhenAny(x => x.Exists),
(_, e) => e)
// Do any download work on background thread
.ObserveOn(RxApp.TaskpoolScheduler)
.SelectTask(async (exists) =>
{
if (!exists)
{
try
{
var success = await Download();
if (!success)
{
Error = ErrorResponse.Fail("Download was marked unsuccessful");
return false;
}
}
catch (Exception ex)
{
Error = ErrorResponse.Fail(ex);
return false;
}
2019-11-30 09:08:04 +00:00
// Return an updated check on exists
return File.Exists(Location);
}
return exists;
})
.Where(exists => exists)
2019-11-30 09:08:04 +00:00
// Do any install page swap over on GUI thread
.ObserveOnGuiThread()
.Select(_ =>
2019-11-30 09:08:04 +00:00
{
_parent.MWVM.OpenInstaller(Path.GetFullPath(Location));
// Wait for modlist member to be filled, then open its readme
return _parent.MWVM.Installer.Value.WhenAny(x => x.ModList)
.NotNull()
.Take(1)
.Do(modList =>
{
try
{
modList.OpenReadmeWindow();
}
catch (Exception ex)
{
Utils.Error(ex);
}
});
})
.Switch()
.Unit());
2019-11-30 09:08:04 +00:00
_Exists = Observable.Interval(TimeSpan.FromSeconds(0.5))
.Unit()
.StartWith(Unit.Default)
.FlowSwitch(_parent.WhenAny(x => x.IsActive))
.Select(_ =>
{
try
{
return !metadata.NeedsDownload(Location);
}
catch (Exception)
{
return true;
}
})
.ToGuiProperty(this, nameof(Exists));
var imageObs = Observable.Return(Metadata.Links.ImageUri)
.DownloadBitmapImage((ex) => Utils.Log($"Error downloading modlist image {Metadata.Title}"));
_Image = imageObs
.ToGuiProperty(this, nameof(Image));
_LoadingImage = imageObs
.Select(x => false)
.StartWith(true)
.ToGuiProperty(this, nameof(LoadingImage));
}
private async Task<bool> Download()
{
2019-11-30 09:08:04 +00:00
ProgressPercent = 0d;
using (var queue = new WorkQueue(1))
using (queue.Status.Select(i => i.ProgressPercent)
.Subscribe(percent => ProgressPercent = percent))
{
var tcs = new TaskCompletionSource<bool>();
queue.QueueTask(async () =>
{
try
{
var downloader = DownloadDispatcher.ResolveArchive(Metadata.Links.Download);
var result = await downloader.Download(new Archive { Name = Metadata.Title, Size = Metadata.DownloadMetadata?.Size ?? 0 }, Location);
// Want to rehash to current file, even if failed?
Location.FileHashCached();
tcs.SetResult(result);
}
catch (Exception ex)
{
tcs.SetException(ex);
}
});
await Metrics.Send(Metrics.Downloading, Metadata.Title);
return await tcs.Task;
}
}
}
}