1

I am trying to take an image and turn it into tiles that are laid out in GridLayout, but receive a runtime error.

Here is the code:

public class TileList extends JPanel {

private static final int width = 16;            //width of a tile
private static final int height = width;
private int col = 1;
private int row = 1;

private BufferedImage image;
File tilesetImage = new File("image.png");
BufferedImage tileset[];

public void loadAndSplitImage (File loadImage) {
    try{
        image = ImageIO.read(loadImage);
    }catch(Exception error) {
        System.out.println("Error: cannot read tileset image.");
    }// end try/catch
    col = image.getWidth()/width;
    row = image.getHeight()/height;
    BufferedImage tileset[] = new BufferedImage[col*row];
}// end loadAndSplitImage

public TileList() {
    loadAndSplitImage(tilesetImage);
    setLayout(new GridLayout(row,col,1,1));
    setBackground(Color.black);

    int x=0;
    int y=0;
    for (int i = 0; i < (col*row); i++) {
        JPanel panel = new JPanel();
        tileset[i] = new BufferedImage(width, height, image.getType());  //first error
        tileset[i] = image.getSubimage(x,y,x+width,y+height);
        panel.add(new JLabel(new ImageIcon(tileset[i])));
        add(panel);
        x+=width;
        y+=height;
    }// end for loop
}// end constructor

public static void createAndShowGui() {
    JFrame frame = new JFrame("TileList");
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.getContentPane().add(new TileList());      //second error
    frame.pack();
    frame.setLocationRelativeTo(null);
    frame.setVisible(true);
}//end createAndShowGui

public static void main (String [] args) {
    SwingUtilities.invokeLater(new Runnable() {
        public void run () {
            createAndShowGui();           //third error
        }// end run
    });// end invokeLater
  }// end main
}// end class

This is the error I receive:

Exception in thread "AWT-EventQueue-0" java.lang.NullPointerException
    at TileList.<init>(TileList.java:55)
    at TileList.createAndShowGui(TileList.java:73)
    at TileList$1.run(TileList.java:82)
    at java.awt.event.InvocationEvent.dispatch(InvocationEvent.java:251)
    at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:721)
    at java.awt.EventQueue.access$200(EventQueue.java:103)
    at java.awt.EventQueue$3.run(EventQueue.java:682)
    at java.awt.EventQueue$3.run(EventQueue.java:680)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDo
main.java:76)
    at java.awt.EventQueue.dispatchEvent(EventQueue.java:691)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThre
ad.java:244)
    at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.
java:163)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThre
ad.java:151)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:147)

    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:139)

    at java.awt.EventDispatchThread.run(EventDispatchThread.java:97)

What does this error mean and how did I write the code wrong?

Edit: I changed my code as Hovercraft suggested, however now I have new errors:

tileset[] = new BufferedImage[col*row];
       ^                           error: not a statement

tileset[] = new BufferedImage[col*row];
         ^                         error: ';' expected

tileset[] = new BufferedImage[col*row];
                             ^     error: not a statement
Emily
  • 111
  • 1
  • 1
  • 8

1 Answers1

2

Your error is that you are shadowing a variable -- you're re-declaring the variable tileSet inside constructor when it was already declared in the class. Thus the tileSet variable declared in the class is never initialized since only the local variable, the one declared in the constructor is initialized.

The solution: Only declare tileSet once, in the class.

Thus, don't do this:

public class TileList extends JPanel {

  //.... deleted for brevity

  BufferedImage tileset[];

  public void loadAndSplitImage (File loadImage) {
    try{
        image = ImageIO.read(loadImage);
    }catch(Exception error) {
        System.out.println("Error: cannot read tileset image.");
    }// end try/catch
    col = image.getWidth()/width;
    row = image.getHeight()/height;
    BufferedImage tileset[] = new BufferedImage[col*row]; // *** re-declaring variable!
  }

but instead do this:

public class TileList extends JPanel {

  //.... deleted for brevity

  BufferedImage tileset[];

  public void loadAndSplitImage (File loadImage) {
    try{
        image = ImageIO.read(loadImage);
    }catch(Exception error) {
        System.out.println("Error: cannot read tileset image.");
    }// end try/catch
    col = image.getWidth()/width;
    row = image.getHeight()/height;

    // ************
    // BufferedImage tileset[] = new BufferedImage[col*row]; // *****
    tileset[] = new BufferedImage[col*row]; // **** note the difference? ****
  }

Note that even more important than the solution for this isolated problem is to gain an understanding on how to solve NPE (NullPointerExceptions). The key is to check all the variables on the line throwing the exception, check to see which one(s) is null, and then look back to see where one of them is not being initialized properly before you try using it.

Hovercraft Full Of Eels
  • 283,665
  • 25
  • 256
  • 373
  • When I move the line to where I declare my class variables, I have to take out the [i] and replace with []. That leads to another error for incompatible types 'BufferedImage tileset[] = new BufferedImage(width,height,image.getType());' – Emily Jan 06 '13 at 02:40
  • @Emily: no don't move the line, just don't ***re-declare*** the variable. Please see my edit above. And it doesn't make sense to try to have a BufferedImage **array** be assigned a BufferedImage. – Hovercraft Full Of Eels Jan 06 '13 at 02:41
  • Oh, I didn't realize I was redeclaring it. I looked at your new edit and removed the 'BufferedImage' before tileset like you did, but now the whole line has error: not a statement – Emily Jan 06 '13 at 02:49
  • @Emily: If you're seeing an error, then you'd better show it to us, else we're not going to be able to help you. – Hovercraft Full Of Eels Jan 06 '13 at 05:17
  • @Emily: tileset has been declared as an array. Why are you using the `[]` when assigning the variable? Makes no sense, so get rid of that. – Hovercraft Full Of Eels Jan 07 '13 at 03:03
  • Sorry, I haven't worked with arrays much. That fixed my code! Thank you so much for your help! – Emily Jan 08 '13 at 03:16