using MediaBrowser.Controller.Providers; using MediaBrowser.Model.Configuration; using MediaBrowser.Model.Entities; using System; using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; using MediaBrowser.Model.IO; using MediaBrowser.Model.Providers; using MediaBrowser.Model.Serialization; namespace MediaBrowser.Controller.Entities.Movies { /// /// Class Movie /// public class Movie : Video, IHasSpecialFeatures, IHasTrailers, IHasLookupInfo, ISupportsBoxSetGrouping { public Guid[] SpecialFeatureIds { get; set; } public Movie() { SpecialFeatureIds = new Guid[] {}; RemoteTrailers = EmptyMediaUrlArray; LocalTrailerIds = new Guid[] {}; RemoteTrailerIds = new Guid[] {}; } public Guid[] LocalTrailerIds { get; set; } public Guid[] RemoteTrailerIds { get; set; } /// /// Gets or sets the name of the TMDB collection. /// /// The name of the TMDB collection. public string TmdbCollectionName { get; set; } [IgnoreDataMember] public string CollectionName { get { return TmdbCollectionName; } set { TmdbCollectionName = value; } } public override double GetDefaultPrimaryImageAspectRatio() { // hack for tv plugins if (SourceType == SourceType.Channel) { return 0; } return 2 / 3; } protected override async Task RefreshedOwnedItems(MetadataRefreshOptions options, List fileSystemChildren, CancellationToken cancellationToken) { var hasChanges = await base.RefreshedOwnedItems(options, fileSystemChildren, cancellationToken).ConfigureAwait(false); // Must have a parent to have special features // In other words, it must be part of the Parent/Child tree if (IsFileProtocol && SupportsOwnedItems && !IsInMixedFolder) { var specialFeaturesChanged = await RefreshSpecialFeatures(options, fileSystemChildren, cancellationToken).ConfigureAwait(false); if (specialFeaturesChanged) { hasChanges = true; } } return hasChanges; } private async Task RefreshSpecialFeatures(MetadataRefreshOptions options, List fileSystemChildren, CancellationToken cancellationToken) { var newItems = LibraryManager.FindExtras(this, fileSystemChildren, options.DirectoryService).ToList(); var newItemIds = newItems.Select(i => i.Id).ToArray(); var itemsChanged = !SpecialFeatureIds.SequenceEqual(newItemIds); var ownerId = Id; var tasks = newItems.Select(i => { var subOptions = new MetadataRefreshOptions(options); if (i.OwnerId != ownerId) { i.OwnerId = ownerId; subOptions.ForceSave = true; } return RefreshMetadataForOwnedItem(i, false, subOptions, cancellationToken); }); await Task.WhenAll(tasks).ConfigureAwait(false); SpecialFeatureIds = newItemIds; return itemsChanged; } public override UnratedItem GetBlockUnratedType() { return UnratedItem.Movie; } public MovieInfo GetLookupInfo() { var info = GetItemLookupInfo(); if (!IsInMixedFolder) { var name = System.IO.Path.GetFileName(ContainingFolderPath); if (VideoType == VideoType.VideoFile || VideoType == VideoType.Iso) { if (string.Equals(name, System.IO.Path.GetFileName(Path), StringComparison.OrdinalIgnoreCase)) { // if the folder has the file extension, strip it name = System.IO.Path.GetFileNameWithoutExtension(name); } } info.Name = name; } return info; } public override bool BeforeMetadataRefresh(bool replaceAllMetdata) { var hasChanges = base.BeforeMetadataRefresh(replaceAllMetdata); if (!ProductionYear.HasValue) { var info = LibraryManager.ParseName(Name); var yearInName = info.Year; if (yearInName.HasValue) { ProductionYear = yearInName; hasChanges = true; } else { // Try to get the year from the folder name if (!IsInMixedFolder) { info = LibraryManager.ParseName(System.IO.Path.GetFileName(ContainingFolderPath)); yearInName = info.Year; if (yearInName.HasValue) { ProductionYear = yearInName; hasChanges = true; } } } } return hasChanges; } public override List GetRelatedUrls() { var list = base.GetRelatedUrls(); var imdbId = this.GetProviderId(MetadataProviders.Imdb); if (!string.IsNullOrEmpty(imdbId)) { list.Add(new ExternalUrl { Name = "Trakt", Url = string.Format("https://trakt.tv/movies/{0}", imdbId) }); } return list; } [IgnoreDataMember] public override bool StopRefreshIfLocalMetadataFound { get { // Need people id's from internet metadata return false; } } } }