1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-14 02:34:09 +02:00
ComfyFactorio/maps/biter_battles_v2/mirror_terrain.lua
Gerkiz e91b6a352f 2.0 changes
Change global -> storage
Rework how rendering works
Game prototypes are now stored inside 'prototypes.#'
Renamed entity names
2024-10-22 21:47:11 +02:00

356 lines
14 KiB
Lua

local Public = {}
local Functions = require 'maps.biter_battles_v2.functions'
local bb_config = require 'maps.biter_battles_v2.config'
local table_remove = table.remove
local table_insert = table.insert
local direction_translation = {
[0] = 4,
[1] = 5,
[2] = 6,
[3] = 7,
[4] = 0,
[5] = 1,
[6] = 2,
[7] = 3
}
local cliff_orientation_translation = {
['east-to-none'] = 'west-to-none',
['east-to-north'] = 'west-to-south',
['east-to-south'] = 'west-to-north',
['east-to-west'] = 'west-to-east',
['north-to-east'] = 'south-to-west',
['north-to-none'] = 'south-to-none',
['north-to-south'] = 'south-to-north',
['north-to-west'] = 'south-to-east',
['south-to-east'] = 'north-to-west',
['south-to-none'] = 'north-to-none',
['south-to-north'] = 'north-to-south',
['south-to-west'] = 'north-to-east',
['west-to-east'] = 'east-to-west',
['west-to-none'] = 'east-to-none',
['west-to-north'] = 'east-to-south',
['west-to-south'] = 'east-to-north',
['none-to-east'] = 'none-to-west',
['none-to-north'] = 'none-to-south',
['none-to-south'] = 'none-to-north',
['none-to-west'] = 'none-to-east'
}
local entity_copy_functions = {
['tree'] = function (surface, entity, target_position)
if not surface.can_place_entity({ name = entity.name, position = target_position }) then
return
end
entity.clone({ position = target_position, surface = surface, force = 'neutral' })
end,
['simple-entity'] = function (surface, entity, target_position)
local mirror_entity = { name = entity.name, position = target_position, direction = direction_translation[entity.direction] }
if not surface.can_place_entity(mirror_entity) then
return
end
local mirror_entity2 = surface.create_entity(mirror_entity)
mirror_entity2.graphics_variation = entity.graphics_variation
end,
['cliff'] = function (surface, entity, target_position)
local mirror_entity = { name = entity.name, position = target_position, cliff_orientation = cliff_orientation_translation[entity.cliff_orientation] }
if not surface.can_place_entity(mirror_entity) then
return
end
surface.create_entity(mirror_entity)
return
end,
['resource'] = function (surface, entity, target_position)
surface.create_entity({ name = entity.name, position = target_position, amount = entity.amount })
end,
['corpse'] = function (surface, entity, target_position)
surface.create_entity({ name = entity.name, position = target_position })
end,
['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' }
if not surface.can_place_entity(mirror_entity) then
return
end
table_insert(storage.unit_spawners[force_name .. '_biters'], surface.create_entity(mirror_entity))
end,
['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' }
if not surface.can_place_entity(mirror_entity) then
return
end
surface.create_entity(mirror_entity)
end,
['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
storage.rocket_silo[force_name] =
surface.create_entity({ name = entity.name, position = target_position, direction = direction_translation[entity.direction], force = force_name })
storage.rocket_silo[force_name].minable = false
Functions.add_target_entity(storage.rocket_silo[force_name])
end,
['ammo-turret'] = function (surface, entity, target_position, force_name)
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)
Functions.add_target_entity(e)
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
end,
['wall'] = function (surface, entity, target_position, force_name)
local e = entity.clone({ position = target_position, surface = surface, force = force_name })
e.active = true
end,
['container'] = function (surface, entity, target_position, force_name)
local e = entity.clone({ position = target_position, surface = surface, force = force_name })
e.active = true
end,
['fish'] = function (surface, entity, target_position)
local mirror_entity = { name = entity.name, position = target_position }
if not surface.can_place_entity(mirror_entity) then
return
end
surface.create_entity(mirror_entity)
end
}
local function process_entity(surface, entity, force_name)
if not entity.valid then
return
end
if not entity_copy_functions[entity.type] then
return
end
local match_mirror = bb_config.match_mirror
local target_position
if force_name == 'north' then
target_position = entity.position
else
if match_mirror then
target_position = { x = entity.position.x, y = entity.position.y * -1 }
else
target_position = { x = entity.position.x * -1, y = entity.position.y * -1 }
end
end
entity_copy_functions[entity.type](surface, entity, target_position, force_name)
end
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
end
local match_mirror = bb_config.match_mirror
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')
end
chunk[2] = chunk[2] + 1
return
end
local decoratives = {}
if match_mirror then
for k, decorative in pairs(source_surface.find_decoratives_filtered { area = source_area }) do
decoratives[k] = { name = decorative.decorative.name, position = { decorative.position.x, decorative.position.y }, amount = decorative.amount }
end
else
for k, decorative in pairs(source_surface.find_decoratives_filtered { area = source_area }) do
decoratives[k] = { name = decorative.decorative.name, position = { (decorative.position.x * -1) - 1, (decorative.position.y * -1) - 1 }, amount = decorative.amount }
end
end
target_surface.create_decoratives({ check_collision = false, decoratives = decoratives })
return true
end
local function mirror_chunk(chunk)
local target_surface = game.surfaces.biter_battles
local match_mirror = bb_config.match_mirror
local source_surface = game.surfaces.bb_source
local source_chunk_position
if match_mirror then
source_chunk_position = { chunk[1][1], chunk[1][2] * -1 - 1 }
else
source_chunk_position = { chunk[1][1] * -1 - 1, chunk[1][2] * -1 - 1 }
end
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
end
if chunk[2] == 1 then
local tiles = {}
if match_mirror then
for k, tile in pairs(source_surface.find_tiles_filtered({ area = source_area })) do
tiles[k] = { name = tile.name, position = { tile.position.x, tile.position.y * -1 - 1 } }
end
else
for k, tile in pairs(source_surface.find_tiles_filtered({ area = source_area })) do
tiles[k] = { name = tile.name, position = { tile.position.x * -1 - 1, tile.position.y * -1 - 1 } }
end
end
target_surface.set_tiles(tiles, true)
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, 'south')
end
chunk[2] = chunk[2] + 1
return
end
local decoratives = {}
for k, decorative in pairs(source_surface.find_decoratives_filtered { area = source_area }) do
decoratives[k] = { name = decorative.decorative.name, position = { decorative.position.x, decorative.position.y * -1 }, amount = decorative.amount }
end
target_surface.create_decoratives({ check_collision = false, decoratives = decoratives })
return true
end
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
end
function Public.add_chunk(event)
local surface = event.surface
if surface.name ~= 'biter_battles' then
return
end
local left_top = event.area.left_top
local terrain_gen = storage.terrain_gen
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 }
end
end
local function clear_source_surface(terrain_gen)
if terrain_gen.counter % 1024 == 1023 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.')
end
end
local function north_work()
local terrain_gen = storage.terrain_gen
--luacheck: ignore 512
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
end
break
end
clear_source_surface(terrain_gen)
end
local function south_work()
local terrain_gen = storage.terrain_gen
--luacheck: ignore 512
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
end
clear_source_surface(terrain_gen)
end
local works = {
[1] = north_work,
[2] = south_work
}
function Public.ticking_work()
local tick = game.ticks_played
if tick < 4 then
return
end
local work = works[tick % 2 + 1]
if storage.server_restart_timer then
return
end
work()
end
return Public