forked from Sokomine/replacer_setter
		
	working test version that does not set the replacer yet
This commit is contained in:
		
						commit
						df14c3200b
					
				
							
								
								
									
										2
									
								
								depends.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								depends.txt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | ||||
| default | ||||
| replacer | ||||
							
								
								
									
										207
									
								
								fs_history.lua
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										207
									
								
								fs_history.lua
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,207 @@ | ||||
| -- apart from the history, the formspec also handles mode switches | ||||
| 
 | ||||
| -- how many patterns are stored in the history? those don't take up much space, | ||||
| -- but a too long list might not be overly helpful for the players either | ||||
| replacer_setter.max_hist_size = 30 | ||||
| 
 | ||||
| 
 | ||||
| -- turn stored pattern string (<node_name> <param1> <param2>) into something readable by human beeings | ||||
| replacer_setter.human_readable_pattern = function(pattern) | ||||
| 	if(not(pattern)) then | ||||
| 		return "(nothing)" | ||||
| 	end | ||||
| 	-- data is stored in the form "<nodename> <param1> <param2>" | ||||
| 	local parts = string.split(pattern, " ") | ||||
| 	if(not(parts) or #parts < 3) then | ||||
| 		return "(corrupted data)" | ||||
| 	end | ||||
| 	local node_name = parts[1] | ||||
| 	local param2 = parts[3] | ||||
| 
 | ||||
| 	local def = minetest.registered_nodes[ node_name ] | ||||
| 	if(not(def)) then | ||||
| 		return "(unknown node)" | ||||
| 	end | ||||
| 	local text = "'"..tostring(def.description or "- no description -").."'" | ||||
| 	if(not(def.description) or def.description == "") then | ||||
| 		text = "- no description -" | ||||
| 	end | ||||
| 	-- facedir is probably the most commonly used rotation variant | ||||
| 	if( def.paramtype2 == "facedir" | ||||
| 	 or def.paramtype2 == "colorfacedir") then | ||||
| 		local axis_names = {"y+ (Ground)", "z+ (North)", "z- (South)", | ||||
| 				    "x+ (East)", "x- (West)", "y- (Sky)"} | ||||
| 		text = text.." Rotated: "..tostring(param2 % 4).. | ||||
| 			" around axis: "..tostring( axis_names[ math.floor( (param2%24) / 4 ) + 1 ]) | ||||
| 	-- wallmounted is diffrent | ||||
| 	elseif( def.paramtype2 == "wallmounted" | ||||
| 	     or def.paramtype2 == "colorwallmounted") then | ||||
| 		local axis_names = {"y+ (Ground)", "y- (Sky)", | ||||
| 				    "z+ (North)", "z- (South)", | ||||
| 				    "x+ (East)", "x- (West)"} | ||||
| 		text = text.." Mounted at wall: "..tostring( axis_names[ (param2 % 6)+ 1 ]) | ||||
| 	end | ||||
| 	return text | ||||
| end | ||||
| 
 | ||||
| 
 | ||||
| -- set the replacer to a new pattern | ||||
| replacer_setter.set_to = function(player_name, pattern, player, itemstack) | ||||
| 	if(not(player_name) or not(player) or not(itemstack)) then | ||||
| 		return itemstack | ||||
| 	end | ||||
| 	-- fallback if nothing is given | ||||
| 	if(not(pattern)) then | ||||
| 		pattern = "default:dirt 0 0" | ||||
| 	end | ||||
| 
 | ||||
| 	local set_to = replacer_setter.human_readable_pattern(pattern) | ||||
| 	-- change the description of the tool so that it's easier to see which replacer (if you | ||||
| 	-- have more than one in your inv) is set to which node | ||||
| 	local meta = itemstack:get_meta() | ||||
| 	-- actually store the new pattern | ||||
| 	meta:set_string("pattern", pattern ) | ||||
| 
 | ||||
| 	meta:set_string("description", "Node replacement tool set to:\n"..set_to.. | ||||
| 					"\n["..tostring(pattern).."]") | ||||
| 
 | ||||
| 	minetest.chat_send_player(player_name, "Node replacement tool set to: "..set_to.. | ||||
| 					"["..tostring(pattern).."].") | ||||
| 
 | ||||
| 	replacer_setter.add_to_hist(player_name, pattern) | ||||
| 	return itemstack -- nothing consumed but data changed | ||||
| end | ||||
| 
 | ||||
| 
 | ||||
