Qt5开发视频播放器

一、播放器界面UI设计

控件对象名位置(坐标点)对象名称组件名称备注
Widget(0, 0, 809, 572)WidgetQWidget
labellabelQLabel播放窗口
label_2label_2QLabel
voice_controlvoice_controlQSlider音量滑动条
btn_openbtn_openQPushButton打开文件按钮
label_4label_4QLabel
label_ctlabel_ctQLabel
progress_reportprogress_reportMySlider当前播放进度标签
label_all_timelabel_all_timeQLabel视频时长标签
btn_htbtn_htQPushButton后退按钮
btn_qjbtn_qjQPushButton前进按钮
btn_startbtn_startQPushButton播放按钮 / 暂停按钮
btn_fullshowbtn_fullshowQPushButton全屏播放按钮
label_5label_5QLabel显示播放列表文本
listWidgetlistWidgetQListWidget播放列表

效果图如下
在这里插入图片描述

二、播放器基本实现

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>
#include <QMediaPlayer>  //播放器
#include <QVideoWidget>  //播放窗口
#include <QMediaPlaylist>//播放列表
#include <QString>
#include <QFileDialog>//文件窗口
#include <QDebug>
#include <QResizeEvent>
#include <QMouseEvent>
#include <QKeyEvent>
#include <QListWidgetItem>
#include <QObject>
#include <QAbstractEventDispatcher>#include "globalkeyfilter.h"namespace Ui
{
class Widget;
}class QMediaPlayer;
class QMediaPlaylist;
class QVideoWidget;class Widget : public QWidget
{Q_OBJECTpublic:explicit Widget(QWidget* parent = 0);void setget_Alltime(qint64 playtime);//获取视频时长并设置到标签void setget_currenttime(qint64 playtime);//获取当时播放位置并设置void settimeslider(qint64 playtime);//设置进度条void exitFullScreen();  // 退出全屏void togglePlayback();  // 暂停/播放~Widget();private slots:void on_btn_open_clicked();//打开音视频void on_btn_start_clicked();//播放void on_voice_control_valueChanged(int value);//控制声音void on_btn_fullshow_clicked();//全屏显示void getduration(qint64 playtime);   //获取时间改变信号void getposition(qint64 playtime);   //获取当前播放位置void on_btn_ht_clicked();//上一个音视频void on_btn_qj_clicked();//下一个音视频void on_listWidget_doubleClicked(const QModelIndex& index);private:Ui::Widget* ui;QStringList         m_strlist;          //文件名列表QMediaPlayer*       m_pMediaPlayer;     //播放器QVideoWidget*       m_pVideoWidget;     //播放窗口QMediaPlaylist*     m_MediaPlaylist;    //播放列表QMediaPlayer::State m_state;            //播放状态int m_hour;                             //视频总时长int m_munete;                           //分钟int m_second;                           //秒bool m_sliderstate;                     //滑块是否被选中的状态int m_slider_crtval;                    //滑块当前值GlobalKeyFilter* m_globalKeyFilter;     //  GlobalKeyFilter 类
};#endif // WIDGET_H

将label组件传入QVideoWidget作为实例化参数,即可获得播放窗口,并通过QMediaPlayer调用setVideoOutput函数,指定为播放器窗口。

三、自定义滑块控件

#ifndef MYSLIDER_H
#define MYSLIDER_H#include <QMouseEvent>
#include <QSlider>class MySlider : public QSlider
{Q_OBJECT
public:explicit MySlider(QWidget* parent = nullptr);
protected:void mousePressEvent(QMouseEvent* ev);void mouseReleaseEvent(QMouseEvent* ev);
signals:void mousepress(int value);void mouserelease(int value);};#endif // MYSLIDER_H

由于进度条数值和视频时长的单位不统一,需要自定义控件MySlider,监听键盘事件,发送计算得到的视频时长数据到Widget类的匿名槽函数中,完成滑动条功能。

四、全局键盘监听事件

创建一个GlobalKeyFilter类,对widget类向前声明,实现全局的键盘监听。

