Pixel-Composer/scripts/node_3d_camera/node_3d_camera.gml
2024-08-12 18:42:05 +07:00

416 lines
No EOL
13 KiB
Text

function Node_3D_Camera(_x, _y, _group = noone) : Node_3D_Object(_x, _y, _group) constructor {
name = "3D Camera";
batch_output = true;
dimension_index = in_d3d + 2;
object = new __3dCamera_object();
camera = new __3dCamera();
lookat = new __3dGizmoSphere(0.5, c_ltgray, 1);
lookLine = noone;
lookRad = new __3dGizmoCircleZ(0.5, c_yellow, 0.5);
w = 128;
scene = new __3dScene(camera);
scene.name = "Camera";
deferData = noone;
global.SKY_SPHERE = new __3dUVSphere(0.5, 16, 8, true);
inputs[in_d3d + 0] = nodeValue_Int("FOV", self, 60 )
.setDisplay(VALUE_DISPLAY.slider, { range: [ 10, 90, 0.1 ] });
inputs[in_d3d + 1] = nodeValue_Vec2("Clipping Distance", self, [ 1, 10 ] );
inputs[in_d3d + 2] = nodeValue_Dimension(self);
inputs[in_d3d + 3] = nodeValue_Enum_Button("Projection", self, 1 , [ "Perspective", "Orthographic" ]);
inputs[in_d3d + 4] = nodeValue_D3Scene("Scene", self, noone )
.setVisible(true, true);
inputs[in_d3d + 5] = nodeValue_Color("Ambient Light", self, c_dkgrey );
inputs[in_d3d + 6] = nodeValue_Bool("Show Background", self, false );
inputs[in_d3d + 7] = nodeValue_Enum_Button("Backface Culling", self, 2 , [ "None", "CW", "CCW" ]);
inputs[in_d3d + 8] = nodeValue_Float("Orthographic Scale", self, 0.5 )
.setDisplay(VALUE_DISPLAY.slider, { range: [ 0.01, 4, 0.01 ] });
inputs[in_d3d + 9] = nodeValue_Enum_Scroll("Postioning Mode", self, 2, [ "Position + Rotation", "Position + Lookat", "Lookat + Rotation" ] );
inputs[in_d3d + 10] = nodeValue_Vec3("Lookat Position", self, [ 0, 0, 0 ] );
inputs[in_d3d + 11] = nodeValue_Rotation("Roll", self, 0);
inputs[in_d3d + 12] = nodeValue_Rotation("Horizontal Angle", self, 45 );
inputs[in_d3d + 13] = nodeValue_Float("Vertical Angle", self, 30 )
.setDisplay(VALUE_DISPLAY.slider, { range: [0, 90, 0.1] });
inputs[in_d3d + 14] = nodeValue_Float("Distance", self, 4 );
inputs[in_d3d + 15] = nodeValue_Bool("Gamma Adjust", self, false );
inputs[in_d3d + 16] = nodeValue_Surface("Environment Texture", self);
inputs[in_d3d + 17] = nodeValue_Bool("Ambient Occlusion", self, false );
inputs[in_d3d + 18] = nodeValue_Float("AO Radius", self, 0.25 );
inputs[in_d3d + 19] = nodeValue_Float("AO Bias", self, 0.05 );
inputs[in_d3d + 20] = nodeValue_Float("AO Strength", self, 1. )
.setDisplay(VALUE_DISPLAY.slider, { range: [ 0.01, 4, 0.01 ] });
inputs[in_d3d + 21] = nodeValue_Int("Round Normal", self, 0 )
.setWindows();
inputs[in_d3d + 22] = nodeValue_Enum_Button("Blend mode", self, 0 , [ "Normal", "Additive" ]);
in_cam = array_length(inputs);
outputs[0] = nodeValue_Output("Rendered", self, VALUE_TYPE.surface, noone );
outputs[1] = nodeValue_Output("Normal", self, VALUE_TYPE.surface, noone )
.setVisible(false);
outputs[2] = nodeValue_Output("Depth", self, VALUE_TYPE.surface, noone )
.setVisible(false);
input_display_list = [ in_d3d + 4,
["Output", false], in_d3d + 2,
["Transform", false], in_d3d + 9, 0, 1, in_d3d + 10, in_d3d + 11, in_d3d + 12, in_d3d + 13, in_d3d + 14,
["Camera", true], in_d3d + 3, in_d3d + 0, in_d3d + 1, in_d3d + 8,
["Render", true], in_d3d + 5, in_d3d + 16, in_d3d + 6, in_d3d + 7, in_d3d + 15, in_d3d + 22,
["Ambient Occlusion", true], in_d3d + 17, in_d3d + 20, in_d3d + 18, in_d3d + 19,
["Effects", true], in_d3d + 21,
];
tool_lookat = new NodeTool( "Move Target", THEME.tools_3d_transform_object );
static getToolSettings = function() {
var _posm = getInputData(in_d3d + 9);
switch(_posm) {
case 0 : return tool_settings;
case 1 :
case 2 : return [];
}
return [];
}
static drawOverlay3D = function(active, params, _mx, _my, _snx, _sny, _panel) {
var _rot = inputs[1].display_data.angle_display;
tools = _rot == QUARTERNION_DISPLAY.quarterion? tool_quate : tool_euler;
if(_rot == QUARTERNION_DISPLAY.euler && isUsingTool("Rotate"))
PANEL_PREVIEW.tool_current = noone;
var object = getPreviewObjects();
if(array_empty(object)) return;
object = object[0];
var _pos = inputs[0].getValue(,,, true);
var _vpos = new __vec3( _pos[0], _pos[1], _pos[2] );
if(isUsingTool("Transform")) drawGizmoPosition(0, object, _vpos, active, params, _mx, _my, _snx, _sny, _panel);
else if(isUsingTool("Rotate")) drawGizmoRotation(1, object, _vpos, active, params, _mx, _my, _snx, _sny, _panel);
else if(isUsingTool("Move Target")) {
var _lkpos = inputs[in_d3d + 10].getValue(,,, true);
var _lkvpos = new __vec3( _lkpos[0], _lkpos[1], _lkpos[2] );
drawGizmoPosition(in_d3d + 10, noone, _lkvpos, active, params, _mx, _my, _snx, _sny, _panel);
}
if(drag_axis != noone && mouse_release(mb_left)) {
drag_axis = noone;
UNDO_HOLDING = false;
}
#region draw result
var _outSurf = outputs[0].getValue();
if(is_array(_outSurf)) _outSurf = array_safe_get_fast(_outSurf, 0);
if(!is_surface(_outSurf)) return;
var _w = _panel.w;
var _h = _panel.h - _panel.toolbar_height;
var _pw = surface_get_width_safe(_outSurf);
var _ph = surface_get_height_safe(_outSurf);
var _ps = min(128 / _ph, 160 / _pw);
var _pws = _pw * _ps;
var _phs = _ph * _ps;
var _px = _w - 16 - _pws;
var _py = _h - 16 - _phs;
draw_surface_ext_safe(_outSurf, _px, _py, _ps, _ps);
draw_set_color(COLORS._main_icon);
draw_rectangle(_px, _py, _px + _pws, _py + _phs, true);
#endregion
}
static onValueUpdate = function(index) {
if(index == in_d3d + 9) PANEL_PREVIEW.tool_current = noone;
}
static step = function() { #region
var _proj = getInputData(in_d3d + 3);
var _posm = getInputData(in_d3d + 9);
var _ao = getInputData(in_d3d + 17);
inputs[in_d3d + 0].setVisible(_proj == 0);
inputs[in_d3d + 8].setVisible(_proj == 1);
inputs[0].setVisible(_posm == 0 || _posm == 1);
inputs[1].setVisible(_posm == 0);
inputs[in_d3d + 10].setVisible(_posm == 1 || _posm == 2);
inputs[in_d3d + 11].setVisible(_posm == 1);
inputs[in_d3d + 12].setVisible(_posm == 2);
inputs[in_d3d + 13].setVisible(_posm == 2);
inputs[in_d3d + 14].setVisible(_posm == 2);
inputs[in_d3d + 18].setVisible(_ao);
inputs[in_d3d + 19].setVisible(_ao);
inputs[in_d3d + 20].setVisible(_ao);
switch(_posm) {
case 0 :
tools = [ tool_pos, tool_rot ];
break;
case 1 :
tools = [ tool_pos, tool_lookat ];
tool_attribute.context = 1;
break;
case 2 :
tools = [ tool_lookat ];
tool_attribute.context = 1;
break;
}
} #endregion
static preProcessData = function(_data) {}
static submitShadow = function() {}
static submitShader = function() {}
static processData = function(_output, _data, _output_index, _array_index = 0) { #region
#region data
var _pos = _data[0];
var _rot = _data[1];
var _fov = _data[in_d3d + 0];
var _clip = _data[in_d3d + 1];
var _dim = _data[in_d3d + 2];
var _proj = _data[in_d3d + 3];
var _sobj = _data[in_d3d + 4];
var _ambt = _data[in_d3d + 5];
var _dbg = _data[in_d3d + 6];
var _back = _data[in_d3d + 7];
var _orts = _data[in_d3d + 8];
var _posm = _data[in_d3d + 9];
var _look = _data[in_d3d + 10];
var _roll = _data[in_d3d + 11];
var _hAng = _data[in_d3d + 12];
var _vAng = _data[in_d3d + 13];
var _dist = _data[in_d3d + 14];
var _gamm = _data[in_d3d + 15];
var _env = _data[in_d3d + 16];
var _aoEn = _data[in_d3d + 17];
var _aoRa = _data[in_d3d + 18];
var _aoBi = _data[in_d3d + 19];
var _aoSr = _data[in_d3d + 20];
var _nrmSmt = _data[in_d3d + 21];
var _blend = _data[in_d3d + 22];
var _qi1 = new BBMOD_Quaternion().FromAxisAngle(new BBMOD_Vec3(0, 1, 0), 90);
var _qi2 = new BBMOD_Quaternion().FromAxisAngle(new BBMOD_Vec3(1, 0, 0), -90);
var _qi3 = new BBMOD_Quaternion().FromAxisAngle(new BBMOD_Vec3(1, 0, 0), 90);
#endregion
surface_depth_disable(false);
switch(_posm) { #region ++++ camera positioning ++++
case 0 :
camera.useFocus = false;
camera.position.set(_pos);
camera.rotation.set(_rot[0], _rot[1], _rot[2], _rot[3]);
break;
case 1 :
camera.useFocus = true;
camera.position.set(_pos);
camera.focus.set(_look);
camera.up.set(0, 0, -1);
var _for = camera.focus.subtract(camera.position);
if(!_for.isZero())
camera.rotation = new BBMOD_Quaternion().FromLookRotation(_for, camera.up).Mul(_qi1).Mul(_qi2);
lookat.transform.position.set(_look);
lookLine = new __3dGizmoLineDashed(camera.position, camera.focus, 0.25, c_gray, 1);
break;
case 2 :
camera.useFocus = true;
camera.focus.set(_look);
camera.setFocusAngle(_hAng, _vAng, _dist);
camera.setCameraLookRotate();
camera.up = camera.getUp()._multiply(-1);
var _for = camera.focus.subtract(camera.position);
if(!_for.isZero()) camera.rotation = new BBMOD_Quaternion().FromLookRotation(_for, camera.up.multiply(-1)).Mul(_qi1).Mul(_qi3);
lookat.transform.position.set(_look);
lookLine = new __3dGizmoLineDashed(camera.position, camera.focus, 0.25, c_gray, 1);
var _camRad = camera.position.subtract(camera.focus);
var _rad = point_distance(0, 0, _camRad.x, _camRad.y) * 2;
lookRad.transform.scale.set(_rad, _rad, 1);
lookRad.transform.position.set(new __vec3(camera.focus.x, camera.focus.y, camera.position.z));
break;
} #endregion
object.transform.position.set(camera.position);
object.transform.rotation = camera.rotation.Clone();
object.transform.scale.set(1, _dim[0] / _dim[1], 1);
preProcessData(_data);
#region camera view project
camera.projection = _proj;
camera.setViewFov(_fov, _clip[0], _clip[1]);
if(_proj == 0) camera.setViewSize(_dim[0], _dim[1]);
else if(_proj == 1) camera.setViewSize(1 / _orts, _dim[0] / _dim[1] / _orts);
camera.setMatrix();
#endregion
#region scene setting
scene.camera = camera;
scene.lightAmbient = _ambt;
scene.gammaCorrection = _gamm;
scene.enviroment_map = _env;
scene.cull_mode = _back;
scene.ssao_enabled = _aoEn;
scene.ssao_radius = _aoRa;
scene.ssao_bias = _aoBi;
scene.ssao_strength = _aoSr;
scene.defer_normal_radius = _nrmSmt;
scene.draw_background = _dbg;
#endregion
#region submit
var _render = outputs[0].getValue();
var _normal = outputs[1].getValue();
var _depth = outputs[2].getValue();
var _bgSurf = _dbg? scene.renderBackground(_dim[0], _dim[1]) : noone;
_render = surface_verify(_render, _dim[0], _dim[1]);
_normal = surface_verify(_normal, _dim[0], _dim[1]);
_depth = surface_verify(_depth , _dim[0], _dim[1]);
if(_sobj) {
_sobj.submitShadow(scene, _sobj);
submitShadow();
deferData = scene.deferPass(_sobj, _dim[0], _dim[1], deferData);
surface_set_target_ext(0, _render);
surface_set_target_ext(1, _normal);
surface_set_target_ext(2, _depth );
DRAW_CLEAR
gpu_set_zwriteenable(true);
gpu_set_cullmode(_back);
if(_blend == 0) {
gpu_set_ztestenable(true);
} else {
BLEND_ADD
gpu_set_ztestenable(false);
}
camera.applyCamera();
scene.reset();
scene.submitShader(_sobj);
submitShader();
scene.apply(deferData);
scene.submit(_sobj);
BLEND_NORMAL
surface_reset_target();
camera.resetCamera();
}
#endregion
#region render
var _finalRender = surface_create(_dim[0], _dim[1]);
surface_set_target(_finalRender);
DRAW_CLEAR
BLEND_ALPHA
if(_dbg) {
draw_surface_safe(_bgSurf);
surface_free(_bgSurf);
}
draw_surface_safe(_render);
if(deferData) {
BLEND_MULTIPLY
draw_surface_safe(deferData.ssao);
BLEND_NORMAL
}
surface_reset_target();
surface_free(_render);
#endregion
surface_depth_disable(true);
return [ _finalRender, _normal, _depth ];
} #endregion
static onDrawNode = function(xx, yy, _mx, _my, _s, _hover, _focus) {}
static getPreviewObject = function() { #region
var _scene = array_safe_get_fast(all_inputs, in_d3d + 4, noone);
if(is_array(_scene))
_scene = array_safe_get_fast(_scene, preview_index, noone);
return _scene;
} #endregion
static getPreviewObjects = function() { #region
var _posm = getInputData(in_d3d + 9);
var _scene = array_safe_get_fast(all_inputs, in_d3d + 4, noone);
if(is_array(_scene))
_scene = array_safe_get_fast(_scene, preview_index, noone);
switch(_posm) {
case 0 : return [ object, _scene ];
case 1 : return [ object, lookat, lookLine, _scene ];
case 2 : return [ object, lookat, lookLine, lookRad, _scene ];
}
return [ object, _scene ];
} #endregion
static getPreviewObjectOutline = function() { return isUsingTool("Move Target")? [ lookat ] : [ object ]; }
static doSerialize = function(_map) { #region
_map.camera_base_length = in_cam;
} #endregion
static postDeserialize = function() { #region
var _tlen = struct_try_get(load_map, "camera_base_length", in_d3d + 22);
for( var i = _tlen; i < in_cam; i++ )
array_insert(load_map.inputs, i, noone);
} #endregion
}