| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580 | using MediaBrowser.Common.Configuration;using MediaBrowser.Common.Security;using MediaBrowser.Controller.Channels;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Movies;using MediaBrowser.Controller.Entities.TV;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.Localization;using MediaBrowser.Model.Channels;using MediaBrowser.Model.Configuration;using MediaBrowser.Model.Entities;using System;using System.Collections.Generic;using System.IO;using System.Linq;using System.Threading;using System.Threading.Tasks;using CommonIO;using MediaBrowser.Common.IO;using MoreLinq;namespace MediaBrowser.Server.Implementations.Intros{    public class DefaultIntroProvider : IIntroProvider    {        private readonly ISecurityManager _security;        private readonly IChannelManager _channelManager;        private readonly ILocalizationManager _localization;        private readonly IConfigurationManager _serverConfig;        private readonly ILibraryManager _libraryManager;        private readonly IFileSystem _fileSystem;        private readonly IMediaSourceManager _mediaSourceManager;        public DefaultIntroProvider(ISecurityManager security, IChannelManager channelManager, ILocalizationManager localization, IConfigurationManager serverConfig, ILibraryManager libraryManager, IFileSystem fileSystem, IMediaSourceManager mediaSourceManager)        {            _security = security;            _channelManager = channelManager;            _localization = localization;            _serverConfig = serverConfig;            _libraryManager = libraryManager;            _fileSystem = fileSystem;            _mediaSourceManager = mediaSourceManager;        }        public async Task<IEnumerable<IntroInfo>> GetIntros(BaseItem item, User user)        {            var config = GetOptions();            if (item is Movie)            {                if (!config.EnableIntrosForMovies)                {                    return new List<IntroInfo>();                }            }            else if (item is Episode)            {                if (!config.EnableIntrosForEpisodes)                {                    return new List<IntroInfo>();                }            }            else            {                return new List<IntroInfo>();            }            var ratingLevel = string.IsNullOrWhiteSpace(item.OfficialRating)                ? null                : _localization.GetRatingLevel(item.OfficialRating);            var random = new Random(Environment.TickCount + Guid.NewGuid().GetHashCode());            var candidates = new List<ItemWithTrailer>();            var itemPeople = _libraryManager.GetPeople(item);            var allPeople = _libraryManager.GetPeople(new InternalPeopleQuery            {                AppearsInItemId = item.Id            });            if (config.EnableIntrosFromMoviesInLibrary)            {                var inputItems = _libraryManager.GetItems(new InternalItemsQuery(user)                {                    IncludeItemTypes = new[] { typeof(Movie).Name }                }, new string[] { });                var itemsWithTrailers = inputItems                    .Where(i =>                    {                        var hasTrailers = i as IHasTrailers;                        if (hasTrailers != null && hasTrailers.LocalTrailerIds.Count > 0)                        {                            if (i is Movie)                            {                                return !IsDuplicate(item, i);                            }                        }                        return false;                    });                candidates.AddRange(itemsWithTrailers.Select(i => new ItemWithTrailer                {                    Item = i,                    Type = ItemWithTrailerType.ItemWithTrailer,                    User = user,                    WatchingItem = item,                    WatchingItemPeople = itemPeople,                    AllPeople = allPeople,                    Random = random,                    LibraryManager = _libraryManager                }));            }            var trailerTypes = new List<TrailerType>();            if (config.EnableIntrosFromUpcomingTrailers)            {                trailerTypes.Add(TrailerType.ComingSoonToTheaters);            }            if (config.EnableIntrosFromUpcomingDvdMovies)            {                trailerTypes.Add(TrailerType.ComingSoonToDvd);            }            if (config.EnableIntrosFromUpcomingStreamingMovies)            {                trailerTypes.Add(TrailerType.ComingSoonToStreaming);            }            if (config.EnableIntrosFromSimilarMovies)            {                trailerTypes.Add(TrailerType.Archive);            }            if (trailerTypes.Count > 0 && IsSupporter)            {                var channelTrailers = await _channelManager.GetAllMediaInternal(new AllChannelMediaQuery                {                    ContentTypes = new[] { ChannelMediaContentType.MovieExtra },                    ExtraTypes = new[] { ExtraType.Trailer },                    UserId = user.Id.ToString("N"),                    TrailerTypes = trailerTypes.ToArray()                }, CancellationToken.None);                candidates.AddRange(channelTrailers.Items.Select(i => new ItemWithTrailer                {                    Item = i,                    Type = ItemWithTrailerType.ChannelTrailer,                    User = user,                    WatchingItem = item,                    WatchingItemPeople = itemPeople,                    AllPeople = allPeople,                    Random = random,                    LibraryManager = _libraryManager                }));            }            return GetResult(item, candidates, config, ratingLevel);        }        private IEnumerable<IntroInfo> GetResult(BaseItem item, IEnumerable<ItemWithTrailer> candidates, CinemaModeConfiguration config, int? ratingLevel)        {            var customIntros = !string.IsNullOrWhiteSpace(config.CustomIntroPath) ?                GetCustomIntros(config) :                new List<IntroInfo>();            var mediaInfoIntros = !string.IsNullOrWhiteSpace(config.MediaInfoIntroPath) ?                GetMediaInfoIntros(config, item) :                new List<IntroInfo>();            var trailerLimit = config.TrailerLimit;            // Avoid implicitly captured closure            return candidates.Where(i =>            {                if (config.EnableIntrosParentalControl && !FilterByParentalRating(ratingLevel, i.Item))                {                    return false;                }                if (!config.EnableIntrosForWatchedContent && i.IsPlayed)                {                    return false;                }                return !IsDuplicate(item, i.Item);            })                .OrderByDescending(i => i.Score)                .ThenBy(i => Guid.NewGuid())                .ThenByDescending(i => (i.IsPlayed ? 0 : 1))                .Select(i => i.IntroInfo)                .Take(trailerLimit)                .Concat(customIntros.Take(1))                .Concat(mediaInfoIntros);        }        private bool IsDuplicate(BaseItem playingContent, BaseItem test)        {            var id = playingContent.GetProviderId(MetadataProviders.Imdb);            if (!string.IsNullOrWhiteSpace(id) && string.Equals(id, test.GetProviderId(MetadataProviders.Imdb), StringComparison.OrdinalIgnoreCase))            {                return true;            }            id = playingContent.GetProviderId(MetadataProviders.Tmdb);            if (!string.IsNullOrWhiteSpace(id) && string.Equals(id, test.GetProviderId(MetadataProviders.Tmdb), StringComparison.OrdinalIgnoreCase))            {                return true;            }            return false;        }        private CinemaModeConfiguration GetOptions()        {            return _serverConfig.GetConfiguration<CinemaModeConfiguration>("cinemamode");        }        private List<IntroInfo> GetCustomIntros(CinemaModeConfiguration options)        {            try            {                return GetCustomIntroFiles(options, true, false)                    .OrderBy(i => Guid.NewGuid())                    .Select(i => new IntroInfo                    {                        Path = i                    }).ToList();            }            catch (IOException)            {                return new List<IntroInfo>();            }        }        private IEnumerable<IntroInfo> GetMediaInfoIntros(CinemaModeConfiguration options, BaseItem item)        {            try            {                var hasMediaSources = item as IHasMediaSources;                if (hasMediaSources == null)                {                    return new List<IntroInfo>();                }                var mediaSource = _mediaSourceManager.GetStaticMediaSources(hasMediaSources, false)                    .FirstOrDefault();                if (mediaSource == null)                {                    return new List<IntroInfo>();                }                var videoStream = mediaSource.MediaStreams.FirstOrDefault(i => i.Type == MediaStreamType.Video);                var audioStream = mediaSource.MediaStreams.FirstOrDefault(i => i.Type == MediaStreamType.Audio);                var allIntros = GetCustomIntroFiles(options, false, true)                    .OrderBy(i => Guid.NewGuid())                    .Select(i => new IntroInfo                    {                        Path = i                    }).ToList();                var returnResult = new List<IntroInfo>();                if (videoStream != null)                {                    returnResult.AddRange(GetMediaInfoIntrosByVideoStream(allIntros, videoStream).Take(1));                }                if (audioStream != null)                {                    returnResult.AddRange(GetMediaInfoIntrosByAudioStream(allIntros, audioStream).Take(1));                }                returnResult.AddRange(GetMediaInfoIntrosByTags(allIntros, item.Tags).Take(1));                                return returnResult.DistinctBy(i => i.Path, StringComparer.OrdinalIgnoreCase);            }            catch (IOException)            {                return new List<IntroInfo>();            }        }        private IEnumerable<IntroInfo> GetMediaInfoIntrosByVideoStream(List<IntroInfo> allIntros, MediaStream stream)        {            var codec = stream.Codec;            if (string.IsNullOrWhiteSpace(codec))            {                return new List<IntroInfo>();            }            return allIntros                .Where(i => IsMatch(i.Path, codec));        }        private IEnumerable<IntroInfo> GetMediaInfoIntrosByAudioStream(List<IntroInfo> allIntros, MediaStream stream)        {            var codec = stream.Codec;            if (string.IsNullOrWhiteSpace(codec))            {                return new List<IntroInfo>();            }            return allIntros                .Where(i => IsAudioMatch(i.Path, stream));        }        private IEnumerable<IntroInfo> GetMediaInfoIntrosByTags(List<IntroInfo> allIntros, List<string> tags)        {            return allIntros                .Where(i => tags.Any(t => IsMatch(i.Path, t)));        }        private bool IsMatch(string file, string attribute)        {            var filename = Path.GetFileNameWithoutExtension(file) ?? string.Empty;            filename = Normalize(filename);            if (string.IsNullOrWhiteSpace(filename))            {                return false;            }            attribute = Normalize(attribute);            if (string.IsNullOrWhiteSpace(attribute))            {                return false;            }            return string.Equals(filename, attribute, StringComparison.OrdinalIgnoreCase);        }        private string Normalize(string value)        {            return value;        }        private bool IsAudioMatch(string path, MediaStream stream)        {            if (!string.IsNullOrWhiteSpace(stream.Codec))            {                if (IsMatch(path, stream.Codec))                {                    return true;                }            }            if (!string.IsNullOrWhiteSpace(stream.Profile))            {                if (IsMatch(path, stream.Profile))                {                    return true;                }            }            return false;        }        private IEnumerable<string> GetCustomIntroFiles(CinemaModeConfiguration options, bool enableCustomIntros, bool enableMediaInfoIntros)        {            var list = new List<string>();            if (enableCustomIntros && !string.IsNullOrWhiteSpace(options.CustomIntroPath))            {                list.AddRange(_fileSystem.GetFilePaths(options.CustomIntroPath, true)                    .Where(_libraryManager.IsVideoFile));            }            if (enableMediaInfoIntros && !string.IsNullOrWhiteSpace(options.MediaInfoIntroPath))            {                list.AddRange(_fileSystem.GetFilePaths(options.MediaInfoIntroPath, true)                    .Where(_libraryManager.IsVideoFile));            }            return list.Distinct(StringComparer.OrdinalIgnoreCase);        }        private bool FilterByParentalRating(int? ratingLevel, BaseItem item)        {            // Only content rated same or lower            if (ratingLevel.HasValue)            {                var level = string.IsNullOrWhiteSpace(item.OfficialRating)                    ? (int?)null                    : _localization.GetRatingLevel(item.OfficialRating);                return level.HasValue && level.Value <= ratingLevel.Value;            }            return true;        }        internal static int GetSimiliarityScore(BaseItem item1, List<PersonInfo> item1People, List<PersonInfo> allPeople, BaseItem item2, Random random, ILibraryManager libraryManager)        {            var points = 0;            if (!string.IsNullOrEmpty(item1.OfficialRating) && string.Equals(item1.OfficialRating, item2.OfficialRating, StringComparison.OrdinalIgnoreCase))            {                points += 10;            }            // Find common genres            points += item1.Genres.Where(i => item2.Genres.Contains(i, StringComparer.OrdinalIgnoreCase)).Sum(i => 10);            // Find common tags            points += GetTags(item1).Where(i => GetTags(item2).Contains(i, StringComparer.OrdinalIgnoreCase)).Sum(i => 10);            // Find common keywords            points += GetKeywords(item1).Where(i => GetKeywords(item2).Contains(i, StringComparer.OrdinalIgnoreCase)).Sum(i => 10);            // Find common studios            points += item1.Studios.Where(i => item2.Studios.Contains(i, StringComparer.OrdinalIgnoreCase)).Sum(i => 5);            var item2PeopleNames = allPeople.Where(i => i.ItemId == item2.Id)                .Select(i => i.Name)                .Where(i => !string.IsNullOrWhiteSpace(i))                .DistinctNames()                .ToDictionary(i => i, StringComparer.OrdinalIgnoreCase);            points += item1People.Where(i => item2PeopleNames.ContainsKey(i.Name)).Sum(i =>            {                if (string.Equals(i.Type, PersonType.Director, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Director, StringComparison.OrdinalIgnoreCase))                {                    return 5;                }                if (string.Equals(i.Type, PersonType.Actor, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Actor, StringComparison.OrdinalIgnoreCase))                {                    return 3;                }                if (string.Equals(i.Type, PersonType.Composer, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Composer, StringComparison.OrdinalIgnoreCase))                {                    return 3;                }                if (string.Equals(i.Type, PersonType.GuestStar, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.GuestStar, StringComparison.OrdinalIgnoreCase))                {                    return 3;                }                if (string.Equals(i.Type, PersonType.Writer, StringComparison.OrdinalIgnoreCase) || string.Equals(i.Role, PersonType.Writer, StringComparison.OrdinalIgnoreCase))                {                    return 2;                }                return 1;            });            // Add some randomization so that you're not always seeing the same ones for a given movie            points += random.Next(0, 50);            return points;        }        private static IEnumerable<string> GetTags(BaseItem item)        {            var hasTags = item as IHasTags;            if (hasTags != null)            {                return hasTags.Tags;            }            return new List<string>();        }        private static IEnumerable<string> GetKeywords(BaseItem item)        {            var hasTags = item as IHasKeywords;            if (hasTags != null)            {                return hasTags.Keywords;            }            return new List<string>();        }        public IEnumerable<string> GetAllIntroFiles()        {            return GetCustomIntroFiles(GetOptions(), true, true);        }        private bool IsSupporter        {            get { return _security.IsMBSupporter; }        }        public string Name        {            get { return "Default"; }        }        internal class ItemWithTrailer        {            internal BaseItem Item;            internal ItemWithTrailerType Type;            internal User User;            internal BaseItem WatchingItem;            internal List<PersonInfo> WatchingItemPeople;            internal List<PersonInfo> AllPeople;            internal Random Random;            internal ILibraryManager LibraryManager;            private bool? _isPlayed;            public bool IsPlayed            {                get                {                    if (!_isPlayed.HasValue)                    {                        _isPlayed = Item.IsPlayed(User);                    }                    return _isPlayed.Value;                }            }            private int? _score;            public int Score            {                get                {                    if (!_score.HasValue)                    {                        _score = GetSimiliarityScore(WatchingItem, WatchingItemPeople, AllPeople, Item, Random, LibraryManager);                    }                    return _score.Value;                }            }            public IntroInfo IntroInfo            {                get                {                    var id = Item.Id;                    if (Type == ItemWithTrailerType.ItemWithTrailer)                    {                        var hasTrailers = Item as IHasTrailers;                        if (hasTrailers != null)                        {                            id = hasTrailers.LocalTrailerIds.FirstOrDefault();                        }                    }                    return new IntroInfo                    {                        ItemId = id                    };                }            }        }        internal enum ItemWithTrailerType        {            LibraryTrailer,            ChannelTrailer,            ItemWithTrailer        }    }    public class CinemaModeConfigurationFactory : IConfigurationFactory    {        public IEnumerable<ConfigurationStore> GetConfigurations()        {            return new[]            {                new ConfigurationStore                {                     ConfigurationType = typeof(CinemaModeConfiguration),                     Key = "cinemamode"                }            };        }    }}
 |