src/spriteloader/grf.cpp
author truebrain
Mon, 23 Jun 2008 13:52:21 +0000
branchnoai
changeset 11058 3305a425f55b
parent 10776 07203fc29812
permissions -rw-r--r--
(svn r13615) [NoAI] -Fix: fixed most, if not all, problems around AIAbstractList. It is now safe to remove values while looping, among other things.
[NoAI] -Add: allow foreach() usage for AIAbstractList
[NoAI] -Add: allow [] usage for AIAbstractList (read-only)
[NoAI] -Add: allow [] usage for AIList (read/write)
9627
6a7c8ead2328 (svn r10120) [NoAI] -Sync with trunk r10096:r10119
glx
parents: 9626
diff changeset
     1
/* $Id$ */
6a7c8ead2328 (svn r10120) [NoAI] -Sync with trunk r10096:r10119
glx
parents: 9626
diff changeset
     2
10455
22c441f5adf9 (svn r12997) [NoAI] -Sync: with trunk r12895:12996.
rubidium
parents: 9724
diff changeset
     3
/** @file grf.cpp Reading graphics data from (New)GRF files. */
9627
6a7c8ead2328 (svn r10120) [NoAI] -Sync with trunk r10096:r10119
glx
parents: 9626
diff changeset
     4
9626
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
     5
#include "../stdafx.h"
9723
eee46cb39750 (svn r11796) [NoAI] -Sync: with trunk r11502:11795.
rubidium
parents: 9703
diff changeset
     6
#include "../gfx_func.h"
9626
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
     7
#include "../fileio.h"
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
     8
#include "../debug.h"
9723
eee46cb39750 (svn r11796) [NoAI] -Sync: with trunk r11502:11795.
rubidium
parents: 9703
diff changeset
     9
#include "../core/alloc_func.hpp"
9626
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    10
#include "grf.hpp"
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    11
10776
07203fc29812 (svn r13326) [NoAI] -Sync with trunk r13264:13325
glx
parents: 10455
diff changeset
    12
bool SpriteLoaderGrf::LoadSprite(SpriteLoader::Sprite *sprite, uint8 file_slot, size_t file_pos)
9626
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    13
{
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    14
	/* Open the right file and go to the correct position */
9703
d2a6acdbd665 (svn r11146) [NoAI] -Sync: with trunk r11035:11045.
rubidium
parents: 9628
diff changeset
    15
	FioSeekToFile(file_slot, file_pos);
9626
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    16
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    17
	/* Read the size and type */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    18
	int num = FioReadWord();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    19
	byte type = FioReadByte();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    20
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    21
	/* Type 0xFF indicates either a colormap or some other non-sprite info; we do not handle them here */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    22
	if (type == 0xFF) return false;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    23
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    24
	sprite->height = FioReadByte();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    25
	sprite->width  = FioReadWord();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    26
	sprite->x_offs = FioReadWord();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    27
	sprite->y_offs = FioReadWord();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    28
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    29
	/* 0x02 indicates it is a compressed sprite, so we can't rely on 'num' to be valid.
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    30
	 *  In case it is uncompressed, the size is 'num' - 8 (header-size). */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    31
	num = (type & 0x02) ? sprite->width * sprite->height : num - 8;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    32
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    33
	/* XXX -- We should use a pre-located memory segment for this, malloc/free is pretty expensive */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    34
	byte *dest_orig = MallocT<byte>(num);
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    35
	byte *dest = dest_orig;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    36
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    37
	/* Read the file, which has some kind of compression */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    38
	while (num > 0) {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    39
		int8 code = FioReadByte();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    40
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    41
		if (code >= 0) {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    42
			/* Plain bytes to read */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    43
			int size = (code == 0) ? 0x80 : code;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    44
			num -= size;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    45
			for (; size > 0; size--) {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    46
				*dest = FioReadByte();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    47
				dest++;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    48
			}
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    49
		} else {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    50
			/* Copy bytes from earlier in the sprite */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    51
			const uint data_offset = ((code & 7) << 8) | FioReadByte();
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    52
			int size = -(code >> 3);
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    53
			num -= size;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    54
			for (; size > 0; size--) {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    55
				*dest = *(dest - data_offset);
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    56
				dest++;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    57
			}
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    58
		}
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    59
	}
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    60
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    61
	assert(num == 0);
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    62
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    63
	sprite->data = CallocT<SpriteLoader::CommonPixel>(sprite->width * sprite->height);
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    64
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    65
	/* When there are transparency pixels, this format has an other trick.. decode it */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    66
	if (type & 0x08) {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    67
		for (int y = 0; y < sprite->height; y++) {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    68
			bool last_item = false;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    69
			/* Look up in the header-table where the real data is stored for this row */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    70
			int offset = (dest_orig[y * 2 + 1] << 8) | dest_orig[y * 2];
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    71
			/* Go to that row */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    72
			dest = &dest_orig[offset];
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    73
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    74
			do {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    75
				SpriteLoader::CommonPixel *data;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    76
				/* Read the header:
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    77
				 *  0 .. 14  - length
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    78
				 *  15       - last_item
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    79
				 *  16 .. 31 - transparency bytes */
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    80
				last_item  = ((*dest) & 0x80) != 0;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    81
				int length =  (*dest++) & 0x7F;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    82
				int skip   =   *dest++;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    83
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    84
				data = &sprite->data[y * sprite->width + skip];
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    85
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    86
				for (int x = 0; x < length; x++) {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    87
					data->m = *dest;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    88
					dest++;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    89
					data++;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    90
				}
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    91
			} while (!last_item);
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    92
		}
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    93
	} else {
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    94
		dest = dest_orig;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    95
		for (int i = 0; i < sprite->width * sprite->height; i++)
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    96
			sprite->data[i].m = dest[i];
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    97
	}
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
    98
9628
b5c2449616b5 (svn r10195) [NoAI] -Sync: with trunk r10119:10194.
rubidium
parents: 9627
diff changeset
    99
	/* Make sure to mark all transparent pixels transparent on the alpha channel too */
b5c2449616b5 (svn r10195) [NoAI] -Sync: with trunk r10119:10194.
rubidium
parents: 9627
diff changeset
   100
	for (int i = 0; i < sprite->width * sprite->height; i++)
b5c2449616b5 (svn r10195) [NoAI] -Sync: with trunk r10119:10194.
rubidium
parents: 9627
diff changeset
   101
		if (sprite->data[i].m != 0) sprite->data[i].a = 0xFF;
b5c2449616b5 (svn r10195) [NoAI] -Sync: with trunk r10119:10194.
rubidium
parents: 9627
diff changeset
   102
9626
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
   103
	free(dest_orig);
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
   104
	return true;
79f2b5a0cdd7 (svn r10118) [NoAI] -Sync with trunk r10015:r10096
glx
parents:
diff changeset
   105
}