1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-24 03:47:58 +02:00
2022-06-01 19:50:36 +01:00

393 lines
13 KiB
Lua

-- This file is part of thesixthroc's Pirate Ship softmod, licensed under GPLv3 and stored at https://github.com/danielmartin0/ComfyFactorio-Pirates.
local Memory = require 'maps.pirates.memory'
local Math = require 'maps.pirates.math'
-- local Balance = require 'maps.pirates.balance'
local Structures = require 'maps.pirates.structures.structures'
local Common = require 'maps.pirates.common'
-- local Utils = require 'maps.pirates.utils_local'
-- local Ores = require 'maps.pirates.ores'
local _inspect = require 'utils.inspect'.inspect
local Public = {}
local enum = {
STANDARD = '1',
FIRST = '2',
WALKWAYS = '3',
RED_DESERT = '4',
RADIOACTIVE = '5',
STANDARD_VARIANT = '6',
HORSESHOE = '7',
SWAMP = '8',
MAZE = '9',
}
Public.enum = enum
function Public.place_water_tile(args)
if args.static_params and args.static_params.deepwater_terraingenframe_xposition and args.p.x <= args.static_params.deepwater_terraingenframe_xposition - 0.5
then
args.tiles[#args.tiles + 1] = {name = 'deepwater', position = args.p}
local fishrng = Math.random(350)
if fishrng == 350 then
args.entities[#args.entities + 1] = {name = 'fish', position = args.p}
end
return true
end
if not args.noise_generator['height'] then return end
local height_noise = args.noise_generator['height'](args.p)
if height_noise < 0 then
args.tiles[#args.tiles + 1] = {name = 'water', position = args.p}
local fishrng = Math.random(350)
if fishrng == 350 then
args.entities[#args.entities + 1] = {name = 'fish', position = args.p}
end
return true
end
return false
end
function Public.island_height_1(args)
local noise_name = 'height'
if not args.noise_generator[noise_name] then
args.noise_generator:addNoise(noise_name,
function(p)
local r2 = (p.x)^2 + (p.y)^2
local r = Math.sqrt(r2)
-- 'noise testing suite':
-- local height_noise
-- if args.noise_generator.forest then
-- height_noise = args.noise_generator.forest(p)
-- else return 0 end
-- local height_noise = args.noise_generator.height_background(p)
-- local height_noise = (
-- 1 - r/(args.noise_generator.radius{x = p.x/r, y = p.y/r})
-- )
local height_noise = (
1 - r/(args.noise_generator.radius{x = p.x/r, y = p.y/r})
) + args.noise_generator.height_background(p)
return height_noise
end)
end
return args.noise_generator[noise_name]
end
function Public.island_height_mostly_circular(args)
local noise_name = 'height'
if not args.noise_generator[noise_name] then
args.noise_generator:addNoise(noise_name,
function(p)
local r2 = (p.x)^2 + (p.y)^2
local r = Math.sqrt(r2)
-- 'noise testing suite':
-- local height_noise
-- if args.noise_generator.forest then
-- height_noise = args.noise_generator.forest(p)
-- else return 0 end
-- local height_noise = args.noise_generator.height_background(p)
-- local height_noise = (
-- 1 - r/(args.noise_generator.radius{x = p.x/r, y = p.y/r})
-- )
local height_noise = (
1 - r/(args.noise_generator.radius{x = p.x/r, y = p.y/r})
)
return height_noise
end)
end
return args.noise_generator[noise_name]
end
function Public.island_height_horseshoe(args)
local noise_name = 'height'
if not args.noise_generator[noise_name] then
args.noise_generator:addNoise(noise_name,
function(p)
local r12 = (p.x)^2 + (p.y)^2
local r1 = Math.sqrt(r12)
local offsetp = {x = p.x + 80, y = p.y}
local r22 = (offsetp.x)^2 + (offsetp.y)^2
local r2 = Math.sqrt(r22)
-- 'noise testing suite':
-- local height_noise
-- if args.noise_generator.forest then
-- height_noise = args.noise_generator.forest(p)
-- else return 0 end
-- local height_noise = args.noise_generator.height_background(p)
-- local height_noise = (
-- 1 - r/(args.noise_generator.radius{x = p.x/r, y = p.y/r})
-- )
local height_noise = (
1 - r1/(args.noise_generator.radius1{x = p.x/r1, y = p.y/r1})
) - Math.max(0,
1 - r2/(args.noise_generator.radius2{x = offsetp.x/r2, y = offsetp.y/r2})
) + args.noise_generator.height_background(p)
return height_noise
end)
end
return args.noise_generator[noise_name]
end
function Public.island_farness_1(args)
--on a scale from 0 to 1, how 'far' the point is from the boat dropoff point
if not args.static_params.width and args.static_params.islandcenter_position and args.static_params.terraingen_coordinates_offset then return end -- can only call after detailed static_params are generated
local noise_name = 'farness' --might as well remember as a noise just to memoize
if not args.noise_generator[noise_name] then
args.noise_generator:addNoise(noise_name,
function(p)
local island_width = args.static_params.width - 2*Math.abs(args.static_params.islandcenter_position.x)
local nexus_of_boredom = {x = args.static_params.terraingen_coordinates_offset.x + args.static_params.islandcenter_position.x - 2/5*island_width, y = args.static_params.terraingen_coordinates_offset.y + args.static_params.islandcenter_position.y}
local relativeradius2 = Math.distance(p, nexus_of_boredom)
local farness = Math.slopefromto(relativeradius2, island_width/12, island_width)
if p.x < nexus_of_boredom.x then
local num = Math.abs(nexus_of_boredom.y - p.y)
local denom = Math.abs(nexus_of_boredom.x - p.x)
if denom < 1 then denom = 1 end
farness = farness * Math.slopefromto(num/denom, 1, 5)
end
return farness
end)
end
return args.noise_generator[noise_name]
end
function Public.island_farness_horseshoe(args)
--on a scale from 0 to 1, how 'far' the point is from the boat dropoff point
--compared to first farness function this one is much more simply just distance from boat
if not args.static_params.width and args.static_params.islandcenter_position and args.static_params.terraingen_coordinates_offset then return end -- can only call after detailed static_params are generated
local noise_name = 'farness' --might as well remember as a noise just to memoize
if not args.noise_generator[noise_name] then
args.noise_generator:addNoise(noise_name,
function(p)
local island_width = args.static_params.width - 2*Math.abs(args.static_params.islandcenter_position.x)
local nexus_of_boredom = {x = args.static_params.terraingen_coordinates_offset.x + args.static_params.islandcenter_position.x - 1/6*island_width, y = args.static_params.terraingen_coordinates_offset.y + args.static_params.islandcenter_position.y}
local relativeradius2 = Math.distance(p, nexus_of_boredom)
local farness = Math.slopefromto(relativeradius2, island_width/12, 62/100*island_width)
return farness
end)
end
return args.noise_generator[noise_name]
end
function Public.enemies_1(args, spec, no_worms, worm_evo_bonus)
worm_evo_bonus = worm_evo_bonus or 0
for x = args.left_top.x, args.left_top.x + 31 do
for y = args.left_top.y, args.left_top.y + 31 do
local p = {x = x, y = y}
local spec2 = spec(p)
if spec2.placeable and Math.random() < spec2.density_perchunk/(32*32) then
local memory = Memory.get_crew_memory()
local enemy_force_name = memory.enemy_force_name
local rng = Math.random(10)
if rng >= 4 then
args.entities[#args.entities + 1] = {name = 'biter-spawner', position = p, force = enemy_force_name, indestructible = spec2.spawners_indestructible or false}
elseif rng >= 3 then
args.entities[#args.entities + 1] = {name = 'spitter-spawner', position = p, force = enemy_force_name, indestructible = spec2.spawners_indestructible or false}
elseif not no_worms then
local evolution = memory.evolution_factor + worm_evo_bonus
args.entities[#args.entities + 1] = {name = Common.get_random_worm_type(evolution + 0.05), position = p, force = enemy_force_name}
end
end
end
end
end
function Public.enemies_specworms_separate(args, spec)
for x = args.left_top.x, args.left_top.x + 31 do
for y = args.left_top.y, args.left_top.y + 31 do
local p = {x = x, y = y}
local spec2 = spec(p)
if spec2.placeable and Math.random() < spec2.spawners_density_perchunk/(32*32) then
local memory = Memory.get_crew_memory()
local enemy_force_name = memory.enemy_force_name
local rng = Math.random(10)
if rng >=8 then
args.entities[#args.entities + 1] = {name = 'spitter-spawner', position = p, force = enemy_force_name, indestructible = spec2.spawners_indestructible or false}
else
args.entities[#args.entities + 1] = {name = 'biter-spawner', position = p, force = enemy_force_name, indestructible = spec2.spawners_indestructible or false}
end
elseif spec2.placeable and Math.random() < spec2.worms_density_perchunk/(32*32) then
local memory = Memory.get_crew_memory()
local enemy_force_name = memory.enemy_force_name
local evolution = game.forces[enemy_force_name].evolution_factor
args.entities[#args.entities + 1] = {name = Common.get_random_worm_type(evolution + 0.05), position = p, force = enemy_force_name}
end
end
end
end
function Public.assorted_structures_1(args, spec)
local memory = Memory.get_crew_memory()
local overworldx = memory.overworldx or 0
local rng = Math.random()
local left_top = args.left_top
-- initial attempt, to avoid placing two structures too close to each other, is to divide up the map into 2x2 chonks, and spawn once in each
local bool1, bool2 = left_top.x % 64 < 32, left_top.y % 64 < 32
local all_four_chunks = {
{x = left_top.x, y = left_top.y},
{x = left_top.x + (bool1 and 32 or -32), y = left_top.y},
{x = left_top.x, y = left_top.y + (bool2 and 32 or -32)},
{x = left_top.x + (bool1 and 32 or -32), y = left_top.y + (bool2 and 32 or -32)},
}
if not args.other_map_generation_data.chunks_loaded then args.other_map_generation_data.chunks_loaded = {} end
local chunks_loaded = args.other_map_generation_data.chunks_loaded
if not chunks_loaded[args.left_top.x] then chunks_loaded[args.left_top.x] = {} end
chunks_loaded[args.left_top.x][args.left_top.y] = true
local nearby_chunks_generated_count = 0
for i=1,4 do
if chunks_loaded[all_four_chunks[i].x] and chunks_loaded[all_four_chunks[i].x][all_four_chunks[i].y] then
nearby_chunks_generated_count = nearby_chunks_generated_count + 1
end
end
if nearby_chunks_generated_count == 4 then --should trigger only once per 4 chunks
local avgleft_top = {
x = (all_four_chunks[1].x + all_four_chunks[4].x)/2,
y = (all_four_chunks[1].y + all_four_chunks[4].y)/2,
}
local leftmost_topmost = {
x = avgleft_top.x - 16,
y = avgleft_top.y - 16,
}
local spec2 = spec{x = avgleft_top.x + 16, y = avgleft_top.y + 16}
if rng < spec2.chanceper4chunks then
local rng2 = Math.random()
local struct
if overworldx <= 120 then
if rng2 < 20/100 then
struct = Structures.IslandStructures.ROC.lonely_storage_tank
elseif rng2 < 40/100 then
struct = Structures.IslandStructures.MATTISSO.small_crashed_ship
elseif rng2 < 50/100 then
struct = Structures.IslandStructures.MATTISSO.small_oilrig_base
elseif rng2 < 60/100 then
struct = Structures.IslandStructures.MATTISSO.small_abandoned_refinery
elseif rng2 < 70/100 then
struct = Structures.IslandStructures.MATTISSO.small_mining_base
else
struct = Structures.IslandStructures.MATTISSO.small_primitive_mining_base
end
elseif overworldx <= 240 then
if rng2 < 30/100 then
struct = Structures.IslandStructures.ROC.lonely_storage_tank
elseif rng2 < 40/100 then
struct = Structures.IslandStructures.MATTISSO.small_crashed_ship
elseif rng2 < 50/100 then
struct = Structures.IslandStructures.MATTISSO.small_oilrig_base
elseif rng2 < 70/100 then
struct = Structures.IslandStructures.MATTISSO.small_abandoned_refinery
elseif rng2 < 80/100 then
struct = Structures.IslandStructures.MATTISSO.small_mining_base
else
struct = Structures.IslandStructures.MATTISSO.small_solar_base
end
else
if rng2 < 10/100 then
struct = Structures.IslandStructures.ROC.lonely_storage_tank
elseif rng2 < 20/100 then
struct = Structures.IslandStructures.MATTISSO.small_crashed_ship
elseif rng2 < 40/100 then
struct = Structures.IslandStructures.MATTISSO.small_oilrig_base
elseif rng2 < 50/100 then
struct = Structures.IslandStructures.MATTISSO.small_abandoned_refinery
elseif rng2 < 60/100 then
struct = Structures.IslandStructures.MATTISSO.small_mining_base
elseif rng2 < 80/100 then
struct = Structures.IslandStructures.MATTISSO.small_solar_base
else
struct = Structures.IslandStructures.MATTISSO.small_roboport_base
end
end
if struct then
Structures.try_place(struct, args.specials, leftmost_topmost, 64, 64, function(p) return spec(p).placeable end)
end
end
end
end
function Public.random_rock_1(p)
local rock_raffle = {'sand-rock-big','sand-rock-big','rock-big','rock-big','rock-big','rock-big','rock-huge','rock-huge'}
local s_rock_raffle = #rock_raffle
return {name = rock_raffle[Math.random(1, s_rock_raffle)], position = p}
end
function Public.random_tree_1(p)
local tree_raffle = {
'tree-01',
'tree-02',
'tree-02-red',
'tree-03',
'tree-04',
'tree-05',
'tree-06',
'tree-06-brown',
'tree-07',
'tree-08',
'tree-08-brown',
'tree-08-red',
'tree-09',
'tree-09-brown',
'tree-09-red'
}
local s_tree_raffle = #tree_raffle
return {name = tree_raffle[Math.random(1, s_tree_raffle)], position = p}
end
return Public