mirror of
https://github.com/Refactorio/RedMew.git
synced 2024-12-12 10:04:40 +02:00
26e1c28dc0
* Init Factorio 2.0 update * add credits * fix test module * I know luackeck, I know * Fixes * Fix bad event.player_index handling * Hotfixes * Remove all filter inserters * Migrate removed items * Deprecating spidertron control and landfill features
172 lines
4.5 KiB
Lua
172 lines
4.5 KiB
Lua
local perlin = require 'map_gen.shared.perlin_noise'
|
|
local Event = require 'utils.event'
|
|
local RS = require 'map_gen.shared.redmew_surface'
|
|
|
|
local block_size = 1 -- in tiles
|
|
local start_size = 64 -- in blocks
|
|
local strike_time = 1 -- in ticks
|
|
|
|
-- dud blocks don't spawn meteors, with a block_weight = 1 and dud weight = 3, every 3 out of 4 blocks will be a dud block
|
|
local block_weight = 1
|
|
local dud_weight = 0
|
|
local min_blocks_in_list = 10 -- no dud meteors if the number of blocks in the list is less than or equal to this
|
|
|
|
storage.blocks = nil
|
|
storage.used_blocks = nil
|
|
storage.strike_time = strike_time
|
|
storage.weight_count = 0
|
|
|
|
local half_start_size = (start_size * block_size) / 2
|
|
local total_weight = block_weight + dud_weight
|
|
|
|
local function init_blocks()
|
|
local blocks = {}
|
|
local used_blocks = {}
|
|
local half = start_size / 2
|
|
for i = -half, half - 1 do
|
|
table.insert(blocks, {x = i, y = -half - 1})
|
|
used_blocks[i .. ',' .. (-half - 1)] = true
|
|
table.insert(blocks, {x = i, y = half})
|
|
used_blocks[i .. ',' .. half] = true
|
|
table.insert(blocks, {x = -half - 1, y = i})
|
|
used_blocks[(-half - 1) .. ',' .. i] = true
|
|
table.insert(blocks, {x = half, y = i})
|
|
used_blocks[half .. ',' .. i] = true
|
|
|
|
for j = -half, half - 1 do
|
|
used_blocks[i .. ',' .. j] = true
|
|
end
|
|
end
|
|
|
|
storage.blocks = blocks
|
|
storage.used_blocks = used_blocks
|
|
end
|
|
|
|
local function get_resource(x, y)
|
|
local value = perlin.noise(x / 16, y / 16)
|
|
value = value + 1
|
|
value = value * 500
|
|
|
|
local name
|
|
|
|
if value < 450 then
|
|
return nil
|
|
elseif value < 550 then
|
|
name = 'iron-ore'
|
|
elseif value < 650 then
|
|
name = 'copper-ore'
|
|
elseif value < 750 then
|
|
name = 'coal'
|
|
elseif value < 850 then
|
|
name = 'stone'
|
|
else
|
|
return nil
|
|
end
|
|
|
|
value = perlin.noise(y / 64, x / 64)
|
|
value = value + 1
|
|
value = value * 500
|
|
|
|
return {name = name, position = {x, y}, amount = value}
|
|
end
|
|
|
|
function run_combined_module(event) -- luacheck: globals run_combined_module
|
|
if not storage.blocks then
|
|
init_blocks()
|
|
end
|
|
|
|
local area = event.area
|
|
local surface = event.surface
|
|
local top_x = area.left_top.x
|
|
local top_y = area.left_top.y
|
|
|
|
local tiles = {}
|
|
local entities = {}
|
|
|
|
for y = top_y, top_y + 31 do
|
|
for x = top_x, top_x + 31 do
|
|
if -x > half_start_size or x >= half_start_size or -y > half_start_size or y >= half_start_size then
|
|
table.insert(tiles, {name = 'out-of-map', position = {x, y}})
|
|
end
|
|
|
|
local e = get_resource(x, y)
|
|
if e then
|
|
table.insert(entities, e)
|
|
end
|
|
end
|
|
end
|
|
|
|
surface.set_tiles(tiles, false)
|
|
|
|
for _, e in ipairs(entities) do
|
|
if surface.can_place_entity(e) then
|
|
surface.create_entity(e)
|
|
end
|
|
end
|
|
end
|
|
|
|
local function get_block()
|
|
local blocks = storage.blocks
|
|
local count = storage.weight_count
|
|
while count >= block_weight and count < total_weight and #blocks > min_blocks_in_list do
|
|
local index = math.random(#blocks)
|
|
table.remove(blocks, index)
|
|
count = count + 1
|
|
end
|
|
|
|
if count < block_weight then
|
|
count = count + 1
|
|
end
|
|
|
|
if count == total_weight then
|
|
storage.weight_count = 0
|
|
else
|
|
storage.weight_count = count
|
|
end
|
|
|
|
local index = math.random(#blocks)
|
|
return table.remove(blocks, index)
|
|
end
|
|
|
|
local function do_strike()
|
|
local block = get_block()
|
|
|
|
local function add(x, y)
|
|
local key = x .. ',' .. y
|
|
if not storage.used_blocks[key] then
|
|
table.insert(storage.blocks, {x = x, y = y})
|
|
storage.used_blocks[key] = true
|
|
end
|
|
end
|
|
|
|
add(block.x, block.y - 1)
|
|
add(block.x + 1, block.y)
|
|
add(block.x, block.y + 1)
|
|
add(block.x - 1, block.y)
|
|
|
|
local tiles = {}
|
|
local bx = block.x * block_size
|
|
local by = block.y * block_size
|
|
|
|
for x = bx, bx + block_size - 1 do
|
|
for y = by, by + block_size - 1 do
|
|
table.insert(tiles, {name = 'dry-dirt', position = {x, y}})
|
|
end
|
|
end
|
|
local surface = RS.get_surface()
|
|
surface.set_tiles(tiles, false)
|
|
|
|
game.forces.player.chart(surface, {{bx, by}, {bx + block_size, by + block_size}})
|
|
end
|
|
|
|
local function on_tick()
|
|
if storage.strike_time == 0 then
|
|
do_strike()
|
|
storage.strike_time = strike_time
|
|
else
|
|
storage.strike_time = storage.strike_time - 1
|
|
end
|
|
end
|
|
|
|
Event.add(defines.events.on_tick, on_tick)
|