你永远不应该给自己打电话paintComponent
(或paint
)。这是由RepaintManager
.
您实际上遇到的问题是事实speedx
and speedy
or 0
,这意味着您的球永远不会移动...
另一个问题是ballAdder
该类正在使用 aFlowLayout
而您的Ball
类没有提供有关其首选大小的任何详细信息,这意味着Ball
面板的首选大小为0x0
.
审查
您的设计存在可扩展性的重大问题。除了您会发现由于布局问题而难以在 UI 中添加一个以上的球之外......
每个Ball
都有它自己的线程。这意味着,添加的球越多,运行的线程就越多。这将持续消耗资源并影响应用程序的性能。
最好提供一个Drawable
对象的概念,它知道它应该在它的容器的概念中显示在哪里,并且可以painted
来自paintComponent
. 通过使用单个javax.swing.Timer
它应该更有能力支持越来越多的随机球。
第一次修复
要解决你的第一个问题,你可以做这样的事情......
public class ballAdder extends JPanel {
public ballAdder() {
setLayout(new BorderLayout());
add(new Ball(5, 5));
}
}
此修复程序的问题在于,您只能Ball
在容器上放置一个,因为它需要占用最大的可用空间。
您可能需要通读Using Layout Managers以了解更多详细信息
一个(可能的)更好的解决方案
一个(可能的)更好的解决方案是使用单个JPanel
作为“球坑”,它保持对球列表的引用。
然后,您将使用BallPitPane
'paintComponent
方法绘制所有球(在球列表中)。
通过使用单个javax.swing.Timer
,您可以遍历球列表并更新那里的位置(在BallPitPane
恕我直言,这比尝试与布局管理器或编写自己的布局管理器更容易......
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.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.Timer;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
public class Bounce {
public static void main(String[] args) {
new Bounce();
}
public Bounce() {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
}
JFrame frame = new JFrame("Testing");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new BorderLayout());
frame.add(new BallPitPane());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
public class BallPitPane extends JPanel {
private List<Ball> balls;
private Random rand;
public BallPitPane() {
rand = new Random(System.currentTimeMillis());
balls = new ArrayList<>(25);
Timer timer = new Timer(40, new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
if (balls.isEmpty()) {
balls.add(new Ball(BallPitPane.this));
}
if (rand.nextBoolean()) {
balls.add(new Ball(BallPitPane.this));
}
for (Ball ball : balls) {
ball.move();
}
repaint();
}
});
timer.start();
}
@Override
public Dimension getPreferredSize() {
return new Dimension(200, 200);
}
@Override
protected void paintComponent(Graphics g) {
super.paintComponent(g);
Graphics2D g2d = (Graphics2D) g.create();
for (Ball ball : balls) {
ball.paint(g2d);
}
g2d.dispose();
}
}
protected static int random(int min, int max) {
return (int)Math.round(Math.random() * (max - min)) + min;
}
public static class Ball {
public static final int WIDTH = 10;
public static final int HEIGHT = 10;
private int x;
private int y;
private int deltaX;
private int deltaY;
private Color color;
private BallPitPane parent;
public Ball(BallPitPane parent) {
this.parent = parent;
x = parent.getWidth() / 2;
y = parent.getHeight() / 2;
deltaX = random(-4, 4);
deltaY = random(-4, 4);
color = new Color(random(0, 255), random(0, 255), random(0, 255));
}
public void move() {
x += deltaX;
y += deltaY;
if (x + WIDTH > parent.getWidth()) {
x = parent.getWidth() - WIDTH;
deltaX *= -1;
} else if (x < 0) {
x = 0;
deltaX *= -1;
}
if (y + HEIGHT > parent.getHeight()) {
y = parent.getHeight() - HEIGHT;
deltaY *= -1;
} else if (y < 0) {
y = 0;
deltaY *= -1;
}
}
public Color getColor() {
return color;
}
public void paint(Graphics2D g2d) {
g2d.setColor(getColor());
g2d.fillOval(x, y, WIDTH, HEIGHT);
g2d.setColor(Color.BLACK);
g2d.drawOval(x, y, WIDTH, HEIGHT);
}
}
}