Source

love / src / modules / image / devil / ImageData.cpp

/**
* Copyright (c) 2006-2011 LOVE Development Team
*
* This software is provided 'as-is', without any express or implied
* warranty.  In no event will the authors be held liable for any damages
* arising from the use of this software.
*
* Permission is granted to anyone to use this software for any purpose,
* including commercial applications, and to alter it and redistribute it
* freely, subject to the following restrictions:
*
* 1. The origin of this software must not be misrepresented; you must not
*    claim that you wrote the original software. If you use this software
*    in a product, an acknowledgment in the product documentation would be
*    appreciated but is not required.
* 2. Altered source versions must be plainly marked as such, and must not be
*    misrepresented as being the original software.
* 3. This notice may not be removed or altered from any source distribution.
**/

#include "ImageData.h"

// STD
#include <iostream>

// LOVE
#include <common/Exception.h>

namespace love
{
namespace image
{
namespace devil
{
	void ImageData::load(Data * data)
	{
		// Generate DevIL image.
		ilGenImages(1, &image);

		// Bind the image.
		ilBindImage(image);

		// Try to load the image.
		ILboolean success = ilLoadL(IL_TYPE_UNKNOWN, (void*)data->getData(), data->getSize());

		// Check for errors
		if(!success)
		{
			throw love::Exception("Could not decode image!");
			return;
		}

		width = ilGetInteger(IL_IMAGE_WIDTH);
		height = ilGetInteger(IL_IMAGE_HEIGHT);
		origin = ilGetInteger(IL_IMAGE_ORIGIN);

		// Make sure the image is in RGBA format.
		ilConvertImage(IL_RGBA, IL_UNSIGNED_BYTE);

		// This should always be four.
		bpp = ilGetInteger(IL_IMAGE_BPP);

		if(bpp != 4)
		{
			std::cerr << "Bits per pixel != 4" << std::endl;
			return;
		}
	}

	ImageData::ImageData(Data * data)
	{
		load(data);
	}

	ImageData::ImageData(filesystem::File * file)
	{
		Data * data = file->read();
		load(data);
		data->release();
	}

	ImageData::ImageData(int width, int height)
		: width(width), height(height), origin(IL_ORIGIN_UPPER_LEFT), bpp(4)
	{
		// Generate DevIL image.
		ilGenImages(1, &image);

		// Bind the image.
		ilBindImage(image);

		bool success = (ilTexImage(width, height, 1, bpp, IL_RGBA, IL_UNSIGNED_BYTE, 0) == IL_TRUE);
		int err = ilGetError();

		if(!success) {
			if (err != IL_NO_ERROR){
				switch (err) {
					case IL_ILLEGAL_OPERATION:
						throw love::Exception("Error: Illegal operation");
						break;
					case IL_INVALID_PARAM:
						throw love::Exception("Error: invalid parameters");
						break;
					case IL_OUT_OF_MEMORY:
						throw love::Exception("Error: out of memory");
						break;
					default:
						throw love::Exception("Error: unknown error");
						break;
				}
			}
			throw love::Exception("Could not decode image data.");
		}

		// Set to black.
		memset((void*)ilGetData(), 0, width*height*4);
	}

	ImageData::ImageData(int width, int height, void *data)
	: width(width), height(height), origin(IL_ORIGIN_UPPER_LEFT), bpp(4)
	{
		// Generate DevIL image.
		ilGenImages(1, &image);
		// Bind the image.
		ilBindImage(image);
		// Try to load the data.
		bool success = (ilTexImage(width, height, 1, bpp, IL_RGBA, IL_UNSIGNED_BYTE, data) == IL_TRUE);

		if(!success) {
			if (err != IL_NO_ERROR){
				switch (err) {
					case IL_ILLEGAL_OPERATION:
						throw love::Exception("Error: Illegal operation");
						break;
					case IL_INVALID_PARAM:
						throw love::Exception("Error: invalid parameters");
						break;
					case IL_OUT_OF_MEMORY:
						throw love::Exception("Error: out of memory");
						break;
					default:
						throw love::Exception("Error: unknown error");
						break;
				}
			}
			throw love::Exception("Could not decode image data.");
		}
	}

	ImageData::~ImageData()
	{
		ilDeleteImages(1, &image);
	}

	int ImageData::getWidth() const
	{
		return width;
	}

	int ImageData::getHeight() const
	{
		return height;
	}

	void * ImageData::getData() const
	{
		ilBindImage(image);
		return ilGetData();
	}

	int ImageData::getSize() const
	{
		return width*height*bpp;
	}

	void ImageData::setPixel(int x, int y, pixel c)
	{
		//int tx = x > width-1 ? width-1 : x;
		//int ty = y > height-1 ? height-1 : y; // not using these seems to not break anything
		if (x > width-1 || y > height-1 || x < 0 || y < 0) throw love::Exception("Attempt to set out-of-range pixel!");
		pixel * pixels = (pixel *)getData();
		pixels[y*width+x] = c;
	}

	pixel ImageData::getPixel(int x, int y) const
	{
		//int tx = x > width-1 ? width-1 : x;
		//int ty = y > height-1 ? height-1 : y; // not using these seems to not break anything
		if (x > width-1 || y > height-1 || x < 0 || y < 0) throw love::Exception("Attempt to get out-of-range pixel!");
		pixel * pixels = (pixel *)getData();
		return pixels[y*width+x];
	}

