Test of Time Sprite File Format
Many thanks to Apolyton's Zarion (DracoOmega) and Angelo Scotto. I would never have been able to decipher the sprite file format without them.
MercatorSTATIC.SPR The unit images are aligned to the bottom-left corner of the bounding box of the terrain tile. Since the unit health bar is centered at the top, making the images taller will move up the health bar. FILE HEADER bytes: value ---------------------------------------- HEADER 4 bytes: 0, due to absence of Animations section 4 bytes: 12, offset of Image Offsets section 4 bytes: Offset of Images section IMAGE OFFSETS N x 4 bytes: Image offsets relative to start of Images section. The first image has offset 0. They refer to the 5 facing directions (N, NE, E, SE, S) of all units. The last offset points to the end of the file (EOF). * The order is first all facing directions of the first unit, then the second unit etc. * The NW, W and SW facing images are autogenerated by mirroring the images for NE, E and SE, respectively. * In a standard ToT game there are 406 offset: 5 directions x 81 units + EOF = 406 * The Test of Time Patch Project (ToTPP) can increase the number of unit types to 127. The number of sprites used from this file depends on the @COSMIC2 key NumberOfStaticUnitSprites. See http://forums.civfanatics.com/showthread.php?t=517282 for more info about ToTPP. * What happens if there are fewer offsets than there are units? IMAGES bytes: value ---------------------------------------- IMAGE HEADER 16 bytes: 0; no apparent effect 4 bytes: Image width 4 bytes: Image height 4 bytes: Left coordinate of non-empty image bounding box 4 bytes: Top coordinate of non-empty image bounding box 4 bytes: Coordinate of first pixel right of non-empty image bounding box 4 bytes: Coordinate of first pixel below non-empty image bounding box 1 byte : 0xFD, this controls which color is treated as transparent. This only applies to the pixels actually present in the image encoding, not the implicit transparent pixels outside the image's bounding box. The first 5 bits must be 1. The last 3 bits are in the order BGR (blue, green, red) and reflect the RGB values of the transparent color. For example, if the last 3 bits are 111, the transparent color is white. If the last 3 bits are 001 the transparent color is red. The default is 101 which makes the transparent color magenta. 0xFD is 11111101 in binary. 4 bytes: remaining image size (body + footer) IMAGE BODY ([Number of rows] - [Number of leading empty rows] rows) 4 bytes: Number of empty bytes (2 bytes per pixel). This is the number of leading transparent pixels, ignoring the bounding box. 4 bytes: Row data size: N bytes, or N/2 pixels. N bytes: Row image data. This image data is in 15+1 bit format (in Big endian byte order that would be: 1-5-5-5). The first (i.e. highest-order) bit indicates whether the pixel's shade is changed to the civ color (1) or not (0). The other 15 bits are 5 bits each in the order R-G-B. Transparent pixels are represented by magenta (Little endian 0x1F7C, or Big endian 0111110000011111) as described in the header. * Blank lines in the middle of an image are encoded with zeroes for both row header values. * Some resources also use the civ color bit. It's not used. * Existing resources always have a full-size bounding box, so blank lines are encoded too. This is not necessary to work. IMAGE FOOTER 10 bytes: 0; no apparent effect UNITXX.SPR Unit animations are also aligned to the bottom-left corner of the terrain tile's bounding box. Unit numbering follows the order in the Rules.txt, with the first being 00. The last unit is the barbarian leader. 99 or 999 is the railroad train (see below). FILE HEADER bytes: value ---------------------------------------- HEADER 4 bytes: 12, offset of Animations section 4 bytes: Offset of Image Offsets section 4 bytes: Offset of Images section ANIMATIONS Animation Header: 32 x 4 bytes: Start offsets of the first frame of each animation: 8 facing directions x 4 actions = 32 animations The actions are, in order: Attack, Die, Idle, Move. The directions start at North, continuing clockwise to end at Northwest. The offsets indicate first all directions of Attack, then the animations for Die, then Idle and Move. * The train animation also has its own animation file. This is usually Unit99.spr. ToTPP changes this to Unit999.spr if the @COSMIC2 key NumberOfAnimatedUnitSprites is greater than 80. Since the train's Attack and Die animations are never used their offsets always point to an empty animation. * The offsets need not necessarily be in ascending order or even different. You could let the file contain only one animation, with all actions and directions using the same one. Animation Information: N x 4 bytes: For each animation frame... (a signed, long integer "bitmasked" into separate values) 1. Index into the list of image offsets, using 10 bits, so there can be at most 1024 images. 2. .... ..xx (Part of the image index from byte 1) ...x xx.. These 3 bits control transparency: 000 = fully visible 001 ~ 1/8 transparent 010 ~ 2/8 transparent 011 ~ 3/8 transparent 100 ~ 4/8 transparent 101 ~ 5/8 transparent 110 ~ 6/8 transparent 111 ~ 7/8 transparent (units only) ..1. .... Start of animation .1.. .... Loop animation (back to start) 1... .... Mirror frame (horizontally) 3. 0x06 unused? 0x00 on unit animation end frames and some empty resource animations. 4. .... ...1 End of animation (not displayed) .... ..1. Animate continuously (resources only) .000 00.. 0; unused 0... .... 0; sign bit; 1 hangs Civ2 * An animation can be effectively turned off by using a dummy animation with a single frame that is both start and end. RESOURCES: The first frame in an animation MUST have the Start flag. The last frame MUST have either the Loop or End flag. All frames in a looped animation, and only those frames, MUST have the Continuous animation flag. Neither the first, nor the last frame in a non-looping animation are displayed. Error Handling: If the first frame doesn't have the Start flag, Civ2 will first search up to the next End or Loop flag and then search back to find a Start flag. It will then play that animation. Animations that end with the End flag, but do have frames with the Continuous flag will continuously loop only those frames, and will play the entire animation only occasionally. In the continuous animation, the non-continuous frames do take up time, but don't appear, and the animation isn't refreshed during those gaps (unless the tile has focus) so the last continuous frame remains visible. UNITS: The last frame MUST have either the loop or end flag. Die animations SHOULD NOT loop, or they'll go on forever. The first frame doesn't need any flag... There MUST be at least one frame with the start flag, though, if it's a loop. In that case it will play the animation once from the first frame and then loop back to whichever frame has the start flag (usually also the first frame). IMAGE OFFSETS X x 4 bytes: Image offsets (from start of image section), the last one of these offsets points to the end of the file. IMAGES As in Static.spr RESOURCE.SPR Resource animations are aligned to the top-left corner of the terrain tile's bounding box. So making the images taller will cause the bottom to disappear from view. The format is the same as for the unit animations, with some minor differences: ANIMATIONS Animation Header: N x 4 bytes: Terrain resource animations in the same order as specified in the Rules.txt. In a standard ToT game this means there are: 4 maps x 11 terrain types x 2 resources = 88 animations * ToTPP can extend the number of types to 16 and uses as many animations as set in the @COSMIC2's NumberOfAnimatedResources. Animation Information: Some of the animation frame information is treated differently in the units and resources. See the UnitXX.spr description above for more details.
Last updated: 26 May 2020