drivers.py 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  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. class MantisDriver(dict):
  40. pass
  41. def CreateDrivers(drivers):
  42. def brackets(s):
  43. return "[\""+s+"\"]"
  44. from bpy.types import Object, Key
  45. for driver in drivers:
  46. if (isinstance(driver["owner"], Object)):
  47. ob = driver["owner"]
  48. else: # Pose Bone:
  49. ob = driver["owner"].id_data
  50. if isinstance(driver["owner"], Key):
  51. fc = ob.driver_add(driver["prop"])
  52. else:
  53. fc = ob.driver_add(driver["owner"].path_from_id(driver["prop"]), driver["ind"])
  54. drv = fc.driver
  55. try: # annoyingly, this initializes with a modifier
  56. fc.modifiers.remove(fc.modifiers[0])
  57. except IndexError: #haven't seen this happen, but should handle
  58. pass # perhaps this has been fixed for 3.0?
  59. drv.type = driver["type"]
  60. if (expr := driver.get("expression")) and isinstance(expr, str):
  61. drv.expression = expr
  62. fc.extrapolation = "CONSTANT"
  63. if (extrapolation_mode := driver.get("extrapolation")) in ("CONSTANT", "LINEAR"):
  64. fc.extrapolation = extrapolation_mode
  65. else:
  66. prRed(f"Extrapolation Mode in driver has incorrect data: {extrapolation_mode}")
  67. # logic for handling type can go here
  68. # start by clearing
  69. while (len(drv.variables) > 0):
  70. v = drv.variables[0]
  71. dVar = drv.variables.remove(v)
  72. for v in driver["vars"]:
  73. pose_bone = False
  74. bone = ''; target2bone = ''
  75. vob, target2ob = None, None
  76. if (isinstance(v["owner"], Object)):
  77. vob = v["owner"]
  78. else:
  79. pose_bone = True
  80. vob = v["owner"].id_data
  81. bone = v["owner"].name
  82. #
  83. if "xForm 2" in v.keys() and v["xForm 2"]:
  84. if (isinstance(v["xForm 2"], Object)):
  85. target2ob = v["xForm 2"]
  86. else:
  87. target2ob = v["xForm 2"].id_data
  88. target2bone = v["xForm 2"].name
  89. dVar = drv.variables.new()
  90. dVar.name = v["name"]
  91. dVar.type = v["type"]
  92. #for now, assume this is always true:
  93. #dVar.targets[0].id_type = "OBJECT"
  94. #it's possible to use other datablocks, but this is not commonly done
  95. #actually, it looks like Blender figures this out for me.
  96. dVar.targets[0].id = vob
  97. dVar.targets[0].bone_target = bone
  98. if len(dVar.targets) > 1:
  99. dVar.targets[1].id = target2ob
  100. dVar.targets[1].bone_target = target2bone
  101. if (dVar.type == "TRANSFORMS"):
  102. dVar.targets[0].transform_space = v["space"]
  103. dVar.targets[0].transform_type = v["channel"]
  104. if (dVar.type == 'SINGLE_PROP'):
  105. if pose_bone:
  106. stub = "pose.bones[\""+v["owner"].name+"\"]"
  107. dVar.targets[0].data_path = stub + brackets(v["prop"])
  108. if (hasattr( v["owner"], v["prop"] )):
  109. dVar.targets[0].data_path = stub + "."+ (v["prop"])
  110. # else: # the property may be added later.
  111. # TODO BUG I want a guarantee that this property is already there.
  112. else:
  113. if (hasattr( v["owner"], v["prop"] )):
  114. dVar.targets[0].data_path = (v["prop"])
  115. else:
  116. dVar.targets[0].data_path = brackets(v["prop"])
  117. # setup keyframe points
  118. kp = fc.keyframe_points
  119. for key in driver["keys"]:
  120. k = kp.insert(frame=key["co"][0], value = key["co"][1],)
  121. k.interpolation = key["interpolation"]
  122. if (key["interpolation"] == 'BEZIER'):
  123. k.handle_left_type = key["handle_left_type" ]
  124. k.handle_right_type = key["handle_right_type"]
  125. if (k.handle_left_type in ("ALIGNED", "VECTOR", "FREE")):
  126. k.handle_left = (k.co[0] + key["handle_left"][0], k.co[1] + key["handle_left"][1])
  127. if (k.handle_right_type in ("ALIGNED", "VECTOR", "FREE")):
  128. k.handle_right = (k.co[0] + key["handle_right"][0], k.co[1] + key["handle_right"][1])
  129. k.type = key["type"]