123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327 |
- #nullable disable
- #pragma warning disable CS1591
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using Jellyfin.Data.Entities;
- using Jellyfin.Data.Enums;
- using MediaBrowser.Controller.Configuration;
- using MediaBrowser.Controller.Dto;
- using MediaBrowser.Controller.Entities;
- using MediaBrowser.Controller.Library;
- using MediaBrowser.Controller.TV;
- using MediaBrowser.Model.Querying;
- using Episode = MediaBrowser.Controller.Entities.TV.Episode;
- using Series = MediaBrowser.Controller.Entities.TV.Series;
- namespace Emby.Server.Implementations.TV
- {
- public class TVSeriesManager : ITVSeriesManager
- {
- private readonly IUserManager _userManager;
- private readonly IUserDataManager _userDataManager;
- private readonly ILibraryManager _libraryManager;
- private readonly IServerConfigurationManager _configurationManager;
- public TVSeriesManager(IUserManager userManager, IUserDataManager userDataManager, ILibraryManager libraryManager, IServerConfigurationManager configurationManager)
- {
- _userManager = userManager;
- _userDataManager = userDataManager;
- _libraryManager = libraryManager;
- _configurationManager = configurationManager;
- }
- public QueryResult<BaseItem> GetNextUp(NextUpQuery query, DtoOptions options)
- {
- var user = _userManager.GetUserById(query.UserId);
- if (user is null)
- {
- throw new ArgumentException("User not found");
- }
- string presentationUniqueKey = null;
- if (query.SeriesId.HasValue && !query.SeriesId.Value.Equals(default))
- {
- if (_libraryManager.GetItemById(query.SeriesId.Value) is Series series)
- {
- presentationUniqueKey = GetUniqueSeriesKey(series);
- }
- }
- if (!string.IsNullOrEmpty(presentationUniqueKey))
- {
- return GetResult(GetNextUpEpisodes(query, user, new[] { presentationUniqueKey }, options), query);
- }
- BaseItem[] parents;
- if (query.ParentId.HasValue)
- {
- var parent = _libraryManager.GetItemById(query.ParentId.Value);
- if (parent is not null)
- {
- parents = new[] { parent };
- }
- else
- {
- parents = Array.Empty<BaseItem>();
- }
- }
- else
- {
- parents = _libraryManager.GetUserRootFolder().GetChildren(user, true)
- .Where(i => i is Folder)
- .Where(i => !user.GetPreferenceValues<Guid>(PreferenceKind.LatestItemExcludes).Contains(i.Id))
- .ToArray();
- }
- return GetNextUp(query, parents, options);
- }
- public QueryResult<BaseItem> GetNextUp(NextUpQuery request, BaseItem[] parentsFolders, DtoOptions options)
- {
- var user = _userManager.GetUserById(request.UserId);
- if (user is null)
- {
- throw new ArgumentException("User not found");
- }
- string presentationUniqueKey = null;
- int? limit = null;
- if (request.SeriesId.HasValue && !request.SeriesId.Value.Equals(default))
- {
- if (_libraryManager.GetItemById(request.SeriesId.Value) is Series series)
- {
- presentationUniqueKey = GetUniqueSeriesKey(series);
- limit = 1;
- }
- }
- if (!string.IsNullOrEmpty(presentationUniqueKey))
- {
- return GetResult(GetNextUpEpisodes(request, user, new[] { presentationUniqueKey }, options), request);
- }
- if (limit.HasValue)
- {
- limit = limit.Value + 10;
- }
- var items = _libraryManager
- .GetItemList(
- new InternalItemsQuery(user)
- {
- IncludeItemTypes = new[] { BaseItemKind.Episode },
- OrderBy = new[] { (ItemSortBy.DatePlayed, SortOrder.Descending) },
- SeriesPresentationUniqueKey = presentationUniqueKey,
- Limit = limit,
- DtoOptions = new DtoOptions { Fields = new[] { ItemFields.SeriesPresentationUniqueKey }, EnableImages = false },
- GroupBySeriesPresentationUniqueKey = true
- },
- parentsFolders.ToList())
- .Cast<Episode>()
- .Where(episode => !string.IsNullOrEmpty(episode.SeriesPresentationUniqueKey))
- .Select(GetUniqueSeriesKey)
- .ToList();
- // Avoid implicitly captured closure
- var episodes = GetNextUpEpisodes(request, user, items, options);
- return GetResult(episodes, request);
- }
- private IEnumerable<Episode> GetNextUpEpisodes(NextUpQuery request, User user, IReadOnlyList<string> seriesKeys, DtoOptions dtoOptions)
- {
- var allNextUp = seriesKeys.Select(i => GetNextUp(i, user, dtoOptions, false));
- if (request.EnableRewatching)
- {
- allNextUp = allNextUp.Concat(
- seriesKeys.Select(i => GetNextUp(i, user, dtoOptions, true)))
- .OrderByDescending(i => i.LastWatchedDate);
- }
- // If viewing all next up for all series, remove first episodes
- // But if that returns empty, keep those first episodes (avoid completely empty view)
- var alwaysEnableFirstEpisode = request.SeriesId.HasValue && !request.SeriesId.Value.Equals(default);
- var anyFound = false;
- return allNextUp
- .Where(i =>
- {
- if (request.DisableFirstEpisode)
- {
- return i.LastWatchedDate != DateTime.MinValue;
- }
- if (alwaysEnableFirstEpisode || (i.LastWatchedDate != DateTime.MinValue && i.LastWatchedDate.Date >= request.NextUpDateCutoff))
- {
- anyFound = true;
- return true;
- }
- return !anyFound && i.LastWatchedDate == DateTime.MinValue;
- })
- .Select(i => i.GetEpisodeFunction())
- .Where(i => i is not null);
- }
- private static string GetUniqueSeriesKey(Episode episode)
- {
- return episode.SeriesPresentationUniqueKey;
- }
- private static string GetUniqueSeriesKey(Series series)
- {
- return series.GetPresentationUniqueKey();
- }
- /// <summary>
- /// Gets the next up.
- /// </summary>
- /// <returns>Task{Episode}.</returns>
- private (DateTime LastWatchedDate, Func<Episode> GetEpisodeFunction) GetNextUp(string seriesKey, User user, DtoOptions dtoOptions, bool rewatching)
- {
- var lastQuery = new InternalItemsQuery(user)
- {
- AncestorWithPresentationUniqueKey = null,
- SeriesPresentationUniqueKey = seriesKey,
- IncludeItemTypes = new[] { BaseItemKind.Episode },
- IsPlayed = true,
- Limit = 1,
- ParentIndexNumberNotEquals = 0,
- DtoOptions = new DtoOptions
- {
- Fields = new[] { ItemFields.SortName },
- EnableImages = false
- }
- };
- // If rewatching is enabled, sort first by date played and then by season and episode numbers
- lastQuery.OrderBy = rewatching
- ? new[] { (ItemSortBy.DatePlayed, SortOrder.Descending), (ItemSortBy.ParentIndexNumber, SortOrder.Descending), (ItemSortBy.IndexNumber, SortOrder.Descending) }
- : new[] { (ItemSortBy.ParentIndexNumber, SortOrder.Descending), (ItemSortBy.IndexNumber, SortOrder.Descending) };
- var lastWatchedEpisode = _libraryManager.GetItemList(lastQuery).Cast<Episode>().FirstOrDefault();
- Episode GetEpisode()
- {
- var nextQuery = new InternalItemsQuery(user)
- {
- AncestorWithPresentationUniqueKey = null,
- SeriesPresentationUniqueKey = seriesKey,
- IncludeItemTypes = new[] { BaseItemKind.Episode },
- OrderBy = new[] { (ItemSortBy.ParentIndexNumber, SortOrder.Ascending), (ItemSortBy.IndexNumber, SortOrder.Ascending) },
- Limit = 1,
- IsPlayed = rewatching,
- IsVirtualItem = false,
- ParentIndexNumberNotEquals = 0,
- DtoOptions = dtoOptions
- };
- // Locate the next up episode based on the last watched episode's season and episode number
- var lastWatchedParentIndexNumber = lastWatchedEpisode?.ParentIndexNumber;
- var lastWatchedIndexNumber = lastWatchedEpisode?.IndexNumberEnd ?? lastWatchedEpisode?.IndexNumber;
- if (lastWatchedParentIndexNumber.HasValue && lastWatchedIndexNumber.HasValue)
- {
- nextQuery.MinParentAndIndexNumber = (lastWatchedParentIndexNumber.Value, lastWatchedIndexNumber.Value + 1);
- }
- var nextEpisode = _libraryManager.GetItemList(nextQuery).Cast<Episode>().FirstOrDefault();
- if (_configurationManager.Configuration.DisplaySpecialsWithinSeasons)
- {
- var consideredEpisodes = _libraryManager.GetItemList(new InternalItemsQuery(user)
- {
- AncestorWithPresentationUniqueKey = null,
- SeriesPresentationUniqueKey = seriesKey,
- ParentIndexNumber = 0,
- IncludeItemTypes = new[] { BaseItemKind.Episode },
- IsPlayed = rewatching,
- IsVirtualItem = false,
- DtoOptions = dtoOptions
- })
- .Cast<Episode>()
- .Where(episode => episode.AirsBeforeSeasonNumber is not null || episode.AirsAfterSeasonNumber is not null)
- .ToList();
- if (lastWatchedEpisode is not null)
- {
- // Last watched episode is added, because there could be specials that aired before the last watched episode
- consideredEpisodes.Add(lastWatchedEpisode);
- }
- if (nextEpisode is not null)
- {
- consideredEpisodes.Add(nextEpisode);
- }
- var sortedConsideredEpisodes = _libraryManager.Sort(consideredEpisodes, user, new[] { (ItemSortBy.AiredEpisodeOrder, SortOrder.Ascending) })
- .Cast<Episode>();
- if (lastWatchedEpisode is not null)
- {
- sortedConsideredEpisodes = sortedConsideredEpisodes.SkipWhile(episode => !episode.Id.Equals(lastWatchedEpisode.Id)).Skip(1);
- }
- nextEpisode = sortedConsideredEpisodes.FirstOrDefault();
- }
- if (nextEpisode is not null)
- {
- var userData = _userDataManager.GetUserData(user, nextEpisode);
- if (userData.PlaybackPositionTicks > 0)
- {
- return null;
- }
- }
- return nextEpisode;
- }
- if (lastWatchedEpisode is not null)
- {
- var userData = _userDataManager.GetUserData(user, lastWatchedEpisode);
- var lastWatchedDate = userData.LastPlayedDate ?? DateTime.MinValue.AddDays(1);
- return (lastWatchedDate, GetEpisode);
- }
- // Return the first episode
- return (DateTime.MinValue, GetEpisode);
- }
- private static QueryResult<BaseItem> GetResult(IEnumerable<BaseItem> items, NextUpQuery query)
- {
- int totalCount = 0;
- if (query.EnableTotalRecordCount)
- {
- var list = items.ToList();
- totalCount = list.Count;
- items = list;
- }
- if (query.StartIndex.HasValue)
- {
- items = items.Skip(query.StartIndex.Value);
- }
- if (query.Limit.HasValue)
- {
- items = items.Take(query.Limit.Value);
- }
- return new QueryResult<BaseItem>(
- query.StartIndex,
- totalCount,
- items.ToArray());
- }
- }
- }
|