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.
1385 lines
46 KiB
1385 lines
46 KiB
using MediaBrowser.Common.Extensions;
|
|
using MediaBrowser.Common.IO;
|
|
using MediaBrowser.Controller.Channels;
|
|
using MediaBrowser.Controller.Configuration;
|
|
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.Providers;
|
|
using MediaBrowser.Model.Drawing;
|
|
using MediaBrowser.Model.Dto;
|
|
using MediaBrowser.Model.Entities;
|
|
using MediaBrowser.Model.Logging;
|
|
using MediaBrowser.Model.Querying;
|
|
using MoreLinq;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
|
|
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;
|
|
|
|
public DtoService(ILogger logger, ILibraryManager libraryManager, IUserDataManager userDataRepository, IItemRepository itemRepo, IImageProcessor imageProcessor, IServerConfigurationManager config, IFileSystem fileSystem, IProviderManager providerManager, Func<IChannelManager> channelManagerFactory)
|
|
{
|
|
_logger = logger;
|
|
_libraryManager = libraryManager;
|
|
_userDataRepository = userDataRepository;
|
|
_itemRepo = itemRepo;
|
|
_imageProcessor = imageProcessor;
|
|
_config = config;
|
|
_fileSystem = fileSystem;
|
|
_providerManager = providerManager;
|
|
_channelManagerFactory = channelManagerFactory;
|
|
}
|
|
|
|
/// <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 byName = item as IItemByName;
|
|
|
|
if (byName != null)
|
|
{
|
|
var libraryItems = user != null ?
|
|
user.RootFolder.GetRecursiveChildren(user) :
|
|
_libraryManager.RootFolder.RecursiveChildren;
|
|
|
|
var dto = GetBaseItemDtoInternal(item, fields, user);
|
|
|
|
SetItemByNameInfo(item, dto, byName.GetTaggedItems(libraryItems).ToList(), user);
|
|
|
|
return dto;
|
|
}
|
|
|
|
return GetBaseItemDtoInternal(item, fields, user, owner);
|
|
}
|
|
|
|
private BaseItemDto GetBaseItemDtoInternal(BaseItem item, List<ItemFields> fields, User user = null, BaseItem owner = null)
|
|
{
|
|
if (item == null)
|
|
{
|
|
throw new ArgumentNullException("item");
|
|
}
|
|
|
|
if (fields == null)
|
|
{
|
|
throw new ArgumentNullException("fields");
|
|
}
|
|
|
|
var dto = new BaseItemDto();
|
|
|
|
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);
|
|
}
|
|
|
|
var hasMediaSources = item as IHasMediaSources;
|
|
if (hasMediaSources != null)
|
|
{
|
|
if (fields.Contains(ItemFields.MediaSources))
|
|
{
|
|
if (user == null)
|
|
{
|
|
dto.MediaSources = hasMediaSources.GetMediaSources(true).ToList();
|
|
}
|
|
else
|
|
{
|
|
dto.MediaSources = hasMediaSources.GetMediaSources(true, user).ToList();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (fields.Contains(ItemFields.Studios))
|
|
{
|
|
AttachStudios(dto, item);
|
|
}
|
|
|
|
AttachBasicFields(dto, item, owner, fields);
|
|
|
|
if (fields.Contains(ItemFields.SoundtrackIds))
|
|
{
|
|
var hasSoundtracks = item as IHasSoundtracks;
|
|
|
|
if (hasSoundtracks != null)
|
|
{
|
|
dto.SoundtrackIds = hasSoundtracks.SoundtrackIds
|
|
.Select(i => i.ToString("N"))
|
|
.ToArray();
|
|
}
|
|
}
|
|
|
|
return dto;
|
|
}
|
|
|
|
public BaseItemDto GetItemByNameDto<T>(T item, List<ItemFields> fields, List<BaseItem> taggedItems, User user = null)
|
|
where T : BaseItem, IItemByName
|
|
{
|
|
var dto = GetBaseItemDtoInternal(item, fields, user);
|
|
|
|
SetItemByNameInfo(item, dto, taggedItems, user);
|
|
|
|
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.AdultVideoCount = taggedItems.Count(i => i is AdultVideo);
|
|
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.MusicVideoCount = taggedItems.Count(i => i is MusicVideo);
|
|
dto.SeriesCount = taggedItems.Count(i => i is Series);
|
|
dto.SongCount = taggedItems.Count(i => i is Audio);
|
|
dto.TrailerCount = taggedItems.Count(i => i is Trailer);
|
|
}
|
|
|
|
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>
|
|
private void AttachUserSpecificInfo(BaseItemDto dto, BaseItem item, User user, List<ItemFields> fields)
|
|
{
|
|
if (item.IsFolder)
|
|
{
|
|
var userData = _userDataRepository.GetUserData(user.Id, item.GetUserDataKey());
|
|
|
|
// 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;
|
|
|
|
dto.ChildCount = GetChildCount(folder, user);
|
|
|
|
if (!(folder is UserRootFolder))
|
|
{
|
|
SetSpecialCounts(folder, user, dto, fields);
|
|
}
|
|
|
|
dto.UserData.Played = dto.PlayedPercentage.HasValue && dto.PlayedPercentage.Value >= 100;
|
|
dto.UserData.PlayedPercentage = dto.PlayedPercentage;
|
|
dto.UserData.UnplayedItemCount = dto.RecursiveUnplayedItemCount;
|
|
}
|
|
|
|
else
|
|
{
|
|
dto.UserData = _userDataRepository.GetUserDataDto(item, user);
|
|
}
|
|
|
|
dto.PlayAccess = item.GetPlayAccess(user);
|
|
}
|
|
|
|
private int GetChildCount(Folder folder, User user)
|
|
{
|
|
return folder.GetChildren(user, true)
|
|
.Count();
|
|
}
|
|
|
|
public UserDto GetUserDto(User user)
|
|
{
|
|
if (user == null)
|
|
{
|
|
throw new ArgumentNullException("user");
|
|
}
|
|
|
|
var dto = new UserDto
|
|
{
|
|
Id = user.Id.ToString("N"),
|
|
Name = user.Name,
|
|
HasPassword = !String.IsNullOrEmpty(user.Password),
|
|
LastActivityDate = user.LastActivityDate,
|
|
LastLoginDate = user.LastLoginDate,
|
|
Configuration = user.Configuration
|
|
};
|
|
|
|
var image = user.GetImageInfo(ImageType.Primary, 0);
|
|
|
|
if (image != null)
|
|
{
|
|
dto.PrimaryImageTag = GetImageCacheTag(user, image);
|
|
|
|
try
|
|
{
|
|
AttachPrimaryImageAspectRatio(dto, user);
|
|
}
|
|
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, user.Name);
|
|
}
|
|
}
|
|
|
|
return dto;
|
|
}
|
|
|
|
/// <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 SetMusicVideoProperties(BaseItemDto dto, MusicVideo item)
|
|
{
|
|
if (!string.IsNullOrEmpty(item.Album))
|
|
{
|
|
var parentAlbum = _libraryManager.RootFolder
|
|
.GetRecursiveChildren()
|
|
.Where(i => i is MusicAlbum)
|
|
.FirstOrDefault(i => string.Equals(i.Name, item.Album, StringComparison.OrdinalIgnoreCase));
|
|
|
|
if (parentAlbum != null)
|
|
{
|
|
dto.AlbumId = GetDtoId(parentAlbum);
|
|
}
|
|
}
|
|
|
|
dto.Album = item.Album;
|
|
dto.Artists = string.IsNullOrEmpty(item.Artist) ? new List<string>() : new List<string> { item.Artist };
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the backdrop image tags.
|
|
/// </summary>
|
|
/// <param name="item">The item.</param>
|
|
/// <returns>List{System.String}.</returns>
|
|
private List<string> GetBackdropImageTags(BaseItem item)
|
|
{
|
|
return GetCacheTags(item, ImageType.Backdrop).ToList();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the screenshot image tags.
|
|
/// </summary>
|
|
/// <param name="item">The item.</param>
|
|
/// <returns>List{Guid}.</returns>
|
|
private List<string> GetScreenshotImageTags(BaseItem item)
|
|
{
|
|
var hasScreenshots = item as IHasScreenshots;
|
|
if (hasScreenshots == null)
|
|
{
|
|
return new List<string>();
|
|
}
|
|
return GetCacheTags(item, ImageType.Screenshot).ToList();
|
|
}
|
|
|
|
private IEnumerable<string> GetCacheTags(BaseItem item, ImageType type)
|
|
{
|
|
return item.GetImages(type)
|
|
.Select(p => GetImageCacheTag(item, p))
|
|
.Where(i => i != null)
|
|
.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 = item.People.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();
|
|
|
|
// Attach People by transforming them into BaseItemPerson (DTO)
|
|
dto.People = new BaseItemPerson[people.Count];
|
|
|
|
var dictionary = people.Select(p => p.Name)
|
|
.Distinct(StringComparer.OrdinalIgnoreCase).Select(c =>
|
|
{
|
|
try
|
|
{
|
|
return _libraryManager.GetPerson(c);
|
|
}
|
|
catch (IOException ex)
|
|
{
|
|
_logger.ErrorException("Error getting person {0}", ex, c);
|
|
return null;
|
|
}
|
|
|
|
}).Where(i => i != null)
|
|
.DistinctBy(i => i.Name)
|
|
.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");
|
|
}
|
|
|
|
dto.People[i] = baseItemPerson;
|
|
}
|
|
}
|
|
|
|
/// <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)
|
|
.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.Parent ?? owner;
|
|
|
|
while (parent != null)
|
|
{
|
|
if (parent.GetImages(ImageType.Backdrop).Any())
|
|
{
|
|
return parent;
|
|
}
|
|
|
|
parent = parent.Parent;
|
|
}
|
|
|
|
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.Parent ?? owner;
|
|
|
|
while (parent != null)
|
|
{
|
|
if (parent.HasImage(type))
|
|
{
|
|
return parent;
|
|
}
|
|
|
|
parent = parent.Parent;
|
|
}
|
|
|
|
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>
|
|
public 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;
|
|
}
|
|
|
|
/// <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="fields">The fields.</param>
|
|
private void AttachBasicFields(BaseItemDto dto, BaseItem item, BaseItem owner, List<ItemFields> fields)
|
|
{
|
|
if (fields.Contains(ItemFields.DateCreated))
|
|
{
|
|
dto.DateCreated = item.DateCreated;
|
|
}
|
|
|
|
if (fields.Contains(ItemFields.DisplayMediaType))
|
|
{
|
|
dto.DisplayMediaType = item.DisplayMediaType;
|
|
}
|
|
|
|
dto.IsUnidentified = item.IsUnidentified;
|
|
|
|
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))
|
|
{
|
|
var hasTags = item as IHasTags;
|
|
if (hasTags != null)
|
|
{
|
|
dto.Tags = hasTags.Tags;
|
|
}
|
|
|
|
if (dto.Tags == null)
|
|
{
|
|
dto.Tags = new List<string>();
|
|
}
|
|
}
|
|
|
|
if (fields.Contains(ItemFields.Keywords))
|
|
{
|
|
var hasTags = item as IHasKeywords;
|
|
if (hasTags != null)
|
|
{
|
|
dto.Keywords = hasTags.Keywords;
|
|
}
|
|
|
|
if (dto.Keywords == null)
|
|
{
|
|
dto.Keywords = new List<string>();
|
|
}
|
|
}
|
|
|
|
if (fields.Contains(ItemFields.ProductionLocations))
|
|
{
|
|
SetProductionLocations(item, dto);
|
|
}
|
|
|
|
var hasAspectRatio = item as IHasAspectRatio;
|
|
if (hasAspectRatio != null)
|
|
{
|
|
dto.AspectRatio = hasAspectRatio.AspectRatio;
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
dto.BackdropImageTags = GetBackdropImageTags(item);
|
|
|
|
if (fields.Contains(ItemFields.ScreenshotImageTags))
|
|
{
|
|
dto.ScreenshotImageTags = GetScreenshotImageTags(item);
|
|
}
|
|
|
|
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)))
|
|
{
|
|
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;
|
|
|
|
var hasLang = item as IHasPreferredMetadataLanguage;
|
|
|
|
if (hasLang != null)
|
|
{
|
|
dto.PreferredMetadataCountryCode = hasLang.PreferredMetadataCountryCode;
|
|
dto.PreferredMetadataLanguage = hasLang.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.LocalTrailerIds.Count;
|
|
}
|
|
|
|
var hasDisplayOrder = item as IHasDisplayOrder;
|
|
if (hasDisplayOrder != null)
|
|
{
|
|
dto.DisplayOrder = hasDisplayOrder.DisplayOrder;
|
|
}
|
|
|
|
var collectionFolder = item as CollectionFolder;
|
|
if (collectionFolder != null)
|
|
{
|
|
dto.CollectionType = collectionFolder.CollectionType;
|
|
}
|
|
|
|
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))
|
|
{
|
|
// TODO: Remove this after a while, since it's been moved to the providers
|
|
if (item is MusicArtist)
|
|
{
|
|
var strippedOverview = string.IsNullOrEmpty(item.Overview) ? item.Overview : item.Overview.StripHtml();
|
|
|
|
dto.Overview = strippedOverview;
|
|
}
|
|
else
|
|
{
|
|
dto.Overview = item.Overview;
|
|
}
|
|
}
|
|
|
|
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 (dto.BackdropImageTags.Count == 0)
|
|
{
|
|
var parentWithBackdrop = GetParentBackdropItem(item, owner);
|
|
|
|
if (parentWithBackdrop != null)
|
|
{
|
|
dto.ParentBackdropItemId = GetDtoId(parentWithBackdrop);
|
|
dto.ParentBackdropImageTags = GetBackdropImageTags(parentWithBackdrop);
|
|
}
|
|
}
|
|
|
|
if (item.Parent != null && fields.Contains(ItemFields.ParentId))
|
|
{
|
|
dto.ParentId = GetDtoId(item.Parent);
|
|
}
|
|
|
|
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)
|
|
{
|
|
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)
|
|
{
|
|
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)
|
|
{
|
|
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;
|
|
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.Artists = audio.Artists;
|
|
|
|
var albumParent = audio.FindParent<MusicAlbum>();
|
|
|
|
if (albumParent != null)
|
|
{
|
|
dto.AlbumId = GetDtoId(albumParent);
|
|
|
|
dto.AlbumPrimaryImageTag = GetImageCacheTag(albumParent, ImageType.Primary);
|
|
}
|
|
|
|
dto.MediaSourceCount = 1;
|
|
}
|
|
|
|
var album = item as MusicAlbum;
|
|
|
|
if (album != null)
|
|
{
|
|
dto.Artists = album.Artists;
|
|
|
|
dto.SoundtrackIds = album.SoundtrackIds
|
|
.Select(i => i.ToString("N"))
|
|
.ToArray();
|
|
}
|
|
|
|
var hasAlbumArtist = item as IHasAlbumArtist;
|
|
|
|
if (hasAlbumArtist != null)
|
|
{
|
|
dto.AlbumArtist = hasAlbumArtist.AlbumArtists.FirstOrDefault();
|
|
}
|
|
|
|
// Add video info
|
|
var video = item as Video;
|
|
if (video != null)
|
|
{
|
|
dto.VideoType = video.VideoType;
|
|
dto.Video3DFormat = video.Video3DFormat;
|
|
dto.IsoType = video.IsoType;
|
|
dto.IsHD = video.IsHD;
|
|
|
|
dto.PartCount = video.AdditionalPartIds.Count + 1;
|
|
dto.MediaSourceCount = video.MediaSourceCount;
|
|
|
|
if (fields.Contains(ItemFields.Chapters))
|
|
{
|
|
List<ChapterInfoDto> chapters;
|
|
|
|
if (dto.MediaSources != null && dto.MediaSources.Count > 0)
|
|
{
|
|
chapters = _itemRepo.GetChapters(item.Id).Select(c => GetChapterInfoDto(c, item)).ToList();
|
|
}
|
|
else
|
|
{
|
|
chapters = _itemRepo.GetChapters(video.Id)
|
|
.Select(c => GetChapterInfoDto(c, item))
|
|
.ToList();
|
|
}
|
|
|
|
dto.Chapters = chapters;
|
|
}
|
|
}
|
|
|
|
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 = iHasMediaSources.GetMediaSources(true).First().MediaStreams;
|
|
}
|
|
|
|
dto.MediaStreams = mediaStreams;
|
|
}
|
|
}
|
|
|
|
// Add MovieInfo
|
|
var movie = item as Movie;
|
|
|
|
if (movie != null)
|
|
{
|
|
var specialFeatureCount = movie.SpecialFeatureIds.Count;
|
|
|
|
if (specialFeatureCount > 0)
|
|
{
|
|
dto.SpecialFeatureCount = specialFeatureCount;
|
|
}
|
|
|
|
if (fields.Contains(ItemFields.TmdbCollectionName))
|
|
{
|
|
dto.TmdbCollectionName = movie.TmdbCollectionName;
|
|
}
|
|
}
|
|
|
|
// Add EpisodeInfo
|
|
var episode = item as Episode;
|
|
|
|
if (episode != null)
|
|
{
|
|
dto.IndexNumberEnd = episode.IndexNumberEnd;
|
|
|
|
dto.DvdSeasonNumber = episode.DvdSeasonNumber;
|
|
dto.DvdEpisodeNumber = episode.DvdEpisodeNumber;
|
|
dto.AirsAfterSeasonNumber = episode.AirsAfterSeasonNumber;
|
|
dto.AirsBeforeEpisodeNumber = episode.AirsBeforeEpisodeNumber;
|
|
dto.AirsBeforeSeasonNumber = episode.AirsBeforeSeasonNumber;
|
|
dto.AbsoluteEpisodeNumber = episode.AbsoluteEpisodeNumber;
|
|
|
|
var seasonId = episode.SeasonId;
|
|
if (seasonId.HasValue)
|
|
{
|
|
dto.SeasonId = seasonId.Value.ToString("N");
|
|
}
|
|
}
|
|
|
|
// Add SeriesInfo
|
|
var series = item as Series;
|
|
|
|
if (series != null)
|
|
{
|
|
dto.AirDays = series.AirDays;
|
|
dto.AirTime = series.AirTime;
|
|
dto.Status = series.Status;
|
|
|
|
dto.SpecialFeatureCount = series.SpecialFeatureIds.Count;
|
|
|
|
dto.SeasonCount = series.SeasonCount;
|
|
|
|
if (fields.Contains(ItemFields.Settings))
|
|
{
|
|
dto.DisplaySpecialsWithSeasons = series.DisplaySpecialsWithSeasons;
|
|
}
|
|
|
|
dto.AnimeSeriesIndex = series.AnimeSeriesIndex;
|
|
}
|
|
|
|
if (episode != null)
|
|
{
|
|
series = episode.Series;
|
|
|
|
if (series != null)
|
|
{
|
|
dto.SeriesId = GetDtoId(series);
|
|
dto.SeriesName = series.Name;
|
|
dto.AirTime = series.AirTime;
|
|
dto.SeriesStudio = series.Studios.FirstOrDefault();
|
|
dto.SeriesThumbImageTag = GetImageCacheTag(series, ImageType.Thumb);
|
|
dto.SeriesPrimaryImageTag = GetImageCacheTag(series, ImageType.Primary);
|
|
}
|
|
}
|
|
|
|
// 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();
|
|
|
|
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 tvChannel = item as LiveTvChannel;
|
|
|
|
if (tvChannel != null)
|
|
{
|
|
dto.MediaSources = tvChannel.GetMediaSources(true).ToList();
|
|
}
|
|
|
|
var channelItem = item as IChannelItem;
|
|
|
|
if (channelItem != null)
|
|
{
|
|
dto.ChannelId = channelItem.ChannelId;
|
|
dto.ChannelName = _channelManagerFactory().GetChannel(channelItem.ChannelId).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 = _fileSystem.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>
|
|
/// <returns>Task.</returns>
|
|
private void SetSpecialCounts(Folder folder, User user, BaseItemDto dto, List<ItemFields> fields)
|
|
{
|
|
var recursiveItemCount = 0;
|
|
var unplayed = 0;
|
|
long runtime = 0;
|
|
|
|
DateTime? dateLastMediaAdded = null;
|
|
double totalPercentPlayed = 0;
|
|
|
|
IEnumerable<BaseItem> children;
|
|
|
|
var season = folder as Season;
|
|
|
|
if (season != null)
|
|
{
|
|
children = season.GetEpisodes(user).Where(i => i.LocationType != LocationType.Virtual);
|
|
}
|
|
else
|
|
{
|
|
children = folder.GetRecursiveChildren(user)
|
|
.Where(i => !i.IsFolder && i.LocationType != LocationType.Virtual);
|
|
}
|
|
|
|
// Loop through each recursive child
|
|
foreach (var child in children)
|
|
{
|
|
if (!dateLastMediaAdded.HasValue)
|
|
{
|
|
dateLastMediaAdded = child.DateCreated;
|
|
}
|
|
else
|
|
{
|
|
dateLastMediaAdded = new[] { dateLastMediaAdded.Value, child.DateCreated }.Max();
|
|
}
|
|
|
|
var userdata = _userDataRepository.GetUserData(user.Id, child.GetUserDataKey());
|
|
|
|
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++;
|
|
}
|
|
|
|
runtime += child.RunTimeTicks ?? 0;
|
|
}
|
|
|
|
dto.RecursiveItemCount = recursiveItemCount;
|
|
dto.RecursiveUnplayedItemCount = unplayed;
|
|
|
|
if (recursiveItemCount > 0)
|
|
{
|
|
dto.PlayedPercentage = totalPercentPlayed / recursiveItemCount;
|
|
}
|
|
|
|
if (runtime > 0 && fields.Contains(ItemFields.CumulativeRunTimeTicks))
|
|
{
|
|
dto.CumulativeRunTimeTicks = runtime;
|
|
}
|
|
|
|
if (fields.Contains(ItemFields.DateLastMediaAdded))
|
|
{
|
|
dto.DateLastMediaAdded = dateLastMediaAdded;
|
|
}
|
|
}
|
|
|
|
/// <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)
|
|
{
|
|
var imageInfo = item.GetImageInfo(ImageType.Primary, 0);
|
|
|
|
if (imageInfo == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
var path = imageInfo.Path;
|
|
|
|
// See if we can avoid a file system lookup by looking for the file in ResolveArgs
|
|
var dateModified = imageInfo.DateModified;
|
|
|
|
ImageSize size;
|
|
|
|
try
|
|
{
|
|
size = _imageProcessor.GetImageSize(path, dateModified);
|
|
}
|
|
catch (FileNotFoundException)
|
|
{
|
|
_logger.Error("Image file does not exist: {0}", path);
|
|
return;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.ErrorException("Failed to determine primary image aspect ratio for {0}", ex, path);
|
|
return;
|
|
}
|
|
|
|
dto.OriginalPrimaryImageAspectRatio = size.Width / size.Height;
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
dto.PrimaryImageAspectRatio = size.Width / size.Height;
|
|
}
|
|
}
|
|
}
|