|
@@ -1,167 +1,691 @@
|
|
|
-using MediaBrowser.Controller.Entities;
|
|
|
+using MediaBrowser.Common.Net;
|
|
|
+using MediaBrowser.Controller.Drawing;
|
|
|
+using MediaBrowser.Controller.Dto;
|
|
|
+using MediaBrowser.Controller.Entities;
|
|
|
using MediaBrowser.Controller.Entities.Audio;
|
|
|
+using MediaBrowser.Controller.Entities.Movies;
|
|
|
+using MediaBrowser.Controller.Entities.TV;
|
|
|
+using MediaBrowser.Model.Dlna;
|
|
|
+using MediaBrowser.Model.Drawing;
|
|
|
using MediaBrowser.Model.Entities;
|
|
|
using System;
|
|
|
using System.Collections.Generic;
|
|
|
+using System.Globalization;
|
|
|
using System.Linq;
|
|
|
+using System.Xml;
|
|
|
|
|
|
namespace MediaBrowser.Dlna.Didl
|
|
|
{
|
|
|
public class DidlBuilder
|
|
|
{
|
|
|
- const string CRLF = "\r\n";
|
|
|
- const string UNKNOWN = "Unknown";
|
|
|
-
|
|
|
- const string DIDL_START = @"<item id=""{0}"" parentID=""{1}"" restricted=""1"" xmlns=""urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/"">" + CRLF;
|
|
|
- const string DIDL_TITLE = @" <dc:title xmlns:dc=""http://purl.org/dc/elements/1.1/"">{0}</dc:title>" + CRLF;
|
|
|
- const string DIDL_ARTIST = @"<upnp:artist xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">{0}</upnp:artist>" + CRLF;
|
|
|
- const string DIDL_ALBUM = @"<upnp:album xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">{0}</upnp:album>" + CRLF;
|
|
|
- const string DIDL_TRACKNUM = @"<upnp:originalTrackNumber xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">{0}</upnp:originalTrackNumber>" + CRLF;
|
|
|
- const string DIDL_VIDEOCLASS = @" <upnp:class xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">object.item.videoItem</upnp:class>" + CRLF;
|
|
|
- const string DIDL_AUDIOCLASS = @" <upnp:class xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">object.item.audioItem.musicTrack</upnp:class>" + CRLF;
|
|
|
- const string DIDL_IMAGE = @" <upnp:albumArtURI dlna:profileID=""JPEG_TN"" xmlns:dlna=""urn:schemas-dlna-org:metadata-1-0/"" xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">{0}</upnp:albumArtURI>" + CRLF +
|
|
|
- @" <upnp:icon xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">{0}</upnp:icon>" + CRLF;
|
|
|
- const string DIDL_RELEASEDATE = @" <dc:date xmlns:dc=""http://purl.org/dc/elements/1.1/"">{0}</dc:date>" + CRLF;
|
|
|
- const string DIDL_GENRE = @" <upnp:genre xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">{0}</upnp:genre>" + CRLF;
|
|
|
- const string DESCRIPTION = @" <dc:description xmlns:dc=""http://purl.org/dc/elements/1.1/"">{0}</dc:description>" + CRLF;
|
|
|
- const string DIDL_VIDEO_RES = @" <res bitrate=""{0}"" duration=""{1}"" protocolInfo=""http-get:*:video/x-msvideo:DLNA.ORG_PN=AVI;DLNA.ORG_OP=01;DLNA.ORG_CI=0;DLNA.ORG_FLAGS=01500000000000000000000000000000"" resolution=""{2}x{3}"">{4}</res>" + CRLF;
|
|
|
- const string DIDL_AUDIO_RES = @" <res bitrate=""{0}"" duration=""{1}"" nrAudioChannels=""2"" protocolInfo=""http-get:*:audio/mp3:DLNA.ORG_OP=01;DLNA.ORG_CI=0;DLNA.ORG_FLAGS=01500000000000000000000000000000"" sampleFrequency=""{2}"">{3}</res>" + CRLF;
|
|
|
- const string DIDL_IMAGE_RES = @" <res protocolInfo=""http-get:*:image/jpeg:DLNA.ORG_PN=JPEG_TN;DLNA.ORG_OP=00;DLNA.ORG_CI=1;DLNA.ORG_FLAGS=00D00000000000000000000000000000"" resolution=""212x320"">{0}</res>" + CRLF;
|
|
|
- const string DIDL_ALBUMIMAGE_RES = @" <res protocolInfo=""http-get:*:image/jpeg:DLNA.ORG_PN=JPEG_TN;DLNA.ORG_OP=00;DLNA.ORG_CI=1;DLNA.ORG_FLAGS=00D00000000000000000000000000000"" resolution=""320x320"">{0}</res>" + CRLF;
|
|
|
- const string DIDL_RATING = @" <upnp:rating xmlns:upnp=""urn:schemas-upnp-org:metadata-1-0/upnp/"">{0}</upnp:rating>" + CRLF;
|
|
|
- const string DIDL_END = "</item>";
|
|
|
+ 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 IDtoService _dtoService;
|
|
|
+
|
|
|
+ public DidlBuilder(DeviceProfile profile, IImageProcessor imageProcessor, string serverAddress, IDtoService dtoService)
|
|
|
+ {
|
|
|
+ _profile = profile;
|
|
|
+ _imageProcessor = imageProcessor;
|
|
|
+ _serverAddress = serverAddress;
|
|
|
+ _dtoService = dtoService;
|
|
|
+ }
|
|
|
+
|
|
|
+ public string GetItemDidl(BaseItem item, string deviceId, Filter filter)
|
|
|
+ {
|
|
|
+ var result = new XmlDocument();
|
|
|
+
|
|
|
+ var didl = result.CreateElement(string.Empty, "DIDL-Lite", NS_DIDL);
|
|
|
+ didl.SetAttribute("xmlns:dc", NS_DC);
|
|
|
+ didl.SetAttribute("xmlns:dlna", NS_DLNA);
|
|
|
+ didl.SetAttribute("xmlns:upnp", NS_UPNP);
|
|
|
+ //didl.SetAttribute("xmlns:sec", NS_SEC);
|
|
|
+ result.AppendChild(didl);
|
|
|
+
|
|
|
+ result.DocumentElement.AppendChild(GetItemElement(result, item, deviceId, filter));
|
|
|
+
|
|
|
+ return result.DocumentElement.OuterXml;
|
|
|
+ }
|
|
|
+
|
|
|
+ public XmlElement GetItemElement(XmlDocument doc, BaseItem item, string deviceId, Filter filter)
|
|
|
+ {
|
|
|
+ var element = doc.CreateElement(string.Empty, "item", NS_DIDL);
|
|
|
+ element.SetAttribute("restricted", "1");
|
|
|
+ element.SetAttribute("id", item.Id.ToString("N"));
|
|
|
+
|
|
|
+ if (item.Parent != null)
|
|
|
+ {
|
|
|
+ element.SetAttribute("parentID", item.Parent.Id.ToString("N"));
|
|
|
+ }
|
|
|
+
|
|
|
+ //AddBookmarkInfo(item, user, element);
|
|
|
+
|
|
|
+ AddGeneralProperties(item, element, filter);
|
|
|
+
|
|
|
+ // refID?
|
|
|
+ // storeAttribute(itemNode, object, ClassProperties.REF_ID, false);
|
|
|
+
|
|
|
+ var audio = item as Audio;
|
|
|
+ if (audio != null)
|
|
|
+ {
|
|
|
+ AddAudioResource(element, audio, deviceId, filter);
|
|
|
+ }
|
|
|
+
|
|
|
+ var video = item as Video;
|
|
|
+ if (video != null)
|
|
|
+ {
|
|
|
+ AddVideoResource(element, video, deviceId, filter);
|
|
|
+ }
|
|
|
+
|
|
|
+ AddCover(item, element);
|
|
|
+
|
|
|
+ return element;
|
|
|
+ }
|
|
|
+
|
|
|
+ private void AddVideoResource(XmlElement container, Video video, string deviceId, Filter filter)
|
|
|
+ {
|
|
|
+ var res = container.OwnerDocument.CreateElement(string.Empty, "res", NS_DIDL);
|
|
|
+
|
|
|
+ var sources = _dtoService.GetMediaSources(video);
|
|
|
+
|
|
|
+ int? maxBitrateSetting = null;
|
|
|
+
|
|
|
+ var streamInfo = new StreamBuilder().BuildVideoItem(new VideoOptions
|
|
|
+ {
|
|
|
+ ItemId = video.Id.ToString("N"),
|
|
|
+ MediaSources = sources,
|
|
|
+ Profile = _profile,
|
|
|
+ DeviceId = deviceId,
|
|
|
+ MaxBitrate = maxBitrateSetting
|
|
|
+ });
|
|
|
+
|
|
|
+ var url = streamInfo.ToDlnaUrl(_serverAddress);
|
|
|
+ //res.AppendChild(container.OwnerDocument.CreateCDataSection(url));
|
|
|
+ res.InnerText = url;
|
|
|
+
|
|
|
+ var mediaSource = sources.First(i => string.Equals(i.Id, streamInfo.MediaSourceId));
|
|
|
+
|
|
|
+ if (mediaSource.RunTimeTicks.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
|
|
|
+ }
|
|
|
+
|
|
|
+ if (filter.Contains("res@size"))
|
|
|
+ {
|
|
|
+ if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
|
|
|
+ {
|
|
|
+ var size = streamInfo.TargetSize;
|
|
|
+
|
|
|
+ if (size.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("size", size.Value.ToString(_usCulture));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ var totalBitrate = streamInfo.TotalOutputBitrate;
|
|
|
+ var targetSampleRate = streamInfo.TargetAudioSampleRate;
|
|
|
+ var targetChannels = streamInfo.TargetAudioChannels;
|
|
|
+
|
|
|
+ var targetWidth = streamInfo.TargetWidth;
|
|
|
+ var targetHeight = streamInfo.TargetHeight;
|
|
|
+
|
|
|
+ if (targetChannels.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
|
|
|
+ }
|
|
|
+
|
|
|
+ if (filter.Contains("res@resolution"))
|
|
|
+ {
|
|
|
+ if (targetWidth.HasValue && targetHeight.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("resolution", string.Format("{0}x{1}", targetWidth.Value, targetHeight.Value));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (targetSampleRate.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
|
|
|
+ }
|
|
|
+
|
|
|
+ if (totalBitrate.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("bitrate", totalBitrate.Value.ToString(_usCulture));
|
|
|
+ }
|
|
|
+
|
|
|
+ var mediaProfile = _profile.GetVideoMediaProfile(streamInfo.Container,
|
|
|
+ streamInfo.AudioCodec,
|
|
|
+ streamInfo.VideoCodec);
|
|
|
+
|
|
|
+ var filename = url.Substring(0, url.IndexOf('?'));
|
|
|
+
|
|
|
+ var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
|
|
|
+ ? MimeTypes.GetMimeType(filename)
|
|
|
+ : mediaProfile.MimeType;
|
|
|
+
|
|
|
+ var contentFeatures = new ContentFeatureBuilder(_profile).BuildVideoHeader(streamInfo.Container,
|
|
|
+ streamInfo.VideoCodec,
|
|
|
+ streamInfo.AudioCodec,
|
|
|
+ targetWidth,
|
|
|
+ targetHeight,
|
|
|
+ totalBitrate,
|
|
|
+ streamInfo.TargetTimestamp,
|
|
|
+ streamInfo.IsDirectStream,
|
|
|
+ streamInfo.RunTimeTicks,
|
|
|
+ streamInfo.TranscodeSeekInfo);
|
|
|
+
|
|
|
+ res.SetAttribute("protocolInfo", String.Format(
|
|
|
+ "http-get:*:{0}:{1}",
|
|
|
+ mimeType,
|
|
|
+ contentFeatures
|
|
|
+ ));
|
|
|
+
|
|
|
+ container.AppendChild(res);
|
|
|
+ }
|
|
|
+
|
|
|
+ private void AddAudioResource(XmlElement container, Audio audio, string deviceId, Filter filter)
|
|
|
+ {
|
|
|
+ var res = container.OwnerDocument.CreateElement(string.Empty, "res", NS_DIDL);
|
|
|
+
|
|
|
+ var sources = _dtoService.GetMediaSources(audio);
|
|
|
+
|
|
|
+ var streamInfo = new StreamBuilder().BuildAudioItem(new AudioOptions
|
|
|
+ {
|
|
|
+ ItemId = audio.Id.ToString("N"),
|
|
|
+ MediaSources = sources,
|
|
|
+ Profile = _profile,
|
|
|
+ DeviceId = deviceId
|
|
|
+ });
|
|
|
+
|
|
|
+ var url = streamInfo.ToDlnaUrl(_serverAddress);
|
|
|
+ //res.AppendChild(container.OwnerDocument.CreateCDataSection(url));
|
|
|
+ res.InnerText = url;
|
|
|
+
|
|
|
+ var mediaSource = sources.First(i => string.Equals(i.Id, streamInfo.MediaSourceId));
|
|
|
+
|
|
|
+ if (mediaSource.RunTimeTicks.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
|
|
|
+ }
|
|
|
+
|
|
|
+ if (filter.Contains("res@size"))
|
|
|
+ {
|
|
|
+ if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
|
|
|
+ {
|
|
|
+ var size = streamInfo.TargetSize;
|
|
|
+
|
|
|
+ if (size.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("size", size.Value.ToString(_usCulture));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ var targetAudioBitrate = streamInfo.TargetAudioBitrate;
|
|
|
+ var targetSampleRate = streamInfo.TargetAudioSampleRate;
|
|
|
+ var targetChannels = streamInfo.TargetAudioChannels;
|
|
|
+
|
|
|
+ if (targetChannels.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
|
|
|
+ }
|
|
|
+
|
|
|
+ if (targetSampleRate.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
|
|
|
+ }
|
|
|
+
|
|
|
+ if (targetAudioBitrate.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("bitrate", targetAudioBitrate.Value.ToString(_usCulture));
|
|
|
+ }
|
|
|
+
|
|
|
+ var mediaProfile = _profile.GetAudioMediaProfile(streamInfo.Container,
|
|
|
+ streamInfo.AudioCodec);
|
|
|
+
|
|
|
+ var filename = url.Substring(0, url.IndexOf('?'));
|
|
|
+
|
|
|
+ var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
|
|
|
+ ? MimeTypes.GetMimeType(filename)
|
|
|
+ : mediaProfile.MimeType;
|
|
|
+
|
|
|
+ var contentFeatures = new ContentFeatureBuilder(_profile).BuildAudioHeader(streamInfo.Container,
|
|
|
+ streamInfo.TargetAudioCodec,
|
|
|
+ targetAudioBitrate,
|
|
|
+ targetSampleRate,
|
|
|
+ targetChannels,
|
|
|
+ streamInfo.IsDirectStream,
|
|
|
+ streamInfo.RunTimeTicks,
|
|
|
+ streamInfo.TranscodeSeekInfo);
|
|
|
+
|
|
|
+ res.SetAttribute("protocolInfo", String.Format(
|
|
|
+ "http-get:*:{0}:{1}",
|
|
|
+ mimeType,
|
|
|
+ contentFeatures
|
|
|
+ ));
|
|
|
+
|
|
|
+ container.AppendChild(res);
|
|
|
+ }
|
|
|
+
|
|
|
+ public XmlElement GetFolderElement(XmlDocument doc, Folder folder, int childCount, Filter filter)
|
|
|
+ {
|
|
|
+ var container = doc.CreateElement(string.Empty, "container", NS_DIDL);
|
|
|
+ container.SetAttribute("restricted", "0");
|
|
|
+ container.SetAttribute("searchable", "1");
|
|
|
+ container.SetAttribute("childCount", childCount.ToString(_usCulture));
|
|
|
+ container.SetAttribute("id", folder.Id.ToString("N"));
|
|
|
+
|
|
|
+ var parent = folder.Parent;
|
|
|
+ if (parent == null)
|
|
|
+ {
|
|
|
+ container.SetAttribute("parentID", "0");
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ container.SetAttribute("parentID", parent.Id.ToString("N"));
|
|
|
+ }
|
|
|
+
|
|
|
+ AddCommonFields(folder, container, filter);
|
|
|
+
|
|
|
+ AddCover(folder, container);
|
|
|
+
|
|
|
+ return container;
|
|
|
+ }
|
|
|
+
|
|
|
+ //private void AddBookmarkInfo(BaseItem item, User user, XmlElement element)
|
|
|
+ //{
|
|
|
+ // var userdata = _userDataManager.GetUserData(user.Id, item.GetUserDataKey());
|
|
|
+
|
|
|
+ // if (userdata.PlaybackPositionTicks > 0)
|
|
|
+ // {
|
|
|
+ // var dcmInfo = element.OwnerDocument.CreateElement("sec", "dcmInfo", NS_SEC);
|
|
|
+ // dcmInfo.InnerText = string.Format("BM={0}", Convert.ToInt32(TimeSpan.FromTicks(userdata.PlaybackPositionTicks).TotalSeconds).ToString(_usCulture));
|
|
|
+ // element.AppendChild(dcmInfo);
|
|
|
+ // }
|
|
|
+ //}
|
|
|
|
|
|
/// <summary>
|
|
|
- /// Builds a Didl MetaData object for the specified dto.
|
|
|
+ /// Adds fields used by both items and folders
|
|
|
/// </summary>
|
|
|
- /// <param name="dto">The dto.</param>
|
|
|
- /// <param name="userId">The user identifier.</param>
|
|
|
- /// <param name="serverAddress">The server address.</param>
|
|
|
- /// <param name="streamUrl">The stream URL.</param>
|
|
|
- /// <param name="streams">The streams.</param>
|
|
|
- /// <param name="includeImageRes">if set to <c>true</c> [include image resource].</param>
|
|
|
- /// <returns>System.String.</returns>
|
|
|
- public static string Build(BaseItem dto, string userId, string serverAddress, string streamUrl, IEnumerable<MediaStream> streams, bool includeImageRes)
|
|
|
- {
|
|
|
- string response = string.Format(DIDL_START, dto.Id, userId);
|
|
|
- response += string.Format(DIDL_TITLE, dto.Name.Replace("&", "and"));
|
|
|
- if (IsVideo(dto))
|
|
|
- response += DIDL_VIDEOCLASS;
|
|
|
- else
|
|
|
- response += DIDL_AUDIOCLASS;
|
|
|
+ /// <param name="item">The item.</param>
|
|
|
+ /// <param name="element">The element.</param>
|
|
|
+ /// <param name="filter">The filter.</param>
|
|
|
+ private void AddCommonFields(BaseItem item, XmlElement element, Filter filter)
|
|
|
+ {
|
|
|
+ if (filter.Contains("dc:title"))
|
|
|
+ {
|
|
|
+ AddValue(element, "dc", "title", item.Name, NS_DC);
|
|
|
+ }
|
|
|
|
|
|
- var imageUrl = GetImageUrl(dto, serverAddress);
|
|
|
+ element.AppendChild(CreateObjectClass(element.OwnerDocument, item));
|
|
|
|
|
|
- if (!string.IsNullOrWhiteSpace(imageUrl))
|
|
|
+ if (filter.Contains("dc:date"))
|
|
|
{
|
|
|
- response += string.Format(DIDL_IMAGE, imageUrl);
|
|
|
+ if (item.PremiereDate.HasValue)
|
|
|
+ {
|
|
|
+ AddValue(element, "dc", "date", item.PremiereDate.Value.ToString("o"), NS_DC);
|
|
|
+ }
|
|
|
}
|
|
|
- response += string.Format(DIDL_RELEASEDATE, GetDateString(dto.PremiereDate));
|
|
|
|
|
|
- //TODO Add genres to didl;
|
|
|
- response += string.Format(DIDL_GENRE, UNKNOWN);
|
|
|
+ foreach (var genre in item.Genres)
|
|
|
+ {
|
|
|
+ AddValue(element, "upnp", "genre", genre, NS_UPNP);
|
|
|
+ }
|
|
|
|
|
|
- if (IsVideo(dto))
|
|
|
+ foreach (var studio in item.Studios)
|
|
|
{
|
|
|
- response += string.Format(DESCRIPTION, UNKNOWN);
|
|
|
- response += GetVideoDIDL(dto, streamUrl, streams);
|
|
|
+ AddValue(element, "upnp", "publisher", studio, NS_UPNP);
|
|
|
+ }
|
|
|
|
|
|
- if (includeImageRes && !string.IsNullOrWhiteSpace(imageUrl))
|
|
|
+ if (filter.Contains("dc:description"))
|
|
|
+ {
|
|
|
+ if (!string.IsNullOrWhiteSpace(item.Overview))
|
|
|
+ {
|
|
|
+ AddValue(element, "dc", "description", item.Overview, NS_DC);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (filter.Contains("upnp:longDescription"))
|
|
|
+ {
|
|
|
+ if (!string.IsNullOrWhiteSpace(item.Overview))
|
|
|
+ {
|
|
|
+ AddValue(element, "upnp", "longDescription", item.Overview, NS_UPNP);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!string.IsNullOrEmpty(item.OfficialRating))
|
|
|
+ {
|
|
|
+ if (filter.Contains("dc:rating"))
|
|
|
+ {
|
|
|
+ AddValue(element, "dc", "rating", item.OfficialRating, NS_DC);
|
|
|
+ }
|
|
|
+ if (filter.Contains("upnp:rating"))
|
|
|
{
|
|
|
- response += string.Format(DIDL_IMAGE_RES, imageUrl);
|
|
|
+ AddValue(element, "upnp", "rating", item.OfficialRating, NS_UPNP);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ AddPeople(item, element);
|
|
|
+ }
|
|
|
+
|
|
|
+ private XmlElement CreateObjectClass(XmlDocument result, BaseItem item)
|
|
|
+ {
|
|
|
+ var objectClass = result.CreateElement("upnp", "class", NS_UPNP);
|
|
|
+
|
|
|
+ if (item.IsFolder)
|
|
|
+ {
|
|
|
+ string classType = null;
|
|
|
+
|
|
|
+ if (!_profile.RequiresPlainFolders)
|
|
|
+ {
|
|
|
+ if (item is MusicAlbum)
|
|
|
+ {
|
|
|
+ classType = "object.container.album.musicAlbum";
|
|
|
+ }
|
|
|
+ if (item is MusicArtist)
|
|
|
+ {
|
|
|
+ classType = "object.container.person.musicArtist";
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ objectClass.InnerText = classType ?? "object.container.storageFolder";
|
|
|
+ }
|
|
|
+ else if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase))
|
|
|
+ {
|
|
|
+ objectClass.InnerText = "object.item.audioItem.musicTrack";
|
|
|
+ }
|
|
|
+ else if (string.Equals(item.MediaType, MediaType.Photo, StringComparison.OrdinalIgnoreCase))
|
|
|
+ {
|
|
|
+ objectClass.InnerText = "object.item.imageItem.photo";
|
|
|
+ }
|
|
|
+ else if (string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
|
|
|
+ {
|
|
|
+ if (!_profile.RequiresPlainVideoItems && item is Movie)
|
|
|
+ {
|
|
|
+ objectClass.InnerText = "object.item.videoItem.movie";
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ objectClass.InnerText = "object.item.videoItem";
|
|
|
}
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
- var audio = dto as Audio;
|
|
|
+ throw new NotSupportedException();
|
|
|
+ }
|
|
|
+
|
|
|
+ return objectClass;
|
|
|
+ }
|
|
|
+
|
|
|
+ private void AddPeople(BaseItem item, XmlElement element)
|
|
|
+ {
|
|
|
+ foreach (var actor in item.People)
|
|
|
+ {
|
|
|
+ AddValue(element, "upnp", (actor.Type ?? PersonType.Actor).ToLower(), actor.Name, NS_UPNP);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ private void AddGeneralProperties(BaseItem item, XmlElement element, Filter filter)
|
|
|
+ {
|
|
|
+ AddCommonFields(item, element, filter);
|
|
|
+
|
|
|
+ var audio = item as Audio;
|
|
|
+
|
|
|
+ if (audio != null)
|
|
|
+ {
|
|
|
+ foreach (var artist in audio.Artists)
|
|
|
+ {
|
|
|
+ AddValue(element, "upnp", "artist", artist, NS_UPNP);
|
|
|
+ }
|
|
|
|
|
|
- if (audio != null)
|
|
|
+ if (!string.IsNullOrEmpty(audio.Album))
|
|
|
{
|
|
|
- response += string.Format(DIDL_ARTIST, audio.Artists.FirstOrDefault() ?? UNKNOWN);
|
|
|
- response += string.Format(DIDL_ALBUM, audio.Album);
|
|
|
+ AddValue(element, "upnp", "album", audio.Album, NS_UPNP);
|
|
|
+ }
|
|
|
|
|
|
- response += string.Format(DIDL_TRACKNUM, audio.IndexNumber ?? 0);
|
|
|
+ if (!string.IsNullOrEmpty(audio.AlbumArtist))
|
|
|
+ {
|
|
|
+ AddValue(element, "upnp", "albumArtist", audio.AlbumArtist, NS_UPNP);
|
|
|
}
|
|
|
+ }
|
|
|
|
|
|
- response += GetAudioDIDL(dto, streamUrl, streams);
|
|
|
+ var album = item as MusicAlbum;
|
|
|
|
|
|
- if (includeImageRes && !string.IsNullOrWhiteSpace(imageUrl))
|
|
|
+ if (album != null)
|
|
|
+ {
|
|
|
+ if (!string.IsNullOrEmpty(album.AlbumArtist))
|
|
|
{
|
|
|
- response += string.Format(DIDL_ALBUMIMAGE_RES, imageUrl);
|
|
|
+ AddValue(element, "upnp", "artist", album.AlbumArtist, NS_UPNP);
|
|
|
+ AddValue(element, "upnp", "albumArtist", album.AlbumArtist, NS_UPNP);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- response += DIDL_END;
|
|
|
+ var musicVideo = item as MusicVideo;
|
|
|
|
|
|
- return response;
|
|
|
+ if (musicVideo != null)
|
|
|
+ {
|
|
|
+ if (!string.IsNullOrEmpty(musicVideo.Artist))
|
|
|
+ {
|
|
|
+ AddValue(element, "upnp", "artist", musicVideo.Artist, NS_UPNP);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!string.IsNullOrEmpty(musicVideo.Album))
|
|
|
+ {
|
|
|
+ AddValue(element, "upnp", "album", musicVideo.Album, NS_UPNP);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (item.IndexNumber.HasValue)
|
|
|
+ {
|
|
|
+ AddValue(element, "upnp", "originalTrackNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ private void AddValue(XmlElement elem, string prefix, string name, string value, string namespaceUri)
|
|
|
+ {
|
|
|
+ try
|
|
|
+ {
|
|
|
+ var date = elem.OwnerDocument.CreateElement(prefix, name, namespaceUri);
|
|
|
+ date.InnerText = value;
|
|
|
+ elem.AppendChild(date);
|
|
|
+ }
|
|
|
+ catch (XmlException)
|
|
|
+ {
|
|
|
+ //_logger.Error("Error adding xml value: " + value);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
- private static string GetVideoDIDL(BaseItem dto, string streamUrl, IEnumerable<MediaStream> streams)
|
|
|
+ private void AddCover(BaseItem item, XmlElement element)
|
|
|
{
|
|
|
- var videostream = streams.Where(stream => stream.Type == MediaStreamType.Video).OrderBy(s => s.IsDefault ? 0 : 1).FirstOrDefault();
|
|
|
+ var imageInfo = GetImageInfo(item);
|
|
|
+
|
|
|
+ if (imageInfo == null)
|
|
|
+ {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ var result = element.OwnerDocument;
|
|
|
+
|
|
|
+ var albumartUrlInfo = GetImageUrl(imageInfo, _profile.MaxAlbumArtWidth, _profile.MaxAlbumArtHeight);
|
|
|
+
|
|
|
+ var icon = result.CreateElement("upnp", "albumArtURI", NS_UPNP);
|
|
|
+ var profile = result.CreateAttribute("dlna", "profileID", NS_DLNA);
|
|
|
+ profile.InnerText = _profile.AlbumArtPn;
|
|
|
+ icon.SetAttributeNode(profile);
|
|
|
+ icon.InnerText = albumartUrlInfo.Url;
|
|
|
+ element.AppendChild(icon);
|
|
|
+
|
|
|
+ var iconUrlInfo = GetImageUrl(imageInfo, _profile.MaxIconWidth, _profile.MaxIconHeight);
|
|
|
+ icon = result.CreateElement("upnp", "icon", NS_UPNP);
|
|
|
+ profile = result.CreateAttribute("dlna", "profileID", NS_DLNA);
|
|
|
+ profile.InnerText = _profile.AlbumArtPn;
|
|
|
+ icon.SetAttributeNode(profile);
|
|
|
+ icon.InnerText = iconUrlInfo.Url;
|
|
|
+ element.AppendChild(icon);
|
|
|
|
|
|
- if (videostream == null)
|
|
|
+ if (!_profile.EnableAlbumArtInDidl)
|
|
|
{
|
|
|
- // TOOD: ???
|
|
|
- return string.Empty;
|
|
|
+ return;
|
|
|
}
|
|
|
|
|
|
- return string.Format(DIDL_VIDEO_RES,
|
|
|
- videostream.BitRate.HasValue ? videostream.BitRate.Value / 10 : 0,
|
|
|
- GetDurationString(dto),
|
|
|
- videostream.Width ?? 0,
|
|
|
- videostream.Height ?? 0,
|
|
|
- streamUrl);
|
|
|
+ var res = result.CreateElement(string.Empty, "res", NS_DIDL);
|
|
|
+
|
|
|
+ res.InnerText = albumartUrlInfo.Url;
|
|
|
+
|
|
|
+ var width = albumartUrlInfo.Width;
|
|
|
+ var height = albumartUrlInfo.Height;
|
|
|
+
|
|
|
+ var mediaProfile = new MediaFormatProfileResolver().ResolveImageFormat("jpg", width, height);
|
|
|
+
|
|
|
+ var orgPn = mediaProfile.HasValue ? "DLNA.ORG_PN=:" + mediaProfile.Value + ";" : string.Empty;
|
|
|
+
|
|
|
+ res.SetAttribute("protocolInfo", string.Format(
|
|
|
+ "http-get:*:{1}:{0}DLNA.ORG_OP=01;DLNA.ORG_CI=0;DLNA.ORG_FLAGS={2}",
|
|
|
+ orgPn,
|
|
|
+ "image/jpeg",
|
|
|
+ DlnaMaps.DefaultStreaming
|
|
|
+ ));
|
|
|
+
|
|
|
+ if (width.HasValue && height.HasValue)
|
|
|
+ {
|
|
|
+ res.SetAttribute("resolution", string.Format("{0}x{1}", width.Value, height.Value));
|
|
|
+ }
|
|
|
+
|
|
|
+ element.AppendChild(res);
|
|
|
}
|
|
|
|
|
|
- private static string GetAudioDIDL(BaseItem dto, string streamUrl, IEnumerable<MediaStream> streams)
|
|
|
+ private ImageDownloadInfo GetImageInfo(BaseItem item)
|
|
|
{
|
|
|
- var audiostream = streams.Where(stream => stream.Type == MediaStreamType.Audio).OrderBy(s => s.IsDefault ? 0 : 1).FirstOrDefault();
|
|
|
+ if (item.HasImage(ImageType.Primary))
|
|
|
+ {
|
|
|
+ return GetImageInfo(item, ImageType.Primary);
|
|
|
+ }
|
|
|
+ if (item.HasImage(ImageType.Thumb))
|
|
|
+ {
|
|
|
+ return GetImageInfo(item, ImageType.Thumb);
|
|
|
+ }
|
|
|
|
|
|
- if (audiostream == null)
|
|
|
+ if (item is Audio || item is Episode)
|
|
|
{
|
|
|
- // TOOD: ???
|
|
|
- return string.Empty;
|
|
|
+ item = item.Parents.FirstOrDefault(i => i.HasImage(ImageType.Primary));
|
|
|
+
|
|
|
+ if (item != null)
|
|
|
+ {
|
|
|
+ return GetImageInfo(item, ImageType.Primary);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
- return string.Format(DIDL_AUDIO_RES,
|
|
|
- audiostream.BitRate.HasValue ? audiostream.BitRate.Value / 10 : 16000,
|
|
|
- GetDurationString(dto),
|
|
|
- audiostream.SampleRate ?? 0,
|
|
|
- streamUrl);
|
|
|
+ return null;
|
|
|
}
|
|
|
|
|
|
- private static string GetImageUrl(BaseItem dto, string serverAddress)
|
|
|
+ private ImageDownloadInfo GetImageInfo(BaseItem item, ImageType type)
|
|
|
{
|
|
|
- const ImageType imageType = ImageType.Primary;
|
|
|
+ var imageInfo = item.GetImageInfo(type, 0);
|
|
|
+ string tag = null;
|
|
|
+
|
|
|
+ try
|
|
|
+ {
|
|
|
+ var guid = _imageProcessor.GetImageCacheTag(item, ImageType.Primary);
|
|
|
|
|
|
- if (!dto.HasImage(imageType))
|
|
|
+ tag = guid.HasValue ? guid.Value.ToString("N") : null;
|
|
|
+ }
|
|
|
+ catch
|
|
|
{
|
|
|
- dto = dto.Parents.FirstOrDefault(i => i.HasImage(imageType));
|
|
|
+
|
|
|
}
|
|
|
|
|
|
- return dto == null ? null : string.Format("{0}/Items/{1}/Images/{2}", serverAddress, dto.Id, imageType);
|
|
|
+ int? width = null;
|
|
|
+ int? height = null;
|
|
|
+
|
|
|
+ try
|
|
|
+ {
|
|
|
+ var size = _imageProcessor.GetImageSize(imageInfo.Path, imageInfo.DateModified);
|
|
|
+
|
|
|
+ width = Convert.ToInt32(size.Width);
|
|
|
+ height = Convert.ToInt32(size.Height);
|
|
|
+ }
|
|
|
+ catch
|
|
|
+ {
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ return new ImageDownloadInfo
|
|
|
+ {
|
|
|
+ ItemId = item.Id.ToString("N"),
|
|
|
+ Type = ImageType.Primary,
|
|
|
+ ImageTag = tag,
|
|
|
+ Width = width,
|
|
|
+ Height = height
|
|
|
+ };
|
|
|
}
|
|
|
|
|
|
- private static string GetDurationString(BaseItem dto)
|
|
|
+ class ImageDownloadInfo
|
|
|
{
|
|
|
- var duration = TimeSpan.FromTicks(dto.RunTimeTicks.HasValue ? dto.RunTimeTicks.Value : 0);
|
|
|
+ internal string ItemId;
|
|
|
+ internal string ImageTag;
|
|
|
+ internal ImageType Type;
|
|
|
|
|
|
- // TODO: Bad format string?
|
|
|
- return string.Format("{0}:{1:00}:2{00}.000", duration.Hours, duration.Minutes, duration.Seconds);
|
|
|
+ internal int? Width;
|
|
|
+ internal int? Height;
|
|
|
}
|
|
|
|
|
|
- private static string GetDateString(DateTime? date)
|
|
|
+ class ImageUrlInfo
|
|
|
{
|
|
|
- if (!date.HasValue)
|
|
|
- return UNKNOWN;
|
|
|
+ internal string Url;
|
|
|
|
|
|
- return string.Format("{0}-{1:00}-{2:00}", date.Value.Year, date.Value.Month, date.Value.Day);
|
|
|
+ internal int? Width;
|
|
|
+ internal int? Height;
|
|
|
}
|
|
|
|
|
|
- private static bool IsVideo(BaseItem item)
|
|
|
+ private ImageUrlInfo GetImageUrl(ImageDownloadInfo info, int? maxWidth, int? maxHeight)
|
|
|
{
|
|
|
- return string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase);
|
|
|
+ var url = string.Format("{0}/Items/{1}/Images/{2}?tag={3}&format=jpg",
|
|
|
+ _serverAddress,
|
|
|
+ info.ItemId,
|
|
|
+ info.Type,
|
|
|
+ info.ImageTag);
|
|
|
+
|
|
|
+ if (maxWidth.HasValue)
|
|
|
+ {
|
|
|
+ url += "&maxWidth=" + maxWidth.Value.ToString(_usCulture);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (maxHeight.HasValue)
|
|
|
+ {
|
|
|
+ url += "&maxHeight=" + maxHeight.Value.ToString(_usCulture);
|
|
|
+ }
|
|
|
+
|
|
|
+ var width = info.Width;
|
|
|
+ var height = info.Height;
|
|
|
+
|
|
|
+ if (width.HasValue && height.HasValue)
|
|
|
+ {
|
|
|
+ if (maxWidth.HasValue || maxHeight.HasValue)
|
|
|
+ {
|
|
|
+ var newSize = DrawingUtils.Resize(new ImageSize
|
|
|
+ {
|
|
|
+ Height = height.Value,
|
|
|
+ Width = width.Value
|
|
|
+
|
|
|
+ }, maxWidth: maxWidth, maxHeight: maxHeight);
|
|
|
+
|
|
|
+ width = Convert.ToInt32(newSize.Width);
|
|
|
+ height = Convert.ToInt32(newSize.Height);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return new ImageUrlInfo
|
|
|
+ {
|
|
|
+ Url = url,
|
|
|
+ Width = width,
|
|
|
+ Height = height
|
|
|
+ };
|
|
|
}
|
|
|
}
|
|
|
}
|