在WPF中播放RTSP视频流
在WPF中可以使用LibVLCSharp.WPF、Vlc.DotNet.Wpf组件直接播放RTSP流,此外还可以通过LibVLCSharp + SkiaSharp的方式将VLC视频流通过WriteableBitmap的方式关联到Image对象。
1、LibVLCSharp.WPF
使用VideoView控件可以方便地播放RTSP视频流。该方法使用简单,但是不能很好的对RTSP流图像进行处理。
(1) 引入NuGet包
LibVLCSharp.WPF。
(2) 初始化LibVLCSharp
LibVLCSharp.Shared.Core.Initialize();
(3) 使用VideoView控件
<Grid HorizontalAlignment="Stretch" VerticalAlignment="Stretch"> <wpf:VideoView x:Name="VideoView"/> </Grid>
(4) 初始化LibVLC对象
_libVlc = new LibVLC(true); _mediaPlayer = new MediaPlayer(_libVlc) { Volume = 0, //静音 EnableHardwareDecoding = true //硬件加速 }; _mediaPlayer.EndReached += OnPlayerEndReached; _mediaPlayer.Stopped += OnPlayerStopped; VideoView.IsManipulationEnabled = true; VideoView.IsEnabled = true; VideoView.Loaded += (sender, args) => { VideoView.MediaPlayer = _mediaPlayer; };
(5) Unloaded事件中销毁VLC对象
_mediaPlayer.Stop();
_mediaPlayer.Dispose();
_libVlc.Dispose();
(6) 播放RTSP
var options = new[] { "file-caching=300", "live-capture-caching=300", "disc-caching-caching=300", "network-caching=333", "live-caching=300" }; using (var media = new Media(_libVlc, new Uri(CameraPara.Url), options)) { VideoView.MediaPlayer.Play(media); }
(7) 停止播放
if (VideoView.MediaPlayer.IsPlaying) { VideoView.MediaPlayer.Stop(); }
2、Vlc.DotNet.Wpf
借助于VlcVideoSourceProvider类,可以将RTSP视频流经过处理后轻松绑定到Image对象。同时,通过改造VlcVideoSourceProvider类可以实现自由设置RTSP视频流的播放帧率(FPS)。
(1) 引入NuGet包
Vlc.DotNet.Wpf。
(2) 启用位图缓存
<UserControl.CacheMode> <BitmapCache/> </UserControl.CacheMode>
(3) 初始化SourceProvider
_sourceProvider = new VlcVideoSourceProvider(Dispatcher); _sourceProvider.CreatePlayer(_libDirectory, "--no-audio", "--rtsp-tcp"); _sourceProvider.MediaPlayer.Stopped += OnPlayerStopped; VideoImage.SetBinding(Image.SourceProperty, new Binding(nameof(VlcVideoSourceProvider.VideoSource)) { Source = _sourceProvider });
(4) 播放RTSP视频流
var options = new[] { "file-caching=300", "live-capture-caching=300", "disc-caching-caching=300", "network-caching=333", "live-caching=300", HardDecoding ? "avcodec-hw=any" : "" //硬解码 }; _sourceProvider.MediaPlayer.Play(new Uri(CameraPara.Url), options);
(5) 停止播放
if (_sourceProvider.MediaPlayer.IsPlaying()) { _sourceProvider?.MediaPlayer?.Dispose(); _sourceProvider?.Dispose(); _sourceProvider = null; }
3、LibVLCSharp + SkiaSharp
通过设置MediaPlayer的视频回调方法,在视频VideoLock回调中将视频图像拷贝至缓存中,然后在VideoDisplay回调中将缓存中的数据转换为SKImage绘制在Skia画布中,之后刷新关联的位图即可。
(1) 引入NuGet包
LibVLCSharp、SkiaSharp。
(2) 初始化LibVLCSharp
LibVLCSharp.Shared.Core.Initialize();
(3) 初始化LibVLC对象
_libVlc = new LibVLC(true); _mediaPlayer = new MediaPlayer(_libVlc) { Volume = 0, //静音 EnableHardwareDecoding = true //硬件加速 }; _mediaPlayer.EndReached += OnPlayerEndReached; _mediaPlayer.Stopped += OnPlayerStopped;
(4) Loaded事件中设置视频回调
_mediaPlayer.SetVideoFormatCallbacks(OnLibVLCVideoFormat, null); _mediaPlayer.SetVideoCallbacks(OnLibVLCVideoLock, null, OnLibVLCVideoDisplay);
(5) Unloaded事件中销毁VLC对象
_mediaPlayer.Stop();
_mediaPlayer.Dispose();
_libVlc.Dispose();
(6) 视频回调处理
private uint OnLibVLCVideoFormat(ref IntPtr opaque, IntPtr chroma, ref uint width, ref uint height, ref uint pitches, ref uint lines) // ReSharper restore RedundantAssignment { var bytes = Encoding.ASCII.GetBytes("RV32"); //I420, RV32, AVC1 for (var i = 0; i < bytes.Length; i++) { Marshal.WriteByte(chroma, i, bytes[i]); } if (_mediaPlayer.Media is Media media) { foreach (MediaTrack track in media.Tracks) { if (track.TrackType == TrackType.Video) { var trackInfo = track.Data.Video; if (trackInfo.Width > 0 && trackInfo.Height > 0) { width = trackInfo.Width; height = trackInfo.Height; } break; } } } var pixelFormat = PixelFormats.Bgra32; pitches = (uint) (width * pixelFormat.BitsPerPixel) / 8; lines = height; _videoWidth = (int) width; _videoHeight = (int) height; _buffer = new byte[_videoWidth * _videoHeight * 4]; _plane = Marshal.UnsafeAddrOfPinnedArrayElement(_buffer, 0); Dispatcher.Invoke(delegate { _bitmap = new WriteableBitmap(_videoWidth, _videoHeight, 96, 96, PixelFormats.Bgra32, null); _imageInfo = new SKImageInfo(_videoWidth, _videoHeight, SKColorType.Bgra8888); _surface = SKSurface.Create( new SKImageInfo(_videoWidth, _videoHeight, SKImageInfo.PlatformColorType, SKAlphaType.Premul), _bitmap.BackBuffer, _bitmap.BackBufferStride); _rect = new Int32Rect(0, 0, _videoWidth, _videoHeight); VideoImage.Source = _bitmap; VideoImage.Stretch = IsMainControl ? Stretch.Fill : Stretch.Uniform; }); return 1; } private IntPtr OnLibVLCVideoLock(IntPtr opaque, IntPtr planes) { IntPtr[] dataArray = {_plane}; Marshal.Copy(dataArray, 0, planes, dataArray.Length); return IntPtr.Zero; } private void OnLibVLCVideoDisplay(IntPtr opaque, IntPtr picture) { var image = SKImage.FromPixels(_imageInfo, _plane); _surface.Canvas.DrawImage(image, new SKPoint(0, 0)); Dispatcher.Invoke(delegate { _bitmap.Lock(); _bitmap.AddDirtyRect(_rect); _bitmap.Unlock(); }); }
(7) 播放RTSP视频
var options = new[] { "file-caching=300", "live-capture-caching=300", "disc-caching-caching=300", "network-caching=333", "live-caching=300" }; using (var media = new Media(_libVlc, new Uri(CameraPara.Url), options)) { _mediaPlayer.Play(media); }
(8) 停止播放
if (_mediaPlayer.IsPlaying) { _mediaPlayer.Stop(); }