aboutsummaryrefslogtreecommitdiff
path: root/MediaBrowser.LocalMetadata/Savers
diff options
context:
space:
mode:
Diffstat (limited to 'MediaBrowser.LocalMetadata/Savers')
-rw-r--r--MediaBrowser.LocalMetadata/Savers/AlbumXmlSaver.cs68
-rw-r--r--MediaBrowser.LocalMetadata/Savers/ArtistXmlSaver.cs68
-rw-r--r--MediaBrowser.LocalMetadata/Savers/BoxSetXmlSaver.cs68
-rw-r--r--MediaBrowser.LocalMetadata/Savers/ChannelXmlSaver.cs73
-rw-r--r--MediaBrowser.LocalMetadata/Savers/EpisodeXmlSaver.cs151
-rw-r--r--MediaBrowser.LocalMetadata/Savers/FolderXmlSaver.cs80
-rw-r--r--MediaBrowser.LocalMetadata/Savers/GameSystemXmlSaver.cs75
-rw-r--r--MediaBrowser.LocalMetadata/Savers/GameXmlSaver.cs112
-rw-r--r--MediaBrowser.LocalMetadata/Savers/MovieXmlSaver.cs133
-rw-r--r--MediaBrowser.LocalMetadata/Savers/PersonXmlSaver.cs81
-rw-r--r--MediaBrowser.LocalMetadata/Savers/SeasonXmlSaver.cs87
-rw-r--r--MediaBrowser.LocalMetadata/Savers/SeriesXmlSaver.cs135
-rw-r--r--MediaBrowser.LocalMetadata/Savers/XmlSaverHelpers.cs718
13 files changed, 1849 insertions, 0 deletions
diff --git a/MediaBrowser.LocalMetadata/Savers/AlbumXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/AlbumXmlSaver.cs
new file mode 100644
index 000000000..05022464d
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/AlbumXmlSaver.cs
@@ -0,0 +1,68 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.Audio;
+using MediaBrowser.Controller.Library;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ class AlbumXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is MusicAlbum && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ XmlSaverHelpers.AddCommonNodes((MusicAlbum)item, builder);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string> { });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "album.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/ArtistXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/ArtistXmlSaver.cs
new file mode 100644
index 000000000..b932c5c7c
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/ArtistXmlSaver.cs
@@ -0,0 +1,68 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.Audio;
+using MediaBrowser.Controller.Library;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ class ArtistXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is MusicArtist && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ XmlSaverHelpers.AddCommonNodes((MusicArtist)item, builder);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string> { });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "artist.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/BoxSetXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/BoxSetXmlSaver.cs
new file mode 100644
index 000000000..db7b40c7d
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/BoxSetXmlSaver.cs
@@ -0,0 +1,68 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.Movies;
+using MediaBrowser.Controller.Library;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ public class BoxSetXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is BoxSet && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ XmlSaverHelpers.AddCommonNodes((BoxSet)item, builder);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string> { });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "collection.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/ChannelXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/ChannelXmlSaver.cs
new file mode 100644
index 000000000..3b7783012
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/ChannelXmlSaver.cs
@@ -0,0 +1,73 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.LiveTv;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ /// <summary>
+ /// Class PersonXmlSaver
+ /// </summary>
+ public class ChannelXmlSaver : IMetadataFileSaver
+ {
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is LiveTvChannel && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ XmlSaverHelpers.AddCommonNodes((LiveTvChannel)item, builder);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string>
+ {
+ });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "channel.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/EpisodeXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/EpisodeXmlSaver.cs
new file mode 100644
index 000000000..275ec2fe8
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/EpisodeXmlSaver.cs
@@ -0,0 +1,151 @@
+using System.Collections.Generic;
+using System.Globalization;
+using System.IO;
+using System.Security;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.TV;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Persistence;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ public class EpisodeXmlSaver : IMetadataFileSaver
+ {
+ private readonly IItemRepository _itemRepository;
+
+ private readonly CultureInfo _usCulture = new CultureInfo("en-US");
+
+ public EpisodeXmlSaver(IItemRepository itemRepository)
+ {
+ _itemRepository = itemRepository;
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is Episode && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var episode = (Episode)item;
+
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ if (!string.IsNullOrEmpty(item.Name))
+ {
+ builder.Append("<EpisodeName>" + SecurityElement.Escape(episode.Name) + "</EpisodeName>");
+ }
+
+ if (episode.IndexNumber.HasValue)
+ {
+ builder.Append("<EpisodeNumber>" + SecurityElement.Escape(episode.IndexNumber.Value.ToString(_usCulture)) + "</EpisodeNumber>");
+ }
+
+ if (episode.IndexNumberEnd.HasValue)
+ {
+ builder.Append("<EpisodeNumberEnd>" + SecurityElement.Escape(episode.IndexNumberEnd.Value.ToString(_usCulture)) + "</EpisodeNumberEnd>");
+ }
+
+ if (episode.AirsAfterSeasonNumber.HasValue)
+ {
+ builder.Append("<airsafter_season>" + SecurityElement.Escape(episode.AirsAfterSeasonNumber.Value.ToString(_usCulture)) + "</airsafter_season>");
+ }
+ if (episode.AirsBeforeEpisodeNumber.HasValue)
+ {
+ builder.Append("<airsbefore_episode>" + SecurityElement.Escape(episode.AirsBeforeEpisodeNumber.Value.ToString(_usCulture)) + "</airsbefore_episode>");
+ }
+ if (episode.AirsBeforeSeasonNumber.HasValue)
+ {
+ builder.Append("<airsbefore_season>" + SecurityElement.Escape(episode.AirsBeforeSeasonNumber.Value.ToString(_usCulture)) + "</airsbefore_season>");
+ }
+
+ if (episode.ParentIndexNumber.HasValue)
+ {
+ builder.Append("<SeasonNumber>" + SecurityElement.Escape(episode.ParentIndexNumber.Value.ToString(_usCulture)) + "</SeasonNumber>");
+ }
+
+ if (episode.AbsoluteEpisodeNumber.HasValue)
+ {
+ builder.Append("<absolute_number>" + SecurityElement.Escape(episode.AbsoluteEpisodeNumber.Value.ToString(_usCulture)) + "</absolute_number>");
+ }
+
+ if (episode.DvdEpisodeNumber.HasValue)
+ {
+ builder.Append("<DVD_episodenumber>" + SecurityElement.Escape(episode.DvdEpisodeNumber.Value.ToString(_usCulture)) + "</DVD_episodenumber>");
+ }
+
+ if (episode.DvdSeasonNumber.HasValue)
+ {
+ builder.Append("<DVD_season>" + SecurityElement.Escape(episode.DvdSeasonNumber.Value.ToString(_usCulture)) + "</DVD_season>");
+ }
+
+ if (episode.PremiereDate.HasValue)
+ {
+ builder.Append("<FirstAired>" + SecurityElement.Escape(episode.PremiereDate.Value.ToLocalTime().ToString("yyyy-MM-dd")) + "</FirstAired>");
+ }
+
+ XmlSaverHelpers.AddCommonNodes(episode, builder);
+ XmlSaverHelpers.AddMediaInfo(episode, builder, _itemRepository);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string>
+ {
+ "FirstAired",
+ "SeasonNumber",
+ "EpisodeNumber",
+ "EpisodeName",
+ "EpisodeNumberEnd",
+ "airsafter_season",
+ "airsbefore_episode",
+ "airsbefore_season",
+ "DVD_episodenumber",
+ "DVD_season",
+ "absolute_number"
+ });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ var filename = Path.ChangeExtension(Path.GetFileName(item.Path), ".xml");
+
+ return Path.Combine(Path.GetDirectoryName(item.Path), "metadata", filename);
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/FolderXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/FolderXmlSaver.cs
new file mode 100644
index 000000000..6dd65b69c
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/FolderXmlSaver.cs
@@ -0,0 +1,80 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.Audio;
+using MediaBrowser.Controller.Entities.Movies;
+using MediaBrowser.Controller.Entities.TV;
+using MediaBrowser.Controller.Library;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ public class FolderXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ if (item is Folder)
+ {
+ if (!(item is Series) && !(item is BoxSet) && !(item is MusicArtist) && !(item is MusicAlbum) &&
+ !(item is Season) &&
+ !(item is GameSystem))
+ {
+ return updateType >= ItemUpdateType.MetadataDownload;
+ }
+ }
+
+ return false;
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ XmlSaverHelpers.AddCommonNodes((Folder)item, builder);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string> { });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "folder.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/GameSystemXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/GameSystemXmlSaver.cs
new file mode 100644
index 000000000..163c79ce2
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/GameSystemXmlSaver.cs
@@ -0,0 +1,75 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Security;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ public class GameSystemXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is GameSystem && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var gameSystem = (GameSystem)item;
+
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ if (!string.IsNullOrEmpty(gameSystem.GameSystemName))
+ {
+ builder.Append("<GameSystem>" + SecurityElement.Escape(gameSystem.GameSystemName) + "</GameSystem>");
+ }
+
+ XmlSaverHelpers.AddCommonNodes(gameSystem, builder);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string> { });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "gamesystem.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/GameXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/GameXmlSaver.cs
new file mode 100644
index 000000000..7eeaa211f
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/GameXmlSaver.cs
@@ -0,0 +1,112 @@
+using System.Collections.Generic;
+using System.Globalization;
+using System.IO;
+using System.Security;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Model.Entities;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ /// <summary>
+ /// Saves game.xml for games
+ /// </summary>
+ public class GameXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is Game && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ private static readonly CultureInfo UsCulture = new CultureInfo("en-US");
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ var game = (Game)item;
+
+ if (game.PlayersSupported.HasValue)
+ {
+ builder.Append("<Players>" + SecurityElement.Escape(game.PlayersSupported.Value.ToString(UsCulture)) + "</Players>");
+ }
+
+ if (!string.IsNullOrEmpty(game.GameSystem))
+ {
+ builder.Append("<GameSystem>" + SecurityElement.Escape(game.GameSystem) + "</GameSystem>");
+ }
+
+ var val = game.GetProviderId(MetadataProviders.NesBox);
+
+ if (!string.IsNullOrEmpty(val))
+ {
+ builder.Append("<NesBox>" + SecurityElement.Escape(val) + "</NesBox>");
+ }
+
+ val = game.GetProviderId(MetadataProviders.NesBoxRom);
+
+ if (!string.IsNullOrEmpty(val))
+ {
+ builder.Append("<NesBoxRom>" + SecurityElement.Escape(val) + "</NesBoxRom>");
+ }
+
+ XmlSaverHelpers.AddCommonNodes(game, builder);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string>
+ {
+ "Players",
+ "GameSystem",
+ "NesBox",
+ "NesBoxRom"
+ });
+ }
+
+ public string GetSavePath(IHasMetadata item)
+ {
+ return GetGameSavePath((Game)item);
+ }
+
+ public static string GetGameSavePath(Game item)
+ {
+ if (item.IsInMixedFolder)
+ {
+ return Path.ChangeExtension(item.Path, ".xml");
+ }
+
+ return Path.Combine(item.ContainingFolderPath, "game.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/MovieXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/MovieXmlSaver.cs
new file mode 100644
index 000000000..ef81790a6
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/MovieXmlSaver.cs
@@ -0,0 +1,133 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Security;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.Movies;
+using MediaBrowser.Controller.Entities.TV;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Persistence;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ /// <summary>
+ /// Saves movie.xml for movies, trailers and music videos
+ /// </summary>
+ public class MovieXmlSaver : IMetadataFileSaver
+ {
+ private readonly IItemRepository _itemRepository;
+
+ public MovieXmlSaver(IItemRepository itemRepository)
+ {
+ _itemRepository = itemRepository;
+ }
+
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ var video = item as Video;
+
+ // Check parent for null to avoid running this against things like video backdrops
+ if (video != null && !(item is Episode) && !video.IsOwnedItem)
+ {
+ return updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ return false;
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var video = (Video)item;
+
+ var builder = new StringBuilder();
+
+ builder.Append("<Title>");
+
+ XmlSaverHelpers.AddCommonNodes(video, builder);
+
+ var musicVideo = item as MusicVideo;
+
+ if (musicVideo != null)
+ {
+ if (!string.IsNullOrEmpty(musicVideo.Artist))
+ {
+ builder.Append("<Artist>" + SecurityElement.Escape(musicVideo.Artist) + "</Artist>");
+ }
+ if (!string.IsNullOrEmpty(musicVideo.Album))
+ {
+ builder.Append("<Album>" + SecurityElement.Escape(musicVideo.Album) + "</Album>");
+ }
+ }
+
+ var movie = item as Movie;
+
+ if (movie != null)
+ {
+ if (!string.IsNullOrEmpty(movie.TmdbCollectionName))
+ {
+ builder.Append("<TmdbCollectionName>" + SecurityElement.Escape(movie.TmdbCollectionName) + "</TmdbCollectionName>");
+ }
+ }
+
+ XmlSaverHelpers.AddMediaInfo(video, builder, _itemRepository);
+
+ builder.Append("</Title>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string>
+ {
+ // Deprecated. No longer saving in this field.
+ "IMDBrating",
+
+ // Deprecated. No longer saving in this field.
+ "Description",
+
+ "Artist",
+ "Album",
+ "TmdbCollectionName"
+ });
+ }
+
+ public string GetSavePath(IHasMetadata item)
+ {
+ return GetMovieSavePath((Video)item);
+ }
+
+ public static string GetMovieSavePath(Video item)
+ {
+ if (item.IsInMixedFolder)
+ {
+ return Path.ChangeExtension(item.Path, ".xml");
+ }
+
+ return Path.Combine(item.ContainingFolderPath, "movie.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/PersonXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/PersonXmlSaver.cs
new file mode 100644
index 000000000..2ea60f47c
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/PersonXmlSaver.cs
@@ -0,0 +1,81 @@
+using System.Collections.Generic;
+using System.IO;
+using System.Security;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ /// <summary>
+ /// Class PersonXmlSaver
+ /// </summary>
+ public class PersonXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is Person && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var person = (Person)item;
+
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ XmlSaverHelpers.AddCommonNodes(person, builder);
+
+ if (!string.IsNullOrEmpty(person.PlaceOfBirth))
+ {
+ builder.Append("<PlaceOfBirth>" + SecurityElement.Escape(person.PlaceOfBirth) + "</PlaceOfBirth>");
+ }
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string>
+ {
+ "PlaceOfBirth"
+ });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "person.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/SeasonXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/SeasonXmlSaver.cs
new file mode 100644
index 000000000..b9908875d
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/SeasonXmlSaver.cs
@@ -0,0 +1,87 @@
+using System.Collections.Generic;
+using System.Globalization;
+using System.IO;
+using System.Security;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.TV;
+using MediaBrowser.Controller.Library;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ public class SeasonXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ if (!(item is Season))
+ {
+ return false;
+ }
+
+ return updateType >= ItemUpdateType.MetadataDownload || (updateType >= ItemUpdateType.MetadataImport && File.Exists(GetSavePath(item)));
+ }
+
+ private readonly CultureInfo _usCulture = new CultureInfo("en-US");
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var builder = new StringBuilder();
+
+ builder.Append("<Item>");
+
+ var season = (Season)item;
+
+ if (season.IndexNumber.HasValue)
+ {
+ builder.Append("<SeasonNumber>" + SecurityElement.Escape(season.IndexNumber.Value.ToString(_usCulture)) + "</SeasonNumber>");
+ }
+
+ XmlSaverHelpers.AddCommonNodes((Season)item, builder);
+
+ builder.Append("</Item>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string>
+ {
+ "SeasonNumber"
+ });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "season.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/SeriesXmlSaver.cs b/MediaBrowser.LocalMetadata/Savers/SeriesXmlSaver.cs
new file mode 100644
index 000000000..23ea52820
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/SeriesXmlSaver.cs
@@ -0,0 +1,135 @@
+using System.Collections.Generic;
+using System.Globalization;
+using System.IO;
+using System.Security;
+using System.Text;
+using System.Threading;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.TV;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Model.Entities;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ public class SeriesXmlSaver : IMetadataFileSaver
+ {
+ public string Name
+ {
+ get
+ {
+ return "Media Browser Xml";
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is enabled for] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="updateType">Type of the update.</param>
+ /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
+ public bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType)
+ {
+ if (!item.SupportsLocalMetadata)
+ {
+ return false;
+ }
+
+ return item is Series && updateType >= ItemUpdateType.MetadataDownload;
+ }
+
+ private static readonly CultureInfo UsCulture = new CultureInfo("en-US");
+
+ /// <summary>
+ /// Saves the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ public void Save(IHasMetadata item, CancellationToken cancellationToken)
+ {
+ var series = (Series)item;
+
+ var builder = new StringBuilder();
+
+ builder.Append("<Series>");
+
+ var tvdb = item.GetProviderId(MetadataProviders.Tvdb);
+
+ if (!string.IsNullOrEmpty(tvdb))
+ {
+ builder.Append("<id>" + SecurityElement.Escape(tvdb) + "</id>");
+ }
+
+ if (series.Status.HasValue)
+ {
+ builder.Append("<Status>" + SecurityElement.Escape(series.Status.Value.ToString()) + "</Status>");
+ }
+
+ if (series.Studios.Count > 0)
+ {
+ builder.Append("<Network>" + SecurityElement.Escape(series.Studios[0]) + "</Network>");
+ }
+
+ if (!string.IsNullOrEmpty(series.AirTime))
+ {
+ builder.Append("<Airs_Time>" + SecurityElement.Escape(series.AirTime) + "</Airs_Time>");
+ }
+
+ if (series.AirDays != null)
+ {
+ if (series.AirDays.Count == 7)
+ {
+ builder.Append("<Airs_DayOfWeek>" + SecurityElement.Escape("Daily") + "</Airs_DayOfWeek>");
+ }
+ else if (series.AirDays.Count > 0)
+ {
+ builder.Append("<Airs_DayOfWeek>" + SecurityElement.Escape(series.AirDays[0].ToString()) + "</Airs_DayOfWeek>");
+ }
+ }
+
+ if (series.PremiereDate.HasValue)
+ {
+ builder.Append("<FirstAired>" + SecurityElement.Escape(series.PremiereDate.Value.ToLocalTime().ToString("yyyy-MM-dd")) + "</FirstAired>");
+ }
+
+ if (series.AnimeSeriesIndex.HasValue)
+ {
+ builder.Append("<AnimeSeriesIndex>" + SecurityElement.Escape(series.AnimeSeriesIndex.Value.ToString(UsCulture)) + "</AnimeSeriesIndex>");
+ }
+
+ XmlSaverHelpers.AddCommonNodes(series, builder);
+
+ builder.Append("</Series>");
+
+ var xmlFilePath = GetSavePath(item);
+
+ XmlSaverHelpers.Save(builder, xmlFilePath, new List<string>
+ {
+ "id",
+ "Status",
+ "Network",
+ "Airs_Time",
+ "Airs_DayOfWeek",
+ "FirstAired",
+
+ // Don't preserve old series node
+ "Series",
+
+ "SeriesName",
+
+ // Deprecated. No longer saving in this field.
+ "AnimeSeriesIndex"
+ });
+ }
+
+ /// <summary>
+ /// Gets the save path.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>System.String.</returns>
+ public string GetSavePath(IHasMetadata item)
+ {
+ return Path.Combine(item.Path, "series.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.LocalMetadata/Savers/XmlSaverHelpers.cs b/MediaBrowser.LocalMetadata/Savers/XmlSaverHelpers.cs
new file mode 100644
index 000000000..c43875b04
--- /dev/null
+++ b/MediaBrowser.LocalMetadata/Savers/XmlSaverHelpers.cs
@@ -0,0 +1,718 @@
+using System;
+using System.Collections.Generic;
+using System.Globalization;
+using System.IO;
+using System.Linq;
+using System.Security;
+using System.Text;
+using System.Xml;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.Movies;
+using MediaBrowser.Controller.Entities.TV;
+using MediaBrowser.Controller.Persistence;
+using MediaBrowser.Model.Entities;
+
+namespace MediaBrowser.LocalMetadata.Savers
+{
+ /// <summary>
+ /// Class XmlHelpers
+ /// </summary>
+ public static class XmlSaverHelpers
+ {
+ private static readonly Dictionary<string, string> CommonTags = new[] {
+
+ "Added",
+ "AspectRatio",
+ "AudioDbAlbumId",
+ "AudioDbArtistId",
+ "AwardSummary",
+ "BirthDate",
+ "Budget",
+
+ // Deprecated. No longer saving in this field.
+ "certification",
+
+ "Chapters",
+ "ContentRating",
+ "Countries",
+ "CustomRating",
+ "CriticRating",
+ "CriticRatingSummary",
+ "DeathDate",
+ "DisplayOrder",
+ "EndDate",
+ "Genres",
+ "Genre",
+ "GamesDbId",
+
+ // Deprecated. No longer saving in this field.
+ "IMDB_ID",
+
+ "IMDB",
+
+ // Deprecated. No longer saving in this field.
+ "IMDbId",
+
+ "Language",
+ "LocalTitle",
+ "LockData",
+ "LockedFields",
+ "Format3D",
+ "Metascore",
+
+ // Deprecated. No longer saving in this field.
+ "MPAARating",
+
+ "MusicBrainzArtistId",
+ "MusicBrainzAlbumArtistId",
+ "MusicBrainzAlbumId",
+ "MusicBrainzReleaseGroupId",
+
+ // Deprecated. No longer saving in this field.
+ "MusicbrainzId",
+
+ "Overview",
+ "ShortOverview",
+ "Persons",
+ "PlotKeywords",
+ "PremiereDate",
+ "ProductionYear",
+ "Rating",
+ "Revenue",
+ "RottenTomatoesId",
+ "RunningTime",
+
+ // Deprecated. No longer saving in this field.
+ "Runtime",
+
+ "SortTitle",
+ "Studios",
+ "Tags",
+
+ // Deprecated. No longer saving in this field.
+ "TagLine",
+
+ "Taglines",
+ "TMDbCollectionId",
+ "TMDbId",
+
+ // Deprecated. No longer saving in this field.
+ "Trailer",
+
+ "Trailers",
+ "TVcomId",
+ "TvDbId",
+ "Type",
+ "TVRageId",
+ "VoteCount",
+ "Website",
+ "Zap2ItId",
+ "CollectionItems"
+
+ }.ToDictionary(i => i, StringComparer.OrdinalIgnoreCase);
+
+ /// <summary>
+ /// The us culture
+ /// </summary>
+ private static readonly CultureInfo UsCulture = new CultureInfo("en-US");
+
+ /// <summary>
+ /// Saves the specified XML.
+ /// </summary>
+ /// <param name="xml">The XML.</param>
+ /// <param name="path">The path.</param>
+ /// <param name="xmlTagsUsed">The XML tags used.</param>
+ public static void Save(StringBuilder xml, string path, List<string> xmlTagsUsed)
+ {
+ if (File.Exists(path))
+ {
+ var position = xml.ToString().LastIndexOf("</", StringComparison.OrdinalIgnoreCase);
+ xml.Insert(position, GetCustomTags(path, xmlTagsUsed));
+ }
+
+ var xmlDocument = new XmlDocument();
+ xmlDocument.LoadXml(xml.ToString());
+
+ //Add the new node to the document.
+ xmlDocument.InsertBefore(xmlDocument.CreateXmlDeclaration("1.0", "UTF-8", "yes"), xmlDocument.DocumentElement);
+
+ Directory.CreateDirectory(Path.GetDirectoryName(path));
+
+ var wasHidden = false;
+
+ var file = new FileInfo(path);
+
+ // This will fail if the file is hidden
+ if (file.Exists)
+ {
+ if ((file.Attributes & FileAttributes.Hidden) == FileAttributes.Hidden)
+ {
+ file.Attributes &= ~FileAttributes.Hidden;
+
+ wasHidden = true;
+ }
+ }
+
+ using (var filestream = new FileStream(path, FileMode.Create, FileAccess.Write, FileShare.Read))
+ {
+ using (var streamWriter = new StreamWriter(filestream, Encoding.UTF8))
+ {
+ xmlDocument.Save(streamWriter);
+ }
+ }
+
+ if (wasHidden)
+ {
+ file.Refresh();
+
+ // Add back the attribute
+ file.Attributes |= FileAttributes.Hidden;
+ }
+ }
+
+ /// <summary>
+ /// Gets the custom tags.
+ /// </summary>
+ /// <param name="path">The path.</param>
+ /// <param name="xmlTagsUsed">The XML tags used.</param>
+ /// <returns>System.String.</returns>
+ private static string GetCustomTags(string path, List<string> xmlTagsUsed)
+ {
+ var settings = new XmlReaderSettings
+ {
+ CheckCharacters = false,
+ IgnoreProcessingInstructions = true,
+ IgnoreComments = true,
+ ValidationType = ValidationType.None
+ };
+
+ var builder = new StringBuilder();
+
+ using (var streamReader = new StreamReader(path, Encoding.UTF8))
+ {
+ // Use XmlReader for best performance
+ using (var reader = XmlReader.Create(streamReader, settings))
+ {
+ reader.MoveToContent();
+
+ // Loop through each element
+ while (reader.Read())
+ {
+ if (reader.NodeType == XmlNodeType.Element)
+ {
+ var name = reader.Name;
+
+ if (!CommonTags.ContainsKey(name) && !xmlTagsUsed.Contains(name, StringComparer.OrdinalIgnoreCase))
+ {
+ builder.AppendLine(reader.ReadOuterXml());
+ }
+ else
+ {
+ reader.Skip();
+ }
+ }
+ }
+ }
+ }
+
+ return builder.ToString();
+ }
+
+ /// <summary>
+ /// Adds the common nodes.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="builder">The builder.</param>
+ public static void AddCommonNodes(BaseItem item, StringBuilder builder)
+ {
+ if (!string.IsNullOrEmpty(item.OfficialRating))
+ {
+ builder.Append("<ContentRating>" + SecurityElement.Escape(item.OfficialRating) + "</ContentRating>");
+ }
+
+ builder.Append("<Added>" + SecurityElement.Escape(item.DateCreated.ToLocalTime().ToString("G")) + "</Added>");
+
+ builder.Append("<LockData>" + item.IsLocked.ToString().ToLower() + "</LockData>");
+
+ if (item.LockedFields.Count > 0)
+ {
+ builder.Append("<LockedFields>" + string.Join("|", item.LockedFields.Select(i => i.ToString()).ToArray()) + "</LockedFields>");
+ }
+
+ if (!string.IsNullOrEmpty(item.DisplayMediaType))
+ {
+ builder.Append("<Type>" + SecurityElement.Escape(item.DisplayMediaType) + "</Type>");
+ }
+
+ var hasCriticRating = item as IHasCriticRating;
+ if (hasCriticRating != null)
+ {
+ if (hasCriticRating.CriticRating.HasValue)
+ {
+ builder.Append("<CriticRating>" + SecurityElement.Escape(hasCriticRating.CriticRating.Value.ToString(UsCulture)) + "</CriticRating>");
+ }
+
+ if (!string.IsNullOrEmpty(hasCriticRating.CriticRatingSummary))
+ {
+ builder.Append("<CriticRatingSummary><![CDATA[" + hasCriticRating.CriticRatingSummary + "]]></CriticRatingSummary>");
+ }
+ }
+
+ if (!string.IsNullOrEmpty(item.Overview))
+ {
+ builder.Append("<Overview><![CDATA[" + item.Overview + "]]></Overview>");
+ }
+
+ var hasShortOverview = item as IHasShortOverview;
+ if (hasShortOverview != null)
+ {
+ if (!string.IsNullOrEmpty(hasShortOverview.ShortOverview))
+ {
+ builder.Append("<ShortOverview><![CDATA[" + hasShortOverview.ShortOverview + "]]></ShortOverview>");
+ }
+ }
+
+ if (!string.IsNullOrEmpty(item.CustomRating))
+ {
+ builder.Append("<CustomRating>" + SecurityElement.Escape(item.CustomRating) + "</CustomRating>");
+ }
+
+ if (!string.IsNullOrEmpty(item.Name) && !(item is Episode))
+ {
+ builder.Append("<LocalTitle>" + SecurityElement.Escape(item.Name) + "</LocalTitle>");
+ }
+
+ if (!string.IsNullOrEmpty(item.ForcedSortName))
+ {
+ builder.Append("<SortTitle>" + SecurityElement.Escape(item.ForcedSortName) + "</SortTitle>");
+ }
+
+ if (item.PremiereDate.HasValue)
+ {
+ if (item is Person)
+ {
+ builder.Append("<BirthDate>" + SecurityElement.Escape(item.PremiereDate.Value.ToLocalTime().ToString("yyyy-MM-dd")) + "</BirthDate>");
+ }
+ else if (!(item is Episode))
+ {
+ builder.Append("<PremiereDate>" + SecurityElement.Escape(item.PremiereDate.Value.ToLocalTime().ToString("yyyy-MM-dd")) + "</PremiereDate>");
+ }
+ }
+
+ if (item.EndDate.HasValue)
+ {
+ if (item is Person)
+ {
+ builder.Append("<DeathDate>" + SecurityElement.Escape(item.EndDate.Value.ToString("yyyy-MM-dd")) + "</DeathDate>");
+ }
+ else if (!(item is Episode))
+ {
+ builder.Append("<EndDate>" + SecurityElement.Escape(item.EndDate.Value.ToString("yyyy-MM-dd")) + "</EndDate>");
+ }
+ }
+
+ var hasTrailers = item as IHasTrailers;
+ if (hasTrailers != null)
+ {
+ if (hasTrailers.RemoteTrailers.Count > 0)
+ {
+ builder.Append("<Trailers>");
+
+ foreach (var trailer in hasTrailers.RemoteTrailers)
+ {
+ builder.Append("<Trailer>" + SecurityElement.Escape(trailer.Url) + "</Trailer>");
+ }
+
+ builder.Append("</Trailers>");
+ }
+ }
+
+ var hasProductionLocations = item as IHasProductionLocations;
+ if (hasProductionLocations != null)
+ {
+ if (hasProductionLocations.ProductionLocations.Count > 0)
+ {
+ builder.Append("<Countries>");
+
+ foreach (var name in hasProductionLocations.ProductionLocations)
+ {
+ builder.Append("<Country>" + SecurityElement.Escape(name) + "</Country>");
+ }
+
+ builder.Append("</Countries>");
+ }
+ }
+
+ var hasDisplayOrder = item as IHasDisplayOrder;
+ if (hasDisplayOrder != null && !string.IsNullOrEmpty(hasDisplayOrder.DisplayOrder))
+ {
+ builder.Append("<DisplayOrder>" + SecurityElement.Escape(hasDisplayOrder.DisplayOrder) + "</DisplayOrder>");
+ }
+
+ var hasMetascore = item as IHasMetascore;
+ if (hasMetascore != null && hasMetascore.Metascore.HasValue)
+ {
+ builder.Append("<Metascore>" + SecurityElement.Escape(hasMetascore.Metascore.Value.ToString(UsCulture)) + "</Metascore>");
+ }
+
+ var hasAwards = item as IHasAwards;
+ if (hasAwards != null && !string.IsNullOrEmpty(hasAwards.AwardSummary))
+ {
+ builder.Append("<AwardSummary>" + SecurityElement.Escape(hasAwards.AwardSummary) + "</AwardSummary>");
+ }
+
+ var hasBudget = item as IHasBudget;
+ if (hasBudget != null)
+ {
+ if (hasBudget.Budget.HasValue)
+ {
+ builder.Append("<Budget>" + SecurityElement.Escape(hasBudget.Budget.Value.ToString(UsCulture)) + "</Budget>");
+ }
+
+ if (hasBudget.Revenue.HasValue)
+ {
+ builder.Append("<Revenue>" + SecurityElement.Escape(hasBudget.Revenue.Value.ToString(UsCulture)) + "</Revenue>");
+ }
+ }
+
+ if (item.CommunityRating.HasValue)
+ {
+ builder.Append("<Rating>" + SecurityElement.Escape(item.CommunityRating.Value.ToString(UsCulture)) + "</Rating>");
+ }
+ if (item.VoteCount.HasValue)
+ {
+ builder.Append("<VoteCount>" + SecurityElement.Escape(item.VoteCount.Value.ToString(UsCulture)) + "</VoteCount>");
+ }
+
+ if (item.ProductionYear.HasValue && !(item is Person))
+ {
+ builder.Append("<ProductionYear>" + SecurityElement.Escape(item.ProductionYear.Value.ToString(UsCulture)) + "</ProductionYear>");
+ }
+
+ if (!string.IsNullOrEmpty(item.HomePageUrl))
+ {
+ builder.Append("<Website>" + SecurityElement.Escape(item.HomePageUrl) + "</Website>");
+ }
+
+ var hasAspectRatio = item as IHasAspectRatio;
+ if (hasAspectRatio != null)
+ {
+ if (!string.IsNullOrEmpty(hasAspectRatio.AspectRatio))
+ {
+ builder.Append("<AspectRatio>" + SecurityElement.Escape(hasAspectRatio.AspectRatio) + "</AspectRatio>");
+ }
+ }
+
+ var hasLanguage = item as IHasPreferredMetadataLanguage;
+ if (hasLanguage != null)
+ {
+ if (!string.IsNullOrEmpty(hasLanguage.PreferredMetadataLanguage))
+ {
+ builder.Append("<Language>" + SecurityElement.Escape(hasLanguage.PreferredMetadataLanguage) + "</Language>");
+ }
+ }
+
+ // Use original runtime here, actual file runtime later in MediaInfo
+ var runTimeTicks = item.RunTimeTicks;
+
+ if (runTimeTicks.HasValue)
+ {
+ var timespan = TimeSpan.FromTicks(runTimeTicks.Value);
+
+ builder.Append("<RunningTime>" + Convert.ToInt32(timespan.TotalMinutes).ToString(UsCulture) + "</RunningTime>");
+ }
+
+ var imdb = item.GetProviderId(MetadataProviders.Imdb);
+
+ if (!string.IsNullOrEmpty(imdb))
+ {
+ builder.Append("<IMDB>" + SecurityElement.Escape(imdb) + "</IMDB>");
+ }
+
+ var tmdb = item.GetProviderId(MetadataProviders.Tmdb);
+
+ if (!string.IsNullOrEmpty(tmdb))
+ {
+ builder.Append("<TMDbId>" + SecurityElement.Escape(tmdb) + "</TMDbId>");
+ }
+
+ if (!(item is Series))
+ {
+ var tvdb = item.GetProviderId(MetadataProviders.Tvdb);
+
+ if (!string.IsNullOrEmpty(tvdb))
+ {
+ builder.Append("<TvDbId>" + SecurityElement.Escape(tvdb) + "</TvDbId>");
+ }
+ }
+
+ var externalId = item.GetProviderId(MetadataProviders.Tvcom);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<TVcomId>" + SecurityElement.Escape(externalId) + "</TVcomId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.RottenTomatoes);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<RottenTomatoesId>" + SecurityElement.Escape(externalId) + "</RottenTomatoesId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.Zap2It);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<Zap2ItId>" + SecurityElement.Escape(externalId) + "</Zap2ItId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.MusicBrainzAlbum);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<MusicBrainzAlbumId>" + SecurityElement.Escape(externalId) + "</MusicBrainzAlbumId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.MusicBrainzAlbumArtist);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<MusicBrainzAlbumArtistId>" + SecurityElement.Escape(externalId) + "</MusicBrainzAlbumArtistId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.MusicBrainzArtist);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<MusicBrainzArtistId>" + SecurityElement.Escape(externalId) + "</MusicBrainzArtistId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.MusicBrainzReleaseGroup);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<MusicBrainzReleaseGroupId>" + SecurityElement.Escape(externalId) + "</MusicBrainzReleaseGroupId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.Gamesdb);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<GamesDbId>" + SecurityElement.Escape(externalId) + "</GamesDbId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.TmdbCollection);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<TMDbCollectionId>" + SecurityElement.Escape(externalId) + "</TMDbCollectionId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.AudioDbArtist);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<AudioDbArtistId>" + SecurityElement.Escape(externalId) + "</AudioDbArtistId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.AudioDbAlbum);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<AudioDbAlbumId>" + SecurityElement.Escape(externalId) + "</AudioDbAlbumId>");
+ }
+
+ externalId = item.GetProviderId(MetadataProviders.TvRage);
+
+ if (!string.IsNullOrEmpty(externalId))
+ {
+ builder.Append("<TVRageId>" + SecurityElement.Escape(externalId) + "</TVRageId>");
+ }
+
+ var hasTagline = item as IHasTaglines;
+ if (hasTagline != null)
+ {
+ if (hasTagline.Taglines.Count > 0)
+ {
+ builder.Append("<Taglines>");
+
+ foreach (var tagline in hasTagline.Taglines)
+ {
+ builder.Append("<Tagline>" + SecurityElement.Escape(tagline) + "</Tagline>");
+ }
+
+ builder.Append("</Taglines>");
+ }
+ }
+
+ if (item.Genres.Count > 0)
+ {
+ builder.Append("<Genres>");
+
+ foreach (var genre in item.Genres)
+ {
+ builder.Append("<Genre>" + SecurityElement.Escape(genre) + "</Genre>");
+ }
+
+ builder.Append("</Genres>");
+ }
+
+ if (item.Studios.Count > 0)
+ {
+ builder.Append("<Studios>");
+
+ foreach (var studio in item.Studios)
+ {
+ builder.Append("<Studio>" + SecurityElement.Escape(studio) + "</Studio>");
+ }
+
+ builder.Append("</Studios>");
+ }
+
+ var hasTags = item as IHasTags;
+ if (hasTags != null)
+ {
+ if (hasTags.Tags.Count > 0)
+ {
+ builder.Append("<Tags>");
+
+ foreach (var tag in hasTags.Tags)
+ {
+ builder.Append("<Tag>" + SecurityElement.Escape(tag) + "</Tag>");
+ }
+
+ builder.Append("</Tags>");
+ }
+ }
+
+ var hasKeywords = item as IHasKeywords;
+ if (hasKeywords != null)
+ {
+ if (hasKeywords.Keywords.Count > 0)
+ {
+ builder.Append("<PlotKeywords>");
+
+ foreach (var tag in hasKeywords.Keywords)
+ {
+ builder.Append("<PlotKeyword>" + SecurityElement.Escape(tag) + "</PlotKeyword>");
+ }
+
+ builder.Append("</PlotKeywords>");
+ }
+ }
+
+ if (item.People.Count > 0)
+ {
+ builder.Append("<Persons>");
+
+ foreach (var person in item.People)
+ {
+ builder.Append("<Person>");
+ builder.Append("<Name>" + SecurityElement.Escape(person.Name) + "</Name>");
+ builder.Append("<Type>" + SecurityElement.Escape(person.Type) + "</Type>");
+ builder.Append("<Role>" + SecurityElement.Escape(person.Role) + "</Role>");
+
+ if (person.SortOrder.HasValue)
+ {
+ builder.Append("<SortOrder>" + SecurityElement.Escape(person.SortOrder.Value.ToString(UsCulture)) + "</SortOrder>");
+ }
+
+ builder.Append("</Person>");
+ }
+
+ builder.Append("</Persons>");
+ }
+
+ var folder = item as BoxSet;
+ if (folder != null)
+ {
+ AddCollectionItems(folder, builder);
+ }
+ }
+
+ public static void AddChapters(Video item, StringBuilder builder, IItemRepository repository)
+ {
+ var chapters = repository.GetChapters(item.Id);
+
+ builder.Append("<Chapters>");
+
+ foreach (var chapter in chapters)
+ {
+ builder.Append("<Chapter>");
+ builder.Append("<Name>" + SecurityElement.Escape(chapter.Name) + "</Name>");
+
+ var time = TimeSpan.FromTicks(chapter.StartPositionTicks);
+ var ms = Convert.ToInt64(time.TotalMilliseconds);
+
+ builder.Append("<StartPositionMs>" + SecurityElement.Escape(ms.ToString(UsCulture)) + "</StartPositionMs>");
+ builder.Append("</Chapter>");
+ }
+
+ builder.Append("</Chapters>");
+ }
+
+ /// <summary>
+ /// Appends the media info.
+ /// </summary>
+ /// <typeparam name="T"></typeparam>
+ public static void AddMediaInfo<T>(T item, StringBuilder builder, IItemRepository itemRepository)
+ where T : BaseItem
+ {
+ var video = item as Video;
+
+ if (video != null)
+ {
+ //AddChapters(video, builder, itemRepository);
+
+ if (video.Video3DFormat.HasValue)
+ {
+ switch (video.Video3DFormat.Value)
+ {
+ case Video3DFormat.FullSideBySide:
+ builder.Append("<Format3D>FSBS</Format3D>");
+ break;
+ case Video3DFormat.FullTopAndBottom:
+ builder.Append("<Format3D>FTAB</Format3D>");
+ break;
+ case Video3DFormat.HalfSideBySide:
+ builder.Append("<Format3D>HSBS</Format3D>");
+ break;
+ case Video3DFormat.HalfTopAndBottom:
+ builder.Append("<Format3D>HTAB</Format3D>");
+ break;
+ }
+ }
+ }
+ }
+
+ public static void AddCollectionItems(Folder item, StringBuilder builder)
+ {
+ var items = item.LinkedChildren
+ .Where(i => i.Type == LinkedChildType.Manual && !string.IsNullOrWhiteSpace(i.ItemName))
+ .ToList();
+
+ if (items.Count == 0)
+ {
+ return;
+ }
+
+ builder.Append("<CollectionItems>");
+ foreach (var link in items)
+ {
+ builder.Append("<CollectionItem>");
+
+ builder.Append("<Name>" + SecurityElement.Escape(link.ItemName) + "</Name>");
+ builder.Append("<Type>" + SecurityElement.Escape(link.ItemType) + "</Type>");
+
+ if (link.ItemYear.HasValue)
+ {
+ builder.Append("<Year>" + SecurityElement.Escape(link.ItemYear.Value.ToString(UsCulture)) + "</Year>");
+ }
+
+ builder.Append("</CollectionItem>");
+ }
+ builder.Append("</CollectionItems>");
+ }
+ }
+}