Как распознать QMouseEvent внутри дочерних виджетов?
РЕДАКТИРОВАТЬ и некоторую самокритичность Я испробовал оба приведенных решения, которые оба решили мою проблему, и поэтому я благодарю вас обоих! Я пометил прозрачное решение как принятое, потому что я думал, что это самая простая реализация, когда у меня был только один дочерний виджет, но я хочу поделиться некоторыми соображениями с другими новичками:
Я впервые использовал QLabel
, который в настоящее время включил Qt::WA_TransparentForMouseEvents
по умолчанию и, таким образом, очевидно, работал, но я также хотел, чтобы текст можно было выбрать, используя QPlainTextEdit
вместо. Смехотворно, это никак не возможно, потому что если вы попытаетесь выделить текст (нажав), вы закроете окно! В итоге я сохранил прозрачность и пренебрегал возможностью выбора текста.
Я предполагаю, что на мой следующий вопрос уже был дан ответ где-то раньше, но после часа поиска я теперь отправляю вопрос сам. Я благодарен, если кто-то может указать мне на уже отвеченный вопрос, который решает мою проблему.
Во всяком случае, я создаю всплывающее окно, используя C++ и Qt. Я создал следующее PopupDialog
класс, который работает хорошо и хорошо для всех своих целей. Однако я удалил его фрейм (включая панель, содержащую кнопку закрытия и заголовок окна), чтобы он выглядел минималистично, и теперь я хочу, чтобы он закрывался, если пользователь нажимает / отпускает кнопку мыши в любом месте всплывающего окна (диалоговое окно),
Приведенный ниже код работает, однако таким образом, что я должен щелкнуть и отпустить мышь точно в самом окне QDialog. Он не закроется, когда я нажму, если курсор мыши над дочерним виджетом (ами) внутри QDialog
например, QPlainTextEdit
, который отображает текст.
Следовательно, мне нужно решение для QDialog
распознавать QMouseEvents
внутри своих дочерних виджетов. Пожалуйста, не стесняйтесь спрашивать, если что-то неясно. Я не включил свои файлы mainwindow.h/.cpp или popupdialog.ui
файл, так как я считаю, что было бы слишком много, чтобы разместить здесь, но .ui
очень просто: просто QDialog
окно держит QBoxLayout
, содержащий один виджет, QPlainTextEdit
, Я могу опубликовать их по запросу, если это поможет.
// popupdialog.h
#ifndef POPUPDIALOG_H
#define POPUPDIALOG_H
#include <QDialog>
#include <QString>
namespace Ui {class PopupDialog;}
class PopupDialog : public QDialog
{
Q_OBJECT
public:
explicit PopupDialog(QWidget *parent = 0, QString msgTxt="");
~PopupDialog();
private:
Ui::PopupDialog *ui;
QString messageText;
void mouseReleaseEvent(QMouseEvent*);
};
#endif //POPUPDIALOG_H
...
// popupdialog.cpp
#include "popupdialog.h"
#include "ui_popupdialog.h"
PopupDialog::PopupDialog(QWidget *parent, QString msgTxt) :
QDialog(parent),
ui(new Ui::PopupDialog),
messageText(msgTxt)
{
ui->setupUi(this);
setWindowFlags(Qt::Window | Qt::FramelessWindowHint);
setModal(true);
ui->message_text_display->setText(messageText);
// The message_text_display is an instance of the class,
// "PlainTextEdit". Using "QLabel" partly solves my
// problem, but does not allow text selection.
}
PopupDialog::~PopupDialog()
{
delete ui;
}
void PopupDialog::mouseReleaseEvent(QMouseEvent *e)
{
this->close();
}
2 ответа
Как вы уже заметили, события мыши обрабатываются дочерними виджетами и передаются родителям, если они не принимаются. Вы можете прочитать больше об этом здесь
Чтобы закрыть всплывающее окно, когда щелчок сделан внутри дочернего виджета, вы можете сделать две вещи. Вы можете попробовать установить installEventFilter и настроить его на каждом дочернем виджете для вызова close()
,
Другой вариант потребует от вас иметь вид CentralWidget (например, MainWindow
обычно имеет) - просто сгруппировать все ваши дочерние виджеты. Таким образом, вы можете вызвать setAttribute() для него, чтобы установить Qt::WA_TransparentForMouseEvents
свойство просто пропустить обработку событий мыши на виджете и всех его дочерних элементах.
groupWidget->setAttribute(Qt::WA_TransparentForMouseEvents);
В соответствии с Qt
документы:
Когда этот параметр включен, этот атрибут отключает доставку событий мыши для виджета и его дочерних элементов. События мыши доставляются в другие виджеты, как если бы виджет и его дочерние элементы отсутствовали в иерархии виджетов; щелчки мыши и другие события эффективно "проходят" через них. Этот атрибут по умолчанию отключен.
Что в основном означает, что событие будет передано по цепочке первому виджету, который может обработать событие. В вашем случае это будет PopupDialog
и уже переопределено mouseReleaseEvent
слот.
В заголовочном файле
class PopupDialog : public QDialog
{
Q_OBJECT
public:
explicit PopupDialog(QWidget *parent = 0, QString msgTxt="");
~PopupDialog();
//////////////////////////////////
protected:
bool eventFilter(QObject *obj, QEvent *event);
//////////////////////////////////////
private:
Ui::PopupDialog *ui;
QString messageText;
void mouseReleaseEvent(QMouseEvent*);
};
в кпп
PopupDialog::PopupDialog(QWidget *parent, QString msgTxt) :
QDialog(parent),
ui(new Ui::PopupDialog),
messageText(msgTxt)
{
ui->setupUi(this);
setWindowFlags(Qt::Window | Qt::FramelessWindowHint);
setModal(true);
ui->message_text_display->setText(messageText);
// The message_text_display is an instance of the class,
// "PlainTextEdit". Using "QLabel" partly solves my
// problem, but does not allow text selection.
///////////////////////////////////////
foreach (QObject *child, children())
{
child->installEventFilter(this);
}
///////////////////////////////////////
}
///////////////////////////////////////
bool PopupDialog::eventFilter(QObject *obj, QEvent *event)
{
if(event->type() == QEvent::MouseButtonRelease)
{
this->close();
}
}