From bc5c825939440400eff9c4b6356a1c2528635006 Mon Sep 17 00:00:00 2001 From: Athul Raj Date: Mon, 17 Jun 2024 17:05:20 +0530 Subject: [PATCH] - show episodes section in about anime in a grid view - using binding in selector bar behavior --- Totoro.Core/ViewModels/AboutAnimeViewModel.cs | 92 +++++++++---------- Totoro.WinUI/Behaviors/SelectorBarBehavior.cs | 16 +++- .../Views/AboutSections/EpisodesSection.xaml | 30 ++++-- 3 files changed, 77 insertions(+), 61 deletions(-) diff --git a/Totoro.Core/ViewModels/AboutAnimeViewModel.cs b/Totoro.Core/ViewModels/AboutAnimeViewModel.cs index 9a241ac..77e6f67 100644 --- a/Totoro.Core/ViewModels/AboutAnimeViewModel.cs +++ b/Totoro.Core/ViewModels/AboutAnimeViewModel.cs @@ -7,8 +7,6 @@ namespace Totoro.Core.ViewModels; public class AboutAnimeViewModel : NavigatableViewModel { - private readonly SourceList _sectionsList = new(); - private readonly ReadOnlyObservableCollection _sections; public ObservableCollection Pages { get; } @@ -19,45 +17,39 @@ public AboutAnimeViewModel(IAnimeServiceContext animeService, ISimklService simklService, IEpisodesInfoProvider episodesInfoProvider) { - _sectionsList - .Connect() - .RefCount() - .AutoRefresh(x => x.Visible) - .Filter(x => x.Visible) - .Bind(out _sections) - .Subscribe() - .DisposeWith(Garbage); - - _sectionsList.Add(new PivotItemModel - { - Header = "Previews", - ViewModel = typeof(PreviewsViewModel) - }); - _sectionsList.Add(new PivotItemModel - { - Header = "Episodes", - ViewModel = typeof(AnimeEpisodesViewModel) - }); - _sectionsList.Add(new PivotItemModel - { - Header = "Related", - ViewModel = typeof(AnimeCardListViewModel), - }); - _sectionsList.Add(new PivotItemModel - { - Header = "Recommended", - ViewModel = typeof(AnimeCardListViewModel), - }); - _sectionsList.Add(new PivotItemModel - { - Header = "OST", - ViewModel = typeof(OriginalSoundTracksViewModel), - }); - _sectionsList.Add(new PivotItemModel - { - Header = "Torrents", - ViewModel = typeof(AnimeEpisodesTorrentViewModel) - }); + Sections = new( + [ + new PivotItemModel + { + Header = "Previews", + ViewModel = typeof(PreviewsViewModel) + }, + new PivotItemModel + { + Header = "Episodes", + ViewModel = typeof(AnimeEpisodesViewModel) + }, + new PivotItemModel + { + Header = "Related", + ViewModel = typeof(AnimeCardListViewModel), + }, + new PivotItemModel + { + Header = "Recommended", + ViewModel = typeof(AnimeCardListViewModel), + }, + new PivotItemModel + { + Header = "OST", + ViewModel = typeof(OriginalSoundTracksViewModel), + }, + new PivotItemModel + { + Header = "Torrents", + ViewModel = typeof(AnimeEpisodesTorrentViewModel) + } + ]); ListType = settings.DefaultListService; if (PluginFactory.Instance.Plugins.FirstOrDefault(x => x.Name == settings.DefaultProviderType) is { } provider) @@ -70,15 +62,16 @@ public AboutAnimeViewModel(IAnimeServiceContext animeService, .ObserveOn(RxApp.MainThreadScheduler) .Subscribe(async anime => { - var previewsItem = _sectionsList.Items.ElementAt(0); - var episodesItem = _sectionsList.Items.ElementAt(1); - var relatedItem = _sectionsList.Items.ElementAt(2); - var recommendedItem = _sectionsList.Items.ElementAt(3); - var ostsItem = _sectionsList.Items.ElementAt(4); - var torrentsItem = _sectionsList.Items.ElementAt(5); var sounds = animeSoundService.GetThemes(anime.Id); var episodes = await episodesInfoProvider.GetEpisodeInfos(anime.Id, GetServiceName(settings.DefaultListService)).ToListAsync(); + var previewsItem = Sections[0]; + var episodesItem = Sections[1]; + var relatedItem = Sections[2]; + var recommendedItem = Sections[3]; + var ostsItem = Sections[4]; + var torrentsItem = Sections[5]; + if (episodes.FirstOrDefault() is { EpisodeNumber: > 1 } first) { var offset = first.EpisodeNumber - 1; @@ -133,10 +126,12 @@ public AboutAnimeViewModel(IAnimeServiceContext animeService, if (episodes is not { Count: > 0 }) { episodesItem.Visible = false; + torrentsItem.Visible = false; } SelectedSection = null; SelectedSection = Sections.FirstOrDefault(); + IsLoading = false; }); this.ObservableForProperty(x => x.Id, x => x) @@ -147,7 +142,6 @@ public AboutAnimeViewModel(IAnimeServiceContext animeService, .Subscribe(x => { Anime = x; - IsLoading = false; }, RxApp.DefaultExceptionHandler.OnError); this.WhenAnyValue(x => x.Anime) @@ -194,7 +188,7 @@ public AboutAnimeViewModel(IAnimeServiceContext animeService, [ObservableAsProperty] public bool CanWatch { get; } [ObservableAsProperty] public bool HasTracking { get; } - public ReadOnlyObservableCollection Sections => _sections; + public ReadOnlyObservableCollection Sections { get; } public string DefaultProviderType { get; } public ListServiceType ListType { get; } diff --git a/Totoro.WinUI/Behaviors/SelectorBarBehavior.cs b/Totoro.WinUI/Behaviors/SelectorBarBehavior.cs index 0ccba30..5317bcc 100644 --- a/Totoro.WinUI/Behaviors/SelectorBarBehavior.cs +++ b/Totoro.WinUI/Behaviors/SelectorBarBehavior.cs @@ -1,5 +1,6 @@ using Microsoft.UI.Xaml; using Microsoft.UI.Xaml.Controls; +using Microsoft.UI.Xaml.Data; using Microsoft.Xaml.Interactivity; using ReactiveMarbles.ObservableEvents; using Totoro.Core.ViewModels; @@ -35,13 +36,22 @@ protected override void OnAttached() .Subscribe(values => { AssociatedObject.Items.Clear(); - foreach (var item in values.Where(x => x.Visible)) + foreach (var item in values) { - AssociatedObject.Items.Add(new SelectorBarItem + var barItem = new SelectorBarItem { Text = item.Header, - FontSize = 20 + FontSize = 20, + }; + + barItem.SetBinding(UIElement.VisibilityProperty, new Binding + { + Source = item, + Path = new PropertyPath(nameof(item.Visible)), + Mode = BindingMode.OneWay }); + + AssociatedObject.Items.Add(barItem); } if (SelectedItem is null) diff --git a/Totoro.WinUI/Views/AboutSections/EpisodesSection.xaml b/Totoro.WinUI/Views/AboutSections/EpisodesSection.xaml index c18f38e..80fd7c7 100644 --- a/Totoro.WinUI/Views/AboutSections/EpisodesSection.xaml +++ b/Totoro.WinUI/Views/AboutSections/EpisodesSection.xaml @@ -31,20 +31,27 @@ - + + + + + + - - - + + + - + @@ -53,14 +60,19 @@ - - + + + + - +