#include "globalkeyfilter.h"
#include "widget.h"
#include <QKeyEvent>
#include <Windows.h>GlobalKeyFilter::GlobalKeyFilter(Widget* widget) : m_widget(widget)
{qDebug() << "键盘监听事件启动了" << "\n";
}bool GlobalKeyFilter::nativeEventFilter(const QByteArray& eventType, void* message, long* result)
{if (eventType == "windows_generic_MSG"){MSG* msg = static_cast<MSG*>(message);if (msg->message == WM_KEYDOWN){int key = msg->wParam;// 处理 ESC 键if (key == VK_ESCAPE){if (m_widget){m_widget->exitFullScreen();}}// 处理空格键if (key == VK_SPACE){m_widget->togglePlayback();}}}// 返回 false 以便继续传递事件return false;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/33511.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在vue3中定义组件的5种方式

在vue3中定义组件的5种方式 Vue 正在不断发展&#xff0c;目前在 Vue3 中定义组件的方法有多种。从选项式到组合式再到类API&#xff0c;情况截然不同。本文将会定义一个简单的组件并使用所有可用的方法重构它。 选项式 这是在 Vue 中声明组件的最常见方法。从 Vue1 就开始存…

[RISCV] Generate Debian distribution

获取keyring $ wget http://mirrors.ustc.edu.cn/debian/pool/main/d/debian-ports-archive-keyring/debian-ports-archive-keyring_2023.02.01~deb11u1_all.deb安装keyring $ sudo dpkg -i debian-ports-archive-keyring_2023.02.01~deb11u1_all.deb安装debootstrap $ sudo…

ThreadLLocal的学习

ThreadLocal的学习 ThreadLocal的学习1.ThreadLocal是什么&#xff1f;2.ThreadLocal的数据结构Java的四种引用类型 3.ThreadLocal为什么会出现内存泄露&#xff1f;既然会出现内存泄露为什么Entry的key还要使用弱引用&#xff1f;如何避免内存泄露&#xff1f; ThreadLocal的学…

最新版本2023UI千月影视APP源码 开源完美版前后端完美匹配 后端基于ThinkPHP框架

最新版本的2023UI千月影视APP源码是一款开源的完美版应用程序&#xff0c;具备前后端完美匹配的特点。该应用的后端开发基于ThinkPHP框架&#xff0c;这是一个广泛使用的PHP开发框架&#xff0c;具有稳定性和安全性方面的优势。 2023UI千月影视APP是一款提供电影、电视剧、综艺…

ES6 解构

解构的语法 … {} 解构的语法中&#xff0c;...&#xff08;展开运算符&#xff09;和 {}&#xff08;对象字面量&#xff09;扮演着不同的角色。 ...&#xff08;展开运算符&#xff09;&#xff1a; 在解构中&#xff0c;... 被用作展开运算符&#xff0c;用于将数组或对象中…

深入理解MySQL表的操作和管理

MySQL是一种广泛使用的关系型数据库管理系统&#xff0c;用于存储和管理大量结构化数据。在MySQL中&#xff0c;表是数据的基本组织单位&#xff0c;对表的操作和管理能力对于数据库的性能和数据完整性至关重要。本文将深入讨论MySQL表的操作和管理&#xff0c;包括创建表、修改…

Clickhouse学习系列——一条SQL完成gourp by分组与不分组数值计算

笔者在近一两年接触了Clickhouse数据库&#xff0c;在项目中也进行了一些实践&#xff0c;但一直都没有一些技术文章的沉淀&#xff0c;近期打算做个系列&#xff0c;通过一些具体的场景将Clickhouse的用法进行沉淀和分享&#xff0c;供大家参考。 首先我们假设一个Clickhouse数…

智能合约 -- 常规漏洞分析 + 实例

1.重入攻击 漏洞分析 攻击者利用合约漏洞&#xff0c;通过fallback()或者receive()函数进行函数递归进行持续取钱。 刚才试了一下可以递归10次&#xff0c;貌似就结束了(version: 0.8.20)。 直接看代码: 银行合约&#xff1a;有存钱、取钱、查看账户余额等函数。攻击合约:…

第04天 Spring是如何解决循环依赖的

✅作者简介&#xff1a;大家好&#xff0c;我是Leo&#xff0c;热爱Java后端开发者&#xff0c;一个想要与大家共同进步的男人&#x1f609;&#x1f609; &#x1f34e;个人主页&#xff1a;Leo的博客 &#x1f49e;当前专栏&#xff1a;每天一个知识点 ✨特色专栏&#xff1a…

【VUE】7、VUE项目中集成watermark实现页面添加水印

在网站浏览中&#xff0c;常常需要网页水印&#xff0c;以便防止用户截图或录屏暴露敏感信息后&#xff0c;方便追踪用户来源。 1、安装 watermark 在 package.json 文件 dependencies 节点增加 watermark-dom 依赖 "watermark-dom": "2.3.0"然后执行命…

Petrel解释二维浅地层数据

Petrel是斯伦贝谢开发的一款地质解释和建模软件&#xff0c;有点像地理信息系统的ArcGIS&#xff0c;主要用于数据分析和展示。它不是用来处理原始数据的&#xff0c;而是集成各种处理后的结果数据进行特征分析和目标拾取。当然&#xff0c;它也能读取原始数据&#xff0c;比如…

Xilinx DDR3学习总结——2、MIG exmaple直接上板

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 Xilinx DDR3学习总结——2、MIG exmaple直接上板查看初始化状态 前言修改内容上板 前言 上一篇&#xff0c;我们生成了一个example&#xff0c;example的测试激励看起来都比…

Vue 3.0中的Treeshaking?

1.treeshaking是什么&#xff1f; Tree shaking 是一种通过清除多余代码方式来优化项目打包体积的技术&#xff0c;专业术语叫 Dead code elimination 简单来讲&#xff0c;就是在保持代码运行结果不变的前提下&#xff0c;去除无用的代码 如果把代码打包比作制作蛋糕&#…

5. 服务发现

当主机较少时&#xff0c;在抓取配置中手动列出它们的IP地址和端口是常见的做法&#xff0c;但不适用于较大规模的集群。尤其不适用使用容器和基于云的实例的动态集群&#xff0c;这些实例经常会变化、创建或销毁的情况。 Prometheus通过使用服务发现解决了这个问题&#xff1…

C# 扩展方法

C# 扩展方法 假设该类是密封的&#xff0c;不能访问代码&#xff0c;或有其他的设计原因使这些方法不能工作&#xff0c;就不得不在另一个类中使用该类的公有可用成员编写一个方法。 假设这个类是密封的&#xff1a; sealed class MyData{double D1;double D2;double D3;publi…

【Mysql】修改definer

修改definer 本文介绍如何修改MySQL中的function、procedure、event、view和trigger的definer 修改function、procedure的definer 首先&#xff0c;我们需要登录MySQL命令行界面&#xff0c;然后执行以下命令&#xff1a; select definer from mysql.proc;这个命令会列出所…

EFLFK——ELK日志分析系统+kafka+filebeat架构(3)

zookeeperkafka分布式消息队列集群的部署 紧接上期&#xff0c;在ELFK的基础上&#xff0c;添加kafka做数据缓冲 附kafka消息队列 nginx服务器配置filebeat收集日志&#xff1a;192.168.116.40&#xff0c;修改配置将采集到的日志转发给kafka&#xff1b; kafka集群&#xff…

(5)所有角色数据分析页面的构建-5

所有角色数据分析页面&#xff0c;包括一个时间轴柱状图、六个散点图、六个柱状图(每个属性角色的生命值/防御力/攻击力的max与min的对比)。 """绘图""" from pyecharts.charts import Timeline from find_type import FindType import pandas …

RN 使用react-navigation写可以滚动的横向导航条(expo项目)

装包&#xff1a; yarn add react-navigation/material-top-tabs react-native-tab-view npx expo install react-native-pager-view import React from react import { View, Text, ScrollView, SafeAreaView } from react-native import { Icon } from ../../../../../compo…

栈和队列详解

目录 栈 栈的概念及结构&#xff1a; 栈的实现&#xff1a; 代码实现&#xff1a; Stack.h stack.c 队列&#xff1a; 概念及结构&#xff1a; 队列的实现&#xff1a; 代码实现&#xff1a; Queue.h Queue.c 拓展&#xff1a; 循环队列&#xff08;LeetCode题目链接&#xff0…