--[[ This mod implements a HACK to make 100% sure the digging times of all tools match Minecraft's perfectly. The digging times system of Luanti is very different, so this weird group trickery has to be used. In Minecraft, each block has a hardness and the actual Minecraft digging time is determined by this: 1) The block's hardness 2) The tool being used (the tool speed and its efficiency level) 3) Whether the tool is considered as "eligible" for the block (e.g. only diamond pick eligible for obsidian) See Minecraft Wiki for more information. How the mod is used =================== In VoxeLibre, all diggable nodes have the hardness set in the custom field "_mcl_hardness" (0 by default). These values are used together with digging groups by this mod to create the correct digging times for nodes. Digging groups are registered using the following code: mcl_autogroup.register_diggroup("shovely") mcl_autogroup.register_diggroup("pickaxey", { levels = { "wood", "gold", "stone", "iron", "diamond" } }) The first line registers a simple digging group. The second line registers a digging group with 5 different levels (in this case one for each material of a pickaxes). Nodes indicate that they belong to a particular digging group by being member of the digging group in their node definition. "mcl_core:dirt" for example has shovely=1 in its groups. If the digging group has multiple levels the value of the group indicates which digging level the node requires. "mcl_core:stone_with_gold" for example has pickaxey=4 because it requires a pickaxe of level 4 be mined. For tools to be able to dig nodes of digging groups they need to use the have the custom field "_mcl_diggroups" function to get the groupcaps. The value of this field is a table which defines which groups the tool can dig and how efficiently. _mcl_diggroups = { handy = { speed = 1, level = 1, uses = 0 }, pickaxey = { speed = 1, level = 0, uses = 0 }, } The "uses" field indicate how many uses (0 for infinite) a tool has when used on the specified digging group. The "speed" field is a multiplier to the dig speed on that digging group. The "level" field indicates which levels of the group the tool can harvest. A level of 0 means that the tool cannot harvest blocks of that node. A level of 1 or above means that the tool can harvest nodes with that level or below. See "mcl_tools/init.lua" for examples on how "_mcl_diggroups" is used in practice. Information about the mod ========================= The mod is split up into two parts, mcl_autogroup and _mcl_autogroup. mcl_autogroup contains the API functions used to register custom digging groups. _mcl_autogroup contains most of the code. The leading underscore in the name "_mcl_autogroup" is used to force Luanti to load that part of the mod as late as possible. Luanti loads mods in reverse alphabetical order. This also means that it is very important that no mod adds _mcl_autogroup as a dependency. --]] assert(minetest.get_modpath("mcl_autogroup"), "This mod requires the mod mcl_autogroup to function") -- Returns a table containing the unique "_mcl_hardness" for nodes belonging to -- each diggroup. local function get_hardness_values_for_groups() local maps = {} local values = {} for g, _ in pairs(mcl_autogroup.registered_diggroups) do maps[g] = {} values[g] = {} end for _, ndef in pairs(minetest.registered_nodes) do for g, _ in pairs(mcl_autogroup.registered_diggroups) do if ndef.groups[g] then maps[g][ndef._mcl_hardness or 0] = true end end end for g, map in pairs(maps) do for k, _ in pairs(map) do table.insert(values[g], k) end end for g, _ in pairs(mcl_autogroup.registered_diggroups) do table.sort(values[g]) end return values end -- Returns a table containing a table indexed by "_mcl_hardness_value" to get -- its index in the list of unique hardnesses for each diggroup. local function get_hardness_lookup_for_groups(hardness_values) local map = {} for g, values in pairs(hardness_values) do map[g] = {} for k, v in pairs(values) do map[g][v] = k end end return map end -- Array of unique hardness values for each group which affects dig time. local hardness_values = get_hardness_values_for_groups() --[[local function compute_creativetimes(group) local creativetimes = {} for index, hardness in pairs(hardness_values[group]) do table.insert(creativetimes, 0) end return creativetimes end]] -- Get the list of digging times for using a specific tool on a specific -- diggroup. -- -- Parameters: -- group - the group which it is digging -- can_harvest - if the tool can harvest the block -- speed - dig speed multiplier for tool (default 1) -- efficiency - efficiency level for the tool if applicable local function get_digtimes(group, can_harvest, speed, efficiency) local speed = speed or 1 if efficiency then speed = speed + efficiency * efficiency + 1 end local digtimes = {} for _, hardness in pairs(hardness_values[group]) do local digtime = (hardness or 0) / speed if can_harvest then digtime = digtime * 1.5 else digtime = digtime * 5 end if digtime <= 0.05 then digtime = 0 else digtime = math.ceil(digtime * 20) / 20 end table.insert(digtimes, digtime) end return digtimes end -- Get one groupcap field for using a specific tool on a specific group. local function get_groupcap(group, can_harvest, multiplier, efficiency, uses) return { times = get_digtimes(group, can_harvest, multiplier, efficiency), uses = uses, maxlevel = 0, } end -- Add the groupcaps from a field in "_mcl_diggroups" to the groupcaps of a -- tool. local function add_groupcaps(toolname, groupcaps, groupcaps_def, efficiency) if not groupcaps_def then return end for g, capsdef in pairs(groupcaps_def) do local mult = capsdef.speed or 1 local uses = capsdef.uses local def = mcl_autogroup.registered_diggroups[g] assert(def, toolname .. " has unknown diggroup '" .. dump(g) .. "'") local max_level = def.levels and #def.levels or 1 assert(capsdef.level, toolname .. ' is missing level for ' .. g) local level = math.min(capsdef.level, max_level) if def.levels then groupcaps[g .. "_dig_default"] = get_groupcap(g, false, mult, efficiency, uses) if level > 0 then groupcaps[g .. "_dig_" .. def.levels[level]] = get_groupcap(g, true, mult, efficiency, uses) end else groupcaps[g .. "_dig"] = get_groupcap(g, level > 0, mult, efficiency, uses) end end end -- Checks if the given node would drop its useful drop if dug by a given tool. -- Returns true if it will yield its useful drop, false otherwise. function mcl_autogroup.can_harvest(nodename, toolname, player) local ndef = minetest.registered_nodes[nodename] if not ndef then return false end if minetest.get_item_group(nodename, "dig_immediate") >= 2 then return true end if minetest.get_item_group(nodename, "dig_immediate_piston") >= 1 then return true end -- Check if it can be dug by tool local tdef = minetest.registered_tools[toolname] if tdef and tdef._mcl_diggroups then for g, gdef in pairs(tdef._mcl_diggroups) do if ndef.groups[g] then if ndef.groups[g] <= gdef.level then return true end end end end -- Check if it can be dug by hand if not player or not player:is_player() then return false end local name = player:get_inventory():get_stack("hand", 1):get_name() tdef = minetest.registered_items[name] if tdef then for g, gdef in pairs(tdef._mcl_diggroups) do if ndef.groups[g] then if ndef.groups[g] <= gdef.level then return true end end end end return false end -- Get one groupcap field for using a specific tool on a specific group. --[[local function get_groupcap(group, can_harvest, multiplier, efficiency, uses) return { times = get_digtimes(group, can_harvest, multiplier, efficiency), uses = uses, maxlevel = 0, } end]] -- Returns the tool_capabilities from a tool definition or a default set of -- tool_capabilities local function get_tool_capabilities(tdef) if tdef.tool_capabilities then return tdef.tool_capabilities end -- If the damage group and punch interval from hand is not included, -- then the user will not be able to attack with the tool. local hand_toolcaps = mcl_meshhand.survival_hand_tool_caps return { full_punch_interval = hand_toolcaps.full_punch_interval, damage_groups = hand_toolcaps.damage_groups } end -- Get the groupcaps for a tool. This function returns "groupcaps" table of -- digging which should be put in the "tool_capabilities" of the tool definition -- or in the metadata of an enchanted tool. -- -- Parameters: -- toolname - Name of the tool being enchanted (like "mcl_tools:diamond_pickaxe") -- efficiency - The efficiency level the tool is enchanted with (default 0) -- -- NOTE: -- This function can only be called after mod initialization. Otherwise a mod -- would have to add _mcl_autogroup as a dependency which would break the mod -- loading order. function mcl_autogroup.get_groupcaps(toolname, efficiency) local tdef = minetest.registered_items[toolname] local groupcaps = table.copy(get_tool_capabilities(tdef).groupcaps or {}) add_groupcaps(toolname, groupcaps, tdef._mcl_diggroups, efficiency) return groupcaps end -- Get the wear from using a tool on a digging group. -- -- Parameters -- toolname - Name of the tool used -- diggroup - The name of the diggroup the tool is used on -- -- NOTE: -- This function can only be called after mod initialization. Otherwise a mod -- would have to add _mcl_autogroup as a dependency which would break the mod -- loading order. function mcl_autogroup.get_wear(toolname, diggroup) local tdef = minetest.registered_tools[toolname] if not tdef then minetest.log("warning", "Adding wear for tool: " .. tostring(toolname) .. " failed with diggroup: " .. tostring(diggroup)) return nil end local uses = tdef._mcl_diggroups[diggroup].uses return math.ceil(65535 / uses) end local GROUP_MAP = { ["choppy"] = "axey", ["oddly_breakable_by_hand"] = "handy", ["cracky"] = "pickaxey", ["crumbly"] = "shovely", ["snappy"] = "shearsy", } function mcl_autogroup.group_compatibility(groups) local grouped = false for name,old_group_value in pairs(groups) do local new_group = GROUP_MAP[name] if new_group then groups[new_group] = old_group_value end if mcl_autogroup.registered_diggroups[name] then grouped = true end end if not grouped then groups.handy = 1 end end local function overwrite() -- Refresh, now that all groups are known. hardness_values = get_hardness_values_for_groups() -- Map indexed by hardness values which return the index of that value in -- hardness_value. Used for quick lookup. local hardness_lookup = get_hardness_lookup_for_groups(hardness_values) local count = 0 for nname, ndef in pairs(minetest.registered_nodes) do count = count + 1 local newgroups = table.copy(ndef.groups) if not newgroups.unbreakable and not newgroups.indestructible then ndef.diggable = true mcl_autogroup.group_compatibility(newgroups) if not ndef._mcl_hardness then ndef._mcl_hardness = 0 end end -- Make sure compatibility groups are present for the below logic ndef.groups = newgroups if (nname ~= "ignore" and ndef.diggable) then -- Automatically assign the "solid" group for solid nodes if (ndef.walkable == nil or ndef.walkable == true) and (ndef.collision_box == nil or ndef.collision_box.type == "regular") and (ndef.node_box == nil or ndef.node_box.type == "regular") and (ndef.groups.not_solid == 0 or ndef.groups.not_solid == nil) then newgroups.solid = 1 end -- Automatically assign the "opaque" group for opaque nodes if (not (ndef.paramtype == "light" or ndef.sunlight_propagates)) and (ndef.groups.not_opaque == 0 or ndef.groups.not_opaque == nil) then newgroups.opaque = 1 end --local creative_breakable = false -- Assign groups used for digging this node depending on -- the registered digging groups for g, gdef in pairs(mcl_autogroup.registered_diggroups) do --creative_breakable = true local index = hardness_lookup[g][ndef._mcl_hardness or 0] if ndef.groups[g] then if gdef.levels then newgroups[g .. "_dig_default"] = index for i = ndef.groups[g], #gdef.levels do newgroups[g .. "_dig_" .. gdef.levels[i]] = index end else newgroups[g .. "_dig"] = index end end end -- Automatically assign the node to the -- creative_breakable group if it belongs to any digging -- group. newgroups["creative_breakable"] = 1 minetest.override_item(nname, { groups = newgroups }) end end minetest.log("verbose","Total registered nodes: "..count) for tname, tdef in pairs(minetest.registered_items) do -- Assign groupcaps for digging the registered digging groups -- depending on the _mcl_diggroups in the tool definition if tdef._mcl_diggroups then local toolcaps = table.copy(get_tool_capabilities(tdef)) toolcaps.groupcaps = mcl_autogroup.get_groupcaps(tname) minetest.override_item(tname, { tool_capabilities = toolcaps }) end end end -- Make sure all tools and groups are registered minetest.register_on_mods_loaded(overwrite)