【三十八】【QT开发应用】vlcplayer视频播放器(一)实现视频播放,视频暂停,视频停止,进度条调节,音量调节,时长显示功能

ops/2024/11/1 5:38:28/

效果展示

vlcplayer_test视频播放器

MainWidget.ui

在这里插入图片描述

  • 注意控件的布局和命名,控件的命名和信号与槽函数的绑定有关,所以这点很重要。

下载VLC组件和环境配置

  • videolan下载地址
  • 我下载的是vlc-3.0.8-win64版本.
    在这里插入图片描述
    在这里插入图片描述
  • 将下载的文件复制粘贴到项目文件中.
    在这里插入图片描述
    在这里插入图片描述
  • 复制粘贴这三个文件到项目的debug文件夹里面.
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
  • 然后我们在主文件里面添加#include <vlc/vlc.h>头文件.

MainWidget.h

#pragma once#include <QtWidgets>
#include "ui_MainWidget.h"
#include <vlc/vlc.h>class MainWidget : public QWidget
{Q_OBJECTpublic:MainWidget(QWidget *parent = nullptr);~MainWidget();libvlc_media_player_t* getMediaPlayer();//获取media player对象qint64 getDuration();//获取总时长,单位是秒void setTimeSliderPos(int value);//设置时间滑动条的值void setVolumeSliderPos(int value);//设置音量滑动条的值void setTimeText(QString str);//设置时间标签文本private slots:void on_btnOpen_clicked();//打开按钮槽函数void on_btnPlay_clicked();//播放按钮槽函数void on_btnPause_clicked();//暂停按钮槽函数void on_btnStop_clicked();//停止按钮槽函数void onTimeSliderMoved(int value);//时间滑动条移动槽函数void onVolumeSliderMoved(int value);//音量滑动条移动槽函数
private:Ui::MainWidgetClass ui;libvlc_instance_t* m_pInstance = nullptr;//视频示例,负责管理全局配置和资源libvlc_media_player_t* m_pMediaPlayer = nullptr;//VLC 媒体播放器的指针,用于控制媒体的播放、暂停、停止等操作libvlc_media_t* m_pMedia = nullptr;// VLC 媒体的指针,用于加载和管理特定的媒体内容(如音频或视频文件)libvlc_event_manager_t* m_pEvent_manager = nullptr;//事件管理器的指针,负责处理 VLC 播放器的各种事件(例如媒体播放完毕、音量变化等)qint64 m_totalSecs = -1;//总时长,单位是秒
};
  • 我们利用vlc库去实现视频播放器,那么我们就需要按照他人的要求去编写代码,因为vlc库是别人写出来的东西,而他人的要求就是需要我们去定义示例,媒体播放器,媒体,事件等对象,并按照要求去使用.

MainWidget.cpp

