当前位置:   article > 正文

libVLC 提取视频帧

libVLC 提取视频帧

在前面的文章中,我们使用libvlc_media_player_set_hwnd设置了视频的显示的窗口。

libvlc_media_player_set_hwnd(vlc_mediaPlayer, (void *)ui.widgetShow->winId());

如果我们想要提取每一帧数据,将数据保存到本地,该如何操作呢?答案肯定是有的。

默认情况下,VLC 会使用自己的渲染机制来显示视频。但如果你想要在自己的应用程序中处理视频帧(例如进行视频编辑、分析或其他自定义渲染),可以使用 libvlc_video_set_callbacks 来指定自定义的回调函数。
以下是libvlc_video_set_callbacks函数声明。

  1. LIBVLC_API
  2. void libvlc_video_set_callbacks( libvlc_media_player_t *mp,
  3. libvlc_video_lock_cb lock,
  4. libvlc_video_unlock_cb unlock,
  5. libvlc_video_display_cb display,
  6. void *opaque );
  • mp是指向libvlc_media_player_t的指针,代表一个媒体播放器实例。
  • lock是一个回调函数,当VLC需要访问视频帧时会被调用。
  • unlock是一个回调函数,当VLC完成对视频帧的处理后会被调用。
  • display是一个回调函数,用于显示视频帧。
  • opaque是一个用户数据指针,会被传递给上述回调函数。

以下是libvlc_video_lock_cb声明。

typedef void *(*libvlc_video_lock_cb)(void *opaque, void **planes);
  • opaque是一个用户数据指针,libvlc_video_set_callbacks最后一个参数会传递给lock。
  • planes是一个指向指针的指针,它指向一个指针数组,每个指针指向一个视频平面(对于 YUV 格式,通常有三个平面:Y、U 和 V)。对于 RGB 格式,通常只有一个平面。
  • 返回值传递给unlock的参数2。

如果应用程序需要在视频渲染前对视频帧进行一些处理,那么可以在libvlc_video_lock_cb中进行这些处理,并将处理后的帧数据地址赋值给 *planes。

以下是libvlc_video_unlock_cb声明。

  1. typedef void (*libvlc_video_unlock_cb)(void *opaque, void *picture,
  2. void *const *planes);
  • opaque是一个用户数据指针,libvlc_video_set_callbacks最后一个参数会传递给lock。
  • picture是libvlc_video_lock_cb返回值。
  • planes是平面像素数据。

在打开一个新的视频文件的时候,我们不知道视频的宽和高等数据。这时候需要使用libvlc_video_set_format_callbacks来获取视频格式,函数声明如下。

  1. LIBVLC_API
  2. void libvlc_video_set_format_callbacks( libvlc_media_player_t *mp,
  3. libvlc_video_format_cb setup,
  4. libvlc_video_cleanup_cb cleanup );
  • mp是指向libvlc_media_player_t 的指针,代表一个媒体播放器实例。
  • setup是一个回调函数,用于设置视频像素格式和缓冲区配置。
  • cleanup是一个回调函数,用于清理视频像素格式和缓冲区配置。

 以下是libvlc_video_format_cb声明,用于在视频解码开始之前设置视频像素格式和缓冲区配置。

  1. typedef unsigned (*libvlc_video_format_cb)(void **opaque, char *chroma,
  2. unsigned *width, unsigned *height,
  3. unsigned *pitches,
  4. unsigned *lines);
  • opaque调用libvlc_video_set_callbacks时指定,可以被传递给回调函数,用于存储应用程序特定的数据。
  • chroma是一个指向字符数组的指针,用于返回像素格式(例如 “RGBA” 或 “YUV420P”)。
  • width和height返回视频帧宽度和高度。
  • pitches返回每个视频平面(对于 YUV 格式,通常有三个平面:Y、U 和 V)的行跨度(即一行像素所占的字节数)。
  • lines返回每个视频平面的行数。

代码示例:保存指定的数据帧。

