jellyfin/MediaBrowser.Model/Dlna/StreamInfo.cs

816 lines
28 KiB
C#
Raw Normal View History

2014-10-12 03:46:02 +02:00
using MediaBrowser.Model.Drawing;
2014-04-23 04:47:46 +02:00
using MediaBrowser.Model.Dto;
2014-04-18 19:16:25 +02:00
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Extensions;
2014-05-08 23:23:24 +02:00
using MediaBrowser.Model.MediaInfo;
using MediaBrowser.Model.Session;
2014-04-06 19:53:23 +02:00
using System;
2014-04-02 00:23:07 +02:00
using System.Collections.Generic;
namespace MediaBrowser.Model.Dlna
{
/// <summary>
/// Class StreamInfo.
/// </summary>
public class StreamInfo
{
2016-06-26 18:21:10 +02:00
public StreamInfo()
{
AudioCodecs = new string[] { };
}
2014-04-02 00:23:07 +02:00
public string ItemId { get; set; }
public PlayMethod PlayMethod { get; set; }
2015-01-02 06:36:27 +01:00
public EncodingContext Context { get; set; }
2014-04-02 00:23:07 +02:00
public DlnaProfileType MediaType { get; set; }
public string Container { get; set; }
public string SubProtocol { get; set; }
2014-04-02 00:23:07 +02:00
public long StartPositionTicks { get; set; }
public string VideoCodec { get; set; }
2014-04-24 07:08:10 +02:00
public string VideoProfile { get; set; }
2014-04-02 00:23:07 +02:00
public bool CopyTimestamps { get; set; }
2016-07-13 21:16:51 +02:00
public bool EnableSubtitlesInManifest { get; set; }
2016-10-16 19:11:32 +02:00
public bool EnableSplittingOnNonKeyFrames { get; set; }
2016-06-26 18:21:10 +02:00
public string[] AudioCodecs { get; set; }
2014-04-02 00:23:07 +02:00
public int? AudioStreamIndex { get; set; }
public int? SubtitleStreamIndex { get; set; }
2016-05-14 07:40:01 +02:00
public int? TranscodingMaxAudioChannels { get; set; }
2014-04-02 00:23:07 +02:00
public int? MaxAudioChannels { get; set; }
public int? AudioBitrate { get; set; }
public int? VideoBitrate { get; set; }
public int? VideoLevel { get; set; }
public int? MaxWidth { get; set; }
public int? MaxHeight { get; set; }
2014-09-23 06:05:29 +02:00
public int? MaxVideoBitDepth { get; set; }
public int? MaxRefFrames { get; set; }
2015-03-30 21:57:37 +02:00
2014-06-23 18:05:19 +02:00
public float? MaxFramerate { get; set; }
2014-04-02 00:23:07 +02:00
2015-02-05 06:29:37 +01:00
public DeviceProfile DeviceProfile { get; set; }
2014-04-02 00:23:07 +02:00
public string DeviceProfileId { get; set; }
public string DeviceId { get; set; }
2014-04-18 07:03:01 +02:00
public long? RunTimeTicks { get; set; }
public TranscodeSeekInfo TranscodeSeekInfo { get; set; }
2014-04-18 19:16:25 +02:00
public bool EstimateContentLength { get; set; }
public MediaSourceInfo MediaSource { get; set; }
public SubtitleDeliveryMethod SubtitleDeliveryMethod { get; set; }
public string SubtitleFormat { get; set; }
2015-04-01 23:55:50 +02:00
public string PlaySessionId { get; set; }
2015-04-04 21:35:29 +02:00
public List<MediaSourceInfo> AllMediaSources { get; set; }
2015-03-23 18:19:21 +01:00
2014-04-18 19:16:25 +02:00
public string MediaSourceId
{
get
{
return MediaSource == null ? null : MediaSource.Id;
}
}
public bool IsDirectStream
{
2015-03-30 21:57:37 +02:00
get
{
2014-10-11 22:38:13 +02:00
return PlayMethod == PlayMethod.DirectStream ||
PlayMethod == PlayMethod.DirectPlay;
}
}
public string ToUrl(string baseUrl, string accessToken)
2014-04-02 00:23:07 +02:00
{
2015-03-26 19:24:13 +01:00
if (PlayMethod == PlayMethod.DirectPlay)
{
return MediaSource.Path;
}
if (string.IsNullOrEmpty(baseUrl))
{
throw new ArgumentNullException(baseUrl);
}
List<string> list = new List<string>();
2015-04-16 16:59:39 +02:00
foreach (NameValuePair pair in BuildParams(this, accessToken, false))
2015-03-26 19:24:13 +01:00
{
2015-03-26 20:18:21 +01:00
if (string.IsNullOrEmpty(pair.Value))
2015-03-26 19:24:13 +01:00
{
2015-03-26 20:18:21 +01:00
continue;
2015-03-26 19:24:13 +01:00
}
2015-03-26 20:18:21 +01:00
// Try to keep the url clean by omitting defaults
if (StringHelper.EqualsIgnoreCase(pair.Name, "StartTimeTicks") &&
StringHelper.EqualsIgnoreCase(pair.Value, "0"))
{
continue;
}
if (StringHelper.EqualsIgnoreCase(pair.Name, "SubtitleStreamIndex") &&
StringHelper.EqualsIgnoreCase(pair.Value, "-1"))
{
continue;
}
if (StringHelper.EqualsIgnoreCase(pair.Name, "Static") &&
StringHelper.EqualsIgnoreCase(pair.Value, "false"))
{
continue;
}
list.Add(string.Format("{0}={1}", pair.Name, pair.Value));
2015-03-26 19:24:13 +01:00
}
string queryString = string.Join("&", list.ToArray());
return GetUrl(baseUrl, queryString);
2014-04-02 00:23:07 +02:00
}
public string ToDlnaUrl(string baseUrl, string accessToken)
2014-04-02 00:23:07 +02:00
{
2014-10-12 03:46:02 +02:00
if (PlayMethod == PlayMethod.DirectPlay)
{
return MediaSource.Path;
}
2015-03-26 19:24:13 +01:00
string dlnaCommand = BuildDlnaParam(this, accessToken);
return GetUrl(baseUrl, dlnaCommand);
}
private string GetUrl(string baseUrl, string queryString)
{
2014-04-02 00:23:07 +02:00
if (string.IsNullOrEmpty(baseUrl))
{
throw new ArgumentNullException(baseUrl);
}
2014-05-08 22:09:53 +02:00
string extension = string.IsNullOrEmpty(Container) ? string.Empty : "." + Container;
2014-04-02 00:23:07 +02:00
baseUrl = baseUrl.TrimEnd('/');
if (MediaType == DlnaProfileType.Audio)
{
2015-05-24 20:33:28 +02:00
if (StringHelper.EqualsIgnoreCase(SubProtocol, "hls"))
{
return string.Format("{0}/audio/{1}/master.m3u8?{2}", baseUrl, ItemId, queryString);
}
2015-03-26 19:24:13 +01:00
return string.Format("{0}/audio/{1}/stream{2}?{3}", baseUrl, ItemId, extension, queryString);
2014-04-02 00:23:07 +02:00
}
if (StringHelper.EqualsIgnoreCase(SubProtocol, "hls"))
{
2015-03-26 19:24:13 +01:00
return string.Format("{0}/videos/{1}/master.m3u8?{2}", baseUrl, ItemId, queryString);
}
2015-03-26 19:24:13 +01:00
return string.Format("{0}/videos/{1}/stream{2}?{3}", baseUrl, ItemId, extension, queryString);
2014-04-02 00:23:07 +02:00
}
2015-03-26 19:24:13 +01:00
private static string BuildDlnaParam(StreamInfo item, string accessToken)
2014-04-02 00:23:07 +02:00
{
2015-03-26 19:24:13 +01:00
List<string> list = new List<string>();
2014-09-23 06:05:29 +02:00
2015-04-16 16:59:39 +02:00
foreach (NameValuePair pair in BuildParams(item, accessToken, true))
2015-03-26 19:24:13 +01:00
{
list.Add(pair.Value);
}
2015-03-30 21:57:37 +02:00
2014-04-02 00:23:07 +02:00
return string.Format("Params={0}", string.Join(";", list.ToArray()));
}
2015-04-16 16:59:39 +02:00
private static List<NameValuePair> BuildParams(StreamInfo item, string accessToken, bool isDlna)
2015-03-26 19:24:13 +01:00
{
List<NameValuePair> list = new List<NameValuePair>();
2016-06-26 18:21:10 +02:00
string audioCodecs = item.AudioCodecs.Length == 0 ?
string.Empty :
string.Join(",", item.AudioCodecs);
2015-03-26 19:24:13 +01:00
list.Add(new NameValuePair("DeviceProfileId", item.DeviceProfileId ?? string.Empty));
list.Add(new NameValuePair("DeviceId", item.DeviceId ?? string.Empty));
list.Add(new NameValuePair("MediaSourceId", item.MediaSourceId ?? string.Empty));
2016-03-27 23:11:27 +02:00
list.Add(new NameValuePair("Static", item.IsDirectStream.ToString().ToLower()));
2015-03-26 19:24:13 +01:00
list.Add(new NameValuePair("VideoCodec", item.VideoCodec ?? string.Empty));
2016-06-26 18:21:10 +02:00
list.Add(new NameValuePair("AudioCodec", audioCodecs));
2015-03-26 19:24:13 +01:00
list.Add(new NameValuePair("AudioStreamIndex", item.AudioStreamIndex.HasValue ? StringHelper.ToStringCultureInvariant(item.AudioStreamIndex.Value) : string.Empty));
list.Add(new NameValuePair("SubtitleStreamIndex", item.SubtitleStreamIndex.HasValue && item.SubtitleDeliveryMethod != SubtitleDeliveryMethod.External ? StringHelper.ToStringCultureInvariant(item.SubtitleStreamIndex.Value) : string.Empty));
list.Add(new NameValuePair("VideoBitrate", item.VideoBitrate.HasValue ? StringHelper.ToStringCultureInvariant(item.VideoBitrate.Value) : string.Empty));
list.Add(new NameValuePair("AudioBitrate", item.AudioBitrate.HasValue ? StringHelper.ToStringCultureInvariant(item.AudioBitrate.Value) : string.Empty));
list.Add(new NameValuePair("MaxAudioChannels", item.MaxAudioChannels.HasValue ? StringHelper.ToStringCultureInvariant(item.MaxAudioChannels.Value) : string.Empty));
list.Add(new NameValuePair("MaxFramerate", item.MaxFramerate.HasValue ? StringHelper.ToStringCultureInvariant(item.MaxFramerate.Value) : string.Empty));
list.Add(new NameValuePair("MaxWidth", item.MaxWidth.HasValue ? StringHelper.ToStringCultureInvariant(item.MaxWidth.Value) : string.Empty));
list.Add(new NameValuePair("MaxHeight", item.MaxHeight.HasValue ? StringHelper.ToStringCultureInvariant(item.MaxHeight.Value) : string.Empty));
2015-04-03 04:35:15 +02:00
2016-09-30 08:50:06 +02:00
var forceStartPosition = false;
long startPositionTicks = item.StartPositionTicks;
//if (item.MediaSource.DateLiveStreamOpened.HasValue && startPositionTicks == 0)
//{
// var elapsed = DateTime.UtcNow - item.MediaSource.DateLiveStreamOpened.Value;
// elapsed -= TimeSpan.FromSeconds(20);
// if (elapsed.TotalSeconds >= 0)
// {
// startPositionTicks = elapsed.Ticks + startPositionTicks;
// forceStartPosition = true;
// }
//}
if (StringHelper.EqualsIgnoreCase(item.SubProtocol, "hls") && !forceStartPosition)
2015-04-03 04:35:15 +02:00
{
list.Add(new NameValuePair("StartTimeTicks", string.Empty));
}
else
{
2016-09-30 08:50:06 +02:00
list.Add(new NameValuePair("StartTimeTicks", StringHelper.ToStringCultureInvariant(startPositionTicks)));
2015-04-03 04:35:15 +02:00
}
2015-03-26 19:24:13 +01:00
list.Add(new NameValuePair("Level", item.VideoLevel.HasValue ? StringHelper.ToStringCultureInvariant(item.VideoLevel.Value) : string.Empty));
2015-03-30 21:57:37 +02:00
2015-03-26 19:24:13 +01:00
list.Add(new NameValuePair("MaxRefFrames", item.MaxRefFrames.HasValue ? StringHelper.ToStringCultureInvariant(item.MaxRefFrames.Value) : string.Empty));
list.Add(new NameValuePair("MaxVideoBitDepth", item.MaxVideoBitDepth.HasValue ? StringHelper.ToStringCultureInvariant(item.MaxVideoBitDepth.Value) : string.Empty));
list.Add(new NameValuePair("Profile", item.VideoProfile ?? string.Empty));
2016-04-04 02:01:03 +02:00
// no longer used
list.Add(new NameValuePair("Cabac", string.Empty));
2015-03-26 19:24:13 +01:00
2015-04-01 23:55:50 +02:00
list.Add(new NameValuePair("PlaySessionId", item.PlaySessionId ?? string.Empty));
2015-03-26 19:24:13 +01:00
list.Add(new NameValuePair("api_key", accessToken ?? string.Empty));
2015-03-29 20:16:40 +02:00
string liveStreamId = item.MediaSource == null ? null : item.MediaSource.LiveStreamId;
list.Add(new NameValuePair("LiveStreamId", liveStreamId ?? string.Empty));
2015-04-24 22:06:37 +02:00
if (isDlna)
{
list.Add(new NameValuePair("ItemId", item.ItemId));
}
2016-03-27 23:11:27 +02:00
list.Add(new NameValuePair("CopyTimestamps", item.CopyTimestamps.ToString().ToLower()));
2016-03-07 05:56:45 +01:00
list.Add(new NameValuePair("SubtitleMethod", item.SubtitleStreamIndex.HasValue && item.SubtitleDeliveryMethod != SubtitleDeliveryMethod.External ? item.SubtitleDeliveryMethod.ToString() : string.Empty));
2016-05-14 07:40:01 +02:00
list.Add(new NameValuePair("TranscodingMaxAudioChannels", item.TranscodingMaxAudioChannels.HasValue ? StringHelper.ToStringCultureInvariant(item.TranscodingMaxAudioChannels.Value) : string.Empty));
2016-07-13 21:16:51 +02:00
list.Add(new NameValuePair("EnableSubtitlesInManifest", item.EnableSubtitlesInManifest.ToString().ToLower()));
2016-05-14 07:40:01 +02:00
list.Add(new NameValuePair("Tag", item.MediaSource.ETag ?? string.Empty));
2016-10-16 19:11:32 +02:00
list.Add(new NameValuePair("EnableSplittingOnNonKeyFrames", item.EnableSplittingOnNonKeyFrames.ToString().ToLower()));
2015-03-26 19:24:13 +01:00
return list;
}
2015-04-01 23:55:50 +02:00
public List<SubtitleStreamInfo> GetExternalSubtitles(bool includeSelectedTrackOnly, string baseUrl, string accessToken)
{
return GetExternalSubtitles(includeSelectedTrackOnly, false, baseUrl, accessToken);
}
2015-03-31 18:24:16 +02:00
public List<SubtitleStreamInfo> GetExternalSubtitles(bool includeSelectedTrackOnly, bool enableAllProfiles, string baseUrl, string accessToken)
{
2015-03-31 18:24:16 +02:00
List<SubtitleStreamInfo> list = GetSubtitleProfiles(includeSelectedTrackOnly, enableAllProfiles, baseUrl, accessToken);
2015-03-30 21:57:37 +02:00
List<SubtitleStreamInfo> newList = new List<SubtitleStreamInfo>();
// First add the selected track
2015-03-30 21:57:37 +02:00
foreach (SubtitleStreamInfo stream in list)
{
2015-03-30 21:57:37 +02:00
if (stream.DeliveryMethod == SubtitleDeliveryMethod.External)
{
2015-03-30 21:57:37 +02:00
newList.Add(stream);
}
}
2015-03-30 21:57:37 +02:00
return newList;
}
2015-03-30 21:57:37 +02:00
public List<SubtitleStreamInfo> GetSubtitleProfiles(bool includeSelectedTrackOnly, string baseUrl, string accessToken)
2015-03-31 18:24:16 +02:00
{
return GetSubtitleProfiles(includeSelectedTrackOnly, false, baseUrl, accessToken);
}
public List<SubtitleStreamInfo> GetSubtitleProfiles(bool includeSelectedTrackOnly, bool enableAllProfiles, string baseUrl, string accessToken)
{
2014-08-06 06:18:13 +02:00
List<SubtitleStreamInfo> list = new List<SubtitleStreamInfo>();
// HLS will preserve timestamps so we can just grab the full subtitle stream
long startPositionTicks = StringHelper.EqualsIgnoreCase(SubProtocol, "hls")
? 0
2016-06-26 18:21:10 +02:00
: (PlayMethod == PlayMethod.Transcode && !CopyTimestamps ? StartPositionTicks : 0);
2014-11-16 21:44:08 +01:00
// First add the selected track
if (SubtitleStreamIndex.HasValue)
{
foreach (MediaStream stream in MediaSource.MediaStreams)
{
2015-02-05 06:29:37 +01:00
if (stream.Type == MediaStreamType.Subtitle && stream.Index == SubtitleStreamIndex.Value)
2014-11-16 21:44:08 +01:00
{
2015-03-31 18:24:16 +02:00
AddSubtitleProfiles(list, stream, enableAllProfiles, baseUrl, accessToken, startPositionTicks);
2014-11-16 21:44:08 +01:00
}
}
}
if (!includeSelectedTrackOnly)
{
foreach (MediaStream stream in MediaSource.MediaStreams)
{
2015-02-05 06:29:37 +01:00
if (stream.Type == MediaStreamType.Subtitle && (!SubtitleStreamIndex.HasValue || stream.Index != SubtitleStreamIndex.Value))
2014-11-16 21:44:08 +01:00
{
2015-03-31 18:24:16 +02:00
AddSubtitleProfiles(list, stream, enableAllProfiles, baseUrl, accessToken, startPositionTicks);
2014-11-16 21:44:08 +01:00
}
}
}
2015-03-30 21:57:37 +02:00
2014-11-16 21:44:08 +01:00
return list;
}
2015-03-31 18:24:16 +02:00
private void AddSubtitleProfiles(List<SubtitleStreamInfo> list, MediaStream stream, bool enableAllProfiles, string baseUrl, string accessToken, long startPositionTicks)
2014-11-16 21:44:08 +01:00
{
2015-03-31 18:24:16 +02:00
if (enableAllProfiles)
2015-02-05 06:29:37 +01:00
{
2015-03-31 18:24:16 +02:00
foreach (SubtitleProfile profile in DeviceProfile.SubtitleProfiles)
2015-03-30 21:57:37 +02:00
{
2015-03-31 18:24:16 +02:00
SubtitleStreamInfo info = GetSubtitleStreamInfo(stream, baseUrl, accessToken, startPositionTicks, new[] { profile });
list.Add(info);
2015-03-30 21:57:37 +02:00
}
2015-02-05 06:29:37 +01:00
}
2015-03-31 18:24:16 +02:00
else
{
SubtitleStreamInfo info = GetSubtitleStreamInfo(stream, baseUrl, accessToken, startPositionTicks, DeviceProfile.SubtitleProfiles);
2015-02-05 06:29:37 +01:00
2015-03-31 18:24:16 +02:00
list.Add(info);
}
}
2015-03-31 18:24:16 +02:00
private SubtitleStreamInfo GetSubtitleStreamInfo(MediaStream stream, string baseUrl, string accessToken, long startPositionTicks, SubtitleProfile[] subtitleProfiles)
{
2016-04-03 01:39:08 +02:00
SubtitleProfile subtitleProfile = StreamBuilder.GetSubtitleProfile(stream, subtitleProfiles, PlayMethod);
2015-03-31 18:24:16 +02:00
SubtitleStreamInfo info = new SubtitleStreamInfo
2014-08-06 04:26:12 +02:00
{
2014-11-16 21:44:08 +01:00
IsForced = stream.IsForced,
Language = stream.Language,
Name = stream.Language ?? "Unknown",
2015-03-31 18:24:16 +02:00
Format = subtitleProfile.Format,
2015-03-30 21:57:37 +02:00
Index = stream.Index,
2016-05-22 18:45:28 +02:00
DeliveryMethod = subtitleProfile.Method,
DisplayTitle = stream.DisplayTitle
};
2015-03-31 18:24:16 +02:00
if (info.DeliveryMethod == SubtitleDeliveryMethod.External)
{
if (MediaSource.Protocol == MediaProtocol.File || !StringHelper.EqualsIgnoreCase(stream.Codec, subtitleProfile.Format))
{
info.Url = string.Format("{0}/Videos/{1}/{2}/Subtitles/{3}/{4}/Stream.{5}",
baseUrl,
ItemId,
MediaSourceId,
StringHelper.ToStringCultureInvariant(stream.Index),
StringHelper.ToStringCultureInvariant(startPositionTicks),
subtitleProfile.Format);
2015-04-02 04:47:59 +02:00
2016-03-16 17:53:53 +01:00
if (!string.IsNullOrEmpty(accessToken))
2016-03-15 20:12:10 +01:00
{
info.Url += "?api_key=" + accessToken;
}
2015-04-02 04:47:59 +02:00
info.IsExternalUrl = false;
2015-03-31 18:24:16 +02:00
}
else
{
info.Url = stream.Path;
2015-04-02 04:47:59 +02:00
info.IsExternalUrl = true;
2015-03-31 18:24:16 +02:00
}
}
return info;
}
2014-04-18 19:16:25 +02:00
/// <summary>
/// Returns the audio stream that will be used
/// </summary>
public MediaStream TargetAudioStream
{
get
{
if (MediaSource != null)
{
2015-03-23 18:19:21 +01:00
return MediaSource.GetDefaultAudioStream(AudioStreamIndex);
2014-04-18 19:16:25 +02:00
}
return null;
}
}
/// <summary>
/// Returns the video stream that will be used
/// </summary>
public MediaStream TargetVideoStream
{
get
{
if (MediaSource != null)
{
2014-05-10 01:08:08 +02:00
return MediaSource.VideoStream;
2014-04-18 19:16:25 +02:00
}
return null;
}
}
/// <summary>
/// Predicts the audio sample rate that will be in the output stream
/// </summary>
public int? TargetAudioSampleRate
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetAudioStream;
2014-04-18 19:16:25 +02:00
return stream == null ? null : stream.SampleRate;
}
}
2014-04-24 07:08:10 +02:00
/// <summary>
/// Predicts the audio sample rate that will be in the output stream
/// </summary>
public int? TargetVideoBitDepth
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetVideoStream;
2014-04-24 07:08:10 +02:00
return stream == null || !IsDirectStream ? null : stream.BitDepth;
}
}
2014-09-09 03:15:31 +02:00
/// <summary>
/// Gets the target reference frames.
/// </summary>
/// <value>The target reference frames.</value>
public int? TargetRefFrames
{
get
{
MediaStream stream = TargetVideoStream;
return stream == null || !IsDirectStream ? null : stream.RefFrames;
}
}
2014-04-24 07:08:10 +02:00
/// <summary>
/// Predicts the audio sample rate that will be in the output stream
/// </summary>
2014-06-23 18:05:19 +02:00
public float? TargetFramerate
2014-04-24 07:08:10 +02:00
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetVideoStream;
2014-04-24 07:08:10 +02:00
return MaxFramerate.HasValue && !IsDirectStream
? MaxFramerate
: stream == null ? null : stream.AverageFrameRate ?? stream.RealFrameRate;
}
}
/// <summary>
/// Predicts the audio sample rate that will be in the output stream
/// </summary>
public double? TargetVideoLevel
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetVideoStream;
2014-04-24 07:08:10 +02:00
return VideoLevel.HasValue && !IsDirectStream
? VideoLevel
: stream == null ? null : stream.Level;
}
}
/// <summary>
/// Predicts the audio sample rate that will be in the output stream
/// </summary>
public int? TargetPacketLength
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetVideoStream;
2014-04-24 07:08:10 +02:00
return !IsDirectStream
? null
: stream == null ? null : stream.PacketLength;
}
}
/// <summary>
/// Predicts the audio sample rate that will be in the output stream
/// </summary>
public string TargetVideoProfile
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetVideoStream;
2014-04-24 07:08:10 +02:00
return !string.IsNullOrEmpty(VideoProfile) && !IsDirectStream
? VideoProfile
: stream == null ? null : stream.Profile;
}
}
2015-10-19 18:05:03 +02:00
/// <summary>
/// Gets the target video codec tag.
/// </summary>
/// <value>The target video codec tag.</value>
public string TargetVideoCodecTag
{
get
{
MediaStream stream = TargetVideoStream;
return !IsDirectStream
? null
: stream == null ? null : stream.CodecTag;
}
}
2014-04-18 19:16:25 +02:00
/// <summary>
/// Predicts the audio bitrate that will be in the output stream
/// </summary>
public int? TargetAudioBitrate
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetAudioStream;
2014-04-18 19:16:25 +02:00
return AudioBitrate.HasValue && !IsDirectStream
? AudioBitrate
: stream == null ? null : stream.BitRate;
}
}
/// <summary>
/// Predicts the audio channels that will be in the output stream
/// </summary>
public int? TargetAudioChannels
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetAudioStream;
int? streamChannels = stream == null ? null : stream.Channels;
2014-04-18 19:16:25 +02:00
2014-12-24 07:28:40 +01:00
if (MaxAudioChannels.HasValue && !IsDirectStream)
{
if (streamChannels.HasValue)
{
return Math.Min(MaxAudioChannels.Value, streamChannels.Value);
}
return MaxAudioChannels.Value;
}
return streamChannels;
2014-04-18 19:16:25 +02:00
}
}
/// <summary>
/// Predicts the audio codec that will be in the output stream
/// </summary>
public string TargetAudioCodec
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetAudioStream;
2014-04-18 19:16:25 +02:00
2016-06-26 18:21:10 +02:00
string inputCodec = stream == null ? null : stream.Codec;
if (IsDirectStream)
{
return inputCodec;
}
foreach (string codec in AudioCodecs)
{
if (StringHelper.EqualsIgnoreCase(codec, inputCodec))
{
return codec;
}
}
return AudioCodecs.Length == 0 ? null : AudioCodecs[0];
2014-04-18 19:16:25 +02:00
}
}
/// <summary>
/// Predicts the audio channels that will be in the output stream
/// </summary>
public long? TargetSize
{
get
{
if (IsDirectStream)
{
2014-04-25 19:30:41 +02:00
return MediaSource.Size;
2014-04-18 19:16:25 +02:00
}
if (RunTimeTicks.HasValue)
{
2014-05-08 22:09:53 +02:00
int? totalBitrate = TargetTotalBitrate;
2014-04-18 19:16:25 +02:00
2014-07-01 06:06:28 +02:00
double totalSeconds = RunTimeTicks.Value;
// Convert to ms
totalSeconds /= 10000;
// Convert to seconds
totalSeconds /= 1000;
2014-04-28 17:05:28 +02:00
return totalBitrate.HasValue ?
2014-07-01 06:06:28 +02:00
Convert.ToInt64(totalBitrate.Value * totalSeconds) :
2014-04-28 17:05:28 +02:00
(long?)null;
2014-04-18 19:16:25 +02:00
}
2014-04-28 17:05:28 +02:00
return null;
2014-04-18 19:16:25 +02:00
}
}
2014-04-23 04:47:46 +02:00
2014-04-24 07:08:10 +02:00
public int? TargetVideoBitrate
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream stream = TargetVideoStream;
2014-04-24 07:08:10 +02:00
return VideoBitrate.HasValue && !IsDirectStream
? VideoBitrate
: stream == null ? null : stream.BitRate;
}
}
public TransportStreamTimestamp TargetTimestamp
{
get
{
TransportStreamTimestamp defaultValue = StringHelper.EqualsIgnoreCase(Container, "m2ts")
2014-04-25 04:45:06 +02:00
? TransportStreamTimestamp.Valid
: TransportStreamTimestamp.None;
2014-04-28 17:05:28 +02:00
return !IsDirectStream
? defaultValue
2014-04-25 04:45:06 +02:00
: MediaSource == null ? defaultValue : MediaSource.Timestamp ?? TransportStreamTimestamp.None;
2014-04-24 07:08:10 +02:00
}
}
public int? TargetTotalBitrate
2014-04-23 04:47:46 +02:00
{
get
{
2014-04-24 07:08:10 +02:00
return (TargetAudioBitrate ?? 0) + (TargetVideoBitrate ?? 0);
2014-04-23 04:47:46 +02:00
}
}
2014-06-22 18:25:47 +02:00
public bool? IsTargetAnamorphic
{
get
{
if (IsDirectStream)
{
return TargetVideoStream == null ? null : TargetVideoStream.IsAnamorphic;
}
return false;
}
}
2016-10-03 08:28:45 +02:00
public bool? IsTargetAVC
{
get
{
if (IsDirectStream)
{
return TargetVideoStream == null ? null : TargetVideoStream.IsAVC;
}
return true;
}
}
2014-04-23 04:47:46 +02:00
public int? TargetWidth
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream videoStream = TargetVideoStream;
2014-04-23 04:47:46 +02:00
if (videoStream != null && videoStream.Width.HasValue && videoStream.Height.HasValue)
{
2014-05-08 22:09:53 +02:00
ImageSize size = new ImageSize
2014-04-23 04:47:46 +02:00
{
Width = videoStream.Width.Value,
Height = videoStream.Height.Value
};
2014-07-01 06:06:28 +02:00
double? maxWidth = MaxWidth.HasValue ? (double)MaxWidth.Value : (double?)null;
double? maxHeight = MaxHeight.HasValue ? (double)MaxHeight.Value : (double?)null;
2014-05-08 22:09:53 +02:00
ImageSize newSize = DrawingUtils.Resize(size,
2014-04-23 04:47:46 +02:00
null,
null,
2014-07-01 06:06:28 +02:00
maxWidth,
maxHeight);
2014-04-23 04:47:46 +02:00
return Convert.ToInt32(newSize.Width);
}
return MaxWidth;
}
}
public int? TargetHeight
{
get
{
2014-05-08 22:09:53 +02:00
MediaStream videoStream = TargetVideoStream;
2014-04-23 04:47:46 +02:00
if (videoStream != null && videoStream.Width.HasValue && videoStream.Height.HasValue)
{
2014-05-08 22:09:53 +02:00
ImageSize size = new ImageSize
2014-04-23 04:47:46 +02:00
{
Width = videoStream.Width.Value,
Height = videoStream.Height.Value
};
2014-07-01 06:06:28 +02:00
double? maxWidth = MaxWidth.HasValue ? (double)MaxWidth.Value : (double?)null;
double? maxHeight = MaxHeight.HasValue ? (double)MaxHeight.Value : (double?)null;
2014-05-08 22:09:53 +02:00
ImageSize newSize = DrawingUtils.Resize(size,
2014-04-23 04:47:46 +02:00
null,
null,
2014-07-01 06:06:28 +02:00
maxWidth,
maxHeight);
2014-04-23 04:47:46 +02:00
return Convert.ToInt32(newSize.Height);
}
return MaxHeight;
}
}
2015-03-12 22:13:00 +01:00
public int? TargetVideoStreamCount
{
get
{
if (IsDirectStream)
{
return GetMediaStreamCount(MediaStreamType.Video, int.MaxValue);
}
return GetMediaStreamCount(MediaStreamType.Video, 1);
}
}
public int? TargetAudioStreamCount
{
get
{
if (IsDirectStream)
{
return GetMediaStreamCount(MediaStreamType.Audio, int.MaxValue);
}
return GetMediaStreamCount(MediaStreamType.Audio, 1);
}
}
private int? GetMediaStreamCount(MediaStreamType type, int limit)
{
var count = MediaSource.GetStreamCount(type);
if (count.HasValue)
{
count = Math.Min(count.Value, limit);
}
return count;
}
2015-03-12 22:13:00 +01:00
public List<MediaStream> GetSelectableAudioStreams()
{
return GetSelectableStreams(MediaStreamType.Audio);
}
public List<MediaStream> GetSelectableSubtitleStreams()
{
return GetSelectableStreams(MediaStreamType.Subtitle);
}
public List<MediaStream> GetSelectableStreams(MediaStreamType type)
{
List<MediaStream> list = new List<MediaStream>();
foreach (MediaStream stream in MediaSource.MediaStreams)
{
if (type == stream.Type)
{
list.Add(stream);
}
}
return list;
}
2014-04-02 00:23:07 +02:00
}
}