summaryrefslogblamecommitdiff
path: root/tools/Reorganize.java
blob: 0b9c860605d4307bee7410a46584c25538daa65f (plain) (tree)





































































































































































































                                                                               
/* Reorganize (c) 2006 Bjørn Lindeijer
 * License: GPL, v2 or later
 */

import java.awt.Color;
import java.awt.Graphics;
import java.awt.Rectangle;
import java.awt.image.BufferedImage;
import java.util.Vector;
import java.io.IOException;
import java.io.File;
import javax.imageio.ImageIO;

/**
 * Tool to reorganize the headgears.
 */
public class Reorganize
{
    private static final int SPRITE_WIDTH = 60;
    private static final int SPRITE_HEIGHT = 60;
    private static final int FRAMES = 10;
    private static final int DIRECTIONS = 4;

    private static final int HAIR_COLORS = 10;
    private static final int HAIR_FRAMES = 9;
    private static final int HAIR_SPRITE_WIDTH = 40;
    private static final int HAIR_SPRITE_HEIGHT = 40;

    private static final int TRANSPARENT = new Color(255, 0, 255).getRGB();

    public static void main(String[] arg)
    {
        if (arg.length != 2) {
            System.out.println("Usage:\n   java Reorganize [source] [target]");
            return;
        }

        BufferedImage source = null;
        try {
            source = ImageIO.read(new File(arg[0]));
        } catch (IOException e) {
            System.out.println("Error while trying to read " + arg[0] + ".");
            e.printStackTrace();
            System.exit(1);
        }

        // Read the existing frames into a vector
        Vector<BufferedImage> spriteSet = gridCut(source,
                HAIR_SPRITE_WIDTH, HAIR_SPRITE_HEIGHT,
                HAIR_FRAMES, 1);

        // Determine minimal rectangle that can still contain the contents of
        // any frame
        /*
        Rectangle cropRect = minimumCropRect(spriteSet);

        if (cropRect == null) {
            System.out.println(
                    "Error: no optimal crop rect could be determined.");
            System.exit(1);
        }

        System.out.println(arg[0] + ": width=\"" +
                cropRect.width + "\" height=\"" + cropRect.height + "\"");
        */

        filterHeadgear(spriteSet);

        BufferedImage target = gridDraw(
                spriteSet,
                new Rectangle(0, 0, HAIR_SPRITE_WIDTH, HAIR_SPRITE_HEIGHT),
                HAIR_FRAMES - 4, 1);

        // Save the target image
        try {
            ImageIO.write(target, "png", new File(arg[1]));
        } catch (IOException e) {
            System.out.println("Error while trying to write " + arg[1] + ".");
            e.printStackTrace();
            System.exit(1);
        }
    }

    private static Vector<BufferedImage> gridCut(
            BufferedImage source,
            int width, int height, int xFrames, int yFrames)
    {
        Vector<BufferedImage> spriteSet = new Vector<BufferedImage>();

        for (int y = 0; y < yFrames; y++) {
            for (int x = 0; x < xFrames; x++) {
                BufferedImage sprite = source.getSubimage(
                        x * width,
                        y * height,
                        width,
                        height);

                spriteSet.add(sprite);
            }
        }

        return spriteSet;
    }

    private static BufferedImage gridDraw(Vector<BufferedImage> spriteSet,
            Rectangle cropRect, int xFrames, int yFrames)
    {
        // Create a new image
        BufferedImage target = new BufferedImage(
                xFrames * cropRect.width,
                yFrames * cropRect.height,
                BufferedImage.TYPE_INT_ARGB);

        // Draw the frames onto the target image
        Graphics g = target.getGraphics();
        for (int y = 0; y < yFrames; y++) {
            for (int x = 0; x < xFrames; x++) {
                g.drawImage(
                        spriteSet.get(x + xFrames * y).getSubimage(
                            cropRect.x,
                            cropRect.y,
                            cropRect.width,
                            cropRect.height),
                        x * cropRect.width,
                        y * cropRect.height,
                        null);
            }
        }

        return target;
    }

    private static Rectangle minimumCropRect(Vector<BufferedImage> spriteSet)
    {
        Rectangle cropRect = null;

        for (BufferedImage sprite : spriteSet) {
            Rectangle frameCropRect = determineCropRect(sprite);

            if (cropRect == null) {
                cropRect = frameCropRect;
            } else {
                cropRect.add(frameCropRect);
            }
        }

        // Make crop rect one pixel larger (since we want an inclusive rect)
        if (cropRect != null) {
            cropRect.add(
                    cropRect.x + cropRect.width + 1,
                    cropRect.y + cropRect.height + 1);
        }

        return cropRect;
    }

    private static Rectangle determineCropRect(BufferedImage image)
    {
        // Loop through all the pixels, ignoring transparent ones.
        Rectangle rect = null;

        for (int y = 0; y < image.getHeight(); y++) {
            for (int x = 0; x < image.getWidth(); x++) {
                int color = image.getRGB(x, y);

                if (color != TRANSPARENT && (color & 0xFF000000) != 0) {
                    if (rect == null) {
                        rect = new Rectangle(x, y, 0, 0);
                    } else {
                        rect.add(x, y);
                    }
                }
            }
        }

        return rect;
    }

    private static void filterHairstyle(Vector<BufferedImage> spriteSet)
    {
        // Remove frame 1, 2, 6 and 7 from each color
        for (int i = HAIR_COLORS - 1; i >= 0; i--) {
            spriteSet.remove(i * HAIR_FRAMES + 7);
            spriteSet.remove(i * HAIR_FRAMES + 6);
            spriteSet.remove(i * HAIR_FRAMES + 2);
            spriteSet.remove(i * HAIR_FRAMES + 1);
        }
    }

    private static void filterHeadgear(Vector<BufferedImage> spriteSet)
    {
        // Remove frame 1, 2, 6 and 7
        spriteSet.remove(7);
        spriteSet.remove(6);
        spriteSet.remove(2);
        spriteSet.remove(1);
    }
}