如何在Qt中实现QHoverEvent?

问题描述 投票:7回答:6

我只是用C ++学习Qt。我已经成功地实现了信号和插槽以捕获标准事件,如ButtonPushed()等。但是,我希望有一个函数调用当我鼠标移开并且鼠标移出QLabel。看起来QHoverEvent会做我需要的,但我似乎找不到任何有关如何实现它的教程或示例。它是否与信号和插槽完全相同?我试过了:

connect(ui.lbl_test, SIGNAL(QHoverEvent), this, SLOT(TestFunc(QEvent::Type type, const QPoint & pos, const QPoint & oldPos)));

..但是当我在标签上盘旋时,功能没有被调用。

这是函数,在头文件中列为公共插槽:

void MyDialog::TestFunc(QEvent::Type type, const QPoint & pos, const QPoint & oldPos) {
     QMessageBox::information(this, tr("Hey"), tr("Listen!"));
}

任何人都可以帮我解决这个问题或者给我一个很好的例子吗?

编辑:

阅读下面的帖子后,我发现没有setFlag()会员要求我的标签小部件,但我确实尝试:

    ui.lbl_test->setMouseTracking(true);
    connect(ui.lbl_test, SIGNAL(ui.lbl_test->mouseMoveEvent()), this, SLOT(TestFunc(QMouseEvent *event)));

并据此更新TestFunc()。但是当我鼠标移动时仍然没有任何反应。

看后我不确定QLabel甚至从QWidget继承了mouseMoveEvent()。如果这是真的,是否有一个小部件,或者在某个地方继承它的对象列表?我从他们网站上的文档中可以看出,对象有多少继承的函数。

c++ qt events
6个回答
7
投票

为此目的使用信号和插槽不起作用。

mouseMoveEvent()不是信号或元方法,不能连接到插槽。

对窗口小部件类进行子类化并覆盖mouseMoveEvent()将允许您获取鼠标移动事件,但这是实现此目的的一种非常重要的方法(并为您的源代码添加一个类)。

相反,考虑在eventFilter()类上实现MyDialog方法并将其安装在QLabel上。使用此事件过滤器方法,您可以拦截给定QObject实例的所有事件。

以下是有关事件过滤器的文档。

http://doc.qt.io/qt-4.8/eventsandfilters.html#event-filters

此外,通过查看代码示例,我建议您花一点时间来研究SIGNAL()SLOT()宏的作用。您可以在$QTDIR/src/corelib/kernel/qobjectdefs.h中看到它们的定义方式


13
投票

根据您提供的文档链接,如果您的小部件具有Qt::WA_Hover标志,那么您将只获得此QHoverEvent。

构建小部件调用后:

widget->setAttribute(Qt::WA_Hover);

看看它是否有效

实现相同结果的另一种方法是覆盖窗口小部件中的mouseMoveEvent() 请注意,除非你打电话,否则通常不会调用此函数:

widget->setMouseTracking(true);

这基本上是QT如何在内部实现悬停事件。


8
投票

http://qt-project.org/doc/qt-5/qwidget.html#enterEvent

http://qt-project.org/doc/qt-5/qwidget.html#leaveEvent

http://qt-project.org/doc/qt-5/qt.html#widget-attributes

Qt :: _ _ ho r

迫使Qt在鼠标进入或离开小部件时生成绘画事件。实现自定义样式时通常使用此功能;有关详细信息,请参阅样式示例。

http://qt-project.org/doc/qt-5/qtwidgets-widgets-styles-example.html#norwegianwoodstyle-class-implementation

这个QStyle::polish()重载在使用该样式绘制的每个小部件上调用一次。我们重新实现它以在Qt::WA_HoverQPushButtons上设置QComboBoxes属性。设置此属性后,Qt会在鼠标指针进入或离开窗口小部件时生成绘制事件。这使得当鼠标指针悬停在按钮和组合框上时,可以以不同方式呈现按钮和组合框。

如何在QWidget上接收进入和离开事件

  1. 设置WA_Hover的Widget属性 // in your widget's constructor (probably) this->setAttribute(Qt::WA_HOVER, true);
  2. 实施QWidget::enterEvent()QWidget::leaveEvent()void Widget::enterEvent(QEvent * event) { qDebug() << Q_FUNC_INFO << this->objectName(); QWidget::enterEvent(event); } void Widget::leaveEvent(QEvent * event) { qDebug() << Q_FUNC_INFO << this->objectName(); QWidget::leaveEvent(event); }
  3. 完成

QWidget中的QHoverEvent

http://qt-project.org/doc/qt-5/qhoverevent.html#details

http://qt-project.org/doc/qt-5/qobject.html#event

