forked from simboden/BVtkNodes
-
Notifications
You must be signed in to change notification settings - Fork 20
/
animation_helper.py
262 lines (227 loc) · 9.84 KB
/
animation_helper.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
"""
This file is responsible for helping in animating properties until
https://developer.blender.org/T66392
is resolved.
"""
from .core import *
from .core import l
from .core import assert_bvtk
import bpy
import numpy as np
invalid_frame = int(-1e6)
def iterate_f_curves(f_curves):
for node_group in bpy.data.node_groups.values():
node_tree_name = node_group.name
for key, val in bpy.data.actions.items():
# Skip unrelated node trees
if node_tree_name + "Action" in key:
for f_curve in f_curves:
# Extract the complete data path from the f-curve
prop_path = f_curve.data_path
# Split the data path into parent and component (i.e. vtkTransformFilter.m_Rotation -> vtkTransformFilter, m_Rotation)
# Actual names are e.g. 'nodes["Time Selector"].time_step'
delimiter_index = prop_path.rindex(".")
node_name = prop_path[7 : delimiter_index - 2]
attribute_name = prop_path[delimiter_index + 1 :]
# Properties can be arrays (e.g. rotation is a 3D vector)
# The associated index is queried here
arr_ind = f_curve.array_index
interpolation_modes = [
kf.interpolation for kf in f_curve.keyframe_points
]
keyframes = [int(kf.co[0]) for kf in f_curve.keyframe_points]
keyframe_values = [kf.co[1] for kf in f_curve.keyframe_points]
first_interpolation_mode = interpolation_modes[0]
single_mode = all(
x == first_interpolation_mode for x in interpolation_modes
)
if not single_mode:
l.warning(
"Global Time Keeper: Mixed interpolation type encountered. This should not happen"
)
interpolation_mode = (
first_interpolation_mode if single_mode else "Mixed"
)
# Load current value of the property
try:
try:
current_val = eval(
f_curve.data_path + "[{}]".format(arr_ind),
{"nodes": node_group.nodes},
)
except TypeError as err:
if arr_ind == 0: # May be a scalar
current_val = eval(
f_curve.data_path, {"nodes": node_group.nodes}
)
else:
l.error(
"Could not load property "
+ prop_path
+ ". Error: "
+ str(err)
)
continue
except Exception as ex:
l.error(
"Could not load property "
+ prop_path
+ ". Error: "
+ str(ex)
)
continue
yield (
f_curve,
node_group,
prop_path,
node_name,
attribute_name,
arr_ind,
keyframes,
keyframe_values,
current_val,
interpolation_modes,
interpolation_mode,
)
class AnimationHelper:
current_frame = invalid_frame
def setup(self):
self.current_frame = invalid_frame
self.animated_properties = None
def get_animated_property_list(self):
"""Fetches the animated properties in a presentable way"""
# Get fcurves
if "NodeTreeAction" not in bpy.data.actions:
return {}
self.f_curves = bpy.data.actions["NodeTreeAction"].fcurves
animated_properties = {}
for items in iterate_f_curves(self.f_curves):
(
prop_path,
node_name,
attribute_name,
arr_ind,
keyframes,
keyframe_values,
current_val,
interpolation_mode,
) = (items[2:9] + items[-1:])
key = prop_path
# Array structures
if key in animated_properties:
old_props = animated_properties[key]
old_keyframes, old_keyframe_values, old_current_val = [
old_props[i] for i in [2, 3, 4]
]
if np.any(
np.array(old_keyframes) != np.array(keyframes)
) or arr_ind != len(old_keyframe_values):
l.warning(
"Global Time Keeper: Keyframe times of property "
+ str(prop_path)
+ " do not match up for different array indices. Skipping"
)
continue
animated_properties[key] = (
node_name,
attribute_name,
keyframes,
old_keyframe_values + (keyframe_values,),
old_current_val + (current_val,),
interpolation_mode,
)
else:
animated_properties[key] = (
node_name,
attribute_name,
keyframes,
(keyframe_values,),
(current_val,),
interpolation_mode,
)
animated_properties = {
k: (v[0], v[1], v[2], [tmp for tmp in zip(*v[3])], v[4], v[5])
for k, v in animated_properties.items()
}
return animated_properties
def update_animated_properties(self, scene):
"""Update the animated properties upon change in Global Time."""
self.animated_properties = []
self.animated_properties_info = []
self.interpolation_modes = []
self.animated_values = {}
# Stop if fcurves are not accessible
if "NodeTreeAction" not in bpy.data.actions:
return []
current_frame = scene.frame_current
AnimationHelper.current_frame = current_frame
self.f_curves = bpy.data.actions["NodeTreeAction"].fcurves
updated_nodes = set()
for items in iterate_f_curves(self.f_curves):
# (f_curve, node_group, prop_path, node_name, attribute_name, arr_ind, keyframes, keyframe_values, current_val, interpolation_modes, interpolation_mode) = items
(
f_curve,
node_group,
prop_path,
node_name,
arr_ind,
current_val,
interpolation_modes,
) = (items[0:4] + items[5:6] + items[-3:-1])
# TODO: Make other interpolation modes available
# All interpolation modes will be set to linear in all cases
for mode_i, mode in enumerate(interpolation_modes):
if mode != "LINEAR":
f_curve.keyframe_points[mode_i].interpolation = "LINEAR"
interpolation_modes = ["LINEAR" for _ in interpolation_modes]
new_val = f_curve.evaluate(current_frame)
if prop_path not in self.animated_properties:
self.animated_properties.append(prop_path)
old_vals = (
self.animated_values[prop_path]
if prop_path in self.animated_values
else ()
)
self.animated_values[prop_path] = old_vals + (new_val,)
# No update necessary if the value hasn't changed
if np.isclose(current_val, new_val):
continue
# Update the property with the new value
delimiter_index = prop_path.rindex(".")
try:
# Time Selector nodes will not be updated if they use the scene time
node = node_group.nodes[node_name]
if (
node.bl_idname == "BVTK_Node_TimeSelectorType"
and node.use_scene_time
):
continue
try:
exec(
f_curve.data_path + "[{}] = {}".format(arr_ind, new_val),
{"nodes": node_group.nodes},
)
except TypeError as err:
if arr_ind == 0: # May be a scalar update
exec(
f_curve.data_path + " = {}".format(new_val),
{"nodes": node_group.nodes},
)
else:
l.error(
"Could not update property "
+ prop_path
+ ". Error: "
+ str(err)
)
updated_node = eval(
prop_path[:delimiter_index], {"nodes": node_group.nodes}
)
updated_nodes = updated_nodes.union(set([updated_node]))
except Exception as ex:
l.error(
"Could not update property " + prop_path + ". Error: " + str(ex)
)
# This helps in better displaying vectors by concatenating into a list of tuples
self.animated_properties = self.get_animated_property_list()
return updated_nodes