mirror of
https://github.com/ComfyFactory/ComfyFactorio.git
synced 2025-01-24 03:47:58 +02:00
475 lines
18 KiB
Lua
475 lines
18 KiB
Lua
local Price_raffle = require 'maps.expanse.price_raffle'
|
|
local BiterRaffle = require 'functions.biter_raffle'
|
|
local Task = require 'utils.task'
|
|
local Token = require 'utils.token'
|
|
local Public = {}
|
|
|
|
local ores = {'copper-ore', 'iron-ore', 'stone', 'coal'}
|
|
local price_modifiers = {
|
|
['unit-spawner'] = -256,
|
|
['unit'] = -16,
|
|
['turret'] = -128,
|
|
['tree'] = -8,
|
|
['simple-entity'] = 2,
|
|
['cliff'] = -128,
|
|
['water'] = -5,
|
|
['water-green'] = -5,
|
|
['deepwater'] = -5,
|
|
['deepwater-green'] = -5,
|
|
['water-mud'] = -6,
|
|
['water-shallow'] = -6
|
|
}
|
|
|
|
--- Some mods like to destroy the infini tree.
|
|
--- So we solve it by delaying the creation.
|
|
local delay_infini_tree_token =
|
|
Token.register(
|
|
function(event)
|
|
local surface = event.surface
|
|
local position = event.position
|
|
|
|
surface.create_entity({name = 'tree-0' .. math.random(1, 9), position = position})
|
|
end
|
|
)
|
|
|
|
local function reward_tokens(expanse, entity)
|
|
local chance = expanse.token_chance % 1
|
|
local count = math.floor(expanse.token_chance)
|
|
|
|
if chance > 0 then
|
|
chance = math.floor(chance * 1000)
|
|
if math.random(1, 1000) <= chance then
|
|
entity.surface.spill_item_stack(entity.position, {name = 'coin', count = 1}, true, nil, false)
|
|
end
|
|
end
|
|
if count > 0 then
|
|
for _ = 1, count, 1 do
|
|
entity.surface.spill_item_stack(entity.position, {name = 'coin', count = 1}, true, nil, false)
|
|
end
|
|
end
|
|
end
|
|
|
|
local function get_cell_value(expanse, left_top)
|
|
local square_size = expanse.square_size
|
|
local value = square_size ^ 2
|
|
value = value * 8
|
|
|
|
local source_surface = game.surfaces[expanse.source_surface]
|
|
local area = {{left_top.x, left_top.y}, {left_top.x + square_size, left_top.y + square_size}}
|
|
local entities = source_surface.find_entities(area)
|
|
local tiles = source_surface.find_tiles_filtered({area = area})
|
|
|
|
for _, tile in pairs(tiles) do
|
|
if price_modifiers[tile.name] then
|
|
value = value + price_modifiers[tile.name]
|
|
end
|
|
end
|
|
for _, entity in pairs(entities) do
|
|
if price_modifiers[entity.type] then
|
|
value = value + price_modifiers[entity.type]
|
|
end
|
|
end
|
|
|
|
local distance = math.sqrt(left_top.x ^ 2 + left_top.y ^ 2)
|
|
value = value * ((distance ^ 1.1) * expanse.price_distance_modifier)
|
|
local ore_modifier = distance * (expanse.price_distance_modifier / 20)
|
|
if ore_modifier > expanse.max_ore_price_modifier then
|
|
ore_modifier = expanse.max_ore_price_modifier
|
|
end
|
|
|
|
for _, entity in pairs(entities) do
|
|
if entity.type == 'resource' then
|
|
if entity.prototype.resource_category == 'basic-fluid' then
|
|
value = value + (entity.amount * ore_modifier * 0.01)
|
|
else
|
|
value = value + (entity.amount * ore_modifier)
|
|
end
|
|
end
|
|
end
|
|
|
|
value = math.floor(value)
|
|
if value < 16 then
|
|
value = 16
|
|
end
|
|
|
|
return value
|
|
end
|
|
|
|
local function get_left_top(expanse, position)
|
|
local vectors = {{-1, 0}, {1, 0}, {0, 1}, {0, -1}}
|
|
table.shuffle_table(vectors)
|
|
|
|
local surface = game.surfaces.expanse
|
|
|
|
for _, v in pairs(vectors) do
|
|
local tile = surface.get_tile({position.x + v[1], position.y + v[2]})
|
|
if tile.name == 'out-of-map' then
|
|
local left_top = tile.position
|
|
left_top.x = left_top.x - left_top.x % expanse.square_size
|
|
left_top.y = left_top.y - left_top.y % expanse.square_size
|
|
if not expanse.grid[tostring(left_top.x .. '_' .. left_top.y)] then
|
|
return left_top
|
|
end
|
|
end
|
|
end
|
|
|
|
return false
|
|
end
|
|
|
|
local function is_container_position_valid(expanse, position)
|
|
if game.tick == 0 then
|
|
return true
|
|
end
|
|
|
|
local left_top = get_left_top(expanse, position)
|
|
if not left_top then
|
|
return false
|
|
end
|
|
|
|
if
|
|
game.surfaces.expanse.count_entities_filtered(
|
|
{
|
|
name = 'logistic-chest-requester',
|
|
force = 'neutral',
|
|
area = {{left_top.x - 1, left_top.y - 1}, {left_top.x + expanse.square_size + 1, left_top.y + expanse.square_size + 1}}
|
|
}
|
|
) > 0
|
|
then
|
|
return false
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
local function create_costs_render(entity, name, offset)
|
|
local id = rendering.draw_sprite{
|
|
sprite = 'virtual-signal/signal-grey',
|
|
surface = entity.surface,
|
|
target = entity,
|
|
x_scale = 1.1,
|
|
y_scale = 1.1,
|
|
render_layer = '190',
|
|
target_offset = {offset, -1.5},
|
|
only_in_alt_mode = true
|
|
}
|
|
local id2 = rendering.draw_sprite{
|
|
sprite = 'item/' ..name,
|
|
surface = entity.surface,
|
|
target = entity,
|
|
x_scale = 0.75,
|
|
y_scale = 0.75,
|
|
render_layer = '191',
|
|
target_offset = {offset, -1.5},
|
|
only_in_alt_mode = true
|
|
}
|
|
return {id, id2}
|
|
end
|
|
|
|
local function remove_one_render(container, key)
|
|
if rendering.is_valid(container.price[key].render[1]) then
|
|
rendering.destroy(container.price[key].render[1])
|
|
end
|
|
if rendering.is_valid(container.price[key].render[2]) then
|
|
rendering.destroy(container.price[key].render[2])
|
|
end
|
|
end
|
|
|
|
local function remove_old_renders(container)
|
|
for key, _ in pairs(container.price) do
|
|
remove_one_render(container, key)
|
|
end
|
|
end
|
|
|
|
function Public.spawn_units(spawner)
|
|
local evolution = game.forces.enemy.evolution_factor
|
|
local position = spawner.position
|
|
for i = 1, 4 + math.floor(8 * evolution), 1 do
|
|
local biter_roll = BiterRaffle.roll('mixed', evolution)
|
|
local free_pos = spawner.surface.find_non_colliding_position(biter_roll, {x = position.x + math.random(-8,8), y = position.y + math.random(-8,8)}, 12, 0.05)
|
|
spawner.surface.create_entity({name = biter_roll, position = free_pos or position, force = 'enemy'})
|
|
end
|
|
end
|
|
|
|
function Public.get_item_tooltip(name)
|
|
return {'expanse.stats_item_tooltip', game.item_prototypes[name].localised_name, Price_raffle.get_item_worth(name)}
|
|
end
|
|
|
|
function Public.invasion_numbers()
|
|
local evo = game.forces.enemy.evolution_factor
|
|
return {candidates = 3 + math.floor(evo * 10), groups = 1 + math.floor(evo * 4)}
|
|
end
|
|
|
|
function Public.invasion_warn(event)
|
|
local seconds = (120 * 60 - event.delay) / 60
|
|
game.print({'expanse.biters_invasion_warning', seconds, event.size}, {r = 0.88, g = 0.22, b = 0.22})
|
|
end
|
|
|
|
function Public.invasion_detonate(event)
|
|
local surface = event.surface
|
|
local position = event.position
|
|
local entities_close = surface.find_entities_filtered{position = position, radius = 8}
|
|
for _, entity in pairs(entities_close) do
|
|
if entity.valid then
|
|
entity.die('enemy')
|
|
end
|
|
end
|
|
local entities_nearby = surface.find_entities_filtered{position = position, radius = 16}
|
|
for _, entity in pairs(entities_nearby) do
|
|
if entity.valid and entity.is_entity_with_health then
|
|
entity.damage(entity.prototype.max_health * 0.75, 'enemy')
|
|
end
|
|
end
|
|
surface.create_entity({name = 'nuke-explosion', position = position})
|
|
end
|
|
|
|
function Public.invasion_trigger(event)
|
|
local surface = event.surface
|
|
local position = event.position
|
|
local round = event.round
|
|
local evolution = game.forces.enemy.evolution_factor
|
|
local biters = {}
|
|
for i = 1, 5 + math.floor(30 * evolution) + round * 5, 1 do
|
|
local biter_roll = BiterRaffle.roll('mixed', evolution)
|
|
local free_pos = surface.find_non_colliding_position(biter_roll, {x = position.x + math.random(-8,8), y = position.y + math.random(-8,8)}, 12, 0.05)
|
|
biters[#biters + 1] = surface.create_entity({name = biter_roll, position = free_pos or position, force = 'enemy'})
|
|
end
|
|
local group = surface.create_unit_group{position = position, force = 'enemy'}
|
|
for _, biter in pairs(biters) do
|
|
group.add_member(biter)
|
|
end
|
|
group.set_command({type = defines.command.attack_area, destination = position, radius = 80, distraction = defines.distraction.by_anything})
|
|
group.start_moving()
|
|
local worm_roll = BiterRaffle.roll('worm', evolution)
|
|
for i = 1, 3 + math.floor(7 * evolution), 1 do
|
|
local worm_pos = surface.find_non_colliding_position(worm_roll, {x = position.x + math.random(-12,12), y = position.y + math.random(-12,12)}, 12, 0.1)
|
|
if worm_pos then
|
|
surface.create_entity({name = worm_roll, position = worm_pos, force = 'enemy'})
|
|
end
|
|
end
|
|
local nest = {'biter-spawner', 'biter-spawner', 'biter-spawner', 'spitter-spawner'}
|
|
local nest_roll = nest[math.random(1, 4)]
|
|
local nest_pos = surface.find_non_colliding_position(nest_roll, position, 12, 0.1)
|
|
if nest_pos then
|
|
surface.create_entity({name = nest_roll, position = nest_pos, force = 'enemy'})
|
|
end
|
|
end
|
|
|
|
local function schedule_detonation(expanse, surface, position)
|
|
table.insert(expanse.schedule, {tick = game.tick + 120 * 60, event = 'invasion_detonate', parameters = {surface = surface, position = position}})
|
|
end
|
|
|
|
local function schedule_warning(expanse, size, delay)
|
|
table.insert(expanse.schedule, {tick = game.tick + 2 * 60 + delay, event = 'invasion_warn', parameters = {size = size, delay = delay}})
|
|
end
|
|
|
|
local function schedule_biters(expanse, surface, position, delay, round)
|
|
table.insert(expanse.schedule, {tick = game.tick + delay + 120 * 60, event = 'invasion_trigger', parameters = {surface = surface, position = position, round = round}})
|
|
end
|
|
|
|
local function plan_invasion(expanse, invasion_numbers)
|
|
local candidates = expanse.invasion_candidates
|
|
table.shuffle_table(candidates)
|
|
schedule_warning(expanse, invasion_numbers.groups, 0)
|
|
schedule_warning(expanse, invasion_numbers.groups, 60 * 60)
|
|
schedule_warning(expanse, invasion_numbers.groups, 90 * 60)
|
|
local rounds = 4 + math.random(1, 8)
|
|
for i = 1, invasion_numbers.groups, 1 do
|
|
local surface = candidates[i].surface
|
|
local position = candidates[i].position
|
|
schedule_detonation(expanse, surface, position)
|
|
for ii = 1, rounds, 1 do
|
|
schedule_biters(expanse, surface, position, 120 + (ii - 1) * 300, ii)
|
|
end
|
|
rendering.set_time_to_live(candidates[i].render, 122 * 60 + rounds * 300)
|
|
end
|
|
for j = invasion_numbers.groups + 1, #candidates, 1 do
|
|
rendering.set_time_to_live(candidates[j].render, 122 * 60)
|
|
end
|
|
expanse.invasion_candidates = {}
|
|
end
|
|
|
|
function Public.check_invasion(expanse)
|
|
local invasion_numbers = Public.invasion_numbers()
|
|
if #expanse.invasion_candidates >= invasion_numbers.candidates then
|
|
plan_invasion(expanse, invasion_numbers)
|
|
end
|
|
end
|
|
|
|
function Public.expand(expanse, left_top)
|
|
expanse.grid[tostring(left_top.x .. '_' .. left_top.y)] = true
|
|
|
|
local source_surface = game.surfaces[expanse.source_surface]
|
|
if not source_surface then
|
|
return
|
|
end
|
|
source_surface.request_to_generate_chunks(left_top, 3)
|
|
source_surface.force_generate_chunk_requests()
|
|
|
|
local square_size = expanse.square_size
|
|
local area = {{left_top.x, left_top.y}, {left_top.x + square_size, left_top.y + square_size}}
|
|
local surface = game.surfaces.expanse
|
|
|
|
source_surface.clone_area(
|
|
{
|
|
source_area = area,
|
|
destination_area = area,
|
|
destination_surface = surface,
|
|
clone_tiles = true,
|
|
clone_entities = true,
|
|
clone_decoratives = true,
|
|
clear_destination_entities = false,
|
|
clear_destination_decoratives = true,
|
|
expand_map = true
|
|
}
|
|
)
|
|
|
|
local positions = {
|
|
{x = left_top.x + math.random(1, square_size - 2), y = left_top.y},
|
|
{x = left_top.x, y = left_top.y + math.random(1, square_size - 2)},
|
|
{x = left_top.x + math.random(1, square_size - 2), y = left_top.y + (square_size - 1)},
|
|
{x = left_top.x + (square_size - 1), y = left_top.y + math.random(1, square_size - 2)}
|
|
}
|
|
|
|
for _, position in pairs(positions) do
|
|
if is_container_position_valid(expanse, position) then
|
|
local e = surface.create_entity({name = 'logistic-chest-requester', position = position, force = 'neutral'})
|
|
e.destructible = false
|
|
e.minable = false
|
|
end
|
|
end
|
|
|
|
if game.tick == 0 then
|
|
local a = math.floor(expanse.square_size * 0.5)
|
|
for x = 1, 3, 1 do
|
|
for y = 1, 3, 1 do
|
|
surface.set_tiles({{name = 'water', position = {a + x + 2, a + y + 2}}}, true)
|
|
end
|
|
end
|
|
surface.create_entity({name = 'crude-oil', position = {a - 4, a - 4}, amount = 1500000})
|
|
Task.set_timeout_in_ticks(30, delay_infini_tree_token, {surface = surface, position = {a - 4, a + 4}})
|
|
surface.create_entity({name = 'rock-big', position = {a + 4, a - 4}})
|
|
surface.spill_item_stack({a, a + 2}, {name = 'coin', count = 1}, false, nil, false)
|
|
surface.spill_item_stack({a + 0.5, a + 2.5}, {name = 'coin', count = 1}, false, nil, false)
|
|
surface.spill_item_stack({a - 0.5, a + 2.5}, {name = 'coin', count = 1}, false, nil, false)
|
|
|
|
for x = 0, square_size, 1 do
|
|
for y = 0, square_size, 1 do
|
|
if surface.can_place_entity({name = 'wooden-chest', position = {x, y}}) and surface.can_place_entity({name = 'coal', position = {x, y}, amount = 1}) then
|
|
surface.create_entity({name = ores[(x + y) % 4 + 1], position = {x, y}, amount = 1500})
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local function init_container(expanse, entity, budget)
|
|
local left_top = get_left_top(expanse, entity.position)
|
|
if not left_top then
|
|
return
|
|
end
|
|
local cell_value = budget or get_cell_value(expanse, left_top)
|
|
local item_stacks = {}
|
|
local roll_count = 3
|
|
for _ = 1, roll_count, 1 do
|
|
for _, stack in pairs(Price_raffle.roll(math.floor(cell_value / roll_count), 3, nil, math.max(4, cell_value / (roll_count * 6)))) do
|
|
if not item_stacks[stack.name] then
|
|
item_stacks[stack.name] = stack.count
|
|
else
|
|
item_stacks[stack.name] = item_stacks[stack.name] + stack.count
|
|
end
|
|
end
|
|
end
|
|
|
|
local price = {}
|
|
local offset = -3
|
|
for k, v in pairs(item_stacks) do
|
|
table.insert(price, {name = k, count = v, render = create_costs_render(entity, k, offset)})
|
|
offset = offset + 1
|
|
end
|
|
|
|
local containers = expanse.containers
|
|
containers[entity.unit_number] = {entity = entity, left_top = left_top, price = price}
|
|
end
|
|
|
|
local function get_remaining_budget(container)
|
|
local budget = 0
|
|
for _, item_stack in pairs(container.price) do
|
|
budget = budget + (item_stack.count * Price_raffle.get_item_worth(item_stack.name))
|
|
end
|
|
return budget
|
|
end
|
|
|
|
function Public.set_container(expanse, entity)
|
|
if entity.name ~= 'logistic-chest-requester' then
|
|
return
|
|
end
|
|
if not expanse.containers[entity.unit_number] then
|
|
init_container(expanse, entity)
|
|
end
|
|
|
|
local container = expanse.containers[entity.unit_number]
|
|
if not container or not container.entity or not container.entity.valid then
|
|
expanse.containers[entity.unit_number] = nil
|
|
return
|
|
end
|
|
|
|
local inventory = container.entity.get_inventory(defines.inventory.chest)
|
|
|
|
if not inventory.is_empty() then
|
|
local contents = inventory.get_contents()
|
|
if contents['coin'] then
|
|
local count_removed = inventory.remove({name = 'coin', count = 1})
|
|
if count_removed > 0 then
|
|
expanse.cost_stats['coin'] = (expanse.cost_stats['coin'] or 0) + count_removed
|
|
script.raise_event(expanse.events.gui_update, {item = 'coin'})
|
|
remove_old_renders(container)
|
|
init_container(expanse, entity, get_remaining_budget(container))
|
|
container = expanse.containers[entity.unit_number]
|
|
game.print({'expanse.chest_reset', {'expanse.gps', math.floor(entity.position.x), math.floor(entity.position.y), 'expanse'}})
|
|
end
|
|
end
|
|
if contents['infinity-chest'] then
|
|
remove_old_renders(container)
|
|
container.price = {}
|
|
end
|
|
end
|
|
|
|
for key, item_stack in pairs(container.price) do
|
|
local name = item_stack.name
|
|
local count_removed = inventory.remove({name = name, count = item_stack.count})
|
|
container.price[key].count = container.price[key].count - count_removed
|
|
expanse.cost_stats[name] = (expanse.cost_stats[name] or 0) + count_removed
|
|
script.raise_event(expanse.events.gui_update, {item = name})
|
|
if container.price[key].count <= 0 then
|
|
remove_one_render(container, key)
|
|
table.remove(container.price, key)
|
|
end
|
|
end
|
|
|
|
if #container.price == 0 then
|
|
Public.expand(expanse, container.left_top)
|
|
local a = math.floor(expanse.square_size * 0.5)
|
|
local expansion_position = {x = expanse.containers[entity.unit_number].left_top.x + a, y = expanse.containers[entity.unit_number].left_top.y + a}
|
|
expanse.containers[entity.unit_number] = nil
|
|
if not inventory.is_empty() then
|
|
for name, count in pairs(inventory.get_contents()) do
|
|
entity.surface.spill_item_stack(entity.position, {name = name, count = count}, true, nil, false)
|
|
end
|
|
end
|
|
reward_tokens(expanse, entity)
|
|
entity.destructible = true
|
|
entity.die()
|
|
return expansion_position
|
|
end
|
|
|
|
for slot = 1, 30, 1 do
|
|
entity.clear_request_slot(slot)
|
|
end
|
|
|
|
for slot, item_stack in pairs(container.price) do
|
|
container.entity.set_request_slot(item_stack, slot)
|
|
end
|
|
end
|
|
|
|
return Public
|