mirror of
				https://github.com/jellyfin/jellyfin.git
				synced 2025-10-26 00:02:44 -04:00 
			
		
		
		
	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" /> | ||||||
|  | |||||||
| @ -42,26 +42,6 @@ namespace MediaBrowser.Controller.MediaEncoding | |||||||
|         /// <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. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | |||||||
| @ -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); | ||||||
| 
 | 
 | ||||||
|  |         /// <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, |         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…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user