author | truebrain |
Mon, 23 Jun 2008 13:52:21 +0000 | |
branch | noai |
changeset 11058 | 3305a425f55b |
parent 10776 | 07203fc29812 |
permissions | -rw-r--r-- |
9627 | 1 |
/* $Id$ */ |
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 | 4 |
|
9626 | 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 | 7 |
#include "../fileio.h" |
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 | 10 |
#include "grf.hpp" |
11 |
||
10776 | 12 |
bool SpriteLoaderGrf::LoadSprite(SpriteLoader::Sprite *sprite, uint8 file_slot, size_t file_pos) |
9626 | 13 |
{ |
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 | 16 |
|
17 |
/* Read the size and type */ |
|
18 |
int num = FioReadWord(); |
|
19 |
byte type = FioReadByte(); |
|
20 |
||
21 |
/* Type 0xFF indicates either a colormap or some other non-sprite info; we do not handle them here */ |
|
22 |
if (type == 0xFF) return false; |
|
23 |
||
24 |
sprite->height = FioReadByte(); |
|
25 |
sprite->width = FioReadWord(); |
|
26 |
sprite->x_offs = FioReadWord(); |
|
27 |
sprite->y_offs = FioReadWord(); |
|
28 |
||
29 |
/* 0x02 indicates it is a compressed sprite, so we can't rely on 'num' to be valid. |
|
30 |
* In case it is uncompressed, the size is 'num' - 8 (header-size). */ |
|
31 |
num = (type & 0x02) ? sprite->width * sprite->height : num - 8; |
|
32 |
||
33 |
/* XXX -- We should use a pre-located memory segment for this, malloc/free is pretty expensive */ |
|
34 |
byte *dest_orig = MallocT<byte>(num); |
|
35 |
byte *dest = dest_orig; |
|
36 |
||
37 |
/* Read the file, which has some kind of compression */ |
|
38 |
while (num > 0) { |
|
39 |
int8 code = FioReadByte(); |
|
40 |
||
41 |
if (code >= 0) { |
|
42 |
/* Plain bytes to read */ |
|
43 |
int size = (code == 0) ? 0x80 : code; |
|
44 |
num -= size; |
|
45 |
for (; size > 0; size--) { |
|
46 |
*dest = FioReadByte(); |
|
47 |
dest++; |
|
48 |
} |
|
49 |
} else { |
|
50 |
/* Copy bytes from earlier in the sprite */ |
|
51 |
const uint data_offset = ((code & 7) << 8) | FioReadByte(); |
|
52 |
int size = -(code >> 3); |
|
53 |
num -= size; |
|
54 |
for (; size > 0; size--) { |
|
55 |
*dest = *(dest - data_offset); |
|
56 |
dest++; |
|
57 |
} |
|
58 |
} |
|
59 |
} |
|
60 |
||
61 |
assert(num == 0); |
|
62 |
||
63 |
sprite->data = CallocT<SpriteLoader::CommonPixel>(sprite->width * sprite->height); |
|
64 |
||
65 |
/* When there are transparency pixels, this format has an other trick.. decode it */ |
|
66 |
if (type & 0x08) { |
|
67 |
for (int y = 0; y < sprite->height; y++) { |
|
68 |
bool last_item = false; |
|
69 |
/* Look up in the header-table where the real data is stored for this row */ |
|
70 |
int offset = (dest_orig[y * 2 + 1] << 8) | dest_orig[y * 2]; |
|
71 |
/* Go to that row */ |
|
72 |
dest = &dest_orig[offset]; |
|
73 |
||
74 |
do { |
|
75 |
SpriteLoader::CommonPixel *data; |
|
76 |
/* Read the header: |
|
77 |
* 0 .. 14 - length |
|
78 |
* 15 - last_item |
|
79 |
* 16 .. 31 - transparency bytes */ |
|
80 |
last_item = ((*dest) & 0x80) != 0; |
|
81 |
int length = (*dest++) & 0x7F; |
|
82 |
int skip = *dest++; |
|
83 |
||
84 |
data = &sprite->data[y * sprite->width + skip]; |
|
85 |
||
86 |
for (int x = 0; x < length; x++) { |
|
87 |
data->m = *dest; |
|
88 |
dest++; |
|
89 |
data++; |
|
90 |
} |
|
91 |
} while (!last_item); |
|
92 |
} |
|
93 |
} else { |
|
94 |
dest = dest_orig; |
|
95 |
for (int i = 0; i < sprite->width * sprite->height; i++) |
|
96 |
sprite->data[i].m = dest[i]; |
|
97 |
} |
|
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 | 103 |
free(dest_orig); |
104 |
return true; |
|
105 |
} |