新建一个项目:06Event

新建cpp文件

06Event.pro

HEADERS
+=
\

MyWidget.h

SOURCES
+=
\

MyWidget.cpp

QT
+=
widgets gui

MyWidget.h

#ifndef
MYWIDGET_H

#define
MYWIDGET_H

#include
<QWidget>

#include
<QPushButton>

#include
<QLineEdit>

class
MyWidget
:
public
QWidget

{

Q_OBJECT

public:

explicit
MyWidget(QWidget
*parent
);

bool
event(QEvent
*);

void
mousePressEvent(QMouseEvent
*);

void
mouseReleaseEvent(QMouseEvent
*);

void
mouseMoveEvent(QMouseEvent
*);

/*不要用,因为它默认调用两次mousePressEvent*/

//void
mouseDoubleClickEvent(QMouseEvent
*);

void
keyPressEvent(QKeyEvent
*);

void
keyReleaseEvent(QKeyEvent
*);

void
closeEvent(QCloseEvent
*);

//void
showEvent(QShowEvent
*);

//void
hideEvent(QHideEvent
*);

void
paintEvent(QPaintEvent
*);

QPushButton*
button;

QLineEdit*
edit;

signals:

public
slots:

void
slotButtonClicked();

};

#endif
//
MYWIDGET_H

MyWidget.cpp

#include "MyWidget.h"
#include <QApplication>
#include <QEvent>
#include <QDebug>
#include <QMouseEvent>
#include <QKeyEvent>
#include <QVBoxLayout>
#include <QPainter>
 
MyWidget::MyWidget(QWidget *parent) :
    QWidget(parent)
{
//  QVBoxLayout* lay = new QVBoxLayout(this);
#if 0
    QPushButton* button2;
    lay->addWidget(button = new QPushButton("OK", this));
    lay->addWidget(button2 = new QPushButton("Button2"));
    //设置它为默认的焦点,当点击tab键之后可以切换焦点
    button->setDefault(true);
 
    // 鼠标不需要按下,mouseMove就能得到调用
    this->setMouseTracking(true);
 
    connect(button2, SIGNAL(clicked()), this, SLOT(slotButtonClicked()));
    connect(button, SIGNAL(clicked()), this, SLOT(slotButtonClicked()));
#endif
//    lay->addWidget(edit = new QLineEdit());
//    connect(edit, SIGNAL(returnPressed()), this, SLOT(slotButtonClicked()));
}
 
//通过这一句实现点击按钮的时候获得按钮上的text()内容
void MyWidget::slotButtonClicked()
{
    QLineEdit* button = (QLineEdit*)sender();
    qDebug() << button->text();
}
 
/* QApplication先得到->具体应该处理的窗口::event()->event()根据消息类型来调用具体的虚函数 */
/* 1)可以重载具体的虚函数,来实现对消息的响应
   2)可以重载event函数,用来处理或者截取消息 */
 
/* 截取消息 */
bool MyWidget::event(QEvent *ev)
{
    // 鼠标消息被截断
   // if(ev->type() == QEvent::MouseButtonPress)
     //   return true;
    ev->accept();
 
    return QWidget::event(ev);
}
 
/**
 * @brief MyWidget::closeEvent 窗口关闭的事件
 */
void MyWidget::closeEvent(QCloseEvent *)
{
    qDebug() << "closeEvent";
}
 
void MyWidget::paintEvent(QPaintEvent *)
{
    QPainter p(this);
    //通过下面的方式实现画线
    p.drawLine(QPoint(0, 0), QPoint(100, 100));
}
 
/**
 * @brief MyWidget::mousePressEvent 鼠标按下的事件
 * @param ev
 */
void MyWidget::mousePressEvent(QMouseEvent *ev)
{
#if 0
    QPoint pt = ev->pos();
    qDebug() << pt;
    //如果鼠标按下的是左键的处理
    if(ev->button() == Qt::LeftButton){}
 
    //如果按下的Shift键了
    if(ev->modifiers() == Qt::ShiftModifier)
    {
        qDebug() << "shift press";
    }
#endif
    //改进办法:先判断是否有左键,然后如果判断是否还按了Ctrl键
    if(ev->button() == Qt::LeftButton)
    {
        if(ev->modifiers() == Qt::ControlModifier)
        {
            // handle with Control;
            return;
        }
        // handle2 without control;
    } else {}
}
 
/**
 * @brief MyWidget::mouseReleaseEvent 鼠标释放的按键
 */
void MyWidget::mouseReleaseEvent(QMouseEvent *){}
 
/**
 * @brief MyWidget::mouseMoveEvent 鼠标移动的事件
 */
