blockbench/lib/wintersky.umd.js
JannisX11 28315784d7 Add particle tick rate setting
Fix two Wintersky issues
2020-10-19 22:11:04 +02:00

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;
})));