赞
踩
在前面的文章中,我们使用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函数声明。
- LIBVLC_API
- void libvlc_video_set_callbacks( libvlc_media_player_t *mp,
- libvlc_video_lock_cb lock,
- libvlc_video_unlock_cb unlock,
- libvlc_video_display_cb display,
- void *opaque );
以下是libvlc_video_lock_cb声明。
typedef void *(*libvlc_video_lock_cb)(void *opaque, void **planes);
如果应用程序需要在视频渲染前对视频帧进行一些处理,那么可以在libvlc_video_lock_cb中进行这些处理,并将处理后的帧数据地址赋值给 *planes。
以下是libvlc_video_unlock_cb声明。
- typedef void (*libvlc_video_unlock_cb)(void *opaque, void *picture,
- void *const *planes);
在打开一个新的视频文件的时候,我们不知道视频的宽和高等数据。这时候需要使用libvlc_video_set_format_callbacks来获取视频格式,函数声明如下。
- LIBVLC_API
- void libvlc_video_set_format_callbacks( libvlc_media_player_t *mp,
- libvlc_video_format_cb setup,
- libvlc_video_cleanup_cb cleanup );
以下是libvlc_video_format_cb声明,用于在视频解码开始之前设置视频像素格式和缓冲区配置。
- typedef unsigned (*libvlc_video_format_cb)(void **opaque, char *chroma,
- unsigned *width, unsigned *height,
- unsigned *pitches,
- unsigned *lines);
代码示例:保存指定的数据帧。
头文件。
- #pragma once
-
- #include <QtWidgets/QWidget>
- #include "ui_showWidget.h"
- #include <QMenu>
- #include <QActionGroup>
- #include <vlc/vlc.h>
- #include <QDebug>
- #include <QFileDialog>
- #include <QThread>
- #include <QMouseEvent>
- #include <QKeyEvent>
-
- enum Rate
- {
- Rate2X,
- Rate1_5X,
- Rate1_25X,
- Rate1_0X,
- Rate0_75X,
- Rate0_5X
- };
-
- class showWidget : public QWidget
- {
- Q_OBJECT
-
- public:
- showWidget(QWidget *parent = nullptr);
- ~showWidget();
-
- private slots:
- void slotOpenFile();
- void slotPlay();
- void slotPause();
- void slotStop();
- void slotValueChanged(int value);
- void slotCurrentIndexChanged(int index);
-
- private:
- //事件处理回调
- static void vlcEvents(const libvlc_event_t *ev, void *param);
-
- private:
- Ui::showWidgetClass ui;
-
- private:
- libvlc_instance_t *vlc_base = nullptr;
- libvlc_media_t *vlc_media = nullptr;
- libvlc_media_player_t *vlc_mediaPlayer = nullptr;
-
- QList<float> m_lstRate;
- QList<QString> m_lstAudioDevice;
- };

