From d5a3a2b344406a6755702bdd4ea73a17841df4fd Mon Sep 17 00:00:00 2001 From: JannisX11 Date: Fri, 9 Oct 2020 14:14:29 +0200 Subject: [PATCH] Add wintersky module --- index.html | 2 +- lib/wintersky.umd.js | 1106 ++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 1107 insertions(+), 1 deletion(-) create mode 100644 lib/wintersky.umd.js diff --git a/index.html b/index.html index 6ebaf6c9..8bbd071f 100644 --- a/index.html +++ b/index.html @@ -53,7 +53,7 @@ - + diff --git a/lib/wintersky.umd.js b/lib/wintersky.umd.js new file mode 100644 index 00000000..e8c89b35 --- /dev/null +++ b/lib/wintersky.umd.js @@ -0,0 +1,1106 @@ +(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.value == '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().start(); + this.particles.slice().forEach(particle => { + particle.remove(); + }); + } 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; + +})));