• category: swing folder: TextOverflowFadeLabel title: JLabelの文字列があふれる場合、フェードアウト効果を適用して端付近の文字を透明化 tags: [JLabel, HTML, JTextField, BufferedImage, AlphaComposite] author: aterai pubdate: 2018-12-03T02:58:34+09:00 description: JLabelなどで文字列があふれる場合、デフォルトの省略記号…ではなく、フェードアウト効果を適用して端付近の文字を透明表示します。 image: https://drive.google.com/uc?export=view&id=16SQWlFfTk5X8LOxTKkjobFZblb-hgj5uGA

概要

JLabelなどで文字列があふれる場合、デフォルトの省略記号ではなく、フェードアウト効果を適用して端付近の文字を透明表示します。

サンプルコード

@Override public void paintComponent(Graphics g) {
  Insets i = getInsets();
  int w = getWidth() - i.left - i.right;
  int h = getHeight() - i.top - i.bottom;

  Rectangle rect = new Rectangle(i.left, i.top, w - LENGTH, h);

  Graphics2D g2 = (Graphics2D) g.create();
  g2.setFont(g.getFont());
  g2.setClip(rect);
  g2.setComposite(AlphaComposite.SrcOver.derive(.99999f));
  super.paintComponent(g2);

  rect.width = 1;
  float alpha = 1f;
  for (int x = w - LENGTH; x < w; x++) {
    rect.x = x;
    alpha = Math.max(0f, alpha - DIFF);
    g2.setComposite(AlphaComposite.SrcOver.derive(alpha));
    g2.setClip(rect);
    super.paintComponent(g2);
  }
  g2.dispose();
}
View in GitHub: Java, Kotlin

解説

  • defalut JLabel ellipsis
    • JLabelのデフォルトは、省略記号であふれる文字列を省略
  • html JLabel fade out
    • JLabelの文字列先頭に<html>タグを付加してデフォルト省略記号によるあふれ省略を無効化
    • 代わりにJLabel#paintComponent(...)をオーバーライドして、右端付近の文字列をフェードアウト効果で透明化
    • Graphics2D#setComposite(...)を使用すると文字列にアンチエイリアスが掛かってしまう?ため、透明化しない文字列にもほぼ1fのアルファ値`を設定して描画
  • JTextField fade out
    • JLabelの代わりに編集不可にしたJTextFieldを使用して省略記号によるあふれ省略を無効化
    • 文字列をBufferedImageに描画し、その右端付近のピクセル値をBufferedImage#getRGB(...)で取得後、アルファ成分を変更してBufferedImage#setRGB(...)で戻す
class FadingOutLabel extends JTextField {
  private static final int LENGTH = 20;
  private final Dimension dim = new Dimension();
  private transient BufferedImage img;

  protected FadingOutLabel(String text) {
    super(text);
  }

  @Override public void updateUI() {
    super.updateUI();
    setOpaque(false);
    setEditable(false);
    setFocusable(false);
    setEnabled(false);
    setBorder(BorderFactory.createEmptyBorder());
  }

  @Override public void paintComponent(Graphics g) {
    // super.paintComponent(g);
    int w = getWidth();
    int h = getHeight();
    if (img == null || dim.width != w || dim.height != h) {
      dim.setSize(w, h);
      img = updateImage(dim);
    }
    g.drawImage(img, 0, 0, this);
  }

  private BufferedImage updateImage(Dimension d) {
    img = new BufferedImage(d.width, d.height, BufferedImage.TYPE_INT_ARGB);
    Graphics2D g2 = img.createGraphics();
    g2.setFont(getFont());
    g2.setPaint(getForeground());
    FontRenderContext frc = g2.getFontRenderContext();
    TextLayout tl = new TextLayout(getText(), getFont(), frc);
    int baseline = getBaseline(d.width, d.height);
    tl.draw(g2, getInsets().left, baseline);
    g2.dispose();

    int spx = Math.max(0, d.width - LENGTH);
    for (int x = 0; x < LENGTH; x++) {
      double factor = 1d - x / (double) LENGTH;
      for (int y = 0; y < d.height; y++) {
        int argb = img.getRGB(spx + x, y);
        int rgb = argb & 0x00FFFFFF;
        int a = (argb >> 24) & 0xFF;
        img.setRGB(spx + x, y, ((int) (a * factor) << 24) | rgb);
      }
    }
    return img;
  }
}

参考リンク

コメント