You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
170 lines
5.7 KiB
170 lines
5.7 KiB
using MediaBrowser.Controller.IO;
|
|
using MediaBrowser.Model.Entities;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Runtime.Serialization;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace MediaBrowser.Controller.Entities.Movies
|
|
{
|
|
/// <summary>
|
|
/// Class Movie
|
|
/// </summary>
|
|
public class Movie : Video
|
|
{
|
|
/// <summary>
|
|
/// Should be overridden to return the proper folder where metadata lives
|
|
/// </summary>
|
|
/// <value>The meta location.</value>
|
|
[IgnoreDataMember]
|
|
public override string MetaLocation
|
|
{
|
|
get
|
|
{
|
|
return VideoType == VideoType.VideoFile || VideoType == VideoType.Iso ? System.IO.Path.GetDirectoryName(Path) : Path;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the user data key.
|
|
/// </summary>
|
|
/// <returns>System.String.</returns>
|
|
public override string GetUserDataKey()
|
|
{
|
|
return this.GetProviderId(MetadataProviders.Tmdb) ?? this.GetProviderId(MetadataProviders.Imdb) ?? base.GetUserDataKey();
|
|
}
|
|
|
|
/// <summary>
|
|
/// The _special features
|
|
/// </summary>
|
|
private List<Video> _specialFeatures;
|
|
/// <summary>
|
|
/// The _special features initialized
|
|
/// </summary>
|
|
private bool _specialFeaturesInitialized;
|
|
/// <summary>
|
|
/// The _special features sync lock
|
|
/// </summary>
|
|
private object _specialFeaturesSyncLock = new object();
|
|
/// <summary>
|
|
/// Gets the special features.
|
|
/// </summary>
|
|
/// <value>The special features.</value>
|
|
[IgnoreDataMember]
|
|
public List<Video> SpecialFeatures
|
|
{
|
|
get
|
|
{
|
|
LazyInitializer.EnsureInitialized(ref _specialFeatures, ref _specialFeaturesInitialized, ref _specialFeaturesSyncLock, () => LoadSpecialFeatures().ToList());
|
|
return _specialFeatures;
|
|
}
|
|
private set
|
|
{
|
|
_specialFeatures = value;
|
|
|
|
if (value == null)
|
|
{
|
|
_specialFeaturesInitialized = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Needed because the resolver stops at the movie folder and we find the video inside.
|
|
/// </summary>
|
|
/// <value><c>true</c> if [use parent path to create resolve args]; otherwise, <c>false</c>.</value>
|
|
protected override bool UseParentPathToCreateResolveArgs
|
|
{
|
|
get
|
|
{
|
|
return VideoType == VideoType.VideoFile || VideoType == VideoType.Iso;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Overrides the base implementation to refresh metadata for special features
|
|
/// </summary>
|
|
/// <param name="cancellationToken">The cancellation token.</param>
|
|
/// <param name="forceSave">if set to <c>true</c> [is new item].</param>
|
|
/// <param name="forceRefresh">if set to <c>true</c> [force].</param>
|
|
/// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param>
|
|
/// <param name="resetResolveArgs">if set to <c>true</c> [reset resolve args].</param>
|
|
/// <returns>Task{System.Boolean}.</returns>
|
|
public override async Task<bool> RefreshMetadata(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true, bool resetResolveArgs = true)
|
|
{
|
|
// Lazy load these again
|
|
SpecialFeatures = null;
|
|
|
|
// Kick off a task to refresh the main item
|
|
var result = await base.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders, resetResolveArgs).ConfigureAwait(false);
|
|
|
|
var tasks = SpecialFeatures.Select(item => item.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders));
|
|
|
|
await Task.WhenAll(tasks).ConfigureAwait(false);
|
|
|
|
cancellationToken.ThrowIfCancellationRequested();
|
|
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Loads the special features.
|
|
/// </summary>
|
|
/// <returns>IEnumerable{Video}.</returns>
|
|
private IEnumerable<Video> LoadSpecialFeatures()
|
|
{
|
|
if (LocationType != LocationType.FileSystem)
|
|
{
|
|
return new List<Video>();
|
|
}
|
|
|
|
FileSystemInfo folder;
|
|
|
|
try
|
|
{
|
|
folder = ResolveArgs.GetFileSystemEntryByName("specials");
|
|
}
|
|
catch (IOException ex)
|
|
{
|
|
Logger.ErrorException("Error getting ResolveArgs for {0}", ex, Path);
|
|
return new List<Video>();
|
|
}
|
|
|
|
// Path doesn't exist. No biggie
|
|
if (folder == null)
|
|
{
|
|
return new List<Video>();
|
|
}
|
|
|
|
IEnumerable<FileSystemInfo> files;
|
|
|
|
try
|
|
{
|
|
files = new DirectoryInfo(folder.FullName).EnumerateFiles();
|
|
}
|
|
catch (IOException ex)
|
|
{
|
|
Logger.ErrorException("Error loading trailers for {0}", ex, Name);
|
|
return new List<Video>();
|
|
}
|
|
|
|
return LibraryManager.ResolvePaths<Video>(files, null).Select(video =>
|
|
{
|
|
// Try to retrieve it from the db. If we don't find it, use the resolved version
|
|
var dbItem = LibraryManager.RetrieveItem(video.Id) as Video;
|
|
|
|
if (dbItem != null)
|
|
{
|
|
dbItem.ResolveArgs = video.ResolveArgs;
|
|
video = dbItem;
|
|
}
|
|
|
|
return video;
|
|
});
|
|
}
|
|
|
|
}
|
|
}
|