1
0
mirror of https://github.com/Refactorio/RedMew.git synced 2025-01-03 22:52:13 +02:00
RedMew/map_gen/presets/spiral_of_spirals.lua
2018-07-10 13:37:45 +01:00

180 lines
4.4 KiB
Lua

local b = require 'map_gen.shared.builders'
local Random = require 'map_gen.shared.random'
local ore_seed1 = 2000
local ore_seed2 = ore_seed1 * 2
local random = Random.new(ore_seed1, ore_seed2)
local spiral = b.rectangular_spiral(1)
local factor = 9
local map = b.single_spiral_rotate_pattern(spiral, factor)
map = b.single_spiral_rotate_pattern(map, factor * factor)
map = b.scale(map, 64)
local function value(base, mult, pow)
return function(x, y)
local d = math.sqrt(x * x + y * y)
return base + mult * d ^ pow
end
end
local patch = b.rectangular_spiral(5)
local bounds = b.rectangle(46, 43)
bounds = b.translate(bounds, 0, 3)
patch = b.all {bounds, patch}
local patch_value = value(600, 0.75, 1.2)
local ores = {
{resource = b.resource(b.full_shape, 'iron-ore', patch_value), weight = 6},
{resource = b.resource(b.full_shape, 'copper-ore', patch_value), weight = 4},
{resource = b.resource(b.full_shape, 'stone', patch_value), weight = 1},
{resource = b.resource(b.full_shape, 'coal', patch_value), weight = 1}
}
local total_ore_weights = {}
local ore_t = 0
for _, v in ipairs(ores) do
ore_t = ore_t + v.weight
table.insert(total_ore_weights, ore_t)
end
local function do_ore(x, y, world)
if not patch(x, y) then
return nil
end
local i = math.random(ore_t)
local index = table.binary_search(total_ore_weights, i)
if (index < 0) then
index = bit32.bnot(index)
end
return ores[index].resource(x, y, world)
end
local small_patch = b.rectangular_spiral(5)
local small_bounds = b.rectangle(25, 33)
small_bounds = b.translate(small_bounds, -1, -2)
small_patch = b.all {small_bounds, small_patch}
local oil_patch = b.throttle_world_xy(small_patch, 1, 5, 1, 5)
local resources = {
{resource = do_ore, weight = 16},
{resource = b.resource(oil_patch, 'crude-oil', value(75000, 50, 1.025)), weight = 5},
{resource = b.resource(small_patch, 'uranium-ore', value(200, 0.3, 1.025)), weight = 2},
{resource = b.no_entity, weight = 130}
}
local total_resource_weights = {}
local resource_t = 0
for _, v in ipairs(resources) do
resource_t = resource_t + v.weight
table.insert(total_resource_weights, resource_t)
end
local resource_pattern = {}
for r = 1, 50 do
local row = {}
resource_pattern[r] = row
for c = 1, 50 do
local i = random:next_int(1, resource_t)
local index = table.binary_search(total_resource_weights, i)
if (index < 0) then
index = bit32.bnot(index)
end
row[c] = resources[index].resource
end
end
local resources_shape = b.grid_pattern(resource_pattern, 50, 50, 64, 64)
local start_stone =
b.resource(
patch,
'stone',
function()
return 600
end
)
local start_coal =
b.resource(
patch,
'coal',
function()
return 2400
end
)
local start_copper =
b.resource(
patch,
'copper-ore',
function()
return 800
end
)
local start_iron =
b.resource(
patch,
'iron-ore',
function()
return 1600
end
)
local start_spiral = b.segment_pattern({start_iron, start_copper, start_coal, start_stone})
start_spiral = b.translate(start_spiral, 0, -5)
local start_bounds = b.rectangle(64)
resources_shape = b.choose(start_bounds, start_spiral, resources_shape)
local worm_names = {'small-worm-turret', 'medium-worm-turret', 'big-worm-turret'}
local safe_d = 300
local half_spawn_d = 100000 -- distance at which there is a half chance of a worm spawning
local max_spawn_rate = 1 / 18
local level_factor = 32 -- higher factor -> more likly to spawn higher level worms
local min_big_worm_d = 900
local hd = 1 / (2 * half_spawn_d)
local inv_level_factor = 1 / level_factor
local function worms(_, _, world)
local x, y = world.x, world.y
local d = math.sqrt(x * x + y * y)
d = d - safe_d
if d <= 0 then
return nil
end
local chance = d * hd
if math.random() > math.min(chance, max_spawn_rate) then
return nil
end
local lf = inv_level_factor / chance
local lvl = math.floor(math.random() ^ lf * 3) + 1
if d < min_big_worm_d and lvl == 3 then
lvl = 2
end
return {name = worm_names[lvl]}
end
map = b.apply_entity(map, resources_shape)
map = b.apply_entity(map, worms)
map = b.change_map_gen_collision_tile(map, 'water-tile', 'grass-1')
map = b.change_tile(map, false, 'water')
map = b.fish(map, 0.0025)
return map