|
@@ -1,3 +1,5 @@
|
|
|
+#pragma warning disable CS1591
|
|
|
+
|
|
|
using System;
|
|
|
using System.Collections.Generic;
|
|
|
using System.Globalization;
|
|
@@ -44,7 +46,6 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
private const string NS_UPNP = "urn:schemas-upnp-org:metadata-1-0/upnp/";
|
|
|
|
|
|
private readonly int _systemUpdateId;
|
|
|
- private readonly CultureInfo _usCulture = new CultureInfo("en-US");
|
|
|
|
|
|
private readonly DidlBuilder _didlBuilder;
|
|
|
|
|
@@ -58,7 +59,8 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
string accessToken,
|
|
|
IImageProcessor imageProcessor,
|
|
|
IUserDataManager userDataManager,
|
|
|
- User user, int systemUpdateId,
|
|
|
+ User user,
|
|
|
+ int systemUpdateId,
|
|
|
IServerConfigurationManager config,
|
|
|
ILocalizationManager localization,
|
|
|
IMediaSourceManager mediaSourceManager,
|
|
@@ -76,117 +78,143 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
_profile = profile;
|
|
|
_config = config;
|
|
|
|
|
|
- _didlBuilder = new DidlBuilder(profile, user, imageProcessor, serverAddress, accessToken, userDataManager, localization, mediaSourceManager, _logger, mediaEncoder);
|
|
|
+ _didlBuilder = new DidlBuilder(
|
|
|
+ profile,
|
|
|
+ user,
|
|
|
+ imageProcessor,
|
|
|
+ serverAddress,
|
|
|
+ accessToken,
|
|
|
+ userDataManager,
|
|
|
+ localization,
|
|
|
+ mediaSourceManager,
|
|
|
+ Logger,
|
|
|
+ mediaEncoder,
|
|
|
+ libraryManager);
|
|
|
}
|
|
|
|
|
|
- protected override IEnumerable<KeyValuePair<string, string>> GetResult(string methodName, IDictionary<string, string> methodParams)
|
|
|
+ /// <inheritdoc />
|
|
|
+ protected override void WriteResult(string methodName, IDictionary<string, string> methodParams, XmlWriter xmlWriter)
|
|
|
{
|
|
|
- var deviceId = "test";
|
|
|
-
|
|
|
- var user = _user;
|
|
|
+ const string DeviceId = "test";
|
|
|
|
|
|
if (string.Equals(methodName, "GetSearchCapabilities", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleGetSearchCapabilities();
|
|
|
+ {
|
|
|
+ HandleGetSearchCapabilities(xmlWriter);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "GetSortCapabilities", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleGetSortCapabilities();
|
|
|
+ {
|
|
|
+ HandleGetSortCapabilities(xmlWriter);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "GetSortExtensionCapabilities", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleGetSortExtensionCapabilities();
|
|
|
+ {
|
|
|
+ HandleGetSortExtensionCapabilities(xmlWriter);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "GetSystemUpdateID", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleGetSystemUpdateID();
|
|
|
+ {
|
|
|
+ HandleGetSystemUpdateID(xmlWriter);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "Browse", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleBrowse(methodParams, user, deviceId);
|
|
|
+ {
|
|
|
+ HandleBrowse(xmlWriter, methodParams, DeviceId);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "X_GetFeatureList", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleXGetFeatureList();
|
|
|
+ {
|
|
|
+ HandleXGetFeatureList(xmlWriter);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "GetFeatureList", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleGetFeatureList();
|
|
|
+ {
|
|
|
+ HandleGetFeatureList(xmlWriter);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "X_SetBookmark", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleXSetBookmark(methodParams, user);
|
|
|
+ {
|
|
|
+ HandleXSetBookmark(methodParams);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "Search", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleSearch(methodParams, user, deviceId);
|
|
|
+ {
|
|
|
+ HandleSearch(xmlWriter, methodParams, DeviceId);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
if (string.Equals(methodName, "X_BrowseByLetter", StringComparison.OrdinalIgnoreCase))
|
|
|
- return HandleX_BrowseByLetter(methodParams, user, deviceId);
|
|
|
+ {
|
|
|
+ HandleXBrowseByLetter(xmlWriter, methodParams, DeviceId);
|
|
|
+ return;
|
|
|
+ }
|
|
|
|
|
|
throw new ResourceNotFoundException("Unexpected control request name: " + methodName);
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleXSetBookmark(IDictionary<string, string> sparams, User user)
|
|
|
+ private void HandleXSetBookmark(IDictionary<string, string> sparams)
|
|
|
{
|
|
|
var id = sparams["ObjectID"];
|
|
|
|
|
|
- var serverItem = GetItemFromObjectId(id, user);
|
|
|
+ var serverItem = GetItemFromObjectId(id);
|
|
|
|
|
|
var item = serverItem.Item;
|
|
|
|
|
|
- var newbookmark = int.Parse(sparams["PosSecond"], _usCulture);
|
|
|
+ var newbookmark = int.Parse(sparams["PosSecond"], CultureInfo.InvariantCulture);
|
|
|
|
|
|
- var userdata = _userDataManager.GetUserData(user, item);
|
|
|
+ var userdata = _userDataManager.GetUserData(_user, item);
|
|
|
|
|
|
userdata.PlaybackPositionTicks = TimeSpan.FromSeconds(newbookmark).Ticks;
|
|
|
|
|
|
- _userDataManager.SaveUserData(user, item, userdata, UserDataSaveReason.TogglePlayed,
|
|
|
+ _userDataManager.SaveUserData(_user, item, userdata, UserDataSaveReason.TogglePlayed,
|
|
|
CancellationToken.None);
|
|
|
-
|
|
|
- return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleGetSearchCapabilities()
|
|
|
+ private void HandleGetSearchCapabilities(XmlWriter xmlWriter)
|
|
|
{
|
|
|
- return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
|
|
|
- {
|
|
|
- { "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" }
|
|
|
- };
|
|
|
+ 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");
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleGetSortCapabilities()
|
|
|
+ private void HandleGetSortCapabilities(XmlWriter xmlWriter)
|
|
|
{
|
|
|
- return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
|
|
|
- {
|
|
|
- { "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" }
|
|
|
- };
|
|
|
+ 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");
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleGetSortExtensionCapabilities()
|
|
|
+ private void HandleGetSortExtensionCapabilities(XmlWriter xmlWriter)
|
|
|
{
|
|
|
- return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
|
|
|
- {
|
|
|
- { "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" }
|
|
|
- };
|
|
|
+ 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");
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleGetSystemUpdateID()
|
|
|
+ private void HandleGetSystemUpdateID(XmlWriter xmlWriter)
|
|
|
{
|
|
|
- var headers = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
|
|
|
- headers.Add("Id", _systemUpdateId.ToString(_usCulture));
|
|
|
- return headers;
|
|
|
+ xmlWriter.WriteElementString("Id", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleGetFeatureList()
|
|
|
+ private void HandleGetFeatureList(XmlWriter xmlWriter)
|
|
|
{
|
|
|
- return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
|
|
|
- {
|
|
|
- { "FeatureList", GetFeatureListXml() }
|
|
|
- };
|
|
|
+ xmlWriter.WriteElementString("FeatureList", WriteFeatureListXml());
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleXGetFeatureList()
|
|
|
- {
|
|
|
- return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
|
|
|
- {
|
|
|
- { "FeatureList", GetFeatureListXml() }
|
|
|
- };
|
|
|
- }
|
|
|
+ private void HandleXGetFeatureList(XmlWriter xmlWriter)
|
|
|
+ => HandleGetFeatureList(xmlWriter);
|
|
|
|
|
|
- private string GetFeatureListXml()
|
|
|
+ private string WriteFeatureListXml()
|
|
|
{
|
|
|
+ // TODO: clean this up
|
|
|
var builder = new StringBuilder();
|
|
|
|
|
|
builder.Append("<?xml version=\"1.0\" encoding=\"UTF-8\"?>");
|
|
@@ -213,7 +241,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
return defaultValue;
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleBrowse(IDictionary<string, string> sparams, User user, string deviceId)
|
|
|
+ private void HandleBrowse(XmlWriter xmlWriter, IDictionary<string, string> sparams, string deviceId)
|
|
|
{
|
|
|
var id = sparams["ObjectID"];
|
|
|
var flag = sparams["BrowseFlag"];
|
|
@@ -237,101 +265,95 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
start = startVal;
|
|
|
}
|
|
|
|
|
|
- var settings = new XmlWriterSettings
|
|
|
- {
|
|
|
- Encoding = Encoding.UTF8,
|
|
|
- CloseOutput = false,
|
|
|
- OmitXmlDeclaration = true,
|
|
|
- ConformanceLevel = ConformanceLevel.Fragment
|
|
|
- };
|
|
|
-
|
|
|
- StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8);
|
|
|
-
|
|
|
int totalCount;
|
|
|
|
|
|
- var dlnaOptions = _config.GetDlnaConfiguration();
|
|
|
-
|
|
|
- using (var writer = XmlWriter.Create(builder, settings))
|
|
|
+ using (StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8))
|
|
|
{
|
|
|
- //writer.WriteStartDocument();
|
|
|
-
|
|
|
- writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
|
|
|
-
|
|
|
- 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);
|
|
|
-
|
|
|
- DidlBuilder.WriteXmlRootAttributes(_profile, writer);
|
|
|
-
|
|
|
- var serverItem = GetItemFromObjectId(id, user);
|
|
|
- var item = serverItem.Item;
|
|
|
+ var settings = new XmlWriterSettings()
|
|
|
+ {
|
|
|
+ Encoding = Encoding.UTF8,
|
|
|
+ CloseOutput = false,
|
|
|
+ OmitXmlDeclaration = true,
|
|
|
+ ConformanceLevel = ConformanceLevel.Fragment
|
|
|
+ };
|
|
|
|
|
|
- if (string.Equals(flag, "BrowseMetadata"))
|
|
|
+ using (var writer = XmlWriter.Create(builder, settings))
|
|
|
{
|
|
|
- totalCount = 1;
|
|
|
+ writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
|
|
|
|
|
|
- if (item.IsDisplayedAsFolder || serverItem.StubType.HasValue)
|
|
|
- {
|
|
|
- var childrenResult = GetUserItems(item, serverItem.StubType, user, sortCriteria, start, requestedCount);
|
|
|
+ writer.WriteAttributeString("xmlns", "dc", null, NS_DC);
|
|
|
+ writer.WriteAttributeString("xmlns", "dlna", null, NS_DLNA);
|
|
|
+ writer.WriteAttributeString("xmlns", "upnp", null, NS_UPNP);
|
|
|
|
|
|
- _didlBuilder.WriteFolderElement(writer, item, serverItem.StubType, null, childrenResult.TotalRecordCount, filter, id);
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- _didlBuilder.WriteItemElement(dlnaOptions, writer, item, user, null, null, deviceId, filter);
|
|
|
- }
|
|
|
+ DidlBuilder.WriteXmlRootAttributes(_profile, writer);
|
|
|
|
|
|
- provided++;
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- var childrenResult = GetUserItems(item, serverItem.StubType, user, sortCriteria, start, requestedCount);
|
|
|
- totalCount = childrenResult.TotalRecordCount;
|
|
|
+ var serverItem = GetItemFromObjectId(id);
|
|
|
+ var item = serverItem.Item;
|
|
|
|
|
|
- provided = childrenResult.Items.Count;
|
|
|
|
|
|
- foreach (var i in childrenResult.Items)
|
|
|
+ if (string.Equals(flag, "BrowseMetadata", StringComparison.Ordinal))
|
|
|
{
|
|
|
- var childItem = i.Item;
|
|
|
- var displayStubType = i.StubType;
|
|
|
+ totalCount = 1;
|
|
|
|
|
|
- if (childItem.IsDisplayedAsFolder || displayStubType.HasValue)
|
|
|
+ if (item.IsDisplayedAsFolder || serverItem.StubType.HasValue)
|
|
|
{
|
|
|
- var childCount = (GetUserItems(childItem, displayStubType, user, sortCriteria, null, 0))
|
|
|
- .TotalRecordCount;
|
|
|
+ var childrenResult = GetUserItems(item, serverItem.StubType, _user, sortCriteria, start, requestedCount);
|
|
|
|
|
|
- _didlBuilder.WriteFolderElement(writer, childItem, displayStubType, item, childCount, filter);
|
|
|
+ _didlBuilder.WriteFolderElement(writer, item, serverItem.StubType, null, childrenResult.TotalRecordCount, filter, id);
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
- _didlBuilder.WriteItemElement(dlnaOptions, writer, childItem, user, item, serverItem.StubType, deviceId, filter);
|
|
|
+ var dlnaOptions = _config.GetDlnaConfiguration();
|
|
|
+ _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;
|
|
|
+
|
|
|
+ var dlnaOptions = _config.GetDlnaConfiguration();
|
|
|
+ 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.WriteFullEndElement();
|
|
|
- //writer.WriteEndDocument();
|
|
|
+ xmlWriter.WriteElementString("Result", builder.ToString());
|
|
|
}
|
|
|
|
|
|
- var resXML = builder.ToString();
|
|
|
-
|
|
|
- return new[]
|
|
|
- {
|
|
|
- new KeyValuePair<string,string>("Result", resXML),
|
|
|
- new KeyValuePair<string,string>("NumberReturned", provided.ToString(_usCulture)),
|
|
|
- new KeyValuePair<string,string>("TotalMatches", totalCount.ToString(_usCulture)),
|
|
|
- new KeyValuePair<string,string>("UpdateID", _systemUpdateId.ToString(_usCulture))
|
|
|
- };
|
|
|
+ xmlWriter.WriteElementString("NumberReturned", provided.ToString(CultureInfo.InvariantCulture));
|
|
|
+ xmlWriter.WriteElementString("TotalMatches", totalCount.ToString(CultureInfo.InvariantCulture));
|
|
|
+ xmlWriter.WriteElementString("UpdateID", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleX_BrowseByLetter(IDictionary<string, string> sparams, User user, string deviceId)
|
|
|
+ private void HandleXBrowseByLetter(XmlWriter xmlWriter, IDictionary<string, string> sparams, string deviceId)
|
|
|
{
|
|
|
// TODO: Implement this method
|
|
|
- return HandleSearch(sparams, user, deviceId);
|
|
|
+ HandleSearch(xmlWriter, sparams, deviceId);
|
|
|
}
|
|
|
|
|
|
- private IEnumerable<KeyValuePair<string, string>> HandleSearch(IDictionary<string, string> sparams, User user, string deviceId)
|
|
|
+ private void HandleSearch(XmlWriter xmlWriter, IDictionary<string, string> sparams, string deviceId)
|
|
|
{
|
|
|
var searchCriteria = new SearchCriteria(GetValueOrDefault(sparams, "SearchCriteria", ""));
|
|
|
var sortCriteria = new SortCriteria(GetValueOrDefault(sparams, "SortCriteria", ""));
|
|
@@ -354,99 +376,86 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
start = startVal;
|
|
|
}
|
|
|
|
|
|
- var settings = new XmlWriterSettings
|
|
|
- {
|
|
|
- Encoding = Encoding.UTF8,
|
|
|
- CloseOutput = false,
|
|
|
- OmitXmlDeclaration = true,
|
|
|
- ConformanceLevel = ConformanceLevel.Fragment
|
|
|
- };
|
|
|
+ QueryResult<BaseItem> childrenResult;
|
|
|
|
|
|
- StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8);
|
|
|
- int totalCount = 0;
|
|
|
- int provided = 0;
|
|
|
-
|
|
|
- using (var writer = XmlWriter.Create(builder, settings))
|
|
|
+ using (StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8))
|
|
|
{
|
|
|
- //writer.WriteStartDocument();
|
|
|
-
|
|
|
- writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
|
|
|
-
|
|
|
- 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);
|
|
|
+ var settings = new XmlWriterSettings()
|
|
|
+ {
|
|
|
+ Encoding = Encoding.UTF8,
|
|
|
+ CloseOutput = false,
|
|
|
+ OmitXmlDeclaration = true,
|
|
|
+ ConformanceLevel = ConformanceLevel.Fragment
|
|
|
+ };
|
|
|
|
|
|
- DidlBuilder.WriteXmlRootAttributes(_profile, writer);
|
|
|
+ using (var writer = XmlWriter.Create(builder, settings))
|
|
|
+ {
|
|
|
+ writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
|
|
|
|
|
|
- var serverItem = GetItemFromObjectId(sparams["ContainerID"], user);
|
|
|
+ writer.WriteAttributeString("xmlns", "dc", null, NS_DC);
|
|
|
+ writer.WriteAttributeString("xmlns", "dlna", null, NS_DLNA);
|
|
|
+ writer.WriteAttributeString("xmlns", "upnp", null, NS_UPNP);
|
|
|
|
|
|
- var item = serverItem.Item;
|
|
|
+ DidlBuilder.WriteXmlRootAttributes(_profile, writer);
|
|
|
|
|
|
- var childrenResult = (GetChildrenSorted(item, user, searchCriteria, sortCriteria, start, requestedCount));
|
|
|
+ var serverItem = GetItemFromObjectId(sparams["ContainerID"]);
|
|
|
|
|
|
- totalCount = childrenResult.TotalRecordCount;
|
|
|
+ var item = serverItem.Item;
|
|
|
|
|
|
- provided = childrenResult.Items.Count;
|
|
|
+ childrenResult = GetChildrenSorted(item, _user, searchCriteria, sortCriteria, start, requestedCount);
|
|
|
|
|
|
- var dlnaOptions = _config.GetDlnaConfiguration();
|
|
|
+ var dlnaOptions = _config.GetDlnaConfiguration();
|
|
|
|
|
|
- foreach (var i in childrenResult.Items)
|
|
|
- {
|
|
|
- if (i.IsDisplayedAsFolder)
|
|
|
+ foreach (var i in childrenResult.Items)
|
|
|
{
|
|
|
- var childCount = (GetChildrenSorted(i, user, searchCriteria, sortCriteria, null, 0))
|
|
|
- .TotalRecordCount;
|
|
|
+ if (i.IsDisplayedAsFolder)
|
|
|
+ {
|
|
|
+ var childCount = GetChildrenSorted(i, _user, searchCriteria, sortCriteria, null, 0)
|
|
|
+ .TotalRecordCount;
|
|
|
|
|
|
- _didlBuilder.WriteFolderElement(writer, i, null, item, childCount, filter);
|
|
|
- }
|
|
|
- else
|
|
|
- {
|
|
|
- _didlBuilder.WriteItemElement(dlnaOptions, writer, i, user, item, serverItem.StubType, deviceId, filter);
|
|
|
+ _didlBuilder.WriteFolderElement(writer, i, null, item, childCount, filter);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ _didlBuilder.WriteItemElement(writer, i, _user, item, serverItem.StubType, deviceId, filter);
|
|
|
+ }
|
|
|
}
|
|
|
+
|
|
|
+ writer.WriteFullEndElement();
|
|
|
}
|
|
|
|
|
|
- writer.WriteFullEndElement();
|
|
|
- //writer.WriteEndDocument();
|
|
|
+ xmlWriter.WriteElementString("Result", builder.ToString());
|
|
|
}
|
|
|
|
|
|
- var resXML = builder.ToString();
|
|
|
-
|
|
|
- return new List<KeyValuePair<string, string>>
|
|
|
- {
|
|
|
- new KeyValuePair<string,string>("Result", resXML),
|
|
|
- new KeyValuePair<string,string>("NumberReturned", provided.ToString(_usCulture)),
|
|
|
- new KeyValuePair<string,string>("TotalMatches", totalCount.ToString(_usCulture)),
|
|
|
- new KeyValuePair<string,string>("UpdateID", _systemUpdateId.ToString(_usCulture))
|
|
|
- };
|
|
|
+ xmlWriter.WriteElementString("NumberReturned", childrenResult.Items.Count.ToString(CultureInfo.InvariantCulture));
|
|
|
+ xmlWriter.WriteElementString("TotalMatches", childrenResult.TotalRecordCount.ToString(CultureInfo.InvariantCulture));
|
|
|
+ xmlWriter.WriteElementString("UpdateID", _systemUpdateId.ToString(CultureInfo.InvariantCulture));
|
|
|
}
|
|
|
|
|
|
private QueryResult<BaseItem> GetChildrenSorted(BaseItem item, User user, SearchCriteria search, SortCriteria sort, int? startIndex, int? limit)
|
|
|
{
|
|
|
var folder = (Folder)item;
|
|
|
|
|
|
- var sortOrders = new List<(string, SortOrder)>();
|
|
|
- if (!folder.IsPreSorted)
|
|
|
- {
|
|
|
- sortOrders.Add((ItemSortBy.SortName, sort.SortOrder));
|
|
|
- }
|
|
|
+ var sortOrders = folder.IsPreSorted
|
|
|
+ ? Array.Empty<(string, SortOrder)>()
|
|
|
+ : new[] { (ItemSortBy.SortName, sort.SortOrder) };
|
|
|
|
|
|
- var mediaTypes = new List<string>();
|
|
|
+ string[] mediaTypes = Array.Empty<string>();
|
|
|
bool? isFolder = null;
|
|
|
|
|
|
if (search.SearchType == SearchType.Audio)
|
|
|
{
|
|
|
- mediaTypes.Add(MediaType.Audio);
|
|
|
+ mediaTypes = new[] { MediaType.Audio };
|
|
|
isFolder = false;
|
|
|
}
|
|
|
else if (search.SearchType == SearchType.Video)
|
|
|
{
|
|
|
- mediaTypes.Add(MediaType.Video);
|
|
|
+ mediaTypes = new[] { MediaType.Video };
|
|
|
isFolder = false;
|
|
|
}
|
|
|
else if (search.SearchType == SearchType.Image)
|
|
|
{
|
|
|
- mediaTypes.Add(MediaType.Photo);
|
|
|
+ mediaTypes = new[] { MediaType.Photo };
|
|
|
isFolder = false;
|
|
|
}
|
|
|
else if (search.SearchType == SearchType.Playlist)
|
|
@@ -470,7 +479,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
IsMissing = false,
|
|
|
ExcludeItemTypes = new[] { typeof(Book).Name },
|
|
|
IsFolder = isFolder,
|
|
|
- MediaTypes = mediaTypes.ToArray(),
|
|
|
+ MediaTypes = mediaTypes,
|
|
|
DtoOptions = GetDtoOptions()
|
|
|
});
|
|
|
}
|
|
@@ -514,11 +523,11 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
}
|
|
|
else if (string.Equals(CollectionType.Folders, collectionFolder.CollectionType, StringComparison.OrdinalIgnoreCase))
|
|
|
{
|
|
|
- return GetFolders(item, user, stubType, sort, startIndex, limit);
|
|
|
+ return GetFolders(user, startIndex, limit);
|
|
|
}
|
|
|
else if (string.Equals(CollectionType.LiveTv, collectionFolder.CollectionType, StringComparison.OrdinalIgnoreCase))
|
|
|
{
|
|
|
- return GetLiveTvChannels(item, user, stubType, sort, startIndex, limit);
|
|
|
+ return GetLiveTvChannels(user, sort, startIndex, limit);
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -549,7 +558,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
return ToResult(queryResult);
|
|
|
}
|
|
|
|
|
|
- private QueryResult<ServerItem> GetLiveTvChannels(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
|
|
|
+ private QueryResult<ServerItem> GetLiveTvChannels(User user, SortCriteria sort, int? startIndex, int? limit)
|
|
|
{
|
|
|
var query = new InternalItemsQuery(user)
|
|
|
{
|
|
@@ -581,7 +590,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
|
|
|
if (stubType.HasValue && stubType.Value == StubType.Playlists)
|
|
|
{
|
|
|
- return GetMusicPlaylists(item, user, query);
|
|
|
+ return GetMusicPlaylists(user, query);
|
|
|
}
|
|
|
|
|
|
if (stubType.HasValue && stubType.Value == StubType.Albums)
|
|
@@ -709,7 +718,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
|
|
|
if (stubType.HasValue && stubType.Value == StubType.Collections)
|
|
|
{
|
|
|
- return GetMovieCollections(item, user, query);
|
|
|
+ return GetMovieCollections(user, query);
|
|
|
}
|
|
|
|
|
|
if (stubType.HasValue && stubType.Value == StubType.Favorites)
|
|
@@ -722,46 +731,42 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
return GetGenres(item, user, query);
|
|
|
}
|
|
|
|
|
|
- var list = new List<ServerItem>();
|
|
|
-
|
|
|
- list.Add(new ServerItem(item)
|
|
|
+ var array = new ServerItem[]
|
|
|
{
|
|
|
- StubType = StubType.ContinueWatching
|
|
|
- });
|
|
|
-
|
|
|
- list.Add(new ServerItem(item)
|
|
|
- {
|
|
|
- StubType = StubType.Latest
|
|
|
- });
|
|
|
-
|
|
|
- list.Add(new ServerItem(item)
|
|
|
- {
|
|
|
- StubType = StubType.Movies
|
|
|
- });
|
|
|
-
|
|
|
- list.Add(new ServerItem(item)
|
|
|
- {
|
|
|
- StubType = StubType.Collections
|
|
|
- });
|
|
|
-
|
|
|
- list.Add(new ServerItem(item)
|
|
|
- {
|
|
|
- StubType = StubType.Favorites
|
|
|
- });
|
|
|
-
|
|
|
- list.Add(new ServerItem(item)
|
|
|
- {
|
|
|
- StubType = StubType.Genres
|
|
|
- });
|
|
|
+ new ServerItem(item)
|
|
|
+ {
|
|
|
+ StubType = StubType.ContinueWatching
|
|
|
+ },
|
|
|
+ new ServerItem(item)
|
|
|
+ {
|
|
|
+ StubType = StubType.Latest
|
|
|
+ },
|
|
|
+ new ServerItem(item)
|
|
|
+ {
|
|
|
+ StubType = StubType.Movies
|
|
|
+ },
|
|
|
+ new ServerItem(item)
|
|
|
+ {
|
|
|
+ StubType = StubType.Collections
|
|
|
+ },
|
|
|
+ new ServerItem(item)
|
|
|
+ {
|
|
|
+ StubType = StubType.Favorites
|
|
|
+ },
|
|
|
+ new ServerItem(item)
|
|
|
+ {
|
|
|
+ StubType = StubType.Genres
|
|
|
+ }
|
|
|
+ };
|
|
|
|
|
|
return new QueryResult<ServerItem>
|
|
|
{
|
|
|
- Items = list,
|
|
|
- TotalRecordCount = list.Count
|
|
|
+ Items = array,
|
|
|
+ TotalRecordCount = array.Length
|
|
|
};
|
|
|
}
|
|
|
|
|
|
- private QueryResult<ServerItem> GetFolders(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
|
|
|
+ private QueryResult<ServerItem> GetFolders(User user, int? startIndex, int? limit)
|
|
|
{
|
|
|
var folders = _libraryManager.GetUserRootFolder().GetChildren(user, true)
|
|
|
.OrderBy(i => i.SortName)
|
|
@@ -771,11 +776,11 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
})
|
|
|
.ToArray();
|
|
|
|
|
|
- return new QueryResult<ServerItem>
|
|
|
+ return ApplyPaging(new QueryResult<ServerItem>
|
|
|
{
|
|
|
Items = folders,
|
|
|
TotalRecordCount = folders.Length
|
|
|
- };
|
|
|
+ }, startIndex, limit);
|
|
|
}
|
|
|
|
|
|
private QueryResult<ServerItem> GetTvFolders(BaseItem item, User user, StubType? stubType, SortCriteria sort, int? startIndex, int? limit)
|
|
@@ -794,7 +799,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
|
|
|
if (stubType.HasValue && stubType.Value == StubType.NextUp)
|
|
|
{
|
|
|
- return GetNextUp(item, user, query);
|
|
|
+ return GetNextUp(item, query);
|
|
|
}
|
|
|
|
|
|
if (stubType.HasValue && stubType.Value == StubType.Latest)
|
|
@@ -912,7 +917,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
return ToResult(result);
|
|
|
}
|
|
|
|
|
|
- private QueryResult<ServerItem> GetMovieCollections(BaseItem parent, User user, InternalItemsQuery query)
|
|
|
+ private QueryResult<ServerItem> GetMovieCollections(User user, InternalItemsQuery query)
|
|
|
{
|
|
|
query.Recursive = true;
|
|
|
//query.Parent = parent;
|
|
@@ -1107,7 +1112,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
return ToResult(result);
|
|
|
}
|
|
|
|
|
|
- private QueryResult<ServerItem> GetMusicPlaylists(BaseItem parent, User user, InternalItemsQuery query)
|
|
|
+ private QueryResult<ServerItem> GetMusicPlaylists(User user, InternalItemsQuery query)
|
|
|
{
|
|
|
query.Parent = null;
|
|
|
query.IncludeItemTypes = new[] { typeof(Playlist).Name };
|
|
@@ -1136,7 +1141,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
return ToResult(items);
|
|
|
}
|
|
|
|
|
|
- private QueryResult<ServerItem> GetNextUp(BaseItem parent, User user, InternalItemsQuery query)
|
|
|
+ private QueryResult<ServerItem> GetNextUp(BaseItem parent, InternalItemsQuery query)
|
|
|
{
|
|
|
query.OrderBy = Array.Empty<(string, SortOrder)>();
|
|
|
|
|
@@ -1291,24 +1296,24 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
return result;
|
|
|
}
|
|
|
|
|
|
- private ServerItem GetItemFromObjectId(string id, User user)
|
|
|
+ private ServerItem GetItemFromObjectId(string id)
|
|
|
{
|
|
|
return DidlBuilder.IsIdRoot(id)
|
|
|
|
|
|
? new ServerItem(_libraryManager.GetUserRootFolder())
|
|
|
- : ParseItemId(id, user);
|
|
|
+ : ParseItemId(id);
|
|
|
}
|
|
|
|
|
|
- private ServerItem ParseItemId(string id, User user)
|
|
|
+ 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);
|
|
|
+ const string ParamsSrch = "Params=";
|
|
|
+ var paramsIndex = id.IndexOf(ParamsSrch, StringComparison.OrdinalIgnoreCase);
|
|
|
if (paramsIndex != -1)
|
|
|
{
|
|
|
- id = id.Substring(paramsIndex + paramsSrch.Length);
|
|
|
+ id = id.Substring(paramsIndex + ParamsSrch.Length);
|
|
|
|
|
|
var parts = id.Split(';');
|
|
|
id = parts[23];
|
|
@@ -1336,7 +1341,7 @@ namespace Emby.Dlna.ContentDirectory
|
|
|
};
|
|
|
}
|
|
|
|
|
|
- _logger.LogError("Error parsing item Id: {id}. Returning user root folder.", id);
|
|
|
+ Logger.LogError("Error parsing item Id: {id}. Returning user root folder.", id);
|
|
|
|
|
|
return new ServerItem(_libraryManager.GetUserRootFolder());
|
|
|
}
|