| -- keep a history of stored patterns for each player (not for each replacer); | ||||
| -- this history is not saved over server shutdown | ||||
| replacer_setter.add_to_hist = function(player_name, pattern) | ||||
| 	if(not(player_name) or not(pattern) or pattern == "") then | ||||
| 		return | ||||
| 	end | ||||
| 	if(not(replacer_setter.history)) then | ||||
| 		replacer_setter.history = {} | ||||
| 	end | ||||
| 	if(not(replacer_setter.history[ player_name ])) then | ||||
| 		replacer_setter.history[ player_name ] = {} | ||||
| 	end | ||||
| 	local index = table.indexof(replacer_setter.history[ player_name ], pattern) | ||||
| 	-- only add new entries; do not store duplicates | ||||
| 	if(index and index > -1) then | ||||
| 		return | ||||
| 	end | ||||
| 	-- remove the oldest entry | ||||
| 	if(#replacer_setter.history[ player_name ] >= replacer_setter.max_hist_size) then | ||||
| 		table.remove(replacer_setter.history[ player_name ], 1) | ||||
| 	end | ||||
| 	-- insert at the beginning of the list | ||||
| 	table.insert(replacer_setter.history[ player_name ], 1, pattern) | ||||
| end | ||||
| 
 | ||||
| 
 | ||||
| -- show a formspec with a history of stored patterns to select from | ||||
| replacer_setter.get_formspec = function(player_name, current_pattern, player) | ||||
| 	-- is the player in creative mode? | ||||
| 	local in_creative_mode = (minetest.settings:get_bool("creative_mode") | ||||
| 				or minetest.check_player_privs(player_name, {creative=true})) | ||||
| 	-- make sure all variables exist and the current entry is stored | ||||
| 	replacer_setter.add_to_hist(player_name, current_pattern) | ||||
| 	-- count how many blocks of each type the player has in his inventory | ||||
| 	local counted_inv = {} | ||||
| 	if(not(in_creative_mode)) then | ||||
| 		local inv_main = player:get_inventory():get_list("main") | ||||
| 		for i, v in ipairs(inv_main) do | ||||
| 			local stack_name = v:get_name() | ||||
| 			if(not(counted_inv[ stack_name ])) then | ||||
| 				counted_inv[ stack_name ] = 0 | ||||
| 			end | ||||
| 			counted_inv[ stack_name ] = counted_inv[ stack_name ] + v:get_count() | ||||
| 		end | ||||
| 	end | ||||
| 
 | ||||
| 	-- find out which mode the player has currently selected | ||||
| 	local current_mode = 1 | ||||
| 	if(replacer_setter.user_mode and replacer_setter.user_mode[ player_name ]) then | ||||
| 		current_mode = table.indexof(replacer_setter.mode_names, replacer_setter.user_mode[ player_name ]) | ||||
| 		if(current_mode == -1) then | ||||
| 			current_mode = 1 | ||||
| 		end | ||||
| 	end | ||||
| 
 | ||||
| 	local formspec = "size[18,10]".. | ||||
| 		"label[6,0;Node Replacement Tool Setup and History]".. | ||||
| 		"button_exit[8,9.4;2,0.8;quit;Exit]".. | ||||
| 		"label[0.2,8.5;Note: Selected mode and history are reset on server restart.\n".. | ||||
| 			"Note: This selection is valid only for the replacer to the ".. | ||||
| 			"*left* of this tool here in your inventory.]".. | ||||
| 		"label[0.2,0.6;Select mode: When replacing (punching, left-click) or ".. | ||||
| 			"placing (right-click) a block, ..]".. | ||||
| 		"dropdown[0.2,1.0;17;select_mode;".. | ||||
| 			table.concat(replacer_setter.mode_descriptions, ",").. | ||||
| 			";"..tostring(current_mode)..";]".. | ||||
| 		"label[0.2,2.1;Click here to set the replacer to a pattern you have stored before:]".. | ||||
| 		"tablecolumns[color;".. | ||||
| 			"text,align=right,tooltip=Amount of nodes of this type left in your inventory:".. | ||||
| 			";color;text,align=left,tooltip=Stored pattern:]".. | ||||
| 		"table[0.2,2.5;17,6;replacer_history;" | ||||
| 	local hist_entries = {} | ||||
| 	local selected = 1 | ||||
| 	for i, v in ipairs(replacer_setter.history[ player_name ]) do | ||||
| 		if(v == current_pattern) then | ||||
| 			selected = i | ||||
| 		end | ||||
| 		local amount_left = "#00FF00,infinite supply:,#00FF00" | ||||
| 		if(not(in_creative_mode)) then | ||||
| 			-- which item are we looking for? | ||||
| 			local parts = v:split(" ") | ||||
| 			if(not(parts) or #parts<1) then | ||||
| 				parts = {"does not exist"} | ||||
| 			-- TODO: handle this in a more general way | ||||
| 			elseif(parts[1] == "default:dirt_with_grass") then | ||||
| 				parts[1] = "default:dirt" | ||||
| 			end | ||||
| 			if(counted_inv[ parts[1] ]) then | ||||
| 				amount_left = "#00FF00,"..tostring(counted_inv[ parts[1] ]).." available:".. | ||||
| 					",#00FF00" | ||||
| 			else | ||||
| 				amount_left = "#FF0000,none left!,#CFCFCF" | ||||
| 			end | ||||
| 		end | ||||
| 		hist_entries[ i ] = tostring(amount_left)..",".. | ||||
| 			minetest.formspec_escape(replacer_setter.human_readable_pattern(v).." ["..v.."]") | ||||
| 	end | ||||
| 	return formspec..table.concat(hist_entries, ",")..";"..tostring(selected).."]" | ||||
| end | ||||
| 
 | ||||
| 
 | ||||
| -- the player has interacted with our formspec | ||||
| minetest.register_on_player_receive_fields( function(player, formname, fields) | ||||
| 	if(not(formname) or formname ~= "replacer:menu") then | ||||
| 		return false | ||||
| 	end | ||||
| 	local player_name = player:get_player_name() | ||||
| 	-- the player clicked on an entry in the history | ||||
| 	if(fields and fields.replacer_history | ||||
| 	   and replacer_setter.history and replacer_setter.history[ player_name ]) then | ||||
| 		-- find out which line it was | ||||
| 		local selected = minetest.explode_table_event(fields.replacer_history) | ||||
| 		if(selected and (selected.type == "CHG" or selected.type == "DLC") | ||||
| 		   and selected.row <= #replacer_setter.history[ player_name ]) then | ||||
| 			local itemstack = player:get_wielded_item() | ||||
| 			itemstack = replacer_setter.set_to(player_name, | ||||
| 				replacer_setter.history[ player_name ][ selected.row ], | ||||
| 				player, itemstack) | ||||
| 			player:set_wielded_item(itemstack) | ||||
| 			return true | ||||
| 		end | ||||
| 	end | ||||
| 	-- the player selected a mode | ||||
| 	if(fields and fields.select_mode) then | ||||
| 		local index = table.indexof(replacer_setter.mode_descriptions, | ||||
| 				minetest.formspec_escape(fields.select_mode)) | ||||
| 		if(index and index > -1 and replacer_setter.mode_names[ index ]) then | ||||
| 			replacer_setter.user_mode[ player_name ] = replacer_setter.mode_names[ index ] | ||||
| 		end | ||||
| 	end | ||||
| 	return true | ||||
| end) | ||||
							
								
								
									
										70
									
								
								init.lua
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								init.lua
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,70 @@ | ||||
| --[[ | ||||
|     Helper Tool for setting the Replacer to new values when using | ||||
|     mobile devices. Comes with a history. | ||||
|     Copyright (C) 2022 Sokomine | ||||
| 
 | ||||
|     License: GPLv3.0 | ||||
| --]] | ||||
| 
 | ||||
| replacer_setter = {} | ||||
| replacer_setter.history = {} | ||||
| 
 | ||||
| -- no idea which other modes other replacers may support... | ||||
| replacer_setter.mode_descriptions = { | ||||
|         "[ normal ] replace material, shape and orientation according to the stored pattern" | ||||
| } | ||||
| -- internal names for the above modes (will be stored in replacer.user_mode[ player_name ]) | ||||
| replacer_setter.mode_names = {"normal"}  | ||||
|          | ||||
| -- make sure all mode descriptions are properly escaped for the dropdown menu | ||||
| for i, v in ipairs(replacer_setter.mode_descriptions) do | ||||
|         replacer_setter.mode_descriptions[i] = minetest.formspec_escape(v) | ||||
| end      | ||||
| 
 | ||||
| replacer_setter.user_mode = {} | ||||
| 
 | ||||
| dofile(minetest.get_modpath("replacer_setter").."/fs_history.lua"); | ||||
| 
 | ||||
| 
 | ||||
| minetest.register_tool("replacer_setter:setter", | ||||
| { | ||||
|     description = "Replacer Set Tool", | ||||
|     groups = {},  | ||||
|     inventory_image = "default_sign_wall_wood.png^replacer_replacer.png", | ||||
|     wield_image = "", | ||||
|     wield_scale = {x=1,y=1,z=1}, | ||||
|     stack_max = 1, -- there is no reason to have more than one | ||||
|     liquids_pointable = true, -- it is ok to painit in/with water | ||||
| 
 | ||||
|     node_placement_prediction = nil, | ||||
| 
 | ||||
|     on_use = function(itemstack, user, pointed_thing) | ||||
| 	if(not(user) or not(pointed_thing)) then | ||||
| 		return | ||||
| 	end | ||||
| 	local new_pattern = "" | ||||
| 	if(pointed_thing.type == "node") then | ||||
| 		local pos  = minetest.get_pointed_thing_position( pointed_thing, false ) | ||||
| 		local node = minetest.get_node_or_nil( pos ) | ||||
| 		new_pattern = "default:dirt 0 0"; | ||||
| 		if( node ~= nil and node.name ) then | ||||
| 			new_pattern = node.name..' '..node.param1..' '..node.param2; | ||||
| 		end | ||||
| 	end | ||||
| 	local pname = user:get_player_name() | ||||
| 	if(not(replacer_setter.history[pname])) then | ||||
| 		replacer_setter.history[pname] = {} | ||||
| 	end | ||||
| 	minetest.show_formspec(pname, "replacer:menu", replacer_setter.get_formspec(pname, new_pattern, user)) | ||||
| 	return nil | ||||
|     end, | ||||
| }) | ||||
| 
 | ||||
| 
 | ||||
| minetest.register_craft({ | ||||
|         output = 'replacer_setter:setter', | ||||
|         recipe = { | ||||
| 		{"replacer:replacer"}, | ||||
| 	} | ||||
| }) | ||||
| 
 | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user