From 8cf9ae288b3d246bbacf3593028b95d29a2cffcc Mon Sep 17 00:00:00 2001 From: Tsanie Lily Date: Thu, 7 May 2020 15:19:24 +0800 Subject: [PATCH] feature: favorite illusts --- Pixiview/AppShell.xaml | 5 ++ Pixiview/Illust/FavoritesPage.xaml | 25 ++++++ Pixiview/Illust/FavoritesPage.xaml.cs | 53 +++++++++++++ Pixiview/Illust/IllustCollectionPage.cs | 101 ++++++++++++++++-------- Pixiview/Illust/MainPage.xaml | 20 +++-- Pixiview/Illust/MainPage.xaml.cs | 27 +++++-- Pixiview/Illust/RankingPage.xaml | 20 ++--- Pixiview/Illust/RankingPage.xaml.cs | 36 ++++++++- Pixiview/Illust/RecommendsPage.xaml | 24 +++--- Pixiview/Illust/RecommendsPage.xaml.cs | 47 +++++++++-- Pixiview/Illust/ViewIllustPage.xaml | 6 +- Pixiview/Illust/ViewIllustPage.xaml.cs | 60 ++++++++++++-- Pixiview/OptionPage.xaml | 6 +- Pixiview/Resources/Languages/zh-CN.xml | 1 + Pixiview/UI/AdaptedPage.cs | 3 + Pixiview/UI/StyleDefinition.cs | 2 + Pixiview/UI/Theme/ThemeBase.cs | 6 ++ Pixiview/Utils/IllustData.cs | 18 +++++ Pixiview/Utils/Stores.cs | 80 +++++++++++++++++++ 19 files changed, 446 insertions(+), 94 deletions(-) create mode 100644 Pixiview/Illust/FavoritesPage.xaml create mode 100644 Pixiview/Illust/FavoritesPage.xaml.cs diff --git a/Pixiview/AppShell.xaml b/Pixiview/AppShell.xaml index 63cbb30..a59005e 100644 --- a/Pixiview/AppShell.xaml +++ b/Pixiview/AppShell.xaml @@ -85,6 +85,11 @@ Route="{x:Static util:Routes.Ranking}"> + + + + + + + + + + + + + diff --git a/Pixiview/Illust/FavoritesPage.xaml.cs b/Pixiview/Illust/FavoritesPage.xaml.cs new file mode 100644 index 0000000..0b0954d --- /dev/null +++ b/Pixiview/Illust/FavoritesPage.xaml.cs @@ -0,0 +1,53 @@ +using System.Collections.Generic; +using System.Linq; +using System.Windows.Input; +using Pixiview.Utils; + +namespace Pixiview.Illust +{ + public partial class FavoritesPage : FavoriteIllustCollectionPage + { + public FavoritesPage() + { + Resources.Add("cardView", GetCardViewTemplate()); + InitializeComponent(); + } + + protected override void OnAppearing() + { + base.OnAppearing(); + + StartLoad(true); + } + + protected override void OnDisappearing() + { + base.OnDisappearing(); + + var illusts = Illusts; + if (illusts != null) + { + illusts.Clear(); + } + } + + protected override IEnumerable DoGetIllustList(IllustItem[] data, ICommand command) + { + return data.Select(i => + { + i.IllustTapped = command; + return i; + }); + } + + protected override IllustItem[] DoLoadIllustData(bool force) + { + var favorites = Stores.LoadFavoritesIllusts(); + if (favorites == null) + { + return null; + } + return favorites.Illusts; + } + } +} diff --git a/Pixiview/Illust/IllustCollectionPage.cs b/Pixiview/Illust/IllustCollectionPage.cs index b47f04a..69ff7d2 100644 --- a/Pixiview/Illust/IllustCollectionPage.cs +++ b/Pixiview/Illust/IllustCollectionPage.cs @@ -1,8 +1,10 @@ -using System.Collections.Generic; +using System; +using System.Collections.Generic; using System.Collections.ObjectModel; using System.Linq; using System.Threading.Tasks; using System.Windows.Input; +using Newtonsoft.Json; using Pixiview.Resources; using Pixiview.UI; using Pixiview.UI.Theme; @@ -12,20 +14,28 @@ using Xamarin.Forms; namespace Pixiview.Illust { - public abstract class IllustCollectionPage : AdaptedPage + public interface IIllustCollectionPage + { + List Favorites { get; } + } + + public abstract class IllustDataCollectionPage : IllustCollectionPage { } + public abstract class FavoriteIllustCollectionPage : IllustCollectionPage { } + + public abstract class IllustCollectionPage : AdaptedPage, IIllustCollectionPage { #region - Properties - public static readonly BindableProperty IllustsProperty = BindableProperty.Create( - nameof(Illusts), typeof(IllustCollection), typeof(IllustCollectionPage)); + nameof(Illusts), typeof(IllustCollection), typeof(IllustCollectionPage)); public static readonly BindableProperty ColumnsProperty = BindableProperty.Create( - nameof(Columns), typeof(int), typeof(IllustCollectionPage), 2); + nameof(Columns), typeof(int), typeof(IllustCollectionPage), 2); public static readonly BindableProperty LoadingProperty = BindableProperty.Create( - nameof(Loading), typeof(bool), typeof(IllustCollectionPage), propertyChanged: OnLoadingPropertyChanged); + nameof(Loading), typeof(bool), typeof(IllustCollectionPage), propertyChanged: OnLoadingPropertyChanged); private static void OnLoadingPropertyChanged(BindableObject obj, object oldValue, object newValue) { - var page = (IllustCollectionPage)obj; + var page = (IllustCollectionPage)obj; var now = (bool)newValue; if (!page.loaded && now && Stores.NetworkAvailable) { @@ -50,11 +60,13 @@ namespace Pixiview.Illust set => SetValue(LoadingProperty, value); } + public List Favorites { get; } = new List(); + #endregion protected bool loaded; - private IllustData illustData; + private T illustData; private readonly ParallelOptions parallelOptions = new ParallelOptions { MaxDegreeOfParallelism = Configs.MaxThreads }; private readonly Command commandIllustImageTapped; @@ -62,6 +74,12 @@ namespace Pixiview.Illust { BindingContext = this; commandIllustImageTapped = new Command(IllustImageTapped); + + var favorites = Stores.LoadFavoritesIllusts(); + if (favorites != null) + { + Favorites.AddRange(favorites.Illusts); + } } private void IllustImageTapped(IllustItem illust) @@ -128,11 +146,11 @@ namespace Pixiview.Illust #endregion - protected abstract IllustData DoLoadIllustData(bool force); - protected abstract IEnumerable DoGetIllustList(IllustData data); + protected abstract T DoLoadIllustData(bool force); + protected abstract IEnumerable DoGetIllustList(T data, ICommand command); protected virtual void OnIllustImageTapped(IllustItem illust) { - var page = new ViewIllustPage(illust); + var page = new ViewIllustPage(illust, this); Navigation.PushAsync(page); } @@ -270,33 +288,12 @@ namespace Pixiview.Illust illustData = DoLoadIllustData(force); if (illustData == null) { - App.DebugError("illusts.load", "failed to load illusts data."); + //App.DebugError("illusts.load", "failed to load illusts data."); + Loading = false; return; } - var data = DoGetIllustList(illustData).Select(id => - { - var illust = illustData.body.thumbnails.illust.FirstOrDefault(l => l.illustId == id); - if (illust == null) - { - return null; - } - return new IllustItem - { - Id = illust.illustId, - ImageUrl = illust.urls.x360 ?? illust.url, - Title = illust.illustTitle, - IsRestrict = illust.xRestrict == 1, - ProfileUrl = illust.profileImageUrl, - UserId = illust.userId, - UserName = illust.userName, - Width = illust.width, - Height = illust.height, - PageCount = illust.pageCount, - - IllustTapped = commandIllustImageTapped - }; - }).Where(i => i != null); + var data = DoGetIllustList(illustData, commandIllustImageTapped).Where(i => i != null); var collection = new IllustCollection(data); Illusts = collection; @@ -340,7 +337,24 @@ namespace Pixiview.Illust public class IllustCollection : ObservableCollection { private static readonly object sync = new object(); + private static IllustCollection empty; + public static IllustCollection Empty + { + get + { + if (empty == null) + { + empty = new IllustCollection(); + } + return empty; + } + } + + public IllustCollection() : base() + { + running = true; + } public IllustCollection(IEnumerable illusts) : base(illusts) { running = true; @@ -360,6 +374,13 @@ namespace Pixiview.Illust } } + public class IllustFavorite + { + public DateTime LastFavoriteUtc { get; set; } + public IllustItem[] Illusts { get; set; } + } + + [JsonObject(MemberSerialization.OptIn)] public class IllustItem : BindableObject { public static readonly BindableProperty ImageProperty = BindableProperty.Create( @@ -379,6 +400,7 @@ namespace Pixiview.Illust get => (ImageSource)GetValue(ProfileImageProperty); set => SetValue(ProfileImageProperty, value); } + [JsonProperty] public GridLength ImageHeight { get => (GridLength)GetValue(ImageHeightProperty); @@ -386,16 +408,27 @@ namespace Pixiview.Illust } public ICommand IllustTapped { get; set; } + [JsonProperty] public string Id { get; set; } + [JsonProperty] public string ImageUrl { get; set; } + [JsonProperty] public string Title { get; set; } + [JsonProperty] public bool IsRestrict { get; set; } + [JsonProperty] public string ProfileUrl { get; set; } + [JsonProperty] public string UserId { get; set; } + [JsonProperty] public string UserName { get; set; } + [JsonProperty] public int Width { get; set; } + [JsonProperty] public int Height { get; set; } + [JsonProperty] public int PageCount { get; set; } + public string PageCountText => $"{StyleDefinition.IconLayer} {PageCount}"; public bool IsPageVisible => PageCount > 1; } diff --git a/Pixiview/Illust/MainPage.xaml b/Pixiview/Illust/MainPage.xaml index 5b28a13..bfbe213 100644 --- a/Pixiview/Illust/MainPage.xaml +++ b/Pixiview/Illust/MainPage.xaml @@ -1,14 +1,12 @@  - + @@ -28,4 +26,4 @@ Color="{DynamicResource WindowColor}"/> - \ No newline at end of file + diff --git a/Pixiview/Illust/MainPage.xaml.cs b/Pixiview/Illust/MainPage.xaml.cs index 3376229..8951355 100644 --- a/Pixiview/Illust/MainPage.xaml.cs +++ b/Pixiview/Illust/MainPage.xaml.cs @@ -1,26 +1,41 @@ using System; using System.Collections.Generic; using System.Linq; +using System.Windows.Input; using Pixiview.Utils; namespace Pixiview.Illust { - // Learn more about making custom code visible in the Xamarin.Forms previewer - // by visiting https://aka.ms/xamarinforms-previewer - //[DesignTimeVisible(false)] - public partial class MainPage : IllustCollectionPage + public partial class MainPage : IllustDataCollectionPage { public MainPage() { Resources.Add("cardView", GetCardViewTemplate()); InitializeComponent(); + } + public override void OnLoad() + { StartLoad(); } - protected override IEnumerable DoGetIllustList(IllustData data) + public override void OnUnload() { - return data.body.page.follow.Select(i => i.ToString()); + Illusts = IllustCollection.Empty; + loaded = false; + } + + protected override IEnumerable DoGetIllustList(IllustData data, ICommand command) + { + return data.body.page.follow.Select(i => + { + var item = data.body.thumbnails.illust.FirstOrDefault(l => l.illustId == i.ToString())?.ConvertToItem(); + if (item != null) + { + item.IllustTapped = command; + } + return item; + }); } protected override IllustData DoLoadIllustData(bool force) diff --git a/Pixiview/Illust/RankingPage.xaml b/Pixiview/Illust/RankingPage.xaml index 5f0766a..491deab 100644 --- a/Pixiview/Illust/RankingPage.xaml +++ b/Pixiview/Illust/RankingPage.xaml @@ -1,12 +1,14 @@  - + + + + - + diff --git a/Pixiview/Illust/RankingPage.xaml.cs b/Pixiview/Illust/RankingPage.xaml.cs index bb4cfcd..770d383 100644 --- a/Pixiview/Illust/RankingPage.xaml.cs +++ b/Pixiview/Illust/RankingPage.xaml.cs @@ -1,23 +1,42 @@ -using System.Collections.Generic; +using System; +using System.Collections.Generic; using System.Linq; +using System.Windows.Input; using Pixiview.Utils; using Xamarin.Forms; namespace Pixiview.Illust { - public partial class RankingPage : IllustCollectionPage + public partial class RankingPage : IllustDataCollectionPage { public RankingPage() { Resources.Add("cardView", GetCardViewTemplate()); InitializeComponent(); + } + public override void OnLoad() + { StartLoad(); } - protected override IEnumerable DoGetIllustList(IllustData data) + public override void OnUnload() { - return data.body.page.ranking.items.Select(i => i.id); + Illusts = IllustCollection.Empty; + loaded = false; + } + + protected override IEnumerable DoGetIllustList(IllustData data, ICommand command) + { + return data.body.page.ranking.items.Select(i => + { + var item = data.body.thumbnails.illust.FirstOrDefault(l => l.illustId == i.id)?.ConvertToItem(); + if (item != null) + { + item.IllustTapped = command; + } + return item; + }); } protected override IllustData DoLoadIllustData(bool force) @@ -34,5 +53,14 @@ namespace Pixiview.Illust } return data; } + + private void Refresh_Clicked(object sender, EventArgs e) + { + if (Loading) + { + return; + } + StartLoad(true); + } } } diff --git a/Pixiview/Illust/RecommendsPage.xaml b/Pixiview/Illust/RecommendsPage.xaml index 13793c1..8993efb 100644 --- a/Pixiview/Illust/RecommendsPage.xaml +++ b/Pixiview/Illust/RecommendsPage.xaml @@ -1,14 +1,16 @@  - + + + + - + diff --git a/Pixiview/Illust/RecommendsPage.xaml.cs b/Pixiview/Illust/RecommendsPage.xaml.cs index b71bbda..3c2d2b2 100644 --- a/Pixiview/Illust/RecommendsPage.xaml.cs +++ b/Pixiview/Illust/RecommendsPage.xaml.cs @@ -1,10 +1,12 @@ -using System.Collections.Generic; +using System; +using System.Collections.Generic; using System.Linq; +using System.Windows.Input; using Pixiview.Utils; namespace Pixiview.Illust { - public partial class RecommendsPage : IllustCollectionPage + public partial class RecommendsPage : IllustDataCollectionPage { public bool ByUser { get; set; } @@ -12,19 +14,45 @@ namespace Pixiview.Illust { Resources.Add("cardView", GetCardViewTemplate()); InitializeComponent(); + } + public override void OnLoad() + { StartLoad(); } - protected override IEnumerable DoGetIllustList(IllustData data) + public override void OnUnload() + { + Illusts = IllustCollection.Empty; + loaded = false; + } + + protected override IEnumerable DoGetIllustList(IllustData data, ICommand command) { if (ByUser) { - return data.body.page.recommendUser.SelectMany(i => i.illustIds); + return data.body.page.recommendUser.SelectMany(i => i.illustIds) + .Select(id => + { + var item = data.body.thumbnails.illust.FirstOrDefault(l => l.illustId == id)?.ConvertToItem(); + if (item != null) + { + item.IllustTapped = command; + } + return item; + }); } else { - return data.body.page.recommend; + return data.body.page.recommend.Select(id => + { + var item = data.body.thumbnails.illust.FirstOrDefault(l => l.illustId == id)?.ConvertToItem(); + if (item != null) + { + item.IllustTapped = command; + } + return item; + }); } } @@ -32,5 +60,14 @@ namespace Pixiview.Illust { return Stores.LoadIllustData(force); } + + private void Refresh_Clicked(object sender, EventArgs e) + { + if (Loading) + { + return; + } + StartLoad(true); + } } } diff --git a/Pixiview/Illust/ViewIllustPage.xaml b/Pixiview/Illust/ViewIllustPage.xaml index 0bcae70..7420cbf 100644 --- a/Pixiview/Illust/ViewIllustPage.xaml +++ b/Pixiview/Illust/ViewIllustPage.xaml @@ -3,17 +3,15 @@ xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml" xmlns:mdl="clr-namespace:Pixiview.Illust" xmlns:u="clr-namespace:Pixiview.UI" - xmlns:util="clr-namespace:Pixiview.Utils" xmlns:ios="clr-namespace:Xamarin.Forms.PlatformConfiguration.iOSSpecific;assembly=Xamarin.Forms.Core" x:Class="Pixiview.Illust.ViewIllustPage" - util:Screen.StatusBarStyle="{DynamicResource StatusBarStyle}" ios:Page.UseSafeArea="False" Shell.TabBarIsVisible="False" BackgroundColor="{DynamicResource WindowColor}" Title="{Binding IllustItem.Title}"> - + (ImageSource)GetValue(IsFavoriteProperty); + public IllustDetailItem[] Illusts { get => (IllustDetailItem[])GetValue(IllustsProperty); @@ -50,23 +56,38 @@ namespace Pixiview.Illust public int CurrentPage { get; private set; } - public ViewIllustPage(IllustItem illust) + private readonly IIllustCollectionPage collectionPage; + private readonly object fontIconLove; + private readonly object fontIconNotLove; + + public ViewIllustPage(IllustItem illust, IIllustCollectionPage page) { IllustItem = illust; + collectionPage = page; BindingContext = this; - InitializeComponent(); - } + fontIconLove = Application.Current.Resources[ThemeBase.FontIconLove]; + fontIconNotLove = Application.Current.Resources[ThemeBase.FontIconNotLove]; + if (page.Favorites != null) + { + SetValue(IsFavoriteProperty, page.Favorites.Any(i => i.Id == illust.Id) + ? fontIconLove + : fontIconNotLove); + } + + InitializeComponent(); - public override void OnLoad() - { - var illust = IllustItem; if (illust != null) { LoadIllust(illust); } } + public override void OnLoad() + { + OnOrientationChanged(CurrentOrientation); + } + private void LoadIllust(IllustItem illust) { if (illust == null) @@ -95,7 +116,6 @@ namespace Pixiview.Illust } Illusts = items; - OnOrientationChanged(CurrentOrientation); Task.Run(DoLoadImages); } @@ -134,6 +154,13 @@ namespace Pixiview.Illust protected override void OnDisappearing() { base.OnDisappearing(); + + var favorite = new IllustFavorite + { + LastFavoriteUtc = DateTime.UtcNow, + Illusts = collectionPage.Favorites.ToArray() + }; + Stores.SaveFavoritesIllusts(favorite); Screen.SetHomeIndicatorAutoHidden(Shell.Current, false); } @@ -220,6 +247,25 @@ namespace Pixiview.Illust item.Loading = false; } + private void Favorite_Clicked(object sender, EventArgs e) + { + if (collectionPage.Favorites == null) + { + return; + } + var index = collectionPage.Favorites.FindIndex(i => i.Id == IllustItem.Id); + if (index < 0) + { + collectionPage.Favorites.Insert(0, IllustItem); + SetValue(IsFavoriteProperty, fontIconLove); + } + else + { + collectionPage.Favorites.RemoveAt(index); + SetValue(IsFavoriteProperty, fontIconNotLove); + } + } + private async void Download_Clicked(object sender, EventArgs e) { var status = await Permissions.CheckStatusAsync(); diff --git a/Pixiview/OptionPage.xaml b/Pixiview/OptionPage.xaml index 03bdddf..2bf334a 100644 --- a/Pixiview/OptionPage.xaml +++ b/Pixiview/OptionPage.xaml @@ -2,10 +2,10 @@ + BackgroundColor="{DynamicResource WindowColor}" + Title="{r:Text Option}"> diff --git a/Pixiview/Resources/Languages/zh-CN.xml b/Pixiview/Resources/Languages/zh-CN.xml index 359d63e..1ae7584 100644 --- a/Pixiview/Resources/Languages/zh-CN.xml +++ b/Pixiview/Resources/Languages/zh-CN.xml @@ -7,6 +7,7 @@ 推荐 按用户 排行榜 + 收藏夹 预览 成功保存图片到照片库。 diff --git a/Pixiview/UI/AdaptedPage.cs b/Pixiview/UI/AdaptedPage.cs index f3f5411..74f3c0a 100644 --- a/Pixiview/UI/AdaptedPage.cs +++ b/Pixiview/UI/AdaptedPage.cs @@ -1,4 +1,6 @@ using System; +using Pixiview.UI.Theme; +using Pixiview.Utils; using Xamarin.Forms; namespace Pixiview.UI @@ -13,6 +15,7 @@ namespace Pixiview.UI public AdaptedPage() { + SetDynamicResource(Screen.StatusBarStyleProperty, ThemeBase.StatusBarStyle); Shell.SetNavBarHasShadow(this, true); } diff --git a/Pixiview/UI/StyleDefinition.cs b/Pixiview/UI/StyleDefinition.cs index 76d939f..393cded 100644 --- a/Pixiview/UI/StyleDefinition.cs +++ b/Pixiview/UI/StyleDefinition.cs @@ -25,8 +25,10 @@ namespace Pixiview.UI public const string IconOrder = "\uf88f"; public const string IconLayer = "\uf302"; public const string IconRefresh = "\uf2f1"; + public const string IconLove = "\uf004"; public const string IconOption = "\uf013"; public const string IconDownload = "\uf019"; + public const string IconFavorite = "\uf02e"; static StyleDefinition() { diff --git a/Pixiview/UI/Theme/ThemeBase.cs b/Pixiview/UI/Theme/ThemeBase.cs index 6ad95ec..b675f9e 100644 --- a/Pixiview/UI/Theme/ThemeBase.cs +++ b/Pixiview/UI/Theme/ThemeBase.cs @@ -10,8 +10,11 @@ namespace Pixiview.UI.Theme public const string FontIconSparkles = nameof(FontIconSparkles); public const string FontIconOrder = nameof(FontIconOrder); public const string FontIconRefresh = nameof(FontIconRefresh); + public const string FontIconLove = nameof(FontIconLove); + public const string FontIconNotLove = nameof(FontIconNotLove); public const string FontIconOption = nameof(FontIconOption); public const string FontIconDownload = nameof(FontIconDownload); + public const string FontIconFavorite = nameof(FontIconFavorite); public const string StatusBarStyle = nameof(StatusBarStyle); public const string WindowColor = nameof(WindowColor); @@ -87,8 +90,11 @@ namespace Pixiview.UI.Theme Add(FontIconSparkles, GetSolidIcon(StyleDefinition.IconSparkles, solidFontFamily, mainColor)); Add(FontIconOrder, GetSolidIcon(StyleDefinition.IconOrder, solidFontFamily, mainColor)); Add(FontIconRefresh, GetSolidIcon(StyleDefinition.IconRefresh, solidFontFamily, mainColor)); + Add(FontIconLove, GetSolidIcon(StyleDefinition.IconLove, solidFontFamily, mainColor)); + Add(FontIconNotLove, GetSolidIcon(StyleDefinition.IconLove, (string)this[IconRegularFontFamily], mainColor)); Add(FontIconOption, GetSolidIcon(StyleDefinition.IconOption, solidFontFamily, mainColor)); Add(FontIconDownload, GetSolidIcon(StyleDefinition.IconDownload, solidFontFamily, mainColor)); + Add(FontIconFavorite, GetSolidIcon(StyleDefinition.IconFavorite, solidFontFamily, mainColor)); } private FontImageSource GetSolidIcon(string icon, string family, Color color) diff --git a/Pixiview/Utils/IllustData.cs b/Pixiview/Utils/IllustData.cs index 2442de9..8195602 100644 --- a/Pixiview/Utils/IllustData.cs +++ b/Pixiview/Utils/IllustData.cs @@ -1,4 +1,5 @@ using Newtonsoft.Json; +using Pixiview.Illust; namespace Pixiview.Utils { @@ -89,6 +90,23 @@ namespace Pixiview.Utils [JsonProperty("540x540")] public string x540; } + + public IllustItem ConvertToItem() + { + return new IllustItem + { + Id = illustId, + ImageUrl = urls.x360 ?? url, + Title = illustTitle, + IsRestrict = xRestrict == 1, + ProfileUrl = profileImageUrl, + UserId = userId, + UserName = userName, + Width = width, + Height = height, + PageCount = pageCount + }; + } } } diff --git a/Pixiview/Utils/Stores.cs b/Pixiview/Utils/Stores.cs index 97e1d0b..cf3f1ba 100644 --- a/Pixiview/Utils/Stores.cs +++ b/Pixiview/Utils/Stores.cs @@ -5,6 +5,7 @@ using System.Net; using System.Net.Http; using System.Text; using Newtonsoft.Json; +using Pixiview.Illust; using Xamarin.Essentials; using Xamarin.Forms; @@ -14,12 +15,17 @@ namespace Pixiview.Utils { public static readonly string PersonalFolder = Environment.GetFolderPath(Environment.SpecialFolder.Personal); public static readonly string CacheFolder = Environment.GetFolderPath(Environment.SpecialFolder.InternetCache); + private const string pagesFolder = "pages"; private const string imageFolder = "img-original"; private const string previewFolder = "img-master"; private const string thumbFolder = "img-thumb"; private const string userFolder = "user-profile"; + private const string illustFile = "illust.json"; + private const string favoriteFile = "favorites.json"; + + private static readonly object sync = new object(); public static bool NetworkAvailable { @@ -87,6 +93,64 @@ namespace Pixiview.Utils } } + private static T ReadObject(string file) + { + string content = null; + if (File.Exists(file)) + { + try + { + content = File.ReadAllText(file); + } + catch (Exception ex) + { + App.DebugError("read", $"failed to read file: {file}, error: {ex.Message}"); + } + } + else + { + App.DebugError("read", $"file not found: {file}"); + return default; + } + try + { + return JsonConvert.DeserializeObject(content); + } + catch (Exception ex) + { + App.DebugError("read", $"failed to parse illust JSON object, error: {ex.Message}"); + return default; + } + } + + private static void WriteObject(string file, object obj) + { + var dir = Path.GetDirectoryName(file); + if (!Directory.Exists(dir)) + { + Directory.CreateDirectory(dir); + } + string content; + try + { + content = JsonConvert.SerializeObject(obj, Formatting.None); + } + catch (Exception ex) + { + App.DebugError("write", $"failed to serialize object, error: {ex.Message}"); + return; + } + + try + { + File.WriteAllText(file, content, Encoding.UTF8); + } + catch (Exception ex) + { + App.DebugError("write", $"failed to write file: {file}, error: {ex.Message}"); + } + } + public static IllustData LoadIllustData(bool force = false) { var file = Path.Combine(PersonalFolder, illustFile); @@ -118,6 +182,21 @@ namespace Pixiview.Utils return result; } + public static IllustFavorite LoadFavoritesIllusts() + { + var file = Path.Combine(PersonalFolder, favoriteFile); + return ReadObject(file); + } + + public static void SaveFavoritesIllusts(IllustFavorite data) + { + var file = Path.Combine(PersonalFolder, favoriteFile); + lock (sync) + { + WriteObject(file, data); + } + } + public static ImageSource LoadIllustImage(string url) { return LoadImage(url, PersonalFolder, imageFolder); @@ -306,6 +385,7 @@ namespace Pixiview.Utils public const string Recommends = "recommends"; public const string ByUser = "byuser"; public const string Ranking = "ranking"; + public const string Favorites = "favorites"; public const string Option = "option"; } }