__init__.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422
  1. from . import ( ops_nodegroup,
  2. base_definitions,
  3. socket_definitions,
  4. link_definitions,
  5. xForm_definitions,
  6. misc_nodes_ui,
  7. primitives_definitions,
  8. deformer_definitions,
  9. math_definitions,
  10. i_o,
  11. schema_definitions,
  12. )
  13. from .ops_generate_tree import GenerateMantisTree
  14. from .utilities import prRed
  15. MANTIS_VERSION_MAJOR=0
  16. MANTIS_VERSION_MINOR=10
  17. MANTIS_VERSION_SUB=6
  18. classLists = [module.TellClasses() for module in [
  19. link_definitions,
  20. xForm_definitions,
  21. misc_nodes_ui,
  22. socket_definitions,
  23. ops_nodegroup,
  24. primitives_definitions,
  25. deformer_definitions,
  26. math_definitions,
  27. i_o,
  28. schema_definitions,
  29. base_definitions,
  30. ]]
  31. classLists.append( [GenerateMantisTree] )
  32. #
  33. classes = []
  34. while (classLists):
  35. classes.extend(classLists.pop())
  36. interface_classes = []
  37. import nodeitems_utils
  38. from nodeitems_utils import NodeCategory, NodeItem
  39. class MantisNodeCategory(NodeCategory):
  40. @classmethod
  41. def poll(cls, context):
  42. return (context.space_data.tree_type in ['MantisTree', 'SchemaTree'])
  43. class SchemaNodeCategory(NodeCategory):
  44. @classmethod
  45. def poll(cls, context):
  46. return (context.space_data.path[len(context.space_data.path)-1].node_tree.bl_idname == 'SchemaTree')
  47. class MantisGroupCategory(NodeCategory):
  48. @classmethod
  49. def poll(cls, context):
  50. return (context.space_data.path[len(context.space_data.path)-1].node_tree.bl_idname in ['MantisTree'] and len(context.space_data.path)>1)
  51. input_category=[
  52. NodeItem("InputFloatNode"),
  53. NodeItem("InputVectorNode"),
  54. NodeItem("InputBooleanNode"),
  55. NodeItem("InputStringNode"),
  56. NodeItem("InputIntNode"),
  57. NodeItem("InputMatrixNode"),
  58. NodeItem("InputExistingGeometryObject"),
  59. NodeItem("InputExistingGeometryData"),
  60. ]
  61. link_transform_category = [
  62. NodeItem("LinkCopyLocation"),
  63. NodeItem("LinkCopyRotation"),
  64. NodeItem("LinkCopyScale"),
  65. NodeItem("LinkCopyTransforms"),
  66. NodeItem("LinkLimitLocation"),
  67. NodeItem("LinkLimitScale"),
  68. NodeItem("LinkLimitRotation"),
  69. NodeItem("LinkLimitDistance"),
  70. NodeItem("LinkTransformation"),
  71. ]
  72. link_tracking_category = [
  73. NodeItem("LinkInverseKinematics"),
  74. NodeItem("LinkSplineIK"),
  75. NodeItem("LinkStretchTo"),
  76. NodeItem("LinkDampedTrack"),
  77. NodeItem("LinkLockedTrack"),
  78. NodeItem("LinkTrackTo"),
  79. ]
  80. link_relationship_category = [
  81. NodeItem("linkInherit"),
  82. NodeItem("LinkInheritConstraint"),
  83. NodeItem("LinkArmature"),
  84. ]
  85. deformer_category=[NodeItem(cls.bl_idname) for cls in deformer_definitions.TellClasses()]
  86. xForm_category = [
  87. NodeItem("xFormGeometryObject"),
  88. NodeItem("xFormBoneNode"),
  89. NodeItem("xFormArmatureNode"),
  90. NodeItem("xFormObjectInstance"),
  91. NodeItem("xFormCurvePin"),
  92. ]
  93. driver_category = [
  94. NodeItem("LinkDrivenParameter"),
  95. NodeItem("UtilityFCurve"),
  96. NodeItem("UtilityBoneProperties"),
  97. NodeItem("UtilityDriverVariable"),
  98. NodeItem("UtilitySwitch"),
  99. NodeItem("UtilityDriver"),
  100. NodeItem("UtilityKeyframe"),
  101. ]
  102. geometry_category = [
  103. NodeItem("GeometryCirclePrimitive"),
  104. ]
  105. utility_category = [
  106. NodeItem("MathStaticInt"),
  107. NodeItem("MathStaticFloat"),
  108. NodeItem("MathStaticVector"),
  109. NodeItem("UtilityCatStrings"),
  110. NodeItem("UtilityGeometryOfXForm"),
  111. NodeItem("UtilityNameOfXForm"),
  112. NodeItem("UtilityCombineThreeBool"),
  113. NodeItem("UtilityCombineVector"),
  114. NodeItem("UtilityIntToString"),
  115. NodeItem("UtilityArrayGet"),
  116. NodeItem("UtilityArrayLength"),
  117. NodeItem("UtilityChoose"),
  118. NodeItem("UtilityCompare"),
  119. NodeItem("UtilityPrint"),
  120. NodeItem("UtilitySeparateVector"),
  121. NodeItem("UtilityGetNearestFactorOnCurve"),
  122. NodeItem("UtilityKDChoosePoint"),
  123. NodeItem("UtilityKDChooseXForm"),
  124. ]
  125. matrix_category = [
  126. NodeItem("UtilityMetaRig"),
  127. NodeItem("UtilityMatrixFromCurve"),
  128. NodeItem("UtilityMatricesFromCurve"),
  129. NodeItem("UtilityNumberOfCurveSegments"),
  130. NodeItem("UtilityMatrixFromCurveSegment"),
  131. NodeItem("UtilityPointFromCurve"),
  132. NodeItem("UtilityGetCurvePoint"),
  133. NodeItem("UtilityPointFromBoneMatrix"),
  134. NodeItem("UtilitySetBoneLength"),
  135. NodeItem("UtilityGetBoneLength"),
  136. NodeItem("UtilityBoneMatrixHeadTailFlip"),
  137. NodeItem("UtilityMatrixSetLocation"),
  138. NodeItem("UtilityMatrixFromXForm"),
  139. NodeItem("UtilityAxesFromMatrix"),
  140. NodeItem("UtilityMatrixTransform"),
  141. NodeItem("UtilityTransformationMatrix"),
  142. NodeItem("UtilitySetBoneMatrixTail"),
  143. ]
  144. groups_category = [
  145. NodeItem("MantisNodeGroup"),
  146. NodeItem("MantisSchemaGroup"),
  147. ]
  148. group_interface_category = [
  149. NodeItem("NodeGroupInput"),
  150. NodeItem("NodeGroupOutput"),
  151. ]
  152. node_categories = [
  153. # identifier, label, items list
  154. MantisNodeCategory('INPUT', "Input", items=input_category),
  155. MantisNodeCategory('LINK_TRANSFORM', "Link (Transform)", items=link_transform_category),
  156. MantisNodeCategory('LINK_TRACKING', "Link (Tracking)", items=link_tracking_category),
  157. MantisNodeCategory('LINK_RELATIONSHIP', "Link (Inheritance)", items=link_relationship_category),
  158. MantisNodeCategory('DEFORMER', "Deformer", items=deformer_category),
  159. MantisNodeCategory('XFORM', "Transform", items=xForm_category),
  160. MantisNodeCategory('DRIVER', "Driver", items=driver_category),
  161. MantisNodeCategory('GEOMETRY', "Geometry", items =geometry_category),
  162. MantisNodeCategory('UTILITIES', "Utility", items=utility_category),
  163. MantisNodeCategory('MATRIX', "Matrix", items=matrix_category),
  164. MantisNodeCategory('GROUPS', "Groups", items=groups_category),
  165. MantisGroupCategory('GROUP_INTERFACE', "Group In/Out", items=group_interface_category),
  166. ]
  167. schema_category=[NodeItem(cls.bl_idname) for cls in schema_definitions.TellClasses()]
  168. schema_categories = [
  169. SchemaNodeCategory('SCHEMA_SCHEMA', "Schema", items=schema_category),
  170. ]
  171. import bpy
  172. def init_keymaps():
  173. kc = bpy.context.window_manager.keyconfigs.addon
  174. km = kc.keymaps.new(name="Node Generic", space_type='NODE_EDITOR')
  175. kmi = [
  176. # Normal operation
  177. km.keymap_items.new("mantis.group_nodes", 'G', 'PRESS', ctrl=True),
  178. km.keymap_items.new("mantis.edit_group", 'TAB', 'PRESS'),
  179. km.keymap_items.new("mantis.execute_node_tree", 'E', 'PRESS'),
  180. km.keymap_items.new("mantis.mute_node", 'M', 'PRESS'),
  181. km.keymap_items.new("mantis.nodes_cleanup", "C", 'PRESS', shift=True,),
  182. # Testing
  183. km.keymap_items.new("mantis.query_sockets", 'Q', 'PRESS'),
  184. km.keymap_items.new("mantis.test_operator", 'T', 'PRESS'),
  185. km.keymap_items.new("mantis.visualize_output", 'V', 'PRESS'),
  186. # Saving, Loading, Reloading, etc.
  187. km.keymap_items.new("mantis.export_save_choose", "S", 'PRESS', alt=True,),
  188. km.keymap_items.new("mantis.export_save_as", "S", 'PRESS', alt=True, shift=True),
  189. km.keymap_items.new("mantis.reload_tree", "R", 'PRESS', alt=True,),
  190. km.keymap_items.new("mantis.import_tree", "O", 'PRESS', ctrl=True,),
  191. ]
  192. return km, kmi
  193. addon_keymaps = []
  194. # handlers! these have to be persistent
  195. from bpy.app.handlers import persistent
  196. @persistent
  197. def update_handler(scene):
  198. context=bpy.context
  199. if context.space_data:
  200. if not hasattr(context.space_data, "path"):
  201. return
  202. trees = [p.node_tree for p in context.space_data.path]
  203. if not trees: return
  204. if (node_tree := trees[0]).bl_idname in ['MantisTree']:
  205. if node_tree.is_exporting:
  206. return
  207. if node_tree.prevent_next_exec : pass
  208. elif node_tree.do_live_update and not (node_tree.is_executing):
  209. prev_links = node_tree.num_links
  210. node_tree.num_links = len(node_tree.links)
  211. if (prev_links == -1):
  212. return
  213. if prev_links != node_tree.num_links:
  214. node_tree.tree_valid = False
  215. if node_tree.tree_valid == False:
  216. scene.render.use_lock_interface = True
  217. node_tree.update_tree(context)
  218. scene.render.use_lock_interface = False
  219. @persistent
  220. def execute_handler(scene):
  221. context = bpy.context
  222. if context.space_data:
  223. if not hasattr(context.space_data, "path"):
  224. return
  225. trees = [p.node_tree for p in context.space_data.path]
  226. if not trees: return
  227. if (node_tree := trees[0]).bl_idname in ['MantisTree']:
  228. if node_tree.is_exporting:
  229. return
  230. if node_tree.prevent_next_exec : node_tree.prevent_next_exec = False
  231. elif node_tree.tree_valid and node_tree.do_live_update and not (node_tree.is_executing):
  232. scene.render.use_lock_interface = True
  233. node_tree.execute_tree(context)
  234. scene.render.use_lock_interface = False
  235. node_tree.tree_valid = False
  236. versioning_node_tasks = [
  237. #relevant bl_idnames # task
  238. #(['LinkTransformation'], transformation_constraint_radians_to_degrees)
  239. ]
  240. def do_version_update(node_tree):
  241. from .base_definitions import NODES_REMOVED, SOCKETS_REMOVED, SOCKETS_RENAMED, SOCKETS_ADDED
  242. for n in node_tree.nodes:
  243. rename_jobs = []
  244. if n.bl_idname in NODES_REMOVED:
  245. print(f"INFO: removing node {n.name} of type {n.bl_idname} because it has been deprecated.")
  246. n.inputs.remove(socket)
  247. continue
  248. for i, socket in enumerate(n.inputs.values()):
  249. if (n.bl_idname, socket.identifier) in SOCKETS_REMOVED:
  250. print(f"INFO: removing socket {socket.identifier} of node {n.name} of type {n.bl_idname} because it has been deprecated.")
  251. n.inputs.remove(socket)
  252. for old_class, old_bl_idname, old_name, new_bl_idname, new_name, multi in SOCKETS_RENAMED:
  253. if (n.bl_idname == old_class and socket.bl_idname == old_bl_idname and socket.name == old_name):
  254. rename_jobs.append((socket, i, new_bl_idname, new_name, multi))
  255. for i, socket in enumerate(n.outputs.values()):
  256. if (n.bl_idname, socket.identifier) in SOCKETS_REMOVED:
  257. print(f"INFO: removing socket {socket.identifier} of node {n.name} of type {n.bl_idname} because it has been deprecated.")
  258. n.outputs.remove(socket)
  259. for old_class, old_bl_idname, old_name, new_bl_idname, new_name, multi in SOCKETS_RENAMED:
  260. if (n.bl_idname == old_class and socket.bl_idname == old_bl_idname and socket.name == old_name):
  261. rename_jobs.append((socket, i, new_bl_idname, new_name, multi))
  262. for bl_idname, in_out, socket_type, socket_name, index, use_multi_input, default_val in SOCKETS_ADDED:
  263. if n.bl_idname != bl_idname:
  264. continue
  265. if in_out == 'INPUT' and n.inputs.get(socket_name) is None:
  266. print(f"INFO: adding socket \"{socket_name}\" of type {socket_type} to node {n.name} of type {n.bl_idname}.")
  267. s = n.inputs.new(socket_type, socket_name, use_multi_input=use_multi_input)
  268. s.default_value = default_val
  269. n.inputs.move(len(n.inputs)-1, index)
  270. socket_map = None
  271. if rename_jobs:
  272. from .utilities import get_socket_maps
  273. socket_maps = get_socket_maps(n)
  274. for socket, socket_index, new_bl_idname, new_name, multi in rename_jobs:
  275. old_id = socket.identifier
  276. print (f"Renaming socket {socket.identifier} to {new_name} in node {n.name}")
  277. from .utilities import do_relink
  278. if socket.is_output:
  279. index = 1
  280. in_out = "OUTPUT"
  281. n.outputs.remove(socket)
  282. s = n.outputs.new(new_bl_idname, new_name, identifier=new_name, use_multi_input=multi)
  283. n.outputs.move(len(n.outputs)-1, socket_index)
  284. socket_map = socket_maps[1]
  285. else:
  286. index = 0
  287. in_out = "INPUT"
  288. n.inputs.remove(socket)
  289. s = n.inputs.new(new_bl_idname, new_name, identifier=new_name, use_multi_input=multi)
  290. n.inputs.move(len(n.inputs)-1, socket_index)
  291. socket_map = socket_maps[0]
  292. socket_map[new_name] = socket_map[old_id]; del socket_map[old_id]
  293. do_relink(n, s, socket_map)
  294. for bl_idname, task in versioning_node_tasks:
  295. if n.bl_idname in bl_idname: task(n)
  296. # increment the version at the end
  297. node_tree.mantis_version[0] = MANTIS_VERSION_MAJOR
  298. node_tree.mantis_version[1] = MANTIS_VERSION_MINOR
  299. node_tree.mantis_version[2] = MANTIS_VERSION_SUB
  300. @persistent
  301. def version_update_handler(filename):
  302. for node_tree in bpy.data.node_groups: # ensure it can update again after file load.
  303. if node_tree.bl_idname in ["MantisTree", "SchemaTree"]:
  304. node_tree.is_exporting=False; node_tree.is_executing=False
  305. for node_tree in bpy.data.node_groups:
  306. if node_tree.bl_idname in ["MantisTree", "SchemaTree"]:
  307. if (node_tree.mantis_version[0] < MANTIS_VERSION_MAJOR) or \
  308. (node_tree.mantis_version[1] < MANTIS_VERSION_MINOR) or \
  309. (node_tree.mantis_version[2] < MANTIS_VERSION_SUB):
  310. print (f"Updating tree {node_tree.name} to {MANTIS_VERSION_MAJOR}.{MANTIS_VERSION_MINOR}.{MANTIS_VERSION_SUB}")
  311. do_version_update(node_tree)
  312. # I'll need to do some fiddling here when it comes time to try
  313. # and make rig definitions animatable.
  314. @persistent
  315. def on_animation_playback_pre_handler(scene,depsgraph):
  316. for t in bpy.data.node_groups:
  317. if t.bl_idname in ['MantisTree', 'SchemaTree']:
  318. t.is_executing = True
  319. @persistent
  320. def on_animation_playback_post_handler(scene,depsgraph):
  321. for t in bpy.data.node_groups:
  322. if t.bl_idname in ['MantisTree', 'SchemaTree']:
  323. t.is_executing = False
  324. @persistent
  325. def on_undo_pre_handler(scene): # the undo will trigger a depsgraph update
  326. for t in bpy.data.node_groups: # so we enable prevent_next_exec.
  327. if t.bl_idname in ['MantisTree', 'SchemaTree']:
  328. t.prevent_next_exec = True
  329. @persistent
  330. def on_undo_post_handler(scene): # the undo will trigger a depsgraph update
  331. for t in bpy.data.node_groups: # so we enable prevent_next_exec.
  332. if t.bl_idname in ['MantisTree', 'SchemaTree']:
  333. t.prevent_next_exec = True
  334. @persistent
  335. def on_save_pre_handler(scene): # save-as will trigger a depsgraph update
  336. for t in bpy.data.node_groups: # so we enable prevent_next_exec.
  337. if t.bl_idname in ['MantisTree', 'SchemaTree']:
  338. t.prevent_next_exec = True
  339. # annoyingly, regular save does not trigger a DG update so we also need this:
  340. @persistent
  341. def on_save_post_handler(scene): # The DG has already updated and we can disable this.
  342. for t in bpy.data.node_groups:
  343. if t.bl_idname in ['MantisTree', 'SchemaTree']:
  344. t.prevent_next_exec = False
  345. def register():
  346. from bpy.utils import register_class
  347. for cls in classes:
  348. try:
  349. register_class(cls)
  350. except RuntimeError as e:
  351. prRed(f"Registration error for class: {cls.__name__}")
  352. raise e
  353. nodeitems_utils.register_node_categories('MantisNodeCategories', node_categories)
  354. nodeitems_utils.register_node_categories('SchemaNodeCategories', schema_categories)
  355. km, kmi = init_keymaps()
  356. for k in kmi:
  357. k.active = True
  358. addon_keymaps.append((km, k))
  359. # add the handlers
  360. bpy.app.handlers.depsgraph_update_pre.insert(0, update_handler)
  361. bpy.app.handlers.depsgraph_update_post.insert(0, execute_handler)
  362. bpy.app.handlers.save_pre.insert(0, on_save_pre_handler)
  363. bpy.app.handlers.save_post.insert(0, on_save_post_handler)
  364. bpy.app.handlers.load_post.insert(0, version_update_handler)
  365. bpy.app.handlers.animation_playback_pre.insert(0, on_animation_playback_pre_handler)
  366. bpy.app.handlers.animation_playback_post.insert(0, on_animation_playback_post_handler)
  367. bpy.app.handlers.undo_pre.insert(0, on_undo_pre_handler)
  368. # I'm adding mine in first to ensure other addons don't mess up mine
  369. # but I am a good citizen! so my addon won't mess up yours! probably...
  370. def unregister():
  371. for tree in bpy.data.node_groups: # ensure it doesn't try to update while quitting.
  372. if tree.bl_idname in ['MantisTree, SchemaTree']:
  373. tree.is_exporting=True; tree.is_executing=True
  374. nodeitems_utils.unregister_node_categories('MantisNodeCategories')
  375. nodeitems_utils.unregister_node_categories('SchemaNodeCategories')
  376. from bpy.utils import unregister_class
  377. for cls in reversed(classes):
  378. unregister_class(cls)
  379. for km, kmi in addon_keymaps:
  380. km.keymap_items.remove(kmi)
  381. addon_keymaps.clear()