Qt——事件简单介绍

目录

事件的概念

事件和处理函数的关联

鼠标移入和移出事件

题外话------自定义类如何使用QtDesigner

鼠标按下事件

鼠标释放事件

鼠标双击事件

鼠标移动事件

题外话------setMouseTracking(true)

鼠标滚动事件

键盘事件

定时器事件

窗口移动和大小改变事件


事件的概念

用户进行的各种操作,不仅会产生信号,也会产生事件。我们也可以给事件关联对应的处理函数,当事件触发时就执行到相应代码

实际上,事件本身是操作系统提供的机制Qt将操作系统的事件封装成Qt事件。尽管封装了,Qt事件使用起来还是比较复杂,所以Qt把Qt事件再封装成信号槽

不过Qt信号槽并没有提供所有用户动作对应的信号,因此我们有时还需要利用Qt事件机制来获取并处理用户动作**。**

所有的Qt事件都继承自QEvent:


事件和处理函数的关联

之前我们使用connect函数来把信号和信号槽关联起来。而在Qt事件机制中,我们需要做的是****重写Qt内置控件的事件处理函数,即创建继承自Qt内置控件类的新控件类并重写其事件处理函数,我们使用新的控件类自然就绑定了新的事件处理函数。

一句话,如果想对某控件的某些事件做出处理,就使用继承自该控件的子类控件并重写父类控件中对应事件的虚函数。


鼠标移入和移出事件

cpp 复制代码
#include "label.h"

label::label(QWidget * ptr):QLabel(ptr)
{
    this->setStyleSheet("border: 1px solid black; padding: 5px;");
}

void label::enterEvent(QEvent *event)
{
    qDebug()<<"鼠标进来了!";
}
void label::leaveEvent(QEvent *event)
{
    qDebug()<<"鼠标出去了!";
}

题外话------自定义类如何使用QtDesigner

我们现在的需求是,使用我们自己写的继承自Qt内置控件的控件,但是QtDesigner中的控件都是Qt内置的控件。

实际上我们可以先在QtDesigner拖入一个Qt内置控件,然后将其提升为我们写的子类控件,具体做法如下:

拖入一个内置QLabel控件,通过右侧对象树可知,当前使用的不是子类控件

右击选中的控件,点击提升为:

在上面两处填写你自己编写的控件类名称以及头文件名称并点击添加:

选中出现的行,然后点击提升,我们发现对象树上的类型已经改变了:


鼠标按下事件

cpp 复制代码
#include "label.h"

Label::Label(QWidget * ptr):QLabel(ptr)
{
    this->setStyleSheet("border: 1px solid black; padding: 5px;");
}

void Label::mousePressEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)
    {
        qDebug()<<"点击鼠标左键";
    }
    else if(event->button() == Qt::RightButton)
    {
         qDebug()<<"点击鼠标右键";
    }
    qDebug()<<"相对于lebel的坐标x:"<< event->x() <<"相对于lebel的坐标y"<<event->y();
    qDebug()<<"全局坐标x:"<< event->globalX() <<"全局坐标y"<<event->globalY();
}

鼠标释放事件

cpp 复制代码
#include "label.h"

Label::Label(QWidget * ptr):QLabel(ptr)
{
    this->setStyleSheet("border: 1px solid black; padding: 5px;");
}

void Label::mousePressEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)
    {
        qDebug()<<"点击鼠标左键";
    }
    else if(event->button() == Qt::RightButton)
    {
         qDebug()<<"点击鼠标右键";
    }
}

void Label::mouseReleaseEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)
    {
        qDebug()<<"释放鼠标左键";
    }
    else if(event->button() == Qt::RightButton)
    {
         qDebug()<<"释放鼠标右键";
    }
}

鼠标双击事件

cpp 复制代码
#include "label.h"

Label::Label(QWidget * ptr):QLabel(ptr)
{
    this->setStyleSheet("border: 1px solid black; padding: 5px;");
}

void Label::mousePressEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)
    {
        qDebug()<<"点击鼠标左键";
    }
    else if(event->button() == Qt::RightButton)
    {
         qDebug()<<"点击鼠标右键";
    }
}

void Label::mouseReleaseEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)
    {
        qDebug()<<"释放鼠标左键";
    }
    else if(event->button() == Qt::RightButton)
    {
         qDebug()<<"释放鼠标右键";
    }
}

void Label::mouseDoubleClickEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)
    {
        qDebug()<<"双击鼠标左键";
    }
    else if(event->button() == Qt::RightButton)
    {
         qDebug()<<"双击鼠标右键";
    }
}

