摆动漆法

时间:2017-04-06 20:07:04

标签: java swing paint repaint

我在JPanel的paint方法中遇到问题。 首先,我创建了一个扩展JFrame的类。 然后我创建了一个扩展面板的类,并从该类创建一个对象并添加到框架中。

我想通过按键盘上的右键(VK_RIGHT),在2秒内在面板上绘制小矩形。 (这意味着矩形清晰,并在一个新的位置绘制。我的问题是:我不能做定时绘画方法和重绘方法。我希望每两秒矩形前进5次(一步一步)但矩形继续只有一个

我的代码是:

Panel

import java.awt.Color;

import java.awt.Graphics;

import javax.swing.JPanel;

public class Panel extends JPanel{

    private int x;
    private int y;
    public Panel()
    {
        x=100;
        y=100;
        this.setBackground(Color.RED);
    }

    //--------------------------------

    @Override
    public void paint(Graphics g)
    {
        super.paint(g);

        g.setColor(Color.BLACK);

        g.fillRect(x, y, 20, 20);
    }

    //--------------------------------

    @Override
    public void repaint()
    {
        super.repaint();

        x+=20;
    }

    //--------------------------------
}

Frame

import java.awt.event.KeyEvent;

import java.awt.event.KeyListener;

import javax.swing.JFrame;

public class Frame extends JFrame {

    private Panel p;

    public Frame()
    {
        super("Test");
        this.setBounds(1200, 300, 400, 400);
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        p=new Panel();
        this.add(p);


        this.addKeyListener(new KeyLis());
    }

    //----------------------------------

    public static void main(String[] args) {

        Frame a=new Frame();

        a.setVisible(true);
    }

    //-----------------------------------

    private class KeyLis implements KeyListener
    {

        @Override
        public void keyPressed(KeyEvent arg0) {

            if(arg0.getKeyCode() == KeyEvent.VK_RIGHT)
            {
                for(int i=0;i<5;i++)
                {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                    p.repaint();
                }
            }
        }
        //---------------
        @Override
        public void keyReleased(KeyEvent arg0) {
            // TODO Auto-generated method stub

        }
        //---------------
        @Override
        public void keyTyped(KeyEvent arg0) {
            // TODO Auto-generated method stub

        }
        //---------------
    }
    //-----------------------------------------
}

2 个答案:

答案 0 :(得分:2)

您的问题主要是for-loop中的KeyListener,它会阻止事件调度线程,阻止更新被绘制。

简单的解决方案是使用Swing Timer,它在后台等待,触发EDT上的滴答,安全地更新ui并定期更新。

有关详细信息,请参阅Concurrency in SwingHow to use Swing Timers

如果由于某种原因,您发现Swing Timer难以理解,您还可以尝试使用SwingWorker,这将允许您使用后台线程循环,但是提供在EDT上下文中更新状态的方法。

有关详细信息,请参阅Worker Threads and SwingWorker

建议

作为一般规则,覆盖paintComponent而不是paint。有关详细信息,请参阅Performing Custom Painting

不要使用repaint更新您的状态,您可能不是唯一一个调用它的人,而是创建一个可以在视图和您的密钥控制器之间共享的模型。

请参阅Model-View-ControllerObserver Pattern了解一些想法

我也不鼓励使用KeyListener并使用密钥绑定API

有关详细信息,请参阅How to Use Key Bindings

答案 1 :(得分:0)

如果您开始使用线程:

将您的KeyPressed事件更改为:

        @Override
        public void keyPressed(KeyEvent arg0) {

            if (arg0.getKeyCode() == KeyEvent.VK_RIGHT) {
                PaintThread th = new PaintThread(p);
                th.start();
            }
        }

创建了一个主题:

import java.util.logging.Level;
import java.util.logging.Logger;

public class PaintThread extends Thread{

    Panel panel;

    PaintThread(Panel panel){
        this.panel = panel;
    }

    @Override
    public void run() {
        for(int i = 0; i<5; i++){
            try {
                this.sleep(1000);
                panel.repaint();
            } catch (InterruptedException ex) {
                Logger.getLogger(PaintThread.class.getName()).log(Level.SEVERE, null, ex);
            }
        }
    }

}