我有一个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,我试图在一个独立的示例中重现这一点,但不能让它一直失败。
我感谢任何想法/提示。
跟进 -
这似乎是我第一次打开对话框,它会获得焦点,然后当我关闭并重新打开对话框时,焦点并不总是被设置。
有趣的是,关闭对话框后,如果我在再次打开对话框之前更改父对焦,则焦点似乎总是被设置。
答案 0 :(得分:5)
焦点请求失败的原因有很多种。
首先,Component#requestFocus
的Java文档实际上是
因为此方法的焦点行为取决于平台, 强烈建议开发人员在使用requestFocusInWindow时 可能的。
和
此组件必须是可显示的,可聚焦的,可见的以及所有组件 祖先(顶级窗口除外)必须是可见的 要求被授予
为了使组件变得可聚焦,组件及其所有祖先必须有效(可显示)。我看到的一个常见错误是人们在创建新窗口时使用requestFocus
或requestFocusInWindow
,但在该窗口实际显示在屏幕上之前(setVisible
并不能保证窗口将立即可见,只有在未来的某个时间它才会变得可见。
在这种情况下,最好的方法是使用WindowListener
并监控windowOpened
事件。事件然后,我很想使用SwingUtilities#invokeLater
来确保窗口实际上可以在屏幕上显示。
另一个问题是依赖isDisplayable
。即使组件所在的窗口不在(显示在屏幕上),此方法也可能返回true
。
当组件连接到本机屏幕资源时,该组件将变为可显示。这可能发生在它的祖先窗口被打包或可见时......实际上我发现很难确定何时可能发生这种情况。
<强>更新强>
我还应该补充说requestFocus
只是一个“请求”。焦点管理子系统可能会否决该请求,因为另一个字段拒绝放弃焦点(例如,当字段InputVerifier#shouldYieldFocus
返回false
时)
答案 1 :(得分:1)
找到问题的答案。
MadProgrammer的分析是正确的,但问题有点无关,从setVisible返回对话框后有一个线程休眠2秒。
删除此睡眠会使问题停止发生。
现在......至于为什么有2秒的延迟(在setVisible返回后的父面板中)重要是有点神秘,但至少我知道解决方案