DidlBuilder.cs 42 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210
  1. using MediaBrowser.Model.Extensions;
  2. using MediaBrowser.Controller.Channels;
  3. using MediaBrowser.Controller.Drawing;
  4. using MediaBrowser.Controller.Entities;
  5. using MediaBrowser.Controller.Entities.Audio;
  6. using MediaBrowser.Controller.Entities.Movies;
  7. using MediaBrowser.Controller.Entities.TV;
  8. using MediaBrowser.Controller.Library;
  9. using MediaBrowser.Controller.Playlists;
  10. using Emby.Dlna.ContentDirectory;
  11. using MediaBrowser.Model.Dlna;
  12. using MediaBrowser.Model.Drawing;
  13. using MediaBrowser.Model.Entities;
  14. using MediaBrowser.Model.Logging;
  15. using MediaBrowser.Model.Net;
  16. using System;
  17. using System.Globalization;
  18. using System.IO;
  19. using System.Linq;
  20. using System.Text;
  21. using System.Threading.Tasks;
  22. using System.Xml;
  23. using MediaBrowser.Controller.MediaEncoding;
  24. using MediaBrowser.Model.Configuration;
  25. using MediaBrowser.Model.Globalization;
  26. namespace Emby.Dlna.Didl
  27. {
  28. public class DidlBuilder
  29. {
  30. private readonly CultureInfo _usCulture = new CultureInfo("en-US");
  31. private const string NS_DIDL = "urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/";
  32. private const string NS_DC = "http://purl.org/dc/elements/1.1/";
  33. private const string NS_UPNP = "urn:schemas-upnp-org:metadata-1-0/upnp/";
  34. private const string NS_DLNA = "urn:schemas-dlna-org:metadata-1-0/";
  35. private readonly DeviceProfile _profile;
  36. private readonly IImageProcessor _imageProcessor;
  37. private readonly string _serverAddress;
  38. private readonly string _accessToken;
  39. private readonly User _user;
  40. private readonly IUserDataManager _userDataManager;
  41. private readonly ILocalizationManager _localization;
  42. private readonly IMediaSourceManager _mediaSourceManager;
  43. private readonly ILogger _logger;
  44. private readonly ILibraryManager _libraryManager;
  45. private readonly IMediaEncoder _mediaEncoder;
  46. public DidlBuilder(DeviceProfile profile, User user, IImageProcessor imageProcessor, string serverAddress, string accessToken, IUserDataManager userDataManager, ILocalizationManager localization, IMediaSourceManager mediaSourceManager, ILogger logger, ILibraryManager libraryManager, IMediaEncoder mediaEncoder)
  47. {
  48. _profile = profile;
  49. _imageProcessor = imageProcessor;
  50. _serverAddress = serverAddress;
  51. _userDataManager = userDataManager;
  52. _localization = localization;
  53. _mediaSourceManager = mediaSourceManager;
  54. _logger = logger;
  55. _libraryManager = libraryManager;
  56. _mediaEncoder = mediaEncoder;
  57. _accessToken = accessToken;
  58. _user = user;
  59. }
  60. public string GetItemDidl(DlnaOptions options, BaseItem item, User user, BaseItem context, string deviceId, Filter filter, StreamInfo streamInfo)
  61. {
  62. var settings = new XmlWriterSettings
  63. {
  64. Encoding = Encoding.UTF8,
  65. CloseOutput = false,
  66. OmitXmlDeclaration = true,
  67. ConformanceLevel = ConformanceLevel.Fragment
  68. };
  69. StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8);
  70. using (XmlWriter writer = XmlWriter.Create(builder, settings))
  71. {
  72. //writer.WriteStartDocument();
  73. writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
  74. writer.WriteAttributeString("xmlns", "dc", null, NS_DC);
  75. writer.WriteAttributeString("xmlns", "dlna", null, NS_DLNA);
  76. writer.WriteAttributeString("xmlns", "upnp", null, NS_UPNP);
  77. //didl.SetAttribute("xmlns:sec", NS_SEC);
  78. WriteXmlRootAttributes(_profile, writer);
  79. WriteItemElement(options, writer, item, user, context, null, deviceId, filter, streamInfo);
  80. writer.WriteFullEndElement();
  81. //writer.WriteEndDocument();
  82. }
  83. return builder.ToString();
  84. }
  85. public static void WriteXmlRootAttributes(DeviceProfile profile, XmlWriter writer)
  86. {
  87. foreach (var att in profile.XmlRootAttributes)
  88. {
  89. var parts = att.Name.Split(new[] { ':' }, StringSplitOptions.RemoveEmptyEntries);
  90. if (parts.Length == 2)
  91. {
  92. writer.WriteAttributeString(parts[0], parts[1], null, att.Value);
  93. }
  94. else
  95. {
  96. writer.WriteAttributeString(att.Name, att.Value);
  97. }
  98. }
  99. }
  100. public void WriteItemElement(DlnaOptions options,
  101. XmlWriter writer,
  102. BaseItem item,
  103. User user,
  104. BaseItem context,
  105. StubType? contextStubType,
  106. string deviceId,
  107. Filter filter,
  108. StreamInfo streamInfo = null)
  109. {
  110. var clientId = GetClientId(item, null);
  111. writer.WriteStartElement(string.Empty, "item", NS_DIDL);
  112. writer.WriteAttributeString("restricted", "1");
  113. writer.WriteAttributeString("id", clientId);
  114. if (context != null)
  115. {
  116. writer.WriteAttributeString("parentID", GetClientId(context, contextStubType));
  117. }
  118. else
  119. {
  120. var parent = item.DisplayParentId;
  121. if (parent.HasValue)
  122. {
  123. writer.WriteAttributeString("parentID", GetClientId(parent.Value, null));
  124. }
  125. }
  126. AddGeneralProperties(item, null, context, writer, filter);
  127. AddSamsungBookmarkInfo(item, user, writer);
  128. // refID?
  129. // storeAttribute(itemNode, object, ClassProperties.REF_ID, false);
  130. var hasMediaSources = item as IHasMediaSources;
  131. if (hasMediaSources != null)
  132. {
  133. if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase))
  134. {
  135. AddAudioResource(options, writer, hasMediaSources, deviceId, filter, streamInfo);
  136. }
  137. else if (string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
  138. {
  139. AddVideoResource(options, writer, hasMediaSources, deviceId, filter, streamInfo);
  140. }
  141. }
  142. AddCover(item, context, null, writer);
  143. writer.WriteFullEndElement();
  144. }
  145. private ILogger GetStreamBuilderLogger(DlnaOptions options)
  146. {
  147. if (options.EnableDebugLog)
  148. {
  149. return _logger;
  150. }
  151. return new NullLogger();
  152. }
  153. private string GetMimeType(string input)
  154. {
  155. var mime = MimeTypes.GetMimeType(input);
  156. if (string.Equals(mime, "video/mp2t", StringComparison.OrdinalIgnoreCase))
  157. {
  158. mime = "video/mpeg";
  159. }
  160. return mime;
  161. }
  162. private void AddVideoResource(DlnaOptions options, XmlWriter writer, IHasMediaSources video, string deviceId, Filter filter, StreamInfo streamInfo = null)
  163. {
  164. if (streamInfo == null)
  165. {
  166. var sources = _mediaSourceManager.GetStaticMediaSources(video, true, _user).ToList();
  167. streamInfo = new StreamBuilder(_mediaEncoder, GetStreamBuilderLogger(options)).BuildVideoItem(new VideoOptions
  168. {
  169. ItemId = GetClientId(video),
  170. MediaSources = sources,
  171. Profile = _profile,
  172. DeviceId = deviceId,
  173. MaxBitrate = _profile.MaxStreamingBitrate
  174. });
  175. }
  176. var targetWidth = streamInfo.TargetWidth;
  177. var targetHeight = streamInfo.TargetHeight;
  178. var contentFeatureList = new ContentFeatureBuilder(_profile).BuildVideoHeader(streamInfo.Container,
  179. streamInfo.TargetVideoCodec,
  180. streamInfo.TargetAudioCodec,
  181. targetWidth,
  182. targetHeight,
  183. streamInfo.TargetVideoBitDepth,
  184. streamInfo.TargetVideoBitrate,
  185. streamInfo.TargetTimestamp,
  186. streamInfo.IsDirectStream,
  187. streamInfo.RunTimeTicks,
  188. streamInfo.TargetVideoProfile,
  189. streamInfo.TargetVideoLevel,
  190. streamInfo.TargetFramerate,
  191. streamInfo.TargetPacketLength,
  192. streamInfo.TranscodeSeekInfo,
  193. streamInfo.IsTargetAnamorphic,
  194. streamInfo.IsTargetInterlaced,
  195. streamInfo.TargetRefFrames,
  196. streamInfo.TargetVideoStreamCount,
  197. streamInfo.TargetAudioStreamCount,
  198. streamInfo.TargetVideoCodecTag,
  199. streamInfo.IsTargetAVC);
  200. foreach (var contentFeature in contentFeatureList)
  201. {
  202. AddVideoResource(writer, video, deviceId, filter, contentFeature, streamInfo);
  203. }
  204. var subtitleProfiles = streamInfo.GetSubtitleProfiles(false, _serverAddress, _accessToken)
  205. .Where(subtitle => subtitle.DeliveryMethod == SubtitleDeliveryMethod.External)
  206. .ToList();
  207. foreach (var subtitle in subtitleProfiles)
  208. {
  209. var subtitleAdded = AddSubtitleElement(writer, subtitle);
  210. if (subtitleAdded && _profile.EnableSingleSubtitleLimit)
  211. {
  212. break;
  213. }
  214. }
  215. }
  216. private bool AddSubtitleElement(XmlWriter writer, SubtitleStreamInfo info)
  217. {
  218. var subtitleProfile = _profile.SubtitleProfiles
  219. .FirstOrDefault(i => string.Equals(info.Format, i.Format, StringComparison.OrdinalIgnoreCase) && i.Method == SubtitleDeliveryMethod.External);
  220. if (subtitleProfile == null)
  221. {
  222. return false;
  223. }
  224. var subtitleMode = subtitleProfile.DidlMode;
  225. if (string.Equals(subtitleMode, "CaptionInfoEx", StringComparison.OrdinalIgnoreCase))
  226. {
  227. // <sec:CaptionInfoEx sec:type="srt">http://192.168.1.3:9999/video.srt</sec:CaptionInfoEx>
  228. // <sec:CaptionInfo sec:type="srt">http://192.168.1.3:9999/video.srt</sec:CaptionInfo>
  229. writer.WriteStartElement("sec", "CaptionInfoEx", null);
  230. writer.WriteAttributeString("sec", "type", null, info.Format.ToLower());
  231. writer.WriteString(info.Url);
  232. writer.WriteFullEndElement();
  233. }
  234. else if (string.Equals(subtitleMode, "smi", StringComparison.OrdinalIgnoreCase))
  235. {
  236. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  237. writer.WriteAttributeString("protocolInfo", "http-get:*:smi/caption:*");
  238. writer.WriteString(info.Url);
  239. writer.WriteFullEndElement();
  240. }
  241. else
  242. {
  243. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  244. var protocolInfo = string.Format("http-get:*:text/{0}:*", info.Format.ToLower());
  245. writer.WriteAttributeString("protocolInfo", protocolInfo);
  246. writer.WriteString(info.Url);
  247. writer.WriteFullEndElement();
  248. }
  249. return true;
  250. }
  251. private void AddVideoResource(XmlWriter writer, IHasMediaSources video, string deviceId, Filter filter, string contentFeatures, StreamInfo streamInfo)
  252. {
  253. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  254. var url = streamInfo.ToDlnaUrl(_serverAddress, _accessToken);
  255. var mediaSource = streamInfo.MediaSource;
  256. if (mediaSource.RunTimeTicks.HasValue)
  257. {
  258. writer.WriteAttributeString("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
  259. }
  260. if (filter.Contains("res@size"))
  261. {
  262. if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
  263. {
  264. var size = streamInfo.TargetSize;
  265. if (size.HasValue)
  266. {
  267. writer.WriteAttributeString("size", size.Value.ToString(_usCulture));
  268. }
  269. }
  270. }
  271. var totalBitrate = streamInfo.TargetTotalBitrate;
  272. var targetSampleRate = streamInfo.TargetAudioSampleRate;
  273. var targetChannels = streamInfo.TargetAudioChannels;
  274. var targetWidth = streamInfo.TargetWidth;
  275. var targetHeight = streamInfo.TargetHeight;
  276. if (targetChannels.HasValue)
  277. {
  278. writer.WriteAttributeString("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
  279. }
  280. if (filter.Contains("res@resolution"))
  281. {
  282. if (targetWidth.HasValue && targetHeight.HasValue)
  283. {
  284. writer.WriteAttributeString("resolution", string.Format("{0}x{1}", targetWidth.Value, targetHeight.Value));
  285. }
  286. }
  287. if (targetSampleRate.HasValue)
  288. {
  289. writer.WriteAttributeString("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
  290. }
  291. if (totalBitrate.HasValue)
  292. {
  293. writer.WriteAttributeString("bitrate", totalBitrate.Value.ToString(_usCulture));
  294. }
  295. var mediaProfile = _profile.GetVideoMediaProfile(streamInfo.Container,
  296. streamInfo.TargetAudioCodec,
  297. streamInfo.TargetVideoCodec,
  298. streamInfo.TargetAudioBitrate,
  299. targetWidth,
  300. targetHeight,
  301. streamInfo.TargetVideoBitDepth,
  302. streamInfo.TargetVideoProfile,
  303. streamInfo.TargetVideoLevel,
  304. streamInfo.TargetFramerate,
  305. streamInfo.TargetPacketLength,
  306. streamInfo.TargetTimestamp,
  307. streamInfo.IsTargetAnamorphic,
  308. streamInfo.IsTargetInterlaced,
  309. streamInfo.TargetRefFrames,
  310. streamInfo.TargetVideoStreamCount,
  311. streamInfo.TargetAudioStreamCount,
  312. streamInfo.TargetVideoCodecTag,
  313. streamInfo.IsTargetAVC);
  314. var filename = url.Substring(0, url.IndexOf('?'));
  315. var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
  316. ? GetMimeType(filename)
  317. : mediaProfile.MimeType;
  318. writer.WriteAttributeString("protocolInfo", String.Format(
  319. "http-get:*:{0}:{1}",
  320. mimeType,
  321. contentFeatures
  322. ));
  323. writer.WriteString(url);
  324. writer.WriteFullEndElement();
  325. }
  326. private string GetDisplayName(BaseItem item, StubType? itemStubType, BaseItem context)
  327. {
  328. if (itemStubType.HasValue && itemStubType.Value == StubType.People)
  329. {
  330. if (item is Video)
  331. {
  332. return _localization.GetLocalizedString("HeaderCastCrew");
  333. }
  334. return _localization.GetLocalizedString("HeaderPeople");
  335. }
  336. var episode = item as Episode;
  337. var season = context as Season;
  338. if (episode != null && season != null)
  339. {
  340. // This is a special embedded within a season
  341. if (item.ParentIndexNumber.HasValue && item.ParentIndexNumber.Value == 0)
  342. {
  343. if (season.IndexNumber.HasValue && season.IndexNumber.Value != 0)
  344. {
  345. return string.Format(_localization.GetLocalizedString("ValueSpecialEpisodeName"), item.Name);
  346. }
  347. }
  348. if (item.IndexNumber.HasValue)
  349. {
  350. var number = item.IndexNumber.Value.ToString("00", CultureInfo.InvariantCulture);
  351. if (episode.IndexNumberEnd.HasValue)
  352. {
  353. number += "-" + episode.IndexNumberEnd.Value.ToString("00", CultureInfo.InvariantCulture);
  354. }
  355. return number + " - " + item.Name;
  356. }
  357. }
  358. return item.Name;
  359. }
  360. private void AddAudioResource(DlnaOptions options, XmlWriter writer, IHasMediaSources audio, string deviceId, Filter filter, StreamInfo streamInfo = null)
  361. {
  362. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  363. if (streamInfo == null)
  364. {
  365. var sources = _mediaSourceManager.GetStaticMediaSources(audio, true, _user).ToList();
  366. streamInfo = new StreamBuilder(_mediaEncoder, GetStreamBuilderLogger(options)).BuildAudioItem(new AudioOptions
  367. {
  368. ItemId = GetClientId(audio),
  369. MediaSources = sources,
  370. Profile = _profile,
  371. DeviceId = deviceId
  372. });
  373. }
  374. var url = streamInfo.ToDlnaUrl(_serverAddress, _accessToken);
  375. var mediaSource = streamInfo.MediaSource;
  376. if (mediaSource.RunTimeTicks.HasValue)
  377. {
  378. writer.WriteAttributeString("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
  379. }
  380. if (filter.Contains("res@size"))
  381. {
  382. if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
  383. {
  384. var size = streamInfo.TargetSize;
  385. if (size.HasValue)
  386. {
  387. writer.WriteAttributeString("size", size.Value.ToString(_usCulture));
  388. }
  389. }
  390. }
  391. var targetAudioBitrate = streamInfo.TargetAudioBitrate;
  392. var targetSampleRate = streamInfo.TargetAudioSampleRate;
  393. var targetChannels = streamInfo.TargetAudioChannels;
  394. var targetAudioBitDepth = streamInfo.TargetAudioBitDepth;
  395. if (targetChannels.HasValue)
  396. {
  397. writer.WriteAttributeString("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
  398. }
  399. if (targetSampleRate.HasValue)
  400. {
  401. writer.WriteAttributeString("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
  402. }
  403. if (targetAudioBitrate.HasValue)
  404. {
  405. writer.WriteAttributeString("bitrate", targetAudioBitrate.Value.ToString(_usCulture));
  406. }
  407. var mediaProfile = _profile.GetAudioMediaProfile(streamInfo.Container,
  408. streamInfo.TargetAudioCodec,
  409. targetChannels,
  410. targetAudioBitrate,
  411. targetSampleRate,
  412. targetAudioBitDepth);
  413. var filename = url.Substring(0, url.IndexOf('?'));
  414. var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
  415. ? GetMimeType(filename)
  416. : mediaProfile.MimeType;
  417. var contentFeatures = new ContentFeatureBuilder(_profile).BuildAudioHeader(streamInfo.Container,
  418. streamInfo.TargetAudioCodec,
  419. targetAudioBitrate,
  420. targetSampleRate,
  421. targetChannels,
  422. targetAudioBitDepth,
  423. streamInfo.IsDirectStream,
  424. streamInfo.RunTimeTicks,
  425. streamInfo.TranscodeSeekInfo);
  426. writer.WriteAttributeString("protocolInfo", String.Format(
  427. "http-get:*:{0}:{1}",
  428. mimeType,
  429. contentFeatures
  430. ));
  431. writer.WriteString(url);
  432. writer.WriteFullEndElement();
  433. }
  434. public static bool IsIdRoot(string id)
  435. {
  436. if (string.IsNullOrWhiteSpace(id) ||
  437. string.Equals(id, "0", StringComparison.OrdinalIgnoreCase)
  438. // Samsung sometimes uses 1 as root
  439. || string.Equals(id, "1", StringComparison.OrdinalIgnoreCase))
  440. {
  441. return true;
  442. }
  443. return false;
  444. }
  445. public void WriteFolderElement(XmlWriter writer, BaseItem folder, StubType? stubType, BaseItem context, int childCount, Filter filter, string requestedId = null)
  446. {
  447. writer.WriteStartElement(string.Empty, "container", NS_DIDL);
  448. writer.WriteAttributeString("restricted", "0");
  449. writer.WriteAttributeString("searchable", "1");
  450. writer.WriteAttributeString("childCount", childCount.ToString(_usCulture));
  451. var clientId = GetClientId(folder, stubType);
  452. if (string.Equals(requestedId, "0"))
  453. {
  454. writer.WriteAttributeString("id", "0");
  455. writer.WriteAttributeString("parentID", "-1");
  456. }
  457. else
  458. {
  459. writer.WriteAttributeString("id", clientId);
  460. if (context != null)
  461. {
  462. writer.WriteAttributeString("parentID", GetClientId(context, null));
  463. }
  464. else
  465. {
  466. var parent = folder.DisplayParentId;
  467. if (!parent.HasValue)
  468. {
  469. writer.WriteAttributeString("parentID", "0");
  470. }
  471. else
  472. {
  473. writer.WriteAttributeString("parentID", GetClientId(parent.Value, null));
  474. }
  475. }
  476. }
  477. AddGeneralProperties(folder, stubType, context, writer, filter);
  478. AddCover(folder, context, stubType, writer);
  479. writer.WriteFullEndElement();
  480. }
  481. private void AddSamsungBookmarkInfo(BaseItem item, User user, XmlWriter writer)
  482. {
  483. if (!item.SupportsPositionTicksResume || item is Folder)
  484. {
  485. return;
  486. }
  487. XmlAttribute secAttribute = null;
  488. foreach (var attribute in _profile.XmlRootAttributes)
  489. {
  490. if (string.Equals(attribute.Name, "xmlns:sec", StringComparison.OrdinalIgnoreCase))
  491. {
  492. secAttribute = attribute;
  493. break;
  494. }
  495. }
  496. // Not a samsung device
  497. if (secAttribute == null)
  498. {
  499. return;
  500. }
  501. var userdata = _userDataManager.GetUserData(user.Id, item);
  502. if (userdata.PlaybackPositionTicks > 0)
  503. {
  504. var elementValue = string.Format("BM={0}", Convert.ToInt32(TimeSpan.FromTicks(userdata.PlaybackPositionTicks).TotalSeconds).ToString(_usCulture));
  505. AddValue(writer, "sec", "dcmInfo", elementValue, secAttribute.Value);
  506. }
  507. }
  508. /// <summary>
  509. /// Adds fields used by both items and folders
  510. /// </summary>
  511. private void AddCommonFields(BaseItem item, StubType? itemStubType, BaseItem context, XmlWriter writer, Filter filter)
  512. {
  513. // Don't filter on dc:title because not all devices will include it in the filter
  514. // MediaMonkey for example won't display content without a title
  515. //if (filter.Contains("dc:title"))
  516. {
  517. AddValue(writer, "dc", "title", GetDisplayName(item, itemStubType, context), NS_DC);
  518. }
  519. WriteObjectClass(writer, item, itemStubType);
  520. if (filter.Contains("dc:date"))
  521. {
  522. if (item.PremiereDate.HasValue)
  523. {
  524. AddValue(writer, "dc", "date", item.PremiereDate.Value.ToString("o"), NS_DC);
  525. }
  526. }
  527. if (filter.Contains("upnp:genre"))
  528. {
  529. foreach (var genre in item.Genres)
  530. {
  531. AddValue(writer, "upnp", "genre", genre, NS_UPNP);
  532. }
  533. }
  534. foreach (var studio in item.Studios)
  535. {
  536. AddValue(writer, "upnp", "publisher", studio, NS_UPNP);
  537. }
  538. if (filter.Contains("dc:description"))
  539. {
  540. var desc = item.Overview;
  541. if (!string.IsNullOrWhiteSpace(desc))
  542. {
  543. AddValue(writer, "dc", "description", desc, NS_DC);
  544. }
  545. }
  546. if (filter.Contains("upnp:longDescription"))
  547. {
  548. if (!string.IsNullOrWhiteSpace(item.Overview))
  549. {
  550. AddValue(writer, "upnp", "longDescription", item.Overview, NS_UPNP);
  551. }
  552. }
  553. if (!string.IsNullOrEmpty(item.OfficialRating))
  554. {
  555. if (filter.Contains("dc:rating"))
  556. {
  557. AddValue(writer, "dc", "rating", item.OfficialRating, NS_DC);
  558. }
  559. if (filter.Contains("upnp:rating"))
  560. {
  561. AddValue(writer, "upnp", "rating", item.OfficialRating, NS_UPNP);
  562. }
  563. }
  564. AddPeople(item, writer);
  565. }
  566. private void WriteObjectClass(XmlWriter writer, BaseItem item, StubType? stubType)
  567. {
  568. // More types here
  569. // http://oss.linn.co.uk/repos/Public/LibUpnpCil/DidlLite/UpnpAv/Test/TestDidlLite.cs
  570. writer.WriteStartElement("upnp", "class", NS_UPNP);
  571. if (item.IsDisplayedAsFolder || stubType.HasValue)
  572. {
  573. string classType = null;
  574. if (!_profile.RequiresPlainFolders)
  575. {
  576. if (item is MusicAlbum)
  577. {
  578. classType = "object.container.album.musicAlbum";
  579. }
  580. else if (item is MusicArtist)
  581. {
  582. classType = "object.container.person.musicArtist";
  583. }
  584. else if (item is Series || item is Season || item is BoxSet || item is Video)
  585. {
  586. classType = "object.container.album.videoAlbum";
  587. }
  588. else if (item is Playlist)
  589. {
  590. classType = "object.container.playlistContainer";
  591. }
  592. else if (item is PhotoAlbum)
  593. {
  594. classType = "object.container.album.photoAlbum";
  595. }
  596. }
  597. writer.WriteString(classType ?? "object.container.storageFolder");
  598. }
  599. else if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase))
  600. {
  601. writer.WriteString("object.item.audioItem.musicTrack");
  602. }
  603. else if (string.Equals(item.MediaType, MediaType.Photo, StringComparison.OrdinalIgnoreCase))
  604. {
  605. writer.WriteString("object.item.imageItem.photo");
  606. }
  607. else if (string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
  608. {
  609. if (!_profile.RequiresPlainVideoItems && item is Movie)
  610. {
  611. writer.WriteString("object.item.videoItem.movie");
  612. }
  613. else if (!_profile.RequiresPlainVideoItems && item is MusicVideo)
  614. {
  615. writer.WriteString("object.item.videoItem.musicVideoClip");
  616. }
  617. else
  618. {
  619. writer.WriteString("object.item.videoItem");
  620. }
  621. }
  622. else if (item is MusicGenre)
  623. {
  624. writer.WriteString(_profile.RequiresPlainFolders ? "object.container.storageFolder" : "object.container.genre.musicGenre");
  625. }
  626. else if (item is Genre || item is GameGenre)
  627. {
  628. writer.WriteString(_profile.RequiresPlainFolders ? "object.container.storageFolder" : "object.container.genre");
  629. }
  630. else
  631. {
  632. writer.WriteString("object.item");
  633. }
  634. writer.WriteFullEndElement();
  635. }
  636. private void AddPeople(BaseItem item, XmlWriter writer)
  637. {
  638. var types = new[]
  639. {
  640. PersonType.Director,
  641. PersonType.Writer,
  642. PersonType.Producer,
  643. PersonType.Composer,
  644. "Creator"
  645. };
  646. var people = _libraryManager.GetPeople(item);
  647. var index = 0;
  648. // Seeing some LG models locking up due content with large lists of people
  649. // The actual issue might just be due to processing a more metadata than it can handle
  650. var limit = 6;
  651. foreach (var actor in people)
  652. {
  653. var type = types.FirstOrDefault(i => string.Equals(i, actor.Type, StringComparison.OrdinalIgnoreCase) || string.Equals(i, actor.Role, StringComparison.OrdinalIgnoreCase))
  654. ?? PersonType.Actor;
  655. AddValue(writer, "upnp", type.ToLower(), actor.Name, NS_UPNP);
  656. index++;
  657. if (index >= limit)
  658. {
  659. break;
  660. }
  661. }
  662. }
  663. private void AddGeneralProperties(BaseItem item, StubType? itemStubType, BaseItem context, XmlWriter writer, Filter filter)
  664. {
  665. AddCommonFields(item, itemStubType, context, writer, filter);
  666. var audio = item as Audio;
  667. if (audio != null)
  668. {
  669. foreach (var artist in audio.Artists)
  670. {
  671. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  672. }
  673. if (!string.IsNullOrEmpty(audio.Album))
  674. {
  675. AddValue(writer, "upnp", "album", audio.Album, NS_UPNP);
  676. }
  677. foreach (var artist in audio.AlbumArtists)
  678. {
  679. AddAlbumArtist(writer, artist);
  680. }
  681. }
  682. var album = item as MusicAlbum;
  683. if (album != null)
  684. {
  685. foreach (var artist in album.AlbumArtists)
  686. {
  687. AddAlbumArtist(writer, artist);
  688. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  689. }
  690. foreach (var artist in album.Artists)
  691. {
  692. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  693. }
  694. }
  695. var musicVideo = item as MusicVideo;
  696. if (musicVideo != null)
  697. {
  698. foreach (var artist in musicVideo.Artists)
  699. {
  700. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  701. AddAlbumArtist(writer, artist);
  702. }
  703. if (!string.IsNullOrEmpty(musicVideo.Album))
  704. {
  705. AddValue(writer, "upnp", "album", musicVideo.Album, NS_UPNP);
  706. }
  707. }
  708. if (item.IndexNumber.HasValue)
  709. {
  710. AddValue(writer, "upnp", "originalTrackNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
  711. if (item is Episode)
  712. {
  713. AddValue(writer, "upnp", "episodeNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
  714. }
  715. }
  716. }
  717. private void AddAlbumArtist(XmlWriter writer, string name)
  718. {
  719. try
  720. {
  721. writer.WriteStartElement("upnp", "artist", NS_UPNP);
  722. writer.WriteAttributeString("role", "AlbumArtist");
  723. writer.WriteString(name);
  724. writer.WriteFullEndElement();
  725. }
  726. catch (XmlException)
  727. {
  728. //_logger.Error("Error adding xml value: " + value);
  729. }
  730. }
  731. private void AddValue(XmlWriter writer, string prefix, string name, string value, string namespaceUri)
  732. {
  733. try
  734. {
  735. writer.WriteElementString(prefix, name, namespaceUri, value);
  736. }
  737. catch (XmlException)
  738. {
  739. //_logger.Error("Error adding xml value: " + value);
  740. }
  741. }
  742. private void AddCover(BaseItem item, BaseItem context, StubType? stubType, XmlWriter writer)
  743. {
  744. if (stubType.HasValue && stubType.Value == StubType.People)
  745. {
  746. AddEmbeddedImageAsCover("people", writer);
  747. return;
  748. }
  749. ImageDownloadInfo imageInfo = null;
  750. if (context is UserView)
  751. {
  752. var episode = item as Episode;
  753. if (episode != null)
  754. {
  755. var parent = episode.Series;
  756. if (parent != null)
  757. {
  758. imageInfo = GetImageInfo(parent);
  759. }
  760. }
  761. }
  762. // Finally, just use the image from the item
  763. if (imageInfo == null)
  764. {
  765. imageInfo = GetImageInfo(item);
  766. }
  767. if (imageInfo == null)
  768. {
  769. return;
  770. }
  771. var playbackPercentage = 0;
  772. var unplayedCount = 0;
  773. if (item is Video)
  774. {
  775. var userData = _userDataManager.GetUserDataDto(item, _user);
  776. playbackPercentage = Convert.ToInt32(userData.PlayedPercentage ?? 0);
  777. if (playbackPercentage >= 100 || userData.Played)
  778. {
  779. playbackPercentage = 100;
  780. }
  781. }
  782. else if (item is Series || item is Season || item is BoxSet)
  783. {
  784. var userData = _userDataManager.GetUserDataDto(item, _user);
  785. if (userData.Played)
  786. {
  787. playbackPercentage = 100;
  788. }
  789. else
  790. {
  791. unplayedCount = userData.UnplayedItemCount ?? 0;
  792. }
  793. }
  794. var albumartUrlInfo = GetImageUrl(imageInfo, _profile.MaxAlbumArtWidth, _profile.MaxAlbumArtHeight, playbackPercentage, unplayedCount, "jpg");
  795. writer.WriteStartElement("upnp", "albumArtURI", NS_UPNP);
  796. writer.WriteAttributeString("dlna", "profileID", NS_DLNA, _profile.AlbumArtPn);
  797. writer.WriteString(albumartUrlInfo.Url);
  798. writer.WriteFullEndElement();
  799. // TOOD: Remove these default values
  800. var iconUrlInfo = GetImageUrl(imageInfo, _profile.MaxIconWidth ?? 48, _profile.MaxIconHeight ?? 48, playbackPercentage, unplayedCount, "jpg");
  801. writer.WriteElementString("upnp", "icon", NS_UPNP, iconUrlInfo.Url);
  802. if (!_profile.EnableAlbumArtInDidl)
  803. {
  804. if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase)
  805. || string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
  806. {
  807. if (!stubType.HasValue)
  808. {
  809. return;
  810. }
  811. }
  812. }
  813. AddImageResElement(item, writer, 160, 160, playbackPercentage, unplayedCount, "jpg", "JPEG_TN");
  814. if (!_profile.EnableSingleAlbumArtLimit)
  815. {
  816. AddImageResElement(item, writer, 4096, 4096, playbackPercentage, unplayedCount, "jpg", "JPEG_LRG");
  817. AddImageResElement(item, writer, 1024, 768, playbackPercentage, unplayedCount, "jpg", "JPEG_MED");
  818. AddImageResElement(item, writer, 640, 480, playbackPercentage, unplayedCount, "jpg", "JPEG_SM");
  819. AddImageResElement(item, writer, 4096, 4096, playbackPercentage, unplayedCount, "png", "PNG_LRG");
  820. AddImageResElement(item, writer, 160, 160, playbackPercentage, unplayedCount, "png", "PNG_TN");
  821. }
  822. }
  823. private void AddEmbeddedImageAsCover(string name, XmlWriter writer)
  824. {
  825. writer.WriteStartElement("upnp", "albumArtURI", NS_UPNP);
  826. writer.WriteAttributeString("dlna", "profileID", NS_DLNA, _profile.AlbumArtPn);
  827. writer.WriteString(_serverAddress + "/Dlna/icons/people480.jpg");
  828. writer.WriteFullEndElement();
  829. writer.WriteElementString("upnp", "icon", NS_UPNP, _serverAddress + "/Dlna/icons/people48.jpg");
  830. }
  831. private void AddImageResElement(BaseItem item,
  832. XmlWriter writer,
  833. int maxWidth,
  834. int maxHeight,
  835. int playbackPercentage,
  836. int unplayedCount,
  837. string format,
  838. string org_Pn)
  839. {
  840. var imageInfo = GetImageInfo(item);
  841. if (imageInfo == null)
  842. {
  843. return;
  844. }
  845. var albumartUrlInfo = GetImageUrl(imageInfo, maxWidth, maxHeight, playbackPercentage, unplayedCount, format);
  846. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  847. var width = albumartUrlInfo.Width;
  848. var height = albumartUrlInfo.Height;
  849. var contentFeatures = new ContentFeatureBuilder(_profile)
  850. .BuildImageHeader(format, width, height, imageInfo.IsDirectStream, org_Pn);
  851. writer.WriteAttributeString("protocolInfo", String.Format(
  852. "http-get:*:{0}:{1}",
  853. GetMimeType("file." + format),
  854. contentFeatures
  855. ));
  856. if (width.HasValue && height.HasValue)
  857. {
  858. writer.WriteAttributeString("resolution", string.Format("{0}x{1}", width.Value, height.Value));
  859. }
  860. writer.WriteString(albumartUrlInfo.Url);
  861. writer.WriteFullEndElement();
  862. }
  863. private ImageDownloadInfo GetImageInfo(BaseItem item)
  864. {
  865. if (item.HasImage(ImageType.Primary))
  866. {
  867. return GetImageInfo(item, ImageType.Primary);
  868. }
  869. if (item.HasImage(ImageType.Thumb))
  870. {
  871. return GetImageInfo(item, ImageType.Thumb);
  872. }
  873. if (item.HasImage(ImageType.Backdrop))
  874. {
  875. if (item is Channel)
  876. {
  877. return GetImageInfo(item, ImageType.Backdrop);
  878. }
  879. }
  880. item = item.GetParents().FirstOrDefault(i => i.HasImage(ImageType.Primary));
  881. if (item != null)
  882. {
  883. if (item.HasImage(ImageType.Primary))
  884. {
  885. return GetImageInfo(item, ImageType.Primary);
  886. }
  887. }
  888. return null;
  889. }
  890. private ImageDownloadInfo GetImageInfo(BaseItem item, ImageType type)
  891. {
  892. var imageInfo = item.GetImageInfo(type, 0);
  893. string tag = null;
  894. try
  895. {
  896. tag = _imageProcessor.GetImageCacheTag(item, type);
  897. }
  898. catch
  899. {
  900. }
  901. int? width = null;
  902. int? height = null;
  903. try
  904. {
  905. var size = _imageProcessor.GetImageSize(imageInfo);
  906. width = Convert.ToInt32(size.Width);
  907. height = Convert.ToInt32(size.Height);
  908. }
  909. catch
  910. {
  911. }
  912. var inputFormat = (Path.GetExtension(imageInfo.Path) ?? string.Empty)
  913. .TrimStart('.')
  914. .Replace("jpeg", "jpg", StringComparison.OrdinalIgnoreCase);
  915. return new ImageDownloadInfo
  916. {
  917. ItemId = item.Id.ToString("N"),
  918. Type = type,
  919. ImageTag = tag,
  920. Width = width,
  921. Height = height,
  922. Format = inputFormat,
  923. ItemImageInfo = imageInfo
  924. };
  925. }
  926. class ImageDownloadInfo
  927. {
  928. internal string ItemId;
  929. internal string ImageTag;
  930. internal ImageType Type;
  931. internal int? Width;
  932. internal int? Height;
  933. internal bool IsDirectStream;
  934. internal string Format;
  935. internal ItemImageInfo ItemImageInfo;
  936. }
  937. class ImageUrlInfo
  938. {
  939. internal string Url;
  940. internal int? Width;
  941. internal int? Height;
  942. }
  943. public static string GetClientId(BaseItem item, StubType? stubType)
  944. {
  945. return GetClientId(item.Id, stubType);
  946. }
  947. public static string GetClientId(Guid idValue, StubType? stubType)
  948. {
  949. var id = idValue.ToString("N");
  950. if (stubType.HasValue)
  951. {
  952. id = stubType.Value.ToString().ToLower() + "_" + id;
  953. }
  954. return id;
  955. }
  956. public static string GetClientId(IHasMediaSources item)
  957. {
  958. var id = item.Id.ToString("N");
  959. return id;
  960. }
  961. private ImageUrlInfo GetImageUrl(ImageDownloadInfo info, int maxWidth, int maxHeight, int playbackPercentage, int unplayedCount, string format)
  962. {
  963. var url = string.Format("{0}/Items/{1}/Images/{2}/0/{3}/{4}/{5}/{6}/{7}/{8}",
  964. _serverAddress,
  965. info.ItemId,
  966. info.Type,
  967. info.ImageTag,
  968. format,
  969. maxWidth.ToString(CultureInfo.InvariantCulture),
  970. maxHeight.ToString(CultureInfo.InvariantCulture),
  971. playbackPercentage.ToString(CultureInfo.InvariantCulture),
  972. unplayedCount.ToString(CultureInfo.InvariantCulture)
  973. );
  974. var width = info.Width;
  975. var height = info.Height;
  976. info.IsDirectStream = false;
  977. if (width.HasValue && height.HasValue)
  978. {
  979. var newSize = DrawingUtils.Resize(new ImageSize
  980. {
  981. Height = height.Value,
  982. Width = width.Value
  983. }, null, null, maxWidth, maxHeight);
  984. width = Convert.ToInt32(newSize.Width);
  985. height = Convert.ToInt32(newSize.Height);
  986. var normalizedFormat = format
  987. .Replace("jpeg", "jpg", StringComparison.OrdinalIgnoreCase);
  988. if (string.Equals(info.Format, normalizedFormat, StringComparison.OrdinalIgnoreCase))
  989. {
  990. info.IsDirectStream = maxWidth >= width.Value && maxHeight >= height.Value;
  991. }
  992. }
  993. return new ImageUrlInfo
  994. {
  995. Url = url,
  996. Width = width,
  997. Height = height
  998. };
  999. }
  1000. }
  1001. }