0

我有一个非常简单的程序,要求用户单击与其背景颜色匹配的四个面板之一,以匹配更大的面板(又名显示面板),它随机将其背景颜色设置为四个之一。如果单击了错误的面板 Joptionpane出来; displaypanels 背景暂时设置为黑色。如果单击是,则显示面板应重新设置其背景颜色。问题是我并不总是看到背景被更新。相反,有时背景会保持黑色。...更令人困惑的是,如果您将另一个窗口拖到该程序的窗口上,如果您看到颜色在显示面板上移动时部分更新,或者只是切换到不同的窗口并重新调整,那么它会保持黑色,然后您会看到完成更新的背景颜色。

那么为什么调用 setMethod 但只是偶尔由幕后的绘制方法执行呢?为什么铸造其他窗户或框架使其可见?它与正在处理的鼠标单击事件有什么关系吗?

我感谢所有这一切的任何解释,谢谢大家

public class MainPanel extends JPanel{


    Subpanel panel1;
    Subpanel panel2;
    Subpanel panel3;
    Subpanel panel4;
    Subpanel displaypanel;

    Color[] color; //stores all the  colors that display on the subpanels


    public static void main(String[] args) {
        JFrame window=new JFrame("This is a test");
        window.setContentPane(new MainPanel());
        window.setLocation(100,30);
        window.setSize(600,500);
        window.setVisible(true);

    }


    public MainPanel(){


        setLayout(new FlowLayout(FlowLayout.CENTER,80,30));


        panel1= new Subpanel();
        panel2= new Subpanel();
        panel3= new Subpanel();
        panel4= new Subpanel();

        //the big sub panel
        displaypanel= new Subpanel();

        color=new Color[4];
        color[0]=Color.BLUE;
        color[1]=Color.RED;
        color[2]=Color.YELLOW;
        color[3]=Color.GREEN;



        setBackground(Color.GRAY);
        displaypanel.setBackground(displayRandomColor());
        panel1.setBackground(color[0]);
        panel2.setBackground(color[1]);
        panel3.setBackground(color[2]);
        panel4.setBackground(color[3]); 

        displaypanel.setPreferredSize(new Dimension(400,250));
        panel1.setPreferredSize(new Dimension(70,70));
        panel2.setPreferredSize(new Dimension(70,70));
        panel3.setPreferredSize(new Dimension(70,70));
        panel4.setPreferredSize(new Dimension(70,70));

        add(displaypanel);
        add(panel1);
        add(panel2);
        add(panel3);
        add(panel4);





    }


    public void paintComponent(Graphics g){
        super.paintComponent(g);



    }

    public Color displayRandomColor(){
            Color i=Color.WHITE;
        switch ((int)(Math.random()*4)+1){
            case 1:
            i= Color.YELLOW;
            break;

            case 2:
            i= Color.BLUE;
            break;

            case 3:
            i= Color.GREEN;
            break;

            case 4:
            i= Color.RED;
        }   
        return i;
    }


    public class Subpanel extends JPanel{

        public Subpanel(){
                this.addMouseListener(new MouseAdapter(){
                public void mouseClicked(MouseEvent evt){
                    Component source=(Component)evt.getSource();

                        if((source.getBackground()).equals(displaypanel.getBackground())){
                            //do nothing for this test..
                    }
                        else{
                                displaypanel.setBackground(Color.BLACK);

                                //ask user to reset the background color
                                //**the following 2 lines introduces the problem
                            if(JOptionPane.showOptionDialog(null,"click Yes to see a new Color","Incorrect",JOptionPane.YES_NO_OPTION,JOptionPane.PLAIN_MESSAGE,null,null,null)==JOptionPane.YES_OPTION){
                                displaypanel.setBackground(displayRandomColor());

                            }
                        }

                        }





        });
        } //end of constructor
        public void paintComponent(Graphics g){
            super.paintComponent(g);

        }

    }

}
4

1 回答 1

1

正确同步时,您的程序运行良好。几点注意事项:

  • 尽可能早地在程序执行中初始化对象一次。

  • 使用 的实例Random获取随机整数;注意更简单的实现displayRandomColor()

    private Color displayRandomColor() {
        return color[r.nextInt(color.length)];
    }
    
  • Swing GUI 对象应该事件分派线程上构建和操作。

  • setPreferredSize()当您真正想要覆盖getPreferredSize()]时,请勿使用,如此处建议那样。

  • 使用布局pack()封闭窗口。

图片

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.util.Random;
import javax.swing.JFrame;
import javax.swing.JOptionPane;
import javax.swing.JPanel;

public class MainPanel extends JPanel {

    private static final Random r = new Random();
    private final JPanel displayPanel;
    private Color[] color = {Color.RED, Color.GREEN, Color.BLUE, Color.YELLOW};
    private Subpanel panel1 = new Subpanel(color[0]);
    private Subpanel panel2 = new Subpanel(color[1]);
    private Subpanel panel3 = new Subpanel(color[2]);
    private Subpanel panel4 = new Subpanel(color[3]);

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                JFrame window = new JFrame("This is a test");
                window.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                window.add(new MainPanel());
                window.pack();
                window.setLocationRelativeTo(null);
                window.setVisible(true);
            }
        });
    }

    public MainPanel() {
        setLayout(new BorderLayout());
        setBackground(Color.GRAY);

        //the big sub panel
        displayPanel = new JPanel() {

            @Override
            public Dimension getPreferredSize() {
                return new Dimension(320, 240);
            }
        };
        displayPanel.setBackground(displayRandomColor());

        //the control panel
        JPanel p = new JPanel();
        p.add(panel1);
        p.add(panel2);
        p.add(panel3);
        p.add(panel4);

        add(displayPanel, BorderLayout.CENTER);
        add(p, BorderLayout.SOUTH);
    }

    private Color displayRandomColor() {
        return color[r.nextInt(color.length)];
    }

    public class Subpanel extends JPanel {

        public Subpanel(Color color) {
            this.setBackground(color);
            this.addMouseListener(new MouseAdapter() {
                @Override
                public void mousePressed(MouseEvent evt) {
                    Component source = (Component) evt.getSource();
                    if ((source.getBackground()).equals(displayPanel.getBackground())) {
                        System.out.println(source.getBackground());
                    } else {
                        displayPanel.setBackground(Color.BLACK);
                        if (JOptionPane.showOptionDialog(null,
                            "Click Yes to see a new Color", "Incorrect",
                            JOptionPane.YES_NO_OPTION, JOptionPane.PLAIN_MESSAGE,
                            null, null, null) == JOptionPane.YES_OPTION) {
                            displayPanel.setBackground(displayRandomColor());
                        }
                    }
                }
            });
        }

        @Override
        public Dimension getPreferredSize() {
            return new Dimension(70, 70);
        }
    }
}
于 2016-02-29T12:51:07.873 回答