如何在Qt 4.4.1中检测应用程序级别聚焦?

时间:2008-09-15 13:03:48

标签: c++ qt

我需要确定我的Qt 4.4.1应用程序何时获得焦点。

我已经提出了两种可能的解决方案,但它们都不能完全按照我的意愿工作。

在第一种可能的解决方案中,我将来自qApp的focusChanged()信号连接到SLOT。在插槽中,我检查'旧'指针。如果它为'0',那么我知道我们已经切换到这个应用程序,我做我想做的事情。这似乎是让应用程序检测这里介绍的两个解决方案焦点的最可靠方法,但是遇到了下面描述的问题。

在第二种可能的解决方案中,我覆盖了'focusInEvent()'例程,如果原因是'ActiveWindowFocusReason',我会按照自己的意愿行事。

在这两种解决方案中,代码都会在我不希望的时候执行。

例如,我有这个代码覆盖focusInEvent()例程:

void
ApplicationWindow::focusInEvent( QFocusEvent* p_event )
{

  Qt::FocusReason reason = p_event->reason();

  if( reason == Qt::ActiveWindowFocusReason && 
      hasNewUpstreamData() )
  {
    switch( QMessageBox::warning( this, "New Upstream Data Found!",
                                  "New upstream data exists!\n"
                                  "Do you want to refresh this simulation?",
                                  "&Yes", "&No", 0, 0, 1 ) )
    { 
    case 0: // Yes
      refreshSimulation();
      break;
    case 1: // No
      break;
    }
  }
}

执行此操作时,将显示QMessageBox对话框。但是,当通过按“是”或“否”关闭对话框时,会立即再次调用此函数,因为我认为焦点在此时使用ActiveWindowFocusReason更改回应用程序窗口。显然我不希望这种情况发生。

同样,如果用户正在使用应用程序打开&关闭对话框和窗口等,我不想激活这个例程。注意:我不确定这个例程被激活的情况,因为我已经尝试了一下,并且它不会发生在所有窗口和&对话框,尽管它至少发生在示例代码中显示的那个。

如果应用程序是从此应用程序外部聚焦的,而不是从其他对话框窗口聚焦主窗口时,我只想激活它。

这可能吗?怎么办呢?

感谢您提供任何信息,因为这对我们的申请非常重要。

雷蒙德。

3 个答案:

答案 0 :(得分:7)

我认为您需要跟踪QEvent::ApplicationActivate事件。

您可以在QApplication实例上放置event filter,然后查找它。

bool
ApplicationWindow::eventFilter( QObject * watched, QEvent * event )
{
    if ( watched != qApp )
        goto finished;

    if ( event->type() != QEvent::ApplicationActivate )
        goto finished;

    // Invariant: we are now looking at an application activate event for
    //            the application object
    if ( !hasNewUpstreamData() )
        goto finished;

    QMessageBox::StandardButton response =
            QMessageBox::warning( this, "New Upstream Data Found!",
                                  "New upstream data exists!\n"
                                  "Do you want to refresh this simulation?",
                                  QMessageBox::Yes | QMessageBox::No) );

    if ( response == QMessageBox::Yes )
      refreshSimulation();

finished:
    return <The-Superclass-here>::eventFilter( watched, event );
}

ApplicationWindow::ApplicationWindow(...)
{
    if (qApp)
        qApp->installEventFilter( this );
    ...
}

答案 1 :(得分:1)

当您的对话框打开时,键盘事件不会进入主窗口。关闭对话框后,他们会这样做。这是一个焦点变化。如果要忽略焦点从应用程序中的另一个窗口切换的情况,则需要知道应用程序中的任何窗口何时具有焦点。创建一个变量并为您的函数添加更多逻辑。这需要注意,因为在主窗口获得焦点之前对话框将失去焦点。

答案 2 :(得分:0)

查看Qt文档时,似乎每次窗口小部件获得焦点时都会创建焦点事件,因此您发布的示例代码将因您说明的原因而无法工作。

我猜测QApplication :: focusedChanged不能按照你想要的方式工作,因为有些小部件不接受键盘事件,所以即使在同一个应用程序中更改焦点,也会返回null作为“旧”小部件。

我想知道你是否可以对QApplication :: activeWindow()

做任何事情
  

返回具有键盘输入焦点的应用程序顶级窗口,如果没有应用程序窗口具有焦点,则返回0。请注意,即使没有focusWidget(),也可能存在activeWindow(),例如,如果该窗口中没有窗口小部件接受关键事件。