2022-03-23 15:02:50 -04:00
|
|
|
import bpy
|
|
|
|
import math
|
|
|
|
|
2022-03-26 01:55:44 -04:00
|
|
|
# Ensures that the Compify Footage shader group exists.
|
2022-03-23 15:02:50 -04:00
|
|
|
#
|
|
|
|
# It will create it if it doesn't exist, and returns the group.
|
2022-03-26 01:55:44 -04:00
|
|
|
def ensure_footage_group():
|
|
|
|
NAME = "Compify Footage"
|
2022-03-23 15:02:50 -04:00
|
|
|
if NAME in bpy.data.node_groups:
|
|
|
|
return bpy.data.node_groups[NAME]
|
2022-03-26 01:55:44 -04:00
|
|
|
|
|
|
|
# Create the group.
|
|
|
|
group = bpy.data.node_groups.new(NAME, type='ShaderNodeTree')
|
|
|
|
for node in group.nodes:
|
|
|
|
group.nodes.remove(node)
|
|
|
|
|
|
|
|
# Create the group inputs and outputs.
|
|
|
|
group.inputs.new(type="NodeSocketColor", name="Footage")
|
|
|
|
group.inputs['Footage'].default_value = (1.0, 0.0, 1.0, 1.0)
|
|
|
|
group.inputs['Footage'].hide_value = True
|
|
|
|
|
2022-04-10 10:05:16 -04:00
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Footage Alpha")
|
|
|
|
group.inputs['Footage Alpha'].default_value = 1.0
|
|
|
|
group.inputs['Footage Alpha'].min_value = 0.0
|
|
|
|
group.inputs['Footage Alpha'].max_value = 1.0
|
|
|
|
|
2022-03-26 01:55:44 -04:00
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Footage Emit")
|
|
|
|
group.inputs['Footage Emit'].default_value = 0.0
|
2022-04-10 10:05:16 -04:00
|
|
|
group.inputs['Footage Emit'].min_value = 0.0
|
|
|
|
group.inputs['Footage Emit'].max_value = 1.0
|
2022-03-26 01:55:44 -04:00
|
|
|
|
|
|
|
group.inputs.new(type="NodeSocketColor", name="Background")
|
|
|
|
group.inputs['Background'].default_value = (1.0, 0.0, 1.0, 1.0)
|
|
|
|
group.inputs['Background'].hide_value = True
|
|
|
|
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Background Alpha")
|
|
|
|
group.inputs['Background Alpha'].default_value = 0.0
|
|
|
|
group.inputs['Background Alpha'].min_value = 0.0
|
|
|
|
group.inputs['Background Alpha'].max_value = 1.0
|
|
|
|
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Background Emit")
|
|
|
|
group.inputs['Background Emit'].default_value = 0.0
|
|
|
|
group.inputs['Background Emit'].min_value = 0.0
|
|
|
|
group.inputs['Background Emit'].max_value = 1.0
|
|
|
|
|
|
|
|
group.inputs.new(type="NodeSocketColor", name="Baked Lighting")
|
|
|
|
group.inputs['Baked Lighting'].default_value = (1.0, 1.0, 1.0, 1.0)
|
|
|
|
group.inputs['Baked Lighting'].hide_value = True
|
|
|
|
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Do Bake")
|
|
|
|
group.inputs['Do Bake'].default_value = 0.0
|
|
|
|
group.inputs['Do Bake'].min_value = 0.0
|
|
|
|
group.inputs['Do Bake'].max_value = 1.0
|
|
|
|
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Debug")
|
|
|
|
group.inputs['Debug'].default_value = 0.0
|
|
|
|
group.inputs['Debug'].min_value = 0.0
|
|
|
|
group.inputs['Debug'].max_value = 1.0
|
|
|
|
|
|
|
|
group.outputs.new(type="NodeSocketShader", name="Shader")
|
|
|
|
|
|
|
|
#-------------------
|
|
|
|
# Footage nodes.
|
|
|
|
|
|
|
|
# Create the nodes.
|
|
|
|
footage_frame = group.nodes.new(type='NodeFrame')
|
|
|
|
footage_input = group.nodes.new(type='NodeGroupInput')
|
|
|
|
footage_debug = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
footage_delight = group.nodes.new(type='ShaderNodeMixRGB')
|
|
|
|
footage_1 = group.nodes.new(type='ShaderNodeBsdfDiffuse')
|
|
|
|
footage_2 = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
|
|
|
|
# Label the nodes.
|
|
|
|
footage_frame.label = "Footage"
|
|
|
|
footage_input.label = "Footage Input"
|
|
|
|
footage_debug.label = "Footage Debug"
|
|
|
|
footage_delight.label = "Footage Delight"
|
|
|
|
footage_1.label = "Footage 1"
|
|
|
|
footage_2.label = "Footage 2"
|
|
|
|
|
|
|
|
# Put the nodes in their frame.
|
|
|
|
footage_input.parent = footage_frame
|
|
|
|
footage_debug.parent = footage_frame
|
|
|
|
footage_delight.parent = footage_frame
|
|
|
|
footage_1.parent = footage_frame
|
|
|
|
footage_2.parent = footage_frame
|
|
|
|
|
|
|
|
# Position the nodes.
|
|
|
|
hs = 200.0
|
|
|
|
x = 200.0
|
|
|
|
y = -330.0
|
|
|
|
|
|
|
|
footage_input.location = (x, y)
|
|
|
|
x += hs
|
|
|
|
footage_delight.location = (x, y)
|
|
|
|
footage_debug.location = (x, y + 50.0)
|
|
|
|
x += hs
|
|
|
|
footage_1.location = (x, y)
|
|
|
|
x += hs
|
|
|
|
footage_2.location = (x, y)
|
|
|
|
|
|
|
|
# Configure the nodes.
|
|
|
|
for socket in footage_input.outputs:
|
|
|
|
socket.hide = True
|
|
|
|
footage_input.outputs["Footage"].hide = False
|
|
|
|
footage_input.outputs["Footage Emit"].hide = False
|
|
|
|
footage_input.outputs["Baked Lighting"].hide = False
|
|
|
|
footage_input.outputs["Do Bake"].hide = False
|
|
|
|
|
|
|
|
footage_delight.blend_type = 'DIVIDE'
|
|
|
|
footage_delight.use_clamp = False
|
|
|
|
footage_delight.inputs[0].default_value = 1.0
|
|
|
|
|
|
|
|
footage_1.inputs['Roughness'].default_value = 0.0
|
|
|
|
footage_1.hide = True
|
|
|
|
|
|
|
|
footage_debug.operation = 'MAXIMUM'
|
|
|
|
footage_debug.use_clamp = True
|
|
|
|
footage_debug.hide = True
|
|
|
|
|
|
|
|
# Hook up the nodes.
|
|
|
|
group.links.new(footage_input.outputs['Footage'], footage_delight.inputs['Color1'])
|
|
|
|
group.links.new(footage_input.outputs['Footage'], footage_2.inputs[2])
|
|
|
|
group.links.new(footage_input.outputs['Footage Emit'], footage_debug.inputs[0])
|
|
|
|
group.links.new(footage_input.outputs['Baked Lighting'], footage_delight.inputs['Color2'])
|
|
|
|
group.links.new(footage_input.outputs['Do Bake'], footage_debug.inputs[1])
|
|
|
|
group.links.new(footage_delight.outputs['Color'], footage_1.inputs['Color'])
|
|
|
|
group.links.new(footage_1.outputs['BSDF'], footage_2.inputs[1])
|
|
|
|
group.links.new(footage_debug.outputs['Value'], footage_2.inputs['Fac'])
|
|
|
|
|
|
|
|
#-------------------
|
|
|
|
# Background nodes.
|
|
|
|
|
|
|
|
# Create the nodes.
|
|
|
|
background_frame = group.nodes.new(type='NodeFrame')
|
|
|
|
background_input = group.nodes.new(type='NodeGroupInput')
|
|
|
|
background_debug = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
background_delight = group.nodes.new(type='ShaderNodeMixRGB')
|
|
|
|
background_transparent = group.nodes.new(type='ShaderNodeBsdfTransparent')
|
|
|
|
background_1 = group.nodes.new(type='ShaderNodeBsdfDiffuse')
|
|
|
|
background_2 = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
background_3 = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
|
|
|
|
# Label the nodes.
|
|
|
|
background_frame.label = "Background"
|
|
|
|
background_input.label = "Background Input"
|
|
|
|
background_debug.label = "Background Debug"
|
|
|
|
background_delight.label = "Background Delight"
|
|
|
|
background_transparent.label = "Background Transparent"
|
|
|
|
background_1.label = "Background 1"
|
|
|
|
background_2.label = "Background 2"
|
|
|
|
background_3.label = "Background 3"
|
|
|
|
|
|
|
|
# Put the nodes in their frame.
|
|
|
|
background_input.parent = background_frame
|
|
|
|
background_debug.parent = background_frame
|
|
|
|
background_delight.parent = background_frame
|
|
|
|
background_transparent.parent = background_frame
|
|
|
|
background_1.parent = background_frame
|
|
|
|
background_2.parent = background_frame
|
|
|
|
background_3.parent = background_frame
|
|
|
|
|
|
|
|
# Position the nodes.
|
|
|
|
hs = 200.0
|
|
|
|
x = 0.0
|
|
|
|
y = 0.0
|
|
|
|
|
|
|
|
background_input.location = (x, y)
|
|
|
|
x += hs
|
|
|
|
background_delight.location = (x, y)
|
|
|
|
background_debug.location = (x, y + 80.0)
|
|
|
|
x += hs
|
|
|
|
background_1.location = (x, y + 30)
|
|
|
|
x += hs
|
|
|
|
background_transparent.location = (x, y - 80.0)
|
|
|
|
background_2.location = (x, y + 100.0)
|
|
|
|
x += hs
|
|
|
|
background_3.location = (x, y)
|
|
|
|
|
|
|
|
# Configure the nodes.
|
|
|
|
for socket in background_input.outputs:
|
|
|
|
socket.hide = True
|
|
|
|
background_input.outputs["Background"].hide = False
|
|
|
|
background_input.outputs["Background Alpha"].hide = False
|
|
|
|
background_input.outputs["Background Emit"].hide = False
|
|
|
|
background_input.outputs["Baked Lighting"].hide = False
|
|
|
|
background_input.outputs["Do Bake"].hide = False
|
|
|
|
|
|
|
|
background_delight.blend_type = 'DIVIDE'
|
|
|
|
background_delight.use_clamp = False
|
|
|
|
background_delight.inputs[0].default_value = 1.0
|
|
|
|
|
|
|
|
background_transparent.inputs['Color'].default_value = (1.0, 1.0, 1.0, 1.0)
|
|
|
|
|
|
|
|
background_1.inputs['Roughness'].default_value = 0.0
|
|
|
|
background_1.hide = True
|
|
|
|
|
|
|
|
background_debug.operation = 'MAXIMUM'
|
|
|
|
background_debug.use_clamp = True
|
|
|
|
background_debug.hide = True
|
|
|
|
|
|
|
|
# Hook up the nodes.
|
|
|
|
group.links.new(background_input.outputs['Background'], background_delight.inputs['Color1'])
|
|
|
|
group.links.new(background_input.outputs['Background'], background_2.inputs[2])
|
|
|
|
group.links.new(background_input.outputs['Background Alpha'], background_3.inputs['Fac'])
|
|
|
|
group.links.new(background_input.outputs['Background Emit'], background_debug.inputs[0])
|
|
|
|
group.links.new(background_input.outputs['Baked Lighting'], background_delight.inputs['Color2'])
|
|
|
|
group.links.new(background_input.outputs['Do Bake'], background_debug.inputs[1])
|
|
|
|
group.links.new(background_delight.outputs['Color'], background_1.inputs['Color'])
|
|
|
|
group.links.new(background_1.outputs['BSDF'], background_2.inputs[1])
|
|
|
|
group.links.new(background_debug.outputs['Value'], background_2.inputs['Fac'])
|
|
|
|
group.links.new(background_transparent.outputs['BSDF'], background_3.inputs[1])
|
|
|
|
group.links.new(background_2.outputs['Shader'], background_3.inputs[2])
|
|
|
|
|
|
|
|
#-------------------
|
|
|
|
# Camera Ray nodes.
|
|
|
|
|
|
|
|
# Create the nodes.
|
|
|
|
camera_ray_frame = group.nodes.new(type='NodeFrame')
|
|
|
|
camera_input = group.nodes.new(type='NodeGroupInput')
|
|
|
|
light_path = group.nodes.new(type='ShaderNodeLightPath')
|
|
|
|
debug_invert = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
camera_ray = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
|
|
|
|
# Label the nodes.
|
|
|
|
camera_ray_frame.label = "Camera Ray"
|
|
|
|
camera_input.label = "Camera Input"
|
|
|
|
light_path.label = "Light Path"
|
|
|
|
debug_invert.label = "Debug Invert"
|
|
|
|
camera_ray.label = "Camera Ray"
|
|
|
|
|
|
|
|
# Put the nodes in their frame.
|
|
|
|
camera_input.parent = camera_ray_frame
|
|
|
|
light_path.parent = camera_ray_frame
|
|
|
|
debug_invert.parent = camera_ray_frame
|
|
|
|
camera_ray.parent = camera_ray_frame
|
|
|
|
|
|
|
|
# Position the nodes.
|
|
|
|
hs = 200.0
|
|
|
|
x = 400.0
|
|
|
|
y = 260.0
|
|
|
|
|
|
|
|
camera_input.location = (x, y)
|
|
|
|
x += hs
|
|
|
|
light_path.location = (x, y)
|
|
|
|
debug_invert.location = (x, y + 50.0)
|
|
|
|
x += hs
|
|
|
|
camera_ray.location = (x, y)
|
|
|
|
|
|
|
|
# Configure the nodes.
|
|
|
|
for socket in camera_input.outputs:
|
|
|
|
socket.hide = True
|
|
|
|
camera_input.outputs["Debug"].hide = False
|
|
|
|
|
|
|
|
for socket in light_path.outputs:
|
|
|
|
socket.hide = True
|
|
|
|
light_path.outputs["Is Camera Ray"].hide = False
|
|
|
|
|
|
|
|
debug_invert.operation = 'MULTIPLY_ADD'
|
|
|
|
debug_invert.use_clamp = True
|
|
|
|
debug_invert.inputs[1].default_value = -1.0
|
|
|
|
debug_invert.inputs[2].default_value = 1.0
|
|
|
|
debug_invert.hide = True
|
|
|
|
|
|
|
|
camera_ray.operation = 'MINIMUM'
|
|
|
|
camera_ray.use_clamp = True
|
|
|
|
camera_ray.hide = True
|
|
|
|
|
|
|
|
# Hook up the nodes.
|
|
|
|
group.links.new(camera_input.outputs['Debug'], debug_invert.inputs['Value'])
|
|
|
|
group.links.new(light_path.outputs['Is Camera Ray'], camera_ray.inputs[0])
|
|
|
|
group.links.new(debug_invert.outputs['Value'], camera_ray.inputs[1])
|
|
|
|
|
|
|
|
#---------------
|
|
|
|
# Baking nodes.
|
|
|
|
|
|
|
|
# Create the nodes.
|
|
|
|
baking_frame = group.nodes.new(type='NodeFrame')
|
|
|
|
baking_input = group.nodes.new(type='NodeGroupInput')
|
|
|
|
baking_transparent = group.nodes.new(type='ShaderNodeBsdfTransparent')
|
|
|
|
baking_diffuse = group.nodes.new(type='ShaderNodeBsdfDiffuse')
|
|
|
|
baking_1 = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
baking_2 = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
baking_3 = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
|
|
|
|
# Label the nodes.
|
|
|
|
baking_frame.label = "Baking"
|
|
|
|
baking_input.label = "Baking Input"
|
|
|
|
baking_transparent.label = "Baking Transparent"
|
|
|
|
baking_diffuse.label = "Baking Diffuse"
|
|
|
|
baking_1.label = "Baking 1"
|
|
|
|
baking_2.label = "Baking 2"
|
|
|
|
baking_3.label = "Baking 3"
|
|
|
|
|
|
|
|
# Put the nodes in their frame.
|
|
|
|
baking_input.parent = baking_frame
|
|
|
|
baking_transparent.parent = baking_frame
|
|
|
|
baking_diffuse.parent = baking_frame
|
|
|
|
baking_1.parent = baking_frame
|
|
|
|
baking_2.parent = baking_frame
|
|
|
|
baking_3.parent = baking_frame
|
|
|
|
|
|
|
|
# Position the nodes.
|
|
|
|
hs = 200.0
|
|
|
|
x = 850.0
|
|
|
|
y = 700.0
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
baking_transparent.location = (x, y)
|
|
|
|
baking_input.location = (x, y - 140.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
baking_1.location = (x, y)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
baking_2.location = (x, y)
|
|
|
|
baking_diffuse.location = (x, y - 140.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
baking_3.location = (x, y)
|
|
|
|
|
|
|
|
# Configure the nodes.
|
|
|
|
for socket in baking_input.outputs:
|
|
|
|
socket.hide = True
|
|
|
|
baking_input.outputs["Footage"].hide = False
|
|
|
|
baking_input.outputs["Background"].hide = False
|
|
|
|
baking_input.outputs["Background Alpha"].hide = False
|
2022-04-10 10:05:16 -04:00
|
|
|
baking_input.outputs["Footage Alpha"].hide = False
|
2022-03-26 01:55:44 -04:00
|
|
|
|
|
|
|
baking_transparent.inputs['Color'].default_value = (1.0, 1.0, 1.0, 1.0)
|
|
|
|
|
|
|
|
baking_diffuse.inputs['Color'].default_value = (1.0, 1.0, 1.0, 1.0)
|
|
|
|
baking_diffuse.inputs['Roughness'].default_value = 0.0
|
|
|
|
|
|
|
|
# Hook up the nodes.
|
|
|
|
group.links.new(baking_transparent.outputs['BSDF'], baking_1.inputs[1])
|
|
|
|
group.links.new(baking_input.outputs['Footage'], baking_2.inputs[2])
|
|
|
|
group.links.new(baking_input.outputs['Background'], baking_1.inputs[2])
|
|
|
|
group.links.new(baking_input.outputs['Background Alpha'], baking_1.inputs['Fac'])
|
2022-04-10 10:05:16 -04:00
|
|
|
group.links.new(baking_input.outputs['Footage Alpha'], baking_2.inputs['Fac'])
|
2022-03-26 01:55:44 -04:00
|
|
|
group.links.new(baking_1.outputs['Shader'], baking_2.inputs[1])
|
|
|
|
group.links.new(baking_2.outputs['Shader'], baking_3.inputs[1])
|
|
|
|
group.links.new(baking_diffuse.outputs['BSDF'], baking_3.inputs[2])
|
|
|
|
|
|
|
|
#----------------------
|
|
|
|
# The remaining nodes.
|
|
|
|
|
|
|
|
# Create the nodes.
|
|
|
|
mix_input = group.nodes.new(type='NodeGroupInput')
|
|
|
|
background_mask = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
camera_switch = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
bake_switch = group.nodes.new(type='ShaderNodeMixShader')
|
|
|
|
output = group.nodes.new(type='NodeGroupOutput')
|
|
|
|
|
|
|
|
# Label the nodes.
|
|
|
|
mix_input.label = "Mix Input"
|
|
|
|
background_mask.label = "Background Mask"
|
|
|
|
camera_switch.label = "Camera Switch"
|
|
|
|
bake_switch.label = "Bake Switch"
|
|
|
|
|
|
|
|
# Position the nodes.
|
|
|
|
hs = 200.0
|
|
|
|
x = 1400.0
|
|
|
|
y = 300.0
|
|
|
|
|
|
|
|
mix_input.location = (x, y)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
background_mask.location = (x, y - 100.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
camera_switch.location = (x, y - 100.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
bake_switch.location = (x, y)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
output.location = (x, y)
|
|
|
|
|
|
|
|
# Configure the nodes.
|
|
|
|
for socket in mix_input.outputs:
|
|
|
|
socket.hide = True
|
2022-04-10 10:05:16 -04:00
|
|
|
mix_input.outputs["Footage Alpha"].hide = False
|
2022-03-26 01:55:44 -04:00
|
|
|
mix_input.outputs["Do Bake"].hide = False
|
|
|
|
|
|
|
|
# Hook up the nodes.
|
2022-04-10 10:05:16 -04:00
|
|
|
group.links.new(mix_input.outputs['Footage Alpha'], background_mask.inputs['Fac'])
|
2022-03-26 01:55:44 -04:00
|
|
|
group.links.new(mix_input.outputs['Do Bake'], bake_switch.inputs['Fac'])
|
|
|
|
group.links.new(background_mask.outputs['Shader'], camera_switch.inputs[1])
|
|
|
|
group.links.new(camera_switch.outputs['Shader'], bake_switch.inputs[1])
|
|
|
|
group.links.new(bake_switch.outputs['Shader'], output.inputs['Shader'])
|
|
|
|
|
|
|
|
#-------------------------------------------------
|
|
|
|
# Final hook up of all the groups of nodes above.
|
|
|
|
|
|
|
|
group.links.new(camera_ray.outputs['Value'], baking_3.inputs['Fac'])
|
|
|
|
group.links.new(camera_ray.outputs['Value'], camera_switch.inputs['Fac'])
|
|
|
|
group.links.new(background_3.outputs['Shader'], background_mask.inputs[1])
|
|
|
|
group.links.new(footage_2.outputs['Shader'], background_mask.inputs[2])
|
|
|
|
group.links.new(footage_2.outputs['Shader'], camera_switch.inputs[2])
|
|
|
|
group.links.new(baking_3.outputs['Shader'], bake_switch.inputs[2])
|
|
|
|
|
|
|
|
return group
|
2022-03-23 15:02:50 -04:00
|
|
|
|
|
|
|
|
|
|
|
# Ensures that the Feathered Square shader group exists.
|
|
|
|
#
|
|
|
|
# It will create it if it doesn't exist, and returns the group.
|
|
|
|
def ensure_feathered_square_group():
|
|
|
|
NAME = "Feathered Square"
|
|
|
|
|
|
|
|
# If it already exists, just return it.
|
|
|
|
if NAME in bpy.data.node_groups:
|
|
|
|
return bpy.data.node_groups[NAME]
|
|
|
|
|
|
|
|
# Create the group.
|
|
|
|
group = bpy.data.node_groups.new(NAME, type='ShaderNodeTree')
|
|
|
|
for node in group.nodes:
|
|
|
|
group.nodes.remove(node)
|
|
|
|
|
|
|
|
# Create the group inputs and outputs.
|
|
|
|
group.inputs.new(type="NodeSocketVector", name="Vector")
|
|
|
|
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Feather")
|
|
|
|
group.inputs['Feather'].default_value = 0.0
|
|
|
|
group.inputs['Feather'].min_value = 0.0
|
|
|
|
group.inputs['Feather'].max_value = 1.0
|
|
|
|
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Dilate")
|
|
|
|
group.inputs['Dilate'].default_value = 0.0
|
|
|
|
group.inputs['Dilate'].min_value = 0.0
|
|
|
|
group.inputs['Dilate'].max_value = 0.1
|
|
|
|
|
|
|
|
group.outputs.new(type="NodeSocketFloat", name="Value")
|
|
|
|
|
|
|
|
#-------------------
|
|
|
|
# Create the nodes.
|
|
|
|
input = group.nodes.new(type='NodeGroupInput')
|
|
|
|
output = group.nodes.new(type='NodeGroupOutput')
|
|
|
|
|
|
|
|
xyz = group.nodes.new(type='ShaderNodeSeparateXYZ')
|
|
|
|
feather_clamp = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
|
|
|
|
madd_x = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
madd_y = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
|
|
|
|
abs_x = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
abs_y = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
|
|
|
|
xy_max = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
xy_invert = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
xy_add = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
xy_divide = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
|
|
|
|
smoothstep1 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
smoothstep2 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
smoothstep3 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
smoothstep4 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
smoothstep5 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
|
|
|
|
#------------------
|
|
|
|
# Label the nodes.
|
|
|
|
xyz.label = "XYZ"
|
|
|
|
feather_clamp.label = "Feather Clamp"
|
|
|
|
|
|
|
|
madd_x.label = "Multiply-Add X"
|
|
|
|
madd_y.label = "Multiply-Add Y"
|
|
|
|
|
|
|
|
abs_x.label = "Abs X"
|
|
|
|
abs_y.label = "Abs Y"
|
|
|
|
|
|
|
|
xy_max.label = "XY Max"
|
|
|
|
xy_invert.label = "XY Invert"
|
|
|
|
xy_add.label = "XY Add"
|
|
|
|
xy_divide.label = "XY Divide"
|
|
|
|
|
|
|
|
smoothstep1.label = "Smoothstep 1"
|
|
|
|
smoothstep2.label = "Smoothstep 2"
|
|
|
|
smoothstep3.label = "Smoothstep 3"
|
|
|
|
smoothstep4.label = "Smoothstep 4"
|
|
|
|
smoothstep5.label = "Smoothstep 5"
|
|
|
|
|
|
|
|
#---------------------
|
|
|
|
# Position the nodes.
|
|
|
|
hs = 250.0
|
|
|
|
x = 0.0
|
|
|
|
|
|
|
|
input.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
xyz.location = (x, 0.0)
|
|
|
|
feather_clamp.location = (x, -200.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
madd_x.location = (x, 0.0)
|
|
|
|
madd_y.location = (x, -200.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
abs_x.location = (x, 0.0)
|
|
|
|
abs_y.location = (x, -200.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
xy_max.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
xy_invert.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
xy_add.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
xy_divide.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
smoothstep1.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
smoothstep2.location = (x, -200.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
smoothstep3.location = (x, 0.0)
|
|
|
|
smoothstep4.location = (x, -200.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
smoothstep5.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
output.location = (x, 0.0)
|
|
|
|
|
|
|
|
#----------------------
|
|
|
|
# Configure the nodes.
|
|
|
|
feather_clamp.operation = 'MAXIMUM'
|
|
|
|
feather_clamp.use_clamp = False
|
|
|
|
feather_clamp.inputs[1].default_value = 0.000001
|
|
|
|
|
|
|
|
madd_x.operation = 'MULTIPLY_ADD'
|
|
|
|
madd_x.use_clamp = False
|
|
|
|
madd_x.inputs[1].default_value = 2.0
|
|
|
|
madd_x.inputs[2].default_value = -1.0
|
|
|
|
madd_y.operation = 'MULTIPLY_ADD'
|
|
|
|
madd_y.use_clamp = False
|
|
|
|
madd_y.inputs[1].default_value = 2.0
|
|
|
|
madd_y.inputs[2].default_value = -1.0
|
|
|
|
|
|
|
|
abs_x.operation = 'ABSOLUTE'
|
|
|
|
abs_x.use_clamp = False
|
|
|
|
abs_y.operation = 'ABSOLUTE'
|
|
|
|
abs_y.use_clamp = False
|
|
|
|
|
|
|
|
xy_max.operation = 'MAXIMUM'
|
|
|
|
xy_max.use_clamp = False
|
|
|
|
xy_invert.operation = 'MULTIPLY_ADD'
|
|
|
|
xy_invert.use_clamp = False
|
|
|
|
xy_invert.inputs[1].default_value = -1.0
|
|
|
|
xy_invert.inputs[2].default_value = 1.0
|
|
|
|
xy_add.operation = 'ADD'
|
|
|
|
xy_add.use_clamp = False
|
|
|
|
xy_divide.operation = 'DIVIDE'
|
|
|
|
xy_divide.use_clamp = True
|
|
|
|
|
|
|
|
smoothstep1.operation = 'MULTIPLY'
|
|
|
|
smoothstep1.use_clamp = False
|
|
|
|
smoothstep2.operation = 'MULTIPLY'
|
|
|
|
smoothstep2.use_clamp = False
|
|
|
|
smoothstep3.operation = 'MULTIPLY'
|
|
|
|
smoothstep3.use_clamp = False
|
|
|
|
smoothstep3.inputs[1].default_value = 3.0
|
|
|
|
smoothstep4.operation = 'MULTIPLY'
|
|
|
|
smoothstep4.use_clamp = False
|
|
|
|
smoothstep4.inputs[1].default_value = 2.0
|
|
|
|
smoothstep5.operation = 'SUBTRACT'
|
|
|
|
smoothstep5.use_clamp = True
|
|
|
|
|
|
|
|
#--------------------
|
|
|
|
# Hook up the nodes.
|
|
|
|
group.links.new(input.outputs['Vector'], xyz.inputs[0])
|
|
|
|
group.links.new(input.outputs['Feather'], feather_clamp.inputs[0])
|
|
|
|
group.links.new(input.outputs['Dilate'], xy_add.inputs[1])
|
|
|
|
|
|
|
|
group.links.new(xyz.outputs['X'], madd_x.inputs[0])
|
|
|
|
group.links.new(xyz.outputs['Y'], madd_y.inputs[0])
|
|
|
|
group.links.new(feather_clamp.outputs[0], xy_divide.inputs[1])
|
|
|
|
|
|
|
|
group.links.new(madd_x.outputs[0], abs_x.inputs[0])
|
|
|
|
group.links.new(madd_y.outputs[0], abs_y.inputs[0])
|
|
|
|
|
|
|
|
group.links.new(abs_x.outputs[0], xy_max.inputs[0])
|
|
|
|
group.links.new(abs_y.outputs[0], xy_max.inputs[1])
|
|
|
|
|
|
|
|
group.links.new(xy_max.outputs[0], xy_invert.inputs['Value'])
|
|
|
|
|
|
|
|
group.links.new(xy_invert.outputs[0], xy_add.inputs[0])
|
|
|
|
|
|
|
|
group.links.new(xy_add.outputs[0], xy_divide.inputs[0])
|
|
|
|
|
|
|
|
group.links.new(xy_divide.outputs[0], smoothstep1.inputs[0])
|
|
|
|
group.links.new(xy_divide.outputs[0], smoothstep1.inputs[1])
|
|
|
|
group.links.new(xy_divide.outputs[0], smoothstep2.inputs[1])
|
|
|
|
group.links.new(smoothstep1.outputs[0], smoothstep2.inputs[0])
|
|
|
|
|
|
|
|
group.links.new(smoothstep1.outputs[0], smoothstep3.inputs[0])
|
|
|
|
group.links.new(smoothstep2.outputs[0], smoothstep4.inputs[0])
|
|
|
|
|
|
|
|
group.links.new(smoothstep3.outputs[0], smoothstep5.inputs[0])
|
|
|
|
group.links.new(smoothstep4.outputs[0], smoothstep5.inputs[1])
|
|
|
|
|
|
|
|
group.links.new(smoothstep5.outputs[0], output.inputs['Value'])
|
|
|
|
|
|
|
|
return group
|
|
|
|
|
|
|
|
|
|
|
|
# Takes a camera object, and ensures there is a node group for
|
|
|
|
# projecting textures from that camera.
|
|
|
|
#
|
|
|
|
# It will create it if it doesn't exist, and returns the group.
|
|
|
|
def ensure_camera_project_group(camera):
|
|
|
|
name = "Camera Project | " + camera.name
|
|
|
|
|
|
|
|
# Fetch or create group.
|
|
|
|
group = None
|
|
|
|
if name in bpy.data.node_groups:
|
|
|
|
group = bpy.data.node_groups[name]
|
|
|
|
else:
|
|
|
|
group = bpy.data.node_groups.new(name, type='ShaderNodeTree')
|
|
|
|
|
|
|
|
# Clear all nodes, to start from a clean slate.
|
|
|
|
for node in group.nodes:
|
|
|
|
group.nodes.remove(node)
|
|
|
|
|
|
|
|
# Create the group inputs and outputs.
|
|
|
|
if not "Aspect Ratio" in group.inputs:
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Aspect Ratio")
|
|
|
|
group.inputs['Aspect Ratio'].default_value = 1.0
|
|
|
|
if not "Rotation" in group.inputs:
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Rotation")
|
|
|
|
if not "Loc X" in group.inputs:
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Loc X")
|
|
|
|
if not "Loc Y" in group.inputs:
|
|
|
|
group.inputs.new(type="NodeSocketFloat", name="Loc Y")
|
|
|
|
if not "Vector" in group.outputs:
|
|
|
|
group.outputs.new(type="NodeSocketVector", name="Vector")
|
|
|
|
|
|
|
|
#-------------------
|
|
|
|
# Create the nodes.
|
|
|
|
input = group.nodes.new(type='NodeGroupInput')
|
|
|
|
output = group.nodes.new(type='NodeGroupOutput')
|
|
|
|
|
2022-04-10 10:05:16 -04:00
|
|
|
camera_transform = group.nodes.new(type='ShaderNodeTexCoord')
|
2022-03-23 15:02:50 -04:00
|
|
|
lens = group.nodes.new(type='ShaderNodeValue')
|
|
|
|
sensor_width = group.nodes.new(type='ShaderNodeValue')
|
|
|
|
lens_shift_x = group.nodes.new(type='ShaderNodeValue')
|
|
|
|
lens_shift_y = group.nodes.new(type='ShaderNodeValue')
|
|
|
|
|
|
|
|
zoom_1 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
zoom_2 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
lens_shift_1 = group.nodes.new(type='ShaderNodeCombineXYZ')
|
|
|
|
to_radians = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
user_location = group.nodes.new(type='ShaderNodeCombineXYZ')
|
|
|
|
|
|
|
|
perspective_1 = group.nodes.new(type='ShaderNodeSeparateXYZ')
|
|
|
|
perspective_2 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
perspective_3 = group.nodes.new(type='ShaderNodeMath')
|
|
|
|
perspective_4 = group.nodes.new(type='ShaderNodeCombineXYZ')
|
|
|
|
zoom_3 = group.nodes.new(type='ShaderNodeVectorMath')
|
|
|
|
lens_shift_2 = group.nodes.new(type='ShaderNodeVectorMath')
|
|
|
|
|
|
|
|
user_translate = group.nodes.new(type='ShaderNodeVectorMath')
|
|
|
|
user_rotate = group.nodes.new(type='ShaderNodeVectorRotate')
|
|
|
|
aspect_ratio = group.nodes.new(type='ShaderNodeCombineXYZ')
|
|
|
|
user_transforms = group.nodes.new(type='ShaderNodeVectorMath')
|
|
|
|
|
|
|
|
recenter = group.nodes.new(type='ShaderNodeVectorMath')
|
|
|
|
|
|
|
|
#--------------------
|
|
|
|
# Label the nodes.
|
2022-04-10 10:05:16 -04:00
|
|
|
camera_transform.label = "Camera Transform"
|
2022-03-23 15:02:50 -04:00
|
|
|
lens.label = "Lens"
|
|
|
|
sensor_width.label = "Sensor Width"
|
|
|
|
lens_shift_x.label = "Lens Shift X"
|
|
|
|
lens_shift_y.label = "Lens Shift Y"
|
|
|
|
|
|
|
|
zoom_1.label = "Zoom 1"
|
|
|
|
zoom_2.label = "Zoom 2"
|
|
|
|
lens_shift_1.label = "Lens Shift 1"
|
|
|
|
to_radians.label = "Degrees to Radians"
|
|
|
|
user_location.label = "User Location"
|
|
|
|
|
|
|
|
perspective_1.label = "Perspective 1"
|
|
|
|
perspective_2.label = "Perspective 2"
|
|
|
|
perspective_3.label = "Perspective 3"
|
|
|
|
perspective_4.label = "Perspective 4"
|
|
|
|
zoom_3.label = "Zoom 3"
|
|
|
|
lens_shift_2.label = "Lens Shift 2"
|
|
|
|
|
|
|
|
user_translate.label = "User Translate"
|
|
|
|
user_rotate.label = "User Rotate"
|
|
|
|
aspect_ratio.label = "Aspect Ratio"
|
|
|
|
user_transforms.label = "User Transforms"
|
|
|
|
|
|
|
|
recenter.label = "Recenter"
|
|
|
|
|
|
|
|
#---------------------
|
|
|
|
# Position the nodes.
|
|
|
|
hs = 250.0
|
|
|
|
x = 0.0
|
|
|
|
|
2022-04-10 10:05:16 -04:00
|
|
|
camera_transform.location = (x, 0.0)
|
2022-03-23 15:02:50 -04:00
|
|
|
lens.location = (x, -700.0)
|
|
|
|
sensor_width.location = (x, -900.0)
|
|
|
|
lens_shift_x.location = (x, -1100.0)
|
|
|
|
lens_shift_y.location = (x, -1300.0)
|
|
|
|
input.location = (x, -1500.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
zoom_1.location = (x, -700.0)
|
|
|
|
lens_shift_1.location = (x, -1100.0)
|
|
|
|
to_radians.location = (x, -1500.0)
|
|
|
|
user_location.location = (x, -1700.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
zoom_2.location = (x, -700.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
perspective_1.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
perspective_2.location = (x, 0.0)
|
|
|
|
perspective_3.location = (x, -200.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
perspective_4.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
zoom_3.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
lens_shift_2.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
user_translate.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
user_rotate.location = (x, 0.0)
|
|
|
|
aspect_ratio.location = (x, -300.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
user_transforms.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
recenter.location = (x, 0.0)
|
|
|
|
|
|
|
|
x += hs
|
|
|
|
output.location = (x, 0.0)
|
|
|
|
|
|
|
|
#---------------------
|
|
|
|
# Set up the drivers.
|
|
|
|
|
|
|
|
drv_lens = lens.outputs['Value'].driver_add("default_value").driver
|
|
|
|
drv_lens.type = 'SUM'
|
|
|
|
var = drv_lens.variables.new()
|
|
|
|
var.type = 'SINGLE_PROP'
|
|
|
|
var.targets[0].id_type = 'CAMERA'
|
|
|
|
var.targets[0].id = camera.data
|
|
|
|
var.targets[0].data_path = 'lens'
|
|
|
|
|
|
|
|
drv_width = sensor_width.outputs['Value'].driver_add("default_value").driver
|
|
|
|
drv_width.type = 'SUM'
|
|
|
|
var = drv_width.variables.new()
|
|
|
|
var.type = 'SINGLE_PROP'
|
|
|
|
var.targets[0].id_type = 'CAMERA'
|
|
|
|
var.targets[0].id = camera.data
|
|
|
|
var.targets[0].data_path = 'sensor_width'
|
|
|
|
|
|
|
|
drv_shift_x = lens_shift_x.outputs['Value'].driver_add("default_value").driver
|
|
|
|
drv_shift_x.type = 'SUM'
|
|
|
|
var = drv_shift_x.variables.new()
|
|
|
|
var.type = 'SINGLE_PROP'
|
|
|
|
var.targets[0].id_type = 'CAMERA'
|
|
|
|
var.targets[0].id = camera.data
|
|
|
|
var.targets[0].data_path = 'shift_x'
|
|
|
|
|
|
|
|
drv_shift_y = lens_shift_y.outputs['Value'].driver_add("default_value").driver
|
|
|
|
drv_shift_y.type = 'SUM'
|
|
|
|
var = drv_shift_y.variables.new()
|
|
|
|
var.type = 'SINGLE_PROP'
|
|
|
|
var.targets[0].id_type = 'CAMERA'
|
|
|
|
var.targets[0].id = camera.data
|
|
|
|
var.targets[0].data_path = 'shift_y'
|
|
|
|
|
|
|
|
#----------------------
|
|
|
|
# Configure the nodes.
|
2022-04-10 10:05:16 -04:00
|
|
|
camera_transform.object = camera
|
|
|
|
|
2022-03-23 15:02:50 -04:00
|
|
|
zoom_1.operation = 'DIVIDE'
|
|
|
|
zoom_1.use_clamp = False
|
|
|
|
zoom_2.operation = 'MULTIPLY'
|
|
|
|
zoom_2.use_clamp = False
|
|
|
|
zoom_2.inputs[1].default_value = -1.0
|
|
|
|
lens_shift_1.inputs[2].default_value = 0.0
|
|
|
|
to_radians.operation = 'MULTIPLY'
|
|
|
|
to_radians.use_clamp = False
|
|
|
|
to_radians.inputs[1].default_value = math.pi / 180.0
|
|
|
|
user_location.inputs[2].default_value = 0.0
|
|
|
|
|
|
|
|
perspective_2.operation = 'DIVIDE'
|
|
|
|
perspective_2.use_clamp = False
|
|
|
|
perspective_3.operation = 'DIVIDE'
|
|
|
|
perspective_3.use_clamp = False
|
|
|
|
zoom_3.operation = 'MULTIPLY'
|
|
|
|
lens_shift_2.operation = 'SUBTRACT'
|
|
|
|
|
|
|
|
user_translate.operation = 'SUBTRACT'
|
|
|
|
user_rotate.rotation_type = 'Z_AXIS'
|
|
|
|
user_rotate.invert = False
|
|
|
|
user_rotate.inputs['Center'].default_value = (0.0, 0.0, 0.0)
|
|
|
|
aspect_ratio.inputs['X'].default_value = 1.0
|
|
|
|
aspect_ratio.inputs['Z'].default_value = 0.0
|
|
|
|
user_transforms.operation = 'MULTIPLY'
|
|
|
|
|
|
|
|
recenter.operation = 'ADD'
|
|
|
|
recenter.inputs[1].default_value = (0.5, 0.5, 0.0)
|
|
|
|
|
|
|
|
#--------------------
|
|
|
|
# Hook up the nodes.
|
2022-04-10 10:05:16 -04:00
|
|
|
group.links.new(camera_transform.outputs['Object'], perspective_1.inputs['Vector'])
|
2022-03-23 15:02:50 -04:00
|
|
|
group.links.new(lens.outputs['Value'], zoom_1.inputs[0])
|
|
|
|
group.links.new(sensor_width.outputs['Value'], zoom_1.inputs[1])
|
|
|
|
group.links.new(zoom_1.outputs['Value'], zoom_2.inputs[0])
|
|
|
|
group.links.new(zoom_2.outputs['Value'], zoom_3.inputs[1])
|
|
|
|
group.links.new(lens_shift_x.outputs['Value'], lens_shift_1.inputs['X'])
|
|
|
|
group.links.new(lens_shift_y.outputs['Value'], lens_shift_1.inputs['Y'])
|
|
|
|
group.links.new(lens_shift_1.outputs['Vector'], lens_shift_2.inputs[1])
|
|
|
|
|
|
|
|
group.links.new(input.outputs['Aspect Ratio'], aspect_ratio.inputs[1])
|
|
|
|
group.links.new(input.outputs['Rotation'], to_radians.inputs[0])
|
|
|
|
group.links.new(to_radians.outputs['Value'], user_rotate.inputs['Angle'])
|
|
|
|
group.links.new(input.outputs['Loc X'], user_location.inputs['X'])
|
|
|
|
group.links.new(input.outputs['Loc Y'], user_location.inputs['Y'])
|
|
|
|
group.links.new(user_location.outputs['Vector'], user_translate.inputs[1])
|
|
|
|
|
|
|
|
group.links.new(perspective_1.outputs['X'], perspective_2.inputs[0])
|
|
|
|
group.links.new(perspective_1.outputs['Y'], perspective_3.inputs[0])
|
|
|
|
group.links.new(perspective_1.outputs['Z'], perspective_2.inputs[1])
|
|
|
|
group.links.new(perspective_1.outputs['Z'], perspective_3.inputs[1])
|
|
|
|
group.links.new(perspective_1.outputs['Z'], perspective_4.inputs['Z'])
|
|
|
|
group.links.new(perspective_2.outputs['Value'], perspective_4.inputs['X'])
|
|
|
|
group.links.new(perspective_3.outputs['Value'], perspective_4.inputs['Y'])
|
|
|
|
group.links.new(perspective_4.outputs['Vector'], zoom_3.inputs[0])
|
|
|
|
group.links.new(zoom_3.outputs['Vector'], lens_shift_2.inputs[0])
|
|
|
|
group.links.new(lens_shift_2.outputs['Vector'], user_translate.inputs[0])
|
|
|
|
|
|
|
|
group.links.new(user_translate.outputs['Vector'], user_rotate.inputs['Vector'])
|
|
|
|
group.links.new(user_rotate.outputs['Vector'], user_transforms.inputs[0])
|
|
|
|
group.links.new(aspect_ratio.outputs['Vector'], user_transforms.inputs[1])
|
|
|
|
group.links.new(user_transforms.outputs['Vector'], recenter.inputs[0])
|
|
|
|
|
|
|
|
group.links.new(recenter.outputs['Vector'], output.inputs['Vector'])
|
|
|
|
|
|
|
|
return group
|