再談Qt Focus事件


[2010年07月30日文檔]

在我的前一篇文章<focusInEvent()與focusOutEvent>中,我實現了QLineEdit獲得焦點高亮顯示與失去焦點恢復原樣的操作,是通過重新繼承該類,再重構該事件函數的方式。這篇文章緊跟那篇文章,這里要實現的功能也是一樣的,而是通過另外不同的方式——事件過濾器(eventFilter)。

Qt的事件模型中提供的事件過濾功能使得一個QObject對象可以監視另一個QObject對象中的事件,通過在一個QObject對象中安裝事件過濾器可以在事件到達該對象前捕獲事件,從而起到監視該對象事件的效果。

實現類似功能的另一種方式是通過分別繼承不同的控件類,並重構各控件的事件響應函數,但若窗體中包含大量不同的控件時,每一個控件都必須重新繼承,然后分別重構不同的事件函數,實現比較復雜。事件過濾器可以實現在窗體中監視全部控件的不同事件,方便實現功能擴展。

#ifndef WIDGET_H
#define WIDGET_H
#include <QWidget>
namespace Ui {
class Widget;
}
class Widget : public QWidget
{
     Q_OBJECT
 public:
     explicit Widget(QWidget *parent = 0);
     ~Widget();
 public slots:
     bool eventFilter(QObject *,QEvent *);    //注意這里
 private:
     Ui::Widget *ui;
};
#endif // WIDGET_H
`

#include "widget.h"
#include "ui_widget.h"
#include <QPalette>
Widget::Widget(QWidget *parent) :
        QWidget(parent),
        ui(new Ui::Widget)
{
     ui->setupUi(this);
     ui->lineEdit1->installEventFilter(this);  //在窗體上為lineEdit1安裝過濾器
     ui->lineEdit2->installEventFilter(this);  //在窗體上為lineEdit2安裝過濾器
}

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

bool Widget::eventFilter(QObject *watched, QEvent *event)
{
     if (watched==ui->lineEdit1)         //首先判斷控件(這里指 lineEdit1)
     {
          if (event->type()==QEvent::FocusIn)     //然后再判斷控件的具體事件 (這里指獲得焦點事件)
          {
              QPalette p=QPalette();
              p.setColor(QPalette::Base,Qt::green);
              ui->lineEdit1->setPalette(p);
          }
          else if (event->type()==QEvent::FocusOut)    // 這里指 lineEdit1 控件的失去焦點事件
          {
              QPalette p=QPalette();
              p.setColor(QPalette::Base,Qt::white);
              ui->lineEdit1->setPalette(p);
           }
     }
     if (watched==ui->lineEdit2)           //這里來處理 lineEdit2 , 和處理lineEdit1 是一樣的
     {
          if (event->type()==QEvent::FocusIn)
         {
              QPalette p=QPalette();
              p.setColor(QPalette::Base,Qt::green);
              ui->lineEdit2->setPalette(p);
          }
         else if (event->type()==QEvent::FocusOut)
         {
              QPalette p=QPalette();
              p.setColor(QPalette::Base,Qt::white);
              ui->lineEdit2->setPalette(p);
         }
     }
 return QWidget::eventFilter(watched,event);     // 最后將事件交給上層對話框
}

另外,我在一本書上看到作者有一個例子是關於動態按鈕的:鼠標未按下時沒有任何反應,當鼠標左鍵按下時圖片變大,松開鼠標后又恢復原來的狀態。其實這個效果和我這個例子是一個道理,也就是監聽按鈕的按下事件(QEvent::MouseButtonPress)和釋放事件(QEvent::MouseButtonRelease)

bool EventFilter::eventFilter(QObject *watched,QEvent *event)
{
  if (watched==Label1)
  {
      if (event->type()==QEvent::MouseButtonPress)
      {
            QMouseEvent *mouseEvent=static_cast<QMouseEvent *>event;
            if (mouseEvent->buttons() && Qt::LeftButton)
            {  // 更換一張大一點的圖片    ..........
            }
      if (event->type()==QEvent::MouseButtonRelease)
      {    // 重新換回最初那張圖片 ...........
      }
  return QWidget::eventFilter(watched,event);
}

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM