using MediaBrowser.Controller.Providers; using System; using System.Collections.Generic; using System.Linq; using System.Runtime.Serialization; using MediaBrowser.Common.Extensions; using MediaBrowser.Controller.Extensions; using MediaBrowser.Model.Entities; using MediaBrowser.Model.Extensions; namespace MediaBrowser.Controller.Entities { /// /// This is the full Person object that can be retrieved with all of it's data. /// public class Person : BaseItem, IItemByName, IHasLookupInfo { /// /// Gets or sets the place of birth. /// /// The place of birth. public string PlaceOfBirth { get; set; } public override List GetUserDataKeys() { var list = base.GetUserDataKeys(); list.Insert(0, GetType().Name + "-" + (Name ?? string.Empty).RemoveDiacritics()); return list; } public override string CreatePresentationUniqueKey() { return GetUserDataKeys()[0]; } public PersonLookupInfo GetLookupInfo() { return GetItemLookupInfo(); } public IEnumerable GetTaggedItems(InternalItemsQuery query) { query.Person = Name; return LibraryManager.GetItemList(query); } /// /// Returns the folder containing the item. /// If the item is a folder, it returns the folder itself /// /// The containing folder path. [IgnoreDataMember] public override string ContainingFolderPath { get { return Path; } } public override bool CanDelete() { return false; } public override bool IsSaveLocalMetadataEnabled() { return true; } [IgnoreDataMember] public override bool EnableAlphaNumericSorting { get { return false; } } /// /// Gets a value indicating whether this instance is owned item. /// /// true if this instance is owned item; otherwise, false. [IgnoreDataMember] public override bool IsOwnedItem { get { return false; } } public IEnumerable GetTaggedItems(IEnumerable inputItems) { var itemsWithPerson = LibraryManager.GetItemIds(new InternalItemsQuery { Person = Name }); return inputItems.Where(i => itemsWithPerson.Contains(i.Id)); } public Func GetItemFilter() { return i => LibraryManager.GetPeople(i).Any(p => string.Equals(p.Name, Name, StringComparison.OrdinalIgnoreCase)); } [IgnoreDataMember] public override bool SupportsPeople { get { return false; } } [IgnoreDataMember] public override bool SupportsAncestors { get { return false; } } public static string GetPath(string name, bool normalizeName = true) { // Trim the period at the end because windows will have a hard time with that var validFilename = normalizeName ? FileSystem.GetValidFilename(name).Trim().TrimEnd('.') : name; string subFolderPrefix = null; foreach (char c in validFilename) { if (char.IsLetterOrDigit(c)) { subFolderPrefix = c.ToString(); break; } } var path = ConfigurationManager.ApplicationPaths.PeoplePath; return string.IsNullOrEmpty(subFolderPrefix) ? System.IO.Path.Combine(path, validFilename) : System.IO.Path.Combine(path, subFolderPrefix, validFilename); } private string GetRebasedPath() { return GetPath(System.IO.Path.GetFileName(Path), false); } public override bool RequiresRefresh() { var newPath = GetRebasedPath(); if (!string.Equals(Path, newPath, StringComparison.Ordinal)) { Logger.Debug("{0} path has changed from {1} to {2}", GetType().Name, Path, newPath); return true; } return base.RequiresRefresh(); } /// /// This is called before any metadata refresh and returns true or false indicating if changes were made /// public override bool BeforeMetadataRefresh() { var hasChanges = base.BeforeMetadataRefresh(); var newPath = GetRebasedPath(); if (!string.Equals(Path, newPath, StringComparison.Ordinal)) { Path = newPath; hasChanges = true; } return hasChanges; } } /// /// This is the small Person stub that is attached to BaseItems /// public class PersonInfo : IHasProviderIds { public PersonInfo() { ProviderIds = new Dictionary(StringComparer.OrdinalIgnoreCase); } public Guid ItemId { get; set; } /// /// Gets or sets the name. /// /// The name. public string Name { get; set; } /// /// Gets or sets the role. /// /// The role. public string Role { get; set; } /// /// Gets or sets the type. /// /// The type. public string Type { get; set; } /// /// Gets or sets the sort order - ascending /// /// The sort order. public int? SortOrder { get; set; } public string ImageUrl { get; set; } public Dictionary ProviderIds { get; set; } /// /// Returns a that represents this instance. /// /// A that represents this instance. public override string ToString() { return Name; } public bool IsType(string type) { return string.Equals(Type, type, StringComparison.OrdinalIgnoreCase) || string.Equals(Role, type, StringComparison.OrdinalIgnoreCase); } } }