From aef4fe4380eec250758a888892b0d6b297aa53f5 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Tue, 23 Feb 2021 00:19:24 +0000 Subject: [PATCH 01/19] fillcaves: treat liquids as air --- CHANGELOG.md | 1 + worldeditadditions/lib/fillcaves.lua | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 747829a..9bf1ec6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,6 +15,7 @@ It's about time I started a changelog! This will serve from now on as the main c - If you encounter any other issues with it over large areas (particularly 2000x150x2000 and larger), please let me know - Bugfix: Fix obscure crash in calls to `human_size` ("unknown" will now be returned if passed junk) - `//many` can now be used with commands with no arguments. + - `//fillcaves`: Treat liquids as air when filling caves in ## v1.10 (16th January 2021) - `//maze`: Fix some parts of generated mazes staying solid diff --git a/worldeditadditions/lib/fillcaves.lua b/worldeditadditions/lib/fillcaves.lua index debb983..5a1706f 100644 --- a/worldeditadditions/lib/fillcaves.lua +++ b/worldeditadditions/lib/fillcaves.lua @@ -27,7 +27,7 @@ function worldeditadditions.fillcaves(pos1, pos2, node_name) for y = pos2.y, pos1.y, -1 do local i = area:index(x, y, z) - local is_air = worldeditadditions.is_airlike(data[i]) + local is_air = worldeditadditions.is_airlike(data[i]) or worldeditadditions.is_liquidlike(data[i]) local is_ignore = data[i] == node_id_ignore -- If the previous node was air and this one isn't, then we've found the top level From 4ba59a6ec3d23a8c6c2e86776bf987b3433e36d0 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Tue, 23 Feb 2021 00:20:57 +0000 Subject: [PATCH 02/19] //conv, //erode: treat liquids as air --- CHANGELOG.md | 3 ++- worldeditadditions/utils/terrain.lua | 2 +- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 9bf1ec6..7f5ecd8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,7 +15,8 @@ It's about time I started a changelog! This will serve from now on as the main c - If you encounter any other issues with it over large areas (particularly 2000x150x2000 and larger), please let me know - Bugfix: Fix obscure crash in calls to `human_size` ("unknown" will now be returned if passed junk) - `//many` can now be used with commands with no arguments. - - `//fillcaves`: Treat liquids as air when filling caves in + - `//conv`, `//erode`, `//fillcaves`: Treat liquids as air + ## v1.10 (16th January 2021) - `//maze`: Fix some parts of generated mazes staying solid diff --git a/worldeditadditions/utils/terrain.lua b/worldeditadditions/utils/terrain.lua index 4032024..2875309 100644 --- a/worldeditadditions/utils/terrain.lua +++ b/worldeditadditions/utils/terrain.lua @@ -20,7 +20,7 @@ function worldeditadditions.make_heightmap(pos1, pos2, manip, area, data) -- Scan each column top to bottom for y = pos2.y+1, pos1.y, -1 do local i = area:index(x, y, z) - if not worldeditadditions.is_airlike(data[i]) then + if not (worldeditadditions.is_airlike(data[i]) or worldeditadditions.is_liquidlike(data[i])) then -- It's the first non-airlike node in this column -- Start heightmap values from 1 (i.e. there's at least 1 node in the column) heightmap[hi] = (y - pos1.y) + 1 From f2aa87b3966939bd20804e16ee1a7b57de739f67 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Tue, 23 Feb 2021 01:31:24 +0000 Subject: [PATCH 03/19] terrain: pick the node to replace with from higher up in the column in question ....unless we hit the ignore node, in which case we default to air. --- worldeditadditions/utils/terrain.lua | 18 ++++++++++++++++-- 1 file changed, 16 insertions(+), 2 deletions(-) diff --git a/worldeditadditions/utils/terrain.lua b/worldeditadditions/utils/terrain.lua index 2875309..e54b254 100644 --- a/worldeditadditions/utils/terrain.lua +++ b/worldeditadditions/utils/terrain.lua @@ -82,6 +82,7 @@ end function worldeditadditions.apply_heightmap_changes(pos1, pos2, area, data, heightmap_old, heightmap_new, heightmap_size) local stats = { added = 0, removed = 0 } local node_id_air = minetest.get_content_id("air") + local node_id_ignore = minetest.get_content_id("ignore") for z = heightmap_size[0], 0, -1 do for x = heightmap_size[1], 0, -1 do @@ -95,12 +96,23 @@ function worldeditadditions.apply_heightmap_changes(pos1, pos2, area, data, heig if height_old == height_new then -- noop elseif height_new < height_old then + local node_id_replace = data[area:index( + pos1.x + x, + pos1.y + height_old + 1, + pos1.z + z + )] + -- Unlikely, but if it can happen, it *will* happen..... + if node_id_replace == node_id_ignore then + node_id_replace = node_id_air + end stats.removed = stats.removed + (height_old - height_new) local y = height_new while y < height_old do local ci = area:index(pos1.x + x, pos1.y + y, pos1.z + z) -- print("[conv/save] remove at y", y, "→", pos1.y + y, "current:", minetest.get_name_from_content_id(data[ci])) - data[ci] = node_id_air + if data[ci] ~= node_id_ignore then + data[ci] = node_id_replace + end y = y + 1 end else -- height_new > height_old @@ -113,7 +125,9 @@ function worldeditadditions.apply_heightmap_changes(pos1, pos2, area, data, heig while y < height_new do local ci = area:index(pos1.x + x, pos1.y + y, pos1.z + z) -- print("[conv/save] add at y", y, "→", pos1.y + y, "current:", minetest.get_name_from_content_id(data[ci])) - data[ci] = node_id + if data[ci] ~= node_id_ignore then + data[ci] = node_id + end y = y + 1 end end From 78572e453e674f276520c5df0bc5fc7f16160a6f Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Tue, 23 Feb 2021 22:56:56 +0000 Subject: [PATCH 04/19] Implement point cloud wand --- worldeditadditions/utils/vector.lua | 33 ++++++++++ .../edit/worldedit_wand.piskel | 2 +- worldeditadditions_farwand/init.lua | 2 + worldeditadditions_farwand/lib/cloudwand.lua | 58 ++++++++++++++++++ worldeditadditions_farwand/lib/do_raycast.lua | 17 +++++ .../textures/worldeditadditions_cloudwand.png | Bin 0 -> 181 bytes 6 files changed, 111 insertions(+), 1 deletion(-) create mode 100644 worldeditadditions_farwand/lib/cloudwand.lua create mode 100644 worldeditadditions_farwand/lib/do_raycast.lua create mode 100644 worldeditadditions_farwand/textures/worldeditadditions_cloudwand.png diff --git a/worldeditadditions/utils/vector.lua b/worldeditadditions/utils/vector.lua index b7c2357..75f2b75 100644 --- a/worldeditadditions/utils/vector.lua +++ b/worldeditadditions/utils/vector.lua @@ -40,3 +40,36 @@ function worldeditadditions.vector.floor(v) -- Some vectors are 2d if v.z then v.z = math.floor(v.z) end end + +--- Determines if the target point is contained within the defined worldedit region. +-- @param pos1 Vector pos1 of the defined region. +-- @param pos2 Vector pos2 of the defined region. +-- @param target Vector The target vector to check. +-- @return boolean Whether the given target is contained within the defined worldedit region. +function worldeditadditions.vector.is_contained(pos1, pos2, target) + local pos1, pos2 = worldedit.sort_pos(pos1, pos2) + return pos1.x >= target.x + and pos1.y >= target.y + and pos1.z >= target.z + and pos2.x <= target.x + and pos2.y <= target.y + and pos2.z <= target.z +end + +--- Expands the defined region to include the given point. +-- @param pos1 Vector pos1 of the defined region. +-- @param pos2 Vector pos2 of the defined region. +-- @param target Vector The target vector to include. +function worldeditadditions.vector.expand_region(pos1, pos2, target) + local pos1, pos2 = worldedit.sort_pos(pos1, pos2) + + if target.x < pos1.x then pos1.x = target.x end + if target.y < pos1.y then pos1.y = target.y end + if target.z < pos1.z then pos1.z = target.z end + + if target.x > pos2.x then pos2.x = target.x end + if target.y > pos2.y then pos2.y = target.y end + if target.z > pos2.z then pos2.z = target.z end + + return pos1, pos2 +end diff --git a/worldeditadditions_farwand/edit/worldedit_wand.piskel b/worldeditadditions_farwand/edit/worldedit_wand.piskel index 4960e21..5a41af1 100644 --- a/worldeditadditions_farwand/edit/worldedit_wand.piskel +++ b/worldeditadditions_farwand/edit/worldedit_wand.piskel @@ -1 +1 @@ -{"modelVersion":2,"piskel":{"name":"worldedit_wand","description":"","fps":0,"height":16,"width":16,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":3,\"chunks\":[{\"layout\":[[0],[1],[2]],\"base64PNG\":\"\"}]}"]}} \ No newline at end of file +{"modelVersion":2,"piskel":{"name":"worldedit_wand","description":"","fps":0,"height":16,"width":16,"layers":["{\"name\":\"Layer 1\",\"opacity\":1,\"frameCount\":4,\"chunks\":[{\"layout\":[[0],[1],[2],[3]],\"base64PNG\":\"\"}]}"]}} \ No newline at end of file diff --git a/worldeditadditions_farwand/init.lua b/worldeditadditions_farwand/init.lua index 26cc902..c569ede 100644 --- a/worldeditadditions_farwand/init.lua +++ b/worldeditadditions_farwand/init.lua @@ -4,6 +4,8 @@ worldeditadditions.farwand = { local modpath = minetest.get_modpath("worldeditadditions_farwand") +dofile(modpath.."/lib/do_raycast.lua") dofile(modpath.."/lib/farwand.lua") +dofile(modpath.."/lib/cloudwand.lua") dofile(modpath.."/lib/chatcommand.lua") dofile(modpath.."/lib/settings.lua") diff --git a/worldeditadditions_farwand/lib/cloudwand.lua b/worldeditadditions_farwand/lib/cloudwand.lua new file mode 100644 index 0000000..b65dec4 --- /dev/null +++ b/worldeditadditions_farwand/lib/cloudwand.lua @@ -0,0 +1,58 @@ +local function add_point(name, pos) + if pos ~= nil then + -- print("[set_pos1]", name, "("..pos.x..", "..pos.y..", "..pos.z..")") + if not worldedit.pos1[name] then worldedit.pos1[name] = vector.new(pos) end + if not worldedit.pos2[name] then worldedit.pos2[name] = vector.new(pos) end + + worldedit.marker_update(name) + + local volume_before = worldedit.volume(worldedit.pos1[name], worldedit.pos2[name]) + + worldedit.pos1[name], worldedit.pos2[name] = worldeditadditions.vector.expand_region(worldedit.pos1[name], worldedit.pos2[name], pos) + + local volume_after = worldedit.volume(worldedit.pos1[name], worldedit.pos2[name]) + + local volume_difference = volume_after - volume_before + + worldedit.marker_update(name) + worldedit.player_notify(name, "Expanded region by "..volume_difference.." nodes") + else + worldedit.player_notify(name, "Error: Too far away (try raising your maxdist with //farwand maxdist )") + -- print("[set_pos1]", name, "nil") + end +end +local function clear_points(name, pos) + worldedit.pos1[name] = nil + worldedit.pos2[name] = nil + worldedit.marker_update(name) + worldedit.set_pos[name] = nil +end + +minetest.register_tool(":worldeditadditions:cloudwand", { + description = "WorldEditAdditions far-reaching point cloud wand", + inventory_image = "worldeditadditions_cloudwand.png", + + on_place = function(itemstack, player, pointed_thing) + local name = player:get_player_name() + -- print("[farwand] on_place", name) + -- Right click when pointing at something + -- Pointed thing: https://rubenwardy.com/minetest_modding_book/lua_api.html#pointed_thing + clear_points(name) + end, + + on_use = function(itemstack, player, pointed_thing) + local name = player:get_player_name() + -- print("[farwand] on_use", name) + local looking_pos, node_id = worldeditadditions.farwand.do_raycast(player) + add_point(name, looking_pos) + -- Left click when pointing at something or nothing + end, + + on_secondary_use = function(itemstack, player, pointed_thing) + local name = player:get_player_name() + -- Right click when pointing at nothing + -- print("[farwand] on_secondary_use", name) + + clear_points(name) + end +}) diff --git a/worldeditadditions_farwand/lib/do_raycast.lua b/worldeditadditions_farwand/lib/do_raycast.lua new file mode 100644 index 0000000..11c3c5c --- /dev/null +++ b/worldeditadditions_farwand/lib/do_raycast.lua @@ -0,0 +1,17 @@ + +--- worldeditadditions.raycast() wrapper +function worldeditadditions.farwand.do_raycast(player) + if player == nil then return nil end + local player_name = player:get_player_name() + + if worldeditadditions.farwand.player_data[player_name] == nil then + worldeditadditions.farwand.player_data[player_name] = { maxdist = 1000, skip_liquid = true } + end + + local looking_pos, node_id = worldeditadditions.raycast( + player, + worldeditadditions.farwand.player_data[player_name].maxdist, + worldeditadditions.farwand.player_data[player_name].skip_liquid + ) + return looking_pos, node_id +end diff --git a/worldeditadditions_farwand/textures/worldeditadditions_cloudwand.png b/worldeditadditions_farwand/textures/worldeditadditions_cloudwand.png new file mode 100644 index 0000000000000000000000000000000000000000..fb3c1fc0b44d39090af4ac1e8cfee21d60fda019 GIT binary patch literal 181 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!73?$#)eFPFv5AX?b1=3}v9L?4ov%LftEZw#6M_dL4*kThZij=SxQcx75|bErN%ZEMu6O;?-m%A>ia@pqR?2(R@)rO~q%9 a0K>}J9QWj(gc}3RVeoYIb6Mw<&;$Tr&^}oJ literal 0 HcmV?d00001 From a6b34f9311e24bea33e632077f835ef53c2775c4 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Tue, 23 Feb 2021 23:01:17 +0000 Subject: [PATCH 05/19] Update chat command reference --- Chat-Command-Reference.md | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/Chat-Command-Reference.md b/Chat-Command-Reference.md index 617296a..03cd8a7 100644 --- a/Chat-Command-Reference.md +++ b/Chat-Command-Reference.md @@ -530,3 +530,10 @@ You can change the maximum range with the `maxdist` subcommand: ``` Note that the number there isn't in blocks (because hard maths). It is however proportional to the distance the wand will raycast looks for nodes, so a higher value will result in it raycasting further. + +## Cloud Wand +The cloud wand (`worldeditadditions:cloudwand`) is a another variant the above _Far Wand_. It looks like this: ![A picture of the far wand](https://raw.githubusercontent.com/sbrl/Minetest-WorldEditAdditions/master/worldeditadditions_farwand/textures/worldeditadditions_cloudwand.png) + +Unlike the other 2 wands, this wand functions in an additive manner. Left-click on a node to expand the currently defined region (creating a new one if one isn't defined already) to include that node. Right click to clear the currently defined region. + +It has the range of the _Far Wand_ mentioned above too, so you can select nodes from a great distance. It also abides by preferences set via the `//farwand` chat command. From 566b37096081a8ec9fdb978e6f5c43df1fa3d528 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Tue, 23 Feb 2021 23:04:32 +0000 Subject: [PATCH 06/19] Add cloud wand to README & changelog --- CHANGELOG.md | 1 + README.md | 1 + 2 files changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 7f5ecd8..2ba43fa 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -16,6 +16,7 @@ It's about time I started a changelog! This will serve from now on as the main c - Bugfix: Fix obscure crash in calls to `human_size` ("unknown" will now be returned if passed junk) - `//many` can now be used with commands with no arguments. - `//conv`, `//erode`, `//fillcaves`: Treat liquids as air + - Add new [cloud wand](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#cloud-wand) ## v1.10 (16th January 2021) diff --git a/README.md b/README.md index a7a2065..1239291 100644 --- a/README.md +++ b/README.md @@ -13,6 +13,7 @@ _(Do you have a cool build that you used WorldEditAdditions to build? [Get in to ## Table of Contents - [Quick Command Reference](#quick-command-reference) (including links to detailed explanations) - [Using the Far Wand](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#far-wand) + - [Using the Cloud Wand](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#cloud-wand) - [Detailed Chat Command Explanations](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md) - [Chat Command Cookbook](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Cookbook.md) - [Troubleshooting](#troubleshooting) From 478d1269a1694af7754385acde0d9ee71b18f2ea Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Tue, 23 Feb 2021 23:06:44 +0000 Subject: [PATCH 07/19] cloud wand: send chat message when clearing region --- worldeditadditions_farwand/lib/cloudwand.lua | 2 ++ 1 file changed, 2 insertions(+) diff --git a/worldeditadditions_farwand/lib/cloudwand.lua b/worldeditadditions_farwand/lib/cloudwand.lua index b65dec4..06c0582 100644 --- a/worldeditadditions_farwand/lib/cloudwand.lua +++ b/worldeditadditions_farwand/lib/cloudwand.lua @@ -26,6 +26,8 @@ local function clear_points(name, pos) worldedit.pos2[name] = nil worldedit.marker_update(name) worldedit.set_pos[name] = nil + + worldedit.player_notify(name, "Region cleared") end minetest.register_tool(":worldeditadditions:cloudwand", { From a0c20a150325ace9ef25b6e65b3f55b6cc4c8bb8 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Tue, 23 Feb 2021 23:10:45 +0000 Subject: [PATCH 08/19] farwand: add too far away message It works fabulously for the cloud wand, so let's add it to the far wand too --- worldeditadditions_farwand/lib/farwand.lua | 2 ++ 1 file changed, 2 insertions(+) diff --git a/worldeditadditions_farwand/lib/farwand.lua b/worldeditadditions_farwand/lib/farwand.lua index 4d41eac..649654b 100644 --- a/worldeditadditions_farwand/lib/farwand.lua +++ b/worldeditadditions_farwand/lib/farwand.lua @@ -5,6 +5,7 @@ local function set_pos1(name, pos) worldedit.mark_pos1(name) worldedit.player_notify(name, "pos1 set to "..worldeditadditions.vector.tostring(pos)) else + worldedit.player_notify(name, "Error: Too far away (try raising your maxdist with //farwand maxdist )") -- print("[set_pos1]", name, "nil") end end @@ -15,6 +16,7 @@ local function set_pos2(name, pos) worldedit.mark_pos2(name) worldedit.player_notify(name, "pos2 set to "..worldeditadditions.vector.tostring(pos)) else + worldedit.player_notify(name, "Error: Too far away (try raising your maxdist with //farwand maxdist )") -- print("[set_pos2]", name, "nil") end end From 756bd4947075a2e9b64b1b944e157c1bc16bd717 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Thu, 25 Feb 2021 22:52:47 +0000 Subject: [PATCH 09/19] //scale: remove debug statement --- worldeditadditions_commands/commands/scale.lua | 1 - 1 file changed, 1 deletion(-) diff --git a/worldeditadditions_commands/commands/scale.lua b/worldeditadditions_commands/commands/scale.lua index 66a31be..d3e724e 100644 --- a/worldeditadditions_commands/commands/scale.lua +++ b/worldeditadditions_commands/commands/scale.lua @@ -28,7 +28,6 @@ worldedit.register_command("scale", { privs = { worldedit = true }, require_pos = 2, parse = function(params_text) - print("[DEBUG//scale] got params_text '"..params_text.."'") if not params_text then params_text = "" end local parts = worldeditadditions.split(params_text, "%s+", false) From 56380f149bd968ce88e4337aa3af6ce5e475ff7f Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Fri, 26 Feb 2021 02:20:53 +0000 Subject: [PATCH 10/19] heightmap_size: refactor to use { x, z } instead of [0] / [1] --- worldeditadditions/lib/conv/conv.lua | 6 +--- worldeditadditions/lib/conv/convolve.lua | 14 ++++---- worldeditadditions/lib/erode/erode.lua | 9 ++--- worldeditadditions/lib/erode/snowballs.lua | 10 +++--- worldeditadditions/utils/terrain.lua | 39 ++++++++++++---------- 5 files changed, 40 insertions(+), 38 deletions(-) diff --git a/worldeditadditions/lib/conv/conv.lua b/worldeditadditions/lib/conv/conv.lua index 93b0dbc..9489058 100644 --- a/worldeditadditions/lib/conv/conv.lua +++ b/worldeditadditions/lib/conv/conv.lua @@ -57,13 +57,9 @@ function worldeditadditions.convolve(pos1, pos2, kernel, kernel_size) local node_id_air = minetest.get_content_id("air") - local heightmap = worldeditadditions.make_heightmap(pos1, pos2, manip, area, data) + local heightmap, heightmap_size = worldeditadditions.make_heightmap(pos1, pos2, manip, area, data) local heightmap_conv = worldeditadditions.shallowcopy(heightmap) - local heightmap_size = {} - heightmap_size[0] = (pos2.z - pos1.z) + 1 - heightmap_size[1] = (pos2.x - pos1.x) + 1 - worldeditadditions.conv.convolve( heightmap_conv, heightmap_size, diff --git a/worldeditadditions/lib/conv/convolve.lua b/worldeditadditions/lib/conv/convolve.lua index 9404bf0..bcf3e4b 100644 --- a/worldeditadditions/lib/conv/convolve.lua +++ b/worldeditadditions/lib/conv/convolve.lua @@ -18,18 +18,18 @@ function worldeditadditions.conv.convolve(heightmap, heightmap_size, matrix, mat border_size[1] = (matrix_size[1]-1) / 2 -- width -- print("[convolve] matrix_size", matrix_size[0], matrix_size[1]) -- print("[convolve] border_size", border_size[0], border_size[1]) - -- print("[convolve] heightmap_size: ", heightmap_size[0], heightmap_size[1]) + -- print("[convolve] heightmap_size: ", heightmap_size.z, heightmap_size.x) -- - -- print("[convolve] z: from", (heightmap_size[0]-border_size[0]) - 1, "to", border_size[0], "step", -1) - -- print("[convolve] x: from", (heightmap_size[1]-border_size[1]) - 1, "to", border_size[1], "step", -1) + -- print("[convolve] z: from", (heightmap_size.z-border_size[0]) - 1, "to", border_size[0], "step", -1) + -- print("[convolve] x: from", (heightmap_size.x-border_size[1]) - 1, "to", border_size[1], "step", -1) -- Convolve over only the bit that allows us to use the full convolution matrix - for z = (heightmap_size[0]-border_size[0]) - 1, border_size[0], -1 do - for x = (heightmap_size[1]-border_size[1]) - 1, border_size[1], -1 do + for z = (heightmap_size.z-border_size[0]) - 1, border_size[0], -1 do + for x = (heightmap_size.x-border_size[1]) - 1, border_size[1], -1 do local total = 0 - local hi = (z * heightmap_size[1]) + x + local hi = (z * heightmap_size.x) + x -- print("[convolve/internal] z", z, "x", x, "hi", hi) -- No continue statement in Lua :-/ @@ -40,7 +40,7 @@ function worldeditadditions.conv.convolve(heightmap, heightmap_size, matrix, mat local cz = z + (mz - border_size[0]) local cx = x + (mx - border_size[1]) - local i = (cz * heightmap_size[1]) + cx + local i = (cz * heightmap_size.x) + cx -- A value of -1 = nothing in this column (so we should ignore it) if heightmap[i] ~= -1 then diff --git a/worldeditadditions/lib/erode/erode.lua b/worldeditadditions/lib/erode/erode.lua index f85e51b..4fb5127 100644 --- a/worldeditadditions/lib/erode/erode.lua +++ b/worldeditadditions/lib/erode/erode.lua @@ -9,9 +9,10 @@ function worldeditadditions.erode.run(pos1, pos2, algorithm, params) local manip, area = worldedit.manip_helpers.init(pos1, pos2) local data = manip:get_data() - local heightmap_size = {} - heightmap_size[0] = (pos2.z - pos1.z) + 1 - heightmap_size[1] = (pos2.x - pos1.x) + 1 + local heightmap_size = { + z = (pos2.z - pos1.z) + 1, + x = (pos2.x - pos1.x) + 1 + } local region_height = (pos2.y - pos1.y) + 1 @@ -20,7 +21,7 @@ function worldeditadditions.erode.run(pos1, pos2, algorithm, params) -- print("[erode.run] algorithm: "..algorithm..", params:"); -- print(worldeditadditions.map_stringify(params)) - -- worldeditadditions.print_2d(heightmap, heightmap_size[1]) + -- worldeditadditions.print_2d(heightmap, heightmap_size.x) local success, msg, stats if algorithm == "snowballs" then success, msg = worldeditadditions.erode.snowballs(heightmap, heightmap_eroded, heightmap_size, region_height, params) diff --git a/worldeditadditions/lib/erode/snowballs.lua b/worldeditadditions/lib/erode/snowballs.lua index 7fcc646..ced0766 100644 --- a/worldeditadditions/lib/erode/snowballs.lua +++ b/worldeditadditions/lib/erode/snowballs.lua @@ -18,11 +18,11 @@ local function snowball(heightmap, normalmap, heightmap_size, startpos, params) for i = 1, params.max_steps do local x = pos.x local z = pos.z - local hi = math.floor(z+0.5)*heightmap_size[1] + math.floor(x+0.5) + local hi = math.floor(z+0.5)*heightmap_size.x + math.floor(x+0.5) -- Stop if we go out of bounds if x < 0 or z < 0 - or x >= heightmap_size[1]-1 or z >= heightmap_size[0]-1 then - -- print("[snowball] hit edge; stopping at ("..x..", "..z.."), (bounds @ "..(heightmap_size[1]-1)..", "..(heightmap_size[0]-1)..")", "x", x, "/", heightmap_size[1]-1, "z", z, "/", heightmap_size[0]-1) + or x >= heightmap_size.x-1 or z >= heightmap_size.z-1 then + -- print("[snowball] hit edge; stopping at ("..x..", "..z.."), (bounds @ "..(heightmap_size.x-1)..", "..(heightmap_size.z-1)..")", "x", x, "/", heightmap_size.x-1, "z", z, "/", heightmap_size.z-1) return true, i end @@ -100,8 +100,8 @@ function worldeditadditions.erode.snowballs(heightmap_initial, heightmap, height local success, steps = snowball( heightmap, normals, heightmap_size, { - x = math.random() * (heightmap_size[1] - 1), - z = math.random() * (heightmap_size[0] - 1) + x = math.random() * (heightmap_size.x - 1), + z = math.random() * (heightmap_size.z - 1) }, params ) diff --git a/worldeditadditions/utils/terrain.lua b/worldeditadditions/utils/terrain.lua index e54b254..4e42d94 100644 --- a/worldeditadditions/utils/terrain.lua +++ b/worldeditadditions/utils/terrain.lua @@ -7,7 +7,7 @@ -- @param manip VoxelManip The VoxelManip object. -- @param area area The associated area object. -- @param data table The associated data object. --- @return table The ZERO-indexed heightmap data (as 1 single flat array). +-- @return table,table The ZERO-indexed heightmap data (as 1 single flat array), followed by the size of the heightmap in the form { 0 = size_z, 1 = size_x }. function worldeditadditions.make_heightmap(pos1, pos2, manip, area, data) -- z y x (in reverse for little-endian machines) is the preferred loop order, but that isn't really possible here @@ -34,7 +34,12 @@ function worldeditadditions.make_heightmap(pos1, pos2, manip, area, data) end end - return heightmap + local heightmap_size = { + z = (pos2.z - pos1.z) + 1, + x = (pos2.x - pos1.x) + 1 + } + + return heightmap, heightmap_size end --- Calculates a normal map for the given heightmap. @@ -45,27 +50,27 @@ end -- @param heightmap_size int[] The size of the heightmap in the form [ z, x ] -- @return Vector[] The calculated normal map, in the same form as the input heightmap. Each element of the array is a 3D Vector (i.e. { x, y, z }) representing a normal. function worldeditadditions.calculate_normals(heightmap, heightmap_size) - -- print("heightmap_size: "..heightmap_size[1].."x"..heightmap_size[0]) + -- print("heightmap_size: "..heightmap_size.x.."x"..heightmap_size.z) local result = {} - for z = heightmap_size[0]-1, 0, -1 do - for x = heightmap_size[1]-1, 0, -1 do + for z = heightmap_size.z-1, 0, -1 do + for x = heightmap_size.x-1, 0, -1 do -- Algorithm ref https://stackoverflow.com/a/13983431/1460422 -- Also ref Vector.mjs, which I implemented myself (available upon request) - local hi = z*heightmap_size[1] + x + local hi = z*heightmap_size.x + x -- Default to this pixel's height local up = heightmap[hi] local down = heightmap[hi] local left = heightmap[hi] local right = heightmap[hi] - if z - 1 > 0 then up = heightmap[(z-1)*heightmap_size[1] + x] end - if z + 1 < heightmap_size[0]-1 then down = heightmap[(z+1)*heightmap_size[1] + x] end - if x - 1 > 0 then left = heightmap[z*heightmap_size[1] + (x-1)] end - if x + 1 < heightmap_size[1]-1 then right = heightmap[z*heightmap_size[1] + (x+1)] end + if z - 1 > 0 then up = heightmap[(z-1)*heightmap_size.x + x] end + if z + 1 < heightmap_size.z-1 then down = heightmap[(z+1)*heightmap_size.x + x] end + if x - 1 > 0 then left = heightmap[z*heightmap_size.x + (x-1)] end + if x + 1 < heightmap_size.x-1 then right = heightmap[z*heightmap_size.x + (x+1)] end - -- print("[normals] UP | index", (z-1)*heightmap_size[1] + x, "z", z, "z-1", z - 1, "up", up, "limit", 0) - -- print("[normals] DOWN | index", (z+1)*heightmap_size[1] + x, "z", z, "z+1", z + 1, "down", down, "limit", heightmap_size[1]-1) - -- print("[normals] LEFT | index", z*heightmap_size[1] + (x-1), "x", x, "x-1", x - 1, "left", left, "limit", 0) - -- print("[normals] RIGHT | index", z*heightmap_size[1] + (x+1), "x", x, "x+1", x + 1, "right", right, "limit", heightmap_size[1]-1) + -- print("[normals] UP | index", (z-1)*heightmap_size.x + x, "z", z, "z-1", z - 1, "up", up, "limit", 0) + -- print("[normals] DOWN | index", (z+1)*heightmap_size.x + x, "z", z, "z+1", z + 1, "down", down, "limit", heightmap_size.x-1) + -- print("[normals] LEFT | index", z*heightmap_size.x + (x-1), "x", x, "x-1", x - 1, "left", left, "limit", 0) + -- print("[normals] RIGHT | index", z*heightmap_size.x + (x+1), "x", x, "x+1", x + 1, "right", right, "limit", heightmap_size.x-1) result[hi] = worldeditadditions.vector.normalize({ x = left - right, @@ -84,9 +89,9 @@ function worldeditadditions.apply_heightmap_changes(pos1, pos2, area, data, heig local node_id_air = minetest.get_content_id("air") local node_id_ignore = minetest.get_content_id("ignore") - for z = heightmap_size[0], 0, -1 do - for x = heightmap_size[1], 0, -1 do - local hi = z*heightmap_size[1] + x + for z = heightmap_size.z, 0, -1 do + for x = heightmap_size.x, 0, -1 do + local hi = z*heightmap_size.x + x local height_old = heightmap_old[hi] local height_new = heightmap_new[hi] From eae005dc362c990f0517851103373e80647658be Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Fri, 26 Feb 2021 02:35:52 +0000 Subject: [PATCH 11/19] Prepare v1.11 release --- CHANGELOG.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2ba43fa..3cf4f9f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,7 +2,7 @@ It's about time I started a changelog! This will serve from now on as the main changelog for WorldEditAdditions. -## v1.11 (unreleased) +## v1.11: The big data update (25th January 2021) - Add `//scale` (currently **experimental**) - Scale operations that scale up and down at the same time are split into 2 separate operations automatically (scaling up is always performed first) - `//count`: Make numbers human-readable @@ -19,7 +19,7 @@ It's about time I started a changelog! This will serve from now on as the main c - Add new [cloud wand](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#cloud-wand) -## v1.10 (16th January 2021) +## v1.10: The tidyup update (16th January 2021) - `//maze`: Fix some parts of generated mazes staying solid - `//maze`, `//maze3d`: Allow non-number seeds (existing seeds aren't affected - they will still produce identical output) - `//many`: Improve format of progress messages, add ETA From 1034e592bd157c10f496da7eb427011a3a767534 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Fri, 26 Feb 2021 02:36:35 +0000 Subject: [PATCH 12/19] Update changelog --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 3cf4f9f..b458293 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -17,6 +17,7 @@ It's about time I started a changelog! This will serve from now on as the main c - `//many` can now be used with commands with no arguments. - `//conv`, `//erode`, `//fillcaves`: Treat liquids as air - Add new [cloud wand](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#cloud-wand) + - `//conv`, `//erode`: minor refactoring to improve code clarity ## v1.10: The tidyup update (16th January 2021) From a9603e29ec95cbe6ee014b39ce7a4ef7917564d7 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Fri, 26 Feb 2021 02:44:09 +0000 Subject: [PATCH 13/19] README: mark //scale as experimental --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 1239291..5a267b5 100644 --- a/README.md +++ b/README.md @@ -39,7 +39,7 @@ The detailed explanations have moved! Check them out [here](https://github.com/s ### Misc - [`//replacemix [] [] [ []] [ []] ....`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#replacemix-target_node-chance-replace_node_a-chance_a-replace_node_b-chance_b-replace_node_n-chance_n-) - [`//floodfill [ []]`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#floodfill-replace_node-radius-floodfill) - - [`//scale | [ [ ]]`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#scale-axis-scale_factor--factor_x-factor_y-factor_z-anchor_x-anchor_y-anchor_z) + - [`//scale | [ [ ]]`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#scale-axis-scale_factor--factor_x-factor_y-factor_z-anchor_x-anchor_y-anchor_z) **experimental** ### Terrain - [`//overlay [] [] [ []] ...`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#overlay-node_name_a-chance_a-node_name_b-chance_b-node_name_n-chance_n-) From e1755ce8e13d2ce1679f0ef09d522452049fe39f Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Mon, 1 Mar 2021 21:08:32 +0000 Subject: [PATCH 14/19] srect: localify parsing function --- .../commands/selectors/srect.lua | 55 +++++++++---------- 1 file changed, 25 insertions(+), 30 deletions(-) diff --git a/worldeditadditions_commands/commands/selectors/srect.lua b/worldeditadditions_commands/commands/selectors/srect.lua index ee25709..22b05e0 100644 --- a/worldeditadditions_commands/commands/selectors/srect.lua +++ b/worldeditadditions_commands/commands/selectors/srect.lua @@ -3,42 +3,37 @@ -- ███████ ██████ █████ ██ ██ -- ██ ██ ██ ██ ██ ██ -- ███████ ██ ██ ███████ ██████ ██ --- local -- TODO: set this to local once development is finished -function parse_params_srect(params_text) - local wea = worldeditadditions - local find = wea.split(params_text, "%s", false) - local ax1, ax2, len = find[1], find[2], find[table.maxn(find)] - - -- If ax1 is bad set to player facing dir - if ax1 == len or not ax1:match('[xyz]') then ax1 = "get" - else - local success, value = wea.getsign(ax1, "int") - if not success then return success, value - else ax1 = { value, ax1:gsub('[^xyz]',''):sub(1,1) } - end - end - -- If ax2 is bad set to +y - if not ax2 or ax2 == len or not ax2:match('[xyz]') then ax2 = "y" end - local success, value = wea.getsign(ax2, "int") - if not success then return success, value end - ax2 = { value, ax2:gsub('[^xyz]',''):sub(1,1) } - - len = tonumber(len) - -- If len == nill cancel the operation - if len == nil then - return false, "No length specified." - end - - return true, ax1, ax2, len -end worldedit.register_command("srect", { params = "[ []] ", description = "Set WorldEdit region position 2 at a set distance along 2 axes.", privs = {worldedit=true}, require_pos = 1, parse = function(params_text) - local values = {parse_params_srect(params_text)} - return unpack(values) + local wea = worldeditadditions + local find = wea.split(params_text, "%s", false) + local ax1, ax2, len = find[1], find[2], find[table.maxn(find)] + + -- If ax1 is bad set to player facing dir + if ax1 == len or not ax1:match('[xyz]') then ax1 = "get" + else + local success, value = wea.getsign(ax1, "int") + if not success then return success, value + else ax1 = { value, ax1:gsub('[^xyz]',''):sub(1,1) } + end + end + -- If ax2 is bad set to +y + if not ax2 or ax2 == len or not ax2:match('[xyz]') then ax2 = "y" end + local success, value = wea.getsign(ax2, "int") + if not success then return success, value end + ax2 = { value, ax2:gsub('[^xyz]',''):sub(1,1) } + + len = tonumber(len) + -- If len == nill cancel the operation + if len == nil then + return false, "No length specified." + end + + return true, ax1, ax2, len end, func = function(name, axis1, axis2, len) if axis1 == "get" then axis1 = worldeditadditions.player_axis2d(name) end From 9f01703744cd7897a0107ed84f9bc424c49fbb17 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Mon, 1 Mar 2021 21:14:33 +0000 Subject: [PATCH 15/19] Chat Command Reference: add sectiono on srect --- Chat-Command-Reference.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/Chat-Command-Reference.md b/Chat-Command-Reference.md index 03cd8a7..14623c6 100644 --- a/Chat-Command-Reference.md +++ b/Chat-Command-Reference.md @@ -482,6 +482,15 @@ Executes the given command, and then clips the result to the largest ellipsoid t //ellipsoidapply layers desert_sand sand 2 desert_sandstone 4 sandstone 10 ``` +## `//srect [ []] ` +Short for _select rectangle_. Sets the pos2 at a set distance along 2 axes from pos1. If the axes aren't specified, defaults to positive y and the direction you are facing. Implementation thanks to @VorTechnix. + +``` +//srect x z 10 +//srect 3 +//srect -z y 25 +``` + ## `//y` Confirms the execution of a command if it could potentially affect a large number of nodes and take a while. This is a regular WorldEdit command. From 8cce699313aaffb35ab29adb023c66b2e170376a Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Mon, 1 Mar 2021 21:16:23 +0000 Subject: [PATCH 16/19] srect: minor formatting tidyup --- worldeditadditions_commands/commands/selectors/srect.lua | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/worldeditadditions_commands/commands/selectors/srect.lua b/worldeditadditions_commands/commands/selectors/srect.lua index 22b05e0..1ddc602 100644 --- a/worldeditadditions_commands/commands/selectors/srect.lua +++ b/worldeditadditions_commands/commands/selectors/srect.lua @@ -6,7 +6,7 @@ worldedit.register_command("srect", { params = "[ []] ", description = "Set WorldEdit region position 2 at a set distance along 2 axes.", - privs = {worldedit=true}, + privs = { worldedit = true }, require_pos = 1, parse = function(params_text) local wea = worldeditadditions @@ -14,18 +14,17 @@ worldedit.register_command("srect", { local ax1, ax2, len = find[1], find[2], find[table.maxn(find)] -- If ax1 is bad set to player facing dir - if ax1 == len or not ax1:match('[xyz]') then ax1 = "get" + if ax1 == len or not ax1:match('[xyz]') then ax1 = "get" else local success, value = wea.getsign(ax1, "int") if not success then return success, value - else ax1 = { value, ax1:gsub('[^xyz]',''):sub(1,1) } - end + else ax1 = { value, ax1:gsub('[^xyz]', ''):sub(1, 1) } end end -- If ax2 is bad set to +y if not ax2 or ax2 == len or not ax2:match('[xyz]') then ax2 = "y" end local success, value = wea.getsign(ax2, "int") if not success then return success, value end - ax2 = { value, ax2:gsub('[^xyz]',''):sub(1,1) } + ax2 = { value, ax2:gsub('[^xyz]', ''):sub(1, 1) } len = tonumber(len) -- If len == nill cancel the operation From 76339239113c415e0e89aa902839d1fb2cace6ac Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Mon, 1 Mar 2021 21:17:18 +0000 Subject: [PATCH 17/19] Update readme & changelog --- CHANGELOG.md | 5 ++++- README.md | 1 + 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index b458293..cced82b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,9 @@ It's about time I started a changelog! This will serve from now on as the main changelog for WorldEditAdditions. +## v1.12 (unreleased) + - Add `//srect` (_select rectangle_) - thanks, @VorTechnix! + ## v1.11: The big data update (25th January 2021) - Add `//scale` (currently **experimental**) - Scale operations that scale up and down at the same time are split into 2 separate operations automatically (scaling up is always performed first) @@ -17,7 +20,7 @@ It's about time I started a changelog! This will serve from now on as the main c - `//many` can now be used with commands with no arguments. - `//conv`, `//erode`, `//fillcaves`: Treat liquids as air - Add new [cloud wand](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#cloud-wand) - - `//conv`, `//erode`: minor refactoring to improve code clarity + - `//conv`, `//erode`: Minor refactoring to improve code clarity ## v1.10: The tidyup update (16th January 2021) diff --git a/README.md b/README.md index 5a267b5..105dcb7 100644 --- a/README.md +++ b/README.md @@ -40,6 +40,7 @@ The detailed explanations have moved! Check them out [here](https://github.com/s - [`//replacemix [] [] [ []] [ []] ....`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#replacemix-target_node-chance-replace_node_a-chance_a-replace_node_b-chance_b-replace_node_n-chance_n-) - [`//floodfill [ []]`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#floodfill-replace_node-radius-floodfill) - [`//scale | [ [ ]]`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#scale-axis-scale_factor--factor_x-factor_y-factor_z-anchor_x-anchor_y-anchor_z) **experimental** + - [`//srect [ []] `](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#srect-axis1-axis2-length) ### Terrain - [`//overlay [] [] [ []] ...`](https://github.com/sbrl/Minetest-WorldEditAdditions/blob/master/Chat-Command-Reference.md#overlay-node_name_a-chance_a-node_name_b-chance_b-node_name_n-chance_n-) From e9e7df73b244483661d53971b98f8f039b9e26c4 Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Mon, 1 Mar 2021 21:26:09 +0000 Subject: [PATCH 18/19] Cookbook: Add @VorTechnix's suggestions Fixes #37. --- Cookbook.md | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/Cookbook.md b/Cookbook.md index 302f9f4..2835be5 100644 --- a/Cookbook.md +++ b/Cookbook.md @@ -42,3 +42,29 @@ Clearing large amounts of foliage is easy! ``` Adjust the numbers (and direction in the `//shift` command) to match your scenario. + + +## Flower Field +Make a flower field with no grass. + +``` +//overlay air 20 flowers:geranium 1 +``` + +Adjust the air value to change flower density. + +``` +//overlay air 80 flowers:rose 1 flowers:tulip 1 flowers:dandelion_yellow 1 flowers:chrysanthemum_green 1 flowers:geranium 1 flowers:viola 1 flowers:dandelion_white 1 flowers:tulip_black 1 +``` + +When working with many types of flowers the air values need to be higher to compensate. The best equation for the air value that I've found is ` * `. + + +## Grass Field +Make a grass field + +``` +//overlay air 36 default:grass_2 2 default:grass_3 2 default:grass_4 1 default:grass_5 1 +``` + +Adjust the air value to change grass density. As with flower field the best equation for the air value is ` * `. From 108c280843c8a544dba0300fabadff9e81a7002e Mon Sep 17 00:00:00 2001 From: Starbeamrainbowlabs Date: Mon, 1 Mar 2021 21:26:33 +0000 Subject: [PATCH 19/19] Start laying out stuff for our noise experiment, but it isn't hooked up yet and isn't finished. --- worldeditadditions/lib/noise/engines.lua | 3 ++ worldeditadditions/lib/noise/make_2d.lua | 9 ++++++ worldeditadditions/lib/noise/noise.lua | 3 ++ worldeditadditions/lib/noise/noise2d.lua | 38 ++++++++++++++++++++++++ 4 files changed, 53 insertions(+) create mode 100644 worldeditadditions/lib/noise/engines.lua create mode 100644 worldeditadditions/lib/noise/make_2d.lua create mode 100644 worldeditadditions/lib/noise/noise.lua create mode 100644 worldeditadditions/lib/noise/noise2d.lua diff --git a/worldeditadditions/lib/noise/engines.lua b/worldeditadditions/lib/noise/engines.lua new file mode 100644 index 0000000..7d172fd --- /dev/null +++ b/worldeditadditions/lib/noise/engines.lua @@ -0,0 +1,3 @@ +function worldeditadditions.noise.engine_perlin_2d(x, y) + +end diff --git a/worldeditadditions/lib/noise/make_2d.lua b/worldeditadditions/lib/noise/make_2d.lua new file mode 100644 index 0000000..9810d44 --- /dev/null +++ b/worldeditadditions/lib/noise/make_2d.lua @@ -0,0 +1,9 @@ + +-- ███ ███ █████ ██ ██ ███████ ██████ ██████ +-- ████ ████ ██ ██ ██ ██ ██ ██ ██ ██ +-- ██ ████ ██ ███████ █████ █████ █████ ██ ██ +-- ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ +-- ██ ██ ██ ██ ██ ██ ███████ ███████ ███████ ██████ +function worldeditadditions.noise.make_2d(size, noise_func) + -- TODO: Follow https://www.redblobgames.com/maps/terrain-from-noise/ +end diff --git a/worldeditadditions/lib/noise/noise.lua b/worldeditadditions/lib/noise/noise.lua new file mode 100644 index 0000000..30b2549 --- /dev/null +++ b/worldeditadditions/lib/noise/noise.lua @@ -0,0 +1,3 @@ +worldeditadditions.noise = {} + +dofile(worldeditadditions.modpath.."/lib/noise/alg_perlin.lua") diff --git a/worldeditadditions/lib/noise/noise2d.lua b/worldeditadditions/lib/noise/noise2d.lua new file mode 100644 index 0000000..5661628 --- /dev/null +++ b/worldeditadditions/lib/noise/noise2d.lua @@ -0,0 +1,38 @@ +--- Applies a layer of 2D noise over the terrain in the defined region. +-- @module worldeditadditions.noise2d + +-- ███ ██ ██████ ██ ███████ ███████ ██████ ██████ +-- ████ ██ ██ ██ ██ ██ ██ ██ ██ ██ +-- ██ ██ ██ ██ ██ ██ ███████ █████ █████ ██ ██ +-- ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ +-- ██ ████ ██████ ██ ███████ ███████ ███████ ██████ +--- Applies a layer of 2d noise over the terrain in the defined region. +-- @param pos1 Vector pos1 of the defined region +-- @param pos2 Vector pos2 of the defined region +-- @param noise_params table A noise parameters table. Will be passed unmodified to PerlinNoise() from the Minetest API. +function worldeditadditions.noise2d(pos1, pos2, noise_params) + pos1, pos2 = worldedit.sort_pos(pos1, pos2) + -- pos2 will always have the highest co-ordinates now + + -- Fetch the nodes in the specified area + local manip, area = worldedit.manip_helpers.init(pos1, pos2) + local data = manip:get_data() + + local heightmap_old, heightmap_size = worldeditadditions.make_heightmap( + pos1, pos2, + manip, area, + data + ) + local heightmap_new = worldeditadditions.shallowcopy(heightmap_old) + + local perlin_map = PerlinNoiseMap(noise_params, heightmap_size) + + + local stats = { added = 0, removed = 0 } + + -- Save the modified nodes back to disk & return + -- No need to save - this function doesn't actually change anything + worldedit.manip_helpers.finish(manip, data) + + return true, stats +end