#include "MainWidget.h"void vlc_callback(const struct libvlc_event_t* p_event, void* p_data) {MainWidget* pMain = static_cast<MainWidget*>(p_data);if (pMain) {switch (p_event->type) {case libvlc_MediaPlayerPositionChanged:{float pos = libvlc_media_player_get_position(pMain->getMediaPlayer());pMain->setTimeSliderPos(pos * 100);qint64 curSecs = libvlc_media_player_get_time(pMain->getMediaPlayer()) / 1000;int curH = curSecs / 3600;int curMinute = (curSecs - curH * 3600) / 60;int curSec = curSecs - curH * 3600 - curMinute * 60;QString str1 = QString("%1:%2:%3").arg(curH, 2, 10, QChar('0')).arg(curMinute, 2, 10, QChar('0')).arg(curSec, 2, 10, QChar('0'));qint64 totalSecs = pMain->getDuration();int totalH = totalSecs / 3600;int totalMinute = (totalSecs - totalH * 3600) / 60;int totalSec = totalSecs - totalH * 3600 - totalMinute * 60;QString str2 = QString("%1:%2:%3").arg(totalH, 2, 10, QChar('0')).arg(totalMinute, 2, 10, QChar('0')).arg(totalSec, 2, 10, QChar('0'));QString text = str1 + "/" + str2;pMain->setTimeText(text);break;}case libvlc_MediaPlayerAudioVolume:{int volume = libvlc_audio_get_volume(pMain->getMediaPlayer());pMain->setVolumeSliderPos(volume);break;}}}
}MainWidget::MainWidget(QWidget* parent): QWidget(parent) {ui.setupUi(this);m_pInstance = libvlc_new(0, nullptr);if (m_pInstance) {m_pMediaPlayer = libvlc_media_player_new(m_pInstance);if (m_pMediaPlayer) {m_pEvent_manager = libvlc_media_player_event_manager(m_pMediaPlayer);libvlc_event_attach(m_pEvent_manager, libvlc_MediaPlayerPositionChanged, vlc_callback, this);libvlc_event_attach(m_pEvent_manager, libvlc_MediaPlayerAudioVolume, vlc_callback, this);} else {libvlc_release(m_pInstance);QMessageBox::information(this, "提示", "libvlc_media_player_new failed");exit(EXIT_FAILURE);}} else {QMessageBox::information(this, "提示", "libvlc_new failed");exit(EXIT_FAILURE);}ui.label_time->setText("");connect(ui.timeSlider, &QSlider::sliderMoved, this, &MainWidget::onTimeSliderMoved);connect(ui.volumeSlider, &QSlider::sliderMoved, this, &MainWidget::onVolumeSliderMoved);
}MainWidget::~MainWidget() {if (m_pMediaPlayer) {libvlc_media_player_release(m_pMediaPlayer);m_pMediaPlayer = nullptr;}if (m_pInstance) {libvlc_release(m_pInstance);m_pInstance = nullptr;}}libvlc_media_player_t* MainWidget::getMediaPlayer() {return m_pMediaPlayer;
}qint64 MainWidget::getDuration() {return m_totalSecs;
}void MainWidget::setTimeSliderPos(int value) {ui.timeSlider->setValue(value);
}void MainWidget::setVolumeSliderPos(int value) {ui.volumeSlider->setValue(value);
}void MainWidget::setTimeText(QString str) {ui.label_time->setText(str);
}void MainWidget::on_btnOpen_clicked() {QString filename = QFileDialog::getOpenFileName(this, "请选择视频文件", "D:","视频文件(*.mp4 *.flv);;所有文件(*.*);;");if (filename.isEmpty()) return;filename = QDir::toNativeSeparators(filename);m_pMedia = libvlc_media_new_path(m_pInstance, filename.toStdString().c_str());if (m_pMedia) {libvlc_media_parse(m_pMedia);libvlc_media_player_set_media(m_pMediaPlayer, m_pMedia);libvlc_media_player_set_hwnd(m_pMediaPlayer, (void*)(ui.video_widget->winId()));libvlc_media_release(m_pMedia);m_totalSecs = libvlc_media_get_duration(m_pMedia) / 1000;m_pMedia = nullptr;libvlc_media_player_play(m_pMediaPlayer);} else {if (m_pMediaPlayer) {libvlc_media_player_release(m_pMediaPlayer);m_pMediaPlayer = nullptr;}if (m_pInstance) {libvlc_release(m_pInstance);m_pInstance = nullptr;}QMessageBox::information(this, "提示", "libvlc_media_new_path failed");exit(EXIT_FAILURE);}}void MainWidget::on_btnPlay_clicked() {if (libvlc_media_player_get_state(m_pMediaPlayer) == libvlc_state_t::libvlc_Paused|| libvlc_media_player_get_state(m_pMediaPlayer) == libvlc_state_t::libvlc_Stopped) {libvlc_media_player_play(m_pMediaPlayer);}
}void MainWidget::on_btnPause_clicked() {if (libvlc_media_player_get_state(m_pMediaPlayer) == libvlc_state_t::libvlc_Playing) {libvlc_media_player_pause(m_pMediaPlayer);}
}void MainWidget::on_btnStop_clicked() {if (libvlc_media_player_get_state(m_pMediaPlayer) == libvlc_state_t::libvlc_Playing|| libvlc_media_player_get_state(m_pMediaPlayer) == libvlc_state_t::libvlc_Paused) {libvlc_media_player_stop(m_pMediaPlayer);}
}void MainWidget::onTimeSliderMoved(int value) {libvlc_media_player_set_position(m_pMediaPlayer, value / 100.0);
}void MainWidget::onVolumeSliderMoved(int value) {libvlc_audio_set_volume(m_pMediaPlayer, value);
}

