base_definitions.py 48 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103
  1. #Mantis Nodes Base
  2. import bpy
  3. from bpy.props import (BoolProperty, StringProperty, EnumProperty, CollectionProperty, \
  4. IntProperty, IntVectorProperty, PointerProperty, BoolVectorProperty)
  5. from . import ops_nodegroup
  6. from bpy.types import NodeTree, Node, PropertyGroup, Operator, UIList, Panel
  7. from .utilities import (prRed, prGreen, prPurple, prWhite,
  8. prOrange,
  9. wrapRed, wrapGreen, wrapPurple, wrapWhite,
  10. wrapOrange,)
  11. from .utilities import get_socket_maps, relink_socket_map, do_relink
  12. FLOAT_EPSILON=0.0001 # used to check against floating point inaccuracy
  13. def TellClasses():
  14. #Why use a function to do this? Because I don't need every class to register.
  15. return [ MantisTree,
  16. SchemaTree,
  17. MantisNodeGroup,
  18. SchemaGroup,
  19. ]
  20. def error_popup_draw(self, context):
  21. self.layout.label(text="Error executing tree. See Console.")
  22. mantis_root = ".".join(__name__.split('.')[:-1]) # absolute HACK
  23. from os import environ
  24. # https://docs.blender.org/api/master/bpy.types.NodeTree.html#bpy.types.NodeTree.valid_socket_type
  25. # thank you, Sverchok
  26. def valid_interface_types(cls : NodeTree, socket_idname : str):
  27. from .socket_definitions import tell_valid_bl_idnames, TellClasses
  28. #TODO: do the versioning code to handle this so it can be in all versions
  29. if bpy.app.version <= (4,4,0): # should work in 4.4.1
  30. return socket_idname in [cls.bl_idname for cls in TellClasses()]
  31. else: # once versioning is finished this will be unnecesary.
  32. return socket_idname in tell_valid_bl_idnames()
  33. def fix_reroute_colors(tree):
  34. context = bpy.context
  35. if any((tree.is_executing, tree.is_exporting, tree.do_live_update==False, context.space_data is None) ):
  36. return
  37. from collections import deque
  38. from .utilities import socket_seek
  39. from .socket_definitions import MantisSocket
  40. reroutes_without_color = deque()
  41. for n in tree.nodes:
  42. if n.bl_idname=='NodeReroute' and n.inputs[0].bl_idname == "NodeSocketColor":
  43. reroutes_without_color.append(n)
  44. try:
  45. while reroutes_without_color:
  46. rr = reroutes_without_color.pop()
  47. if rr.inputs[0].is_linked:
  48. link = rr.inputs[0].links[0]
  49. socket = socket_seek(link, tree.links)
  50. if isinstance(socket, MantisSocket):
  51. rr.socket_idname = socket.bl_idname
  52. except Exception as e:
  53. print(wrapOrange("WARN: Updating reroute color failed with exception: ")+wrapWhite(f"{e.__class__.__name__}"))
  54. #functions to identify the state of the system using hashes
  55. # this function runs a lot so it should be optimized as well as possible.
  56. def hash_tree(tree):
  57. trees=set(); links=[]; hash_data=""
  58. for node in tree.nodes:
  59. hash_data+=str(node.name)
  60. if hasattr(node, 'node_tree') and node.node_tree:
  61. trees.add(node.node_tree)
  62. for other_tree in trees:
  63. hash_data+=str(hash_tree(other_tree))
  64. for link in tree.links:
  65. links.append( link.from_node.name+link.from_socket.name+
  66. link.to_node.name+link.to_socket.name+
  67. str(link.multi_input_sort_id) )
  68. links.sort(); hash_data+=''.join(links)
  69. return hash(hash_data)
  70. class MantisTree(NodeTree):
  71. '''A custom node tree type that will show up in the editor type list'''
  72. bl_idname = 'MantisTree'
  73. bl_label = "Rigging Nodes"
  74. bl_icon = 'OUTLINER_OB_ARMATURE'
  75. tree_valid:BoolProperty(default=False)
  76. hash:StringProperty(default='')
  77. do_live_update:BoolProperty(default=True) # use this to disable updates for e.g. scripts
  78. num_links:IntProperty(default=-1)
  79. filepath:StringProperty(default="", subtype='FILE_PATH')
  80. is_executing:BoolProperty(default=False)
  81. is_exporting:BoolProperty(default=False)
  82. execution_id:StringProperty(default='')
  83. # prev_execution_id:StringProperty(default='')
  84. mantis_version:IntVectorProperty(default=[0,9,2])
  85. # this prevents the node group from executing on the next depsgraph update
  86. # because I don't always have control over when the dg update happens.
  87. prevent_next_exec:BoolProperty(default=False)
  88. parsed_tree={}
  89. if (bpy.app.version < (4, 4, 0)): # in 4.4 this leads to a crash
  90. @classmethod
  91. def valid_socket_type(cls : NodeTree, socket_idname: str):
  92. return valid_interface_types(cls, socket_idname)
  93. def update(self): # set the reroute colors
  94. if (bpy.app.version >= (4,4,0)):
  95. fix_reroute_colors(self)
  96. def update_tree(self, context = None, force=False, error_popups=False):
  97. if self.is_exporting:
  98. return
  99. my_hash = str( hash_tree(self) )
  100. if my_hash != self.hash or force:
  101. self.hash = my_hash
  102. self.is_executing = True
  103. from . import readtree
  104. prGreen("Validating Tree: %s" % self.name)
  105. try:
  106. import bpy # I am importing here so that the context passed in
  107. # is used for display update... but I always want to do this
  108. scene = bpy.context.scene
  109. scene.render.use_lock_interface = True
  110. self.parsed_tree = readtree.parse_tree(self, error_popups)
  111. if context:
  112. self.display_update(context)
  113. self.tree_valid = True
  114. except Exception as e:
  115. prRed("Failed to update node tree due to error.")
  116. self.tree_valid = False
  117. self.hash='' # unset the hash to mark the tree as un-parsed.
  118. raise e
  119. finally:
  120. scene.render.use_lock_interface = False
  121. self.is_executing = False
  122. def display_update(self, context):
  123. if self.is_exporting:
  124. return
  125. self.is_executing = True
  126. current_tree = bpy.context.space_data.path[-1].node_tree
  127. for node in current_tree.nodes:
  128. if hasattr(node, "display_update"):
  129. try:
  130. node.display_update(self.parsed_tree, context)
  131. except Exception as e:
  132. print("Node \"%s\" failed to update display with error: %s" %(wrapGreen(node.name), wrapRed(e)))
  133. self.is_executing = False
  134. # TODO: deal with invalid links properly.
  135. # - Non-hierarchy links should be ignored in the circle-check and so the links should be marked valid in such a circle
  136. # - hierarchy-links should be marked invalid and prevent the tree from executing.
  137. def execute_tree(self,context, error_popups = False):
  138. self.prevent_next_exec = False
  139. if not self.hash:
  140. return
  141. if self.is_exporting or self.is_executing:
  142. return
  143. prGreen("Executing Tree: %s" % self.name)
  144. self.is_executing = True
  145. from . import readtree
  146. try:
  147. context.scene.render.use_lock_interface = True
  148. readtree.execute_tree(self.parsed_tree, self, context, error_popups)
  149. except RecursionError as e:
  150. prRed("Recursion error while parsing tree.")
  151. finally:
  152. context.scene.render.use_lock_interface = False
  153. self.is_executing = False
  154. class SchemaTree(NodeTree):
  155. '''A node tree representing a schema to generate a Mantis tree'''
  156. bl_idname = 'SchemaTree'
  157. bl_label = "Rigging Nodes Schema"
  158. bl_icon = 'RIGID_BODY_CONSTRAINT'
  159. # these are only needed for consistent interface, but should not be used
  160. do_live_update:BoolProperty(default=True) # default to true so that updates work
  161. is_executing:BoolProperty(default=False)
  162. is_exporting:BoolProperty(default=False)
  163. mantis_version:IntVectorProperty(default=[0,9,2])
  164. if (bpy.app.version < (4, 4, 0)): # in 4.4 this leads to a crash
  165. @classmethod
  166. def valid_socket_type(cls : NodeTree, socket_idname: str):
  167. return valid_interface_types(cls, socket_idname)
  168. def update(self): # set the reroute colors
  169. if (bpy.app.version >= (4,4,0)):
  170. fix_reroute_colors(self)
  171. from dataclasses import dataclass, field
  172. from typing import Any
  173. @dataclass
  174. class MantisSocketTemplate():
  175. name : str = field(default="")
  176. bl_idname : str = field(default="")
  177. traverse_target : str = field(default="")
  178. identifier : str = field(default="")
  179. display_shape : str = field(default="") # for arrays
  180. category : str = field(default="") # for use in display update
  181. blender_property : str | tuple[str] = field(default="") # for props_sockets -> evaluate sockets
  182. is_input : bool = field(default=False)
  183. hide : bool = field(default=False)
  184. use_multi_input : bool = field(default=False)
  185. default_value : Any = field(default=None)
  186. #TODO: do a better job explaining how MantisNode and MantisUINode relate.
  187. class MantisUINode:
  188. """
  189. This class contains the common user-interface features of Mantis nodes.
  190. MantisUINode objects will spawn one or more MantisNode objects when the graph is evaluated.
  191. The MantisNode objects will pull the data from the UI node and use it to generate the graph.
  192. """
  193. mantis_node_library=''
  194. mantis_node_class_name=''
  195. mantis_class=None
  196. @classmethod
  197. def poll(cls, ntree):
  198. return (ntree.bl_idname in ['MantisTree', 'SchemaTree'])
  199. @classmethod
  200. def set_mantis_class(self):
  201. from importlib import import_module
  202. # do not catch errors, they should cause a failure.
  203. try:
  204. module = import_module(self.mantis_node_library, package=mantis_root)
  205. self.mantis_class=getattr(module, self.mantis_node_class_name)
  206. except Exception as e:
  207. print(self)
  208. raise e
  209. def insert_link(self, link):
  210. if (bpy.app.version >= (4, 4, 0)):
  211. return # this causes a crash due to a bug.
  212. context = bpy.context
  213. if context.space_data:
  214. node_tree = context.space_data.path[0].node_tree
  215. if node_tree.do_live_update:
  216. node_tree.update_tree(context)
  217. if (link.to_socket.is_linked == False):
  218. node_tree.num_links+=1
  219. elif (link.to_socket.is_multi_input):
  220. node_tree.num_links+=1
  221. def init_sockets(self, socket_templates : tuple[MantisSocketTemplate]):
  222. for template in socket_templates:
  223. collection = self.outputs
  224. if template.is_input:
  225. collection = self.inputs
  226. identifier = template.name
  227. if template.identifier:
  228. identifier = template.identifier
  229. use_multi_input = template.use_multi_input if template.is_input else False
  230. socket = collection.new(
  231. template.bl_idname,
  232. template.name,
  233. identifier=identifier,
  234. use_multi_input=use_multi_input
  235. )
  236. socket.hide= template.hide
  237. if template.category:
  238. # a custom property for the UI functions to use.
  239. socket['category'] = template.category
  240. if template.default_value is not None:
  241. socket.default_value = template.default_value
  242. # this can throw a TypeError - it is the caller's
  243. # responsibility to send the right type.
  244. if template.use_multi_input: # this is an array
  245. socket.display_shape = 'SQUARE_DOT'
  246. class SchemaUINode(MantisUINode):
  247. mantis_node_library='.schema_containers'
  248. @classmethod
  249. def poll(cls, ntree):
  250. return (ntree.bl_idname in ['SchemaTree'])
  251. class LinkNode(MantisUINode):
  252. mantis_node_library='.link_containers'
  253. @classmethod
  254. def poll(cls, ntree):
  255. return (ntree.bl_idname in ['MantisTree', 'SchemaTree'])
  256. class xFormNode(MantisUINode):
  257. mantis_node_library='.xForm_containers'
  258. @classmethod
  259. def poll(cls, ntree):
  260. return (ntree.bl_idname in ['MantisTree', 'SchemaTree'])
  261. class DeformerNode(MantisUINode):
  262. mantis_node_library='.deformer_containers'
  263. @classmethod
  264. def poll(cls, ntree):
  265. return (ntree.bl_idname in ['MantisTree', 'SchemaTree'])
  266. def poll_node_tree(self, object):
  267. forbid = []
  268. context = bpy.context
  269. if context.space_data:
  270. if context.space_data.path:
  271. for path_item in context.space_data.path:
  272. forbid.append(path_item.node_tree.name)
  273. if isinstance(object, MantisTree) and object.name not in forbid:
  274. return True
  275. return False
  276. # TODO: try and remove the extra loop used here... but it is OK for now
  277. def should_remove_socket(node, socket):
  278. # a function to check if the socket is in the interface
  279. id_found = False
  280. for item in node.node_tree.interface.items_tree:
  281. if item.item_type != "SOCKET": continue
  282. if item.identifier == socket.identifier:
  283. id_found = True; break
  284. return not id_found
  285. # TODO: try to check identifiers instead of name.
  286. def node_group_update(node, force = False):
  287. if not node.is_updating:
  288. raise RuntimeError("Cannot update node while it is not marked as updating.")
  289. if not force:
  290. if (node.id_data.do_live_update == False) or \
  291. (node.id_data.is_executing == True) or \
  292. (node.id_data.is_exporting == True):
  293. return
  294. # note: if (node.id_data.is_exporting == True) I need to be able to update so I can make links.
  295. toggle_update = node.id_data.do_live_update
  296. node.id_data.do_live_update = False
  297. identifiers_in={socket.identifier:socket for socket in node.inputs}
  298. identifiers_out={socket.identifier:socket for socket in node.outputs}
  299. indices_in,indices_out={},{} # check by INDEX to see if the socket's name/type match.
  300. for collection, map in [(node.inputs, indices_in), (node.outputs, indices_out)]:
  301. for i, socket in enumerate(collection):
  302. map[socket.identifier]=i
  303. if node.node_tree is None:
  304. node.inputs.clear(); node.outputs.clear()
  305. node.id_data.do_live_update = toggle_update
  306. return
  307. found_in, found_out = [], []
  308. update_input, update_output = False, False
  309. for item in node.node_tree.interface.items_tree:
  310. if item.item_type != "SOCKET": continue
  311. if item.in_out == 'OUTPUT':
  312. if s:= identifiers_out.get(item.identifier): # if the requested output doesn't exist, update
  313. found_out.append(item.identifier)
  314. if (indices_out[s.identifier]!=item.index): update_output=True; continue
  315. if update_output: continue
  316. if s.bl_idname != item.socket_type: update_output = True; continue
  317. else: update_output = True; continue
  318. else:
  319. if s:= identifiers_in.get(item.identifier): # if the requested input doesn't exist, update
  320. found_in.append(item.identifier)
  321. if (indices_in[s.identifier]!=item.index): update_input=True; continue
  322. if update_input: continue # done here
  323. if s.bl_idname != item.socket_type: update_input = True; continue
  324. else: update_input = True; continue
  325. # Schema has an extra input for Length and for Extend.
  326. if node.bl_idname == 'MantisSchemaGroup':
  327. found_in.extend(['Schema Length', ''])
  328. # if we have too many elements, just get rid of the ones we don't need
  329. if len(node.inputs) > len(found_in):#
  330. for inp in node.inputs:
  331. if inp.identifier in found_in: continue
  332. node.inputs.remove(inp)
  333. if len(node.outputs) > len(found_out):
  334. for out in node.outputs:
  335. if out.identifier in found_out: continue
  336. node.outputs.remove(out)
  337. #
  338. if len(node.inputs) > 0 and (inp := node.inputs[-1]).bl_idname == 'WildcardSocket' and inp.is_linked:
  339. update_input = True
  340. #
  341. if not (update_input or update_output):
  342. node.id_data.do_live_update = toggle_update
  343. return
  344. if update_input or update_output:
  345. socket_maps = get_socket_maps(node,)
  346. if socket_maps:
  347. socket_map_in, socket_map_out = socket_maps
  348. if node.bl_idname == "MantisSchemaGroup" and \
  349. len(node.inputs)+len(node.outputs)<=2 and\
  350. len(node.node_tree.interface.items_tree) > 0:
  351. socket_map_in, socket_map_out = None, None
  352. # We have to initialize the node because it only has its base inputs.
  353. elif socket_maps is None:
  354. node.id_data.do_live_update = toggle_update
  355. return
  356. if update_input :
  357. if node.bl_idname == 'MantisSchemaGroup':
  358. schema_length=0
  359. if sl := node.inputs.get("Schema Length"):
  360. schema_length = sl.default_value
  361. # sometimes this isn't available yet # TODO not happy about this solution
  362. remove_me=[]
  363. # remove all found map items but the Schema Length input (reuse it)
  364. for i, socket in enumerate(node.inputs):
  365. if socket.identifier == "Schema Length" and i == 0:
  366. continue
  367. elif (socket_map_in is None) or socket.identifier in socket_map_in.keys():
  368. remove_me.append(socket)
  369. elif should_remove_socket(node, socket):
  370. remove_me.append(socket)
  371. while remove_me:
  372. node.inputs.remove(remove_me.pop())
  373. if update_output:
  374. remove_me=[]
  375. for socket in node.outputs:
  376. if (socket_map_out is None) or socket.identifier in socket_map_out.keys():
  377. remove_me.append(socket)
  378. elif should_remove_socket(node, socket):
  379. remove_me.append(socket)
  380. while remove_me:
  381. node.inputs.remove(remove_me.pop())
  382. from .utilities import relink_socket_map_add_socket
  383. reorder_me_input = []; input_index = 0
  384. reorder_me_output = []; output_index = 0
  385. def update_group_sockets(interface_item, is_input):
  386. socket_map = socket_map_in if is_input else socket_map_out
  387. socket_collection = node.inputs if is_input else node.outputs
  388. counter = input_index if is_input else output_index
  389. reorder_collection = reorder_me_input if is_input else reorder_me_output
  390. if socket_map:
  391. if item.identifier in socket_map.keys():
  392. socket = relink_socket_map_add_socket(node, socket_collection, item)
  393. do_relink(node, socket, socket_map, item.in_out)
  394. else:
  395. for has_socket in socket_collection:
  396. if has_socket.bl_idname == item.socket_type and \
  397. has_socket.name == item.name:
  398. reorder_collection.append((has_socket, counter))
  399. break
  400. else:
  401. socket = relink_socket_map_add_socket(node, socket_collection, item)
  402. else:
  403. socket = relink_socket_map_add_socket(node, socket_collection, item)
  404. counter += 1
  405. for item in node.node_tree.interface.items_tree:
  406. if item.item_type != "SOCKET": continue
  407. if (item.in_out == 'INPUT' and update_input):
  408. update_group_sockets(item, True)
  409. if (item.in_out == 'OUTPUT' and update_output):
  410. update_group_sockets(item, False)
  411. both_reorders = zip([reorder_me_input, reorder_me_output], [node.inputs, node.outputs])
  412. for reorder_task, collection in both_reorders:
  413. for socket, position in reorder_task:
  414. for i, s in enumerate(collection): # get the index
  415. if s.identifier == socket.identifier: break
  416. else:
  417. prRed(f"WARN: could not reorder socket {socket.name}")
  418. to_index = position
  419. if (not socket.is_output) and node.bl_idname == "MantisSchemaGroup":
  420. to_index+=1
  421. collection.move(i, to_index)
  422. # at this point there is no wildcard socket
  423. if socket_map_in and '__extend__' in socket_map_in.keys():
  424. do_relink(node, None, socket_map_in, in_out='INPUT', parent_name='Constant' )
  425. node.id_data.do_live_update = toggle_update
  426. def node_tree_prop_update(self, context):
  427. if self.is_updating: # update() can be called from update() and that leads to an infinite loop.
  428. return # so we check if an update is currently running.
  429. self.is_updating = True
  430. def init_schema(self, context):
  431. if len(self.inputs) == 0:
  432. self.inputs.new("UnsignedIntSocket", "Schema Length", identifier='Schema Length')
  433. if self.inputs[-1].bl_idname != "WildcardSocket":
  434. self.inputs.new("WildcardSocket", "", identifier="__extend__")
  435. init_schema(self, context)
  436. try:
  437. node_group_update(self, force=True)
  438. finally: # ensure this line is run even if there is an error
  439. self.is_updating = False
  440. if self.bl_idname in ['MantisSchemaGroup'] and self.node_tree is not None:
  441. init_schema(self, context)
  442. from bpy.types import NodeCustomGroup
  443. def group_draw_buttons(self, context, layout):
  444. row = layout.row(align=True)
  445. row.prop(self, "node_tree", text="")
  446. if self.node_tree is None:
  447. row.operator("mantis.new_node_tree", text="", icon='PLUS', emboss=True)
  448. else:
  449. row.operator("mantis.edit_group", text="", icon='NODETREE', emboss=True)
  450. class MantisNodeGroup(Node, MantisUINode):
  451. bl_idname = "MantisNodeGroup"
  452. bl_label = "Node Group"
  453. node_tree:PointerProperty(type=NodeTree, poll=poll_node_tree, update=node_tree_prop_update,)
  454. is_updating:BoolProperty(default=False)
  455. def draw_label(self):
  456. if self.node_tree is None:
  457. return "Node Group"
  458. else:
  459. return self.node_tree.name
  460. def draw_buttons(self, context, layout):
  461. group_draw_buttons(self, context, layout)
  462. def update(self):
  463. if self.node_tree is None:
  464. return
  465. if self.is_updating: # update() can be called from update() and that leads to an infinite loop.
  466. return # so we check if an update is currently running.
  467. live_update = self.id_data.do_live_update
  468. self.is_updating = True
  469. try:
  470. node_group_update(self)
  471. finally: # we need to reset this regardless of whether or not the operation succeeds!
  472. self.is_updating = False
  473. self.id_data.do_live_update = live_update # ensure this remains the same
  474. class GraphError(Exception):
  475. pass
  476. def get_signature_from_edited_tree(node, context):
  477. sig_path=[None,]
  478. for item in context.space_data.path[:-1]:
  479. sig_path.append(item.node_tree.nodes.active.name)
  480. return tuple(sig_path+[node.name])
  481. def poll_node_tree_schema(self, object):
  482. if isinstance(object, SchemaTree):
  483. return True
  484. return False
  485. # TODO tiny UI problem - inserting new links into the tree will not place them in the right place.
  486. class SchemaGroup(Node, MantisUINode):
  487. bl_idname = "MantisSchemaGroup"
  488. bl_label = "Node Schema"
  489. node_tree:PointerProperty(type=NodeTree, poll=poll_node_tree_schema, update=node_tree_prop_update,)
  490. is_updating:BoolProperty(default=False)
  491. def draw_buttons(self, context, layout):
  492. group_draw_buttons(self, context, layout)
  493. def draw_label(self):
  494. if self.node_tree is None:
  495. return "Schema Group"
  496. else:
  497. return self.node_tree.name
  498. def update(self):
  499. if self.is_updating: # update() can be called from update() and that leads to an infinite loop.
  500. return # so we check if an update is currently running.
  501. if self.node_tree is None:
  502. return
  503. live_update = self.id_data.do_live_update
  504. self.is_updating = True
  505. try:
  506. node_group_update(self)
  507. # reset things if necessary:
  508. if self.node_tree:
  509. if len(self.inputs) == 0:
  510. self.inputs.new("UnsignedIntSocket", "Schema Length", identifier='Schema Length')
  511. if self.inputs[-1].identifier != "__extend__":
  512. self.inputs.new("WildcardSocket", "", identifier="__extend__")
  513. finally: # we need to reset this regardless of whether or not the operation succeeds!
  514. self.is_updating = False
  515. self.id_data.do_live_update = live_update # ensure this remains the same
  516. NODES_REMOVED=["xFormRootNode"]
  517. # Node bl_idname, # Socket Name
  518. SOCKETS_REMOVED=[("UtilityDriverVariable", "Transform Channel"),
  519. ("xFormRootNode","World Out"),
  520. ("UtilitySwitch","xForm"),
  521. ("LinkDrivenParameter", "Enable")]
  522. # Node Class #Prior bl_idname # prior name # new bl_idname # new name, # Multi
  523. SOCKETS_RENAMED=[ ("LinkDrivenParameter", "DriverSocket", "Driver", "FloatSocket", "Value", False),
  524. ("DeformerHook", "IntSocket", "Index", "UnsignedIntSocket", "Point Index", False)]
  525. # NODE CLASS NAME IN_OUT SOCKET TYPE SOCKET NAME INDEX MULTI DEFAULT
  526. SOCKETS_ADDED=[("DeformerMorphTargetDeform", 'INPUT', 'BooleanSocket', "Use Shape Key", 1, False, False),
  527. ("DeformerMorphTargetDeform", 'INPUT', 'BooleanSocket', "Use Offset", 2, False, True),
  528. ("UtilityFCurve", 'INPUT', "eFCrvExtrapolationMode", "Extrapolation Mode", 0, False, 'CONSTANT'),
  529. ("LinkCopyScale", 'INPUT', "BooleanSocket", "Additive", 3, False, False),
  530. ("DeformerHook", 'INPUT', "FloatFactorSocket", "Influence",3, False, 1.0),
  531. ("DeformerHook", 'INPUT', "UnsignedIntSocket", "Spline Index", 2, False, 0),
  532. ("DeformerHook", 'INPUT', "BooleanSocket", "Auto-Bezier", 5, False, True),
  533. ("UtilityCompare", 'INPUT', "EnumCompareOperation", "Comparison", 0, False, 'EQUAL'),
  534. ("UtilityMatrixFromCurve", 'INPUT', "UnsignedIntSocket", "Spline Index", 1, False, 0),
  535. ("UtilityMatricesFromCurve", 'INPUT', "UnsignedIntSocket", "Spline Index", 1, False, 0),
  536. ("UtilityPointFromCurve", 'INPUT', "UnsignedIntSocket", "Spline Index", 1, False, 0),
  537. ("LinkCopyScale", 'INPUT', "FloatFactorSocket", "Power", 5, False, 1.0),
  538. ]
  539. # replace names with bl_idnames for reading the tree and solving schemas.
  540. replace_types = ["NodeGroupInput", "NodeGroupOutput", "SchemaIncomingConnection",
  541. "SchemaArrayInput", "SchemaArrayInputAll", "SchemaConstInput", "SchemaConstOutput",
  542. "SchemaIndex", "SchemaOutgoingConnection", "SchemaArrayOutput","SchemaArrayInputGet",
  543. ]
  544. # anything that gets properties added in the graph... this is a clumsy approach but I need to watch for this
  545. # in schema generation and this is the easiest way to do it for now.
  546. custom_props_types = ["LinkArmature", "UtilityKeyframe", "UtilityFCurve", "UtilityDriver", "xFormBone"]
  547. # filters for determining if a link is a hierarchy link or a non-hierarchy (cyclic) link.
  548. from_name_filter = ["Driver",]
  549. to_name_filter = [
  550. "Custom Object xForm Override",
  551. "Custom Object",
  552. "Deform Bones",
  553. ]
  554. # nodes that must be solved as if they were Schema because they send arrays out.
  555. array_output_types = [
  556. 'UtilityArrayGet', 'UtilityKDChoosePoint', 'UtilityKDChooseXForm',
  557. ]
  558. def can_remove_socket_for_autogen(node, socket) -> bool:
  559. """ Whether to enable socket removal optimization for the socket
  560. This should be disallowed if e.g. it is a custom property.
  561. """
  562. if node.socket_templates:
  563. for s_template in node.socket_templates:
  564. if s_template.name == socket:
  565. # raise NotImplementedError
  566. return True
  567. elif node.node_type == 'UTILITY':
  568. return True # HACK because most utilities don't have socket templates yet
  569. return False
  570. # TODO:
  571. # - get the execution context in the execution code
  572. # - from there, begin to use it for stuff I can't do without it
  573. # - and slowly start transferring stuff to it
  574. # The Mantis Overlay class is used to store node-tree specific information
  575. # such as inputs and outputs
  576. # used for e.g. allowing strings to pass as $variables in node names
  577. class MantisOverlay():
  578. def __init__( self, parent, inputs, outputs, ):
  579. pass
  580. # The MantisExecutionContext class is used to store the execution-specific variables
  581. # that are used when executing the tree
  582. # Importantly, it is NOT used to store variables between solutions, these belong to the
  583. # tree itself.
  584. class MantisExecutionContext():
  585. def __init__(
  586. self,
  587. base_tree,
  588. ):
  589. self.base_tree = base_tree
  590. self.execution_id = base_tree.execution_id
  591. self.b_objects={} # objects created by Mantis during execution
  592. class MantisNode:
  593. """
  594. This class contains the basic interface for a Mantis Node.
  595. A MantisNode is used internally by Mantis to represent the final evaluated node graph.
  596. It gets generated with data from a MantisUINode when the graph is read.
  597. """
  598. def __init__(self, signature : tuple,
  599. base_tree : bpy.types.NodeTree,
  600. socket_templates : list[MantisSocketTemplate]=[],):
  601. self.base_tree=base_tree
  602. self.signature = signature
  603. self.ui_signature = signature
  604. self.inputs = MantisNodeSocketCollection(node=self, is_input=True)
  605. self.outputs = MantisNodeSocketCollection(node=self, is_input=False)
  606. self.parameters, self.drivers = {}, {}; self.bObject=None
  607. self.node_type='UNINITIALIZED'
  608. self.hierarchy_connections, self.connections = [], []
  609. self.hierarchy_dependencies, self.dependencies = [], []
  610. self.prepared, self.executed = False, False
  611. self.execution_prepared = False
  612. # the above is for tracking prep state in execution, so that I can avoid preparing nodes
  613. # again without changing the readtree code much.
  614. self.socket_templates = socket_templates
  615. self.mContext = None # for now I am gonna set this at runtime
  616. # I know it isn't "beautiful OOP" or whatever, but it is just easier
  617. # code should be simple and do things in the simplest way.
  618. # in the future I can refactor it, but it will require changes to 100+
  619. # classes, instead of adding about 5 lines of code elsewhere.
  620. if self.socket_templates:
  621. self.init_sockets()
  622. @property
  623. def name(self):
  624. return self.ui_signature[-1]
  625. @property
  626. def bl_idname(self): # this and the above exist solely to maintain interface w/bpy.types.Node
  627. from .utilities import get_node_prototype
  628. return get_node_prototype(self.ui_signature, self.base_tree).bl_idname
  629. def reset_execution(self) -> None:
  630. """ Reset the node for additional execution without re-building the tree."""
  631. self.drivers={}; self.bObject=None
  632. self.executed = False
  633. self.execution_prepared = False
  634. def init_sockets(self) -> None:
  635. self.inputs.init_sockets(self.socket_templates)
  636. self.outputs.init_sockets(self.socket_templates)
  637. def init_parameters(self, additional_parameters = {}) -> None:
  638. for socket in self.inputs:
  639. self.parameters[socket.name] = None
  640. for socket in self.outputs:
  641. self.parameters[socket.name] = None
  642. for key, value in additional_parameters.items():
  643. self.parameters[key]=value
  644. def gen_property_socket_map(self) -> dict:
  645. props_sockets = {}
  646. for s_template in self.socket_templates:
  647. if not s_template.blender_property:
  648. continue
  649. if isinstance(s_template.blender_property, str):
  650. props_sockets[s_template.blender_property]=(s_template.name, s_template.default_value)
  651. elif isinstance(s_template.blender_property, (tuple, list)):
  652. for index, sub_prop in enumerate(s_template.blender_property):
  653. props_sockets[sub_prop]=( (s_template.name, index),s_template.default_value[index] )
  654. return props_sockets
  655. def set_traverse(self, traversal_pairs = [(str, str)]) -> None:
  656. for (a, b) in traversal_pairs:
  657. self.inputs[a].set_traverse_target(self.outputs[b])
  658. self.outputs[b].set_traverse_target(self.inputs[a])
  659. def flush_links(self) -> None:
  660. for inp in self.inputs.values():
  661. inp.flush_links()
  662. for out in self.outputs.values():
  663. out.flush_links()
  664. def update_socket_value(self, blender_property, value) -> bool:
  665. change_handled=False
  666. if self.node_type == 'LINK':
  667. if len(self.bObject) == 0: # - there are no downstream xForms
  668. return True # so there is nothing to do here
  669. for b_ob in self.bObject:
  670. try:
  671. setattr(b_ob, blender_property, value)
  672. change_handled=True
  673. except Exception as e:
  674. print("Failed to update mantis socket because of %s" % e,
  675. "Updating tree instead.")
  676. else:
  677. try:
  678. b_ob = self.bObject
  679. if self.node_type == 'XFORM': # HACK
  680. b_ob = self.bGetObject()
  681. setattr(b_ob, blender_property, value)
  682. change_handled=True
  683. except Exception as e:
  684. print("Failed to update mantis socket because of %s" % e,
  685. "Updating tree instead.")
  686. return change_handled
  687. def ui_modify_socket(self, ui_socket, socket_name=None) -> bool:
  688. """ Handle changes in the node's UI. Updates the rig if possible."""
  689. # Always update the node's data
  690. change_handled=False
  691. if socket_name is None: socket_name = ui_socket.name
  692. value = ui_socket.default_value
  693. if socket_name == 'Enable': value = not value
  694. try:
  695. self.parameters[ui_socket.name]=value
  696. except KeyError:
  697. prRed(f"Unhandled change occured in socket {ui_socket.name} in node"
  698. f" {ui_socket.node.name} in tree {ui_socket.node.id_data.name}.")
  699. for s_template in self.socket_templates:
  700. if s_template.name==ui_socket.name:
  701. change_handled = True
  702. if not s_template.blender_property: return False
  703. elif isinstance(s_template.blender_property, str):
  704. change_handled &= self.update_socket_value(
  705. s_template.blender_property, value)
  706. else: # it is a tuple
  707. for i, prop in enumerate(s_template.blender_property):
  708. try:
  709. change_handled &= self.update_socket_value(
  710. prop, value[i])
  711. except IndexError:
  712. prRed(f"{ui_socket.name} does not have enough values to unpack"
  713. " to update the Mantis tree. Please report this as a bug.")
  714. change_handled=False
  715. break # we don't have to look through any more socket templates
  716. return change_handled
  717. # the goal here is to tag the node as unprepared
  718. # but some nodes are always prepared, so we have to kick it forward.
  719. def reset_execution_recursive(self):
  720. self.reset_execution()
  721. if self.prepared==False: return # all good from here
  722. for conn in self.hierarchy_connections:
  723. conn.reset_execution_recursive()
  724. def evaluate_input(self, input_name, index=0) -> Any:
  725. from .node_container_common import trace_single_line
  726. if not (self.inputs.get(input_name)): # get the named parameter if there is no input
  727. return self.parameters.get(input_name) # this will return None if the parameter does not exist.
  728. # this trace() should give a key error if there is a problem
  729. # it is NOT handled here because it should NOT happen - so I want the error message.
  730. trace = trace_single_line(self, input_name, index)
  731. prop = trace[0][-1].parameters[trace[1].name] #trace[0] = the list of traced nodes; read its parameters
  732. return prop
  733. def fill_parameters(self, ui_node=None) -> None:
  734. from .utilities import get_node_prototype
  735. from .node_container_common import get_socket_value
  736. if not ui_node:
  737. if ( (self.signature[0] in ["MANTIS_AUTOGENERATED", "SCHEMA_AUTOGENERATED" ]) or
  738. (self.signature[-1] in ["NodeGroupOutput", "NodeGroupInput"]) ): # I think this is harmless
  739. return None
  740. else:
  741. ui_node = get_node_prototype(self.signature, self.base_tree)
  742. if not ui_node:
  743. raise RuntimeError(wrapRed("No node prototype found for... %s" % ( [self.base_tree] + list(self.signature[1:]) ) ) )
  744. for key in self.parameters.keys():
  745. node_socket = ui_node.inputs.get(key)
  746. if self.parameters[key] is not None: # the parameters are usually initialized as None.
  747. continue # will be filled by the node itself
  748. if not node_socket: #maybe the node socket has no name
  749. if ( ( len(ui_node.inputs) == 0) and ( len(ui_node.outputs) == 1) ):
  750. node_socket = ui_node.outputs[0] # this is a simple input node.
  751. elif key == 'Name': # for Links we just use the Node Label, or if there is no label, the name.
  752. self.parameters[key] = ui_node.label if ui_node.label else ui_node.name
  753. continue
  754. if node_socket:
  755. if node_socket.bl_idname in ['RelationshipSocket', 'xFormSocket']: continue
  756. elif node_socket.is_linked and (not node_socket.is_output): continue
  757. # we will get the value from the link, because this is a linked input port.
  758. # very importantly, we do not pass linked outputs
  759. # fill these because they are probably Input nodes.
  760. elif hasattr(node_socket, "default_value"):
  761. if (value := get_socket_value(node_socket)) is not None:
  762. self.parameters[key] = value
  763. else:
  764. raise RuntimeError(wrapRed("No value found for " + self.__repr__() + " when filling out node parameters for " + ui_node.name + "::"+node_socket.name))
  765. # I don't think this works! but I like the idea
  766. def call_on_all_ancestors(self, *args, **kwargs):
  767. """Resolve the dependencies of this node with the named method and its arguments.
  768. First, dependencies are discovered by walking backwards through the tree. Once the root
  769. nodes are discovered, the method is called by each node in dependency order.
  770. The first argument MUST be the name of the method as a string.
  771. """
  772. prGreen(self)
  773. if args[0] == 'call_on_all_ancestors': raise RuntimeError("Very funny!")
  774. from .utilities import get_all_dependencies
  775. from collections import deque
  776. # get all dependencies by walking backward through the tree.
  777. all_dependencies = get_all_dependencies(self)
  778. # get just the roots
  779. can_solve = deque(filter(lambda a : len(a.hierarchy_connections) == 0, all_dependencies))
  780. solved = set()
  781. while can_solve:
  782. node = can_solve.pop()
  783. print(node)
  784. method = getattr(node, args[0])
  785. method(*args[0:], **kwargs)
  786. solved.add(node)
  787. can_solve.extendleft(filter(lambda a : a in all_dependencies, node.hierarchy_connections))
  788. if self in solved:
  789. break
  790. return
  791. # gets targets for constraints and deformers and should handle all cases
  792. def get_target_and_subtarget(self, constraint_or_deformer, input_name = "Target"):
  793. from bpy.types import PoseBone, Object, SplineIKConstraint
  794. subtarget = ''; target = self.evaluate_input(input_name)
  795. if target:
  796. if not hasattr(target, "bGetObject"):
  797. if hasattr(constraint_or_deformer, 'name'):
  798. name = constraint_or_deformer.name
  799. else:
  800. name = 'NAME NOT FOUND'
  801. prRed(f"No {input_name} target found for {name} in {self} because there is no connected node, or node is wrong type")
  802. return
  803. if (isinstance(target.bGetObject(), PoseBone)):
  804. subtarget = target.bGetObject().name
  805. target = target.bGetParentArmature()
  806. elif (isinstance(target.bGetObject(), Object) ):
  807. target = target.bGetObject()
  808. else:
  809. raise RuntimeError("Cannot interpret constraint or deformer target!")
  810. if (isinstance(constraint_or_deformer, SplineIKConstraint)):
  811. if target and target.type not in ["CURVE"]:
  812. raise GraphError(wrapRed("Error: %s requires a Curve input, not %s" %
  813. (self, type(target))))
  814. constraint_or_deformer.target = target# don't get a subtarget
  815. if (input_name == 'Pole Target'):
  816. constraint_or_deformer.pole_target, constraint_or_deformer.pole_subtarget = target, subtarget
  817. else:
  818. if hasattr(constraint_or_deformer, "target"):
  819. constraint_or_deformer.target = target
  820. if hasattr(constraint_or_deformer, "object"):
  821. constraint_or_deformer.object = target
  822. if hasattr(constraint_or_deformer, "subtarget"):
  823. constraint_or_deformer.subtarget = subtarget
  824. def bPrepare(self, bContext=None):
  825. return
  826. def bExecute(self, bContext=None):
  827. return
  828. def bFinalize(self, bContext=None):
  829. return
  830. if environ.get("DOERROR"):
  831. def __repr__(self):
  832. return self.signature.__repr__()
  833. else:
  834. def __repr__(self):
  835. return self.ui_signature.__repr__()
  836. # do I need this and the link class above?
  837. class DummyLink:
  838. #gonna use this for faking links to keep the interface consistent
  839. def __init__(self, from_socket, to_socket, nc_from=None, nc_to=None, original_from=None, multi_input_sort_id=0):
  840. self.from_socket = from_socket
  841. self.to_socket = to_socket
  842. self.nc_from = nc_from
  843. self.nc_to = nc_to
  844. self.multi_input_sort_id = multi_input_sort_id
  845. # self.from_node = from_socket.node
  846. # self.to_node = to_socket.node
  847. if (original_from):
  848. self.original_from = original_from
  849. else:
  850. self.original_from = self.from_socket
  851. def __repr__(self):
  852. return(self.nc_from.__repr__()+":"+self.from_socket.name + " -> " + self.nc_to.__repr__()+":"+self.to_socket.name)
  853. def detect_hierarchy_link(from_node, from_socket, to_node, to_socket,):
  854. if to_node.node_type in ['DUMMY_SCHEMA', 'SCHEMA']:
  855. return False #TODO: find out if filtering SCHEMA types is wise
  856. if (from_socket in from_name_filter) or (to_socket in to_name_filter):
  857. return False
  858. # if from_node.__class__.__name__ in ["UtilityCombineVector", "UtilityCombineThreeBool"]:
  859. # return False
  860. return True
  861. class NodeLink:
  862. from_node = None
  863. from_socket = None
  864. to_node = None
  865. to_socket = None
  866. def __init__(self, from_node, from_socket, to_node, to_socket, multi_input_sort_id=0):
  867. if from_node.signature == to_node.signature:
  868. raise RuntimeError("Cannot connect a node to itself.")
  869. self.from_node = from_node
  870. self.from_socket = from_socket
  871. self.to_node = to_node
  872. self.to_socket = to_socket
  873. self.from_node.outputs[self.from_socket].links.append(self)
  874. # it is the responsibility of the node that uses these links to sort them correctly based on the sort_id
  875. self.multi_input_sort_id = multi_input_sort_id
  876. self.to_node.inputs[self.to_socket].links.append(self)
  877. self.is_hierarchy = detect_hierarchy_link(from_node, from_socket, to_node, to_socket,)
  878. self.is_alive = True
  879. def __repr__(self):
  880. return self.from_node.outputs[self.from_socket].__repr__() + " --> " + self.to_node.inputs[self.to_socket].__repr__()
  881. # link_string = # if I need to colorize output for debugging.
  882. # if self.is_hierarchy:
  883. # return wrapOrange(link_string)
  884. # else:
  885. # return wrapWhite(link_string)
  886. def die(self):
  887. self.is_alive = False
  888. self.to_node.inputs[self.to_socket].flush_links()
  889. self.from_node.outputs[self.from_socket].flush_links()
  890. def insert_node(self, middle_node, middle_node_in, middle_node_out, re_init_hierarchy = True):
  891. to_node = self.to_node
  892. to_socket = self.to_socket
  893. self.to_node = middle_node
  894. self.to_socket = middle_node_in
  895. middle_node.outputs[middle_node_out].connect(to_node, to_socket)
  896. if re_init_hierarchy:
  897. from .utilities import init_connections, init_dependencies
  898. init_connections(self.from_node)
  899. init_connections(middle_node)
  900. init_dependencies(middle_node)
  901. init_dependencies(to_node)
  902. class NodeSocket:
  903. # @property # this is a read-only property.
  904. # def is_linked(self):
  905. # return bool(self.links)
  906. def __init__(self, is_input = False,
  907. node = None, name = None,
  908. traverse_target = None):
  909. self.can_traverse = False # to/from the other side of the parent node
  910. self.traverse_target = None
  911. self.node = node
  912. self.name = name
  913. self.is_input = is_input
  914. self.links = []
  915. self.is_linked = False
  916. if (traverse_target):
  917. self.can_traverse = True
  918. def connect(self, node, socket, sort_id=0):
  919. if (self.is_input):
  920. to_node = self.node; from_node = node
  921. to_socket = self.name; from_socket = socket
  922. else:
  923. from_node = self.node; to_node = node
  924. from_socket = self.name; to_socket = socket
  925. from_node.outputs[from_socket].is_linked = True
  926. to_node.inputs[to_socket].is_linked = True
  927. if sort_id == 0: # otherwise it is an array and dupe links are valid.
  928. for l in from_node.outputs[from_socket].links:
  929. if l.to_node==to_node and l.to_socket==to_socket:
  930. return None
  931. new_link = NodeLink(
  932. from_node,
  933. from_socket,
  934. to_node,
  935. to_socket,
  936. sort_id)
  937. return new_link
  938. def set_traverse_target(self, traverse_target):
  939. self.traverse_target = traverse_target
  940. self.can_traverse = True
  941. def flush_links(self):
  942. """ Removes dead links from this socket."""
  943. self.links = [l for l in self.links if l.is_alive]
  944. self.links.sort(key=lambda a : -a.multi_input_sort_id)
  945. self.is_linked = bool(self.links)
  946. @property
  947. def is_connected(self):
  948. return len(self.links)>0
  949. def __repr__(self):
  950. return self.node.__repr__() + "::" + self.name
  951. class MantisNodeSocketCollection(dict):
  952. def __init__(self, node, is_input=False):
  953. self.is_input = is_input
  954. self.node = node
  955. def init_sockets(self, sockets):
  956. for socket in sockets:
  957. if isinstance(socket, str):
  958. self[socket] = NodeSocket(is_input=self.is_input, name=socket, node=self.node)
  959. elif isinstance(socket, MantisSocketTemplate):
  960. if socket.is_input != self.is_input: continue
  961. self[socket.name] = NodeSocket(is_input=self.is_input, name=socket.name, node=self.node)
  962. else:
  963. raise RuntimeError(f"NodeSocketCollection keys must be str or MantisSocketTemplate, not {type(socket)}")
  964. def __delitem__(self, key):
  965. """Deletes a node socket by name, and all its links."""
  966. socket = self[key]
  967. for l in socket.links:
  968. l.die()
  969. super().__delitem__(key)
  970. def __iter__(self):
  971. """Makes the class iterable"""
  972. return iter(self.values())