• title: JLayerを使ってJTabbedPaneのタブの挿入位置を描画する tags: [JLayer, JTabbedPane, DragAndDrop, TransferHandler, JWindow, JLabel] author: aterai pubdate: 2012-01-23T18:01:31+09:00 description: JLayerを使って、タブのドラッグ&ドロップでの移動先をJTabbedPane上に描画します。 hreflang:
       href: http://java-swing-tips.blogspot.com/2012/01/sharing-tabs-between-2-jframes.html
       lang: en

概要

JLayerを使って、タブのドラッグ&ドロップでの移動先をJTabbedPane上に描画します。

サンプルコード

class DropLocationLayerUI extends LayerUI<DnDTabbedPane> {
  private static final int LINEWIDTH = 3;
  private final Rectangle lineRect = new Rectangle();
  @Override public void paint(Graphics g, JComponent c) {
    super.paint(g, c);
    if (c instanceof JLayer) {
      JLayer layer = (JLayer) c;
      DnDTabbedPane tabbedPane = (DnDTabbedPane) layer.getView();
      DnDTabbedPane.DropLocation loc = tabbedPane.getDropLocation();
      if (loc != null && loc.isDropable() && loc.getIndex() >= 0) {
        Graphics2D g2 = (Graphics2D) g.create();
        g2.setComposite(
            AlphaComposite.getInstance(AlphaComposite.SRC_OVER, .5f));
        g2.setColor(Color.RED);
        initLineRect(tabbedPane, loc);
        g2.fill(lineRect);
        g2.dispose();
      }
    }
  }
  private void initLineRect(
      DnDTabbedPane tabbedPane, DnDTabbedPane.DropLocation loc) {
    int index = loc.getIndex();
    boolean isZero = index == 0;
    Rectangle r = tabbedPane.getBoundsAt(isZero ? 0 : index - 1);
    Rectangle rect = new Rectangle();
    int a = isZero ? 0 : 1;
    if (tabbedPane.getTabPlacement() == JTabbedPane.TOP ||
        tabbedPane.getTabPlacement() == JTabbedPane.BOTTOM) {
      rect.x = r.x - LINEWIDTH / 2 + r.width * a;
      rect.y = r.y;
      rect.width  = LINEWIDTH;
      rect.height = r.height;
    } else {
      rect.x = r.x;
      rect.y = r.y - LINEWIDTH / 2 + r.height * a;
      rect.width  = r.width;
      rect.height = LINEWIDTH;
    }
    lineRect.setRect(rect);
  }
}
View in GitHub: Java, Kotlin

解説

上記のサンプルでは、JTabbedPaneのタブをドラッグ&ドロップや、JTabbedPane間でタブのドラッグ&ドロップ移動のようにGlassPaneを使用する代わりに、JDK 1.7.0で導入されたJLayerを使用して、タブの挿入先を描画しています。JLayerを使用することで、別ウィンドウにあるJTabbedPaneへのタブ移動などの描画が簡単にできるようになっています。


メニューバーから、ドラッグ中の半透明タブイメージの描画方法を切り替えてテストすることができます。

  • Lightweight
    • JDK1.7.0で導入された、TransferHandler#setDragImage(...)メソッドを使用して描画
    • ウィンドウの外では非表示
  • Heavyweight
    • 半透明のJWindowJLabelを追加して表示
    • ウィンドウの外でも表示可能
    • 表示位置のオフセットが(0, 0)の場合、DragOverイベントが元のJFrameに伝わらない?
      • オフセットが(0, 0)でも、JLabel#contains(...)が常にfalseなら問題なし
private final JLabel label = new JLabel() {
  @Override public boolean contains(int x, int y) {
    return false;
  }
};
private final JWindow dialog = new JWindow();
public TabTransferHandler() {
  dialog.add(label);
  //dialog.setAlwaysOnTop(true); // Web Start
  dialog.setOpacity(0.5f);
  //com.sun.awt.AWTUtilities.setWindowOpacity(dialog, .5f); // JDK 1.6.0
  DragSource.getDefaultDragSource().addDragSourceMotionListener(
      new DragSourceMotionListener() {
    @Override public void dragMouseMoved(DragSourceDragEvent dsde) {
      Point pt = dsde.getLocation();
      pt.translate(5, 5); // offset
      dialog.setLocation(pt);
    }
  });
//...

参考リンク

コメント