TITLE:JInternalFrameをModalにする
#navi(../)
RIGHT:Posted by [[aterai]] at 2007-10-15
*JInternalFrameをModalにする [#le7c13f6]
JInternalFrameをModalにして、他のJInternalFrameなどを操作できないようにブロックします。

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

//#screenshot
#ref(http://lh6.ggpht.com/_9Z4BYR88imo/TQTP9wW-lJI/AAAAAAAAAe0/xQ9vJrX3MuQ/s800/ModalInternalFrame.png)

**サンプルコード [#d9d01b15]
#code{{
//menuItem.setMnemonic(KeyEvent.VK_1);
class ModalInternalFrameAction1 extends AbstractAction {
  public ModalInternalFrameAction1(String label) {
    super(label);
  }
  public void actionPerformed(ActionEvent e) {
    setJMenuEnabled(false);
    JOptionPane.showInternalMessageDialog(
      desktop, "information", "modal1", JOptionPane.INFORMATION_MESSAGE);
    setJMenuEnabled(true);
  }
}
}}
#code{{
//menuItem.setMnemonic(KeyEvent.VK_2);
class ModalInternalFrameAction2 extends AbstractAction {
  private final JPanel glass = new MyGlassPane();
  public ModalInternalFrameAction2(String label) {
    super(label);
    Rectangle screen = frame.getGraphicsConfiguration().getBounds();
    glass.setBorder(BorderFactory.createEmptyBorder());
    glass.setLocation(0,0);
    glass.setSize(screen.width, screen.height);
    glass.setOpaque(false);
    glass.setVisible(false);
    desktop.add(glass, JLayeredPane.MODAL_LAYER);
  }
  public void actionPerformed(ActionEvent e) {
    setJMenuEnabled(false);
    glass.setVisible(true);
    JOptionPane.showInternalMessageDialog(
      desktop, "information", "modal2", JOptionPane.INFORMATION_MESSAGE);
    glass.setVisible(false);
    setJMenuEnabled(true);
  }
}
}}
#code{{
//menuItem.setMnemonic(KeyEvent.VK_3);
//Creating Modal Internal Frames -- Approach 1 and Approach 2
//http://java.sun.com/developer/JDCTechTips/2001/tt1220.html
class ModalInternalFrameAction3 extends AbstractAction {
  private final Component orgGlassPane;
  private final JPanel glass = new PrintGlassPane();
  public ModalInternalFrameAction3(String label) {
    super(label);
    orgGlassPane = frame.getGlassPane();
    glass.setVisible(false);
  }
  public void actionPerformed(ActionEvent e) {
    JOptionPane optionPane = new JOptionPane();
    optionPane.setMessage("Hello, World");
    optionPane.setMessageType(JOptionPane.INFORMATION_MESSAGE);
    JInternalFrame modal = optionPane.createInternalFrame(desktop, "modal3");
    removeSystemMenuListener(modal);
    modal.addInternalFrameListener(new InternalFrameAdapter() {
      public void internalFrameClosed(InternalFrameEvent e) {
        glass.setVisible(false);
        frame.setGlassPane(orgGlassPane);
      }
    });
    glass.add(modal);
    Rectangle screen = desktop.getBounds();
    modal.setLocation(screen.x + screen.width/2  - modal.getSize().width/2,
                      screen.y + screen.height/2 - modal.getSize().height/2);
    frame.setGlassPane(glass);
    glass.setVisible(true);
    modal.setVisible(true);
    try{
      modal.setSelected(true);
    }catch(java.beans.PropertyVetoException ex) {}
  }
}
}}

