当前位置:   article > 正文

【QT入门】 Qt自定义控件与样式设计之QPushButton实现鼠标悬浮按钮弹出对话框

【QT入门】 Qt自定义控件与样式设计之QPushButton实现鼠标悬浮按钮弹出对话框

往期回顾:

【QT入门】 Qt自定义控件与样式设计之qss选择器-CSDN博客

【QT入门】 Qt自定义控件与样式设计之QLineEdit的qss使用-CSDN博客

【QT入门】Qt自定义控件与样式设计之QPushButton常用qss-CSDN博客

QT入门】 Qt自定义控件与样式设计之QPushButton实现鼠标悬浮按钮弹出对话框

一、最终效果

鼠标悬浮弹出对话框的功能:最终要实现纯代码设计出一个音量按钮,当鼠标悬浮在上面的时候,显示滑块,并可以自己调节大小,鼠标离开后滑块消失。

二、具体实现

 1、鼠标事件捕捉

首先实现一个基本功能,就是窗口能识别鼠标,鼠标放上去就显示,不放上去就不显示,这个功能很简单只需要重写两个函数

void enterEvent(QEvent * event) override;
void leaveEvent(QEvent * event) override;

在Widget主窗口写这两个类,你那么就能实现鼠标放进主窗口有反应,拿出来也有反应

  1. void Widget::enterEvent(QEvent * event)
  2. {
  3. qDebug()<<"11111";
  4. }
  5. void Widget::leaveEvent(QEvent * event)
  6. {
  7. qDebug()<<"22222";
  8. }

2、加载主布局 

Widget类加载主布局,主类没操作,就是设置大小,创建布局,把按钮放进去。

  1. Widget::Widget(QWidget *parent)
  2. : QWidget(parent)
  3. , ui(new Ui::Widget)
  4. {
  5. ui->setupUi(this);
  6. resize(800, 600);
  7. //创建水平布局
  8. QHBoxLayout *pHlay = new QHBoxLayout(this);
  9. //创建音量按钮并添加到布局水平布局里
  10. CVolumeButton* pVolumeButton = new CVolumeButton(this);
  11. pHlay->addWidget(pVolumeButton);
  12. }

3、创建滑块 

 3.1cvolumesliderdialog类创建滑块

这个类本身继承自QDialog,主要作用是创建QSlider对象并添加到布局中。

QSlider是用于控制边界值的经典小部件。它允许用户沿水平或垂直凹槽移动QSlider 的滑块,并将 滑块 的位置转换为合法范围内的整数值。

  1. //设置对话框的固定大小为(40, 200)
  2. this->setFixedSize(40, 200);
  3. //创建一个垂直布局管理器pVLay,
  4. QVBoxLayout* pVLay = new QVBoxLayout(this);
  5. m_pSlider = new QSlider(this);
  6. //将一个垂直方向的QSlider对象m_pSlider添加到布局中
  7. m_pSlider->setOrientation(Qt::Vertical);
  8. pVLay->addWidget(m_pSlider);
3.2设置窗口标志

另外比较重要的一点,把设置窗口标志为无边框窗口和ToolTip提示框样式

这里用到一个ToolTip,ToolTip = Popup | Sheet ,可以实现滑块悬浮时显示,离开时消失

  1. setFixedSize(40, 120);
  2. //设置窗口标志为无边框窗口和ToolTip提示框样式
  3. //ToolTip = Popup | Sheet,
  4. setWindowFlags(Qt::FramelessWindowHint | Qt::ToolTip); //ToolTip : 悬浮时显示,离开时消失
  5. setStyleSheet("QDialog{background-color: rgba(54, 54, 54, 0.5);}"); //0.5表示透明度,0表示全透明、1表示不透明;
  6. //也可以使用百分百表示如: frm->setStyleSheet(“QFrame{background-color: rgba(255, 0, 0, 50%);}”);
3.3修复bug

bug: bool QWidget::event(QEvent *event)这个方法设置popup后,dialog会有窗口阴影

需要去除就重写event函数,

  1. bool CVolumeSliderDialog::event(QEvent* event)
  2. {
  3. //定义一个静态布尔变量class_amended,用于跟踪是否已经修改了窗口类。
  4. static bool class_amended = false;
  5. //事件类型为WinIdChange,表示窗口标识已更改。
  6. if (event->type() == QEvent::WinIdChange)
  7. {
  8. //获取对话框的窗口句柄
  9. HWND hwnd = (HWND)winId();
  10. //检查是否已经修改了窗口类
  11. if (class_amended == false)
  12. {
  13. class_amended = true;//class_amended设置为true,表示已经修改了窗口类
  14. DWORD class_style = ::GetClassLong(hwnd, GCL_STYLE);//获取窗口类的样式
  15. //将CS_DROPSHADOW样式从窗口类样式中移除,这样可以去除窗口阴影
  16. class_style &= ~CS_DROPSHADOW;
  17. //将修改后的窗口类样式应用到窗口
  18. ::SetClassLong(hwnd, GCL_STYLE, class_style); // windows系统函数
  19. }
  20. }
  21. //调用基类QWidget的event函数来处理事件,确保其他事件得到正常处理
  22. return QWidget::event(event);
  23. }

注意这个完全可以作为一个通用类,需要的时候直接使用。

4、创建音量按钮

cvolumebutton类实现音量按钮的创建和滑块的绑定以及逻辑实现

主要是几个逻辑实现
 

