提问者:小点点

JPanel不显示所有定制组件


我在JFrame中显示组件时遇到了一个奇怪的问题。

我必须在不使用Swing的情况下编写自己的GUI引擎(按钮、文本框等…)。只允许使用JFrame/JPanel。

假设我想放置3个按钮。

我的按钮类:

public class Button extends JPanel implements MouseListener {

Rectangle r = new Rectangle();
String text;

int X,Y,W,H;


public Button(int x, int y, int w, int h, String t)
{
    X=x;
    Y=y;
    W=w;
    H=h;

    this.setBackground(Color.CYAN);
    addMouseListener(this);

    r.setSize(w, h);
    r.setLocation(x, y);
    this.text = t;
}



@Override
public void paintComponent(Graphics g) {
    super.paintComponent(g);
    Graphics2D g2d = (Graphics2D)g;
    g2d.draw(r);
    g2d.drawString(text, X+W/2, Y+H/2);


}

@Override
public void mouseClicked(MouseEvent arg0) {
    // TODO Auto-generated method stub

    if((arg0.getButton()==1) && r.contains(arg0.getPoint())) 
        System.out.println(arg0.getPoint().toString()); 

}


@Override
public void mouseEntered(MouseEvent arg0) {
    // TODO Auto-generated method stub

}


@Override
public void mouseExited(MouseEvent arg0) {
    // TODO Auto-generated method stub

}


@Override
public void mousePressed(MouseEvent arg0) {
    // TODO Auto-generated method stub

}


@Override
public void mouseReleased(MouseEvent arg0) {
    // TODO Auto-generated method stub

}

}

在主类中,我创建了一个JFrame和JPanel。我向JPanel添加了3个按钮,最后将JPanel添加到JFrame,但只显示了最后一个声明的按钮。

public static void main(String[] args) {
    // TODO Auto-generated method stub


    JFrame f = new JFrame("Demo");
    f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    f.setBackground(Color.cyan);

    JPanel j = new JPanel(new BorderLayout());

    j.add(new Button(10,10,100,50,"text"));
    j.add(new Button(10,100,100,50,"text2"));
    j.add(new Button(300,10,100,50,"text3"));

    f.add(j);
    f.pack();
    f.setSize(640, 400);;
    f.setVisible(true);


}

我做错了什么?


共2个答案

匿名用户

您的代码不遵守BorderLayout规则。当使用容器而未指定BorderLayout位置向BorderLayout添加3个组件时,它们都会被添加到默认BorderLayout.CENTER点,最后添加的一个覆盖了其他3个。添加组件或使用其他布局管理器时,请考虑使用BorderLayout常量。

说到这里,我认为你最好完全改变路线。相反,考虑…

  • 使您的Button类成为逻辑和非GUI类(即,通过不扩展JPanel),
  • 让一个非按钮JPanel持有一个List

这将大大简化事情。

import java.awt.Color;
import java.awt.Dimension;
import java.awt.FontMetrics;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Rectangle;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.geom.Rectangle2D;
import java.util.ArrayList;
import java.util.List;

import javax.swing.*;

public class FooGui extends JPanel {
    private static final int PREF_W = 640;
    private static final int PREF_H = 400;
    private List<MyButton> btnList = new ArrayList<>();

    public FooGui() {
        addMouseListener(new MyMouse());
    }

    public void addButton(MyButton btn) {
        btnList.add(btn);
    }

    @Override
    protected void paintComponent(Graphics g) {
        super.paintComponent(g);
        for (MyButton myButton : btnList) {
            myButton.draw(g);
        }
    }

    @Override
    public Dimension getPreferredSize() {
        if (isPreferredSizeSet()) {
            return super.getPreferredSize();
        }
        return new Dimension(PREF_W, PREF_H);
    }

    private class MyMouse extends MouseAdapter {
        @Override
        public void mousePressed(MouseEvent e) {
            for (MyButton myButton : btnList) {
                if (myButton.getRect().contains(e.getPoint())) {
                    System.out.println("Text: " + myButton.getText());
                }
            }
        }
    }

    private static void createAndShowGui() {
        FooGui fooGui = new FooGui();
        fooGui.addButton(new MyButton(10,10,100,50,"text"));
        fooGui.addButton(new MyButton(10,100,100,50,"text2"));
        fooGui.addButton(new MyButton(300,10,100,50,"text3"));

        JFrame frame = new JFrame("FooGui");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.getContentPane().add(fooGui);
        frame.pack();
        frame.setLocationRelativeTo(null);
        frame.setVisible(true);
    }

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                createAndShowGui();
            }
        });
    }
}

class MyButton {
    private static final Color BK = Color.CYAN;
    private static final Color TEXT_COLOR = Color.BLACK;
    private int x;
    private int y;
    private int w;
    private int h;
    private String text;
    private Rectangle rect;

    public MyButton(int x, int y, int w, int h, String t) {
        this.x = x;
        this.y = y;
        this.w = w;
        this.h = h;
        this.text = t;
        rect = new Rectangle(x, y, w, h);
    }

    public void draw(Graphics g) {
        Graphics2D g2 = (Graphics2D) g;
        g2.setColor(BK);
        g2.fill(rect);
        g2.setColor(TEXT_COLOR);
        FontMetrics metrics = g2.getFontMetrics();
        Rectangle2D bounds = metrics.getStringBounds(text, g2);
        int textX = (int) (x + (w - bounds.getWidth()) / 2);
        int textY = (int) (y + (h + bounds.getHeight()) / 2);

        g2.drawString(text, textX, textY);
    }

    public Rectangle getRect() {
        return rect;
    }

    public int getX() {
        return x;
    }

    public int getY() {
        return y;
    }

    public int getW() {
        return w;
    }

    public int getH() {
        return h;
    }

    public String getText() {
        return text;
    }
}

匿名用户

这个从属关系的目的是创建一个GUI,当例如某个设备(如旧手机)中没有实现JButton、JLabel等时可以使用

嗯,这并没有真正回答我的问题。你提到了自定义绘制纽扣和标签的问题,但是还有更多的GUI然后只是绘制东西。

我还问您是否可以使用AWT的其他功能,如MouseListeners、KeyListeners、制表符、布局管理器等。因为如果您可以使用这些功能,那么就没有理由像气垫船在回答中所做的那样完全重新发明轮子。

如果你需要做的只是扩展JPanel并为按钮或标签进行自定义绘制,那么你的代码的问题是你没有正确使用布局管理器。这是框架的默认布局是边界布局,你不能将多个组件添加到边界布局的中心。

发布代码的第二个问题是您没有覆盖组件的getPreferredSize()方法。因此大小将为0,布局管理器无法正常工作。