function Node_Scatter(_x, _y, _group = -1) : Node_Processor(_x, _y, _group) constructor { name = "Scatter"; shader = sh_blend_normal_dim; uniform_dim = shader_get_uniform(shader, "dimension"); uniform_pos = shader_get_uniform(shader, "position"); uniform_sca = shader_get_uniform(shader, "scale"); uniform_rot = shader_get_uniform(shader, "rotation"); uniform_for = shader_get_sampler_index(shader, "fore"); inputs[| 0] = nodeValue(0, "Surface in", self, JUNCTION_CONNECT.input, VALUE_TYPE.surface, 0); inputs[| 1] = nodeValue(1, "Dimension", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, def_surf_size2 ) .setDisplay(VALUE_DISPLAY.vector); inputs[| 2] = nodeValue(2, "Amount", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 8); inputs[| 3] = nodeValue(3, "Scale", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 1, 1, 1, 1 ] ) .setDisplay(VALUE_DISPLAY.vector_range); inputs[| 4] = nodeValue(4, "Angle", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [0, 0]) .setDisplay(VALUE_DISPLAY.rotation_range); inputs[| 5] = nodeValue(5, "Area", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [ def_surf_size / 2, def_surf_size / 2, def_surf_size / 2, def_surf_size / 2, AREA_SHAPE.rectangle ]) .setDisplay(VALUE_DISPLAY.area, function() { return inputs[| 1].getValue(); }); inputs[| 6] = nodeValue(6, "Distribution", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0) .setDisplay(VALUE_DISPLAY.enum_button, [ "Area", "Border" ]); inputs[| 7] = nodeValue(7, "Point at center", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, false, "Rotate each copy to face the spawn center."); inputs[| 8] = nodeValue(8, "Uniform scaling", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, true); inputs[| 9] = nodeValue(9, "Scatter", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 1) .setDisplay(VALUE_DISPLAY.enum_button, [ "Uniform", "Random" ]); inputs[| 10] = nodeValue(10, "Seed", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, irandom(9999999)); inputs[| 11] = nodeValue(11, "Random blend", self, JUNCTION_CONNECT.input, VALUE_TYPE.color, c_white) .setDisplay(VALUE_DISPLAY.gradient); inputs[| 12] = nodeValue(12, "Alpha", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 1, 1 ]) .setDisplay(VALUE_DISPLAY.slider_range, [0, 1, 0.01]); outputs[| 0] = nodeValue(0, "Surface out", self, JUNCTION_CONNECT.output, VALUE_TYPE.surface, PIXEL_SURFACE); input_display_list = [ 0, 1, 10, ["Scatter", false], 5, 6, 9, 2, ["Transform", false], 3, 8, 7, 4, ["Render", false], 11, 12 ]; temp_surf = [ PIXEL_SURFACE, PIXEL_SURFACE ]; static drawOverlay = function(active, _x, _y, _s, _mx, _my, _snx, _sny) { inputs[| 5].drawOverlay(active, _x, _y, _s, _mx, _my, _snx, _sny); } static process_data = function(_outSurf, _data, _output_index, _array_index) { var _inSurf = _data[0], surf; if(_inSurf == 0) return; var _dim = _data[1]; var _amount = _data[2]; var _scale = _data[3]; var _rota = _data[4]; var _area = _data[5]; var _dist = _data[6]; var _scat = _data[9]; var _pint = _data[7]; var _unis = _data[8]; var seed = _data[10]; var color = inputs[| 11].getValue(); var _bldTyp = inputs[| 11].getExtraData(); var alpha = _data[12]; var _in_w, _in_h; _outSurf = surface_verify(_outSurf, _dim[0], _dim[1]); var ww = surface_get_width(_outSurf); var hh = surface_get_height(_outSurf); for(var i = 0; i < 2; i++) { temp_surf[i] = surface_verify(temp_surf[i], ww, hh); surface_set_target(temp_surf[i]); draw_clear_alpha(0, 0); surface_reset_target(); } BLEND_OVERRIDE var _sed = seed; var res_index = 0, bg = 0; for(var i = 0; i < _amount; i++) { var sp = area_get_random_point(_area, _dist, _scat, i, _amount, _sed); _sed += 20; var _x = sp[0]; var _y = sp[1]; var _scx = random_range_seed(_scale[0], _scale[1], _sed); _sed++; var _scy = random_range_seed(_scale[2], _scale[3], _sed); _sed++; if(_unis) _scy = _scx; var _r = (_pint? point_direction(_area[0], _area[1], _x, _y) : 0) + random_range_seed(_rota[0], _rota[1], _sed); _sed++; surf = _inSurf; if(is_array(_inSurf)) surf = _inSurf[irandom_seed(array_length(_inSurf) - 1, _sed)]; _sed++; var sw = surface_get_width(surf); var sh = surface_get_height(surf); if(_dist != AREA_DISTRIBUTION.area || _scat != AREA_SCATTER.uniform) { _x -= sw / 2; _y -= sh / 2; } var clr = gradient_eval(color, random_seed(1, _sed), _bldTyp[| 0]); _sed++; var alp = random_range_seed(alpha[0], alpha[1], _sed); _sed++; surface_set_target(temp_surf[bg]); shader_set(shader); shader_set_uniform_f_array(uniform_dim, [ sw / ww, sh / hh ]); shader_set_uniform_f_array(uniform_pos, [ _x / ww, _y / hh]); shader_set_uniform_f_array(uniform_sca, [ _scx, _scy ]) shader_set_uniform_f(uniform_rot, degtorad(_r)); texture_set_stage(uniform_for, surface_get_texture(surf)); draw_surface_ext_safe(temp_surf[!bg], 0, 0, 1, 1, 0, clr, alp); shader_reset(); surface_reset_target(); res_index = bg; bg = !bg; } BLEND_NORMAL surface_set_target(_outSurf); draw_clear_alpha(0, 0); BLEND_OVERRIDE draw_surface_safe(temp_surf[res_index], 0, 0); BLEND_NORMAL surface_reset_target(); return _outSurf; } }