using MediaBrowser.Controller.Providers; using MediaBrowser.Model.Configuration; using MediaBrowser.Model.Entities; using System; using System.Collections.Generic; using System.Globalization; using System.Linq; using MediaBrowser.Model.IO; using MediaBrowser.Model.Serialization; using Microsoft.Extensions.Logging; namespace MediaBrowser.Controller.Entities.TV { /// /// Class Episode /// public class Episode : Video, IHasTrailers, IHasLookupInfo, IHasSeries { public Episode() { RemoteTrailers = EmptyMediaUrlArray; LocalTrailerIds = new Guid[] {}; RemoteTrailerIds = new Guid[] {}; } public Guid[] LocalTrailerIds { get; set; } public Guid[] RemoteTrailerIds { get; set; } public MediaUrl[] RemoteTrailers { get; set; } /// /// Gets the season in which it aired. /// /// The aired season. public int? AirsBeforeSeasonNumber { get; set; } public int? AirsAfterSeasonNumber { get; set; } public int? AirsBeforeEpisodeNumber { get; set; } /// /// This is the ending episode number for double episodes. /// /// The index number. public int? IndexNumberEnd { get; set; } public string FindSeriesSortName() { var series = Series; return series == null ? SeriesName : series.SortName; } [IgnoreDataMember] protected override bool SupportsOwnedItems { get { return IsStacked || MediaSourceCount > 1; } } [IgnoreDataMember] public override bool SupportsInheritedParentImages { get { return true; } } [IgnoreDataMember] public override bool SupportsPeople { get { return true; } } [IgnoreDataMember] public int? AiredSeasonNumber { get { return AirsAfterSeasonNumber ?? AirsBeforeSeasonNumber ?? ParentIndexNumber; } } [IgnoreDataMember] public override Folder LatestItemsIndexContainer { get { return Series; } } [IgnoreDataMember] public override Guid DisplayParentId { get { return SeasonId; } } [IgnoreDataMember] protected override bool EnableDefaultVideoUserDataKeys { get { return false; } } public override double GetDefaultPrimaryImageAspectRatio() { // hack for tv plugins if (SourceType == SourceType.Channel) { return 0; } double value = 16; value /= 9; return value; } public override List GetUserDataKeys() { var list = base.GetUserDataKeys(); var series = Series; if (series != null && ParentIndexNumber.HasValue && IndexNumber.HasValue) { var seriesUserDataKeys = series.GetUserDataKeys(); var take = seriesUserDataKeys.Count; if (seriesUserDataKeys.Count > 1) { take--; } list.InsertRange(0, seriesUserDataKeys.Take(take).Select(i => i + ParentIndexNumber.Value.ToString("000") + IndexNumber.Value.ToString("000"))); } return list; } /// /// This Episode's Series Instance /// /// The series. [IgnoreDataMember] public Series Series { get { var seriesId = SeriesId; if (seriesId.Equals(Guid.Empty)) { seriesId = FindSeriesId(); } return !seriesId.Equals(Guid.Empty) ? (LibraryManager.GetItemById(seriesId) as Series) : null; } } [IgnoreDataMember] public Season Season { get { var seasonId = SeasonId; if (seasonId.Equals(Guid.Empty)) { seasonId = FindSeasonId(); } return !seasonId.Equals(Guid.Empty) ? (LibraryManager.GetItemById(seasonId) as Season) : null; } } [IgnoreDataMember] public bool IsInSeasonFolder { get { return FindParent() != null; } } [IgnoreDataMember] public string SeriesPresentationUniqueKey { get; set; } [IgnoreDataMember] public string SeriesName { get; set; } [IgnoreDataMember] public string SeasonName { get; set; } public string FindSeriesPresentationUniqueKey() { var series = Series; return series == null ? null : series.PresentationUniqueKey; } public string FindSeasonName() { var season = Season; if (season == null) { if (ParentIndexNumber.HasValue) { return "Season " + ParentIndexNumber.Value.ToString(CultureInfo.InvariantCulture); } return "Season Unknown"; } return season.Name; } public string FindSeriesName() { var series = Series; return series == null ? SeriesName : series.Name; } public Guid FindSeasonId() { var season = FindParent(); // Episodes directly in series folder if (season == null) { var series = Series; if (series != null && ParentIndexNumber.HasValue) { var findNumber = ParentIndexNumber.Value; season = series.Children .OfType() .FirstOrDefault(i => i.IndexNumber.HasValue && i.IndexNumber.Value == findNumber); } } return season == null ? Guid.Empty : season.Id; } /// /// Creates the name of the sort. /// /// System.String. protected override string CreateSortName() { return (ParentIndexNumber != null ? ParentIndexNumber.Value.ToString("000 - ") : "") + (IndexNumber != null ? IndexNumber.Value.ToString("0000 - ") : "") + Name; } /// /// Determines whether [contains episode number] [the specified number]. /// /// The number. /// true if [contains episode number] [the specified number]; otherwise, false. public bool ContainsEpisodeNumber(int number) { if (IndexNumber.HasValue) { if (IndexNumberEnd.HasValue) { return number >= IndexNumber.Value && number <= IndexNumberEnd.Value; } return IndexNumber.Value == number; } return false; } [IgnoreDataMember] public override bool SupportsRemoteImageDownloading { get { if (IsMissingEpisode) { return false; } return true; } } [IgnoreDataMember] public bool IsMissingEpisode { get { return LocationType == LocationType.Virtual; } } [IgnoreDataMember] public Guid SeasonId { get; set; } [IgnoreDataMember] public Guid SeriesId { get; set; } public Guid FindSeriesId() { var series = FindParent(); return series == null ? Guid.Empty : series.Id; } public override IEnumerable GetAncestorIds() { var list = base.GetAncestorIds().ToList(); var seasonId = SeasonId; if (!seasonId.Equals(Guid.Empty) && !list.Contains(seasonId)) { list.Add(seasonId); } return list; } public override IEnumerable GetDeletePaths() { return new[] { new FileSystemMetadata { FullName = Path, IsDirectory = IsFolder } }.Concat(GetLocalMetadataFilesToDelete()); } public override UnratedItem GetBlockUnratedType() { return UnratedItem.Series; } public EpisodeInfo GetLookupInfo() { var id = GetItemLookupInfo(); var series = Series; if (series != null) { id.SeriesProviderIds = series.ProviderIds; id.SeriesDisplayOrder = series.DisplayOrder; } id.IsMissingEpisode = IsMissingEpisode; id.IndexNumberEnd = IndexNumberEnd; return id; } public override bool BeforeMetadataRefresh(bool replaceAllMetdata) { var hasChanges = base.BeforeMetadataRefresh(replaceAllMetdata); if (!IsLocked) { if (SourceType == SourceType.Library) { try { if (LibraryManager.FillMissingEpisodeNumbersFromPath(this, replaceAllMetdata)) { hasChanges = true; } } catch (Exception ex) { Logger.LogError(ex, "Error in FillMissingEpisodeNumbersFromPath. Episode: {Episode}", Path ?? Name ?? Id.ToString()); } } } return hasChanges; } } }