2021-10-24 23:31:01 +02:00
|
|
|
local replacements = {fuel = {}}
|
2021-12-06 14:44:40 +01:00
|
|
|
local http = ...
|
2021-10-24 23:31:01 +02:00
|
|
|
|
2021-11-29 19:15:14 +01:00
|
|
|
IMPORT("maxn", "copy", "insert", "sort", "match", "sub")
|
|
|
|
IMPORT("is_group", "extract_groups", "item_has_groups", "groups_to_items")
|
2022-06-18 21:50:13 +02:00
|
|
|
IMPORT("fmt", "reg_items", "reg_aliases", "reg_nodes", "is_cube", "get_cube", "ItemStack")
|
|
|
|
IMPORT("true_str", "is_table", "valid_item", "table_merge", "table_replace", "rcp_eq")
|
2021-10-24 23:31:01 +02:00
|
|
|
|
|
|
|
local function get_burntime(item)
|
|
|
|
return core.get_craft_result{method = "fuel", items = {item}}.time
|
|
|
|
end
|
|
|
|
|
|
|
|
local function cache_fuel(item)
|
|
|
|
local burntime = get_burntime(item)
|
|
|
|
if burntime > 0 then
|
|
|
|
i3.fuel_cache[item] = {
|
|
|
|
type = "fuel",
|
|
|
|
items = {item},
|
|
|
|
burntime = burntime,
|
|
|
|
replacements = replacements.fuel[item],
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-06-18 19:47:23 +02:00
|
|
|
local function cache_groups(groupname, groups)
|
|
|
|
i3.groups[groupname] = {}
|
|
|
|
i3.groups[groupname].groups = groups
|
|
|
|
i3.groups[groupname].items = groups_to_items(groups, true)
|
2022-06-18 21:50:13 +02:00
|
|
|
|
|
|
|
local items = i3.groups[groupname].items
|
|
|
|
local nb_items = #items
|
|
|
|
|
|
|
|
if nb_items > 1 then
|
|
|
|
local px = 256
|
|
|
|
local sprite = fmt("[combine:%ux%u", px, nb_items + (px * nb_items) - 1)
|
|
|
|
|
|
|
|
for i = 1, nb_items do
|
|
|
|
local item = items[i]
|
|
|
|
local def = reg_items[item]
|
|
|
|
local texture = def.inventory_image or def.wield_image
|
|
|
|
|
|
|
|
if is_cube(def.drawtype) then
|
|
|
|
texture = get_cube(def.tiles)
|
|
|
|
end
|
|
|
|
|
|
|
|
sprite = sprite .. fmt(":0,%u=%s", (i - 1) + ((i - 1) * px), texture)
|
|
|
|
end
|
|
|
|
|
|
|
|
i3.groups[groupname].sprite = sprite
|
|
|
|
end
|
2022-06-18 19:47:23 +02:00
|
|
|
end
|
|
|
|
|
2022-06-18 21:50:13 +02:00
|
|
|
|
2021-10-24 23:31:01 +02:00
|
|
|
local function get_item_usages(item, recipe, added)
|
2022-06-18 19:47:23 +02:00
|
|
|
if is_group(item) then
|
|
|
|
local groupname = item:sub(7)
|
|
|
|
local group_cache = i3.groups[groupname]
|
|
|
|
local groups = group_cache and group_cache.groups or extract_groups(item)
|
|
|
|
|
|
|
|
if not group_cache then
|
|
|
|
cache_groups(groupname, groups)
|
|
|
|
end
|
2021-10-24 23:31:01 +02:00
|
|
|
|
|
|
|
for name, def in pairs(reg_items) do
|
2021-11-24 00:51:58 +01:00
|
|
|
if not added[name] and valid_item(def) and item_has_groups(def.groups, groups) then
|
2021-10-24 23:31:01 +02:00
|
|
|
local usage = copy(recipe)
|
|
|
|
table_replace(usage.items, item, name)
|
|
|
|
|
|
|
|
i3.usages_cache[name] = i3.usages_cache[name] or {}
|
|
|
|
insert(i3.usages_cache[name], 1, usage)
|
|
|
|
|
|
|
|
added[name] = true
|
|
|
|
end
|
|
|
|
end
|
2021-11-24 00:51:58 +01:00
|
|
|
elseif valid_item(reg_items[item]) then
|
2021-10-24 23:31:01 +02:00
|
|
|
i3.usages_cache[item] = i3.usages_cache[item] or {}
|
|
|
|
insert(i3.usages_cache[item], 1, recipe)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function get_usages(recipe)
|
|
|
|
local added = {}
|
|
|
|
|
|
|
|
for _, item in pairs(recipe.items) do
|
|
|
|
item = reg_aliases[item] or item
|
|
|
|
|
|
|
|
if not added[item] then
|
|
|
|
get_item_usages(item, recipe, added)
|
|
|
|
added[item] = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function cache_usages(item)
|
|
|
|
local recipes = i3.recipes_cache[item] or {}
|
|
|
|
|
|
|
|
for i = 1, #recipes do
|
|
|
|
get_usages(recipes[i])
|
|
|
|
end
|
|
|
|
|
|
|
|
if i3.fuel_cache[item] then
|
|
|
|
i3.usages_cache[item] = table_merge(i3.usages_cache[item] or {}, {i3.fuel_cache[item]})
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function drop_table(name, drop)
|
|
|
|
local count_sure = 0
|
|
|
|
local drop_items = drop.items or {}
|
|
|
|
local max_items = drop.max_items
|
|
|
|
|
|
|
|
for i = 1, #drop_items do
|
|
|
|
local di = drop_items[i]
|
|
|
|
local valid_rarity = di.rarity and di.rarity > 1
|
|
|
|
|
|
|
|
if di.rarity or not max_items or
|
|
|
|
(max_items and not di.rarity and count_sure < max_items) then
|
|
|
|
for j = 1, #di.items do
|
|
|
|
local dstack = ItemStack(di.items[j])
|
|
|
|
local dname = dstack:get_name()
|
|
|
|
local dcount = dstack:get_count()
|
|
|
|
local empty = dstack:is_empty()
|
|
|
|
|
|
|
|
if not empty and (dname ~= name or (dname == name and dcount > 1)) then
|
|
|
|
local rarity = valid_rarity and di.rarity
|
|
|
|
|
|
|
|
i3.register_craft {
|
|
|
|
type = rarity and "digging_chance" or "digging",
|
|
|
|
items = {name},
|
|
|
|
output = fmt("%s %u", dname, dcount),
|
|
|
|
rarity = rarity,
|
|
|
|
tools = di.tools,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if not di.rarity then
|
2021-11-29 02:47:36 +01:00
|
|
|
count_sure++
|
2021-10-24 23:31:01 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function cache_drops(name, drop)
|
|
|
|
if true_str(drop) then
|
|
|
|
local dstack = ItemStack(drop)
|
|
|
|
local dname = dstack:get_name()
|
|
|
|
local empty = dstack:is_empty()
|
|
|
|
|
|
|
|
if not empty and dname ~= name then
|
|
|
|
i3.register_craft {
|
|
|
|
type = "digging",
|
|
|
|
items = {name},
|
|
|
|
output = drop,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
elseif is_table(drop) then
|
|
|
|
drop_table(name, drop)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function cache_recipes(item)
|
|
|
|
local recipes = core.get_all_craft_recipes(item)
|
|
|
|
|
|
|
|
if replacements[item] then
|
|
|
|
local _recipes = {}
|
|
|
|
|
|
|
|
for k, v in ipairs(recipes or {}) do
|
|
|
|
_recipes[#recipes + 1 - k] = v
|
|
|
|
end
|
|
|
|
|
|
|
|
local shift = 0
|
|
|
|
local size_rpl = maxn(replacements[item])
|
|
|
|
local size_rcp = #_recipes
|
|
|
|
|
|
|
|
if size_rpl > size_rcp then
|
|
|
|
shift = size_rcp - size_rpl
|
|
|
|
end
|
|
|
|
|
|
|
|
for k, v in pairs(replacements[item]) do
|
2021-11-29 02:47:36 +01:00
|
|
|
k += shift
|
2021-10-24 23:31:01 +02:00
|
|
|
|
|
|
|
if _recipes[k] then
|
|
|
|
_recipes[k].replacements = v
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
recipes = _recipes
|
|
|
|
end
|
|
|
|
|
|
|
|
if recipes then
|
|
|
|
i3.recipes_cache[item] = table_merge(recipes, i3.recipes_cache[item] or {})
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
--[[ As `core.get_craft_recipe` and `core.get_all_craft_recipes` do not
|
|
|
|
return the fuel, replacements and toolrepair recipes, we have to
|
|
|
|
override `core.register_craft` and do some reverse engineering.
|
|
|
|
See engine's issues #4901, #5745 and #8920. ]]
|
|
|
|
|
|
|
|
local old_register_craft = core.register_craft
|
|
|
|
local rcp_num = {}
|
|
|
|
|
|
|
|
core.register_craft = function(def)
|
|
|
|
old_register_craft(def)
|
|
|
|
|
|
|
|
if def.type == "toolrepair" then
|
|
|
|
i3.toolrepair = def.additional_wear * -100
|
|
|
|
end
|
|
|
|
|
|
|
|
local output = def.output or (true_str(def.recipe) and def.recipe) or nil
|
|
|
|
if not output then return end
|
|
|
|
output = {match(output, "%S+")}
|
|
|
|
|
|
|
|
local groups
|
|
|
|
|
|
|
|
if is_group(output[1]) then
|
|
|
|
groups = extract_groups(output[1])
|
|
|
|
output = groups_to_items(groups, true)
|
|
|
|
end
|
|
|
|
|
|
|
|
for i = 1, #output do
|
|
|
|
local item = output[i]
|
|
|
|
rcp_num[item] = (rcp_num[item] or 0) + 1
|
|
|
|
|
|
|
|
if def.replacements then
|
|
|
|
if def.type == "fuel" then
|
|
|
|
replacements.fuel[item] = def.replacements
|
|
|
|
else
|
|
|
|
replacements[item] = replacements[item] or {}
|
|
|
|
replacements[item][rcp_num[item]] = def.replacements
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local old_clear_craft = core.clear_craft
|
|
|
|
|
|
|
|
core.clear_craft = function(def)
|
|
|
|
old_clear_craft(def)
|
|
|
|
|
|
|
|
if true_str(def) then
|
|
|
|
return -- TODO
|
|
|
|
elseif is_table(def) then
|
|
|
|
return -- TODO
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function resolve_aliases(hash)
|
|
|
|
for oldname, newname in pairs(reg_aliases) do
|
|
|
|
cache_recipes(oldname)
|
|
|
|
local recipes = i3.recipes_cache[oldname]
|
|
|
|
|
|
|
|
if recipes then
|
|
|
|
if not i3.recipes_cache[newname] then
|
|
|
|
i3.recipes_cache[newname] = {}
|
|
|
|
end
|
|
|
|
|
|
|
|
local similar
|
|
|
|
|
|
|
|
for i = 1, #i3.recipes_cache[oldname] do
|
|
|
|
local rcp_old = i3.recipes_cache[oldname][i]
|
|
|
|
|
|
|
|
for j = 1, #i3.recipes_cache[newname] do
|
|
|
|
local rcp_new = copy(i3.recipes_cache[newname][j])
|
2021-11-04 00:53:14 +01:00
|
|
|
rcp_new.output = oldname
|
2021-10-24 23:31:01 +02:00
|
|
|
|
2021-11-04 00:53:14 +01:00
|
|
|
if rcp_eq(rcp_old, rcp_new) then
|
2021-10-24 23:31:01 +02:00
|
|
|
similar = true
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if not similar then
|
|
|
|
insert(i3.recipes_cache[newname], rcp_old)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-04-25 16:00:49 +02:00
|
|
|
if newname ~= "" and i3.recipes_cache[oldname] and reg_items[newname] and not hash[newname] then
|
2021-11-15 23:50:48 +01:00
|
|
|
insert(i3.init_items, newname)
|
2021-10-24 23:31:01 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-10-25 07:09:21 +02:00
|
|
|
local function init_recipes()
|
|
|
|
local _select, _preselect = {}, {}
|
|
|
|
|
|
|
|
for name, def in pairs(reg_items) do
|
2021-11-24 00:51:58 +01:00
|
|
|
if name ~= "" and valid_item(def) then
|
2021-10-25 07:09:21 +02:00
|
|
|
cache_drops(name, def.drop)
|
|
|
|
cache_fuel(name)
|
|
|
|
cache_recipes(name)
|
|
|
|
|
|
|
|
_preselect[name] = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
for name in pairs(_preselect) do
|
|
|
|
cache_usages(name)
|
|
|
|
|
2021-11-15 23:50:48 +01:00
|
|
|
insert(i3.init_items, name)
|
2021-10-25 07:09:21 +02:00
|
|
|
_select[name] = true
|
|
|
|
end
|
|
|
|
|
|
|
|
resolve_aliases(_select)
|
|
|
|
sort(i3.init_items)
|
|
|
|
|
2021-12-06 14:44:40 +01:00
|
|
|
if http and true_str(i3.export_url) then
|
2021-10-25 07:09:21 +02:00
|
|
|
local post_data = {
|
|
|
|
recipes = i3.recipes_cache,
|
|
|
|
usages = i3.usages_cache,
|
|
|
|
}
|
|
|
|
|
2021-12-06 14:44:40 +01:00
|
|
|
http.fetch_async {
|
2021-10-25 07:09:21 +02:00
|
|
|
url = i3.export_url,
|
|
|
|
post_data = core.write_json(post_data),
|
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-11-25 04:22:46 +01:00
|
|
|
local function init_cubes()
|
|
|
|
for name, def in pairs(reg_nodes) do
|
2021-11-29 04:36:37 +01:00
|
|
|
if def then
|
2021-11-25 04:22:46 +01:00
|
|
|
local id = core.get_content_id(name)
|
2021-11-29 04:36:37 +01:00
|
|
|
|
2022-06-18 19:56:47 +02:00
|
|
|
if is_cube(def.drawtype) then
|
2021-11-29 04:36:37 +01:00
|
|
|
i3.cubes[id] = get_cube(def.tiles)
|
|
|
|
elseif sub(def.drawtype, 1, 9) == "plantlike" or sub(def.drawtype, 1, 8) == "firelike" then
|
2022-06-18 19:56:47 +02:00
|
|
|
i3.plants[id] = def.inventory_image
|
2021-11-29 04:36:37 +01:00
|
|
|
end
|
2021-11-25 04:22:46 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return function()
|
|
|
|
init_recipes()
|
|
|
|
init_cubes()
|
|
|
|
end
|