**解説 [#v497998b]
-Alt+1: JOptionPane.showInternalMessageDialogメソッドを使用して、簡単なメッセージを表示するModalなDialogをJDeskTopPane内に表示します。
-- Mnemonicがうまく無効に出来ないため、Alt+Bでボタンを押すことが出来てしまいます(Mnemonicを設定したコンポーネントはsetEnabled(false)とする必要がある)。
-- MnemonicをJMenuに設定しているとsetEnabled(false)としても、Altキーに反応してしまう(Windows L&Fだけ?)ので、ダイアログを表示している間は、JMenuBarをダミーと入れ替えています。
-- このダイアログを閉じない限り、アプリケーションをAlt+F4などで閉じることは出来ません。
-- ダイアログのシステムメニュー(左上のアイコンをクリックすると表示される)がマウスで操作できません。

-Alt+2: Alt+1と同様ですが、%%もう一枚タイトルバーなどを削除した半透明なJInternalFrameを下の%% 半透明なGlassPaneをJLayeredPane.MODAL_LAYERに追加して表示しています。
-- JDeskTop内にマスクが掛かります。

-Alt+3: JFrameに半透明なGlassPaneを追加し、そこにJInternalFrameを追加することでModalにしています。
-- JFrame内全体(JMenuBarなども含む)にマスクが掛かります。
-- %%ダイアログのシステムメニューが自身のレイヤーより奥に表示されるため、アイコン(JLabel)をクリックしても反応しないようにリスナーを取り除いています。%%
-- このダイアログを開いていても、アプリケーションをAlt+F4などで閉じることが出来てしまいます。

----
[http://weblogs.java.net/blog/alexfromsun/archive/2008/01/ Alexander Potochkin's Blog: Disabling Swing Containers, the final solution?]を参考に((paint ではなく、print を使うのがポイント。これは便利、他にも色々使えそう。))して、GlassPane を以下のように修正すると、上記のサンプルのAlt+3((Alt+2の場合は、描画が乱れる))は、Mnemonic もうまくブロックできるようです。
-JFrameのメニューバーのMnemonicもブロックできる
-JFrameのシステムメニューはブロックできない
-モーダルにしたJInternalFrameのシステムメニューは表示されない
--ただし表示されないだけで、クリックしてからカーソル移動やダブルクリックなどが動いてしまう
-モーダルにしたJInternalFrameの右上の閉じるボタンのJToolTip がJDesktopPane内で表示される場合、空白になる %%JDK 6 ではJToolTipは表示されない%%

#code{{
class PrintGlassPane extends JPanel {
  TexturePaint texture = TextureFactory.createCheckerTexture(4);
  public PrintGlassPane() {
    super((LayoutManager)null);
    setOpaque(false);
  }
  @Override
  public void setVisible(boolean isVisible) {
    boolean oldVisible = isVisible();
    super.setVisible(isVisible);
    JRootPane rootPane = SwingUtilities.getRootPane(this);
    if(rootPane!=null && isVisible()!=oldVisible) {
      rootPane.getLayeredPane().setVisible(!isVisible);
    }
  }
  @Override
  public void paintComponent(Graphics g) {
    JRootPane rootPane = SwingUtilities.getRootPane(this);
    if(rootPane!=null) {
      //http://weblogs.java.net/blog/alexfromsun/archive/2008/01/
      // it is important to call print() instead of paint() here
      // because print() doesn't affect the frame's double buffer
      rootPane.getLayeredPane().print(g);
    }
    Graphics2D g2 = (Graphics2D) g;
    g2.setPaint(texture);
    g2.fillRect(0,0,getWidth(),getHeight());
  }
}
}}

----
JDK 6 の場合、[http://tips4java.wordpress.com/2008/11/07/disabled-glass-pane/ Disabled Glass Pane « Java Tips Weblog]のようにキー入力を無効にするキーリスナーを追加する方法もあります。

この方法は、JDK 5 などの場合、WindowsLnF で、Altキーを押すとメニューバーにフォーカスが移ることがあります。

**参考リンク [#n6c5a455]
-[http://java.sun.com/developer/JDCTechTips/2001/tt1220.html Creating Modal Internal Frames -- Approach 1 and Approach 2]
-[http://java.sun.com/docs/books/tutorial/uiswing/components/rootpane.html How to Use Root Panes]
-[http://weblogs.java.net/blog/alexfromsun/archive/2008/01/ Disabling Swing Containers, the final solution?]
--[[Cursorを砂時計に変更>Swing/WaitCursor]]

**コメント [#h8218ae6]
- %%[[JInternalFrameを半透明にする>Swing/TransparentFrame]]と、同様にGlassPaneがUbuntu(GNOME)などで半透明にならない場合があります。%% -- [[aterai]] &new{2007-10-15 (月) 13:16:07};
-- (Alt+2)で開いた場合、JInternalFrameにGlassPaneを乗せるのではなく、直接JDeskTopPaneのJLayeredPane.MODAL_LAYERに追加するように変更しました。 -- [[aterai]] &new{2007-10-16 (火) 17:31:50};
- メモ: [http://weblogs.java.net/blog/alexfromsun/archive/2008/01/ Alexander Potochkin's Blog: Disabling Swing Containers, the final solution?]のサンプルでは、Mnemonic もちゃんとブロックできているようなので、「あとで調べる & 参考にする」こと。 -- [[aterai]] &new{2008-01-25 (金) 17:28:21};
- Mnemonic を数字(1-3)に変更 -- [[aterai]] &new{2008-04-25 (金) 20:51:49};

#comment