deformer_nodes_ui.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. import bpy
  2. from bpy.types import NodeTree, Node, NodeSocket
  3. from .base_definitions import MantisUINode, DeformerNode, get_signature_from_edited_tree
  4. from.deformer_socket_templates import *
  5. from .utilities import (prRed, prGreen, prPurple, prWhite, prOrange,
  6. wrapRed, wrapGreen, wrapPurple, wrapWhite,
  7. wrapOrange,)
  8. def TellClasses():
  9. return [
  10. DeformerArmatureNode,
  11. DeformerHook,
  12. DeformerMorphTargetDeform,
  13. DeformerMorphTarget,
  14. DeformerSurfaceDeform,
  15. DeformerMeshDeform,
  16. DeformerLatticeDeform,
  17. DeformerSmoothCorrectiveDeform,
  18. ]
  19. def default_traverse(self, socket):
  20. if (socket == self.outputs["Deformer"]):
  21. return self.inputs["Deformer"]
  22. if (socket == self.inputs["Deformer"]):
  23. return self.outputs["Deformer"]
  24. return None
  25. class DeformerArmatureNode(Node, DeformerNode):
  26. '''A node representing an Armature Deformer'''
  27. bl_idname = 'DeformerArmature'
  28. bl_label = "Armature Deform"
  29. bl_icon = 'MOD_ARMATURE'
  30. initialized : bpy.props.BoolProperty(default = False)
  31. mantis_node_class_name=bl_idname
  32. def init(self, context):
  33. # self.inputs.new ("RelationshipSocket", "Input Relationship")
  34. self.inputs.new('xFormSocket', "Armature Object")
  35. self.inputs.new('StringSocket', "Blend Vertex Group")
  36. # self.inputs.new('StringSocket', "Preserve Volume Vertex Group") #TODO figure out the right UX for automatic dual-quat blending
  37. self.inputs.new('BooleanSocket', "Invert Vertex Group")
  38. self.inputs.new('BooleanSocket', "Preserve Volume")
  39. # TODO: make the above controlled by a vertex group instead.
  40. self.inputs.new('BooleanSocket', "Use Multi Modifier")# might just set this auto
  41. self.inputs.new('BooleanSocket', "Use Envelopes")
  42. self.inputs.new('BooleanSocket', "Use Vertex Groups")
  43. self.inputs.new("DeformerSocket", "Deformer")
  44. s = self.inputs.new("xFormSocket", "Copy Skin Weights From")
  45. s.hide = True
  46. self.inputs.new("EnumSkinning", "Skinning Method")
  47. self.outputs.new('DeformerSocket', "Deformer")
  48. self.initialized = True
  49. def display_update(self, parsed_tree, context):
  50. self.inputs["Copy Skin Weights From"].hide = True
  51. node_tree = context.space_data.path[0].node_tree
  52. nc = parsed_tree.get(get_signature_from_edited_tree(self, context))
  53. if nc:
  54. self.inputs["Copy Skin Weights From"].hide = not (nc.evaluate_input("Skinning Method") == "COPY_FROM_OBJECT")
  55. class DeformerHook(Node, DeformerNode):
  56. '''A node representing a Hook Deformer'''
  57. bl_idname = 'DeformerHook'
  58. bl_label = "Hook Deform"
  59. bl_icon = 'HOOK'
  60. initialized : bpy.props.BoolProperty(default = False)
  61. mantis_node_class_name=bl_idname
  62. def init(self, context):
  63. self.init_sockets(HookSockets)
  64. self.initialized = True
  65. def display_update(self, parsed_tree, context):
  66. curve_sockets = [
  67. self.inputs["Affect Curve Radius"],
  68. self.inputs['Auto-Bezier'],
  69. self.inputs['Spline Index'],
  70. ]
  71. is_curve_hook=True
  72. if self.outputs["Deformer"].is_linked:
  73. from bpy.types import Object
  74. if (mantis_node := parsed_tree.get(get_signature_from_edited_tree(self, context))):
  75. if (xforms := mantis_node.GetxForm()):
  76. for xF_node in xforms:
  77. if (ob := xF_node.bObject) and isinstance (xF_node, Object):
  78. if ob.type != 'CURVE': is_curve_hook=False
  79. for socket in curve_sockets:
  80. socket.hide=not is_curve_hook
  81. from .utilities import get_socket_maps, relink_socket_map
  82. # TODO this should probably not be in this file but intstead in Utilities or something
  83. def simple_do_relink(node, map, in_out='INPUT'):
  84. from bpy.types import NodeSocket
  85. for key, val in map.items():
  86. s = node.inputs.get(key) if in_out == "INPUT" else node.outputs.get(key)
  87. if s is None:
  88. if in_out == "INPUT":
  89. if node.num_targets > 0:
  90. s = node.inputs["Target."+str(node.num_targets-1).zfill(3)]
  91. else:
  92. continue
  93. if isinstance(val, list):
  94. for sub_val in val:
  95. if isinstance(sub_val, NodeSocket):
  96. if in_out =='INPUT':
  97. node.id_data.links.new(input=sub_val, output=s)
  98. else:
  99. node.id_data.links.new(input=s, output=sub_val)
  100. else:
  101. try:
  102. s.default_value = val
  103. except (AttributeError, ValueError, TypeError): # must be readonly or maybe it doesn't have a d.v.. TypeError if the d.v. is None at this point
  104. pass
  105. # Dynamic
  106. # - each Morph Target gets a MT input
  107. # - each Morph Target gets an influence input
  108. class DeformerMorphTargetDeform(Node, DeformerNode):
  109. '''A node representing a Morph Target Deformer'''
  110. bl_idname = 'DeformerMorphTargetDeform'
  111. bl_label = "Morph Deform"
  112. bl_icon = 'MOD_ARMATURE'
  113. initialized : bpy.props.BoolProperty(default = False)
  114. num_targets : bpy.props.IntProperty(default = 0)
  115. mantis_node_class_name=bl_idname
  116. def init(self, context):
  117. self.id_data.do_live_update = False
  118. self.inputs.new('DeformerSocket', 'Deformer', )
  119. self.inputs.new('BooleanSocket', 'Use Shape Key', )
  120. s = self.inputs.new('BooleanSocket', 'Use Offset', )
  121. s.default_value = True
  122. self.inputs.new('WildcardSocket', '', identifier='__extend__')
  123. self.outputs.new('DeformerSocket', "Deformer")
  124. self.update_morph_deformer()
  125. def update_morph_deformer(self, force=False):
  126. self.initialized = False
  127. # use_offset = self.inputs["Use Offset"].default_value
  128. socket_maps = get_socket_maps(self)
  129. if socket_maps is None:
  130. return
  131. input_map = socket_maps[0]
  132. # checc to see if targets have been removed... then modify the input map if necessary
  133. targets_deleted = 0 # this should usually be either 0 or 1
  134. for i in range(self.num_targets):
  135. name = "Target."+str(i).zfill(3); inf_name = "Value."+str(i).zfill(3)
  136. if self.inputs[name].is_linked == False:
  137. del input_map[name]; del input_map[inf_name]
  138. targets_deleted+=1
  139. elif targets_deleted: # move it back
  140. new_name = "Target."+str(i-targets_deleted).zfill(3); new_inf_name = "Value."+str(i-targets_deleted).zfill(3)
  141. input_map[new_name] = input_map[name]; input_map[new_inf_name] = input_map[inf_name]
  142. del input_map[name]; del input_map[inf_name]
  143. self.num_targets-=targets_deleted
  144. if self.inputs[-1].is_linked and self.inputs[-1].bl_idname == 'WildcardSocket':
  145. self.num_targets+=1
  146. self.inputs.clear()
  147. self.inputs.new('DeformerSocket', 'Deformer', )
  148. self.inputs.new('BooleanSocket', 'Use Shape Key', )
  149. self.inputs.new('BooleanSocket', 'Use Offset', )
  150. for i in range(self.num_targets):
  151. self.inputs.new("MorphTargetSocket", "Target."+str(i).zfill(3))
  152. self.inputs.new("FloatSocket", "Value."+str(i).zfill(3))
  153. simple_do_relink(self, input_map, in_out='INPUT')
  154. if self.inputs[-1].bl_idname not in ["WildcardSocket"]:
  155. self.inputs.new('WildcardSocket', '', identifier='__extend__')
  156. self.initialized = True
  157. def update(self):
  158. if self.id_data.is_exporting:
  159. return # so that we don't update it while saving/loading the tree
  160. self.update_morph_deformer(force=False)
  161. def display_update(self, parsed_tree, context):
  162. if self.inputs["Deformer"].is_linked:
  163. if self.inputs["Deformer"].links[0].from_node.bl_idname not in [self.bl_idname, "NodeGroupInput"]:
  164. self.inputs["Use Shape Key"].default_value = False
  165. self.inputs["Use Shape Key"].hide = True
  166. elif self.inputs["Deformer"].links[0].from_node.inputs["Use Shape Key"].default_value == False:
  167. self.inputs["Use Shape Key"].default_value = False
  168. self.inputs["Use Shape Key"].hide = True
  169. if self.inputs["Use Offset"] == False:
  170. self.inputs["Use Shape Key"].hide = True
  171. self.inputs["Use Shape Key"].default_value = False
  172. # TODO: there is no reason for this to be a separate node!
  173. class DeformerMorphTarget(Node, DeformerNode):
  174. '''A node representing a single Morph Target'''
  175. bl_idname = 'DeformerMorphTarget'
  176. bl_label = "Morph Target"
  177. bl_icon = 'SHAPEKEY_DATA'
  178. initialized : bpy.props.BoolProperty(default = False)
  179. num_targets : bpy.props.IntProperty(default = 0)
  180. mantis_node_class_name=bl_idname
  181. def init(self, context):
  182. self.inputs.new('xFormSocket', "Relative to")
  183. self.inputs.new('xFormSocket', "Object")
  184. self.inputs.new('StringSocket', "Vertex Group")
  185. self.outputs.new('MorphTargetSocket', "Morph Target")
  186. self.initialized = True
  187. class DeformerSurfaceDeform(Node, DeformerNode):
  188. '''A node representing a Surface Deform modifier'''
  189. bl_idname = 'DeformerSurfaceDeform'
  190. bl_label = "Surface Deform"
  191. bl_icon = 'MOD_SOFT'
  192. initialized : bpy.props.BoolProperty(default = False)
  193. num_targets : bpy.props.IntProperty(default = 0)
  194. mantis_node_class_name=bl_idname
  195. def init(self, context):
  196. self.init_sockets(SurfaceDeformSockets)
  197. self.initialized = True
  198. class DeformerMeshDeform(Node, DeformerNode):
  199. '''A node representing a Mesh Deform modifier'''
  200. bl_idname = 'DeformerMeshDeform'
  201. bl_label = "Mesh Deform"
  202. bl_icon = 'MOD_SOFT'
  203. initialized : bpy.props.BoolProperty(default = False)
  204. num_targets : bpy.props.IntProperty(default = 0)
  205. mantis_node_class_name=bl_idname
  206. def init(self, context):
  207. self.init_sockets(MeshDeformSockets)
  208. self.initialized = True
  209. class DeformerLatticeDeform(Node, DeformerNode):
  210. '''A node representing a Lattice Deformer'''
  211. bl_idname = 'DeformerLatticeDeform'
  212. bl_label = "Lattice Deform"
  213. bl_icon = 'MOD_LATTICE'
  214. initialized : bpy.props.BoolProperty(default = False)
  215. num_targets : bpy.props.IntProperty(default = 0)
  216. mantis_node_class_name=bl_idname
  217. def init(self, context):
  218. self.init_sockets(LatticeDeformSockets)
  219. self.initialized = True
  220. class DeformerSmoothCorrectiveDeform(Node, DeformerNode):
  221. '''A node representing a Corrective Smooth Deformer'''
  222. bl_idname = 'DeformerSmoothCorrectiveDeform'
  223. bl_label = "Smooth Deform"
  224. bl_icon = 'MOD_SMOOTH'
  225. initialized : bpy.props.BoolProperty(default = False)
  226. num_targets : bpy.props.IntProperty(default = 0)
  227. mantis_node_class_name=bl_idname
  228. def init(self, context):
  229. self.init_sockets(SmoothDeformSockets)
  230. self.initialized = True
  231. def display_update(self, parsed_tree, context):
  232. use_corrective_smooth = self.inputs["Use Corrective Smooth"]
  233. if ( use_corrective_smooth.is_linked or
  234. use_corrective_smooth.default_value == False): # this is inverted
  235. self.inputs['Corrective Smooth Scale'].hide = False
  236. else:
  237. self.inputs['Corrective Smooth Scale'].hide = True
  238. # Set up the class property that ties the UI classes to the Mantis classes.
  239. for cls in TellClasses():
  240. cls.set_mantis_class()