添加qt相关内容

This commit is contained in:
taynpg 2024-03-08 14:07:44 +08:00
parent 276e3411c6
commit 99748deb93
3 changed files with 353 additions and 0 deletions

134
qt/demo/CVlcKit.cpp Normal file
View File

@ -0,0 +1,134 @@
//
// Created by Administrator on 2023-09-17.
//
#include "CVlcKit.h"
CVlcKit::CVlcKit() = default;
CVlcKit::~CVlcKit()
{
libvlc_media_player_release(m_pMediaPlayer);
libvlc_release(m_pInstance);
}
libvlc_media_player_t *CVlcKit::getPlayerPoint() const {
return m_pMediaPlayer;
}
bool CVlcKit::InitVLC() {
m_pInstance = libvlc_new(0, nullptr);
if (!m_pInstance) {
m_szError = u8"libvlc new failed!";
return false;
}
m_pMediaPlayer = libvlc_media_player_new(m_pInstance);
if (!m_pMediaPlayer) {
libvlc_release(m_pInstance);
m_szError = u8"libvlc_media_player_new failed!";
return false;
}
m_event = libvlc_media_player_event_manager(m_pMediaPlayer);
libvlc_event_attach(m_event, libvlc_MediaPlayerPositionChanged, vlc_call, this);
libvlc_event_attach(m_event, libvlc_MediaPlayerAudioVolume, vlc_call, this);
return true;
}
void CVlcKit::vlc_call(const struct libvlc_event_t *p_event, void *p_data) {
auto* pThis = static_cast<CVlcKit*>(p_data);
if (!pThis) {
return;
}
switch (p_event->type) {
case libvlc_MediaPlayerPositionChanged:
{
float pos = libvlc_media_player_get_position(pThis->m_pMediaPlayer);
emit pThis->sigSetTimeSlider(static_cast<int>(pos * 100));
long long current = libvlc_media_player_get_time(pThis->m_pMediaPlayer) / 1000;
emit pThis->sigSetStringTime(current);
break;
}
case libvlc_MediaPlayerAudioVolume:
{
int vol = libvlc_audio_get_volume(pThis->m_pMediaPlayer);
emit pThis->sigSetVolumeSlider(vol);
break;
}
default:
break;
}
}
void CVlcKit::stopVideo() {
if (!m_pMediaPlayer) {
return;
}
if (libvlc_media_player_get_state(m_pMediaPlayer) == libvlc_state_t::libvlc_Playing) {
libvlc_media_player_stop(m_pMediaPlayer);
}
}
void CVlcKit::pauseVideo() {
if (!m_pMediaPlayer) {
return;
}
if (libvlc_media_player_get_state(m_pMediaPlayer) == libvlc_state_t::libvlc_Playing) {
libvlc_media_player_pause(m_pMediaPlayer);
}
}
void CVlcKit::playVideo() {
if (!m_pMediaPlayer) {
return;
}
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);
}
}
bool CVlcKit::openVideo(const char* path, void* pWindow) {
// 设置路径
m_pMedia = libvlc_media_new_path(m_pInstance, path);
if (!m_pMedia) {
m_szError = u8"libvlc_media_new_path failed!";
return false;
}
// 解析media
libvlc_media_parse(m_pMedia);
// 获取总时长 (ms)
m_totalSec = libvlc_media_get_duration(m_pMedia);
emit sigSetTotalSecond(m_totalSec / 1000);
// 设置media
libvlc_media_player_set_media(m_pMediaPlayer, m_pMedia);
// 设置播放窗口句柄
libvlc_media_player_set_hwnd(m_pMediaPlayer, pWindow);
// 释放media
libvlc_media_release(m_pMedia);
m_pMedia = nullptr;
// 播放视频
libvlc_media_player_play(m_pMediaPlayer);
return true;
}
std::string CVlcKit::getLastError() const {
return m_szError;
}
void CVlcKit::setPlayPostion(float pos) {
if (!m_pMediaPlayer) {
return ;
}
libvlc_media_player_set_position(m_pMediaPlayer, static_cast<float >(pos / 100.0));
}
void CVlcKit::setVolume(int vol) {
if (!m_pMediaPlayer) {
return ;
}
libvlc_audio_set_volume(m_pMediaPlayer, vol);
}