头文件。

  1. #pragma once
  2. #include <QtWidgets/QWidget>
  3. #include "ui_showWidget.h"
  4. #include <QMenu>
  5. #include <QActionGroup>
  6. #include <vlc/vlc.h>
  7. #include <QDebug>
  8. #include <QFileDialog>
  9. #include <QThread>
  10. #include <QMouseEvent>
  11. #include <QKeyEvent>
  12. enum Rate
  13. {
  14. Rate2X,
  15. Rate1_5X,
  16. Rate1_25X,
  17. Rate1_0X,
  18. Rate0_75X,
  19. Rate0_5X
  20. };
  21. class showWidget : public QWidget
  22. {
  23. Q_OBJECT
  24. public:
  25. showWidget(QWidget *parent = nullptr);
  26. ~showWidget();
  27. private slots:
  28. void slotOpenFile();
  29. void slotPlay();
  30. void slotPause();
  31. void slotStop();
  32. void slotValueChanged(int value);
  33. void slotCurrentIndexChanged(int index);
  34. private:
  35. //事件处理回调
  36. static void vlcEvents(const libvlc_event_t *ev, void *param);
  37. private:
  38. Ui::showWidgetClass ui;
  39. private:
  40. libvlc_instance_t *vlc_base = nullptr;
  41. libvlc_media_t *vlc_media = nullptr;
  42. libvlc_media_player_t *vlc_mediaPlayer = nullptr;
  43. QList<float> m_lstRate;
  44. QList<QString> m_lstAudioDevice;
  45. };

