Support subtitle offset
This commit is contained in:
parent
f86e8a415a
commit
dd7825f6c8
|
@ -34,6 +34,9 @@ namespace MediaBrowser.Api.Library
|
||||||
|
|
||||||
[ApiMember(Name = "Format", Description = "Format", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
|
[ApiMember(Name = "Format", Description = "Format", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
|
||||||
public string Format { get; set; }
|
public string Format { get; set; }
|
||||||
|
|
||||||
|
[ApiMember(Name = "StartPositionTicks", Description = "StartPositionTicks", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
|
||||||
|
public long StartPositionTicks { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
[Route("/Videos/{Id}/Subtitles/{Index}", "DELETE", Summary = "Deletes an external subtitle file")]
|
[Route("/Videos/{Id}/Subtitles/{Index}", "DELETE", Summary = "Deletes an external subtitle file")]
|
||||||
|
@ -127,10 +130,12 @@ namespace MediaBrowser.Api.Library
|
||||||
|
|
||||||
private async Task<Stream> GetSubtitles(GetSubtitle request)
|
private async Task<Stream> GetSubtitles(GetSubtitle request)
|
||||||
{
|
{
|
||||||
var stream = await _subtitleEncoder.GetSubtitles(request.Id, request.MediaSourceId, request.Index, request.Format,
|
return await _subtitleEncoder.GetSubtitles(request.Id,
|
||||||
CancellationToken.None);
|
request.MediaSourceId,
|
||||||
|
request.Index,
|
||||||
return stream;
|
request.Format,
|
||||||
|
request.StartPositionTicks,
|
||||||
|
CancellationToken.None).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Delete(DeleteSubtitle request)
|
public void Delete(DeleteSubtitle request)
|
||||||
|
|
|
@ -65,30 +65,24 @@ namespace MediaBrowser.Api.Playback
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <value>The media encoder.</value>
|
/// <value>The media encoder.</value>
|
||||||
protected IMediaEncoder MediaEncoder { get; private set; }
|
protected IMediaEncoder MediaEncoder { get; private set; }
|
||||||
protected IEncodingManager EncodingManager { get; private set; }
|
|
||||||
protected IDtoService DtoService { get; private set; }
|
|
||||||
|
|
||||||
protected IFileSystem FileSystem { get; private set; }
|
protected IFileSystem FileSystem { get; private set; }
|
||||||
|
|
||||||
protected IItemRepository ItemRepository { get; private set; }
|
|
||||||
protected ILiveTvManager LiveTvManager { get; private set; }
|
protected ILiveTvManager LiveTvManager { get; private set; }
|
||||||
protected IDlnaManager DlnaManager { get; private set; }
|
protected IDlnaManager DlnaManager { get; private set; }
|
||||||
protected IChannelManager ChannelManager { get; private set; }
|
protected IChannelManager ChannelManager { get; private set; }
|
||||||
protected IHttpClient HttpClient { get; private set; }
|
protected ISubtitleEncoder SubtitleEncoder { get; private set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="BaseStreamingService" /> class.
|
/// Initializes a new instance of the <see cref="BaseStreamingService" /> class.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
protected BaseStreamingService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IDtoService dtoService, IFileSystem fileSystem, IItemRepository itemRepository, ILiveTvManager liveTvManager, IEncodingManager encodingManager, IDlnaManager dlnaManager, IChannelManager channelManager, IHttpClient httpClient)
|
protected BaseStreamingService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder)
|
||||||
{
|
{
|
||||||
HttpClient = httpClient;
|
SubtitleEncoder = subtitleEncoder;
|
||||||
ChannelManager = channelManager;
|
ChannelManager = channelManager;
|
||||||
DlnaManager = dlnaManager;
|
DlnaManager = dlnaManager;
|
||||||
EncodingManager = encodingManager;
|
|
||||||
LiveTvManager = liveTvManager;
|
LiveTvManager = liveTvManager;
|
||||||
ItemRepository = itemRepository;
|
|
||||||
FileSystem = fileSystem;
|
FileSystem = fileSystem;
|
||||||
DtoService = dtoService;
|
|
||||||
ServerConfigurationManager = serverConfig;
|
ServerConfigurationManager = serverConfig;
|
||||||
UserManager = userManager;
|
UserManager = userManager;
|
||||||
LibraryManager = libraryManager;
|
LibraryManager = libraryManager;
|
||||||
|
@ -587,7 +581,7 @@ namespace MediaBrowser.Api.Playback
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(state.SubtitleStream.Language))
|
if (!string.IsNullOrEmpty(state.SubtitleStream.Language))
|
||||||
{
|
{
|
||||||
var charenc = MediaEncoder.GetSubtitleLanguageEncodingParam(subtitlePath, state.SubtitleStream.Language);
|
var charenc = SubtitleEncoder.GetSubtitleFileCharacterSet(subtitlePath, state.SubtitleStream.Language);
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(charenc))
|
if (!string.IsNullOrEmpty(charenc))
|
||||||
{
|
{
|
||||||
|
@ -1374,6 +1368,12 @@ namespace MediaBrowser.Api.Playback
|
||||||
var path = recording.RecordingInfo.Path;
|
var path = recording.RecordingInfo.Path;
|
||||||
var mediaUrl = recording.RecordingInfo.Url;
|
var mediaUrl = recording.RecordingInfo.Url;
|
||||||
|
|
||||||
|
var source = string.IsNullOrEmpty(request.MediaSourceId)
|
||||||
|
? recording.GetMediaSources(false).First()
|
||||||
|
: recording.GetMediaSources(false).First(i => string.Equals(i.Id, request.MediaSourceId));
|
||||||
|
|
||||||
|
mediaStreams = source.MediaStreams;
|
||||||
|
|
||||||
if (string.IsNullOrWhiteSpace(path) && string.IsNullOrWhiteSpace(mediaUrl))
|
if (string.IsNullOrWhiteSpace(path) && string.IsNullOrWhiteSpace(mediaUrl))
|
||||||
{
|
{
|
||||||
var streamInfo = await LiveTvManager.GetRecordingStream(request.Id, cancellationToken).ConfigureAwait(false);
|
var streamInfo = await LiveTvManager.GetRecordingStream(request.Id, cancellationToken).ConfigureAwait(false);
|
||||||
|
@ -1453,51 +1453,45 @@ namespace MediaBrowser.Api.Playback
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
var mediaSource = string.IsNullOrWhiteSpace(request.MediaSourceId)
|
var hasMediaSources = (IHasMediaSources)item;
|
||||||
? item
|
var mediaSource = string.IsNullOrEmpty(request.MediaSourceId)
|
||||||
: LibraryManager.GetItemById(request.MediaSourceId);
|
? hasMediaSources.GetMediaSources(false).First()
|
||||||
|
: hasMediaSources.GetMediaSources(false).First(i => string.Equals(i.Id, request.MediaSourceId));
|
||||||
|
|
||||||
|
mediaStreams = mediaSource.MediaStreams;
|
||||||
|
|
||||||
state.MediaPath = mediaSource.Path;
|
state.MediaPath = mediaSource.Path;
|
||||||
state.IsRemote = mediaSource.LocationType == LocationType.Remote;
|
state.IsRemote = mediaSource.LocationType == LocationType.Remote;
|
||||||
|
state.InputContainer = mediaSource.Container;
|
||||||
|
|
||||||
var video = mediaSource as Video;
|
if (item is Video)
|
||||||
|
|
||||||
if (video != null)
|
|
||||||
{
|
{
|
||||||
state.IsInputVideo = true;
|
state.IsInputVideo = true;
|
||||||
state.VideoType = video.VideoType;
|
|
||||||
state.IsoType = video.IsoType;
|
|
||||||
|
|
||||||
state.PlayableStreamFileNames = video.PlayableStreamFileNames == null
|
if (mediaSource.VideoType.HasValue)
|
||||||
? new List<string>()
|
|
||||||
: video.PlayableStreamFileNames.ToList();
|
|
||||||
|
|
||||||
state.DeInterlace = string.Equals(video.Container, "wtv", StringComparison.OrdinalIgnoreCase);
|
|
||||||
|
|
||||||
if (video.Timestamp.HasValue)
|
|
||||||
{
|
{
|
||||||
state.InputTimestamp = video.Timestamp.Value;
|
state.VideoType = mediaSource.VideoType.Value;
|
||||||
}
|
}
|
||||||
|
|
||||||
state.InputContainer = video.Container;
|
state.IsoType = mediaSource.IsoType;
|
||||||
}
|
|
||||||
|
|
||||||
var audio = mediaSource as Audio;
|
//state.PlayableStreamFileNames = mediaSource.PlayableStreamFileNames.ToList();
|
||||||
if (audio != null)
|
|
||||||
{
|
if (mediaSource.Timestamp.HasValue)
|
||||||
state.InputContainer = audio.Container;
|
{
|
||||||
|
state.InputTimestamp = mediaSource.Timestamp.Value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
state.RunTimeTicks = mediaSource.RunTimeTicks;
|
state.RunTimeTicks = mediaSource.RunTimeTicks;
|
||||||
}
|
}
|
||||||
|
|
||||||
var videoRequest = request as VideoStreamRequest;
|
if (string.Equals(state.InputContainer, "wtv", StringComparison.OrdinalIgnoreCase))
|
||||||
|
|
||||||
mediaStreams = mediaStreams ?? ItemRepository.GetMediaStreams(new MediaStreamQuery
|
|
||||||
{
|
{
|
||||||
ItemId = new Guid(string.IsNullOrWhiteSpace(request.MediaSourceId) ? request.Id : request.MediaSourceId)
|
state.DeInterlace = true;
|
||||||
|
}
|
||||||
|
|
||||||
}).ToList();
|
var videoRequest = request as VideoStreamRequest;
|
||||||
|
|
||||||
AttachMediaStreamInfo(state, mediaStreams, videoRequest, url);
|
AttachMediaStreamInfo(state, mediaStreams, videoRequest, url);
|
||||||
|
|
||||||
|
|
|
@ -25,7 +25,7 @@ namespace MediaBrowser.Api.Playback.Hls
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public abstract class BaseHlsService : BaseStreamingService
|
public abstract class BaseHlsService : BaseStreamingService
|
||||||
{
|
{
|
||||||
protected BaseHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IDtoService dtoService, IFileSystem fileSystem, IItemRepository itemRepository, ILiveTvManager liveTvManager, IEncodingManager encodingManager, IDlnaManager dlnaManager, IChannelManager channelManager, IHttpClient httpClient) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, dtoService, fileSystem, itemRepository, liveTvManager, encodingManager, dlnaManager, channelManager, httpClient)
|
protected BaseHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, liveTvManager, dlnaManager, channelManager, subtitleEncoder)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -62,7 +62,7 @@ namespace MediaBrowser.Api.Playback.Hls
|
||||||
|
|
||||||
public class DynamicHlsService : BaseHlsService
|
public class DynamicHlsService : BaseHlsService
|
||||||
{
|
{
|
||||||
public DynamicHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IDtoService dtoService, IFileSystem fileSystem, IItemRepository itemRepository, ILiveTvManager liveTvManager, IEncodingManager encodingManager, IDlnaManager dlnaManager, IChannelManager channelManager, IHttpClient httpClient) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, dtoService, fileSystem, itemRepository, liveTvManager, encodingManager, dlnaManager, channelManager, httpClient)
|
public DynamicHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, liveTvManager, dlnaManager, channelManager, subtitleEncoder)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -56,7 +56,7 @@ namespace MediaBrowser.Api.Playback.Hls
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class VideoHlsService : BaseHlsService
|
public class VideoHlsService : BaseHlsService
|
||||||
{
|
{
|
||||||
public VideoHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IDtoService dtoService, IFileSystem fileSystem, IItemRepository itemRepository, ILiveTvManager liveTvManager, IEncodingManager encodingManager, IDlnaManager dlnaManager, IChannelManager channelManager, IHttpClient httpClient) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, dtoService, fileSystem, itemRepository, liveTvManager, encodingManager, dlnaManager, channelManager, httpClient)
|
public VideoHlsService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, liveTvManager, dlnaManager, channelManager, subtitleEncoder)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -44,7 +44,7 @@ namespace MediaBrowser.Api.Playback.Progressive
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class AudioService : BaseProgressiveStreamingService
|
public class AudioService : BaseProgressiveStreamingService
|
||||||
{
|
{
|
||||||
public AudioService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IDtoService dtoService, IFileSystem fileSystem, IItemRepository itemRepository, ILiveTvManager liveTvManager, IEncodingManager encodingManager, IDlnaManager dlnaManager, IChannelManager channelManager, IHttpClient httpClient, IImageProcessor imageProcessor) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, dtoService, fileSystem, itemRepository, liveTvManager, encodingManager, dlnaManager, channelManager, httpClient, imageProcessor)
|
public AudioService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder, IImageProcessor imageProcessor, IHttpClient httpClient) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, liveTvManager, dlnaManager, channelManager, subtitleEncoder, imageProcessor, httpClient)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4,11 +4,9 @@ using MediaBrowser.Controller.Channels;
|
||||||
using MediaBrowser.Controller.Configuration;
|
using MediaBrowser.Controller.Configuration;
|
||||||
using MediaBrowser.Controller.Dlna;
|
using MediaBrowser.Controller.Dlna;
|
||||||
using MediaBrowser.Controller.Drawing;
|
using MediaBrowser.Controller.Drawing;
|
||||||
using MediaBrowser.Controller.Dto;
|
|
||||||
using MediaBrowser.Controller.Library;
|
using MediaBrowser.Controller.Library;
|
||||||
using MediaBrowser.Controller.LiveTv;
|
using MediaBrowser.Controller.LiveTv;
|
||||||
using MediaBrowser.Controller.MediaEncoding;
|
using MediaBrowser.Controller.MediaEncoding;
|
||||||
using MediaBrowser.Controller.Persistence;
|
|
||||||
using MediaBrowser.Model.IO;
|
using MediaBrowser.Model.IO;
|
||||||
using ServiceStack.Web;
|
using ServiceStack.Web;
|
||||||
using System;
|
using System;
|
||||||
|
@ -25,10 +23,12 @@ namespace MediaBrowser.Api.Playback.Progressive
|
||||||
public abstract class BaseProgressiveStreamingService : BaseStreamingService
|
public abstract class BaseProgressiveStreamingService : BaseStreamingService
|
||||||
{
|
{
|
||||||
protected readonly IImageProcessor ImageProcessor;
|
protected readonly IImageProcessor ImageProcessor;
|
||||||
|
protected readonly IHttpClient HttpClient;
|
||||||
|
|
||||||
protected BaseProgressiveStreamingService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IDtoService dtoService, IFileSystem fileSystem, IItemRepository itemRepository, ILiveTvManager liveTvManager, IEncodingManager encodingManager, IDlnaManager dlnaManager, IChannelManager channelManager, IHttpClient httpClient, IImageProcessor imageProcessor) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, dtoService, fileSystem, itemRepository, liveTvManager, encodingManager, dlnaManager, channelManager, httpClient)
|
protected BaseProgressiveStreamingService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder, IImageProcessor imageProcessor, IHttpClient httpClient) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, liveTvManager, dlnaManager, channelManager, subtitleEncoder)
|
||||||
{
|
{
|
||||||
ImageProcessor = imageProcessor;
|
ImageProcessor = imageProcessor;
|
||||||
|
HttpClient = httpClient;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|
|
@ -61,7 +61,7 @@ namespace MediaBrowser.Api.Playback.Progressive
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class VideoService : BaseProgressiveStreamingService
|
public class VideoService : BaseProgressiveStreamingService
|
||||||
{
|
{
|
||||||
public VideoService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IDtoService dtoService, IFileSystem fileSystem, IItemRepository itemRepository, ILiveTvManager liveTvManager, IEncodingManager encodingManager, IDlnaManager dlnaManager, IChannelManager channelManager, IHttpClient httpClient, IImageProcessor imageProcessor) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, dtoService, fileSystem, itemRepository, liveTvManager, encodingManager, dlnaManager, channelManager, httpClient, imageProcessor)
|
public VideoService(IServerConfigurationManager serverConfig, IUserManager userManager, ILibraryManager libraryManager, IIsoManager isoManager, IMediaEncoder mediaEncoder, IFileSystem fileSystem, ILiveTvManager liveTvManager, IDlnaManager dlnaManager, IChannelManager channelManager, ISubtitleEncoder subtitleEncoder, IImageProcessor imageProcessor, IHttpClient httpClient) : base(serverConfig, userManager, libraryManager, isoManager, mediaEncoder, fileSystem, liveTvManager, dlnaManager, channelManager, subtitleEncoder, imageProcessor, httpClient)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,6 @@ using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Runtime.Serialization;
|
using System.Runtime.Serialization;
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace MediaBrowser.Controller.Entities.Audio
|
namespace MediaBrowser.Controller.Entities.Audio
|
||||||
{
|
{
|
||||||
|
@ -15,7 +14,6 @@ namespace MediaBrowser.Controller.Entities.Audio
|
||||||
/// Class Audio
|
/// Class Audio
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Audio : BaseItem,
|
public class Audio : BaseItem,
|
||||||
IHasMediaStreams,
|
|
||||||
IHasAlbumArtist,
|
IHasAlbumArtist,
|
||||||
IHasArtist,
|
IHasArtist,
|
||||||
IHasMusicGenres,
|
IHasMusicGenres,
|
||||||
|
|
|
@ -1,10 +0,0 @@
|
||||||
|
|
||||||
namespace MediaBrowser.Controller.Entities
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// This is essentially a marker interface
|
|
||||||
/// </summary>
|
|
||||||
public interface IHasMediaStreams
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -20,7 +20,6 @@ namespace MediaBrowser.Controller.Entities
|
||||||
/// Class Video
|
/// Class Video
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Video : BaseItem,
|
public class Video : BaseItem,
|
||||||
IHasMediaStreams,
|
|
||||||
IHasAspectRatio,
|
IHasAspectRatio,
|
||||||
IHasTags,
|
IHasTags,
|
||||||
ISupportsPlaceHolders,
|
ISupportsPlaceHolders,
|
||||||
|
|
|
@ -6,7 +6,7 @@ using System.Threading.Tasks;
|
||||||
|
|
||||||
namespace MediaBrowser.Controller.LiveTv
|
namespace MediaBrowser.Controller.LiveTv
|
||||||
{
|
{
|
||||||
public interface ILiveTvRecording : IHasImages, IHasMediaStreams
|
public interface ILiveTvRecording : IHasImages, IHasMediaSources
|
||||||
{
|
{
|
||||||
string ServiceName { get; set; }
|
string ServiceName { get; set; }
|
||||||
|
|
||||||
|
|
|
@ -128,7 +128,6 @@
|
||||||
<Compile Include="Entities\IHasImages.cs" />
|
<Compile Include="Entities\IHasImages.cs" />
|
||||||
<Compile Include="Entities\IHasKeywords.cs" />
|
<Compile Include="Entities\IHasKeywords.cs" />
|
||||||
<Compile Include="Entities\IHasMediaSources.cs" />
|
<Compile Include="Entities\IHasMediaSources.cs" />
|
||||||
<Compile Include="Entities\IHasMediaStreams.cs" />
|
|
||||||
<Compile Include="Entities\IHasMetascore.cs" />
|
<Compile Include="Entities\IHasMetascore.cs" />
|
||||||
<Compile Include="Entities\IHasPreferredMetadataLanguage.cs" />
|
<Compile Include="Entities\IHasPreferredMetadataLanguage.cs" />
|
||||||
<Compile Include="Entities\IHasProductionLocations.cs" />
|
<Compile Include="Entities\IHasProductionLocations.cs" />
|
||||||
|
|
|
@ -41,26 +41,6 @@ namespace MediaBrowser.Controller.MediaEncoding
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
/// <returns>Task{Stream}.</returns>
|
/// <returns>Task{Stream}.</returns>
|
||||||
Task<Stream> ExtractVideoImage(string[] inputFiles, InputType type, Video3DFormat? threedFormat, TimeSpan? offset, CancellationToken cancellationToken);
|
Task<Stream> ExtractVideoImage(string[] inputFiles, InputType type, Video3DFormat? threedFormat, TimeSpan? offset, CancellationToken cancellationToken);
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Extracts the text subtitle.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="inputFiles">The input files.</param>
|
|
||||||
/// <param name="type">The type.</param>
|
|
||||||
/// <param name="subtitleStreamIndex">Index of the subtitle stream.</param>
|
|
||||||
/// <param name="copySubtitleStream">if set to true, copy stream instead of converting.</param>
|
|
||||||
/// <param name="outputPath">The output path.</param>
|
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
|
||||||
/// <returns>Task.</returns>
|
|
||||||
Task ExtractTextSubtitle(string[] inputFiles, InputType type, int subtitleStreamIndex, bool copySubtitleStream, string outputPath, CancellationToken cancellationToken);
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets the subtitle language encoding parameter.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="path">The path.</param>
|
|
||||||
/// <param name="language">The language.</param>
|
|
||||||
/// <returns>System.String.</returns>
|
|
||||||
string GetSubtitleLanguageEncodingParam(string path, string language);
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the media info.
|
/// Gets the media info.
|
||||||
|
|
|
@ -6,16 +6,46 @@ namespace MediaBrowser.Controller.MediaEncoding
|
||||||
{
|
{
|
||||||
public interface ISubtitleEncoder
|
public interface ISubtitleEncoder
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Converts the subtitles.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="stream">The stream.</param>
|
||||||
|
/// <param name="inputFormat">The input format.</param>
|
||||||
|
/// <param name="outputFormat">The output format.</param>
|
||||||
|
/// <param name="startTimeTicks">The start time ticks.</param>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>Task{Stream}.</returns>
|
||||||
Task<Stream> ConvertSubtitles(
|
Task<Stream> ConvertSubtitles(
|
||||||
Stream stream,
|
Stream stream,
|
||||||
string inputFormat,
|
string inputFormat,
|
||||||
string outputFormat,
|
string outputFormat,
|
||||||
|
long startTimeTicks,
|
||||||
CancellationToken cancellationToken);
|
CancellationToken cancellationToken);
|
||||||
|
|
||||||
Task<Stream> GetSubtitles(string itemId,
|
/// <summary>
|
||||||
|
/// Gets the subtitles.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="itemId">The item identifier.</param>
|
||||||
|
/// <param name="mediaSourceId">The media source identifier.</param>
|
||||||
|
/// <param name="subtitleStreamIndex">Index of the subtitle stream.</param>
|
||||||
|
/// <param name="outputFormat">The output format.</param>
|
||||||
|
/// <param name="startTimeTicks">The start time ticks.</param>
|
||||||
|
/// <param name="cancellationToken">The cancellation token.</param>
|
||||||
|
/// <returns>Task{Stream}.</returns>
|
||||||
|
Task<Stream> GetSubtitles(string itemId,
|
||||||
string mediaSourceId,
|
string mediaSourceId,
|
||||||
int subtitleStreamIndex,
|
int subtitleStreamIndex,
|
||||||
string outputFormat,
|
string outputFormat,
|
||||||
|
long startTimeTicks,
|
||||||
CancellationToken cancellationToken);
|
CancellationToken cancellationToken);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the subtitle language encoding parameter.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="path">The path.</param>
|
||||||
|
/// <param name="language">The language.</param>
|
||||||
|
/// <returns>System.String.</returns>
|
||||||
|
string GetSubtitleFileCharacterSet(string path, string language);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -285,289 +285,6 @@ namespace MediaBrowser.MediaEncoding.Encoder
|
||||||
return string.Empty;
|
return string.Empty;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets the subtitle language encoding param.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="path">The path.</param>
|
|
||||||
/// <param name="language">The language.</param>
|
|
||||||
/// <returns>System.String.</returns>
|
|
||||||
public string GetSubtitleLanguageEncodingParam(string path, string language)
|
|
||||||
{
|
|
||||||
if (GetFileEncoding(path).Equals(Encoding.UTF8))
|
|
||||||
{
|
|
||||||
return string.Empty;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (language.ToLower())
|
|
||||||
{
|
|
||||||
case "pol":
|
|
||||||
case "cze":
|
|
||||||
case "ces":
|
|
||||||
case "slo":
|
|
||||||
case "slk":
|
|
||||||
case "hun":
|
|
||||||
case "slv":
|
|
||||||
case "srp":
|
|
||||||
case "hrv":
|
|
||||||
case "rum":
|
|
||||||
case "ron":
|
|
||||||
case "rup":
|
|
||||||
case "alb":
|
|
||||||
case "sqi":
|
|
||||||
return "windows-1250";
|
|
||||||
case "ara":
|
|
||||||
return "windows-1256";
|
|
||||||
case "heb":
|
|
||||||
return "windows-1255";
|
|
||||||
case "grc":
|
|
||||||
case "gre":
|
|
||||||
return "windows-1253";
|
|
||||||
case "crh":
|
|
||||||
case "ota":
|
|
||||||
case "tur":
|
|
||||||
return "windows-1254";
|
|
||||||
case "rus":
|
|
||||||
return "windows-1251";
|
|
||||||
case "vie":
|
|
||||||
return "windows-1258";
|
|
||||||
case "kor":
|
|
||||||
return "cp949";
|
|
||||||
default:
|
|
||||||
return "windows-1252";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Encoding GetFileEncoding(string srcFile)
|
|
||||||
{
|
|
||||||
// *** Detect byte order mark if any - otherwise assume default
|
|
||||||
var buffer = new byte[5];
|
|
||||||
|
|
||||||
using (var file = new FileStream(srcFile, FileMode.Open))
|
|
||||||
{
|
|
||||||
file.Read(buffer, 0, 5);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (buffer[0] == 0xef && buffer[1] == 0xbb && buffer[2] == 0xbf)
|
|
||||||
return Encoding.UTF8;
|
|
||||||
if (buffer[0] == 0xfe && buffer[1] == 0xff)
|
|
||||||
return Encoding.Unicode;
|
|
||||||
if (buffer[0] == 0 && buffer[1] == 0 && buffer[2] == 0xfe && buffer[3] == 0xff)
|
|
||||||
return Encoding.UTF32;
|
|
||||||
if (buffer[0] == 0x2b && buffer[1] == 0x2f && buffer[2] == 0x76)
|
|
||||||
return Encoding.UTF7;
|
|
||||||
|
|
||||||
// It's ok - anything aside from utf is ok since that's what we're looking for
|
|
||||||
return Encoding.Default;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Extracts the text subtitle.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="inputFiles">The input files.</param>
|
|
||||||
/// <param name="type">The type.</param>
|
|
||||||
/// <param name="subtitleStreamIndex">Index of the subtitle stream.</param>
|
|
||||||
/// <param name="copySubtitleStream">if set to true, copy stream instead of converting.</param>
|
|
||||||
/// <param name="outputPath">The output path.</param>
|
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
|
||||||
/// <returns>Task.</returns>
|
|
||||||
/// <exception cref="System.ArgumentException">Must use inputPath list overload</exception>
|
|
||||||
public async Task ExtractTextSubtitle(string[] inputFiles, InputType type, int subtitleStreamIndex,
|
|
||||||
bool copySubtitleStream, string outputPath, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
var semaphore = GetLock(outputPath);
|
|
||||||
|
|
||||||
await semaphore.WaitAsync(cancellationToken).ConfigureAwait(false);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (!File.Exists(outputPath))
|
|
||||||
{
|
|
||||||
await
|
|
||||||
ExtractTextSubtitleInternal(GetInputArgument(inputFiles, type), subtitleStreamIndex,
|
|
||||||
copySubtitleStream, outputPath, cancellationToken).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
semaphore.Release();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Extracts the text subtitle.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="inputPath">The input path.</param>
|
|
||||||
/// <param name="subtitleStreamIndex">Index of the subtitle stream.</param>
|
|
||||||
/// <param name="copySubtitleStream">if set to true, copy stream instead of converting.</param>
|
|
||||||
/// <param name="outputPath">The output path.</param>
|
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
|
||||||
/// <returns>Task.</returns>
|
|
||||||
/// <exception cref="System.ArgumentNullException">inputPath
|
|
||||||
/// or
|
|
||||||
/// outputPath
|
|
||||||
/// or
|
|
||||||
/// cancellationToken</exception>
|
|
||||||
/// <exception cref="System.ApplicationException"></exception>
|
|
||||||
private async Task ExtractTextSubtitleInternal(string inputPath, int subtitleStreamIndex,
|
|
||||||
bool copySubtitleStream, string outputPath, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
if (string.IsNullOrEmpty(inputPath))
|
|
||||||
{
|
|
||||||
throw new ArgumentNullException("inputPath");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(outputPath))
|
|
||||||
{
|
|
||||||
throw new ArgumentNullException("outputPath");
|
|
||||||
}
|
|
||||||
|
|
||||||
string processArgs = string.Format("-i {0} -map 0:{1} -an -vn -c:s ass \"{2}\"", inputPath,
|
|
||||||
subtitleStreamIndex, outputPath);
|
|
||||||
|
|
||||||
if (copySubtitleStream)
|
|
||||||
{
|
|
||||||
processArgs = string.Format("-i {0} -map 0:{1} -an -vn -c:s copy \"{2}\"", inputPath,
|
|
||||||
subtitleStreamIndex, outputPath);
|
|
||||||
}
|
|
||||||
|
|
||||||
var process = new Process
|
|
||||||
{
|
|
||||||
StartInfo = new ProcessStartInfo
|
|
||||||
{
|
|
||||||
CreateNoWindow = true,
|
|
||||||
UseShellExecute = false,
|
|
||||||
|
|
||||||
RedirectStandardOutput = false,
|
|
||||||
RedirectStandardError = true,
|
|
||||||
|
|
||||||
FileName = FFMpegPath,
|
|
||||||
Arguments = processArgs,
|
|
||||||
WindowStyle = ProcessWindowStyle.Hidden,
|
|
||||||
ErrorDialog = false
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
_logger.Debug("{0} {1}", process.StartInfo.FileName, process.StartInfo.Arguments);
|
|
||||||
|
|
||||||
var logFilePath = Path.Combine(_appPaths.LogDirectoryPath, "ffmpeg-sub-extract-" + Guid.NewGuid() + ".txt");
|
|
||||||
Directory.CreateDirectory(Path.GetDirectoryName(logFilePath));
|
|
||||||
|
|
||||||
var logFileStream = _fileSystem.GetFileStream(logFilePath, FileMode.Create, FileAccess.Write, FileShare.Read,
|
|
||||||
true);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
process.Start();
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
logFileStream.Dispose();
|
|
||||||
|
|
||||||
_logger.ErrorException("Error starting ffmpeg", ex);
|
|
||||||
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
|
|
||||||
process.StandardError.BaseStream.CopyToAsync(logFileStream);
|
|
||||||
|
|
||||||
var ranToCompletion = process.WaitForExit(60000);
|
|
||||||
|
|
||||||
if (!ranToCompletion)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
_logger.Info("Killing ffmpeg subtitle extraction process");
|
|
||||||
|
|
||||||
process.Kill();
|
|
||||||
|
|
||||||
process.WaitForExit(1000);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
_logger.ErrorException("Error killing subtitle extraction process", ex);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
logFileStream.Dispose();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var exitCode = ranToCompletion ? process.ExitCode : -1;
|
|
||||||
|
|
||||||
process.Dispose();
|
|
||||||
|
|
||||||
var failed = false;
|
|
||||||
|
|
||||||
if (exitCode == -1)
|
|
||||||
{
|
|
||||||
failed = true;
|
|
||||||
|
|
||||||
if (File.Exists(outputPath))
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
_logger.Info("Deleting extracted subtitle due to failure: ", outputPath);
|
|
||||||
File.Delete(outputPath);
|
|
||||||
}
|
|
||||||
catch (IOException ex)
|
|
||||||
{
|
|
||||||
_logger.ErrorException("Error deleting extracted subtitle {0}", ex, outputPath);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (!File.Exists(outputPath))
|
|
||||||
{
|
|
||||||
failed = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (failed)
|
|
||||||
{
|
|
||||||
var msg = string.Format("ffmpeg subtitle extraction failed for {0} to {1}", inputPath, outputPath);
|
|
||||||
|
|
||||||
_logger.Error(msg);
|
|
||||||
|
|
||||||
throw new ApplicationException(msg);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var msg = string.Format("ffmpeg subtitle extraction completed for {0} to {1}", inputPath, outputPath);
|
|
||||||
|
|
||||||
_logger.Info(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
await SetAssFont(outputPath).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Sets the ass font.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="file">The file.</param>
|
|
||||||
/// <returns>Task.</returns>
|
|
||||||
private async Task SetAssFont(string file)
|
|
||||||
{
|
|
||||||
_logger.Info("Setting ass font within {0}", file);
|
|
||||||
|
|
||||||
string text;
|
|
||||||
Encoding encoding;
|
|
||||||
|
|
||||||
using (var reader = new StreamReader(file, detectEncodingFromByteOrderMarks: true))
|
|
||||||
{
|
|
||||||
encoding = reader.CurrentEncoding;
|
|
||||||
|
|
||||||
text = await reader.ReadToEndAsync().ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
var newText = text.Replace(",Arial,", ",Arial Unicode MS,");
|
|
||||||
|
|
||||||
if (!string.Equals(text, newText))
|
|
||||||
{
|
|
||||||
using (var writer = new StreamWriter(file, false, encoding))
|
|
||||||
{
|
|
||||||
writer.Write(newText);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Task<Stream> ExtractAudioImage(string path, CancellationToken cancellationToken)
|
public Task<Stream> ExtractAudioImage(string path, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
return ExtractImage(new[] { path }, InputType.File, true, null, null, cancellationToken);
|
return ExtractImage(new[] { path }, InputType.File, true, null, null, cancellationToken);
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
|
using System.Text;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
|
||||||
|
|
|
@ -47,13 +47,16 @@ namespace MediaBrowser.MediaEncoding.Subtitles
|
||||||
public async Task<Stream> ConvertSubtitles(Stream stream,
|
public async Task<Stream> ConvertSubtitles(Stream stream,
|
||||||
string inputFormat,
|
string inputFormat,
|
||||||
string outputFormat,
|
string outputFormat,
|
||||||
|
long startTimeTicks,
|
||||||
CancellationToken cancellationToken)
|
CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var ms = new MemoryStream();
|
var ms = new MemoryStream();
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (string.Equals(inputFormat, outputFormat, StringComparison.OrdinalIgnoreCase))
|
// Return the original without any conversions, if possible
|
||||||
|
if (startTimeTicks == 0 &&
|
||||||
|
string.Equals(inputFormat, outputFormat, StringComparison.OrdinalIgnoreCase))
|
||||||
{
|
{
|
||||||
await stream.CopyToAsync(ms, 81920, cancellationToken).ConfigureAwait(false);
|
await stream.CopyToAsync(ms, 81920, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
|
@ -61,6 +64,8 @@ namespace MediaBrowser.MediaEncoding.Subtitles
|
||||||
{
|
{
|
||||||
var trackInfo = await GetTrackInfo(stream, inputFormat, cancellationToken).ConfigureAwait(false);
|
var trackInfo = await GetTrackInfo(stream, inputFormat, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
UpdateStartingPosition(trackInfo, startTimeTicks);
|
||||||
|
|
||||||
var writer = GetWriter(outputFormat);
|
var writer = GetWriter(outputFormat);
|
||||||
|
|
||||||
writer.Write(trackInfo, ms, cancellationToken);
|
writer.Write(trackInfo, ms, cancellationToken);
|
||||||
|
@ -76,10 +81,26 @@ namespace MediaBrowser.MediaEncoding.Subtitles
|
||||||
return ms;
|
return ms;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void UpdateStartingPosition(SubtitleTrackInfo track, long startPositionTicks)
|
||||||
|
{
|
||||||
|
if (startPositionTicks == 0) return;
|
||||||
|
|
||||||
|
foreach (var trackEvent in track.TrackEvents)
|
||||||
|
{
|
||||||
|
trackEvent.EndPositionTicks -= startPositionTicks;
|
||||||
|
trackEvent.StartPositionTicks -= startPositionTicks;
|
||||||
|
}
|
||||||
|
|
||||||
|
track.TrackEvents = track.TrackEvents
|
||||||
|
.SkipWhile(i => i.StartPositionTicks < 0 || i.EndPositionTicks < 0)
|
||||||
|
.ToList();
|
||||||
|
}
|
||||||
|
|
||||||
public async Task<Stream> GetSubtitles(string itemId,
|
public async Task<Stream> GetSubtitles(string itemId,
|
||||||
string mediaSourceId,
|
string mediaSourceId,
|
||||||
int subtitleStreamIndex,
|
int subtitleStreamIndex,
|
||||||
string outputFormat,
|
string outputFormat,
|
||||||
|
long startTimeTicks,
|
||||||
CancellationToken cancellationToken)
|
CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var subtitle = await GetSubtitleStream(itemId, mediaSourceId, subtitleStreamIndex, cancellationToken)
|
var subtitle = await GetSubtitleStream(itemId, mediaSourceId, subtitleStreamIndex, cancellationToken)
|
||||||
|
@ -89,7 +110,7 @@ namespace MediaBrowser.MediaEncoding.Subtitles
|
||||||
{
|
{
|
||||||
var inputFormat = subtitle.Item2;
|
var inputFormat = subtitle.Item2;
|
||||||
|
|
||||||
return await ConvertSubtitles(stream, inputFormat, outputFormat, cancellationToken).ConfigureAwait(false);
|
return await ConvertSubtitles(stream, inputFormat, outputFormat, startTimeTicks, cancellationToken).ConfigureAwait(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -127,11 +148,36 @@ namespace MediaBrowser.MediaEncoding.Subtitles
|
||||||
|
|
||||||
var fileInfo = await GetReadableFile(mediaSource.Path, inputFiles, inputType, subtitleStream, cancellationToken).ConfigureAwait(false);
|
var fileInfo = await GetReadableFile(mediaSource.Path, inputFiles, inputType, subtitleStream, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
var stream = File.OpenRead(fileInfo.Item1);
|
var stream = await GetSubtitleStream(fileInfo.Item1, subtitleStream.Language).ConfigureAwait(false);
|
||||||
|
|
||||||
return new Tuple<Stream, string>(stream, fileInfo.Item2);
|
return new Tuple<Stream, string>(stream, fileInfo.Item2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task<Stream> GetSubtitleStream(string path, string language)
|
||||||
|
{
|
||||||
|
if (!string.IsNullOrEmpty(language))
|
||||||
|
{
|
||||||
|
var charset = GetSubtitleFileCharacterSet(path, language);
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(charset))
|
||||||
|
{
|
||||||
|
using (var fs = _fileSystem.GetFileStream(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite, true))
|
||||||
|
{
|
||||||
|
using (var reader = new StreamReader(fs, Encoding.GetEncoding(charset)))
|
||||||
|
{
|
||||||
|
var text = await reader.ReadToEndAsync().ConfigureAwait(false);
|
||||||
|
|
||||||
|
var bytes = Encoding.UTF8.GetBytes(text);
|
||||||
|
|
||||||
|
return new MemoryStream(bytes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return File.OpenRead(path);
|
||||||
|
}
|
||||||
|
|
||||||
private async Task<Tuple<string, string>> GetReadableFile(string mediaPath,
|
private async Task<Tuple<string, string>> GetReadableFile(string mediaPath,
|
||||||
string[] inputFiles,
|
string[] inputFiles,
|
||||||
InputType type,
|
InputType type,
|
||||||
|
@ -282,10 +328,11 @@ namespace MediaBrowser.MediaEncoding.Subtitles
|
||||||
throw new ArgumentNullException("outputPath");
|
throw new ArgumentNullException("outputPath");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Directory.CreateDirectory(Path.GetDirectoryName(outputPath));
|
||||||
|
|
||||||
var encodingParam = string.IsNullOrEmpty(language)
|
var encodingParam = string.IsNullOrEmpty(language)
|
||||||
? string.Empty
|
? string.Empty
|
||||||
: _mediaEncoder.GetSubtitleLanguageEncodingParam(inputPath, language);
|
: GetSubtitleFileCharacterSet(inputPath, language);
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(encodingParam))
|
if (!string.IsNullOrEmpty(encodingParam))
|
||||||
{
|
{
|
||||||
|
@ -456,7 +503,9 @@ namespace MediaBrowser.MediaEncoding.Subtitles
|
||||||
throw new ArgumentNullException("outputPath");
|
throw new ArgumentNullException("outputPath");
|
||||||
}
|
}
|
||||||
|
|
||||||
string processArgs = string.Format("-i {0} -map 0:{1} -an -vn -c:s ass \"{2}\"", inputPath,
|
Directory.CreateDirectory(Path.GetDirectoryName(outputPath));
|
||||||
|
|
||||||
|
var processArgs = string.Format("-i {0} -map 0:{1} -an -vn -c:s ass \"{2}\"", inputPath,
|
||||||
subtitleStreamIndex, outputPath);
|
subtitleStreamIndex, outputPath);
|
||||||
|
|
||||||
if (copySubtitleStream)
|
if (copySubtitleStream)
|
||||||
|
@ -615,5 +664,80 @@ namespace MediaBrowser.MediaEncoding.Subtitles
|
||||||
|
|
||||||
return Path.Combine(SubtitleCachePath, prefix, filename);
|
return Path.Combine(SubtitleCachePath, prefix, filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the subtitle language encoding param.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="path">The path.</param>
|
||||||
|
/// <param name="language">The language.</param>
|
||||||
|
/// <returns>System.String.</returns>
|
||||||
|
public string GetSubtitleFileCharacterSet(string path, string language)
|
||||||
|
{
|
||||||
|
if (GetFileEncoding(path).Equals(Encoding.UTF8))
|
||||||
|
{
|
||||||
|
return string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (language.ToLower())
|
||||||
|
{
|
||||||
|
case "pol":
|
||||||
|
case "cze":
|
||||||
|
case "ces":
|
||||||
|
case "slo":
|
||||||
|
case "slk":
|
||||||
|
case "hun":
|
||||||
|
case "slv":
|
||||||
|
case "srp":
|
||||||
|
case "hrv":
|
||||||
|
case "rum":
|
||||||
|
case "ron":
|
||||||
|
case "rup":
|
||||||
|
case "alb":
|
||||||
|
case "sqi":
|
||||||
|
return "windows-1250";
|
||||||
|
case "ara":
|
||||||
|
return "windows-1256";
|
||||||
|
case "heb":
|
||||||
|
return "windows-1255";
|
||||||
|
case "grc":
|
||||||
|
case "gre":
|
||||||
|
return "windows-1253";
|
||||||
|
case "crh":
|
||||||
|
case "ota":
|
||||||
|
case "tur":
|
||||||
|
return "windows-1254";
|
||||||
|
case "rus":
|
||||||
|
return "windows-1251";
|
||||||
|
case "vie":
|
||||||
|
return "windows-1258";
|
||||||
|
case "kor":
|
||||||
|
return "cp949";
|
||||||
|
default:
|
||||||
|
return "windows-1252";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Encoding GetFileEncoding(string srcFile)
|
||||||
|
{
|
||||||
|
// *** Detect byte order mark if any - otherwise assume default
|
||||||
|
var buffer = new byte[5];
|
||||||
|
|
||||||
|
using (var file = new FileStream(srcFile, FileMode.Open))
|
||||||
|
{
|
||||||
|
file.Read(buffer, 0, 5);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (buffer[0] == 0xef && buffer[1] == 0xbb && buffer[2] == 0xbf)
|
||||||
|
return Encoding.UTF8;
|
||||||
|
if (buffer[0] == 0xfe && buffer[1] == 0xff)
|
||||||
|
return Encoding.Unicode;
|
||||||
|
if (buffer[0] == 0 && buffer[1] == 0 && buffer[2] == 0xfe && buffer[3] == 0xff)
|
||||||
|
return Encoding.UTF32;
|
||||||
|
if (buffer[0] == 0x2b && buffer[1] == 0x2f && buffer[2] == 0x76)
|
||||||
|
return Encoding.UTF7;
|
||||||
|
|
||||||
|
// It's ok - anything aside from utf is ok since that's what we're looking for
|
||||||
|
return Encoding.Default;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -130,7 +130,6 @@ namespace MediaBrowser.Model.Entities
|
||||||
/// <value><c>true</c> if this instance is external; otherwise, <c>false</c>.</value>
|
/// <value><c>true</c> if this instance is external; otherwise, <c>false</c>.</value>
|
||||||
public bool IsExternal { get; set; }
|
public bool IsExternal { get; set; }
|
||||||
|
|
||||||
[IgnoreDataMember]
|
|
||||||
public bool IsGraphicalSubtitleStream
|
public bool IsGraphicalSubtitleStream
|
||||||
{
|
{
|
||||||
get
|
get
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
using System.Threading;
|
using MediaBrowser.Common.Extensions;
|
||||||
using MediaBrowser.Common.Extensions;
|
|
||||||
using MediaBrowser.Common.IO;
|
using MediaBrowser.Common.IO;
|
||||||
using MediaBrowser.Controller.Channels;
|
using MediaBrowser.Controller.Channels;
|
||||||
using MediaBrowser.Controller.Configuration;
|
using MediaBrowser.Controller.Configuration;
|
||||||
|
@ -30,7 +29,6 @@ namespace MediaBrowser.Server.Implementations.Dto
|
||||||
{
|
{
|
||||||
private readonly ILogger _logger;
|
private readonly ILogger _logger;
|
||||||
private readonly ILibraryManager _libraryManager;
|
private readonly ILibraryManager _libraryManager;
|
||||||
private readonly IUserManager _userManager;
|
|
||||||
private readonly IUserDataManager _userDataRepository;
|
private readonly IUserDataManager _userDataRepository;
|
||||||
private readonly IItemRepository _itemRepo;
|
private readonly IItemRepository _itemRepo;
|
||||||
|
|
||||||
|
@ -41,11 +39,10 @@ namespace MediaBrowser.Server.Implementations.Dto
|
||||||
|
|
||||||
private readonly Func<IChannelManager> _channelManagerFactory;
|
private readonly Func<IChannelManager> _channelManagerFactory;
|
||||||
|
|
||||||
public DtoService(ILogger logger, ILibraryManager libraryManager, IUserManager userManager, IUserDataManager userDataRepository, IItemRepository itemRepo, IImageProcessor imageProcessor, IServerConfigurationManager config, IFileSystem fileSystem, IProviderManager providerManager, Func<IChannelManager> channelManagerFactory)
|
public DtoService(ILogger logger, ILibraryManager libraryManager, IUserDataManager userDataRepository, IItemRepository itemRepo, IImageProcessor imageProcessor, IServerConfigurationManager config, IFileSystem fileSystem, IProviderManager providerManager, Func<IChannelManager> channelManagerFactory)
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_libraryManager = libraryManager;
|
_libraryManager = libraryManager;
|
||||||
_userManager = userManager;
|
|
||||||
_userDataRepository = userDataRepository;
|
_userDataRepository = userDataRepository;
|
||||||
_itemRepo = itemRepo;
|
_itemRepo = itemRepo;
|
||||||
_imageProcessor = imageProcessor;
|
_imageProcessor = imageProcessor;
|
||||||
|
@ -993,9 +990,9 @@ namespace MediaBrowser.Server.Implementations.Dto
|
||||||
if (fields.Contains(ItemFields.MediaStreams))
|
if (fields.Contains(ItemFields.MediaStreams))
|
||||||
{
|
{
|
||||||
// Add VideoInfo
|
// Add VideoInfo
|
||||||
var iHasMediaStreams = item as IHasMediaStreams;
|
var iHasMediaSources = item as IHasMediaSources;
|
||||||
|
|
||||||
if (iHasMediaStreams != null)
|
if (iHasMediaSources != null)
|
||||||
{
|
{
|
||||||
List<MediaStream> mediaStreams;
|
List<MediaStream> mediaStreams;
|
||||||
|
|
||||||
|
@ -1007,11 +1004,7 @@ namespace MediaBrowser.Server.Implementations.Dto
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
mediaStreams = _itemRepo.GetMediaStreams(new MediaStreamQuery
|
mediaStreams = iHasMediaSources.GetMediaSources(true).First().MediaStreams;
|
||||||
{
|
|
||||||
ItemId = item.Id
|
|
||||||
|
|
||||||
}).ToList();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
dto.MediaStreams = mediaStreams;
|
dto.MediaStreams = mediaStreams;
|
||||||
|
|
|
@ -498,7 +498,7 @@ namespace MediaBrowser.ServerApplication
|
||||||
ImageProcessor = new ImageProcessor(LogManager.GetLogger("ImageProcessor"), ServerConfigurationManager.ApplicationPaths, FileSystemManager, JsonSerializer, MediaEncoder);
|
ImageProcessor = new ImageProcessor(LogManager.GetLogger("ImageProcessor"), ServerConfigurationManager.ApplicationPaths, FileSystemManager, JsonSerializer, MediaEncoder);
|
||||||
RegisterSingleInstance(ImageProcessor);
|
RegisterSingleInstance(ImageProcessor);
|
||||||
|
|
||||||
DtoService = new DtoService(Logger, LibraryManager, UserManager, UserDataManager, ItemRepository, ImageProcessor, ServerConfigurationManager, FileSystemManager, ProviderManager, () => ChannelManager);
|
DtoService = new DtoService(Logger, LibraryManager, UserDataManager, ItemRepository, ImageProcessor, ServerConfigurationManager, FileSystemManager, ProviderManager, () => ChannelManager);
|
||||||
RegisterSingleInstance(DtoService);
|
RegisterSingleInstance(DtoService);
|
||||||
|
|
||||||
SessionManager = new SessionManager(UserDataManager, ServerConfigurationManager, Logger, UserRepository, LibraryManager, UserManager, musicManager, DtoService, ImageProcessor, ItemRepository, JsonSerializer, this, HttpClient);
|
SessionManager = new SessionManager(UserDataManager, ServerConfigurationManager, Logger, UserRepository, LibraryManager, UserManager, musicManager, DtoService, ImageProcessor, ItemRepository, JsonSerializer, this, HttpClient);
|
||||||
|
|
|
@ -46,18 +46,12 @@ namespace MediaBrowser.ServerApplication
|
||||||
|
|
||||||
var json = FormatJson(_jsonSerializer.SerializeToString(item));
|
var json = FormatJson(_jsonSerializer.SerializeToString(item));
|
||||||
|
|
||||||
if (item is IHasMediaStreams)
|
var hasMediaSources = item as IHasMediaSources;
|
||||||
|
if (hasMediaSources != null)
|
||||||
{
|
{
|
||||||
var mediaStreams = _itemRepository.GetMediaStreams(new MediaStreamQuery
|
var sources = hasMediaSources.GetMediaSources(false).ToList();
|
||||||
{
|
|
||||||
ItemId = item.Id
|
|
||||||
|
|
||||||
}).ToList();
|
json += "\n\nMedia Sources:\n\n" + FormatJson(_jsonSerializer.SerializeToString(sources));
|
||||||
|
|
||||||
if (mediaStreams.Count > 0)
|
|
||||||
{
|
|
||||||
json += "\n\nMedia Streams:\n\n" + FormatJson(_jsonSerializer.SerializeToString(mediaStreams));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
txtJson.Text = json;
|
txtJson.Text = json;
|
||||||
|
|
Loading…
Reference in New Issue
Block a user