概要

JComboBoxのドロップダウンリスト(ポップアップメニュー)で、マウスの右クリックを無効にします。

サンプルコード

class BasicComboPopup2 extends BasicComboPopup {
  private transient Handler2 handler2;
  @Override public void uninstallingUI() {
    super.uninstallingUI();
    handler2 = null;
  }
  public BasicComboPopup2(JComboBox combo) {
    super(combo);
  }
  @Override protected MouseListener createListMouseListener() {
    if (handler2 == null) {
      handler2 = new Handler2();
    }
    return handler2;
  }
  private class Handler2 extends MouseAdapter {
    @Override public void mouseReleased(MouseEvent e) {
      if (e.getSource().equals(list)) {
        if (list.getModel().getSize() > 0) {
          // <ins>
          if (!SwingUtilities.isLeftMouseButton(e) || !comboBox.isEnabled()) {
            return;
          }
          // </ins>
          // JList mouse listener
          if (comboBox.getSelectedIndex() == list.getSelectedIndex()) {
            comboBox.getEditor().setItem(list.getSelectedValue());
          }
          comboBox.setSelectedIndex(list.getSelectedIndex());
        }
        comboBox.setPopupVisible(false);
        // workaround for cancelling an edited item (bug 4530953)
        if (comboBox.isEditable() && comboBox.getEditor() != null) {
          comboBox.configureEditor(comboBox.getEditor(), comboBox.getSelectedItem());
        }
      }
    }
  }
}
view all

解説

上記のサンプルでは、ComboBoxUI#createPopup()をオーバーライドして、ドロップダウンリストに設定するMouseListenerを入れ替えたBasicComboPopupを追加しています。

combo02.setUI(new BasicComboBoxUI() {
  @Override protected ComboPopup createPopup() {
    return new BasicComboPopup2(comboBox);
  }
});

元のMouseListenerは、JComboBox全体のHandlerになっていますが、必要なのはドロップダウンリスト関係のみなので、e.getSource() == listな部分だけ元のHandlerからコピーし、この中でif(!SwingUtilities.isLeftMouseButton(e) || !comboBox.isEnabled()) return;と右クリックを無視しています。


以下のような方法もあります。

class BasicComboPopup3 extends BasicComboPopup {
  @SuppressWarnings("unchecked")
  @Override protected JList createList() {
    return new JList(comboBox.getModel()) {
      @Override public void processMouseEvent(MouseEvent e) {
        if (SwingUtilities.isRightMouseButton(e)) {
          return;
        }
        MouseEvent ev = e;
        if (e.isControlDown()) {
          // Fix for 4234053. Filter out the Control Key from the list.
          // ie., don't allow CTRL key deselection.
          Toolkit toolkit = Toolkit.getDefaultToolkit();
          ev = new MouseEvent(e.getComponent(), e.getID(), e.getWhen(),
                              //e.getModifiers() ^ InputEvent.CTRL_MASK,
                              e.getModifiers() ^ toolkit.getMenuShortcutKeyMask(),
                              e.getX(), e.getY(),
                              e.getXOnScreen(), e.getYOnScreen(),
                              e.getClickCount(),
                              e.isPopupTrigger(),
                              MouseEvent.NOBUTTON);
        }
        super.processMouseEvent(ev);
      }
    };
  }
}

コメント

//Test
class BasicComboPopup3 extends BasicComboPopup {
  public BasicComboPopup3(JComboBox combo) {
    super(combo);
  }
  @Override protected JScrollPane createScroller() {
    JScrollPane sp = new JScrollPane(list,
                     ScrollPaneConstants.VERTICAL_SCROLLBAR_AS_NEEDED,
                     ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER) {
      @Override protected void processEvent(AWTEvent e) {
        if (e instanceof MouseWheelEvent) {
          JScrollBar toScroll = getVerticalScrollBar();
          if (toScroll == null || !toScroll.isVisible()) {
            ((MouseWheelEvent) e).consume();
            return;
          }
        }
        super.processEvent(e);
      }
    };
    sp.setHorizontalScrollBar(null);
    return sp;
  }
//...
  • 上記の方法では、ドロップダウンリスト内では閉じなくなるが、ドロップダウンリスト外でホイールを回転するとポップアップが閉じてしまう。Toolkit.getDefaultToolkit().addAWTEventListener(...)でもうまくいかない。 -- aterai
  • #JDK-8033069 mouse wheel scroll closes combobox popup - Java Bug Systemで、ホイール回転の方は修正されるかも。 -- aterai