MusicArtist.cs 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. using MediaBrowser.Controller.Library;
  2. using MediaBrowser.Controller.Providers;
  3. using MediaBrowser.Model.Configuration;
  4. using MediaBrowser.Model.Entities;
  5. using MediaBrowser.Model.Users;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Runtime.Serialization;
  10. using System.Threading;
  11. using System.Threading.Tasks;
  12. using MediaBrowser.Common.Extensions;
  13. namespace MediaBrowser.Controller.Entities.Audio
  14. {
  15. /// <summary>
  16. /// Class MusicArtist
  17. /// </summary>
  18. public class MusicArtist : Folder, IMetadataContainer, IItemByName, IHasMusicGenres, IHasDualAccess, IHasProductionLocations, IHasLookupInfo<ArtistInfo>
  19. {
  20. [IgnoreDataMember]
  21. public bool IsAccessedByName
  22. {
  23. get { return ParentId == Guid.Empty; }
  24. }
  25. public List<string> ProductionLocations { get; set; }
  26. [IgnoreDataMember]
  27. public override bool IsFolder
  28. {
  29. get
  30. {
  31. return !IsAccessedByName;
  32. }
  33. }
  34. [IgnoreDataMember]
  35. public override bool SupportsCumulativeRunTimeTicks
  36. {
  37. get
  38. {
  39. return true;
  40. }
  41. }
  42. [IgnoreDataMember]
  43. public override bool SupportsAddingToPlaylist
  44. {
  45. get { return true; }
  46. }
  47. public override bool CanDelete()
  48. {
  49. return !IsAccessedByName;
  50. }
  51. public IEnumerable<BaseItem> GetTaggedItems(InternalItemsQuery query)
  52. {
  53. if (query.IncludeItemTypes.Length == 0)
  54. {
  55. query.IncludeItemTypes = new[] { typeof(Audio).Name, typeof(MusicVideo).Name, typeof(MusicAlbum).Name };
  56. query.ArtistNames = new[] { Name };
  57. }
  58. return LibraryManager.GetItemList(query);
  59. }
  60. [IgnoreDataMember]
  61. protected override IEnumerable<BaseItem> ActualChildren
  62. {
  63. get
  64. {
  65. if (IsAccessedByName)
  66. {
  67. return new List<BaseItem>();
  68. }
  69. return base.ActualChildren;
  70. }
  71. }
  72. public override int GetChildCount(User user)
  73. {
  74. if (IsAccessedByName)
  75. {
  76. return 0;
  77. }
  78. return base.GetChildCount(user);
  79. }
  80. public override bool IsSaveLocalMetadataEnabled()
  81. {
  82. if (IsAccessedByName)
  83. {
  84. return true;
  85. }
  86. return base.IsSaveLocalMetadataEnabled();
  87. }
  88. private readonly Task _cachedTask = Task.FromResult(true);
  89. protected override Task ValidateChildrenInternal(IProgress<double> progress, CancellationToken cancellationToken, bool recursive, bool refreshChildMetadata, MetadataRefreshOptions refreshOptions, IDirectoryService directoryService)
  90. {
  91. if (IsAccessedByName)
  92. {
  93. // Should never get in here anyway
  94. return _cachedTask;
  95. }
  96. return base.ValidateChildrenInternal(progress, cancellationToken, recursive, refreshChildMetadata, refreshOptions, directoryService);
  97. }
  98. public MusicArtist()
  99. {
  100. ProductionLocations = new List<string>();
  101. }
  102. public override List<string> GetUserDataKeys()
  103. {
  104. var list = base.GetUserDataKeys();
  105. list.InsertRange(0, GetUserDataKeys(this));
  106. return list;
  107. }
  108. /// <summary>
  109. /// Returns the folder containing the item.
  110. /// If the item is a folder, it returns the folder itself
  111. /// </summary>
  112. /// <value>The containing folder path.</value>
  113. [IgnoreDataMember]
  114. public override string ContainingFolderPath
  115. {
  116. get
  117. {
  118. return Path;
  119. }
  120. }
  121. /// <summary>
  122. /// Gets a value indicating whether this instance is owned item.
  123. /// </summary>
  124. /// <value><c>true</c> if this instance is owned item; otherwise, <c>false</c>.</value>
  125. [IgnoreDataMember]
  126. public override bool IsOwnedItem
  127. {
  128. get
  129. {
  130. return false;
  131. }
  132. }
  133. /// <summary>
  134. /// Gets the user data key.
  135. /// </summary>
  136. /// <param name="item">The item.</param>
  137. /// <returns>System.String.</returns>
  138. private static List<string> GetUserDataKeys(MusicArtist item)
  139. {
  140. var list = new List<string>();
  141. var id = item.GetProviderId(MetadataProviders.MusicBrainzArtist);
  142. if (!string.IsNullOrEmpty(id))
  143. {
  144. list.Add("Artist-Musicbrainz-" + id);
  145. }
  146. list.Add("Artist-" + (item.Name ?? string.Empty).RemoveDiacritics());
  147. return list;
  148. }
  149. public override string CreatePresentationUniqueKey()
  150. {
  151. return "Artist-" + (Name ?? string.Empty).RemoveDiacritics();
  152. }
  153. protected override bool GetBlockUnratedValue(UserPolicy config)
  154. {
  155. return config.BlockUnratedItems.Contains(UnratedItem.Music);
  156. }
  157. public override UnratedItem GetBlockUnratedType()
  158. {
  159. return UnratedItem.Music;
  160. }
  161. public async Task RefreshAllMetadata(MetadataRefreshOptions refreshOptions, IProgress<double> progress, CancellationToken cancellationToken)
  162. {
  163. var items = GetRecursiveChildren().ToList();
  164. var songs = items.OfType<Audio>().ToList();
  165. var others = items.Except(songs).ToList();
  166. var totalItems = songs.Count + others.Count;
  167. var numComplete = 0;
  168. var childUpdateType = ItemUpdateType.None;
  169. // Refresh songs
  170. foreach (var item in songs)
  171. {
  172. cancellationToken.ThrowIfCancellationRequested();
  173. var updateType = await item.RefreshMetadata(refreshOptions, cancellationToken).ConfigureAwait(false);
  174. childUpdateType = childUpdateType | updateType;
  175. numComplete++;
  176. double percent = numComplete;
  177. percent /= totalItems;
  178. progress.Report(percent * 100);
  179. }
  180. var parentRefreshOptions = refreshOptions;
  181. if (childUpdateType > ItemUpdateType.None)
  182. {
  183. parentRefreshOptions = new MetadataRefreshOptions(refreshOptions);
  184. parentRefreshOptions.MetadataRefreshMode = MetadataRefreshMode.FullRefresh;
  185. }
  186. // Refresh current item
  187. await RefreshMetadata(parentRefreshOptions, cancellationToken).ConfigureAwait(false);
  188. // Refresh all non-songs
  189. foreach (var item in others)
  190. {
  191. cancellationToken.ThrowIfCancellationRequested();
  192. var updateType = await item.RefreshMetadata(parentRefreshOptions, cancellationToken).ConfigureAwait(false);
  193. numComplete++;
  194. double percent = numComplete;
  195. percent /= totalItems;
  196. progress.Report(percent * 100);
  197. }
  198. progress.Report(100);
  199. }
  200. public ArtistInfo GetLookupInfo()
  201. {
  202. var info = GetItemLookupInfo<ArtistInfo>();
  203. info.SongInfos = GetRecursiveChildren(i => i is Audio)
  204. .Cast<Audio>()
  205. .Select(i => i.GetLookupInfo())
  206. .ToList();
  207. return info;
  208. }
  209. public IEnumerable<BaseItem> GetTaggedItems(IEnumerable<BaseItem> inputItems)
  210. {
  211. return inputItems.Where(GetItemFilter());
  212. }
  213. public Func<BaseItem, bool> GetItemFilter()
  214. {
  215. return i =>
  216. {
  217. var hasArtist = i as IHasArtist;
  218. return hasArtist != null && hasArtist.HasAnyArtist(Name);
  219. };
  220. }
  221. [IgnoreDataMember]
  222. public override bool SupportsPeople
  223. {
  224. get
  225. {
  226. return false;
  227. }
  228. }
  229. public static string GetPath(string name, bool normalizeName = true)
  230. {
  231. // Trim the period at the end because windows will have a hard time with that
  232. var validName = normalizeName ?
  233. FileSystem.GetValidFilename(name).Trim().TrimEnd('.') :
  234. name;
  235. return System.IO.Path.Combine(ConfigurationManager.ApplicationPaths.ArtistsPath, validName);
  236. }
  237. private string GetRebasedPath()
  238. {
  239. return GetPath(System.IO.Path.GetFileName(Path), false);
  240. }
  241. public override bool RequiresRefresh()
  242. {
  243. if (IsAccessedByName)
  244. {
  245. var newPath = GetRebasedPath();
  246. if (!string.Equals(Path, newPath, StringComparison.Ordinal))
  247. {
  248. Logger.Debug("{0} path has changed from {1} to {2}", GetType().Name, Path, newPath);
  249. return true;
  250. }
  251. }
  252. return base.RequiresRefresh();
  253. }
  254. /// <summary>
  255. /// This is called before any metadata refresh and returns true or false indicating if changes were made
  256. /// </summary>
  257. public override bool BeforeMetadataRefresh()
  258. {
  259. var hasChanges = base.BeforeMetadataRefresh();
  260. if (IsAccessedByName)
  261. {
  262. var newPath = GetRebasedPath();
  263. if (!string.Equals(Path, newPath, StringComparison.Ordinal))
  264. {
  265. Path = newPath;
  266. hasChanges = true;
  267. }
  268. }
  269. return hasChanges;
  270. }
  271. }
  272. }