UNPKG

polygonjs-engine

Version:

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

275 lines (274 loc) 10.6 kB
import {TypedSopNode} from "./_Base"; import { AttribClassMenuEntries, AttribTypeMenuEntries, AttribClass, AttribType, ATTRIBUTE_CLASSES, ATTRIBUTE_TYPES } from "../../../core/geometry/Constant"; import {CoreAttribute} from "../../../core/geometry/Attribute"; import {TypeAssert} from "../../poly/Assert"; import {AttribCreateSopOperation} from "../../../core/operations/sop/AttribCreate"; import {NodeParamsConfig, ParamConfig} from "../utils/params/ParamsConfig"; const DEFAULT = AttribCreateSopOperation.DEFAULT_PARAMS; class AttribCreateSopParamsConfig extends NodeParamsConfig { constructor() { super(...arguments); this.group = ParamConfig.STRING(DEFAULT.group); this.class = ParamConfig.INTEGER(DEFAULT.class, { menu: { entries: AttribClassMenuEntries } }); this.type = ParamConfig.INTEGER(DEFAULT.type, { menu: { entries: AttribTypeMenuEntries } }); this.name = ParamConfig.STRING(DEFAULT.name); this.size = ParamConfig.INTEGER(DEFAULT.size, { range: [1, 4], rangeLocked: [true, true], visibleIf: {type: AttribType.NUMERIC} }); this.value1 = ParamConfig.FLOAT(DEFAULT.value1, { visibleIf: {type: AttribType.NUMERIC, size: 1}, expression: {forEntities: true} }); this.value2 = ParamConfig.VECTOR2(DEFAULT.value2, { visibleIf: {type: AttribType.NUMERIC, size: 2}, expression: {forEntities: true} }); this.value3 = ParamConfig.VECTOR3(DEFAULT.value3, { visibleIf: {type: AttribType.NUMERIC, size: 3}, expression: {forEntities: true} }); this.value4 = ParamConfig.VECTOR4(DEFAULT.value4, { visibleIf: {type: AttribType.NUMERIC, size: 4}, expression: {forEntities: true} }); this.string = ParamConfig.STRING(DEFAULT.string, { visibleIf: {type: AttribType.STRING}, expression: {forEntities: true} }); } } const ParamsConfig2 = new AttribCreateSopParamsConfig(); export class AttribCreateSopNode extends TypedSopNode { constructor() { super(...arguments); this.params_config = ParamsConfig2; this._x_arrays_by_geometry_uuid = {}; this._y_arrays_by_geometry_uuid = {}; this._z_arrays_by_geometry_uuid = {}; this._w_arrays_by_geometry_uuid = {}; } static type() { return "attribCreate"; } initializeNode() { this.io.inputs.setCount(1); this.io.inputs.initInputsClonedState(AttribCreateSopOperation.INPUT_CLONED_STATE); this.scene().dispatchController.onAddListener(() => { this.params.onParamsCreated("params_label", () => { this.params.label.init([this.p.name]); }); }); } cook(input_contents) { if (this._is_using_expression()) { if (this.pv.name && this.pv.name.trim() != "") { this._add_attribute(ATTRIBUTE_CLASSES[this.pv.class], input_contents[0]); } else { this.states.error.set("attribute name is not valid"); } } else { this._operation = this._operation || new AttribCreateSopOperation(this.scene(), this.states); const core_group = this._operation.cook(input_contents, this.pv); this.setCoreGroup(core_group); } } async _add_attribute(attrib_class, core_group) { const attrib_type = ATTRIBUTE_TYPES[this.pv.type]; switch (attrib_class) { case AttribClass.VERTEX: await this.add_point_attribute(attrib_type, core_group); return this.setCoreGroup(core_group); case AttribClass.OBJECT: await this.add_object_attribute(attrib_type, core_group); return this.setCoreGroup(core_group); } TypeAssert.unreachable(attrib_class); } async add_point_attribute(attrib_type, core_group) { const core_objects = core_group.coreObjects(); switch (attrib_type) { case AttribType.NUMERIC: { for (let i = 0; i < core_objects.length; i++) { await this.add_numeric_attribute_to_points(core_objects[i]); } return; } case AttribType.STRING: { for (let i = 0; i < core_objects.length; i++) { await this.add_string_attribute_to_points(core_objects[i]); } return; } } TypeAssert.unreachable(attrib_type); } async add_object_attribute(attrib_type, core_group) { const core_objects = core_group.coreObjectsFromGroup(this.pv.group); switch (attrib_type) { case AttribType.NUMERIC: await this.add_numeric_attribute_to_object(core_objects); return; case AttribType.STRING: await this.add_string_attribute_to_object(core_objects); return; } TypeAssert.unreachable(attrib_type); } async add_numeric_attribute_to_points(core_object) { const core_geometry = core_object.coreGeometry(); if (!core_geometry) { return; } const points = core_object.pointsFromGroup(this.pv.group); const param = [this.p.value1, this.p.value2, this.p.value3, this.p.value4][this.pv.size - 1]; if (param.has_expression()) { if (!core_geometry.hasAttrib(this.pv.name)) { core_geometry.addNumericAttrib(this.pv.name, this.pv.size, param.value); } const geometry = core_geometry.geometry(); const array = geometry.getAttribute(this.pv.name).array; if (this.pv.size == 1) { if (this.p.value1.expression_controller) { await this.p.value1.expression_controller.compute_expression_for_points(points, (point, value) => { array[point.index() * this.pv.size + 0] = value; }); } } else { const vparam = [this.p.value2, this.p.value3, this.p.value4][this.pv.size - 2]; let params = vparam.components; const tmp_arrays = new Array(params.length); let component_param; const arrays_by_geometry_uuid = [ this._x_arrays_by_geometry_uuid, this._y_arrays_by_geometry_uuid, this._z_arrays_by_geometry_uuid, this._w_arrays_by_geometry_uuid ]; for (let i = 0; i < params.length; i++) { component_param = params[i]; if (component_param.has_expression() && component_param.expression_controller) { tmp_arrays[i] = this._init_array_if_required(geometry, arrays_by_geometry_uuid[i], points.length); await component_param.expression_controller.compute_expression_for_points(points, (point, value) => { tmp_arrays[i][point.index()] = value; }); } else { const value = component_param.value; for (let point of points) { array[point.index() * this.pv.size + i] = value; } } } for (let j = 0; j < tmp_arrays.length; j++) { const tmp_array = tmp_arrays[j]; if (tmp_array) { for (let i = 0; i < tmp_array.length; i++) { array[i * this.pv.size + j] = tmp_array[i]; } } } } } else { } } async add_numeric_attribute_to_object(core_objects) { const param = [this.p.value1, this.p.value2, this.p.value3, this.p.value4][this.pv.size - 1]; if (param.has_expression()) { if (this.pv.size == 1) { if (this.p.value1.expression_controller) { await this.p.value1.expression_controller.compute_expression_for_objects(core_objects, (core_object, value) => { core_object.setAttribValue(this.pv.name, value); }); } } else { const vparam = [this.p.value2, this.p.value3, this.p.value4][this.pv.size - 2]; let params = vparam.components; let values_by_core_object_index = {}; for (let core_object of core_objects) { values_by_core_object_index[core_object.index()] = []; } for (let component_index = 0; component_index < params.length; component_index++) { const component_param = params[component_index]; if (component_param.has_expression() && component_param.expression_controller) { await component_param.expression_controller.compute_expression_for_objects(core_objects, (core_object, value) => { values_by_core_object_index[core_object.index()][component_index] = value; }); } else { for (let core_object of core_objects) { values_by_core_object_index[core_object.index()][component_index] = component_param.value; } } } for (let i = 0; i < core_objects.length; i++) { const core_object = core_objects[i]; const value = values_by_core_object_index[core_object.index()]; core_object.setAttribValue(this.pv.name, value); } } } else { } } async add_string_attribute_to_points(core_object) { const points = core_object.pointsFromGroup(this.pv.group); const param = this.p.string; const string_values = new Array(points.length); if (param.has_expression() && param.expression_controller) { await param.expression_controller.compute_expression_for_points(points, (point, value) => { string_values[point.index()] = value; }); } else { } const index_data = CoreAttribute.array_to_indexed_arrays(string_values); const geometry = core_object.coreGeometry(); if (geometry) { geometry.setIndexedAttribute(this.pv.name, index_data["values"], index_data["indices"]); } } async add_string_attribute_to_object(core_objects) { const param = this.p.string; if (param.has_expression() && param.expression_controller) { await param.expression_controller.compute_expression_for_objects(core_objects, (core_object, value) => { core_object.setAttribValue(this.pv.name, value); }); } else { } } _init_array_if_required(geometry, arrays_by_geometry_uuid, points_count) { const uuid = geometry.uuid; const current_array = arrays_by_geometry_uuid[uuid]; if (current_array) { if (current_array.length < points_count) { arrays_by_geometry_uuid[uuid] = new Array(points_count); } } else { arrays_by_geometry_uuid[uuid] = new Array(points_count); } return arrays_by_geometry_uuid[uuid]; } _is_using_expression() { const attrib_type = ATTRIBUTE_TYPES[this.pv.type]; switch (attrib_type) { case AttribType.NUMERIC: const param = [this.p.value1, this.p.value2, this.p.value3, this.p.value4][this.pv.size - 1]; return param.has_expression(); case AttribType.STRING: return this.p.string.has_expression(); } } }