mirror of
https://github.com/ComfyFactory/ComfyFactorio.git
synced 2025-01-22 03:38:48 +02:00
169 lines
5.5 KiB
Lua
169 lines
5.5 KiB
Lua
local Public = {}
|
|
|
|
local Global = require 'utils.global'
|
|
local explosives = {}
|
|
Global.register(
|
|
explosives,
|
|
function(tbl)
|
|
explosives = tbl
|
|
end
|
|
)
|
|
|
|
local explosives_effects = {"explosion", "grenade-explosion", "explosion", "grenade-explosion", "big-artillery-explosion", "massive-explosion", "land-mine-explosion", "storage-tank-explosion"}
|
|
local table_insert = table.insert
|
|
local table_remove = table.remove
|
|
local math_floor = math.floor
|
|
local math_sqrt = math.sqrt
|
|
local math_random = math.random
|
|
local speed = 6
|
|
local valid_container_types = {
|
|
['car'] = true,
|
|
['cargo-wagon'] = true,
|
|
['container'] = true,
|
|
['logistic-container'] = true,
|
|
['spider-vehicle'] = true,
|
|
}
|
|
|
|
local maximum_radius = 64
|
|
local explosive_vectors = {}
|
|
for x = maximum_radius * -1, maximum_radius, 1 do
|
|
for y = maximum_radius * -1, maximum_radius, 1 do
|
|
local d = math_floor(math_sqrt(x^2 + y^2)) + 1
|
|
if d <= maximum_radius then
|
|
if not explosive_vectors[d] then explosive_vectors[d] = {} end
|
|
table_insert(explosive_vectors[d], {x,y})
|
|
end
|
|
end
|
|
end
|
|
|
|
local function draw_effects(surface, instance)
|
|
local vectors = explosive_vectors[instance.current_r]
|
|
if not vectors then return end
|
|
local center_position_x = instance.position.x
|
|
local center_position_y = instance.position.y
|
|
for _, v in pairs(vectors) do
|
|
if math_random(0, instance.current_r * 0.15 * (explosives.instance_count * 0.5)) == 0 then
|
|
local position = {center_position_x + v[1], center_position_y + v[2]}
|
|
local name = explosives_effects[math_random(1, 8)]
|
|
surface.create_entity({name = name, position = position, target = position})
|
|
end
|
|
end
|
|
end
|
|
|
|
local function damage_entity(entity, amount)
|
|
if not entity.valid then return end
|
|
if not entity.health then return end
|
|
if entity.health <= 0 then return end
|
|
if not entity.destructible then return end
|
|
entity.damage(amount, 'player', 'explosion')
|
|
return true
|
|
end
|
|
|
|
local function damage_entities(surface, instance)
|
|
local vectors = explosive_vectors[instance.current_r]
|
|
if not vectors then return end
|
|
local center_position_x = instance.position.x
|
|
local center_position_y = instance.position.y
|
|
for _, v in pairs(vectors) do
|
|
local position = {center_position_x + v[1], center_position_y + v[2]}
|
|
for _, entity in pairs(surface.find_entities_filtered({area = {position, {position[1] + 1, position[2] + 1}}, type = {"corpse", "explosion"}, invert = true,})) do
|
|
if instance.damage_remaining < 200 then
|
|
if damage_entity(entity, instance.damage_remaining) then return end
|
|
else
|
|
if damage_entity(entity, 200) then instance.damage_remaining = instance.damage_remaining - 200 end
|
|
end
|
|
end
|
|
end
|
|
return true
|
|
end
|
|
|
|
local function damage_tiles(surface, instance)
|
|
local vectors = explosive_vectors[instance.current_r]
|
|
if not vectors then return end
|
|
local center_position_x = instance.position.x
|
|
local center_position_y = instance.position.y
|
|
local destructible_tiles = explosives.destructible_tiles
|
|
for _, v in pairs(vectors) do
|
|
local position = {center_position_x + v[1], center_position_y + v[2]}
|
|
local tile = surface.get_tile(position)
|
|
local tile_health = destructible_tiles[tile.name]
|
|
if tile_health then
|
|
if instance.damage_remaining < tile_health then
|
|
return
|
|
end
|
|
instance.damage_remaining = instance.damage_remaining - tile_health
|
|
surface.set_tiles({{name = "nuclear-ground", position = position}}, true, false, false, false)
|
|
end
|
|
end
|
|
return true
|
|
end
|
|
|
|
local function process_explosion(instance)
|
|
local surface = game.surfaces[instance.surface_index]
|
|
if not surface then return end
|
|
if not surface.valid then return end
|
|
draw_effects(surface, instance)
|
|
if not damage_entities(surface, instance) then return end
|
|
if not damage_tiles(surface, instance) then return end
|
|
instance.current_r = instance.current_r + 1
|
|
if instance.current_r > instance.max_r then return end
|
|
return true
|
|
end
|
|
|
|
function spawn_explosion(surface, position, amount)
|
|
if not explosives.instances then explosives.instances = {} end
|
|
table_insert(explosives.instances, {
|
|
surface_index = surface.index,
|
|
damage_remaining = amount * 500,
|
|
position = position,
|
|
current_r = 1,
|
|
max_r = math_floor(amount / 50) + 5,
|
|
})
|
|
end
|
|
|
|
local function on_tick()
|
|
if not explosives.instances then return end
|
|
explosives.instance_count = #explosives.instances
|
|
for k, instance in pairs(explosives.instances) do
|
|
local success = process_explosion(instance)
|
|
if not success then table_remove(explosives.instances, k) end
|
|
end
|
|
if #explosives.instances == 0 then explosives.instances = nil end
|
|
end
|
|
|
|
local function on_entity_died(event)
|
|
local entity = event.entity
|
|
if not entity.valid then return end
|
|
if not valid_container_types[entity.type] then return end
|
|
|
|
local inventory = defines.inventory.chest
|
|
if entity.type == 'car' or entity.type == "spider-vehicle" then
|
|
inventory = defines.inventory.car_trunk
|
|
end
|
|
|
|
local i = entity.get_inventory(inventory)
|
|
local amount = i.get_item_count('explosives')
|
|
if not amount then return end
|
|
if amount < 1 then return end
|
|
|
|
spawn_explosion(entity.surface, {x = entity.position.x, y = entity.position.y}, amount)
|
|
end
|
|
|
|
function Public.set_destructible_tile(tile_name, health)
|
|
explosives.destructible_tiles[tile_name] = health
|
|
end
|
|
|
|
function Public.get_table()
|
|
return explosives
|
|
end
|
|
|
|
local function on_init()
|
|
explosives.destructible_tiles = {}
|
|
end
|
|
|
|
local Event = require 'utils.event'
|
|
Event.on_init(on_init)
|
|
Event.on_nth_tick(speed, on_tick)
|
|
Event.add(defines.events.on_entity_died, on_entity_died)
|
|
|
|
return Public |