local perlin = require 'map_gen.shared.perlin_noise' local Task = require 'utils.Task' local Event = require 'utils.event' 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 global.blocks = nil global.used_blocks = nil global.strike_time = strike_time global.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 global.blocks = blocks global.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) if not global.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 = global.blocks local count = global.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 global.weight_count = 0 else global.weight_count = count end local index = math.random(#blocks) return table.remove(blocks, index) end local function do_strike() local block = get_block() function add(x, y) local key = x .. ',' .. y if not global.used_blocks[key] then table.insert(global.blocks, {x = x, y = y}) global.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 = game.surfaces[1] surface.set_tiles(tiles, false) game.forces.player.chart(surface, {{bx, by}, {bx + block_size, by + block_size}}) end local function on_tick() if global.strike_time == 0 then do_strike() global.strike_time = strike_time else global.strike_time = global.strike_time - 1 end end Event.add(defines.events.on_tick, on_tick)