# Π‘ΠΎΠ·Π΄Π°Π½ΠΈΠ΅ ΡΠ»ΠΎΠ²Π°ΡΡ Ρ Π³ΡΠ°ΡΠΈΠΊΠΎΠΌ FCurves ΡΠΈΠΏΠ° location (ΠΏΠΎΠ·ΠΈΡΠΈΡ)
fcus = {}
for fcu in action.fcurves:
if fcu.data_path == 'location':
fcus[fcu.array_index] = fcu
print(fcus.items())
# ΠΠΎΠ±Π°Π²Π»Π΅Π½ΠΈΠ΅ Π½ΠΎΠ²ΡΡ ΠΊΠ»ΡΡΠ΅Π²ΡΡ ΡΠΎΡΠ΅ΠΊ ΠΊ x ΠΈ z
kpts_x = fcus[0].keyframe_points
kpts_z = fcus[2].keyframe_points
(x0,y0,z0) = origin
omega = 2*math.pi/20
z *= 0.67
for t in range(101, 201):
xt = 20 + 0.2*(t-101)
zt = z*(1-math.cos(omega*(t - 101)))
z *= 0.98
kpts_z.insert(t, zt+z0, options={'FAST'})
kpts_x.insert(t, xt+x0)
# ΠΠ·ΠΌΠ΅Π½Π΅Π½ΠΈΠ΅ ΡΠΈΠΏΠ° ΡΠΊΡΡΡΠ°ΠΏΠΎΠ»ΡΡΠΈΠΈ ΠΈ ΠΈΠ½ΡΠ΅ΡΠΏΠΎΠ»ΡΡΠΈΠΈ
# Π΄Π»Ρ ΠΊΡΠΈΠ²ΠΎΠΉ X Π½Π° Π»ΠΈΠ½Π΅ΠΉΠ½ΡΠΉ
fcus[0].extrapolation = 'LINEAR'
for kp in kpts_x:
kp.interpolation = 'LINEAR'
# ΠΠΎΠ·ΠΈΡΠΈΡ Y - ΠΊΠΎΠ½ΡΡΠ°Π½ΡΠ° ΠΈ ΠΌΠΎΠΆΠ΅Ρ Π±ΡΡΡ ΡΠ΄Π°Π»Π΅Π½Π°
action.fcurves.remove(fcus[1])
bpy.ops.object.paths_calculate()
return
if __name__ == "__main__":
run((0,0,10))
bpy.ops.screen.animation_play(reverse=False, sync=False)
ΠΡΠ° ΠΏΡΠΎΠ³ΡΠ°ΠΌΠΌΠ° ΡΠΎΠ·Π΄Π°Π΅Ρ Π°ΡΠΌΠ°ΡΡΡΡ Ρ Π΄Π²ΡΠΌΡ ΠΊΠΎΡΡΡΠΌΠΈ, ΠΊΠΎΡΠΎΡΡΠ΅ Π²ΡΠ°ΡΠ°ΡΡΡΡ ΠΏΠΎ Π½Π΅ΠΊΠΎΡΠΎΡΡΠΌ ΡΠ»ΠΎΠΆΠ½ΡΠΌ ΠΊΡΠΈΠ²ΡΠΌ.
#--------------------------------------------------
# File pose_action.py
#--------------------------------------------------
import bpy
import math
def run(origin):
# Π£ΡΡΠ°Π½ΠΎΠ²ΠΊΠ° Π½Π°ΡΠ°Π»Π° ΠΈ ΠΊΠΎΠ½ΡΠ° Π°Π½ΠΈΠΌΠ°ΡΠΈΠΈ
scn = bpy.context.scene
scn.frame_start = 1
scn.frame_end = 250
# Π‘ΠΎΠ·Π΄Π°Π½ΠΈΠ΅ Π°ΡΠΌΠ°ΡΡΡΡ ΠΈ ΠΎΠ±ΡΠ΅ΠΊΡΠ°
bpy.ops.object.armature_add()
ob = bpy.context.object
amt = ob.data
# ΠΠ΅ΡΠ΅ΠΈΠΌΠ΅Π½ΠΎΠ²Π°Π½ΠΈΠ΅ ΠΏΠ΅ΡΠ²ΠΎΠΉ ΠΊΠΎΡΡΠΈ ΠΈ ΡΠΎΠ·Π΄Π°Π½ΠΈΠ΅ Π²ΡΠΎΡΠΎΠΉ ΠΊΠΎΡΡΠΈ
bpy.ops.object.mode_set(mode='EDIT')
base = amt.edit_bones['Bone']
base.name = 'Base'
tip = amt.edit_bones.new('Tip')
tip.head = (0,0,1)
tip.tail = (0,0,2)
tip.parent = base
tip.use_connect = True
# Π£ΡΡΠ°Π½ΠΎΠ²ΠΊΠ° ΠΏΠΎΠ·ΠΈΡΠΈΠΈ ΠΎΠ±ΡΠ΅ΠΊΡΠ° Π² ΡΠ΅ΠΆΠΈΠΌΠ΅ ΠΎΠ±ΡΠ΅ΠΊΡΠΎΠ²
bpy.ops.object.mode_set(mode='OBJECT')
ob.location=origin
# Π£ΡΡΠ°Π½ΠΎΠ²ΠΊΠ° ΠΠΉΠ»Π΅ΡΠΎΠ²Π° ΡΠ΅ΠΆΠΈΠΌΠ° Π²ΡΠ°ΡΠ΅Π½ΠΈΡ (Euler ZYX)
bpy.ops.object.mode_set(mode='POSE')
pbase = ob.pose.bones['Base']
pbase.rotation_mode = 'ZYX'
ptip = ob.pose.bones['Tip']
ptip.rotation_mode = 'ZYX'
# ΠΡΡΠ°Π²ΠΊΠ° 26 ΠΊΠ»ΡΡΠ΅Π²ΡΡ ΠΊΠ°Π΄ΡΠΎΠ² Π΄Π»Ρ Π΄Π²ΡΡ Π²ΡΠ°ΡΠ΅Π½ΠΈΠΉ FCurves
# ΠΠΎΡΠ»Π΅Π΄Π½ΠΈΠΉ ΠΊΠ»ΡΡΠ΅Π²ΠΎΠΉ ΠΊΠ°Π΄Ρ Π±ΡΠ΄Π΅Ρ Π²ΠΎΠ²Π½Π΅ Π΄ΠΈΠΏΠ°Π·ΠΎΠ½Π° Π°Π½ΠΈΠΌΠ°ΡΠΈΠΈ
for n in range(26):
pbase.keyframe_insert(
'rotation_euler',
index=0,
frame=n,
group='Base')
ptip.keyframe_insert(
'rotation_euler',
index=2,
frame=n,
group='Tip')
# ΠΠΎΠ»ΡΡΠ΅Π½ΠΈΠ΅ FCurves ΠΈΠ· Π²Π½ΠΎΠ²Ρ ΡΠΎΠ·Π΄Π°Π½Π½ΠΎΠ³ΠΎ Π΄Π΅ΠΉΡΡΠ²ΠΈΡ
action = ob.animation_data.action
fcus = {}
for fcu in action.fcurves:
bone = fcu.data_path.split('"')[1]
fcus[(bone, fcu.array_index)] = fcu
# ΠΠΎΠ΄ΠΈΡΠΈΠΊΠ°ΡΠΈΡ ΠΊΠ»ΡΡΠ΅Π²ΡΡ ΡΠΎΡΠ΅ΠΊ
baseKptsRotX = fcus[('Base', 0)].keyframe_points
tipKptsRotZ = fcus[('Tip', 2)].keyframe_points
omega = 2*math.pi/250
for n in range(26):
t = 10*n
phi = omega*t
kp = baseKptsRotX[n]
kp.co = (t+1,phi+0.7*math.sin(phi))
kp.interpolation = 'LINEAR'
kp = tipKptsRotZ[n]
kp.co = (t+1, -3*phi+2.7*math.cos(2*phi))
kp.interpolation = 'LINEAR'
# ΠΡΡΠΈΡΠ»Π΅Π½ΠΈΠ΅ ΠΏΡΡΠ΅ΠΉ Π΄Π»Ρ ΠΏΠΎΠ· ΠΊΠΎΡΡΠ΅ΠΉ
bpy.ops.pose.select_all(action='SELECT')
bpy.ops.pose.paths_calculate()
return
if __name__ == "__main__":
run((10,0,0))
bpy.ops.screen.animation_play(reverse=False, sync=False)
ΠΡΠ° ΠΏΡΠΎΠ³ΡΠ°ΠΌΠΌΠ° ΡΠΎΠ·Π΄Π°Π΅Ρ ΡΠ»ΠΎΠΆΠ½ΠΎΠ΅ Π΄Π²ΠΈΠΆΠ΅Π½ΠΈΠ΅, ΠΏΠΎΡΠ»Π΅Π΄ΠΎΠ²Π°ΡΠ΅Π»ΡΠ½ΠΎ Π½Π°Π·Π½Π°ΡΠ°Ρ ΡΠΎΠ΄ΠΈΡΠ΅Π»Π΅ΠΌ Π½Π΅ΡΠΊΠΎΠ»ΡΠΊΠΎ ΠΏΡΡΡΡΡΠ΅ΠΊ ΠΎΡ ΠΎΠ΄Π½ΠΎΠΉ ΠΊ ΡΠ»Π΅Π΄ΡΡΡΠ΅ΠΉ, ΠΈ Π½Π°Π·Π½Π°ΡΠ°Ρ ΠΏΡΠΎΡΡΠΎΠ΅ Π²ΡΠ°ΡΠ΅Π½ΠΈΠ΅ Π΄Π»Ρ ΠΊΠ°ΠΆΠ΄ΠΎΠΉ ΠΈΠ· Π½ΠΈΡ .
#----------------------------------------------------------
# File epicycle.py
#----------------------------------------------------------
import bpy
import math from math
import pi
def createEpiCycle(origin):
periods = [1, 5, 8, 17]
radii = [1.0, 0.3, 0.5, 0.1]
axes = [0, 2, 1, 0]
phases = [0, pi/4, pi/2, 0]
# ΠΠΎΠ±Π°Π²Π»Π΅Π½ΠΈΠ΅ ΠΏΡΡΡΡΡΠ΅ΠΊ
scn = bpy.context.scene
empties = []
nEmpties = len(periods)
for n in range(nEmpties):
empty = bpy.data.objects.new('Empty_%d' % n, None)
scn.objects.link(empty)
empties.append(empty)
# ΠΠ°Π·Π½Π°ΡΠ΅Π½ΠΈΠ΅ ΠΊΠ°ΠΆΠ΄ΠΎΠΉ ΠΏΡΡΡΡΡΠΊΠ΅ ΡΠΎΠ΄ΠΈΡΠ΅Π»Ρ ΠΏΠΎΡΠ»Π΅Π΄ΠΎΠ²Π°ΡΠ΅Π»ΡΠ½ΠΎ
for n in range(1, nEmpties):
empties[n].parent = empties[n-1]
empties[n].location = (0, radii[n-1], 0)
# ΠΡΡΠ°Π²ΠΊΠ° Π΄Π²ΡΡ ΠΊΠ»ΡΡΠ΅Π²ΡΡ ΠΊΠ°Π΄ΡΠΎΠ² Π΄Π»Ρ ΠΊΠ°ΠΆΠ΄ΠΎΠΉ ΠΏΡΡΡΡΡΠΊΠΈ
for n in range(nEmpties):
empty = empties[n]
empty.keyframe_insert(
'rotation_euler',
index=axes[n],
frame=0,
group=empty.name)
empty.keyframe_insert(
'rotation_euler',
index=axes[n],
frame=periods[n],
group=empty.name)
fcu = empty.animation_data.action.fcurves[0]
print(empty, fcu.data_path, fcu.array_index)
kp0 = fcu.keyframe_points[0]
kp0.co = (0, phases[n])
kp0.interpolation = 'LINEAR'
kp1 = fcu.keyframe_points[1]
kp1.co = (250.0/periods[n], 2*pi + phases[n])
kp1.interpolation = 'LINEAR'
fcu.extrapolation = 'LINEAR'
last = empties[nEmpties-1]
bpy.ops.mesh.primitive_ico_sphere_add(
size = 0.2,
location=last.location)
ob = bpy.context.object
ob.parent = last
empties[0].location = origin
return
def run(origin):
createEpiCycle(origin)
bpy.ops.object.paths_calculate()
return
if __name__ == "__main__":
run((0,0,0))
bpy.ops.screen.animation_play(reverse=False, sync=False)
ΠΡΠ° ΠΏΡΠΎΠ³ΡΠ°ΠΌΠΌΠ° Π΄ΠΎΠ±Π°Π²Π»ΡΠ΅Ρ Π°ΡΠΌΠ°ΡΡΡΡ Ρ ΠΎΠ΄Π½ΠΎΠΉ ΡΠΏΡΠ°Π²Π»ΡΡΡΠ΅ΠΉ ΠΊΠΎΡΡΡΡ ΠΈ Π΄Π²ΡΠΌΡ ΡΠΏΡΠ°Π²Π»ΡΠ΅ΠΌΡΠΌΠΈ ΠΊΠΎΡΡΡΠΌΠΈ. ΠΡΠ°ΡΠ΅Π½ΠΈΠ΅ ΠΠΎΠ½ΡΠ° (tip) ΠΏΠΎ Z ΡΠΏΡΠ°Π²Π»ΡΠ΅ΡΡΡ ΠΏΠΎΠ·ΠΈΡΠΈΠ΅ΠΉ ΠΏΠΎ X ΡΠΏΡΠ°Π²Π»ΡΡΡΠ΅ΠΉ ΠΊΠΎΡΡΠΈ. ΠΡΠ°ΡΠ΅Π½ΠΈΠ΅ ΠΠ°Π·Ρ (base) ΠΏΠΎ Z ΡΠΏΡΠ°Π²Π»ΡΠ΅ΡΡΡ ΠΊΠ°ΠΊ ΠΏΠΎΠ·ΠΈΡΠΈΠ΅ΠΉ ΠΏΠΎ Y, ΡΠ°ΠΊ ΠΈ Π²ΡΠ°ΡΠ΅Π½ΠΈΠ΅ΠΌ ΠΏΠΎ Z ΡΠΏΡΠ°Π²Π»ΡΡΡΠ΅ΠΉ ΠΊΠΎΡΡΠΈ.
#----------------------------------------------------------
# File driver.py
#----------------------------------------------------------
import bpy
def run(origin):
# Π‘ΠΎΠ·Π΄Π°Π½ΠΈΠ΅ Π°ΡΠΌΠ°ΡΡΡΡ ΠΈ ΠΎΠ±ΡΠ΅ΠΊΡΠ°
amt = bpy.data.armatures.new('MyRigData')
rig = bpy.data.objects.new('MyRig', amt)
rig.location = origin
amt.show_names = True
# ΠΡΠΈΠ²ΡΠ·ΠΊΠ° ΠΎΠ±ΡΠ΅ΠΊΡΠ° ΠΊ ΡΡΠ΅Π½Π΅
scn = bpy.context.scene
scn.objects.link(rig)
scn.objects.active = rig
scn.update()
# Π‘ΠΎΠ·Π΄Π°Π½ΠΈΠ΅ ΠΊΠΎΡΡΠ΅ΠΉ
bpy.ops.object.mode_set(mode='EDIT')
base = amt.edit_bones.new('Base')
base.head = (0,0,0)
base.tail = (0,0,1)
tip = amt.edit_bones.new('Tip')
tip.head = (0,0,1)
tip.tail = (0,0,2)
tip.parent = base
tip.use_connect = True
driver = amt.edit_bones.new('Driver')
driver.head = (2,0,0)
driver.tail = (2,0,1)
bpy.ops.object.mode_set(mode='POSE')
# ΠΠΎΠ±Π°Π²Π»Π΅Π½ΠΈΠ΅ ΡΠΏΡΠ°Π²Π»ΡΡΡΠ΅Π³ΠΎ ΡΠ»Π΅ΠΌΠ΅Π½ΡΠ° Π΄Π»Ρ Π²ΡΠ°ΡΠ΅Π½ΠΈΡ ΠΏΠΎ Z ΠΊΠΎΡΡΠΈ Tip
# Tip.rotz = 1.0 - 1.0*x, Π³Π΄Π΅ x = Driver.locx
fcurve = rig.pose.bones["Tip"].driver_add('rotation_quaternion', 3)
drv = fcurve.driver
drv.type = 'AVERAGE'
drv.show_debug_info = True
var = drv.variables.new()
var.name = 'x'
var.type = 'TRANSFORMS'
targ = var.targets[0]
targ.id = rig
targ.transform_type = 'LOC_X'
targ.bone_target = 'Driver'
targ.use_local_space_transform = True
fmod = fcurve.modifiers[0]
fmod.mode = 'POLYNOMIAL'
fmod.poly_order = 1
fmod.coefficients = (1.0, -1.0)
# ΠΠΎΠ±Π°Π²Π»Π΅Π½ΠΈΠ΅ ΡΠΏΡΠ°Π²Π»ΡΡΡΠ΅Π³ΠΎ ΡΠ»Π΅ΠΌΠ΅Π½ΡΠ° Π΄Π»Ρ Π²ΡΠ°ΡΠ΅Π½ΠΈΡ ΠΏΠΎ Z ΠΊΠΎΡΡΠΈ Base
# Base.rotz = z*z - 3*y, Π³Π΄Π΅ y = Driver.locy ΠΈ z = Driver.rotz
fcurve = rig.pose.bones["Base"].driver_add('rotation_quaternion', 3)
drv = fcurve.driver
drv.type = 'SCRIPTED'
drv.expression = 'z*z - 3*y'
drv.show_debug_info = True
var1 = drv.variables.new()