cpp文件。
- #include "showWidget.h"
- #include <QTimer>
- #include <QTime>
- #include <QMutex>
- #include <stdlib.h>
-
- #pragma execution_character_set("utf-8")
-
- struct Frame
- {
- int width;
- int height;
- uchar * pixels;
- QMutex mutex;
- };
-
-
- int g_frameNum = 0;
- static Frame *g_frame = nullptr;
-
- // 自定义视频输出模块的回调函数
- static void *lock(void *opaque, void **planes) {
- g_frame->mutex.lock();
- *planes = g_frame->pixels;
-
- return 0;
- }
-
- //保存100~110帧
- static void unlock(void *opaque, void *picture, void *const *planes) {
- // 这里可以释放视频帧的锁
- if (g_frameNum > 100 && g_frameNum < 110)
- {
- char *buffer = (char *)*planes; //planes即为帧数据
- QImage image((unsigned char*)buffer, g_frame->width, g_frame->height, QImage::Format_ARGB32);
- QString filePath;
- filePath.sprintf("./img%d.jpg", g_frameNum);
-
- image.save(filePath);
- }
- g_frameNum++;
- g_frame->mutex.unlock();
- }
-
- static void display(void *opaque, void *picture) {
- // 这里可以进行视频帧的显示或其他处理
- (void)opaque;
- }
-
-
- static unsigned setup(void **opaque, char *chroma,
- unsigned *width, unsigned *height,
- unsigned *pitches,
- unsigned *lines)
- {
- qDebug() << "chroma:" << QString(chroma) << "width:" << *width << ", height:" << *height;
-
- /* 开辟存放图像数据的内存块 */
- if (g_frame)
- {
- if (g_frame->pixels)
- {
- delete[] g_frame->pixels;
- g_frame->pixels = NULL;
- }
-
- delete g_frame;
- g_frame = NULL;
- }
-
- int w = *width;
- int h = *height;
- g_frame = new Frame;
- g_frame->pixels = new uchar[w * h * 4]; // 申请大小也为4通道的像素
-
- memset(g_frame->pixels, 0, w * h * 4);
- memcpy(chroma, "RV32", 4);
- g_frame->width = w;
- g_frame->height = h;
- *pitches = w * 4;
- *lines = h;
-
- return 1;
- }
-
- showWidget::showWidget(QWidget *parent)
- : QWidget(parent)
- {
- ui.setupUi(this);
-
- this->setWindowTitle("视频播放器");
-
- vlc_base = libvlc_new(0, NULL);
-
- ui.cbxRate->setCurrentIndex(Rate1_0X);
-
- m_lstRate << 2.0 << 1.5 << 1.25 << 1.0 << 0.75 << 0.5;
-
- ui.btnOpen->setFocusPolicy(Qt::NoFocus);
- ui.btnPlay->setFocusPolicy(Qt::NoFocus);
- ui.btnPause->setFocusPolicy(Qt::NoFocus);
- ui.btnStop->setFocusPolicy(Qt::NoFocus);
- ui.hSliderVolumn->setFocusPolicy(Qt::NoFocus);
- ui.cbxRate->setFocusPolicy(Qt::NoFocus);
-
- connect(ui.btnOpen, &QPushButton::clicked, this, &showWidget::slotOpenFile);
- connect(ui.btnPlay, &QPushButton::clicked, this, &showWidget::slotPlay);
- connect(ui.btnPause, &QPushButton::clicked, this, &showWidget::slotPause);
- connect(ui.btnStop, &QPushButton::clicked, this, &showWidget::slotStop);
- connect(ui.hSliderVolumn, &QSlider::valueChanged, this, &showWidget::slotValueChanged);
- connect(ui.cbxRate,SIGNAL(currentIndexChanged(int)), this, SLOT(slotCurrentIndexChanged(int)));
- }
-
- showWidget::~showWidget()
- {
- libvlc_release(vlc_base); //减少libvlc实例的引用计数,并销毁
- }
-
-
- void showWidget::slotOpenFile()
- {
- /*选择文件*/
- QString filename = QFileDialog::getOpenFileName(this, "选择打开的文件", "D:/", tr("*.*"));
- std::replace(filename.begin(), filename.end(), QChar('/'), QChar('\\'));
-
- vlc_media = libvlc_media_new_path(vlc_base, filename.toUtf8().data());
- if (!vlc_media) {
- return;
- }
-
- // 创建libvlc实例和媒体播放器
- vlc_mediaPlayer = libvlc_media_player_new_from_media(vlc_media);
- if (!vlc_mediaPlayer) {
- return;
- }
-
- libvlc_video_set_format_callbacks(vlc_mediaPlayer, setup, NULL);
-
- // 设置自定义视频输出
- libvlc_video_set_callbacks(vlc_mediaPlayer, lock, unlock, display, NULL);
-
- // 等待元数据加载完成
- libvlc_media_parse(vlc_media);
-
- // 获取各种元数据
- const char *title = libvlc_media_get_meta(vlc_media, libvlc_meta_Title);
- const char *artist = libvlc_media_get_meta(vlc_media, libvlc_meta_Artist);
- const char *album = libvlc_media_get_meta(vlc_media, libvlc_meta_Album);
- const char *url = libvlc_media_get_meta(vlc_media, libvlc_meta_URL);
- const char *date = libvlc_media_get_meta(vlc_media, libvlc_meta_Date);
- const char *lang = libvlc_media_get_meta(vlc_media, libvlc_meta_Language);
- int duration = libvlc_media_get_duration(vlc_media); // 获取时长(单位:毫秒)
-
- qDebug("Title: %s", title ? title : "N/A");
- qDebug("Artist: %s", artist ? artist : "N/A");
- qDebug("Album: %s", album ? album : "N/A");
- qDebug("Duration: %d ms", duration);
- qDebug("url: %s", url ? url : "N/A");
- qDebug("date: %s", date ? date : "N/A");
- qDebug("lang: %s", lang ? lang : "N/A");
-
- libvlc_media_track_t **tracks;
- int track_count = libvlc_media_tracks_get(vlc_media,&tracks);
- for (unsigned i = 0; i < track_count; i++)
- {
- libvlc_media_track_t* track = tracks[i];
-
- // 显示轨道信息
- printf("Track #%u: %s\n", i, track->psz_description);
-
- // 这里可以获取到每一个轨道的信息,比如轨道类型 track->i_type
- // 可能是 libvlc_track_video, libvlc_track_audio 或者 libvlc_track_text (字幕)
-
- if (track->i_type == libvlc_track_video) {
- // 处理视频轨道信息
- qDebug("width = %d",track->video->i_width);
- qDebug("height = %d", track->video->i_height);
- qDebug("rate_num = %d", track->video->i_frame_rate_num);
- qDebug("rate_den = %d", track->video->i_frame_rate_den);
- }
- else if (track->i_type == libvlc_track_audio) {
- // 处理音频轨道信息
- qDebug("channels = %d", track->audio->i_channels);
- qDebug("rate = %d", track->audio->i_rate);
- }
- else if (track->i_type == libvlc_track_text) {
- // 处理字幕轨道信息
- }
- }
-
- //获取事件管理器
- libvlc_event_manager_t *em = libvlc_media_player_event_manager(vlc_mediaPlayer);
-
- // 注册事件监听器
- libvlc_event_attach(em, libvlc_MediaPlayerTimeChanged, vlcEvents, this);
- libvlc_event_attach(em, libvlc_MediaPlayerEndReached, vlcEvents, this);
- libvlc_event_attach(em, libvlc_MediaPlayerStopped, vlcEvents, this);
- libvlc_event_attach(em, libvlc_MediaPlayerPlaying, vlcEvents, this);
- libvlc_event_attach(em, libvlc_MediaPlayerPaused, vlcEvents, this);
-
- QTimer::singleShot(1000, this, &showWidget::slotPlay);
- libvlc_video_filter_list_get(vlc_base);
- }
-
- void showWidget::slotPlay()
- {
- if (vlc_mediaPlayer)
- {
- libvlc_media_player_play(vlc_mediaPlayer);
- }
- }
-
- void showWidget::slotPause()
- {
- if (vlc_mediaPlayer)
- libvlc_media_player_pause(vlc_mediaPlayer);
- }
-
- void showWidget::slotStop()
- {
- if (vlc_mediaPlayer)
- libvlc_media_player_stop(vlc_mediaPlayer);
- }
-
- void showWidget::slotValueChanged(int value)
- {
- if (vlc_mediaPlayer)
- libvlc_audio_set_volume(vlc_mediaPlayer, value);
- }
-
- void showWidget::slotCurrentIndexChanged(int index)
- {
- if (vlc_mediaPlayer)
- libvlc_media_player_set_rate(vlc_mediaPlayer, m_lstRate[index]);
- }
-
- //事件回调
- void showWidget::vlcEvents(const libvlc_event_t *ev, void *param)
- {
- showWidget *w = (showWidget*)param;
- //处理不同的事件
- switch (ev->type) {
- case libvlc_MediaPlayerTimeChanged:
- {
- //qDebug() << "VLC媒体播放器时间已更改";
- qint64 len = libvlc_media_player_get_time(w->vlc_mediaPlayer);
- libvlc_time_t lenSec = len / 1000;
-
- libvlc_time_t totalLen = libvlc_media_player_get_length(w->vlc_mediaPlayer);
- libvlc_time_t totalLenSec = totalLen / 1000;
-
- int thh, tmm, tss;
- thh = lenSec / 3600;
- tmm = (lenSec % 3600) / 60;
- tss = (lenSec % 60);
- QTime time(thh, tmm, tss);
- w->ui.lbCurTime->setText(time.toString("hh:mm:ss"));
-
- thh = totalLenSec / 3600;
- tmm = (totalLenSec % 3600) / 60;
- tss = (totalLenSec % 60);
- QTime TotalTime(thh, tmm, tss);
- w->ui.lbTotalTime->setText(TotalTime.toString("hh:mm:ss"));
-
- double pos = (double)lenSec / totalLenSec * 100;
- w->ui.horizontalSlider->setValue(pos);
- }
- break;
- case libvlc_MediaPlayerEndReached:
- qDebug() << "VLC播放完毕.";
- break;
- case libvlc_MediaPlayerStopped:
- qDebug() << "VLC停止播放";
- break;
- case libvlc_MediaPlayerPlaying:
- qDebug() << "VLC开始播放";
- break;
- case libvlc_MediaPlayerPaused:
- qDebug() << "VLC暂停播放";
- break;
- }
- }

更多参考:
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。