forked from LeenkxTeam/LNXSDK
		
	
		
			
				
	
	
		
			111 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			111 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
from bpy.props import *
 | 
						|
from bpy.types import Node, NodeSocket
 | 
						|
 | 
						|
import lnx
 | 
						|
from lnx.material.lnx_nodes.lnx_nodes import add_node
 | 
						|
from lnx.material.parser_state import ParserState
 | 
						|
from lnx.material.shader import Shader
 | 
						|
 | 
						|
if lnx.is_reload(__name__):
 | 
						|
    lnx.material.lnx_nodes.lnx_nodes = lnx.reload_module(lnx.material.lnx_nodes.lnx_nodes)
 | 
						|
    from lnx.material.lnx_nodes.lnx_nodes import add_node
 | 
						|
    lnx.material.parser_state = lnx.reload_module(lnx.material.parser_state)
 | 
						|
    from lnx.material.parser_state import ParserState
 | 
						|
    lnx.material.shader = lnx.reload_module(lnx.material.shader)
 | 
						|
    from lnx.material.shader import Shader
 | 
						|
else:
 | 
						|
    lnx.enable_reload(__name__)
 | 
						|
 | 
						|
 | 
						|
class ShaderDataNode(Node):
 | 
						|
    """Allows access to shader data such as uniforms and inputs."""
 | 
						|
    bl_idname = 'LnxShaderDataNode'
 | 
						|
    bl_label = 'Shader Data'
 | 
						|
    bl_icon = 'NONE'
 | 
						|
 | 
						|
    input_type: EnumProperty(
 | 
						|
        items = [('input', 'Input', 'Shader Input'),
 | 
						|
                 ('uniform', 'Uniform', 'Uniform value')],
 | 
						|
        name='Input Type',
 | 
						|
        default='input',
 | 
						|
        description="The kind of data that should be retrieved")
 | 
						|
 | 
						|
    input_source: EnumProperty(
 | 
						|
        items = [('frag', 'Fragment Shader', 'Take the input from the fragment shader'),
 | 
						|
                 ('vert', 'Vertex Shader', 'Take the input from the vertex shader and pass it through to the fragment shader')],
 | 
						|
        name='Input Source',
 | 
						|
        default='vert',
 | 
						|
        description="Where to take the input value from")
 | 
						|
 | 
						|
    variable_type: EnumProperty(
 | 
						|
        items = [('int', 'int', 'int'),
 | 
						|
                 ('float', 'float', 'float'),
 | 
						|
                 ('vec2', 'vec2', 'vec2'),
 | 
						|
                 ('vec3', 'vec3', 'vec3'),
 | 
						|
                 ('vec4', 'vec4', 'vec4'),
 | 
						|
                 ('sampler2D', 'sampler2D', 'sampler2D')],
 | 
						|
        name='Variable Type',
 | 
						|
        default='vec3',
 | 
						|
        description="The type of the variable")
 | 
						|
 | 
						|
    variable_name: StringProperty(name="Variable Name", description="The name of the variable")
 | 
						|
 | 
						|
    def draw_buttons(self, context, layout):
 | 
						|
        col = layout.column(align=True)
 | 
						|
        col.label(text="Input Type:")
 | 
						|
        # Use a row to expand horizontally
 | 
						|
        col.row().prop(self, "input_type", expand=True)
 | 
						|
 | 
						|
        split = layout.split(factor=0.5, align=True)
 | 
						|
        col_left = split.column()
 | 
						|
        col_right = split.column()
 | 
						|
 | 
						|
        if self.input_type == "input":
 | 
						|
            col_left.label(text="Input Source")
 | 
						|
            col_right.prop(self, "input_source", text="")
 | 
						|
 | 
						|
        col_left.label(text="Variable Type")
 | 
						|
        col_right.prop(self, "variable_type", text="")
 | 
						|
        col_left.label(text="Variable Name")
 | 
						|
        col_right.prop(self, "variable_name", text="")
 | 
						|
 | 
						|
    def init(self, context):
 | 
						|
        self.outputs.new('NodeSocketColor', 'Color')
 | 
						|
        self.outputs.new('NodeSocketVector', 'Vector')
 | 
						|
        self.outputs.new('NodeSocketFloat', 'Float')
 | 
						|
        self.outputs.new('NodeSocketInt', 'Int')
 | 
						|
 | 
						|
    def __parse(self, out_socket: NodeSocket, state: ParserState) -> str:
 | 
						|
        if self.input_type == "uniform":
 | 
						|
            state.frag.add_uniform(f'{self.variable_type} {self.variable_name}', link=self.variable_name)
 | 
						|
            state.vert.add_uniform(f'{self.variable_type} {self.variable_name}', link=self.variable_name)
 | 
						|
 | 
						|
            if self.variable_type == "sampler2D":
 | 
						|
                state.frag.add_uniform('vec2 screenSize', link='_screenSize')
 | 
						|
                return f'textureLod({self.variable_name}, gl_FragCoord.xy / screenSize, 0.0).rgb'
 | 
						|
 | 
						|
            if self.variable_type == "vec2":
 | 
						|
                return f'vec3({self.variable_name}.xy, 0)'
 | 
						|
 | 
						|
            return self.variable_name
 | 
						|
 | 
						|
        else:
 | 
						|
            if self.input_source == "frag":
 | 
						|
                state.frag.add_in(f'{self.variable_type} {self.variable_name}')
 | 
						|
                return self.variable_name
 | 
						|
 | 
						|
            # Reroute input from vertex shader to fragment shader (input must exist!)
 | 
						|
            else:
 | 
						|
                state.vert.add_out(f'{self.variable_type} out_{self.variable_name}')
 | 
						|
                state.frag.add_in(f'{self.variable_type} out_{self.variable_name}')
 | 
						|
 | 
						|
                state.vert.write(f'out_{self.variable_name} = {self.variable_name};')
 | 
						|
                return 'out_' + self.variable_name
 | 
						|
 | 
						|
    @staticmethod
 | 
						|
    def parse(node: 'ShaderDataNode', out_socket: NodeSocket, state: ParserState) -> str:
 | 
						|
        return node.__parse(out_socket, state)
 | 
						|
 | 
						|
 | 
						|
add_node(ShaderDataNode, category='Leenkx')
 |