2016-01-27 16:33:28 +03:00
|
|
|
"""
|
|
|
|
A Pillow loader for .ftc and .ftu files (FTEX)
|
|
|
|
Jerome Leclanche <jerome@leclan.ch>
|
|
|
|
|
|
|
|
The contents of this file are hereby released in the public domain (CC0)
|
|
|
|
Full text of the CC0 license:
|
|
|
|
https://creativecommons.org/publicdomain/zero/1.0/
|
|
|
|
|
|
|
|
Independence War 2: Edge Of Chaos - Texture File Format - 16 October 2001
|
|
|
|
|
|
|
|
The textures used for 3D objects in Independence War 2: Edge Of Chaos are in a
|
2018-06-24 15:32:25 +03:00
|
|
|
packed custom format called FTEX. This file format uses file extensions FTC
|
|
|
|
and FTU.
|
2016-01-27 16:33:28 +03:00
|
|
|
* FTC files are compressed textures (using standard texture compression).
|
|
|
|
* FTU files are not compressed.
|
|
|
|
Texture File Format
|
2016-11-19 02:55:08 +03:00
|
|
|
The FTC and FTU texture files both use the same format. This
|
2016-01-27 16:33:28 +03:00
|
|
|
has the following structure:
|
|
|
|
{header}
|
|
|
|
{format_directory}
|
|
|
|
{data}
|
|
|
|
Where:
|
2018-10-21 10:26:08 +03:00
|
|
|
{header} = {
|
|
|
|
u32:magic,
|
|
|
|
u32:version,
|
|
|
|
u32:width,
|
|
|
|
u32:height,
|
|
|
|
u32:mipmap_count,
|
|
|
|
u32:format_count
|
|
|
|
}
|
2016-01-27 16:33:28 +03:00
|
|
|
|
|
|
|
* The "magic" number is "FTEX".
|
|
|
|
* "width" and "height" are the dimensions of the texture.
|
|
|
|
* "mipmap_count" is the number of mipmaps in the texture.
|
2018-06-24 15:32:25 +03:00
|
|
|
* "format_count" is the number of texture formats (different versions of the
|
|
|
|
same texture) in this file.
|
2016-01-27 16:33:28 +03:00
|
|
|
|
|
|
|
{format_directory} = format_count * { u32:format, u32:where }
|
|
|
|
|
2018-06-24 15:32:25 +03:00
|
|
|
The format value is 0 for DXT1 compressed textures and 1 for 24-bit RGB
|
|
|
|
uncompressed textures.
|
2016-01-27 16:33:28 +03:00
|
|
|
The texture data for a format starts at the position "where" in the file.
|
|
|
|
|
|
|
|
Each set of texture data in the file has the following structure:
|
|
|
|
{data} = format_count * { u32:mipmap_size, mipmap_size * { u8 } }
|
2018-06-24 15:32:25 +03:00
|
|
|
* "mipmap_size" is the number of bytes in that mip level. For compressed
|
|
|
|
textures this is the size of the texture data compressed with DXT1. For 24 bit
|
|
|
|
uncompressed textures, this is 3 * width * height. Following this are the image
|
|
|
|
bytes for that mipmap level.
|
2016-01-27 16:33:28 +03:00
|
|
|
|
|
|
|
Note: All data is stored in little-Endian (Intel) byte order.
|
|
|
|
"""
|
2024-02-05 20:18:49 +03:00
|
|
|
|
2023-12-21 14:13:31 +03:00
|
|
|
from __future__ import annotations
|
2016-01-27 16:33:28 +03:00
|
|
|
|
|
|
|
import struct
|
2022-01-15 01:02:31 +03:00
|
|
|
from enum import IntEnum
|
2016-01-27 16:33:28 +03:00
|
|
|
from io import BytesIO
|
|
|
|
|
2019-07-06 23:40:53 +03:00
|
|
|
from . import Image, ImageFile
|
2016-01-27 16:33:28 +03:00
|
|
|
|
|
|
|
MAGIC = b"FTEX"
|
2022-01-15 01:02:31 +03:00
|
|
|
|
|
|
|
|
|
|
|
class Format(IntEnum):
|
|
|
|
DXT1 = 0
|
|
|
|
UNCOMPRESSED = 1
|
|
|
|
|
|
|
|
|
2016-01-27 16:33:28 +03:00
|
|
|
class FtexImageFile(ImageFile.ImageFile):
|
|
|
|
format = "FTEX"
|
|
|
|
format_description = "Texture File Format (IW2:EOC)"
|
|
|
|
|
2024-05-11 03:48:09 +03:00
|
|
|
def _open(self) -> None:
|
2022-02-27 06:47:07 +03:00
|
|
|
if not _accept(self.fp.read(4)):
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = "not an FTEX file"
|
|
|
|
raise SyntaxError(msg)
|
2018-10-18 21:49:14 +03:00
|
|
|
struct.unpack("<i", self.fp.read(4)) # version
|
2018-09-30 05:58:02 +03:00
|
|
|
self._size = struct.unpack("<2i", self.fp.read(8))
|
2016-01-27 16:33:28 +03:00
|
|
|
mipmap_count, format_count = struct.unpack("<2i", self.fp.read(8))
|
|
|
|
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "RGB"
|
2016-02-05 01:57:13 +03:00
|
|
|
|
2018-06-24 15:32:25 +03:00
|
|
|
# Only support single-format files.
|
|
|
|
# I don't know of any multi-format file.
|
2016-01-27 16:33:28 +03:00
|
|
|
assert format_count == 1
|
|
|
|
|
|
|
|
format, where = struct.unpack("<2i", self.fp.read(8))
|
|
|
|
self.fp.seek(where)
|
2019-10-29 14:42:34 +03:00
|
|
|
(mipmap_size,) = struct.unpack("<i", self.fp.read(4))
|
2016-01-27 16:33:28 +03:00
|
|
|
|
|
|
|
data = self.fp.read(mipmap_size)
|
|
|
|
|
2022-01-15 01:02:31 +03:00
|
|
|
if format == Format.DXT1:
|
2023-07-29 02:28:18 +03:00
|
|
|
self._mode = "RGBA"
|
2022-04-10 19:25:40 +03:00
|
|
|
self.tile = [("bcn", (0, 0) + self.size, 0, 1)]
|
2022-01-15 01:02:31 +03:00
|
|
|
elif format == Format.UNCOMPRESSED:
|
2016-01-27 16:33:28 +03:00
|
|
|
self.tile = [("raw", (0, 0) + self.size, 0, ("RGB", 0, 1))]
|
|
|
|
else:
|
2022-12-22 00:51:35 +03:00
|
|
|
msg = f"Invalid texture compression format: {repr(format)}"
|
|
|
|
raise ValueError(msg)
|
2016-01-27 16:33:28 +03:00
|
|
|
|
|
|
|
self.fp.close()
|
|
|
|
self.fp = BytesIO(data)
|
|
|
|
|
2024-05-15 13:19:09 +03:00
|
|
|
def load_seek(self, pos: int) -> None:
|
2016-01-27 16:33:28 +03:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
2024-04-06 05:58:53 +03:00
|
|
|
def _accept(prefix: bytes) -> bool:
|
2016-01-27 16:33:28 +03:00
|
|
|
return prefix[:4] == MAGIC
|
|
|
|
|
|
|
|
|
2021-04-03 13:51:23 +03:00
|
|
|
Image.register_open(FtexImageFile.format, FtexImageFile, _accept)
|
2016-04-25 07:59:02 +03:00
|
|
|
Image.register_extensions(FtexImageFile.format, [".ftc", ".ftu"])
|