aboutsummaryrefslogtreecommitdiff
path: root/MediaBrowser.Server.Implementations/TV
diff options
context:
space:
mode:
authorLuke Pulverenti <luke.pulverenti@gmail.com>2014-09-01 16:10:54 -0400
committerLuke Pulverenti <luke.pulverenti@gmail.com>2014-09-01 16:10:54 -0400
commit6f45ea08237eefde317088459c4a87669be981f4 (patch)
tree17eb19d11d72e0225d45323e79b0711ec30c1584 /MediaBrowser.Server.Implementations/TV
parent383b9999da15e8bc6421527c381c6aa80f14c97f (diff)
fixes #912 - Add special views for Dlna
Diffstat (limited to 'MediaBrowser.Server.Implementations/TV')
-rw-r--r--MediaBrowser.Server.Implementations/TV/TVSeriesManager.cs202
1 files changed, 202 insertions, 0 deletions
diff --git a/MediaBrowser.Server.Implementations/TV/TVSeriesManager.cs b/MediaBrowser.Server.Implementations/TV/TVSeriesManager.cs
new file mode 100644
index 000000000..e71a5d514
--- /dev/null
+++ b/MediaBrowser.Server.Implementations/TV/TVSeriesManager.cs
@@ -0,0 +1,202 @@
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.TV;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.TV;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Querying;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+
+namespace MediaBrowser.Server.Implementations.TV
+{
+ public class TVSeriesManager : ITVSeriesManager
+ {
+ private readonly IUserManager _userManager;
+ private readonly IUserDataManager _userDataManager;
+ private readonly ILibraryManager _libraryManager;
+
+ public TVSeriesManager(IUserManager userManager, IUserDataManager userDataManager, ILibraryManager libraryManager)
+ {
+ _userManager = userManager;
+ _userDataManager = userDataManager;
+ _libraryManager = libraryManager;
+ }
+
+ public QueryResult<BaseItem> GetNextUp(NextUpQuery request)
+ {
+ var user = _userManager.GetUserById(new Guid(request.UserId));
+
+ if (user == null)
+ {
+ throw new ArgumentException("User not found");
+ }
+
+ var parentIds = string.IsNullOrEmpty(request.ParentId)
+ ? new string[] { }
+ : new[] { request.ParentId };
+
+ var items = GetAllLibraryItems(user, parentIds)
+ .OfType<Series>();
+
+ // Avoid implicitly captured closure
+ var episodes = GetNextUpEpisodes(request, user, items);
+
+ return GetResult(episodes, null, request);
+ }
+
+ public QueryResult<BaseItem> GetNextUp(NextUpQuery request, IEnumerable<Folder> parentsFolders)
+ {
+ var user = _userManager.GetUserById(new Guid(request.UserId));
+
+ if (user == null)
+ {
+ throw new ArgumentException("User not found");
+ }
+
+ var items = parentsFolders.SelectMany(i => i.GetRecursiveChildren(user))
+ .OfType<Series>();
+
+ // Avoid implicitly captured closure
+ var episodes = GetNextUpEpisodes(request, user, items);
+
+ return GetResult(episodes, null, request);
+ }
+
+ private IEnumerable<BaseItem> GetAllLibraryItems(User user, string[] parentIds)
+ {
+ if (parentIds.Length > 0)
+ {
+ return parentIds.SelectMany(i =>
+ {
+ var folder = (Folder)_libraryManager.GetItemById(new Guid(i));
+
+ return folder.GetRecursiveChildren(user);
+
+ });
+ }
+
+ if (user == null)
+ {
+ throw new ArgumentException("User not found");
+ }
+
+ return user.RootFolder.GetRecursiveChildren(user);
+ }
+
+ public IEnumerable<Episode> GetNextUpEpisodes(NextUpQuery request, User user, IEnumerable<Series> series)
+ {
+ // Avoid implicitly captured closure
+ var currentUser = user;
+
+ return FilterSeries(request, series)
+ .AsParallel()
+ .Select(i => GetNextUp(i, currentUser))
+ .Where(i => i.Item1 != null)
+ .OrderByDescending(i =>
+ {
+ var episode = i.Item1;
+
+ var seriesUserData = _userDataManager.GetUserData(user.Id, episode.Series.GetUserDataKey());
+
+ if (seriesUserData.IsFavorite)
+ {
+ return 2;
+ }
+
+ if (seriesUserData.Likes.HasValue)
+ {
+ return seriesUserData.Likes.Value ? 1 : -1;
+ }
+
+ return 0;
+ })
+ .ThenByDescending(i => i.Item2)
+ .ThenByDescending(i => i.Item1.PremiereDate ?? DateTime.MinValue)
+ .Select(i => i.Item1);
+ }
+
+ /// <summary>
+ /// Gets the next up.
+ /// </summary>
+ /// <param name="series">The series.</param>
+ /// <param name="user">The user.</param>
+ /// <returns>Task{Episode}.</returns>
+ private Tuple<Episode, DateTime> GetNextUp(Series series, User user)
+ {
+ // Get them in display order, then reverse
+ var allEpisodes = series.GetSeasons(user, true, true)
+ .SelectMany(i => i.GetEpisodes(user, true, true))
+ .Reverse()
+ .ToList();
+
+ Episode lastWatched = null;
+ var lastWatchedDate = DateTime.MinValue;
+ Episode nextUp = null;
+
+ // Go back starting with the most recent episodes
+ foreach (var episode in allEpisodes)
+ {
+ var userData = _userDataManager.GetUserData(user.Id, episode.GetUserDataKey());
+
+ if (userData.Played)
+ {
+ if (lastWatched != null || nextUp == null)
+ {
+ break;
+ }
+
+ lastWatched = episode;
+ lastWatchedDate = userData.LastPlayedDate ?? DateTime.MinValue;
+ }
+ else
+ {
+ if (episode.LocationType != LocationType.Virtual)
+ {
+ nextUp = episode;
+ }
+ }
+ }
+
+ if (lastWatched != null)
+ {
+ return new Tuple<Episode, DateTime>(nextUp, lastWatchedDate);
+ }
+
+ return new Tuple<Episode, DateTime>(null, lastWatchedDate);
+ }
+
+ private IEnumerable<Series> FilterSeries(NextUpQuery request, IEnumerable<Series> items)
+ {
+ if (!string.IsNullOrWhiteSpace(request.SeriesId))
+ {
+ var id = new Guid(request.SeriesId);
+
+ items = items.Where(i => i.Id == id);
+ }
+
+ return items;
+ }
+
+ private QueryResult<BaseItem> GetResult(IEnumerable<BaseItem> items, int? totalRecordLimit, NextUpQuery query)
+ {
+ var itemsArray = totalRecordLimit.HasValue ? items.Take(totalRecordLimit.Value).ToArray() : items.ToArray();
+ var totalCount = itemsArray.Length;
+
+ if (query.Limit.HasValue)
+ {
+ itemsArray = itemsArray.Skip(query.StartIndex ?? 0).Take(query.Limit.Value).ToArray();
+ }
+ else if (query.StartIndex.HasValue)
+ {
+ itemsArray = itemsArray.Skip(query.StartIndex.Value).ToArray();
+ }
+
+ return new QueryResult<BaseItem>
+ {
+ TotalRecordCount = totalCount,
+ Items = itemsArray
+ };
+ }
+ }
+}