UNPKG

polygonjs-engine

Version:

node-based webgl 3D engine https://polygonjs.com

245 lines (218 loc) 8.48 kB
import {BaseJsNodeType} from '../../_Base'; import {TypedNodeTraverser} from '../../../utils/shaders/NodeTraverser'; import {BaseNodeType} from '../../../_Base'; import {BaseJsFunctionAssembler} from '../assemblers/_Base'; import {MapUtils} from '../../../../../core/MapUtils'; import {JsDefinitionType, BaseJsDefinition} from '../../utils/JsDefinition'; import {TypedJsDefinitionCollection} from '../../utils/JsDefinitionCollection'; import {ParamConfigsController} from '../../../../nodes/utils/code/controllers/ParamConfigsController'; import {LinesController} from './LinesController'; import {JsCodeFormatter} from './CodeFormatter'; import {JsLineType} from './LineType'; import {JsParamConfig} from './ParamConfig'; import {ParamType} from '../../../../poly/ParamType'; import {NodeContext} from '../../../../poly/NodeContext'; import {ShaderName} from '../../../utils/shaders/ShaderName'; import {CoreGraphNodeId} from '../../../../../core/graph/CoreGraph'; import {ArrayUtils} from '../../../../../core/ArrayUtils'; export class JsCodeBuilder { _param_configs_controller: ParamConfigsController<JsParamConfig<ParamType>> = new ParamConfigsController(); _param_configs_set_allowed: boolean = true; private _lines_controller: LinesController | undefined; _lines: Map<ShaderName, Map<JsLineType, string[]>> = new Map(); constructor(private _assembler: BaseJsFunctionAssembler, private _gl_parent_node: BaseNodeType) {} async build_from_nodes(root_nodes: BaseJsNodeType[]) { const node_traverser = new TypedNodeTraverser<NodeContext.JS>( this._gl_parent_node, this._assembler.shader_names, (root_node, shader_name) => { return this._assembler.input_names_for_shader_name(root_node, shader_name); } ); node_traverser.traverse(root_nodes); const nodes_by_shader_name: Map<ShaderName, BaseJsNodeType[]> = new Map(); for (let shader_name of this.shader_names()) { const nodes = node_traverser.nodes_for_shader_name(shader_name); nodes_by_shader_name.set(shader_name, nodes); } const sorted_nodes = node_traverser.sorted_nodes(); for (let shader_name of this.shader_names()) { const root_nodes_for_shader = this._assembler.root_nodes_by_shader_name(shader_name); for (let root_node of root_nodes_for_shader) { MapUtils.push_on_array_at_entry(nodes_by_shader_name, shader_name, root_node); } } // ensure nodes are not added if already present const sorted_node_ids: Map<CoreGraphNodeId, boolean> = new Map(); for (let node of sorted_nodes) { sorted_node_ids.set(node.graphNodeId(), true); } for (let root_node of root_nodes) { if (!sorted_node_ids.get(root_node.graphNodeId())) { sorted_nodes.push(root_node); sorted_node_ids.set(root_node.graphNodeId(), true); } } for (let node of sorted_nodes) { // node.set_assembler(this._assembler); node.reset_code(); } for (let node of sorted_nodes) { await node.params.eval_all(); } this._lines_controller = new LinesController(); this.reset(); for (let shader_name of this.shader_names()) { const nodes = ArrayUtils.uniq(nodes_by_shader_name.get(shader_name) || []); if (nodes) { for (let node of nodes) { // node.set_shader_name(shader_name); if (this._param_configs_set_allowed) { node.set_param_configs(); } node.set_lines(this._lines_controller); } } } if (this._param_configs_set_allowed) { this.set_param_configs(sorted_nodes); } this.set_code_lines(sorted_nodes); } disallow_new_param_configs() { this._param_configs_set_allowed = false; } allow_new_param_configs() { this._param_configs_set_allowed = true; } shader_names() { return this._assembler.shader_names; } private reset() { for (let shader_name of this.shader_names()) { const lines_map = new Map(); // for (let line_type of LINE_TYPES) { // lines_map.set(line_type, []); // } this._lines.set(shader_name, lines_map); // this._function_declared.set(shader_name, new Map()); } } param_configs() { return this._param_configs_controller.list || []; } lines(line_type: JsLineType) { const shader_name = ShaderName.VERTEX; return this._lines.get(shader_name)!.get(line_type); } all_lines() { return this._lines; } set_param_configs(nodes: BaseJsNodeType[]) { this._param_configs_controller.reset(); for (let node of nodes) { const param_configs = node.param_configs(); if (param_configs) { for (let param_config of param_configs) { this._param_configs_controller.push(param_config); } } } } set_code_lines(nodes: BaseJsNodeType[]) { // for (let shader_name of this.shader_names()) { // nodes.forEach((node, i)=>{ this.add_code_lines(nodes); // }) // } } add_code_lines(nodes: BaseJsNodeType[]) { this.add_definitions(nodes, JsDefinitionType.FUNCTION, JsLineType.FUNCTION_DECLARATION); this.add_definitions(nodes, JsDefinitionType.UNIFORM, JsLineType.DEFINE); this.add_definitions(nodes, JsDefinitionType.ATTRIBUTE, JsLineType.DEFINE); this.add_code_line_for_nodes_and_line_type(nodes, JsLineType.BODY); } private add_definitions(nodes: BaseJsNodeType[], definition_type: JsDefinitionType, line_type: JsLineType) { if (!this._lines_controller) { return; } const definitions = []; for (let node of nodes) { let node_definitions = this._lines_controller.definitions(node); if (node_definitions) { node_definitions = node_definitions.filter((d) => d.definition_type == definition_type); for (let definition of node_definitions) { definitions.push(definition); } } } if (definitions.length > 0) { const collection = new TypedJsDefinitionCollection<JsDefinitionType>(definitions); const uniq_definitions = collection.uniq(); if (collection.errored) { // TODO: handle error throw `code builder error: ${collection.error_message}`; } const definitions_by_node_id: Map<CoreGraphNodeId, BaseJsDefinition[]> = new Map(); const node_ids: Map<CoreGraphNodeId, boolean> = new Map(); for (let definition of uniq_definitions) { const node_id = definition.node.graphNodeId(); if (!node_ids.has(node_id)) { node_ids.set(node_id, true); } MapUtils.push_on_array_at_entry(definitions_by_node_id, node_id, definition); } const shader_name = ShaderName.VERTEX; const lines_for_shader = this._lines.get(shader_name)!; node_ids.forEach((_, node_id) => { const definitions = definitions_by_node_id.get(node_id); if (definitions) { const first_definition = definitions[0]; if (first_definition) { const comment = JsCodeFormatter.node_comment(first_definition.node, line_type); MapUtils.push_on_array_at_entry(lines_for_shader, line_type, comment); for (let definition of definitions) { const line = JsCodeFormatter.line_wrap(definition.line, line_type); MapUtils.push_on_array_at_entry(lines_for_shader, line_type, line); } const separator = JsCodeFormatter.post_line_separator(line_type); MapUtils.push_on_array_at_entry(lines_for_shader, line_type, separator); } } }); } } add_code_line_for_nodes_and_line_type(nodes: BaseJsNodeType[], line_type: JsLineType) { nodes = nodes.filter((node) => { if (this._lines_controller) { const lines = this._lines_controller.body_lines(node); return lines && lines.length > 0; } }); var nodes_count = nodes.length; for (let i = 0; i < nodes_count; i++) { const is_last = i == nodes.length - 1; this.add_code_line_for_node_and_line_type(nodes[i], line_type, is_last); } } add_code_line_for_node_and_line_type(node: BaseJsNodeType, line_type: JsLineType, is_last: boolean): void { if (!this._lines_controller) { return; } const lines = this._lines_controller.body_lines(node); if (lines && lines.length > 0) { const shader_name = ShaderName.VERTEX; const lines_for_shader = this._lines.get(shader_name)!; const comment = JsCodeFormatter.node_comment(node, line_type); MapUtils.push_on_array_at_entry(lines_for_shader, line_type, comment); ArrayUtils.uniq(lines).forEach((line) => { line = JsCodeFormatter.line_wrap(line, line_type); MapUtils.push_on_array_at_entry(lines_for_shader, line_type, line); }); if (!(line_type == JsLineType.BODY && is_last)) { const separator = JsCodeFormatter.post_line_separator(line_type); MapUtils.push_on_array_at_entry(lines_for_shader, line_type, separator); } } } }