什么原因导致组件requestFocus有时会失败?

时间:2012-12-10 22:58:53

标签: java swing focus

我有一个jDialog,其中包含一些需要聚焦的字段。

我看到一些奇怪的行为,有时聚焦失败,如果你点击Tab键,你可以在下面的底层父窗口中看到焦点变化,所以很明显焦点没有被转移。

我在关注时阅读了有趣的文章(由camickr撰写): http://tips4java.wordpress.com/2010/03/14/dialog-focus/ 但这并没有解决问题。

虽然使用该侦听器,我可以轻松地添加调试以尝试查看正在发生的事情......

public class RequestFocusListener implements AncestorListener
{
private boolean removeListener;
    protected static org.slf4j.Logger logger = LoggerFactory.getLogger(RequestFocusListener.class);

/*
 *  Convenience constructor. The listener is only used once and then it is
 *  removed from the component.
 */
public RequestFocusListener() {
    this(true);
}

/*
 *  Constructor that controls whether this listen can be used once or
 *  multiple times.
 *
 *  @param removeListener when true this listener is only invoked once
 *                        otherwise it can be invoked multiple times.
 */
public RequestFocusListener(boolean removeListener) {
        logger.debug("creating RequestFocusListener, removeListener = " + removeListener);
    this.removeListener = removeListener;
}

@Override
public void ancestorAdded(AncestorEvent e)
{
        logger.debug("ancestorAdded detected");
        JComponent component = e.getComponent();
        logger.debug("requesting focus");
        boolean success = component.requestFocusInWindow();
        logger.debug("request focus in window result was: " + success);
        if (!success) {
            logger.debug("KeyboardFocusManager says focus failed.\nfocus owner is " + KeyboardFocusManager.getCurrentKeyboardFocusManager().getFocusOwner());
            logger.debug("displayable="+component.isDisplayable());
            logger.debug("lightweight="+component.isLightweight());
            logger.debug("enabled="+component.isEnabled());
            logger.debug("focusable="+component.isFocusable());
            logger.debug("showing="+component.isShowing());
            logger.debug("isRequestFocusEnabled="+component.isRequestFocusEnabled());
        } else {
            logger.debug("KeyboardFocusManager says we got focus.  focus owner is " + KeyboardFocusManager.getCurrentKeyboardFocusManager().getFocusOwner());
        }
        if (removeListener) {
    component.removeAncestorListener( this );
        }
}

@Override
public void ancestorMoved(AncestorEvent e) {
    }

@Override
public void ancestorRemoved(AncestorEvent e) {
    }

}

然后我将监听器添加到JDialog主面板中的组件

radioButton.addAncestorListener(new RequestFocusAncestorListener());

我得到的输出显示:

displayable=true
lightweight=true
enabled=true
focusable=true
showing=true
isRequestFocusEnabled=true

单步执行代码以查看使请求失败的原因,我看到它在Component.requestFocusHelper上停止:

boolean success = peer.requestFocus(this, temporary, focusedWindowChangeAllowed, time, cause);

我已经读过该组件必须是可显示/可见/可聚焦的,但调试显示没问题。

任何人都可以了解还有什么可能导致requestFocus失败? (并将焦点留在调用父面板中,在本例中为jtable)

很抱歉提前没有提供完整的SSCCE,我试图在一个独立的示例中重现这一点,但不能让它一直失败。

我感谢任何想法/提示。


跟进 -

这似乎是我第一次打开对话框,它会获得焦点,然后当我关闭并重新打开对话框时,焦点并不总是被设置。

有趣的是,关闭对话框后,如果我在再次打开对话框之前更改父对焦,则焦点似乎总是被设置。

2 个答案:

答案 0 :(得分:5)

焦点请求失败的原因有很多种。

首先,Component#requestFocus的Java文档实际上是

  

因为此方法的焦点行为取决于平台,   强烈建议开发人员在使用requestFocusInWindow时   可能的。

  

此组件必须是可显示的,可聚焦的,可见的以及所有组件   祖先(顶级窗口除外)必须是可见的   要求被授予

为了使组件变得可聚焦,组件及其所有祖先必须有效(可显示)。我看到的一个常见错误是人们在创建新窗口时使用requestFocusrequestFocusInWindow,但在该窗口实际显示在屏幕上之前(setVisible并不能保证窗口将立即可见,只有在未来的某个时间它才会变得可见。

在这种情况下,最好的方法是使用WindowListener并监控windowOpened事件。事件然后,我很想使用SwingUtilities#invokeLater来确保窗口实际上可以在屏幕上显示。

另一个问题是依赖isDisplayable。即使组件所在的窗口不在(显示在屏幕上),此方法也可能返回true

当组件连接到本机屏幕资源时,该组件将变为可显示。这可能发生在它的祖先窗口被打包或可见时......实际上我发现很难确定何时可能发生这种情况。

<强>更新

我还应该补充说requestFocus只是一个“请求”。焦点管理子系统可能会否决该请求,因为另一个字段拒绝放弃焦点(例如,当字段InputVerifier#shouldYieldFocus返回false时)

答案 1 :(得分:1)

找到问题的答案。

MadProgrammer的分析是正确的,但问题有点无关,从setVisible返回对话框后有一个线程休眠2秒。

删除此睡眠会使问题停止发生。

现在......至于为什么有2秒的延迟(在setVisible返回后的父面板中)重要是有点神秘,但至少我知道解决方案