简体   繁体   中英

Java MouseListener MouseClicked only works once

Im using a JPanel to make a GUI for Game Of Life, my class extends JFrame and implements MouseListener, but when im clicking on a JLabel, the MouseListener works only once.

Any idea what to do?its not a problem in the code in my GameOfLife class because im also trying to print some string and it only works once.

Thanks! here is the class:

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) {
    }
}

In your Constructor you assign a mouse listener to all of your cells which you have in a multidimensional array of JLabel . However, then in your updateGeneration() you create another JLabel[][] array but don't assign your mouse listener to cells again.

To fix you need to add a mouse listener to all the cells again in updateGeneration() . Or instead of creating a new JLabel[][] just update the state of the JLabel in the existing array.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM