调整JPanel BorderLayout Java的大小

时间:2018-02-20 16:59:59

标签: java swing border-layout

我尝试使用挥杆,但我遇到了一个无法解决的问题。我想做的很简单:我只想使用BorderLayout在JFrame中使用JPanel。 问题是我的中心面板总是位于我的North Jpanel上方。事实上,无论我给我的北面板的大小只有10像素,在中心pannel开始之后(就像这个image)。

注意:当我将第二个面板放在南方时,第一个面板有足够的位置可以绘制,但即使它有更多的位置,第二个面板也只需要10个像素,这是不够的(如此图像)。

这是我的Plateau构造函数类,它扩展了JFrame:

public Plateau(){
    super("arkanoid");

    this.setLayout(new BorderLayout());

    setFocusable(true);
    setLocationRelativeTo(null);
    setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);


    this.getContentPane().add(affich,BorderLayout.NORTH);
    this.getContentPane().add(jeu, BorderLayout.CENTER);



    setVisible(true);
    this.setResizable(false);  
    this.setMinimumSize(new Dimension(700,800));
    this.setLocationRelativeTo(null);

} 

这里我的面板的一部分位于中心(其余部分是变量修改和绘图功能):

public class Jeu extends JPanel {
    public Jeu(int score, int plateauX, int balleX, int balleY, boolean perdu){

        super();
    }
    public void paint(Graphics g){


        Graphics2D g2 = (Graphics2D)g;
        this.setSize(new Dimension(Width,Heigth));
    }
 }

这是我的全班应该在北方:

public class Affich extends JPanel {
    public Affich() {
        super();
        this.setSize(100,100);
    }

    public void paint(Graphics g){
        this.setSize(100,100);
        g.drawOval(0, 0, 50, 50);
    }
}

我希望我足够清楚

1 个答案:

答案 0 :(得分:4)

从不在绘画方法中调用setSize(...)或类似内容。这些方法应仅用于绘画和绘画,如果你试图改变大小状态,你最终可能会遇到无休止的调用的恶性循环 - 设置大小调用重绘的大小,这会调用重绘。

相反:

  • 覆盖JPanel的paintComponent 而非绘制,因为paint不仅仅是绘制JPanel,而且覆盖它可能会对JPanel的边框和子组件产生意想不到的后果。 。
  • 在覆盖
  • 中调用super的paintComponent
  • 再次,在绘画方法中仅绘制
  • 不要设置大小,而是设置首选大小,并设置一次调用的代码,而不是绘制方法。

例如

import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.RenderingHints;
import javax.swing.*;

public class MyDrawing {
    public static void main(String[] args) {
        SwingUtilities.invokeLater(() -> {
            JFrame frame = new JFrame("GUI");
            frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

            int affichW = 100;
            int affichH = 100;
            Affich affich = new Affich(affichW , affichH );
            Jeu jeu = new Jeu();

            frame.add(affich, BorderLayout.PAGE_START);
            frame.add(jeu, BorderLayout.CENTER);

            frame.pack();
            frame.setLocationRelativeTo(null);
            frame.setVisible(true);
        });
    }
}

@SuppressWarnings("serial")
class Jeu extends JPanel {
    private static final int JEU_W = 600;
    private static final int JEU_H = 450;

    public Jeu(int score, int plateauX, int balleX, int balleY, boolean perdu) {
        super();
        setBorder(BorderFactory.createTitledBorder("Jeu"));
    }

    @Override
    public Dimension getPreferredSize() {
        if (isPreferredSizeSet()) {
            return super.getPreferredSize();            
        } else {
            return new Dimension(JEU_W, JEU_H);
        }
    }

    public Jeu() {
        this(0, 0, 0, 0, false);
    }

    @Override
    protected void paintComponent(Graphics g) {
        super.paintComponent(g);

        Graphics2D g2 = (Graphics2D) g;
        // draw with g2 here
    }
}

@SuppressWarnings("serial")
class Affich extends JPanel {
    private int width = 0;
    private int height = 0;

    public Affich(int width, int height) {
        super();
        this.width = width;
        this.height = height;
    }

    @Override
    public Dimension getPreferredSize() {
        if (isPreferredSizeSet()) {
            return super.getPreferredSize();
        } else {
            return new Dimension(width, height);
        }
    }

    @Override
    protected void paintComponent(Graphics g) {
        super.paintComponent(g);
        Graphics2D g2 = (Graphics2D) g;
        // draw smooth oval
        g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
        g.drawOval(0, 0, 50, 50);
    }
}