TITLE:Windowを開いたときのフォーカスを指定
#navi(../)
#tags()
RIGHT:Posted by &author(aterai); at 2004-10-11
*Windowを開いたときのフォーカスを指定 [#q0754f35]
JFrameやJDialogなどのWindowを開いたときに、デフォルトでフォーカスを持つコンポーネントを指定します。

-&jnlp;
-&jar;
-&zip;

//#screenshot
#ref(http://lh3.ggpht.com/_9Z4BYR88imo/TQTKp09XXEI/AAAAAAAAAWU/p3YhSijyS90/s800/DefaultFocus.png)

**サンプルコード [#c961d3aa]
#code(link){{
EventQueue.invokeLater(new Runnable() {
  @Override public void run() {
    field.requestFocusInWindow();
  }
});
}}

**解説 [#u7271535]
上記のサンプルでは、JTextFieldがデフォルトのフォーカスを持つように、JComponent#requestFocusInWindowメソッドを使用しています。

requestFocusInWindowメソッドは、チュートリアル([http://docs.oracle.com/javase/tutorial/uiswing/misc/focus.html How to Use the Focus Subsystem])にあるように、frame.pack();した後で実行する必要があります。このため、このサンプルではEventQueue.invokeLaterを使って、待ち状態のすべてのイベントが処理されたあとで実行するようにしています。

----
以下のように、FocusTraversalPolicyやWindowListenerを使う方法でも、同様にデフォルトのフォーカスを持つコンポーネントを指定することが出来ます。

-FocusTraversalPolicyを使用
--FocusTraversalPolicy を設定して、最初にフォーカスの当たるコンポーネントを指定(JDK 1.4 以降)
#code{{
frame.setFocusTraversalPolicy(new LayoutFocusTraversalPolicy() {
  @Override public Component getInitialComponent(Window w) {
    return field;
  }
});
}}
-WindowListener#windowOpened で、requestFocusInWindow
--フレームにWindowListenerを設定して、windowOpenedが呼び出されたときに、requestFocusInWindow
#code{{
frame.addWindowListener(new WindowAdapter() {
  @Override public void windowOpened(WindowEvent e) {
    field.requestFocusInWindow();
  }
});
}}
-ComponentListener#componentShown で、requestFocusInWindow
--フレームにComponentListenerを設定して、componentShownが呼び出されたとき(=フレームがsetVisible(true)されたとき)に、requestFocusInWindow
#code{{
frame.addComponentListener(new ComponentAdapter() {
  @Override public void componentShown(ComponentEvent e) {
    field.requestFocusInWindow();
  }
});
}}

-KeyboardFocusManager#addPropertyChangeListener で、requestFocusInWindow
--KeyboardFocusManagerにPropertyChangeListenerを設定して、propertyChangeが呼び出され、PropertyNameが、"activeWindow"、かつPropertyChangeEvent#getNewValueがnullでないときに、requestFocusInWindow
#code{{
KeyboardFocusManager kfm = KeyboardFocusManager.getCurrentKeyboardFocusManager();
kfm.addPropertyChangeListener(new PropertyChangeListener() {
  @Override public void propertyChange(PropertyChangeEvent e) {
    String prop = e.getPropertyName();
    if("activeWindow".equals(prop) && e.getNewValue()!=null) {
      System.out.println("activeWindow");
      field.requestFocusInWindow();
    }
  }
});
}}

//----
//このサンプルでは、ウィンドウを開いたときにフォーカスを持つコンポーネントとは別に、フォーカスがボタンにない場合((ルート区画内に JTextPane やフォーカスのあるJButtonなどの起動イベントを消費する別のコンポーネントがある場合は除く))でも、Enterキーを押したときに起動されるボタンを以下のメソッドを使って設定しています([http://docs.oracle.com/javase/jp/6/api/javax/swing/JRootPane.html#setDefaultButton(javax.swing.JButton) JRootPane#setDefaultButton(javax.swing.JButton)])。
//#code{{
//frame.getRootPane().setDefaultButton(eb);
//}}

**参考リンク [#z4f652d0]
-[[Focusの移動>Swing/FocusTraversal]]
-[http://forums.sun.com/thread.jspa?threadID=579294 Swing - When does requestFocusInWindow() fail]
-[[JOptionPaneのデフォルトフォーカス>Swing/OptionPaneDefaultFocus]]

**コメント [#x1950fff]
- JFrame#getRootPane()#setDefaultButton()は使用方法がよくわからない… -- [[aterai]] 
- setDefaultButtonって、Enterしたときに押されたとみなすボタンだったかな…… --  &new{2004-10-14 (木) 23:21:53};
-- ありがとうございます。おかげでようやく理解できました。JTextFieldなどを編集状態にしたあと、EnterするとsetDefaultButtonしたボタンが押されるのですね。 -- [[aterai]] &new{2004-10-18 (月) 12:14:15};
-- というわけで、frame.getRootPane().setDefaultButton(eb);を追加してみました。上記のサンプルでは、中央のJTextFieldにフォーカスがある状態で、リターンキーを押すと、EASTボタンが押されたことになります。 -- [[aterai]] &new{2004-10-18 (月) 12:20:58};
-- %%いつか、DefaultButton のページを別に作成すること。 -- [[aterai]] &new{2008-05-07 (水) 19:19:44};%%
-- [[DefaultButtonの設定>Swing/DefaultButton]]に移動。 -- [[aterai]] &new{2008-05-12 (月) 14:40:15};
- HierarchyListenerを使用する場合のテスト -- [[aterai]] &new{2009-03-19 (木) 14:51:30};
-- [[JOptionPaneのデフォルトフォーカス>Swing/OptionPaneDefaultFocus]]に移動

#comment