2019-11-14 16:44:08 +01:00
|
|
|
local Public = {}
|
|
|
|
local simplex_noise = require "utils.simplex_noise".d2
|
|
|
|
local math_random = math.random
|
|
|
|
local math_abs = math.abs
|
|
|
|
local math_sqrt = math.sqrt
|
2019-11-14 17:54:28 +01:00
|
|
|
local level_depth = require "maps.mountain_fortress_v2.terrain"
|
2019-11-14 16:44:08 +01:00
|
|
|
local math_floor = math.floor
|
|
|
|
local table_remove = table.remove
|
|
|
|
local table_insert = table.insert
|
2019-11-14 18:06:59 +01:00
|
|
|
local chart_radius = 30
|
2019-11-14 17:54:28 +01:00
|
|
|
local tile_conversion = {
|
|
|
|
["concrete"] = "stone-path",
|
|
|
|
["hazard-concrete-left"] = "stone-path",
|
|
|
|
["hazard-concrete-right"] = "stone-path",
|
|
|
|
["refined-concrete"] = "concrete",
|
|
|
|
["refined-hazard-concrete-left"] = "hazard-concrete-left",
|
|
|
|
["refined-hazard-concrete-right"] = "hazard-concrete-right",
|
|
|
|
["stone-path"] = "landfill",
|
|
|
|
}
|
2019-11-14 16:44:08 +01:00
|
|
|
|
|
|
|
local function get_collapse_vectors(radius)
|
|
|
|
local vectors = {}
|
|
|
|
local i = 1
|
|
|
|
local m = 1 / (radius * 2)
|
|
|
|
local seed = math_random(1, 9999999)
|
|
|
|
for x = radius * -1, radius, 1 do
|
|
|
|
for y = radius * -1, radius, 1 do
|
2019-11-14 17:54:28 +01:00
|
|
|
local noise = math_abs(simplex_noise(x * m, y * m, seed) * radius * 1.3)
|
2019-11-14 16:44:08 +01:00
|
|
|
local d = math_sqrt(x ^ 2 + y ^ 2)
|
|
|
|
if d + noise < radius then
|
|
|
|
vectors[i] = {x, y}
|
|
|
|
i = i + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return vectors
|
|
|
|
end
|
|
|
|
|
2019-11-14 17:54:28 +01:00
|
|
|
local function set_x_positions()
|
|
|
|
local x_positions = global.map_collapse.x_positions
|
2019-11-14 18:06:59 +01:00
|
|
|
for x = level_depth * -1, level_depth - 1, 1 do
|
2019-11-14 17:54:28 +01:00
|
|
|
table_insert(x_positions, x)
|
|
|
|
end
|
|
|
|
table.shuffle_table(x_positions)
|
|
|
|
end
|
|
|
|
|
|
|
|
local function get_position(surface)
|
|
|
|
local x_positions = global.map_collapse.x_positions
|
|
|
|
if #x_positions == 0 then set_x_positions() end
|
|
|
|
local x = x_positions[1]
|
|
|
|
local position = false
|
2019-11-14 18:06:59 +01:00
|
|
|
for y = 160, -100000, -1 do
|
|
|
|
y = y + math_random(1, 16)
|
2019-11-14 17:54:28 +01:00
|
|
|
local tile = surface.get_tile({x, y})
|
|
|
|
if tile.valid then
|
|
|
|
if tile.name ~= "out-of-map" then
|
|
|
|
position = {x = x, y = y}
|
|
|
|
break
|
|
|
|
end
|
|
|
|
else
|
|
|
|
y = y + 96
|
|
|
|
end
|
|
|
|
end
|
|
|
|
table_remove(x_positions, 1)
|
2019-11-14 16:44:08 +01:00
|
|
|
return position
|
|
|
|
end
|
|
|
|
|
|
|
|
local function sort_list_by_distance(center_position, list)
|
|
|
|
local sorted_list = {}
|
|
|
|
for _, item in pairs(list) do
|
|
|
|
local d = math_floor(math_sqrt((item.position.x - center_position.x)^2 + (item.position.y - center_position.y)^2)) + 1
|
|
|
|
if not sorted_list[d] then sorted_list[d] = {} end
|
|
|
|
sorted_list[d][#sorted_list[d] + 1] = item
|
|
|
|
end
|
|
|
|
local final_list = {}
|
|
|
|
for _, row in pairs(sorted_list) do
|
|
|
|
table.shuffle_table(row)
|
|
|
|
for _, tile in pairs(row) do
|
|
|
|
table_insert(final_list, tile)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return final_list
|
|
|
|
end
|
|
|
|
|
|
|
|
local function set_collapse_tiles(surface, position, vectors)
|
|
|
|
local tiles = {}
|
|
|
|
local i = 1
|
|
|
|
for _, vector in pairs(vectors) do
|
|
|
|
local position = {x = position.x + vector[1], y = position.y + vector[2]}
|
|
|
|
local tile = surface.get_tile(position)
|
2019-11-14 17:54:28 +01:00
|
|
|
if tile.valid then
|
2019-11-14 16:44:08 +01:00
|
|
|
tiles[i] = tile
|
|
|
|
i = i + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
local sorted_tiles = sort_list_by_distance(position, tiles)
|
|
|
|
table_insert(global.map_collapse.processing, sorted_tiles)
|
|
|
|
end
|
|
|
|
|
2019-11-14 17:54:28 +01:00
|
|
|
local function collapse_map()
|
|
|
|
local surface = game.surfaces[global.active_surface_index]
|
|
|
|
local vectors = get_collapse_vectors(math_random(8, 24))
|
|
|
|
local position = get_position(surface)
|
2019-11-14 18:06:59 +01:00
|
|
|
if not position then return end
|
|
|
|
|
|
|
|
local last_position = global.map_collapse.last_position
|
|
|
|
game.forces.player.chart(surface, {{last_position.x - chart_radius, last_position.y - chart_radius},{last_position.x + chart_radius, last_position.y + chart_radius}})
|
|
|
|
global.map_collapse.last_position = {x = position.x, y = position.y}
|
|
|
|
|
|
|
|
game.forces.player.chart(surface, {{position.x - chart_radius, position.y - chart_radius},{position.x + chart_radius, position.y + chart_radius}})
|
|
|
|
|
2019-11-14 17:54:28 +01:00
|
|
|
set_collapse_tiles(surface, position, vectors)
|
|
|
|
end
|
|
|
|
|
2019-11-14 16:44:08 +01:00
|
|
|
function Public.process()
|
|
|
|
local processing = global.map_collapse.processing
|
2019-11-14 17:54:28 +01:00
|
|
|
if #processing == 0 then collapse_map() return end
|
2019-11-14 16:44:08 +01:00
|
|
|
local surface = game.surfaces[global.active_surface_index]
|
|
|
|
for k1, tile_set in pairs(processing) do
|
|
|
|
for k2, tile in pairs(tile_set) do
|
2019-11-14 17:54:28 +01:00
|
|
|
local conversion_tile = tile_conversion[tile.name]
|
|
|
|
if conversion_tile then
|
|
|
|
surface.set_tiles({{name = conversion_tile, position = tile.position}}, true)
|
|
|
|
surface.create_trivial_smoke({name="train-smoke", position = tile.position})
|
|
|
|
else
|
|
|
|
surface.set_tiles({{name = "out-of-map", position = tile.position}}, true)
|
|
|
|
end
|
2019-11-14 16:44:08 +01:00
|
|
|
table_remove(tile_set, k2)
|
|
|
|
break
|
|
|
|
end
|
|
|
|
if #tile_set == 0 then table_remove(processing, k1) end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function Public.init()
|
|
|
|
global.map_collapse = {}
|
2019-11-14 17:54:28 +01:00
|
|
|
global.map_collapse.x_positions = {}
|
2019-11-14 16:44:08 +01:00
|
|
|
global.map_collapse.processing = {}
|
2019-11-14 18:06:59 +01:00
|
|
|
global.map_collapse.last_position = {x = 0, y = 0}
|
2019-11-14 16:44:08 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
local event = require 'utils.event'
|
|
|
|
event.on_init(Public.init())
|
|
|
|
|
|
|
|
return Public
|