Java MouseListener MouseClicked只能运行一次

时间:2014-10-07 14:56:50

标签: java swing mouselistener

我使用JPanel为Game Of Life制作GUI,我的类扩展了JFrame并实现了MouseListener,但是当我点击JLabel时,MouseListener只能运行一次。

任何想法该怎么做?它在我的GameOfLife类的代码中不是问题,因为我也试图打印一些字符串,它只能工作一次。

谢谢! 这是班级:

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.Graphics;
import java.awt.GridLayout;
import java.awt.Image;
import java.awt.Toolkit;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.InputEvent;
import java.awt.event.KeyEvent;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import java.awt.image.BufferedImage;

import javax.swing.*;
import javax.swing.border.Border;


public class GameOfLifeWindow extends JFrame implements ActionListener, MouseListener
{
    private static final int DEF_ROWS = 16;
    private static final int DEF_COLS = 16;
    private JLabel[][] cells;
    GameOfLife gol;
    private JPanel panClear;
    private JPanel panCenter;
    private JPanel panNextGen;
    private JButton nextGen;
    private JButton clear;
    private JMenuBar menuBar;
    private JMenu file;
    private JMenuItem newGame;
    private JMenuItem loadFile;

    public GameOfLifeWindow()
    {
        super("Game Of Life");
        gol = new GameOfLife(DEF_ROWS, DEF_COLS);

        //sets layout and dimension
        this.getContentPane().setLayout(new BorderLayout());
        this.getContentPane().setPreferredSize(new Dimension(300,400));

        //Sets the panels
        panClear = new JPanel(new FlowLayout());
        panCenter = new JPanel(new GridLayout(gol.getRows(), gol.getCols()));
        panNextGen = new JPanel(new FlowLayout());

        //Adds the panel to the JFrame
        this.getContentPane().add(panClear, BorderLayout.NORTH);
        this.getContentPane().add(panCenter, BorderLayout.CENTER);
        this.getContentPane().add(panNextGen, BorderLayout.SOUTH);

        //Sets the next generation button
        this.nextGen = new JButton("Next Generation");
        panNextGen.add(this.nextGen);
        this.nextGen.addActionListener(this);
        //Sets the clear button
        ImageIcon btnIcon = new ImageIcon(getClass().getResource("images/clear.png"));
        Image tmpImg = btnIcon.getImage();
        BufferedImage bi = new BufferedImage(30, 30, BufferedImage.TYPE_INT_ARGB);
        Graphics g = bi.createGraphics();
        g.drawImage(tmpImg, 0, 0, 30, 30, null);
        this.clear = new JButton(new ImageIcon(bi));
        this.clear.addActionListener(this);
        panClear.add(clear);

        //Set the GridLayout
        this.updateGeneration();        

        //Set the mouse listener to each cell
        for(int i = 0 ; i < this.cells.length ; i++)
            for(int j = 0 ; j < this.cells[i].length ; j++)
                this.cells[i][j].addMouseListener(this);

        //Sets MenuBar
        this.menuBar = new JMenuBar();
        this.setJMenuBar(menuBar);

        //Sets the File menu
        this.file = new JMenu("File");
        menuBar.add(file);
        //Sets the New Game in the file
        this.newGame = new JMenuItem("New Game");
        this.newGame.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_F1, InputEvent.CTRL_MASK));
        this.newGame.addActionListener(this);
        file.add(this.newGame);

        //Sets the Load File in the file
        this.loadFile = new JMenuItem("Load File");
        this.loadFile.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_F2, InputEvent.CTRL_MASK));
        this.loadFile.addActionListener(this);
        file.add(this.loadFile);
    }

    public void updateGeneration()
    //updates the JLabels according to the Game Of Life Screen
    {
        Border border = BorderFactory.createLineBorder(Color.LIGHT_GRAY);
        this.getContentPane().remove(this.panCenter);
        this.panCenter = new JPanel(new GridLayout(gol.getRows(), gol.getCols()));
        this.cells = new JLabel[gol.getRows()][gol.getCols()];
        this.panCenter.removeAll();
        for(int i = 0 ; i < cells.length ; i++)
            for(int j = 0 ; j < cells[i].length ; j++)
            {
                this.cells[i][j] = new JLabel();
                this.cells[i][j] .setOpaque(true); // make the color visible
                this.cells[i][j].setBorder(border);// sets borders
                if(gol.isAlive(i, j))
                    this.cells[i][j].setBackground(Color.BLACK);
                else
                    this.cells[i][j].setBackground(Color.white);
                panCenter.add(this.cells[i][j]);
            }

        this.getContentPane().add(this.panCenter);
        this.setVisible(true);      
    }

    @Override
    public void actionPerformed(ActionEvent e)
    {
        if(e.getSource() == this.nextGen)
        {
            this.gol.nexGeneration();
            this.updateGeneration();
        }
        else if(e.getSource() == this.clear)
        {
                this.gol = new GameOfLife(DEF_ROWS, DEF_COLS);
                this.updateGeneration();
        }
        else if(e.getSource() == this.newGame)
        {
            this.gol = new GameOfLife(DEF_ROWS, DEF_COLS);
            this.updateGeneration();
        }
        else if(e.getSource() == this.loadFile)
        {
            String fileName = JOptionPane.showInputDialog("Please Enter A File Name");
            this.gol = new GameOfLife(fileName);
            System.out.println(gol.toString());
            this.updateGeneration();
        }
    }

    @Override
    public void mouseClicked(MouseEvent e) 
    {
        boolean found = false;
        for(int i = 0 ; i < cells.length && !found ; i++)
            for(int j = 0 ; j < cells[i].length && !found; j++)
                if(e.getSource() == this.cells[i][j])
                {
                    this.gol.setForGUI(i, j);
                    found = true;
                }
        this.updateGeneration();
        System.out.println("asdsad");
    }

    @Override
    public void mouseEntered(MouseEvent e) {

    }

    @Override
    public void mouseExited(MouseEvent e) {

    }

    @Override
    public void mousePressed(MouseEvent e) {

    }

    @Override
    public void mouseReleased(MouseEvent e) {
    }
}

1 个答案:

答案 0 :(得分:1)

在构造函数中,您将鼠标侦听器分配给多维数组JLabel中的所有单元格。但是,然后在您的updateGeneration()中创建另一个JLabel[][]数组,但不要再将鼠标侦听器分配给单元格。

要解决此问题,您需要在updateGeneration()中再次向所有单元格添加鼠标侦听器。或者,只需更新现有数组中JLabel[][]的状态,而不是创建新的JLabel