enum ANIM_END_ACTION { loop, pingpong, destroy, } enum PARTICLE_BLEND_MODE { normal, alpha, additive } enum PARTICLE_RENDER_TYPE { surface, line, } function __particleObject() constructor { active = false; x = 0; y = 0; rot = 0; scx = 1; scy = 1; x_history = []; y_history = []; surf = noone; blend = c_white; alp = 1; static kill = function() {} static step = function() {} __temp_pt = [ 0, 0 ]; static draw = function(exact, surf_w, surf_h) { #region if(!surface_exists(surf)) return; var _sw = surface_get_width(surf) * scx; var _sh = surface_get_height(surf) * scy; point_rotate(-_sw / 2, -_sh / 2, 0, 0, rot, __temp_pt); draw_surface_ext(surf, x + __temp_pt[0], y + __temp_pt[1], scx, scy, rot, blend, alp); } #endregion static clone = function() { #region var _p = new __particleObject(); struct_override(_p, self); return _p; } #endregion } function __part(_node) : __particleObject() constructor { seed = irandom(99999); node = _node; /////////////////////// Lifespans /////////////////////// life = 0; life_total = 0; life_incr = 0; step_int = 0; /////////////////////// Transforms /////////////////////// prevx = 0; prevy = 0; speedx = 0; speedy = 0; turning = 0; turnSpd = 0; accel = 0; spVec = [ 0, 0 ]; grav = 0; gravDir = -90; gravX = 0; gravY = 0; sc_sx = 1; sc_sy = 1; sct = noone; scx_history = []; scy_history = []; follow = false; rot_s = 0; path = noone; pathIndex = 0; pathPos = new __vec2(); pathDiv = noone; /////////////////////// Render /////////////////////// render_type = PARTICLE_RENDER_TYPE.surface; arr_type = 0; drawx = 0; drawy = 0; drawrot = 0; drawsx = 0; drawsy = 0; col = -1; alp_draw = alp; alp_fade = 0; currColor = c_white; blend_history = []; alp_history = []; anim_speed = 1; anim_end = ANIM_END_ACTION.loop; anim_stre = false; anim_len = 1; line_draw = 1; /////////////////////// Physics /////////////////////// ground = false; ground_y = 0; ground_bounce = 0; ground_friction = 1; trailLife = 0; trailActive = false; frame = 0; static reset = function() { #region INLINE surf = noone; prevx = undefined; prevy = undefined; } #endregion static create = function(_surf, _x, _y, _life) { #region INLINE active = true; surf = _surf; x = _x; y = _y; drawx = x; drawy = y; anim_len = is_array(surf)? array_length(surf) : 1; life_incr = 0; life = _life; life_total = life; if(node.onPartCreate != noone) node.onPartCreate(self); trailLife = 0; x_history = array_create(life); y_history = array_create(life); scx_history = array_create(life); scy_history = array_create(life); blend_history = array_create(life); alp_history = array_create(life); } #endregion static setPhysic = function(_sx, _sy, _ac, _g, _gDir, _turn, _turnSpd) { #region INLINE speedx = _sx; speedy = _sy; accel = _ac; grav = _g; gravDir = _gDir; gravX = lengthdir_x(grav, gravDir); gravY = lengthdir_y(grav, gravDir); turning = _turn; turnSpd = _turnSpd; spVec[0] = point_distance(0, 0, speedx, speedy); spVec[1] = point_direction(0, 0, speedx, speedy); } #endregion static setWiggle = function(wiggle_maps) { #region INLINE wig_psx = wiggle_maps.wig_psx; wig_psy = wiggle_maps.wig_psy; wig_rot = wiggle_maps.wig_rot; wig_scx = wiggle_maps.wig_scx; wig_scy = wiggle_maps.wig_scy; wig_dir = wiggle_maps.wig_dir; } #endregion static setGround = function(_ground, _ground_offset, _ground_bounce, _ground_frict) { #region INLINE ground = _ground; ground_y = y + _ground_offset; ground_bounce = _ground_bounce; ground_friction = clamp(1 - _ground_frict, 0, 1); } #endregion static setTransform = function(_scx, _scy, _sct, _rot, _rots, _follow) { #region INLINE sc_sx = _scx; sc_sy = _scy; sct = _sct; rot = _rot; rot_s = _rots; follow = _follow; } #endregion static setDraw = function(_col, _blend, _alp, _fade) { #region INLINE col = _col; blend = _blend; alp = _alp; alp_draw = _alp; alp_fade = _fade; } #endregion static setPath = function(_path, _pathDiv) { #region INLINE path = _path; pathDiv = _pathDiv; } #endregion static kill = function(callDestroy = true) { #region INLINE active = false; if(callDestroy && node.onPartDestroy != noone) node.onPartDestroy(self); } #endregion static step = function(frame = 0) { #region INLINE trailLife++; if(!active) return; x += speedx; self.frame = frame; random_set_seed(seed + life); if(ground && y + speedy > ground_y) { y = ground_y; speedy = -speedy * ground_bounce; if(abs(speedy) < 0.1) speedx *= ground_friction; } else y += speedy; var dirr = point_direction(0, 0, speedx, speedy); var diss = point_distance(0, 0, speedx, speedy); diss = max(0, diss + accel); if(speedx != 0 || speedy != 0) { dirr += wig_dir.get(seed + life); if(turning != 0) { var trn = turning; if(turnSpd > 0) trn = turning * diss * turnSpd; else if(turnSpd < 0) trn = turning / diss * turnSpd; dirr += trn } } speedx = lengthdir_x(diss, dirr) + gravX; speedy = lengthdir_y(diss, dirr) + gravY; if(follow) rot = spVec[1]; else rot += rot_s; if(node.onPartStep != noone && step_int > 0 && safe_mod(life, step_int) == 0) node.onPartStep(self); if(life-- < 0) kill(); if(prevx != undefined) { spVec[0] = point_distance(prevx, prevy, x, y); spVec[1] = point_direction(prevx, prevy, x, y); } x_history[life_incr] = drawx; y_history[life_incr] = drawy; life_incr++; prevx = x; prevy = y; drawx = x; drawy = y; drawrot = rot; drawsx = sc_sx; drawsy = sc_sy; drawx += wig_psx.get(seed + life); drawy += wig_psy.get(seed + life); drawrot += wig_rot.get(seed + life); drawsx += wig_scy.get(seed + life); drawsy += wig_scy.get(seed + life); } #endregion static draw = function(exact, surf_w, surf_h) { #region INLINE if(render_type == PARTICLE_RENDER_TYPE.line) { var _trail_ed = min(life_incr, life_total); var _trail_st = max(0, trailLife - line_draw); var _trail_len = _trail_ed - _trail_st; if(_trail_len <= 0) return; } var ss = surf; var lifeRat = 1 - life / life_total; var scCurve = sct == noone? 1 : sct.get(lifeRat); scx = drawsx * scCurve; scy = drawsy * scCurve; if(arr_type == 2 && surf != noone && is_array(surf)) { var _life_prog = life_total - life; var ind = anim_stre? _life_prog / life_total * anim_speed * (anim_len - 1) : _life_prog * anim_speed; ind = abs(round(ind)); switch(anim_end) { case ANIM_END_ACTION.loop: ss = surf[safe_mod(ind, anim_len)]; break; case ANIM_END_ACTION.pingpong: var ping = safe_mod(ind, (anim_len - 1) * 2 + 1); ss = surf[ping >= anim_len? (anim_len - 1) * 2 - ping : ping]; break; case ANIM_END_ACTION.destroy: if(ind >= anim_len) { kill(); return; } ss = surf[ind]; break; } } else if(arr_type == 3) { var _sca = round(min(scx, scy)); ss = array_safe_get_fast(surf, clamp(_sca, 0, array_length(surf) - 1)); } var surface = node.surface_cache[$ ss]; var _useS = is_surface(surface); if(arr_type == 3) { scx = 1; scy = 1; } scx_history[life_incr - 1] = scx; scy_history[life_incr - 1] = scy; var _xx, _yy; var s_w = (_useS? surface_get_width_safe(surface) : 1) * scx; var s_h = (_useS? surface_get_height_safe(surface) : 1) * scy; var _pp = point_rotate(-s_w / 2, -s_h / 2, 0, 0, rot); _xx = drawx + _pp[0]; _yy = drawy + _pp[1]; if(path != noone) { var _div = pathDiv.get(lifeRat); pathPos = path.getPointRatio(clamp(lifeRat, 0, 0.99), pathIndex, pathPos); _xx = _xx * _div + pathPos.x; _yy = _yy * _div + pathPos.y; } if(exact) { _xx = round(_xx); _yy = round(_yy); } var x0 = _xx - s_w * 1.5; var y0 = _yy - s_h * 1.5; var x1 = _xx + s_w * 1.5; var y1 = _yy + s_h * 1.5; var cc = (col == -1)? c_white : col.eval(lifeRat); if(blend != c_white) cc = colorMultiply(blend, cc); alp_draw = alp * (alp_fade == noone? 1 : alp_fade.get(lifeRat)) * _color_get_alpha(cc); blend_history[life_incr - 1] = cc; alp_history[life_incr - 1] = alp_draw; currColor = cola(cc, alp_draw); if(_useS && (x0 > surf_w || y0 > surf_h || x1 < 0 || y1 < 0)) return; switch(render_type) { case PARTICLE_RENDER_TYPE.surface : if(_useS) draw_surface_ext_safe(surface, _xx, _yy, scx, scy, drawrot, cc, alp_draw); else { var ss = round(min(scx, scy)); if(round(ss) == 0) return; var _s = shader_current(); shader_reset(); draw_set_color(cc); draw_set_alpha(alp_draw); dynaSurf_circle_fill(_xx, _yy, exact? round(ss) : ss); draw_set_alpha(1); shader_set(_s); } break; case PARTICLE_RENDER_TYPE.line : var _ox, _nx, _oy, _ny; var _osx, _nsx, _osy, _nsy; var _oc, _nc, _oa, _na; for( var j = 0; j < _trail_len; j++ ) { var _index = _trail_st + j; _nx = x_history[ _index]; _ny = y_history[ _index]; _nsx = scx_history[ _index]; _nsy = scy_history[ _index]; _nc = blend_history[_index]; _na = alp_history[ _index]; if(j) { draw_set_color(_nc); draw_set_alpha(_na); if(_osx == 1 && _nsx == 1) draw_line(_ox, _oy, _nx, _ny); else if(_osx == _nsx) draw_line_width(_ox, _oy, _nx, _ny, _osx); else draw_line_width2(_ox, _oy, _nx, _ny, _osx, _nsx, false); draw_set_alpha(1); } _ox = _nx ; _oy = _ny ; _osx = _nsx; _osy = _nsy; _oc = _nc ; _oa = _na ; } break; } } #endregion static getPivot = function() { #region INLINE return [x, y]; } #endregion static clone = function() { #region var _p = new __part(node); struct_override(_p, self); return _p; } #endregion } #region helper #macro UPDATE_PART_FORWARD static updateParticleForward = function() { \ var pt = outputs[| 0]; \ for( var i = 0; i < array_length(pt.value_to); i++ ) { \ var _n = pt.value_to[i]; \ if(_n.value_from != pt) continue; \ \ if(variable_struct_exists(_n.node, "updateParticleForward")) \ _n.node.updateParticleForward(); \ } \ } #endregion