Метод Java Paint не рисует?
Вопрос
Я работаю над простым маленьким качающимся компонентом и рву на себе волосы, пытаясь понять, почему мой метод рисования не работает.
Идея этого компонента заключается в том, что это небольшая JPanel с меткой.Фон (за этикеткой) должен быть белым, с цветным прямоугольником слева, обозначающим соотношение двух измерений:«фактическое» и «ожидаемое».
Если бы у вас была группа этих компонентов, выровненных по вертикали, они образовали бы гистограмму, состоящую из горизонтальных полос.
Подобные вещи должны быть очень простыми.
Во всяком случае, вот код:
package com.mycompany.view;
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Rectangle;
import javax.swing.JLabel;
import javax.swing.JPanel;
public class BarGraphPanel extends JPanel {
private static final Color BACKGROUND = Color.WHITE;
private static final Color FOREGROUND = Color.BLACK;
private static final Color BORDER_COLOR = new Color(229, 172, 0);
private static final Color BAR_GRAPH_COLOR = new Color(255, 255, 165);
private int actual = 0;
private int expected = 1;
private JLabel label;
public BarGraphPanel() {
super();
label = new JLabel();
label.setOpaque(false);
label.setForeground(FOREGROUND);
super.add(label);
super.setOpaque(true);
}
public void setActualAndExpected(int actual, int expected) {
this.actual = actual;
this.expected = expected;
}
@Override
public void paint(Graphics g) {
double proportion = (expected == 0) ? 0 : ((double) actual) / expected;
Rectangle bounds = super.getBounds();
g.setColor(BACKGROUND);
g.fillRect(bounds.x, bounds.y, bounds.width, bounds.height);
g.setColor(BAR_GRAPH_COLOR);
g.fillRect(bounds.x, bounds.y, (int) (bounds.width * proportion), bounds.height);
g.setColor(BORDER_COLOR);
g.drawRect(bounds.x, bounds.y, bounds.width, bounds.height);
label.setText(String.format("%s of %s (%.1f%%)", actual, expected, proportion * 100));
super.paint(g);
g.dispose();
}
}
А вот простой тестовый жгут:
package com.mycompany.view;
import java.awt.Dimension;
import java.awt.GridLayout;
import javax.swing.JFrame;
import javax.swing.UIManager;
public class MyFrame extends JFrame {
public MyFrame() {
super();
super.setLayout(new GridLayout(3, 1));
super.setPreferredSize(new Dimension(300, 200));
BarGraphPanel a = new BarGraphPanel();
BarGraphPanel b = new BarGraphPanel();
BarGraphPanel c = new BarGraphPanel();
a.setActualAndExpected(75, 100);
b.setActualAndExpected(85, 200);
c.setActualAndExpected(20, 300);
super.add(a);
super.add(b);
super.add(c);
}
public static void main(String[] args) {
javax.swing.SwingUtilities.invokeLater(new Runnable() {
public void run() {
createAndShowGUI();
}
});
}
public static void createAndShowGUI() {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (Throwable t) { }
MyFrame frame = new MyFrame();
frame.pack();
frame.setVisible(true);
}
}
Тестовая программа создает простой кадр, а затем добавляет три таких элемента управления.
Все метки отображаются правильно, что указывает на то, что метод Paint() действительно вызывается, но прямоугольники не рисуются в объекте Graphics.
Что я делаю не так?
И почему программирование на Swing так отстойно?
Вот мой окончательный код.Спасибо всем за вашу помощь!
public void paintComponent(Graphics g) {
double proportion = (expected == 0) ? 0 : ((double) actual) / expected;
Rectangle bounds = super.getBounds();
g.setColor(BACKGROUND);
g.fillRect(0, 0, bounds.width, bounds.height);
g.setColor(BAR_GRAPH_COLOR);
g.fillRect(0, 0, (int) (bounds.width * proportion), bounds.height);
g.setColor(BORDER_COLOR);
g.drawRect(0, 0, bounds.width - 1, bounds.height - 1);
FontMetrics metrics = g.getFontMetrics();
String label = String.format("%s of %s (%.1f%%)", actual, expected, proportion * 100);
Rectangle2D textBounds = metrics.getStringBounds(label, g);
g.setColor(FOREGROUND);
g.drawString(label, 5, (int) ((bounds.height + textBounds.getHeight()) / 2));
}
Решение
Я думаю, вы почти ответили на свой вопрос в комментариях вместе с ответом Дэвида.Изменять paint(Graphics g)
к paintComponent(Graphics g)
и удалите последние две строки метода, и все будет в порядке.
РЕДАКТИРОВАТЬ: Как ни странно, это работает только для первого такта из трех.Проводятся дополнительные испытания...
Кстати, у вас ошибка на единицу в коде рисования границ.Должен быть:
g.setColor(BORDER_COLOR);
g.drawRect(bounds.x, bounds.y, bounds.width - 1, bounds.height - 1);
РЕДАКТИРОВАТЬ2: Хорошо понял.Ваш полный paintComponent
метод должен быть следующим:
@Override
public void paintComponent(Graphics g) {
double proportion = (expected == 0) ? 0 : ((double) actual) / expected;
Rectangle bounds = super.getBounds();
g.setColor(BACKGROUND);
g.fillRect(0, 0, bounds.width, bounds.height);
g.setColor(BAR_GRAPH_COLOR);
g.fillRect(0, 0, (int) (bounds.width * proportion), bounds.height);
g.setColor(BORDER_COLOR);
g.drawRect(0, 0, bounds.width-1, bounds.height-1);
label.setText(String.format("%s of %s (%.1f%%)", actual, expected, proportion * 100));
}
Обратите внимание, что координаты, данные g.fillRect()
и g.drawRect()
относятся к компоненту, поэтому они должны начинаться с (0,0).
И нет, я не могу помочь тебе с твоим последним вопросом, хотя и чувствую твою боль.:)
Другие советы
В JPanel вы вызвали super.setOpaque(true).JPanel полностью заполнит фон, когда вы вызовете super.paint() и перезапишете свои переплетения.
Не уверен, что это источник вашей проблемы, но в Swing вы должны переопределить paintComponent(Graphics2D)
вместо...
Во всяком случае, я думаю, вам следует вызвать super.paint(g);наверху вашего метода, а не в самом низу.Возможно, суперкласс опирается на ваши данные.