事件(event)是由串口系统或者Qt自身产生的,用以响应所发生的各类事情。当用户按下或者松开键盘或者鼠标上的按键时,就可以产生一个键盘或者鼠标事件;当某个窗口第一次显示的时候,就会产生一个绘制事件。用来告知窗口需要重绘制它本身,从而使得该窗口可见。
使用Qt进行编程开发时,基本不需要考虑事件,Qt窗口部件都会发射信号。但是当我们需要编写自己的自定义窗口部件,或者是当我们希望改变已经存在的Qt窗口部件的行为时,事件就变得非常有用了。
事件和信号是两个概念。在使用窗口部件时,信号是十分有用的,而在实现窗口部件时,事件则是十分有用的。
7.1重新实现事件处理器
在Qt中,事件就是QEvent子类的一个实例。
7.2安装事件过滤器
Qt事件模型一个非常强大的功能是:QObject实例在看到它自己的事件之前,可以通过设置另外一个QObject实例先监视这个事件。
7.3处理密集时的相应保持
当调用QApplication:exec()时,就启动了Qt的事件循环。在开始时,Qt会发出一些事件命令来显示和绘制窗口部件。在这之后,事件循环就开始运行,它不断检查是否有事件发生并且把这个事件发送给应用程序的QObject。
定时器事件,定时器事件允许应用程序可以再一定的时间间隔后执行事件处理。定时器事件可以用来实现光标的闪烁和其他动画的播放,或者只简单地用作显示的刷新。
Ticker窗口部件实例:显示了一串文本标语,每30毫秒向左移动一个像素,如果窗口部件比文本宽,那么文本将会被多次重复,直到能够填满整个窗口部件的宽度为止。
源码如下:
ticker.h
#ifndef TICKER_H
#define TICKER_H
#include <QWidget>
class Ticker : public QWidget
{
Q_OBJECT
Q_PROPERTY(QString text READ text WRITE setText)
public:
Ticker(QWidget *parent = 0); //析构
void setText(const QString &newText); //设置要显示的文本
QString text() const { return myText; }
QSize sizeHint() const;
protected:
void paintEvent(QPaintEvent *event);
void timerEvent(QTimerEvent *event);
void showEvent(QShowEvent *event);
void hideEvent(QHideEvent *event);
private:
QString myText;
int offset;
int myTimerId;
};
#endif
ticker.cpp
#include <QtGui>
#include "ticker.h"
Ticker::Ticker(QWidget *parent)
: QWidget(parent)
{
offset = 0;
myTimerId = 0;
}
void Ticker::setText(const QString &newText) //设置要显示的文本
{
myText = newText;
update(); //强制执行一个重绘操作
updateGeometry(); //通知对Ticker窗口部件负责的任意布局管理器,提示该窗口部件的大小发生了变化。
}
QSize Ticker::sizeHint() const //返回文本所需的空间大小,并以此作为窗口部件的理想尺寸。
{
return fontMetrics().size(0, text()); //返回一个QFontMetrics对象,可以用这个对象查询并获得与这个窗口部件字体相关的信息
}
void Ticker::paintEvent(QPaintEvent * /* event */) //使用QPainter::drawText()绘制文本
{
QPainter painter(this);
int textWidth = fontMetrics().width(text()); //使用fontMetrics()确定文本在水平方向上所需的空间
if (textWidth < 1)
return;
int x = -offset;
while (x < width()) { //考虑offset的值,多次绘制文本,填充整个窗口部件的宽度为止
painter.drawText(x, 0, textWidth, height(),
Qt::AlignLeft | Qt::AlignVCenter, text());
x += textWidth;
}
}
void Ticker::showEvent(QShowEvent * /* event */) //启动定时器
{
myTimerId = startTimer(30);
}
void Ticker::timerEvent(QTimerEvent *event)
{
if (event->timerId() == myTimerId) {
++offset;
if (offset >= fontMetrics().width(text()))
offset = 0;
scroll(-1, 0); //把窗口部件的内容向左滚动一个像素
} else {
QWidget::timerEvent(event);
}
}
void Ticker::hideEvent(QHideEvent * /* event */)
{
killTimer(myTimerId); 停止定时器
myTimerId = 0;
}