52
qt/demo/CVlcKit.h Normal file
View File

@ -0,0 +1,52 @@
//
// Created by Administrator on 2023-09-17.
//
#ifndef VLCPLAYER_CVLCKIT_H
#define VLCPLAYER_CVLCKIT_H
#ifdef _WIN32
#include <basetsd.h>
typedef SSIZE_T ssize_t;
#endif
#include <QObject>
#include <vlc/vlc.h>
#include <string>
class CVlcKit : public QObject
{
Q_OBJECT
public:
CVlcKit();
~CVlcKit() override;
public:
bool InitVLC();
[[nodiscard]] std::string getLastError() const;
[[nodiscard]] libvlc_media_player_t* getPlayerPoint() const;
public:
void pauseVideo();
void playVideo();
void stopVideo();
bool openVideo(const char* path, void* pWindow);
void setPlayPostion(float pos);
void setVolume(int vol);
signals:
void sigSetTimeSlider(int value);
void sigSetVolumeSlider(int value);
void sigSetStringTime(long long value);
void sigSetTotalSecond(long long value);
private:
static void vlc_call( const struct libvlc_event_t *p_event, void *p_data );
private:
libvlc_instance_t* m_pInstance{};
libvlc_media_player_t* m_pMediaPlayer{};
libvlc_media_t* m_pMedia{};
libvlc_event_manager_t* m_event{};
libvlc_time_t m_totalSec = -1;
std::string m_szError;
};
#endif //VLCPLAYER_CVLCKIT_H

167
qt/qt-knowledge.txt Normal file
View File

