226 lines
8.7 KiB
Python
226 lines
8.7 KiB
Python
import os
|
|
import subprocess
|
|
from typing import Dict, List, Tuple
|
|
|
|
import bpy
|
|
from bpy.types import Material
|
|
from bpy.types import Object
|
|
|
|
import lnx.api
|
|
import lnx.assets as assets
|
|
import lnx.exporter
|
|
import lnx.log as log
|
|
import lnx.material.cycles as cycles
|
|
import lnx.material.make_decal as make_decal
|
|
import lnx.material.make_depth as make_depth
|
|
import lnx.material.make_mesh as make_mesh
|
|
import lnx.material.make_overlay as make_overlay
|
|
import lnx.material.make_transluc as make_transluc
|
|
import lnx.material.make_refract as make_refract
|
|
import lnx.material.make_voxel as make_voxel
|
|
import lnx.material.mat_state as mat_state
|
|
import lnx.material.mat_utils as mat_utils
|
|
from lnx.material.shader import Shader, ShaderContext, ShaderData
|
|
import lnx.utils
|
|
|
|
if lnx.is_reload(__name__):
|
|
lnx.api = lnx.reload_module(lnx.api)
|
|
assets = lnx.reload_module(assets)
|
|
lnx.exporter = lnx.reload_module(lnx.exporter)
|
|
log = lnx.reload_module(log)
|
|
cycles = lnx.reload_module(cycles)
|
|
make_decal = lnx.reload_module(make_decal)
|
|
make_depth = lnx.reload_module(make_depth)
|
|
make_mesh = lnx.reload_module(make_mesh)
|
|
make_overlay = lnx.reload_module(make_overlay)
|
|
make_transluc = lnx.reload_module(make_transluc)
|
|
make_voxel = lnx.reload_module(make_voxel)
|
|
mat_state = lnx.reload_module(mat_state)
|
|
mat_utils = lnx.reload_module(mat_utils)
|
|
lnx.material.shader = lnx.reload_module(lnx.material.shader)
|
|
from lnx.material.shader import Shader, ShaderContext, ShaderData
|
|
lnx.utils = lnx.reload_module(lnx.utils)
|
|
else:
|
|
lnx.enable_reload(__name__)
|
|
|
|
rpass_hook = None
|
|
|
|
|
|
def build(material: Material, mat_users: Dict[Material, List[Object]], mat_lnxusers) -> Tuple:
|
|
mat_state.mat_users = mat_users
|
|
mat_state.mat_lnxusers = mat_lnxusers
|
|
mat_state.material = material
|
|
mat_state.nodes = material.node_tree.nodes
|
|
mat_state.data = ShaderData(material)
|
|
mat_state.output_node = cycles.node_by_type(mat_state.nodes, 'OUTPUT_MATERIAL')
|
|
if mat_state.output_node is None:
|
|
# Place empty material output to keep compiler happy..
|
|
mat_state.output_node = mat_state.nodes.new('ShaderNodeOutputMaterial')
|
|
|
|
wrd = bpy.data.worlds['Lnx']
|
|
rpdat = lnx.utils.get_rp()
|
|
rpasses = mat_utils.get_rpasses(material)
|
|
matname = lnx.utils.safesrc(lnx.utils.asset_name(material))
|
|
rel_path = lnx.utils.build_dir() + '/compiled/Shaders/'
|
|
full_path = lnx.utils.get_fp() + '/' + rel_path
|
|
if not os.path.exists(full_path):
|
|
os.makedirs(full_path)
|
|
|
|
make_instancing_and_skinning(material, mat_users)
|
|
|
|
bind_constants = dict()
|
|
bind_textures = dict()
|
|
|
|
for rp in rpasses:
|
|
car = []
|
|
bind_constants[rp] = car
|
|
mat_state.bind_constants = car
|
|
tar = []
|
|
bind_textures[rp] = tar
|
|
mat_state.bind_textures = tar
|
|
|
|
con = None
|
|
|
|
if rpdat.rp_driver != 'Leenkx' and lnx.api.drivers[rpdat.rp_driver]['make_rpass'] is not None:
|
|
con = lnx.api.drivers[rpdat.rp_driver]['make_rpass'](rp)
|
|
|
|
if con is not None:
|
|
pass
|
|
|
|
elif rp == 'mesh':
|
|
con = make_mesh.make(rp, rpasses)
|
|
|
|
elif rp == 'shadowmap':
|
|
con = make_depth.make(rp, rpasses, shadowmap=True)
|
|
|
|
elif rp == 'shadowmap_transparent':
|
|
con = make_depth.make(rp, rpasses, shadowmap=True, shadowmap_transparent=True)
|
|
|
|
elif rp == 'translucent':
|
|
con = make_transluc.make(rp)
|
|
|
|
elif rp == 'refraction':
|
|
con = make_refract.make(rp)
|
|
|
|
elif rp == 'overlay':
|
|
con = make_overlay.make(rp)
|
|
|
|
elif rp == 'decal':
|
|
con = make_decal.make(rp)
|
|
|
|
elif rp == 'depth':
|
|
con = make_depth.make(rp, rpasses)
|
|
|
|
elif rp == 'voxel':
|
|
con = make_voxel.make(rp)
|
|
|
|
elif rpass_hook is not None:
|
|
con = rpass_hook(rp)
|
|
|
|
write_shaders(rel_path, con, rp, matname)
|
|
|
|
shader_data_name = matname + '_data'
|
|
|
|
if wrd.lnx_single_data_file:
|
|
if 'shader_datas' not in lnx.exporter.current_output:
|
|
lnx.exporter.current_output['shader_datas'] = []
|
|
lnx.exporter.current_output['shader_datas'].append(mat_state.data.get()['shader_datas'][0])
|
|
else:
|
|
lnx.utils.write_lnx(full_path + '/' + matname + '_data.lnx', mat_state.data.get())
|
|
shader_data_path = lnx.utils.get_fp_build() + '/compiled/Shaders/' + shader_data_name + '.lnx'
|
|
assets.add_shader_data(shader_data_path)
|
|
|
|
return rpasses, mat_state.data, shader_data_name, bind_constants, bind_textures
|
|
|
|
|
|
def write_shaders(rel_path: str, con: ShaderContext, rpass: str, matname: str) -> None:
|
|
keep_cache = mat_state.material.lnx_cached
|
|
write_shader(rel_path, con.vert, 'vert', rpass, matname, keep_cache=keep_cache)
|
|
write_shader(rel_path, con.frag, 'frag', rpass, matname, keep_cache=keep_cache)
|
|
write_shader(rel_path, con.geom, 'geom', rpass, matname, keep_cache=keep_cache)
|
|
write_shader(rel_path, con.tesc, 'tesc', rpass, matname, keep_cache=keep_cache)
|
|
write_shader(rel_path, con.tese, 'tese', rpass, matname, keep_cache=keep_cache)
|
|
|
|
|
|
def write_shader(rel_path: str, shader: Shader, ext: str, rpass: str, matname: str, keep_cache=True) -> None:
|
|
if shader is None or shader.is_linked:
|
|
return
|
|
|
|
# TODO: blend context
|
|
if rpass == 'mesh' and mat_state.material.lnx_blending:
|
|
rpass = 'blend'
|
|
|
|
file_ext = '.glsl'
|
|
if shader.noprocessing:
|
|
# Use hlsl directly
|
|
hlsl_dir = lnx.utils.build_dir() + '/compiled/Hlsl/'
|
|
if not os.path.exists(hlsl_dir):
|
|
os.makedirs(hlsl_dir)
|
|
file_ext = '.hlsl'
|
|
rel_path = rel_path.replace('/compiled/Shaders/', '/compiled/Hlsl/')
|
|
|
|
shader_file = matname + '_' + rpass + '.' + ext + file_ext
|
|
shader_path = lnx.utils.get_fp() + '/' + rel_path + '/' + shader_file
|
|
assets.add_shader(shader_path)
|
|
if not os.path.isfile(shader_path) or not keep_cache:
|
|
with open(shader_path, 'w') as f:
|
|
f.write(shader.get())
|
|
|
|
if shader.noprocessing:
|
|
cwd = os.getcwd()
|
|
os.chdir(lnx.utils.get_fp() + '/' + rel_path)
|
|
hlslbin_path = lnx.utils.get_sdk_path() + '/lib/leenkx_tools/hlslbin/hlslbin.exe'
|
|
prof = 'vs_5_0' if ext == 'vert' else 'ps_5_0' if ext == 'frag' else 'gs_5_0'
|
|
# noprocessing flag - gets renamed to .d3d11
|
|
args = [hlslbin_path.replace('/', '\\').replace('\\\\', '\\'), shader_file, shader_file[:-4] + 'glsl', prof]
|
|
if ext == 'vert':
|
|
args.append('-i')
|
|
args.append('pos')
|
|
proc = subprocess.call(args)
|
|
os.chdir(cwd)
|
|
|
|
|
|
def make_instancing_and_skinning(mat: Material, mat_users: Dict[Material, List[Object]]) -> None:
|
|
"""Build material with instancing or skinning if enabled.
|
|
If the material is a custom material, only validation checks for instancing are performed."""
|
|
global_elems = []
|
|
if mat_users is not None and mat in mat_users:
|
|
# Whether there are both an instanced object and a not instanced object with this material
|
|
instancing_usage = [False, False]
|
|
mat_state.uses_instancing = False
|
|
|
|
for bo in mat_users[mat]:
|
|
if mat.lnx_custom_material == '':
|
|
# Morph Targets
|
|
if lnx.utils.export_morph_targets(bo):
|
|
global_elems.append({'name': 'morph', 'data': 'short2norm'})
|
|
# GPU Skinning
|
|
if lnx.utils.export_bone_data(bo):
|
|
global_elems.append({'name': 'bone', 'data': 'short4norm'})
|
|
global_elems.append({'name': 'weight', 'data': 'short4norm'})
|
|
|
|
# Instancing
|
|
inst = bo.lnx_instanced
|
|
if inst != 'Off' or mat.lnx_particle_flag:
|
|
instancing_usage[0] = True
|
|
mat_state.uses_instancing = True
|
|
|
|
if mat.lnx_custom_material == '':
|
|
global_elems.append({'name': 'ipos', 'data': 'float3'})
|
|
if 'Rot' in inst:
|
|
global_elems.append({'name': 'irot', 'data': 'float3'})
|
|
if 'Scale' in inst:
|
|
global_elems.append({'name': 'iscl', 'data': 'float3'})
|
|
|
|
elif inst == 'Off':
|
|
# Ignore children of instanced objects, they are instanced even when set to 'Off'
|
|
instancing_usage[1] = bo.parent is None or bo.parent.lnx_instanced == 'Off'
|
|
|
|
if instancing_usage[0] and instancing_usage[1]:
|
|
# Display a warning for invalid instancing configurations
|
|
# See https://github.com/leenkx3d/leenkx/issues/2072
|
|
log.warn(f'Material "{mat.name}" has both instanced and not instanced objects, objects might flicker!')
|
|
|
|
if mat.lnx_custom_material == '':
|
|
mat_state.data.global_elems = global_elems
|