Swing/LeftClippedComboBox のバックアップの現在との差分(No.13)
-
category: swing
folder: LeftClippedComboBox
title: JComboBoxのアイテム文字列を左側からクリップ
tags: [JComboBox, ListCellRenderer, ArrowButton]
author: aterai
pubdate: 2007-06-18T19:09:42+09:00
description: JComboBoxのアイテム文字列がコンポーネントより長い場合、これを左側からクリップします。
image:
hreflang:
href: http://java-swing-tips.blogspot.com/2009/05/left-clipped-jcombobox.html href: https://java-swing-tips.blogspot.com/2009/05/left-clipped-jcombobox.html lang: en
概要
概要
JComboBox
のアイテム文字列がコンポーネントより長い場合、これを左側からクリップします。
Screenshot
Advertisement
サンプルコード
サンプルコード
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
// @see BasicComboBoxUI#rectangleForCurrentValue
int buttonSize = combo02.getHeight() - itb;
if (arrowButton != null) {
buttonSize = arrowButton.getWidth();
}
availableWidth -= buttonSize;
JTextField tf = (JTextField)combo02.getEditor().getEditorComponent();
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
// <blockquote cite="https://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>
// </blockquote>
return this;
}
});
View in GitHub: Java, Kotlin解説
標準のJComboBox
では、長い文字列は右側をクリップするので、上記のサンプルでは左側を切り取り、...
で置き換えるようにセルレンダラーを変更しています。
解説
- 標準の
JComboBox
が使用するDefaultListCellRenderer
はJLabel
を継承しているので、長い文字列は右側から省略される - 上記のサンプルでは左側から省略し
...
で置き換えるようにセルレンダラーを変更- 長いファイル名でも拡張子は省略されない
- エディタ部分(
index < 0
の場合)を描画するときは矢印ボタンの幅を考慮する必要がある -
LookAndFeel
によって余白などのサイズが微妙に異なる場合がある? - 補助文字(サロゲートペアなど)を含む文字列を扱う場合は、
String#charAt(int)
ではなくString#codePointAt(int)
やCharacter.charCount(codePoint)
などを使用する必要がある -
#spanend #spanadd FontMetrics fm = getFontMetrics(getFont()); #spanend #spanadd if (fm.stringWidth(cellText) > availableWidth) { #spanend 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)); #spanadd } #spanend #spanadd
index<0
の場合)を描画するときは、矢印ボタンの幅を考慮する必要があります。
LookAndFeel
によって余白などのサイズが微妙に異なる場合がある?ため、うまく表示されないことがあります。
参考リンク
参考リンク
- Swing - JTable - right align in cell even if the text is wider than the cell
- camickr さんの投稿(2005/06/10 5:52)した
JTable
でのサンプルを参考にしています。
- camickr さんの投稿(2005/06/10 5:52)した
- Left Dot Renderer « Java Tips Weblog
- camickr さんが投稿(2005/06/10 5:52)した
JTable
でのサンプルがある
- camickr さんが投稿(2005/06/10 5:52)した
- Left Dot Renderer « Java Tips Weblog
- Java による Unicode サロゲートプログラミング