重绘 - 立即绘画和删除

时间:2013-03-26 21:53:18

标签: java swing repaint runnable

我有一个乒乓球项目(种类)并且它有效,但run()函数存在问题。如果我用我写入面板的功能绘制框架(它们工作,我检查)它给出了图形的问题,如果我使用重绘(如我所说)它绘制框架并立即删除它,每个解决方案将有所帮助(更好的是我的代码级别中的一个简单的:)

import java.awt.*;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.JPanel;
import java.util.Random;
import javax.swing.*;
import sun.org.mozilla.javascript.internal.Kit;



public class Picture extends JPanel implements MouseListener, Runnable{

    private int k = 0;
    Thread MyThread;
    private DrawPic img;

    private Rectangle r1, r3;

    public Picture(DrawPic img, Rectangle rect1, Rectangle rect3) {
        super();
        this.setLocation(0, 85);
        this.setLayout(new FlowLayout());
        this.setSize(1280, 1024);

        this.addMouseListener(this);
        this.setFocusable(true);

        this.r1 = rect1;

        this.r3 = rect3;
        this.img = img;
        this.MyThread = new Thread(this);
        MyThread.start();
        this.setVisible(true);

    }




    public void paintRectangleL(Rectangle rect, Graphics g) {
        k = 3;

        rect.DrawRectangle(g);
        rect.FillRectangle(g);


    }
    public void paintRectangleR(Rectangle rect, Graphics g) {
        k = 1;

        rect.DrawRectangle(g);
        rect.FillRectangle(g);

    }
    public void paintImage(DrawPic img, Graphics g) {
        k = 2;

        //g.clearRect(0, 0, this.getWidth(), this.getHeight());
        img.DrawImg(g, this);



    }
    public void changeK(int k1){
        k = k1;
    }

    @Override
    public void mouseClicked(MouseEvent e) {
        // throw new UnsupportedOperationException("Not supported yet.");



    }

    @Override
    public void mousePressed(MouseEvent e) {
        //throw new UnsupportedOperationException("Not supported yet.");

            Point p = r3.FindCenter();
            double dx, dy;

            dy = e.getY() - p.getY();
            r3.Move(0, dy);
              this.getGraphics().clearRect(0, 0, this.getWidth(), this.getHeight());
            this.paintRectangleL(r3, this.getGraphics());
            this.paintRectangleR(r1, this.getGraphics());
            this.paintImage(img, this.getGraphics());
    }

    @Override
    public void mouseReleased(MouseEvent e) {
        //throw new UnsupportedOperationException("Not supported yet.");

    }

    @Override
    public void mouseEntered(MouseEvent e) {
        //throw new UnsupportedOperationException("Not supported yet.");
    }

    @Override
    public void mouseExited(MouseEvent e) {
        //   throw new UnsupportedOperationException("Not supported yet.");
    }
    public void animate(){
        double dx = 0, dy = 2;
        if ((this.img.getX() + 160 + this.r1.RightPoint().getX() - this.r1.LeftPoint().getX() > this.getWidth() || this.img.getX() < this.r3.RightPoint().getX() - this.r3.LeftPoint().getX())) {
                    dx = -1 * dx;


                }
                if (this.img.getY() + 120> this.getHeight() || this.img.getY() < 0 ) {
                    dy = -1 * dy;
                }
                img.Move(dx, dy);
             //   this.getGraphics().clearRect(0, 0, this.getWidth(), this.getHeight());
              //  this.paintImage(img, this.getGraphics());
              //  this.paintRectangleL(r3, this.getGraphics());
              //  this.paintRectangleR(r1, this.getGraphics());
                repaint();            
    }

    @Override
    public void run() {

        Color col;
        while (true) {
                 animate();

                try {
                    MyThread.sleep(35);
                } catch (InterruptedException ex) {
                    Logger.getLogger(Picture.class.getName()).log(Level.SEVERE, null, ex);
                }
            }
        }
        //   throw new UnsupportedOperationException("Not supported yet.");
    }

1 个答案:

答案 0 :(得分:5)

您不应该使用getGraphics()进行自定义绘制,因为它是一个临时缓冲区,可在下次重绘时循环使用。你在paintComponent()画画吗?

有关更多信息和示例,请参阅Performing Custom Painting。第A Closer Look at the Paint Mechanism部分概述了paint()paintComponent()方法。另请参阅Painting in AWT and Swing

编辑:

你的程序的逻辑和结构不适合Swing的绘画过程。您需要重构您的程序,以便它可以插入该过程并在正确的时间绘制正确的东西。 通常,您可以通过覆盖其paintComponent()来自定义组件。在那种方法中,所有的绘画都发生了这个方法应该尽可能快,避免在那里放任何/太多的应用程序逻辑。

状态更改问题repaint()后,您应该保持绘制对象的某种状态(即坐标,颜色等)。这将安排重新绘制,最终Swing将在将调用paint()的组件上执行paintComponent()

在您的情况下,您有一个定期触发的计时器。您可以覆盖您使用的paintComponent的{​​{1}}。你已经有了计算坐标的逻辑。将这些坐标存储在成员变量中。然后,发出JPanel。在repaint()中,根据计算出的坐标绘制图像。

编辑:

关于线程的另一个注释。 Swing有一个单线程绘画模型。所有UI交互和绘画都在Swing的事件调度线程(EDT)上进行。有关EDT的更多信息,请查看Concurrency in Swing。请注意,方法paintComponent不会在EDT上执行。您没有显示animate()的作用,但以这种方式执行可能不安全。 invokeLater可以帮助确保代码在EDT上执行。但是,在这种特殊情况下,使用Swing timers可能更容易,这可以确保在EDT上执行操作。