java - 如何使用 javax.swing.Timer 暂停 GUI 程序

标签 java swing user-interface timer event-dispatch-thread

在问这个问题之前,我已经做了研究并尝试了一些方法,但似乎没有任何效果。

下面我包含了代码的简化版本。

我有一个程序,基本上可以移动棋 table 上的棋子。该表是使用类 JTable 实现的。我想要完成的是在每次移动之间创建几秒钟的延迟。

表是一个扩展 JTable 的类。构造函数在桌面上创建一个棋子。

public Table (int rows, int columns)
{
    ChessPiece piece = new ChessPiece();
}

table 上有一个菜单,其中包含移动棋子的选项。该选项将通过调用 moveForward() 方法将棋子向前移动。

JMenuBar menuBar = new JMenuBar();
JMenu menu       = new JMenu("Menu");
JMenuItem item   = new JMenuItem("Move Chess Piece");
menu.add(item);
menuBar.add(menu);
item.addActionListener(new ActionListener()
                       {
                           public void actionPerformed(ActionEvent event)
                           {
                               piece.moveForward();
                           }
                       });
setJMenuBar(menuBar);

要启动该程序,我执行以下操作:

Table table = new Table();
javax.swing.SwingUtilities.invokeLater( new Runnable()
                                        { public void run()
                                          {
                                              table.setVisible(true);
                                          }
                                        } );

因此,为了创建几秒钟的延迟,我首先尝试了以下操作:

public void moveForward()
{
    // Sleep for a couple of seconds before moving.
    try { Thread.sleep(SLEEP_TIME);}
    catch(InterruptedException ex) { Thread.currentThread().interrupt(); }

    .....<code to move the chess piece forward>....
}

当我单击“移动棋子”选项时,这会导致整个 GUI 卡住。我不清楚为什么程序会这样。如果有人能用简单的英语向我解释,我将不胜感激!

我环顾四周,有人建议我应该使用 java.swing.Timer。所以我将代码更改为以下内容:

public void moveForward()
{
    ActionListener taskPerformer = new ActionListener() {
                                       public void actionPerformed(ActionEvent evt)
                                       {
                                           System.out.println("Hello!");

                                       }
                                     };
    javax.swing.Timer t = new javax.swing.Timer(10000, taskPerformer);
    t.setRepeats(false);
    t.start();

    .....<code to move the chess piece forward>....
}

当我单击“移动棋子”选项时,GUI 不再卡住,但它什么也不做。没有我预期的 10 秒延迟。

请指出这里有什么问题!

提前致谢。

最佳答案

这似乎工作得很完美,将棋子从起始位置一个接一个地移动到棋盘中央的前面。

为了更快地获得更好的帮助,请发布 Minimal Complete Tested and Readable Example (就像这段代码)行为不同。此 MCTRE 基于 Making a robust, resizable Swing Chess GUI 中看到的代码.

import java.awt.*;
import java.awt.event.*;
import java.awt.image.BufferedImage;
import javax.swing.*;
import javax.swing.border.*;
import java.net.URL;
import javax.imageio.ImageIO;

public class AnimatedChessGui {

    private final JPanel gui = new JPanel(new BorderLayout(3, 3));
    private JButton[][] chessBoardSquares = new JButton[8][8];
    private Image[][] chessPieceImages = new Image[2][6];
    private JPanel chessBoard;
    private final JLabel message = new JLabel(
            "Chess Champ is ready to play!");
    private static final String COLS = "ABCDEFGH";
    public static final int QUEEN = 0, KING = 1,
            ROOK = 2, KNIGHT = 3, BISHOP = 4, PAWN = 5;
    public static final int[] STARTING_ROW = {
        ROOK, KNIGHT, BISHOP, KING, QUEEN, BISHOP, KNIGHT, ROOK
    };
    public static final int BLACK = 0, WHITE = 1;
    private Timer timer;

    AnimatedChessGui() {
        initializeGui();
    }

