Series.cs 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. using MediaBrowser.Controller.Localization;
  2. using MediaBrowser.Controller.Providers;
  3. using MediaBrowser.Model.Configuration;
  4. using MediaBrowser.Model.Entities;
  5. using MediaBrowser.Model.Querying;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Runtime.Serialization;
  10. namespace MediaBrowser.Controller.Entities.TV
  11. {
  12. /// <summary>
  13. /// Class Series
  14. /// </summary>
  15. public class Series : Folder, IHasSoundtracks, IHasTrailers, IHasPreferredMetadataLanguage, IHasDisplayOrder, IHasLookupInfo<SeriesInfo>, IHasSpecialFeatures
  16. {
  17. public List<Guid> SpecialFeatureIds { get; set; }
  18. public List<Guid> SoundtrackIds { get; set; }
  19. public int SeasonCount { get; set; }
  20. public int? AnimeSeriesIndex { get; set; }
  21. /// <summary>
  22. /// Gets or sets the preferred metadata country code.
  23. /// </summary>
  24. /// <value>The preferred metadata country code.</value>
  25. public string PreferredMetadataCountryCode { get; set; }
  26. public Series()
  27. {
  28. AirDays = new List<DayOfWeek>();
  29. SpecialFeatureIds = new List<Guid>();
  30. SoundtrackIds = new List<Guid>();
  31. RemoteTrailers = new List<MediaUrl>();
  32. LocalTrailerIds = new List<Guid>();
  33. RemoteTrailerIds = new List<Guid>();
  34. DisplaySpecialsWithSeasons = true;
  35. }
  36. [IgnoreDataMember]
  37. public override bool SupportsAddingToPlaylist
  38. {
  39. get { return true; }
  40. }
  41. [IgnoreDataMember]
  42. public override bool IsPreSorted
  43. {
  44. get
  45. {
  46. return true;
  47. }
  48. }
  49. public bool DisplaySpecialsWithSeasons { get; set; }
  50. public List<Guid> LocalTrailerIds { get; set; }
  51. public List<Guid> RemoteTrailerIds { get; set; }
  52. public List<MediaUrl> RemoteTrailers { get; set; }
  53. /// <summary>
  54. /// airdate, dvd or absolute
  55. /// </summary>
  56. public string DisplayOrder { get; set; }
  57. /// <summary>
  58. /// Gets or sets the status.
  59. /// </summary>
  60. /// <value>The status.</value>
  61. public SeriesStatus? Status { get; set; }
  62. /// <summary>
  63. /// Gets or sets the air days.
  64. /// </summary>
  65. /// <value>The air days.</value>
  66. public List<DayOfWeek> AirDays { get; set; }
  67. /// <summary>
  68. /// Gets or sets the air time.
  69. /// </summary>
  70. /// <value>The air time.</value>
  71. public string AirTime { get; set; }
  72. /// <summary>
  73. /// Gets or sets the date last episode added.
  74. /// </summary>
  75. /// <value>The date last episode added.</value>
  76. public DateTime DateLastEpisodeAdded { get; set; }
  77. /// <summary>
  78. /// Series aren't included directly in indices - Their Episodes will roll up to them
  79. /// </summary>
  80. /// <value><c>true</c> if [include in index]; otherwise, <c>false</c>.</value>
  81. [IgnoreDataMember]
  82. public override bool IncludeInIndex
  83. {
  84. get
  85. {
  86. return false;
  87. }
  88. }
  89. /// <summary>
  90. /// Gets the user data key.
  91. /// </summary>
  92. /// <returns>System.String.</returns>
  93. public override string GetUserDataKey()
  94. {
  95. return this.GetProviderId(MetadataProviders.Tvdb) ?? this.GetProviderId(MetadataProviders.Tvcom) ?? base.GetUserDataKey();
  96. }
  97. /// <summary>
  98. /// Gets the trailer ids.
  99. /// </summary>
  100. /// <returns>List&lt;Guid&gt;.</returns>
  101. public List<Guid> GetTrailerIds()
  102. {
  103. var list = LocalTrailerIds.ToList();
  104. list.AddRange(RemoteTrailerIds);
  105. return list;
  106. }
  107. // Studio, Genre and Rating will all be the same so makes no sense to index by these
  108. protected override IEnumerable<string> GetIndexByOptions()
  109. {
  110. return new List<string> {
  111. {LocalizedStrings.Instance.GetString("NoneDispPref")},
  112. {LocalizedStrings.Instance.GetString("PerformerDispPref")},
  113. {LocalizedStrings.Instance.GetString("DirectorDispPref")},
  114. {LocalizedStrings.Instance.GetString("YearDispPref")},
  115. };
  116. }
  117. [IgnoreDataMember]
  118. public bool ContainsEpisodesWithoutSeasonFolders
  119. {
  120. get
  121. {
  122. return Children.OfType<Video>().Any();
  123. }
  124. }
  125. public override IEnumerable<BaseItem> GetChildren(User user, bool includeLinkedChildren)
  126. {
  127. return GetSeasons(user);
  128. }
  129. public IEnumerable<Season> GetSeasons(User user)
  130. {
  131. var config = user.Configuration;
  132. return GetSeasons(user, config.DisplayMissingEpisodes, config.DisplayUnairedEpisodes);
  133. }
  134. public IEnumerable<Season> GetSeasons(User user, bool includeMissingSeasons, bool includeVirtualUnaired)
  135. {
  136. var seasons = base.GetChildren(user, true)
  137. .OfType<Season>();
  138. if (!includeMissingSeasons && !includeVirtualUnaired)
  139. {
  140. seasons = seasons.Where(i => !i.IsMissingOrVirtualUnaired);
  141. }
  142. else
  143. {
  144. if (!includeMissingSeasons)
  145. {
  146. seasons = seasons.Where(i => !i.IsMissingSeason);
  147. }
  148. if (!includeVirtualUnaired)
  149. {
  150. seasons = seasons.Where(i => !i.IsVirtualUnaired);
  151. }
  152. }
  153. return LibraryManager
  154. .Sort(seasons, user, new[] { ItemSortBy.SortName }, SortOrder.Ascending)
  155. .Cast<Season>();
  156. }
  157. public IEnumerable<Episode> GetEpisodes(User user, int seasonNumber)
  158. {
  159. var config = user.Configuration;
  160. return GetEpisodes(user, seasonNumber, config.DisplayMissingEpisodes, config.DisplayUnairedEpisodes);
  161. }
  162. public IEnumerable<Episode> GetEpisodes(User user, int seasonNumber, bool includeMissingEpisodes, bool includeVirtualUnairedEpisodes)
  163. {
  164. return GetEpisodes(user, seasonNumber, includeMissingEpisodes, includeVirtualUnairedEpisodes,
  165. new List<Episode>());
  166. }
  167. internal IEnumerable<Episode> GetEpisodes(User user, int seasonNumber, bool includeMissingEpisodes, bool includeVirtualUnairedEpisodes, IEnumerable<Episode> additionalEpisodes)
  168. {
  169. var episodes = GetRecursiveChildren(user)
  170. .OfType<Episode>();
  171. episodes = FilterEpisodesBySeason(episodes, seasonNumber, DisplaySpecialsWithSeasons);
  172. episodes = episodes.Concat(additionalEpisodes).Distinct();
  173. if (!includeMissingEpisodes)
  174. {
  175. episodes = episodes.Where(i => !i.IsMissingEpisode);
  176. }
  177. if (!includeVirtualUnairedEpisodes)
  178. {
  179. episodes = episodes.Where(i => !i.IsVirtualUnaired);
  180. }
  181. var sortBy = seasonNumber == 0 ? ItemSortBy.SortName : ItemSortBy.AiredEpisodeOrder;
  182. return LibraryManager.Sort(episodes, user, new[] { sortBy }, SortOrder.Ascending)
  183. .Cast<Episode>();
  184. }
  185. /// <summary>
  186. /// Filters the episodes by season.
  187. /// </summary>
  188. /// <param name="episodes">The episodes.</param>
  189. /// <param name="seasonNumber">The season number.</param>
  190. /// <param name="includeSpecials">if set to <c>true</c> [include specials].</param>
  191. /// <returns>IEnumerable{Episode}.</returns>
  192. public static IEnumerable<Episode> FilterEpisodesBySeason(IEnumerable<Episode> episodes, int seasonNumber, bool includeSpecials)
  193. {
  194. if (!includeSpecials || seasonNumber < 1)
  195. {
  196. return episodes.Where(i => (i.PhysicalSeasonNumber ?? -1) == seasonNumber);
  197. }
  198. return episodes.Where(i =>
  199. {
  200. var episode = i;
  201. if (episode != null)
  202. {
  203. var currentSeasonNumber = episode.AiredSeasonNumber;
  204. return currentSeasonNumber.HasValue && currentSeasonNumber.Value == seasonNumber;
  205. }
  206. return false;
  207. });
  208. }
  209. protected override bool GetBlockUnratedValue(UserConfiguration config)
  210. {
  211. return config.BlockUnratedItems.Contains(UnratedItem.Series);
  212. }
  213. public string PreferredMetadataLanguage { get; set; }
  214. public SeriesInfo GetLookupInfo()
  215. {
  216. var info = GetItemLookupInfo<SeriesInfo>();
  217. info.AnimeSeriesIndex = AnimeSeriesIndex;
  218. return info;
  219. }
  220. public override bool BeforeMetadataRefresh()
  221. {
  222. var hasChanges = base.BeforeMetadataRefresh();
  223. if (!ProductionYear.HasValue)
  224. {
  225. var info = LibraryManager.ParseName(Name);
  226. var yearInName = info.Year;
  227. if (yearInName.HasValue)
  228. {
  229. ProductionYear = yearInName;
  230. hasChanges = true;
  231. }
  232. }
  233. return hasChanges;
  234. }
  235. }
  236. }