-
Notifications
You must be signed in to change notification settings - Fork 76
/
animatrix.py
109 lines (83 loc) · 3.59 KB
/
animatrix.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
bl_info = {
"name": "Animatrix Export",
"author": "cam72cam",
"blender": (3,4,0),
"version": (1,0,4),
"location": "File > Import-Export",
"description": "Export Animatrix data",
"category": "Import-Export",
}
import bpy
from bpy_extras.io_utils import ExportHelper
from bpy.props import StringProperty, BoolProperty, EnumProperty
from bpy.types import Operator
import math
import mathutils
# Inspired by FriedrichLP's render lib exporter
class ExportAnimatrixData(Operator, ExportHelper):
"""Export Animatrix Data"""
bl_idname = "animatrix.exporter"
bl_label = "Export Animatrix"
# ExportHelper mixin class uses this
filename_ext = ".anim"
filter_glob: StringProperty(
default="*.anim",
options={'HIDDEN'},
maxlen=255, # Max internal buffer length, longer would be clamped.
)
skip_setting: BoolProperty(
name="Skip First Frame",
description="Skip the first animation frame which is typically used as a reference to the exported positions",
default=False,
)
use_setting: EnumProperty(
name="Export:",
description="Choose between Scene/Selected",
items=(
('SELECTED', "Selected Objects", "Only export selected objects"),
('SCENE', "Entire Scene", "Export all objects in the scene"),
),
default='SELECTED',
)
def execute(self, context):
orig_frame = bpy.context.scene.frame_current
with open(self.filepath, 'w', encoding='utf-8') as f:
objs = bpy.context.selected_objects if self.use_setting == 'SELECTED' else bpy.context.scene.objects
for obj in objs:
if obj.type != 'MESH':
continue
data = []
bpy.context.scene.frame_set(bpy.context.scene.frame_start)
def obj_matrix():
res = obj.matrix_world
for arm in [m.object for m in obj.modifiers if m.type == "ARMATURE"]:
if arm is None:
continue
return arm.matrix_world @ arm.pose.bones[obj.vertex_groups[0].name].matrix
return res
orig = obj_matrix().inverted()
for frame in range(bpy.context.scene.frame_start,bpy.context.scene.frame_end + 1):
bpy.context.scene.frame_set(frame)
offset = obj_matrix() @ orig
offset = offset @ (mathutils.Euler((math.radians(90), 0, 0)).to_matrix().to_4x4())
m = [offset[0], offset[2], [-z for z in offset[1]], offset[3]]
data.append("M " + ",".join(["%.16f" % y for x in m for y in x]) + '\n')
if len([line for line in data if line != data[0]]) != 0:
f.write("O " + obj.name + '\n')
f.write("A " + obj.name + "_" + obj.data.name + '\n')
if self.skip_setting:
del data[0]
for line in data:
f.write(line)
bpy.context.scene.frame_set(orig_frame)
return {'FINISHED'}
def menu_func_export(self, context):
self.layout.operator(ExportAnimatrixData.bl_idname, text="Animatrix (.anim)")
def register():
bpy.utils.register_class(ExportAnimatrixData)
bpy.types.TOPBAR_MT_file_export.prepend(menu_func_export)
def unregister():
bpy.utils.unregister_class(ExportAnimatrixData)
bpy.types.TOPBAR_MT_file_export.remove(menu_func_export)
if __name__ == "__main__":
register()