mirror of
				https://github.com/Uberi/Minetest-WorldEdit.git
				synced 2025-10-25 19:35:28 +02:00 
			
		
		
		
	Compare commits
	
		
			7 Commits
		
	
	
		
			1.3
			...
			c223ca4cec
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | c223ca4cec | ||
|  | c8afa95542 | ||
|  | 670e421f57 | ||
|  | 770601dd5d | ||
|  | 2f2f5a7def | ||
|  | 7f87f1658e | ||
|  | 4378750498 | 
							
								
								
									
										11
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | name: test | ||||||
|  |  | ||||||
|  | on: [push, pull_request] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   test: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v2 | ||||||
|  |     - name: Run tests | ||||||
|  |       run: MINETEST_VER=latest ./.util/run_tests.sh | ||||||
							
								
								
									
										30
									
								
								.util/run_tests.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										30
									
								
								.util/run_tests.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | #!/bin/bash | ||||||
|  | tempdir=/tmp/mt | ||||||
|  | confpath=$tempdir/minetest.conf | ||||||
|  | worldpath=$tempdir/world | ||||||
|  |  | ||||||
|  | use_docker=y | ||||||
|  | [ -x ../../bin/minetestserver ] && use_docker= | ||||||
|  |  | ||||||
|  | rm -rf $tempdir | ||||||
|  | mkdir -p $worldpath | ||||||
|  | # the docker image doesn't have devtest | ||||||
|  | [ -n "$use_docker" ] || printf '%s\n' gameid=devtest >$worldpath/world.mt | ||||||
|  | printf '%s\n' mg_name=singlenode '[end_of_params]' >$worldpath/map_meta.txt | ||||||
|  | printf '%s\n' worldedit_run_tests=true max_forceloaded_blocks=9999 >$confpath | ||||||
|  |  | ||||||
|  | if [ -n "$use_docker" ]; then | ||||||
|  | 	chmod -R 777 $tempdir | ||||||
|  | 	docker run --rm -i \ | ||||||
|  | 		-v $confpath:/etc/minetest/minetest.conf \ | ||||||
|  | 		-v $tempdir:/var/lib/minetest/.minetest \ | ||||||
|  | 		-v "$PWD/worldedit":/var/lib/minetest/.minetest/world/worldmods/worldedit \ | ||||||
|  | 		registry.gitlab.com/minetest/minetest/server:${MINETEST_VER} | ||||||
|  | else | ||||||
|  | 	mkdir $worldpath/worldmods | ||||||
|  | 	ln -s "$PWD/worldedit" $worldpath/worldmods/worldedit | ||||||
|  | 	../../bin/minetestserver --config $confpath --world $worldpath --logfile /dev/null | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | test -f $worldpath/tests_ok || exit 1 | ||||||
|  | exit 0 | ||||||
| @@ -23,7 +23,7 @@ There is a nice installation guide over at the [Minetest Wiki](http://wiki.minet | |||||||
| 8. You should have a mod selection screen. Select the one named something like `Minetest-WorldEdit` by left clicking once and press the **Enable Modpack** button. | 8. You should have a mod selection screen. Select the one named something like `Minetest-WorldEdit` by left clicking once and press the **Enable Modpack** button. | ||||||
| 9. Press the **Save** button. You can now use WorldEdit in that world. Repeat steps 7 to 9 to enable WorldEdit for other worlds too. | 9. Press the **Save** button. You can now use WorldEdit in that world. Repeat steps 7 to 9 to enable WorldEdit for other worlds too. | ||||||
|  |  | ||||||
| If you are having trouble, try asking for help in the [IRC channel](https://webchat.freenode.net/?channels=#minetest) (faster but may not always have helpers online) | If you are having trouble, try asking for help in the [IRC channel](https://web.libera.chat/#minetest) (faster but may not always have helpers online) | ||||||
| or ask on the [forum topic](https://forum.minetest.net/viewtopic.php?id=572) (slower but more likely to get help). | or ask on the [forum topic](https://forum.minetest.net/viewtopic.php?id=572) (slower but more likely to get help). | ||||||
|  |  | ||||||
| Usage | Usage | ||||||
|   | |||||||
| @@ -38,3 +38,7 @@ if minetest.settings:get_bool("log_mods") then | |||||||
| 	print("[WorldEdit] Loaded!") | 	print("[WorldEdit] Loaded!") | ||||||
| end | end | ||||||
|  |  | ||||||
|  | if minetest.settings:get_bool("worldedit_run_tests") then | ||||||
|  | 	dofile(path .. "/test.lua") | ||||||
|  | 	minetest.after(0, worldedit.run_tests) | ||||||
|  | end | ||||||
|   | |||||||
| @@ -640,10 +640,34 @@ function worldedit.clear_objects(pos1, pos2) | |||||||
|  |  | ||||||
| 	worldedit.keep_loaded(pos1, pos2) | 	worldedit.keep_loaded(pos1, pos2) | ||||||
|  |  | ||||||
|  | 	local function should_delete(obj) | ||||||
|  | 		-- Avoid players and WorldEdit entities | ||||||
|  | 		if obj:is_player() then | ||||||
|  | 			return false | ||||||
|  | 		end | ||||||
|  | 		local entity = obj:get_luaentity() | ||||||
|  | 		return not entity or not entity.name:find("^worldedit:") | ||||||
|  | 	end | ||||||
|  |  | ||||||
| 	-- Offset positions to include full nodes (positions are in the center of nodes) | 	-- Offset positions to include full nodes (positions are in the center of nodes) | ||||||
| 	local pos1x, pos1y, pos1z = pos1.x - 0.5, pos1.y - 0.5, pos1.z - 0.5 | 	local pos1x, pos1y, pos1z = pos1.x - 0.5, pos1.y - 0.5, pos1.z - 0.5 | ||||||
| 	local pos2x, pos2y, pos2z = pos2.x + 0.5, pos2.y + 0.5, pos2.z + 0.5 | 	local pos2x, pos2y, pos2z = pos2.x + 0.5, pos2.y + 0.5, pos2.z + 0.5 | ||||||
|  |  | ||||||
|  | 	local count = 0 | ||||||
|  | 	if minetest.get_objects_in_area then | ||||||
|  | 		local objects = minetest.get_objects_in_area({x=pos1x, y=pos1y, z=pos1z}, | ||||||
|  | 			{x=pos2x, y=pos2y, z=pos2z}) | ||||||
|  |  | ||||||
|  | 		for _, obj in pairs(objects) do | ||||||
|  | 			if should_delete(obj) then | ||||||
|  | 				obj:remove() | ||||||
|  | 				count = count + 1 | ||||||
|  | 			end | ||||||
|  | 		end | ||||||
|  | 		return count | ||||||
|  | 	end | ||||||
|  |  | ||||||
|  | 	-- Fallback implementation via get_objects_inside_radius | ||||||
| 	-- Center of region | 	-- Center of region | ||||||
| 	local center = { | 	local center = { | ||||||
| 		x = pos1x + ((pos2x - pos1x) / 2), | 		x = pos1x + ((pos2x - pos1x) / 2), | ||||||
| @@ -655,12 +679,8 @@ function worldedit.clear_objects(pos1, pos2) | |||||||
| 			(center.x - pos1x) ^ 2 + | 			(center.x - pos1x) ^ 2 + | ||||||
| 			(center.y - pos1y) ^ 2 + | 			(center.y - pos1y) ^ 2 + | ||||||
| 			(center.z - pos1z) ^ 2) | 			(center.z - pos1z) ^ 2) | ||||||
| 	local count = 0 |  | ||||||
| 	for _, obj in pairs(minetest.get_objects_inside_radius(center, radius)) do | 	for _, obj in pairs(minetest.get_objects_inside_radius(center, radius)) do | ||||||
| 		local entity = obj:get_luaentity() | 		if should_delete(obj) then | ||||||
| 		-- Avoid players and WorldEdit entities |  | ||||||
| 		if not obj:is_player() and (not entity or |  | ||||||
| 				not entity.name:find("^worldedit:")) then |  | ||||||
| 			local pos = obj:get_pos() | 			local pos = obj:get_pos() | ||||||
| 			if pos.x >= pos1x and pos.x <= pos2x and | 			if pos.x >= pos1x and pos.x <= pos2x and | ||||||
| 					pos.y >= pos1y and pos.y <= pos2y and | 					pos.y >= pos1y and pos.y <= pos2y and | ||||||
|   | |||||||
| @@ -118,7 +118,7 @@ end | |||||||
| -- by ChillCode, available under the MIT license. | -- by ChillCode, available under the MIT license. | ||||||
| local function deserialize_workaround(content) | local function deserialize_workaround(content) | ||||||
| 	local nodes | 	local nodes | ||||||
| 	if not jit then | 	if not minetest.global_exists("jit") then | ||||||
| 		nodes = minetest.deserialize(content, true) | 		nodes = minetest.deserialize(content, true) | ||||||
| 	else | 	else | ||||||
| 		-- XXX: This is a filthy hack that works surprisingly well | 		-- XXX: This is a filthy hack that works surprisingly well | ||||||
|   | |||||||
							
								
								
									
										448
									
								
								worldedit/test.lua
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										448
									
								
								worldedit/test.lua
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,448 @@ | |||||||
|  | --------------------- | ||||||
|  | -- Helpers | ||||||
|  | --------------------- | ||||||
|  |  | ||||||
|  | local vec = vector.new | ||||||
|  | local vecw = function(axis, n, base) | ||||||
|  | 	local ret = vec(base) | ||||||
|  | 	ret[axis] = n | ||||||
|  | 	return ret | ||||||
|  | end | ||||||
|  | local pos2str = minetest.pos_to_string | ||||||
|  | local get_node = minetest.get_node | ||||||
|  | local set_node = minetest.set_node | ||||||
|  |  | ||||||
|  | --------------------- | ||||||
|  | -- Nodes | ||||||
|  | --------------------- | ||||||
|  | local air = "air" | ||||||
|  | local testnode1 | ||||||
|  | local testnode2 | ||||||
|  | local testnode3 | ||||||
|  | -- Loads nodenames to use for tests | ||||||
|  | local function init_nodes() | ||||||
|  | 	testnode1 = minetest.registered_aliases["mapgen_stone"] | ||||||
|  | 	testnode2 = minetest.registered_aliases["mapgen_dirt"] | ||||||
|  | 	testnode3 = minetest.registered_aliases["mapgen_cobble"] or minetest.registered_aliases["mapgen_dirt_with_grass"] | ||||||
|  | 	assert(testnode1 and testnode2 and testnode3) | ||||||
|  | end | ||||||
|  | -- Writes repeating pattern into given area | ||||||
|  | local function place_pattern(pos1, pos2, pattern) | ||||||
|  | 	local pos = vec() | ||||||
|  | 	local node = {name=""} | ||||||
|  | 	local i = 1 | ||||||
|  | 	for z = pos1.z, pos2.z do | ||||||
|  | 		pos.z = z | ||||||
|  | 	for y = pos1.y, pos2.y do | ||||||
|  | 		pos.y = y | ||||||
|  | 	for x = pos1.x, pos2.x do | ||||||
|  | 		pos.x = x | ||||||
|  | 		node.name = pattern[i] | ||||||
|  | 		set_node(pos, node) | ||||||
|  | 		i = i % #pattern + 1 | ||||||
|  | 	end | ||||||
|  | 	end | ||||||
|  | 	end | ||||||
|  | end | ||||||
|  |  | ||||||
|  |  | ||||||
|  | --------------------- | ||||||
|  | -- Area management | ||||||
|  | --------------------- | ||||||
|  | assert(minetest.get_mapgen_setting("mg_name") == "singlenode") | ||||||
|  | local area = {} | ||||||
|  | do | ||||||
|  | 	local areamin, areamax | ||||||
|  | 	local off | ||||||
|  | 	local c_air = minetest.get_content_id(air) | ||||||
|  | 	local vbuffer = {} | ||||||
|  | 	-- Assign a new area for use, will emerge and then call ready() | ||||||
|  | 	area.assign = function(min, max, ready) | ||||||
|  | 		areamin = min | ||||||
|  | 		areamax = max | ||||||
|  | 		minetest.emerge_area(min, max, function(bpos, action, remaining) | ||||||
|  | 			assert(action ~= minetest.EMERGE_ERRORED) | ||||||
|  | 			if remaining > 0 then return end | ||||||
|  | 			minetest.after(0, function() | ||||||
|  | 				area.clear() | ||||||
|  | 				ready() | ||||||
|  | 			end) | ||||||
|  | 		end) | ||||||
|  | 	end | ||||||
|  | 	-- Reset area contents and state | ||||||
|  | 	area.clear = function() | ||||||
|  | 		local vmanip = minetest.get_voxel_manip(areamin, areamax) | ||||||
|  | 		local vpos1, vpos2 = vmanip:get_emerged_area() | ||||||
|  | 		local vcount = (vpos2.x - vpos1.x + 1) * (vpos2.y - vpos1.y + 1) * (vpos2.z - vpos1.z + 1) | ||||||
|  | 		if #vbuffer ~= vcount then | ||||||
|  | 			vbuffer = {} | ||||||
|  | 			for i = 1, vcount do | ||||||
|  | 				vbuffer[i] = c_air | ||||||
|  | 			end | ||||||
|  | 		end | ||||||
|  | 		vmanip:set_data(vbuffer) | ||||||
|  | 		vmanip:write_to_map() | ||||||
|  | 		off = vec(0, 0, 0) | ||||||
|  | 	end | ||||||
|  | 	-- Returns an usable area [pos1, pos2] that does not overlap previous ones | ||||||
|  | 	area.get = function(sizex, sizey, sizez) | ||||||
|  | 		local size | ||||||
|  | 		if sizey == nil or sizez == nil then | ||||||
|  | 			size = {x=sizex, y=sizex, z=sizex} | ||||||
|  | 		else | ||||||
|  | 			size = {x=sizex, y=sizey, z=sizez} | ||||||
|  | 		end | ||||||
|  | 		local pos1 = vector.add(areamin, off) | ||||||
|  | 		local pos2 = vector.subtract(vector.add(pos1, size), 1) | ||||||
|  | 		if pos2.x > areamax.x or pos2.y > areamax.y or pos2.z > areamax.z then | ||||||
|  | 			error("Internal failure: out of space") | ||||||
|  | 		end | ||||||
|  | 		off = vector.add(off, size) | ||||||
|  | 		return pos1, pos2 | ||||||
|  | 	end | ||||||
|  | 	-- Returns an axis and count (= n) relative to the last-requested area that is unoccupied | ||||||
|  | 	area.dir = function(n) | ||||||
|  | 		local pos1 = vector.add(areamin, off) | ||||||
|  | 		if pos1.x + n <= areamax.x then | ||||||
|  | 			off.x = off.x + n | ||||||
|  | 			return "x", n | ||||||
|  | 		elseif pos1.x + n <= areamax.y then | ||||||
|  | 			off.y = off.y + n | ||||||
|  | 			return "y", n | ||||||
|  | 		elseif pos1.z + n <= areamax.z then | ||||||
|  | 			off.z = off.z + n | ||||||
|  | 			return "z", n | ||||||
|  | 		end | ||||||
|  | 		error("Internal failure: out of space") | ||||||
|  | 	end | ||||||
|  | 	-- Returns [XYZ] margin (list of pos pairs) of n around last-requested area | ||||||
|  | 	-- (may actually be larger but doesn't matter) | ||||||
|  | 	area.margin = function(n) | ||||||
|  | 		local pos1, pos2 = area.get(n) | ||||||
|  | 		return { | ||||||
|  | 			{ vec(areamin.x, areamin.y, pos1.z), pos2 }, -- X/Y | ||||||
|  | 			{ vec(areamin.x, pos1.y, areamin.z), pos2 }, -- X/Z | ||||||
|  | 			{ vec(pos1.x, areamin.y, areamin.z), pos2 }, -- Y/Z | ||||||
|  | 		} | ||||||
|  | 	end | ||||||
|  | end | ||||||
|  | -- Split an existing area into two non-overlapping [pos1, half1], [half2, pos2] parts; returns half1, half2 | ||||||
|  | area.split = function(pos1, pos2) | ||||||
|  | 	local axis | ||||||
|  | 	if pos2.x - pos1.x >= 1 then | ||||||
|  | 		axis = "x" | ||||||
|  | 	elseif pos2.y - pos1.y >= 1 then | ||||||
|  | 		axis = "y" | ||||||
|  | 	elseif pos2.z - pos1.z >= 1 then | ||||||
|  | 		axis = "z" | ||||||
|  | 	else | ||||||
|  | 		error("Internal failure: area too small to split") | ||||||
|  | 	end | ||||||
|  | 	local hspan = math.floor((pos2[axis] - pos1[axis] + 1) / 2) | ||||||
|  | 	local half1 = vecw(axis, pos1[axis] + hspan - 1, pos2) | ||||||
|  | 	local half2 = vecw(axis, pos1[axis] + hspan, pos2) | ||||||
|  | 	return half1, half2 | ||||||
|  | end | ||||||
|  |  | ||||||
|  |  | ||||||
|  | --------------------- | ||||||
|  | -- Checks | ||||||
|  | --------------------- | ||||||
|  | local check = {} | ||||||
|  | -- Check that all nodes in [pos1, pos2] are the node(s) specified | ||||||
|  | check.filled = function(pos1, pos2, nodes) | ||||||
|  | 	if type(nodes) == "string" then | ||||||
|  | 		nodes = { nodes } | ||||||
|  | 	end | ||||||
|  | 	local _, counts = minetest.find_nodes_in_area(pos1, pos2, nodes) | ||||||
|  | 	local total = worldedit.volume(pos1, pos2) | ||||||
|  | 	local sum = 0 | ||||||
|  | 	for _, n in pairs(counts) do | ||||||
|  | 		sum = sum + n | ||||||
|  | 	end | ||||||
|  | 	if sum ~= total then | ||||||
|  | 		error((total - sum) .. " " .. table.concat(nodes, ",") .. " nodes missing in " .. | ||||||
|  | 			pos2str(pos1) .. " -> " .. pos2str(pos2)) | ||||||
|  | 	end | ||||||
|  | end | ||||||
|  | -- Check that none of the nodes in [pos1, pos2] are the node(s) specified | ||||||
|  | check.not_filled = function(pos1, pos2, nodes) | ||||||
|  | 	if type(nodes) == "string" then | ||||||
|  | 		nodes = { nodes } | ||||||
|  | 	end | ||||||
|  | 	local _, counts = minetest.find_nodes_in_area(pos1, pos2, nodes) | ||||||
|  | 	for nodename, n in pairs(counts) do | ||||||
|  | 		if n ~= 0 then | ||||||
|  | 			error(counts[nodename] .. " " .. nodename .. " nodes found in " .. | ||||||
|  | 				pos2str(pos1) .. " -> " .. pos2str(pos2)) | ||||||
|  | 		end | ||||||
|  | 	end | ||||||
|  | end | ||||||
|  | -- Check that all of the areas are only made of node(s) specified | ||||||
|  | check.filled2 = function(list, nodes) | ||||||
|  | 	for _, pos in ipairs(list) do | ||||||
|  | 		check.filled(pos[1], pos[2], nodes) | ||||||
|  | 	end | ||||||
|  | end | ||||||
|  | -- Check that none of the areas contain the node(s) specified | ||||||
|  | check.not_filled2 = function(list, nodes) | ||||||
|  | 	for _, pos in ipairs(list) do | ||||||
|  | 		check.not_filled(pos[1], pos[2], nodes) | ||||||
|  | 	end | ||||||
|  | end | ||||||
|  | -- Checks presence of a repeating pattern in [pos1, po2] (cf. place_pattern) | ||||||
|  | check.pattern = function(pos1, pos2, pattern) | ||||||
|  | 	local pos = vec() | ||||||
|  | 	local i = 1 | ||||||
|  | 	for z = pos1.z, pos2.z do | ||||||
|  | 		pos.z = z | ||||||
|  | 	for y = pos1.y, pos2.y do | ||||||
|  | 		pos.y = y | ||||||
|  | 	for x = pos1.x, pos2.x do | ||||||
|  | 		pos.x = x | ||||||
|  | 		local node = get_node(pos) | ||||||
|  | 		if node.name ~= pattern[i] then | ||||||
|  | 			error(pattern[i] .. " not found at " .. pos2str(pos) .. " (i=" .. i .. ")") | ||||||
|  | 		end | ||||||
|  | 		i = i % #pattern + 1 | ||||||
|  | 	end | ||||||
|  | 	end | ||||||
|  | 	end | ||||||
|  | end | ||||||
|  |  | ||||||
|  |  | ||||||
|  | --------------------- | ||||||
|  | -- The actual tests | ||||||
|  | --------------------- | ||||||
|  | local tests = {} | ||||||
|  | local function register_test(name, func, opts) | ||||||
|  | 	assert(type(name) == "string") | ||||||
|  | 	assert(func == nil or type(func) == "function") | ||||||
|  | 	if not opts then | ||||||
|  | 		opts = {} | ||||||
|  | 	else | ||||||
|  | 		opts = table.copy(opts) | ||||||
|  | 	end | ||||||
|  | 	opts.name = name | ||||||
|  | 	opts.func = func | ||||||
|  | 	table.insert(tests, opts) | ||||||
|  | end | ||||||
|  | -- How this works: | ||||||
|  | --   register_test registers a test with a name and function | ||||||
|  | --   The function should return if the test passes or otherwise cause a Lua error | ||||||
|  | --   The basic structure is: get areas + do operations + check results | ||||||
|  | -- Helpers: | ||||||
|  | --   area.get must be used to retrieve areas that can be operated on (these will be cleared before each test) | ||||||
|  | --   check.filled / check.not_filled can be used to check the result | ||||||
|  | --   area.margin + check.filled2 is useful to make sure nodes weren't placed too far | ||||||
|  | --   place_pattern + check.pattern is useful to test ops that operate on existing data | ||||||
|  |  | ||||||
|  |  | ||||||
|  | register_test("Internal self-test") | ||||||
|  | register_test("is area loaded?", function() | ||||||
|  | 	local pos1, _ = area.get(1) | ||||||
|  | 	assert(get_node(pos1).name == "air") | ||||||
|  | end, {dry=true}) | ||||||
|  |  | ||||||
|  | register_test("area.split", function() | ||||||
|  | 	for i = 2, 6 do | ||||||
|  | 		local pos1, pos2 = area.get(1, 1, i) | ||||||
|  | 		local half1, half2 = area.split(pos1, pos2) | ||||||
|  | 		assert(pos1.x == half1.x and pos1.y == half1.y) | ||||||
|  | 		assert(half1.x == half2.x and half1.y == half2.y) | ||||||
|  | 		assert(half1.z + 1 == half2.z) | ||||||
|  | 		if i % 2 == 0 then | ||||||
|  | 			assert((half1.z - pos1.z) == (pos2.z - half2.z)) -- divided equally | ||||||
|  | 		end | ||||||
|  | 	end | ||||||
|  | end, {dry=true}) | ||||||
|  |  | ||||||
|  | register_test("check.filled", function() | ||||||
|  | 	local pos1, pos2 = area.get(1, 2, 1) | ||||||
|  | 	set_node(pos1, {name=testnode1}) | ||||||
|  | 	set_node(pos2, {name=testnode2}) | ||||||
|  | 	check.filled(pos1, pos1, testnode1) | ||||||
|  | 	check.filled(pos1, pos2, {testnode1, testnode2}) | ||||||
|  | 	check.not_filled(pos1, pos1, air) | ||||||
|  | 	check.not_filled(pos1, pos2, {air, testnode3}) | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | register_test("pattern", function() | ||||||
|  | 	local pos1, pos2 = area.get(3, 2, 1) | ||||||
|  | 	local pattern = {testnode1, testnode3} | ||||||
|  | 	place_pattern(pos1, pos2, pattern) | ||||||
|  | 	assert(get_node(pos1).name == testnode1) | ||||||
|  | 	check.pattern(pos1, pos2, pattern) | ||||||
|  | end) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | register_test("Generic node manipulations") | ||||||
|  | register_test("worldedit.set", function() | ||||||
|  | 	local pos1, pos2 = area.get(10) | ||||||
|  | 	local m = area.margin(1) | ||||||
|  |  | ||||||
|  | 	worldedit.set(pos1, pos2, testnode1) | ||||||
|  |  | ||||||
|  | 	check.filled(pos1, pos2, testnode1) | ||||||
|  | 	check.filled2(m, air) | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | register_test("worldedit.set mix", function() | ||||||
|  | 	local pos1, pos2 = area.get(10) | ||||||
|  | 	local m = area.margin(1) | ||||||
|  |  | ||||||
|  | 	worldedit.set(pos1, pos2, {testnode1, testnode2}) | ||||||
|  |  | ||||||
|  | 	check.filled(pos1, pos2, {testnode1, testnode2}) | ||||||
|  | 	check.filled2(m, air) | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | register_test("worldedit.replace", function() | ||||||
|  | 	local pos1, pos2 = area.get(10) | ||||||
|  | 	local half1, half2 = area.split(pos1, pos2) | ||||||
|  |  | ||||||
|  | 	worldedit.set(pos1, half1, testnode1) | ||||||
|  | 	worldedit.set(half2, pos2, testnode2) | ||||||
|  | 	worldedit.replace(pos1, pos2, testnode1, testnode3) | ||||||
|  |  | ||||||
|  | 	check.not_filled(pos1, pos2, testnode1) | ||||||
|  | 	check.filled(pos1, half1, testnode3) | ||||||
|  | 	check.filled(half2, pos2, testnode2) | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | register_test("worldedit.replace inverse", function() | ||||||
|  | 	local pos1, pos2 = area.get(10) | ||||||
|  | 	local half1, half2 = area.split(pos1, pos2) | ||||||
|  |  | ||||||
|  | 	worldedit.set(pos1, half1, testnode1) | ||||||
|  | 	worldedit.set(half2, pos2, testnode2) | ||||||
|  | 	worldedit.replace(pos1, pos2, testnode1, testnode3, true) | ||||||
|  |  | ||||||
|  | 	check.filled(pos1, half1, testnode1) | ||||||
|  | 	check.filled(half2, pos2, testnode3) | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | -- FIXME?: this one looks overcomplicated | ||||||
|  | register_test("worldedit.copy", function() | ||||||
|  | 	local pos1, pos2 = area.get(4) | ||||||
|  | 	local axis, n = area.dir(2) | ||||||
|  | 	local m = area.margin(1) | ||||||
|  | 	local b = pos1[axis] | ||||||
|  |  | ||||||
|  | 	-- create one slice with testnode1, one with testnode2 | ||||||
|  | 	worldedit.set(pos1, vecw(axis, b + 1, pos2), testnode1) | ||||||
|  | 	worldedit.set(vecw(axis, b + 2, pos1), pos2, testnode2) | ||||||
|  | 	worldedit.copy(pos1, pos2, axis, n) | ||||||
|  |  | ||||||
|  | 	-- should have three slices now | ||||||
|  | 	check.filled(pos1, vecw(axis, b + 1, pos2), testnode1) | ||||||
|  | 	check.filled(vecw(axis, b + 2, pos1), pos2, testnode1) | ||||||
|  | 	check.filled(vecw(axis, b + 4, pos1), vector.add(pos2, vecw(axis, n)), testnode2) | ||||||
|  | 	check.filled2(m, "air") | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | register_test("worldedit.copy2", function() | ||||||
|  | 	local pos1, pos2 = area.get(6) | ||||||
|  | 	local m1 = area.margin(1) | ||||||
|  | 	local pos1_, pos2_ = area.get(6) | ||||||
|  | 	local m2 = area.margin(1) | ||||||
|  |  | ||||||
|  | 	local pattern = {testnode1, testnode2, testnode3, testnode1, testnode2} | ||||||
|  | 	place_pattern(pos1, pos2, pattern) | ||||||
|  | 	worldedit.copy2(pos1, pos2, vector.subtract(pos1_, pos1)) | ||||||
|  |  | ||||||
|  | 	check.pattern(pos1, pos2, pattern) | ||||||
|  | 	check.pattern(pos1_, pos2_, pattern) | ||||||
|  | 	check.filled2(m1, "air") | ||||||
|  | 	check.filled2(m2, "air") | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | register_test("worldedit.move (overlap)", function() | ||||||
|  | 	local pos1, pos2 = area.get(7) | ||||||
|  | 	local axis, n = area.dir(2) | ||||||
|  | 	local m = area.margin(1) | ||||||
|  |  | ||||||
|  | 	local pattern = {testnode2, testnode1, testnode2, testnode3, testnode3} | ||||||
|  | 	place_pattern(pos1, pos2, pattern) | ||||||
|  | 	worldedit.move(pos1, pos2, axis, n) | ||||||
|  |  | ||||||
|  | 	check.filled(pos1, vecw(axis, pos1[axis] + n - 1, pos2), "air") | ||||||
|  | 	check.pattern(vecw(axis, pos1[axis] + n, pos1), vecw(axis, pos2[axis] + n, pos2), pattern) | ||||||
|  | 	check.filled2(m, "air") | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | register_test("worldedit.move", function() | ||||||
|  | 	local pos1, pos2 = area.get(10) | ||||||
|  | 	local axis, n = area.dir(10) | ||||||
|  | 	local m = area.margin(1) | ||||||
|  |  | ||||||
|  | 	local pattern = {testnode1, testnode3, testnode3, testnode2} | ||||||
|  | 	place_pattern(pos1, pos2, pattern) | ||||||
|  | 	worldedit.move(pos1, pos2, axis, n) | ||||||
|  |  | ||||||
|  | 	check.filled(pos1, pos2, "air") | ||||||
|  | 	check.pattern(vecw(axis, pos1[axis] + n, pos1), vecw(axis, pos2[axis] + n, pos2), pattern) | ||||||
|  | 	check.filled2(m, "air") | ||||||
|  | end) | ||||||
|  |  | ||||||
|  | -- TODO: the rest (also testing param2 + metadata) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | --------------------- | ||||||
|  | -- Main function | ||||||
|  | --------------------- | ||||||
|  | worldedit.run_tests = function() | ||||||
|  | 	do | ||||||
|  | 		local v = minetest.get_version() | ||||||
|  | 		print("Running " .. #tests .. " tests for WorldEdit " .. | ||||||
|  | 			worldedit.version_string .. " on " .. v.project .. " " .. (v.hash or v.string)) | ||||||
|  | 	end | ||||||
|  |  | ||||||
|  | 	init_nodes() | ||||||
|  |  | ||||||
|  | 	-- emerge area from (0,0,0) ~ (56,56,56) and keep it loaded | ||||||
|  | 	-- Note: making this area smaller speeds up tests | ||||||
|  | 	local wanted = vec(56, 56, 56) | ||||||
|  | 	for x = 0, math.floor(wanted.x/16) do | ||||||
|  | 	for y = 0, math.floor(wanted.y/16) do | ||||||
|  | 	for z = 0, math.floor(wanted.z/16) do | ||||||
|  | 		assert(minetest.forceload_block({x=x*16, y=y*16, z=z*16}, true)) | ||||||
|  | 	end | ||||||
|  | 	end | ||||||
|  | 	end | ||||||
|  | 	area.assign(vec(0, 0, 0), wanted, function() | ||||||
|  |  | ||||||
|  | 		local failed = 0 | ||||||
|  | 		for _, test in ipairs(tests) do | ||||||
|  | 			if not test.func then | ||||||
|  | 				local s = "---- " .. test.name .. " " | ||||||
|  | 				print(s .. string.rep("-", 60 - #s)) | ||||||
|  | 			else | ||||||
|  | 				if not test.dry then | ||||||
|  | 					area.clear() | ||||||
|  | 				end | ||||||
|  | 				local ok, err = pcall(test.func) | ||||||
|  | 				print(string.format("%-60s %s", test.name, ok and "pass" or "FAIL")) | ||||||
|  | 				if not ok then | ||||||
|  | 					print("   " .. err) | ||||||
|  | 					failed = failed + 1 | ||||||
|  | 				end | ||||||
|  | 			end | ||||||
|  | 		end | ||||||
|  |  | ||||||
|  | 		print("Done, " .. failed .. " tests failed.") | ||||||
|  | 		if failed == 0 then | ||||||
|  | 			io.close(io.open(minetest.get_worldpath() .. "/tests_ok", "w")) | ||||||
|  | 		end | ||||||
|  | 		minetest.request_shutdown() | ||||||
|  | 	end) | ||||||
|  | end | ||||||
|  |  | ||||||
|  | -- for debug purposes | ||||||
|  | minetest.register_on_joinplayer(function(player) | ||||||
|  | 	minetest.set_player_privs(player:get_player_name(), | ||||||
|  | 		minetest.string_to_privs("fly,fast,noclip,basic_debug,debug,interact")) | ||||||
|  | end) | ||||||
|  | minetest.register_on_punchnode(function(pos, node, puncher) | ||||||
|  | 	minetest.chat_send_player(puncher:get_player_name(), pos2str(pos)) | ||||||
|  | end) | ||||||
| @@ -11,6 +11,7 @@ local gui_count2 = {} --mapping of player names to a quantity (arbitrary strings | |||||||
| local gui_count3 = {} --mapping of player names to a quantity (arbitrary strings may also appear as values) | local gui_count3 = {} --mapping of player names to a quantity (arbitrary strings may also appear as values) | ||||||
| local gui_angle = {} --mapping of player names to an angle (one of 90, 180, 270, representing the angle in degrees clockwise) | local gui_angle = {} --mapping of player names to an angle (one of 90, 180, 270, representing the angle in degrees clockwise) | ||||||
| local gui_filename = {} --mapping of player names to file names | local gui_filename = {} --mapping of player names to file names | ||||||
|  | local gui_param2 = {} --mapping of player names to param2 values | ||||||
|  |  | ||||||
| --set default values | --set default values | ||||||
| setmetatable(gui_nodename1, {__index = function() return "Cobblestone" end}) | setmetatable(gui_nodename1, {__index = function() return "Cobblestone" end}) | ||||||
| @@ -25,6 +26,7 @@ setmetatable(gui_count2,     {__index = function() return "6" end}) | |||||||
| setmetatable(gui_count3,     {__index = function() return "4" end}) | setmetatable(gui_count3,     {__index = function() return "4" end}) | ||||||
| setmetatable(gui_angle,     {__index = function() return 90 end}) | setmetatable(gui_angle,     {__index = function() return 90 end}) | ||||||
| setmetatable(gui_filename,  {__index = function() return "building" end}) | setmetatable(gui_filename,  {__index = function() return "building" end}) | ||||||
|  | setmetatable(gui_param2,    {__index = function() return "0" end}) | ||||||
|  |  | ||||||
| local axis_indices = {["X axis"]=1, ["Y axis"]=2, ["Z axis"]=3, ["Look direction"]=4} | local axis_indices = {["X axis"]=1, ["Y axis"]=2, ["Z axis"]=3, ["Look direction"]=4} | ||||||
| local axis_values = {"x", "y", "z", "?"} | local axis_values = {"x", "y", "z", "?"} | ||||||
| @@ -904,3 +906,31 @@ worldedit.register_gui_function("worldedit_gui_clearobjects", { | |||||||
| 		execute_worldedit_command("clearobjects", name, "") | 		execute_worldedit_command("clearobjects", name, "") | ||||||
| 	end, | 	end, | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  | worldedit.register_gui_function("worldedit_gui_param2", { | ||||||
|  | 	name = "Set Param2", | ||||||
|  | 	privs = we_privs("param2"), | ||||||
|  | 	get_formspec = function(name) | ||||||
|  | 		local value = gui_param2[name] or "0" | ||||||
|  | 		return "size[6.5,3]" .. worldedit.get_formspec_header("worldedit_gui_param2") .. | ||||||
|  | 			"textarea[0.5,1;5,2;;;Some values may break the node!]".. | ||||||
|  | 			string.format("field[0.5,2.5;2,0.8;worldedit_gui_param2_value;New Param2;%s]", minetest.formspec_escape(value)) .. | ||||||
|  | 			"field_close_on_enter[worldedit_gui_param2_value;false]" .. | ||||||
|  | 			"button_exit[3.5,2.5;3,0.8;worldedit_gui_param2_submit;Set Param2]" | ||||||
|  | 	end, | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | worldedit.register_gui_handler("worldedit_gui_param2", function(name, fields) | ||||||
|  | 	local cg = { | ||||||
|  | 		worldedit_gui_param2_value = gui_param2, | ||||||
|  | 	} | ||||||
|  | 	local ret = handle_changes(name, "worldedit_gui_param2", fields, cg) | ||||||
|  | 	if fields.worldedit_gui_param2_submit then | ||||||
|  | 		copy_changes(name, fields, cg) | ||||||
|  | 		worldedit.show_page(name, "worldedit_gui_param2") | ||||||
|  |  | ||||||
|  | 		execute_worldedit_command("param2", name, gui_param2[name]) | ||||||
|  | 		return true | ||||||
|  | 	end | ||||||
|  | 	return ret | ||||||
|  | end) | ||||||
|   | |||||||
| @@ -216,7 +216,7 @@ elseif minetest.global_exists("sfinv") then -- sfinv installed | |||||||
| 		end | 		end | ||||||
| 	end | 	end | ||||||
| else | else | ||||||
| 	error( | 	return minetest.log("error", | ||||||
| 		"worldedit_gui requires a supported gui management mod to be installed.\n".. | 		"worldedit_gui requires a supported gui management mod to be installed.\n".. | ||||||
| 		"To use the it you need to either:\n".. | 		"To use the it you need to either:\n".. | ||||||
| 		"* use minetest_game or another sfinv-compatible subgame\n".. | 		"* use minetest_game or another sfinv-compatible subgame\n".. | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user