forked from cataclysmbnteam/Cataclysm-BN
-
Notifications
You must be signed in to change notification settings - Fork 0
/
merge_maps.py
executable file
·346 lines (304 loc) · 12.5 KB
/
merge_maps.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
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
#!/usr/bin/env python3
import json
import argparse
import copy
import math
from operator import itemgetter
STRIDE_X = 3
STRIDE_Y = 2
MIN_X = 10000
MIN_Y = 10000
OM_SPEC_TYPES = ["overmap_special", "city_building"]
KEYED_TERMS = [
"terrain", "furniture", "fields", "npcs", "signs", "vendingmachines", "toilets", "gaspumps",
"items", "monsters", "vehicles", "item", "traps", "monster", "rubble", "liquids",
"sealed_item", "graffiti", "mapping"
]
PLACE_TERMS = ["set", "place_groups"]
MAP_ROTATE = [
{"dir": "_south", "x": 1, "y": 1},
{"dir": "_north", "x": 0, "y": 0}
]
def x_y_bucket(x, y):
return f"{math.floor((x - MIN_X) / STRIDE_X)}__{math.floor((y - MIN_Y) / STRIDE_Y)}"
def x_y_sub(x, y, is_north):
if is_north:
return f"{(x - MIN_X) % STRIDE_X}__{(y - MIN_Y) % STRIDE_Y}"
else:
return f"{(x - MIN_X - 1) % STRIDE_X}__{(y - MIN_Y - 1) % STRIDE_Y}"
def x_y_simple(x, y):
return f"{x}__{y}"
def get_data(argsDict, resource_name):
resource = []
resource_sources = argsDict.get(resource_name, [])
if not isinstance(resource_sources, list):
resource_sources = [resource_sources]
for resource_filename in resource_sources:
if resource_filename.endswith(".json"):
try:
with open(resource_filename) as resource_file:
resource += json.load(resource_file)
except FileNotFoundError:
exit(f"Failed: could not find {resource_filename}")
else:
print(f"Invalid filename {resource_filename}")
return resource
def adjacent_to_set(x, y, coord_set):
for coords in coord_set:
if y == coords["y"] and abs(x - coords["x"]) == 1:
return True
if x == coords["x"] and abs(y - coords["y"]) == 1:
return True
def validate_keyed(key_term, old_obj, entry):
new_keyset = entry["object"].get(term, {})
old_keyset = old_obj.get(key_term, {})
if new_keyset:
for old_key, old_val in old_keyset.items():
new_keyset.setdefault(old_key, old_val)
if new_keyset[old_key] != old_val:
return False
else:
new_keyset = old_keyset
return new_keyset
# make sure that all keyed entries have the same key values and the maps have the same weight
# and fill_ter. Don't try to resolve them.
def validate_old_map(old_map, entry):
old_obj = old_map.get("object", {})
if entry["weight"] and old_map.get("weight") and entry["weight"] != old_map.get("weight"):
return False
if entry["object"].get("fill_ter") and old_obj.get("fill_ter") and \
entry["object"]["fill_ter"] != old_obj.get("fill_ter"):
return False
new_palettes = entry.get("palettes", {})
old_palettes = old_obj.get("palettes")
if new_palettes:
for palette in old_palettes:
if palette not in new_palettes:
return False
else:
new_palettes = old_palettes
keysets = {}
for key_term in KEYED_TERMS:
new_keyset = validate_keyed(key_term, old_obj, entry)
if new_keyset:
keysets[key_term] = new_keyset
elif new_keyset != {}:
return False
if not entry["weight"]:
entry["weight"] = old_map.get("weight", 0)
if not entry["object"].get("fill_ter"):
entry["object"]["fill_ter"] = old_obj.get("fill_ter", "")
for key_term, new_keyset in keysets.items():
entry["object"][key_term] = new_keyset
if new_palettes:
entry["object"]["palettes"] = new_palettes
return True
# adjust the X, Y co-ords of a place_ entry to match the new map
def adjust_place(term, old_obj, offset_x, offset_y):
def adjust_coord(x_or_y, new_entry, old_entry, offset):
val = old_entry.get(x_or_y, "False")
if val == "False":
return False
if isinstance(val, list):
val[0] += offset
val[1] += offset
else:
val += offset
new_entry[x_or_y] = val
results = []
for old_entry in old_obj.get(term, []):
new_entry = copy.deepcopy(old_entry)
if offset_x:
adjust_coord("x", new_entry, old_entry, offset_x)
adjust_coord("x2", new_entry, old_entry, offset_x)
if offset_y:
adjust_coord("y", new_entry, old_entry, offset_y)
adjust_coord("y2", new_entry, old_entry, offset_y)
results.append(new_entry)
return results
args = argparse.ArgumentParser(description="Merge individual OMT maps into blocks of maps.")
args.add_argument("mapgen_sources", action="store", nargs="+",
help="specify jsons file to convert to blocks.")
args.add_argument("specials_sources", action="store", nargs="+",
help="specify json file with overmap special data.")
args.add_argument("--x", dest="stride_x", action="store",
help=f"number of horizontal maps in each block. Defaults to {STRIDE_X}.")
args.add_argument("--y", dest="stride_y", action="store",
help=f"number of vertictal maps in each block. Defaults to {STRIDE_Y}.")
args.add_argument("--output", dest="output_name", action="store",
help="Name of output file. Defaults to the command line.")
argsDict = vars(args.parse_args())
mapgen = get_data(argsDict, "mapgen_sources")
specials = get_data(argsDict, "specials_sources")
string_x = argsDict.get("stride_x")
if string_x and int(string_x):
STRIDE_X = int(string_x)
string_y = argsDict.get("stride_y")
if string_y and int(string_y):
STRIDE_Y = int(string_y)
output_name = argsDict.get("output_name", "")
if output_name and not output_name.endswith(".json"):
output_name += ".json"
# very first pass, sort the overmaps and find the minimum X and Y values
for special in specials:
if special.get("type") in OM_SPEC_TYPES:
overmaps = special.get("overmaps")
if not overmaps:
continue
overmaps.sort(key=lambda om_data: om_data.get("point", [1000, 0])[0])
MIN_X = overmaps[0].get("point", [1000, 0])[0]
overmaps.sort(key=lambda om_data: om_data.get("point", [0, 1000])[1])
MIN_Y = overmaps[0].get("point", [0, 1000])[1]
# create the merge sets of maps
merge_sets = {}
for special in specials:
if special.get("type") in OM_SPEC_TYPES:
overmaps = special.get("overmaps")
for om_data in overmaps:
om_map = om_data.get("overmap")
for map_dir in MAP_ROTATE:
if om_map.endswith(map_dir["dir"]):
om_map = om_map.split(map_dir["dir"])[0]
break
om_point = om_data.get("point", [])
if len(om_point) == 3:
is_north = map_dir["dir"] == "_north"
x = om_point[0]
y = om_point[1]
z = om_point[2]
merge_sets.setdefault(z, {})
merge_sets[z].setdefault(x_y_bucket(x, y), {})
merge_sets[z][x_y_bucket(x, y)][x_y_sub(x, y, is_north)] = om_map
# convert the mapgen list into a dictionary for easier access
map_dict = {}
new_mapgen = []
for om_map in mapgen:
if om_map.get("type") != "mapgen":
new_mapgen.append(om_map)
continue
om_id = om_map["om_terrain"]
if isinstance(om_id, list):
if len(om_id) == 1:
om_id = om_id[0]
else:
continue
map_dict[om_id] = om_map
# dynamically expand the list of "place_" terms
for term in KEYED_TERMS:
PLACE_TERMS.append(f"place_{term}")
basic_entry = {
"method": "json",
"object": {
"fill_ter": "",
"rows": [],
"palettes": [],
},
"om_terrain": [],
"type": "mapgen",
"weight": 0
}
# debug: make sure the merge sets look correct
#print("mergesets: {}".format(json.dumps(merge_sets, indent=2)))
# finally start merging maps
for z, zlevel in merge_sets.items():
for x_y, mapset in zlevel.items():
# first, split the mergeset into chunks with common KEYED_TERMS using a weird floodfill
chunks = []
chunks = [{ "maps": [], "entry": copy.deepcopy(basic_entry)}]
for y in range(0, STRIDE_Y):
for x in range(0, STRIDE_X):
om_id = mapset.get(x_y_simple(x, y), "")
if not om_id:
continue
old_map = map_dict.get(om_id, {})
if not old_map:
continue
validated = False
for chunk_data in chunks:
# try to filter out maps that are no longer adjacent to the rest of the
# merge set due to other maps not being valid
if chunk_data["maps"] and not adjacent_to_set(x, y, chunk_data["maps"]):
continue
# check that this map's keyed terms match the other keyed terms in this chunk
if validate_old_map(old_map, chunk_data["entry"]):
chunk_data["maps"].append({"x": x, "y": y})
validated = True
if not validated:
new_entry = copy.deepcopy(basic_entry)
chunks.append({ "maps": [{"x": x, "y": y}], "entry": new_entry })
# then split up any irregular shapes that made it past the screen
# T and L shapes are possible because every map is adjacent, for instance
final_chunks = []
for chunk_data in chunks:
maps = chunk_data["maps"]
if len(maps) < 3:
final_chunks.append(chunk_data)
continue
maps.sort(key=itemgetter("x"))
max_x = maps[-1]["x"]
min_x = maps[0]["x"]
maps.sort(key=itemgetter("y"))
max_y = maps[-1]["y"]
min_y = maps[0]["y"]
# if this is a line, square, or rectangle, it's continguous
if len(maps) == ((max_x - min_x + 1) * (max_y - min_y + 1)):
final_chunks.append(chunk_data)
continue
# if not, just break it into individual maps
for coords in maps:
final_chunks.append({
"maps": [{"x": coords["x"], "y": coords["y"]}],
"entry": copy.deepcopy(chunk_data["entry"])
})
if not final_chunks:
continue
# debug: do the final chunks look sane?
#print("chunks: {}".format(json.dumps(chunks, indent=2)))
# go through the final chunks and merge them
for chunk_data in final_chunks:
new_rows = []
entry = chunk_data["entry"]
maps = chunk_data["maps"]
if not maps:
continue
first_x = maps[0]["x"]
first_y = maps[0]["y"]
for coords in maps:
x = coords["x"]
y = coords["y"]
row_offset = 24 * (y - first_y)
col_offset = 24 * (x - first_x)
om_id = mapset.get(x_y_simple(x, y), "")
old_map = map_dict.get(om_id, {})
old_obj = old_map.get("object", {})
old_rows = old_obj.get("rows", {})
# merge the rows, obviously
for i in range(0, 24):
if not col_offset:
new_rows.append(old_rows[i])
else:
new_rows[i + row_offset] = f"{new_rows[i + row_offset]}{old_rows[i]}"
if len(maps) == 1:
entry["om_terrain"] = om_id
else:
if len(entry["om_terrain"]) < (y - first_y + 1):
entry["om_terrain"].append([om_id])
else:
entry["om_terrain"][y - first_y].append(om_id)
# adjust the offsets for place_ entries before potentially converting set entries
for place_term in PLACE_TERMS:
new_terms = adjust_place(place_term, old_obj, col_offset, row_offset)
if new_terms:
entry["object"].setdefault(place_term, [])
for term_entry in new_terms:
entry["object"][place_term].append(term_entry)
# finally done with the chunk, so add it to the list
entry["object"]["rows"] = new_rows
new_mapgen.append(entry)
# debug: make sure sure the final chunk is correct
#print("{}".format(json.dumps(entry, indent=2)))
if output_name:
with open(output_name, 'w') as output_file:
output_file.write(json.dumps(new_mapgen))
exit()
print(f"{json.dumps(new_mapgen, indent=2)}")