如何防止 JPanel 被更新?

How to prevent the JPanel from being updated?

我正在创建一种绘画应用程序。用户可以通过 pressing/dragging 鼠标在 JPanel 中移动一个圆圈。

我的 JMenu 中有一个 JCheckBoxMenuItem

JCheckBoxMenuItem checkitem = new JCheckBoxMenuItem("Draw mode",false);

我的代码的简化版本:

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

class GUI extends JFrame implements MouseListener, MouseMotionListener, ActionListener, ItemListener
{
    JPanel mainPan, colorPan;
    Color color = Color.BLACK;
    JCheckBoxMenuItem checkitem;
    boolean clear = true;

    public GUI(String header)
    {
        maker();

        mainPan.addMouseListener(this);
        mainPan.addMouseMotionListener(this);

        add(mainPan , BorderLayout.CENTER);
        add(colorPan, BorderLayout.PAGE_END); 
    }

    public void maker()
    {
        colorPan = new JPanel();
        colorPan.setLayout(new GridLayout(1, 0));

        mainPan = new JPanel(){
            @Override
            public void paintComponent(Graphics g)
            {
                //g.setColor(Color.WHITE);
                //g.fillRect(0,0,getWidth(),getHeight());
                if(clear)
                    super.paintComponent(g); //Do the same thing as above(Clear JPanel)

                g.setColor(color);
                g.fillOval(x,y,50,50); //x and y are integer variables that I use in my full program
            }
        };

        checkitem = new JCheckBoxMenuItem("Draw mode",false);
        //After adding this to a JMenu,
        checkitem.addItemListener(this);
    }

    public void itemStateChanged(ItemEvent e)
    {
        if(e.getStateChange() == ItemEvent.SELECTED)
        {
            clear = false;
        }
        else
        {
            clear = true;
        }
    }
}

下面的屏幕截图显示了我的完整程序的输出:

colorPan 是充满不同颜色 JButton 的 JPanel。最上面是mainPan.

现在,"Draw mode" 没有按预期工作。我一直认为 super.paintComponent(g); 是调用 repaint() 时 clears/resets 屏幕的那个。但我删除了它并且很惊讶地看到程序以相同的方式运行。

基本上,我的问题在这里:

if(clear)
    super.paintComponent(g);

我需要防止在调用 repaint() 时清除所有内容。我如何实现我想要的?

这段代码中没有需要修改的地方。而且应该更改的不是绘画方法。每当您或系统需要时油漆油漆。当 window 被调整大小或移动或部分覆盖时 - 它使用绘画再次绘制图片。 你真正应该做的是停止更新你绘制的椭圆的坐标。它可以在鼠标侦听器或坐标 setter 中完成,或者更好的是,在管理这些坐标的控制部分中完成。您的复选框应控制更改模型的能力。它不应该控制绘画。 有常用的模式 Model-View-Controller - 看看它。对于这样的小应用程序,它可能看起来有点矫枉过正,但即使 Swing 本身也是建立在这种模式之上的,所以您已经遵循了它。当你试图打破它时,问题就会出现。所以 - 不要。

不能 "prevent the JPanel from being updated;" paintComponent()会被系统要求异步调用。相反,您的视图 class 的条件属性允许您的 paintComponent() 的实现在任何时候 被调用时呈现所有内容。

在下面的示例中,每次单击鼠标都会更改前景色,paintComponent() 使用修改后的设置。在引用 here, ClearAction clears the List<Node> and List<Edge> that define the graph's model. Absent a call to super.paintComponent(g), otherwise required for an opaque 组件的更详细示例中,在 paintComponent() 中调用 fillRect() 会清除所有剩余的选择工件。

public void actionPerformed(ActionEvent e) {
    nodes.clear();
    edges.clear();
    repaint();
}

import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Point;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.event.MouseMotionAdapter;
import java.util.Random;
import javax.swing.JFrame;
import javax.swing.JPanel;

/** @see  */
public class MouseDragTest extends JPanel {

    private static final String TITLE = "Drag me!";
    private static final Random r = new Random();
    private static final int W = 640;
    private static final int H = 480;
    private Point textPt = new Point(W / 2, H / 2);
    private Point mousePt;
    private Color color = Color.black;

    public MouseDragTest() {
        this.setFont(new Font("Serif", Font.ITALIC + Font.BOLD, 32));
        this.addMouseListener(new MouseAdapter() {

            @Override
            public void mousePressed(MouseEvent e) {
                mousePt = e.getPoint();
                setColor(Color.getHSBColor(r.nextFloat(), 1, 1));
                repaint();
            }
        });
        this.addMouseMotionListener(new MouseMotionAdapter() {

            @Override
            public void mouseDragged(MouseEvent e) {
                int dx = e.getX() - mousePt.x;
                int dy = e.getY() - mousePt.y;
                textPt.setLocation(textPt.x + dx, textPt.y + dy);
                mousePt = e.getPoint();
                repaint();
            }
        });
    }

    public void setColor(Color color) {
        this.color = color;
    }

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

    @Override
    public void paintComponent(Graphics g) {
        super.paintComponent(g);
        g.setColor(color);
        int w2 = g.getFontMetrics().stringWidth(TITLE) / 2;
        g.drawString(TITLE, textPt.x - w2, textPt.y);
    }

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

            @Override
            public void run() {
                JFrame f = new JFrame(TITLE);
                f.add(new MouseDragTest());
                f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                f.pack();
                f.setLocationRelativeTo(null);
                f.setVisible(true);
            }
        });
    }
}