如何在 Java 中分层绘制组件?



我在编写应用程序的图形部分时遇到了很大的问题,我需要将组件一个堆叠在一起:

  1. 首先我有JFrame(固定大小)

  2. 在其中,我有两个JPanel组件。我希望它们有彩色背景。 这是容易的部分。

  3. 在其中一个JPanel组件上,我想绘制固定的轮廓 - 矩形、车道等。在这里我有问题,我有两个类:一个扩展JPanel并且是这部分的背景,第二个扩展JComponent并表示我绘制的元素(有几个元素)。我不知道如何在JPanel中绘制元素 - 我尝试了几种方法,但没有任何结果。对我来说重要的是,JComponents应该只与这个JPanel绘制和连接,而不是整个框架。

  4. 最重要的是,我想要移动的形状。当我只有框架和矩形时,这很容易,因为我只更改位置并调用repaint()方法,但是如何做到这一点以使移动形状连接到并位于JPanel内部并将以前的图层留在它们的位置?

在我的尝试中,我创建了几个带有矩形的类:

import javax.swing.*;
import java.awt.*;
import java.util.ArrayList;
public class Main{
public Main() {
JFrame frame = new JFrame();
frame.setSize(1200, 900);
frame.setResizable(false);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setVisible(true);
JPanel background = new JPanel();
background.setBackground(Color.lightGray);
GreenRect gr = new GreenRect();
gr.setPreferredSize(new Dimension(500,800));
background.add(gr, BorderLayout.WEST);

RedRect rr = new RedRect();
rr.setPreferredSize(new Dimension(500,800));
background.add(rr, BorderLayout.EAST);
frame.add(background);
}
public static void main(String[] args) {
new Main();
}
}
class GreenRect extends JPanel {
ArrayList<BlackRect> r = new ArrayList<>();
ArrayList<MovingRec> m = new ArrayList<>();
public GreenRect() {
setBackground(Color.green);
addRec(10,10);
addRec(50,50);
addRec(100,100);
addRec(1000,1000);
}
public void addRec(int x, int y) {
r.add(new BlackRect(x,y));
}
}

class RedRect extends JPanel {
public RedRect() {
setBackground(Color.red);
}
}
class BlackRect extends JComponent {
int x, y;
int w = 100, h = 100;
public BlackRect (int x, int y){
this.x = x;
this.y = y;
}
@Override
public void paintComponent(Graphics g) {
super.paintComponents(g);
Graphics2D g2d = (Graphics2D) g;
g2d.setColor(Color.BLACK);
g2d.fillRect(x, y, w, h);
}
}
class MovingRec  extends JComponent {
int x, y;
int w = 20, h = 20;
public MovingRec (int x, int y){
this.x = x;
this.y = y;
}
public void paintComponent(Graphics g) {
super.paintComponents(g);
Graphics2D g2d = (Graphics2D) g;
g2d.setColor(Color.BLUE);
g2d.fillRect(x, y, w, h);
}
public void update() {
x += 5;
y += 5;
}
}

现在我在第 3 点和第 4 点上遇到了问题,因为我无法在背景上放置黑色矩形并在顶部放置移动矩形。

我将不胜感激所有的帮助:)

您不需要(也不应该)从JComponent扩展BlackRectMovingRect

例如,BlackRect可以是一个简单的对象,例如:

class BlackRect {
int x, y;
int w = 100, h = 100;
public BlackRect(int x, int y) {
this.x = x;
this.y = y;
}
public void paint(Graphics2D g2d) {
g2d.setColor(Color.BLACK);
g2d.fillRect(x, y, w, h);
}
}

您应该重写GreenRect的绘制方法,以在该面板上绘制矩形:

public GreenRect extends JPanel {
// Existing members 
@Override
public void paintComponent(Graphics g) {
super.paintComponent(g);
Graphics2D g2d = (Graphics2D) g;
for (BlackRect black_rect : r) {
black_rect.paint(g2d);
}
// Also paint list of moving rectangles here
}
}

调用GreenRect.repaint()时,它将绘制其背景以及r中的所有矩形(并在添加该代码时m列表中)。 如果m矩形的位置已更新,它们将绘制到新位置,因此它们看起来正在移动。 由于移动矩形是最后绘制的,因此它们将显示在"顶部"。

使用摆动Timer驱动动画。 当计时器到期时,它应该稍微移动所有移动的矩形(即,调用MovingRec.update()),并在GreenRect上调用repaint()

最新更新