@ -0,0 +1,167 @@
1.使用信号槽发送结构体信息,使用 qRegisterMetaType 注册一下。
qRegisterMetaType<SParam>("SParam");
2.sendParamValue 尽管设置成了 const reference,但是还会有一次构造
因为 Qt 一下两个原因:
1.线程安全性:Qt 的信号槽机制支持多线程,在多线程环境中,如果参数是引用,可能会引发线程安全问题。
2.松耦合性:通过传值而不是引用,信号发送方和接收方可以更独立,不必担心在信号槽连接时传递的对象是否仍然有效。
1.Qt的三个窗口类的区别
(1)QMainWindow:包含菜单栏、工具栏、状态栏。
(2)QWidget:普通窗口、大部分窗口要做成无边框的,这个使用较多。
(3)QDialog:对话框,常用于做登录窗口、弹出窗口。
2.类似保存上次访问信息这存储
(1)使用QSettings
示例:ini, /LastPath/path
(2)一些常用的位置在 QStandardPaths::writableLocation(enum);
3.信号
signals: void xxx(int value); --> emit xxx(nValue);
private slots: void yyy();
4.子线程无法直接修改UI,需要使用信号处理。
5.在VS中写QT时开启cmd窗口:
项目--属性--链接器--系统--子系统--控制台 (/SUBSYSTEM:CONSOLE)
6.信号重载:
(1)使用 Qt4 的写法
(2)Qt5 的写法,参数二 => QOverlaod<int>::of(xx)
7.moc全称是Meta-Object Compiler(元对象编译器)
moc -> 分析C++(若有Q_OBJECT) -> moc_xx.cpp(包含QOBJECT实现)
--> 进入编译(预处理器之前,与源文件一起编译而不是替换)
可以理解为:moc把Qt中不是C++的关键字解析让其认识。
8.界面布局
(1)设定min和max就是固定
(2)配合弹簧布局
(3)无边框:setWindowFlags(Qt::FramelessWindowHint)
(4)任务栏单机隐藏显示 Qt::WindowMinMaxButtonsHint
(5)Layout:
setMargin: 与边界间隙。
setSpacing: 与相邻空间间隙(默认大概是7)。
addSpacing: 在 setSpacing 基础上加减。
addStretch: 添加一个 QSpacerItem
(6)QSplitter分裂器布局
setOpaqueResize(false): 鼠标松开才变换。
(7)全局setLayout如果不清除的话仅第一次生效。
(8)清空布局内的所有元素:
QLayoutItem* pChild; layout = this->layout();
while((pChild = layout->takeAt(0)) != 0) if (pChild->widget()) xx->setParent(nullptr) delete;
(9)Widget也可以作为一个容器用于布局。
(10) resize 可以设定首次启动时的界面大小。
9.QSS相关
(1)给QPushButton设置背景图:
QPushButton {
background-image: url(:/resource/xx.png);
border: none // 无边框;
QPushButton::hover {
background-color: rgb(192, 192, 192); // 鼠标悬浮时
}
}
(2)QTabWidget
QTabWidget::pane {
border-top: 1px solid #EAEAEA; // 上边框
position: absolute;
top: -0.1px;
}
QTabBar::tab {
width: 100px;
height: 32px;
font-size: 16px;
font-family: Microsoft YaHei;
font-weight: 400;
background: #FFFFFF;
border: 2px solid #FFFFFF;
border-bottom-color: #FFFFFF;
border-top-left-radius: 4px;
border-top-right-radium: 4px;
padding: 2px;
}
QTabBar::tab:selected {
color: #333333;
border-color: #FFFFFF;
border-bottom-color: #4BA4F2;
}
QTabBar::tab:!selected {
color: #B2B2B2;
border-color: #FFFFFF;
border-bottom-color: #4BA4F2;
}
10.基本控件简记
(1)QToolButton可以图片文字一起显示
QToolButton->setIcon, setIconSize, setToolButtonStyle
(2)QLineEidt
setPlacceholderText: 未填写内容的提示信息
setEchoMode: 显示形式,比如密码形式。
(3)右键菜单, menu->exec(QCursor::pos())
setContextMenuPolicy(Qt::CustomContextMenu);
void contextMenuEvent(QContextMenuEvent* event) override;
(4)QMenu可以添加QAction
(5)QTableWiget可以直接insert Wiget作为一个新的tab页。
setTabPostion: 位置
setTabShape: 形状
setMovable: 可移动
(6)QListWidget
Item项也可以是一个widget。
ListeItem 怎么添加右键菜单:
①必须设置菜单策略 listWidget->setContextMenuPolicy(Qt::CustomContextMenu);
connect(xxListwidget, &QListWidget::customContextMenuRequested, this, xxx);
图标模式:setViewMode
焦点:setFocusPolicy(Qt::NoFocus)
(7)QScrollArea
setFrameShape(QFrame::NoFrame)
setHorizontalScrollBarPolicy()
11.无边框窗口
Qt很多问题都可以从QWidget中找到答案,学习Qt大部分就是在与QWidget打交道。
(1)无边框界面移动和拉伸、大小改变
重写
mousePressEvent
mouseMoveEvent
this->pos() 窗口左上角相对桌面位置
event->pos() 鼠标相对exe位置
event->globalPos() 鼠标相对桌面位置
使用 this->move(pos)
重写nativeEvent()
setAttributte(Qt::WA_Hover)防止鼠标边无法选中。
(2)自定义标题栏(自己布局)
setAttribute(Qt::WA_StyleBackground)禁止父窗口影响子窗口样式。
qobject_cast<QPushButton*>(sender())可以获取发送者。
(3)实现窗口阴影
需要两层窗口
$设置窗口透明 this->setAttribute(Qt::WA_TranslucentBackground, true);
$设置无边框 Qt::Windows
$给顶层widget设置背景颜色,不然看不见。
Qt窗口阴影类:QGraphicsDropShadowEffect -> shadow
设置阴影距离: setOffset
设置阴影颜色: setColor
设置区域: setBlurRadius
给顶层widget设置阴影:setGraphicsEffect(shadow);
相当于是:将实际窗口外包了一层,外面层用于透明。
(4)圆角窗口(也要透明,无边框)
$1: 重写paintEvent
QPainter painter(this);
painter.setRenderHint(QPainter::Antialiasing); // 抗锯齿
painter.setBrush(QBrush(QColor(255, 255, 255));
painter.setPen(Qt::transparent);
auto rect = this->rect();
painter.drawRoundedRect(rect, 15, 15); // 可以画圆
$2: 使用 qss
this->setStyleSheet("QWidget{background-color:gray; border-radius: 30px}");
// 窗口透明后,需要重写该方法
void paintEvent(xxx)
{
QStyleOption opt;
opt.init(this);
QPainter p(this);
style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
QWidget::paintEvent(event);
}
12.QSS样式表(Qt Style Sheet) 类似css,但是没有css强大。
13.Qt CEF, QCefView