在一个JPanel中添加另一个JPanel——嵌套的JPanel不会显示

3

新代码

package test;

import javax.swing.*;
import java.awt.*;


public class TestWindow extends JFrame{
//------------------------------------------------------------------------------
    public static void main(String[] args) {
        new TestWindow();
    }
//------------------------------------------------------------------------------
    public TestWindow(){
        setSize(300,300);
        this.setUndecorated(true);
        add(new Background());
        setVisible(true);
    }
//------------------------------------------------------------------------------

    private class Background extends JPanel{
        public Background(){
            add(b);
            repaint();
        }
//------------------------------------------------------------------------------    
        Bubble b = new Bubble();
        @Override
        public void paintComponent(Graphics g){
            super.paintComponent(g);
            Color c = Color.cyan;
            g.setColor(c);
            g.fillRect(0, 0,getWidth(), getHeight());
        }
//------------------------------------------------------------------------------
        private class Bubble extends JPanel{
            @Override
            public void paintComponent(Graphics g){
                super.paintComponent(g);
                g.setColor(Color.green);
                g.drawOval(0, 0, Background.this.getWidth(), Background.this.getHeight());
            }
        }
//------------------------------------------------------------------------------
    }
}  

输出

enter image description here

问题

目标是绘制一个带有绿色圆圈的青色窗口。稍后我将向绿色圆圈添加组件,使其看起来像是具有青色背景和绿色圆圈及其中组件的窗口。
然而,输出只显示了青色背景。没有圆圈。

我尝试设置异或模式为青色,但这也没有起作用。我是否嵌套了类?


1
你在Background类的paintComponent()方法中调用了add(),你需要在构造函数中调用它。 - Extreme Coders
尝试过了,有一个非常小的正方形在中心处被绘制出来。稍后会上传一张图片。 - An SO User
在旧代码下方展示新代码。如果我们看不到“无法运行”的情况,我们怎么能帮助它呢?当然,@ExtremeCoders是正确的。你绝对不应该在paintComponent(...)内添加或删除组件,或尝试进行程序逻辑。这是一个非常糟糕的想法。 - Hovercraft Full Of Eels
2
不要在构造函数中调用repaint()。除非是动画效果,否则应该让系统来处理重绘。 - Extreme Coders
1
对于sscce,请点赞。 - trashgod
显示剩余2条评论
2个回答

8

主要问题在这里...

    public void paintComponent(Graphics g){
        super.paintComponent(g);
        Color c = Color.cyan;
        g.setColor(c);
        g.fillRect(0, 0,getWidth(), getHeight());
        add(b);
        repaint();
    }

不仅在你的绘制方法中添加组件,还调用了repaint方法,这些都会对你造成不利影响。当组件需要更新时,无论出于何种原因,repaint管理器都会调用paint方法。你不应该调用任何可能使其无效或需要重新绘制的方法,否则将导致CPU过度负荷。相反,你应该在Background组件的构造函数中添加Bubble组件,并重写这些组件的getPreferredSize方法,提供有用的提示,以便布局管理器有一些关于组件实际使用多少空间的想法。你面临的主要问题(除了糟糕的绘制)是组件报告自己不需要高度或宽度,这意味着当布局管理器来进行布局时,它们实际上是不可见的。
更新
我建议你看一下

彩蛋

为了表彰你的努力和取得的成果,让我给你一点帮助...

enter image description here

我建议你阅读代码,回到Java文档和教程中,尝试弄清楚发生了什么 :)

public class CircleControl {

    public static void main(String[] args) {
        new CircleControl();
    }

