如何防止JPanel更新?

时间:2015-05-14 11:53:22

标签: java swing graphics jpanel

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

我的一个JCheckBoxMenuItem s中有一个JMenu

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;
        }
    }
}

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

SCREENSHOT

colorPan是JPanel,其中包含不同颜色的JButton。它的顶部是mainPan

现在,"绘制模式"没有像预期的那样工作。我一直认为super.paintComponent(g);是在调用repaint()时清除/重置屏幕的那个。但我删除了它,并且看到程序的行为方式相同,我感到非常惊讶。

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

if(clear)
    super.paintComponent(g);

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

2 个答案:

答案 0 :(得分:3)

此代码中不应进行更改。它不是应该改变的油漆方法。无论是您的系统还是系统都需要油漆。当窗口调整大小或移动或部分覆盖时 - 它使用油漆再次绘制图片。 你应该做的是停止更新你的彩绘椭圆的坐标。它可以在鼠标监听器或坐标设置器中完成,或者更好地在管理这些坐标的控制部分中完成。您的复选框应控制更改模型的能力。它不应该控制绘画。 有常用的模式Model-View-Controller - 看看它。对于这样的小应用程序来说,它可能看起来有点过分,但即使Swing本身也是建立在这种模式上的,所以你已经遵循它了。当你试图打破它时,问题就会上升。所以 - 不要。

答案 1 :(得分:1)

无法“阻止JPanel更新;”将根据系统要求异步调用paintComponent()。相反,视图类的条件属性允许您实现paintComponent()在调用它时呈现所有

在下面的示例中,每次单击鼠标都会更改前景色,paintComponent()会使用修改后的设置。在引用的here更精细的示例中,ClearAction清除了定义图表模型的List<Node>List<Edge>。如果super.paintComponent(g)组件无法拨打fillRect(),则paintComponent()public void actionPerformed(ActionEvent e) { nodes.clear(); edges.clear(); repaint(); } 的{​​{1}}调用会清除所有剩余的选择工件。

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 https://stackoverflow.com/a/5312702/230513 */
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);
            }
        });
    }
}

image

git diff rev1:file1 rev2:file2