| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749 | using MediaBrowser.Common;using MediaBrowser.Controller.Channels;using MediaBrowser.Controller.Configuration;using MediaBrowser.Controller.Devices;using MediaBrowser.Controller.Drawing;using MediaBrowser.Controller.Dto;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Audio;using MediaBrowser.Controller.Entities.Movies;using MediaBrowser.Controller.Entities.TV;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.LiveTv;using MediaBrowser.Controller.Persistence;using MediaBrowser.Controller.Playlists;using MediaBrowser.Controller.Providers;using MediaBrowser.Controller.Sync;using MediaBrowser.Model.Drawing;using MediaBrowser.Model.Dto;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Logging;using MediaBrowser.Model.Querying;using MediaBrowser.Model.Sync;using MoreLinq;using System;using System.Collections.Generic;using System.IO;using System.Linq;using System.Threading.Tasks;using CommonIO;namespace MediaBrowser.Server.Implementations.Dto{    public class DtoService : IDtoService    {        private readonly ILogger _logger;        private readonly ILibraryManager _libraryManager;        private readonly IUserDataManager _userDataRepository;        private readonly IItemRepository _itemRepo;        private readonly IImageProcessor _imageProcessor;        private readonly IServerConfigurationManager _config;        private readonly IFileSystem _fileSystem;        private readonly IProviderManager _providerManager;        private readonly Func<IChannelManager> _channelManagerFactory;        private readonly ISyncManager _syncManager;        private readonly IApplicationHost _appHost;        private readonly Func<IDeviceManager> _deviceManager;        private readonly Func<IMediaSourceManager> _mediaSourceManager;        private readonly Func<ILiveTvManager> _livetvManager;        public DtoService(ILogger logger, ILibraryManager libraryManager, IUserDataManager userDataRepository, IItemRepository itemRepo, IImageProcessor imageProcessor, IServerConfigurationManager config, IFileSystem fileSystem, IProviderManager providerManager, Func<IChannelManager> channelManagerFactory, ISyncManager syncManager, IApplicationHost appHost, Func<IDeviceManager> deviceManager, Func<IMediaSourceManager> mediaSourceManager, Func<ILiveTvManager> livetvManager)        {            _logger = logger;            _libraryManager = libraryManager;            _userDataRepository = userDataRepository;            _itemRepo = itemRepo;            _imageProcessor = imageProcessor;            _config = config;            _fileSystem = fileSystem;            _providerManager = providerManager;            _channelManagerFactory = channelManagerFactory;            _syncManager = syncManager;            _appHost = appHost;            _deviceManager = deviceManager;            _mediaSourceManager = mediaSourceManager;            _livetvManager = livetvManager;        }        /// <summary>        /// Converts a BaseItem to a DTOBaseItem        /// </summary>        /// <param name="item">The item.</param>        /// <param name="fields">The fields.</param>        /// <param name="user">The user.</param>        /// <param name="owner">The owner.</param>        /// <returns>Task{DtoBaseItem}.</returns>        /// <exception cref="System.ArgumentNullException">item</exception>        public BaseItemDto GetBaseItemDto(BaseItem item, List<ItemFields> fields, User user = null, BaseItem owner = null)        {            var options = new DtoOptions            {                Fields = fields            };            return GetBaseItemDto(item, options, user, owner);        }        public IEnumerable<BaseItemDto> GetBaseItemDtos(IEnumerable<BaseItem> items, DtoOptions options, User user = null, BaseItem owner = null)        {            var syncJobItems = GetSyncedItemProgress(options);            var syncDictionary = GetSyncedItemProgressDictionary(syncJobItems);            var list = new List<BaseItemDto>();            var programTuples = new List<Tuple<BaseItem, BaseItemDto>> { };            var channelTuples = new List<Tuple<BaseItemDto, LiveTvChannel>> { };            foreach (var item in items)            {                var dto = GetBaseItemDtoInternal(item, options, syncDictionary, user, owner);                var tvChannel = item as LiveTvChannel;                if (tvChannel != null)                {                    channelTuples.Add(new Tuple<BaseItemDto, LiveTvChannel>(dto, tvChannel));                }                else if (item is LiveTvProgram)                {                    programTuples.Add(new Tuple<BaseItem, BaseItemDto>(item, dto));                }                var byName = item as IItemByName;                if (byName != null)                {                    if (options.Fields.Contains(ItemFields.ItemCounts))                    {                        var libraryItems = byName.GetTaggedItems(new InternalItemsQuery(user)                        {                            Recursive = true                        });                        SetItemByNameInfo(item, dto, libraryItems.ToList(), user);                    }                }                FillSyncInfo(dto, item, syncJobItems, options, user);                list.Add(dto);            }            if (programTuples.Count > 0)            {                var task = _livetvManager().AddInfoToProgramDto(programTuples, options.Fields, user);                Task.WaitAll(task);            }            if (channelTuples.Count > 0)            {                _livetvManager().AddChannelInfo(channelTuples, options, user);            }            return list;        }        private Dictionary<string, SyncedItemProgress> GetSyncedItemProgressDictionary(IEnumerable<SyncedItemProgress> items)        {            var dict = new Dictionary<string, SyncedItemProgress>();            foreach (var item in items)            {                dict[item.ItemId] = item;            }            return dict;        }        public BaseItemDto GetBaseItemDto(BaseItem item, DtoOptions options, User user = null, BaseItem owner = null)        {            var syncProgress = GetSyncedItemProgress(options);            var dto = GetBaseItemDtoInternal(item, options, GetSyncedItemProgressDictionary(syncProgress), user, owner);            var tvChannel = item as LiveTvChannel;            if (tvChannel != null)            {                var list = new List<Tuple<BaseItemDto, LiveTvChannel>> { new Tuple<BaseItemDto, LiveTvChannel>(dto, tvChannel) };                _livetvManager().AddChannelInfo(list, options, user);            }            else if (item is LiveTvProgram)            {                var list = new List<Tuple<BaseItem, BaseItemDto>> { new Tuple<BaseItem, BaseItemDto>(item, dto) };                var task = _livetvManager().AddInfoToProgramDto(list, options.Fields, user);                Task.WaitAll(task);            }            var byName = item as IItemByName;            if (byName != null)            {                if (options.Fields.Contains(ItemFields.ItemCounts))                {                    SetItemByNameInfo(item, dto, GetTaggedItems(byName, user), user);                }                FillSyncInfo(dto, item, options, user, syncProgress);                return dto;            }            FillSyncInfo(dto, item, options, user, syncProgress);            return dto;        }        private List<BaseItem> GetTaggedItems(IItemByName byName, User user)        {            var items = byName.GetTaggedItems(new InternalItemsQuery(user)            {                Recursive = true            }).ToList();            return items;        }        private SyncedItemProgress[] GetSyncedItemProgress(DtoOptions options)        {            if (!options.Fields.Contains(ItemFields.SyncInfo))            {                return new SyncedItemProgress[] { };            }            var deviceId = options.DeviceId;            if (string.IsNullOrWhiteSpace(deviceId))            {                return new SyncedItemProgress[] { };            }            var caps = _deviceManager().GetCapabilities(deviceId);            if (caps == null || !caps.SupportsSync)            {                return new SyncedItemProgress[] { };            }            return _syncManager.GetSyncedItemProgresses(new SyncJobItemQuery            {                TargetId = deviceId,                Statuses = new[]                {                    SyncJobItemStatus.Converting,                    SyncJobItemStatus.Queued,                    SyncJobItemStatus.Transferring,                    SyncJobItemStatus.ReadyToTransfer,                    SyncJobItemStatus.Synced                }            }).Items;        }        public void FillSyncInfo(IEnumerable<Tuple<BaseItem, BaseItemDto>> tuples, DtoOptions options, User user)        {            if (options.Fields.Contains(ItemFields.SyncInfo))            {                var syncProgress = GetSyncedItemProgress(options);                foreach (var tuple in tuples)                {                    var item = tuple.Item1;                    FillSyncInfo(tuple.Item2, item, syncProgress, options, user);                }            }        }        private void FillSyncInfo(IHasSyncInfo dto, BaseItem item, DtoOptions options, User user, SyncedItemProgress[] syncProgress)        {            if (options.Fields.Contains(ItemFields.SyncInfo))            {                var userCanSync = user != null && user.Policy.EnableSync;                dto.SupportsSync = userCanSync && _syncManager.SupportsSync(item);            }            if (dto.SupportsSync ?? false)            {                dto.HasSyncJob = syncProgress.Any(i => i.Status != SyncJobItemStatus.Synced && string.Equals(i.ItemId, dto.Id, StringComparison.OrdinalIgnoreCase));                dto.IsSynced = syncProgress.Any(i => i.Status == SyncJobItemStatus.Synced && string.Equals(i.ItemId, dto.Id, StringComparison.OrdinalIgnoreCase));                if (dto.IsSynced.Value)                {                    dto.SyncStatus = SyncJobItemStatus.Synced;                }                else if (dto.HasSyncJob.Value)                {                    dto.SyncStatus = syncProgress.Where(i => string.Equals(i.ItemId, dto.Id, StringComparison.OrdinalIgnoreCase)).Select(i => i.Status).Max();                }            }        }        private void FillSyncInfo(IHasSyncInfo dto, BaseItem item, SyncedItemProgress[] syncProgress, DtoOptions options, User user)        {            if (options.Fields.Contains(ItemFields.SyncInfo))            {                var userCanSync = user != null && user.Policy.EnableSync;                dto.SupportsSync = userCanSync && _syncManager.SupportsSync(item);            }            if (dto.SupportsSync ?? false)            {                dto.HasSyncJob = syncProgress.Any(i => i.Status != SyncJobItemStatus.Synced && string.Equals(i.ItemId, dto.Id, StringComparison.OrdinalIgnoreCase));                dto.IsSynced = syncProgress.Any(i => i.Status == SyncJobItemStatus.Synced && string.Equals(i.ItemId, dto.Id, StringComparison.OrdinalIgnoreCase));                if (dto.IsSynced.Value)                {                    dto.SyncStatus = SyncJobItemStatus.Synced;                }                else if (dto.HasSyncJob.Value)                {                    dto.SyncStatus = syncProgress.Where(i => string.Equals(i.ItemId, dto.Id, StringComparison.OrdinalIgnoreCase)).Select(i => i.Status).Max();                }            }        }        private BaseItemDto GetBaseItemDtoInternal(BaseItem item, DtoOptions options, Dictionary<string, SyncedItemProgress> syncProgress, User user = null, BaseItem owner = null)        {            var fields = options.Fields;            if (item == null)            {                throw new ArgumentNullException("item");            }            if (fields == null)            {                throw new ArgumentNullException("fields");            }            var dto = new BaseItemDto            {                ServerId = _appHost.SystemId            };            if (item.SourceType == SourceType.Channel)            {                dto.SourceType = item.SourceType.ToString();            }            if (fields.Contains(ItemFields.People))            {                AttachPeople(dto, item);            }            if (fields.Contains(ItemFields.PrimaryImageAspectRatio))            {                try                {                    AttachPrimaryImageAspectRatio(dto, item);                }                catch (Exception ex)                {                    // Have to use a catch-all unfortunately because some .net image methods throw plain Exceptions                    _logger.ErrorException("Error generating PrimaryImageAspectRatio for {0}", ex, item.Name);                }            }            if (fields.Contains(ItemFields.DisplayPreferencesId))            {                dto.DisplayPreferencesId = item.DisplayPreferencesId.ToString("N");            }            if (user != null)            {                AttachUserSpecificInfo(dto, item, user, fields, syncProgress);            }            var hasMediaSources = item as IHasMediaSources;            if (hasMediaSources != null)            {                if (fields.Contains(ItemFields.MediaSources))                {                    if (user == null)                    {                        dto.MediaSources = _mediaSourceManager().GetStaticMediaSources(hasMediaSources, true).ToList();                    }                    else                    {                        dto.MediaSources = _mediaSourceManager().GetStaticMediaSources(hasMediaSources, true, user).ToList();                    }                }            }            if (fields.Contains(ItemFields.Studios))            {                AttachStudios(dto, item);            }            AttachBasicFields(dto, item, owner, options);            var collectionFolder = item as ICollectionFolder;            if (collectionFolder != null)            {                dto.OriginalCollectionType = collectionFolder.CollectionType;                dto.CollectionType = user == null ?                    collectionFolder.CollectionType :                    collectionFolder.GetViewType(user);            }            if (fields.Contains(ItemFields.CanDelete))            {                dto.CanDelete = user == null                    ? item.CanDelete()                    : item.CanDelete(user);            }            if (fields.Contains(ItemFields.CanDownload))            {                dto.CanDownload = user == null                    ? item.CanDownload()                    : item.CanDownload(user);            }            if (fields.Contains(ItemFields.Etag))            {                dto.Etag = item.GetEtag(user);            }            if (item is ILiveTvRecording)            {                _livetvManager().AddInfoToRecordingDto(item, dto, user);            }            return dto;        }        public BaseItemDto GetItemByNameDto(BaseItem item, DtoOptions options, List<BaseItem> taggedItems, User user = null)        {            var syncProgress = GetSyncedItemProgress(options);            var dto = GetBaseItemDtoInternal(item, options, GetSyncedItemProgressDictionary(syncProgress), user);            if (options.Fields.Contains(ItemFields.ItemCounts))            {                SetItemByNameInfo(item, dto, taggedItems, user);            }            FillSyncInfo(dto, item, options, user, syncProgress);            return dto;        }        private void SetItemByNameInfo(BaseItem item, BaseItemDto dto, List<BaseItem> taggedItems, User user = null)        {            if (item is MusicArtist || item is MusicGenre)            {                dto.AlbumCount = taggedItems.Count(i => i is MusicAlbum);                dto.MusicVideoCount = taggedItems.Count(i => i is MusicVideo);                dto.SongCount = taggedItems.Count(i => i is Audio);            }            else if (item is GameGenre)            {                dto.GameCount = taggedItems.Count(i => i is Game);            }            else            {                // This populates them all and covers Genre, Person, Studio, Year                dto.AlbumCount = taggedItems.Count(i => i is MusicAlbum);                dto.EpisodeCount = taggedItems.Count(i => i is Episode);                dto.GameCount = taggedItems.Count(i => i is Game);                dto.MovieCount = taggedItems.Count(i => i is Movie);                dto.TrailerCount = taggedItems.Count(i => i is Trailer);                dto.MusicVideoCount = taggedItems.Count(i => i is MusicVideo);                dto.SeriesCount = taggedItems.Count(i => i is Series);                dto.SongCount = taggedItems.Count(i => i is Audio);            }            dto.ChildCount = taggedItems.Count;        }        /// <summary>        /// Attaches the user specific info.        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        /// <param name="user">The user.</param>        /// <param name="fields">The fields.</param>        /// <param name="syncProgress">The synchronize progress.</param>        private void AttachUserSpecificInfo(BaseItemDto dto, BaseItem item, User user, List<ItemFields> fields, Dictionary<string, SyncedItemProgress> syncProgress)        {            if (item.IsFolder)            {                var userData = _userDataRepository.GetUserData(user, item);                // Skip the user data manager because we've already looped through the recursive tree and don't want to do it twice                // TODO: Improve in future                dto.UserData = GetUserItemDataDto(userData);                var folder = (Folder)item;                if (item.SourceType == SourceType.Library)                {                    dto.ChildCount = GetChildCount(folder, user);                    if (folder.SupportsUserDataFromChildren)                    {                        SetSpecialCounts(folder, user, dto, fields, syncProgress);                    }                }                if (fields.Contains(ItemFields.CumulativeRunTimeTicks))                {                    dto.CumulativeRunTimeTicks = item.RunTimeTicks;                }                if (fields.Contains(ItemFields.DateLastMediaAdded))                {                    dto.DateLastMediaAdded = folder.DateLastMediaAdded;                }                dto.UserData.Played = dto.UserData.PlayedPercentage.HasValue && dto.UserData.PlayedPercentage.Value >= 100;            }            else            {                dto.UserData = _userDataRepository.GetUserDataDto(item, user);            }            dto.PlayAccess = item.GetPlayAccess(user);            if (fields.Contains(ItemFields.SeasonUserData))            {                var episode = item as Episode;                if (episode != null)                {                    var season = episode.Season;                    if (season != null)                    {                        dto.SeasonUserData = _userDataRepository.GetUserDataDto(season, user);                    }                }            }            var userView = item as UserView;            if (userView != null)            {                dto.HasDynamicCategories = userView.ContainsDynamicCategories(user);            }            var collectionFolder = item as ICollectionFolder;            if (collectionFolder != null)            {                dto.HasDynamicCategories = false;            }        }        private int GetChildCount(Folder folder, User user)        {            return folder.GetChildren(user, true)                .Count();        }        /// <summary>        /// Gets client-side Id of a server-side BaseItem        /// </summary>        /// <param name="item">The item.</param>        /// <returns>System.String.</returns>        /// <exception cref="System.ArgumentNullException">item</exception>        public string GetDtoId(BaseItem item)        {            if (item == null)            {                throw new ArgumentNullException("item");            }            return item.Id.ToString("N");        }        /// <summary>        /// Converts a UserItemData to a DTOUserItemData        /// </summary>        /// <param name="data">The data.</param>        /// <returns>DtoUserItemData.</returns>        /// <exception cref="System.ArgumentNullException"></exception>        public UserItemDataDto GetUserItemDataDto(UserItemData data)        {            if (data == null)            {                throw new ArgumentNullException("data");            }            return new UserItemDataDto            {                IsFavorite = data.IsFavorite,                Likes = data.Likes,                PlaybackPositionTicks = data.PlaybackPositionTicks,                PlayCount = data.PlayCount,                Rating = data.Rating,                Played = data.Played,                LastPlayedDate = data.LastPlayedDate,                Key = data.Key            };        }        private void SetBookProperties(BaseItemDto dto, Book item)        {            dto.SeriesName = item.SeriesName;        }        private void SetPhotoProperties(BaseItemDto dto, Photo item)        {            dto.Width = item.Width;            dto.Height = item.Height;            dto.CameraMake = item.CameraMake;            dto.CameraModel = item.CameraModel;            dto.Software = item.Software;            dto.ExposureTime = item.ExposureTime;            dto.FocalLength = item.FocalLength;            dto.ImageOrientation = item.Orientation;            dto.Aperture = item.Aperture;            dto.ShutterSpeed = item.ShutterSpeed;            dto.Latitude = item.Latitude;            dto.Longitude = item.Longitude;            dto.Altitude = item.Altitude;            dto.IsoSpeedRating = item.IsoSpeedRating;            var album = item.Album;            if (album != null)            {                dto.Album = album.Name;                dto.AlbumId = album.Id.ToString("N");            }        }        private void SetMusicVideoProperties(BaseItemDto dto, MusicVideo item)        {            if (!string.IsNullOrEmpty(item.Album))            {                var parentAlbum = _libraryManager.GetItemList(new InternalItemsQuery                {                    IncludeItemTypes = new[] { typeof(MusicAlbum).Name },                    Name = item.Album                }).FirstOrDefault();                if (parentAlbum != null)                {                    dto.AlbumId = GetDtoId(parentAlbum);                }            }            dto.Album = item.Album;        }        private void SetGameProperties(BaseItemDto dto, Game item)        {            dto.Players = item.PlayersSupported;            dto.GameSystem = item.GameSystem;            dto.MultiPartGameFiles = item.MultiPartGameFiles;        }        private void SetGameSystemProperties(BaseItemDto dto, GameSystem item)        {            dto.GameSystem = item.GameSystemName;        }        private List<string> GetBackdropImageTags(BaseItem item, int limit)        {            return GetCacheTags(item, ImageType.Backdrop, limit).ToList();        }        private List<string> GetScreenshotImageTags(BaseItem item, int limit)        {            var hasScreenshots = item as IHasScreenshots;            if (hasScreenshots == null)            {                return new List<string>();            }            return GetCacheTags(item, ImageType.Screenshot, limit).ToList();        }        private IEnumerable<string> GetCacheTags(BaseItem item, ImageType type, int limit)        {            return item.GetImages(type)                // Convert to a list now in case GetImageCacheTag is slow                .ToList()                .Select(p => GetImageCacheTag(item, p))                .Where(i => i != null)                .Take(limit)                .ToList();        }        private string GetImageCacheTag(BaseItem item, ImageType type)        {            try            {                return _imageProcessor.GetImageCacheTag(item, type);            }            catch (Exception ex)            {                _logger.ErrorException("Error getting {0} image info", ex, type);                return null;            }        }        private string GetImageCacheTag(BaseItem item, ItemImageInfo image)        {            try            {                return _imageProcessor.GetImageCacheTag(item, image);            }            catch (Exception ex)            {                _logger.ErrorException("Error getting {0} image info for {1}", ex, image.Type, image.Path);                return null;            }        }        /// <summary>        /// Attaches People DTO's to a DTOBaseItem        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        /// <returns>Task.</returns>        private void AttachPeople(BaseItemDto dto, BaseItem item)        {            // Ordering by person type to ensure actors and artists are at the front.            // This is taking advantage of the fact that they both begin with A            // This should be improved in the future            var people = _libraryManager.GetPeople(item).OrderBy(i => i.SortOrder ?? int.MaxValue)                .ThenBy(i =>                {                    if (i.IsType(PersonType.Actor))                    {                        return 0;                    }                    if (i.IsType(PersonType.GuestStar))                    {                        return 1;                    }                    if (i.IsType(PersonType.Director))                    {                        return 2;                    }                    if (i.IsType(PersonType.Writer))                    {                        return 3;                    }                    if (i.IsType(PersonType.Producer))                    {                        return 4;                    }                    if (i.IsType(PersonType.Composer))                    {                        return 4;                    }                    return 10;                })                .ToList();            var list = new List<BaseItemPerson>();            var dictionary = people.Select(p => p.Name)                .Distinct(StringComparer.OrdinalIgnoreCase).Select(c =>                {                    try                    {                        return _libraryManager.GetPerson(c);                    }                    catch (Exception ex)                    {                        _logger.ErrorException("Error getting person {0}", ex, c);                        return null;                    }                }).Where(i => i != null)                .DistinctBy(i => i.Name, StringComparer.OrdinalIgnoreCase)                .ToDictionary(i => i.Name, StringComparer.OrdinalIgnoreCase);            for (var i = 0; i < people.Count; i++)            {                var person = people[i];                var baseItemPerson = new BaseItemPerson                {                    Name = person.Name,                    Role = person.Role,                    Type = person.Type                };                Person entity;                if (dictionary.TryGetValue(person.Name, out entity))                {                    baseItemPerson.PrimaryImageTag = GetImageCacheTag(entity, ImageType.Primary);                    baseItemPerson.Id = entity.Id.ToString("N");                    list.Add(baseItemPerson);                }            }            dto.People = list.ToArray();        }        /// <summary>        /// Attaches the studios.        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        /// <returns>Task.</returns>        private void AttachStudios(BaseItemDto dto, BaseItem item)        {            var studios = item.Studios.ToList();            dto.Studios = new StudioDto[studios.Count];            var dictionary = studios.Distinct(StringComparer.OrdinalIgnoreCase).Select(name =>            {                try                {                    return _libraryManager.GetStudio(name);                }                catch (IOException ex)                {                    _logger.ErrorException("Error getting studio {0}", ex, name);                    return null;                }            })            .Where(i => i != null)            .DistinctBy(i => i.Name, StringComparer.OrdinalIgnoreCase)            .ToDictionary(i => i.Name, StringComparer.OrdinalIgnoreCase);            for (var i = 0; i < studios.Count; i++)            {                var studio = studios[i];                var studioDto = new StudioDto                {                    Name = studio                };                Studio entity;                if (dictionary.TryGetValue(studio, out entity))                {                    studioDto.Id = entity.Id.ToString("N");                    studioDto.PrimaryImageTag = GetImageCacheTag(entity, ImageType.Primary);                }                dto.Studios[i] = studioDto;            }        }        /// <summary>        /// If an item does not any backdrops, this can be used to find the first parent that does have one        /// </summary>        /// <param name="item">The item.</param>        /// <param name="owner">The owner.</param>        /// <returns>BaseItem.</returns>        private BaseItem GetParentBackdropItem(BaseItem item, BaseItem owner)        {            var parent = item.GetParent() ?? owner;            while (parent != null)            {                if (parent.GetImages(ImageType.Backdrop).Any())                {                    return parent;                }                parent = parent.GetParent();            }            return null;        }        /// <summary>        /// If an item does not have a logo, this can be used to find the first parent that does have one        /// </summary>        /// <param name="item">The item.</param>        /// <param name="type">The type.</param>        /// <param name="owner">The owner.</param>        /// <returns>BaseItem.</returns>        private BaseItem GetParentImageItem(BaseItem item, ImageType type, BaseItem owner)        {            var parent = item.GetParent() ?? owner;            while (parent != null)            {                if (parent.HasImage(type))                {                    return parent;                }                parent = parent.GetParent();            }            return null;        }        /// <summary>        /// Gets the chapter info dto.        /// </summary>        /// <param name="chapterInfo">The chapter info.</param>        /// <param name="item">The item.</param>        /// <returns>ChapterInfoDto.</returns>        private ChapterInfoDto GetChapterInfoDto(ChapterInfo chapterInfo, BaseItem item)        {            var dto = new ChapterInfoDto            {                Name = chapterInfo.Name,                StartPositionTicks = chapterInfo.StartPositionTicks            };            if (!string.IsNullOrEmpty(chapterInfo.ImagePath))            {                dto.ImageTag = GetImageCacheTag(item, new ItemImageInfo                {                    Path = chapterInfo.ImagePath,                    Type = ImageType.Chapter,                    DateModified = _fileSystem.GetLastWriteTimeUtc(chapterInfo.ImagePath)                });            }            return dto;        }        public List<ChapterInfoDto> GetChapterInfoDtos(BaseItem item)        {            return _itemRepo.GetChapters(item.Id)                .Select(c => GetChapterInfoDto(c, item))                .ToList();        }        /// <summary>        /// Sets simple property values on a DTOBaseItem        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        /// <param name="owner">The owner.</param>        /// <param name="options">The options.</param>        private void AttachBasicFields(BaseItemDto dto, BaseItem item, BaseItem owner, DtoOptions options)        {            var fields = options.Fields;            if (fields.Contains(ItemFields.DateCreated))            {                dto.DateCreated = item.DateCreated;            }            if (fields.Contains(ItemFields.DisplayMediaType))            {                dto.DisplayMediaType = item.DisplayMediaType;            }            if (fields.Contains(ItemFields.Settings))            {                dto.LockedFields = item.LockedFields;                dto.LockData = item.IsLocked;                dto.ForcedSortName = item.ForcedSortName;            }            var hasBudget = item as IHasBudget;            if (hasBudget != null)            {                if (fields.Contains(ItemFields.Budget))                {                    dto.Budget = hasBudget.Budget;                }                if (fields.Contains(ItemFields.Revenue))                {                    dto.Revenue = hasBudget.Revenue;                }            }            dto.EndDate = item.EndDate;            if (fields.Contains(ItemFields.HomePageUrl))            {                dto.HomePageUrl = item.HomePageUrl;            }            if (fields.Contains(ItemFields.ExternalUrls))            {                dto.ExternalUrls = _providerManager.GetExternalUrls(item).ToArray();            }            if (fields.Contains(ItemFields.Tags))            {                dto.Tags = item.Tags;            }            if (fields.Contains(ItemFields.Keywords))            {                dto.Keywords = item.Keywords;            }            if (fields.Contains(ItemFields.ProductionLocations))            {                SetProductionLocations(item, dto);            }            var hasAspectRatio = item as IHasAspectRatio;            if (hasAspectRatio != null)            {                dto.AspectRatio = hasAspectRatio.AspectRatio;            }            if (fields.Contains(ItemFields.Metascore))            {                var hasMetascore = item as IHasMetascore;                if (hasMetascore != null)                {                    dto.Metascore = hasMetascore.Metascore;                }            }            if (fields.Contains(ItemFields.AwardSummary))            {                var hasAwards = item as IHasAwards;                if (hasAwards != null)                {                    dto.AwardSummary = hasAwards.AwardSummary;                }            }            var backdropLimit = options.GetImageLimit(ImageType.Backdrop);            if (backdropLimit > 0)            {                dto.BackdropImageTags = GetBackdropImageTags(item, backdropLimit);            }            if (fields.Contains(ItemFields.ScreenshotImageTags))            {                var screenshotLimit = options.GetImageLimit(ImageType.Screenshot);                if (screenshotLimit > 0)                {                    dto.ScreenshotImageTags = GetScreenshotImageTags(item, screenshotLimit);                }            }            if (fields.Contains(ItemFields.Genres))            {                dto.Genres = item.Genres;            }            dto.ImageTags = new Dictionary<ImageType, string>();            // Prevent implicitly captured closure            var currentItem = item;            foreach (var image in currentItem.ImageInfos.Where(i => !currentItem.AllowsMultipleImages(i.Type))                .ToList())            {                if (options.GetImageLimit(image.Type) > 0)                {                    var tag = GetImageCacheTag(item, image);                    if (tag != null)                    {                        dto.ImageTags[image.Type] = tag;                    }                }            }            dto.Id = GetDtoId(item);            dto.IndexNumber = item.IndexNumber;            dto.IsFolder = item.IsFolder;            dto.MediaType = item.MediaType;            dto.LocationType = item.LocationType;            if (item.IsHD.HasValue && item.IsHD.Value)            {                dto.IsHD = item.IsHD;            }            dto.Audio = item.Audio;            dto.PreferredMetadataCountryCode = item.PreferredMetadataCountryCode;            dto.PreferredMetadataLanguage = item.PreferredMetadataLanguage;            var hasCriticRating = item as IHasCriticRating;            if (hasCriticRating != null)            {                dto.CriticRating = hasCriticRating.CriticRating;                if (fields.Contains(ItemFields.CriticRatingSummary))                {                    dto.CriticRatingSummary = hasCriticRating.CriticRatingSummary;                }            }            var hasTrailers = item as IHasTrailers;            if (hasTrailers != null)            {                dto.LocalTrailerCount = hasTrailers.GetTrailerIds().Count;            }            var hasDisplayOrder = item as IHasDisplayOrder;            if (hasDisplayOrder != null)            {                dto.DisplayOrder = hasDisplayOrder.DisplayOrder;            }            var userView = item as UserView;            if (userView != null)            {                dto.CollectionType = userView.ViewType;            }            if (fields.Contains(ItemFields.RemoteTrailers))            {                dto.RemoteTrailers = hasTrailers != null ?                    hasTrailers.RemoteTrailers :                    new List<MediaUrl>();            }            dto.Name = item.Name;            dto.OfficialRating = item.OfficialRating;            if (fields.Contains(ItemFields.Overview))            {                dto.Overview = item.Overview;            }            if (fields.Contains(ItemFields.OriginalTitle))            {                dto.OriginalTitle = item.OriginalTitle;            }            if (fields.Contains(ItemFields.ShortOverview))            {                var hasShortOverview = item as IHasShortOverview;                if (hasShortOverview != null)                {                    dto.ShortOverview = hasShortOverview.ShortOverview;                }            }            // If there are no backdrops, indicate what parent has them in case the Ui wants to allow inheritance            if (backdropLimit > 0 && dto.BackdropImageTags.Count == 0)            {                var parentWithBackdrop = GetParentBackdropItem(item, owner);                if (parentWithBackdrop != null)                {                    dto.ParentBackdropItemId = GetDtoId(parentWithBackdrop);                    dto.ParentBackdropImageTags = GetBackdropImageTags(parentWithBackdrop, backdropLimit);                }            }            if (fields.Contains(ItemFields.ParentId))            {                var displayParentId = item.DisplayParentId;                if (displayParentId.HasValue)                {                    dto.ParentId = displayParentId.Value.ToString("N");                }            }            dto.ParentIndexNumber = item.ParentIndexNumber;            // If there is no logo, indicate what parent has one in case the Ui wants to allow inheritance            if (!dto.HasLogo && options.GetImageLimit(ImageType.Logo) > 0)            {                var parentWithLogo = GetParentImageItem(item, ImageType.Logo, owner);                if (parentWithLogo != null)                {                    dto.ParentLogoItemId = GetDtoId(parentWithLogo);                    dto.ParentLogoImageTag = GetImageCacheTag(parentWithLogo, ImageType.Logo);                }            }            // If there is no art, indicate what parent has one in case the Ui wants to allow inheritance            if (!dto.HasArtImage && options.GetImageLimit(ImageType.Art) > 0)            {                var parentWithImage = GetParentImageItem(item, ImageType.Art, owner);                if (parentWithImage != null)                {                    dto.ParentArtItemId = GetDtoId(parentWithImage);                    dto.ParentArtImageTag = GetImageCacheTag(parentWithImage, ImageType.Art);                }            }            // If there is no thumb, indicate what parent has one in case the Ui wants to allow inheritance            if (!dto.HasThumb && options.GetImageLimit(ImageType.Thumb) > 0)            {                var parentWithImage = GetParentImageItem(item, ImageType.Thumb, owner);                if (parentWithImage != null)                {                    dto.ParentThumbItemId = GetDtoId(parentWithImage);                    dto.ParentThumbImageTag = GetImageCacheTag(parentWithImage, ImageType.Thumb);                }            }            if (fields.Contains(ItemFields.Path))            {                dto.Path = GetMappedPath(item);            }            dto.PremiereDate = item.PremiereDate;            dto.ProductionYear = item.ProductionYear;            if (fields.Contains(ItemFields.ProviderIds))            {                dto.ProviderIds = item.ProviderIds;            }            dto.RunTimeTicks = item.RunTimeTicks;            if (fields.Contains(ItemFields.SortName))            {                dto.SortName = item.SortName;            }            if (fields.Contains(ItemFields.CustomRating))            {                dto.CustomRating = item.CustomRating;            }            if (fields.Contains(ItemFields.Taglines))            {                var hasTagline = item as IHasTaglines;                if (hasTagline != null)                {                    dto.Taglines = hasTagline.Taglines;                }                if (dto.Taglines == null)                {                    dto.Taglines = new List<string>();                }            }            dto.Type = item.GetClientTypeName();            dto.CommunityRating = item.CommunityRating;            if (fields.Contains(ItemFields.VoteCount))            {                dto.VoteCount = item.VoteCount;            }            //if (item.IsFolder)            //{            //    var folder = (Folder)item;            //    if (fields.Contains(ItemFields.IndexOptions))            //    {            //        dto.IndexOptions = folder.IndexByOptionStrings.ToArray();            //    }            //}            var supportsPlaceHolders = item as ISupportsPlaceHolders;            if (supportsPlaceHolders != null)            {                dto.IsPlaceHolder = supportsPlaceHolders.IsPlaceHolder;            }            // Add audio info            var audio = item as Audio;            if (audio != null)            {                dto.Album = audio.Album;                dto.ExtraType = audio.ExtraType;                var albumParent = audio.AlbumEntity;                if (albumParent != null)                {                    dto.AlbumId = GetDtoId(albumParent);                    dto.AlbumPrimaryImageTag = GetImageCacheTag(albumParent, ImageType.Primary);                }                //if (fields.Contains(ItemFields.MediaSourceCount))                //{                // Songs always have one                //}            }            var hasArtist = item as IHasArtist;            if (hasArtist != null)            {                dto.Artists = hasArtist.Artists;                dto.ArtistItems = hasArtist                    .Artists                    .Select(i =>                    {                        try                        {                            var artist = _libraryManager.GetArtist(i);                            return new NameIdPair                            {                                Name = artist.Name,                                Id = artist.Id.ToString("N")                            };                        }                        catch (Exception ex)                        {                            _logger.ErrorException("Error getting artist", ex);                            return null;                        }                    })                    .Where(i => i != null)                    .ToList();            }            var hasAlbumArtist = item as IHasAlbumArtist;            if (hasAlbumArtist != null)            {                dto.AlbumArtist = hasAlbumArtist.AlbumArtists.FirstOrDefault();                dto.AlbumArtists = hasAlbumArtist                    .AlbumArtists                    .Select(i =>                    {                        try                        {                            var artist = _libraryManager.GetArtist(i);                            return new NameIdPair                            {                                Name = artist.Name,                                Id = artist.Id.ToString("N")                            };                        }                        catch (Exception ex)                        {                            _logger.ErrorException("Error getting album artist", ex);                            return null;                        }                    })                    .Where(i => i != null)                    .ToList();            }            // Add video info            var video = item as Video;            if (video != null)            {                dto.VideoType = video.VideoType;                dto.Video3DFormat = video.Video3DFormat;                dto.IsoType = video.IsoType;                if (video.HasSubtitles)                {                    dto.HasSubtitles = video.HasSubtitles;                }                if (video.AdditionalParts.Count != 0)                {                    dto.PartCount = video.AdditionalParts.Count + 1;                }                if (fields.Contains(ItemFields.MediaSourceCount))                {                    var mediaSourceCount = video.MediaSourceCount;                    if (mediaSourceCount != 1)                    {                        dto.MediaSourceCount = mediaSourceCount;                    }                }                if (fields.Contains(ItemFields.Chapters))                {                    dto.Chapters = GetChapterInfoDtos(item);                }                dto.ExtraType = video.ExtraType;            }            if (fields.Contains(ItemFields.MediaStreams))            {                // Add VideoInfo                var iHasMediaSources = item as IHasMediaSources;                if (iHasMediaSources != null)                {                    List<MediaStream> mediaStreams;                    if (dto.MediaSources != null && dto.MediaSources.Count > 0)                    {                        mediaStreams = dto.MediaSources.Where(i => new Guid(i.Id) == item.Id)                            .SelectMany(i => i.MediaStreams)                            .ToList();                    }                    else                    {                        mediaStreams = _mediaSourceManager().GetStaticMediaSources(iHasMediaSources, true).First().MediaStreams;                    }                    dto.MediaStreams = mediaStreams;                }            }            var hasSpecialFeatures = item as IHasSpecialFeatures;            if (hasSpecialFeatures != null)            {                var specialFeatureCount = hasSpecialFeatures.SpecialFeatureIds.Count;                if (specialFeatureCount > 0)                {                    dto.SpecialFeatureCount = specialFeatureCount;                }            }            // Add EpisodeInfo            var episode = item as Episode;            if (episode != null)            {                dto.IndexNumberEnd = episode.IndexNumberEnd;                dto.SeriesName = episode.SeriesName;                if (fields.Contains(ItemFields.AlternateEpisodeNumbers))                {                    dto.DvdSeasonNumber = episode.DvdSeasonNumber;                    dto.DvdEpisodeNumber = episode.DvdEpisodeNumber;                    dto.AbsoluteEpisodeNumber = episode.AbsoluteEpisodeNumber;                }                if (fields.Contains(ItemFields.SpecialEpisodeNumbers))                {                    dto.AirsAfterSeasonNumber = episode.AirsAfterSeasonNumber;                    dto.AirsBeforeEpisodeNumber = episode.AirsBeforeEpisodeNumber;                    dto.AirsBeforeSeasonNumber = episode.AirsBeforeSeasonNumber;                }                var seasonId = episode.SeasonId;                if (seasonId.HasValue)                {                    dto.SeasonId = seasonId.Value.ToString("N");                }                var episodeSeason = episode.Season;                if (episodeSeason != null)                {                    if (fields.Contains(ItemFields.SeasonName))                    {                        dto.SeasonName = episodeSeason.Name;                    }                }                var episodeSeries = episode.Series;                if (episodeSeries != null)                {                    if (fields.Contains(ItemFields.SeriesGenres))                    {                        dto.SeriesGenres = episodeSeries.Genres.ToList();                    }                    dto.SeriesId = GetDtoId(episodeSeries);                    if (fields.Contains(ItemFields.AirTime))                    {                        dto.AirTime = episodeSeries.AirTime;                    }                    if (options.GetImageLimit(ImageType.Thumb) > 0)                    {                        dto.SeriesThumbImageTag = GetImageCacheTag(episodeSeries, ImageType.Thumb);                    }                    if (options.GetImageLimit(ImageType.Primary) > 0)                    {                        dto.SeriesPrimaryImageTag = GetImageCacheTag(episodeSeries, ImageType.Primary);                    }                    if (fields.Contains(ItemFields.SeriesStudio))                    {                        dto.SeriesStudio = episodeSeries.Studios.FirstOrDefault();                    }                }            }            // Add SeriesInfo            var series = item as Series;            if (series != null)            {                dto.AirDays = series.AirDays;                dto.AirTime = series.AirTime;                dto.SeriesStatus = series.Status;                dto.AnimeSeriesIndex = series.AnimeSeriesIndex;            }            // Add SeasonInfo            var season = item as Season;            if (season != null)            {                series = season.Series;                if (series != null)                {                    dto.SeriesId = GetDtoId(series);                    dto.SeriesName = series.Name;                    dto.AirTime = series.AirTime;                    dto.SeriesStudio = series.Studios.FirstOrDefault();                    if (options.GetImageLimit(ImageType.Primary) > 0)                    {                        dto.SeriesPrimaryImageTag = GetImageCacheTag(series, ImageType.Primary);                    }                }            }            var game = item as Game;            if (game != null)            {                SetGameProperties(dto, game);            }            var gameSystem = item as GameSystem;            if (gameSystem != null)            {                SetGameSystemProperties(dto, gameSystem);            }            var musicVideo = item as MusicVideo;            if (musicVideo != null)            {                SetMusicVideoProperties(dto, musicVideo);            }            var book = item as Book;            if (book != null)            {                SetBookProperties(dto, book);            }            var photo = item as Photo;            if (photo != null)            {                SetPhotoProperties(dto, photo);            }            dto.ChannelId = item.ChannelId;            if (item.SourceType == SourceType.Channel && !string.IsNullOrWhiteSpace(item.ChannelId))            {                var channel = _libraryManager.GetItemById(item.ChannelId);                if (channel != null)                {                    dto.ChannelName = channel.Name;                }            }        }        private string GetMappedPath(IHasMetadata item)        {            var path = item.Path;            var locationType = item.LocationType;            if (locationType == LocationType.FileSystem || locationType == LocationType.Offline)            {                foreach (var map in _config.Configuration.PathSubstitutions)                {                    path = _libraryManager.SubstitutePath(path, map.From, map.To);                }            }            return path;        }        private void SetProductionLocations(BaseItem item, BaseItemDto dto)        {            var hasProductionLocations = item as IHasProductionLocations;            if (hasProductionLocations != null)            {                dto.ProductionLocations = hasProductionLocations.ProductionLocations;            }            var person = item as Person;            if (person != null)            {                dto.ProductionLocations = new List<string>();                if (!string.IsNullOrEmpty(person.PlaceOfBirth))                {                    dto.ProductionLocations.Add(person.PlaceOfBirth);                }            }            if (dto.ProductionLocations == null)            {                dto.ProductionLocations = new List<string>();            }        }        /// <summary>        /// Since it can be slow to make all of these calculations independently, this method will provide a way to do them all at once        /// </summary>        /// <param name="folder">The folder.</param>        /// <param name="user">The user.</param>        /// <param name="dto">The dto.</param>        /// <param name="fields">The fields.</param>        /// <param name="syncProgress">The synchronize progress.</param>        /// <returns>Task.</returns>        private void SetSpecialCounts(Folder folder, User user, BaseItemDto dto, List<ItemFields> fields, Dictionary<string, SyncedItemProgress> syncProgress)        {            var recursiveItemCount = 0;            var unplayed = 0;            double totalPercentPlayed = 0;            double totalSyncPercent = 0;            var addSyncInfo = fields.Contains(ItemFields.SyncInfo);            var children = folder.GetItems(new InternalItemsQuery            {                IsFolder = false,                Recursive = true,                ExcludeLocationTypes = new[] { LocationType.Virtual },                User = user            }).Result.Items;            // Loop through each recursive child            foreach (var child in children)            {                var userdata = _userDataRepository.GetUserData(user, child);                recursiveItemCount++;                var isUnplayed = true;                // Incrememt totalPercentPlayed                if (userdata != null)                {                    if (userdata.Played)                    {                        totalPercentPlayed += 100;                        isUnplayed = false;                    }                    else if (userdata.PlaybackPositionTicks > 0 && child.RunTimeTicks.HasValue && child.RunTimeTicks.Value > 0)                    {                        double itemPercent = userdata.PlaybackPositionTicks;                        itemPercent /= child.RunTimeTicks.Value;                        totalPercentPlayed += itemPercent;                    }                }                if (isUnplayed)                {                    unplayed++;                }                if (addSyncInfo)                {                    double percent = 0;                    SyncedItemProgress syncItemProgress;                    if (syncProgress.TryGetValue(child.Id.ToString("N"), out syncItemProgress))                    {                        switch (syncItemProgress.Status)                        {                            case SyncJobItemStatus.Synced:                                percent = 100;                                break;                            case SyncJobItemStatus.Converting:                            case SyncJobItemStatus.ReadyToTransfer:                            case SyncJobItemStatus.Transferring:                                percent = 50;                                break;                        }                    }                    totalSyncPercent += percent;                }            }            dto.RecursiveItemCount = recursiveItemCount;            dto.UserData.UnplayedItemCount = unplayed;            if (recursiveItemCount > 0)            {                dto.UserData.PlayedPercentage = totalPercentPlayed / recursiveItemCount;                if (addSyncInfo)                {                    var pct = totalSyncPercent / recursiveItemCount;                    if (pct > 0)                    {                        dto.SyncPercent = pct;                    }                }            }        }        /// <summary>        /// Attaches the primary image aspect ratio.        /// </summary>        /// <param name="dto">The dto.</param>        /// <param name="item">The item.</param>        /// <returns>Task.</returns>        public void AttachPrimaryImageAspectRatio(IItemDto dto, IHasImages item)        {            dto.PrimaryImageAspectRatio = GetPrimaryImageAspectRatio(item);        }        public double? GetPrimaryImageAspectRatio(IHasImages item)        {            var imageInfo = item.GetImageInfo(ImageType.Primary, 0);            if (imageInfo == null || !imageInfo.IsLocalFile)            {                return null;            }            ImageSize size;            try            {                size = _imageProcessor.GetImageSize(imageInfo);            }            catch (Exception ex)            {                //_logger.ErrorException("Failed to determine primary image aspect ratio for {0}", ex, path);                return null;            }            var supportedEnhancers = _imageProcessor.GetSupportedEnhancers(item, ImageType.Primary).ToList();            foreach (var enhancer in supportedEnhancers)            {                try                {                    size = enhancer.GetEnhancedImageSize(item, ImageType.Primary, 0, size);                }                catch (Exception ex)                {                    _logger.ErrorException("Error in image enhancer: {0}", ex, enhancer.GetType().Name);                }            }            var width = size.Width;            var height = size.Height;            if (width == 0 || height == 0)            {                return null;            }            var photo = item as Photo;            if (photo != null && photo.Orientation.HasValue)            {                switch (photo.Orientation.Value)                {                    case ImageOrientation.LeftBottom:                    case ImageOrientation.LeftTop:                    case ImageOrientation.RightBottom:                    case ImageOrientation.RightTop:                        var temp = height;                        height = width;                        width = temp;                        break;                }            }            return width / height;        }    }}
 |