cpp文件。 

  1. #include "showWidget.h"
  2. #include <QTimer>
  3. #include <QTime>
  4. #include <QMutex>
  5. #include <stdlib.h>
  6. #pragma execution_character_set("utf-8")
  7. struct Frame
  8. {
  9. int width;
  10. int height;
  11. uchar * pixels;
  12. QMutex mutex;
  13. };
  14. int g_frameNum = 0;
  15. static Frame *g_frame = nullptr;
  16. // 自定义视频输出模块的回调函数
  17. static void *lock(void *opaque, void **planes) {
  18. g_frame->mutex.lock();
  19. *planes = g_frame->pixels;
  20. return 0;
  21. }
  22. //保存100~110帧
  23. static void unlock(void *opaque, void *picture, void *const *planes) {
  24. // 这里可以释放视频帧的锁
  25. if (g_frameNum > 100 && g_frameNum < 110)
  26. {
  27. char *buffer = (char *)*planes; //planes即为帧数据
  28. QImage image((unsigned char*)buffer, g_frame->width, g_frame->height, QImage::Format_ARGB32);
  29. QString filePath;
  30. filePath.sprintf("./img%d.jpg", g_frameNum);
  31. image.save(filePath);
  32. }
  33. g_frameNum++;
  34. g_frame->mutex.unlock();
  35. }
  36. static void display(void *opaque, void *picture) {
  37. // 这里可以进行视频帧的显示或其他处理
  38. (void)opaque;
  39. }
  40. static unsigned setup(void **opaque, char *chroma,
  41. unsigned *width, unsigned *height,
  42. unsigned *pitches,
  43. unsigned *lines)
  44. {
  45. qDebug() << "chroma:" << QString(chroma) << "width:" << *width << ", height:" << *height;
  46. /* 开辟存放图像数据的内存块 */
  47. if (g_frame)
  48. {
  49. if (g_frame->pixels)
  50. {
  51. delete[] g_frame->pixels;
  52. g_frame->pixels = NULL;
  53. }
  54. delete g_frame;
  55. g_frame = NULL;
  56. }
  57. int w = *width;
  58. int h = *height;
  59. g_frame = new Frame;
  60. g_frame->pixels = new uchar[w * h * 4]; // 申请大小也为4通道的像素
  61. memset(g_frame->pixels, 0, w * h * 4);
  62. memcpy(chroma, "RV32", 4);
  63. g_frame->width = w;
  64. g_frame->height = h;
  65. *pitches = w * 4;
  66. *lines = h;
  67. return 1;
  68. }
  69. showWidget::showWidget(QWidget *parent)
  70. : QWidget(parent)
  71. {
  72. ui.setupUi(this);
  73. this->setWindowTitle("视频播放器");
  74. vlc_base = libvlc_new(0, NULL);
  75. ui.cbxRate->setCurrentIndex(Rate1_0X);
  76. m_lstRate << 2.0 << 1.5 << 1.25 << 1.0 << 0.75 << 0.5;
  77. ui.btnOpen->setFocusPolicy(Qt::NoFocus);
  78. ui.btnPlay->setFocusPolicy(Qt::NoFocus);
  79. ui.btnPause->setFocusPolicy(Qt::NoFocus);
  80. ui.btnStop->setFocusPolicy(Qt::NoFocus);
  81. ui.hSliderVolumn->setFocusPolicy(Qt::NoFocus);
  82. ui.cbxRate->setFocusPolicy(Qt::NoFocus);
  83. connect(ui.btnOpen, &QPushButton::clicked, this, &showWidget::slotOpenFile);
  84. connect(ui.btnPlay, &QPushButton::clicked, this, &showWidget::slotPlay);
  85. connect(ui.btnPause, &QPushButton::clicked, this, &showWidget::slotPause);
  86. connect(ui.btnStop, &QPushButton::clicked, this, &showWidget::slotStop);
  87. connect(ui.hSliderVolumn, &QSlider::valueChanged, this, &showWidget::slotValueChanged);
  88. connect(ui.cbxRate,SIGNAL(currentIndexChanged(int)), this, SLOT(slotCurrentIndexChanged(int)));
  89. }
  90. showWidget::~showWidget()
  91. {
  92. libvlc_release(vlc_base); //减少libvlc实例的引用计数,并销毁
  93. }
  94. void showWidget::slotOpenFile()
  95. {
  96. /*选择文件*/
  97. QString filename = QFileDialog::getOpenFileName(this, "选择打开的文件", "D:/", tr("*.*"));
  98. std::replace(filename.begin(), filename.end(), QChar('/'), QChar('\\'));
  99. vlc_media = libvlc_media_new_path(vlc_base, filename.toUtf8().data());
  100. if (!vlc_media) {
  101. return;
  102. }
  103. // 创建libvlc实例和媒体播放器
  104. vlc_mediaPlayer = libvlc_media_player_new_from_media(vlc_media);
  105. if (!vlc_mediaPlayer) {
  106. return;
  107. }
  108. libvlc_video_set_format_callbacks(vlc_mediaPlayer, setup, NULL);
  109. // 设置自定义视频输出
  110. libvlc_video_set_callbacks(vlc_mediaPlayer, lock, unlock, display, NULL);
  111. // 等待元数据加载完成
  112. libvlc_media_parse(vlc_media);
  113. // 获取各种元数据
  114. const char *title = libvlc_media_get_meta(vlc_media, libvlc_meta_Title);
  115. const char *artist = libvlc_media_get_meta(vlc_media, libvlc_meta_Artist);
  116. const char *album = libvlc_media_get_meta(vlc_media, libvlc_meta_Album);
  117. const char *url = libvlc_media_get_meta(vlc_media, libvlc_meta_URL);
  118. const char *date = libvlc_media_get_meta(vlc_media, libvlc_meta_Date);
  119. const char *lang = libvlc_media_get_meta(vlc_media, libvlc_meta_Language);
  120. int duration = libvlc_media_get_duration(vlc_media); // 获取时长(单位:毫秒)
  121. qDebug("Title: %s", title ? title : "N/A");
  122. qDebug("Artist: %s", artist ? artist : "N/A");
  123. qDebug("Album: %s", album ? album : "N/A");
  124. qDebug("Duration: %d ms", duration);
  125. qDebug("url: %s", url ? url : "N/A");
  126. qDebug("date: %s", date ? date : "N/A");
  127. qDebug("lang: %s", lang ? lang : "N/A");
  128. libvlc_media_track_t **tracks;
  129. int track_count = libvlc_media_tracks_get(vlc_media,&tracks);
  130. for (unsigned i = 0; i < track_count; i++)
  131. {
  132. libvlc_media_track_t* track = tracks[i];
  133. // 显示轨道信息
  134. printf("Track #%u: %s\n", i, track->psz_description);
  135. // 这里可以获取到每一个轨道的信息,比如轨道类型 track->i_type
  136. // 可能是 libvlc_track_video, libvlc_track_audio 或者 libvlc_track_text (字幕)
  137. if (track->i_type == libvlc_track_video) {
  138. // 处理视频轨道信息
  139. qDebug("width = %d",track->video->i_width);
  140. qDebug("height = %d", track->video->i_height);
  141. qDebug("rate_num = %d", track->video->i_frame_rate_num);
  142. qDebug("rate_den = %d", track->video->i_frame_rate_den);
  143. }
  144. else if (track->i_type == libvlc_track_audio) {
  145. // 处理音频轨道信息
  146. qDebug("channels = %d", track->audio->i_channels);
  147. qDebug("rate = %d", track->audio->i_rate);
  148. }
  149. else if (track->i_type == libvlc_track_text) {
  150. // 处理字幕轨道信息
  151. }
  152. }
  153. //获取事件管理器
  154. libvlc_event_manager_t *em = libvlc_media_player_event_manager(vlc_mediaPlayer);
  155. // 注册事件监听器
  156. libvlc_event_attach(em, libvlc_MediaPlayerTimeChanged, vlcEvents, this);
  157. libvlc_event_attach(em, libvlc_MediaPlayerEndReached, vlcEvents, this);
  158. libvlc_event_attach(em, libvlc_MediaPlayerStopped, vlcEvents, this);
  159. libvlc_event_attach(em, libvlc_MediaPlayerPlaying, vlcEvents, this);
  160. libvlc_event_attach(em, libvlc_MediaPlayerPaused, vlcEvents, this);
  161. QTimer::singleShot(1000, this, &showWidget::slotPlay);
  162. libvlc_video_filter_list_get(vlc_base);
  163. }
  164. void showWidget::slotPlay()
  165. {
  166. if (vlc_mediaPlayer)
  167. {
  168. libvlc_media_player_play(vlc_mediaPlayer);
  169. }
  170. }
  171. void showWidget::slotPause()
  172. {
  173. if (vlc_mediaPlayer)
  174. libvlc_media_player_pause(vlc_mediaPlayer);
  175. }
  176. void showWidget::slotStop()
  177. {
  178. if (vlc_mediaPlayer)
  179. libvlc_media_player_stop(vlc_mediaPlayer);
  180. }
  181. void showWidget::slotValueChanged(int value)
  182. {
  183. if (vlc_mediaPlayer)
  184. libvlc_audio_set_volume(vlc_mediaPlayer, value);
  185. }
  186. void showWidget::slotCurrentIndexChanged(int index)
  187. {
  188. if (vlc_mediaPlayer)
  189. libvlc_media_player_set_rate(vlc_mediaPlayer, m_lstRate[index]);
  190. }
  191. //事件回调
  192. void showWidget::vlcEvents(const libvlc_event_t *ev, void *param)
  193. {
  194. showWidget *w = (showWidget*)param;
  195. //处理不同的事件
  196. switch (ev->type) {
  197. case libvlc_MediaPlayerTimeChanged:
  198. {
  199. //qDebug() << "VLC媒体播放器时间已更改";
  200. qint64 len = libvlc_media_player_get_time(w->vlc_mediaPlayer);
  201. libvlc_time_t lenSec = len / 1000;
  202. libvlc_time_t totalLen = libvlc_media_player_get_length(w->vlc_mediaPlayer);
  203. libvlc_time_t totalLenSec = totalLen / 1000;
  204. int thh, tmm, tss;
  205. thh = lenSec / 3600;
  206. tmm = (lenSec % 3600) / 60;
  207. tss = (lenSec % 60);
  208. QTime time(thh, tmm, tss);
  209. w->ui.lbCurTime->setText(time.toString("hh:mm:ss"));
  210. thh = totalLenSec / 3600;
  211. tmm = (totalLenSec % 3600) / 60;
  212. tss = (totalLenSec % 60);
  213. QTime TotalTime(thh, tmm, tss);
  214. w->ui.lbTotalTime->setText(TotalTime.toString("hh:mm:ss"));
  215. double pos = (double)lenSec / totalLenSec * 100;
  216. w->ui.horizontalSlider->setValue(pos);
  217. }
  218. break;
  219. case libvlc_MediaPlayerEndReached:
  220. qDebug() << "VLC播放完毕.";
  221. break;
  222. case libvlc_MediaPlayerStopped:
  223. qDebug() << "VLC停止播放";
  224. break;
  225. case libvlc_MediaPlayerPlaying:
  226. qDebug() << "VLC开始播放";
  227. break;
  228. case libvlc_MediaPlayerPaused:
  229. qDebug() << "VLC暂停播放";
  230. break;
  231. }
  232. }

更多参考:

libVLC 事件机制-CSDN博客

libVLC windows开发环境搭建-CSDN博客

libVLC 元数据-CSDN博客

libVLC 添加图片和文本水印-CSDN博客

libVLC 音频输出设备切换-CSDN博客

libVLC 音频立体声模式切换-CSDN博客

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/weixin_40725706/article/detail/430195
推荐阅读
相关标签
  

闽ICP备14008679号