Summary

JToolTipの形状を吹き出し風に変更し、JListのセル上にこれを表示します。

Source Code Examples

class BalloonToolTip extends JToolTip {
  private HierarchyListener listener;
  @Override public void updateUI() {
    removeHierarchyListener(listener);
    super.updateUI();
    listener = e -> {
      Component c = e.getComponent();
      if ((e.getChangeFlags() & HierarchyEvent.SHOWING_CHANGED) != 0
          && c.isShowing()) {
        Window w = SwingUtilities.getWindowAncestor(c);
        if (w != null && w.getType() == Window.Type.POPUP) {
          // Popup$HeavyWeightWindow
          w.setBackground(new Color(0x0, true));
        }
      }
    };
    addHierarchyListener(listener);
    setOpaque(false);
    setBorder(BorderFactory.createEmptyBorder(8, 5, 0, 5));
  }

  @Override public Dimension getPreferredSize() {
    Dimension d = super.getPreferredSize();
    d.height = 28;
    return d;
  }

  @Override public void paintComponent(Graphics g) {
    Shape s = makeBalloonShape();
    Graphics2D g2 = (Graphics2D) g.create();
    g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING,
                        RenderingHints.VALUE_ANTIALIAS_ON);
    g2.setColor(getBackground());
    g2.fill(s);
    g2.setColor(getForeground());
    g2.draw(s);
    g2.dispose();
    super.paintComponent(g);
  }

  private Shape makeBalloonShape() {
    Insets i = getInsets();
    int w = getWidth() - 1;
    int h = getHeight() - 1;
    int v = i.top / 2;
    Polygon triangle = new Polygon();
    triangle.addPoint(i.left + v + v, 0);
    triangle.addPoint(i.left + v, v);
    triangle.addPoint(i.left + v + v + v, v);
    Area area = new Area(new RoundRectangle2D.Float(
        0, v, w, h - i.bottom - v, i.top, i.top));
    area.add(new Area(triangle));
    return area;
  }
}
View in GitHub: Java, Kotlin

Explanation

  • setOpaque(false)で背景色を描画しないJToolTopを作成しJToolTop#paintComponent(...)をオーバーライドして吹き出し風の背景を描画
  • JToolTopHierarchyListenerを追加しそのJToolTopが表示状態になった時親がJWindowWindow#getType() == Window.Type.POPUPかどうかを調べてこれを透明に設定
    • JToolTopが元JFrame外に表示される場合はHeavyWeightWindowに配置して表示されるためこのJWindowを透明にする必要がある
    • JToolTopが元JFrame内に表示される場合はそのJFrameJLayeredPanePOPUP_LAYERJToolTopが描画されるのでsetOpaque(false)JToolTopの背景は描画されない
    • JWindow#setShape(...)メソッドでも形状の変更が可能だが、この場合フチを滑らかにすることが難しい
  • JList#createToolTip()をオーバーライドし通常のJToolTipの代わりにBalloonToolTip(上記のJToolTip)を作成して返すように設定
JList<String> list = new JList<String>(model) {
  @Override public JToolTip createToolTip() {
    JToolTip tip = new BalloonToolTip();
    tip.setComponent(this);
    return tip;
  }
};

Reference

Comment