    public void animateChessPieces() {
        timer = new Timer(1000, new ActionListener() {

            int count = 0;

            @Override
            public void actionPerformed(ActionEvent e) {
                if (count % 2 == 0) {
                    chessBoardSquares[count/2][1].setIcon(null);
                    chessBoardSquares[count/2][3].setIcon(new ImageIcon(
                            chessPieceImages[BLACK][PAWN]));
                } else {
                    chessBoardSquares[count/2][6].setIcon(null);
                    chessBoardSquares[count/2][4].setIcon(new ImageIcon(
                            chessPieceImages[WHITE][PAWN]));
                }
                count++;
                if (count == 16) {
                    timer.stop();
                }
            }
        });
        timer.start();
    }

    public final void initializeGui() {
        // create the images for the chess pieces
        createImages();

        // set up the main GUI
        gui.setBorder(new EmptyBorder(5, 5, 5, 5));
        JToolBar tools = new JToolBar();
        tools.setFloatable(false);
        gui.add(tools, BorderLayout.PAGE_START);
        Action newGameAction = new AbstractAction("New") {

            @Override
            public void actionPerformed(ActionEvent e) {
                setupNewGame();
            }
        };
        tools.add(newGameAction);
        tools.add(new JButton("Save")); // TODO - add functionality!
        tools.add(new JButton("Restore")); // TODO - add functionality!
        tools.addSeparator();
        tools.add(new JButton("Resign")); // TODO - add functionality!
        tools.addSeparator();
        tools.add(message);

        gui.add(new JLabel("?"), BorderLayout.LINE_START);

        chessBoard = new JPanel(new GridLayout(0, 9)) {

            /**
             * Override the preferred size to return the largest it can, in a
             * square shape. Must (must, must) be added to a GridBagLayout as
             * the only component (it uses the parent as a guide to size) with
             * no GridBagConstaint (so it is centered).
             */
            @Override
            public final Dimension getPreferredSize() {
                Dimension d = super.getPreferredSize();
                Dimension prefSize = null;
                Component c = getParent();
                if (c == null) {
                    prefSize = new Dimension(
                            (int) d.getWidth(), (int) d.getHeight());
                } else if (c != null
                        && c.getWidth() > d.getWidth()
                        && c.getHeight() > d.getHeight()) {
                    prefSize = c.getSize();
                } else {
                    prefSize = d;
                }
                int w = (int) prefSize.getWidth();
                int h = (int) prefSize.getHeight();
                // the smaller of the two sizes
                int s = (w > h ? h : w);
                return new Dimension(s, s);
            }
        };
        chessBoard.setBorder(new CompoundBorder(
                new EmptyBorder(8, 8, 8, 8),
                new LineBorder(Color.BLACK)));
        // Set the BG to be ochre
        Color ochre = new Color(204, 119, 34);
        chessBoard.setBackground(ochre);
        JPanel boardConstrain = new JPanel(new GridBagLayout());
        boardConstrain.setBackground(ochre);
        boardConstrain.add(chessBoard);
        gui.add(boardConstrain);

        // create the chess board squares
        Insets buttonMargin = new Insets(0, 0, 0, 0);
        for (int ii = 0; ii < chessBoardSquares.length; ii++) {
            for (int jj = 0; jj < chessBoardSquares[ii].length; jj++) {
                JButton b = new JButton();
                b.setMargin(buttonMargin);
                // our chess pieces are 64x64 px in size, so we'll
                // 'fill this in' using a transparent icon..
                ImageIcon icon = new ImageIcon(
                        new BufferedImage(64, 64, BufferedImage.TYPE_INT_ARGB));
                b.setIcon(icon);
                if ((jj % 2 == 1 && ii % 2 == 1)
                        //) {
                        || (jj % 2 == 0 && ii % 2 == 0)) {
                    b.setBackground(Color.WHITE);
                } else {
                    b.setBackground(Color.BLACK);
                }
                chessBoardSquares[jj][ii] = b;
            }
        }

        /*
         * fill the chess board
         */
        chessBoard.add(new JLabel(""));
        // fill the top row
        for (int ii = 0; ii < 8; ii++) {
            chessBoard.add(
                    new JLabel(COLS.substring(ii, ii + 1),
                    SwingConstants.CENTER));
        }
        // fill the black non-pawn piece row
        for (int ii = 0; ii < 8; ii++) {
            for (int jj = 0; jj < 8; jj++) {
                switch (jj) {
                    case 0:
                        chessBoard.add(new JLabel("" + (9 - (ii + 1)),
                                SwingConstants.CENTER));
                    default:
                        chessBoard.add(chessBoardSquares[jj][ii]);
                }
            }
        }
    }

