1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-26 03:52:22 +02:00
ComfyFactorio/modules/wave_defense/threat_events.lua

196 lines
8.0 KiB
Lua
Raw Normal View History

2019-10-28 17:38:36 +01:00
local WD = require "modules.wave_defense.table"
2019-10-08 17:41:15 +02:00
local threat_values = require "modules.wave_defense.threat_values"
2019-10-28 17:38:36 +01:00
local BiterRolls = require 'modules.wave_defense.biter_rolls'
2019-10-08 17:41:15 +02:00
local math_random = math.random
2019-10-28 17:38:36 +01:00
local Public = {}
2019-10-10 04:33:51 +02:00
local function remove_unit(entity)
2019-10-28 17:38:36 +01:00
local wave_defense_table = WD.get_table()
local unit_number = entity.unit_number
2019-11-14 16:43:54 +01:00
if not wave_defense_table.active_biters[unit_number] then return end
local m = 1
if global.biter_health_boost_units[unit_number] then
m = 1 / global.biter_health_boost_units[unit_number][2]
end
local active_threat_loss = math.round(threat_values[entity.name] * m, 2)
wave_defense_table.active_biter_threat = wave_defense_table.active_biter_threat - active_threat_loss
2019-10-28 17:38:36 +01:00
wave_defense_table.active_biter_count = wave_defense_table.active_biter_count - 1
wave_defense_table.active_biters[unit_number] = nil
2019-10-10 04:33:51 +02:00
end
2019-10-29 12:26:04 +01:00
local function place_nest_near_unit_group(wave_defense_table)
2019-10-28 17:38:36 +01:00
local group = wave_defense_table.unit_groups[math_random(1, #wave_defense_table.unit_groups)]
2019-10-15 04:20:40 +02:00
if not group then return end
if not group.valid then return end
if not group.members then return end
if not group.members[1] then return end
local unit = group.members[math_random(1, #group.members)]
if not unit.valid then return end
local name = "biter-spawner"
if math_random(1, 3) == 1 then name = "spitter-spawner" end
local position = unit.surface.find_non_colliding_position(name, unit.position, 12, 1)
2019-10-10 04:33:51 +02:00
if not position then return end
2019-10-28 17:38:36 +01:00
local r = wave_defense_table.nest_building_density
2019-10-10 07:24:49 +02:00
if unit.surface.count_entities_filtered({type = "unit-spawner", area = {{position.x - r, position.y - r},{position.x + r, position.y + r}}}) > 0 then return end
unit.surface.create_entity({name = name, position = position, force = unit.force})
2019-10-10 04:33:51 +02:00
unit.surface.create_entity({name = "blood-explosion-huge", position = position})
unit.surface.create_entity({name = "blood-explosion-huge", position = unit.position})
remove_unit(unit)
unit.destroy()
wave_defense_table.threat = wave_defense_table.threat - threat_values[name]
2019-10-29 12:26:04 +01:00
return true
end
function Public.build_nest()
local wave_defense_table = WD.get_table()
if wave_defense_table.threat < 1024 then return end
2019-10-29 12:26:04 +01:00
if #wave_defense_table.unit_groups == 0 then return end
2019-10-31 16:18:17 +01:00
for _ = 1, 2, 1 do
2019-10-29 12:26:04 +01:00
if place_nest_near_unit_group(wave_defense_table) then return end
end
2019-10-10 04:33:51 +02:00
end
2019-10-28 17:38:36 +01:00
function Public.build_worm()
local wave_defense_table = WD.get_table()
if wave_defense_table.threat < 512 then return end
2019-10-28 17:38:36 +01:00
if math_random(1, wave_defense_table.worm_building_chance) ~= 1 then return end
if #wave_defense_table.unit_groups == 0 then return end
local group = wave_defense_table.unit_groups[math_random(1, #wave_defense_table.unit_groups)]
2019-10-15 04:20:40 +02:00
if not group then return end
if not group.valid then return end
if not group.members then return end
if not group.members[1] then return end
local unit = group.members[math_random(1, #group.members)]
if not unit.valid then return end
2019-10-26 13:56:02 +02:00
local position = unit.surface.find_non_colliding_position("assembling-machine-1", unit.position, 8, 1)
BiterRolls.wave_defense_set_worm_raffle(wave_defense_table.wave_number)
2019-10-28 17:38:36 +01:00
local worm = BiterRolls.wave_defense_roll_worm_name()
2019-10-10 04:33:51 +02:00
if not position then return end
2019-10-28 17:38:36 +01:00
local r = wave_defense_table.worm_building_density
2019-10-10 07:24:49 +02:00
if unit.surface.count_entities_filtered({type = "turret", area = {{position.x - r, position.y - r},{position.x + r, position.y + r}}}) > 0 then return end
2019-10-10 04:33:51 +02:00
unit.surface.create_entity({name = worm, position = position, force = unit.force})
unit.surface.create_entity({name = "blood-explosion-huge", position = position})
unit.surface.create_entity({name = "blood-explosion-huge", position = unit.position})
remove_unit(unit)
unit.destroy()
2019-10-28 17:38:36 +01:00
wave_defense_table.threat = wave_defense_table.threat - threat_values[worm]
2019-10-10 04:33:51 +02:00
end
2019-10-31 10:13:47 +01:00
--[[
2019-10-08 17:41:15 +02:00
local function get_circle_vectors(radius)
local vectors = {}
for x = radius * -1, radius, 1 do
for y = radius * -1, radius, 1 do
if math.sqrt(x^2 + y^2) <= radius then
vectors[#vectors + 1] = {x, y}
end
end
end
return vectors
end
local acid_nova_entities = {
2019-10-31 10:13:47 +01:00
["small-biter"] = {projectile = "acid-stream-worm-small", vectors = get_circle_vectors(3), threat_cost = 32},
["medium-biter"] = {projectile = "acid-stream-worm-medium", vectors = get_circle_vectors(4), threat_cost = 64},
["big-biter"] = {projectile = "acid-stream-worm-big", vectors = get_circle_vectors(5), threat_cost = 96},
["behemoth-biter"] = {projectile = "acid-stream-worm-behemoth", vectors = get_circle_vectors(6), threat_cost = 128},
2019-10-08 17:41:15 +02:00
}
local function acid_nova(entity)
2019-10-28 17:38:36 +01:00
local wave_defense_table = WD.get_table()
2019-10-08 17:41:15 +02:00
if not acid_nova_entities[entity.name] then return end
2019-10-28 17:38:36 +01:00
if wave_defense_table.threat < 100000 then return end
2019-10-31 10:13:47 +01:00
if math.random(1, 32) ~= 1 then return end
for _ = 1, 8, 1 do
2019-10-08 23:50:57 +02:00
local i = math_random(1, #acid_nova_entities[entity.name].vectors)
2019-10-08 17:41:15 +02:00
entity.surface.create_entity({
name = acid_nova_entities[entity.name].projectile,
position = entity.position,
force = entity.force.name,
source = entity.position,
target = {x = entity.position.x + acid_nova_entities[entity.name].vectors[i][1], y = entity.position.y + acid_nova_entities[entity.name].vectors[i][2]},
2019-10-15 04:20:40 +02:00
max_range = 10,
2019-10-08 17:41:15 +02:00
speed = 0.001
})
end
2019-10-28 17:38:36 +01:00
wave_defense_table.threat = wave_defense_table.threat - acid_nova_entities[entity.name].threat_cost
2019-10-08 17:41:15 +02:00
return true
end
2019-10-31 10:13:47 +01:00
]]
2019-10-08 23:50:57 +02:00
local function shred_simple_entities(entity)
2019-10-28 17:38:36 +01:00
local wave_defense_table = WD.get_table()
if wave_defense_table.threat < 25000 then return end
2019-10-26 11:47:46 +02:00
local simple_entities = entity.surface.find_entities_filtered({type = "simple-entity", area = {{entity.position.x - 3, entity.position.y - 3},{entity.position.x + 3, entity.position.y + 3}}})
2019-10-08 23:50:57 +02:00
if #simple_entities == 0 then return end
2019-10-09 03:25:00 +02:00
if #simple_entities > 1 then table.shuffle_table(simple_entities) end
2019-10-28 17:38:36 +01:00
local r = math.floor(wave_defense_table.threat * 0.00004)
2019-10-26 11:47:46 +02:00
if r < 1 then r = 1 end
local count = math.random(1, r)
--local count = 1
2019-10-08 23:50:57 +02:00
local damage_dealt = 0
for i = 1, count, 1 do
if not simple_entities[i] then break end
if simple_entities[i].valid then
if simple_entities[i].health then
damage_dealt = damage_dealt + simple_entities[i].health
simple_entities[i].die("neutral", simple_entities[i])
end
end
end
if damage_dealt == 0 then return end
2019-10-28 17:38:36 +01:00
local threat_cost = math.floor(damage_dealt * wave_defense_table.simple_entity_shredding_cost_modifier)
2019-10-08 23:50:57 +02:00
if threat_cost < 1 then threat_cost = 1 end
2019-10-28 17:38:36 +01:00
wave_defense_table.threat = wave_defense_table.threat - threat_cost
2019-10-08 23:50:57 +02:00
end
local function spawn_unit_spawner_inhabitants(wave_defense_table, entity)
if entity.type ~= "unit-spawner" then return end
local count = 8 + math.floor(wave_defense_table.wave_number * 0.02)
if count > 128 then count = 128 end
BiterRolls.wave_defense_set_unit_raffle(wave_defense_table.wave_number)
for _ = 1, count, 1 do
2019-10-29 20:07:19 +01:00
local position = {entity.position.x + (-4 + math.random(0, 8)), entity.position.y + (-4 + math.random(0, 8))}
if math.random(1,4) == 1 then
entity.surface.create_entity({name = BiterRolls.wave_defense_roll_spitter_name(), position = position, force = "enemy"})
else
entity.surface.create_entity({name = BiterRolls.wave_defense_roll_biter_name(), position = position, force = "enemy"})
end
end
end
2019-10-08 23:50:57 +02:00
local function on_entity_died(event)
2019-10-28 17:38:36 +01:00
local wave_defense_table = WD.get_table()
2019-10-25 08:09:39 +02:00
local entity = event.entity
if not entity.valid then return end
2019-10-08 23:50:57 +02:00
2019-10-25 08:09:39 +02:00
if entity.type == "unit" then
2019-10-28 17:38:36 +01:00
wave_defense_table.threat = math.round(wave_defense_table.threat - threat_values[entity.name] * global.biter_health_boost, 2)
2019-10-25 08:09:39 +02:00
remove_unit(entity)
2019-10-31 10:13:47 +01:00
--acid_nova(entity)
2019-10-25 08:09:39 +02:00
else
if entity.force.index == 2 then
if entity.health then
if threat_values[entity.name] then
2019-10-28 17:38:36 +01:00
wave_defense_table.threat = wave_defense_table.threat - threat_values[entity.name]
2019-10-25 08:09:39 +02:00
end
spawn_unit_spawner_inhabitants(wave_defense_table, entity)
2019-10-25 08:09:39 +02:00
end
end
2019-10-08 23:50:57 +02:00
end
2019-10-25 08:09:39 +02:00
if entity.force.index == 3 then
2019-10-08 23:50:57 +02:00
if event.cause then
if event.cause.valid then
if event.cause.force.index == 2 then
2019-10-25 08:09:39 +02:00
shred_simple_entities(entity)
2019-10-08 23:50:57 +02:00
end
end
end
end
2019-10-08 17:41:15 +02:00
end
local event = require 'utils.event'
2019-10-28 17:38:36 +01:00
event.add(defines.events.on_entity_died, on_entity_died)
return Public