    public CircleControl() {
        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 TestPane());
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    public class TestPane extends JPanel {

        public TestPane() {
            setLayout(new GridBagLayout());
            add(new Bubble());
        }

        @Override
        public Dimension getPreferredSize() {
            return new Dimension(200, 300);
        }

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

            Graphics2D g2d = (Graphics2D) g.create();

            Color startColor = brighten(Color.CYAN, 0.75f);
            Color endColor = brighten(Color.CYAN, 0.5f);

            LinearGradientPaint lgp = new LinearGradientPaint(
                            new Point(0, 0),
                            new Point(0, getHeight()),
                            new float[]{0f, 1f},
                            new Color[]{startColor, endColor});

            g2d.setPaint(lgp);
            g2d.fill(new Rectangle(getWidth(), getHeight()));
            g2d.dispose();

        }
    }

    public class Bubble extends JPanel {

        public Bubble() {
            Font font = UIManager.getFont("Label.font");
            setFont(font.deriveFont(Font.BOLD, 48));
            setForeground(Color.WHITE);
            setBackground(darken(Color.CYAN, 0.3f));
            setOpaque(false);
        }

        @Override
        public Dimension getPreferredSize() {
            return new Dimension(150, 150);
        }

        @Override
        protected void paintComponent(Graphics g) {
            super.paintComponent(g);
            Graphics2D g2d = (Graphics2D) g.create();
            g2d.setRenderingHint(RenderingHints.KEY_ALPHA_INTERPOLATION, RenderingHints.VALUE_ALPHA_INTERPOLATION_QUALITY);
            g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
            g2d.setRenderingHint(RenderingHints.KEY_COLOR_RENDERING, RenderingHints.VALUE_COLOR_RENDER_QUALITY);
            g2d.setRenderingHint(RenderingHints.KEY_DITHERING, RenderingHints.VALUE_DITHER_ENABLE);
            g2d.setRenderingHint(RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_ON);
            g2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR);
            g2d.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
            g2d.setRenderingHint(RenderingHints.KEY_STROKE_CONTROL, RenderingHints.VALUE_STROKE_PURE);
            g2d.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
            
            int x = (getWidth() - 150) / 2;
            int y = (getHeight() - 150) / 2;

            Color startColor = brighten(getBackground(), 0.05f);
            Color endColor = getBackground();

            LinearGradientPaint lgp = new LinearGradientPaint(
                            new Point(x, y),
                            new Point(x, y + 150),
                            new float[]{0f, 1f},
                            new Color[]{startColor, endColor});
            g2d.setPaint(lgp);
            g2d.fill(new Ellipse2D.Double(x, y, 150, 150));
            
            FontMetrics fm = g2d.getFontMetrics();
            x = x + ((150 - fm.stringWidth("22")) / 2);
            y = y + ((150 / 2) + fm.getAscent());
            g2d.setColor(getForeground());
            g2d.drawString("22", x, y);

        }
    }

    public static Color brighten(Color color, double fraction) {
        int red = (int) Math.round(Math.min(255, color.getRed() + 255 * fraction));
        int green = (int) Math.round(Math.min(255, color.getGreen() + 255 * fraction));
        int blue = (int) Math.round(Math.min(255, color.getBlue() + 255 * fraction));
        int alpha = color.getAlpha();
        return new Color(red, green, blue, alpha);
    }

    public static Color darken(Color color, double fraction) {
        int red = (int) Math.round(Math.max(0, color.getRed() - 255 * fraction));
        int green = (int) Math.round(Math.max(0, color.getGreen() - 255 * fraction));
        int blue = (int) Math.round(Math.max(0, color.getBlue() - 255 * fraction));
        int alpha = color.getAlpha();
        return new Color(red, green, blue, alpha);
    }
}

噢,好吧...从之前的问题中,你知道了我的天气应用程序,所以我可以假设这个例子是想嘲笑我? :D 开玩笑的。 - An SO User
2
@LittleChild:模拟?不是。联系?是的。 - trashgod
@trashgod 我只是开玩笑 :) - An SO User
@trashgod:这里需要更多的帮助。 (http://stackoverflow.com/questions/14910124/how-to-meaningfully-extract-information-from-a-json-reply/14910200#14910200) - An SO User
1
@LittleChild 是的。我知道你的天气应用程序,是的,我建议你在遇到问题时尝试一些东西并发布一个问题,这就是你所做的。我只是认为为你提供一点额外的帮助可能会很好,作为奖励,因为你采纳了周围人的建议,这在今天是如此罕见。 - MadProgrammer
@MadProgrammer,现在GUI已经处理好了,我正在使用GSON进行JSON解析。建议已经被积极采纳 :) - An SO User

2
我已经运行了你的代码并做出了更改。这是修改后的代码。然而,你的程序没有处理窗口关闭的情况。你需要自己编写处理程序。
package test;
import javax.swing.*;
import java.awt.*;

public class TestWindow extends JFrame
{
    //------------------------------------------------------------------------------
    public static void main(String[] args) {
        new TestWindow();
    }
    //------------------------------------------------------------------------------
    public TestWindow(){
        setSize(300,310);
        this.setUndecorated(true);
        add(new Background());
        setVisible(true);
    }
//------------------------------------------------------------------------------

    private class Background extends JPanel{
        public Background(){
            Bubble b = new Bubble();
            add(b);
        }
        //------------------------------------------------------------------------------    

        @Override public Dimension getPreferredSize()
        {
            return new Dimension(300,300);
        }

        @Override
        public void paintComponent(Graphics g){
            super.paintComponent(g);
            Color c = Color.cyan;
            g.setColor(c);
            g.fillRect(0, 0,getWidth(), getHeight());
        }
        //------------------------------------------------------------------------------
        private class Bubble extends JPanel{

            Bubble()
            {
                setOpaque(false);
            }

            @Override
            public void paintComponent(Graphics g){
                super.paintComponent(g);
                g.setColor(Color.green);
                g.fillOval(0, 0, getWidth(), getHeight());
            }

            @Override public Dimension getPreferredSize()
            {
                return new Dimension(300,300);
            }
        }
//------------------------------------------------------------------------------
    }
}  

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接