-- tunnelmaker -- Another tunnel digging mod for minetest. -- Digs tunnels and builds simple bridges for advtrains track, supporting -- all 16 track directions, along with slopes up and down. -- -- by David G (kestral246@gmail.com) -- and by Mikola -- Version 2.0-beta-6 - 2019-01-24 -- Add silver_sandstone. -- Add back minetest 4.x compatibility. -- Fix knight direction so doesn't remove diagonal down steps. -- All slopes need to have one same direction level dig at top and bottom before turning. -- First pass verification of water tunnels -- fixed several bugs. Some minor issues need more review. -- Fix bug #1, in definition of dig_patterns[11]. -- Feature complete, moving from experimental to beta branch. Now on to validation. -- Config option changes WIP, they won't be permanent. -- Controls for operation ------------------------- -- Left-click: dig one node. -- Shift left-click: bring up user config menu. -- Right-click: dig tunnel based on direction player pointing. -- Shift right-click: cycle through vertical digging directions. -- Icon display based on compassgps 2.7 and compass 0.5 -- To the extent possible under law, the author(s) have dedicated all copyright and related -- and neighboring rights to this software to the public domain worldwide. This software is -- distributed without any warranty. -- You should have received a copy of the CC0 Public Domain Dedication along with this -- software. If not, see . -- User config defaults (for minetest.conf) ------------------------------------------- -- Initial digging mode for user config (1, 2, or 3). local tunnel_mode_default = tonumber(minetest.settings:get("tunnel_digging_mode") or 2) -- Train tunnels can be lined with a coating. local add_lined_tunnels_default = minetest.settings:get_bool("add_lined_tunnels", false) -- Continuous updown digging, which allows digging up/down multiple times without resetting mode. local continuous_updown_default = minetest.settings:get_bool("continuous_updown_digging", false) -- Enable desert mode - can use different materials when in the desert. Requires Minetest 5.0+. -- When desert mode is enabled, user gets additional option to Lock desert mode to current state -- of being in desert or not. Useful to define materials used when in desert transition regions. local add_desert_material = minetest.settings:get_bool("add_desert_material", false) -- Can use other lights in tunnels instead of torches. local lighting = minetest.settings:get("tunnel_lights") or "default:torch" -- Configuration variables -- (If you'd really like some of these available to minetest.conf, request them.) -------------------------- -- Tunnel height, can vary for each digging mode. local tunnel_height_general = 4 local tunnel_height_train = 6 local tunnel_height_bike = 5 -- Train tunnels (only) can have "arches" along the sides. local add_arches_config = true -- Material for walls and floors (general and train path beyond embankment). local tunnel_material = "default:stone" local tunnel_material_desert = "default:desert_stone" -- Material for train track embankment local embankment = "default:gravel" -- Material for reference marks to help laying advtrains track. -- This should be a fairly uncommon material with a distinctive look. -- If this is changed, old reference marks won't be able to be removed by tunnelmaker tool. local reference_marks = "default:stone_block" -- Time that reference marks are removed when this command enabled by the user. local remove_refs_enable_time = 120 -- Material for bike paths. local bike_path_material = "default:cobble" local slab_not_desert = "stairs:slab_cobble" local angled_slab_not_desert = "angledstairs:angled_slab_left_cobble" local angled_stair_not_desert = "angledstairs:angled_stair_left_cobble" local bike_path_material_desert = "default:desert_cobble" local slab_desert = "stairs:slab_desert_cobble" local angled_slab_desert = "angledstairs:angled_slab_left_desert_cobble" local angled_stair_desert = "angledstairs:angled_stair_left_desert_cobble" -- Material for coating for walls in the water. local glass_walls = "default:glass" -- Max height to clear trees and other brush, when clear tree cover enabled. local clear_trees_max = 30 -- Lights are placed in tunnel ceilings to light the way. local add_lighting = true -- Light spacing: 1 for default:torch, 2 for brighter lights like default:mese_post_light. -- Any mods with lights will have to be added to depends to make sure they're found here. -- Originally I used minetest.register_on_mods_loaded, but that's only a 5.0 feature. local lighting_search_radius = 1 if minetest.registered_nodes[lighting] and minetest.registered_nodes[lighting].light_source > 13 then lighting_search_radius = 2 end -- Angledstairs mod gives much nicer bike path diagonal slopes. local angledstairs_exists = false if angledstairs ~= nil then angledstairs_exists = true end -- End of configuration -- Require "tunneling" priviledge to be able to user tunnelmaker tool. minetest.register_privilege("tunneling", {description = "Allow use of tunnelmaker tool"}) -- Define top level variable to maintain per player state tunnelmaker = {} local user_config = {} -- Initialize player's state when player joins minetest.register_on_joinplayer(function(player) local pname = player:get_player_name() tunnelmaker[pname] = {updown = 0, lastdir = -1, lastpos = {x = 0, y = 0, z = 0}} if tunnel_mode_default == 1 then user_config[pname] = { digging_mode = 1, -- General purpose mode height = tunnel_height_general, add_arches = false, add_embankment = false, add_refs = false, add_floors = add_lined_tunnels_default, add_wide_floors = add_lined_tunnels_default, add_bike_ramps = false, add_lined_tunnels = add_lined_tunnels_default, continuous_updown = continuous_updown_default, lock_desert_mode = false, clear_trees = 0, remove_refs = 0, use_desert_material = add_desert_material and minetest.get_biome_data and string.match(minetest.get_biome_name(minetest.get_biome_data(player:get_pos()).biome), "desert"), coating_not_desert = tunnel_material, coating_desert = tunnel_material_desert, } elseif tunnel_mode_default == 2 then user_config[pname] = { digging_mode = 2, -- Advanced train mode height = tunnel_height_train, add_arches = add_arches_config, add_embankment = true, add_refs = true, add_floors = true, add_wide_floors = add_lined_tunnels_default, add_bike_ramps = false, add_lined_tunnels = add_lined_tunnels_default, continuous_updown = continuous_updown_default, lock_desert_mode = false, clear_trees = 0, remove_refs = 0, use_desert_material = add_desert_material and minetest.get_biome_data and string.match(minetest.get_biome_name(minetest.get_biome_data(player:get_pos()).biome), "desert"), coating_not_desert = tunnel_material, coating_desert = tunnel_material_desert, } else user_config[pname] = { digging_mode = 3, -- Bike path mode height = tunnel_height_bike, add_arches = false, add_embankment = false, add_refs = true, add_floors = true, add_wide_floors = add_lined_tunnels_default, add_bike_ramps = true, add_lined_tunnels = add_lined_tunnels_default, continuous_updown = continuous_updown_default, lock_desert_mode = false, clear_trees = 0, remove_refs = 0, use_desert_material = add_desert_material and minetest.get_biome_data and string.match(minetest.get_biome_name(minetest.get_biome_data(player:get_pos()).biome), "desert"), coating_not_desert = bike_path_material, coating_desert = bike_path_material_desert, } end end) -- Delete player's state when player leaves minetest.register_on_leaveplayer(function(player) local pname = player:get_player_name() if tunnelmaker[pname] then tunnelmaker[pname] = nil end if user_config[pname] then user_config[pname] = nil end end) local activewidth=8 -- until I can find some way to get it from minetest minetest.register_globalstep(function(dtime) local players = minetest.get_connected_players() for i,player in ipairs(players) do local gotatunnelmaker=false local wielded=false local activeinv=nil local stackidx=0 -- first check to see if the user has a tunnelmaker, because if they don't -- there is no reason to waste time calculating bookmarks or spawnpoints. local wielded_item = player:get_wielded_item():get_name() if string.sub(wielded_item, 0, 16) == "tunnelmaker:tool" then -- if the player is wielding a tunnelmaker, change the wielded image wielded=true stackidx=player:get_wield_index() gotatunnelmaker=true else -- check to see if tunnelmaker is in active inventory if player:get_inventory() then -- is there a way to only check the activewidth items instead of entire list? -- problem being that arrays are not sorted in lua for i,stack in ipairs(player:get_inventory():get_list("main")) do if i<=activewidth and string.sub(stack:get_name(), 0, 16) == "tunnelmaker:tool" then activeinv=stack -- store the stack so we can update it later with new image stackidx=i -- store the index so we can add image at correct location gotatunnelmaker=true break end end end end -- don't mess with the rest of this if they don't have a tunnelmaker if gotatunnelmaker then local pname = player:get_player_name() local dir = player:get_look_horizontal() local angle_relative = math.deg(dir) local rawdir = math.floor((angle_relative/22.5) + 0.5)%16 local distance2 = function(x, y, z) return x*x + y*y + z*z end -- Calculate distance player has moved since setting up or down local delta = distance2((player:getpos().x - tunnelmaker[pname].lastpos.x), (player:getpos().y - tunnelmaker[pname].lastpos.y), (player:getpos().z - tunnelmaker[pname].lastpos.z)) -- If rotate to different direction, or move far enough from set position, reset to horizontal if rawdir ~= tunnelmaker[pname].lastdir or (not user_config[pname].continuous_updown and delta > 0.2) then -- tune to make distance moved feel right tunnelmaker[pname].lastdir = rawdir -- tunnelmaker[pname].lastpos = pos tunnelmaker[pname].updown = 0 -- reset updown to horizontal end local tunnelmaker_image = rawdir -- horizontal digging maps to 0-15 if tunnelmaker[pname].updown ~= 0 and rawdir % 2 == 0 then -- only 0,45,90 are updown capable (U:16-23,D:24-31) tunnelmaker_image = 16 + (tunnelmaker[pname].updown - 1) * 8 + (rawdir / 2) end -- update tunnelmaker image to point at target if wielded then player:set_wielded_item("tunnelmaker:tool"..tunnelmaker_image) elseif activeinv then player:get_inventory():set_stack("main",stackidx,"tunnelmaker:tool"..tunnelmaker_image) end end end end) local images = { "tunnelmaker_0.png", "tunnelmaker_1.png", "tunnelmaker_2.png", "tunnelmaker_3.png", "tunnelmaker_4.png", "tunnelmaker_5.png", "tunnelmaker_6.png", "tunnelmaker_7.png", "tunnelmaker_8.png", "tunnelmaker_9.png", "tunnelmaker_10.png", "tunnelmaker_11.png", "tunnelmaker_12.png", "tunnelmaker_13.png", "tunnelmaker_14.png", "tunnelmaker_15.png", -- up [0, 2, .., 14] "tunnelmaker_16.png", "tunnelmaker_17.png", "tunnelmaker_18.png", "tunnelmaker_19.png", "tunnelmaker_20.png", "tunnelmaker_21.png", "tunnelmaker_22.png", "tunnelmaker_23.png", -- down [0, 2, .., 14] "tunnelmaker_24.png", "tunnelmaker_25.png", "tunnelmaker_26.png", "tunnelmaker_27.png", "tunnelmaker_28.png", "tunnelmaker_29.png", "tunnelmaker_30.png", "tunnelmaker_31.png", } -- Tests whether position is in desert-type biomes, such as desert, sandstone_desert, cold_desert, etc. -- Always just returns false if can't determine biome (i.e., using 0.4.x version). local is_desert = function(user, pos) local pname = user:get_player_name() if add_desert_material and minetest.get_biome_data then if user_config[pname].lock_desert_mode then return user_config[pname].use_desert_material else local cur_biome = minetest.get_biome_name( minetest.get_biome_data(pos).biome ) return string.match(cur_biome, "desert") end else return false end end -- Returns correct lining material based on whether to use desert or not. local lining_material = function(user, pos) local pname = user:get_player_name() if is_desert(user, pos) then return user_config[pname].coating_desert else return user_config[pname].coating_not_desert end end -- Tests whether node is flammable, mainly vegetation. local is_flammable = function(name) local group_flammable = false if minetest.registered_nodes[name] then group_flammable = minetest.registered_nodes[name].groups.flammable and minetest.registered_nodes[name].groups.flammable > 0 end return group_flammable end -- Tests whether node is a light. (Note that it could also be flammable). local is_light = function(name) if minetest.registered_nodes[name] then return minetest.registered_nodes[name].light_source > 0 else return false end end -- Lookup table to map direction to appropriate rotation for angled_stair_left nodes. local astairs_lu = { [-1] ={[2]=1,[6]=0,[10]=3,[14]=2}, -- down [1] = {[2]=3,[6]=2,[10]=1,[14]=0} -- up } local region -- Declare so I can use these functions recursively. region = { [0] = -- Null. function(x, y, z, dir, user, pointed_thing) end, [1] = -- Air. Don't delete lights or track. (Works with torches, but not with ceiling mounted lamps.) function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local name = minetest.get_node(pos).name if not (name == "air" or is_light(name) or string.match(name, "dtrack")) then minetest.set_node(pos, {name = "air"}) end end end, [2] = -- Ceiling. function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() local name = minetest.get_node(pos).name if string.match(name, "water") then -- Always line water with glass. minetest.set_node(pos, {name = glass_walls}) elseif user_config[pname].add_lined_tunnels and user_config[pname].digging_mode ~= 3 then -- Line tunnel ... if not (name == "air" or name == glass_walls or name == "default:snow" or is_flammable(name)) then -- except for these. minetest.set_node(pos, {name = lining_material(user, pos)}) end else -- Don't line tunnel, but convert sand to sandstone and gravel to cobble. if string.match(name, "default:sand") then minetest.set_node(pos, {name = "default:sandstone"}) elseif string.match(name, "default:desert_sand") then minetest.set_node(pos, {name = "default:desert_sandstone"}) elseif string.match(name, "default:silver_sand") then minetest.set_node(pos, {name = "default:silver_sandstone"}) elseif string.match(name, "default:gravel") then minetest.set_node(pos, {name = "default:cobble"}) end end if user_config[pname].clear_trees > 0 then -- Check if need to clear tree cover above dig. for i = y, clear_trees_max do local posi = vector.add(pointed_thing.under, {x=x, y=i, z=z}) local namei = minetest.get_node(posi).name if namei == "default:snow" or is_flammable(namei) then minetest.set_node(posi, {name = "air"}) elseif namei ~= "air" then break end end end end end, [3] = -- Side walls. function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() local name = minetest.get_node(pos).name if string.match(name, "water") then minetest.set_node(pos, {name = glass_walls}) -- Always line water with glass. elseif user_config[pname].add_lined_tunnels and user_config[pname].digging_mode ~= 3 then -- Line tunnel ... if not (name == "air" or name == glass_walls or name == "default:snow" or is_flammable(name) or string.match(name, "dtrack")) then -- except for these. minetest.set_node(pos, {name = lining_material(user,pos)}) end end end end, [4] = -- Temporary endcaps. function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local name = minetest.get_node(pos).name if string.match(name, "water") then -- Place temporary endcap if water. minetest.set_node(pos, {name = glass_walls}) end end end, [5] = -- Reference markers. function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() local name = minetest.get_node(pos).name -- Figure out what replacement material should be. local rep_mat if user_config[pname].add_refs then -- Add reference marks. if user_config[pname].add_embankment then rep_mat = embankment else rep_mat = lining_material(user, pos) end minetest.set_node(pos, {name = reference_marks}) local meta = minetest.get_meta(pos) meta:set_string("replace_with", rep_mat) else -- No refs. if user_config[pname].add_floors or string.match(name, "water") or name == "air" or name == glass_walls or name == "default:snow" or is_flammable(name) then minetest.set_node(pos, {name = lining_material(user, pos)}) end end end end, [6] = -- Embankment area. function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() local name = minetest.get_node(pos).name if user_config[pname].add_floors then -- Going to set all. if user_config[pname].add_embankment then minetest.set_node(pos, {name = embankment, param2 = 42}) else minetest.set_node(pos, {name = lining_material(user, pos)}) end else -- Only fill holes. if string.match(name, "water") or name == "air" or name == glass_walls or name == "default:snow" or is_flammable(name) then minetest.set_node(pos, {name = lining_material(user, pos)}) end end end end, [7] = -- Wide floors. (starting to refine) function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() local name = minetest.get_node(pos).name if user_config[pname].add_floors and user_config[pname].add_wide_floors then local param2 = minetest.get_node(pos).param2 local pos0 = vector.add(pos, {x=0, y=-1, z=0}) local node0 = minetest.get_node(pos0) if not ((node0.name == user_config[pname].coating_desert or node0.name == user_config[pname].coating_not_desert) and node0.param2 == 7) and -- Exception to match diagonal up and down digging. not (user_config[pname].add_embankment and ((name == embankment and param2 == 42) or name == reference_marks)) and -- Don't overwrite embankment or refs in train mode. not (user_config[pname].add_bike_ramps and name == reference_marks) then -- Don't overwrite refs in bike mode. minetest.set_node(pos, {name = lining_material(user, pos), param2 = 7}) end else -- Not wide. However, this makes double-wide glass when digging at water surface level. if string.match(name, "water") then minetest.set_node(pos, {name = glass_walls}) end end end end, [8] = -- Underfloor, only used directly for slope up and slope down where embankment or brace is always needed. function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then minetest.set_node(pos, {name = lining_material(user, pos)}) end end, [10] = -- Bike slope down narrow (air or angled slab). function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() local name = minetest.get_node(pos).name if user_config[pname].add_bike_ramps and angledstairs_exists then if not (name == angled_slab_desert or name == angled_slab_not_desert) then -- Don't overwrite angled_slab on ref when going diagonally down. region[1](x, y, z, dir, user, pointed_thing) end else region[1](x, y, z, dir, user, pointed_thing) end end end, [11] = -- Bike slope up or down narrow (air or angled slab). function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() if user_config[pname].add_bike_ramps and angledstairs_exists then if is_desert(user, pos) then minetest.set_node(pos, {name = angled_slab_desert, param2 = astairs_lu[dir.vert][dir.horiz]}) else minetest.set_node(pos, {name = angled_slab_not_desert, param2 = astairs_lu[dir.vert][dir.horiz]}) end else region[1](x, y, z, dir, user, pointed_thing) end end end, [12] = -- Bike slope up or down narrow (slab or angled stair). function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() if user_config[pname].add_bike_ramps then if angledstairs_exists and math.fmod(dir.horiz, 4) == 2 then -- diagonal slope if is_desert(user, pos) then minetest.set_node(pos, {name = angled_stair_desert, param2 = astairs_lu[dir.vert][dir.horiz]}) else minetest.set_node(pos, {name = angled_stair_not_desert, param2 = astairs_lu[dir.vert][dir.horiz]}) end else -- no angledstairs if is_desert(user, pos) then minetest.set_node(pos, {name = slab_desert, param2 = 2}) else minetest.set_node(pos, {name = slab_not_desert, param2 = 2}) end end else region[1](x, y, z, dir, user, pointed_thing) end end end, [13] = -- Bike slope wide up or down (slabs). function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() if user_config[pname].add_bike_ramps and user_config[pname].add_wide_floors then if is_desert(user, pos) then minetest.set_node(pos, {name = slab_desert, param2 = 2}) else minetest.set_node(pos, {name = slab_not_desert, param2 = 2}) end else region[1](x, y, z, dir, user, pointed_thing) end end end, [14] = -- Bike slope wide up or down (air or angled slab). function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() if user_config[pname].add_bike_ramps and user_config[pname].add_wide_floors and angledstairs_exists then if is_desert(user, pos) then minetest.set_node(pos, {name = angled_slab_desert, param2 = astairs_lu[dir.vert][dir.horiz]}) else minetest.set_node(pos, {name = angled_slab_not_desert, param2 = astairs_lu[dir.vert][dir.horiz]}) end else region[1](x, y, z, dir, user, pointed_thing) end end end, [19] = -- Bike slopes. Don't remove bike slopes placed by previous down slope. function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() local node = minetest.get_node(pos) if not user_config[pname].add_bike_ramps or (user_config[pname].add_bike_ramps and not ((node.name == slab_not_desert or node.name == slab_desert) and node.param2 == 2) and not (node.name == angled_slab_not_desert or node.name == angled_slab_desert)) then region[1](x, y, z, dir, user, pointed_thing) end end end, [21] = -- Arch or air, (use for arch). function(x, y, z, dir, user, pointed_thing) local pname = user:get_player_name() if user_config[pname].add_arches then -- arches region[2](x, y, z, dir, user, pointed_thing) else region[1](x, y, z, dir, user, pointed_thing) end end, [30] = -- Wall or null (based on arches). function(x, y, z, dir, user, pointed_thing) local pname = user:get_player_name() if not user_config[pname].add_arches then region[3](x, y, z, dir, user, pointed_thing) end end, [32] = -- Wall or ceiling, (use above arch). function(x, y, z, dir, user, pointed_thing) local pname = user:get_player_name() if user_config[pname].add_arches then region[3](x, y, z, dir, user, pointed_thing) else region[2](x, y, z, dir, user, pointed_thing) end end, [37] = -- Floor under wall. Only place floor under wall if wall right above floor. function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then local pname = user:get_player_name() local name = minetest.get_node(pos).name local pos1 = vector.add(pos, {x=0, y=1, z=0}) local name1 = minetest.get_node(pos1).name if string.match(name, "water") then minetest.set_node(pos, {name = glass_walls}) elseif name1 == user_config[pname].coating_not_desert or name1 == user_config[pname].coating_desert then minetest.set_node(pos, {name = lining_material(user, pos)}) end end end, [40] = -- Endcap or null (based on arches). function(x, y, z, dir, user, pointed_thing) local pname = user:get_player_name() if not user_config[pname].add_arches then region[4](x, y, z, dir, user, pointed_thing) end end, [77] = -- Add light. Need to figure out how to handle ceiling lamps. They get placed, but cleared by [1] function(x, y, z, dir, user, pointed_thing) local pos = vector.add(pointed_thing.under, {x=x, y=y, z=z}) if not minetest.is_protected(pos, user) then region[1](x, y, z, dir, user, pointed_thing) local pname = user:get_player_name() local name = minetest.get_node(pos).name local pos1 = vector.add(pos, {x=0, y=1, z=0}) --local name1 = minetest.get_node(pos1).name if name == "air" then local ceiling = minetest.get_node(pos1).name if add_lighting and (ceiling == user_config[pname].coating_not_desert or ceiling == "default:stone" or ceiling == user_config[pname].coating_desert or ceiling == "default:desert_stone" or ceiling == glass_walls) and minetest.find_node_near(pos, lighting_search_radius, {name = lighting}) == nil then minetest.set_node(pos, {name = lighting}) end end end end, [86] = -- Underfloor under embankment. function(x, y, z, dir, user, pointed_thing) local pname = user:get_player_name() if user_config[pname].add_floors and user_config[pname].add_embankment then region[8](x, y, z, dir, user, pointed_thing) end end, [87] = -- Underfloor under wide floor. function(x, y, z, dir, user, pointed_thing) local pname = user:get_player_name() if user_config[pname].add_floors and user_config[pname].add_wide_floors then region[8](x, y, z, dir, user, pointed_thing) end end, } -- Add flips and rotates so I only need to define the seven basic digging patterns. -- For flip: 1 = vertical, -1 = horizontal, 2 = both. -- For rotate: 1 = clockwise, -1 = counterclockwise. local fliprot = function(xzpos, f, r) local xzres = {xzpos[1], xzpos[2]} -- identity if f == 2 then -- double flip xzres[1] = -xzpos[1] xzres[2] = -xzpos[2] elseif f ~= 0 and r == 0 then -- single flip xzres[1] = f * xzpos[1] xzres[2] = -f * xzpos[2] elseif f == 0 and r ~= 0 then -- rotate xzres[1] = r * xzpos[2] xzres[2] = -r * xzpos[1] elseif f ~= 0 and r ~= 0 then -- flip + rotate xzres[1] = f * r * xzpos[2] xzres[2] = f * r * xzpos[1] end return xzres end local run_list = function(dir_list, f, r, dir, user, pointed_thing) local pname = user:get_player_name() local height = user_config[pname].height for _,v in ipairs(dir_list) do local newpos = fliprot(v[1], f, r) for i = 9, 6, -1 do -- ceiling region[v[2][i]](newpos[1], i + height - 7, newpos[2], dir, user, pointed_thing) end for y = (height - 2), 2, -1 do -- variable mid region repeats element 5 region[v[2][5]](newpos[1], y, newpos[2], dir, user, pointed_thing) end for i = 4, 1, -1 do -- floor region[v[2][i]](newpos[1], i-3, newpos[2], dir, user, pointed_thing) end end end -- Dig tunnel based on direction given. -- [9] = h + 2 (up ceiling) -- [8] = h + 1 (default ceiling) -- -- [7] = h (default arch) -- [6] = h - 1 (down arch) -- [5] = 2 to h - 2 (middle repeated, hmin = 3, zero instances,) -- [4] = 1 (up floor) -- -- [3] = 0 (default floor) -- [2] = -1 (default base, down floor,) -- [1] = -2 (down base) local dig_tunnel = function(cdir, user, pointed_thing) if minetest.check_player_privs(user, "tunneling") then local dig_patterns = { -- Orthogonal (north reference). [1] = { {{-3, 3},{0,0, 4, 4,4,4,4, 40,0}}, {{-2, 3},{0,0,4, 4,4,4, 4, 4,0}}, {{-1, 3},{0, 0,4, 4,4,4,4, 4,0}}, {{ 0, 3},{0, 0,4, 4,4,4, 4, 4,0}}, {{ 1, 3},{0, 0,4, 4,4,4,4, 4,0}}, {{ 2, 3},{0,0,4, 4,4,4, 4, 4,0}}, {{ 3, 3},{0,0, 4, 4,4,4,4, 40,0}}, {{-3, 2},{0,0,37, 3,3,3,3, 30,0}}, {{-2, 2},{0,0,7, 1,1,1,21, 32,0}}, {{-1, 2},{0,86,6, 1,1,1,1, 2,0}}, {{ 0, 2},{0,86,5, 1,1,1, 1, 2,0}}, {{ 1, 2},{0,86,6, 1,1,1,1, 2,0}}, {{ 2, 2},{0,0,7, 1,1,1,21, 32,0}}, {{ 3, 2},{0,0,37, 3,3,3,3, 30,0}}, {{-3, 1},{0,0,37, 3,3,3,3, 30,0}}, {{-2, 1},{0,0,7, 1,1,1,21, 32,0}}, {{-1, 1},{0,86,6, 1,1,1,1, 2,0}}, {{ 0, 1},{0,86,6, 1,1,1,77, 2,0}}, {{ 1, 1},{0,86,6, 1,1,1,1, 2,0}}, {{ 2, 1},{0,0,7, 1,1,1,21, 32,0}}, {{ 3, 1},{0,0,37, 3,3,3,3, 30,0}}, {{-3, 0},{0,0,37, 3,3,3,3, 30,0}}, {{-2, 0},{0,0,7, 1,1,1,21, 32,0}}, {{-1, 0},{0,86,6, 1,1,1,1, 2,0}}, {{ 0, 0},{0,86,5, 1,1,1, 1, 2,0}}, {{ 1, 0},{0,86,6, 1,1,1,1, 2,0}}, {{ 2, 0},{0,0,7, 1,1,1,21, 32,0}}, {{ 3, 0},{0,0,37, 3,3,3,3, 30,0}}, {{-3,-1},{0,0, 4, 4,4,4,4, 40,0}}, {{-2,-1},{0,0,4, 4,4,4, 4, 4,0}}, {{-1,-1},{0, 0,4, 4,4,4,4, 4,0}}, {{ 0,-1},{0, 0,4, 4,4,4, 4, 4,0}}, {{ 1,-1},{0, 0,4, 4,4,4,4, 4,0}}, {{ 2,-1},{0,0,4, 4,4,4, 4, 4,0}}, {{ 3,-1},{0,0, 4, 4,4,4,4, 40,0}}, }, -- Knight move (north-northwest reference). [2] = { {{-4, 3},{0,0, 4, 4,4,4,4, 40,0}}, {{-3, 3},{0,0, 4, 4,4,4, 4, 4,0}}, {{-2, 3},{0, 0,4, 4,4,4, 4, 4,0}}, {{-1, 3},{0, 0,4, 4,4,4, 4, 4,0}}, {{ 0, 3},{0, 0,4, 4,4,4,4, 4,0}}, {{ 1, 3},{0, 0,4, 4,4,4, 4, 4,0}}, {{ 2, 3},{0,0,37, 3,3,3, 3, 30,0}}, {{-4, 2},{0,0,37, 3,3,3,3, 30,0}}, {{-3, 2},{0,0, 7, 1,1,1,21, 32,0}}, {{-2, 2},{0,86,6, 1,1,1, 1, 2,0}}, {{-1, 2},{0,86,5, 1,1,1, 1, 2,0}}, {{ 0, 2},{0,86,6, 1,1,1,1, 2,0}}, {{ 1, 2},{0, 0,7, 1,1,1,21, 32,0}}, {{ 2, 2},{0,0,37, 3,3,3, 3, 3,0}}, {{ 3, 2},{0,0,37, 3,3,3,3, 30,0}}, {{-4, 1},{0,0,37, 3,3,3,3, 30,0}}, {{-3, 1},{0,0, 7, 1,1,1,21, 32,0}}, {{-2, 1},{0,86,6, 1,1,1, 1, 2,0}}, {{-1, 1},{0,86,6, 1,1,1,77, 2,0}}, {{ 0, 1},{0,86,6, 1,1,1,1, 2,0}}, {{ 1, 1},{0,86,6, 1,1,1, 1, 2,0}}, {{ 2, 1},{0,0, 7, 1,1,1,21, 32,0}}, {{ 3, 1},{0,0,37, 3,3,3,3, 30,0}}, {{-4, 0},{0,0,37, 3,3,3,3, 30,0}}, {{-3, 0},{0,0,37, 3,3,3, 3, 3,0}}, {{-2, 0},{0, 0,7, 1,1,1,21, 32,0}}, {{-1, 0},{0,86,6, 1,1,1, 1, 2,0}}, {{ 0, 0},{0,86,5, 1,1,1,1, 2,0}}, {{ 1, 0},{0,86,6, 1,1,1, 1, 2,0}}, {{ 2, 0},{0,0, 7, 19,1,1,21, 32,0}}, {{ 3, 0},{0,0,37, 3,3,3,3, 30,0}}, {{-3,-1},{0,0,37, 3,3,3, 3, 30,0}}, {{-2,-1},{0, 0,4, 4,4,4, 4, 4,0}}, {{-1,-1},{0, 0,4, 4,4,4, 4, 4,0}}, {{ 0,-1},{0, 0,4, 4,4,4,4, 4,0}}, {{ 1,-1},{0, 0,4, 4,4,4, 4, 4,0}}, {{ 2,-1},{0,0, 4, 4,4,4, 4, 4,0}}, {{ 3,-1},{0,0, 4, 4,4,4,4, 40,0}}, }, -- Diagonal (northwest reference). [3] = { {{-1, 4},{0, 0, 4, 4,4,4, 4, 40,0}}, {{ 0, 4},{0, 0,37, 3,3,3, 3, 30,0}}, {{ 1, 4},{0, 0,37, 3,3,3, 3, 30,0}}, {{-2, 3},{0, 0, 4, 4,4,4, 4, 4,0}}, {{-1, 3},{0, 0, 4, 4,4,4, 4, 4,0}}, {{ 0, 3},{0, 0, 7, 1,1,1,21, 32,0}}, {{ 1, 3},{0, 0,37, 3,3,3, 3, 2,0}}, {{ 2, 3},{0,0,37, 3,3,3,3, 30,0}}, {{-3, 2},{0,0, 4, 4,4,4, 4, 4,0}}, {{-2, 2},{0, 0, 4, 4,4,4, 4, 4,0}}, {{-1, 2},{0, 0, 4, 4,4,4, 4, 4,0}}, {{ 0, 2},{0,86, 6, 1,1,1, 1, 2,0}}, {{ 1, 2},{0, 0, 7, 1,1,1,21, 32,0}}, {{ 2, 2},{0,0,37, 3,3,3,3, 2,0}}, {{ 3, 2},{0,0,37, 3,3,3,3, 30,0}}, {{-4, 1},{0,0, 4, 4,4,4,4, 40,0}}, {{-3, 1},{0,0, 4, 4,4,4, 4, 4,0}}, {{-2, 1},{0, 0, 4, 4,4,4, 4, 4,0}}, {{-1, 1},{0,86, 5, 1,1,1,77, 2,0}}, {{ 0, 1},{0,86, 6, 1,1,1, 1, 2,0}}, {{ 1, 1},{0,86, 6, 10,1,1, 1, 2,0}}, {{ 2, 1},{0,0, 7, 19,1,1,21, 32,0}}, {{ 3, 1},{0,0,37, 3,3,3,3, 30,0}}, {{-4, 0},{0,0,37, 3,3,3,3, 30,0}}, {{-3, 0},{0,0, 7, 1,1,1,21, 32,0}}, {{-2, 0},{0,86, 6, 1,1,1, 1, 2,0}}, {{-1, 0},{0,86, 6, 1,1,1, 1, 2,0}}, {{ 0, 0},{0,86, 5, 10,1,1, 1, 2,0}}, {{ 1, 0},{0, 0, 4, 4,4,4, 4, 4,0}}, {{ 2, 0},{0,0, 4, 4,4,4,4, 4,0}}, {{ 3, 0},{0,0, 4, 4,4,4,4, 40,0}}, {{-4,-1},{0,0,37, 3,3,3,3, 30,0}}, {{-3,-1},{0,0,37, 3,3,3, 3, 2,0}}, {{-2,-1},{0, 0, 7, 1,1,1,21, 32,0}}, {{-1,-1},{0,86, 6, 10,1,1, 1, 2,0}}, {{ 0,-1},{0, 0, 4, 4,4,4, 4, 4,0}}, {{ 1,-1},{0, 0, 4, 4,4,4, 4, 4,0}}, {{-3,-2},{0,0,37, 3,3,3, 3, 30,0}}, {{-2,-2},{0, 0,37, 3,3,3, 3, 2,0}}, {{-1,-2},{0, 0, 7, 19,1,1,21, 32,0}}, {{ 0,-2},{0, 0, 4, 4,4,4, 4, 4,0}}, {{-2,-3},{0, 0,37, 3,3,3, 3, 30,0}}, {{-1,-3},{0, 0,37, 3,3,3, 3, 30,0}}, {{ 0,-3},{0, 0, 4, 4,4,4, 4, 40,0}}, }, -- Orthogonal slope down (north reference). -- Review water tunnel (-3,0), (-2,0), (2,0), (3,0) all 1 below ref level (0 and 87) -- not solid seal, but water's not going to go up. [10] = {{{-3, 3},{0, 4, 4, 4,4,4,40, 0,0}}, {{-2, 3},{0, 4, 4, 4,4, 4, 4, 0,0}}, {{-1, 3},{ 0,4, 4, 4,4,4,4, 0,0}}, {{ 0, 3},{ 0,4, 4, 4,4,4, 4, 0,0}}, {{ 1, 3},{ 0,4, 4, 4,4,4,4, 0,0}}, {{ 2, 3},{0, 4, 4, 4,4, 4, 4, 0,0}}, {{ 3, 3},{0, 4, 4, 4,4,4,40, 0,0}}, {{-3, 2},{0,37, 3, 3,3,3, 3, 30,0}}, {{-2, 2},{0, 7, 1, 1,1,21,32, 32,0}}, {{-1, 2},{86,6, 1, 1,1,1,2, 2,0}}, {{ 0, 2},{86,5, 1, 1,1,1, 2, 2,0}}, {{ 1, 2},{86,6, 1, 1,1,1,2, 2,0}}, {{ 2, 2},{0, 7, 1, 1,1,21,32, 32,0}}, {{ 3, 2},{0,37, 3, 3,3,3, 3, 30,0}}, {{-3, 1},{0,37, 3, 3,3,3, 3, 30,0}}, {{-2, 1},{0, 7,13, 1,1, 1,21, 32,0}}, {{-1, 1},{86,6,12, 1,1,1,1, 2,0}}, {{ 0, 1},{86,6,12, 1,1,1,77, 2,0}}, {{ 1, 1},{86,6,12, 1,1,1,1, 2,0}}, {{ 2, 1},{0, 7,13, 1,1, 1,21, 32,0}}, {{ 3, 1},{0,37, 3, 3,3,3, 3, 30,0}}, {{-3, 0},{0, 0,37, 3,3,3, 3, 30,0}}, {{-2, 0},{0,87, 7, 1,1, 1,21, 32,0}}, {{-1, 0},{86,8, 6, 1,1,1,1, 2,0}}, {{ 0, 0},{86,8, 5, 1,1,1, 1, 2,0}}, {{ 1, 0},{86,8, 6, 1,1,1,1, 2,0}}, {{ 2, 0},{0,87, 7, 1,1, 1,21, 32,0}}, {{ 3, 0},{0, 0,37, 3,3,3, 3, 30,0}}, {{-3,-1},{0, 0, 4, 4,4,4, 4, 40,0}}, {{-2,-1},{0, 0, 4, 4,4, 4, 4, 4,0}}, {{-1,-1},{ 0,0, 4, 4,4,4,4, 4,0}}, {{ 0,-1},{ 0,0, 4, 4,4,4, 4, 4,0}}, {{ 1,-1},{ 0,0, 4, 4,4,4,4, 4,0}}, {{ 2,-1},{0, 0, 4, 4,4, 4, 4, 4,0}}, {{ 3,-1},{0, 0, 4, 4,4,4, 4, 40,0}}, }, -- Orthogonal slope up (north reference). -- Review water tunnel (-3,2), (-2,2), (2,2), 3,2) all at ref level (0 and 87) -- not solid seal, but water's not going to go up. [11] = {{{-3, 3},{0,0, 0, 4,4,4,4, 4,40}}, {{-2, 3},{0,0, 0, 4,4,4, 4, 4, 4}}, {{-1, 3},{0, 0,0, 4,4,4,4, 4,4}}, {{ 0, 3},{0, 0,0, 4,4,4,4, 4,4}}, {{ 1, 3},{0, 0,0, 4,4,4,4, 4,4}}, {{ 2, 3},{0,0, 0, 4,4,4, 4, 4, 4}}, {{ 3, 3},{0,0, 0, 4,4,4,4, 4,40}}, {{-3, 2},{0,0, 0, 37,3,3,3, 3,30}}, {{-2, 2},{0,0,87, 7,1,1, 1, 21,32}}, {{-1, 2},{0,86,8, 6,1,1,1, 1,2}}, {{ 0, 2},{0,86,8, 5,1,1,1, 1,2}}, {{ 1, 2},{0,86,8, 6,1,1,1, 1,2}}, {{ 2, 2},{0,0,87, 7,1,1, 1, 21,32}}, {{ 3, 2},{0,0, 0, 37,3,3,3, 3,30}}, {{-3, 1},{0,0,37, 3,3,3,3, 3,30}}, {{-2, 1},{0,0, 7, 13,1,1, 1, 21,32}}, {{-1, 1},{0,86,6, 12,1,1,1, 1,2}}, {{ 0, 1},{0,86,6, 12,1,1,1, 77,2}}, {{ 1, 1},{0,86,6, 12,1,1,1, 1,2}}, {{ 2, 1},{0,0, 7, 13,1,1, 1, 21,32}}, {{ 3, 1},{0,0,37, 3,3,3,3, 3,30}}, {{-3, 0},{0,0,37, 3,3,3,3, 3,30}}, {{-2, 0},{0,0, 7, 1,1,1,21, 32, 4}}, {{-1, 0},{0,86,6, 1,1,1,1, 2,4}}, {{ 0, 0},{0,86,5, 1,1,1,1, 2,4}}, {{ 1, 0},{0,86,6, 1,1,1,1, 2,4}}, {{ 2, 0},{0,0, 7, 1,1,1,21, 32, 4}}, {{ 3, 0},{0,0,37, 3,3,3,3, 3,30}}, {{-3,-1},{0,0, 4, 4,4,4,4, 40, 0}}, {{-2,-1},{0,0, 4, 4,4,4, 4, 4, 0}}, {{-1,-1},{0, 0,4, 4,4,4,4, 4,0}}, {{ 0,-1},{0, 0,4, 4,4,4,4, 4,0}}, {{ 1,-1},{0, 0,4, 4,4,4,4, 4,0}}, {{ 2,-1},{0,0, 4, 4,4,4, 4, 4, 0}}, {{ 3,-1},{0,0, 4, 4,4,4,4, 40, 0}}, }, -- Diagonal slope down (northwest reference). -- Review water tunnel (-3,0), (0,3) stacked 32 -- with arch, overdoing seal, could raise height of 21 or remove top 32. -- Review water tunnel (-1,-2), (2,1) -- not solid seal, but water's not going to go up. [30] = { {{-1, 4},{ 0, 4, 4, 4,4,4, 4, 0,0}}, {{ 0, 4},{ 0,37, 3, 3,3, 3, 3, 0,0}}, {{ 1, 4},{ 0,37, 3, 3,3,3, 3, 0,0}}, {{-1, 3},{ 0, 4, 4, 4,4,4, 4, 4,0}}, {{ 0, 3},{ 0, 7, 1, 1,1,21,32, 32,0}}, {{ 1, 3},{ 0,37, 3, 3,3,3, 3, 2,0}}, {{ 2, 3},{0,37,3, 3,3,3, 3, 30,0}}, {{-2, 2},{ 0, 4, 4, 4,4,4, 4, 4,0}}, {{-1, 2},{ 0, 4, 4, 4,4,4, 4, 4,0}}, {{ 0, 2},{86, 6,14, 1,1, 1, 1, 2,0}}, {{ 1, 2},{ 0, 7,13, 1,1,1,21, 32,0}}, {{ 2, 2},{0,37,3, 3,3,3, 3, 2,0}}, {{ 3, 2},{0,0,37, 3,3,3,3, 30,0}}, {{-4, 1},{0, 4,4, 4,4,4,4, 0,0}}, {{-3, 1},{0, 4,4, 4,4, 4, 4, 4,0}}, {{-2, 1},{ 0, 4, 4, 4,4,4, 4, 4,0}}, {{-1, 1},{86, 5,11, 1,1,1,77, 2,0}}, {{ 0, 1},{86, 6,12, 1,1, 1, 1, 2,0}}, {{ 1, 1},{86, 8, 6, 10,1,1, 1, 2,0}}, {{ 2, 1},{0,87,7, 19,1,1,21, 32,0}}, {{ 3, 1},{0,0,37, 3,3,3,3, 30,0}}, {{-4, 0},{0,37,3, 3,3,3,3, 0,0}}, {{-3, 0},{0, 7,1, 1,1,21,32, 32,0}}, {{-2, 0},{86, 6,14, 1,1,1, 1, 2,0}}, {{-1, 0},{86, 6,12, 1,1,1, 1, 2,0}}, {{ 0, 0},{86, 8, 5, 10,1, 1, 1, 2,0}}, {{ 1, 0},{ 0, 0, 4, 4,4,4, 4, 4,0}}, {{ 2, 0},{0, 0,4, 4,4,4, 4, 4,0}}, {{ 3, 0},{0,0, 4, 4,4,4,4, 40,0}}, {{-4,-1},{0,37,3, 3,3,3,3, 0,0}}, {{-3,-1},{0,37,3, 3,3, 3, 3, 2,0}}, {{-2,-1},{ 0, 7,13, 1,1,1,21, 32,0}}, {{-1,-1},{86, 8, 6, 10,1,1, 1, 2,0}}, {{ 0,-1},{ 0, 0, 4, 4,4, 4, 4, 4,0}}, {{ 1,-1},{ 0, 0, 4, 4,4,4, 4, 4,0}}, {{-3,-2},{0,37,3, 3,3, 3, 3, 30,0}}, {{-2,-2},{ 0,37, 3, 3,3,3, 3, 2,0}}, {{-1,-2},{ 0,87, 7, 19,1,1,21, 32,0}}, {{ 0,-2},{ 0, 0, 4, 4,4, 4, 4, 4,0}}, {{-2,-3},{ 0, 0,37, 3,3,3, 3, 30,0}}, {{-1,-3},{ 0, 0,37, 3,3,3, 3, 30,0}}, {{ 0,-3},{ 0, 0, 4, 4,4, 4, 4, 40,0}}, }, -- Diagonal slope up (northwest reference). -- Review water tunnel (-2,-3), (-1,-3), (0,-3), (3,0), (3,1), (3,2) -- without arch, same as slope down, overdoing seal. -- Review water tunnel (-3,0), (0,3) -- not solid seal, but water's not going to go up. [31] = { {{-1, 4},{0, 0, 0, 4,4,4, 4, 4,40}}, {{ 0, 4},{0, 0, 0, 37,3,3,3, 3,30}}, {{ 1, 4},{0, 0,37, 3,3,3,3, 3,30}}, {{-1, 3},{0, 0, 0, 4,4,4, 4, 4, 4}}, {{ 0, 3},{0, 0,87, 7,1,1,1, 21,32}}, {{ 1, 3},{0, 0,37, 3,3,3,3, 3, 3}}, {{ 2, 3},{0,0,37, 3,3,3,3, 3,30}}, {{-2, 2},{0, 0, 0, 4,4,4,4, 4, 4}}, {{-1, 2},{0, 0, 0, 4,4,4, 4, 4, 4}}, {{ 0, 2},{0,86, 8, 6,1,1,1, 1, 2}}, {{ 1, 2},{0, 0, 7, 13,1,1,1, 21,32}}, {{ 2, 2},{0,0,37, 3,3,3,3, 3, 3}}, {{ 3, 2},{0,0,37, 3,3,3,3, 3,0}}, {{-4, 1},{0,0,0, 4,4,4,4, 4,40}}, {{-3, 1},{0,0, 0, 4,4,4,4, 4, 4}}, {{-2, 1},{0, 0, 0, 4,4,4,4, 4, 4}}, {{-1, 1},{0,86, 8, 5,1,1, 1, 77, 2}}, {{ 0, 1},{0,86, 6, 12,1,1,1, 1, 2}}, {{ 1, 1},{0,86, 6, 14,1,1,1, 1, 2}}, {{ 2, 1},{0,0, 7, 1,1,1,21, 32, 4}}, {{ 3, 1},{0,0,37, 3,3,3,3, 3,0}}, {{-4, 0},{0,0,0, 37,3,3,3, 3,30}}, {{-3, 0},{0,0,87, 7,1,1,1, 21,32}}, {{-2, 0},{0,86, 8, 6,1,1,1, 1, 2}}, {{-1, 0},{0,86, 6, 12,1,1, 1, 1, 2}}, {{ 0, 0},{0,86, 5, 11,1,1,1, 1, 2}}, {{ 1, 0},{0, 0, 4, 4,4,4,4, 4, 4}}, {{ 2, 0},{0,0, 4, 4,4,4,4, 4, 4}}, {{ 3, 0},{0,0, 4, 4,4,4,4, 4,0}}, {{-4,-1},{0,0,3, 37,3,3,3, 3,30}}, {{-3,-1},{0,0,37, 3,3,3,3, 3, 3}}, {{-2,-1},{0, 0, 7, 13,1,1,1, 21,32}}, {{-1,-1},{0,86, 6, 14,1,1, 1, 1, 2}}, {{ 0,-1},{0, 0, 4, 4,4,4,4, 4, 4}}, {{ 1,-1},{0, 0, 4, 4,4,4,4, 4, 4}}, {{-3,-2},{0,0,37, 3,3,3,3, 3,30}}, {{-2,-2},{0, 0,37, 3,3,3,3, 3, 3}}, {{-1,-2},{0, 0, 7, 1,1,1,21, 32, 4}}, {{ 0,-2},{0, 0, 4, 4,4,4,4, 4, 4}}, {{-2,-3},{0, 0,37, 3,3,3,3, 3, 0}}, {{-1,-3},{0, 0,37, 3,3,3, 3, 3, 0}}, {{ 0,-3},{0, 0, 4, 4,4,4,4, 4, 0}}, }, } local dig_lookup = { -- Defines dig pattern, flip, and rotation for each direction. [0] = {1, 0, 0}, [1] = {2, 0, 0}, [2] = {3, 0, 0}, [3] = {2, 1, -1}, [4] = {1, 0, -1}, [5] = {2, 0, -1}, [6] = {3, 1, 0}, [7] = {2, 1, 0}, [8] = {1, 1, 0}, [9] = {2, 2, 0}, [10] = {3, 2, 0}, [11] = {2, 1, 1}, [12] = {1, 0, 1}, [13] = {2, 0, 1}, [14] = {3, 0, 1}, [15] = {2, -1, 0}, [16] = {11, 0, 0}, [17] = {31, 0, 0}, [18] = {11, 0, -1}, [19] = {31, 1, 0}, [20] = {11, 1, 0}, [21] = {31, 2, 0}, [22] = {11, 0, 1}, [23] = {31, -1, 0}, [24] = {10, 0, 0}, [25] = {30, 0, 0}, [26] = {10, 0, -1}, [27] = {30, 1, 0}, [28] = {10, 1, 0}, [29] = {30, 2, 0}, [30] = {10, 0, 1}, [31] = {30, -1, 0} } -- Pass horizontal and vertical directions to region functions. -- Horizontal needed for angledstairs nodes, vertical helped combine functions. local dir = {} if cdir >= 24 then dir.horiz = (cdir - 24) * 2 dir.vert = -1 elseif cdir >= 16 then dir.horiz = (cdir - 16) * 2 dir.vert = 1 else dir.horiz = cdir dir.vert = 0 end local dig_list = dig_patterns[dig_lookup[cdir][1]] local flip = dig_lookup[cdir][2] local rotation = dig_lookup[cdir][3] run_list(dig_list, flip, rotation, dir, user, pointed_thing) end end local i for i,img in ipairs(images) do local inv = 1 if i == 2 then inv = 0 end minetest.register_tool("tunnelmaker:tool"..(i-1), { description = "Tunnel Maker", groups = {not_in_creative_inventory=inv}, inventory_image = img, wield_image = img, stack_max = 1, range = 7.0, -- Dig single node with left mouse click. on_use = function(itemstack, player, pointed_thing) local pname = player:get_player_name() local pos = pointed_thing.under local key_stats = player:get_player_control() -- If sneak button held down when left-clicking tunnelmaker, brings up User Config formspec. if key_stats.sneak then -- Configuration formspec local remove_refs_on = false if user_config[pname].remove_refs > 0 then remove_refs_on = true end local clear_trees_on = false if user_config[pname].clear_trees > 0 then clear_trees_on = true end local formspec = "size[5,6.5]".. "label[0.25,0.25;Tunnelmaker User Options]".. "dropdown[0.25,1.00;4;digging_mode;General purpose mode,Advanced trains mode,Bike path mode;"..tostring(user_config[pname].digging_mode).."]".. "checkbox[0.25,1.75;add_lined_tunnels;Wide paths / lined tunnels;"..tostring(user_config[pname].add_lined_tunnels).."]".. "checkbox[0.25,2.20;continuous_updown;Continuous up/down digging;"..tostring(user_config[pname].continuous_updown).."]".. "checkbox[0.25,2.75;clear_trees;Clear tree cover above*;"..tostring(clear_trees_on).."]".. "checkbox[0.25,3.20;remove_refs;Remove reference nodes*;"..tostring(remove_refs_on).."]".. "button_exit[2,5.00;1,0.4;exit;Exit]".. "label[0.25,5.75;"..minetest.colorize("#888","* Automatically disabled after 2 mins.").."]" local formspec_dm = "" local dmat = "" local use_desert_material = user_config[pname].use_desert_material if add_desert_material and minetest.get_biome_data then if not user_config[pname].lock_desert_mode then use_desert_material = string.match(minetest.get_biome_name(minetest.get_biome_data(player:get_pos()).biome), "desert") user_config[pname].use_desert_material = use_desert_material end if use_desert_material then dmat = "Desert" else dmat = "Non-desert" end formspec_dm = "checkbox[0.25,3.75;lock_desert_mode;Lock desert mode to: "..dmat..";"..tostring(user_config[pname].lock_desert_mode).."]" end minetest.show_formspec(pname, "tunnelmaker:form", formspec..formspec_dm) else -- Dig single node, if pointing to one if pos ~= nil then minetest.node_dig(pos, minetest.get_node(pos), player) minetest.sound_play("default_dig_dig_immediate", {pos=pos, max_hear_distance = 8, gain = 0.5}) end end end, -- Dig tunnel with right mouse click (double tap on android) on_place = function(itemstack, placer, pointed_thing) local pname = placer and placer:get_player_name() or "" -- If sneak button held down when right-clicking tunnelmaker, toggle updown dig direction: up, down, horizontal, ... -- Rotating or moving will reset to horizontal. if placer:get_player_control().sneak then tunnelmaker[pname].updown = (tunnelmaker[pname].updown + 1) % 3 tunnelmaker[pname].lastpos = { x = placer:getpos().x, y = placer:getpos().y, z = placer:getpos().z } -- Otherwise dig tunnel based on direction pointed and current updown direction elseif pointed_thing.type=="node" then -- if advtrains_track, I lower positions of pointed_thing to right below track, but keep name the same. Same with snow cover. local name = minetest.get_node(pointed_thing.under).name -- if minetest.registered_nodes[name].groups.advtrains_track == 1 then if string.match(name, "dtrack") or name == "default:snow" or name == angled_slab_not_desert or name == angled_slab_desert then pointed_thing.under = vector.add(pointed_thing.under, {x=0, y=-1, z=0}) --pointed_thing.above = vector.add(pointed_thing.above, {x=0, y=-1, z=0}) -- don't currently use this end minetest.sound_play("default_dig_dig_immediate", {pos=pointed_thing.under, max_hear_distance = 8, gain = 1.0}) dig_tunnel(i-1, placer, pointed_thing) if not user_config[pname].continuous_updown then tunnelmaker[pname].updown = 0 -- reset to horizontal after one use end end end, } ) end -- Register configuration callback minetest.register_on_player_receive_fields(function(player, formname, fields) if formname ~= "tunnelmaker:form" then return false end local pname = player:get_player_name() if fields.continuous_updown == "true" then user_config[pname].continuous_updown = true elseif fields.continuous_updown == "false" then user_config[pname].continuous_updown = false elseif fields.add_lined_tunnels == "true" then user_config[pname].add_lined_tunnels = true user_config[pname].add_floors = true user_config[pname].add_wide_floors = true elseif fields.add_lined_tunnels == "false" then user_config[pname].add_lined_tunnels = false user_config[pname].add_floors = user_config[pname].digging_mode ~= 1 -- gp no floors when not lined user_config[pname].add_wide_floors = false elseif fields.clear_trees == "true" then user_config[pname].clear_trees = remove_refs_enable_time elseif fields.clear_trees == "false" then user_config[pname].clear_trees = 0 elseif fields.remove_refs == "true" then user_config[pname].remove_refs = remove_refs_enable_time elseif fields.remove_refs == "false" then user_config[pname].remove_refs = 0 elseif fields.lock_desert_mode == "false" then user_config[pname].lock_desert_mode = false elseif fields.lock_desert_mode == "true" then user_config[pname].lock_desert_mode = true elseif fields.digging_mode == "General purpose mode" then user_config[pname].digging_mode = 1 user_config[pname].height = tunnel_height_general user_config[pname].add_arches = false user_config[pname].add_embankment = false user_config[pname].add_refs = false user_config[pname].add_floors = user_config[pname].add_lined_tunnels user_config[pname].add_wide_floors = user_config[pname].add_lined_tunnels user_config[pname].add_bike_ramps = false user_config[pname].coating_not_desert = tunnel_material user_config[pname].coating_desert = tunnel_material_desert elseif fields.digging_mode == "Advanced trains mode" then user_config[pname].digging_mode = 2 user_config[pname].height = tunnel_height_train user_config[pname].add_arches = add_arches_config user_config[pname].add_embankment = true user_config[pname].add_refs = true user_config[pname].add_floors = true user_config[pname].add_wide_floors = user_config[pname].add_lined_tunnels user_config[pname].add_bike_ramps = false user_config[pname].coating_not_desert = tunnel_material user_config[pname].coating_desert = tunnel_material_desert elseif fields.digging_mode == "Bike path mode" then user_config[pname].digging_mode = 3 user_config[pname].height = tunnel_height_bike user_config[pname].add_arches = false user_config[pname].add_embankment = false user_config[pname].add_refs = true user_config[pname].add_floors = true user_config[pname].add_wide_floors = user_config[pname].add_lined_tunnels user_config[pname].add_bike_ramps = true user_config[pname].coating_not_desert = bike_path_material user_config[pname].coating_desert = bike_path_material_desert end return true end) -- Decrement remove_refs countdown timers. minetest.register_globalstep(function(dtime) local players = minetest.get_connected_players() for _,player in ipairs(players) do local pname = player:get_player_name() local rr = user_config[pname].remove_refs if rr > 0 then rr = rr - dtime if rr <= 0 then user_config[pname].remove_refs = 0 else user_config[pname].remove_refs = rr end end local ct = user_config[pname].clear_trees if ct > 0 then ct = ct - dtime if ct <= 0 then user_config[pname].clear_trees = 0 else user_config[pname].clear_trees = ct end end end end) -- Remove reference marks local remove_refs = function(player) local ppos = player:get_pos() local refpos = minetest.find_node_near(ppos, 1, reference_marks) if refpos then if not minetest.is_protected(refpos, player) then local meta = minetest.get_meta(refpos) local rep_mat = meta:get("replace_with") if rep_mat and string.len(rep_mat) > 0 then minetest.set_node(refpos, {name = rep_mat, param2 = 42}) end end end end -- Replaces reference marks with appropriate material. minetest.register_globalstep(function(dtime) for _, player in ipairs(minetest.get_connected_players()) do if user_config[player:get_player_name()].remove_refs > 0 then remove_refs(player) end end end)