-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathexport_tmd.py
453 lines (393 loc) · 19.2 KB
/
export_tmd.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
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
import os
import time
import bpy
import mathutils
from struct import pack, unpack_from
from .utils.tristrip import stripify
from .common_tmd import errors, log_error, correction_local, correction_global, name_to_blender, name_to_tmd
def get_armature():
src_armatures = [ob for ob in bpy.data.objects if type(ob.data) == bpy.types.Armature]
#do we have armatures?
if src_armatures:
#see if one of these is selected
if len(src_armatures) > 1:
sel_armatures = [ob for ob in src_armatures if ob.select]
if sel_armatures:
return sel_armatures[0]
return src_armatures[0]
def flatten(mat):
return [v for row in mat for v in row]
def save(operator, context, filepath = '', export_anims = False, pad_anims = False):
MAX_BONES_PER_PIECE = 27
MAX_PIECES = 10
PIECE_LEN = 7500
print("\nStarting export to",filepath)
starttime = time.process_time()
out_dir = os.path.dirname(filepath)
armature = get_armature()
try:
tmd_in_path = armature["tmd_path"]
except KeyError:
log_error("Custom property 'tmd_path' is missing from the armature. Add it in the armature's object tab!")
return errors
#read the original TMD - do three things max:
#1) always get some of the magic numbers from the header
print("Reading data from original",tmd_in_path)
try:
with open(tmd_in_path, 'rb') as f:
header = f.read(130)
remaining_bytes, tkl_ref, magic_value1, magic_value2, lod_data_offset, salt, u1, u2 = unpack_from("I 8s 2L 4I", header, 8)
tkl_ref = tkl_ref.split(b"\x00")[0].decode("utf-8")
scene_block_bytes, num_nodes, u3, num_anims, u4 = unpack_from("I 4H", header, 60)
aux_node_data, node_data, anim_pointer = unpack_from("3I", header, 60+56)
#decrypt the addresses
aux_node_data += 60 - salt
node_data += 60 - salt
anim_pointer += 60 - salt
if aux_node_data == 124:
anim_pointer = node_data
node_data = aux_node_data
#2) set the bone names list into the original order
#3) copy the anim_bytes block from the imported file
if not export_anims:
f.seek(anim_pointer)
anim_bytes = f.read(lod_data_offset + 60 - anim_pointer)
f.seek(node_data)
node_bytes = f.read(176 * num_nodes)
pos = 0
#note that these are not necessarily sorted the same way as in blender's armature!
bone_names = []
for i in range(0, num_nodes):
name_len = unpack_from("B", node_bytes, pos+144)[0]
bone_name = name_to_blender(unpack_from(str(name_len)+"s", node_bytes, pos+145)[0])
bone_names.append(bone_name)
pos+=176
#do all bones match up between TMD and blender?
if set(bone_names) != set(armature.data.bones.keys()):
log_error("Bone mismatch between source TMD and blender armature. If you have imported another model since, re-import the desired source model, delete it and try again. Alternatively, you might want to export custom anims.")
return errors
else:
# create a new sorting from blender bones, updated bones before non-updated bones.
b_bones = armature.data.bones.keys()
bone_names = [b for b in b_bones if armature.data.bones[b].use_deform] + [b for b in b_bones if not armature.data.bones[b].use_deform]
except FileNotFoundError:
log_error("Original tmd file not found! Make sure the armature's custom property 'tmd_path' points to an existing tmd file!")
return errors
node_data = 124
anim_pointer = node_data + 176 * len(armature.data.bones)
bones_bytes = []
fallback_matrix = {}
#export all bones in the correct order
for bone_name in bone_names:
bone = armature.data.bones[bone_name]
#we get the original bind like this:
bind = correction_global.inverted() @ correction_local.inverted() @ bone.matrix_local @ correction_local
mat_local = bind
#only non-skeletal nodes can ignore updates
updates = 0 if bone.use_deform else 1
parent_id = -1
if bone.parent:
parent_id = bone_names.index(bone.parent.name)
p_bind_restored = correction_global.inverted() @ correction_local.inverted() @ bone.parent.matrix_local @ correction_local
mat_local = p_bind_restored.inverted() @ mat_local
fallback_matrix[bone_name] = mat_local
#mind the order of quat keys when packing!
q = mat_local.to_quaternion()
l = mat_local.to_translation()
#note that on import, the bind is transposed right after reading, so here we do it in the very end
bones_bytes.append( pack('4f 16f 16f B 15s hH 3f', q.x, q.y, q.z, q.w, *flatten(bind.transposed()), *flatten(bind.inverted().transposed()), len(bone.name), name_to_tmd(bone_name).encode("utf-8"), parent_id, updates, *l) )
bones_bytes = b"".join(bones_bytes)
if export_anims:
animations = [action for action in bpy.data.actions if not action.name.startswith("*")]
num_anims = len(animations)
anim_bytes = []
channels_bytes = []
all_quats = []
all_locs = []
#just get all vars
tkl_in_path = os.path.join(os.path.dirname(tmd_in_path), tkl_ref+".tkl")
with open(tkl_in_path, 'rb') as f:
tklstream = f.read(60)
tkl_b00, tkl_b01, tkl_b02, tkl_b03, tkl_remaining_bytes, tkl_name, tkl_b04, tkl_b05, tkl_b06, tkl_b07, tkl_b08, tkl_b09, tkl_b10, tkl_b11, tkl_b12, tkl_b13, num_loc, num_rot, tkl_i00, tkl_i01, tkl_i02, tkl_i03, tkl_i04 = unpack_from("4B I 6s 10B 2I 5I", tklstream, 4)
fps = bpy.context.scene.render.fps
#note this is not encrypted here
offset = anim_pointer + len(animations) * 4
for action in animations:
print(action.name)
#Animation Pointer Block
#offsets are encrypted
anim_bytes.append(pack('I', offset - 60 + salt))
#every bone, and only every bone, is written
offset += 32 + len(bone_names) * 4
channel_pointer_bytes = []
channel_bytes = []
#decode the action name
action_name = action.name[:-2]
ub1 = int(action.name[-2])
ub2 = int(action.name[-1])
channel_pointer_bytes.append(pack('B 15s 3I f', len(action_name), action_name.encode("utf-8"), ub1, ub2, len(bone_names), action.frame_range[1]/fps))
for bone_name in bone_names:
channel_pointer_bytes.append(pack('I', offset - 60 + salt))
if bone_name in action.groups:
group = action.groups[bone_name]
rotations = [fcurve for fcurve in group.channels if fcurve.data_path.endswith("quaternion")]
translations = [fcurve for fcurve in group.channels if fcurve.data_path.endswith("location")]
else:
rotations = []
translations = []
#prepare the fcurves for export -> make sure they are properly sampled, with no keys missing
fcurves = rotations+translations
times = []
same_amount_of_keys = all(len(fcu.keyframe_points) == len(fcurves[0].keyframe_points) for fcu in fcurves)
if not same_amount_of_keys:
print(bone_name+" has differing keyframe numbers for each fcurve")
#get all times
for fcu in fcurves:
for key in fcu.keyframe_points:
key_time = key.co[0]
if key_time not in times: times.append(key_time)
times.sort()
#sample and recreate all fcurves according to the full times
for fcu in fcurves:
samples = [fcu.evaluate(key_time) for key_time in times]
fcu_dp, fcu_i = fcu.data_path, fcu.array_index
action.fcurves.remove(fcu)
fcu = action.fcurves.new(fcu_dp, index=fcu_i, action_group=bone_name)
fcu.keyframe_points.add(count=len(times))
fcu.keyframe_points.foreach_set("co", [x for co in zip(times, samples) for x in co])
fcu.update()
#get the new curves because we deleted the original ones
rotations = [fcurve for fcurve in group.channels if fcurve.data_path.endswith("quaternion")]
translations = [fcurve for fcurve in group.channels if fcurve.data_path.endswith("location")]
#first, create define how to get the timestamp and key matrix for this bone
if (not rotations) and (not translations):
channel_mode = 2
num_keys = 0
elif not translations:
channel_mode = 0
num_keys = len(rotations[0].keyframe_points)
def get_key(rotations, translations, i):
key_matrix = mathutils.Quaternion([fcurve.keyframe_points[i].co[1] for fcurve in rotations]).to_matrix().to_4x4()
return rotations[0].keyframe_points[i].co[0]/fps, key_matrix
elif not rotations:
channel_mode = 3
num_keys = len(translations[0].keyframe_points)
def get_key(rotations, translations, i):
key_matrix = mathutils.Matrix()
key_matrix.translation = [fcurve.keyframe_points[i].co[1] for fcurve in translations]
return translations[0].keyframe_points[i].co[0]/fps, key_matrix
else:
channel_mode = 1
num_keys = len(rotations[0].keyframe_points)
def get_key(rotations, translations, i):
key_matrix = mathutils.Quaternion([fcurve.keyframe_points[i].co[1] for fcurve in rotations]).to_matrix().to_4x4()
key_matrix.translation = [fcurve.keyframe_points[i].co[1] for fcurve in translations]
return rotations[0].keyframe_points[i].co[0]/fps, key_matrix
#now, we assume that all curves are the same length and the keys are in columns
channel_bytes.append(pack('2H', channel_mode, num_keys ))
for i in range(0, num_keys):
try:
timestamp, key_matrix = get_key(rotations, translations, i)
except:
log_error("Bone "+bone_name+" in "+action_name+" has incomplete / faulty keyframes.")
#space conversion, simply inverse of import
key_matrix = correction_local.inverted() @ key_matrix @ correction_local
key_matrix = fallback_matrix[group.name] @ key_matrix
q = key_matrix.to_quaternion()
l = key_matrix.to_translation()
#even if use_channel says no, things still have to be written!
if l not in all_locs: all_locs.append(l)
if q not in all_quats: all_quats.append(q)
#super primitive indexing, it should be using a KD-Tree for closest neighbour search.
l_index = all_locs.index(l)
q_index = all_quats.index(q)
channel_bytes.append(pack('f2H', timestamp, l_index, q_index ))
# size of this channel: channelinfo (4b) + num_keys * key (8b)
offset += 4 + num_keys * 8
channels_bytes += channel_pointer_bytes + channel_bytes
anim_bytes += channels_bytes
anim_bytes = b"".join(anim_bytes)
print("Final Loc keys:",len(all_locs))
print("Final Rot keys:",len(all_quats))
#create an individualized dummy file which has to be merged, otherwise keep the old tkl_ref
if not pad_anims: tkl_ref = os.path.basename(filepath)[:-4][:6]
#create the TKL file
tkl_out_path = os.path.join(out_dir, tkl_ref+".tkl")
print("\nWriting",tkl_out_path)
#this is used for testing, so we just fill the TKL with dummy data to avoid crashes
if pad_anims:
all_locs.extend([all_locs[0] for x in range(num_loc-len(all_locs))])
all_quats.extend([all_quats[0] for x in range(num_rot-len(all_quats))])
tkl_locs = [pack("3f", *l) for l in all_locs]
tkl_quats = [pack("4f", q.x, q.y, q.z, q.w) for q in all_quats]
tkl_len_data = len(tkl_locs)*16 + len(tkl_quats)*12
tkl_header = pack("4s 4B I 6s 10B 2I 5I", b"TPKL", tkl_b00, tkl_b01, tkl_b02, tkl_b03, tkl_len_data+44, tkl_ref.encode("utf-8"), tkl_b04, tkl_b05, tkl_b06, tkl_b07, tkl_b08, tkl_b09, tkl_b10, tkl_b11, tkl_b12, tkl_b13, len(all_locs), len(all_quats), tkl_i00, tkl_i01, tkl_i02, tkl_i03, tkl_len_data)
try:
with open(tkl_out_path, 'wb') as f:
f.write(b"".join( (tkl_header, b"".join(tkl_locs), b"".join(tkl_quats) ) ))
except PermissionError:
log_error("You do not have writing permissions for "+out_dir+". Gain writing permissions there or export to another folder!")
return errors
#find all models
lod_bytes = []
lods = []
for i in range(0,10):
lod = [ob for ob in armature.children if "_LOD"+str(i) in ob.name]
if lod: lods.append(lod)
else: break
if not lods:
log_error("Could not find any LODs! Follow the naming convention of imported TMDs!")
return errors
max_lod_distance = 2 * max(max(ob.dimensions) for ob in armature.children)
lod_bytes.append(pack('I f', len(lods), max_lod_distance))
for lod in lods:
#todo: get these from the bounding box or center of gravity; note that x and y should be swizzled
s_x = 0.05 * max_lod_distance
s_y = -0.02 * max_lod_distance
s_z = 0.1 * max_lod_distance
d = 0.9 * max_lod_distance
lod_bytes.append(pack('I f 4f ',len(lod), 0, s_x, s_y, s_z, d))
for ob in lod:
print("\nProcessing mesh of",ob.name)
#remove unneeded modifiers
for mod in ob.modifiers:
if mod.type in ('ARMATURE','TRIANGULATE'):
ob.modifiers.remove(mod)
ob.modifiers.new('Triangulate', 'TRIANGULATE')
# make a copy with all modifiers applied
dg = bpy.context.evaluated_depsgraph_get()
eval_obj = ob.evaluated_get(dg)
me = eval_obj.to_mesh(preserve_all_data_layers=True, depsgraph=dg)
if not me.polygons:
log_error("Mesh "+ob.name+" contains no faces and will be ignored!")
continue
me.calc_normals_split()
#and restore the armature modifier
ob.modifiers.new('SkinDeform', 'ARMATURE').object = armature
#initialize the piece lists
bones_pieces = [ [] for i in range(0, MAX_PIECES) ]
tris_pieces = [ [] for i in range(0, MAX_PIECES) ]
print("Splitting into pieces of <= 28 bones")
#first step:
#go over all triangles, see which bones their verts use, see which tri goes into which piece
for polygon in me.polygons:
tri_bones = set()
for loop_index in polygon.loop_indices:
vertex = me.vertices[me.loops[loop_index].vertex_index]
#we can only look up the name here, and index it per piece
for vertex_group in vertex.groups:
bone_name = ob.vertex_groups[vertex_group.group].name
bone_weight = vertex_group.weight
#should this vertex group be used?
if bone_weight > 0 and bone_name in bone_names:
#add it to the set
tri_bones.add(bone_name)
#go over all pieces and see where we can add this tri
for i in range(0, MAX_PIECES):
#see how many bones this triangle would add to the piece
bones_to_add = sum([1 for bone_name in tri_bones if bone_name not in bones_pieces[i]])
#so can we add it to this piece? if not go to the next loop/piece
if len(bones_pieces[i]) + bones_to_add > MAX_BONES_PER_PIECE:
continue
#ok we can add it, so add it to the bones
#add only unique - don't use a set here because it must be sorted. then we can do the bone lookup in the same step
#all second order pieces will use the same bone list!
for bone_name in tri_bones:
if bone_name not in bones_pieces[i]:
bones_pieces[i].append(bone_name)
#also add the tri
tris_pieces[i].append(polygon.loop_indices)
#ok, done, skip the other pieces because the tri is already added
break
uv_layer = me.uv_layers[0].data
piece_data = []
mesh_vertices = []
dummy_vertices = []
#do the second splitting
for temp_piece_i in range(0, MAX_PIECES):
if bones_pieces[temp_piece_i]:
print("Processing temp piece", temp_piece_i)
#at this point we have the tris in the right pieces, so all verts that are used in piece 0 will exist for piece 1 (incase we want to reuse them)
tmd_piece_tris = []
for tri in tris_pieces[temp_piece_i]:
tmd_tri=[]
for loop_index in tri:
vertex = me.vertices[me.loops[loop_index].vertex_index]
co = vertex.co
no = me.loops[loop_index].normal
w = []
#we can only look up the name here, and index it per piece
for vertex_group in vertex.groups:
bone_name = ob.vertex_groups[vertex_group.group].name
bone_weight = vertex_group.weight
#should this vertex group be used?
if bone_weight > 0 and bone_name in bone_names:
w.append((int(bones_pieces[temp_piece_i].index(bone_name) * 3), bone_weight))
if not w:
log_error("Weight painting error, at least one vertex is not weighted!")
return errors
#only use the 4 biggest keys
w_s = sorted(w, key = lambda x:x[1], reverse = True)[0:4]
#for normalization
w_sum = sum([weight for id, weight in w_s])
#pad the weight list to 4 bones, ie. add empty bones if missing
for i in range(0, 4-len(w_s)): w_s.append((0,0))
#index the bone names, and build the list of bones used in this piece's strip
b = [id for id, weight in w_s]
w = [int(weight / w_sum * 255) for id, weight in w_s]
dummy = pack('3f 2f 4B', co.x, co.y, co.z, uv_layer[loop_index].uv.x, -uv_layer[loop_index].uv.y, *b)
#we could probably spread them out by pieces, but it doesn't seem to be required
if dummy not in dummy_vertices:
dummy_vertices.append(dummy)
#save the final vert
mesh_vertices.append( pack('3f 3f 4B 4B 2f', co.x, co.y, co.z, no.x, no.y, no.z, *w, *b, uv_layer[loop_index].uv.x, -uv_layer[loop_index].uv.y ) )
# get the corrected index for this tri
tmd_tri.append(dummy_vertices.index(dummy))
tmd_piece_tris.append(tmd_tri)
#there is just one input strip created from the triangles
in_strip = stripify(tmd_piece_tris, stitchstrips = True)[0]
#then we must split
for n in range(0, len(in_strip), PIECE_LEN):
piece_data.append((in_strip[n : PIECE_LEN+n+2], bones_pieces[temp_piece_i]))
num_pieces = len(piece_data)
num_all_strip_indices = sum([len(strip) for strip, piece_bone_names in piece_data])
num_all_verts = len(mesh_vertices)
print("\nWriting mesh",ob.name)
try:
material_name = me.materials[0].name
except:
material_name = "none"
log_error(ob.name+" has no material, set to 'none'!")
lod_bytes.append(pack("3I 32s ", num_pieces, num_all_strip_indices, num_all_verts, material_name.encode("utf-8")))
for piece_i in range(0, len(piece_data)):
print("Writing piece",piece_i)
strip, piece_bone_names = piece_data[piece_i]
#note that these are for the whole object and not the piece - might have to be adjusted
bbc_x, bbc_y, bbc_z = 0.125 * sum((mathutils.Vector(b) for b in ob.bound_box), mathutils.Vector())
bbe_x, bbe_y, bbe_z = ob.dimensions
#just dump all verts into the last piece
piece_verts = []
#if piece_i == len(piece_data)-1:
if piece_i == 0:
piece_verts = mesh_vertices
#write the mesh_piece header
lod_bytes.append(pack("4I 3f 3f", len(strip), len(piece_verts), len(piece_bone_names), max(strip), bbc_x, bbc_y, bbc_z, bbe_x, bbe_y, bbe_z))
#write the piece_bones
lod_bytes.append(pack(str(len(piece_bone_names))+"I", *[bone_names.index(bone_name) for bone_name in piece_bone_names]))
#write the verts
lod_bytes.append(b"".join(piece_verts))
#write the whole tristrip
lod_bytes.append(pack(str(len(strip))+"h", *strip))
lod_bytes = b"".join(lod_bytes)
try:
with open(filepath, 'wb') as f:
remaining_bytes = 112 + len(bones_bytes) + len(anim_bytes) + len(lod_bytes)
lod_offset = anim_pointer-60+len(anim_bytes)
header_bytes = pack('8s I 8s 2L 4I 4I', b"TMDL", remaining_bytes, tkl_ref.encode("utf-8"), magic_value1, magic_value2, lod_offset, salt, u1, u2, 0,0,0,0 )+ pack("I 4H 11I", lod_offset, len(bone_names), u3, num_anims, u4, 0,0,0,0,0,0,0,0,0,0,0)+ pack("2I", node_data-60+salt, anim_pointer-60+salt)
f.write(b"".join((header_bytes, bones_bytes, anim_bytes, lod_bytes)))
except PermissionError:
log_error("You do not have writing permissions for "+out_dir+". Gain writing permissions there or export to another folder!")
return errors
success = '\nFinished TMD Export in %.2f seconds\n' %(time.process_time()-starttime)
print(success)
return errors