mirror of
https://github.com/Refactorio/RedMew.git
synced 2024-12-14 10:13:13 +02:00
e86cc2a8f5
Airstrike command
1884 lines
49 KiB
Lua
1884 lines
49 KiB
Lua
--local Random = require 'map_gen.shared.random'
|
|
local Token = require 'utils.token'
|
|
local Global = require 'utils.global'
|
|
local Event = require 'utils.event'
|
|
local Task = require 'utils.task'
|
|
local Retailer = require 'features.retailer'
|
|
local Donator = require 'features.donator'
|
|
local RS = require 'map_gen.shared.redmew_surface'
|
|
local Server = require 'features.server'
|
|
local CrashSiteToast = require 'map_gen.maps.crash_site.crash_site_toast'
|
|
local ScoreTracker = require 'utils.score_tracker'
|
|
local change_for_player = ScoreTracker.change_for_player
|
|
local coins_earned_name = 'coins-earned'
|
|
|
|
local table = require 'utils.table'
|
|
--local next = next
|
|
local pairs = pairs
|
|
local concat = table.concat
|
|
local floor = math.floor
|
|
local format = string.format
|
|
local tostring = tostring
|
|
local draw_text = rendering.draw_text
|
|
local render_mode_game = defines.render_mode.game
|
|
|
|
local b = require 'map_gen.shared.builders'
|
|
|
|
local direction_bit_mask = 0xc0000000
|
|
local section_bit_mask = 0x30000000
|
|
local level_bit_mask = 0x0fffffff
|
|
--local not_level_bit_mask = 0xf0000000
|
|
local direction_bit_shift = 30
|
|
local section_bit_shift = 28
|
|
|
|
--local section_straight = 0
|
|
--local section_outer_corner = 1
|
|
--local section_inner_corner = 2
|
|
|
|
local wall_north_straight = 0x00000001
|
|
local wall_east_straight = 0x40000001
|
|
local wall_south_straight = 0x80000001
|
|
local wall_west_straight = 0xc0000001
|
|
local wall_north_outer = 0x10000001
|
|
local wall_east_outer = 0x50000001
|
|
local wall_south_outer = 0x90000001
|
|
local wall_west_outer = 0xd0000001
|
|
local wall_north_inner = 0x20000001
|
|
local wall_east_inner = 0x60000001
|
|
local wall_south_inner = 0xa0000001
|
|
local wall_west_inner = 0xe0000001
|
|
|
|
local default_part_size = 6
|
|
|
|
local magic_crafters_per_tick = 3
|
|
local magic_fluid_crafters_per_tick = 8
|
|
|
|
local refill_turrets = {index = 1}
|
|
local power_sources = {}
|
|
local turret_to_outpost = {}
|
|
local magic_crafters = {index = 1}
|
|
local magic_fluid_crafters = {index = 1}
|
|
local outposts = {}
|
|
local artillery_outposts = {index = 1}
|
|
local outpost_count = 0
|
|
|
|
Global.register(
|
|
{
|
|
refil_turrets = refill_turrets,
|
|
power_sources = power_sources,
|
|
turret_to_outpost = turret_to_outpost,
|
|
magic_crafters = magic_crafters,
|
|
magic_fluid_crafters = magic_fluid_crafters,
|
|
outposts = outposts,
|
|
artillery_outposts = artillery_outposts
|
|
},
|
|
function(tbl)
|
|
refill_turrets = tbl.refil_turrets
|
|
power_sources = tbl.power_sources
|
|
turret_to_outpost = tbl.turret_to_outpost
|
|
magic_crafters = tbl.magic_crafters
|
|
magic_fluid_crafters = tbl.magic_fluid_crafters
|
|
outposts = tbl.outposts
|
|
artillery_outposts = tbl.artillery_outposts
|
|
end
|
|
)
|
|
|
|
--[[ local function get_direction(part)
|
|
local dir = bit32.band(part, direction_bit_mask)
|
|
return bit32.rshift(dir, direction_bit_shift - 1)
|
|
end ]]
|
|
local function get_4_way_direction(part)
|
|
local dir = bit32.band(part, direction_bit_mask)
|
|
return bit32.rshift(dir, direction_bit_shift)
|
|
end
|
|
|
|
local function get_section(part)
|
|
local sec = bit32.band(part, section_bit_mask)
|
|
return bit32.rshift(sec, section_bit_shift)
|
|
end
|
|
|
|
local function get_level(part)
|
|
return bit32.band(part, level_bit_mask)
|
|
end
|
|
|
|
--[[ local function set_level(part, level)
|
|
local not_level = bit32.band(part)
|
|
return not_level + level
|
|
end
|
|
|
|
local function set_block(tbl, x, y, value)
|
|
tbl[(y - 1) * tbl.size + x] = value
|
|
end
|
|
|
|
local function get_block(tbl, x, y)
|
|
local size = tbl.size
|
|
if x < 1 or x > size or y < 1 or y > size then
|
|
return 0
|
|
end
|
|
return tbl[(y - 1) * size + x] or 0
|
|
end ]]
|
|
local function fast_remove(tbl, index)
|
|
local count = #tbl
|
|
if index > count then
|
|
return
|
|
elseif index < count then
|
|
tbl[index] = tbl[count]
|
|
end
|
|
|
|
tbl[count] = nil
|
|
end
|
|
|
|
local Public = {}
|
|
Public.__index = Public
|
|
|
|
Public.empty_template = {}
|
|
|
|
function Public.new(random)
|
|
local obj = {random = random}
|
|
|
|
return setmetatable(obj, Public)
|
|
end
|
|
|
|
local function do_walls(self, blocks, outpost_variance, outpost_min_step)
|
|
local size = blocks.size
|
|
local random = self.random
|
|
|
|
local max_variance = size - outpost_variance + 1
|
|
local variance_step = outpost_variance + outpost_min_step
|
|
|
|
local x = random:next_int(1, outpost_variance)
|
|
local y = random:next_int(1, outpost_variance)
|
|
|
|
local start_x, start_y = x, y
|
|
|
|
local i = (y - 1) * size + x
|
|
|
|
local pv = -1
|
|
|
|
-- top
|
|
while x < size do
|
|
local tx = x + random:next_int(outpost_min_step, variance_step)
|
|
tx = math.min(tx, size)
|
|
|
|
if pv == 0 then
|
|
blocks[i] = wall_north_straight
|
|
elseif pv == -1 then
|
|
blocks[i] = wall_north_outer
|
|
else
|
|
blocks[i] = wall_east_inner
|
|
end
|
|
|
|
x = x + 1
|
|
i = i + 1
|
|
|
|
while x < tx do
|
|
blocks[i] = wall_north_straight
|
|
x = x + 1
|
|
i = i + 1
|
|
end
|
|
|
|
if x < size - outpost_min_step then
|
|
local ty = random:next_int(1, outpost_variance)
|
|
|
|
if y == ty then
|
|
pv = 0
|
|
elseif y < ty then
|
|
pv = 1
|
|
blocks[i] = wall_east_outer
|
|
y = y + 1
|
|
i = i + size
|
|
while y < ty do
|
|
blocks[i] = wall_east_straight
|
|
y = y + 1
|
|
i = i + size
|
|
end
|
|
else
|
|
pv = -1
|
|
blocks[i] = wall_north_inner
|
|
y = y - 1
|
|
i = i - size
|
|
while y > ty do
|
|
blocks[i] = wall_west_straight
|
|
y = y - 1
|
|
i = i - size
|
|
end
|
|
end
|
|
else
|
|
pv = 0
|
|
end
|
|
end
|
|
|
|
pv = 1
|
|
-- right
|
|
while y < size do
|
|
local ty = y + random:next_int(outpost_min_step, variance_step)
|
|
ty = math.min(ty, size)
|
|
|
|
if pv == 0 then
|
|
blocks[i] = wall_east_straight
|
|
elseif pv == -1 then
|
|
blocks[i] = wall_south_inner
|
|
else
|
|
blocks[i] = wall_east_outer
|
|
end
|
|
|
|
y = y + 1
|
|
i = i + size
|
|
|
|
while y < ty do
|
|
blocks[i] = wall_east_straight
|
|
y = y + 1
|
|
i = i + size
|
|
end
|
|
|
|
if y < size - outpost_min_step then
|
|
local tx = random:next_int(max_variance, size)
|
|
|
|
if x == tx then
|
|
pv = 0
|
|
elseif x < tx then
|
|
pv = 1
|
|
blocks[i] = wall_east_inner
|
|
x = x + 1
|
|
i = i + 1
|
|
while x < tx do
|
|
blocks[i] = wall_north_straight
|
|
x = x + 1
|
|
i = i + 1
|
|
end
|
|
else
|
|
pv = -1
|
|
blocks[i] = wall_south_outer
|
|
x = x - 1
|
|
i = i - 1
|
|
while x > tx do
|
|
blocks[i] = wall_south_straight
|
|
x = x - 1
|
|
i = i - 1
|
|
end
|
|
end
|
|
else
|
|
pv = 0
|
|
end
|
|
end
|
|
|
|
pv = 1
|
|
|
|
-- bottom
|
|
while x > 1 do
|
|
local tx = x - random:next_int(outpost_min_step, variance_step)
|
|
tx = math.max(tx, 1)
|
|
|
|
if pv == 0 then
|
|
blocks[i] = wall_south_straight
|
|
elseif pv == -1 then
|
|
blocks[i] = wall_west_inner
|
|
else
|
|
blocks[i] = wall_south_outer
|
|
end
|
|
|
|
x = x - 1
|
|
i = i - 1
|
|
|
|
while x > tx do
|
|
blocks[i] = wall_south_straight
|
|
x = x - 1
|
|
i = i - 1
|
|
end
|
|
|
|
if x > outpost_min_step + 1 then
|
|
local ty = random:next_int(max_variance, size)
|
|
|
|
if y == ty then
|
|
pv = 0
|
|
elseif y < ty then
|
|
pv = 1
|
|
blocks[i] = wall_south_inner
|
|
y = y + 1
|
|
i = i + size
|
|
while y < ty do
|
|
blocks[i] = wall_east_straight
|
|
y = y + 1
|
|
i = i + size
|
|
end
|
|
else
|
|
pv = -1
|
|
blocks[i] = wall_west_outer
|
|
y = y - 1
|
|
i = i - size
|
|
while y > ty do
|
|
blocks[i] = wall_west_straight
|
|
y = y - 1
|
|
i = i - size
|
|
end
|
|
end
|
|
else
|
|
pv = 0
|
|
end
|
|
end
|
|
|
|
pv = -1
|
|
-- left
|
|
local bottom_left_y = y
|
|
while y > start_y + variance_step do
|
|
local ty = y - random:next_int(outpost_min_step, variance_step)
|
|
ty = math.max(ty, start_y)
|
|
|
|
if pv == 0 then
|
|
blocks[i] = wall_west_straight
|
|
elseif pv == -1 then
|
|
blocks[i] = wall_west_outer
|
|
else
|
|
blocks[i] = wall_north_inner
|
|
end
|
|
|
|
y = y - 1
|
|
i = i - size
|
|
|
|
while y > ty do
|
|
blocks[i] = wall_west_straight
|
|
y = y - 1
|
|
i = i - size
|
|
end
|
|
|
|
if y > start_y + variance_step + outpost_min_step then
|
|
local tx = random:next_int(1, outpost_variance)
|
|
|
|
if x == tx then
|
|
pv = 0
|
|
elseif x < tx then
|
|
pv = 1
|
|
--blocks[i] = wall_west_outer
|
|
blocks[i] = wall_north_outer
|
|
x = x + 1
|
|
i = i + 1
|
|
while x < tx do
|
|
blocks[i] = wall_north_straight
|
|
x = x + 1
|
|
i = i + 1
|
|
end
|
|
else
|
|
pv = -1
|
|
blocks[i] = wall_west_inner
|
|
x = x - 1
|
|
i = i - 1
|
|
while x > tx do
|
|
blocks[i] = wall_south_straight
|
|
x = x - 1
|
|
i = i - 1
|
|
end
|
|
end
|
|
else
|
|
pv = 0
|
|
end
|
|
end
|
|
|
|
-- final connection
|
|
if y == bottom_left_y then
|
|
blocks[i] = wall_west_outer
|
|
|
|
y = y - 1
|
|
i = i - size
|
|
|
|
while y > bottom_left_y - outpost_min_step do
|
|
blocks[i] = wall_west_straight
|
|
y = y - 1
|
|
i = i - size
|
|
end
|
|
end
|
|
|
|
if x == start_x then
|
|
pv = 0
|
|
elseif x < start_x then
|
|
pv = 1
|
|
blocks[i] = wall_north_outer
|
|
x = x + 1
|
|
i = i + 1
|
|
while x < start_x do
|
|
blocks[i] = wall_north_straight
|
|
x = x + 1
|
|
i = i + 1
|
|
end
|
|
else
|
|
pv = -1
|
|
blocks[i] = wall_west_inner
|
|
x = x - 1
|
|
i = i - 1
|
|
while x > start_x do
|
|
blocks[i] = wall_south_straight
|
|
x = x - 1
|
|
i = i - 1
|
|
end
|
|
end
|
|
|
|
if pv == 0 then
|
|
blocks[i] = wall_west_straight
|
|
elseif pv == -1 then
|
|
blocks[i] = wall_west_outer
|
|
else
|
|
blocks[i] = wall_north_inner
|
|
end
|
|
|
|
y = y - 1
|
|
i = i - size
|
|
|
|
while y > start_y do
|
|
blocks[i] = wall_west_straight
|
|
y = y - 1
|
|
i = i - size
|
|
end
|
|
end
|
|
|
|
local function fill(blocks)
|
|
local size = blocks.size
|
|
local anti_set = {size = size}
|
|
local anti_stack = {}
|
|
|
|
local y_offset = (size - 1) * size
|
|
for x = 1, size do
|
|
if blocks[x] == nil then
|
|
anti_stack[#anti_stack + 1] = {x = x, y = 1}
|
|
end
|
|
|
|
if blocks[x + y_offset] == nil then
|
|
anti_stack[#anti_stack + 1] = {x = x, y = size}
|
|
end
|
|
end
|
|
|
|
for y = 2, size do
|
|
y_offset = (y - 1) * size
|
|
if blocks[y_offset + 1] == nil then
|
|
anti_stack[#anti_stack + 1] = {x = 1, y = y}
|
|
end
|
|
|
|
if blocks[y_offset + size] == nil then
|
|
anti_stack[#anti_stack + 1] = {x = size, y = y}
|
|
end
|
|
end
|
|
|
|
while #anti_stack > 0 do
|
|
local point = table.remove(anti_stack)
|
|
local x, y = point.x, point.y
|
|
|
|
local offset = (y - 1) * size + x
|
|
|
|
anti_set[offset] = true
|
|
|
|
if x > 1 then
|
|
local x2 = x - 1
|
|
local offset2 = offset - 1
|
|
|
|
if not anti_set[offset2] and not blocks[offset2] then
|
|
anti_stack[#anti_stack + 1] = {x = x2, y = y}
|
|
end
|
|
end
|
|
if x < size then
|
|
local x2 = x + 1
|
|
local offset2 = offset + 1
|
|
|
|
if not anti_set[offset2] and not blocks[offset2] then
|
|
anti_stack[#anti_stack + 1] = {x = x2, y = y}
|
|
end
|
|
end
|
|
if y > 1 then
|
|
local y2 = y - 1
|
|
local offset2 = offset - size
|
|
|
|
if not anti_set[offset2] and not blocks[offset2] then
|
|
anti_stack[#anti_stack + 1] = {x = x, y = y2}
|
|
end
|
|
end
|
|
if y < size then
|
|
local y2 = y + 1
|
|
local offset2 = offset + size
|
|
|
|
if not anti_set[offset2] and not blocks[offset2] then
|
|
anti_stack[#anti_stack + 1] = {x = x, y = y2}
|
|
end
|
|
end
|
|
end
|
|
|
|
for y = 1, size do
|
|
local offset = (y - 1) * size
|
|
for x = 1, size do
|
|
local i = offset + x
|
|
if not anti_set[i] and not blocks[i] then
|
|
blocks[i] = 2
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local function do_levels(blocks, max_level)
|
|
local size = blocks.size
|
|
local level = 2
|
|
|
|
while level < max_level do
|
|
local next_level = level + 1
|
|
|
|
for y = 1, size do
|
|
local offset = (y - 1) * size
|
|
for x = 1, size do
|
|
local i = offset + x
|
|
|
|
if get_level(blocks[i] or 0) >= level then
|
|
local count = 0
|
|
|
|
if x > 1 and get_level(blocks[i - 1] or 0) >= level then
|
|
count = count + 1
|
|
end
|
|
if x < size and get_level(blocks[i + 1] or 0) >= level then
|
|
count = count + 1
|
|
end
|
|
if y > 1 and get_level(blocks[i - size] or 0) >= level then
|
|
count = count + 1
|
|
end
|
|
if y < size and get_level(blocks[i + size] or 0) >= level then
|
|
count = count + 1
|
|
end
|
|
|
|
if count == 4 then
|
|
blocks[i] = next_level
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
level = level + 1
|
|
end
|
|
|
|
local levels = {}
|
|
blocks.levels = levels
|
|
|
|
for i = 1, max_level do
|
|
levels[i] = {}
|
|
end
|
|
|
|
for y = 1, size do
|
|
local offset = (y - 1) * size
|
|
for x = 1, size do
|
|
local i = offset + x
|
|
local block = blocks[i]
|
|
if block then
|
|
local l = get_level(block)
|
|
|
|
local lvl = levels[l]
|
|
lvl[#lvl + 1] = i
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local function get_template(random, templates, templates_count, counts)
|
|
local template
|
|
if templates_count == 0 then
|
|
return nil
|
|
elseif templates_count == 1 then
|
|
template = templates[1]
|
|
else
|
|
local ti = random:next_int(1, templates_count)
|
|
template = templates[ti]
|
|
end
|
|
|
|
if template == Public.empty_template then
|
|
return nil
|
|
end
|
|
|
|
local count = counts[template] or 0
|
|
local max_count = template.max_count
|
|
|
|
while count == max_count do
|
|
template = template.fallback
|
|
if template == nil then
|
|
return nil
|
|
end
|
|
count = counts[template] or 0
|
|
max_count = template.max_count
|
|
end
|
|
|
|
counts[template] = count + 1
|
|
|
|
return template
|
|
end
|
|
|
|
local function make_blocks(self, blocks, template)
|
|
local random = self.random
|
|
local counts = {}
|
|
|
|
local levels = blocks.levels
|
|
local wall_level = levels[1]
|
|
|
|
local walls = template.walls
|
|
local wall_template_count = #walls
|
|
|
|
while #wall_level > 0 do
|
|
local index = random:next_int(1, #wall_level)
|
|
local i = wall_level[index]
|
|
|
|
fast_remove(wall_level, index)
|
|
|
|
local block = get_template(random, walls, wall_template_count, counts)
|
|
|
|
if block == Public.empty_template then
|
|
blocks[i] = nil
|
|
else
|
|
local block_data = blocks[i]
|
|
|
|
local section = get_section(block_data)
|
|
local dir = get_4_way_direction(block_data)
|
|
|
|
local new_block = block[section + 1][dir + 1]
|
|
blocks[i] = new_block
|
|
end
|
|
end
|
|
|
|
local bases = template.bases
|
|
|
|
for l = 2, #levels do
|
|
local level = levels[l]
|
|
local base_templates = bases[l - 1]
|
|
|
|
if base_templates then
|
|
local base_template_count = #base_templates
|
|
|
|
while #level > 0 do
|
|
local index = random:next_int(1, #level)
|
|
local i = level[index]
|
|
|
|
fast_remove(level, index)
|
|
|
|
blocks[i] = get_template(random, base_templates, base_template_count, counts)
|
|
end
|
|
else
|
|
for index = 1, #level do
|
|
local i = level[index]
|
|
blocks[i] = nil
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local remove_entity_types = {'tree', 'simple-entity'}
|
|
|
|
local function to_shape(blocks, part_size, on_init)
|
|
part_size = part_size or default_part_size
|
|
local inv_part_size = 1 / part_size
|
|
local size = blocks.size
|
|
local t_size = size * part_size
|
|
local half_t_size = t_size * 0.5
|
|
|
|
local outpost_id = outpost_count + 1
|
|
outpost_count = outpost_id
|
|
|
|
if on_init then
|
|
outposts[outpost_id] = {
|
|
outpost_id = outpost_id,
|
|
magic_crafters = {},
|
|
--magic_fluid_crafters = {},
|
|
market = nil,
|
|
turret_count = 0,
|
|
top_left = {nil, nil},
|
|
bottom_right = {nil, nil},
|
|
level = 1,
|
|
upgrade_rate = nil,
|
|
upgrade_base_cost = nil,
|
|
upgrade_cost_base = nil,
|
|
artillery_area = nil,
|
|
artillery_turrets = nil,
|
|
last_fire_tick = nil
|
|
}
|
|
end
|
|
|
|
local function shape(x, y, world)
|
|
x, y = floor(x + half_t_size), floor(y + half_t_size)
|
|
if x < 0 or y < 0 or x >= t_size or y >= t_size then
|
|
return false
|
|
end
|
|
|
|
local x2, y2 = floor(x * inv_part_size), floor(y * inv_part_size)
|
|
|
|
local template = blocks[y2 * size + x2 + 1]
|
|
if not template then
|
|
return false
|
|
end
|
|
|
|
local wx, wy = world.x, world.y
|
|
|
|
local entities =
|
|
world.surface.find_entities_filtered {
|
|
area = {{wx, wy}, {wx + 1, wy + 1}},
|
|
type = remove_entity_types
|
|
}
|
|
for i = 1, #entities do
|
|
local e = entities[i]
|
|
e.destroy()
|
|
end
|
|
|
|
local x3, y3 = x - x2 * part_size, y - y2 * part_size
|
|
|
|
local i = y3 * part_size + x3 + 1
|
|
|
|
local entry = template[i]
|
|
if not entry then
|
|
return false
|
|
end
|
|
|
|
local entity = entry.entity
|
|
local tile = entry.tile or true
|
|
if entity then
|
|
local data
|
|
local callback = entity.callback
|
|
if callback then
|
|
data = {outpost_id = outpost_id}
|
|
local cd = template[callback]
|
|
|
|
callback = cd.callback
|
|
data.callback_data = cd.data
|
|
end
|
|
|
|
return {
|
|
tile = tile,
|
|
entities = {
|
|
{
|
|
name = entity.name,
|
|
direction = entity.direction,
|
|
force = entity.force or template.force,
|
|
callback = callback,
|
|
data = data,
|
|
always_place = true
|
|
}
|
|
}
|
|
}
|
|
end
|
|
return tile
|
|
end
|
|
|
|
return b.change_map_gen_collision_hidden_tile(shape, 'water-tile', 'grass-1')
|
|
end
|
|
|
|
Public.to_shape = to_shape
|
|
|
|
function Public:do_outpost(template, on_init)
|
|
local settings = template.settings
|
|
local blocks = {size = settings.blocks}
|
|
|
|
do_walls(self, blocks, settings.variance, settings.min_step)
|
|
fill(blocks)
|
|
do_levels(blocks, settings.max_level)
|
|
make_blocks(self, blocks, template)
|
|
|
|
return to_shape(blocks, settings.part_size, on_init)
|
|
end
|
|
|
|
local function change_direction(entity, new_dir)
|
|
local copy = {}
|
|
|
|
for k, v in pairs(entity) do
|
|
copy[k] = v
|
|
end
|
|
copy.direction = new_dir
|
|
|
|
return copy
|
|
end
|
|
|
|
function Public.make_1_way(data)
|
|
data.__index = data
|
|
return data
|
|
end
|
|
|
|
local function set_tile(tbl, index, tile)
|
|
local entry = tbl[index]
|
|
if entry then
|
|
entry.tile = tile
|
|
else
|
|
tbl[index] = {tile = tile}
|
|
end
|
|
end
|
|
|
|
local function set_entity(tbl, index, entity)
|
|
local entry = tbl[index]
|
|
|
|
if entry then
|
|
entry.entity = entity
|
|
else
|
|
tbl[index] = {entity = entity}
|
|
end
|
|
end
|
|
|
|
function Public.make_4_way(data)
|
|
local part_size = data.part_size or default_part_size
|
|
local inv_part_size = 1 / part_size
|
|
|
|
local props = {}
|
|
|
|
local north = {}
|
|
local east = {}
|
|
local south = {}
|
|
local west = {}
|
|
local res = {north, east, south, west}
|
|
|
|
for i, entry in pairs(data) do
|
|
if type(i) == 'string' then
|
|
props[i] = entry
|
|
else
|
|
local y = math.ceil(i * inv_part_size)
|
|
local x = i - (y - 1) * part_size
|
|
|
|
local x2 = part_size - y + 1
|
|
local y2 = x
|
|
local x3 = part_size - x + 1
|
|
local y3 = part_size - y + 1
|
|
local x4 = y
|
|
local y4 = part_size - x + 1
|
|
|
|
local i2 = (y2 - 1) * part_size + x2
|
|
local i3 = (y3 - 1) * part_size + x3
|
|
local i4 = (y4 - 1) * part_size + x4
|
|
|
|
local tile = entry.tile
|
|
if tile then
|
|
set_tile(north, i, tile)
|
|
set_tile(east, i2, tile)
|
|
set_tile(south, i3, tile)
|
|
set_tile(west, i4, tile)
|
|
end
|
|
|
|
local entity = entry.entity
|
|
|
|
if entity then
|
|
local offset = entity.offset
|
|
|
|
if offset == 3 then
|
|
i = i + part_size + 1
|
|
i2 = i2 + part_size
|
|
i4 = i4 + 1
|
|
elseif offset == 1 then
|
|
i = i + 1
|
|
i2 = i2 + part_size
|
|
elseif offset == 2 then
|
|
i = i + part_size
|
|
i4 = i4 + 1
|
|
end
|
|
|
|
local dir = entity.direction or 0
|
|
|
|
set_entity(north, i, entity)
|
|
set_entity(east, i2, change_direction(entity, (dir + 2) % 8))
|
|
set_entity(south, i3, change_direction(entity, (dir + 4) % 8))
|
|
set_entity(west, i4, change_direction(entity, (dir + 6) % 8))
|
|
end
|
|
end
|
|
end
|
|
|
|
north.__index = north
|
|
east.__index = east
|
|
south.__index = south
|
|
west.__index = west
|
|
|
|
for k, v in pairs(props) do
|
|
north[k] = v
|
|
east[k] = v
|
|
south[k] = v
|
|
west[k] = v
|
|
end
|
|
|
|
return res
|
|
end
|
|
|
|
function Public.make_walls(data)
|
|
data.__index = data
|
|
return data
|
|
end
|
|
|
|
local function shallow_copy(tbl)
|
|
local copy = {}
|
|
for k, v in pairs(tbl) do
|
|
copy[k] = v
|
|
end
|
|
return copy
|
|
end
|
|
|
|
function Public.extend_1_way(data, tbl)
|
|
return setmetatable(shallow_copy(tbl), data)
|
|
end
|
|
|
|
function Public.extend_4_way(data, tbl)
|
|
return {
|
|
setmetatable(shallow_copy(tbl), data[1]),
|
|
setmetatable(shallow_copy(tbl), data[2]),
|
|
setmetatable(shallow_copy(tbl), data[3]),
|
|
setmetatable(shallow_copy(tbl), data[4])
|
|
}
|
|
end
|
|
|
|
function Public.extend_walls(data, tbl)
|
|
local copy = shallow_copy(tbl)
|
|
|
|
local base = {
|
|
Public.extend_4_way(data[1], copy),
|
|
Public.extend_4_way(data[2], copy),
|
|
Public.extend_4_way(data[3], copy)
|
|
}
|
|
base.__index = base
|
|
|
|
return setmetatable(copy, base)
|
|
end
|
|
|
|
local function update_market_upgrade_description(outpost_data)
|
|
local outpost_id = outpost_data.outpost_id
|
|
|
|
local upgrade_rate = outpost_data.upgrade_rate
|
|
local upgrade_base_cost = outpost_data.upgrade_base_cost
|
|
local upgrade_cost_base = outpost_data.upgrade_cost_base
|
|
local level = outpost_data.level
|
|
local base_outputs = outpost_data.base_outputs
|
|
local outpost_magic_crafters = outpost_data.magic_crafters
|
|
local prototype = Retailer.get_items(outpost_id)['upgrade']
|
|
|
|
prototype.price = upgrade_base_cost * #outpost_magic_crafters * upgrade_cost_base ^ (level - 1)
|
|
prototype.name_label = 'Upgrade Outpost to level ' .. tostring(level + 1)
|
|
|
|
local tooltip_str = {''}
|
|
local mapview_str = {''}
|
|
local count = 2
|
|
for k, v in pairs(base_outputs) do
|
|
local base_rate = v * 60
|
|
local upgrade_per_level = base_rate * upgrade_rate
|
|
local current_rate = base_rate + (level - 1) * upgrade_per_level
|
|
local next_rate = current_rate + upgrade_per_level
|
|
|
|
local name = game.item_prototypes[k]
|
|
if name then
|
|
tooltip_str[count] = concat {'[item=', k, ']'}
|
|
mapview_str[count] = name.localised_name
|
|
else
|
|
name = game.fluid_prototypes[k]
|
|
if name then
|
|
tooltip_str[count] = concat {'[fluid=', k, ']'}
|
|
mapview_str[count] = name.localised_name
|
|
else
|
|
tooltip_str[count] = k
|
|
mapview_str[count] = k
|
|
end
|
|
end
|
|
count = count + 1
|
|
|
|
local str = concat {': ', format('%.2f', current_rate), ' -> ', format('%.2f / sec', next_rate)}
|
|
|
|
tooltip_str[count] = str
|
|
tooltip_str[count + 1] = '\n'
|
|
|
|
mapview_str[count] = str
|
|
mapview_str[count + 1] = ', '
|
|
|
|
count = count + 2
|
|
end
|
|
tooltip_str[count - 1] = nil
|
|
mapview_str[count - 1] = nil
|
|
|
|
prototype.description = tooltip_str
|
|
prototype.disabled = false
|
|
|
|
prototype.mapview_description = mapview_str
|
|
|
|
Retailer.set_item(outpost_id, prototype)
|
|
end
|
|
|
|
local function do_outpost_upgrade(event)
|
|
if event.item.type ~= 'upgrade' then
|
|
return
|
|
end
|
|
|
|
local outpost_id = event.group_name
|
|
local outpost_data = outposts[outpost_id]
|
|
|
|
local outpost_magic_crafters = outpost_data.magic_crafters
|
|
local upgrade_rate = outpost_data.upgrade_rate
|
|
|
|
local level = outpost_data.level + 1
|
|
outpost_data.level = level
|
|
|
|
local outpost_name = Retailer.get_market_group_label(outpost_id)
|
|
local message = concat {outpost_name, ' has been upgraded to level ', level}
|
|
|
|
CrashSiteToast.do_outpost_toast(outpost_data.market, message)
|
|
Server.to_discord_bold(concat {'*** ', message, ' ***'})
|
|
|
|
for i = 1, #outpost_magic_crafters do
|
|
local crafter = outpost_magic_crafters[i]
|
|
|
|
local rate = crafter.rate
|
|
local upgrade_amount = upgrade_rate * crafter.base_rate
|
|
|
|
crafter.rate = rate + upgrade_amount
|
|
end
|
|
|
|
update_market_upgrade_description(outpost_data)
|
|
end
|
|
|
|
local function activate_market_upgrade(outpost_data)
|
|
local outpost_magic_crafters = outpost_data.magic_crafters
|
|
|
|
if #outpost_magic_crafters == 0 then
|
|
local outpost_id = outpost_data.outpost_id
|
|
|
|
local prototype = Retailer.get_items(outpost_id)['upgrade']
|
|
prototype.disabled_reason = 'No machines to upgrade.'
|
|
|
|
Retailer.set_item(outpost_id, prototype)
|
|
|
|
return
|
|
end
|
|
|
|
local base_outputs = {}
|
|
|
|
for i = 1, #outpost_magic_crafters do
|
|
local crafter = outpost_magic_crafters[i]
|
|
local item = crafter.item
|
|
|
|
local count = base_outputs[item] or 0
|
|
count = count + crafter.rate
|
|
|
|
base_outputs[item] = count
|
|
end
|
|
|
|
outpost_data.base_outputs = base_outputs
|
|
|
|
update_market_upgrade_description(outpost_data)
|
|
end
|
|
|
|
function Public.activate_market_upgrade(outpost_id)
|
|
local outpost_data = outposts[outpost_id]
|
|
activate_market_upgrade(outpost_data)
|
|
end
|
|
|
|
local function do_capture_outpost(outpost_data)
|
|
local area = {top_left = outpost_data.top_left, bottom_right = outpost_data.bottom_right}
|
|
local walls = RS.get_surface().find_entities_filtered {area = area, force = 'enemy', name = 'stone-wall'}
|
|
|
|
for i = 1, #walls do
|
|
walls[i].force = 'player'
|
|
end
|
|
|
|
local outpost_id = outpost_data.outpost_id
|
|
|
|
local name = Retailer.get_market_group_label(outpost_id)
|
|
if name == 'Market' then
|
|
return
|
|
end
|
|
|
|
local donators = Donator.get_donators_table()
|
|
if next(donators) then
|
|
local donator = table.get_random_dictionary_entry(donators, true)
|
|
if donator then
|
|
name = concat({donator, "'s ", name})
|
|
Retailer.set_market_group_label(outpost_id, name)
|
|
end
|
|
end
|
|
|
|
local message = 'Outpost captured: ' .. name
|
|
CrashSiteToast.do_outpost_toast(outpost_data.market, message)
|
|
Server.to_discord_bold(concat {'*** ', message, ' ***'})
|
|
|
|
activate_market_upgrade(outpost_data)
|
|
end
|
|
|
|
local function do_refill_turrets()
|
|
local index = refill_turrets.index
|
|
|
|
if index > #refill_turrets then
|
|
refill_turrets.index = 1
|
|
return
|
|
end
|
|
|
|
local turret_data = refill_turrets[index]
|
|
local turret = turret_data.turret
|
|
|
|
if not turret.valid then
|
|
fast_remove(refill_turrets, index)
|
|
return
|
|
end
|
|
|
|
refill_turrets.index = index + 1
|
|
|
|
local data = turret_data.data
|
|
if data.liquid then
|
|
turret.fluidbox[1] = data
|
|
elseif data then
|
|
turret.insert(data)
|
|
end
|
|
end
|
|
|
|
local artillery_target_entities = {
|
|
'character',
|
|
'tank',
|
|
'car',
|
|
'locomotive',
|
|
'cargo-wagon',
|
|
'fluid-wagon',
|
|
'artillery-wagon',
|
|
'spidertron'
|
|
}
|
|
|
|
local artillery_target_callback =
|
|
Token.register(
|
|
function(data)
|
|
local position = data.position
|
|
local entity = data.entity
|
|
|
|
if not entity.valid then
|
|
return
|
|
end
|
|
|
|
local tx, ty = position.x, position.y
|
|
|
|
local pos = entity.position
|
|
local x, y = pos.x, pos.y
|
|
local dx, dy = tx - x, ty - y
|
|
local d = dx * dx + dy * dy
|
|
if d >= 1024 then -- 32 ^ 2
|
|
entity.surface.create_entity {
|
|
name = 'artillery-projectile',
|
|
position = position,
|
|
target = entity,
|
|
speed = 1.5
|
|
}
|
|
if entity.name == "spidertron" then
|
|
-- fire a rocket as well as artillery-projectile since artillery-projectile doesn't damage spidertrons
|
|
-- Balance note: 1 rocket per artillery kills spidertron shield fast but takes ~ 1 minute to kill the spidertron
|
|
entity.surface.create_entity {
|
|
name = 'rocket',
|
|
position = position,
|
|
target = entity,
|
|
speed = 1.5
|
|
}
|
|
end
|
|
end
|
|
end
|
|
)
|
|
|
|
local function do_artillery_turrets_targets()
|
|
local index = artillery_outposts.index
|
|
|
|
if index > #artillery_outposts then
|
|
artillery_outposts.index = 1
|
|
return
|
|
end
|
|
|
|
artillery_outposts.index = index + 1
|
|
|
|
local outpost = artillery_outposts[index]
|
|
|
|
local now = game.tick
|
|
if now - outpost.last_fire_tick < 480 then
|
|
return
|
|
end
|
|
|
|
local turrets = outpost.artillery_turrets
|
|
for i = #turrets, 1, -1 do
|
|
local turret = turrets[i]
|
|
if not turret.valid then
|
|
fast_remove(turrets, i)
|
|
end
|
|
end
|
|
|
|
local count = #turrets
|
|
if count == 0 then
|
|
fast_remove(artillery_outposts, index)
|
|
return
|
|
end
|
|
|
|
outpost.last_fire_tick = now
|
|
|
|
local turret = turrets[1]
|
|
local area = outpost.artillery_area
|
|
local surface = turret.surface
|
|
|
|
local entities = surface.find_entities_filtered {area = area, name = artillery_target_entities}
|
|
|
|
if #entities == 0 then
|
|
return
|
|
end
|
|
|
|
local position = turret.position
|
|
|
|
for i = 1, count do
|
|
local entity = entities[math.random(#entities)]
|
|
if entity and entity.valid then
|
|
local data = {position = position, entity = entity}
|
|
Task.set_timeout_in_ticks(i * 60, artillery_target_callback, data)
|
|
end
|
|
end
|
|
end
|
|
|
|
local function do_magic_crafters()
|
|
local limit = #magic_crafters
|
|
if limit == 0 then
|
|
return
|
|
end
|
|
|
|
local index = magic_crafters.index
|
|
|
|
for i = 1, magic_crafters_per_tick do
|
|
if index > limit then
|
|
index = 1
|
|
end
|
|
|
|
local data = magic_crafters[index]
|
|
|
|
local entity = data.entity
|
|
if not entity.valid then
|
|
fast_remove(magic_crafters, index)
|
|
limit = limit - 1
|
|
if limit == 0 then
|
|
return
|
|
end
|
|
else
|
|
index = index + 1
|
|
|
|
local tick = game.tick
|
|
local last_tick = data.last_tick
|
|
local rate = data.rate
|
|
|
|
local count = (tick - last_tick) * rate
|
|
|
|
local fcount = floor(count)
|
|
|
|
if fcount > 0 then
|
|
entity.get_output_inventory().insert {name = data.item, count = fcount}
|
|
data.last_tick = tick - (count - fcount) / rate
|
|
end
|
|
end
|
|
end
|
|
|
|
magic_crafters.index = index
|
|
end
|
|
|
|
local function do_magic_fluid_crafters()
|
|
local limit = #magic_fluid_crafters
|
|
|
|
if limit == 0 then
|
|
return
|
|
end
|
|
|
|
local index = magic_fluid_crafters.index
|
|
|
|
for i = 1, magic_fluid_crafters_per_tick do
|
|
if index > limit then
|
|
index = 1
|
|
end
|
|
|
|
local data = magic_fluid_crafters[index]
|
|
|
|
local entity = data.entity
|
|
if not entity.valid then
|
|
fast_remove(magic_fluid_crafters, index)
|
|
limit = limit - 1
|
|
if limit == 0 then
|
|
return
|
|
end
|
|
else
|
|
index = index + 1
|
|
|
|
local tick = game.tick
|
|
local last_tick = data.last_tick
|
|
local rate = data.rate
|
|
|
|
local count = (tick - last_tick) * rate
|
|
|
|
local fcount = floor(count)
|
|
|
|
if fcount > 0 then
|
|
local fluidbox_index = data.fluidbox_index
|
|
local fb = entity.fluidbox
|
|
|
|
local fb_data = fb[fluidbox_index] or {name = data.item, amount = 0}
|
|
fb_data.amount = fb_data.amount + fcount
|
|
fb[fluidbox_index] = fb_data
|
|
|
|
data.last_tick = tick - (count - fcount) / rate
|
|
end
|
|
end
|
|
end
|
|
|
|
magic_fluid_crafters.index = index
|
|
end
|
|
|
|
local function tick()
|
|
do_refill_turrets()
|
|
do_artillery_turrets_targets()
|
|
do_magic_crafters()
|
|
do_magic_fluid_crafters()
|
|
end
|
|
|
|
Public.refill_turret_callback =
|
|
Token.register(
|
|
function(turret, data)
|
|
local outpost_id = data.outpost_id
|
|
|
|
refill_turrets[#refill_turrets + 1] = {turret = turret, data = data.callback_data}
|
|
turret_to_outpost[turret.unit_number] = outpost_id
|
|
|
|
local outpost_data = outposts[outpost_id]
|
|
outpost_data.turret_count = outpost_data.turret_count + 1
|
|
end
|
|
)
|
|
|
|
Public.refill_liquid_turret_callback =
|
|
Token.register(
|
|
function(turret, data)
|
|
local callback_data = data.callback_data
|
|
callback_data.liquid = true
|
|
|
|
local outpost_id = data.outpost_id
|
|
|
|
refill_turrets[#refill_turrets + 1] = {turret = turret, data = callback_data}
|
|
turret_to_outpost[turret.unit_number] = outpost_id
|
|
|
|
local outpost_data = outposts[outpost_id]
|
|
outpost_data.turret_count = outpost_data.turret_count + 1
|
|
end
|
|
)
|
|
|
|
Public.refill_artillery_turret_callback =
|
|
Token.register(
|
|
function(turret, data)
|
|
local outpost_id = data.outpost_id
|
|
|
|
refill_turrets[#refill_turrets + 1] = {turret = turret, data = data.callback_data}
|
|
turret_to_outpost[turret.unit_number] = outpost_id
|
|
|
|
local outpost_data = outposts[outpost_id]
|
|
outpost_data.turret_count = outpost_data.turret_count + 1
|
|
|
|
local artillery_turrets = outpost_data.artillery_turrets
|
|
if not artillery_turrets then
|
|
artillery_turrets = {}
|
|
outpost_data.artillery_turrets = artillery_turrets
|
|
|
|
local pos = turret.position
|
|
local x, y = pos.x, pos.y
|
|
outpost_data.artillery_area = {{x - 112, y - 112}, {x + 112, y + 112}}
|
|
outpost_data.last_fire_tick = 0
|
|
|
|
artillery_outposts[#artillery_outposts + 1] = outpost_data
|
|
end
|
|
|
|
artillery_turrets[#artillery_turrets + 1] = turret
|
|
end
|
|
)
|
|
|
|
Public.power_source_callback =
|
|
Token.register(
|
|
function(turret, data)
|
|
local outpost_id = data.outpost_id
|
|
local callback_data = data.callback_data
|
|
|
|
local power_source =
|
|
turret.surface.create_entity {name = 'hidden-electric-energy-interface', position = turret.position}
|
|
power_source.electric_buffer_size = callback_data.buffer_size
|
|
power_source.power_production = callback_data.power_production
|
|
power_source.destructible = false
|
|
|
|
power_sources[turret.unit_number] = {entity = power_source}
|
|
turret_to_outpost[turret.unit_number] = outpost_id
|
|
|
|
local outpost_data = outposts[outpost_id]
|
|
outpost_data.turret_count = outpost_data.turret_count + 1
|
|
end
|
|
)
|
|
|
|
Public.worm_turret_callback =
|
|
Token.register(
|
|
function(turret, data)
|
|
local outpost_id = data.outpost_id
|
|
|
|
turret_to_outpost[turret.unit_number] = outpost_id
|
|
|
|
local outpost_data = outposts[outpost_id]
|
|
outpost_data.turret_count = outpost_data.turret_count + 1
|
|
end
|
|
)
|
|
|
|
local function add_magic_crafter_output(entity, output, distance, outpost_id)
|
|
local rate = output.min_rate + output.distance_factor * distance
|
|
|
|
local fluidbox_index = output.fluidbox_index
|
|
local data = {
|
|
entity = entity,
|
|
last_tick = game.tick,
|
|
base_rate = rate,
|
|
rate = rate,
|
|
item = output.item,
|
|
fluidbox_index = fluidbox_index
|
|
}
|
|
|
|
if fluidbox_index then
|
|
magic_fluid_crafters[#magic_fluid_crafters + 1] = data
|
|
else
|
|
magic_crafters[#magic_crafters + 1] = data
|
|
end
|
|
|
|
local outpost_magic_crafters = outposts[outpost_id].magic_crafters
|
|
outpost_magic_crafters[#outpost_magic_crafters + 1] = data
|
|
end
|
|
|
|
local set_inactive_token =
|
|
Token.register(
|
|
function(entity)
|
|
if entity.valid then
|
|
entity.active = false
|
|
end
|
|
end
|
|
)
|
|
|
|
Public.magic_item_crafting_callback =
|
|
Token.register(
|
|
function(entity, data)
|
|
local outpost_id = data.outpost_id
|
|
local callback_data = data.callback_data
|
|
|
|
entity.minable = false
|
|
entity.destructible = false
|
|
entity.operable = false
|
|
|
|
local recipe = callback_data.recipe
|
|
if recipe then
|
|
entity.set_recipe(recipe)
|
|
else
|
|
local furance_item = callback_data.furance_item
|
|
if furance_item then
|
|
local inv = entity.get_inventory(2) -- defines.inventory.furnace_source
|
|
inv.insert(furance_item)
|
|
end
|
|
end
|
|
|
|
local p = entity.position
|
|
local x, y = p.x, p.y
|
|
local distance = math.sqrt(x * x + y * y)
|
|
|
|
local output = callback_data.output
|
|
if #output == 0 then
|
|
add_magic_crafter_output(entity, output, distance, outpost_id)
|
|
else
|
|
for i = 1, #output do
|
|
local o = output[i]
|
|
add_magic_crafter_output(entity, o, distance, outpost_id)
|
|
end
|
|
end
|
|
|
|
if not callback_data.keep_active then
|
|
Task.set_timeout_in_ticks(2, set_inactive_token, entity) -- causes problems with refineries.
|
|
end
|
|
end
|
|
)
|
|
|
|
Public.magic_item_crafting_callback_weighted =
|
|
Token.register(
|
|
function(entity, data)
|
|
local outpost_id = data.outpost_id
|
|
local callback_data = data.callback_data
|
|
|
|
entity.minable = false
|
|
entity.destructible = false
|
|
entity.operable = false
|
|
|
|
local weights = callback_data.weights
|
|
local loot = callback_data.loot
|
|
|
|
local i = math.random() * weights.total
|
|
|
|
local index = table.binary_search(weights, i)
|
|
if (index < 0) then
|
|
index = bit32.bnot(index)
|
|
end
|
|
|
|
local stack = loot[index].stack
|
|
if not stack then
|
|
return
|
|
end
|
|
|
|
local recipe = stack.recipe
|
|
if recipe then
|
|
entity.set_recipe(recipe)
|
|
else
|
|
local furance_item = stack.furance_item
|
|
if furance_item then
|
|
local inv = entity.get_inventory(2) -- defines.inventory.furnace_source
|
|
inv.insert(furance_item)
|
|
end
|
|
end
|
|
|
|
local p = entity.position
|
|
local x, y = p.x, p.y
|
|
local distance = math.sqrt(x * x + y * y)
|
|
|
|
local output = stack.output
|
|
if #output == 0 then
|
|
add_magic_crafter_output(entity, output, distance, outpost_id)
|
|
else
|
|
for o_i = 1, #output do
|
|
local o = output[o_i]
|
|
add_magic_crafter_output(entity, o, distance, outpost_id)
|
|
end
|
|
end
|
|
|
|
if not callback_data.keep_active then
|
|
Task.set_timeout_in_ticks(2, set_inactive_token, entity) -- causes problems with refineries.
|
|
end
|
|
end
|
|
)
|
|
|
|
Public.wall_callback =
|
|
Token.register(
|
|
function(entity, data)
|
|
if not entity.valid then
|
|
return
|
|
end
|
|
|
|
local position = entity.position
|
|
local px, py = position.x, position.y
|
|
|
|
local outpost_id = data.outpost_id
|
|
local outpost_data = outposts[outpost_id]
|
|
local top_left = outpost_data.top_left
|
|
local bottom_right = outpost_data.bottom_right
|
|
local tx, ty = top_left.x, top_left.y
|
|
local bx, by = bottom_right.x, bottom_right.y
|
|
|
|
if not tx or px < tx then
|
|
top_left.x = px
|
|
end
|
|
if not ty or py < ty then
|
|
top_left.y = py
|
|
end
|
|
|
|
if not bx or px > bx then
|
|
bottom_right.x = px
|
|
end
|
|
if not by or py > by then
|
|
bottom_right.y = py
|
|
end
|
|
end
|
|
)
|
|
|
|
Public.deactivate_callback =
|
|
Token.register(
|
|
function(entity)
|
|
entity.active = false
|
|
entity.operable = false
|
|
entity.destructible = false
|
|
end
|
|
)
|
|
|
|
|
|
Public.scenario_chest_callback = Token.register(function(chest)
|
|
if not chest or not chest.valid then
|
|
return
|
|
end
|
|
|
|
chest.destructible = false
|
|
chest.minable = false
|
|
end)
|
|
|
|
local function turret_died(event)
|
|
local entity = event.entity
|
|
if not entity or not entity.valid then
|
|
return
|
|
end
|
|
|
|
local number = entity.unit_number
|
|
if not number then
|
|
return
|
|
end
|
|
|
|
local ps_data = power_sources[number]
|
|
if ps_data then
|
|
power_sources[number] = nil
|
|
|
|
local ps_entity = ps_data.entity
|
|
|
|
if ps_entity and ps_entity.valid then
|
|
ps_entity.destroy()
|
|
end
|
|
end
|
|
|
|
local outpost_id = turret_to_outpost[number]
|
|
if outpost_id then
|
|
local outpost_data = outposts[outpost_id]
|
|
|
|
local turret_count = outpost_data.turret_count - 1
|
|
outpost_data.turret_count = turret_count
|
|
|
|
if turret_count == 0 then
|
|
do_capture_outpost(outpost_data)
|
|
end
|
|
end
|
|
end
|
|
|
|
Public.market_set_items_callback =
|
|
Token.register(
|
|
function(entity, data)
|
|
if not entity.valid then
|
|
return
|
|
end
|
|
|
|
local callback_data = data.callback_data
|
|
|
|
entity.destructible = false
|
|
|
|
local market_id = data.outpost_id
|
|
local outpost_data = outposts[market_id]
|
|
local upgrade_base_cost = callback_data.upgrade_base_cost or 0
|
|
|
|
outpost_data.market = entity
|
|
outpost_data.upgrade_rate = callback_data.upgrade_rate
|
|
outpost_data.upgrade_base_cost = upgrade_base_cost
|
|
outpost_data.upgrade_cost_base = callback_data.upgrade_cost_base
|
|
|
|
Retailer.add_market(market_id, entity)
|
|
Retailer.set_market_group_label(market_id, callback_data.market_name)
|
|
|
|
Retailer.set_item(
|
|
market_id,
|
|
{
|
|
name = 'upgrade',
|
|
type = 'upgrade',
|
|
name_label = 'Upgrade Outpost',
|
|
sprite = 'item-group/production',
|
|
price = 0,
|
|
stack_limit = 1,
|
|
disabled = true,
|
|
disabled_reason = 'Outpost must be captured first.',
|
|
description = 'Increases output.'
|
|
}
|
|
)
|
|
|
|
local p = entity.position
|
|
local x, y = p.x, p.y
|
|
local d = math.sqrt(x * x + y * y)
|
|
|
|
for i = 1, #callback_data do
|
|
local item = callback_data[i]
|
|
local price = item.price
|
|
|
|
local df = item.distance_factor
|
|
if df then
|
|
local min_price = item.min_price or 1
|
|
|
|
price = item.price - d * df
|
|
price = math.max(price, min_price)
|
|
end
|
|
|
|
Retailer.set_item(
|
|
market_id,
|
|
{name = item.name, type = item.type, price = price, name_label = item.name_label, sprite = item.sprite, description = item.description}
|
|
)
|
|
end
|
|
end
|
|
)
|
|
|
|
Public.firearm_magazine_ammo = {name = 'firearm-magazine', count = 200}
|
|
Public.piercing_rounds_magazine_ammo = {name = 'piercing-rounds-magazine', count = 200}
|
|
Public.uranium_rounds_magazine_ammo = {name = 'uranium-rounds-magazine', count = 200}
|
|
Public.artillery_shell_ammo = {name = 'artillery-shell', count = 15}
|
|
Public.light_oil_ammo = {name = 'light-oil', amount = 100}
|
|
|
|
Public.laser_turrent_power_source = {buffer_size = 2400000, power_production = 40000}
|
|
|
|
function Public.prepare_weighted_loot(loot)
|
|
local total = 0
|
|
local weights = {}
|
|
|
|
for i = 1, #loot do
|
|
local v = loot[i]
|
|
total = total + v.weight
|
|
weights[#weights + 1] = total
|
|
end
|
|
|
|
weights.total = total
|
|
|
|
return weights
|
|
end
|
|
|
|
function Public.do_random_loot(entity, weights, loot)
|
|
if not entity.valid then
|
|
return
|
|
end
|
|
|
|
entity.operable = false
|
|
--entity.destructible = false
|
|
|
|
local i = math.random() * weights.total
|
|
|
|
local index = table.binary_search(weights, i)
|
|
if (index < 0) then
|
|
index = bit32.bnot(index)
|
|
end
|
|
|
|
local stack = loot[index].stack
|
|
if not stack then
|
|
return
|
|
end
|
|
|
|
local df = stack.distance_factor
|
|
local count
|
|
if df then
|
|
local p = entity.position
|
|
local x, y = p.x, p.y
|
|
local d = math.sqrt(x * x + y * y)
|
|
|
|
count = stack.count + d * df
|
|
else
|
|
count = stack.count
|
|
end
|
|
|
|
entity.insert {name = stack.name, count = count}
|
|
end
|
|
|
|
function Public.do_random_fluid_loot(entity, weights, loot)
|
|
if not entity.valid then
|
|
return
|
|
end
|
|
|
|
entity.operable = false
|
|
entity.destructible = false
|
|
|
|
local i = math.random() * weights.total
|
|
|
|
local index = table.binary_search(weights, i)
|
|
if (index < 0) then
|
|
index = bit32.bnot(index)
|
|
end
|
|
|
|
local stack = loot[index].stack
|
|
if not stack then
|
|
return
|
|
end
|
|
|
|
local df = stack.distance_factor
|
|
local count
|
|
if df then
|
|
local p = entity.position
|
|
local x, y = p.x, p.y
|
|
local d = math.sqrt(x * x + y * y)
|
|
|
|
count = stack.count + d * df
|
|
else
|
|
count = stack.count
|
|
end
|
|
|
|
entity.fluidbox[1] = {name = stack.name, amount = count}
|
|
end
|
|
|
|
function Public.do_factory_loot(entity, weights, loot)
|
|
if not entity.valid then
|
|
return
|
|
end
|
|
|
|
entity.operable = false
|
|
entity.destructible = false
|
|
|
|
local i = math.random() * weights.total
|
|
|
|
local index = table.binary_search(weights, i)
|
|
if (index < 0) then
|
|
index = bit32.bnot(index)
|
|
end
|
|
|
|
local stack = loot[index].stack
|
|
if not stack then
|
|
return
|
|
end
|
|
|
|
local df = stack.distance_factor
|
|
local count
|
|
if df then
|
|
local p = entity.position
|
|
local x, y = p.x, p.y
|
|
local d = math.sqrt(x * x + y * y)
|
|
|
|
count = stack.count + d * df
|
|
else
|
|
count = stack.count
|
|
end
|
|
|
|
local name = stack.name
|
|
|
|
entity.set_recipe(name)
|
|
entity.get_output_inventory().insert {name = name, count = count}
|
|
end
|
|
|
|
local function coin_mined(event)
|
|
local stack = event.item_stack
|
|
if stack.name == 'coin' then
|
|
change_for_player(event.player_index, coins_earned_name, stack.count)
|
|
end
|
|
end
|
|
|
|
local function market_selected(event)
|
|
local player = game.get_player(event.player_index)
|
|
if not player or not player.valid then
|
|
return
|
|
end
|
|
|
|
if player.render_mode == render_mode_game then
|
|
return
|
|
end
|
|
|
|
local selected = player.selected
|
|
|
|
if not selected or not selected.valid or selected.name ~= 'market' then
|
|
return
|
|
end
|
|
|
|
local group = Retailer.get_market_group_name(selected)
|
|
local prototype = Retailer.get_items(group)['upgrade']
|
|
|
|
if prototype.disabled then
|
|
return
|
|
end
|
|
|
|
local args = {
|
|
text = nil,
|
|
target = selected,
|
|
target_offset = nil,
|
|
alignment = 'center',
|
|
surface = selected.surface,
|
|
color = {1, 1, 1},
|
|
players = {player},
|
|
scale = 1.5,
|
|
scale_with_zoom = true,
|
|
time_to_live = 180
|
|
}
|
|
|
|
args.text = prototype.name_label
|
|
args.target_offset = {0, -6.5}
|
|
draw_text(args)
|
|
|
|
args.text = 'Price: ' .. prototype.price
|
|
args.target_offset = {0, -5}
|
|
draw_text(args)
|
|
|
|
args.text = prototype.mapview_description
|
|
args.target_offset = {0, -3.5}
|
|
draw_text(args)
|
|
end
|
|
|
|
Event.add(defines.events.on_tick, tick)
|
|
Event.add(defines.events.on_entity_died, turret_died)
|
|
|
|
Event.on_init(
|
|
function()
|
|
game.forces.neutral.recipes['steel-plate'].enabled = true
|
|
end
|
|
)
|
|
|
|
Event.add(defines.events.on_player_mined_item, coin_mined)
|
|
|
|
Event.add(Retailer.events.on_market_purchase, do_outpost_upgrade)
|
|
|
|
Event.add(defines.events.on_selected_entity_changed, market_selected)
|
|
|
|
return Public
|