Swing API уже предоставляет, если довольно многословно, API для выполнения этой операции - SwingUtilities#layoutCompoundLabel
protected TextBounds calculateTextBounds(JLabel label) {
Rectangle paintIconR = new Rectangle();
Rectangle paintTextR = new Rectangle();
Dimension size = label.getSize();
Insets insets = label.getInsets(null);
String text = label.getText();
Icon icon = (label.isEnabled()) ? label.getIcon()
: label.getDisabledIcon();
Rectangle paintViewR = new Rectangle();
paintViewR.x = insets.left;
paintViewR.y = insets.top;
paintViewR.width = size.width - (insets.left + insets.right);
paintViewR.height = size.height - (insets.top + insets.bottom);
paintIconR.x = paintIconR.y = paintIconR.width = paintIconR.height = 0;
paintTextR.x = paintTextR.y = paintTextR.width = paintTextR.height = 0;
String result = SwingUtilities.layoutCompoundLabel(
(JComponent) label,
label.getFontMetrics(label.getFont()),
text,
icon,
label.getVerticalAlignment(),
label.getHorizontalAlignment(),
label.getVerticalTextPosition(),
label.getHorizontalTextPosition(),
paintViewR,
paintIconR,
paintTextR,
label.getIconTextGap());
TextBounds bounds = new TextBounds(paintViewR, paintTextR, paintIconR);
return bounds;
}
По сути, это обеспечит границы «вида» (в основном это область, в которой может быть нарисована метка), границы «текста» и границы «значка», я просто обернул их в простой в использовании класс
Так почему бы вам использовать этот подход?
- Это фактически тот же рабочий процесс, который используется для разметки метки делегатами внешнего вида
- Не требуется создавать собственный класс, что означает, что его можно применять к любому экземпляру
JLabel
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Insets;
import java.awt.Rectangle;
import javax.swing.Icon;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
public class Test {
public static void main(String[] args) {
new Test();
}
public Test() {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
ex.printStackTrace();
}
JFrame frame = new JFrame("Testing");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(new TestPane());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
public class TextBounds {
private Rectangle paintBounds;
private Rectangle textBounds;
private Rectangle iconBounds;
public TextBounds(Rectangle paintBounds, Rectangle textBounds, Rectangle iconBounds) {
this.paintBounds = paintBounds;
this.textBounds = textBounds;
this.iconBounds = iconBounds;
}
public Rectangle getPaintBounds() {
return paintBounds;
}
public Rectangle getTextBounds() {
return textBounds;
}
public Rectangle getIconBounds() {
return iconBounds;
}
}
public class TestPane extends JPanel {
private JLabel label = new JLabel("Hello");
public TestPane() {
setLayout(new BorderLayout());
label.setHorizontalAlignment(JLabel.CENTER);
label.setVerticalAlignment(JLabel.CENTER);
add(label);
}
protected TextBounds calculateTextBounds(JLabel label) {
Rectangle paintIconR = new Rectangle();
Rectangle paintTextR = new Rectangle();
Dimension size = label.getSize();
Insets insets = label.getInsets(null);
String text = label.getText();
Icon icon = (label.isEnabled()) ? label.getIcon()
: label.getDisabledIcon();
Rectangle paintViewR = new Rectangle();
paintViewR.x = insets.left;
paintViewR.y = insets.top;
paintViewR.width = size.width - (insets.left + insets.right);
paintViewR.height = size.height - (insets.top + insets.bottom);
paintIconR.x = paintIconR.y = paintIconR.width = paintIconR.height = 0;
paintTextR.x = paintTextR.y = paintTextR.width = paintTextR.height = 0;
String result = SwingUtilities.layoutCompoundLabel(
(JComponent) label,
label.getFontMetrics(label.getFont()),
text,
icon,
label.getVerticalAlignment(),
label.getHorizontalAlignment(),
label.getVerticalTextPosition(),
label.getHorizontalTextPosition(),
paintViewR,
paintIconR,
paintTextR,
label.getIconTextGap());
TextBounds bounds = new TextBounds(paintViewR, paintTextR, paintIconR);
System.out.println(result);
System.out.println("view " + paintViewR);
System.out.println("paint " + paintIconR);
System.out.println("text " + paintTextR);
return bounds;
}
@Override
public Dimension getPreferredSize() {
return new Dimension(200, 200);
}
protected void paintComponent(Graphics g) {
super.paintComponent(g);
Graphics2D g2d = (Graphics2D) g.create();
TextBounds bounds = calculateTextBounds(label);
g2d.setColor(Color.RED);
g2d.draw(bounds.getPaintBounds());
g2d.setColor(Color.GREEN);
g2d.draw(bounds.getTextBounds());
g2d.setColor(Color.BLUE);
g2d.draw(bounds.getIconBounds());
g2d.dispose();
}
}
}
У других решений есть потенциальное преимущество: не создавать по три новых объекта Rectangle за вызов. Это может быть проблемой, если getTextBounds вызывается в слушателе событий MouseMoved. При некоторой сложности, конечный прямоугольник может быть кэширован вместе с шириной и высотой JLabel.
Rectangle
нужно создавать только один раз, они передаются в API и их значения применяются напрямую, поэтому вы не создаете новые объекты каждый раз.