LocalImageProvider.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.IO;
  5. using System.Linq;
  6. using Jellyfin.Extensions;
  7. using MediaBrowser.Controller.Entities;
  8. using MediaBrowser.Controller.Entities.Audio;
  9. using MediaBrowser.Controller.Entities.Movies;
  10. using MediaBrowser.Controller.Entities.TV;
  11. using MediaBrowser.Controller.Providers;
  12. using MediaBrowser.Model.Entities;
  13. using MediaBrowser.Model.IO;
  14. namespace MediaBrowser.LocalMetadata.Images
  15. {
  16. /// <summary>
  17. /// Local image provider.
  18. /// </summary>
  19. public class LocalImageProvider : ILocalImageProvider, IHasOrder
  20. {
  21. private static readonly string[] _commonImageFileNames =
  22. {
  23. "poster",
  24. "folder",
  25. "cover",
  26. "default"
  27. };
  28. private static readonly string[] _musicImageFileNames =
  29. {
  30. "folder",
  31. "poster",
  32. "cover",
  33. "jacket",
  34. "default",
  35. "albumart"
  36. };
  37. private static readonly string[] _personImageFileNames =
  38. {
  39. "folder",
  40. "poster"
  41. };
  42. private static readonly string[] _seriesImageFileNames =
  43. {
  44. "poster",
  45. "folder",
  46. "cover",
  47. "default",
  48. "show"
  49. };
  50. private static readonly string[] _videoImageFileNames =
  51. {
  52. "poster",
  53. "folder",
  54. "cover",
  55. "default",
  56. "movie"
  57. };
  58. private readonly IFileSystem _fileSystem;
  59. /// <summary>
  60. /// Initializes a new instance of the <see cref="LocalImageProvider"/> class.
  61. /// </summary>
  62. /// <param name="fileSystem">Instance of the <see cref="IFileSystem"/> interface.</param>
  63. public LocalImageProvider(IFileSystem fileSystem)
  64. {
  65. _fileSystem = fileSystem;
  66. }
  67. /// <inheritdoc />
  68. public string Name => "Local Images";
  69. /// <inheritdoc />
  70. public int Order => 0;
  71. /// <inheritdoc />
  72. public bool Supports(BaseItem item)
  73. {
  74. if (item.SupportsLocalMetadata)
  75. {
  76. // Episode has its own provider
  77. if (item is Episode || item is Audio || item is Photo)
  78. {
  79. return false;
  80. }
  81. return true;
  82. }
  83. if (item.LocationType == LocationType.Virtual)
  84. {
  85. var season = item as Season;
  86. var series = season?.Series;
  87. if (series is not null && series.IsFileProtocol)
  88. {
  89. return true;
  90. }
  91. }
  92. return false;
  93. }
  94. private static IEnumerable<FileSystemMetadata> GetFiles(BaseItem item, bool includeDirectories, IDirectoryService directoryService)
  95. {
  96. if (!item.IsFileProtocol)
  97. {
  98. return Enumerable.Empty<FileSystemMetadata>();
  99. }
  100. var path = item.ContainingFolderPath;
  101. // Exit if the cache dir does not exist, alternative solution is to create it, but that's a lot of empty dirs...
  102. if (!Directory.Exists(path))
  103. {
  104. return Enumerable.Empty<FileSystemMetadata>();
  105. }
  106. return directoryService.GetFileSystemEntries(path)
  107. .Where(i =>
  108. (includeDirectories && i.IsDirectory)
  109. || BaseItem.SupportedImageExtensions.Contains(i.Extension, StringComparison.OrdinalIgnoreCase))
  110. .OrderBy(i => Array.IndexOf(BaseItem.SupportedImageExtensions, i.Extension ?? string.Empty));
  111. }
  112. /// <inheritdoc />
  113. public IEnumerable<LocalImageInfo> GetImages(BaseItem item, IDirectoryService directoryService)
  114. {
  115. var files = GetFiles(item, true, directoryService).ToList();
  116. var list = new List<LocalImageInfo>();
  117. PopulateImages(item, list, files, true, directoryService);
  118. return list;
  119. }
  120. /// <summary>
  121. /// Get images for item.
  122. /// </summary>
  123. /// <param name="item">The item.</param>
  124. /// <param name="path">The images path.</param>
  125. /// <param name="directoryService">Instance of the <see cref="IDirectoryService"/> interface.</param>
  126. /// <returns>The local image info.</returns>
  127. public IEnumerable<LocalImageInfo> GetImages(BaseItem item, string path, IDirectoryService directoryService)
  128. {
  129. return GetImages(item, new[] { path }, directoryService);
  130. }
  131. /// <summary>
  132. /// Get images for item from multiple paths.
  133. /// </summary>
  134. /// <param name="item">The item.</param>
  135. /// <param name="paths">The image paths.</param>
  136. /// <param name="directoryService">Instance of the <see cref="IDirectoryService"/> interface.</param>
  137. /// <returns>The local image info.</returns>
  138. public IEnumerable<LocalImageInfo> GetImages(BaseItem item, IEnumerable<string> paths, IDirectoryService directoryService)
  139. {
  140. IEnumerable<FileSystemMetadata> files = paths.SelectMany(i => _fileSystem.GetFiles(i, BaseItem.SupportedImageExtensions, true, false));
  141. files = files
  142. .OrderBy(i => Array.IndexOf(BaseItem.SupportedImageExtensions, i.Extension ?? string.Empty));
  143. var list = new List<LocalImageInfo>();
  144. PopulateImages(item, list, files.ToList(), false, directoryService);
  145. return list;
  146. }
  147. private void PopulateImages(BaseItem item, List<LocalImageInfo> images, List<FileSystemMetadata> files, bool supportParentSeriesFiles, IDirectoryService directoryService)
  148. {
  149. if (supportParentSeriesFiles)
  150. {
  151. if (item is Season season)
  152. {
  153. PopulateSeasonImagesFromSeriesFolder(season, images, directoryService);
  154. }
  155. }
  156. var imagePrefix = item.FileNameWithoutExtension + "-";
  157. var isInMixedFolder = item.IsInMixedFolder;
  158. PopulatePrimaryImages(item, images, files, imagePrefix, isInMixedFolder);
  159. var added = false;
  160. var isEpisode = item is Episode;
  161. var isSong = item.GetType() == typeof(Audio);
  162. var isPerson = item is Person;
  163. // Logo
  164. if (!isEpisode && !isSong && !isPerson)
  165. {
  166. added = AddImage(files, images, "logo", imagePrefix, isInMixedFolder, ImageType.Logo);
  167. if (!added)
  168. {
  169. AddImage(files, images, "clearlogo", imagePrefix, isInMixedFolder, ImageType.Logo);
  170. }
  171. }
  172. // Art
  173. if (!isEpisode && !isSong && !isPerson)
  174. {
  175. AddImage(files, images, "clearart", imagePrefix, isInMixedFolder, ImageType.Art);
  176. }
  177. // For music albums, prefer cdart before disc
  178. if (item is MusicAlbum)
  179. {
  180. added = AddImage(files, images, "cdart", imagePrefix, isInMixedFolder, ImageType.Disc);
  181. if (!added)
  182. {
  183. AddImage(files, images, "disc", imagePrefix, isInMixedFolder, ImageType.Disc);
  184. }
  185. }
  186. else if (item is Video || item is BoxSet)
  187. {
  188. added = AddImage(files, images, "disc", imagePrefix, isInMixedFolder, ImageType.Disc);
  189. if (!added)
  190. {
  191. added = AddImage(files, images, "cdart", imagePrefix, isInMixedFolder, ImageType.Disc);
  192. }
  193. if (!added)
  194. {
  195. AddImage(files, images, "discart", imagePrefix, isInMixedFolder, ImageType.Disc);
  196. }
  197. }
  198. // Banner
  199. if (!isEpisode && !isSong && !isPerson)
  200. {
  201. AddImage(files, images, "banner", imagePrefix, isInMixedFolder, ImageType.Banner);
  202. }
  203. // Thumb
  204. if (!isEpisode && !isSong && !isPerson)
  205. {
  206. added = AddImage(files, images, "landscape", imagePrefix, isInMixedFolder, ImageType.Thumb);
  207. if (!added)
  208. {
  209. AddImage(files, images, "thumb", imagePrefix, isInMixedFolder, ImageType.Thumb);
  210. }
  211. }
  212. if (!isEpisode && !isSong && !isPerson)
  213. {
  214. PopulateBackdrops(item, images, files, imagePrefix, isInMixedFolder);
  215. }
  216. }
  217. private void PopulatePrimaryImages(BaseItem item, List<LocalImageInfo> images, List<FileSystemMetadata> files, string imagePrefix, bool isInMixedFolder)
  218. {
  219. string[] imageFileNames;
  220. if (item is MusicAlbum || item is MusicArtist || item is PhotoAlbum)
  221. {
  222. // these prefer folder
  223. imageFileNames = _musicImageFileNames;
  224. }
  225. else if (item is Person)
  226. {
  227. // these prefer folder
  228. imageFileNames = _personImageFileNames;
  229. }
  230. else if (item is Series)
  231. {
  232. imageFileNames = _seriesImageFileNames;
  233. }
  234. else if (item is Video && item is not Episode)
  235. {
  236. imageFileNames = _videoImageFileNames;
  237. }
  238. else
  239. {
  240. imageFileNames = _commonImageFileNames;
  241. }
  242. var fileNameWithoutExtension = item.FileNameWithoutExtension;
  243. if (!string.IsNullOrEmpty(fileNameWithoutExtension))
  244. {
  245. if (AddImage(files, images, fileNameWithoutExtension, ImageType.Primary))
  246. {
  247. return;
  248. }
  249. }
  250. foreach (var name in imageFileNames)
  251. {
  252. if (AddImage(files, images, name, ImageType.Primary, imagePrefix))
  253. {
  254. return;
  255. }
  256. }
  257. if (!isInMixedFolder)
  258. {
  259. foreach (var name in imageFileNames)
  260. {
  261. if (AddImage(files, images, name, ImageType.Primary))
  262. {
  263. return;
  264. }
  265. }
  266. }
  267. }
  268. private void PopulateBackdrops(BaseItem item, List<LocalImageInfo> images, List<FileSystemMetadata> files, string imagePrefix, bool isInMixedFolder)
  269. {
  270. if (!string.IsNullOrEmpty(item.Path))
  271. {
  272. var name = item.FileNameWithoutExtension;
  273. if (!string.IsNullOrEmpty(name))
  274. {
  275. AddImage(files, images, name + "-fanart", ImageType.Backdrop, imagePrefix);
  276. // Support without the prefix if it's in its own folder
  277. if (!isInMixedFolder)
  278. {
  279. AddImage(files, images, name + "-fanart", ImageType.Backdrop);
  280. }
  281. }
  282. }
  283. PopulateBackdrops(images, files, imagePrefix, "fanart", "fanart-", isInMixedFolder, ImageType.Backdrop);
  284. PopulateBackdrops(images, files, imagePrefix, "background", "background-", isInMixedFolder, ImageType.Backdrop);
  285. PopulateBackdrops(images, files, imagePrefix, "art", "art-", isInMixedFolder, ImageType.Backdrop);
  286. var extraFanartFolder = files
  287. .FirstOrDefault(i => string.Equals(i.Name, "extrafanart", StringComparison.OrdinalIgnoreCase));
  288. if (extraFanartFolder is not null)
  289. {
  290. PopulateBackdropsFromExtraFanart(extraFanartFolder.FullName, images);
  291. }
  292. PopulateBackdrops(images, files, imagePrefix, "backdrop", "backdrop", isInMixedFolder, ImageType.Backdrop);
  293. }
  294. private void PopulateBackdropsFromExtraFanart(string path, List<LocalImageInfo> images)
  295. {
  296. var imageFiles = _fileSystem.GetFiles(path, BaseItem.SupportedImageExtensions, false, false);
  297. images.AddRange(imageFiles.Where(i => i.Length > 0).Select(i => new LocalImageInfo
  298. {
  299. FileInfo = i,
  300. Type = ImageType.Backdrop
  301. }));
  302. }
  303. private void PopulateBackdrops(List<LocalImageInfo> images, List<FileSystemMetadata> files, string imagePrefix, string firstFileName, string subsequentFileNamePrefix, bool isInMixedFolder, ImageType type)
  304. {
  305. AddImage(files, images, imagePrefix + firstFileName, type);
  306. var unfound = 0;
  307. for (var i = 1; i <= 20; i++)
  308. {
  309. // Screenshot Image
  310. var found = AddImage(files, images, imagePrefix + subsequentFileNamePrefix + i, type);
  311. if (!found)
  312. {
  313. unfound++;
  314. if (unfound >= 3)
  315. {
  316. break;
  317. }
  318. }
  319. }
  320. // Support without the prefix
  321. if (!isInMixedFolder)
  322. {
  323. AddImage(files, images, firstFileName, type);
  324. unfound = 0;
  325. for (var i = 1; i <= 20; i++)
  326. {
  327. // Screenshot Image
  328. var found = AddImage(files, images, subsequentFileNamePrefix + i, type);
  329. if (!found)
  330. {
  331. unfound++;
  332. if (unfound >= 3)
  333. {
  334. break;
  335. }
  336. }
  337. }
  338. }
  339. }
  340. private void PopulateSeasonImagesFromSeriesFolder(Season season, List<LocalImageInfo> images, IDirectoryService directoryService)
  341. {
  342. var seasonNumber = season.IndexNumber;
  343. var series = season.Series;
  344. if (!seasonNumber.HasValue || !series.IsFileProtocol)
  345. {
  346. return;
  347. }
  348. var seriesFiles = GetFiles(series, false, directoryService).ToList();
  349. // Try using the season name
  350. var prefix = season.Name.Replace(" ", string.Empty, StringComparison.Ordinal).ToLowerInvariant();
  351. var filenamePrefixes = new List<string> { prefix };
  352. var seasonMarker = seasonNumber.Value == 0
  353. ? "-specials"
  354. : seasonNumber.Value.ToString("00", CultureInfo.InvariantCulture);
  355. // Get this one directly from the file system since we have to go up a level
  356. if (!string.Equals(prefix, seasonMarker, StringComparison.OrdinalIgnoreCase))
  357. {
  358. filenamePrefixes.Add("season" + seasonMarker);
  359. }
  360. foreach (var filename in filenamePrefixes)
  361. {
  362. AddImage(seriesFiles, images, filename + "-poster", ImageType.Primary);
  363. AddImage(seriesFiles, images, filename + "-fanart", ImageType.Backdrop);
  364. AddImage(seriesFiles, images, filename + "-banner", ImageType.Banner);
  365. AddImage(seriesFiles, images, filename + "-landscape", ImageType.Thumb);
  366. }
  367. }
  368. private bool AddImage(List<FileSystemMetadata> files, List<LocalImageInfo> images, string name, string imagePrefix, bool isInMixedFolder, ImageType type)
  369. {
  370. var added = AddImage(files, images, name, type, imagePrefix);
  371. if (!isInMixedFolder)
  372. {
  373. if (AddImage(files, images, name, type))
  374. {
  375. added = true;
  376. }
  377. }
  378. return added;
  379. }
  380. private static bool AddImage(IReadOnlyList<FileSystemMetadata> files, List<LocalImageInfo> images, string name, ImageType type, string? prefix = null)
  381. {
  382. var image = GetImage(files, name, prefix);
  383. if (image is null)
  384. {
  385. return false;
  386. }
  387. images.Add(new LocalImageInfo
  388. {
  389. FileInfo = image,
  390. Type = type
  391. });
  392. return true;
  393. }
  394. private static FileSystemMetadata? GetImage(IReadOnlyList<FileSystemMetadata> files, string name, string? prefix = null)
  395. {
  396. var fileNameLength = name.Length + (prefix?.Length ?? 0);
  397. for (var i = 0; i < files.Count; i++)
  398. {
  399. var file = files[i];
  400. if (file.IsDirectory || file.Length <= 0)
  401. {
  402. continue;
  403. }
  404. var fileName = Path.GetFileNameWithoutExtension(file.FullName.AsSpan());
  405. if (fileName.Length == fileNameLength
  406. && fileName.StartsWith(prefix, StringComparison.OrdinalIgnoreCase)
  407. && fileName.EndsWith(name, StringComparison.OrdinalIgnoreCase))
  408. {
  409. return file;
  410. }
  411. }
  412. return null;
  413. }
  414. }
  415. }