2020-01-22 08:30:41 +01:00
|
|
|
local Public = {}
|
2019-03-13 17:09:06 +01:00
|
|
|
|
2020-04-16 11:42:00 +02:00
|
|
|
local table_remove = table.remove
|
|
|
|
local table_insert = table.insert
|
|
|
|
|
2019-03-13 17:09:06 +01:00
|
|
|
local direction_translation = {
|
|
|
|
[0] = 4,
|
|
|
|
[1] = 5,
|
|
|
|
[2] = 6,
|
|
|
|
[3] = 7,
|
|
|
|
[4] = 0,
|
|
|
|
[5] = 1,
|
|
|
|
[6] = 2,
|
|
|
|
[7] = 3
|
|
|
|
}
|
|
|
|
|
|
|
|
local cliff_orientation_translation = {
|
2019-03-13 20:22:21 +01:00
|
|
|
["east-to-none"] = "west-to-none",
|
|
|
|
["east-to-north"] = "west-to-south",
|
2019-03-13 17:51:49 +01:00
|
|
|
["east-to-south"] = "west-to-north",
|
2019-09-18 21:56:43 +02:00
|
|
|
["east-to-west"] = "west-to-east",
|
2019-03-13 20:22:21 +01:00
|
|
|
["north-to-east"] = "south-to-west",
|
|
|
|
["north-to-none"] = "south-to-none",
|
|
|
|
["north-to-south"] = "south-to-north",
|
2019-03-13 17:51:49 +01:00
|
|
|
["north-to-west"] = "south-to-east",
|
|
|
|
["south-to-east"] = "north-to-west",
|
2019-03-13 20:22:21 +01:00
|
|
|
["south-to-none"] = "north-to-none",
|
|
|
|
["south-to-north"] = "north-to-south",
|
2019-09-18 21:56:43 +02:00
|
|
|
["south-to-west"] = "north-to-east",
|
2019-03-13 20:22:21 +01:00
|
|
|
["west-to-east"] = "east-to-west",
|
2019-03-13 17:51:49 +01:00
|
|
|
["west-to-none"] = "east-to-none",
|
2019-03-13 20:22:21 +01:00
|
|
|
["west-to-north"] = "east-to-south",
|
|
|
|
["west-to-south"] = "east-to-north",
|
2019-03-13 17:51:49 +01:00
|
|
|
["none-to-east"] = "none-to-west",
|
2019-03-13 20:22:21 +01:00
|
|
|
["none-to-north"] = "none-to-south",
|
2019-03-13 17:51:49 +01:00
|
|
|
["none-to-south"] = "none-to-north",
|
2019-03-13 20:22:21 +01:00
|
|
|
["none-to-west"] = "none-to-east"
|
2019-03-13 17:09:06 +01:00
|
|
|
}
|
|
|
|
|
2019-11-17 23:29:11 +01:00
|
|
|
local entity_copy_functions = {
|
2020-04-16 11:42:00 +02:00
|
|
|
["tree"] = function(surface, entity, target_position, force_name)
|
|
|
|
if not surface.can_place_entity({name = entity.name, position = target_position}) then return end
|
|
|
|
entity.clone({position = target_position, surface = surface, force = "neutral"})
|
2019-11-17 23:29:11 +01:00
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["simple-entity"] = function(surface, entity, target_position, force_name)
|
|
|
|
local mirror_entity = {name = entity.name, position = target_position, direction = direction_translation[entity.direction]}
|
2019-11-17 23:29:11 +01:00
|
|
|
if not surface.can_place_entity(mirror_entity) then return end
|
|
|
|
local mirror_entity = surface.create_entity(mirror_entity)
|
|
|
|
mirror_entity.graphics_variation = entity.graphics_variation
|
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["cliff"] = function(surface, entity, target_position, force_name)
|
|
|
|
local mirror_entity = {name = entity.name, position = target_position, cliff_orientation = cliff_orientation_translation[entity.cliff_orientation]}
|
2019-11-17 23:29:11 +01:00
|
|
|
if not surface.can_place_entity(mirror_entity) then return end
|
|
|
|
surface.create_entity(mirror_entity)
|
|
|
|
return
|
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["resource"] = function(surface, entity, target_position, force_name)
|
|
|
|
surface.create_entity({name = entity.name, position = target_position, amount = entity.amount})
|
2019-11-17 23:29:11 +01:00
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["corpse"] = function(surface, entity, target_position, force_name)
|
|
|
|
surface.create_entity({name = entity.name, position = target_position})
|
2019-11-17 23:29:11 +01:00
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["unit-spawner"] = function(surface, entity, target_position, force_name)
|
|
|
|
local mirror_entity = {name = entity.name, position = target_position, direction = direction_translation[entity.direction], force = force_name .. "_biters"}
|
2020-02-08 09:32:10 +01:00
|
|
|
if not surface.can_place_entity(mirror_entity) then return end
|
2020-04-16 11:42:00 +02:00
|
|
|
table_insert(global.unit_spawners[force_name .. "_biters"], surface.create_entity(mirror_entity))
|
2019-11-17 23:29:11 +01:00
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["turret"] = function(surface, entity, target_position, force_name)
|
|
|
|
local mirror_entity = {name = entity.name, position = target_position, direction = direction_translation[entity.direction], force = force_name .. "_biters"}
|
2019-11-17 23:29:11 +01:00
|
|
|
if not surface.can_place_entity(mirror_entity) then return end
|
|
|
|
surface.create_entity(mirror_entity)
|
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["rocket-silo"] = function(surface, entity, target_position, force_name)
|
|
|
|
if surface.count_entities_filtered({name = "rocket-silo", area = {{target_position.x - 8, target_position.y - 8},{target_position.x + 8, target_position.y + 8}}}) > 0 then return end
|
|
|
|
global.rocket_silo[force_name] = surface.create_entity({name = entity.name, position = target_position, direction = direction_translation[entity.direction], force = force_name})
|
|
|
|
global.rocket_silo[force_name].minable = false
|
2019-11-17 23:29:11 +01:00
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["ammo-turret"] = function(surface, entity, target_position, force_name)
|
2020-04-16 13:22:15 +02:00
|
|
|
local direction = 0
|
|
|
|
if force_name == "south" then direction = 4 end
|
|
|
|
local mirror_entity = {name = entity.name, position = target_position, force = force_name, direction = direction}
|
|
|
|
if not surface.can_place_entity(mirror_entity) then return end
|
|
|
|
local e = surface.create_entity(mirror_entity)
|
|
|
|
local inventory = entity.get_inventory(defines.inventory.turret_ammo)
|
|
|
|
if inventory.is_empty() then return end
|
|
|
|
for name, count in pairs(inventory.get_contents()) do e.insert({name = name, count = count}) end
|
2019-11-17 23:29:11 +01:00
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["wall"] = function(surface, entity, target_position, force_name)
|
2020-04-16 13:22:15 +02:00
|
|
|
local e = entity.clone({position = target_position, surface = surface, force = force_name})
|
|
|
|
e.active = true
|
2019-11-17 23:29:11 +01:00
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["container"] = function(surface, entity, target_position, force_name)
|
2020-04-16 13:22:15 +02:00
|
|
|
local e = entity.clone({position = target_position, surface = surface, force = force_name})
|
|
|
|
e.active = true
|
2019-11-17 23:29:11 +01:00
|
|
|
end,
|
2020-04-16 11:42:00 +02:00
|
|
|
["fish"] = function(surface, entity, target_position, force_name)
|
2020-04-16 13:22:15 +02:00
|
|
|
local mirror_entity = {name = entity.name, position = target_position}
|
2019-11-17 23:29:11 +01:00
|
|
|
if not surface.can_place_entity(mirror_entity) then return end
|
|
|
|
local e = surface.create_entity(mirror_entity)
|
|
|
|
end,
|
2019-09-08 00:21:02 +02:00
|
|
|
}
|
|
|
|
|
2020-04-16 11:42:00 +02:00
|
|
|
local function process_entity(surface, entity, force_name)
|
2019-10-05 04:18:18 +02:00
|
|
|
if not entity.valid then return end
|
2019-11-17 23:29:11 +01:00
|
|
|
if not entity_copy_functions[entity.type] then return end
|
2020-04-16 11:42:00 +02:00
|
|
|
|
|
|
|
local target_position
|
|
|
|
if force_name == "north" then
|
|
|
|
target_position = entity.position
|
|
|
|
else
|
|
|
|
target_position = {x = entity.position.x * -1, y = entity.position.y * -1}
|
|
|
|
end
|
|
|
|
|
|
|
|
entity_copy_functions[entity.type](surface, entity, target_position, force_name)
|
2019-03-13 17:09:06 +01:00
|
|
|
end
|
|
|
|
|
2020-04-16 11:42:00 +02:00
|
|
|
local function copy_chunk(chunk)
|
|
|
|
local target_surface = game.surfaces.biter_battles
|
|
|
|
|
|
|
|
local source_surface = game.surfaces.bb_source
|
|
|
|
local source_chunk_position = {chunk[1][1], chunk[1][2]}
|
|
|
|
local source_left_top = {x = source_chunk_position[1] * 32, y = source_chunk_position[2] * 32}
|
|
|
|
local source_area = {{source_left_top.x, source_left_top.y}, {source_left_top.x + 32, source_left_top.y + 32}}
|
|
|
|
|
|
|
|
local target_chunk_position = chunk[1]
|
|
|
|
local target_left_top = {x = target_chunk_position[1] * 32, y = target_chunk_position[2] * 32}
|
|
|
|
local target_area = {{target_left_top.x, target_left_top.y}, {target_left_top.x + 32, target_left_top.y + 32}}
|
|
|
|
|
|
|
|
if not source_surface.is_chunk_generated(source_chunk_position) then
|
|
|
|
source_surface.request_to_generate_chunks({x = source_left_top.x + 16, y = source_left_top.y + 16}, 0)
|
|
|
|
return
|
2019-12-31 02:34:02 +01:00
|
|
|
end
|
|
|
|
|
2020-04-16 11:42:00 +02:00
|
|
|
if chunk[2] == 1 then
|
|
|
|
source_surface.clone_area({
|
|
|
|
source_area = source_area,
|
|
|
|
destination_area = target_area,
|
|
|
|
destination_surface = target_surface,
|
|
|
|
--destination_force = …,
|
|
|
|
clone_tiles = true,
|
|
|
|
clone_entities = false,
|
|
|
|
clone_decoratives = false,
|
|
|
|
clear_destination_entities = false,
|
|
|
|
clear_destination_decoratives = false,
|
|
|
|
expand_map = false
|
|
|
|
})
|
|
|
|
chunk[2] = chunk[2] + 1
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
if chunk[2] == 2 then
|
|
|
|
for _, entity in pairs(source_surface.find_entities_filtered({area = source_area})) do
|
|
|
|
process_entity(target_surface, entity, "north")
|
2019-03-19 04:05:32 +01:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
chunk[2] = chunk[2] + 1
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
for _, decorative in pairs(source_surface.find_decoratives_filtered{area = source_area}) do
|
|
|
|
target_surface.create_decoratives{
|
|
|
|
check_collision = false,
|
|
|
|
decoratives = {{name = decorative.decorative.name, position = decorative.position, amount = decorative.amount}}
|
|
|
|
}
|
2019-03-19 04:05:32 +01:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
|
|
|
|
return true
|
2019-03-19 04:05:32 +01:00
|
|
|
end
|
|
|
|
|
2020-04-16 11:42:00 +02:00
|
|
|
local function mirror_chunk(chunk)
|
|
|
|
local target_surface = game.surfaces.biter_battles
|
|
|
|
|
|
|
|
local source_surface = game.surfaces.bb_source
|
|
|
|
local source_chunk_position = {chunk[1][1] * -1 - 1, chunk[1][2] * -1 - 1}
|
|
|
|
local source_left_top = {x = source_chunk_position[1] * 32, y = source_chunk_position[2] * 32}
|
|
|
|
local source_area = {{source_left_top.x, source_left_top.y}, {source_left_top.x + 32, source_left_top.y + 32}}
|
|
|
|
|
|
|
|
if not source_surface.is_chunk_generated(source_chunk_position) then
|
|
|
|
source_surface.request_to_generate_chunks({x = source_left_top.x + 16, y = source_left_top.y + 16}, 0)
|
|
|
|
return
|
2019-09-18 21:56:43 +02:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
|
|
|
|
if chunk[2] == 1 then
|
|
|
|
for _, tile in pairs(source_surface.find_tiles_filtered({area = source_area})) do
|
|
|
|
target_surface.set_tiles({{name = tile.name, position = {x = tile.position.x * -1, y = (tile.position.y * -1) - 1}}}, true)
|
|
|
|
end
|
|
|
|
chunk[2] = chunk[2] + 1
|
|
|
|
return
|
2019-09-18 21:56:43 +02:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
|
|
|
|
if chunk[2] == 2 then
|
|
|
|
for _, entity in pairs(source_surface.find_entities_filtered({area = source_area})) do
|
|
|
|
process_entity(target_surface, entity, "south")
|
|
|
|
end
|
|
|
|
chunk[2] = chunk[2] + 1
|
|
|
|
return
|
2019-09-18 21:56:43 +02:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
|
|
|
|
for _, decorative in pairs(source_surface.find_decoratives_filtered{area = source_area}) do
|
|
|
|
target_surface.create_decoratives{
|
|
|
|
check_collision = false,
|
|
|
|
decoratives = {{name = decorative.decorative.name, position = {x = decorative.position.x * -1, y = (decorative.position.y * -1) - 1}, amount = decorative.amount}}
|
2019-03-13 18:30:19 +01:00
|
|
|
}
|
2019-03-17 18:19:40 +01:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
|
2020-04-13 06:31:39 +02:00
|
|
|
return true
|
2019-09-18 21:56:43 +02:00
|
|
|
end
|
|
|
|
|
2020-04-16 11:42:00 +02:00
|
|
|
local function reveal_chunk(chunk)
|
|
|
|
local surface = game.surfaces.biter_battles
|
|
|
|
local chunk_position = chunk[1]
|
|
|
|
for _, force_name in pairs({"north", "south"}) do
|
|
|
|
local force = game.forces[force_name]
|
|
|
|
if force.is_chunk_charted(surface, chunk_position) then
|
|
|
|
force.chart(surface, {{chunk_position[1] * 32, chunk_position[2] * 32}, {chunk_position[1] * 32 + 31, chunk_position[2] * 32 + 31}})
|
|
|
|
end
|
|
|
|
end
|
2019-12-31 02:34:02 +01:00
|
|
|
end
|
|
|
|
|
2020-04-16 11:42:00 +02:00
|
|
|
function Public.add_chunk(event)
|
2020-04-13 06:31:39 +02:00
|
|
|
local surface = event.surface
|
|
|
|
if surface.name ~= "biter_battles" then return end
|
2020-04-16 11:42:00 +02:00
|
|
|
local left_top = event.area.left_top
|
|
|
|
local terrain_gen = global.terrain_gen
|
2020-04-13 06:31:39 +02:00
|
|
|
|
2020-04-16 11:42:00 +02:00
|
|
|
if left_top.y < 0 then
|
|
|
|
terrain_gen.size_of_chunk_copy = terrain_gen.size_of_chunk_copy + 1
|
|
|
|
terrain_gen.chunk_copy[terrain_gen.size_of_chunk_copy] = {{left_top.x / 32, left_top.y / 32}, 1}
|
|
|
|
else
|
|
|
|
terrain_gen.size_of_chunk_mirror = terrain_gen.size_of_chunk_mirror + 1
|
|
|
|
terrain_gen.chunk_mirror[terrain_gen.size_of_chunk_mirror] = {{left_top.x / 32, left_top.y / 32}, 1}
|
2019-09-22 01:52:40 +02:00
|
|
|
end
|
2019-09-18 21:56:43 +02:00
|
|
|
end
|
|
|
|
|
2020-01-22 08:30:41 +01:00
|
|
|
function Public.ticking_work()
|
2020-04-16 11:42:00 +02:00
|
|
|
local terrain_gen = global.terrain_gen
|
|
|
|
|
|
|
|
local tick = game.tick
|
|
|
|
|
|
|
|
if tick < 60 then return end
|
|
|
|
|
|
|
|
if game.tick % 2 == 1 then
|
|
|
|
for k, chunk in pairs(terrain_gen.chunk_copy) do
|
|
|
|
if copy_chunk(chunk) then
|
|
|
|
reveal_chunk(chunk)
|
|
|
|
table_remove(terrain_gen.chunk_copy, k)
|
|
|
|
terrain_gen.size_of_chunk_copy = terrain_gen.size_of_chunk_copy - 1
|
|
|
|
terrain_gen.counter = terrain_gen.counter + 1
|
2019-09-18 21:56:43 +02:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
break
|
2019-12-31 02:34:02 +01:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
else
|
|
|
|
for k, chunk in pairs(terrain_gen.chunk_mirror) do
|
|
|
|
if mirror_chunk(chunk) then
|
|
|
|
reveal_chunk(chunk)
|
|
|
|
table_remove(terrain_gen.chunk_mirror, k)
|
|
|
|
terrain_gen.size_of_chunk_mirror = terrain_gen.size_of_chunk_mirror - 1
|
|
|
|
terrain_gen.counter = terrain_gen.counter + 1
|
|
|
|
end
|
|
|
|
break
|
2019-12-31 02:34:02 +01:00
|
|
|
end
|
2019-09-18 21:56:43 +02:00
|
|
|
end
|
2020-04-16 11:42:00 +02:00
|
|
|
|
|
|
|
if terrain_gen.counter % 512 == 511 then
|
|
|
|
terrain_gen.counter = terrain_gen.counter + 1
|
|
|
|
local surface = game.surfaces.bb_source
|
|
|
|
local c = 0
|
|
|
|
for chunk in surface.get_chunks() do
|
|
|
|
surface.delete_chunk({chunk.x, chunk.y})
|
|
|
|
c = c + 1
|
|
|
|
end
|
|
|
|
print("Deleted " .. c .. " source surface chunks.")
|
2019-09-18 21:56:43 +02:00
|
|
|
end
|
2019-03-17 18:19:40 +01:00
|
|
|
end
|
|
|
|
|
2020-01-22 08:30:41 +01:00
|
|
|
return Public
|