注意:双击事件触发之前,单击事件也是被触发的


鼠标移动事件

cpp 复制代码
#include "widget.h"
#include "ui_widget.h"

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);

    this->setMouseTracking(true);

}

void Widget::mouseMoveEvent(QMouseEvent *event)
{
    qDebug()<<"鼠标的x坐标是:"<<event->x()<<"鼠标的y坐标是:"<<event->y();
}

Widget::~Widget()
{
    delete ui;
}

题外话------setMouseTracking(true)

对于鼠标移动用来说,只要我们轻轻一划,就会产生很多移动事件,这会可能会导致客户端变得卡顿。

因此,控件默认是不追踪鼠标的移动的,在这种情况下,我们只有按住鼠标左键或者右键然后移动鼠标才会触发移动事件。

**想要达到鼠标只要移动就触发事件的效果就得告诉Qt你想追踪鼠标移动:**setMouseTracking(true)


鼠标滚动事件

cpp 复制代码
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    //this->setMouseTracking(true);

}

void Widget::wheelEvent(QWheelEvent *event)
{
    //显示出鼠标滚动的距离
    qDebug()<<event->delta();
}

键盘事件

cpp 复制代码
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);

}
void Widget::keyPressEvent(QKeyEvent *event)
{
    //如果用户按下了普通按键按键,会被存放到这儿
    int commonkey = event->key();
    //如果用户按下了ctrl,alt等特按键,会被存放到这儿
    int specialkey = event->modifiers();
    

    //Qt用数字表示按键类型,但提供给我们一些类型使用,我们使用这些类型区分出用户按下的是什么键
    
    //可以使用specialkey和commonkey判断用户是否按下了组合键(如果两个都不为0就是组合键),按下了什么组合键
    if(specialkey == Qt::ControlModifier && commonkey == Qt::Key_A)
    {
        qDebug()<<"用户按下了ctrl+A";
    }
    else if(specialkey == 0 && commonkey == Qt::Key_A)
    {
        qDebug()<<"用户按下了A";
    }
    else if(specialkey == 0 && commonkey == Qt::Key_B)
    {
        qDebug()<<"用户按下了B";
    }
}

定时器事件

cpp 复制代码
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);

    //开启控件的定时器事件并设置间隔时间,返回定时器ID
    timer_id = this->startTimer(1000);
}

//定时器时间处理函数
void Widget::timerEvent(QTimerEvent *event)
{
    //一个控件可能有多个定时器,它们都会触发这个处理函数,所以我们用ID来区分它们
    int id = event->timerId();
    if(id!=timer_id)
    {
        return;
    }

    //如果倒计时小于等于0,就停止定时器
    int value = ui->lcdNumber->intValue();
    if(value<=0)
    {
        this->killTimer(id);
        return;
    }
    //否则就让倒计时-1
    else
    {
        ui->lcdNumber->display(value-1);
    }
}

窗口移动和大小改变事件

cpp 复制代码
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
}

void Widget::moveEvent(QMoveEvent *event)
{
    qDebug()<<event->pos();
}

void Widget::resizeEvent(QResizeEvent *event)
{
    qDebug()<<event->size();
}
相关推荐
188号安全攻城狮1 小时前
【前端基础知识】JavaScript 数组方法总结:从表格速查到分类详解
开发语言·前端·javascript·网络安全
不爱吃炸鸡柳1 小时前
5道经典贪心算法题详解:从入门到进阶
开发语言·数据结构·c++·算法·贪心算法
qq_381338501 小时前
微前端架构深度实践:从 qiankun 到 Module Federation 的企业级方案
前端·架构
鱼干~1 小时前
【全栈知识点】全栈开发知识点
前端·人工智能·c#
英俊潇洒美少年1 小时前
迷你 React 调度器(带优先级+时间切片)手写实现
前端·javascript·react.js
chQHk57BN1 小时前
PWA开发指南:构建可离线使用的渐进式Web应用
前端
xyq20241 小时前
Java 变量命名规则
开发语言
天启HTTP1 小时前
HTTP代理和隧道代理的底层区别与适用场景分析
开发语言·网络协议·tcp/ip·php
小白学大数据1 小时前
告别复杂 XPath:DeepSeek+Python 爬虫快速实践
开发语言·爬虫·python·selenium
weixin_408099671 小时前
【保姆级教程】按键精灵调用 OCR 文字识别 API(从0到1完整实战 + 可运行脚本)
java·前端·人工智能·后端·ocr·api·按键精灵