mirror of
https://github.com/Refactorio/RedMew.git
synced 2025-01-05 22:53:39 +02:00
238f6b9d7b
* Rename task and queue to q * Threading to schedule * processing_queue to q
226 lines
6.9 KiB
Lua
226 lines
6.9 KiB
Lua
local Task = require 'utils.schedule'
|
|
local Global = require 'utils.global'
|
|
local Token = require 'utils.token'
|
|
local Command = require 'utils.command'
|
|
local Event = require 'utils.event'
|
|
local random = math.random
|
|
local ceil = math.ceil
|
|
local floor = math.floor
|
|
local format = string.format
|
|
|
|
local CreateParticles = {}
|
|
|
|
local settings = {
|
|
faction = 1.0,
|
|
particles_spawned_buffer = 0,
|
|
max_particles_per_second = 4000,
|
|
}
|
|
|
|
Global.register({
|
|
settings = settings,
|
|
}, function (tbl)
|
|
settings = tbl.settings
|
|
end)
|
|
|
|
---sets the scale of particles. 1.0 means 100%, 0.5 would mean spawn only 50% of the particles.
|
|
---@param fraction number
|
|
function CreateParticles.set_fraction(fraction)
|
|
if fraction < 0 or fraction > 1 then
|
|
error(format('Fraction must range from 0 to 1'))
|
|
end
|
|
|
|
settings.faction = fraction
|
|
end
|
|
|
|
---Returns the current scale
|
|
function CreateParticles.get_fraction()
|
|
return settings.faction
|
|
end
|
|
|
|
local function get_particle_cap()
|
|
return settings.max_particles_per_second * (settings.faction + 0.1)
|
|
end
|
|
|
|
---Returns whether or not more particles may be spawned, scale minimum is 0.1
|
|
local function may_spawn_particles()
|
|
return settings.particles_spawned_buffer < get_particle_cap()
|
|
end
|
|
|
|
--- resets the amount of particles in the past second so new ones may spawn
|
|
Event.on_nth_tick(63, function ()
|
|
settings.particles_spawned_buffer = 0
|
|
end)
|
|
|
|
Command.add('particle-scale', {
|
|
description = 'Provide a fraction between 0 and 1 to lower or increase the amount of (max) particles. Leave empty to view the current values.',
|
|
arguments = {'fraction'},
|
|
default_values = {fraction = false},
|
|
admin_only = true,
|
|
allowed_by_server = true,
|
|
}, function (arguments, player)
|
|
local p = player and player.print or print
|
|
|
|
local fraction = arguments.fraction
|
|
|
|
if fraction ~= false then
|
|
local scale = tonumber(fraction)
|
|
if scale == nil or scale < 0 or scale > 1 then
|
|
p('Scale must be a valid number ranging from 0 to 1')
|
|
return
|
|
end
|
|
CreateParticles.set_fraction(scale)
|
|
end
|
|
|
|
p(format('Particle fraction: %.2f', CreateParticles.get_fraction()))
|
|
p(format('Particles per second: %d', get_particle_cap()))
|
|
end)
|
|
|
|
---Scales the count to round the fraction up. Always returns at least 1 unless the particle limit is reached.
|
|
---Useful for particle spawning that influences gameplay for visual indications.
|
|
---@param count number
|
|
local function scale_ceil(count)
|
|
if not may_spawn_particles() then
|
|
return 0
|
|
end
|
|
|
|
local scale = settings.faction
|
|
if scale == 0 then
|
|
return 1
|
|
end
|
|
if scale < 1 and count > 1 then
|
|
count = ceil(count * scale)
|
|
end
|
|
|
|
return count
|
|
end
|
|
|
|
---Scales the count to round the fraction down.
|
|
---Useful for particle spawning that doesn't influence gameplay.
|
|
---@param count number
|
|
local function scale_floor(count)
|
|
local scale = settings.faction
|
|
if scale == 0 then
|
|
return 0
|
|
end
|
|
if not may_spawn_particles() then
|
|
return 0
|
|
end
|
|
if scale < 1 then
|
|
count = floor(count * scale)
|
|
end
|
|
|
|
return count
|
|
end
|
|
|
|
local on_play_particle = Token.register(function (params)
|
|
params.surface.create_entity(params.prototype)
|
|
end)
|
|
|
|
local function play_particle_sequence(surface, sequences)
|
|
local create_entity = surface.create_entity
|
|
for i = 1, #sequences do
|
|
local sequence = sequences[i]
|
|
local frame = sequence.frame
|
|
if frame == 1 then
|
|
create_entity(sequence.prototype)
|
|
else
|
|
Task.set_timeout_in_ticks(frame, on_play_particle, {surface = surface, prototype = sequence.prototype})
|
|
end
|
|
end
|
|
end
|
|
|
|
---@param create_entity function a reference to a surface.create_entity
|
|
---@param particle_count number particle count to spawn
|
|
---@param position Position
|
|
function CreateParticles.destroy_rock(create_entity, particle_count, position)
|
|
for _ = scale_floor(particle_count), 1, -1 do
|
|
settings.particles_spawned_buffer = settings.particles_spawned_buffer + 1
|
|
create_entity({
|
|
position = position,
|
|
name = 'stone-particle',
|
|
movement = {random(-5, 5) * 0.01, random(-5, 5) * 0.01},
|
|
frame_speed = 1,
|
|
vertical_speed = random(12, 14) * 0.01,
|
|
height = random(9, 11) * 0.1,
|
|
})
|
|
end
|
|
end
|
|
|
|
---@param create_entity function a reference to a surface.create_entity
|
|
---@param particle_count number particle count to spawn
|
|
---@param position Position
|
|
function CreateParticles.blood_explosion(create_entity, particle_count, position)
|
|
for _ = particle_count, 1, -1 do
|
|
create_entity({
|
|
position = position,
|
|
name = 'blood-particle',
|
|
movement = {random(-5, 5) * 0.01, random(-5, 5) * 0.01},
|
|
frame_speed = 1,
|
|
vertical_speed = random(10, 12) * 0.01,
|
|
height = random(5, 15) * 0.1,
|
|
})
|
|
end
|
|
end
|
|
|
|
---@param create_entity function a reference to a surface.create_entity
|
|
---@param particle_count number particle count to spawn
|
|
---@param position Position
|
|
function CreateParticles.mine_rock(create_entity, particle_count, position)
|
|
for _ = scale_floor(particle_count), 1, -1 do
|
|
settings.particles_spawned_buffer = settings.particles_spawned_buffer + 1
|
|
create_entity({
|
|
position = position,
|
|
name = 'stone-particle',
|
|
movement = {random(-5, 5) * 0.01, random(-5, 5) * 0.01},
|
|
frame_speed = 1,
|
|
vertical_speed = random(8, 10) * 0.01,
|
|
height = random(5, 8) * 0.1,
|
|
})
|
|
end
|
|
end
|
|
|
|
|
|
---Creates a prototype for LuaSurface.create_entity
|
|
---@param particle string name of the particle
|
|
---@param x number
|
|
---@param y number
|
|
local function create_ceiling_prototype(particle, x, y)
|
|
return {
|
|
name = particle,
|
|
position = {x = x + random(0, 1), y = y + random(0, 1)},
|
|
movement = {random(-5, 5) * 0.002, random(-5, 5) * 0.002},
|
|
frame_speed = 1,
|
|
vertical_speed = 0,
|
|
height = 3
|
|
}
|
|
end
|
|
|
|
---Creates a crumbling effect from the ceiling
|
|
---@param surface LuaSurface
|
|
---@param position table
|
|
function CreateParticles.ceiling_crumble(surface, position)
|
|
local sequences = {}
|
|
local x = position.x
|
|
local y = position.y
|
|
local smoke_scale = scale_ceil(2)
|
|
local stone_scale = scale_floor(4)
|
|
|
|
-- pre-calculate how many particles will be spawned. Prevents spawning too many particles over ticks.
|
|
local particles = settings.particles_spawned_buffer
|
|
|
|
for i = 1, smoke_scale do
|
|
particles = particles + 1
|
|
sequences[i] = {frame = i*random(1,15), prototype = create_ceiling_prototype('explosion-remnants-particle', x, y)}
|
|
end
|
|
for i = smoke_scale + 1, smoke_scale + stone_scale do
|
|
particles = particles + 1
|
|
sequences[i] = {frame = i*random(1,15), prototype = create_ceiling_prototype('stone-particle', x, y)}
|
|
end
|
|
|
|
settings.particles_spawned_buffer = particles
|
|
|
|
play_particle_sequence(surface, sequences)
|
|
end
|
|
|
|
return CreateParticles
|