MainWidget

  • m_pInstance = libvlc_new(0, nullptr);利用libvlc_new函数去创建示例对象.
  • m_pMediaPlayer = libvlc_media_player_new(m_pInstance);如果实例创建成功,就创建视频媒体对象,利用libvlc_media_player_new()函数创建对象.
  • m_pEvent_manager = libvlc_media_player_event_manager(m_pMediaPlayer);如果视频媒体对象创建成功,就创建事件管理器的对象,利用libvlc_media_player_event_manager()函数创建对象.
  • libvlc_event_attach(m_pEvent_manager, libvlc_MediaPlayerPositionChanged, vlc_callback, this);视频媒体位置改变了就会发送信号到回调函数vlc_callback也就是说只要视频在播放,那么就会持续不断的发送信号到回调函数`vlc_callback中.
  • libvlc_event_attach(m_pEvent_manager, libvlc_MediaPlayerAudioVolume, vlc_callback, this);视频声音发生改变的时候触发的事件,发送信号到回调函数vlc_callback,这段代码的意义是保证程序一进去的时候音量和滑动条是对应的.
  • connect(ui.timeSlider, &QSlider::sliderMoved, this, &MainWidget::onTimeSliderMoved);时间滑动条被拖动的时候发送信号触发槽函数.
  • connect(ui.volumeSlider, &QSlider::sliderMoved, this, &MainWidget::onVolumeSliderMoved);音量滑动条被拖动的时候发送信号触发槽函数.

vlc_callback

  • void vlc_callback(const struct libvlc_event_t* p_event, void* p_data) 这是回调函数的固定写法.p_event 是指向 libvlc_event_t 类型的常量指针,表示发生的具体事件.在回调函数中,通常将 p_data 转换为具体类型(例如指向 MainWidget 的指针),以便在回调中访问类成员或调用类方法。
  • MainWidget* pMain = static_cast<MainWidget*>(p_data);p_date进行安全的类型转换,转换成MainWidget类型,这样我们就可以通过pMain直接访问主窗口类里面的对象成员.
  • 利用switch语句对传递进来的不同事件进行不同的代码处理,我们只编辑了两个具体的事件,一个是libvlc_MediaPlayerPositionChanged事件,视频只要在播放就会一直发送信号,一个是libvlc_MediaPlayerAudioVolume,视频的音量改变了就会发送信号,用于进入程序时将音量的滑动条和音量进行绑定.
  • libvlc_MediaPlayerAudioVolume内部代码,
    int volume = libvlc_audio_get_volume(pMain->getMediaPlayer());获取VLC播放器的音量,volume 的值通常在 0 到 100 之间,表示音量的百分比.然后
    pMain->setVolumeSliderPos(volume);设置这个音量到音量滑动条中.也就是说我们一进入程序获取音量然后设置音量滑动条的值.之后我们手动改变音量滑动条视频音量和滑动条通过槽函数一直都是绑定在一起的.
  • libvlc_MediaPlayerPositionChanged内部代码,
    float pos = libvlc_media_player_get_position(pMain->getMediaPlayer());获取视频当前进度占总进度的百分比值,返回值时一个0~1的float类型的值,
    pMain->setTimeSliderPos(pos * 100);将返回值乘以100得到的是具体的值,因为事件滑动条默认是0~100,这样我们直接设置乘以100的值为此时滑动条的值就按照百分比绑定了当前视频进度和滑动条位置.
  • qint64 curSecs = libvlc_media_player_get_time(pMain->getMediaPlayer()) / 1000;获取当前视频的时间,单位是毫秒,转化为秒需要除以1000.然后按照计算得到此时对应的时分秒.将时分秒转化为对应的字符串,然后计算总时间的时分秒转化为字符串拼接在一起,设置为Label的text值.

on_btnOpen_clicked

  • 打开按钮需要实现的效果是,点击之后需要有一个弹窗,选择我们需要播放的视频文件,然后播放.
  • getOpenFileName函数实现弹窗选择视频文件的功能,toNativeSeparators函数将得到的路径转换为正确的格式路径,因为得到的路径 \和/可能不符合当前程序QT里面的规则,用这个函数可以进行转化统一.
  • m_pMedia = libvlc_media_new_path(m_pInstance, filename.toStdString().c_str());得到视频的路径之后创建媒体的对象.
  • libvlc_media_parse(m_pMedia);解析媒体文件,以便从文件中提取元数据信息(如时长、标题、格式等)。
    libvlc_media_player_set_media(m_pMediaPlayer, m_pMedia);将媒体 m_pMedia 设置到媒体播放器 m_pMediaPlayer 中。此操作指示播放器应使用该媒体文件作为播放内容。此时 m_pMedia 已经和播放器绑定,不再需要手动管理其生命周期。
    libvlc_media_player_set_hwnd(m_pMediaPlayer, (void*)(ui.video_widget->winId()));设置视频输出窗口的句柄,使播放器的视频内容渲染在指定的窗口控件上。此处 ui.video_widget->winId() 获取了窗口句柄,将其转换为 void* 类型。
  • m_totalSecs = libvlc_media_get_duration(m_pMedia) / 1000;在媒体进行销毁之前计算出视频总时长,得到的时间是毫秒单位需要除以1000转化为秒.

on_btnPlay_clicked等其他按钮

  • ibvlc_media_player_get_state(m_pMediaPlayer)得到当前的状态,libvlc_state_t::libvlc_Paused暂停状态,libvlc_state_t::libvlc_Stopped停止状态,libvlc_state_t::libvlc_Playing播放状态,
  • libvlc_media_player_play(m_pMediaPlayer);播放视频媒体,libvlc_media_player_pause(m_pMediaPlayer);暂停视频媒体,libvlc_media_player_stop(m_pMediaPlayer);停止视频媒体

杂项

  • libvlc_media_player_set_position设置视频媒体时长位置,
    value / 100.0:value 通常是从进度滑块控件中获取的整数值,范围通常是 0 到 100。通过除以 100.0,将其转换为 0.0 到 1.0 之间的小数(即百分比),符合 libvlc 所需的参数范围。0.0 表示播放开始位置,1.0 表示播放结束位置。
  • libvlc_audio_set_volume(m_pMediaPlayer, value);
    value:要设置的音量值,通常范围是 0 到 100,0 表示静音;100 表示最大音量(也可以设置大于 100,视 VLC 配置可能允许更高的音量,但通常建议限制在 0-100 范围内)。

在这里插入图片描述

结尾

最后,感谢您阅读我的文章,希望这些内容能够对您有所启发和帮助。如果您有任何问题或想要分享您的观点,请随时在评论区留言。
同时,不要忘记订阅我的博客以获取更多有趣的内容。在未来的文章中,我将继续探讨这个话题的不同方面,为您呈现更多深度和见解。
谢谢您的支持,期待与您在下一篇文章中再次相遇!


http://www.ppmy.cn/ops/130051.html

相关文章

尚硅谷-react教程-求和案例-数据共享(下篇)-完成数据共享-笔记

#1024程序员节&#xff5c;征文# public/index.html <!DOCTYPE html> <html><head><meta charset"UTF-8"><title>redux</title></head><body><div id"root"></div></body> </html&…

Jetson Xavier nx在Ubuntu18.04下安装ros2 使用奥比中光330

安装ROS2过程 添加ros2 软件源 sudo sh -c ‘echo “deb [arch=amd64] http://packages.ros.org/ros2/ubuntu bionic main” > /etc/apt/sources.list.d/ros2-latest.list’ sudo apt install curl gnupg2 lsb-release curl http://repo.ros2.org/repos.key | sudo apt-key…

Golang | Leetcode Golang题解之523题连续的子数组和

题目&#xff1a; 题解&#xff1a; func checkSubarraySum(nums []int, k int) bool {m : len(nums)if m < 2 {return false}mp : map[int]int{0: -1}remainder : 0for i, num : range nums {remainder (remainder num) % kif prevIndex, has : mp[remainder]; has {if …

神经网络:解析人工智能的智慧基石

神经网络&#xff1a;解析人工智能的智慧基石 一、引言 在当今科技飞速发展的时代&#xff0c;人工智能已经成为了一个备受关注的领域。而神经网络作为人工智能的重要组成部分&#xff0c;正逐渐改变着我们的生活和未来。那么&#xff0c;什么是神经网络呢&#xff1f;它又是…

Redis-概念、安装、基本配置

文章目录 一、Redis及Redis集群概念、分布式系统概念一-1 Redis是什么一-2 什么是分布式系统、分布式缓存一-3 什么是Redis集群、实现Redis集群的方法有哪些、这些跟Redis的sentinel和cluster有什么关系一-4 Redis的库一-5 Redis中的Key与Value是什么、如何进行操作使用它们添加…

EnrichmentMap 该怎么用?方法和流程的建立过程

愿武艺晴小朋友一定得每天都开心 1&#xff09;首先在&#xff1a;g:Profiler – a web server for functional enrichment analysis and conversions of gene lists 的网站上输入Degs的基因列表。 具体步骤如下&#xff1a; 2&#xff09;准备富集出来的通路们&#xff08;…

鸿蒙网络编程系列38-Web组件文件下载示例

1. web组件文件下载能力简介 在本系列的第22篇文章&#xff0c;介绍了web组件的文件上传能力&#xff0c;同样的&#xff0c;web组件也具备文件下载能力&#xff0c;鸿蒙API提供了处理web组件下载事件的委托类型WebDownloadDelegate&#xff0c;该类型包括四个下载事件的回调接…

基于SpringBoot的酒店管理系统的设计与实现

摘要 酒店管理系统的设计旨在提供快捷高效的住宿资源管理方案&#xff0c;帮助管理员实现对酒店内房间、客户信息、订单的全方位管理&#xff0c;同时为用户提供便捷的预订和查询功能。本系统基于Spring Boot框架&#xff0c;结合前端框架和数据库设计&#xff0c;构建一个用户…