	EncodedImageData * ImageData::encode(EncodedImageData::Format f) {
		ilBindImage(image);
		ILubyte * data;
		ILuint w = getWidth();
		int h = getHeight(); // has to be a signed int so we can make it negative for BMPs
		int headerLen, bpp, row, size, padding, filesize;
		switch (f) {
			case EncodedImageData::FORMAT_BMP:
				headerLen = 54;
				bpp = 3;
				row = w * bpp;
				padding = row & 3;
				size = h * (row + padding);
				filesize = size + headerLen;
				data = new ILubyte[filesize];
				// Here's the header for the BMP file format.
				data[0] = 66; // "B"
				data[1] = 77; // "M"
				data[2] = filesize & 255; // size of the file
				data[3] = (filesize >> 8) & 255;
				data[4] = (filesize >> 16) & 255;
				data[5] = (filesize >> 24) & 255;
				data[6] = data[7] = data[8] = data[9] = 0; // useless reserved values
				data[10] = headerLen; // offset where pixel data begins
				data[11] = (headerLen >> 8) & 255;
				data[12] = (headerLen >> 16) & 255;
				data[13] = (headerLen >> 24) & 255;
				data[14] = headerLen - 14; // length of this part of the header
				data[15] = ((headerLen - 14) >> 8) & 255;
				data[16] = ((headerLen - 14) >> 16) & 255;
				data[17] = ((headerLen - 14) >> 24) & 255;
				data[18] = w & 255; // width of the bitmap
				data[19] = (w >> 8) & 255;
				data[20] = (w >> 16) & 255;
				data[21] = (w >> 24) & 255;
				data[22] = -h & 255; // negative height of the bitmap - used so we don't have to flip the data
				data[23] = ((-h) >> 8) & 255;
				data[24] = ((-h) >> 16) & 255;
				data[25] = ((-h) >> 24) & 255;
				data[26] = 1; // number of color planes
				data[27] = 0;
				data[28] = bpp * 8; // bits per pixel
				data[29] = 0;
				data[30] = data[31] = data[32] = data[33] = 0; // RGB - no compression
				data[34] = (row + padding) * h; // length of the pixel data
				data[35] = (((row + padding) * h) >> 8) & 255;
				data[36] = (((row + padding) * h) >> 16) & 255;
				data[37] = (((row + padding) * h) >> 24) & 255;
				data[38] = 2835 & 255; // horizontal pixels per meter
				data[39] = (2835 >> 8) & 255;
				data[40] = (2835 >> 16) & 255;
				data[41] = (2835 >> 24) & 255;
				data[42] = 2835 & 255; // vertical pixels per meter
				data[43] = data[39];
				data[44] = data[40];
				data[45] = data[41];
				data[46] = data[47] = data[48] = data[49] = 0; // number of colors in the palette
				data[50] = data[51] = data[52] = data[53] = 0; // all colors are important!
				// Okay, header's done! Now for the pixel data...
				data += headerLen;
				for (int i = 0; i < h; i++) { // we've got to loop through the rows, adding the pixel data plus padding
					ilCopyPixels(0,i,0,w,1,1,IL_BGR,IL_UNSIGNED_BYTE,data);
					data += row;
				}
				data -= filesize;
				break;
			case EncodedImageData::FORMAT_TGA:
			default: // TGA is the default format
				headerLen = 18;
				bpp = 4;
				size = h * w * bpp;
				data = new ILubyte[size + headerLen];
				// here's the header for the Targa file format.
				data[0] = 0; // ID field size
				data[1] = 0; // colormap type
				data[2] = 2; // image type
				data[3] = data[4] = 0; // colormap start
				data[5] = data[6] = 0; // colormap length
				data[7] = 32; // colormap bits
				data[8] = data[9] = 0; // x origin
				data[10] = data[11] = 0; // y origin
				// Targa is little endian, so:
				data[12] = w & 255; // least significant byte of width
				data[13] = w >> 8; // most significant byte of width
				data[14] = h & 255; // least significant byte of height
				data[15] = h >> 8; // most significant byte of height
				data[16] = bpp * 8; // bits per pixel
				data[17] = 0x20; // descriptor bits (flip bits: 0x10 horizontal, 0x20 vertical)
				// header done. write the pixel data to TGA:
				data += headerLen;
				ilCopyPixels(0,0,0,w,h,1,IL_BGRA,IL_UNSIGNED_BYTE,data); // convert the pixels to BGRA (remember, little-endian) and copy them to data

				data -= headerLen;
		}
		return new EncodedImageData(data, f, size + headerLen, freeData);
	}

	void ImageData::freeData(void *data)
	{
		delete[] (ILubyte*) data;
	}

} // devil
} // image
} // love
Tip: Filter by directory path e.g. /media app.js to search for public/media/app.js.
Tip: Use camelCasing e.g. ProjME to search for ProjectModifiedEvent.java.
Tip: Filter by extension type e.g. /repo .js to search for all .js files in the /repo directory.
Tip: Separate your search with spaces e.g. /ssh pom.xml to search for src/ssh/pom.xml.
Tip: Use ↑ and ↓ arrow keys to navigate and return to view the file.
Tip: You can also navigate files with Ctrl+j (next) and Ctrl+k (previous) and view the file with Ctrl+o.
Tip: You can also navigate files with Alt+j (next) and Alt+k (previous) and view the file with Alt+o.