• category: swing folder: LeftClippedComboBox title: JComboBoxのアイテム文字列を左側からクリップ tags: [JComboBox, ListCellRenderer, ArrowButton] author: aterai pubdate: 2007-06-18T19:09:42+09:00 description: JComboBoxのアイテム文字列がコンポーネントより長い場合、これを左側からクリップします。 image: https://lh3.googleusercontent.com/_9Z4BYR88imo/TQTPEaiR2iI/AAAAAAAAAdY/E5fxUtKW0sM/s800/LeftClippedComboBox.png hreflang:
       href: http://java-swing-tips.blogspot.com/2009/05/left-clipped-jcombobox.html
       lang: en

概要

JComboBoxのアイテム文字列がコンポーネントより長い場合、これを左側からクリップします。

サンプルコード

final JButton arrowButton = getArrowButton(combo02);
combo02.setRenderer(new DefaultListCellRenderer() {
  @Override public Component getListCellRendererComponent(
    JList list, Object value, int index,
    boolean isSelected, boolean cellHasFocus) {
    super.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus);
    int itb = 0, ilr = 0;
    Insets insets = getInsets();
    itb += insets.top + insets.bottom;
    ilr += insets.left + insets.right;
    insets = combo02.getInsets();
    itb += insets.top + insets.bottom;
    ilr += insets.left + insets.right;
    int availableWidth = combo02.getWidth() - ilr;
    if (index < 0) {
      //@see BasicComboBoxUI#rectangleForCurrentValue
      int buttonSize = combo02.getHeight() - itb;
      if (arrowButton != null) {
        buttonSize = arrowButton.getWidth();
      }
      availableWidth -= buttonSize;
      JTextField tf = (JTextField) combo02.getEditor().getEditorComponent();
      insets = tf.getMargin();
      availableWidth -= (insets.left + insets.right);
    }
    String cellText = (value != null) ? value.toString() : "";
    //<blockquote cite="http://tips4java.wordpress.com/2008/11/12/left-dot-renderer/">
    //@title Left Dot Renderer
    //@auther Rob Camick
    FontMetrics fm = getFontMetrics(getFont());
    if (fm.stringWidth(cellText) > availableWidth) {
      String dots = "...";
      int textWidth = fm.stringWidth(dots);
      int nChars = cellText.length() - 1;
      while (nChars > 0) {
        textWidth += fm.charWidth(cellText.charAt(nChars));
        if (textWidth > availableWidth) break;
        nChars--;
      }
      setText(dots + cellText.substring(nChars + 1));
    }
    //</blockquote>
    return this;
  }
});
View in GitHub: Java, Kotlin

解説

  • 標準のJComboBoxが使用するDefaultListCellRendererJLabelを継承しているので、長い文字列は右側から省略される
  • 上記のサンプルでは左側から省略し、...で置き換えるようにセルレンダラーを変更
    • 例えば、コンボボックスのセルよりファイル名が長くても、拡張子が表示できるようにしたいといった場合に使用可能
  • 注:
    • エディタ部分(index < 0の場合)を描画するときは、矢印ボタンの幅を考慮する必要がある
    • LookAndFeelによって余白などのサイズが微妙に異なる場合があるため、うまく表示されない場合がある?
    • 補助文字(サロゲートペアなど)を含む文字列を扱う場合は、String#charAt(int)ではなく、String#codePointAt(int)Character.charCount(codePoint)などを使用する必要がある
      FontMetrics fm = getFontMetrics(getFont());
      if (fm.stringWidth(cellText) > availableWidth) {
        String dots = "...";
        int textWidth = fm.stringWidth(dots);
        int len = cellText.length();
        int[] acp = new int[cellText.codePointCount(0, len)];
        int j = acp.length;
        for (int i = len; i > 0; i = cellText.offsetByCodePoints(i, -1)) {
          int cp = cellText.codePointBefore(i);
          textWidth += fm.charWidth(cp);
          if (textWidth > availableWidth) {
            break;
          }
          acp[--j] = cp;
        }
        setText(dots + new String(acp, j, acp.length - j));
      }
      

参考リンク

コメント