概要

複数のネストしたComponentを配置したJPanelのドラッグイベントを、JScrollPaneをラップしたJLayerで受け取ってスクロール可能にします。

サンプルコード

class DragScrollLayerUI extends LayerUI<JScrollPane> {
  private final Point pp = new Point();

  @Override public void installUI(JComponent c) {
    super.installUI(c);
    if (c instanceof JLayer) {
      ((JLayer) c).setLayerEventMask(
          AWTEvent.MOUSE_EVENT_MASK | AWTEvent.MOUSE_MOTION_EVENT_MASK);
    }
  }

  @Override public void uninstallUI(JComponent c) {
    if (c instanceof JLayer) {
      ((JLayer) c).setLayerEventMask(0);
    }
    super.uninstallUI(c);
  }

  @Override protected void processMouseEvent(
      MouseEvent e, JLayer<? extends JScrollPane> l) {
    Component c = e.getComponent();
    if (c instanceof JScrollBar || c instanceof JSlider) {
      return;
    }
    if (e.getID() == MouseEvent.MOUSE_PRESSED) {
      JViewport vport = l.getView().getViewport();
      Point cp = SwingUtilities.convertPoint(c, e.getPoint(), vport);
      pp.setLocation(cp);
    }
  }

  @Override protected void processMouseMotionEvent(
      MouseEvent e, JLayer<? extends JScrollPane> l) {
    Component c = e.getComponent();
    if (c instanceof JScrollBar || c instanceof JSlider) {
      return;
    }
    if (e.getID() == MouseEvent.MOUSE_DRAGGED) {
      JViewport vport = l.getView().getViewport();
      JComponent cmp = (JComponent) vport.getView();
      Point cp = SwingUtilities.convertPoint(c, e.getPoint(), vport);
      Point vp = vport.getViewPosition();
      vp.translate(pp.x - cp.x, pp.y - cp.y);
      cmp.scrollRectToVisible(new Rectangle(vp, vport.getSize()));
      pp.setLocation(cp);
    }
  }
}
View in GitHub: Java, Kotlin

解説

  • DragScrollListener
    • JTreeの余白をドラッグしてスクロール
    • このリスナーを追加した一つのComponentをドラッグしてスクロールが可能
    • JPanelを掴んでのスクロールが実行可能だが、その子コンポーネントのJTabbedPaneなどをドラッグしてもスクロールは不可
  • DragScrollLayer
    • すべての下位コンポーネントのMouseEventをキャッチするようJLayer#setLayerEventMask(...)と設定したJLayerを作成
    • このJLayerJScrollPaneをラップし、子にマウスイベントを消費するコンポーネントが存在する場合でも、ビューのJPanelをドラッグに応じてスクロール
    • JScrollBarJSliderのようにノブのドラッグを使用するコンポーネントは二重にスクロールしてしまうので除外

参考リンク

コメント