using UnityEngine;
using System;
using System.Collections;
using System.Collections.Generic;
//-----------------------------------------------------------------------------
// Copyright 2015-2021 RenderHeads Ltd. All rights reserved.
//-----------------------------------------------------------------------------
namespace RenderHeads.Media.AVProVideo
{
public interface IMediaPlayer
{
void OnEnable();
void Update();
void EndUpdate();
void Render();
IntPtr GetNativePlayerHandle();
}
///
/// Interface for side loading of subtitles in SRT format
///
public interface IMediaSubtitles
{
bool LoadSubtitlesSRT(string data);
int GetSubtitleIndex();
string GetSubtitleText();
}
public enum BufferedFrameSelectionMode : int
{
// No buffering, just selects the latest decoded frame
None = 0,
// Selects the newest buffered frame, and displays it until a newer frame is available
NewestFrame = 10,
// Selects the oldest buffered frame, and displays it until a newer frame is available
OldestFrame = 11,
// Selects the next buffered frame, and displays it until the number of buffered frames changes
MediaClock = 20,
// Uses Time.deltaTime to keep a clock which is used to select the buffered frame
ElapsedTime = 30,
// Uses VSync delta time to keep a clock which is used to select the buffered frame
// Time.deltaTime is used to calculate the number of vsyncs that have elapsed
ElapsedTimeVsynced = 40,
// Selects the buffered frame corresponding to the external timeStamp (useful for frame-syncing players)
FromExternalTime = 50,
// Selects the closest buffered frame corresponding to the external timeStamp (useful for frame-syncing players)
FromExternalTimeClosest = 51,
}
///
/// Interface for buffering frames for more control over the timing of their display
///
public interface IBufferedDisplay
{
///
/// We need to manually call UpdateBufferedDisplay() in the case of master-slave synced playback so that master is updated before slaves
///
long UpdateBufferedDisplay();
BufferedFramesState GetBufferedFramesState();
void SetSlaves(IBufferedDisplay[] slaves);
void SetBufferedDisplayMode(BufferedFrameSelectionMode mode, IBufferedDisplay master = null);
void SetBufferedDisplayOptions(bool pauseOnPrerollComplete);
}
public interface IMediaControl
{
///
/// Be careful using this method directly. It is best to instead use the OpenMedia() method in the MediaPlayer component as this will set up the events correctly and also perform other checks
/// customHttpHeaders is in the format "key1:value1\r\nkey2:value2\r\n"=
///
bool OpenMedia(string path, long offset, string customHttpHeaders, MediaHints mediahints, int forceFileFormat = 0, bool startWithHighestBitrate = false);
bool OpenMediaFromBuffer(byte[] buffer);
bool StartOpenMediaFromBuffer(ulong length);
bool AddChunkToMediaBuffer(byte[] chunk, ulong offset, ulong length);
bool EndOpenMediaFromBuffer();
#if NETFX_CORE
bool OpenMedia(Windows.Storage.Streams.IRandomAccessStream ras, string path, long offset, string customHttpHeaders);
#endif
void CloseMedia();
void SetLooping(bool bLooping);
bool IsLooping();
bool HasMetaData();
bool CanPlay();
bool IsPlaying();
bool IsSeeking();
bool IsPaused();
bool IsFinished();
bool IsBuffering();
void Play();
void Pause();
void Stop();
void Rewind();
///
/// The time in seconds seeked will be to the exact time
/// This can take a long time is the keyframes are far apart
/// Some platforms don't support this and instead seek to the closest keyframe
///
void Seek(double time);
///
/// The time in seconds seeked will be to the closest keyframe
///
void SeekFast(double time);
///
/// The time in seconds seeked to will be within the range [time-timeDeltaBefore, time+timeDeltaAfter] for efficiency.
/// Only supported on macOS, iOS and tvOS.
/// Other platforms will automatically pass through to Seek()
///
void SeekWithTolerance(double time, double timeDeltaBefore, double timeDeltaAfter);
///
/// Seek to a specific frame, range is [0, GetMaxFrameNumber()]
/// NOTE: For best results the video should be encoded as keyframes only
/// and have no audio track, or an audio track with the same length as the video track
///
void SeekToFrame(int frame, float overrideFrameRate = 0f);
///
/// Seek forwards or backwards relative to the current frame
/// NOTE: For best results the video should be encoded as keyframes only
/// and have no audio track, or an audio track with the same length as the video track
///
void SeekToFrameRelative(int frameOffset, float overrideFrameRate = 0f);
///
/// Returns the current video time in seconds
///
double GetCurrentTime();
///
/// Returns the current video time in frames, range is [0, GetMaxFrameNumber()]
/// NOTE: For best results the video should be encoded as keyframes only
/// and have no audio track, or an audio track with the same length as the video track
///
int GetCurrentTimeFrames(float overrideFrameRate = 0f);
///
/// Returns the current video date and time usually from the
/// EXT-X-PROGRAM-DATE-TIME tag on HLS streams
/// Only supported on macOS, iOS, tvOS and Android (using ExoPlayer API)
/// And Windows 10 using WinRT API
///
System.DateTime GetProgramDateTime();
float GetPlaybackRate();
void SetPlaybackRate(float rate);
void MuteAudio(bool bMute);
bool IsMuted();
void SetVolume(float volume);
void SetBalance(float balance);
float GetVolume();
float GetBalance();
/*int GetCurrentVideoTrack();
void SetVideoTrack(int index);
int GetCurrentAudioTrack();
void SetAudioTrack(int index);*/
///
/// Returns a range of time values that can be seeked in seconds
///
TimeRanges GetSeekableTimes();
///
/// Returns a range of time values that contain fully downloaded segments,
/// which can be seeked to immediately without requiring additional downloading
///
TimeRanges GetBufferedTimes();
ErrorCode GetLastError();
long GetLastExtendedErrorCode();
void SetTextureProperties(FilterMode filterMode = FilterMode.Bilinear, TextureWrapMode wrapMode = TextureWrapMode.Clamp, int anisoLevel = 1);
void GetTextureProperties(out FilterMode filterMode, out TextureWrapMode wrapMode, out int anisoLevel);
// Audio Grabbing
///
/// Copies the specified amount of audio into the buffer
/// If the specified amount is not yet available then nothing no samples are copied
/// The number of audio samples grabbed are returned
///
int GrabAudio(float[] buffer, int sampleCount, int channelCount);
int GetAudioBufferedSampleCount();
int GetAudioChannelCount();
AudioChannelMaskFlags GetAudioChannelMask();
// Audio 360
void SetAudioChannelMode(Audio360ChannelMode channelMode);
void SetAudioHeadRotation(Quaternion q);
void ResetAudioHeadRotation();
void SetAudioFocusEnabled(bool enabled);
void SetAudioFocusProperties(float offFocusLevel, float widthDegrees);
void SetAudioFocusRotation(Quaternion q);
void ResetAudioFocus();
bool WaitForNextFrame(Camera dummyCamera, int previousFrameCount);
[Obsolete("SetPlayWithoutBuffering has been deprecated, see platform specific options for how to enable playback without buffering (if supported).")]
void SetPlayWithoutBuffering(bool playWithoutBuffering);
// Encrypted stream support
//void SetKeyServerURL(string url);
void SetKeyServerAuthToken(string token);
void SetOverrideDecryptionKey(byte[] key);
// External playback support.
///
/// Check to see if external playback is currently active on the player.
///
bool IsExternalPlaybackActive();
///
/// Set whether the player is allowed to switch to external playback, e.g. AirPlay.
///
void SetAllowsExternalPlayback(bool enable);
///
/// Sets the video gravity of the player for external playback only.
///
void SetExternalPlaybackVideoGravity(ExternalPlaybackVideoGravity gravity);
}
public interface IMediaInfo
{
///
/// Returns media duration in seconds
///
double GetDuration();
///
/// Returns media duration in frames
/// NOTE: For best results the video should be encoded as keyframes only
/// and have no audio track, or an audio track with the same length as the video track
///
int GetDurationFrames(float overrideFrameRate = 0f);
///
/// Returns highest frame number that can be seeked to
/// NOTE: For best results the video should be encoded as keyframes only
/// and have no audio track, or an audio track with the same length as the video track
///
int GetMaxFrameNumber(float overrideFrameRate = 0f);
///
/// Returns video width in pixels
///
int GetVideoWidth();
///
/// Returns video height in pixels
///
int GetVideoHeight();
///
/// Returns the frame rate of the media.
///
float GetVideoFrameRate();
///
/// Returns the current achieved display rate in frames per second
///
float GetVideoDisplayRate();
///
/// Returns true if the media has a visual track
///
bool HasVideo();
///
/// Returns true if the media has a audio track
///
bool HasAudio();
///
/// Returns the a description of which playback path is used internally.
/// This can for example expose whether CPU or GPU decoding is being performed
/// For Windows the available player descriptions are:
/// "DirectShow" - legacy Microsoft API but still very useful especially with modern filters such as LAV
/// "MF-MediaEngine-Software" - uses the Windows 8.1 features of the Microsoft Media Foundation API, but software decoding
/// "MF-MediaEngine-Hardware" - uses the Windows 8.1 features of the Microsoft Media Foundation API, but GPU decoding
/// Android has "MediaPlayer" and "ExoPlayer"
/// macOS / tvOS / iOS just has "AVFoundation"
///
string GetPlayerDescription();
#if !AVPRO_NEW_GAMMA
///
/// Whether this MediaPlayer instance supports linear color space
/// If it doesn't then a correction may have to be made in the shader
///
bool PlayerSupportsLinearColorSpace();
#endif
///
/// Checks if the playback is in a stalled state
///
bool IsPlaybackStalled();
///
/// The affine transform of the texture as an array of six floats: a, b, c, d, tx, ty.
///
float[] GetTextureTransform();
///
/// Gets the estimated bandwidth used by all video players (in bits per second)
/// Currently only supported on Android when using ExoPlayer API
///
long GetEstimatedTotalBandwidthUsed();
/*
string GetMediaDescription();
string GetVideoDescription();
string GetAudioDescription();*/
///
/// Checks if the media is compatible with external playback, for instance via AirPlay.
///
bool IsExternalPlaybackSupported();
// Internal method
bool GetDecoderPerformance(ref int activeDecodeThreadCount, ref int decodedFrameCount, ref int droppedFrameCount);
// Internal method
PlaybackQualityStats GetPlaybackQualityStats();
}
#region MediaCaching
/// Options for configuring media caching.
public class MediaCachingOptions
{
/// The minimum bitrate of the media to cache in bits per second.
public double minimumRequiredBitRate;
/// The minimum resolution of the media to cache.
/// Only supported on Android and iOS 14 and later.
public Vector2 minimumRequiredResolution;
/// The maximum bitrate of the media to cache in bits per second.
/// Only supported on Android.
public double maximumRequiredBitRate;
/// The maximum resolution of the media to cache.
/// Only supported on Android.
public Vector2 maximumRequiredResolution;
/// Human readable title for the cached media.
/// iOS: This value will be displayed in the usage pane of the settings app.
public string title;
/// Optional artwork for the cached media in PNG format.
/// iOS: This value will be displayed in the usage pane of the settings app.
public byte[] artwork;
}
/// Status of the media item in the cache.
public enum CachedMediaStatus: int
{
/// The media has not been cached.
NotCached,
/// The media is being cached.
Caching,
/// The media is cached.
Cached,
/// The media is not cached, something went wrong - check the log.
Failed,
/// The media caching is paused.
Paused
}
/// Interface for the media cache.
public interface IMediaCache
{
/// Test to see if the player can cache media.
/// True if media caching is supported.
bool IsMediaCachingSupported();
/// Cache the media specified by url.
/// The url of the media.
///
///
void AddMediaToCache(string url, string headers = null, MediaCachingOptions options = null);
/// Cancels the download of the media specified by url.
/// The url of the media.
void CancelDownloadOfMediaToCache(string url);
/// Pause the download of the media specified by url.
/// The url of the media.
void PauseDownloadOfMediaToCache(string url);
/// Resume the download of the media specified by url.
/// The url of the media.
void ResumeDownloadOfMediaToCache(string url);
/// Remove the cached media specified by url.
/// The url of the media.
void RemoveMediaFromCache(string url);
/// Get the cached status for the media specified.
/// The url of the media.
/// The amount of the media that has been cached in the range [0...1].
/// The status of the media.
CachedMediaStatus GetCachedMediaStatus(string url, ref float progress);
// /// Test if the currently open media is cached.
// /// True if the media is cached, false otherwise.
// bool IsMediaCached();
}
#endregion
public interface ITextureProducer
{
///
/// Gets the number of textures produced by the media player.
///
int GetTextureCount();
///
/// Returns the Unity texture containing the current frame image.
/// The texture pointer will return null while the video is loading
/// This texture usually remains the same for the duration of the video.
/// There are cases when this texture can change, for instance: if the graphics device is recreated,
/// a new video is loaded, or if an adaptive stream (eg HLS) is used and it switches video streams.
///
Texture GetTexture(int index = 0);
///
/// Returns a count of how many times the texture has been updated
///
int GetTextureFrameCount();
///
/// Returns whether this platform supports counting the number of times the texture has been updated
///
bool SupportsTextureFrameCount();
///
/// Returns the presentation time stamp of the current texture
///
long GetTextureTimeStamp();
///
/// Returns the DAR/SAR ratio
///
float GetTexturePixelAspectRatio();
///
/// Returns true if the image on the texture is upside-down
///
bool RequiresVerticalFlip();
///
/// Returns the type of packing used for stereo content
///
StereoPacking GetTextureStereoPacking();
///
/// Returns the whether the texture has transparency
///
TransparencyMode GetTextureTransparency();
///
/// Returns the type of packing used for alpha content
///
AlphaPacking GetTextureAlphaPacking();
///
/// Returns the current transformation required to convert from YpCbCr to RGB colorspaces.
///
Matrix4x4 GetYpCbCrTransform();
#if AVPRO_NEW_GAMMA
///
/// Returns the gamma type of a sampled pixel
/// Is the texture returns samples in linear gamma then no conversion is need when using Unity's linear color space mode
/// If it doesn't then a correction may have to be made in the shader
///
TextureGamma GetTextureSampleGamma();
bool TextureRequiresGammaConversion();
#endif
}
public enum Platform
{
Windows,
MacOSX,
iOS,
tvOS,
Android,
WindowsUWP,
WebGL,
Count = 7,
Unknown = 100,
}
public enum MediaSource
{
Reference,
Path,
}
public enum MediaPathType
{
AbsolutePathOrURL,
RelativeToProjectFolder,
RelativeToStreamingAssetsFolder,
RelativeToDataFolder,
RelativeToPersistentDataFolder,
}
[System.Serializable]
public class MediaPath
{
[SerializeField] MediaPathType _pathType = MediaPathType.RelativeToStreamingAssetsFolder;
public MediaPathType PathType { get { return _pathType; } internal set { _pathType = value; } }
[SerializeField] string _path = string.Empty;
public string Path { get { return _path; } internal set { _path = value; } }
public MediaPath()
{
_pathType = MediaPathType.RelativeToStreamingAssetsFolder;
_path = string.Empty;
}
public MediaPath(MediaPath copy)
{
_pathType = copy.PathType;
_path = copy.Path;
}
public MediaPath(string path, MediaPathType pathType)
{
_pathType = pathType;
_path = path;
}
public string GetResolvedFullPath()
{
string result = Helper.GetFilePath(_path, _pathType);
#if (UNITY_EDITOR_WIN || (!UNITY_EDITOR && UNITY_STANDALONE_WIN))
if (result.Length > 200 && !result.Contains("://"))
{
result = Helper.ConvertLongPathToShortDOS83Path(result);
}
#endif
return result;
}
public static bool operator == (MediaPath a, MediaPath b)
{
if ((object)a == null)
return (object)b == null;
return a.Equals(b);
}
public static bool operator != (MediaPath a, MediaPath b)
{
return !(a == b);
}
public override bool Equals(object obj)
{
if (obj == null || GetType() != obj.GetType())
return false;
var a = (MediaPath)obj;
return (_pathType == a._pathType && _path == a._path);
}
public override int GetHashCode()
{
return _pathType.GetHashCode() ^ _path.GetHashCode();
}
}
public enum OverrideMode
{
None, // No overide, just use internal logic
Override, // Manually override
}
public enum TextureGamma
{
SRGB,
Linear,
// Future HDR support
// PQ,
// HLG,
}
public enum StereoPacking : int
{
None = 0, // Monoscopic
TopBottom = 1, // Top is the left eye, bottom is the right eye
LeftRight = 2, // Left is the left eye, right is the right eye
CustomUV = 3, // Use the mesh UV to unpack, uv0=left eye, uv1=right eye
TwoTextures = 4, // First texture left eye, second texture is right eye
Unknown = 10,
}
[System.Serializable]
public struct MediaHints
{
public TransparencyMode transparency;
public AlphaPacking alphaPacking;
public StereoPacking stereoPacking;
private static MediaHints defaultHints = new MediaHints();
public static MediaHints Default { get { return defaultHints; } }
}
[System.Serializable]
public struct VideoResolveOptions
{
[SerializeField] public bool applyHSBC;
[SerializeField, Range(0f, 1f)] public float hue;
[SerializeField, Range(0f, 1f)] public float saturation;
[SerializeField, Range(0f, 1f)] public float brightness;
[SerializeField, Range(0f, 1f)] public float contrast;
[SerializeField, Range(0.0001f, 10f)] public float gamma;
[SerializeField] public Color tint;
[SerializeField] public bool generateMipmaps;
public bool IsColourAdjust()
{
return (applyHSBC && (hue != 0.0f || saturation != 0.5f || brightness != 0.5f || contrast != 0.5f || gamma != 1.0f));
}
internal void ResetColourAdjust()
{
hue = 0.0f;
saturation = 0.5f;
brightness = 0.5f;
contrast = 0.5f;
gamma = 1.0f;
}
public static VideoResolveOptions Create()
{
VideoResolveOptions result = new VideoResolveOptions()
{
tint = Color.white,
};
result.ResetColourAdjust();
return result;
}
}
/// Transparency Mode
public enum TransparencyMode
{
Opaque,
Transparent,
}
public enum StereoEye
{
Both,
Left,
Right,
}
public enum AlphaPacking
{
None,
TopBottom,
LeftRight,
}
public enum ErrorCode
{
None = 0,
LoadFailed = 100,
DecodeFailed = 200,
}
public enum Orientation
{
Landscape, // Landscape Right (0 degrees)
LandscapeFlipped, // Landscape Left (180 degrees)
Portrait, // Portrait Up (90 degrees)
PortraitFlipped, // Portrait Down (-90 degrees)
PortraitHorizontalMirror, // Portrait that is mirrored horizontally
}
public enum VideoMapping
{
Unknown,
Normal,
EquiRectangular360,
EquiRectangular180,
CubeMap3x2,
}
public enum FileFormat
{
Unknown,
HLS,
DASH,
SmoothStreaming,
}
public static class Windows
{
public enum VideoApi
{
MediaFoundation, // Windows 8.1 and above
DirectShow, // Legacy API
WinRT, // Windows 10 and above
};
public enum AudioOutput
{
System, // Default
Unity, // Media Foundation API only
FacebookAudio360, // Media Foundation API only
None, // Media Foundation API only
}
// WIP: Experimental feature to allow overriding audio device for VR headsets
public const string AudioDeviceOutputName_Vive = "HTC VIVE USB Audio";
public const string AudioDeviceOutputName_Rift = "Headphones (Rift Audio)";
}
public static class WindowsUWP
{
public enum VideoApi
{
MediaFoundation, // UWP 8.1 and above
WinRT, // UWP 10 and above
};
public enum AudioOutput
{
System, // Default
Unity, // Media Foundation API only
FacebookAudio360, // Media Foundation API only
None, // Media Foundation API only
}
}
public static class Android
{
public enum VideoApi
{
MediaPlayer = 1,
ExoPlayer,
}
public enum AudioOutput
{
System, // Default
Unity, // ExoPlayer API only
FacebookAudio360, // ExoPlayer API only
}
public enum TextureFiltering
{
Point,
Bilinear,
Trilinear,
}
public const int Default_MinBufferTimeMs = 50000; // Only valid when using ExoPlayer (default comes from DefaultLoadControl.DEFAULT_MIN_BUFFER_MS)
public const int Default_MaxBufferTimeMs = 50000; // Only valid when using ExoPlayer (default comes from DefaultLoadControl.DEFAULT_MAX_BUFFER_MS)
public const int Default_BufferForPlaybackMs = 2500; // Only valid when using ExoPlayer (default comes from DefaultLoadControl.DEFAULT_BUFFER_FOR_PLAYBACK_MS)
public const int Default_BufferForPlaybackAfterRebufferMs = 5000; // Only valid when using ExoPlayer (default comes from DefaultLoadControl.DEFAULT_BUFFER_FOR_PLAYBACK_AFTER_REBUFFER_MS)
}
public static class WebGL
{
public enum ExternalLibrary
{
None,
DashJs,
HlsJs,
Custom,
}
}
// Facebook Audio 360 channel mapping
public enum Audio360ChannelMode
{
TBE_8_2 = 0, /// 8 channels of hybrid TBE ambisonics and 2 channels of head-locked stereo audio
TBE_8, /// 8 channels of hybrid TBE ambisonics. NO head-locked stereo audio
TBE_6_2, /// 6 channels of hybrid TBE ambisonics and 2 channels of head-locked stereo audio
TBE_6, /// 6 channels of hybrid TBE ambisonics. NO head-locked stereo audio
TBE_4_2, /// 4 channels of hybrid TBE ambisonics and 2 channels of head-locked stereo audio
TBE_4, /// 4 channels of hybrid TBE ambisonics. NO head-locked stereo audio
TBE_8_PAIR0, /// Channels 1 and 2 of TBE hybrid ambisonics
TBE_8_PAIR1, /// Channels 3 and 4 of TBE hybrid ambisonics
TBE_8_PAIR2, /// Channels 5 and 6 of TBE hybrid ambisonics
TBE_8_PAIR3, /// Channels 7 and 8 of TBE hybrid ambisonics
TBE_CHANNEL0, /// Channels 1 of TBE hybrid ambisonics
TBE_CHANNEL1, /// Channels 2 of TBE hybrid ambisonics
TBE_CHANNEL2, /// Channels 3 of TBE hybrid ambisonics
TBE_CHANNEL3, /// Channels 4 of TBE hybrid ambisonics
TBE_CHANNEL4, /// Channels 5 of TBE hybrid ambisonics
TBE_CHANNEL5, /// Channels 6 of TBE hybrid ambisonics
TBE_CHANNEL6, /// Channels 7 of TBE hybrid ambisonics
TBE_CHANNEL7, /// Channels 8 of TBE hybrid ambisonics
HEADLOCKED_STEREO, /// Head-locked stereo audio
HEADLOCKED_CHANNEL0, /// Channels 1 or left of head-locked stereo audio
HEADLOCKED_CHANNEL1, /// Channels 2 or right of head-locked stereo audio
AMBIX_4, /// 4 channels of first order ambiX
AMBIX_4_2, /// 4 channels of first order ambiX with 2 channels of head-locked audio
AMBIX_9, /// 9 channels of second order ambiX
AMBIX_9_2, /// 9 channels of second order ambiX with 2 channels of head-locked audio
AMBIX_16, /// 16 channels of third order ambiX
AMBIX_16_2, /// 16 channels of third order ambiX with 2 channels of head-locked audio
MONO, /// Mono audio
STEREO, /// Stereo audio
UNKNOWN, /// Unknown channel map
INVALID, /// Invalid/unknown map. This must always be last.
}
[System.Flags]
public enum AudioChannelMaskFlags : int
{
Unspecified = 0x0,
FrontLeft = 0x1,
FrontRight = 0x2,
FrontCenter = 0x4,
LowFrequency = 0x8,
BackLeft = 0x10,
BackRight = 0x20,
FrontLeftOfCenter = 0x40,
FrontRightOfCenter = 0x80,
BackCenter = 0x100,
SideLeft = 0x200,
SideRight = 0x400,
TopCenter = 0x800,
TopFrontLeft = 0x1000,
TopFrontCenter = 0x2000,
TopFrontRight = 0x4000,
TopBackLeft = 0x8000,
TopBackCenter = 0x10000,
TopBackRight = 0x20000,
}
public enum TextureFlags : int
{
Unknown = 0,
TopDown = 1 << 0,
SamplingIsLinear = 1 << 1,
}
[System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential, Pack = 1)]
public struct BufferedFramesState
{
public System.Int32 freeFrameCount;
public System.Int32 bufferedFrameCount;
public System.Int64 minTimeStamp;
public System.Int64 maxTimeStamp;
public System.Int32 prerolledCount;
}
[System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential, Pack = 1)]
public struct TextureFrame
{
internal System.IntPtr texturePointer;
internal System.IntPtr auxTexturePointer;
internal System.Int64 timeStamp;
internal System.UInt32 frameCounter;
internal System.UInt32 writtenFrameCount;
internal TextureFlags flags;
internal System.IntPtr internalNativePointer;
}
[System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential, Pack = 1)]
public struct TimeRange
{
public TimeRange(double startTime, double duration)
{
this.startTime = startTime;
this.duration = duration;
}
public double startTime, duration;
public double StartTime { get { return startTime; } }
public double EndTime { get { return startTime + duration; } }
public double Duration { get { return duration; } }
}
public class TimeRanges : IEnumerable
{
internal TimeRanges() {}
public IEnumerator GetEnumerator()
{
return _ranges.GetEnumerator();
}
public TimeRange this[int index]
{
get
{
return _ranges[index];
}
}
internal TimeRanges(TimeRange[] ranges)
{
_ranges = ranges;
CalculateRange();
}
internal void CalculateRange()
{
_minTime = _maxTime = 0.0;
if (_ranges != null && _ranges.Length > 0)
{
double maxTime = 0.0;
double minTime = double.MaxValue;
for (int i = 0; i < _ranges.Length; i++)
{
minTime = System.Math.Min(minTime, _ranges[i].startTime);
maxTime = System.Math.Max(maxTime, _ranges[i].startTime + _ranges[i].duration);
}
_minTime = minTime;
_maxTime = maxTime;
}
}
public int Count { get { return _ranges.Length; } }
public double MinTime { get { return _minTime; } }
public double MaxTime { get { return _maxTime; } }
public double Duration { get { return (_maxTime - _minTime); } }
internal TimeRange[] _ranges = new TimeRange[0];
internal double _minTime = 0.0;
internal double _maxTime = 0.0;
}
///
/// Video gravity to use with external playback.
///
public enum ExternalPlaybackVideoGravity
{
/// Resizes the video to fit the display, may cause stretching.
Resize,
/// Resizes the video whilst preserving the video's aspect ratio to fit the display bounds.
ResizeAspect,
/// Resizes the video whilst preserving aspect to fill the display bounds.
ResizeAspectFill,
};
}