我正在研究一个简单的小摆动组件,我正在撕掉我的头发试图弄清楚为什么我的油漆方法不起作用。
这个组件背后的想法是它是一个带标签的小型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));
}
答案 0 :(得分:4)
我认为你在评论中几乎已经回答了你自己的问题,以及大卫的回答。将paint(Graphics g)
更改为paintComponent(Graphics g)
并删除方法的最后两行,您应该没问题。
编辑:奇怪的是,这只适用于三者中的第一个栏。正在进行更多测试......
顺便说一下,边框绘制代码中有一个一个一个错误。它应该是:
g.setColor(BORDER_COLOR);
g.drawRect(bounds.x, bounds.y, bounds.width - 1, bounds.height - 1);
EDIT2:好的,明白了。您的完整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)开始。
不,我无法帮助你解决你的最后一个问题,尽管我感到痛苦。 :)
答案 1 :(得分:1)
在你的JPanel中,你调用了super.setOpaque(true)。当您调用super.paint()并覆盖您的重新绑定时,JPanel将完全填充后台。
答案 2 :(得分:0)
不确定这是否是您问题的根源,但在Swing中您应该覆盖paintComponent(Graphics2D)
而不是......
答案 3 :(得分:0)
如果有的话,我认为你应该打电话给super.paint(g);在你的方法的顶部,而不是在最底层。超类可以在你的东西之上绘制。