| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266 | using System;using System.ComponentModel;using System.ComponentModel.DataAnnotations;using System.Globalization;using System.Linq;using Jellyfin.Api.Helpers;using Jellyfin.Api.ModelBinders;using Jellyfin.Data.Enums;using Jellyfin.Extensions;using MediaBrowser.Controller.Drawing;using MediaBrowser.Controller.Dto;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Audio;using MediaBrowser.Controller.Entities.TV;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.LiveTv;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Search;using Microsoft.AspNetCore.Authorization;using Microsoft.AspNetCore.Http;using Microsoft.AspNetCore.Mvc;namespace Jellyfin.Api.Controllers;/// <summary>/// Search controller./// </summary>[Route("Search/Hints")][Authorize]public class SearchController : BaseJellyfinApiController{    private readonly ISearchEngine _searchEngine;    private readonly ILibraryManager _libraryManager;    private readonly IDtoService _dtoService;    private readonly IImageProcessor _imageProcessor;    /// <summary>    /// Initializes a new instance of the <see cref="SearchController"/> class.    /// </summary>    /// <param name="searchEngine">Instance of <see cref="ISearchEngine"/> interface.</param>    /// <param name="libraryManager">Instance of <see cref="ILibraryManager"/> interface.</param>    /// <param name="dtoService">Instance of <see cref="IDtoService"/> interface.</param>    /// <param name="imageProcessor">Instance of <see cref="IImageProcessor"/> interface.</param>    public SearchController(        ISearchEngine searchEngine,        ILibraryManager libraryManager,        IDtoService dtoService,        IImageProcessor imageProcessor)    {        _searchEngine = searchEngine;        _libraryManager = libraryManager;        _dtoService = dtoService;        _imageProcessor = imageProcessor;    }    /// <summary>    /// Gets the search hint result.    /// </summary>    /// <param name="startIndex">Optional. The record index to start at. All items with a lower index will be dropped from the results.</param>    /// <param name="limit">Optional. The maximum number of records to return.</param>    /// <param name="userId">Optional. Supply a user id to search within a user's library or omit to search all.</param>    /// <param name="searchTerm">The search term to filter on.</param>    /// <param name="includeItemTypes">If specified, only results with the specified item types are returned. This allows multiple, comma delimited.</param>    /// <param name="excludeItemTypes">If specified, results with these item types are filtered out. This allows multiple, comma delimited.</param>    /// <param name="mediaTypes">If specified, only results with the specified media types are returned. This allows multiple, comma delimited.</param>    /// <param name="parentId">If specified, only children of the parent are returned.</param>    /// <param name="isMovie">Optional filter for movies.</param>    /// <param name="isSeries">Optional filter for series.</param>    /// <param name="isNews">Optional filter for news.</param>    /// <param name="isKids">Optional filter for kids.</param>    /// <param name="isSports">Optional filter for sports.</param>    /// <param name="includePeople">Optional filter whether to include people.</param>    /// <param name="includeMedia">Optional filter whether to include media.</param>    /// <param name="includeGenres">Optional filter whether to include genres.</param>    /// <param name="includeStudios">Optional filter whether to include studios.</param>    /// <param name="includeArtists">Optional filter whether to include artists.</param>    /// <response code="200">Search hint returned.</response>    /// <returns>An <see cref="SearchHintResult"/> with the results of the search.</returns>    [HttpGet]    [Description("Gets search hints based on a search term")]    [ProducesResponseType(StatusCodes.Status200OK)]    public ActionResult<SearchHintResult> GetSearchHints(        [FromQuery] int? startIndex,        [FromQuery] int? limit,        [FromQuery] Guid? userId,        [FromQuery, Required] string searchTerm,        [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] includeItemTypes,        [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] BaseItemKind[] excludeItemTypes,        [FromQuery, ModelBinder(typeof(CommaDelimitedArrayModelBinder))] MediaType[] mediaTypes,        [FromQuery] Guid? parentId,        [FromQuery] bool? isMovie,        [FromQuery] bool? isSeries,        [FromQuery] bool? isNews,        [FromQuery] bool? isKids,        [FromQuery] bool? isSports,        [FromQuery] bool includePeople = true,        [FromQuery] bool includeMedia = true,        [FromQuery] bool includeGenres = true,        [FromQuery] bool includeStudios = true,        [FromQuery] bool includeArtists = true)    {        userId = RequestHelpers.GetUserId(User, userId);        var result = _searchEngine.GetSearchHints(new SearchQuery        {            Limit = limit,            SearchTerm = searchTerm,            IncludeArtists = includeArtists,            IncludeGenres = includeGenres,            IncludeMedia = includeMedia,            IncludePeople = includePeople,            IncludeStudios = includeStudios,            StartIndex = startIndex,            UserId = userId.Value,            IncludeItemTypes = includeItemTypes,            ExcludeItemTypes = excludeItemTypes,            MediaTypes = mediaTypes,            ParentId = parentId,            IsKids = isKids,            IsMovie = isMovie,            IsNews = isNews,            IsSeries = isSeries,            IsSports = isSports        });        return new SearchHintResult(result.Items.Select(GetSearchHintResult).ToArray(), result.TotalRecordCount);    }    /// <summary>    /// Gets the search hint result.    /// </summary>    /// <param name="hintInfo">The hint info.</param>    /// <returns>SearchHintResult.</returns>    private SearchHint GetSearchHintResult(SearchHintInfo hintInfo)    {        var item = hintInfo.Item;        var result = new SearchHint        {            Name = item.Name,            IndexNumber = item.IndexNumber,            ParentIndexNumber = item.ParentIndexNumber,            Id = item.Id,            Type = item.GetBaseItemKind(),            MediaType = item.MediaType,            MatchedTerm = hintInfo.MatchedTerm,            RunTimeTicks = item.RunTimeTicks,            ProductionYear = item.ProductionYear,            ChannelId = item.ChannelId,            EndDate = item.EndDate        };#pragma warning disable CS0618        // Kept for compatibility with older clients        result.ItemId = result.Id;#pragma warning restore CS0618        if (item.IsFolder)        {            result.IsFolder = true;        }        var primaryImageTag = _imageProcessor.GetImageCacheTag(item, ImageType.Primary);        if (primaryImageTag is not null)        {            result.PrimaryImageTag = primaryImageTag;            result.PrimaryImageAspectRatio = _dtoService.GetPrimaryImageAspectRatio(item);        }        SetThumbImageInfo(result, item);        SetBackdropImageInfo(result, item);        switch (item)        {            case IHasSeries hasSeries:                result.Series = hasSeries.SeriesName;                break;            case LiveTvProgram program:                result.StartDate = program.StartDate;                break;            case Series series:                if (series.Status.HasValue)                {                    result.Status = series.Status.Value.ToString();                }                break;            case MusicAlbum album:                result.Artists = album.Artists;                result.AlbumArtist = album.AlbumArtist;                break;            case Audio song:                result.AlbumArtist = song.AlbumArtists?.FirstOrDefault();                result.Artists = song.Artists;                MusicAlbum musicAlbum = song.AlbumEntity;                if (musicAlbum is not null)                {                    result.Album = musicAlbum.Name;                    result.AlbumId = musicAlbum.Id;                }                else                {                    result.Album = song.Album;                }                break;        }        if (!item.ChannelId.IsEmpty())        {            var channel = _libraryManager.GetItemById<BaseItem>(item.ChannelId);            result.ChannelName = channel?.Name;        }        return result;    }    private void SetThumbImageInfo(SearchHint hint, BaseItem item)    {        var itemWithImage = item.HasImage(ImageType.Thumb) ? item : null;        if (itemWithImage is null && item is Episode)        {            itemWithImage = GetParentWithImage<Series>(item, ImageType.Thumb);        }        itemWithImage ??= GetParentWithImage<BaseItem>(item, ImageType.Thumb);        if (itemWithImage is not null)        {            var tag = _imageProcessor.GetImageCacheTag(itemWithImage, ImageType.Thumb);            if (tag is not null)            {                hint.ThumbImageTag = tag;                hint.ThumbImageItemId = itemWithImage.Id.ToString("N", CultureInfo.InvariantCulture);            }        }    }    private void SetBackdropImageInfo(SearchHint hint, BaseItem item)    {        var itemWithImage = (item.HasImage(ImageType.Backdrop) ? item : null)            ?? GetParentWithImage<BaseItem>(item, ImageType.Backdrop);        if (itemWithImage is not null)        {            var tag = _imageProcessor.GetImageCacheTag(itemWithImage, ImageType.Backdrop);            if (tag is not null)            {                hint.BackdropImageTag = tag;                hint.BackdropImageItemId = itemWithImage.Id.ToString("N", CultureInfo.InvariantCulture);            }        }    }    private T? GetParentWithImage<T>(BaseItem item, ImageType type)        where T : BaseItem    {        return item.GetParents().OfType<T>().FirstOrDefault(i => i.HasImage(type));    }}
 |