1、纯代码创建一个音量按钮
2、创建滑块并定位在按钮上方合适位置
3、判断是否点击按钮需要静音来改变滑块上的位置
4、通过定时器判断鼠标是否离开按钮和滑块区域,离开滑块就消失
4.1首先通过重写paintEvent方法,实现了在CVolumeButton控件上绘制一个按钮外观的功能 
  1. //代码可重用
  2. void CVolumeButton::paintEvent(QPaintEvent*)
  3. {
  4. //QStylePainter用于绘制基于QStyle的外观元素
  5. QStylePainter p(this);
  6. //QStyleOptionButton用于存储按钮控件的外观选项,如颜色、边框等
  7. QStyleOptionButton option;
  8. //调用initStyleOption方法初始化option对象
  9. //initStyleOption通常用于设置控件的外观选项,以便正确绘制控件
  10. initStyleOption(&option);
  11. //使用QStylePainter对象p绘制一个CE_PushButton类型的控件,参数为option对象。
  12. //这将根据option中的外观选项绘制按钮的外观,如背景、文本等
  13. p.drawControl(QStyle::CE_PushButton, option);
  14. }
4.2然后在鼠标进入声音按钮时,创建并显示一个音量滑块对象,将其定位在合适的位置,并启动一个定时器。同时,建立了一个信号与槽的连接,以便在滑块数值变化时通知其他部分。 
  1. void CVolumeButton::enterEvent(QEvent* event)
  2. {
  3. //判断并初始化滑块对象,确保只有在需要时才创建滑块对象
  4. if (!m_pVolumeSliderDlg)
  5. m_pVolumeSliderDlg = new CVolumeSliderDialog(this);
  6. QPoint p1 = this->mapToGlobal(QPoint(0, 0)); //声音按钮左上角相对于桌面的绝对位置
  7. QRect rect1 = this->rect(); //获取声音按钮的矩形区域
  8. //获取滑块对象的矩形区域,rect包含标题栏,去掉标题栏后height不变
  9. QRect rect2 = m_pVolumeSliderDlg->rect();
  10. //计算滑块对象在水平方向上的位置,使其水平居中于声音按钮
  11. //这里计算很简单,就是按钮的一半减去滑块的一半,就是多出来的那部分
  12. //按钮的x加上多出来那部分,就能得到滑块居中的x位置
  13. int x = p1.x() + (rect1.width() - rect2.width()) / 2;
  14. //计算滑块对象在垂直方向上的位置,使其位于声音按钮上方并略微偏移5px
  15. //按钮的y减去滑块的长度再减去偏移的5px,就是滑块的左上角y位置
  16. int y = p1.y() - rect2.height() - 5;
  17. //滑块对象移动到计算得到的位置
  18. m_pVolumeSliderDlg->move(x, y); //move是相对于桌面原点的位置
  19. m_pVolumeSliderDlg->show();//显示滑块对象
  20. m_timerId = startTimer(250);//动一个定时器,每250毫秒触发一次定时器事件
  21. connect(m_pVolumeSliderDlg, &CVolumeSliderDialog::sig_SliderValueChanged, [=](int value) {
  22. emit sig_VolumeValue(value);
  23. });
  24. }
4.3检测鼠标按下事件实现静音
  1. void CVolumeButton::mousePressEvent(QMouseEvent* event)
  2. {
  3. if (event->button() == Qt::LeftButton)//检查鼠标按下的按钮是否是左键
  4. {
  5. m_isMute = !m_isMute; //切换静音状态,真假切换
  6. if (m_isMute)//如果当前为静音状态
  7. {
  8. //如果滑块对象存在,将滑块值设置为0,表示静音状态
  9. if (m_pVolumeSliderDlg)
  10. m_pVolumeSliderDlg->setSliderValue(0);
  11. }
  12. else
  13. {
  14. //如果滑块对象存在,将滑块值设置为50,表示非静音状态下的默认音量
  15. if (m_pVolumeSliderDlg)
  16. m_pVolumeSliderDlg->setSliderValue(50);
  17. }
  18. }
  19. }
4.4在定时器事件中检测鼠标位置,如果鼠标移出音量滑块对象或声音按钮的区域,则隐藏音量滑块对象;如果音量滑块对象不存在或不可见,则停止定时器。
  1. void CVolumeButton::timerEvent(QTimerEvent* event)
  2. {
  3. //检查音量滑块对象是否存在且可见
  4. if ((m_pVolumeSliderDlg != nullptr) && (m_pVolumeSliderDlg->isVisible()))
  5. {
  6. QPoint p1 = QCursor::pos(); //鼠标绝对位置
  7. if (m_pVolumeSliderDlg)
  8. {
  9. QRect rect1 = this->rect(); //获取声音按钮的矩形区域
  10. QRect rect2 = m_pVolumeSliderDlg->rect();//获取音量滑块对象的矩形区域
  11. QRect rect3 = m_pVolumeSliderDlg->geometry();//获取音量滑块对象的几何信息
  12. QPoint p2 = this->mapToGlobal(QPoint(0, 0)); //声音按钮左上角相对于桌面的绝对位置
  13. //已知:音量框宽40 > 按钮宽30
  14. //创建一个矩形区域,用于检测鼠标位置是否在音量滑块对象或声音按钮上
  15. QRect area(rect3.left(), rect3.top(), rect2.width(), p2.y() + rect1.height() - rect3.top()); //左上宽高
  16. if (!area.contains(p1))//如果鼠标位置不在矩形区域内
  17. {
  18. m_pVolumeSliderDlg->hide();//隐藏音量滑块对象
  19. }
  20. }
  21. }
  22. else //如果鼠标位置在矩形区域内
  23. {
  24. //停止定时器,即终止定时器事件的触发
  25. killTimer(m_timerId);
  26. }
  27. }

都看到这里了,点个赞再走呗朋友~

加油吧,预祝大家变得更强!

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

闽ICP备14008679号