mirror of
https://github.com/Ttanasart-pt/Pixel-Composer.git
synced 2025-01-12 15:26:33 +01:00
551 lines
No EOL
16 KiB
Text
551 lines
No EOL
16 KiB
Text
function Node_create_Particle(_x, _y) {
|
|
var node = new Node_Particle(_x, _y);
|
|
ds_list_add(PANEL_GRAPH.nodes_list, node);
|
|
return node;
|
|
}
|
|
|
|
enum ANIM_END_ACTION {
|
|
loop,
|
|
pingpong,
|
|
destroy,
|
|
}
|
|
|
|
function __part() constructor {
|
|
seed = irandom(99999);
|
|
active = false;
|
|
surf = noone;
|
|
x = 0;
|
|
y = 0;
|
|
sx = 0;
|
|
sy = 0;
|
|
ac = 0;
|
|
g = 0;
|
|
wig = 0;
|
|
|
|
boundary_data = -1;
|
|
|
|
fx = 0;
|
|
fy = 0;
|
|
|
|
gy = 0;
|
|
|
|
scx = 1;
|
|
scy = 1;
|
|
scx_s = 1;
|
|
scy_s = 1;
|
|
|
|
rot = 0;
|
|
follow = false;
|
|
rot_s = 0;
|
|
|
|
col = -1;
|
|
alp = 1;
|
|
alp_draw = alp;
|
|
alp_fade = 0;
|
|
|
|
life = 0;
|
|
life_total = 0;
|
|
|
|
anim_speed = 1;
|
|
anim_end = ANIM_END_ACTION.loop;
|
|
|
|
is_loop = false;
|
|
|
|
|
|
function create(_surf, _x, _y, _life) {
|
|
active = true;
|
|
surf = _surf;
|
|
x = _x;
|
|
y = _y;
|
|
gy = 0;
|
|
|
|
life = _life;
|
|
life_total = life;
|
|
}
|
|
|
|
function setPhysic(_sx, _sy, _ac, _g, _wig) {
|
|
sx = _sx;
|
|
sy = _sy;
|
|
ac = _ac;
|
|
g = _g;
|
|
|
|
wig = _wig;
|
|
}
|
|
function setTransform(_scx, _scy, _scxs, _scys, _rot, _rots, _follow) {
|
|
scx = _scx;
|
|
scy = _scy;
|
|
scx_s = _scxs;
|
|
scy_s = _scys;
|
|
rot = _rot;
|
|
rot_s = _rots;
|
|
follow = _follow;
|
|
}
|
|
function setDraw(_col, _alp, _fade) {
|
|
col = _col;
|
|
alp = _alp;
|
|
alp_draw = _alp;
|
|
alp_fade = _fade;
|
|
}
|
|
|
|
function kill() {
|
|
active = false;
|
|
}
|
|
|
|
static step = function() {
|
|
if(!active) return;
|
|
var xp = x, yp = y;
|
|
x += sx;
|
|
y += sy;
|
|
|
|
var dirr = point_direction(0, 0, sx, sy);
|
|
var diss = point_distance(0, 0, sx, sy);
|
|
if(diss > 0) {
|
|
diss += ac;
|
|
dirr += random_range(-wig, wig);
|
|
sx = lengthdir_x(diss, dirr);
|
|
sy = lengthdir_y(diss, dirr);
|
|
}
|
|
|
|
gy += g;
|
|
y += gy;
|
|
|
|
if(scx_s < 0) scx = max(scx + scx_s, 0);
|
|
else scx = scx + scx_s;
|
|
if(scy_s < 0) scy = max(scy + scy_s, 0);
|
|
else scy = scy + scy_s;
|
|
|
|
if(follow)
|
|
rot = point_direction(xp, yp, x, y);
|
|
else
|
|
rot += rot_s;
|
|
alp_draw = alp * eval_bezier_cubic(1 - life / life_total, alp_fade[0], alp_fade[1], alp_fade[2], alp_fade[3]);
|
|
|
|
if(life-- < 0) kill();
|
|
}
|
|
|
|
function draw(exact) {
|
|
if(!active) return;
|
|
var ss = surf;
|
|
if(is_array(surf)) {
|
|
var ind = abs(round((life_total - life) * anim_speed));
|
|
var len = array_length(surf);
|
|
|
|
switch(anim_end) {
|
|
case ANIM_END_ACTION.loop:
|
|
ss = surf[safe_mod(ind, len)];
|
|
break;
|
|
case ANIM_END_ACTION.pingpong:
|
|
var ping = safe_mod(ind, (len - 1) * 2 + 1);
|
|
ss = surf[ping >= len? (len - 1) * 2 - ping : ping];
|
|
break;
|
|
case ANIM_END_ACTION.destroy:
|
|
if(ind >= len) return;
|
|
else ss = surf[ind];
|
|
break;
|
|
}
|
|
}
|
|
if(!is_surface(ss)) return;
|
|
|
|
var cc = (col == -1)? c_white : gradient_eval(col, 1 - life / life_total);
|
|
var _xx, _yy;
|
|
var s_w = surface_get_width(ss) * scx;
|
|
var s_h = surface_get_height(ss) * scy;
|
|
|
|
if(boundary_data == -1) {
|
|
var _pp = point_rotate(-s_w / 2, -s_h / 2, 0, 0, rot);
|
|
_xx = x + _pp[0];
|
|
_yy = y + _pp[1];
|
|
} else {
|
|
var ww = boundary_data[2] + boundary_data[0];
|
|
var hh = boundary_data[3] + boundary_data[1];
|
|
|
|
var cx = (boundary_data[0] + boundary_data[2]) / 2;
|
|
var cy = (boundary_data[1] + boundary_data[3]) / 2;
|
|
|
|
var _pp = point_rotate(-cx, -cy, 0, 0, rot);
|
|
|
|
_xx = x + cx + _pp[0] * scx;
|
|
_yy = y + cy + _pp[1] * scy;
|
|
}
|
|
|
|
if(exact) {
|
|
_xx = round(_xx);
|
|
_yy = round(_yy);
|
|
}
|
|
|
|
draw_surface_ext_safe(ss, _xx, _yy, scx, scy, rot, cc, alp_draw);
|
|
}
|
|
|
|
function getPivot() {
|
|
if(boundary_data == -1)
|
|
return [x, y];
|
|
|
|
var ww = (boundary_data[2] - boundary_data[0]) * scx;
|
|
var hh = (boundary_data[3] - boundary_data[1]) * scy;
|
|
var cx = x + boundary_data[0] + ww / 2;
|
|
var cy = y + boundary_data[1] + hh / 2;
|
|
|
|
return [cx, cy];
|
|
}
|
|
}
|
|
|
|
enum PARTICLE_BLEND_MODE {
|
|
normal,
|
|
additive
|
|
}
|
|
|
|
function Node_Particle(_x, _y) : Node(_x, _y) constructor {
|
|
name = "Particle";
|
|
auto_update = false;
|
|
use_cache = true;
|
|
|
|
inputs[| 0] = nodeValue(0, "Particle", self, JUNCTION_CONNECT.input, VALUE_TYPE.surface, 0)
|
|
.setDisplay(noone, "particles");
|
|
|
|
inputs[| 1] = nodeValue(1, "Output dimension", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, def_surf_size2, VALUE_TAG.dimension_2d)
|
|
.setDisplay(VALUE_DISPLAY.vector);
|
|
|
|
inputs[| 2] = nodeValue(2, "Spawn delay", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 4);
|
|
inputs[| 3] = nodeValue(3, "Spawn amount", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 2);
|
|
inputs[| 4] = nodeValue(4, "Spawn area", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 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[| 5] = nodeValue(5, "Spawn distribution", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0)
|
|
.setDisplay(VALUE_DISPLAY.enum_button, [ "Area", "Border" ]);
|
|
|
|
inputs[| 6] = nodeValue(6, "Lifespan", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [ 20, 30 ])
|
|
.setDisplay(VALUE_DISPLAY.range);
|
|
|
|
inputs[| 7] = nodeValue(7, "Spawn direction", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [ 45, 135 ])
|
|
.setDisplay(VALUE_DISPLAY.rotation_range);
|
|
inputs[| 8] = nodeValue(8, "Acceleration", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 0, 0 ])
|
|
.setDisplay(VALUE_DISPLAY.range);
|
|
|
|
inputs[| 9] = nodeValue(9, "Orientation", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [0, 0])
|
|
.setDisplay(VALUE_DISPLAY.rotation_range);
|
|
|
|
inputs[| 10] = nodeValue(10, "Rotational speed", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 0, 0 ])
|
|
.setDisplay(VALUE_DISPLAY.range);
|
|
|
|
inputs[| 11] = nodeValue(11, "Spawn scale", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 1, 1, 1, 1 ] )
|
|
.setDisplay(VALUE_DISPLAY.vector_range);
|
|
inputs[| 12] = nodeValue(12, "Scaling speed", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 0, 0 ] )
|
|
.setDisplay(VALUE_DISPLAY.vector);
|
|
|
|
inputs[| 13] = nodeValue(13, "Color over lifetime", self, JUNCTION_CONNECT.input, VALUE_TYPE.color, c_white)
|
|
.setDisplay(VALUE_DISPLAY.gradient);
|
|
inputs[| 14] = nodeValue(14, "Alpha", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 1, 1 ])
|
|
.setDisplay(VALUE_DISPLAY.range);
|
|
inputs[| 15] = nodeValue(15, "Alpha over time", self, JUNCTION_CONNECT.input, VALUE_TYPE.curve, [1, 1, 1, 1]);
|
|
|
|
inputs[| 16] = nodeValue(16, "Rotate by direction", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, false);
|
|
|
|
inputs[| 17] = nodeValue(17, "Spawn type", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0)
|
|
.setDisplay(VALUE_DISPLAY.enum_button, [ "Stream", "Burst" ]);
|
|
|
|
inputs[| 18] = nodeValue(18, "Spawn size", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [ 1, 1 ] )
|
|
.setDisplay(VALUE_DISPLAY.range);
|
|
|
|
inputs[| 19] = nodeValue(19, "Draw exact", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, true );
|
|
|
|
inputs[| 20] = nodeValue(20, "Spawn velocity", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, [1, 2] )
|
|
.setDisplay(VALUE_DISPLAY.range);
|
|
|
|
inputs[| 21] = nodeValue(21, "Gravity", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, 0 );
|
|
inputs[| 22] = nodeValue(22, "Wiggle", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, 0 );
|
|
|
|
inputs[| 23] = nodeValue(23, "Loop", self, JUNCTION_CONNECT.input, VALUE_TYPE.boolean, true );
|
|
|
|
inputs[| 24] = nodeValue(24, "Blend mode", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0 )
|
|
.setDisplay(VALUE_DISPLAY.enum_scroll, [ "Normal", "Additive" ]);
|
|
|
|
inputs[| 25] = nodeValue(25, "Surface array", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 0 )
|
|
.setDisplay(VALUE_DISPLAY.enum_scroll, [ "Random", "Order", "Animation" ])
|
|
.setVisible(false);
|
|
|
|
inputs[| 26] = nodeValue(26, "Animation speed", self, JUNCTION_CONNECT.input, VALUE_TYPE.float, 1 )
|
|
.setVisible(false);
|
|
|
|
inputs[| 27] = nodeValue(27, "Scatter", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, 1)
|
|
.setDisplay(VALUE_DISPLAY.enum_button, [ "Uniform", "Random", "Data" ]);
|
|
|
|
inputs[| 28] = nodeValue(28, "Boundary data", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, [])
|
|
.setVisible(false, true);
|
|
|
|
inputs[| 29] = nodeValue(29, "On animation end", self, JUNCTION_CONNECT.input, VALUE_TYPE.integer, ANIM_END_ACTION.loop)
|
|
.setDisplay(VALUE_DISPLAY.enum_button, [ "Loop", "Ping pong", "Destroy" ])
|
|
.setVisible(false);
|
|
|
|
input_display_list = [
|
|
["Output", true], 1,
|
|
["Sprite", false], 0, 25, 26, 29,
|
|
["Spawn", true], 17, 2, 3, 4, 5, 27, 28, 6,
|
|
["Movement", true], 7, 20, 8,
|
|
["Physics", true], 21, 22,
|
|
["Rotation", true], 16, 9, 10,
|
|
["Scale", true], 11, 18, 12,
|
|
["Color", true], 13, 14, 15,
|
|
["Render", true], 24, 19, 23,
|
|
];
|
|
|
|
outputs[| 0] = nodeValue(0, "Surface out", self, JUNCTION_CONNECT.output, VALUE_TYPE.surface, PIXEL_SURFACE);
|
|
|
|
seed = irandom(9999999);
|
|
def_surface = -1;
|
|
|
|
parts = ds_list_create();
|
|
for(var i = 0; i < PREF_MAP[? "part_max_amount"]; i++)
|
|
ds_list_add(parts, new __part());
|
|
|
|
outputs[| 1] = nodeValue(1, "Particle data", self, JUNCTION_CONNECT.output, VALUE_TYPE.object, parts );
|
|
|
|
function spawn() {
|
|
randomize();
|
|
var _inSurf = inputs[| 0].getValue();
|
|
|
|
if(_inSurf == 0) {
|
|
if(def_surface == -1 || !surface_exists(def_surface)) {
|
|
def_surface = PIXEL_SURFACE;
|
|
surface_set_target(def_surface);
|
|
draw_clear(c_white);
|
|
surface_reset_target();
|
|
}
|
|
_inSurf = def_surface;
|
|
}
|
|
|
|
var _spawn_amount = inputs[| 3].getValue();
|
|
var _amo = _spawn_amount;
|
|
|
|
var _spawn_area = inputs[| 4].getValue();
|
|
var _distrib = inputs[| 5].getValue();
|
|
var _scatter = inputs[| 27].getValue();
|
|
|
|
var _life = inputs[| 6].getValue();
|
|
var _direction = inputs[| 7].getValue();
|
|
var _velocity = inputs[| 20].getValue();
|
|
|
|
var _accel = inputs[| 8].getValue();
|
|
var _grav = inputs[| 21].getValue();
|
|
var _wigg = inputs[| 22].getValue();
|
|
|
|
var _follow = inputs[| 16].getValue();
|
|
var _rotation = inputs[| 9].getValue();
|
|
var _rotation_speed = inputs[| 10].getValue();
|
|
var _scale = inputs[| 11].getValue();
|
|
var _size = inputs[| 18].getValue();
|
|
var _scale_speed = inputs[| 12].getValue();
|
|
|
|
var _loop = inputs[| 23].getValue();
|
|
|
|
var _color = inputs[| 13].getValue();
|
|
var _alpha = inputs[| 14].getValue();
|
|
var _fade = inputs[| 15].getValue();
|
|
|
|
var _arr_type = inputs[| 25].getValue();
|
|
var _anim_speed = inputs[| 26].getValue();
|
|
var _anim_end = inputs[| 29].getValue();
|
|
|
|
if(_rotation[1] < _rotation[0]) _rotation[1] += 360;
|
|
|
|
for(var i = 0; i < PREF_MAP[? "part_max_amount"]; i++) {
|
|
if(!parts[| i].active) {
|
|
var _spr = _inSurf, _index = 0;
|
|
if(is_array(_inSurf)) {
|
|
if(_arr_type == 0) {
|
|
_index = irandom(array_length(_inSurf) - 1);
|
|
_spr = _inSurf[_index];
|
|
} else if(_arr_type == 1) {
|
|
_index = safe_mod(spawn_index, array_length(_inSurf));
|
|
_spr = _inSurf[_index];
|
|
} else if(_arr_type == 2)
|
|
_spr = _inSurf;
|
|
}
|
|
var xx = 0;
|
|
var yy = 0;
|
|
|
|
if(_scatter == 2) {
|
|
var _b_data = inputs[| 28].getValue();
|
|
if(!is_array(_b_data) || array_length(_b_data) <= 0) return;
|
|
var _b = _b_data[safe_mod(_index, array_length(_b_data))];
|
|
if(!is_array(_b) || array_length(_b) != 4) return;
|
|
|
|
xx = array_safe_get(_spawn_area, 0) - array_safe_get(_spawn_area, 2);
|
|
yy = array_safe_get(_spawn_area, 1) - array_safe_get(_spawn_area, 3);
|
|
|
|
parts[| i].boundary_data = _b;
|
|
} else {
|
|
var sp = area_get_random_point(_spawn_area, _distrib, _scatter, spawn_index, _spawn_amount);
|
|
xx = sp[0];
|
|
yy = sp[1];
|
|
|
|
parts[| i].boundary_data = -1;
|
|
}
|
|
|
|
var _lif = random_range(_life[0], _life[1]);
|
|
|
|
var _rot = random_range(_rotation[0], _rotation[1]);
|
|
var _rot_spd = random_range(_rotation_speed[0], _rotation_speed[1]);
|
|
|
|
var _dirr = random_range(_direction[0], _direction[1]);
|
|
|
|
var _velo = random_range(_velocity[0], _velocity[1]);
|
|
var _vx = lengthdir_x(_velo, _dirr);
|
|
var _vy = lengthdir_y(_velo, _dirr);
|
|
var _acc = random_range(_accel[0], _accel[1]);
|
|
|
|
var _ss = random_range(_size[0], _size[1]);
|
|
var _scx = random_range(_scale[0], _scale[1]) * _ss;
|
|
var _scy = random_range(_scale[2], _scale[3]) * _ss;
|
|
|
|
var _alp = random_range(_alpha[0], _alpha[1]);
|
|
|
|
parts[| i].create(_spr, xx, yy, _lif);
|
|
parts[| i].anim_speed = _anim_speed;
|
|
parts[| i].anim_end = _anim_end;
|
|
|
|
parts[| i].setPhysic(_vx, _vy, _acc, _grav, _wigg);
|
|
parts[| i].setTransform(_scx, _scy, _scale_speed[0], _scale_speed[1], _rot, _rot_spd, _follow);
|
|
parts[| i].setDraw(_color, _alp, _fade);
|
|
setUpPart(parts[| i]);
|
|
spawn_index = safe_mod(spawn_index + 1, PREF_MAP[? "part_max_amount"]);
|
|
|
|
if(_loop && ANIMATOR.current_frame + _lif > ANIMATOR.frames_total)
|
|
parts[| i].is_loop = true;
|
|
|
|
if(--_amo <= 0)
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
function setUpPart(part) {}
|
|
|
|
function reset() {
|
|
spawn_index = 0;
|
|
for(var i = 0; i < PREF_MAP[? "part_max_amount"]; i++) {
|
|
if(parts[| i].is_loop)
|
|
parts[| i].is_loop = false;
|
|
else
|
|
parts[| i].kill();
|
|
}
|
|
render();
|
|
}
|
|
|
|
function updateParticle() {
|
|
var jun = outputs[| 1];
|
|
for(var j = 0; j < ds_list_size(jun.value_to); j++) {
|
|
if(jun.value_to[| j].value_from == jun) {
|
|
jun.value_to[| j].node.doUpdate();
|
|
}
|
|
}
|
|
|
|
render();
|
|
}
|
|
|
|
function resetPartPool() {
|
|
var _part_amo = PREF_MAP[? "part_max_amount"];
|
|
if(_part_amo > ds_list_size(parts)) {
|
|
repeat(_part_amo - ds_list_size(parts)) {
|
|
ds_list_add(parts, new __part());
|
|
}
|
|
} else if(_part_amo < ds_list_size(parts)) {
|
|
repeat(ds_list_size(parts) - _part_amo) {
|
|
ds_list_delete(parts, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
static step = function() {
|
|
var _inSurf = inputs[| 0].getValue();
|
|
var _scatt = inputs[| 27].getValue();
|
|
|
|
inputs[| 25].setVisible(false);
|
|
inputs[| 26].setVisible(false);
|
|
inputs[| 28].setVisible(_scatt == 2);
|
|
|
|
if(is_array(_inSurf)) {
|
|
inputs[| 25].setVisible(true);
|
|
var _type = inputs[| 25].getValue();
|
|
if(_type == 2) {
|
|
inputs[| 26].setVisible(true);
|
|
inputs[| 29].setVisible(true);
|
|
}
|
|
}
|
|
|
|
resetPartPool();
|
|
var _spawn_type = inputs[| 17].getValue();
|
|
if(_spawn_type == 0)
|
|
inputs[| 2].name = "Spawn delay";
|
|
else
|
|
inputs[| 2].name = "Spawn frame";
|
|
|
|
var _spawn_delay = inputs[| 2].getValue();
|
|
|
|
if(ANIMATOR.is_playing && ANIMATOR.frame_progress) {
|
|
if(ANIMATOR.current_frame == 0) reset();
|
|
|
|
if(_spawn_type == 0) {
|
|
if(safe_mod(ANIMATOR.current_frame, _spawn_delay) == 0)
|
|
spawn();
|
|
} else if(_spawn_type == 1) {
|
|
if(ANIMATOR.current_frame == _spawn_delay)
|
|
spawn();
|
|
}
|
|
|
|
for(var i = 0; i < PREF_MAP[? "part_max_amount"]; i++)
|
|
parts[| i].step();
|
|
updateParticle();
|
|
|
|
updateForward();
|
|
}
|
|
|
|
if(ANIMATOR.is_scrubing) {
|
|
recoverCache();
|
|
}
|
|
}
|
|
|
|
static drawOverlay = function(_active, _x, _y, _s, _mx, _my) {
|
|
inputs[| 4].drawOverlay(_active, _x, _y, _s, _mx, _my);
|
|
if(onDrawOverlay != -1)
|
|
onDrawOverlay(_active, _x, _y, _s, _mx, _my);
|
|
}
|
|
|
|
static onDrawOverlay = -1;
|
|
|
|
function render() {
|
|
var _dim = inputs[| 1].getValue();
|
|
var _exact = inputs[| 19].getValue();
|
|
var _blend = inputs[| 24].getValue();
|
|
|
|
var _outSurf = outputs[| 0].getValue();
|
|
|
|
if(is_surface(_outSurf))
|
|
surface_size_to(_outSurf, _dim[0], _dim[1]);
|
|
else {
|
|
_outSurf = surface_create_valid(_dim[0], _dim[1]);
|
|
outputs[| 0].setValue(_outSurf);
|
|
}
|
|
|
|
surface_set_target(_outSurf);
|
|
draw_clear_alpha(c_white, 0);
|
|
|
|
switch(_blend) {
|
|
case PARTICLE_BLEND_MODE.normal : gpu_set_blendmode(bm_normal); break;
|
|
case PARTICLE_BLEND_MODE.additive : gpu_set_blendmode(bm_add); break;
|
|
}
|
|
|
|
for(var i = 0; i < PREF_MAP[? "part_max_amount"]; i++)
|
|
parts[| i].draw(_exact);
|
|
|
|
gpu_set_blendmode(bm_normal);
|
|
surface_reset_target();
|
|
|
|
cacheCurrentFrame(_outSurf);
|
|
}
|
|
|
|
static update = function() {
|
|
reset();
|
|
}
|
|
doUpdate();
|
|
render();
|
|
} |