forked from Deokishisu/CrystalDust
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtileset_ripper.py
117 lines (102 loc) · 4.19 KB
/
tileset_ripper.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
#!/usr/bin/python3
from sys import argv
import os, subprocess, struct
NUM_TILES_IN_PRIMARY = 640
NUM_TILES_TOTAL = 1024
NUM_METATILES_IN_PRIMARY = 640
NUM_METATILES_TOTAL = 1024
def gbagfx(params):
subprocess.call([os.sep.join([os.curdir, 'tools', 'gbagfx', 'gbagfx'])] + params)
def create_image(path, data, is_compressed):
filename = os.sep.join([path, "tiles"])
with open(filename + (".4bpp" if not is_compressed else ".4bpp.lz"), 'wb+') as f:
f.write(data)
if is_compressed:
gbagfx([filename + ".4bpp.lz", filename + ".4bpp"])
gbagfx([filename + ".4bpp", filename + ".png", "-width", "16", "-palette", os.sep.join([path, "palettes/08.gbapal"])])
def create_pals(path, data):
subdir = os.sep.join([path, 'palettes'])
os.makedirs(subdir, exist_ok=True)
for i in range(0, 16):
filename = os.sep.join([subdir, format(i, '02')])
with open(filename + '.gbapal', 'wb+') as f:
f.write(data[i * 32:i * 32 + 32])
gbagfx([filename + '.gbapal', filename + '.pal'])
def create_metatiles(path, data, num_metatiles):
with open(os.sep.join([path, 'metatiles.bin']), 'wb+') as f:
f.write(data)
def create_behavior(path, data, num_metatiles):
new_data = bytearray(num_metatiles * 2)
for i in range(0, num_metatiles):
behavior = data[i * 4]
if behavior == 0x2A: # stairs
behavior = 0
elif behavior >= 0x6C and behavior <= 0x6F: # staircase
behavior -= 0x40
elif behavior == 0x81: # bookshelf
behavior = 0xE2
elif behavior == 0x82: # pokemart shelf
behavior = 0
elif behavior == 0x84: # signpost
behavior = 0x7E
elif behavior == 0x87: # pokemon center
behavior = 0x81
elif behavior == 0x88: # pokemon mart
behavior = 0x82
elif behavior >= 0x89 and behavior <= 0x8B: # dishes, sink, drawers
behavior = 0
elif behavior == 0x90: # fridge
behavior = 0
elif behavior == 0x93: # chart
behavior = 0
elif behavior == 0x94: # picture
behavior = 0
elif behavior == 0x95: # machine
behavior = 0
elif behavior == 0x96: # phone
behavior = 0
elif behavior == 0x97: # computer
behavior = 0
elif behavior == 0x9B: # coffee mug
behavior = 0
elif behavior >= 0x9C and behavior <= 0x9D: # window
behavior = 0x88
elif behavior == 0x9E: # machine
behavior = 0
elif behavior == 0xA3: # card thing
behavior = 0
new_data[i * 2] = behavior
bg = data[i * 4 + 3]
if bg & 0x20 == 0x20:
bg = 0x10
new_data[i * 2 + 1] = bg
with open(os.sep.join([path, 'metatile_attributes.bin']), 'wb+') as f:
f.write(new_data)
if __name__ == '__main__':
with open(argv[1], 'rb') as f:
f.seek(int(argv[2], base=16))
tileset_header = struct.unpack('<??xxIIIII', f.read(24))
is_compressed = tileset_header[0]
is_secondary = tileset_header[1]
num_tiles = NUM_TILES_IN_PRIMARY if not is_secondary else NUM_TILES_TOTAL - NUM_TILES_IN_PRIMARY
num_metatiles = NUM_METATILES_IN_PRIMARY if not is_secondary else NUM_METATILES_TOTAL - NUM_METATILES_IN_PRIMARY
if len(argv) > 4:
num_metatiles = int(argv[4])
image_ptr = tileset_header[2] - 0x8000000
palette_ptr = tileset_header[3] - 0x8000000
metatiles_ptr = tileset_header[4] - 0x8000000
behavior_ptr = tileset_header[6] - 0x8000000
f.seek(image_ptr)
image = f.read(num_tiles * 32)
f.seek(palette_ptr)
palettes = f.read(512)
f.seek(metatiles_ptr)
metatiles = f.read(num_metatiles * 16)
f.seek(behavior_ptr)
behavior = f.read(num_metatiles * 4)
path = os.sep.join([os.curdir, 'data', 'tilesets', 'primary' if not is_secondary else 'secondary', argv[3]])
os.makedirs(path, exist_ok=True)
create_pals(path, palettes)
create_image(path, image, is_compressed)
create_metatiles(path, metatiles, num_metatiles)
create_behavior(path, behavior, num_metatiles)