jellyfin-server/Emby.Dlna/Didl/DidlBuilder.cs

1260 lines
46 KiB
C#
Raw Normal View History

#pragma warning disable CS1591
using System;
2019-01-13 19:16:19 +00:00
using System.Globalization;
using System.IO;
using System.Linq;
using System.Text;
using System.Xml;
using Emby.Dlna.ContentDirectory;
2016-10-29 22:22:20 +00:00
using MediaBrowser.Controller.Channels;
using MediaBrowser.Controller.Drawing;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Entities.Movies;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
2019-01-13 19:16:19 +00:00
using MediaBrowser.Controller.MediaEncoding;
2016-10-29 22:22:20 +00:00
using MediaBrowser.Controller.Playlists;
using MediaBrowser.Model.Dlna;
using MediaBrowser.Model.Drawing;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Globalization;
2019-01-13 19:16:19 +00:00
using MediaBrowser.Model.Net;
using Microsoft.Extensions.Logging;
2016-10-29 22:22:20 +00:00
2016-10-29 22:34:54 +00:00
namespace Emby.Dlna.Didl
2016-10-29 22:22:20 +00:00
{
public class DidlBuilder
{
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
private const string NS_DIDL = "urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/";
private const string NS_DC = "http://purl.org/dc/elements/1.1/";
private const string NS_UPNP = "urn:schemas-upnp-org:metadata-1-0/upnp/";
private const string NS_DLNA = "urn:schemas-dlna-org:metadata-1-0/";
private readonly DeviceProfile _profile;
private readonly IImageProcessor _imageProcessor;
private readonly string _serverAddress;
private readonly string _accessToken;
2020-05-13 02:10:35 +00:00
private readonly Jellyfin.Data.Entities.User _user;
2016-10-29 22:22:20 +00:00
private readonly IUserDataManager _userDataManager;
private readonly ILocalizationManager _localization;
private readonly IMediaSourceManager _mediaSourceManager;
private readonly ILogger _logger;
private readonly IMediaEncoder _mediaEncoder;
2020-04-02 14:49:58 +00:00
private readonly ILibraryManager _libraryManager;
2016-10-29 22:22:20 +00:00
public DidlBuilder(
DeviceProfile profile,
2020-05-13 02:10:35 +00:00
Jellyfin.Data.Entities.User user,
IImageProcessor imageProcessor,
string serverAddress,
string accessToken,
IUserDataManager userDataManager,
ILocalizationManager localization,
IMediaSourceManager mediaSourceManager,
ILogger logger,
2020-04-02 14:49:58 +00:00
IMediaEncoder mediaEncoder,
ILibraryManager libraryManager)
2016-10-29 22:22:20 +00:00
{
_profile = profile;
_user = user;
2016-10-29 22:22:20 +00:00
_imageProcessor = imageProcessor;
_serverAddress = serverAddress;
_accessToken = accessToken;
2016-10-29 22:22:20 +00:00
_userDataManager = userDataManager;
_localization = localization;
_mediaSourceManager = mediaSourceManager;
_logger = logger;
_mediaEncoder = mediaEncoder;
2020-04-02 14:49:58 +00:00
_libraryManager = libraryManager;
2016-10-29 22:22:20 +00:00
}
2018-09-12 17:26:21 +00:00
public static string NormalizeDlnaMediaUrl(string url)
{
return url + "&dlnaheaders=true";
}
2020-05-13 02:10:35 +00:00
public string GetItemDidl(BaseItem item, Jellyfin.Data.Entities.User user, BaseItem context, string deviceId, Filter filter, StreamInfo streamInfo)
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
var settings = new XmlWriterSettings
{
Encoding = Encoding.UTF8,
CloseOutput = false,
OmitXmlDeclaration = true,
ConformanceLevel = ConformanceLevel.Fragment
};
2016-10-29 22:22:20 +00:00
2018-09-12 17:26:21 +00:00
using (StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8))
2016-10-29 22:22:20 +00:00
{
2019-01-13 20:37:13 +00:00
using (var writer = XmlWriter.Create(builder, settings))
2018-09-12 17:26:21 +00:00
{
//writer.WriteStartDocument();
2016-11-04 08:31:05 +00:00
2018-09-12 17:26:21 +00:00
writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
2016-10-29 22:22:20 +00:00
2018-09-12 17:26:21 +00:00
writer.WriteAttributeString("xmlns", "dc", null, NS_DC);
writer.WriteAttributeString("xmlns", "dlna", null, NS_DLNA);
writer.WriteAttributeString("xmlns", "upnp", null, NS_UPNP);
//didl.SetAttribute("xmlns:sec", NS_SEC);
2016-10-29 22:22:20 +00:00
2018-09-12 17:26:21 +00:00
WriteXmlRootAttributes(_profile, writer);
2016-10-29 22:22:20 +00:00
2020-04-02 14:49:58 +00:00
WriteItemElement(writer, item, user, context, null, deviceId, filter, streamInfo);
2016-11-04 08:31:05 +00:00
2018-09-12 17:26:21 +00:00
writer.WriteFullEndElement();
//writer.WriteEndDocument();
}
2016-11-04 08:31:05 +00:00
2018-09-12 17:26:21 +00:00
return builder.ToString();
}
2016-10-29 22:22:20 +00:00
}
2016-11-06 17:30:44 +00:00
public static void WriteXmlRootAttributes(DeviceProfile profile, XmlWriter writer)
{
foreach (var att in profile.XmlRootAttributes)
{
var parts = att.Name.Split(new[] { ':' }, StringSplitOptions.RemoveEmptyEntries);
if (parts.Length == 2)
{
writer.WriteAttributeString(parts[0], parts[1], null, att.Value);
}
else
{
writer.WriteAttributeString(att.Name, att.Value);
}
}
}
public void WriteItemElement(
2017-05-26 06:48:54 +00:00
XmlWriter writer,
BaseItem item,
2020-05-13 02:10:35 +00:00
Jellyfin.Data.Entities.User user,
2017-05-26 06:48:54 +00:00
BaseItem context,
StubType? contextStubType,
string deviceId,
Filter filter,
2016-12-26 17:37:49 +00:00
StreamInfo streamInfo = null)
2016-10-29 22:22:20 +00:00
{
var clientId = GetClientId(item, null);
2016-11-04 08:31:05 +00:00
writer.WriteStartElement(string.Empty, "item", NS_DIDL);
writer.WriteAttributeString("restricted", "1");
writer.WriteAttributeString("id", clientId);
2016-10-29 22:22:20 +00:00
if (context != null)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("parentID", GetClientId(context, contextStubType));
2016-10-29 22:22:20 +00:00
}
else
{
var parent = item.DisplayParentId;
2018-09-12 17:26:21 +00:00
if (!parent.Equals(Guid.Empty))
2016-10-29 22:22:20 +00:00
{
2018-09-12 17:26:21 +00:00
writer.WriteAttributeString("parentID", GetClientId(parent, null));
2016-10-29 22:22:20 +00:00
}
}
2016-11-06 17:30:44 +00:00
AddGeneralProperties(item, null, context, writer, filter);
AddSamsungBookmarkInfo(item, user, writer, streamInfo);
2016-10-29 22:22:20 +00:00
// refID?
// storeAttribute(itemNode, object, ClassProperties.REF_ID, false);
2020-04-02 14:49:58 +00:00
if (item is IHasMediaSources)
2016-10-29 22:22:20 +00:00
{
if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase))
{
2020-04-02 14:49:58 +00:00
AddAudioResource(writer, item, deviceId, filter, streamInfo);
2016-10-29 22:22:20 +00:00
}
else if (string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
{
2020-04-02 14:49:58 +00:00
AddVideoResource(writer, item, deviceId, filter, streamInfo);
2016-10-29 22:22:20 +00:00
}
}
2020-04-02 14:49:58 +00:00
AddCover(item, null, writer);
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
2020-04-02 14:49:58 +00:00
private void AddVideoResource(XmlWriter writer, BaseItem video, string deviceId, Filter filter, StreamInfo streamInfo = null)
2016-10-29 22:22:20 +00:00
{
if (streamInfo == null)
{
2017-08-05 19:02:33 +00:00
var sources = _mediaSourceManager.GetStaticMediaSources(video, true, _user);
2016-10-29 22:22:20 +00:00
2018-12-14 19:17:29 +00:00
streamInfo = new StreamBuilder(_mediaEncoder, _logger).BuildVideoItem(new VideoOptions
2016-10-29 22:22:20 +00:00
{
2018-09-12 17:26:21 +00:00
ItemId = video.Id,
MediaSources = sources.ToArray(),
2016-10-29 22:22:20 +00:00
Profile = _profile,
DeviceId = deviceId,
MaxBitrate = _profile.MaxStreamingBitrate
});
}
var targetWidth = streamInfo.TargetWidth;
var targetHeight = streamInfo.TargetHeight;
var contentFeatureList = new ContentFeatureBuilder(_profile).BuildVideoHeader(streamInfo.Container,
2017-09-25 05:06:15 +00:00
streamInfo.TargetVideoCodec.FirstOrDefault(),
streamInfo.TargetAudioCodec.FirstOrDefault(),
2016-10-29 22:22:20 +00:00
targetWidth,
targetHeight,
streamInfo.TargetVideoBitDepth,
streamInfo.TargetVideoBitrate,
streamInfo.TargetTimestamp,
streamInfo.IsDirectStream,
2018-09-12 17:26:21 +00:00
streamInfo.RunTimeTicks ?? 0,
2016-10-29 22:22:20 +00:00
streamInfo.TargetVideoProfile,
streamInfo.TargetVideoLevel,
2018-09-12 17:26:21 +00:00
streamInfo.TargetFramerate ?? 0,
2016-10-29 22:22:20 +00:00
streamInfo.TargetPacketLength,
streamInfo.TranscodeSeekInfo,
streamInfo.IsTargetAnamorphic,
2017-05-29 12:35:59 +00:00
streamInfo.IsTargetInterlaced,
2016-10-29 22:22:20 +00:00
streamInfo.TargetRefFrames,
streamInfo.TargetVideoStreamCount,
streamInfo.TargetAudioStreamCount,
streamInfo.TargetVideoCodecTag,
2016-12-14 20:58:55 +00:00
streamInfo.IsTargetAVC);
2016-10-29 22:22:20 +00:00
foreach (var contentFeature in contentFeatureList)
{
2020-04-02 14:49:58 +00:00
AddVideoResource(writer, filter, contentFeature, streamInfo);
2016-10-29 22:22:20 +00:00
}
var subtitleProfiles = streamInfo.GetSubtitleProfiles(_mediaEncoder, false, _serverAddress, _accessToken);
2016-12-08 15:40:20 +00:00
foreach (var subtitle in subtitleProfiles)
2016-10-29 22:22:20 +00:00
{
if (subtitle.DeliveryMethod != SubtitleDeliveryMethod.External)
{
continue;
}
2016-12-08 15:40:20 +00:00
var subtitleAdded = AddSubtitleElement(writer, subtitle);
2016-10-29 22:22:20 +00:00
2016-12-08 15:40:20 +00:00
if (subtitleAdded && _profile.EnableSingleSubtitleLimit)
{
break;
2016-10-29 22:22:20 +00:00
}
}
}
2016-11-04 08:31:05 +00:00
private bool AddSubtitleElement(XmlWriter writer, SubtitleStreamInfo info)
2016-10-29 22:22:20 +00:00
{
var subtitleProfile = _profile.SubtitleProfiles
.FirstOrDefault(i => string.Equals(info.Format, i.Format, StringComparison.OrdinalIgnoreCase)
&& i.Method == SubtitleDeliveryMethod.External);
2016-10-29 22:22:20 +00:00
if (subtitleProfile == null)
{
return false;
}
var subtitleMode = subtitleProfile.DidlMode;
if (string.Equals(subtitleMode, "CaptionInfoEx", StringComparison.OrdinalIgnoreCase))
{
// <sec:CaptionInfoEx sec:type="srt">http://192.168.1.3:9999/video.srt</sec:CaptionInfoEx>
// <sec:CaptionInfo sec:type="srt">http://192.168.1.3:9999/video.srt</sec:CaptionInfo>
2016-11-04 08:31:05 +00:00
writer.WriteStartElement("sec", "CaptionInfoEx", null);
2019-01-27 11:03:43 +00:00
writer.WriteAttributeString("sec", "type", null, info.Format.ToLowerInvariant());
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteString(info.Url);
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
else if (string.Equals(subtitleMode, "smi", StringComparison.OrdinalIgnoreCase))
{
2016-11-04 08:31:05 +00:00
writer.WriteStartElement(string.Empty, "res", NS_DIDL);
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("protocolInfo", "http-get:*:smi/caption:*");
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteString(info.Url);
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
else
{
2016-11-04 08:31:05 +00:00
writer.WriteStartElement(string.Empty, "res", NS_DIDL);
2020-04-02 14:49:58 +00:00
var protocolInfo = string.Format(
CultureInfo.InvariantCulture,
"http-get:*:text/{0}:*",
info.Format.ToLowerInvariant());
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("protocolInfo", protocolInfo);
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteString(info.Url);
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
return true;
}
2020-04-02 14:49:58 +00:00
private void AddVideoResource(XmlWriter writer, Filter filter, string contentFeatures, StreamInfo streamInfo)
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
writer.WriteStartElement(string.Empty, "res", NS_DIDL);
2016-10-29 22:22:20 +00:00
2018-09-12 17:26:21 +00:00
var url = NormalizeDlnaMediaUrl(streamInfo.ToUrl(_serverAddress, _accessToken));
2016-10-29 22:22:20 +00:00
var mediaSource = streamInfo.MediaSource;
if (mediaSource.RunTimeTicks.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
2016-10-29 22:22:20 +00:00
}
if (filter.Contains("res@size"))
{
if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
{
var size = streamInfo.TargetSize;
if (size.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("size", size.Value.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
}
}
}
var totalBitrate = streamInfo.TargetTotalBitrate;
var targetSampleRate = streamInfo.TargetAudioSampleRate;
var targetChannels = streamInfo.TargetAudioChannels;
var targetWidth = streamInfo.TargetWidth;
var targetHeight = streamInfo.TargetHeight;
if (targetChannels.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
}
if (filter.Contains("res@resolution"))
{
if (targetWidth.HasValue && targetHeight.HasValue)
{
2020-04-02 14:49:58 +00:00
writer.WriteAttributeString(
"resolution",
string.Format(
CultureInfo.InvariantCulture,
"{0}x{1}",
targetWidth.Value,
targetHeight.Value));
2016-10-29 22:22:20 +00:00
}
}
if (targetSampleRate.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
}
if (totalBitrate.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("bitrate", totalBitrate.Value.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
}
var mediaProfile = _profile.GetVideoMediaProfile(streamInfo.Container,
2017-09-25 05:06:15 +00:00
streamInfo.TargetAudioCodec.FirstOrDefault(),
streamInfo.TargetVideoCodec.FirstOrDefault(),
2016-10-29 22:22:20 +00:00
streamInfo.TargetAudioBitrate,
targetWidth,
targetHeight,
streamInfo.TargetVideoBitDepth,
streamInfo.TargetVideoProfile,
streamInfo.TargetVideoLevel,
2018-09-12 17:26:21 +00:00
streamInfo.TargetFramerate ?? 0,
2016-10-29 22:22:20 +00:00
streamInfo.TargetPacketLength,
streamInfo.TargetTimestamp,
streamInfo.IsTargetAnamorphic,
2017-05-29 12:35:59 +00:00
streamInfo.IsTargetInterlaced,
2016-10-29 22:22:20 +00:00
streamInfo.TargetRefFrames,
streamInfo.TargetVideoStreamCount,
streamInfo.TargetAudioStreamCount,
streamInfo.TargetVideoCodecTag,
2016-12-14 20:58:55 +00:00
streamInfo.IsTargetAVC);
2016-10-29 22:22:20 +00:00
2020-04-02 14:49:58 +00:00
var filename = url.Substring(0, url.IndexOf('?', StringComparison.Ordinal));
2016-10-29 22:22:20 +00:00
var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
2019-02-13 15:37:18 +00:00
? MimeTypes.GetMimeType(filename)
2016-10-29 22:22:20 +00:00
: mediaProfile.MimeType;
2020-04-02 14:49:58 +00:00
writer.WriteAttributeString(
"protocolInfo",
string.Format(
CultureInfo.InvariantCulture,
"http-get:*:{0}:{1}",
mimeType,
contentFeatures));
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteString(url);
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
private string GetDisplayName(BaseItem item, StubType? itemStubType, BaseItem context)
{
if (itemStubType.HasValue)
2017-07-22 23:00:48 +00:00
{
switch (itemStubType.Value)
{
2020-03-24 15:12:06 +00:00
case StubType.Latest: return _localization.GetLocalizedString("Latest");
case StubType.Playlists: return _localization.GetLocalizedString("Playlists");
case StubType.AlbumArtists: return _localization.GetLocalizedString("HeaderAlbumArtists");
case StubType.Albums: return _localization.GetLocalizedString("Albums");
case StubType.Artists: return _localization.GetLocalizedString("Artists");
case StubType.Songs: return _localization.GetLocalizedString("Songs");
case StubType.Genres: return _localization.GetLocalizedString("Genres");
case StubType.FavoriteAlbums: return _localization.GetLocalizedString("HeaderFavoriteAlbums");
case StubType.FavoriteArtists: return _localization.GetLocalizedString("HeaderFavoriteArtists");
case StubType.FavoriteSongs: return _localization.GetLocalizedString("HeaderFavoriteSongs");
case StubType.ContinueWatching: return _localization.GetLocalizedString("HeaderContinueWatching");
2020-03-24 15:12:06 +00:00
case StubType.Movies: return _localization.GetLocalizedString("Movies");
case StubType.Collections: return _localization.GetLocalizedString("Collections");
case StubType.Favorites: return _localization.GetLocalizedString("Favorites");
case StubType.NextUp: return _localization.GetLocalizedString("HeaderNextUp");
case StubType.FavoriteSeries: return _localization.GetLocalizedString("HeaderFavoriteShows");
case StubType.FavoriteEpisodes: return _localization.GetLocalizedString("HeaderFavoriteEpisodes");
2020-03-24 15:12:06 +00:00
case StubType.Series: return _localization.GetLocalizedString("Shows");
default: break;
}
}
2016-10-29 22:22:20 +00:00
return item is Episode episode
? GetEpisodeDisplayName(episode, context)
: item.Name;
}
/// <summary>
/// Gets episode display name appropriate for the given context.
/// </summary>
/// <remarks>
/// If context is a season, this will return a string containing just episode number and name.
/// Otherwise the result will include series nams and season number.
/// </remarks>
/// <param name="episode">The episode.</param>
/// <param name="context">Current context.</param>
/// <returns>Formatted name of the episode.</returns>
private string GetEpisodeDisplayName(Episode episode, BaseItem context)
{
string[] components;
if (context is Season season)
2016-10-29 22:22:20 +00:00
{
// This is a special embedded within a season
if (episode.ParentIndexNumber.HasValue && episode.ParentIndexNumber.Value == 0
&& season.IndexNumber.HasValue && season.IndexNumber.Value != 0)
2016-10-29 22:22:20 +00:00
{
2020-04-02 14:49:58 +00:00
return string.Format(
CultureInfo.InvariantCulture,
_localization.GetLocalizedString("ValueSpecialEpisodeName"),
episode.Name);
2016-10-29 22:22:20 +00:00
}
// inside a season use simple format (ex. '12 - Episode Name')
var epNumberName = GetEpisodeIndexFullName(episode);
components = new[] { epNumberName, episode.Name };
}
else
{
// outside a season include series and season details (ex. 'TV Show - S05E11 - Episode Name')
var epNumberName = GetEpisodeNumberDisplayName(episode);
components = new[] { episode.SeriesName, epNumberName, episode.Name };
}
2016-10-29 22:22:20 +00:00
return string.Join(" - ", components.Where(NotNullOrWhiteSpace));
}
2016-10-29 22:22:20 +00:00
/// <summary>
/// Gets complete episode number.
/// </summary>
/// <param name="episode">The episode.</param>
/// <returns>For single episodes returns just the number. For double episodes - current and ending numbers.</returns>
private string GetEpisodeIndexFullName(Episode episode)
{
var name = string.Empty;
if (episode.IndexNumber.HasValue)
{
name += episode.IndexNumber.Value.ToString("00", CultureInfo.InvariantCulture);
if (episode.IndexNumberEnd.HasValue)
{
name += "-" + episode.IndexNumberEnd.Value.ToString("00", CultureInfo.InvariantCulture);
}
}
return name;
}
/// <summary>
/// Gets episode number formatted as 'S##E##'.
/// </summary>
/// <param name="episode">The episode.</param>
/// <returns>Formatted episode number.</returns>
private string GetEpisodeNumberDisplayName(Episode episode)
{
var name = string.Empty;
var seasonNumber = episode.Season?.IndexNumber;
if (seasonNumber.HasValue)
{
name = "S" + seasonNumber.Value.ToString("00", CultureInfo.InvariantCulture);
}
2016-10-29 22:22:20 +00:00
var indexName = GetEpisodeIndexFullName(episode);
if (!string.IsNullOrWhiteSpace(indexName))
{
name += "E" + indexName;
}
return name;
2016-10-29 22:22:20 +00:00
}
private bool NotNullOrWhiteSpace(string s) => !string.IsNullOrWhiteSpace(s);
2020-04-02 14:49:58 +00:00
private void AddAudioResource(XmlWriter writer, BaseItem audio, string deviceId, Filter filter, StreamInfo streamInfo = null)
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
writer.WriteStartElement(string.Empty, "res", NS_DIDL);
2016-10-29 22:22:20 +00:00
if (streamInfo == null)
{
2017-08-05 19:02:33 +00:00
var sources = _mediaSourceManager.GetStaticMediaSources(audio, true, _user);
2016-10-29 22:22:20 +00:00
2018-12-14 19:17:29 +00:00
streamInfo = new StreamBuilder(_mediaEncoder, _logger).BuildAudioItem(new AudioOptions
2016-11-04 08:31:05 +00:00
{
2018-09-12 17:26:21 +00:00
ItemId = audio.Id,
2018-12-28 15:48:26 +00:00
MediaSources = sources.ToArray(),
2016-11-04 08:31:05 +00:00
Profile = _profile,
DeviceId = deviceId
});
2016-10-29 22:22:20 +00:00
}
2018-09-12 17:26:21 +00:00
var url = NormalizeDlnaMediaUrl(streamInfo.ToUrl(_serverAddress, _accessToken));
2016-10-29 22:22:20 +00:00
var mediaSource = streamInfo.MediaSource;
if (mediaSource.RunTimeTicks.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
2016-10-29 22:22:20 +00:00
}
if (filter.Contains("res@size"))
{
if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
{
var size = streamInfo.TargetSize;
if (size.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("size", size.Value.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
}
}
}
var targetAudioBitrate = streamInfo.TargetAudioBitrate;
var targetSampleRate = streamInfo.TargetAudioSampleRate;
var targetChannels = streamInfo.TargetAudioChannels;
2017-06-26 15:10:52 +00:00
var targetAudioBitDepth = streamInfo.TargetAudioBitDepth;
2016-10-29 22:22:20 +00:00
if (targetChannels.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
}
if (targetSampleRate.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
}
if (targetAudioBitrate.HasValue)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("bitrate", targetAudioBitrate.Value.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
}
var mediaProfile = _profile.GetAudioMediaProfile(streamInfo.Container,
2017-09-25 05:06:15 +00:00
streamInfo.TargetAudioCodec.FirstOrDefault(),
2016-10-29 22:22:20 +00:00
targetChannels,
2017-05-13 19:31:25 +00:00
targetAudioBitrate,
2017-06-26 15:10:52 +00:00
targetSampleRate,
targetAudioBitDepth);
2016-10-29 22:22:20 +00:00
2020-04-02 14:49:58 +00:00
var filename = url.Substring(0, url.IndexOf('?', StringComparison.Ordinal));
2016-10-29 22:22:20 +00:00
var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
2019-02-13 15:37:18 +00:00
? MimeTypes.GetMimeType(filename)
2016-10-29 22:22:20 +00:00
: mediaProfile.MimeType;
var contentFeatures = new ContentFeatureBuilder(_profile).BuildAudioHeader(streamInfo.Container,
2017-09-25 05:06:15 +00:00
streamInfo.TargetAudioCodec.FirstOrDefault(),
2016-10-29 22:22:20 +00:00
targetAudioBitrate,
targetSampleRate,
targetChannels,
2017-06-26 15:10:52 +00:00
targetAudioBitDepth,
2016-10-29 22:22:20 +00:00
streamInfo.IsDirectStream,
2018-09-12 17:26:21 +00:00
streamInfo.RunTimeTicks ?? 0,
2016-10-29 22:22:20 +00:00
streamInfo.TranscodeSeekInfo);
2020-04-02 14:49:58 +00:00
writer.WriteAttributeString(
"protocolInfo",
string.Format(
CultureInfo.InvariantCulture,
"http-get:*:{0}:{1}",
mimeType,
contentFeatures));
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteString(url);
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
public static bool IsIdRoot(string id)
2019-02-13 15:37:18 +00:00
=> string.IsNullOrWhiteSpace(id)
|| string.Equals(id, "0", StringComparison.OrdinalIgnoreCase)
2016-10-29 22:22:20 +00:00
// Samsung sometimes uses 1 as root
2019-02-13 15:37:18 +00:00
|| string.Equals(id, "1", StringComparison.OrdinalIgnoreCase);
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
public void WriteFolderElement(XmlWriter writer, BaseItem folder, StubType? stubType, BaseItem context, int childCount, Filter filter, string requestedId = null)
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
writer.WriteStartElement(string.Empty, "container", NS_DIDL);
2018-09-12 17:26:21 +00:00
writer.WriteAttributeString("restricted", "1");
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("searchable", "1");
writer.WriteAttributeString("childCount", childCount.ToString(_usCulture));
2016-10-29 22:22:20 +00:00
var clientId = GetClientId(folder, stubType);
2020-04-02 14:49:58 +00:00
if (string.Equals(requestedId, "0", StringComparison.Ordinal))
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("id", "0");
writer.WriteAttributeString("parentID", "-1");
2016-10-29 22:22:20 +00:00
}
else
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("id", clientId);
2016-10-29 22:22:20 +00:00
if (context != null)
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("parentID", GetClientId(context, null));
2016-10-29 22:22:20 +00:00
}
else
{
var parent = folder.DisplayParentId;
2018-09-12 17:26:21 +00:00
if (parent.Equals(Guid.Empty))
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("parentID", "0");
2016-10-29 22:22:20 +00:00
}
else
{
2018-09-12 17:26:21 +00:00
writer.WriteAttributeString("parentID", GetClientId(parent, null));
2016-10-29 22:22:20 +00:00
}
}
}
2016-11-06 17:30:44 +00:00
AddGeneralProperties(folder, stubType, context, writer, filter);
2020-04-02 14:49:58 +00:00
AddCover(folder, stubType, writer);
2016-10-29 22:22:20 +00:00
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
2020-05-13 02:10:35 +00:00
private void AddSamsungBookmarkInfo(BaseItem item, Jellyfin.Data.Entities.User user, XmlWriter writer, StreamInfo streamInfo)
2016-12-26 17:37:49 +00:00
{
if (!item.SupportsPositionTicksResume || item is Folder)
{
return;
}
MediaBrowser.Model.Dlna.XmlAttribute secAttribute = null;
2016-12-26 17:37:49 +00:00
foreach (var attribute in _profile.XmlRootAttributes)
{
if (string.Equals(attribute.Name, "xmlns:sec", StringComparison.OrdinalIgnoreCase))
{
secAttribute = attribute;
break;
}
}
// Not a samsung device
if (secAttribute == null)
{
return;
}
2018-09-12 17:26:21 +00:00
var userdata = _userDataManager.GetUserData(user, item);
var playbackPositionTicks = (streamInfo != null && streamInfo.StartPositionTicks > 0) ? streamInfo.StartPositionTicks : userdata.PlaybackPositionTicks;
2016-12-26 17:37:49 +00:00
if (playbackPositionTicks > 0)
2016-12-26 17:37:49 +00:00
{
2020-04-02 14:49:58 +00:00
var elementValue = string.Format(
CultureInfo.InvariantCulture,
"BM={0}",
Convert.ToInt32(TimeSpan.FromTicks(playbackPositionTicks).TotalSeconds));
2016-12-26 17:37:49 +00:00
AddValue(writer, "sec", "dcmInfo", elementValue, secAttribute.Value);
}
}
2016-10-29 22:22:20 +00:00
/// <summary>
/// Adds fields used by both items and folders
/// </summary>
2016-11-04 08:31:05 +00:00
private void AddCommonFields(BaseItem item, StubType? itemStubType, BaseItem context, XmlWriter writer, Filter filter)
2016-10-29 22:22:20 +00:00
{
// Don't filter on dc:title because not all devices will include it in the filter
// MediaMonkey for example won't display content without a title
//if (filter.Contains("dc:title"))
{
2016-11-04 08:31:05 +00:00
AddValue(writer, "dc", "title", GetDisplayName(item, itemStubType, context), NS_DC);
2016-10-29 22:22:20 +00:00
}
2016-11-04 08:31:05 +00:00
WriteObjectClass(writer, item, itemStubType);
2016-10-29 22:22:20 +00:00
if (filter.Contains("dc:date"))
{
if (item.PremiereDate.HasValue)
{
2020-01-27 22:34:40 +00:00
AddValue(writer, "dc", "date", item.PremiereDate.Value.ToString("o", CultureInfo.InvariantCulture), NS_DC);
2016-10-29 22:22:20 +00:00
}
}
if (filter.Contains("upnp:genre"))
{
foreach (var genre in item.Genres)
{
2016-11-04 08:31:05 +00:00
AddValue(writer, "upnp", "genre", genre, NS_UPNP);
2016-10-29 22:22:20 +00:00
}
}
foreach (var studio in item.Studios)
{
2016-11-04 08:31:05 +00:00
AddValue(writer, "upnp", "publisher", studio, NS_UPNP);
2016-10-29 22:22:20 +00:00
}
2018-09-12 17:26:21 +00:00
if (!(item is Folder))
2016-10-29 22:22:20 +00:00
{
2018-09-12 17:26:21 +00:00
if (filter.Contains("dc:description"))
2016-10-29 22:22:20 +00:00
{
2018-09-12 17:26:21 +00:00
var desc = item.Overview;
if (!string.IsNullOrWhiteSpace(desc))
{
AddValue(writer, "dc", "description", desc, NS_DC);
}
2016-10-29 22:22:20 +00:00
}
2018-09-12 17:26:21 +00:00
//if (filter.Contains("upnp:longDescription"))
//{
// if (!string.IsNullOrWhiteSpace(item.Overview))
// {
// AddValue(writer, "upnp", "longDescription", item.Overview, NS_UPNP);
// }
//}
2016-10-29 22:22:20 +00:00
}
if (!string.IsNullOrEmpty(item.OfficialRating))
{
if (filter.Contains("dc:rating"))
{
2016-11-04 08:31:05 +00:00
AddValue(writer, "dc", "rating", item.OfficialRating, NS_DC);
2016-10-29 22:22:20 +00:00
}
if (filter.Contains("upnp:rating"))
{
2016-11-04 08:31:05 +00:00
AddValue(writer, "upnp", "rating", item.OfficialRating, NS_UPNP);
2016-10-29 22:22:20 +00:00
}
}
2016-11-04 08:31:05 +00:00
AddPeople(item, writer);
2016-10-29 22:22:20 +00:00
}
2016-11-04 08:31:05 +00:00
private void WriteObjectClass(XmlWriter writer, BaseItem item, StubType? stubType)
2016-10-29 22:22:20 +00:00
{
// More types here
// http://oss.linn.co.uk/repos/Public/LibUpnpCil/DidlLite/UpnpAv/Test/TestDidlLite.cs
2016-11-04 08:31:05 +00:00
writer.WriteStartElement("upnp", "class", NS_UPNP);
2016-10-29 22:22:20 +00:00
2017-01-06 04:38:03 +00:00
if (item.IsDisplayedAsFolder || stubType.HasValue)
2016-10-29 22:22:20 +00:00
{
string classType = null;
if (!_profile.RequiresPlainFolders)
{
if (item is MusicAlbum)
{
classType = "object.container.album.musicAlbum";
}
else if (item is MusicArtist)
{
classType = "object.container.person.musicArtist";
}
else if (item is Series || item is Season || item is BoxSet || item is Video)
{
classType = "object.container.album.videoAlbum";
}
else if (item is Playlist)
{
classType = "object.container.playlistContainer";
}
else if (item is PhotoAlbum)
{
classType = "object.container.album.photoAlbum";
}
}
2016-11-04 08:31:05 +00:00
writer.WriteString(classType ?? "object.container.storageFolder");
2016-10-29 22:22:20 +00:00
}
else if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase))
{
2016-11-04 08:31:05 +00:00
writer.WriteString("object.item.audioItem.musicTrack");
2016-10-29 22:22:20 +00:00
}
else if (string.Equals(item.MediaType, MediaType.Photo, StringComparison.OrdinalIgnoreCase))
{
2016-11-04 08:31:05 +00:00
writer.WriteString("object.item.imageItem.photo");
2016-10-29 22:22:20 +00:00
}
else if (string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
{
if (!_profile.RequiresPlainVideoItems && item is Movie)
{
2016-11-04 08:31:05 +00:00
writer.WriteString("object.item.videoItem.movie");
2016-10-29 22:22:20 +00:00
}
else if (!_profile.RequiresPlainVideoItems && item is MusicVideo)
{
2016-11-04 08:31:05 +00:00
writer.WriteString("object.item.videoItem.musicVideoClip");
2016-10-29 22:22:20 +00:00
}
else
{
2016-11-04 08:31:05 +00:00
writer.WriteString("object.item.videoItem");
2016-10-29 22:22:20 +00:00
}
}
else if (item is MusicGenre)
{
2016-11-04 08:31:05 +00:00
writer.WriteString(_profile.RequiresPlainFolders ? "object.container.storageFolder" : "object.container.genre.musicGenre");
2016-10-29 22:22:20 +00:00
}
else if (item is Genre)
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
writer.WriteString(_profile.RequiresPlainFolders ? "object.container.storageFolder" : "object.container.genre");
2016-10-29 22:22:20 +00:00
}
else
{
2016-11-04 08:31:05 +00:00
writer.WriteString("object.item");
2016-10-29 22:22:20 +00:00
}
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
2016-11-04 08:31:05 +00:00
private void AddPeople(BaseItem item, XmlWriter writer)
2016-10-29 22:22:20 +00:00
{
2020-04-03 15:30:01 +00:00
if (!item.SupportsPeople)
{
return;
}
2016-10-29 22:22:20 +00:00
2020-04-02 14:49:58 +00:00
var types = new[]
{
PersonType.Director,
PersonType.Writer,
PersonType.Producer,
PersonType.Composer,
2020-04-03 15:30:01 +00:00
"creator"
2020-04-02 14:49:58 +00:00
};
2016-10-29 22:22:20 +00:00
2020-04-03 21:20:04 +00:00
// Seeing some LG models locking up due content with large lists of people
// The actual issue might just be due to processing a more metadata than it can handle
2020-04-03 15:30:01 +00:00
var people = _libraryManager.GetPeople(
new InternalPeopleQuery
{
ItemId = item.Id,
Limit = 6
});
2016-10-29 22:22:20 +00:00
2020-04-02 14:49:58 +00:00
foreach (var actor in people)
{
var type = types.FirstOrDefault(i => string.Equals(i, actor.Type, StringComparison.OrdinalIgnoreCase) || string.Equals(i, actor.Role, StringComparison.OrdinalIgnoreCase))
?? PersonType.Actor;
2016-10-29 22:22:20 +00:00
2020-04-02 14:49:58 +00:00
AddValue(writer, "upnp", type.ToLowerInvariant(), actor.Name, NS_UPNP);
}
2016-10-29 22:22:20 +00:00
}
2016-11-04 08:31:05 +00:00
private void AddGeneralProperties(BaseItem item, StubType? itemStubType, BaseItem context, XmlWriter writer, Filter filter)
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
AddCommonFields(item, itemStubType, context, writer, filter);
2016-10-29 22:22:20 +00:00
2017-10-27 03:50:17 +00:00
var hasAlbumArtists = item as IHasAlbumArtist;
2016-10-29 22:22:20 +00:00
if (item is IHasArtist hasArtists)
2016-10-29 22:22:20 +00:00
{
2017-10-27 03:50:17 +00:00
foreach (var artist in hasArtists.Artists)
2016-10-29 22:22:20 +00:00
{
2016-11-04 08:31:05 +00:00
AddValue(writer, "upnp", "artist", artist, NS_UPNP);
2017-10-27 03:50:17 +00:00
AddValue(writer, "dc", "creator", artist, NS_DC);
2016-10-29 22:22:20 +00:00
2017-10-27 03:50:17 +00:00
// If it doesn't support album artists (musicvideo), then tag as both
if (hasAlbumArtists == null)
{
AddAlbumArtist(writer, artist);
}
2016-10-29 22:22:20 +00:00
}
}
2017-10-27 03:50:17 +00:00
if (hasAlbumArtists != null)
2016-10-29 22:22:20 +00:00
{
2017-10-27 03:50:17 +00:00
foreach (var albumArtist in hasAlbumArtists.AlbumArtists)
2016-10-29 22:22:20 +00:00
{
2017-10-27 03:50:17 +00:00
AddAlbumArtist(writer, albumArtist);
2016-10-29 22:22:20 +00:00
}
}
2017-10-27 03:50:17 +00:00
if (!string.IsNullOrWhiteSpace(item.Album))
2016-10-29 22:22:20 +00:00
{
2017-10-27 03:50:17 +00:00
AddValue(writer, "upnp", "album", item.Album, NS_UPNP);
2016-10-29 22:22:20 +00:00
}
if (item.IndexNumber.HasValue)
{
2016-11-04 08:31:05 +00:00
AddValue(writer, "upnp", "originalTrackNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
2016-10-29 22:22:20 +00:00
if (item is Episode)
{
2016-11-04 08:31:05 +00:00
AddValue(writer, "upnp", "episodeNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
2016-10-29 22:22:20 +00:00
}
}
}
2016-11-04 08:31:05 +00:00
private void AddAlbumArtist(XmlWriter writer, string name)
2016-10-29 22:22:20 +00:00
{
try
{
2016-11-04 08:31:05 +00:00
writer.WriteStartElement("upnp", "artist", NS_UPNP);
writer.WriteAttributeString("role", "AlbumArtist");
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteString(name);
2016-10-29 22:22:20 +00:00
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
2018-12-20 12:11:26 +00:00
catch (XmlException ex)
2016-10-29 22:22:20 +00:00
{
2018-12-20 12:11:26 +00:00
_logger.LogError(ex, "Error adding xml value: {value}", name);
2016-10-29 22:22:20 +00:00
}
}
2016-11-04 08:31:05 +00:00
private void AddValue(XmlWriter writer, string prefix, string name, string value, string namespaceUri)
2016-10-29 22:22:20 +00:00
{
try
{
2016-11-04 08:31:05 +00:00
writer.WriteElementString(prefix, name, namespaceUri, value);
2016-10-29 22:22:20 +00:00
}
2018-12-20 12:11:26 +00:00
catch (XmlException ex)
2016-10-29 22:22:20 +00:00
{
2018-12-20 12:11:26 +00:00
_logger.LogError(ex, "Error adding xml value: {value}", value);
2016-10-29 22:22:20 +00:00
}
}
2020-04-02 14:49:58 +00:00
private void AddCover(BaseItem item, StubType? stubType, XmlWriter writer)
2016-10-29 22:22:20 +00:00
{
2019-01-25 21:51:34 +00:00
ImageDownloadInfo imageInfo = GetImageInfo(item);
2016-10-29 22:22:20 +00:00
if (imageInfo == null)
{
return;
}
2018-09-12 17:26:21 +00:00
var albumartUrlInfo = GetImageUrl(imageInfo, _profile.MaxAlbumArtWidth, _profile.MaxAlbumArtHeight, "jpg");
2016-10-29 22:22:20 +00:00
2020-01-27 22:34:40 +00:00
writer.WriteStartElement("upnp", "albumArtURI", NS_UPNP);
2016-11-04 08:31:05 +00:00
writer.WriteAttributeString("dlna", "profileID", NS_DLNA, _profile.AlbumArtPn);
writer.WriteString(albumartUrlInfo.Url);
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
// TOOD: Remove these default values
2018-09-12 17:26:21 +00:00
var iconUrlInfo = GetImageUrl(imageInfo, _profile.MaxIconWidth ?? 48, _profile.MaxIconHeight ?? 48, "jpg");
2016-11-04 08:31:05 +00:00
writer.WriteElementString("upnp", "icon", NS_UPNP, iconUrlInfo.Url);
2016-10-29 22:22:20 +00:00
if (!_profile.EnableAlbumArtInDidl)
{
if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase)
|| string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
{
if (!stubType.HasValue)
{
return;
}
}
}
2018-09-12 17:26:21 +00:00
if (!_profile.EnableSingleAlbumArtLimit || string.Equals(item.MediaType, MediaType.Photo, StringComparison.OrdinalIgnoreCase))
2016-10-29 22:22:20 +00:00
{
2018-09-12 17:26:21 +00:00
AddImageResElement(item, writer, 4096, 4096, "jpg", "JPEG_LRG");
AddImageResElement(item, writer, 1024, 768, "jpg", "JPEG_MED");
AddImageResElement(item, writer, 640, 480, "jpg", "JPEG_SM");
AddImageResElement(item, writer, 4096, 4096, "png", "PNG_LRG");
AddImageResElement(item, writer, 160, 160, "png", "PNG_TN");
2016-10-29 22:22:20 +00:00
}
AddImageResElement(item, writer, 160, 160, "jpg", "JPEG_TN");
2016-10-29 22:22:20 +00:00
}
2020-04-02 14:49:58 +00:00
private void AddImageResElement(
BaseItem item,
2016-11-04 08:31:05 +00:00
XmlWriter writer,
2016-10-29 22:22:20 +00:00
int maxWidth,
int maxHeight,
string format,
string org_Pn)
{
var imageInfo = GetImageInfo(item);
if (imageInfo == null)
{
return;
}
2018-09-12 17:26:21 +00:00
var albumartUrlInfo = GetImageUrl(imageInfo, maxWidth, maxHeight, format);
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteStartElement(string.Empty, "res", NS_DIDL);
2016-10-29 22:22:20 +00:00
2017-10-17 19:49:55 +00:00
// Images must have a reported size or many clients (Bubble upnp), will only use the first thumbnail
// rather than using a larger one when available
var width = albumartUrlInfo.Width ?? maxWidth;
var height = albumartUrlInfo.Height ?? maxHeight;
2016-10-29 22:22:20 +00:00
var contentFeatures = new ContentFeatureBuilder(_profile)
.BuildImageHeader(format, width, height, imageInfo.IsDirectStream, org_Pn);
2020-04-02 14:49:58 +00:00
writer.WriteAttributeString(
"protocolInfo",
string.Format(
CultureInfo.InvariantCulture,
"http-get:*:{0}:{1}",
MimeTypes.GetMimeType("file." + format),
contentFeatures));
2016-10-29 22:22:20 +00:00
2020-04-02 14:49:58 +00:00
writer.WriteAttributeString(
"resolution",
string.Format(CultureInfo.InvariantCulture, "{0}x{1}", width, height));
2016-10-29 22:22:20 +00:00
2016-11-04 08:31:05 +00:00
writer.WriteString(albumartUrlInfo.Url);
2016-11-06 17:30:44 +00:00
writer.WriteFullEndElement();
2016-10-29 22:22:20 +00:00
}
private ImageDownloadInfo GetImageInfo(BaseItem item)
{
if (item.HasImage(ImageType.Primary))
{
return GetImageInfo(item, ImageType.Primary);
}
if (item.HasImage(ImageType.Thumb))
{
return GetImageInfo(item, ImageType.Thumb);
}
if (item.HasImage(ImageType.Backdrop))
{
if (item is Channel)
{
return GetImageInfo(item, ImageType.Backdrop);
}
}
// For audio tracks without art use album art if available.
if (item is Audio audioItem)
{
var album = audioItem.AlbumEntity;
return album != null && album.HasImage(ImageType.Primary)
? GetImageInfo(album, ImageType.Primary)
: null;
}
2016-10-29 22:22:20 +00:00
// Don't look beyond album/playlist level. Metadata service may assign an image from a different album/show to the parent folder.
if (item is MusicAlbum || item is Playlist)
2016-10-29 22:22:20 +00:00
{
return null;
}
// For other item types check parents, but be aware that image retrieved from a parent may be not suitable for this media item.
var parentWithImage = GetFirstParentWithImageBelowUserRoot(item);
if (parentWithImage != null)
{
return GetImageInfo(parentWithImage, ImageType.Primary);
2016-10-29 22:22:20 +00:00
}
return null;
}
private BaseItem GetFirstParentWithImageBelowUserRoot(BaseItem item)
{
if (item == null)
{
return null;
}
if (item.HasImage(ImageType.Primary))
{
return item;
}
var parent = item.GetParent();
if (parent is UserRootFolder)
{
return null;
}
// terminate in case we went past user root folder (unlikely?)
if (parent is Folder folder && folder.IsRoot)
{
return null;
}
return GetFirstParentWithImageBelowUserRoot(parent);
}
2016-10-29 22:22:20 +00:00
private ImageDownloadInfo GetImageInfo(BaseItem item, ImageType type)
{
var imageInfo = item.GetImageInfo(type, 0);
string tag = null;
try
{
tag = _imageProcessor.GetImageCacheTag(item, type);
}
2018-12-20 12:39:58 +00:00
catch (Exception ex)
2016-10-29 22:22:20 +00:00
{
2018-12-20 12:39:58 +00:00
_logger.LogError(ex, "Error getting image cache tag");
2016-10-29 22:22:20 +00:00
}
2017-10-22 06:22:43 +00:00
int? width = imageInfo.Width;
int? height = imageInfo.Height;
if (width == 0 || height == 0)
{
//_imageProcessor.GetImageSize(item, imageInfo);
width = null;
height = null;
}
else if (width == -1 || height == -1)
{
width = null;
height = null;
}
2016-10-29 22:22:20 +00:00
2017-09-22 20:33:01 +00:00
//try
//{
// var size = _imageProcessor.GetImageSize(imageInfo);
2016-10-29 22:22:20 +00:00
// width = size.Width;
// height = size.Height;
2017-09-22 20:33:01 +00:00
//}
//catch
//{
2016-10-29 22:22:20 +00:00
2017-09-22 20:33:01 +00:00
//}
2016-10-29 22:22:20 +00:00
var inputFormat = (Path.GetExtension(imageInfo.Path) ?? string.Empty)
.TrimStart('.')
.Replace("jpeg", "jpg", StringComparison.OrdinalIgnoreCase);
return new ImageDownloadInfo
{
2018-09-12 17:26:21 +00:00
ItemId = item.Id,
2016-10-29 22:22:20 +00:00
Type = type,
ImageTag = tag,
Width = width,
Height = height,
Format = inputFormat,
ItemImageInfo = imageInfo
};
}
private class ImageDownloadInfo
2016-10-29 22:22:20 +00:00
{
2018-09-12 17:26:21 +00:00
internal Guid ItemId;
2016-10-29 22:22:20 +00:00
internal string ImageTag;
internal ImageType Type;
internal int? Width;
internal int? Height;
internal bool IsDirectStream;
internal string Format;
internal ItemImageInfo ItemImageInfo;
}
private class ImageUrlInfo
2016-10-29 22:22:20 +00:00
{
internal string Url;
internal int? Width;
internal int? Height;
}
public static string GetClientId(BaseItem item, StubType? stubType)
{
return GetClientId(item.Id, stubType);
}
public static string GetClientId(Guid idValue, StubType? stubType)
{
var id = idValue.ToString("N", CultureInfo.InvariantCulture);
2016-10-29 22:22:20 +00:00
if (stubType.HasValue)
{
2019-01-27 11:03:43 +00:00
id = stubType.Value.ToString().ToLowerInvariant() + "_" + id;
2016-10-29 22:22:20 +00:00
}
return id;
}
2018-09-12 17:26:21 +00:00
private ImageUrlInfo GetImageUrl(ImageDownloadInfo info, int maxWidth, int maxHeight, string format)
2016-10-29 22:22:20 +00:00
{
2020-04-02 14:49:58 +00:00
var url = string.Format(
CultureInfo.InvariantCulture,
"{0}/Items/{1}/Images/{2}/0/{3}/{4}/{5}/{6}/0/0",
2016-10-29 22:22:20 +00:00
_serverAddress,
info.ItemId.ToString("N", CultureInfo.InvariantCulture),
2016-10-29 22:22:20 +00:00
info.Type,
info.ImageTag,
format,
maxWidth.ToString(CultureInfo.InvariantCulture),
maxHeight.ToString(CultureInfo.InvariantCulture));
2016-10-29 22:22:20 +00:00
var width = info.Width;
var height = info.Height;
info.IsDirectStream = false;
if (width.HasValue && height.HasValue)
{
var newSize = DrawingUtils.Resize(
new ImageDimensions(width.Value, height.Value), 0, 0, maxWidth, maxHeight);
2016-10-29 22:22:20 +00:00
width = newSize.Width;
height = newSize.Height;
2016-10-29 22:22:20 +00:00
var normalizedFormat = format
.Replace("jpeg", "jpg", StringComparison.OrdinalIgnoreCase);
if (string.Equals(info.Format, normalizedFormat, StringComparison.OrdinalIgnoreCase))
{
info.IsDirectStream = maxWidth >= width.Value && maxHeight >= height.Value;
}
}
2018-09-12 17:26:21 +00:00
// just lie
info.IsDirectStream = true;
2016-10-29 22:22:20 +00:00
return new ImageUrlInfo
{
Url = url,
Width = width,
Height = height
};
}
}
}