GlyphVectorで作成した5段階評価の結果をラベルに描画する
Total: 949
, Today: 1
, Yesterday: 1
Posted by aterai at
Last-modified:
Summary
GlyphVector
を使用して5
段階評価の★のアウトラインを作成し、その内部を評価値の小数点一位まで塗りつぶします。
Screenshot
Advertisement
Source Code Examples
// JComponent label1 = new FiveStarRatingLabel("3.5");
class FiveStarRatingLabel extends JComponent {
private static final String STAR = "★★★★★";
private final int ip;
private final int fp;
protected FiveStarRatingLabel(String rating) {
super();
BigDecimal bd = new BigDecimal(rating);
ip = bd.intValue();
fp = bd.subtract(new BigDecimal(ip)).multiply(BigDecimal.TEN).intValue();
}
@Override protected void paintComponent(Graphics g) {
super.paintComponent(g);
int w = getWidth();
int h = getHeight();
g.setColor(Color.WHITE);
g.fillRect(0, 0, w, h);
Graphics2D g2 = (Graphics2D) g.create();
g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
FontRenderContext frc = g2.getFontRenderContext();
GlyphVector gv = getFont().createGlyphVector(frc, STAR);
Rectangle2D r = gv.getVisualBounds();
double cx = w / 2d - r.getCenterX();
double cy = h / 2d - r.getCenterY();
AffineTransform toCenterAtf = AffineTransform.getTranslateInstance(cx, cy);
double point = 0d;
for (int i = 0; i < gv.getNumGlyphs(); i++) {
GlyphMetrics gm = gv.getGlyphMetrics(i);
if (i <= ip - 1) {
point += gm.getAdvance();
} else if (i <= ip) {
point += gm.getBounds2D().getWidth() * fp / 10d;
}
}
g2.setPaint(Color.GREEN);
Shape s = toCenterAtf.createTransformedShape(gv.getOutline());
g2.draw(s);
Rectangle2D clip = new Rectangle2D.Double(r.getX(), r.getY(), point, r.getHeight());
g2.setClip(toCenterAtf.createTransformedShape(clip));
g2.fill(s);
g2.dispose();
}
}
View in GitHub: Java, KotlinExplanation
5
段階評価用に★を5
文字分用意Font#createGlyphVector(...)
でその文字列★★★★★からGlyphVector
を作成GlyphVector#getGlyphMetrics(index)
で一文字ごとにGlyphMetrics
を取得- 評価値の整数部(★一つ個分)は
GlyphMetrics#getAdvance()
で幅を取得 - 評価値の小数部は小数点一位を
10
倍してBigDecimal
で整数として取得し、GlyphMetrics#getBounds2D()#getWidth() * fractionalPart / 10d
で幅に変換
- 評価値の整数部(★一つ個分)は
- この合計幅の矩形で
Clip
した領域のみ文字列★のアウトラインを塗りつぶして評価値を描画