    public final JComponent getGui() {
        return gui;
    }

    private final void createImages() {
        try {
            URL url = new URL("/image/memI0.png");
            BufferedImage bi = ImageIO.read(url);
            for (int ii = 0; ii < 2; ii++) {
                for (int jj = 0; jj < 6; jj++) {
                    chessPieceImages[ii][jj] = bi.getSubimage(
                            jj * 64, ii * 64, 64, 64);
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
            System.exit(1);
        }
    }

    /**
     * Initializes the icons of the initial chess board piece places
     */
    private final void setupNewGame() {
        message.setText("Make your move!");
        // set up the black pieces
        for (int ii = 0; ii < STARTING_ROW.length; ii++) {
            chessBoardSquares[ii][0].setIcon(new ImageIcon(
                    chessPieceImages[BLACK][STARTING_ROW[ii]]));
        }
        for (int ii = 0; ii < STARTING_ROW.length; ii++) {
            chessBoardSquares[ii][1].setIcon(new ImageIcon(
                    chessPieceImages[BLACK][PAWN]));
        }
        // set up the white pieces
        for (int ii = 0; ii < STARTING_ROW.length; ii++) {
            chessBoardSquares[ii][6].setIcon(new ImageIcon(
                    chessPieceImages[WHITE][PAWN]));
        }
        for (int ii = 0; ii < STARTING_ROW.length; ii++) {
            chessBoardSquares[ii][7].setIcon(new ImageIcon(
                    chessPieceImages[WHITE][STARTING_ROW[ii]]));
        }
    }

    public static void main(String[] args) {
        Runnable r = new Runnable() {

            @Override
            public void run() {
                AnimatedChessGui acg = new AnimatedChessGui();

                JFrame f = new JFrame("ChessChamp");
                f.add(acg.getGui());
                // Ensures JVM closes after frame(s) closed and
                // all non-daemon threads are finished
                f.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
                // See https://stackoverflow.com/a/7143398/418556 for demo.
                f.setLocationByPlatform(true);

                // ensures the frame is the minimum size it needs to be
                // in order display the components within it
                f.pack();
                // ensures the minimum size is enforced.
                f.setMinimumSize(f.getSize());
                f.setVisible(true);
                acg.setupNewGame();
                acg.animateChessPieces();
            }
        };
        // Swing GUIs should be created and updated on the EDT
        // http://docs.oracle.com/javase/tutorial/uiswing/concurrency
        SwingUtilities.invokeLater(r);
    }
}

关于java - 如何使用 javax.swing.Timer 暂停 GUI 程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21507506/

相关文章:

java - 锁定异常 : Failure obtaining db row lock: No operations

java - Spring Boot 过滤器未返回正确的响应

java - JFileChooser组件显示怪异

java - 调整 JMenuItem 的大小?

java - 更新另一个 swing gui 类中的 jTextField

java - BufferedReader 和 InputStreamReader 是否应该显式关闭?

java - JMenuItem 设置对齐方式和最大尺寸

user-interface - Qt Creator - 如何编写 UI?

Java 的 setPreferredSize 不会调整 JPanel 的大小

java - BoxLayout 无法将奇数宽度的单个元素居中对齐