1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-16 02:47:48 +02:00
ComfyFactorio/modules/explosives.lua

444 lines
12 KiB
Lua
Raw Normal View History

2020-05-05 11:49:54 +02:00
local Public = {}
2024-05-28 10:43:10 +02:00
local Event = require 'utils.event'
2020-05-05 11:49:54 +02:00
local Global = require 'utils.global'
local Collapse = require 'modules.collapse'
local this = {
explosives = {},
settings = {
2024-05-28 10:43:10 +02:00
disabled = false,
slow_explode = false,
2024-06-19 21:01:10 +02:00
slow_explode_tick = 300,
check_growth_below_void = false,
valid_items = {
['explosives'] = 500,
['cliff-explosives'] = 750
}
}
}
2020-05-05 11:49:54 +02:00
Global.register(
this,
2024-06-19 21:01:10 +02:00
function (tbl)
this = tbl
2020-05-05 11:49:54 +02:00
end
)
2019-10-11 05:38:27 +02:00
local math_abs = math.abs
local math_floor = math.floor
local math_sqrt = math.sqrt
local math_round = math.round
local math_random = math.random
2020-05-05 11:49:54 +02:00
local shuffle_table = table.shuffle_table
local speed = 6
2019-10-11 07:13:08 +02:00
local density = 1
local density_r = density * 0.5
local valid_container_types = {
2020-07-06 15:45:09 +02:00
['container'] = true,
['logistic-container'] = true,
['car'] = true,
['cargo-wagon'] = true
2019-10-11 05:38:27 +02:00
}
2024-05-30 10:45:26 +02:00
local disabled_container_names = {
['logistic-chest-buffer'] = true
}
2019-10-11 05:38:27 +02:00
local function pos_to_key(position)
2020-07-06 15:45:09 +02:00
return tostring(position.x .. '_' .. position.y)
2019-10-11 05:38:27 +02:00
end
local function check_y_pos(position)
if not this.settings.check_growth_below_void then
return false
end
if not position or not position.y then
return false
end
local collapse_pos = Collapse.get_position()
local radius = 10
local dy = position.y - collapse_pos.y
if dy ^ 2 < radius ^ 2 then
return true
end
if position.y >= collapse_pos.y then
return true
else
return false
end
end
2019-10-11 05:38:27 +02:00
local function get_explosion_name(health)
2020-07-06 15:45:09 +02:00
if health < 2500 then
return 'explosion'
end
if health < 25000 then
return 'big-explosion'
end
return 'big-artillery-explosion'
2019-10-11 05:38:27 +02:00
end
2023-11-21 01:35:47 +02:00
local function cell_birth(surface_index, origin_position, origin_tick, position, health, atomic)
2020-07-06 15:45:09 +02:00
local key = pos_to_key(position)
2019-10-11 05:38:27 +02:00
2020-07-06 15:45:09 +02:00
--Merge cells that are overlapping.
if this.explosives.cells[key] then
this.explosives.cells[key].health = this.explosives.cells[key].health + health
2020-07-06 15:45:09 +02:00
return
end
2019-10-11 05:38:27 +02:00
2023-11-21 01:35:47 +02:00
if not atomic then
atomic = false
end
2020-07-06 15:45:09 +02:00
--Spawn new cell.
this.explosives.cells[key] = {
2020-07-06 15:45:09 +02:00
surface_index = surface_index,
origin_position = origin_position,
origin_tick = origin_tick,
2024-06-19 21:01:10 +02:00
position = { x = position.x, y = position.y },
2020-07-06 15:45:09 +02:00
spawn_tick = game.tick + speed,
2023-11-21 01:35:47 +02:00
health = health,
atomic = atomic
2020-07-06 15:45:09 +02:00
}
2019-10-11 05:38:27 +02:00
end
local function grow_cell(cell)
shuffle_table(this.explosives.vectors)
2020-07-06 15:45:09 +02:00
local radius = math_floor((game.tick - cell.origin_tick) / 9) + 2
local positions = {}
for i = 1, 4, 1 do
local position = {
x = cell.position.x + this.explosives.vectors[i][1],
y = cell.position.y + this.explosives.vectors[i][2]
2020-07-06 15:45:09 +02:00
}
if not this.explosives.cells[pos_to_key(position)] then
2021-03-24 17:46:00 +02:00
local distance = math_sqrt((cell.origin_position.x - position.x) ^ 2 + (cell.origin_position.y - position.y) ^ 2)
2020-07-06 15:45:09 +02:00
if distance < radius then
positions[#positions + 1] = position
end
end
end
if #positions == 0 then
positions[#positions + 1] = {
x = cell.position.x + this.explosives.vectors[1][1],
y = cell.position.y + this.explosives.vectors[1][2]
2020-07-06 15:45:09 +02:00
}
end
local new_cell_health = math_round(cell.health / #positions, 3) - this.explosives.damage_decay
2020-07-06 15:45:09 +02:00
if new_cell_health <= 0 then
return
end
2023-11-21 01:35:47 +02:00
if not cell.atomic then
cell.atomic = false
end
2020-07-06 15:45:09 +02:00
for _, p in pairs(positions) do
2023-11-21 01:35:47 +02:00
cell_birth(cell.surface_index, cell.origin_position, cell.origin_tick, p, new_cell_health, cell.atomic)
2020-07-06 15:45:09 +02:00
end
2019-10-11 05:38:27 +02:00
end
2019-10-13 05:24:11 +02:00
local function damage_entity(entity, cell)
2020-07-06 15:45:09 +02:00
if not entity.valid then
return true
end
if not entity.health then
return true
end
if entity.health <= 0 then
return true
end
if not entity.destructible then
return true
end
if this.explosives.whitelist_entity[entity.name] then
2020-07-06 15:45:09 +02:00
return true
end
local damage_required = entity.health
for _ = 1, 4, 1 do
if damage_required > cell.health then
entity.damage(cell.health, 'player', 'explosion')
return false
end
local damage_dealt = entity.damage(damage_required, 'player', 'explosion')
cell.health = cell.health - damage_required
if not entity then
return true
end
if not entity.valid then
return true
end
if entity.health <= 0 then
return true
end
damage_required = math_floor(entity.health * (damage_required / damage_dealt)) + 1
end
2019-10-13 05:24:11 +02:00
end
2019-10-11 05:38:27 +02:00
local function damage_area(cell)
2020-07-06 15:45:09 +02:00
local surface = game.surfaces[cell.surface_index]
if not surface then
return
end
if not surface.valid then
return
end
if math_random(1, 4) == 1 then
2023-11-21 01:35:47 +02:00
if cell.atomic then
2024-06-19 21:01:10 +02:00
surface.create_entity({ name = 'nuke-explosion', position = cell.position })
2023-11-21 01:35:47 +02:00
else
2024-06-19 21:01:10 +02:00
surface.create_entity({ name = get_explosion_name(cell.health), position = cell.position })
2023-11-21 01:35:47 +02:00
end
2020-07-06 15:45:09 +02:00
end
for _, entity in pairs(
surface.find_entities(
{
2024-06-19 21:01:10 +02:00
{ cell.position.x - density_r, cell.position.y - density_r },
{ cell.position.x + density_r, cell.position.y + density_r }
2020-07-06 15:45:09 +02:00
}
)
) do
if not damage_entity(entity, cell) then
return
end
end
local tile = surface.get_tile(cell.position)
if this.explosives.destructible_tiles[tile.name] then
2020-07-06 15:45:09 +02:00
local key = pos_to_key(tile.position)
if not this.explosives.tiles[key] then
this.explosives.tiles[key] = this.explosives.destructible_tiles[tile.name]
2020-07-06 15:45:09 +02:00
end
if cell.health > this.explosives.tiles[key] then
cell.health = cell.health - this.explosives.tiles[key]
this.explosives.tiles[key] = nil
2021-03-24 17:46:00 +02:00
if math_abs(tile.position.y) < surface.map_gen_settings.height * 0.5 and math_abs(tile.position.x) < surface.map_gen_settings.width * 0.5 then
if not check_y_pos(tile.position) then
2024-06-19 21:01:10 +02:00
surface.set_tiles({ { name = 'landfill', position = tile.position } }, true)
end
2020-07-06 15:45:09 +02:00
end
else
this.explosives.tiles[key] = this.explosives.tiles[key] - cell.health
2020-07-06 15:45:09 +02:00
return
end
end
return true
2019-10-11 05:38:27 +02:00
end
local function life_cycle(cell)
2020-07-06 15:45:09 +02:00
if not damage_area(cell) then
return
end
grow_cell(cell)
2019-10-11 05:38:27 +02:00
end
2020-07-06 15:45:09 +02:00
local function tick()
2024-05-28 10:43:10 +02:00
if this.settings.disabled then
return
end
if this.settings.slow_explode then
local count = 0
for key, cell in pairs(this.explosives.cells) do
if cell.spawn_tick < game.tick then
count = count + 1
2024-06-19 21:01:10 +02:00
if count < this.settings.slow_explode_tick then
life_cycle(cell)
this.explosives.cells[key] = nil
else
cell.spawn_tick = game.tick + speed
end
end
end
else
for key, cell in pairs(this.explosives.cells) do
if cell.spawn_tick < game.tick then
life_cycle(cell)
this.explosives.cells[key] = nil
end
2020-07-06 15:45:09 +02:00
end
end
2020-07-06 15:45:09 +02:00
if game.tick % 216000 == 0 then
this.explosives.tiles = {}
2020-07-06 15:45:09 +02:00
end
2019-10-11 05:38:27 +02:00
end
local function check_entity_for_items(item)
local items = this.settings.valid_items
for name, damage in pairs(items) do
local amount = item.get_item_count(name)
if amount and amount > 1 then
return amount, damage
end
end
return false
end
2019-10-11 07:13:08 +02:00
local function on_entity_died(event)
2024-05-28 10:43:10 +02:00
if this.settings.disabled then
return false
end
2020-07-06 15:45:09 +02:00
local entity = event.entity
if not entity.valid then
return
end
2024-05-30 10:45:26 +02:00
if disabled_container_names[entity.name] then
return
end
2020-07-06 15:45:09 +02:00
if not valid_container_types[entity.type] then
return
end
if this.explosives.surface_whitelist then
if not this.explosives.surface_whitelist[entity.surface.name] then
2020-07-06 15:45:09 +02:00
return
end
end
local inventory = defines.inventory.chest
if entity.type == 'car' then
inventory = defines.inventory.car_trunk
end
local item = entity.get_inventory(inventory)
local amount, damage = check_entity_for_items(item)
2020-07-06 15:45:09 +02:00
if not amount then
return
end
local final_damage = amount * damage
2024-06-19 21:01:10 +02:00
cell_birth(entity.surface.index, { x = entity.position.x, y = entity.position.y }, game.tick, { x = entity.position.x, y = entity.position.y }, final_damage)
2020-05-05 11:49:54 +02:00
end
2021-05-23 21:13:21 +02:00
function Public.detonate_chest(entity)
2024-05-28 10:43:10 +02:00
if this.settings.disabled then
return false
end
2021-06-06 20:14:26 +02:00
if not entity or not entity.valid then
2021-05-23 21:13:21 +02:00
return false
end
if not valid_container_types[entity.type] then
return false
end
if this.explosives.surface_whitelist then
if not this.explosives.surface_whitelist[entity.surface.name] then
2021-05-23 21:13:21 +02:00
return false
end
end
local inventory = defines.inventory.chest
if entity.type == 'car' then
inventory = defines.inventory.car_trunk
end
local item = entity.get_inventory(inventory)
local amount, damage = check_entity_for_items(item)
2021-05-23 21:13:21 +02:00
if not amount then
return false
end
2021-05-24 17:21:25 +02:00
if amount < 99 then
2021-05-23 21:13:21 +02:00
return false
end
2024-06-19 21:01:10 +02:00
cell_birth(entity.surface.index, { x = entity.position.x, y = entity.position.y }, game.tick, { x = entity.position.x, y = entity.position.y }, amount * damage)
2021-05-23 21:13:21 +02:00
return true
end
2023-11-21 01:35:47 +02:00
function Public.detonate_entity(entity, amount, damage)
2024-05-28 10:43:10 +02:00
if this.settings.disabled then
return false
end
2023-11-21 01:35:47 +02:00
if not entity or not entity.valid then
return false
end
if not amount then
amount = 200
end
if not damage then
damage = 700
end
2024-06-19 21:01:10 +02:00
cell_birth(entity.surface.index, { x = entity.position.x, y = entity.position.y }, game.tick, { x = entity.position.x, y = entity.position.y }, amount * damage, true)
2023-11-21 01:35:47 +02:00
return true
end
2020-05-05 11:49:54 +02:00
function Public.reset()
this.explosives.cells = {}
this.explosives.tiles = {}
if not this.explosives.vectors then
2024-06-19 21:01:10 +02:00
this.explosives.vectors = { { density, 0 }, { density * -1, 0 }, { 0, density }, { 0, density * -1 } }
2020-07-06 15:45:09 +02:00
end
if not this.explosives.damage_decay then
this.explosives.damage_decay = 10
2020-07-06 15:45:09 +02:00
end
if not this.explosives.destructible_tiles then
this.explosives.destructible_tiles = {}
2020-07-06 15:45:09 +02:00
end
if not this.explosives.whitelist_entity then
this.explosives.whitelist_entity = {}
2020-07-06 15:45:09 +02:00
end
2020-05-05 11:49:54 +02:00
end
function Public.set_destructible_tile(tile_name, health)
this.explosives.destructible_tiles[tile_name] = health
2020-07-06 15:45:09 +02:00
end
function Public.set_whitelist_entity(entity)
if entity then
this.explosives.whitelist_entity[entity] = true
2020-07-06 15:45:09 +02:00
end
2020-05-05 11:49:54 +02:00
end
2020-05-05 14:14:51 +02:00
function Public.set_surface_whitelist(list)
this.explosives.surface_whitelist = list
2020-05-05 14:14:51 +02:00
end
2024-05-28 10:43:10 +02:00
function Public.disable(state)
this.settings.disabled = state or false
end
2020-05-05 11:49:54 +02:00
function Public.get_table()
return this.explosives
end
function Public.check_growth_below_void(value)
this.settings.check_growth_below_void = value or false
2019-10-11 07:13:08 +02:00
end
function Public.slow_explode(value)
this.settings.slow_explode = value or false
end
2024-06-19 21:01:10 +02:00
function Public.slow_explode_tick(value)
this.settings.slow_explode_tick = value or 300
end
2019-10-27 20:26:55 +02:00
local function on_init()
2020-07-06 15:45:09 +02:00
Public.reset()
2019-10-11 05:38:27 +02:00
end
2020-05-05 11:49:54 +02:00
Event.on_init(on_init)
Event.on_nth_tick(speed, tick)
Event.add(defines.events.on_entity_died, on_entity_died)
2020-07-06 15:45:09 +02:00
return Public