How to detect user inactivity in Qt?
How can I detect user inactivity in a Qt QMainWindow? My idea so far is to have a QTimer that increments a counter, which, if a certain value is passed, locks the application. Any mouse or key interaction should set the timer back to 0. However I need to know how to properly handle input events which reset; I can re-implement:
virtual void keyPressEvent(QKeyEvent *event)
virtual void keyReleaseEvent(QKeyEvent *event)
virtual void mouseDoubleClickEvent(QMouseEvent *event)
virtual void mouseMoveEvent(QMouseEvent *event)
virtual开发者_运维百科 void mousePressEvent(QMouseEvent *event)
virtual void mouseReleaseEvent(QMouseEvent *event)
...but won't the event handlers of all the widgets in the QMainWindow prevent events occurring in those controls from reaching the QMainWindow's? Is there a better architecture for detecting user activity as it is?
You could use a custom event filter to process all keyboard and mouse events received by your application before they are passed on to the child widgets.
class MyEventFilter : public QObject
{
Q_OBJECT
protected:
bool eventFilter(QObject *obj, QEvent *ev)
{
if(ev->type() == QEvent::KeyPress ||
ev->type() == QEvent::MouseMove)
// now reset your timer, for example
resetMyTimer();
return QObject::eventFilter(obj, ev);
}
}
Then use something like
MyApplication app(argc, argv);
MyEventFilter filter;
app.installEventFilter(&filter);
app.exec();
This definitely works (I've tried it myself).
EDIT: And many thanks to ereOn for pointing out that my earlier solution was not very useful.
One of better approach will be to catch xidle signal rather then catching so many events from user. Here one need to capture QEvent:MouseMove event also
The cleanest way is to override the eventFilter
function of your main window widget and set it as event filter on your application object.
Inside the filter you can use dynamic_cast
to check if the event is a QInputEvent
. All events with user interaction are derived from QInputEvent
and are recognized this way.
class MainWindow: public QWidget {
public:
MainWindow() {
QApplication::instance()->installEventFilter(this);
}
protected:
bool eventFilter(QObject* target, QEvent* event) override {
if(dynamic_cast<QInputEvent*>(event)){
// detected user interaction
}
return QWidget::eventFilter(target, event);
}
};
You can replace the base class QWidget
with any class derived from QWidget
. (Including QMainWindow
.) Note that the event
function must pass the event to the base class, and return its return value.
If you have more then one window, you might also want to check, that the event target object is your window or one of its QWidget
children.
class MainWindow: public QWidget {
public:
MainWindow() {
QApplication::instance()->installEventFilter(this);
}
protected:
bool eventFilter(QObject* target, QEvent* event) override {
if(dynamic_cast<QInputEvent*>(event) && (
target == this ||
findChildren<QWidget*>().contains(target))
){
// detected user interaction
}
return QWidget::eventFilter(target, event);
}
};
精彩评论