|  | @@ -1,1250 +0,0 @@
 | 
											
												
													
														|  | -#nullable disable
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -using System;
 |  | 
 | 
											
												
													
														|  | -using System.Collections.Generic;
 |  | 
 | 
											
												
													
														|  | -using System.Globalization;
 |  | 
 | 
											
												
													
														|  | -using System.IO;
 |  | 
 | 
											
												
													
														|  | -using System.Linq;
 |  | 
 | 
											
												
													
														|  | -using System.Text;
 |  | 
 | 
											
												
													
														|  | -using System.Threading;
 |  | 
 | 
											
												
													
														|  | -using System.Xml;
 |  | 
 | 
											
												
													
														|  | -using Emby.Dlna.Didl;
 |  | 
 | 
											
												
													
														|  | -using Emby.Dlna.Service;
 |  | 
 | 
											
												
													
														|  | -using Jellyfin.Data.Entities;
 |  | 
 | 
											
												
													
														|  | -using Jellyfin.Data.Enums;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Common.Extensions;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Controller.Configuration;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Controller.Drawing;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Controller.Dto;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Controller.Entities;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Controller.Entities.Audio;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Controller.Library;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Controller.MediaEncoding;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Controller.TV;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Model.Dlna;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Model.Dto;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Model.Entities;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Model.Globalization;
 |  | 
 | 
											
												
													
														|  | -using MediaBrowser.Model.Querying;
 |  | 
 | 
											
												
													
														|  | -using Microsoft.Extensions.Logging;
 |  | 
 | 
											
												
													
														|  | -using Genre = MediaBrowser.Controller.Entities.Genre;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -namespace Emby.Dlna.ContentDirectory
 |  | 
 | 
											
												
													
														|  | -{
 |  | 
 | 
											
												
													
														|  | -    /// <summary>
 |  | 
 | 
											
												
													
														|  | -    /// Defines the <see cref="ControlHandler" />.
 |  | 
 | 
											
												
													
														|  | -    /// </summary>
 |  | 
 | 
											
												
													
														|  | -    public class ControlHandler : BaseControlHandler
 |  | 
 | 
											
												
													
														|  | -    {
 |  | 
 | 
											
												
													
														|  | -        private const string NsDc = "http://purl.org/dc/elements/1.1/";
 |  | 
 | 
											
												
													
														|  | -        private const string NsDidl = "urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/";
 |  | 
 | 
											
												
													
														|  | -        private const string NsDlna = "urn:schemas-dlna-org:metadata-1-0/";
 |  | 
 | 
											
												
													
														|  | -        private const string NsUpnp = "urn:schemas-upnp-org:metadata-1-0/upnp/";
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        private readonly ILibraryManager _libraryManager;
 |  | 
 | 
											
												
													
														|  | -        private readonly IUserDataManager _userDataManager;
 |  | 
 | 
											
												
													
														|  | -        private readonly User _user;
 |  | 
 | 
											
												
													
														|  | -        private readonly IUserViewManager _userViewManager;
 |  | 
 | 
											
												
													
														|  | -        private readonly ITVSeriesManager _tvSeriesManager;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        private readonly int _systemUpdateId;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        private readonly DidlBuilder _didlBuilder;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        private readonly DeviceProfile _profile;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Initializes a new instance of the <see cref="ControlHandler"/> class.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="logger">The <see cref="ILogger"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="libraryManager">The <see cref="ILibraryManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="profile">The <see cref="DeviceProfile"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="serverAddress">The server address to use in this instance> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="accessToken">The <see cref="string"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="imageProcessor">The <see cref="IImageProcessor"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="userDataManager">The <see cref="IUserDataManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="systemUpdateId">The system id for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="config">The <see cref="IServerConfigurationManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="localization">The <see cref="ILocalizationManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="mediaSourceManager">The <see cref="IMediaSourceManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="userViewManager">The <see cref="IUserViewManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="mediaEncoder">The <see cref="IMediaEncoder"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="tvSeriesManager">The <see cref="ITVSeriesManager"/> for use with the <see cref="ControlHandler"/> instance.</param>
 |  | 
 | 
											
												
													
														|  | -        public ControlHandler(
 |  | 
 | 
											
												
													
														|  | -            ILogger logger,
 |  | 
 | 
											
												
													
														|  | -            ILibraryManager libraryManager,
 |  | 
 | 
											
												
													
														|  | -            DeviceProfile profile,
 |  | 
 | 
											
												
													
														|  | -            string serverAddress,
 |  | 
 | 
											
												
													
														|  | -            string accessToken,
 |  | 
 | 
											
												
													
														|  | -            IImageProcessor imageProcessor,
 |  | 
 | 
											
												
													
														|  | -            IUserDataManager userDataManager,
 |  | 
 | 
											
												
													
														|  | -            User user,
 |  | 
 | 
											
												
													
														|  | -            int systemUpdateId,
 |  | 
 | 
											
												
													
														|  | -            IServerConfigurationManager config,
 |  | 
 | 
											
												
													
														|  | -            ILocalizationManager localization,
 |  | 
 | 
											
												
													
														|  | -            IMediaSourceManager mediaSourceManager,
 |  | 
 | 
											
												
													
														|  | -            IUserViewManager userViewManager,
 |  | 
 | 
											
												
													
														|  | -            IMediaEncoder mediaEncoder,
 |  | 
 | 
											
												
													
														|  | -            ITVSeriesManager tvSeriesManager)
 |  | 
 | 
											
												
													
														|  | -            : base(config, logger)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            _libraryManager = libraryManager;
 |  | 
 | 
											
												
													
														|  | -            _userDataManager = userDataManager;
 |  | 
 | 
											
												
													
														|  | -            _user = user;
 |  | 
 | 
											
												
													
														|  | -            _systemUpdateId = systemUpdateId;
 |  | 
 | 
											
												
													
														|  | -            _userViewManager = userViewManager;
 |  | 
 | 
											
												
													
														|  | -            _tvSeriesManager = tvSeriesManager;
 |  | 
 | 
											
												
													
														|  | -            _profile = profile;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            _didlBuilder = new DidlBuilder(
 |  | 
 | 
											
												
													
														|  | -                profile,
 |  | 
 | 
											
												
													
														|  | -                user,
 |  | 
 | 
											
												
													
														|  | -                imageProcessor,
 |  | 
 | 
											
												
													
														|  | -                serverAddress,
 |  | 
 | 
											
												
													
														|  | -                accessToken,
 |  | 
 | 
											
												
													
														|  | -                userDataManager,
 |  | 
 | 
											
												
													
														|  | -                localization,
 |  | 
 | 
											
												
													
														|  | -                mediaSourceManager,
 |  | 
 | 
											
												
													
														|  | -                Logger,
 |  | 
 | 
											
												
													
														|  | -                mediaEncoder,
 |  | 
 | 
											
												
													
														|  | -                libraryManager);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <inheritdoc />
 |  | 
 | 
											
												
													
														|  | -        protected override void WriteResult(string methodName, IReadOnlyDictionary<string, string> methodParams, XmlWriter xmlWriter)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            ArgumentNullException.ThrowIfNull(xmlWriter);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            ArgumentNullException.ThrowIfNull(methodParams);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            const string DeviceId = "test";
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "GetSearchCapabilities", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleGetSearchCapabilities(xmlWriter);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "GetSortCapabilities", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleGetSortCapabilities(xmlWriter);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "GetSortExtensionCapabilities", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleGetSortExtensionCapabilities(xmlWriter);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "GetSystemUpdateID", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleGetSystemUpdateID(xmlWriter);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "Browse", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleBrowse(xmlWriter, methodParams, DeviceId);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "X_GetFeatureList", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleXGetFeatureList(xmlWriter);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "GetFeatureList", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleGetFeatureList(xmlWriter);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "X_SetBookmark", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleXSetBookmark(methodParams);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "Search", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleSearch(xmlWriter, methodParams, DeviceId);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (string.Equals(methodName, "X_BrowseByLetter", StringComparison.OrdinalIgnoreCase))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                HandleXBrowseByLetter(xmlWriter, methodParams, DeviceId);
 |  | 
 | 
											
												
													
														|  | -                return;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            throw new ResourceNotFoundException("Unexpected control request name: " + methodName);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Adds a "XSetBookmark" element to the xml document.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sparams">The method parameters.</param>
 |  | 
 | 
											
												
													
														|  | -        private void HandleXSetBookmark(IReadOnlyDictionary<string, string> sparams)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var id = sparams["ObjectID"];
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var serverItem = GetItemFromObjectId(id);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var item = serverItem.Item;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var newbookmark = int.Parse(sparams["PosSecond"], CultureInfo.InvariantCulture);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var userdata = _userDataManager.GetUserData(_user, item);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            userdata.PlaybackPositionTicks = TimeSpan.FromSeconds(newbookmark).Ticks;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            _userDataManager.SaveUserData(
 |  | 
 | 
											
												
													
														|  | -                _user,
 |  | 
 | 
											
												
													
														|  | -                item,
 |  | 
 | 
											
												
													
														|  | -                userdata,
 |  | 
 | 
											
												
													
														|  | -                UserDataSaveReason.TogglePlayed,
 |  | 
 | 
											
												
													
														|  | -                CancellationToken.None);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Adds the "SearchCaps" element to the xml document.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        private static void HandleGetSearchCapabilities(XmlWriter xmlWriter)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString(
 |  | 
 | 
											
												
													
														|  | -                "SearchCaps",
 |  | 
 | 
											
												
													
														|  | -                "res@resolution,res@size,res@duration,dc:title,dc:creator,upnp:actor,upnp:artist,upnp:genre,upnp:album,dc:date,upnp:class,@id,@refID,@protocolInfo,upnp:author,dc:description,pv:avKeywords");
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Adds the "SortCaps" element to the xml document.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        private static void HandleGetSortCapabilities(XmlWriter xmlWriter)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString(
 |  | 
 | 
											
												
													
														|  | -                "SortCaps",
 |  | 
 | 
											
												
													
														|  | -                "res@duration,res@size,res@bitrate,dc:date,dc:title,dc:size,upnp:album,upnp:artist,upnp:albumArtist,upnp:episodeNumber,upnp:genre,upnp:originalTrackNumber,upnp:rating");
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Adds the "SortExtensionCaps" element to the xml document.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        private static void HandleGetSortExtensionCapabilities(XmlWriter xmlWriter)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString(
 |  | 
 | 
											
												
													
														|  | -                "SortExtensionCaps",
 |  | 
 | 
											
												
													
														|  | -                "res@duration,res@size,res@bitrate,dc:date,dc:title,dc:size,upnp:album,upnp:artist,upnp:albumArtist,upnp:episodeNumber,upnp:genre,upnp:originalTrackNumber,upnp:rating");
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Adds the "Id" element to the xml document.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        private void HandleGetSystemUpdateID(XmlWriter xmlWriter)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString("Id", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Adds the "FeatureList" element to the xml document.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        private static void HandleGetFeatureList(XmlWriter xmlWriter)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString("FeatureList", WriteFeatureListXml());
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Adds the "FeatureList" element to the xml document.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        private static void HandleXGetFeatureList(XmlWriter xmlWriter)
 |  | 
 | 
											
												
													
														|  | -            => HandleGetFeatureList(xmlWriter);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Builds a static feature list.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The xml feature list.</returns>
 |  | 
 | 
											
												
													
														|  | -        private static string WriteFeatureListXml()
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            return "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"
 |  | 
 | 
											
												
													
														|  | -                + "<Features xmlns=\"urn:schemas-upnp-org:av:avs\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xsi:schemaLocation=\"urn:schemas-upnp-org:av:avs http://www.upnp.org/schemas/av/avs.xsd\">"
 |  | 
 | 
											
												
													
														|  | -                + "<Feature name=\"samsung.com_BASICVIEW\" version=\"1\">"
 |  | 
 | 
											
												
													
														|  | -                + "<container id=\"0\" type=\"object.item.imageItem\"/>"
 |  | 
 | 
											
												
													
														|  | -                + "<container id=\"0\" type=\"object.item.audioItem\"/>"
 |  | 
 | 
											
												
													
														|  | -                + "<container id=\"0\" type=\"object.item.videoItem\"/>"
 |  | 
 | 
											
												
													
														|  | -                + "</Feature>"
 |  | 
 | 
											
												
													
														|  | -                + "</Features>";
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Builds the "Browse" xml response.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sparams">The method parameters.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="deviceId">The device Id to use.</param>
 |  | 
 | 
											
												
													
														|  | -        private void HandleBrowse(XmlWriter xmlWriter, IReadOnlyDictionary<string, string> sparams, string deviceId)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var id = sparams["ObjectID"];
 |  | 
 | 
											
												
													
														|  | -            var flag = sparams["BrowseFlag"];
 |  | 
 | 
											
												
													
														|  | -            var filter = new Filter(sparams.GetValueOrDefault("Filter", "*"));
 |  | 
 | 
											
												
													
														|  | -            var sortCriteria = new SortCriteria(sparams.GetValueOrDefault("SortCriteria", string.Empty));
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var provided = 0;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            // Default to null instead of 0
 |  | 
 | 
											
												
													
														|  | -            // Upnp inspector sends 0 as requestedCount when it wants everything
 |  | 
 | 
											
												
													
														|  | -            int? requestedCount = null;
 |  | 
 | 
											
												
													
														|  | -            int? start = 0;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (sparams.ContainsKey("RequestedCount") && int.TryParse(sparams["RequestedCount"], out var requestedVal) && requestedVal > 0)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                requestedCount = requestedVal;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (sparams.ContainsKey("StartingIndex") && int.TryParse(sparams["StartingIndex"], out var startVal) && startVal > 0)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                start = startVal;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            int totalCount;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var settings = new XmlWriterSettings
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                Encoding = Encoding.UTF8,
 |  | 
 | 
											
												
													
														|  | -                CloseOutput = false,
 |  | 
 | 
											
												
													
														|  | -                OmitXmlDeclaration = true,
 |  | 
 | 
											
												
													
														|  | -                ConformanceLevel = ConformanceLevel.Fragment
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            using (StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8))
 |  | 
 | 
											
												
													
														|  | -            using (var writer = XmlWriter.Create(builder, settings))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                writer.WriteStartElement(string.Empty, "DIDL-Lite", NsDidl);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                writer.WriteAttributeString("xmlns", "dc", null, NsDc);
 |  | 
 | 
											
												
													
														|  | -                writer.WriteAttributeString("xmlns", "dlna", null, NsDlna);
 |  | 
 | 
											
												
													
														|  | -                writer.WriteAttributeString("xmlns", "upnp", null, NsUpnp);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                DidlBuilder.WriteXmlRootAttributes(_profile, writer);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                var serverItem = GetItemFromObjectId(id);
 |  | 
 | 
											
												
													
														|  | -                var item = serverItem.Item;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                if (string.Equals(flag, "BrowseMetadata", StringComparison.Ordinal))
 |  | 
 | 
											
												
													
														|  | -                {
 |  | 
 | 
											
												
													
														|  | -                    totalCount = 1;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                    if (item.IsDisplayedAsFolder || serverItem.StubType.HasValue)
 |  | 
 | 
											
												
													
														|  | -                    {
 |  | 
 | 
											
												
													
														|  | -                        var childrenResult = GetUserItems(item, serverItem.StubType, _user, sortCriteria, start, requestedCount);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                        _didlBuilder.WriteFolderElement(writer, item, serverItem.StubType, null, childrenResult.TotalRecordCount, filter, id);
 |  | 
 | 
											
												
													
														|  | -                    }
 |  | 
 | 
											
												
													
														|  | -                    else
 |  | 
 | 
											
												
													
														|  | -                    {
 |  | 
 | 
											
												
													
														|  | -                        _didlBuilder.WriteItemElement(writer, item, _user, null, null, deviceId, filter);
 |  | 
 | 
											
												
													
														|  | -                    }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                    provided++;
 |  | 
 | 
											
												
													
														|  | -                }
 |  | 
 | 
											
												
													
														|  | -                else
 |  | 
 | 
											
												
													
														|  | -                {
 |  | 
 | 
											
												
													
														|  | -                    var childrenResult = GetUserItems(item, serverItem.StubType, _user, sortCriteria, start, requestedCount);
 |  | 
 | 
											
												
													
														|  | -                    totalCount = childrenResult.TotalRecordCount;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                    provided = childrenResult.Items.Count;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                    foreach (var i in childrenResult.Items)
 |  | 
 | 
											
												
													
														|  | -                    {
 |  | 
 | 
											
												
													
														|  | -                        var childItem = i.Item;
 |  | 
 | 
											
												
													
														|  | -                        var displayStubType = i.StubType;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                        if (childItem.IsDisplayedAsFolder || displayStubType.HasValue)
 |  | 
 | 
											
												
													
														|  | -                        {
 |  | 
 | 
											
												
													
														|  | -                            var childCount = GetUserItems(childItem, displayStubType, _user, sortCriteria, null, 0)
 |  | 
 | 
											
												
													
														|  | -                                .TotalRecordCount;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                            _didlBuilder.WriteFolderElement(writer, childItem, displayStubType, item, childCount, filter);
 |  | 
 | 
											
												
													
														|  | -                        }
 |  | 
 | 
											
												
													
														|  | -                        else
 |  | 
 | 
											
												
													
														|  | -                        {
 |  | 
 | 
											
												
													
														|  | -                            _didlBuilder.WriteItemElement(writer, childItem, _user, item, serverItem.StubType, deviceId, filter);
 |  | 
 | 
											
												
													
														|  | -                        }
 |  | 
 | 
											
												
													
														|  | -                    }
 |  | 
 | 
											
												
													
														|  | -                }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                writer.WriteFullEndElement();
 |  | 
 | 
											
												
													
														|  | -                writer.Flush();
 |  | 
 | 
											
												
													
														|  | -                xmlWriter.WriteElementString("Result", builder.ToString());
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString("NumberReturned", provided.ToString(CultureInfo.InvariantCulture));
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString("TotalMatches", totalCount.ToString(CultureInfo.InvariantCulture));
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString("UpdateID", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Builds the response to the "X_BrowseByLetter request.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The <see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sparams">The method parameters.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="deviceId">The device id.</param>
 |  | 
 | 
											
												
													
														|  | -        private void HandleXBrowseByLetter(XmlWriter xmlWriter, IReadOnlyDictionary<string, string> sparams, string deviceId)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            // TODO: Implement this method
 |  | 
 | 
											
												
													
														|  | -            HandleSearch(xmlWriter, sparams, deviceId);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Builds a response to the "Search" request.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="xmlWriter">The xmlWriter<see cref="XmlWriter"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sparams">The method parameters.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="deviceId">The deviceId<see cref="string"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        private void HandleSearch(XmlWriter xmlWriter, IReadOnlyDictionary<string, string> sparams, string deviceId)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var searchCriteria = new SearchCriteria(sparams.GetValueOrDefault("SearchCriteria", string.Empty));
 |  | 
 | 
											
												
													
														|  | -            var sortCriteria = new SortCriteria(sparams.GetValueOrDefault("SortCriteria", string.Empty));
 |  | 
 | 
											
												
													
														|  | -            var filter = new Filter(sparams.GetValueOrDefault("Filter", "*"));
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            // sort example: dc:title, dc:date
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            // Default to null instead of 0
 |  | 
 | 
											
												
													
														|  | -            // Upnp inspector sends 0 as requestedCount when it wants everything
 |  | 
 | 
											
												
													
														|  | -            int? requestedCount = null;
 |  | 
 | 
											
												
													
														|  | -            int? start = 0;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (sparams.ContainsKey("RequestedCount") && int.TryParse(sparams["RequestedCount"], out var requestedVal) && requestedVal > 0)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                requestedCount = requestedVal;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (sparams.ContainsKey("StartingIndex") && int.TryParse(sparams["StartingIndex"], out var startVal) && startVal > 0)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                start = startVal;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            QueryResult<BaseItem> childrenResult;
 |  | 
 | 
											
												
													
														|  | -            var settings = new XmlWriterSettings
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                Encoding = Encoding.UTF8,
 |  | 
 | 
											
												
													
														|  | -                CloseOutput = false,
 |  | 
 | 
											
												
													
														|  | -                OmitXmlDeclaration = true,
 |  | 
 | 
											
												
													
														|  | -                ConformanceLevel = ConformanceLevel.Fragment
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            using (StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8))
 |  | 
 | 
											
												
													
														|  | -            using (var writer = XmlWriter.Create(builder, settings))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                writer.WriteStartElement(string.Empty, "DIDL-Lite", NsDidl);
 |  | 
 | 
											
												
													
														|  | -                writer.WriteAttributeString("xmlns", "dc", null, NsDc);
 |  | 
 | 
											
												
													
														|  | -                writer.WriteAttributeString("xmlns", "dlna", null, NsDlna);
 |  | 
 | 
											
												
													
														|  | -                writer.WriteAttributeString("xmlns", "upnp", null, NsUpnp);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                DidlBuilder.WriteXmlRootAttributes(_profile, writer);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                var serverItem = GetItemFromObjectId(sparams["ContainerID"]);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                var item = serverItem.Item;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                childrenResult = GetChildrenSorted(item, _user, searchCriteria, sortCriteria, start, requestedCount);
 |  | 
 | 
											
												
													
														|  | -                foreach (var i in childrenResult.Items)
 |  | 
 | 
											
												
													
														|  | -                {
 |  | 
 | 
											
												
													
														|  | -                    if (i.IsDisplayedAsFolder)
 |  | 
 | 
											
												
													
														|  | -                    {
 |  | 
 | 
											
												
													
														|  | -                        var childCount = GetChildrenSorted(i, _user, searchCriteria, sortCriteria, null, 0)
 |  | 
 | 
											
												
													
														|  | -                            .TotalRecordCount;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                        _didlBuilder.WriteFolderElement(writer, i, null, item, childCount, filter);
 |  | 
 | 
											
												
													
														|  | -                    }
 |  | 
 | 
											
												
													
														|  | -                    else
 |  | 
 | 
											
												
													
														|  | -                    {
 |  | 
 | 
											
												
													
														|  | -                        _didlBuilder.WriteItemElement(writer, i, _user, item, serverItem.StubType, deviceId, filter);
 |  | 
 | 
											
												
													
														|  | -                    }
 |  | 
 | 
											
												
													
														|  | -                }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                writer.WriteFullEndElement();
 |  | 
 | 
											
												
													
														|  | -                writer.Flush();
 |  | 
 | 
											
												
													
														|  | -                xmlWriter.WriteElementString("Result", builder.ToString());
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString("NumberReturned", childrenResult.Items.Count.ToString(CultureInfo.InvariantCulture));
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString("TotalMatches", childrenResult.TotalRecordCount.ToString(CultureInfo.InvariantCulture));
 |  | 
 | 
											
												
													
														|  | -            xmlWriter.WriteElementString("UpdateID", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the child items meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="item">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="search">The <see cref="SearchCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{BaseItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private static QueryResult<BaseItem> GetChildrenSorted(BaseItem item, User user, SearchCriteria search, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var folder = (Folder)item;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            MediaType[] mediaTypes = Array.Empty<MediaType>();
 |  | 
 | 
											
												
													
														|  | -            bool? isFolder = null;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            switch (search.SearchType)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                case SearchType.Audio:
 |  | 
 | 
											
												
													
														|  | -                    mediaTypes = new[] { MediaType.Audio };
 |  | 
 | 
											
												
													
														|  | -                    isFolder = false;
 |  | 
 | 
											
												
													
														|  | -                    break;
 |  | 
 | 
											
												
													
														|  | -                case SearchType.Video:
 |  | 
 | 
											
												
													
														|  | -                    mediaTypes = new[] { MediaType.Video };
 |  | 
 | 
											
												
													
														|  | -                    isFolder = false;
 |  | 
 | 
											
												
													
														|  | -                    break;
 |  | 
 | 
											
												
													
														|  | -                case SearchType.Image:
 |  | 
 | 
											
												
													
														|  | -                    mediaTypes = new[] { MediaType.Photo };
 |  | 
 | 
											
												
													
														|  | -                    isFolder = false;
 |  | 
 | 
											
												
													
														|  | -                    break;
 |  | 
 | 
											
												
													
														|  | -                case SearchType.Playlist:
 |  | 
 | 
											
												
													
														|  | -                case SearchType.MusicAlbum:
 |  | 
 | 
											
												
													
														|  | -                    isFolder = true;
 |  | 
 | 
											
												
													
														|  | -                    break;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return folder.GetItems(new InternalItemsQuery
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, folder.IsPreSorted),
 |  | 
 | 
											
												
													
														|  | -                User = user,
 |  | 
 | 
											
												
													
														|  | -                Recursive = true,
 |  | 
 | 
											
												
													
														|  | -                IsMissing = false,
 |  | 
 | 
											
												
													
														|  | -                ExcludeItemTypes = new[] { BaseItemKind.Book },
 |  | 
 | 
											
												
													
														|  | -                IsFolder = isFolder,
 |  | 
 | 
											
												
													
														|  | -                MediaTypes = mediaTypes,
 |  | 
 | 
											
												
													
														|  | -                DtoOptions = GetDtoOptions()
 |  | 
 | 
											
												
													
														|  | -            });
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns a new DtoOptions object.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="DtoOptions"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private static DtoOptions GetDtoOptions()
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            return new DtoOptions(true);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the User items meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="item">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="stubType">The <see cref="StubType"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetUserItems(BaseItem item, StubType? stubType, User user, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            switch (item)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                case MusicGenre:
 |  | 
 | 
											
												
													
														|  | -                    return GetMusicGenreItems(item, user, sort, startIndex, limit);
 |  | 
 | 
											
												
													
														|  | -                case MusicArtist:
 |  | 
 | 
											
												
													
														|  | -                    return GetMusicArtistItems(item, user, sort, startIndex, limit);
 |  | 
 | 
											
												
													
														|  | -                case Genre:
 |  | 
 | 
											
												
													
														|  | -                    return GetGenreItems(item, user, sort, startIndex, limit);
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (stubType != StubType.Folder && item is IHasCollectionType collectionFolder)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                switch (collectionFolder.CollectionType)
 |  | 
 | 
											
												
													
														|  | -                {
 |  | 
 | 
											
												
													
														|  | -                    case CollectionType.Music:
 |  | 
 | 
											
												
													
														|  | -                        return GetMusicFolders(item, user, stubType, sort, startIndex, limit);
 |  | 
 | 
											
												
													
														|  | -                    case CollectionType.Movies:
 |  | 
 | 
											
												
													
														|  | -                        return GetMovieFolders(item, user, stubType, sort, startIndex, limit);
 |  | 
 | 
											
												
													
														|  | -                    case CollectionType.TvShows:
 |  | 
 | 
											
												
													
														|  | -                        return GetTvFolders(item, user, stubType, sort, startIndex, limit);
 |  | 
 | 
											
												
													
														|  | -                    case CollectionType.Folders:
 |  | 
 | 
											
												
													
														|  | -                        return GetFolders(user, startIndex, limit);
 |  | 
 | 
											
												
													
														|  | -                    case CollectionType.LiveTv:
 |  | 
 | 
											
												
													
														|  | -                        return GetLiveTvChannels(user, sort, startIndex, limit);
 |  | 
 | 
											
												
													
														|  | -                }
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (stubType.HasValue && stubType.Value != StubType.Folder)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                // TODO should this be doing something?
 |  | 
 | 
											
												
													
														|  | -                return new QueryResult<ServerItem>();
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var folder = (Folder)item;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var query = new InternalItemsQuery(user)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                IsVirtualItem = false,
 |  | 
 | 
											
												
													
														|  | -                ExcludeItemTypes = new[] { BaseItemKind.Book },
 |  | 
 | 
											
												
													
														|  | -                IsPlaceHolder = false,
 |  | 
 | 
											
												
													
														|  | -                DtoOptions = GetDtoOptions(),
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, folder.IsPreSorted)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var queryResult = folder.GetItems(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(startIndex, queryResult);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the Live Tv Channels meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetLiveTvChannels(User user, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var query = new InternalItemsQuery(user)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                IncludeItemTypes = new[] { BaseItemKind.LiveTvChannel },
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, false)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _libraryManager.GetItemsResult(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(startIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the music folders meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="item">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="stubType">The <see cref="StubType"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMusicFolders(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var query = new InternalItemsQuery(user)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, false)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            switch (stubType)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                case StubType.Latest:
 |  | 
 | 
											
												
													
														|  | -                    return GetLatest(item, query, BaseItemKind.Audio);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Playlists:
 |  | 
 | 
											
												
													
														|  | -                    return GetMusicPlaylists(query);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Albums:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.MusicAlbum);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Artists:
 |  | 
 | 
											
												
													
														|  | -                    return GetMusicArtists(item, query);
 |  | 
 | 
											
												
													
														|  | -                case StubType.AlbumArtists:
 |  | 
 | 
											
												
													
														|  | -                    return GetMusicAlbumArtists(item, query);
 |  | 
 | 
											
												
													
														|  | -                case StubType.FavoriteAlbums:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.MusicAlbum, true);
 |  | 
 | 
											
												
													
														|  | -                case StubType.FavoriteArtists:
 |  | 
 | 
											
												
													
														|  | -                    return GetFavoriteArtists(item, query);
 |  | 
 | 
											
												
													
														|  | -                case StubType.FavoriteSongs:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.Audio, true);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Songs:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.Audio);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Genres:
 |  | 
 | 
											
												
													
														|  | -                    return GetMusicGenres(item, query);
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var serverItems = new ServerItem[]
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Latest),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Playlists),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Albums),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.AlbumArtists),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Artists),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Songs),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Genres),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.FavoriteArtists),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.FavoriteAlbums),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.FavoriteSongs)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (limit < serverItems.Length)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                serverItems = serverItems[..limit.Value];
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return new QueryResult<ServerItem>(
 |  | 
 | 
											
												
													
														|  | -                startIndex,
 |  | 
 | 
											
												
													
														|  | -                serverItems.Length,
 |  | 
 | 
											
												
													
														|  | -                serverItems);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the movie folders meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="item">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="stubType">The <see cref="StubType"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMovieFolders(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var query = new InternalItemsQuery(user)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, false)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            switch (stubType)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                case StubType.ContinueWatching:
 |  | 
 | 
											
												
													
														|  | -                    return GetMovieContinueWatching(item, query);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Latest:
 |  | 
 | 
											
												
													
														|  | -                    return GetLatest(item, query, BaseItemKind.Movie);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Movies:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.Movie);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Collections:
 |  | 
 | 
											
												
													
														|  | -                    return GetMovieCollections(query);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Favorites:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.Movie, true);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Genres:
 |  | 
 | 
											
												
													
														|  | -                    return GetGenres(item, query);
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var array = new ServerItem[]
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.ContinueWatching),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Latest),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Movies),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Collections),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Favorites),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Genres)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (limit < array.Length)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                array = array[..limit.Value];
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return new QueryResult<ServerItem>(
 |  | 
 | 
											
												
													
														|  | -                startIndex,
 |  | 
 | 
											
												
													
														|  | -                array.Length,
 |  | 
 | 
											
												
													
														|  | -                array);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the folders meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetFolders(User user, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var folders = _libraryManager.GetUserRootFolder().GetChildren(user, true);
 |  | 
 | 
											
												
													
														|  | -            var totalRecordCount = folders.Count;
 |  | 
 | 
											
												
													
														|  | -            // Handle paging
 |  | 
 | 
											
												
													
														|  | -            var items = folders
 |  | 
 | 
											
												
													
														|  | -                .OrderBy(i => i.SortName)
 |  | 
 | 
											
												
													
														|  | -                .Skip(startIndex ?? 0)
 |  | 
 | 
											
												
													
														|  | -                .Take(limit ?? int.MaxValue)
 |  | 
 | 
											
												
													
														|  | -                .Select(i => new ServerItem(i, StubType.Folder))
 |  | 
 | 
											
												
													
														|  | -                .ToArray();
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return new QueryResult<ServerItem>(
 |  | 
 | 
											
												
													
														|  | -                startIndex,
 |  | 
 | 
											
												
													
														|  | -                totalRecordCount,
 |  | 
 | 
											
												
													
														|  | -                items);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the TV folders meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="item">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="stubType">The <see cref="StubType"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetTvFolders(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var query = new InternalItemsQuery(user)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, false)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            switch (stubType)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                case StubType.ContinueWatching:
 |  | 
 | 
											
												
													
														|  | -                    return GetMovieContinueWatching(item, query);
 |  | 
 | 
											
												
													
														|  | -                case StubType.NextUp:
 |  | 
 | 
											
												
													
														|  | -                    return GetNextUp(item, query);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Latest:
 |  | 
 | 
											
												
													
														|  | -                    return GetLatest(item, query, BaseItemKind.Episode);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Series:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.Series);
 |  | 
 | 
											
												
													
														|  | -                case StubType.FavoriteSeries:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.Series, true);
 |  | 
 | 
											
												
													
														|  | -                case StubType.FavoriteEpisodes:
 |  | 
 | 
											
												
													
														|  | -                    return GetChildrenOfItem(item, query, BaseItemKind.Episode, true);
 |  | 
 | 
											
												
													
														|  | -                case StubType.Genres:
 |  | 
 | 
											
												
													
														|  | -                    return GetGenres(item, query);
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var serverItems = new ServerItem[]
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.ContinueWatching),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.NextUp),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Latest),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Series),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.FavoriteSeries),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.FavoriteEpisodes),
 |  | 
 | 
											
												
													
														|  | -                new(item, StubType.Genres)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (limit < serverItems.Length)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                serverItems = serverItems[..limit.Value];
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return new QueryResult<ServerItem>(
 |  | 
 | 
											
												
													
														|  | -                startIndex,
 |  | 
 | 
											
												
													
														|  | -                serverItems.Length,
 |  | 
 | 
											
												
													
														|  | -                serverItems);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the Movies that are part watched that meet the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMovieContinueWatching(BaseItem parent, InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            query.Recursive = true;
 |  | 
 | 
											
												
													
														|  | -            query.Parent = parent;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            query.OrderBy = new[]
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                (ItemSortBy.DatePlayed, SortOrder.Descending),
 |  | 
 | 
											
												
													
														|  | -                (ItemSortBy.SortName, SortOrder.Ascending)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            query.IsResumable = true;
 |  | 
 | 
											
												
													
														|  | -            query.Limit ??= 10;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _libraryManager.GetItemsResult(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the Movie collections meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMovieCollections(InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            query.Recursive = true;
 |  | 
 | 
											
												
													
														|  | -            query.IncludeItemTypes = new[] { BaseItemKind.BoxSet };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _libraryManager.GetItemsResult(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the children that meet the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="itemType">The item type.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="isFavorite">A value indicating whether to only fetch favorite items.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetChildrenOfItem(BaseItem parent, InternalItemsQuery query, BaseItemKind itemType, bool isFavorite = false)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            query.Recursive = true;
 |  | 
 | 
											
												
													
														|  | -            query.Parent = parent;
 |  | 
 | 
											
												
													
														|  | -            query.IsFavorite = isFavorite;
 |  | 
 | 
											
												
													
														|  | -            query.IncludeItemTypes = new[] { itemType };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _libraryManager.GetItemsResult(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the genres meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// The GetGenres.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetGenres(BaseItem parent, InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            // Don't sort
 |  | 
 | 
											
												
													
														|  | -            query.OrderBy = Array.Empty<(ItemSortBy, SortOrder)>();
 |  | 
 | 
											
												
													
														|  | -            query.AncestorIds = new[] { parent.Id };
 |  | 
 | 
											
												
													
														|  | -            var genresResult = _libraryManager.GetGenres(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, genresResult);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the music genres meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMusicGenres(BaseItem parent, InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            // Don't sort
 |  | 
 | 
											
												
													
														|  | -            query.OrderBy = Array.Empty<(ItemSortBy, SortOrder)>();
 |  | 
 | 
											
												
													
														|  | -            query.AncestorIds = new[] { parent.Id };
 |  | 
 | 
											
												
													
														|  | -            var genresResult = _libraryManager.GetMusicGenres(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, genresResult);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the music albums by artist that meet the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMusicAlbumArtists(BaseItem parent, InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            // Don't sort
 |  | 
 | 
											
												
													
														|  | -            query.OrderBy = Array.Empty<(ItemSortBy, SortOrder)>();
 |  | 
 | 
											
												
													
														|  | -            query.AncestorIds = new[] { parent.Id };
 |  | 
 | 
											
												
													
														|  | -            var artists = _libraryManager.GetAlbumArtists(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, artists);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the music artists meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMusicArtists(BaseItem parent, InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            // Don't sort
 |  | 
 | 
											
												
													
														|  | -            query.OrderBy = Array.Empty<(ItemSortBy, SortOrder)>();
 |  | 
 | 
											
												
													
														|  | -            query.AncestorIds = new[] { parent.Id };
 |  | 
 | 
											
												
													
														|  | -            var artists = _libraryManager.GetArtists(query);
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, artists);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the artists tagged as favourite that meet the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetFavoriteArtists(BaseItem parent, InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            // Don't sort
 |  | 
 | 
											
												
													
														|  | -            query.OrderBy = Array.Empty<(ItemSortBy, SortOrder)>();
 |  | 
 | 
											
												
													
														|  | -            query.AncestorIds = new[] { parent.Id };
 |  | 
 | 
											
												
													
														|  | -            query.IsFavorite = true;
 |  | 
 | 
											
												
													
														|  | -            var artists = _libraryManager.GetArtists(query);
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, artists);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the music playlists meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The query<see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMusicPlaylists(InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            query.Parent = null;
 |  | 
 | 
											
												
													
														|  | -            query.IncludeItemTypes = new[] { BaseItemKind.Playlist };
 |  | 
 | 
											
												
													
														|  | -            query.Recursive = true;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _libraryManager.GetItemsResult(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the next up item meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetNextUp(BaseItem parent, InternalItemsQuery query)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            query.OrderBy = Array.Empty<(ItemSortBy, SortOrder)>();
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _tvSeriesManager.GetNextUp(
 |  | 
 | 
											
												
													
														|  | -                new NextUpQuery
 |  | 
 | 
											
												
													
														|  | -                {
 |  | 
 | 
											
												
													
														|  | -                    Limit = query.Limit,
 |  | 
 | 
											
												
													
														|  | -                    StartIndex = query.StartIndex,
 |  | 
 | 
											
												
													
														|  | -                    // User cannot be null here as the caller has set it
 |  | 
 | 
											
												
													
														|  | -                    UserId = query.User!.Id
 |  | 
 | 
											
												
													
														|  | -                },
 |  | 
 | 
											
												
													
														|  | -                new[] { parent },
 |  | 
 | 
											
												
													
														|  | -                query.DtoOptions);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the latest items of [itemType] meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="parent">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="query">The <see cref="InternalItemsQuery"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="itemType">The item type.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetLatest(BaseItem parent, InternalItemsQuery query, BaseItemKind itemType)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            query.OrderBy = Array.Empty<(ItemSortBy, SortOrder)>();
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var items = _userViewManager.GetLatestItems(
 |  | 
 | 
											
												
													
														|  | -                new LatestItemsQuery
 |  | 
 | 
											
												
													
														|  | -                {
 |  | 
 | 
											
												
													
														|  | -                    // User cannot be null here as the caller has set it
 |  | 
 | 
											
												
													
														|  | -                    UserId = query.User!.Id,
 |  | 
 | 
											
												
													
														|  | -                    Limit = query.Limit ?? 50,
 |  | 
 | 
											
												
													
														|  | -                    IncludeItemTypes = new[] { itemType },
 |  | 
 | 
											
												
													
														|  | -                    ParentId = parent?.Id ?? Guid.Empty,
 |  | 
 | 
											
												
													
														|  | -                    GroupItems = true
 |  | 
 | 
											
												
													
														|  | -                },
 |  | 
 | 
											
												
													
														|  | -                query.DtoOptions).Select(i => i.Item1 ?? i.Item2.FirstOrDefault()).Where(i => i is not null).ToArray();
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(query.StartIndex, items);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns music artist items that meet the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="item">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMusicArtistItems(BaseItem item, User user, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var query = new InternalItemsQuery(user)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                Recursive = true,
 |  | 
 | 
											
												
													
														|  | -                ArtistIds = new[] { item.Id },
 |  | 
 | 
											
												
													
														|  | -                IncludeItemTypes = new[] { BaseItemKind.MusicAlbum },
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                DtoOptions = GetDtoOptions(),
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, false)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _libraryManager.GetItemsResult(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(startIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the genre items meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="item">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetGenreItems(BaseItem item, User user, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var query = new InternalItemsQuery(user)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                Recursive = true,
 |  | 
 | 
											
												
													
														|  | -                GenreIds = new[] { item.Id },
 |  | 
 | 
											
												
													
														|  | -                IncludeItemTypes = new[]
 |  | 
 | 
											
												
													
														|  | -                {
 |  | 
 | 
											
												
													
														|  | -                    BaseItemKind.Movie,
 |  | 
 | 
											
												
													
														|  | -                    BaseItemKind.Series
 |  | 
 | 
											
												
													
														|  | -                },
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                DtoOptions = GetDtoOptions(),
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, false)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _libraryManager.GetItemsResult(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(startIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Returns the music genre items meeting the criteria.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="item">The <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="user">The <see cref="User"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="limit">The maximum number to return.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private QueryResult<ServerItem> GetMusicGenreItems(BaseItem item, User user, SortCriteria sort, int? startIndex, int? limit)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var query = new InternalItemsQuery(user)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                Recursive = true,
 |  | 
 | 
											
												
													
														|  | -                GenreIds = new[] { item.Id },
 |  | 
 | 
											
												
													
														|  | -                IncludeItemTypes = new[] { BaseItemKind.MusicAlbum },
 |  | 
 | 
											
												
													
														|  | -                Limit = limit,
 |  | 
 | 
											
												
													
														|  | -                StartIndex = startIndex,
 |  | 
 | 
											
												
													
														|  | -                DtoOptions = GetDtoOptions(),
 |  | 
 | 
											
												
													
														|  | -                OrderBy = GetOrderBy(sort, false)
 |  | 
 | 
											
												
													
														|  | -            };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var result = _libraryManager.GetItemsResult(query);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return ToResult(startIndex, result);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Converts <see cref="IReadOnlyCollection{BaseItem}"/> into a <see cref="QueryResult{ServerItem}"/>.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="result">An array of <see cref="BaseItem"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>A <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private static QueryResult<ServerItem> ToResult(int? startIndex, IReadOnlyCollection<BaseItem> result)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var serverItems = result
 |  | 
 | 
											
												
													
														|  | -                .Select(i => new ServerItem(i, null))
 |  | 
 | 
											
												
													
														|  | -                .ToArray();
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return new QueryResult<ServerItem>(
 |  | 
 | 
											
												
													
														|  | -                startIndex,
 |  | 
 | 
											
												
													
														|  | -                result.Count,
 |  | 
 | 
											
												
													
														|  | -                serverItems);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Converts a <see cref="QueryResult{BaseItem}"/> to a <see cref="QueryResult{ServerItem}"/>.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The index the result started at.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="result">A <see cref="QueryResult{BaseItem}"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private static QueryResult<ServerItem> ToResult(int? startIndex, QueryResult<BaseItem> result)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var length = result.Items.Count;
 |  | 
 | 
											
												
													
														|  | -            var serverItems = new ServerItem[length];
 |  | 
 | 
											
												
													
														|  | -            for (var i = 0; i < length; i++)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                serverItems[i] = new ServerItem(result.Items[i], null);
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return new QueryResult<ServerItem>(
 |  | 
 | 
											
												
													
														|  | -                startIndex,
 |  | 
 | 
											
												
													
														|  | -                result.TotalRecordCount,
 |  | 
 | 
											
												
													
														|  | -                serverItems);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Converts a query result to a <see cref="QueryResult{ServerItem}"/>.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="startIndex">The start index.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="result">A <see cref="QueryResult{BaseItem}"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="QueryResult{ServerItem}"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private static QueryResult<ServerItem> ToResult(int? startIndex, QueryResult<(BaseItem Item, ItemCounts ItemCounts)> result)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            var length = result.Items.Count;
 |  | 
 | 
											
												
													
														|  | -            var serverItems = new ServerItem[length];
 |  | 
 | 
											
												
													
														|  | -            for (var i = 0; i < length; i++)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                serverItems[i] = new ServerItem(result.Items[i].Item, null);
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return new QueryResult<ServerItem>(
 |  | 
 | 
											
												
													
														|  | -                startIndex,
 |  | 
 | 
											
												
													
														|  | -                result.TotalRecordCount,
 |  | 
 | 
											
												
													
														|  | -                serverItems);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Gets the sorting method on a query.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="sort">The <see cref="SortCriteria"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="isPreSorted">True if pre-sorted.</param>
 |  | 
 | 
											
												
													
														|  | -        private static (ItemSortBy SortName, SortOrder SortOrder)[] GetOrderBy(SortCriteria sort, bool isPreSorted)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            return isPreSorted ? Array.Empty<(ItemSortBy, SortOrder)>() : new[] { (ItemSortBy.SortName, sort.SortOrder) };
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Retrieves the ServerItem id.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="id">The id<see cref="string"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The <see cref="ServerItem"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private ServerItem GetItemFromObjectId(string id)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            return DidlBuilder.IsIdRoot(id)
 |  | 
 | 
											
												
													
														|  | -                 ? new ServerItem(_libraryManager.GetUserRootFolder(), null)
 |  | 
 | 
											
												
													
														|  | -                 : ParseItemId(id);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -        /// <summary>
 |  | 
 | 
											
												
													
														|  | -        /// Parses the item id into a <see cref="ServerItem"/>.
 |  | 
 | 
											
												
													
														|  | -        /// </summary>
 |  | 
 | 
											
												
													
														|  | -        /// <param name="id">The <see cref="string"/>.</param>
 |  | 
 | 
											
												
													
														|  | -        /// <returns>The corresponding <see cref="ServerItem"/>.</returns>
 |  | 
 | 
											
												
													
														|  | -        private ServerItem ParseItemId(string id)
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            StubType? stubType = null;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            // After using PlayTo, MediaMonkey sends a request to the server trying to get item info
 |  | 
 | 
											
												
													
														|  | -            const string ParamsSrch = "Params=";
 |  | 
 | 
											
												
													
														|  | -            var paramsIndex = id.IndexOf(ParamsSrch, StringComparison.OrdinalIgnoreCase);
 |  | 
 | 
											
												
													
														|  | -            if (paramsIndex != -1)
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                id = id[(paramsIndex + ParamsSrch.Length)..];
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                var parts = id.Split(';');
 |  | 
 | 
											
												
													
														|  | -                id = parts[23];
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            var dividerIndex = id.IndexOf('_', StringComparison.Ordinal);
 |  | 
 | 
											
												
													
														|  | -            if (dividerIndex != -1 && Enum.TryParse<StubType>(id.AsSpan(0, dividerIndex), true, out var parsedStubType))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                id = id[(dividerIndex + 1)..];
 |  | 
 | 
											
												
													
														|  | -                stubType = parsedStubType;
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            if (Guid.TryParse(id, out var itemId))
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                var item = _libraryManager.GetItemById(itemId);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                return new ServerItem(item, stubType);
 |  | 
 | 
											
												
													
														|  | -            }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            Logger.LogError("Error parsing item Id: {Id}. Returning user root folder.", id);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -            return new ServerItem(_libraryManager.GetUserRootFolder(), null);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | -    }
 |  | 
 | 
											
												
													
														|  | -}
 |  | 
 |