123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495 |
- using System;
- using System.Collections.Generic;
- using System.Globalization;
- using System.IO;
- using System.Linq;
- using Jellyfin.Extensions;
- using MediaBrowser.Controller.Entities;
- using MediaBrowser.Controller.Entities.Audio;
- using MediaBrowser.Controller.Entities.Movies;
- using MediaBrowser.Controller.Entities.TV;
- using MediaBrowser.Controller.Providers;
- using MediaBrowser.Model.Entities;
- using MediaBrowser.Model.IO;
- namespace MediaBrowser.LocalMetadata.Images
- {
- /// <summary>
- /// Local image provider.
- /// </summary>
- public class LocalImageProvider : ILocalImageProvider, IHasOrder
- {
- private static readonly string[] _commonImageFileNames =
- {
- "poster",
- "folder",
- "cover",
- "default"
- };
- private static readonly string[] _musicImageFileNames =
- {
- "folder",
- "poster",
- "cover",
- "jacket",
- "default",
- "albumart"
- };
- private static readonly string[] _personImageFileNames =
- {
- "folder",
- "poster"
- };
- private static readonly string[] _seriesImageFileNames =
- {
- "poster",
- "folder",
- "cover",
- "default",
- "show"
- };
- private static readonly string[] _videoImageFileNames =
- {
- "poster",
- "folder",
- "cover",
- "default",
- "movie"
- };
- private readonly IFileSystem _fileSystem;
- /// <summary>
- /// Initializes a new instance of the <see cref="LocalImageProvider"/> class.
- /// </summary>
- /// <param name="fileSystem">Instance of the <see cref="IFileSystem"/> interface.</param>
- public LocalImageProvider(IFileSystem fileSystem)
- {
- _fileSystem = fileSystem;
- }
- /// <inheritdoc />
- public string Name => "Local Images";
- /// <inheritdoc />
- public int Order => 0;
- /// <inheritdoc />
- public bool Supports(BaseItem item)
- {
- if (item.SupportsLocalMetadata)
- {
- // Episode has its own provider
- if (item is Episode || item is Audio || item is Photo)
- {
- return false;
- }
- return true;
- }
- if (item.LocationType == LocationType.Virtual)
- {
- var season = item as Season;
- var series = season?.Series;
- if (series is not null && series.IsFileProtocol)
- {
- return true;
- }
- }
- return false;
- }
- private static IEnumerable<FileSystemMetadata> GetFiles(BaseItem item, bool includeDirectories, IDirectoryService directoryService)
- {
- if (!item.IsFileProtocol)
- {
- return Enumerable.Empty<FileSystemMetadata>();
- }
- var path = item.ContainingFolderPath;
- // Exit if the cache dir does not exist, alternative solution is to create it, but that's a lot of empty dirs...
- if (!Directory.Exists(path))
- {
- return Enumerable.Empty<FileSystemMetadata>();
- }
- return directoryService.GetFileSystemEntries(path)
- .Where(i =>
- (includeDirectories && i.IsDirectory)
- || BaseItem.SupportedImageExtensions.Contains(i.Extension, StringComparison.OrdinalIgnoreCase))
- .OrderBy(i => Array.IndexOf(BaseItem.SupportedImageExtensions, i.Extension ?? string.Empty));
- }
- /// <inheritdoc />
- public IEnumerable<LocalImageInfo> GetImages(BaseItem item, IDirectoryService directoryService)
- {
- var files = GetFiles(item, true, directoryService).ToList();
- var list = new List<LocalImageInfo>();
- PopulateImages(item, list, files, true, directoryService);
- return list;
- }
- /// <summary>
- /// Get images for item.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <param name="path">The images path.</param>
- /// <param name="directoryService">Instance of the <see cref="IDirectoryService"/> interface.</param>
- /// <returns>The local image info.</returns>
- public IEnumerable<LocalImageInfo> GetImages(BaseItem item, string path, IDirectoryService directoryService)
- {
- return GetImages(item, new[] { path }, directoryService);
- }
- /// <summary>
- /// Get images for item from multiple paths.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <param name="paths">The image paths.</param>
- /// <param name="directoryService">Instance of the <see cref="IDirectoryService"/> interface.</param>
- /// <returns>The local image info.</returns>
- public IEnumerable<LocalImageInfo> GetImages(BaseItem item, IEnumerable<string> paths, IDirectoryService directoryService)
- {
- IEnumerable<FileSystemMetadata> files = paths.SelectMany(i => _fileSystem.GetFiles(i, BaseItem.SupportedImageExtensions, true, false));
- files = files
- .OrderBy(i => Array.IndexOf(BaseItem.SupportedImageExtensions, i.Extension ?? string.Empty));
- var list = new List<LocalImageInfo>();
- PopulateImages(item, list, files.ToList(), false, directoryService);
- return list;
- }
- private void PopulateImages(BaseItem item, List<LocalImageInfo> images, List<FileSystemMetadata> files, bool supportParentSeriesFiles, IDirectoryService directoryService)
- {
- if (supportParentSeriesFiles)
- {
- if (item is Season season)
- {
- PopulateSeasonImagesFromSeriesFolder(season, images, directoryService);
- }
- }
- var imagePrefix = item.FileNameWithoutExtension + "-";
- var isInMixedFolder = item.IsInMixedFolder;
- PopulatePrimaryImages(item, images, files, imagePrefix, isInMixedFolder);
- var added = false;
- var isEpisode = item is Episode;
- var isSong = item.GetType() == typeof(Audio);
- var isPerson = item is Person;
- // Logo
- if (!isEpisode && !isSong && !isPerson)
- {
- added = AddImage(files, images, "logo", imagePrefix, isInMixedFolder, ImageType.Logo);
- if (!added)
- {
- AddImage(files, images, "clearlogo", imagePrefix, isInMixedFolder, ImageType.Logo);
- }
- }
- // Art
- if (!isEpisode && !isSong && !isPerson)
- {
- AddImage(files, images, "clearart", imagePrefix, isInMixedFolder, ImageType.Art);
- }
- // For music albums, prefer cdart before disc
- if (item is MusicAlbum)
- {
- added = AddImage(files, images, "cdart", imagePrefix, isInMixedFolder, ImageType.Disc);
- if (!added)
- {
- AddImage(files, images, "disc", imagePrefix, isInMixedFolder, ImageType.Disc);
- }
- }
- else if (item is Video || item is BoxSet)
- {
- added = AddImage(files, images, "disc", imagePrefix, isInMixedFolder, ImageType.Disc);
- if (!added)
- {
- added = AddImage(files, images, "cdart", imagePrefix, isInMixedFolder, ImageType.Disc);
- }
- if (!added)
- {
- AddImage(files, images, "discart", imagePrefix, isInMixedFolder, ImageType.Disc);
- }
- }
- // Banner
- if (!isEpisode && !isSong && !isPerson)
- {
- AddImage(files, images, "banner", imagePrefix, isInMixedFolder, ImageType.Banner);
- }
- // Thumb
- if (!isEpisode && !isSong && !isPerson)
- {
- added = AddImage(files, images, "landscape", imagePrefix, isInMixedFolder, ImageType.Thumb);
- if (!added)
- {
- AddImage(files, images, "thumb", imagePrefix, isInMixedFolder, ImageType.Thumb);
- }
- }
- if (!isEpisode && !isSong && !isPerson)
- {
- PopulateBackdrops(item, images, files, imagePrefix, isInMixedFolder);
- }
- }
- private void PopulatePrimaryImages(BaseItem item, List<LocalImageInfo> images, List<FileSystemMetadata> files, string imagePrefix, bool isInMixedFolder)
- {
- string[] imageFileNames;
- if (item is MusicAlbum || item is MusicArtist || item is PhotoAlbum)
- {
- // these prefer folder
- imageFileNames = _musicImageFileNames;
- }
- else if (item is Person)
- {
- // these prefer folder
- imageFileNames = _personImageFileNames;
- }
- else if (item is Series)
- {
- imageFileNames = _seriesImageFileNames;
- }
- else if (item is Video && item is not Episode)
- {
- imageFileNames = _videoImageFileNames;
- }
- else
- {
- imageFileNames = _commonImageFileNames;
- }
- var fileNameWithoutExtension = item.FileNameWithoutExtension;
- if (!string.IsNullOrEmpty(fileNameWithoutExtension))
- {
- if (AddImage(files, images, fileNameWithoutExtension, ImageType.Primary))
- {
- return;
- }
- }
- foreach (var name in imageFileNames)
- {
- if (AddImage(files, images, name, ImageType.Primary, imagePrefix))
- {
- return;
- }
- }
- if (!isInMixedFolder)
- {
- foreach (var name in imageFileNames)
- {
- if (AddImage(files, images, name, ImageType.Primary))
- {
- return;
- }
- }
- }
- }
- private void PopulateBackdrops(BaseItem item, List<LocalImageInfo> images, List<FileSystemMetadata> files, string imagePrefix, bool isInMixedFolder)
- {
- if (!string.IsNullOrEmpty(item.Path))
- {
- var name = item.FileNameWithoutExtension;
- if (!string.IsNullOrEmpty(name))
- {
- AddImage(files, images, name + "-fanart", ImageType.Backdrop, imagePrefix);
- // Support without the prefix if it's in its own folder
- if (!isInMixedFolder)
- {
- AddImage(files, images, name + "-fanart", ImageType.Backdrop);
- }
- }
- }
- PopulateBackdrops(images, files, imagePrefix, "fanart", "fanart-", isInMixedFolder, ImageType.Backdrop);
- PopulateBackdrops(images, files, imagePrefix, "background", "background-", isInMixedFolder, ImageType.Backdrop);
- PopulateBackdrops(images, files, imagePrefix, "art", "art-", isInMixedFolder, ImageType.Backdrop);
- var extraFanartFolder = files
- .FirstOrDefault(i => string.Equals(i.Name, "extrafanart", StringComparison.OrdinalIgnoreCase));
- if (extraFanartFolder is not null)
- {
- PopulateBackdropsFromExtraFanart(extraFanartFolder.FullName, images);
- }
- PopulateBackdrops(images, files, imagePrefix, "backdrop", "backdrop", isInMixedFolder, ImageType.Backdrop);
- }
- private void PopulateBackdropsFromExtraFanart(string path, List<LocalImageInfo> images)
- {
- var imageFiles = _fileSystem.GetFiles(path, BaseItem.SupportedImageExtensions, false, false);
- images.AddRange(imageFiles.Where(i => i.Length > 0).Select(i => new LocalImageInfo
- {
- FileInfo = i,
- Type = ImageType.Backdrop
- }));
- }
- private void PopulateBackdrops(List<LocalImageInfo> images, List<FileSystemMetadata> files, string imagePrefix, string firstFileName, string subsequentFileNamePrefix, bool isInMixedFolder, ImageType type)
- {
- AddImage(files, images, imagePrefix + firstFileName, type);
- var unfound = 0;
- for (var i = 1; i <= 20; i++)
- {
- // Screenshot Image
- var found = AddImage(files, images, imagePrefix + subsequentFileNamePrefix + i, type);
- if (!found)
- {
- unfound++;
- if (unfound >= 3)
- {
- break;
- }
- }
- }
- // Support without the prefix
- if (!isInMixedFolder)
- {
- AddImage(files, images, firstFileName, type);
- unfound = 0;
- for (var i = 1; i <= 20; i++)
- {
- // Screenshot Image
- var found = AddImage(files, images, subsequentFileNamePrefix + i, type);
- if (!found)
- {
- unfound++;
- if (unfound >= 3)
- {
- break;
- }
- }
- }
- }
- }
- private void PopulateSeasonImagesFromSeriesFolder(Season season, List<LocalImageInfo> images, IDirectoryService directoryService)
- {
- var seasonNumber = season.IndexNumber;
- var series = season.Series;
- if (!seasonNumber.HasValue || !series.IsFileProtocol)
- {
- return;
- }
- var seriesFiles = GetFiles(series, false, directoryService).ToList();
- // Try using the season name
- var prefix = season.Name.Replace(" ", string.Empty, StringComparison.Ordinal).ToLowerInvariant();
- var filenamePrefixes = new List<string> { prefix };
- var seasonMarker = seasonNumber.Value == 0
- ? "-specials"
- : seasonNumber.Value.ToString("00", CultureInfo.InvariantCulture);
- // Get this one directly from the file system since we have to go up a level
- if (!string.Equals(prefix, seasonMarker, StringComparison.OrdinalIgnoreCase))
- {
- filenamePrefixes.Add("season" + seasonMarker);
- }
- foreach (var filename in filenamePrefixes)
- {
- AddImage(seriesFiles, images, filename + "-poster", ImageType.Primary);
- AddImage(seriesFiles, images, filename + "-fanart", ImageType.Backdrop);
- AddImage(seriesFiles, images, filename + "-banner", ImageType.Banner);
- AddImage(seriesFiles, images, filename + "-landscape", ImageType.Thumb);
- }
- }
- private bool AddImage(List<FileSystemMetadata> files, List<LocalImageInfo> images, string name, string imagePrefix, bool isInMixedFolder, ImageType type)
- {
- var added = AddImage(files, images, name, type, imagePrefix);
- if (!isInMixedFolder)
- {
- if (AddImage(files, images, name, type))
- {
- added = true;
- }
- }
- return added;
- }
- private static bool AddImage(IReadOnlyList<FileSystemMetadata> files, List<LocalImageInfo> images, string name, ImageType type, string? prefix = null)
- {
- var image = GetImage(files, name, prefix);
- if (image is null)
- {
- return false;
- }
- images.Add(new LocalImageInfo
- {
- FileInfo = image,
- Type = type
- });
- return true;
- }
- private static FileSystemMetadata? GetImage(IReadOnlyList<FileSystemMetadata> files, string name, string? prefix = null)
- {
- var fileNameLength = name.Length + (prefix?.Length ?? 0);
- for (var i = 0; i < files.Count; i++)
- {
- var file = files[i];
- if (file.IsDirectory || file.Length <= 0)
- {
- continue;
- }
- var fileName = Path.GetFileNameWithoutExtension(file.FullName.AsSpan());
- if (fileName.Length == fileNameLength
- && fileName.StartsWith(prefix, StringComparison.OrdinalIgnoreCase)
- && fileName.EndsWith(name, StringComparison.OrdinalIgnoreCase))
- {
- return file;
- }
- }
- return null;
- }
- }
- }
|