mirror of
https://github.com/JannisX11/blockbench.git
synced 2025-01-18 15:26:19 +08:00
28315784d7
Fix two Wintersky issues
1108 lines
66 KiB
JavaScript
1108 lines
66 KiB
JavaScript
(function (global, factory) {
|
|
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('three'), require('molangjs'), require('tinycolor2')) :
|
|
typeof define === 'function' && define.amd ? define(['three', 'molangjs', 'tinycolor2'], factory) :
|
|
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.Wintersky = factory(global.THREE, global.Molang, global.tinycolor));
|
|
}(this, (function (THREE$1, Molang, tinycolor) { 'use strict';
|
|
|
|
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
|
|
|
|
var THREE__default = /*#__PURE__*/_interopDefaultLegacy(THREE$1);
|
|
var Molang__default = /*#__PURE__*/_interopDefaultLegacy(Molang);
|
|
var tinycolor__default = /*#__PURE__*/_interopDefaultLegacy(tinycolor);
|
|
|
|
const Wintersky = {
|
|
emitters: [],
|
|
space: new THREE__default['default'].Object3D(),
|
|
updateFacingRotation(camera) {
|
|
Wintersky.emitters.forEach(emitter => {
|
|
emitter.updateFacingRotation(camera);
|
|
});
|
|
},
|
|
global_options: {
|
|
max_emitter_particles: 30000,
|
|
tick_rate: 30,
|
|
loop_mode: 'auto', // looping, once
|
|
parent_mode: 'world', // entity, locator
|
|
get scale() {
|
|
return Wintersky.global_options._scale;
|
|
},
|
|
set scale(val) {
|
|
Wintersky.global_options._scale = val;
|
|
Wintersky.emitters.forEach(emitter => {
|
|
emitter.local_space.scale.set(val, val, val);
|
|
emitter.global_space.scale.set(val, val, val);
|
|
});
|
|
//Wintersky.space.scale.set(val, val, val);
|
|
},
|
|
_scale: 1
|
|
}
|
|
};
|
|
|
|
const img = "";
|
|
|
|
const img$1 = "";
|
|
|
|
const img$2 = "";
|
|
|
|
const img$3 = "";
|
|
|
|
const img$4 = "";
|
|
|
|
function parseColor(input) {
|
|
return new tinycolor__default['default'](input).toHexString();
|
|
}
|
|
|
|
class Config {
|
|
constructor(config, options = 0) {
|
|
this.texture = new THREE.Texture(new Image());
|
|
this.reset();
|
|
|
|
if (config && config.particle_effect) {
|
|
this.setFromJSON(config);
|
|
} else if (typeof config == 'object') {
|
|
Object.assign(this, config);
|
|
}
|
|
if (options.path) this.set('file_path', options.path);
|
|
}
|
|
reset() {
|
|
this.texture.image.src = img;
|
|
this.texture.magFilter = THREE.NearestFilter;
|
|
this.texture.minFilter = THREE.NearestFilter;
|
|
|
|
this.identifier = '';
|
|
this.file_path = '';
|
|
this.curves = {};
|
|
this.space_local_position = false;
|
|
this.space_local_rotation = false;
|
|
this.variables_creation_vars = [];
|
|
this.variables_tick_vars = [];
|
|
|
|
this.emitter_rate_mode = 'steady';
|
|
this.emitter_rate_rate = '';
|
|
this.emitter_rate_amount = '';
|
|
this.emitter_rate_maximum = '';
|
|
this.emitter_lifetime_mode = 'once';
|
|
this.emitter_lifetime_active_time = '';
|
|
this.emitter_lifetime_sleep_time = '';
|
|
this.emitter_lifetime_activation = '';
|
|
this.emitter_lifetime_expiration = '';
|
|
this.emitter_shape_mode = 'point';
|
|
this.emitter_shape_offset = [0, 0, 0];
|
|
this.emitter_shape_radius = '';
|
|
this.emitter_shape_half_dimensions = [0, 0, 0];
|
|
this.emitter_shape_plane_normal = [0, 0, 0];
|
|
this.emitter_shape_surface_only = false;
|
|
this.particle_appearance_size = [0, 0];
|
|
this.particle_appearance_facing_camera_mode = 'rotate_xyz';
|
|
this.particle_appearance_material = 'particles_alpha';
|
|
this.particle_direction_mode = 'outwards';
|
|
this.particle_direction_direction = [0, 0, 0];
|
|
this.particle_motion_mode = 'dynamic';
|
|
this.particle_motion_linear_speed = '';
|
|
this.particle_motion_linear_acceleration = [0, 0, 0];
|
|
this.particle_motion_linear_drag_coefficient = '';
|
|
this.particle_motion_relative_position = [];
|
|
this.particle_motion_direction = [];
|
|
this.particle_rotation_mode = 'dynamic';
|
|
this.particle_rotation_initial_rotation = '';
|
|
this.particle_rotation_rotation_rate = '';
|
|
this.particle_rotation_rotation_acceleration = '';
|
|
this.particle_rotation_rotation_drag_coefficient = '';
|
|
this.particle_rotation_rotation = '';
|
|
this.particle_lifetime_mode = 'time';
|
|
this.particle_lifetime_max_lifetime = '';
|
|
this.particle_lifetime_kill_plane = [0, 0, 0, 0];
|
|
this.particle_lifetime_expiration_expression = '';
|
|
this.particle_lifetime_expire_in = [];
|
|
this.particle_lifetime_expire_outside = [];
|
|
this.particle_texture_width = 0;
|
|
this.particle_texture_height = 0;
|
|
this.particle_texture_path = '';
|
|
this.particle_texture_image = '';
|
|
this.particle_texture_mode = 'static';
|
|
this.particle_texture_uv = [0, 0];
|
|
this.particle_texture_uv_size = [0, 0];
|
|
this.particle_texture_uv_step = [0, 0];
|
|
this.particle_texture_frames_per_second = 0;
|
|
this.particle_texture_max_frame = '';
|
|
this.particle_texture_stretch_to_lifetime = false;
|
|
this.particle_texture_loop = false;
|
|
this.particle_color_mode = 'static';
|
|
this.particle_color_static = '#ffffff';
|
|
this.particle_color_interpolant = '';
|
|
this.particle_color_range = 0;
|
|
this.particle_color_gradient = [];
|
|
this.particle_color_expression = [];
|
|
this.particle_color_light = false;
|
|
this.particle_collision_enabled = false;
|
|
this.particle_collision_collision_drag = 0;
|
|
this.particle_collision_coefficient_of_restitution = 0;
|
|
this.particle_collision_collision_radius = 0;
|
|
this.particle_collision_expire_on_contact = false;
|
|
|
|
return this;
|
|
}
|
|
set(key, val) {
|
|
if (this[key] == undefined || val == undefined || val == null) return;
|
|
|
|
if (this[key] instanceof Array) {
|
|
this[key].splice(0, Infinity, ...val);
|
|
} else if (typeof this[key] == 'string') {
|
|
this[key] = val.toString();
|
|
} else if (typeof this[key] == 'number' && typeof val == 'number') {
|
|
this[key] = val;
|
|
} else if (typeof this[key] == 'boolean') {
|
|
this[key] = !!val;
|
|
}
|
|
return this;
|
|
}
|
|
setFromJSON(data) {
|
|
|
|
var comps = data.particle_effect.components;
|
|
var curves = data.particle_effect.curves;
|
|
var desc = data.particle_effect.description;
|
|
if (desc && desc.identifier) {
|
|
this.identifier = desc.identifier;
|
|
}
|
|
if (desc && desc.basic_render_parameters) {
|
|
this.set('particle_texture_path', desc.basic_render_parameters.texture);
|
|
|
|
this.set('particle_appearance_material', desc.basic_render_parameters.material);
|
|
}
|
|
if (curves) {
|
|
for (var key in curves) {
|
|
var json_curve = curves[key];
|
|
var new_curve = {
|
|
id: key,
|
|
mode: json_curve.type,
|
|
input: json_curve.input,
|
|
range: json_curve.horizontal_range,
|
|
nodes: []
|
|
};
|
|
if (json_curve.nodes && json_curve.nodes.length) {
|
|
json_curve.nodes.forEach(value => {
|
|
value = parseFloat(value)||0;
|
|
new_curve.nodes.push(value);
|
|
});
|
|
}
|
|
this.curves[key] = new_curve;
|
|
}
|
|
}
|
|
|
|
if (comps) {
|
|
function comp(id) {
|
|
return comps[`minecraft:${id}`]
|
|
}
|
|
if (comp('emitter_initialization')) {
|
|
var cr_v = comp('emitter_initialization').creation_expression;
|
|
var up_v = comp('emitter_initialization').per_update_expression;
|
|
if (typeof cr_v == 'string') {
|
|
this.variables_creation_vars = cr_v.replace(/;+$/, '').split(';');
|
|
}
|
|
if (typeof up_v == 'string') {
|
|
this.variables_tick_vars = up_v.replace(/;+$/, '').split(';');
|
|
}
|
|
}
|
|
if (comp('emitter_local_space')) {
|
|
this.space_local_position = comp('emitter_local_space').position;
|
|
this.space_local_rotation = comp('emitter_local_space').rotation;
|
|
}
|
|
if (comp('emitter_rate_steady')) {
|
|
this.set('emitter_rate_mode', 'steady');
|
|
this.set('emitter_rate_rate', comp('emitter_rate_steady').spawn_rate);
|
|
this.set('emitter_rate_maximum', comp('emitter_rate_steady').max_particles);
|
|
}
|
|
if (comp('emitter_rate_instant')) {
|
|
this.set('emitter_rate_mode', 'instant');
|
|
this.set('emitter_rate_amount', comp('emitter_rate_instant').num_particles);
|
|
}
|
|
if (comp('emitter_lifetime_once')) {
|
|
this.set('emitter_lifetime_mode', 'once');
|
|
this.set('emitter_lifetime_active_time', comp('emitter_lifetime_once').active_time);
|
|
}
|
|
if (comp('emitter_lifetime_looping')) {
|
|
this.set('emitter_lifetime_mode', 'looping');
|
|
this.set('emitter_lifetime_active_time', comp('emitter_lifetime_looping').active_time);
|
|
this.set('emitter_lifetime_sleep_time', comp('emitter_lifetime_looping').sleep_time);
|
|
}
|
|
if (comp('emitter_lifetime_expression')) {
|
|
this.set('emitter_lifetime_mode', 'expression');
|
|
this.set('emitter_lifetime_activation', comp('emitter_lifetime_expression').activation_expression);
|
|
this.set('emitter_lifetime_expiration', comp('emitter_lifetime_expression').expiration_expression);
|
|
}
|
|
var shape_component = comp('emitter_shape_point') || comp('emitter_shape_custom');
|
|
if (shape_component) {
|
|
this.set('emitter_shape_mode', 'point');
|
|
this.set('emitter_shape_offset', shape_component.offset);
|
|
}
|
|
if (comp('emitter_shape_sphere')) {
|
|
shape_component = comp('emitter_shape_sphere');
|
|
this.set('emitter_shape_mode', 'sphere');
|
|
this.set('emitter_shape_offset', shape_component.offset);
|
|
this.set('emitter_shape_radius', shape_component.radius);
|
|
this.set('emitter_shape_surface_only', shape_component.surface_only);
|
|
}
|
|
if (comp('emitter_shape_box')) {
|
|
shape_component = comp('emitter_shape_box');
|
|
this.set('emitter_shape_mode', 'box');
|
|
this.set('emitter_shape_offset', shape_component.offset);
|
|
this.set('emitter_shape_half_dimensions', shape_component.half_dimensions);
|
|
this.set('emitter_shape_surface_only', shape_component.surface_only);
|
|
}
|
|
if (comp('emitter_shape_disc')) {
|
|
shape_component = comp('emitter_shape_disc');
|
|
this.set('emitter_shape_mode', 'disc');
|
|
this.set('emitter_shape_offset', shape_component.offset);
|
|
switch (shape_component.plane_normal) {
|
|
case 'x': this.set('emitter_shape_plane_normal', [1, 0, 0]); break;
|
|
case 'y': this.set('emitter_shape_plane_normal', [0, 1, 0]); break;
|
|
case 'z': this.set('emitter_shape_plane_normal', [0, 0, 1]); break;
|
|
default: this.set('emitter_shape_plane_normal', shape_component.plane_normal); break;
|
|
}
|
|
this.set('emitter_shape_radius', shape_component.radius);
|
|
this.set('emitter_shape_surface_only', shape_component.surface_only);
|
|
}
|
|
if (comp('emitter_shape_entity_aabb')) {
|
|
this.set('emitter_shape_mode', 'entity_aabb');
|
|
this.set('emitter_shape_surface_only', comp('emitter_shape_entity_aabb').surface_only);
|
|
shape_component = comp('emitter_shape_entity_aabb');
|
|
}
|
|
if (shape_component && shape_component.direction) {
|
|
if (shape_component.direction == 'inwards' || shape_component.direction == 'outwards') {
|
|
this.set('particle_direction_mode', shape_component.direction);
|
|
} else {
|
|
this.set('particle_direction_mode', 'direction');
|
|
this.set('particle_direction_direction', shape_component.direction);
|
|
}
|
|
}
|
|
|
|
if (comp('particle_initial_spin')) {
|
|
this.set('particle_rotation_initial_rotation', comp('particle_initial_spin').rotation);
|
|
this.set('particle_rotation_rotation_rate', comp('particle_initial_spin').rotation_rate);
|
|
}
|
|
if (comp('particle_kill_plane')) {
|
|
this.set('particle_lifetime_kill_plane', comp('particle_kill_plane'));
|
|
}
|
|
|
|
if (comp('particle_motion_dynamic')) {
|
|
this.set('particle_motion_mode', 'dynamic');
|
|
this.set('particle_motion_linear_acceleration', comp('particle_motion_dynamic').linear_acceleration);
|
|
this.set('particle_motion_linear_drag_coefficient', comp('particle_motion_dynamic').linear_drag_coefficient);
|
|
this.set('particle_rotation_rotation_acceleration', comp('particle_motion_dynamic').rotation_acceleration);
|
|
this.set('particle_rotation_rotation_drag_coefficient', comp('particle_motion_dynamic').rotation_drag_coefficient);
|
|
this.set('particle_motion_linear_speed', 1);
|
|
}
|
|
if (comp('particle_motion_parametric')) {
|
|
this.set('particle_motion_mode', 'parametric');
|
|
this.set('particle_motion_relative_position', comp('particle_motion_parametric').relative_position);
|
|
this.set('particle_motion_direction', comp('particle_motion_parametric').direction);
|
|
this.set('particle_rotation_rotation', comp('particle_motion_parametric').rotation);
|
|
}
|
|
if (comp('particle_motion_collision')) {
|
|
this.set('particle_collision_enabled', comp('particle_motion_collision').enabled || true);
|
|
this.set('particle_collision_collision_drag', comp('particle_motion_collision').collision_drag);
|
|
this.set('particle_collision_coefficient_of_restitution', comp('particle_motion_collision').coefficient_of_restitution);
|
|
this.set('particle_collision_collision_radius', comp('particle_motion_collision').collision_radius);
|
|
this.set('particle_collision_expire_on_contact', comp('particle_motion_collision').expire_on_contact);
|
|
}
|
|
if (comp('particle_initial_speed') !== undefined) {
|
|
var c = comp('particle_initial_speed');
|
|
if (typeof c !== 'object') {
|
|
this.set('particle_motion_linear_speed', c);
|
|
} else {
|
|
this.set('particle_direction_mode', 'direction');
|
|
this.set('particle_direction_direction', comp('particle_initial_speed'));
|
|
this.set('particle_motion_linear_speed', 1);
|
|
}
|
|
}
|
|
|
|
if (comp('particle_lifetime_expression')) {
|
|
this.set('particle_lifetime_mode', 'expression');
|
|
if (comp('particle_lifetime_expression').expiration_expression) {
|
|
this.set('particle_lifetime_mode', 'expression');
|
|
this.set('particle_lifetime_expiration_expression', comp('particle_lifetime_expression').expiration_expression);
|
|
} else {
|
|
this.set('particle_lifetime_mode', 'time');
|
|
this.set('particle_lifetime_max_lifetime', comp('particle_lifetime_expression').max_lifetime);
|
|
}
|
|
}
|
|
if (comp('particle_expire_if_in_blocks') instanceof Array) {
|
|
this.set('particle_lifetime_expire_in', comp('particle_expire_if_in_blocks'));
|
|
}
|
|
if (comp('particle_expire_if_not_in_blocks') instanceof Array) {
|
|
this.set('particle_lifetime_expire_outside', comp('particle_expire_if_not_in_blocks'));
|
|
}
|
|
|
|
if (comp('particle_appearance_billboard')) {
|
|
this.set('particle_appearance_size', comp('particle_appearance_billboard').size);
|
|
this.set('particle_appearance_facing_camera_mode', comp('particle_appearance_billboard').facing_camera_mode);
|
|
var uv_tag = comp('particle_appearance_billboard').uv;
|
|
if (uv_tag) {
|
|
if (uv_tag.texture_width) this.set('particle_texture_width', uv_tag.texture_width);
|
|
if (uv_tag.texture_height) this.set('particle_texture_height', uv_tag.texture_height);
|
|
if (uv_tag.flipbook) {
|
|
this.set('particle_texture_mode', 'animated');
|
|
this.set('particle_texture_uv', uv_tag.flipbook.base_UV);
|
|
this.set('particle_texture_uv_size', uv_tag.flipbook.size_UV);
|
|
this.set('particle_texture_uv_step', uv_tag.flipbook.step_UV);
|
|
this.set('particle_texture_frames_per_second', uv_tag.flipbook.frames_per_second);
|
|
this.set('particle_texture_max_frame', uv_tag.flipbook.max_frame);
|
|
this.set('particle_texture_stretch_to_lifetime', uv_tag.flipbook.stretch_to_lifetime);
|
|
this.set('particle_texture_loop', uv_tag.flipbook.loop);
|
|
} else {
|
|
this.set('particle_texture_mode', 'static');
|
|
this.set('particle_texture_uv', uv_tag.uv);
|
|
this.set('particle_texture_uv_size', uv_tag.uv_size);
|
|
}
|
|
}
|
|
}
|
|
if (comp('particle_appearance_lighting')) {
|
|
this.set('particle_color_light', true);
|
|
}
|
|
if (comp('particle_appearance_tinting')) {
|
|
var c = comp('particle_appearance_tinting').color;
|
|
|
|
if (c instanceof Array && c.length >= 3) {
|
|
if ((typeof c[0] + typeof c[1] + typeof c[1]).includes('string')) {
|
|
this.set('particle_color_mode', 'expression');
|
|
this.set('particle_color_expression', c);
|
|
|
|
} else {
|
|
this.set('particle_color_mode', 'static');
|
|
|
|
var color = new tinycolor__default['default']({
|
|
r: c[0] * 255,
|
|
g: c[1] * 255,
|
|
b: c[2] * 255,
|
|
}).toHexString();
|
|
this.set('particle_color_static', color);
|
|
}
|
|
} else if (typeof c == 'object') {
|
|
// Gradient
|
|
this.set('particle_color_mode', 'gradient');
|
|
this.set('particle_color_interpolant', c.interpolant);
|
|
let gradient_points = [];
|
|
if (c.gradient instanceof Array) {
|
|
let distance = 100 / (c.gradient.length-1);
|
|
c.gradient.forEach((color, i) => {
|
|
color = parseColor(color);
|
|
var percent = distance * i;
|
|
gradient_points.push({percent, color});
|
|
});
|
|
} else if (typeof c.gradient == 'object') {
|
|
let max_time = 0;
|
|
for (var time in c.gradient) {
|
|
max_time = Math.max(parseFloat(time), max_time);
|
|
}
|
|
this.particle_color_range = max_time;
|
|
for (var time in c.gradient) {
|
|
var color = parseColor(c.gradient[time]);
|
|
var percent = (parseFloat(time) / max_time) * 100;
|
|
gradient_points.push({color, percent});
|
|
}
|
|
}
|
|
this.set('particle_color_gradient', gradient_points);
|
|
}
|
|
}
|
|
}
|
|
|
|
this.updateTexture();
|
|
return this;
|
|
}
|
|
updateTexture() {
|
|
var url;
|
|
var path = this.particle_texture_path;
|
|
|
|
switch (path) {
|
|
case 'textures/particle/particles':
|
|
url = img$1;
|
|
break;
|
|
case 'textures/flame_atlas': case 'textures/particle/flame_atlas':
|
|
url = img$2;
|
|
break;
|
|
case 'textures/particle/soul':
|
|
url = img$3;
|
|
break;
|
|
case 'textures/particle/campfire_smoke':
|
|
url = img$4;
|
|
break;
|
|
default:
|
|
url = img;
|
|
break;
|
|
}
|
|
if (url == img && typeof Wintersky.fetchTexture == 'function') {
|
|
let result = Wintersky.fetchTexture(this);
|
|
if (result) url = result;
|
|
}
|
|
this.texture.image.src = url;
|
|
this.texture.image.onload = () => {
|
|
this.texture.needsUpdate = true;
|
|
};
|
|
return this;
|
|
}
|
|
}
|
|
Wintersky.Config = Config;
|
|
|
|
const MathUtil = {
|
|
roundTo(num, digits) {
|
|
var d = Math.pow(10,digits);
|
|
return Math.round(num * d) / d
|
|
},
|
|
randomab(a, b) {
|
|
return a + Math.random() * (b-a)
|
|
},
|
|
radToDeg(rad) {
|
|
return rad / Math.PI * 180
|
|
},
|
|
degToRad(deg) {
|
|
return Math.PI / (180 /deg)
|
|
},
|
|
clamp(number, min, max) {
|
|
if (number > max) number = max;
|
|
if (number < min || isNaN(number)) number = min;
|
|
return number;
|
|
},
|
|
roundTo(num, digits) {
|
|
var d = Math.pow(10,digits);
|
|
return Math.round(num * d) / d
|
|
},
|
|
getRandomEuler() {
|
|
return new THREE__default['default'].Euler(
|
|
MathUtil.randomab(-Math.PI, Math.PI),
|
|
MathUtil.randomab(-Math.PI, Math.PI),
|
|
MathUtil.randomab(-Math.PI, Math.PI)
|
|
)
|
|
}
|
|
};
|
|
|
|
const Normals = {
|
|
x: new THREE__default['default'].Vector3(1, 0, 0),
|
|
y: new THREE__default['default'].Vector3(0, 1, 0),
|
|
z: new THREE__default['default'].Vector3(0, 0, 1),
|
|
n: new THREE__default['default'].Vector3(0, 0, 0),
|
|
};
|
|
|
|
function removeFromArray(array, item) {
|
|
let index = array.indexOf(item);
|
|
if (index >= 0) {
|
|
array.splice(index, 1);
|
|
}
|
|
}
|
|
|
|
function calculateGradient(gradient, percent) {
|
|
let index = 0;
|
|
gradient.forEach((point, i) => {
|
|
if (point.percent <= percent) index = i;
|
|
});
|
|
if (gradient[index] && !gradient[index+1]) {
|
|
var color = gradient[index].color;
|
|
|
|
} else if (!gradient[index] && gradient[index+1]) {
|
|
var color = gradient[index+1].color;
|
|
|
|
} else if (gradient[index] && gradient[index+1]) {
|
|
// Interpolate
|
|
var mix = (percent - gradient[index].percent) / (gradient[index+1].percent - gradient[index].percent);
|
|
var color = tinycolor__default['default'].mix(gradient[index].color, gradient[index+1].color, mix*100).toHexString();
|
|
|
|
} else {
|
|
var color = '#ffffff';
|
|
}
|
|
return new THREE__default['default'].Color(color);
|
|
}
|
|
|
|
|
|
class Particle {
|
|
constructor(emitter, data) {
|
|
this.emitter = emitter;
|
|
if (!data) data = 0;
|
|
|
|
this.geometry = new THREE__default['default'].PlaneGeometry(1, 1);
|
|
this.material = this.emitter.material.clone();
|
|
this.mesh = new THREE__default['default'].Mesh(this.geometry, this.material);
|
|
this.position = this.mesh.position;
|
|
|
|
this.speed = data.speed||new THREE__default['default'].Vector3();
|
|
this.acceleration = data.acceleration||new THREE__default['default'].Vector3();
|
|
|
|
this.add();
|
|
}
|
|
params() {
|
|
var obj = this.emitter.params();
|
|
obj["variable.particle_lifetime"] = this.lifetime;
|
|
obj["variable.particle_age"] = this.age;
|
|
obj["variable.particle_random_1"] = this.random_vars[0];
|
|
obj["variable.particle_random_2"] = this.random_vars[1];
|
|
obj["variable.particle_random_3"] = this.random_vars[2];
|
|
obj["variable.particle_random_4"] = this.random_vars[3];
|
|
return obj;
|
|
}
|
|
add() {
|
|
if (!this.emitter.particles.includes(this)) {
|
|
this.emitter.particles.push(this);
|
|
if (this.emitter.config.space_local_position && this.emitter.local_space.parent) {
|
|
// Add the particle to the local space object if local space is enabled and used
|
|
this.emitter.local_space.add(this.mesh);
|
|
} else {
|
|
// Otherwise add to global space
|
|
this.emitter.global_space.add(this.mesh);
|
|
}
|
|
}
|
|
|
|
this.age = this.loop_time = 0;
|
|
this.current_frame = 0;
|
|
this.random_vars = [Math.random(), Math.random(), Math.random(), Math.random()];
|
|
this.material.copy(this.emitter.material);
|
|
this.material.needsUpdate = true;
|
|
var params = this.params();
|
|
|
|
this.position.set(0, 0, 0);
|
|
this.lifetime = this.emitter.calculate(this.emitter.config.particle_lifetime_max_lifetime, params);
|
|
this.initial_rotation = this.emitter.calculate(this.emitter.config.particle_rotation_initial_rotation, params);
|
|
this.rotation_rate = this.emitter.calculate(this.emitter.config.particle_rotation_rotation_rate, params);
|
|
this.rotation = 0;
|
|
|
|
//Init Position:
|
|
var surface = this.emitter.config.emitter_shape_surface_only;
|
|
if (this.emitter.config.emitter_shape_mode === 'box') {
|
|
var size = this.emitter.calculate(this.emitter.config.emitter_shape_half_dimensions, params);
|
|
|
|
this.position.x = MathUtil.randomab(-size.x, size.x);
|
|
this.position.y = MathUtil.randomab(-size.y, size.y);
|
|
this.position.z = MathUtil.randomab(-size.z, size.z);
|
|
|
|
if (surface) {
|
|
var face = Math.floor(MathUtil.randomab(0, 3));
|
|
var side = Math.floor(MathUtil.randomab(0, 2));
|
|
this.position.setComponent(face, size.getComponent(face) * (side?1:-1));
|
|
}
|
|
} else if (this.emitter.config.emitter_shape_mode === 'entity_aabb') {
|
|
var size = new THREE__default['default'].Vector3(0.5, 1, 0.5);
|
|
|
|
this.position.x = MathUtil.randomab(-size.x, size.x);
|
|
this.position.y = MathUtil.randomab(-size.y, size.y);
|
|
this.position.z = MathUtil.randomab(-size.z, size.z);
|
|
|
|
if (surface) {
|
|
var face = Math.floor(MathUtil.randomab(0, 3));
|
|
var side = Math.floor(MathUtil.randomab(0, 2));
|
|
this.position.setComponent(face, size.getComponent(face) * (side?1:-1));
|
|
}
|
|
} else if (this.emitter.config.emitter_shape_mode === 'sphere') {
|
|
|
|
var radius = this.emitter.calculate(this.emitter.config.emitter_shape_radius, params);
|
|
if (surface) {
|
|
this.position.x = radius;
|
|
} else {
|
|
this.position.x = radius * Math.random();
|
|
}
|
|
this.position.applyEuler(MathUtil.getRandomEuler());
|
|
} else if (this.emitter.config.emitter_shape_mode === 'disc') {
|
|
var radius = this.emitter.calculate(this.emitter.config.emitter_shape_radius, params);
|
|
var ang = Math.random()*Math.PI*2;
|
|
var dis = surface ? radius : radius * Math.sqrt(Math.random());
|
|
|
|
this.position.x = dis * Math.cos(ang);
|
|
this.position.z = dis * Math.sin(ang);
|
|
|
|
var normal = this.emitter.calculate(this.emitter.config.emitter_shape_plane_normal, params);
|
|
if (!normal.equals(Normals.n)) {
|
|
var q = new THREE__default['default'].Quaternion().setFromUnitVectors(Normals.y, normal);
|
|
this.position.applyQuaternion(q);
|
|
}
|
|
}
|
|
//Speed
|
|
this.speed = new THREE__default['default'].Vector3();
|
|
var dir = this.emitter.config.particle_direction_mode;
|
|
if (dir == 'inwards' || dir == 'outwards') {
|
|
|
|
if (this.emitter.config.emitter_shape_mode === 'point') {
|
|
this.speed.set(1, 0, 0).applyEuler(MathUtil.getRandomEuler());
|
|
} else {
|
|
this.speed.copy(this.position).normalize();
|
|
if (dir == 'inwards') {
|
|
this.speed.negate();
|
|
}
|
|
}
|
|
} else {
|
|
this.speed = this.emitter.calculate(this.emitter.config.particle_direction_direction, params).normalize();
|
|
}
|
|
var speed = this.emitter.calculate(this.emitter.config.particle_motion_linear_speed, params);
|
|
this.speed.x *= speed;
|
|
this.speed.y *= speed;
|
|
this.speed.z *= speed;
|
|
|
|
this.position.add(this.emitter.calculate(this.emitter.config.emitter_shape_offset, params));
|
|
|
|
if (this.emitter.parent_mode == 'locator') {
|
|
this.position.x *= -1;
|
|
this.position.y *= -1;
|
|
this.speed.x *= -1;
|
|
this.speed.y *= -1;
|
|
} else if (this.emitter.config.space_local_position && !this.emitter.config.space_local_rotation) {
|
|
this.speed.x *= -1;
|
|
this.speed.z *= -1;
|
|
}
|
|
|
|
if (this.emitter.local_space.parent) {
|
|
|
|
if (!this.emitter.config.space_local_rotation) {
|
|
this.position.applyQuaternion(this.emitter.local_space.getWorldQuaternion(new THREE__default['default'].Quaternion()));
|
|
}
|
|
if (!this.emitter.config.space_local_position) {
|
|
let offset = this.emitter.local_space.getWorldPosition(new THREE__default['default'].Vector3());
|
|
this.position.addScaledVector(offset, 1/Wintersky.global_options._scale);
|
|
}
|
|
}
|
|
|
|
//UV
|
|
this.setFrame(0);
|
|
|
|
return this.tick();
|
|
}
|
|
tick(jump) {
|
|
var params = this.params();
|
|
let {tick_rate} = Wintersky.global_options;
|
|
|
|
//Lifetime
|
|
this.age += 1/tick_rate;
|
|
this.loop_time += 1/tick_rate;
|
|
if (this.emitter.config.particle_lifetime_mode === 'time') {
|
|
if (this.age > this.lifetime) {
|
|
this.remove();
|
|
}
|
|
} else {
|
|
if (this.emitter.calculate(this.emitter.config.particle_lifetime_expiration_expression, params)) {
|
|
this.remove();
|
|
}
|
|
}
|
|
//Movement
|
|
if (this.emitter.config.particle_motion_mode === 'dynamic') {
|
|
//Position
|
|
var drag = this.emitter.calculate(this.emitter.config.particle_motion_linear_drag_coefficient, params);
|
|
this.acceleration = this.emitter.calculate(this.emitter.config.particle_motion_linear_acceleration, params);
|
|
if (this.emitter.config.space_local_position) {
|
|
if (this.emitter.parent_mode == 'locator') {
|
|
this.acceleration.x *= -1;
|
|
this.acceleration.y *= -1;
|
|
}
|
|
} else if (this.emitter.parent_mode != 'world') {
|
|
this.acceleration.x *= -1;
|
|
this.acceleration.z *= -1;
|
|
}
|
|
this.acceleration.addScaledVector(this.speed, -drag);
|
|
this.speed.addScaledVector(this.acceleration, 1/tick_rate);
|
|
this.position.addScaledVector(this.speed, 1/tick_rate);
|
|
|
|
//Rotation
|
|
var rot_drag = this.emitter.calculate(this.emitter.config.particle_rotation_rotation_drag_coefficient, params);
|
|
var rot_acceleration = this.emitter.calculate(this.emitter.config.particle_rotation_rotation_acceleration, params);
|
|
rot_acceleration += -rot_drag * this.rotation_rate;
|
|
this.rotation_rate += rot_acceleration*1/tick_rate;
|
|
this.rotation = MathUtil.degToRad(this.initial_rotation + this.rotation_rate*this.age);
|
|
} else if (!jump) {
|
|
if (this.emitter.config.particle_motion_relative_position.join('').length) {
|
|
this.position.copy(this.emitter.calculate(this.emitter.config.particle_motion_relative_position, params));
|
|
}
|
|
this.rotation = MathUtil.degToRad(this.emitter.calculate(this.emitter.config.particle_rotation_rotation, params));
|
|
}
|
|
|
|
if (!jump) {
|
|
//Size
|
|
var size = this.emitter.calculate(this.emitter.config.particle_appearance_size, params);
|
|
this.mesh.scale.x = size.x*2.25 || 0.0001;
|
|
this.mesh.scale.y = size.y*2.25 || 0.0001;
|
|
|
|
|
|
//UV
|
|
if (this.emitter.config.particle_texture_mode === 'animated') {
|
|
var max_frame = this.emitter.calculate(this.emitter.config.particle_texture_max_frame, params);
|
|
if (this.emitter.config.particle_texture_stretch_to_lifetime && max_frame) {
|
|
var fps = max_frame/this.lifetime;
|
|
} else {
|
|
var fps = this.emitter.calculate(this.emitter.config.particle_texture_frames_per_second, params);
|
|
}
|
|
if (Math.floor(this.loop_time*fps) > this.current_frame) {
|
|
this.current_frame = Math.floor(this.loop_time*fps);
|
|
if (max_frame && this.current_frame > max_frame) {
|
|
if (this.emitter.config.particle_texture_loop) {
|
|
this.current_frame = 0;
|
|
this.loop_time = 0;
|
|
this.setFrame(this.current_frame);
|
|
}
|
|
} else {
|
|
this.setFrame(this.current_frame);
|
|
}
|
|
}
|
|
} else {
|
|
this.setFrame(0);
|
|
}
|
|
|
|
//Color (ToDo)
|
|
if (this.emitter.config.particle_color_mode === 'expression') {
|
|
var c = this.emitter.calculate(this.emitter.config.particle_color_expression, params, 'array');
|
|
this.setColor(...c);
|
|
|
|
} else if (this.emitter.config.particle_color_mode === 'gradient') {
|
|
var i = this.emitter.calculate(this.emitter.config.particle_color_interpolant, params);
|
|
var r = this.emitter.calculate(this.emitter.config.particle_color_range, params);
|
|
var c = calculateGradient(this.emitter.config.particle_color_gradient, (i/r) * 100);
|
|
this.setColor(c.r, c.g, c.b);
|
|
|
|
} else {
|
|
var c = tinycolor__default['default'](this.emitter.config.particle_color_static).toRgb();
|
|
this.setColor(c.r/255, c.g/255, c.b/255);
|
|
}
|
|
}
|
|
|
|
return this;
|
|
}
|
|
remove() {
|
|
removeFromArray(this.emitter.particles, this);
|
|
if (this.mesh.parent) this.mesh.parent.remove(this.mesh);
|
|
this.emitter.dead_particles.push(this);
|
|
return this;
|
|
}
|
|
setColor(r, g, b) {
|
|
this.mesh.geometry.faces.forEach(face => {
|
|
face.color.setRGB(r, g, b);
|
|
});
|
|
this.mesh.geometry.colorsNeedUpdate = true;
|
|
}
|
|
setFrame(n) {
|
|
var params = this.params();
|
|
var uv = this.emitter.calculate(this.emitter.config.particle_texture_uv, params);
|
|
var size = this.emitter.calculate(this.emitter.config.particle_texture_uv_size, params);
|
|
if (n) {
|
|
var offset = this.emitter.calculate(this.emitter.config.particle_texture_uv_step, params);
|
|
uv.addScaledVector(offset, n);
|
|
}
|
|
this.setUV(uv.x, uv.y, size.x||this.emitter.config.particle_texture_width, size.y||this.emitter.config.particle_texture_height);
|
|
}
|
|
setUV(x, y, w, h) {
|
|
var epsilon = 0.05;
|
|
var vertex_uvs = this.geometry.faceVertexUvs[0];
|
|
|
|
w = (x+w - 2*epsilon) / this.emitter.config.particle_texture_width;
|
|
h = (y+h - 2*epsilon) / this.emitter.config.particle_texture_height;
|
|
x = (x + (w>0 ? epsilon : -epsilon)) / this.emitter.config.particle_texture_width;
|
|
y = (y + (h>0 ? epsilon : -epsilon)) / this.emitter.config.particle_texture_height;
|
|
|
|
vertex_uvs[0][0].set(x, 1-y);
|
|
vertex_uvs[0][1].set(x, 1-h);
|
|
vertex_uvs[0][2].set(w, 1-y);
|
|
vertex_uvs[1][1].set(w, 1-h);
|
|
|
|
vertex_uvs[1][0] = vertex_uvs[0][1];
|
|
vertex_uvs[1][2] = vertex_uvs[0][2];
|
|
this.geometry.uvsNeedUpdate = true;
|
|
}
|
|
}
|
|
Wintersky.Particle = Particle;
|
|
|
|
const dummy_vec = new THREE__default['default'].Vector3();
|
|
|
|
function calculateCurve(emitter, curve, params) {
|
|
|
|
var position = emitter.Molang.parse(curve.input, params);
|
|
var range = emitter.Molang.parse(curve.range, params);
|
|
|
|
position = (position/range) || 0;
|
|
if (position === Infinity) position = 0;
|
|
|
|
if (curve.mode == 'linear') {
|
|
|
|
var segments = curve.nodes.length-1;
|
|
position *= segments;
|
|
var index = Math.floor(position);
|
|
var blend = position%1;
|
|
var difference = curve.nodes[index+1] - curve.nodes[index];
|
|
var value = curve.nodes[index] + difference * blend;
|
|
return value;
|
|
|
|
} else if (curve.mode == 'catmull_rom') {
|
|
var vectors = [];
|
|
curve.nodes.forEach((val, i) => {
|
|
vectors.push(new THREE__default['default'].Vector2(i-1, val));
|
|
});
|
|
var spline = new THREE__default['default'].SplineCurve(vectors);
|
|
|
|
var segments = curve.nodes.length-3;
|
|
position *= segments;
|
|
var pso = (position+1)/(segments+2);
|
|
return spline.getPoint(pso).y;
|
|
}
|
|
}
|
|
|
|
|
|
class Emitter {
|
|
constructor(config, options = 0) {
|
|
Wintersky.emitters.push(this);
|
|
|
|
this.config = config instanceof Config ? config : new Config(config, options);
|
|
|
|
this.Molang = new Molang__default['default']();
|
|
this.Molang.variableHandler = (key, params) => {
|
|
return this.creation_values[key]
|
|
|| this.tick_values[key]
|
|
|| (this.config.curves[key] && calculateCurve(this, this.config.curves[key], params))
|
|
};
|
|
|
|
let global_scale = Wintersky.global_options._scale;
|
|
this.local_space = new THREE__default['default'].Object3D();
|
|
this.local_space.scale.set(global_scale, global_scale, global_scale);
|
|
this.global_space = new THREE__default['default'].Object3D();
|
|
this.global_space.scale.set(global_scale, global_scale, global_scale);
|
|
this.material = new THREE__default['default'].MeshBasicMaterial({
|
|
color: 0xffffff,
|
|
transparent: true,
|
|
vertexColors: THREE__default['default'].FaceColors,
|
|
alphaTest: 0.2,
|
|
map: this.config.texture
|
|
});
|
|
|
|
this.particles = [];
|
|
this.dead_particles = [];
|
|
this.age = 0;
|
|
this.view_age = 0;
|
|
this.enabled = false;
|
|
this.loop_mode = options.loop_mode || Wintersky.global_options.loop_mode;
|
|
this.parent_mode = options.parent_mode || Wintersky.global_options.parent_mode;
|
|
this.random_vars = [Math.random(), Math.random(), Math.random(), Math.random()];
|
|
this.tick_variables = {};
|
|
this.tick_values = {};
|
|
this.creation_variables = {};
|
|
this.creation_values = {};
|
|
|
|
this.updateMaterial();
|
|
}
|
|
clone() {
|
|
let clone = new Wintersky.Emitter(this.config);
|
|
clone.loop_mode = this.loop_mode;
|
|
return clone;
|
|
}
|
|
params() {
|
|
var obj = {
|
|
"variable.entity_scale": 1
|
|
};
|
|
obj["variable.emitter_lifetime"] = this.active_time;
|
|
obj["variable.emitter_age"] = this.age;
|
|
obj["variable.emitter_random_1"] = this.random_vars[0];
|
|
obj["variable.emitter_random_2"] = this.random_vars[1];
|
|
obj["variable.emitter_random_3"] = this.random_vars[2];
|
|
obj["variable.emitter_random_4"] = this.random_vars[3];
|
|
return obj;
|
|
}
|
|
calculate(input, variables, datatype) {
|
|
|
|
let getV = v => this.Molang.parse(v, variables);
|
|
var data;
|
|
|
|
if (input instanceof Array) {
|
|
if (datatype == 'array') {
|
|
data = [];
|
|
input.forEach(source => {
|
|
data.push(getV(source));
|
|
});
|
|
|
|
} else if (input.length === 4) {
|
|
data = new THREE__default['default'].Plane().setComponents(
|
|
getV(input[0]),
|
|
getV(input[1]),
|
|
getV(input[2]),
|
|
getV(input[3])
|
|
);
|
|
} else if (input.length === 3) {
|
|
data = new THREE__default['default'].Vector3(
|
|
getV(input[0]),
|
|
getV(input[1]),
|
|
getV(input[2])
|
|
);
|
|
} else if (input.length === 2) {
|
|
data = new THREE__default['default'].Vector2(
|
|
getV(input[0]),
|
|
getV(input[1])
|
|
);
|
|
}
|
|
} else if (datatype == 'color') ; else {
|
|
data = getV(input);
|
|
}
|
|
return data;
|
|
}
|
|
updateConfig() {
|
|
this.updateMaterial();
|
|
}
|
|
updateMaterial() {
|
|
this.config.updateTexture();
|
|
}
|
|
start() {
|
|
this.age = 0;
|
|
this.view_age = 0;
|
|
this.enabled = true;
|
|
this.initialized = true;
|
|
Wintersky.space.add(this.global_space);
|
|
var params = this.params();
|
|
this.active_time = this.calculate(this.config.emitter_lifetime_active_time, params);
|
|
this.sleep_time = this.calculate(this.config.emitter_lifetime_sleep_time, params);
|
|
this.random_vars = [Math.random(), Math.random(), Math.random(), Math.random()];
|
|
this.creation_values = {};
|
|
|
|
for (var line of this.config.variables_creation_vars) {
|
|
let [key, value] = line.split(/\s*=(.+)/);
|
|
value = value.replace(/^\s*=\s*/, '');
|
|
this.creation_values[key] = this.Molang.parse(value);
|
|
}
|
|
|
|
if (this.config.emitter_rate_mode === 'instant') {
|
|
this.spawnParticles(this.calculate(this.config.emitter_rate_amount, params));
|
|
}
|
|
return this;
|
|
}
|
|
tick(jump) {
|
|
let params = this.params();
|
|
let {tick_rate} = Wintersky.global_options;
|
|
this.tick_values = {};
|
|
|
|
// Calculate tick values
|
|
for (var line of this.config.variables_tick_vars) {
|
|
let [key, value] = line.split(/\s*=(.+)/);
|
|
value = value.replace(/^\s*=\s*/, '');
|
|
this.tick_values[key] = this.Molang.parse(value);
|
|
}
|
|
// Spawn steady particles
|
|
if (this.enabled && this.config.emitter_rate_mode === 'steady') {
|
|
var p_this_tick = this.calculate(this.config.emitter_rate_rate, params)/tick_rate;
|
|
var x = 1/p_this_tick;
|
|
var c_f = Math.round(this.age*tick_rate);
|
|
if (c_f % Math.round(x) == 0) {
|
|
p_this_tick = Math.ceil(p_this_tick);
|
|
} else {
|
|
p_this_tick = Math.floor(p_this_tick);
|
|
}
|
|
this.spawnParticles(p_this_tick);
|
|
}
|
|
// Tick particles
|
|
this.particles.forEach(p => {
|
|
p.tick(jump);
|
|
});
|
|
|
|
this.age += 1/tick_rate;
|
|
this.view_age += 1/tick_rate;
|
|
|
|
if (this.config.emitter_lifetime_mode === 'expression') {
|
|
//Expressions
|
|
if (this.enabled && this.calculate(this.config.emitter_lifetime_expiration, params)) {
|
|
this.stop();
|
|
}
|
|
if (!this.enabled && this.calculate(this.config.emitter_lifetime_activation, params)) {
|
|
this.start();
|
|
}
|
|
} else if (this.loop_mode == 'looping' || (this.loop_mode == 'auto' && this.config.emitter_lifetime_mode == 'looping')) {
|
|
//Looping
|
|
if (this.enabled && MathUtil.roundTo(this.age, 5) >= this.active_time) {
|
|
this.stop();
|
|
}
|
|
if (!this.enabled && MathUtil.roundTo(this.age, 5) >= this.sleep_time) {
|
|
this.start();
|
|
}
|
|
} else {
|
|
//Once
|
|
if (this.enabled && MathUtil.roundTo(this.age, 5) >= this.active_time) {
|
|
this.stop();
|
|
}
|
|
}
|
|
return this;
|
|
}
|
|
stop() {
|
|
this.enabled = false;
|
|
this.age = 0;
|
|
return this;
|
|
}
|
|
jumpTo(second) {
|
|
let {tick_rate} = Wintersky.global_options;
|
|
let old_time = Math.round(this.view_age * tick_rate);
|
|
let new_time = Math.round(second * tick_rate);
|
|
if (this.loop_mode != 'once') {
|
|
new_time = Math.clamp(new_time, 0, Math.round(this.active_time * tick_rate) - 1);
|
|
}
|
|
if (old_time == new_time) return;
|
|
if (new_time < old_time) {
|
|
this.stop();
|
|
this.particles.slice().forEach(particle => {
|
|
particle.remove();
|
|
});
|
|
this.start();
|
|
} else if (!this.initialized) {
|
|
this.start();
|
|
}
|
|
while (Math.round(this.view_age * tick_rate) < new_time-1) {
|
|
this.tick(true);
|
|
}
|
|
this.tick(false);
|
|
return this;
|
|
}
|
|
updateFacingRotation(camera) {
|
|
this.particles.forEach(p => {
|
|
|
|
switch (this.config.particle_appearance_facing_camera_mode) {
|
|
case 'lookat_xyz':
|
|
p.mesh.lookAt(camera.position);
|
|
break;
|
|
case 'lookat_y':
|
|
var v = new THREE__default['default'].Vector3().copy(camera.position);
|
|
v.y = p.mesh.getWorldPosition(dummy_vec).y;
|
|
p.mesh.lookAt(v);
|
|
break;
|
|
case 'rotate_xyz':
|
|
p.mesh.rotation.copy(camera.rotation);
|
|
break;
|
|
case 'rotate_y':
|
|
p.mesh.rotation.copy(camera.rotation);
|
|
p.mesh.rotation.reorder('YXZ');
|
|
p.mesh.rotation.x = p.mesh.rotation.z = 0;
|
|
break;
|
|
case 'direction':
|
|
var q = new THREE__default['default'].Quaternion().setFromUnitVectors(Normals.z, p.speed);
|
|
p.mesh.rotation.setFromQuaternion(q);
|
|
break;
|
|
}
|
|
p.mesh.rotation.z += p.rotation||0;
|
|
});
|
|
}
|
|
spawnParticles(count) {
|
|
if (!count) return this;
|
|
|
|
if (this.config.emitter_rate_mode == 'steady') {
|
|
var max = this.calculate(this.config.emitter_rate_maximum, this.params())||0;
|
|
max = MathUtil.clamp(max, 0, Wintersky.global_options.max_emitter_particles);
|
|
count = MathUtil.clamp(count, 0, max-this.particles.length);
|
|
} else {
|
|
count = MathUtil.clamp(count, 0, Wintersky.global_options.max_emitter_particles-this.particles.length);
|
|
}
|
|
for (var i = 0; i < count; i++) {
|
|
if (this.dead_particles.length) {
|
|
var p = this.dead_particles.pop();
|
|
} else {
|
|
var p = new Particle(this);
|
|
}
|
|
p.add();
|
|
}
|
|
return count;
|
|
}
|
|
delete() {
|
|
[...this.particles, ...this.dead_particles].forEach(particle => {
|
|
if (particle.mesh.parent) particle.mesh.parent.remove(particle.mesh);
|
|
});
|
|
this.particles.splice(0, Infinity);
|
|
this.dead_particles.splice(0, Infinity);
|
|
if (this.local_space.parent) this.local_space.parent.remove(this.local_space);
|
|
if (this.global_space.parent) this.global_space.parent.remove(this.global_space);
|
|
removeFromArray(Wintersky.emitters, this);
|
|
}
|
|
}
|
|
Wintersky.Emitter = Emitter;
|
|
|
|
return Wintersky;
|
|
|
|
})));
|