void MyWidget::mouseMoveEvent(QMouseEvent *)
{
    static int i=0;
    qDebug() << "mouse move"<< i++;
}
 
/**
 * @brief MyWidget::keyPressEvent 鼠标按下的事件,通过这个可以获得按下的键
 * @param ev
 */
void MyWidget::keyPressEvent(QKeyEvent *ev)
{
    ev->modifiers();
    int key = ev->key();
    qDebug() << key;
    char a = key;
    qDebug() << (char)a;
}
 
/**
 * @brief MyWidget::keyReleaseEvent 按键释放的事件
 */
void MyWidget::keyReleaseEvent(QKeyEvent *){}
 
int main(int argc, char* argv[])
{
    QApplication app(argc, argv);
 
    MyWidget w;
    w.show();
 
    return app.exec();
}
运行结果:


QT消息过滤器

EventFilter.pro

HEADERS
+=
\

MyWidget.h
\

MyApplication.h

SOURCES
+=
\

MyWidget.cpp
\

MyApplication.cpp

QT
+=
widgets gui

MyWidget.h

#ifndef MYWIDGET_H
#define MYWIDGET_H
 
#include <QWidget>
#include <QPushButton>
class MyWidget : public QWidget
{
    Q_OBJECT
public:
    explicit MyWidget(QWidget *parent);
 
    QPushButton* _button;
    bool eventFilter(QObject *, QEvent *);
    bool event(QEvent *);
signals:
 
public slots:
 
};
 
#endif // MYWIDGET_H

MyWidget.cpp

#include "MyWidget.h"
#include <QPushButton>
#include <QEvent>
#include "MyApplication.h"
#include <QDebug>
#include <QApplication>
 
MyWidget::MyWidget(QWidget *parent) :
    QWidget(parent)
{
    QPushButton* button;
 
    button = new QPushButton("This button", this);
    connect(button, SIGNAL(clicked()), this, SLOT(close()));
    _button = button;
 
    /*button给自己安装了一个消息过滤器,那么经过button的消息,
     * 都先要调用它的过滤器的eventFilter函数*/
    button->installEventFilter(this);
}
 
bool MyWidget::eventFilter(QObject *o, QEvent *e)
{
#if 0
    if(0 == (QObject*)_button &&(
                e->type() == QEvent::MouseButtonRelease ||
                e->type() == QEvent::MouseButtonDblClick ||
                e->type() == QEvent::MouseButtonPress)) {
        return true;
    }
#endif
 
    return QWidget::eventFilter(o,e);
}
 
bool MyWidget::event(QEvent *e)
{
    if(e->type() == QEvent::User)
    {
        qDebug() << "User event is comming";
    }
    return QWidget::event(e);
}
 
int main(int argc,char *argv[])
{
    MyApplication app(argc,argv);
    MyWidget w;
    w.show();
 
    //发送一个Event给MyWidget
    qDebug() << "begin send";
    app.postEvent(&w,new QEvent(QEvent::User));
    qDebug() << "end send";
 
    return app.exec();
}

MyApplication.h

#ifndef MYAPPLICATION_H
#define MYAPPLICATION_H
 
#include <QApplication>
 
class MyApplication : public QApplication
{
    Q_OBJECT
public:
    MyApplication(int argc,char *argv[]):QApplication(argc,argv)
    {}
    bool notify(QObject *,QEvent *);
 
signals:
 
public slots:
 
};
 
#endif // MYAPPLICATION_H

MyApplication.cpp

#include "MyApplication.h"
#include <QEvent>
 
#include <QDebug>
bool MyApplication::notify(QObject *o, QEvent *e)
{
    if(this->topLevelWidgets().count()>0)
    {
        QWidget* mainWnd = this->topLevelWidgets().at(0);
        if(o==(QObject*)mainWnd && e->type() == QEvent::MouseButtonPress)
        {
            // do ...
            qDebug() << "mainwnd is clicked";
        }
    }
 
    return QApplication::notify(o, e);
}

运行结果:

3.QT事件处理,消息过滤器的更多相关文章

  1. Qt事件处理机制

    研一的时候开始使用Qt,感觉用Qt开发图形界面比MFC的一套框架来方便的多.后来由于项目的需要,也没有再接触Qt了.现在要重新拾起来,于是要从基础学起. Now,开始学习Qt事件处理机制. 先给出原文 ...

  2. 【转】Qt 事件处理机制 (下篇)

    转自:http://mobile.51cto.com/symbian-272816.htm 在Qt中,事件被封装成一个个对象,所有的事件均继承自抽象类QEvent. 接下来依次谈谈Qt中有谁来产生.分 ...

  3. 9、Qt 事件处理机制

    原文地址:http://mobile.51cto.com/symbian-272812.htm 在Qt中,事件被封装成一个个对象,所有的事件均继承自抽象类QEvent. 接下来依次谈谈Qt中有谁来产生 ...

  4. Qt 事件处理机制

    Qt 事件处理机制 因为这篇文章写得特别好,将Qt的事件处理机制能够阐述的清晰有条理,并且便于学习.于是就装载过来了(本文做了排版,并删减了一些冗余的东西,希望原主勿怪),以供学习之用. 简介 在Qt ...

  5. QT开发(十二)——QT事件处理机制

    一.QT事件简介 QT程序是事件驱动的, 程序的每个动作都是由内部某个事件所触发.QT事件的发生和处理成为程序运行的主线,存在于程序整个生命周期. 常见的QT事件类型如下: 键盘事件: 按键按下和松开 ...

  6. Qt 事件处理机制 (下篇)

    继续我们上一篇文章继续介绍,Qt 事件处理机制 (上篇) 介绍了Qt框架的事件处理机制:事件的产生.分发.接受和处理,并以视窗系统鼠标点击QWidget为例,对代码进行了剖析,向大家分析了Qt框架如何 ...

  7. 【转】解读Qt 事件处理机制(上篇)

    [转自]:http://mobile.51cto.com/symbian-272812.htm 在Qt中,事件被封装成一个个对象,所有的事件均继承自抽象类QEvent. 接下来依次谈谈Qt中有谁来产生 ...

  8. Qt5学习笔记(消息过滤器)

    T06EventFilter.pro HEADERS += \ MyWidget.h SOURCES += \ MyWidget.cpp QT += widgets gui MyWidget.h #i ...

  9. Qt 事件处理机制 (上篇)

    本篇来介绍Qt 事件处理机制 .深入了解事件处理系统对于每个学习Qt人来说非常重要,可以说,Qt是以事件驱动的UI工具集. 大家熟知Signals/Slots在多线程的实现也依赖于Qt的事件处理机制. ...

随机推荐

  1. [PA 2014]Lustra

    Description Byteasar公司专门外包生产带有镜子的衣柜.刚刚举行的招标会上,有n个工厂参加竞标.所有镜子都是长方形的,每个工厂能够制造的镜子都有其各自的最大.最小宽度和最大.最小高度. ...

  2. hdu 4514 并查集+树形dp

    湫湫系列故事——设计风景线 Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Tot ...

  3. [Helvetic Coding Contest 2017 online mirror]

    来自FallDream的博客,未经允许,请勿转载,谢谢, 第一次在cf上打acm...和同校大佬组队打 总共15题,比较鬼畜,最后勉强过了10题. AB一样的题目,不同数据范围,一起讲吧 你有一个背包 ...

  4. SpringCloud学习之SpringCloudBus

    一.spring-cloud-bus是什么? 回答这个问题之前,我们先回顾先前的分布式配置,当配置中心发生变化后,我们需要利用spring-boot-actuator里的refresh端点进行手动刷新 ...

  5. 简述RIP路由协议和OSPF路由协议的相同点和不同点。

    路由协议分为静态路由协议和动态路由协议.动态路由协议有很多种,如RIP.OSPF.EIGRP等. 1.RIP(路由信息协议)是路由器生产商之间使用的第一个开放标准.RIP有两个版本:RIPv1和RIP ...

  6. nginx负载均衡及详细配置

    接上篇nginx配置,然后再准备两台web服务器: nginx服务器:192.168.0.241 web1:192.168.0.141 web2:192.168.0.142 一.两台web服务器先安装 ...

  7. candy(动态规划)

    题目描述 There are N children standing in a line. Each child is assigned a rating value. You are giving ...

  8. 2017-9-19 c语言预备作业

    题目一: (1)我对邹欣老师博客内容的看法 针对邹欣老师的第一种看法,也就是文中所谈的春蚕与园丁的例子.我认为在大学之前的阶段,师生关系可以如此比喻,因为在中学阶段教师与学生的关系,更多地是一个知识的 ...

  9. 微信内置浏览器 如何小窗不全屏播放视频?也可以尝试canvas.

    设置属性: <video height="100%" width="100%" autoplay="autoplay" control ...

  10. 微信小程序 发现之旅(一)—— 项目搭建与页面跳转

    开发微信小程序需要注册一个小程序账号,具体流程可以参照官方教程: https://mp.weixin.qq.com/debug/wxadoc/dev/index.html 开通账户之后,在 “开发设置 ...