drivers.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. from .utilities import (prRed, prGreen, prPurple, prWhite,
  2. prOrange,
  3. wrapRed, wrapGreen, wrapPurple, wrapWhite,
  4. wrapOrange,)
  5. ##########################################################################
  6. # Drivers!
  7. ##########################################################################
  8. # SO: the idea is that the driver's input is a Python dictionary
  9. # with all of the requisite information to build the driver
  10. # I need a generic function to create the driver
  11. # EXAMPLE INPUT:
  12. # example = {"owner":None,
  13. # "prop":None,
  14. # "ind":-1,
  15. # "type":"AVERAGE",
  16. # "vars":[{"id":None,
  17. # "name":"a",
  18. # "type":"TRANSFORMS",
  19. # "space":'LOCAL_SPACE',
  20. # "channel":'LOC_Z',},],
  21. # "keys":[{"co":(0,0.5),
  22. # "interpolation": "BEZIER",
  23. # "handle_left_type": "AUTO_CLAMPED", #if AUTO then handle_left will be ignored
  24. # "handle_right_type": "AUTO_CLAMPED",
  25. # "type":"KEYFRAME",}, #display type
  26. # {"co":(-1,0),
  27. # "interpolation": "BEZIER",
  28. # "handle_left_type": "AUTO_CLAMPED",
  29. # "handle_right_type": "AUTO_CLAMPED",
  30. # "type":"KEYFRAME",},
  31. # {"co":(1,1),
  32. # "interpolation": "BEZIER",
  33. # "handle_left_type": "ALIGNED",
  34. # "handle_right_type": "ALIGNED",
  35. # "handle_left": (-0.4,0), #these are treated as offsets
  36. # "handle_right": ( 0.04,0), #only valid if interp. == BEZIER
  37. # "type":"KEYFRAME",},],
  38. # }
  39. #Explain this ^ ?
  40. #
  41. class MantisDriver(dict):
  42. pass
  43. def CreateDrivers(drivers):
  44. def brackets(s):
  45. return "[\""+s+"\"]"
  46. from bpy.types import Object, Key
  47. for driver in drivers:
  48. if (isinstance(driver["owner"], Object)):
  49. ob = driver["owner"]
  50. else: # Pose Bone:
  51. ob = driver["owner"].id_data
  52. if isinstance(driver["owner"], Key):
  53. fc = ob.driver_add(driver["prop"])
  54. else:
  55. fc = ob.driver_add(driver["owner"].path_from_id(driver["prop"]), driver["ind"])
  56. drv = fc.driver
  57. try: # annoyingly, this initializes with a modifier
  58. fc.modifiers.remove(fc.modifiers[0])
  59. except IndexError: #haven't seen this happen, but should handle
  60. pass # perhaps this has been fixed for 3.0?
  61. drv.type = driver["type"]
  62. if (expr := driver.get("expression")) and isinstance(expr, str):
  63. drv.expression = expr
  64. # logic for handling type can go here
  65. # start by clearing
  66. while (len(drv.variables) > 0):
  67. v = drv.variables[0]
  68. dVar = drv.variables.remove(v)
  69. for v in driver["vars"]:
  70. pose_bone = False
  71. bone = ''; target2bone = ''
  72. vob, target2ob = None, None
  73. if (isinstance(v["owner"], Object)):
  74. vob = v["owner"]
  75. else:
  76. pose_bone = True
  77. vob = v["owner"].id_data
  78. bone = v["owner"].name
  79. #
  80. if "xForm 2" in v.keys() and v["xForm 2"]:
  81. if (isinstance(v["xForm 2"], Object)):
  82. target2ob = v["xForm 2"]
  83. else:
  84. target2ob = v["xForm 2"].id_data
  85. target2bone = v["xForm 2"].name
  86. dVar = drv.variables.new()
  87. dVar.name = v["name"]
  88. dVar.type = v["type"]
  89. #for now, assume this is always true:
  90. #dVar.targets[0].id_type = "OBJECT"
  91. #it's possible to use other datablocks, but this is not commonly done
  92. #actually, it looks like Blender figures this out for me.
  93. dVar.targets[0].id = vob
  94. dVar.targets[0].bone_target = bone
  95. if len(dVar.targets) > 1:
  96. dVar.targets[1].id = target2ob
  97. dVar.targets[1].bone_target = target2bone
  98. if (dVar.type == "TRANSFORMS"):
  99. dVar.targets[0].transform_space = v["space"]
  100. dVar.targets[0].transform_type = v["channel"]
  101. if (dVar.type == 'SINGLE_PROP'):
  102. if pose_bone:
  103. stub = "pose.bones[\""+v["owner"].name+"\"]"
  104. if (hasattr( v["owner"], v["prop"] )):
  105. dVar.targets[0].data_path = stub + "."+ (v["prop"])
  106. else:
  107. dVar.targets[0].data_path = stub + brackets(v["prop"])
  108. else:
  109. if (hasattr( v["owner"], v["prop"] )):
  110. dVar.targets[0].data_path = (v["prop"])
  111. else:
  112. dVar.targets[0].data_path = brackets(v["prop"])
  113. # setup keyframe points
  114. kp = fc.keyframe_points
  115. for key in driver["keys"]:
  116. k = kp.insert(frame=key["co"][0], value = key["co"][1],)
  117. k.interpolation = key["interpolation"]
  118. if (key["interpolation"] == 'BEZIER'):
  119. k.handle_left_type = key["handle_left_type" ]
  120. k.handle_right_type = key["handle_right_type"]
  121. if (k.handle_left_type in ("ALIGNED", "VECTOR", "FREE")):
  122. k.handle_left = (k.co[0] + key["handle_left"][0], k.co[1] + key["handle_left"][1])
  123. if (k.handle_right_type in ("ALIGNED", "VECTOR", "FREE")):
  124. k.handle_right = (k.co[0] + key["handle_right"][0], k.co[1] + key["handle_right"][1])
  125. k.type = key["type"]