• category: swing folder: RollOverTree title: JTreeのノードをハイライト tags: [JTree, TreeCellRenderer, MouseMotionListener] author: aterai pubdate: 2007-05-21T05:44:50+09:00 description: JTreeのノード上にマウスカーソルがきたら、ハイライト表示します。 image: https://lh3.googleusercontent.com/_9Z4BYR88imo/TQTSHWXxwwI/AAAAAAAAAiQ/MP7tLgM--lo/s800/RollOverTree.png

概要

概要

JTreeのノード上にマウスカーソルがきたら、ハイライト表示します。

サンプルコード

サンプルコード

#spandel
class MyTreeCellRenderer extends DefaultTreeCellRenderer implements MouseMotionListener {
#spanend
  private static final Color rollOverRowColor = new Color(220,240,255);
  private final JTree tree;
  private final TreeCellRenderer renderer;
  public MyTreeCellRenderer(JTree tree, TreeCellRenderer renderer) {
    this.tree = tree;
    this.renderer = renderer;
    tree.addMouseMotionListener(this);
  }
  @Override public Component getTreeCellRendererComponent(
          JTree tree, Object value, boolean isSelected,
          boolean expanded, boolean leaf, int row, boolean hasFocus) {
    JComponent c = (JComponent)renderer.getTreeCellRendererComponent(
        tree, value, isSelected, expanded, leaf, row, hasFocus);
    if(row==rollOverRowIndex) {
      c.setOpaque(true);
      c.setBackground(rollOverRowColor);
      if(isSelected) c.setForeground(getTextNonSelectionColor());
    }else{
      c.setOpaque(false);
    }
    return c;
  }
#spanadd
private final JTree tree = new JTree(makeModel()) {
#spanend
  private final Color rolloverRowColor = new Color(220, 240, 255);
  private int rollOverRowIndex = -1;
  @Override public void mouseMoved(MouseEvent e) {
    int row = tree.getRowForLocation(e.getX(), e.getY());
    if(row!=rollOverRowIndex) {
      //System.out.println(row);
      rollOverRowIndex = row;
      tree.repaint();
    }
  private transient MouseMotionListener listener;
  @Override public void updateUI() {
    removeMouseMotionListener(listener);
    super.updateUI();
    setCellRenderer(new DefaultTreeCellRenderer() {
      @Override public Component getTreeCellRendererComponent(
          JTree tree, Object value, boolean selected, boolean expanded,
          boolean leaf, int row, boolean hasFocus) {
        JComponent c = (JComponent) super.getTreeCellRendererComponent(
            tree, value, selected, expanded, leaf, row, hasFocus);
        if (row == rollOverRowIndex) {
          c.setOpaque(true);
          c.setBackground(rolloverRowColor);
          if (selected) {
            c.setForeground(getTextNonSelectionColor());
          }
        } else {
          c.setOpaque(false);
        }
        return c;
      }
    });
    listener = new MouseAdapter() {
      @Override public void mouseMoved(MouseEvent e) {
        int row = getRowForLocation(e.getX(), e.getY());
        if (row != rollOverRowIndex) {
          rollOverRowIndex = row;
          repaint();
        }
      }
    };
    addMouseMotionListener(listener);
  }
  @Override public void mouseDragged(MouseEvent e) {}
#spandel
}
#spanend
#spanadd
};
#spanend
View in GitHub: Java, Kotlin

解説

JTreeからデフォルトのセルレンダラーを取得し、これを委譲してセルレンダラーを作成しています。

解説

  • JTreeMouseMotionListenerを設定して現在マウスカーソルが存在する行を記録
  • DefaultTreeCellRenderer#getTreeCellRendererComponent(...)メソッドをオーバーライドし、カーソル行の場合はsetOpaque(true)でノードの不透明設定とsetForeground(Color)で背景色を変更
  • DefaultTreeCellRendererDefaultTableCellRendererとは異なりDefaultTreeCellRenderer#getTextNonSelectionColor()などのメソッドが使用可能
    • DefaultTableCellRendererには選択時の文字色や背景色を取得するメソッドが存在しないのでJTable#getSelectionBackground()などを使用する必要がある
  • 継承ではなく、委譲を使うのは、ノードをハイライトしない場合は、Windowsなどでのノードアイコンは選択されない、文字列だけ選択されるという描画をそのまま利用するため
  • 継承もしているが、これは、JTableと違いDefaultTreeCellRendererからノードを選択した時の色などを取得するようになっているためで、あまり意味はない
このレンダラーはMouseMotionListenerを実装し、getTreeCellRendererComponentメソッドの実装で、ハイライト表示したいノードの場合は、 委譲したレンダラーから得たコンポーネントをsetOpaque(true)setForeground(Color)などで修飾して返すようになっています。

参考リンク

参考リンク

コメント

コメント