2021-12-27 05:13:28 +00:00
|
|
|
|
using System;
|
2021-12-27 16:24:45 +00:00
|
|
|
|
using System.Linq;
|
2021-12-26 21:56:44 +00:00
|
|
|
|
using System.Reactive;
|
|
|
|
|
using System.Reactive.Linq;
|
|
|
|
|
using System.Windows.Media.Imaging;
|
2021-12-27 05:13:28 +00:00
|
|
|
|
using DynamicData;
|
2021-12-27 16:24:45 +00:00
|
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
2021-12-27 05:13:28 +00:00
|
|
|
|
using ReactiveUI;
|
|
|
|
|
using ReactiveUI.Fody.Helpers;
|
2021-12-27 16:24:45 +00:00
|
|
|
|
using Wabbajack.Common;
|
2021-12-27 00:38:44 +00:00
|
|
|
|
using Wabbajack.DTOs.DownloadStates;
|
2021-12-26 21:56:44 +00:00
|
|
|
|
using Wabbajack.Lib;
|
2021-12-27 00:38:44 +00:00
|
|
|
|
using Wabbajack.Lib.Extensions;
|
2021-12-26 21:56:44 +00:00
|
|
|
|
|
2021-12-27 05:13:28 +00:00
|
|
|
|
namespace Wabbajack.View_Models
|
2021-12-26 21:56:44 +00:00
|
|
|
|
{
|
|
|
|
|
public class SlideShow : ViewModel
|
|
|
|
|
{
|
|
|
|
|
private readonly Random _random = new Random();
|
|
|
|
|
|
|
|
|
|
public InstallerVM Installer { get; }
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public bool ShowNSFW { get; set; }
|
|
|
|
|
|
|
|
|
|
[Reactive]
|
|
|
|
|
public bool Enable { get; set; } = true;
|
|
|
|
|
|
|
|
|
|
private readonly ObservableAsPropertyHelper<BitmapImage> _image;
|
|
|
|
|
public BitmapImage Image => _image.Value;
|
|
|
|
|
|
|
|
|
|
private readonly ObservableAsPropertyHelper<ModVM> _targetMod;
|
|
|
|
|
public ModVM TargetMod => _targetMod.Value;
|
|
|
|
|
|
|
|
|
|
public ReactiveCommand<Unit, Unit> SlideShowNextItemCommand { get; } = ReactiveCommand.Create(() => { });
|
|
|
|
|
public ReactiveCommand<Unit, Unit> VisitURLCommand { get; }
|
|
|
|
|
|
|
|
|
|
public const int PreloadAmount = 4;
|
|
|
|
|
|
2021-12-28 21:39:20 +00:00
|
|
|
|
public SlideShow(InstallerVM appState, IServiceProvider provider)
|
2021-12-26 21:56:44 +00:00
|
|
|
|
{
|
|
|
|
|
Installer = appState;
|
|
|
|
|
|
|
|
|
|
// Wire target slideshow index
|
|
|
|
|
var intervalSeconds = 10;
|
|
|
|
|
// Compile all the sources that trigger a slideshow update, any of which trigger a counter update
|
|
|
|
|
var selectedIndex = Observable.Merge(
|
|
|
|
|
// If user requests one manually
|
|
|
|
|
SlideShowNextItemCommand.StartingExecution(),
|
|
|
|
|
// If the natural timer fires
|
|
|
|
|
Observable.Merge(
|
|
|
|
|
// Start with an initial timer
|
|
|
|
|
Observable.Return(Observable.Interval(TimeSpan.FromSeconds(intervalSeconds))),
|
|
|
|
|
// but reset timer if user requests one
|
|
|
|
|
SlideShowNextItemCommand.StartingExecution()
|
|
|
|
|
.Select(_ => Observable.Interval(TimeSpan.FromSeconds(intervalSeconds))))
|
|
|
|
|
// When a new timer comes in, swap to it
|
|
|
|
|
.Switch()
|
|
|
|
|
.Unit()
|
|
|
|
|
// Only subscribe to timer if enabled and installing
|
|
|
|
|
.FlowSwitch(
|
|
|
|
|
Observable.CombineLatest(
|
|
|
|
|
this.WhenAny(x => x.Enable),
|
|
|
|
|
this.WhenAny(x => x.Installer.Installing),
|
|
|
|
|
resultSelector: (enabled, installing) => enabled && installing)))
|
|
|
|
|
// When filter switch enabled, fire an initial signal
|
|
|
|
|
.StartWith(Unit.Default)
|
|
|
|
|
// Only subscribe to slideshow triggers if started
|
|
|
|
|
.FlowSwitch(this.WhenAny(x => x.Installer.StartedInstallation))
|
|
|
|
|
// Block spam
|
|
|
|
|
.Debounce(TimeSpan.FromMilliseconds(250), RxApp.MainThreadScheduler)
|
|
|
|
|
.Scan(
|
|
|
|
|
seed: 0,
|
|
|
|
|
accumulator: (i, _) => i + 1)
|
|
|
|
|
.Publish()
|
|
|
|
|
.RefCount();
|
|
|
|
|
|
|
|
|
|
// Dynamic list changeset of mod VMs to display
|
|
|
|
|
var modVMs = this.WhenAny(x => x.Installer.ModList)
|
|
|
|
|
// Whenever modlist changes, grab the list of its slides
|
|
|
|
|
.Select(modList =>
|
|
|
|
|
{
|
|
|
|
|
if (modList?.SourceModList?.Archives == null)
|
|
|
|
|
{
|
|
|
|
|
return Observable.Empty<IMetaState>()
|
2021-12-27 05:13:28 +00:00
|
|
|
|
.ToObservableChangeSet(x => x.LinkUrl);
|
2021-12-26 21:56:44 +00:00
|
|
|
|
}
|
|
|
|
|
return modList.SourceModList.Archives
|
|
|
|
|
.Select(m => m.State)
|
|
|
|
|
.OfType<IMetaState>()
|
2021-12-27 16:24:45 +00:00
|
|
|
|
.Where(x => x.LinkUrl != default && x.ImageURL != default)
|
|
|
|
|
.DistinctBy(x => x.LinkUrl)
|
2021-12-26 21:56:44 +00:00
|
|
|
|
// Shuffle it
|
|
|
|
|
.Shuffle(_random)
|
2021-12-27 16:24:45 +00:00
|
|
|
|
.AsObservableChangeSet(x => x.LinkUrl);
|
2021-12-26 21:56:44 +00:00
|
|
|
|
})
|
|
|
|
|
// Switch to the new list after every ModList change
|
|
|
|
|
.Switch()
|
2021-12-27 16:24:45 +00:00
|
|
|
|
.Transform(mod => new ModVM(provider.GetService<ILogger<ModVM>>(), mod))
|
2021-12-26 21:56:44 +00:00
|
|
|
|
.DisposeMany()
|
|
|
|
|
// Filter out any NSFW slides if we don't want them
|
|
|
|
|
.AutoRefreshOnObservable(slide => this.WhenAny(x => x.ShowNSFW))
|
|
|
|
|
.Filter(slide => !slide.State.IsNSFW || ShowNSFW)
|
|
|
|
|
.RefCount();
|
|
|
|
|
|
|
|
|
|
// Find target mod to display by combining dynamic list with currently desired index
|
|
|
|
|
_targetMod = Observable.CombineLatest(
|
|
|
|
|
modVMs.QueryWhenChanged(),
|
|
|
|
|
selectedIndex,
|
|
|
|
|
resultSelector: (query, selected) =>
|
|
|
|
|
{
|
|
|
|
|
var index = selected % (query.Count == 0 ? 1 : query.Count);
|
|
|
|
|
return query.Items.ElementAtOrDefault(index);
|
|
|
|
|
})
|
|
|
|
|
.StartWith(default(ModVM))
|
|
|
|
|
.ToGuiProperty(this, nameof(TargetMod));
|
|
|
|
|
|
|
|
|
|
// Mark interest and materialize image of target mod
|
|
|
|
|
_image = this.WhenAny(x => x.TargetMod)
|
|
|
|
|
// We want to Switch here, not SelectMany, as we want to hotswap to newest target without waiting on old ones
|
|
|
|
|
.Select(x => x?.ImageObservable ?? Observable.Return(default(BitmapImage)))
|
|
|
|
|
.Switch()
|
|
|
|
|
.ToGuiProperty(this, nameof(Image));
|
|
|
|
|
|
|
|
|
|
VisitURLCommand = ReactiveCommand.Create(
|
|
|
|
|
execute: () =>
|
|
|
|
|
{
|
2021-12-27 16:24:45 +00:00
|
|
|
|
UIUtils.OpenWebsite(TargetMod.State.LinkUrl);
|
2021-12-26 21:56:44 +00:00
|
|
|
|
return Unit.Default;
|
|
|
|
|
},
|
2021-12-27 16:24:45 +00:00
|
|
|
|
canExecute: this.WhenAny(x => x.TargetMod.State.LinkUrl)
|
2021-12-26 21:56:44 +00:00
|
|
|
|
.Select(x =>
|
|
|
|
|
{
|
|
|
|
|
//var regex = new Regex("^(http|https):\\/\\/");
|
|
|
|
|
var scheme = x?.Scheme;
|
|
|
|
|
return scheme != null &&
|
|
|
|
|
(scheme.Equals("https", StringComparison.OrdinalIgnoreCase) ||
|
|
|
|
|
scheme.Equals("http", StringComparison.OrdinalIgnoreCase));
|
|
|
|
|
})
|
|
|
|
|
.ObserveOnGuiThread());
|
|
|
|
|
|
|
|
|
|
// Preload upcoming images
|
|
|
|
|
var list = Observable.CombineLatest(
|
|
|
|
|
modVMs.QueryWhenChanged(),
|
|
|
|
|
selectedIndex,
|
|
|
|
|
resultSelector: (query, selected) =>
|
|
|
|
|
{
|
|
|
|
|
// Retrieve the mods that should be preloaded
|
|
|
|
|
var index = selected % (query.Count == 0 ? 1 : query.Count);
|
|
|
|
|
var amountToTake = Math.Min(query.Count - index, PreloadAmount);
|
|
|
|
|
return query.Items.Skip(index).Take(amountToTake).ToObservable();
|
|
|
|
|
})
|
|
|
|
|
.Select(i => i.ToObservableChangeSet())
|
|
|
|
|
.Switch()
|
|
|
|
|
.Transform(mod => mod.ImageObservable.Subscribe())
|
|
|
|
|
.DisposeMany()
|
|
|
|
|
.AsObservableList();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|