http://qt-project.org/doc/qt-5/qwidget.html#event

// in your widget's constructor (probably)
this->setAttribute(Qt::WA_HOVER, true);
// ...

void Widget::hoverEnter(QHoverEvent * event) {qDebug() << Q_FUNC_INFO << this->objectName();}
void Widget::hoverLeave(QHoverEvent * event) {qDebug() << Q_FUNC_INFO << this->objectName();}
void Widget::hoverMove(QHoverEvent * event) {qDebug() << Q_FUNC_INFO << this->objectName();}

bool Widget::event(QEvent * e)
{
    switch(e->type())
    {
    case QEvent::HoverEnter:
        hoverEnter(static_cast<QHoverEvent*>(e));
        return true;
        break;
    case QEvent::HoverLeave:
        hoverLeave(static_cast<QHoverEvent*>(e));
        return true;
        break;
    case QEvent::HoverMove:
        hoverMove(static_cast<QHoverEvent*>(e));
        return true;
        break;
    default:
        break;
    }
    return QWidget::event(e);
}

更新:

简单的例子

将鼠标悬停在按钮上,查看计数更改。查看应用程序输出以获取更多信息。

https://gist.github.com/peteristhegreat/d6564cd0992351f98aa94f869be36f77

希望有所帮助。


1
投票

QHoverEvent仅用于悬停小部件,您希望通过子类化小部件来实现enterEventleaveEvent处理程序。如果要使用事件过滤器,则相应的事件类型为QEvent::EnterQEvent::Leave

如果您只是需要更改窗口小部件的外观,您可能需要查看Qt样式表,因为它们提供了:hover选择器。


1
投票

我有同样的问题,我来到以下解决方案设计:

在我的主要小部件中,我想处理某些选定对象的悬停事件。出于这个原因,我在MainWindow上创建了2个插槽:

public slots:
    void onHoverIn(QObject* object);
    void onHoverOut(QObject* object);

然后我创建了一个这样的事件过滤器类:

hovereventfilter.h

#ifndef HOVEREVENTFILTER_H
#define HOVEREVENTFILTER_H

#include <QObject>
#include <QEvent>

class HoverEventFilter : public QObject
{
    Q_OBJECT
public:
    explicit HoverEventFilter(QObject *parent = 0);

signals:
    void HoverIn(QObject *);
    void HoverOut(QObject *);

public slots:

protected:
    bool eventFilter(QObject *watched, QEvent *event);
};

#endif // HOVEREVENTFILTER_H

hovereventfilter.cpp

#include "hovereventfilter.h"

HoverEventFilter::HoverEventFilter(QObject *parent) : QObject(parent)
{

}

bool HoverEventFilter::eventFilter(QObject *watched, QEvent *event)
{

    QEvent::Type t = event->type();

    switch(t){
    case QEvent::Enter:
        emit HoverIn(watched);
        break;

    case QEvent::Leave:
        emit HoverOut(watched);
        break;
    default:
        return false;
    }

    return true;
}

您可以看到此类将根据发生的情况触发HoverIn或HoverOut。这种方法不需要设置Qt::WA_Hover

现在作为最后一步,我们需要指示要过滤哪些元素并连接信号和插槽。我将在mainwindow.h中创建一个指向事件过滤器的私有指针

class MainWindow : public QWidget
{
    Q_OBJECT

...

public slots:
    void onHoverIn(QObject* object);
    void onHoverOut(QObject* object);

private:

    HoverEventFilter* hoverEventFilter;

...

};

在构造函数中,我添加:

this->hoverEventFilter = new HoverEventFilter(this);

connect(this->hoverEventFilter, SIGNAL(HoverIn(QObject*)), this, SLOT(onHoverIn(QObject*)));
connect(this->hoverEventFilter, SIGNAL(HoverOut(QObject*)), this, SLOT(onHoverOut(QObject*)));

现在,每当我希望在某个对象上接收悬停事件时,我只需在其上设置事件过滤器,如下所示:

this->ui->someLabelOrWhatever->installEventFilter(this->hoverEventFilter);

剩下的是在onHoverIn实施onHoverOutMainWindow。他们都有相同的想法所以我只会展示onHoverIn

void MainWindow::onHoverIn(QObject *object)
{
    QString objectName = object->objectName();

    switch(objectName){
        // do something depending on name of the widget
    }
}

你可以很容易地扩展它,在新项目上处理悬停事件,你只需在它上面设置一个事件监听器,并在onHoverInonHoverOut方法中处理你想做的事情。无需子类化任何小部件。


-3
投票

您需要子类化或过滤该特定小部件的focusInEventfocusOutEvent

© www.soinside.com 2019 - 2024. All rights reserved.