1
0
mirror of https://github.com/Refactorio/RedMew.git synced 2025-01-26 03:52:00 +02:00
RedMew/map_gen/shared/builders.lua

2218 lines
63 KiB
Lua
Raw Normal View History

-- View the docs on the wiki.
-- https://github.com/Refactorio/RedMew/wiki/Using-the-Builders
-- The docs sometimes include examples
-- Some functions may not have been documented, you could always try to search
-- the repository for uses of them.
2018-11-11 15:08:35 +00:00
local math = require 'utils.math'
local table = require 'utils.table'
2019-01-23 19:55:48 -05:00
local pi = math.pi
local random = math.random
local abs = math.abs
local floor = math.floor
local ceil = math.ceil
local max = math.max
local sqrt = math.sqrt
local sin = math.sin
local cos = math.cos
local atan2 = math.atan2
local tau = math.tau
local loga = math.log
local shallow_copy = table.shallow_copy
local remove = table.remove
2019-05-13 12:21:59 +01:00
local binary_search = table.binary_search
local bnot = bit32.bnot
2019-01-23 19:55:48 -05:00
2017-08-12 01:20:08 +01:00
-- helpers
2019-01-23 19:55:48 -05:00
local inv_pi = 1 / pi
2017-08-12 01:20:08 +01:00
2018-08-28 13:31:23 +01:00
local Builders = {}
2018-05-08 21:23:07 +01:00
local function add_entity(tile, entity)
2018-05-16 11:37:50 +01:00
if type(tile) == 'table' then
2018-05-08 21:23:07 +01:00
if tile.entities then
tile.entities[#tile.entities + 1] = entity
2018-05-08 21:23:07 +01:00
else
tile.entities = {entity}
end
elseif tile then
tile = {
tile = tile,
entities = {entity}
}
end
2018-05-08 21:23:07 +01:00
return tile
end
local function add_decorative(tile, decorative)
if type(tile) == 'table' then
if tile.decoratives then
tile.decoratives[#tile.decoratives + 1] = decorative
else
tile.decoratives = {decorative}
end
elseif tile then
tile = {
tile = tile,
decoratives = {decorative}
}
end
return tile
end
--- Docs: MISSING
2018-08-28 13:31:23 +01:00
function Builders.add_entity(tile, entity)
return add_entity(tile, entity)
end
2018-05-08 21:23:07 +01:00
--- Docs: MISSING
function Builders.add_decorative(tile, decorative)
return add_decorative(tile, decorative)
end
2017-08-12 01:20:08 +01:00
-- shape builders
--- Docs: MISSING
2018-05-08 21:23:07 +01:00
function Builders.empty_shape()
2018-06-02 13:28:33 +01:00
return false
2017-08-12 01:20:08 +01:00
end
--- Docs: MISSING
2018-05-08 21:23:07 +01:00
function Builders.full_shape()
2017-08-12 01:20:08 +01:00
return true
end
--- Docs: MISSING
2018-06-02 13:28:33 +01:00
function Builders.no_entity()
return nil
end
--- Docs: MISSING
2018-05-08 21:23:07 +01:00
function Builders.tile(tile)
2018-03-03 12:54:55 +00:00
return function()
return tile
2018-03-03 12:54:55 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderspath
2018-05-08 21:23:07 +01:00
function Builders.path(thickness, optional_thickness_height)
2017-08-25 06:27:50 +01:00
local width = thickness / 2
local thickness2 = optional_thickness_height or thickness
local height = thickness2 / 2
return function(x, y)
2017-08-25 06:27:50 +01:00
return (x > -width and x <= width) or (y > -height and y <= height)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersrectangle
2018-05-08 21:23:07 +01:00
function Builders.rectangle(width, height)
width = width / 2
2018-01-21 15:31:33 +00:00
if height then
height = height / 2
else
height = width
end
return function(x, y)
return x > -width and x <= width and y > -height and y <= height
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersline_x
2018-05-08 21:23:07 +01:00
function Builders.line_x(thickness)
2018-01-21 15:31:33 +00:00
thickness = thickness / 2
2018-05-16 11:37:50 +01:00
return function(_, y)
return y > -thickness and y <= thickness
2018-01-21 15:31:33 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersline_y
2018-05-08 21:23:07 +01:00
function Builders.line_y(thickness)
2018-01-21 15:31:33 +00:00
thickness = thickness / 2
2018-05-16 11:37:50 +01:00
return function(x, _)
return x > -thickness and x <= thickness
2018-01-21 15:31:33 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssquare_diamond
2018-05-08 21:23:07 +01:00
function Builders.square_diamond(size)
2017-08-12 01:20:08 +01:00
size = size / 2
return function(x, y)
2019-01-23 19:55:48 -05:00
return abs(x) + abs(y) <= size
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersrectangle_diamond
2019-01-23 19:55:48 -05:00
local rot = sqrt(2) / 2 -- 45 degree rotation.
2018-05-08 21:23:07 +01:00
function Builders.rectangle_diamond(width, height)
2017-08-12 01:20:08 +01:00
width = width / 2
height = height / 2
return function(x, y)
2017-08-12 01:20:08 +01:00
local rot_x = rot * (x - y)
local rot_y = rot * (x + y)
2019-01-23 19:55:48 -05:00
return abs(rot_x) < width and abs(rot_y) < height
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscircle
2018-05-08 21:23:07 +01:00
function Builders.circle(radius)
2017-08-12 01:20:08 +01:00
local rr = radius * radius
return function(x, y)
return x * x + y * y < rr
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersoval
2018-05-08 21:23:07 +01:00
function Builders.oval(x_radius, y_radius)
2017-08-12 01:20:08 +01:00
local x_rr = x_radius * x_radius
local y_rr = y_radius * y_radius
return function(x, y)
return ((x * x) / x_rr + (y * y) / y_rr) < 1
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssine_fill
2018-05-08 21:23:07 +01:00
function Builders.sine_fill(width, height)
2018-05-16 11:37:50 +01:00
local width_inv = tau / width
local height_inv = -2 / height
2018-03-29 19:23:58 +01:00
return function(x, y)
local x2 = x * width_inv
local y2 = y * height_inv
if y <= 0 then
2019-01-23 19:55:48 -05:00
return y2 < sin(x2)
2018-03-29 19:23:58 +01:00
else
2019-01-23 19:55:48 -05:00
return y2 > sin(x2)
end
2018-03-29 19:23:58 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssine_wave
2018-08-11 12:38:12 +01:00
function Builders.sine_wave(width, height, thickness)
local width_inv = tau / width
local height_inv = 2 / height
thickness = thickness * 0.5
return function(x, y)
local x2 = x * width_inv
2019-01-23 19:55:48 -05:00
local y2 = sin(x2)
2018-08-11 12:38:12 +01:00
y = y * height_inv
2019-01-23 19:55:48 -05:00
local d = abs(y2 - y)
2018-08-11 12:38:12 +01:00
return d < thickness
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersrectangular_spiral
2018-05-29 00:02:25 +01:00
function Builders.rectangular_spiral(x_size, optional_y_size)
optional_y_size = optional_y_size or x_size
x_size = 1 / x_size
optional_y_size = 1 / optional_y_size
return function(x, y)
x, y = x * x_size, y * optional_y_size
2019-01-23 19:55:48 -05:00
x, y = floor(x + 0.5), floor(y + 0.5)
local a = -max(abs(x), abs(y)) -- because of absolutes, it's faster to use max than an if..then..else
2018-05-29 00:02:25 +01:00
if a % 2 == 0 then
return y ~= a or x == a
else
return y == a and x ~= a
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscircular_spiral
2018-08-24 15:31:16 +01:00
function Builders.circular_spiral(in_thickness, total_thickness)
2018-08-21 13:11:06 +01:00
local half_total_thickness = total_thickness * 0.5
return function(x, y)
2019-01-23 19:55:48 -05:00
local d = sqrt(x * x + y * y)
2018-08-21 13:11:06 +01:00
2019-01-23 19:55:48 -05:00
local angle = 1 + inv_pi * atan2(x, y)
2018-08-21 13:11:06 +01:00
local offset = d + (angle * half_total_thickness)
2018-08-24 18:27:46 +01:00
return offset % total_thickness < in_thickness
2018-08-21 13:11:06 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscircular_spiral_grow
2018-08-24 15:31:16 +01:00
function Builders.circular_spiral_grow(in_thickness, total_thickness, grow_factor)
local half_total_thickness = total_thickness * 0.5
local inv_grow_factor = 1 / grow_factor
return function(x, y)
2019-01-23 19:55:48 -05:00
local d = sqrt(x * x + y * y)
2018-08-24 15:31:16 +01:00
local factor = (d * inv_grow_factor) + 1
local total_thickness2 = total_thickness * factor
2018-08-24 18:27:46 +01:00
local in_thickness2 = in_thickness * factor
2018-08-24 15:31:16 +01:00
local half_total_thickness2 = half_total_thickness * factor
2019-01-23 19:55:48 -05:00
local angle = 1 + inv_pi * atan2(x, y)
2018-08-24 15:31:16 +01:00
local offset = d + (angle * half_total_thickness2)
2018-08-24 18:27:46 +01:00
return offset % total_thickness2 < in_thickness2
2018-08-24 15:31:16 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscircular_spiral_n_threads
2018-08-23 10:49:52 +01:00
function Builders.circular_spiral_n_threads(in_thickness, total_thickness, n_threads)
local half_total_thickness = total_thickness * 0.5 * n_threads
return function(x, y)
2019-01-23 19:55:48 -05:00
local d = sqrt(x * x + y * y)
2018-08-23 10:49:52 +01:00
2019-01-23 19:55:48 -05:00
local angle = 1 + inv_pi * atan2(x, y)
2018-08-23 10:49:52 +01:00
local offset = d + (angle * half_total_thickness)
2018-08-24 18:27:46 +01:00
return offset % total_thickness < in_thickness
2018-08-23 10:49:52 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscircular_spiral_grow_n_threads
2018-08-24 15:31:16 +01:00
function Builders.circular_spiral_grow_n_threads(in_thickness, total_thickness, grow_factor, n_threads)
local half_total_thickness = total_thickness * 0.5 * n_threads
local inv_grow_factor = 1 / grow_factor
return function(x, y)
2019-01-23 19:55:48 -05:00
local d = sqrt(x * x + y * y)
2018-08-24 15:31:16 +01:00
local factor = (d * inv_grow_factor) + 1
local total_thickness2 = total_thickness * factor
2018-08-24 18:27:46 +01:00
local in_thickness2 = in_thickness * factor
2018-08-24 15:31:16 +01:00
local half_total_thickness2 = half_total_thickness * factor
2019-01-23 19:55:48 -05:00
local angle = 1 + inv_pi * atan2(x, y)
2018-08-24 15:31:16 +01:00
local offset = d + (angle * half_total_thickness2)
2018-08-24 18:27:46 +01:00
return offset % total_thickness2 < in_thickness2
2018-08-24 15:31:16 +01:00
end
end
local tile_map = {
2017-11-10 13:03:46 +01:00
[1] = false,
[2] = true,
2018-05-16 11:37:50 +01:00
[3] = 'concrete',
[4] = 'deepwater-green',
[5] = 'deepwater',
[6] = 'dirt-1',
[7] = 'dirt-2',
[8] = 'dirt-3',
[9] = 'dirt-4',
[10] = 'dirt-5',
[11] = 'dirt-6',
[12] = 'dirt-7',
[13] = 'dry-dirt',
[14] = 'grass-1',
[15] = 'grass-2',
[16] = 'grass-3',
[17] = 'grass-4',
[18] = 'hazard-concrete-left',
[19] = 'hazard-concrete-right',
[20] = 'lab-dark-1',
[21] = 'lab-dark-2',
[22] = 'lab-white',
[23] = 'out-of-map',
[24] = 'red-desert-0',
[25] = 'red-desert-1',
[26] = 'red-desert-2',
[27] = 'red-desert-3',
[28] = 'sand-1',
[29] = 'sand-2',
[30] = 'sand-3',
[31] = 'stone-path',
[32] = 'water-green',
[33] = 'water'
2017-11-10 13:03:46 +01:00
}
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersdecompress
2018-05-08 21:23:07 +01:00
function Builders.decompress(pic)
2017-11-10 13:03:46 +01:00
local data = pic.data
local width = pic.width
local height = pic.height
2017-11-10 13:03:46 +01:00
local uncompressed = {}
2017-11-10 13:03:46 +01:00
for y = 1, height do
local row = data[y]
local u_row = {}
uncompressed[y] = u_row
local x = 1
for index = 1, #row, 2 do
local pixel = tile_map[row[index]]
local count = row[index + 1]
2018-05-16 11:37:50 +01:00
for _ = 1, count do
2017-11-10 13:03:46 +01:00
u_row[x] = pixel
x = x + 1
end
end
end
2017-11-10 13:03:46 +01:00
return {width = width, height = height, data = uncompressed}
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderspicture
2018-05-08 21:23:07 +01:00
function Builders.picture(pic)
2017-11-10 13:03:46 +01:00
local data = pic.data
local width = pic.width
local height = pic.height
2017-08-25 06:27:50 +01:00
-- the plus one is because lua tables are one based.
2019-01-23 19:55:48 -05:00
local half_width = floor(width / 2) + 1
local half_height = floor(height / 2) + 1
2017-08-12 01:20:08 +01:00
return function(x, y)
2019-01-23 19:55:48 -05:00
x = floor(x)
y = floor(y)
2017-08-25 06:27:50 +01:00
local x2 = x + half_width
local y2 = y + half_height
if y2 > 0 and y2 <= height and x2 > 0 and x2 <= width then
return data[y2][x2]
2018-05-08 17:47:34 +01:00
else
return false
end
2017-08-12 01:20:08 +01:00
end
end
-- transforms and shape helpers
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderstranslate
2018-05-08 21:23:07 +01:00
function Builders.translate(shape, x_offset, y_offset)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
return shape(x - x_offset, y - y_offset, world)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersscale
2018-05-08 21:23:07 +01:00
function Builders.scale(shape, x_scale, y_scale)
2018-05-10 20:42:38 +01:00
y_scale = y_scale or x_scale
2017-08-12 01:20:08 +01:00
x_scale = 1 / x_scale
y_scale = 1 / y_scale
2018-05-10 20:42:38 +01:00
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
return shape(x * x_scale, y * y_scale, world)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersrotate
2018-05-08 21:23:07 +01:00
function Builders.rotate(shape, angle)
2019-01-23 19:55:48 -05:00
local qx = cos(angle)
local qy = sin(angle)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2017-08-12 01:20:08 +01:00
local rot_x = qx * x - qy * y
local rot_y = qy * x + qx * y
2018-05-08 21:23:07 +01:00
return shape(rot_x, rot_y, world)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersflip_x
2018-05-08 21:23:07 +01:00
function Builders.flip_x(shape)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
return shape(-x, y, world)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersflip_y
2018-05-08 21:23:07 +01:00
function Builders.flip_y(shape)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
return shape(x, -y, world)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersflip_xy
2018-05-08 21:23:07 +01:00
function Builders.flip_xy(shape)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
return shape(-x, -y, world)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersany
2018-05-08 21:23:07 +01:00
function Builders.any(shapes)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
for _, s in ipairs(shapes) do
local tile = s(x, y, world)
if tile then
return tile
end
2017-08-12 01:20:08 +01:00
end
2018-05-08 17:47:34 +01:00
return false
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersall
2018-05-08 21:23:07 +01:00
function Builders.all(shapes)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local tile
2018-05-08 21:23:07 +01:00
for _, s in ipairs(shapes) do
tile = s(x, y, world)
if not tile then
2018-05-08 17:47:34 +01:00
return false
end
2017-08-12 01:20:08 +01:00
end
return tile
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscombine
2018-05-10 00:21:44 +01:00
function Builders.combine(shapes)
return function(x, y, world)
local function combine_table(tile, index)
local i, s = next(shapes, index)
while i do
local t = s(x, y, world)
2018-05-16 11:37:50 +01:00
if type(t) == 'table' then
2018-05-10 00:21:44 +01:00
if not tile.tile then
tile.tile = t.tile
end
local es = t.entities
if es then
for _, e in ipairs(es) do
add_entity(tile, e)
end
end
else
if not tile.tile then
tile.tile = t
end
end
2018-05-10 20:42:38 +01:00
i, s = next(shapes, i)
2018-05-10 00:21:44 +01:00
end
return tile
end
local tile = false
local i, s = next(shapes, nil)
while i do
local t = s(x, y, world)
if not tile then
tile = t
2018-05-16 11:37:50 +01:00
elseif type(t) == 'table' then
2018-05-10 00:21:44 +01:00
t.tile = tile
return combine_table(t, i)
end
2018-05-16 11:37:50 +01:00
if type(tile) == 'table' then
2018-05-10 00:21:44 +01:00
return combine_table(tile, i)
end
i, s = next(shapes, i)
end
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersadd
2018-11-14 12:16:42 +00:00
function Builders.add(shape1, shape2)
return function(x, y, world)
return shape1(x, y, world) or shape2(x, y, world)
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssubtract
2018-05-10 00:21:44 +01:00
function Builders.subtract(shape, minus_shape)
return function(x, y, world)
if minus_shape(x, y, world) then
return false
else
return shape(x, y, world)
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersinvert
2018-05-08 21:23:07 +01:00
function Builders.invert(shape)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-10 00:21:44 +01:00
return not shape(x, y, world)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersthrottle_x
2018-05-08 21:23:07 +01:00
function Builders.throttle_x(shape, x_in, x_size)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
if x % x_size < x_in then
2018-05-08 21:23:07 +01:00
return shape(x, y, world)
2017-11-10 13:03:46 +01:00
else
2018-05-08 17:47:34 +01:00
return false
2017-11-10 13:03:46 +01:00
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersthrottle_y
2018-05-08 21:23:07 +01:00
function Builders.throttle_y(shape, y_in, y_size)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
if y % y_size < y_in then
2018-05-08 21:23:07 +01:00
return shape(x, y, world)
2017-08-12 01:20:08 +01:00
else
2018-05-08 17:47:34 +01:00
return false
2017-08-12 01:20:08 +01:00
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersthrottle_xy
2018-05-08 21:23:07 +01:00
function Builders.throttle_xy(shape, x_in, x_size, y_in, y_size)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-02-13 00:49:46 +00:00
if x % x_size < x_in and y % y_size < y_in then
2018-05-08 21:23:07 +01:00
return shape(x, y, world)
2018-02-13 00:49:46 +00:00
else
2018-05-08 17:47:34 +01:00
return false
2018-02-13 00:49:46 +00:00
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersthrottle_world_xy
2018-05-08 21:23:07 +01:00
function Builders.throttle_world_xy(shape, x_in, x_size, y_in, y_size)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
if world.x % x_size < x_in and world.y % y_size < y_in then
2018-05-08 21:23:07 +01:00
return shape(x, y, world)
2018-02-13 00:49:46 +00:00
else
2018-05-08 17:47:34 +01:00
return false
2018-02-13 00:49:46 +00:00
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderschoose
2018-05-08 21:23:07 +01:00
function Builders.choose(condition, true_shape, false_shape)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
if condition(x, y, world) then
return true_shape(x, y, world)
2017-08-13 03:57:43 +01:00
else
2018-05-08 17:47:34 +01:00
return false_shape(x, y, world)
2017-08-13 03:57:43 +01:00
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersif_else
2018-05-08 21:23:07 +01:00
function Builders.if_else(shape, else_shape)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
return shape(x, y, world) or else_shape(x, y, world)
2018-02-23 22:51:28 +00:00
end
end
function Builders.use_world_as_local(shape)
return function (_, _, world)
return shape(world.x, world.y, world)
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderslinear_grow
2018-05-08 21:23:07 +01:00
function Builders.linear_grow(shape, size)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2019-01-23 19:55:48 -05:00
local t = ceil((y / size) + 0.5)
local n = ceil((sqrt(8 * t + 1) - 1) / 2)
2017-08-14 09:38:14 +01:00
local t_upper = n * (n + 1) * 0.5
local t_lower = t_upper - n
2018-05-16 11:37:50 +01:00
y = (y - size * (t_lower + n / 2 - 0.5)) / n
x = x / n
2018-05-08 17:47:34 +01:00
return shape(x, y, world)
2017-08-14 09:38:14 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersgrow
2018-05-08 21:23:07 +01:00
function Builders.grow(in_shape, out_shape, size, offset)
2018-02-04 22:12:52 +00:00
local half_size = size / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2019-01-23 19:55:48 -05:00
local tx = ceil(abs(x) / half_size)
local ty = ceil(abs(y) / half_size)
local t
if tx > ty then
t = tx
else
t = ty
end
2018-02-04 22:24:41 +00:00
for i = t, 2.5 * t, 1 do
2018-02-04 22:12:52 +00:00
local out_t = 1 / (i - offset)
local in_t = 1 / i
2018-05-08 17:47:34 +01:00
if out_shape(out_t * x, out_t * y, world) then
return nil
2018-02-04 22:12:52 +00:00
end
2018-05-08 17:47:34 +01:00
local tile = in_shape(in_t * x, in_t * y, world)
2018-02-04 22:12:52 +00:00
if tile then
return tile
2018-02-04 22:12:52 +00:00
end
end
return nil
end
2018-02-04 22:12:52 +00:00
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersproject
2018-05-08 21:23:07 +01:00
function Builders.project(shape, size, r)
local ln_r = loga(r)
2017-08-14 09:38:14 +01:00
local r2 = 1 / (r - 1)
local a = 1 / size
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2017-08-25 06:27:50 +01:00
local offset = 0.5 * size
2019-01-23 19:55:48 -05:00
local sn = ceil(y + offset)
local n = ceil(loga((r - 1) * sn * a + 1) / ln_r - 1)
2017-08-25 06:27:50 +01:00
local rn = r ^ n
local rn2 = 1 / rn
local c = size * rn
2017-08-14 09:38:14 +01:00
local sn_upper = size * (r ^ (n + 1) - 1) * r2
2018-05-16 11:37:50 +01:00
x = x * rn2
y = (y - (sn_upper - 0.5 * c) + offset) * rn2
2018-05-08 17:47:34 +01:00
return shape(x, y, world)
2017-08-25 06:27:50 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersproject_pattern
2018-06-16 12:57:11 +01:00
function Builders.project_pattern(pattern, size, r, columns, rows)
local ln_r = loga(r)
2018-06-16 12:57:11 +01:00
local r2 = 1 / (r - 1)
local a = 1 / size
local half_size = size / 2
return function(x, y, world)
local offset = 0.5 * size
2019-01-23 19:55:48 -05:00
local sn = ceil(y + offset)
2018-06-16 12:57:11 +01:00
local n = ceil(loga((r - 1) * sn * a + 1) / ln_r - 1)
2018-06-16 12:57:11 +01:00
local rn = r ^ n
local rn2 = 1 / rn
local c = size * rn
local sn_upper = size * (r ^ (n + 1) - 1) * r2
x = x * rn2
y = (y - (sn_upper - 0.5 * c) + offset) * rn2
local row_i = n % rows + 1
local row = pattern[row_i]
local x2 = ((x + half_size) % size) - half_size
2019-01-23 19:55:48 -05:00
local col_pos = floor(x / size + 0.5)
2018-06-16 12:57:11 +01:00
local col_i = col_pos % columns + 1
local shape = row[col_i]
return shape(x2, y, world)
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersproject_overlap
2018-05-08 21:23:07 +01:00
function Builders.project_overlap(shape, size, r)
local ln_r = loga(r)
2017-08-25 06:27:50 +01:00
local r2 = 1 / (r - 1)
local a = 1 / size
local offset = 0.5 * size
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2019-01-23 19:55:48 -05:00
local sn = ceil(y + offset)
local n = ceil(loga((r - 1) * sn * a + 1) / ln_r - 1)
2017-08-25 06:27:50 +01:00
local rn = r ^ n
local rn2 = 1 / rn
local c = size * rn
2017-08-25 06:27:50 +01:00
local sn_upper = size * (r ^ (n + 1) - 1) * r2
2018-05-16 11:37:50 +01:00
x = x * rn2
y = (y - (sn_upper - 0.5 * c) + offset) * rn2
2017-08-25 06:27:50 +01:00
local tile
2018-05-08 17:47:34 +01:00
tile = shape(x, y, world)
2017-08-25 06:27:50 +01:00
if tile then
return tile
2017-08-25 06:27:50 +01:00
end
2017-08-25 06:27:50 +01:00
local rn_above = rn / r
local rn2_above = 1 / rn_above
local c_above = size * rn_above
2017-08-25 06:27:50 +01:00
local sn_upper_above = sn_upper - c
2018-05-08 17:47:34 +01:00
local x_above = x * rn2_above
local y_above = (y - (sn_upper_above - 0.5 * c_above) + offset) * rn2_above
2018-05-08 17:47:34 +01:00
tile = shape(x_above, y_above, world)
if tile then
return tile
end
2017-08-25 06:27:50 +01:00
local rn_below = rn * r
local rn2_below = 1 / rn_below
local c_below = size * rn_below
2017-08-25 06:27:50 +01:00
local sn_upper_below = sn_upper + c_below
2018-05-08 17:47:34 +01:00
local x_below = x * rn2_below
local y_below = (y - (sn_upper_below - 0.5 * c_below) + offset) * rn2_below
2018-05-08 17:47:34 +01:00
return shape(x_below, y_below, world)
2017-08-14 09:38:14 +01:00
end
end
2018-05-10 00:21:44 +01:00
-- Entity generation
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersentity
2018-05-10 00:21:44 +01:00
function Builders.entity(shape, name)
return function(x, y, world)
if shape(x, y, world) then
2018-05-10 20:42:38 +01:00
return {name = name}
2018-05-10 00:21:44 +01:00
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersentity_func
2018-05-17 14:29:18 +01:00
function Builders.entity_func(shape, func)
return function(x, y, world)
if shape(x, y, world) then
return func(x, y, world)
end
end
end
-- Helper function
local function destroy_entities(entities)
for i = 1, #entities do
entities[i].destroy()
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersremove_map_gen_entities_by_filter
function Builders.remove_map_gen_entities_by_filter(shape, filter)
filter = shallow_copy(filter)
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return tile
end
local wx, wy = world.x, world.y
filter.area = {{wx, wy}, {wx + 1, wy + 1}}
local entities = world.surface.find_entities_filtered(filter)
destroy_entities(entities)
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersremove_entities_by_name
function Builders.remove_entities_by_name(shape, names)
if type(names) ~= 'table' then
names = {names}
end
return function(x, y, world)
local tile = shape(x, y, world)
if type(tile) ~= 'table' then
return tile
end
local entities = tile.entities
if not entities then
return tile
end
for e_index = #entities, 1, -1 do
local entity_name = entities[e_index].name
for n_index = 1, #names do
if entity_name == names[n_index] then
remove(entities, e_index)
end
end
end
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersremove_map_gen_decoratives
function Builders.remove_map_gen_decoratives(shape, optional_filter)
if optional_filter then
optional_filter = shallow_copy(optional_filter)
else
optional_filter = {}
end
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return tile
end
local wx, wy = world.x, world.y
optional_filter.area = {{wx, wy}, {wx + 1, wy + 1}}
world.surface.destroy_decoratives(optional_filter)
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders/#buildersremove_decoratives_by_name
function Builders.remove_decoratives_by_name(shape, names)
if type(names) ~= 'table' then
names = {names}
end
return function(x, y, world)
local tile = shape(x, y, world)
if type(tile) ~= 'table' then
return tile
end
local decoratives = tile.decoratives
if not decoratives then
return tile
end
for d_index = #decoratives, 1, -1 do
local decorative_name = decoratives[d_index].name
for n_index = 1, #names do
if decorative_name == names[n_index] then
remove(decoratives, d_index)
end
end
end
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersremove_map_gen_resources
function Builders.remove_map_gen_resources(shape)
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return tile
end
local wx, wy = world.x, world.y
local area = {{wx, wy}, {wx + 1, wy + 1}}
local entities = world.surface.find_entities_filtered {area = area, type = 'resource'}
destroy_entities(entities)
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersremove_map_gen_trees
function Builders.remove_map_gen_trees(shape)
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return tile
end
local wx, wy = world.x, world.y
local area = {{wx, wy}, {wx + 1, wy + 1}}
local entities = world.surface.find_entities_filtered {area = area, type = 'tree'}
destroy_entities(entities)
return tile
end
end
-- Removes simple entities such as rocks: https://wiki.factorio.com/Data.raw#simple-entity
-- Looking for a remove_rocks function? You're welcome ~ Jayefuu
function Builders.remove_map_gen_simple_entity(shape)
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return tile
end
local wx, wy = world.x, world.y
local area = {{wx, wy}, {wx + 1, wy + 1}}
local entities = world.surface.find_entities_filtered {area = area, type = 'simple-entity'}
destroy_entities(entities)
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersremove_map_gen_enemies
function Builders.remove_map_gen_enemies(shape)
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return tile
end
local wx, wy = world.x, world.y
local area = {{wx, wy}, {wx + 1, wy + 1}}
local entities = world.surface.find_entities_filtered {area = area, force = 'enemy'}
destroy_entities(entities)
return tile
end
end
-- Decorative generation
--- Docs: MISSING
function Builders.decorative(shape, name, amount)
return function(x, y, world)
if shape(x, y, world) then
return {name = name, amount = amount or 1}
end
end
end
--- Docs: MISSING
function Builders.decorative_func(shape, func)
return function(x, y, world)
if shape(x, y, world) then
return func(x, y, world)
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersresource
2018-08-25 12:54:55 +01:00
function Builders.resource(shape, resource_type, amount_function, always_place)
2018-05-16 11:37:50 +01:00
amount_function = amount_function or function()
return 404
end
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
if shape(x, y, world) then
return {
2017-08-12 01:20:08 +01:00
name = resource_type,
2018-08-25 12:54:55 +01:00
amount = amount_function(world.x, world.y),
always_place = always_place
}
2017-08-12 01:20:08 +01:00
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersapply_entity
2018-05-08 21:23:07 +01:00
function Builders.apply_entity(shape, entity_shape)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
local tile = shape(x, y, world)
if not tile then
return false
end
local e = entity_shape(x, y, world)
if e and e ~= true then
2018-05-08 21:23:07 +01:00
tile = add_entity(tile, e)
end
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersapply_entities
2018-05-08 21:23:07 +01:00
function Builders.apply_entities(shape, entity_shapes)
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return false
end
for _, es in ipairs(entity_shapes) do
local e = es(x, y, world)
if e and e ~= true then
2018-05-08 21:23:07 +01:00
tile = add_entity(tile, e)
2018-05-08 17:47:34 +01:00
end
2017-08-12 01:20:08 +01:00
end
2018-05-08 17:47:34 +01:00
return tile
2017-08-12 01:20:08 +01:00
end
end
--- Docs: MISSING
function Builders.apply_decorative(shape, decorative_shape)
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return false
end
local d = decorative_shape(x, y, world)
if d then
tile = add_decorative(tile, d)
end
return tile
end
end
--- Docs: MISSING
function Builders.apply_decoratives(shape, decorative_shapes)
return function(x, y, world)
local tile = shape(x, y, world)
if not tile then
return false
end
for _, ds in ipairs(decorative_shapes) do
local d = ds(x, y, world)
if d then
tile = add_decorative(tile, d)
end
end
return tile
end
end
-- pattern builders
--- Loops through the shapes, if any shape is able to place any entities,
--- it'll return that tile. Otherwise, false is returned.
function Builders.any_entity_pattern(pattern)
return function(x, y, world)
for index, shape in pairs(pattern or {}) do
local tile = shape.shape(x, y, world)
if tile and tile.entities then return tile end
end
return false
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssingle_pattern
2018-05-08 21:23:07 +01:00
function Builders.single_pattern(shape, width, height)
2018-05-16 11:37:50 +01:00
shape = shape or Builders.empty_shape
local half_width = width / 2
2018-01-21 15:31:33 +00:00
local half_height
if height then
half_height = height / 2
else
half_height = half_width
end
2018-05-08 17:47:34 +01:00
return function(x, y, world)
y = ((y + half_height) % height) - half_height
x = ((x + half_width) % width) - half_width
2018-05-08 17:47:34 +01:00
return shape(x, y, world)
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssingle_pattern_overlap
2018-05-08 21:23:07 +01:00
function Builders.single_pattern_overlap(shape, width, height)
2018-05-16 11:37:50 +01:00
shape = shape or Builders.empty_shape
local half_width = width / 2
2018-02-04 22:12:52 +00:00
local half_height
if height then
half_height = height / 2
else
half_height = half_width
end
2018-05-08 17:47:34 +01:00
return function(x, y, world)
y = ((y + half_height) % height) - half_height
x = ((x + half_width) % width) - half_width
return shape(x, y, world) or shape(x + width, y, world) or shape(x - width, y, world) or shape(x, y + height, world) or shape(x, y - height, world)
2018-02-04 22:12:52 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssingle_x_pattern
2018-05-08 21:23:07 +01:00
function Builders.single_x_pattern(shape, width)
2018-05-16 11:37:50 +01:00
shape = shape or Builders.empty_shape
local half_width = width / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
x = ((x + half_width) % width) - half_width
2018-05-08 17:47:34 +01:00
return shape(x, y, world)
2018-01-21 15:31:33 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssingle_y_pattern
2018-05-08 21:23:07 +01:00
function Builders.single_y_pattern(shape, height)
2018-05-16 11:37:50 +01:00
shape = shape or Builders.empty_shape
2018-01-21 15:31:33 +00:00
local half_height = height / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
y = ((y + half_height) % height) - half_height
2018-05-08 17:47:34 +01:00
return shape(x, y, world)
2018-01-21 15:31:33 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssingle_grid_pattern
2018-11-14 12:16:42 +00:00
function Builders.single_grid_pattern(shape, width, height)
shape = shape or Builders.empty_shape
local half_width = width / 2
local half_height = height / 2
return function(x, y, world)
x = ((x + half_width) % width) - half_width
y = ((y + half_height) % height) - half_height
return shape(x, y, world)
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersgrid_x_pattern
2018-07-02 10:51:22 +01:00
function Builders.grid_x_pattern(pattern, columns, width)
local half_width = width / 2
return function(x, y, world)
local x2 = ((x + half_width) % width) - half_width
2019-01-23 19:55:48 -05:00
local columns_pos = floor(x / width + 0.5)
2018-07-02 10:51:22 +01:00
local column_i = columns_pos % columns + 1
local shape = pattern[column_i] or Builders.empty_shape
return shape(x2, y, world)
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersgrid_y_pattern
2018-07-02 10:51:22 +01:00
function Builders.grid_y_pattern(pattern, rows, height)
local half_height = height / 2
return function(x, y, world)
local y2 = ((y + half_height) % height) - half_height
2019-01-23 19:55:48 -05:00
local row_pos = floor(y / height + 0.5)
2018-07-02 10:51:22 +01:00
local row_i = row_pos % rows + 1
local shape = pattern[row_i] or Builders.empty_shape
return shape(x, y2, world)
end
end
2021-08-01 14:57:12 +01:00
function Builders.grid_y_no_repeat_weighted_pattern(pattern, height)
local weights = Builders.prepare_weighted_array(pattern)
local total = weights.total
local scale = 1 / height
return function(x, y, world)
local i = ((y * scale) -0.5) % total
local index = binary_search(weights, i)
if index < 0 then
index = bnot(index)
end
local data = pattern[index]
local shape
if data then
shape = data.shape
end
shape = shape or Builders.empty_shape
return shape(x, y, world)
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersgrid_pattern
2018-05-08 21:23:07 +01:00
function Builders.grid_pattern(pattern, columns, rows, width, height)
local half_width = width / 2
2017-08-12 01:20:08 +01:00
local half_height = height / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local y2 = ((y + half_height) % height) - half_height
2019-01-23 19:55:48 -05:00
local row_pos = floor(y / height + 0.5)
2017-08-12 01:20:08 +01:00
local row_i = row_pos % rows + 1
local row = pattern[row_i] or {}
2018-05-08 17:47:34 +01:00
local x2 = ((x + half_width) % width) - half_width
2019-01-23 19:55:48 -05:00
local col_pos = floor(x / width + 0.5)
2017-08-12 01:20:08 +01:00
local col_i = col_pos % columns + 1
2018-05-16 11:37:50 +01:00
local shape = row[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
return shape(x2, y2, world)
2017-08-12 01:20:08 +01:00
end
end
2023-11-26 14:30:24 +00:00
function Builders.grid_pattern_no_repeat(pattern, width, height)
local half_width = width / 2
local half_height = height / 2
return function(x, y, world)
local y2 = ((y + half_height) % height) - half_height
local row_pos = floor(y / height + 0.5)
local row_i = row_pos + 1
local row = pattern[row_i] or {}
local x2 = ((x + half_width) % width) - half_width
local col_pos = floor(x / width + 0.5)
local col_i = col_pos + 1
local shape = row[col_i] or Builders.empty_shape
return shape(x2, y2, world)
end
end
2021-05-03 15:05:26 +01:00
function Builders.grid_pattern_no_offset(pattern, columns, rows, width, height)
return function(x, y, world)
local row_pos = floor(y / height + 0.5)
local row_i = row_pos % rows + 1
local row = pattern[row_i] or {}
local col_pos = floor(x / width + 0.5)
local col_i = col_pos % columns + 1
local shape = row[col_i] or Builders.empty_shape
return shape(x, y, world)
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersgrid_pattern_overlap
2018-05-08 21:23:07 +01:00
function Builders.grid_pattern_overlap(pattern, columns, rows, width, height)
2018-02-23 22:51:28 +00:00
local half_width = width / 2
local half_height = height / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local y2 = ((y + half_height) % height) - half_height
2019-01-23 19:55:48 -05:00
local row_pos = floor(y / height + 0.5)
2018-02-23 22:51:28 +00:00
local row_i = row_pos % rows + 1
local row = pattern[row_i] or {}
2018-05-08 17:47:34 +01:00
local x2 = ((x + half_width) % width) - half_width
2019-01-23 19:55:48 -05:00
local col_pos = floor(x / width + 0.5)
2018-02-23 22:51:28 +00:00
local col_i = col_pos % columns + 1
2018-05-16 11:37:50 +01:00
local shape = row[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
local tile = shape(x2, y2, world)
if tile then
return tile
end
2018-03-03 12:54:55 +00:00
-- edges
local col_i_left = (col_pos - 1) % columns + 1
2018-05-16 11:37:50 +01:00
shape = row[col_i_left] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2 + width, y2, world)
if tile then
return tile
end
2018-03-03 12:54:55 +00:00
local col_i_right = (col_pos + 1) % columns + 1
2018-05-16 11:37:50 +01:00
shape = row[col_i_right] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2 - width, y2, world)
if tile then
return tile
end
2018-03-03 12:54:55 +00:00
local row_i_up = (row_pos - 1) % rows + 1
local row_up = pattern[row_i_up] or {}
2018-05-16 11:37:50 +01:00
shape = row_up[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2, y2 + height, world)
if tile then
return tile
end
2018-03-03 12:54:55 +00:00
local row_i_down = (row_pos + 1) % rows + 1
local row_down = pattern[row_i_down] or {}
2018-05-16 11:37:50 +01:00
shape = row_down[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
return shape(x2, y2 - height, world)
2018-03-03 12:54:55 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersgrid_pattern_full_overlap
2018-05-08 21:23:07 +01:00
function Builders.grid_pattern_full_overlap(pattern, columns, rows, width, height)
2018-03-03 12:54:55 +00:00
local half_width = width / 2
local half_height = height / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local y2 = ((y + half_height) % height) - half_height
2019-01-23 19:55:48 -05:00
local row_pos = floor(y / height + 0.5)
2018-03-03 12:54:55 +00:00
local row_i = row_pos % rows + 1
local row = pattern[row_i] or {}
2018-05-08 17:47:34 +01:00
local x2 = ((x + half_width) % width) - half_width
2019-01-23 19:55:48 -05:00
local col_pos = floor(x / width + 0.5)
2018-03-03 12:54:55 +00:00
local col_i = col_pos % columns + 1
2018-03-27 16:16:37 +01:00
local row_i_up = (row_pos - 1) % rows + 1
local row_up = pattern[row_i_up] or {}
local row_i_down = (row_pos + 1) % rows + 1
local row_down = pattern[row_i_down] or {}
2018-03-27 16:16:37 +01:00
local col_i_left = (col_pos - 1) % columns + 1
local col_i_right = (col_pos + 1) % columns + 1
2018-03-27 16:16:37 +01:00
-- start from top left, move left to right then down
2018-05-16 11:37:50 +01:00
local shape = row_up[col_i_left] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
local tile = shape(x2 + width, y2 + height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_up[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2, y2 + height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_up[col_i_right] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2 - width, y2 + height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row[col_i_left] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2 + width, y2, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2, y2, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row[col_i_right] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2 - width, y2, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_down[col_i_left] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2 + width, y2 - height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_down[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x2, y2 - height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_down[col_i_right] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
return shape(x2 - width, y2 - height, world)
2018-03-27 16:16:37 +01:00
end
end
2018-12-05 13:50:02 +00:00
-- Tile a shape in a circular pattern
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscircular_pattern
2018-12-05 13:50:02 +00:00
function Builders.circular_pattern(shape, quantity, radius)
local pattern = {}
local angle = tau / quantity
for i = 1, quantity do
local shape2 = Builders.rotate(Builders.translate(shape, 0, radius), i * angle)
2019-01-23 23:32:14 -05:00
pattern[i] = shape2
2018-12-05 13:50:02 +00:00
end
return Builders.any(pattern)
end
2018-07-07 12:24:09 +01:00
local function is_spiral(x, y)
2019-01-23 19:55:48 -05:00
local a = -max(abs(x), abs(y)) -- because of absolutes, it's faster to use max than an if..then..else
2018-07-07 12:24:09 +01:00
if a % 2 == 0 then
return y ~= a or x == a
else
return y == a and x ~= a
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssingle_spiral_pattern
2018-07-07 12:24:09 +01:00
function Builders.single_spiral_pattern(shape, width, height)
local inv_width = 1 / width
local inv_height = 1 / height
return function(x, y, world)
2019-01-23 19:55:48 -05:00
local x1 = floor(x * inv_width + 0.5)
local y1 = floor(y * inv_height + 0.5)
2018-07-07 12:24:09 +01:00
if is_spiral(x1, y1) then
x1 = x - x1 * width
y1 = y - y1 * height
return shape(x1, y1, world)
else
return false
end
end
end
local function rotate_0(x, y)
return x, y
end
local function rotate_90(x, y)
return y, -x
end
local function rotate_180(x, y)
return -x, -y
end
local function rotate_270(x, y)
return -y, x
end
local function spiral_rotation(x, y)
2019-01-23 19:55:48 -05:00
local a = -max(abs(x), abs(y)) -- because of absolutes, it's faster to use max than an if..then..else
2018-07-07 12:24:09 +01:00
if a % 2 == 0 then
if y ~= a or x == a then
if x == a then
return rotate_0
elseif y >= x then
return rotate_270
else
return rotate_180
end
end
else
if y == a and x ~= a then
return rotate_90
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssingle_spiral_rotate_pattern
2018-07-10 13:37:45 +01:00
function Builders.single_spiral_rotate_pattern(shape, width, optional_height)
optional_height = optional_height or width
2018-07-07 12:24:09 +01:00
local inv_width = 1 / width
2018-07-10 13:37:45 +01:00
local inv_height = 1 / optional_height
2018-07-07 12:24:09 +01:00
return function(x, y, world)
2019-01-23 19:55:48 -05:00
local x1 = floor(x * inv_width + 0.5)
local y1 = floor(y * inv_height + 0.5)
2018-07-07 12:24:09 +01:00
local t = spiral_rotation(x1, y1)
if t then
x1 = x - x1 * width
2018-07-10 13:37:45 +01:00
y1 = y - y1 * optional_height
2018-07-07 12:24:09 +01:00
x1, y1 = t(x1, y1)
return shape(x1, y1, world)
else
return false
end
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscircular_spiral_pattern
2018-08-24 18:27:46 +01:00
function Builders.circular_spiral_pattern(in_thickness, total_thickness, pattern)
local n_threads = #pattern
total_thickness = total_thickness * n_threads
local half_total_thickness = total_thickness * 0.5
local delta = total_thickness / n_threads
return function(x, y, world)
2019-01-23 19:55:48 -05:00
local d = sqrt(x * x + y * y)
2018-08-24 18:27:46 +01:00
2019-01-23 19:55:48 -05:00
local angle = 1 + inv_pi * atan2(x, y)
2018-08-24 18:27:46 +01:00
local offset = d + (angle * half_total_thickness)
if offset % total_thickness < in_thickness then
return pattern[1](x, y, world)
end
for i = 2, n_threads do
offset = offset + delta
if offset % total_thickness < in_thickness then
return pattern[i](x, y, world)
end
end
return false
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderscircular_spiral_grow_pattern
2018-08-24 18:27:46 +01:00
function Builders.circular_spiral_grow_pattern(in_thickness, total_thickness, grow_factor, pattern)
local n_threads = #pattern
total_thickness = total_thickness * n_threads
local half_total_thickness = total_thickness * 0.5
local inv_grow_factor = 1 / grow_factor
local delta = total_thickness / n_threads
return function(x, y, world)
2019-01-23 19:55:48 -05:00
local d = sqrt(x * x + y * y)
2018-08-24 18:27:46 +01:00
local factor = (d * inv_grow_factor) + 1
local total_thickness2 = total_thickness * factor
local in_thickness2 = in_thickness * factor
local half_total_thickness2 = half_total_thickness * factor
local delta2 = delta * factor
2019-01-23 19:55:48 -05:00
local angle = 1 + inv_pi * atan2(x, y)
2018-08-24 18:27:46 +01:00
local offset = d + (angle * half_total_thickness2)
if offset % total_thickness2 < in_thickness2 then
return pattern[1](x, y, world)
end
for i = 2, n_threads do
offset = offset + delta2
if offset % total_thickness2 < in_thickness2 then
return pattern[i](x, y, world)
end
end
return false
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderssegment_pattern
2018-05-08 21:23:07 +01:00
function Builders.segment_pattern(pattern)
2018-03-27 16:16:37 +01:00
local count = #pattern
2019-05-13 12:21:59 +01:00
local count_by_tau = count / tau
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2019-01-23 19:55:48 -05:00
local angle = atan2(-y, x)
2019-05-13 12:21:59 +01:00
local index = floor(angle * count_by_tau) % count + 1
2018-05-16 11:37:50 +01:00
local shape = pattern[index] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
return shape(x, y, world)
2018-03-27 16:16:37 +01:00
end
end
2019-05-13 12:21:59 +01:00
function Builders.segment_weighted_pattern(pattern)
local weights = Builders.prepare_weighted_array(pattern)
local total = weights.total * 0.5
return function(x, y, world)
local angle = atan2(-y, x)
local i = (angle * inv_pi + 1) * total
local index = binary_search(weights, i)
if index < 0 then
index = bnot(index)
end
local shape = pattern[index].shape or Builders.empty_shape
2019-05-13 12:21:59 +01:00
return shape(x, y, world)
end
end
function Builders.ring_pattern(pattern, thickness)
local count = #pattern
local scale = 1 / thickness
return function(x, y, world)
local d = sqrt(x * x + y * y)
local index = floor(d * scale) % count + 1
local shape = pattern[index] or Builders.empty_shape
return shape(x, y, world)
end
end
function Builders.ring_weighted_pattern(pattern, thickness)
local weights = Builders.prepare_weighted_array(pattern)
local total = weights.total
local scale = 1 / thickness
2021-05-23 11:59:44 +01:00
return function(x, y, world)
local d = sqrt(x * x + y * y)
local i = (d * scale) % total
local index = binary_search(weights, i)
if index < 0 then
index = bnot(index)
end
local data = pattern[index]
local shape
if data then
shape = data.shape
end
shape = shape or Builders.empty_shape
return shape(x, y, world)
end
end
function Builders.ring_weighted_grow_pattern(pattern, thickness, grow_factor)
local weights = Builders.prepare_weighted_array(pattern)
local total = weights.total
local scale = 1 / thickness
local inv_grow_factor = 1 / grow_factor
return function(x, y, world)
local d = sqrt(x * x + y * y)
local factor = 1 + (d * inv_grow_factor)
local i = (d * scale / factor) % total
local index = binary_search(weights, i)
if index < 0 then
index = bnot(index)
end
local data = pattern[index]
local shape
if data then
shape = data.shape
end
shape = shape or Builders.empty_shape
return shape(x, y, world)
end
end
2021-01-09 13:02:34 +00:00
local function rotate_cords(angle)
local qx = cos(angle)
local qy = sin(angle)
return function(x, y)
local rot_x = qx * x - qy * y
local rot_y = qy * x + qx * y
return rot_x, rot_y
end
end
function Builders.gradient_pattern(pattern)
local count = #pattern
local tau_by_count = tau / count
local values = {}
local rotations = {}
for i = 1, count do
rotations[i] = rotate_cords(tau_by_count * (i - 1))
end
return function(x, y, world)
for i = 1, count do
local rot_x, rot_y = rotations[i](x, y)
local angle = atan2(rot_x, rot_y)
local normalised = angle * inv_pi
if normalised < 0 then
normalised = -normalised
end
values[i] = pattern[i].weight(normalised)
end
local sum = 0
for i = 1, count do
sum = sum + values[i]
end
local rand = random() * sum
for i = 1, count - 1 do
local v = values[i]
if rand < v then
return pattern[i].shape(x, y, world)
end
rand = rand - v
end
return pattern[count].shape(x, y, world)
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderspyramid_pattern
2018-05-08 21:23:07 +01:00
function Builders.pyramid_pattern(pattern, columns, rows, width, height)
2018-03-27 16:16:37 +01:00
local half_width = width / 2
local half_height = height / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local y2 = ((y + half_height) % height) - half_height
2019-01-23 19:55:48 -05:00
local row_pos = floor(y / height + 0.5)
2018-03-27 16:16:37 +01:00
local row_i = row_pos % rows + 1
local row = pattern[row_i] or {}
2018-03-27 16:16:37 +01:00
if row_pos % 2 ~= 0 then
2018-05-08 17:47:34 +01:00
x = x - half_width
2018-03-27 16:16:37 +01:00
end
2018-05-08 17:47:34 +01:00
local x2 = ((x + half_width) % width) - half_width
2019-01-23 19:55:48 -05:00
local col_pos = floor(x / width + 0.5)
2018-03-27 16:16:37 +01:00
local col_i = col_pos % columns + 1
2018-03-27 16:16:37 +01:00
if col_pos > row_pos / 2 or -col_pos > (row_pos + 1) / 2 then
return false
end
2018-05-16 11:37:50 +01:00
local shape = row[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
return shape(x2, y2, world)
2018-03-27 16:16:37 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderspyramid_pattern_inner_overlap
2018-05-08 21:23:07 +01:00
function Builders.pyramid_pattern_inner_overlap(pattern, columns, rows, width, height)
2018-03-27 16:16:37 +01:00
local half_width = width / 2
local half_height = height / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local y2 = ((y + half_height) % height) - half_height
2019-01-23 19:55:48 -05:00
local row_pos = floor(y / height + 0.5)
2018-03-27 16:16:37 +01:00
local row_i = row_pos % rows + 1
local row = pattern[row_i] or {}
2018-03-27 16:16:37 +01:00
local x_odd
local x_even
if row_pos % 2 == 0 then
2018-05-08 17:47:34 +01:00
x_even = x
x_odd = x - half_width
2018-03-27 16:16:37 +01:00
else
2018-05-08 17:47:34 +01:00
x_even = x - half_width
x_odd = x
x = x - half_width
end
2018-03-27 16:16:37 +01:00
x_even = ((x_even + half_width) % width) - half_width
x_odd = ((x_odd + half_width) % width) - half_width
2019-01-23 19:55:48 -05:00
local col_pos = floor(x / width + 0.5)
2018-03-27 16:16:37 +01:00
local offset = 1
local offset_odd = 0
if (col_pos % 2) == (row_pos % 2) then
offset = 0
offset_odd = 1
end
local col_i = (col_pos - offset) % columns + 1
local col_i_odd = (col_pos - offset_odd) % columns + 1
2018-03-27 16:16:37 +01:00
if col_pos > row_pos / 2 or -col_pos > (row_pos + 1) / 2 then
return false
end
2018-03-27 16:16:37 +01:00
local row_i_up = (row_pos - 1) % rows + 1
local row_up = pattern[row_i_up] or {}
2018-02-23 22:51:28 +00:00
local row_i_down = (row_pos + 1) % rows + 1
local row_down = pattern[row_i_down] or {}
2018-03-27 16:16:37 +01:00
local col_i_left = (col_pos - 1) % columns + 1
local col_i_right = (col_pos + 1) % columns + 1
2018-03-27 16:16:37 +01:00
-- start from top left, move left to right then down
2018-05-16 11:37:50 +01:00
local shape = row_up[col_i_left] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
local tile = shape(x_even + width, y2 + height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_up[col_i_odd] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x_odd, y2 + height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_up[col_i_right] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x_even - width, y2 + height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row[col_i_left] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x_even + width, y2, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x_even, y2, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row[col_i_right] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x_even - width, y2, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_down[col_i_left] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x_even + width, y2 - height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_down[col_i_odd] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
tile = shape(x_odd, y2 - height, world)
if tile then
return tile
end
2018-05-16 11:37:50 +01:00
shape = row_down[col_i_right] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
return shape(x_even - width, y2 - height, world)
2018-02-23 22:51:28 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersgrid_pattern_offset
2018-05-08 21:23:07 +01:00
function Builders.grid_pattern_offset(pattern, columns, rows, width, height)
2018-03-27 16:16:37 +01:00
local half_width = width / 2
local half_height = height / 2
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local y2 = ((y + half_height) % height) - half_height
2019-01-23 19:55:48 -05:00
local row_pos = floor(y / height + 0.5)
2018-03-27 16:16:37 +01:00
local row_i = row_pos % rows + 1
local row = pattern[row_i] or {}
2018-05-08 17:47:34 +01:00
local x2 = ((x + half_width) % width) - half_width
2019-01-23 19:55:48 -05:00
local col_pos = floor(x / width + 0.5)
2018-03-27 16:16:37 +01:00
local col_i = col_pos % columns + 1
2019-01-23 19:55:48 -05:00
y2 = y2 + height * floor((row_pos + 1) / rows)
x2 = x2 + width * floor((col_pos + 1) / columns)
2018-05-16 11:37:50 +01:00
local shape = row[col_i] or Builders.empty_shape
2018-05-08 17:47:34 +01:00
return shape(x2, y2, world)
2018-01-21 15:31:33 +00:00
end
end
2017-08-14 09:38:14 +01:00
2017-08-12 01:20:08 +01:00
-- tile converters
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderschange_tile
2018-05-08 21:23:07 +01:00
function Builders.change_tile(shape, old_tile, new_tile)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
local tile = shape(x, y, world)
2018-05-16 11:37:50 +01:00
if type(tile) == 'table' then
if tile.tile == old_tile then
tile.tile = new_tile
end
else
if tile == old_tile then
2018-05-08 17:47:34 +01:00
tile = new_tile
end
2017-08-12 01:20:08 +01:00
end
2018-05-08 17:47:34 +01:00
return tile
2017-08-12 01:20:08 +01:00
end
end
2018-11-11 15:08:35 +00:00
local path_tiles = {
['concrete'] = true,
['hazard-concrete-left'] = true,
['hazard-concrete-right'] = true,
['stone-path'] = true,
['refined-concrete'] = true,
['refined-hazard-concrete-left'] = true,
['refined-hazard-concrete-right'] = true
}
Builders.path_tiles = path_tiles
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersset_hidden_tile
2018-11-11 15:08:35 +00:00
function Builders.set_hidden_tile(shape, hidden_tile)
return function(x, y, world)
local tile = shape(x, y, world)
if type(tile) == 'table' and path_tiles[tile.tile] then
tile.hidden_tile = hidden_tile
elseif path_tiles[tile] then
tile = {tile = tile, hidden_tile = hidden_tile}
end
return tile
end
end
local collision_map = {
2024-10-26 23:40:41 +02:00
['concrete'] = 'ground_tile',
['deepwater-green'] = 'water_tile',
['deepwater'] = 'water_tile',
['dirt-1'] = 'ground_tile',
['dirt-2'] = 'ground_tile',
['dirt-3'] = 'ground_tile',
['dirt-4'] = 'ground_tile',
['dirt-5'] = 'ground_tile',
['dirt-6'] = 'ground_tile',
['dirt-7'] = 'ground_tile',
['dry-dirt'] = 'ground_tile',
['grass-1'] = 'ground_tile',
['grass-2'] = 'ground_tile',
['grass-3'] = 'ground_tile',
['grass-4'] = 'ground_tile',
['hazard-concrete-left'] = 'ground_tile',
['hazard-concrete-right'] = 'ground_tile',
['lab-dark-1'] = 'ground_tile',
['lab-dark-2'] = 'ground_tile',
['lab-white'] = 'ground_tile',
2018-11-11 15:08:35 +00:00
['out-of-map'] = false,
2024-10-26 23:40:41 +02:00
['red-desert-0'] = 'ground_tile',
['red-desert-1'] = 'ground_tile',
['red-desert-2'] = 'ground_tile',
['red-desert-3'] = 'ground_tile',
['sand-1'] = 'ground_tile',
['sand-2'] = 'ground_tile',
['sand-3'] = 'ground_tile',
['stone-path'] = 'ground_tile',
['water-green'] = 'water_tile',
['water'] = 'water_tile',
['refined-concrete'] = 'ground_tile',
['refined-hazard-concrete-left'] = 'ground_tile',
['refined-hazard-concrete-right'] = 'ground_tile'
2018-11-11 15:08:35 +00:00
}
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderschange_collision_tile
2018-05-08 21:23:07 +01:00
function Builders.change_collision_tile(shape, collides, new_tile)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
local tile = shape(x, y, world)
2018-05-16 11:37:50 +01:00
if type(tile) == 'table' then
2018-11-11 15:08:35 +00:00
if collision_map[tile.tile] == collides then
tile.tile = new_tile
return tile
end
else
2018-11-11 15:08:35 +00:00
if collision_map[tile] == collides then
return new_tile
end
2017-08-12 01:20:08 +01:00
end
2018-05-08 17:47:34 +01:00
return tile
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderschange_map_gen_tile
2017-08-12 02:12:10 +01:00
-- only changes tiles made by the factorio map generator.
2018-05-08 21:23:07 +01:00
function Builders.change_map_gen_tile(shape, old_tile, new_tile)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local function handle_tile(tile)
2018-05-16 11:37:50 +01:00
if type(tile) == 'boolean' and tile then
2018-05-08 17:47:34 +01:00
local gen_tile = world.surface.get_tile(world.x, world.y).name
if gen_tile == old_tile then
return new_tile
end
2017-08-12 01:20:08 +01:00
end
2018-05-08 17:47:34 +01:00
return tile
end
2018-05-08 21:23:07 +01:00
local tile = shape(x, y, world)
2018-05-08 17:47:34 +01:00
if type(tile) == 'table' then
tile.tile = handle_tile(tile.tile)
else
tile = handle_tile(tile)
end
return tile
end
end
--- Docs: MISSING
-- only changes tiles made by the factorio map generator.
function Builders.change_map_gen_tiles(shape, new_tile_map)
return function(x, y, world)
local function handle_tile(tile)
if type(tile) == 'boolean' and tile then
local gen_tile = world.surface.get_tile(world.x, world.y).name
local new_tile = new_tile_map[gen_tile]
if new_tile ~= nil then
return new_tile
end
end
return tile
end
local tile = shape(x, y, world)
2018-05-16 11:37:50 +01:00
if type(tile) == 'table' then
2018-05-08 17:47:34 +01:00
tile.tile = handle_tile(tile.tile)
else
tile = handle_tile(tile)
2017-08-12 01:20:08 +01:00
end
2018-05-08 17:47:34 +01:00
return tile
2017-08-12 01:20:08 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderschange_map_gen_hidden_tile
2018-11-11 15:08:35 +00:00
function Builders.change_map_gen_hidden_tile(shape, old_tile, hidden_tile)
return function(x, y, world)
local function is_collides()
local gen_tile = world.surface.get_tile(world.x, world.y)
return gen_tile.name == old_tile
end
local tile = shape(x, y, world)
2018-11-14 12:16:42 +00:00
if type(tile) == 'table' then
2018-11-11 15:08:35 +00:00
if path_tiles[tile.tile] and is_collides() then
tile.hidden_tile = hidden_tile
end
elseif path_tiles[tile] and is_collides() then
tile = {tile = tile, hidden_tile = hidden_tile}
end
return tile
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderschange_map_gen_collision_tile
2017-08-12 02:12:10 +01:00
-- only changes tiles made by the factorio map generator.
2018-05-08 21:23:07 +01:00
function Builders.change_map_gen_collision_tile(shape, collides, new_tile)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local function handle_tile(tile)
2018-05-16 11:37:50 +01:00
if type(tile) == 'boolean' and tile then
2018-05-08 17:47:34 +01:00
local gen_tile = world.surface.get_tile(world.x, world.y)
if gen_tile.collides_with(collides) then
return new_tile
end
2017-08-12 01:20:08 +01:00
end
return tile
end
2018-05-08 21:23:07 +01:00
local tile = shape(x, y, world)
2018-05-08 17:47:34 +01:00
2018-05-16 11:37:50 +01:00
if type(tile) == 'table' then
tile.tile = handle_tile(tile.tile)
else
tile = handle_tile(tile)
2017-08-12 01:20:08 +01:00
end
return tile
2017-08-12 01:20:08 +01:00
end
2017-08-25 15:22:22 +01:00
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderschange_map_gen_collision_hidden_tile
2018-11-11 15:08:35 +00:00
function Builders.change_map_gen_collision_hidden_tile(shape, collides, hidden_tile)
return function(x, y, world)
local function is_collides()
local gen_tile = world.surface.get_tile(world.x, world.y)
return gen_tile.collides_with(collides)
end
local tile = shape(x, y, world)
if type(tile) == 'table' then
if path_tiles[tile.tile] and is_collides() then
tile.hidden_tile = hidden_tile
end
elseif path_tiles[tile] and is_collides() then
tile = {tile = tile, hidden_tile = hidden_tile}
end
return tile
end
end
2018-05-10 20:42:38 +01:00
local bad_tiles = {
2018-05-16 11:37:50 +01:00
['out-of-map'] = true,
['water'] = true,
['deepwater'] = true,
['water-green'] = true,
['deepwater-green'] = true
2018-05-10 20:42:38 +01:00
}
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersoverlay_tile_land
2018-05-10 20:42:38 +01:00
function Builders.overlay_tile_land(shape, tile_shape)
return function(x, y, world)
local function handle_tile(tile)
2018-05-16 11:37:50 +01:00
if type(tile) == 'boolean' then
2024-10-26 23:40:41 +02:00
return tile and not world.surface.get_tile(world.x, world.y).collides_with('water_tile')
2018-05-10 20:42:38 +01:00
else
return not bad_tiles[tile]
end
end
local tile = shape(x, y, world)
2018-05-16 11:37:50 +01:00
if type(tile) == 'table' then
2018-05-10 20:42:38 +01:00
if handle_tile(tile.tile) then
tile.tile = tile_shape(x, y, world) or tile.tile
end
else
if handle_tile(tile) then
tile = tile_shape(x, y, world) or tile
end
end
return tile
end
end
local water_tiles = {
2018-05-16 11:37:50 +01:00
['water'] = true,
['deepwater'] = true,
['water-green'] = true,
['deepwater-green'] = true
}
2020-02-08 13:47:51 +00:00
Builders.water_tiles = water_tiles
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersfish
2018-05-08 21:23:07 +01:00
function Builders.fish(shape, spawn_rate)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
local function handle_tile(tile)
2018-05-16 11:37:50 +01:00
if type(tile) == 'string' then
2019-01-23 19:55:48 -05:00
if water_tiles[tile] and spawn_rate >= random() then
2018-05-16 11:37:50 +01:00
return {name = 'fish'}
end
elseif tile then
2024-10-26 23:40:41 +02:00
if world.surface.get_tile(world.x, world.y).collides_with('water_tile') and spawn_rate >= random() then
2018-05-16 11:37:50 +01:00
return {name = 'fish'}
end
end
end
2018-05-08 21:23:07 +01:00
local tile = shape(x, y, world)
2018-05-16 11:37:50 +01:00
if type(tile) == 'table' then
local entity = handle_tile(tile.tile)
if entity then
add_entity(tile, entity)
end
else
local entity = handle_tile(tile)
if entity then
tile = {
tile = tile,
entities = {entity}
}
2018-01-29 17:43:26 +00:00
end
end
return tile
2018-01-29 17:43:26 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersapply_effect
2018-05-08 21:23:07 +01:00
function Builders.apply_effect(shape, func)
2018-05-08 17:47:34 +01:00
return function(x, y, world)
2018-05-08 21:23:07 +01:00
local tile = shape(x, y, world)
2018-11-13 16:21:08 +00:00
if not tile then
return tile
end
2018-05-08 17:47:34 +01:00
return func(x, y, world, tile)
2017-08-25 15:22:22 +01:00
end
end
2018-03-03 12:54:55 +00:00
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersmanhattan_value
2018-05-08 21:23:07 +01:00
function Builders.manhattan_value(base, mult)
2018-03-03 12:54:55 +00:00
return function(x, y)
2019-01-23 19:55:48 -05:00
return mult * (abs(x) + abs(y)) + base
2018-03-03 12:54:55 +00:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#builderseuclidean_value
2018-05-08 21:23:07 +01:00
function Builders.euclidean_value(base, mult)
2018-03-03 12:54:55 +00:00
return function(x, y)
2019-01-23 19:55:48 -05:00
return mult * sqrt(x * x + y * y) + base
2018-03-03 12:54:55 +00:00
end
end
2018-05-08 21:23:07 +01:00
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersexponential_value
2018-08-25 12:54:55 +01:00
function Builders.exponential_value(base, mult, pow)
return function(x, y)
2018-10-02 21:58:22 +02:00
local d_sq = x * x + y * y
return base + mult * d_sq ^ (pow / 2)
2018-08-25 12:54:55 +01:00
end
end
--- Docs: https://github.com/Refactorio/RedMew/wiki/Using-the-Builders#buildersprepare_weighted_array
2018-08-07 00:14:54 +01:00
function Builders.prepare_weighted_array(array)
local total = 0
local weights = {}
2019-01-23 23:32:14 -05:00
local weight_counter = 1
2018-08-07 00:14:54 +01:00
for _, v in ipairs(array) do
total = total + v.weight
2019-01-23 23:32:14 -05:00
weights[weight_counter] = total
weight_counter = weight_counter + 1
2018-08-07 00:14:54 +01:00
end
weights.total = total
return weights
end
2018-05-08 21:23:07 +01:00
return Builders