2023-06-20 14:33:09 +10:00
|
|
|
#include "StarItemDatabase.hpp"
|
|
|
|
#include "StarCodexDatabase.hpp"
|
|
|
|
#include "StarJsonExtra.hpp"
|
|
|
|
#include "StarRoot.hpp"
|
|
|
|
#include "StarAssets.hpp"
|
|
|
|
#include "StarCasting.hpp"
|
|
|
|
#include "StarCurrency.hpp"
|
|
|
|
#include "StarConsumableItem.hpp"
|
|
|
|
#include "StarBlueprintItem.hpp"
|
|
|
|
#include "StarCodexItem.hpp"
|
|
|
|
#include "StarLiquidItem.hpp"
|
|
|
|
#include "StarMaterialItem.hpp"
|
|
|
|
#include "StarObjectItem.hpp"
|
|
|
|
#include "StarItemDrop.hpp"
|
|
|
|
#include "StarInspectionTool.hpp"
|
|
|
|
#include "StarInstrumentItem.hpp"
|
|
|
|
#include "StarThrownItem.hpp"
|
|
|
|
#include "StarUnlockItem.hpp"
|
|
|
|
#include "StarActiveItem.hpp"
|
|
|
|
#include "StarAugmentItem.hpp"
|
|
|
|
#include "StarTools.hpp"
|
|
|
|
#include "StarArmors.hpp"
|
|
|
|
#include "StarObjectDatabase.hpp"
|
|
|
|
#include "StarRootLuaBindings.hpp"
|
|
|
|
#include "StarItemLuaBindings.hpp"
|
|
|
|
#include "StarConfigLuaBindings.hpp"
|
|
|
|
#include "StarUtilityLuaBindings.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
|
|
|
EnumMap<ItemType> ItemTypeNames{
|
|
|
|
{ItemType::Generic, "generic"},
|
|
|
|
{ItemType::LiquidItem, "liquid"},
|
|
|
|
{ItemType::MaterialItem, "material"},
|
|
|
|
{ItemType::ObjectItem, "object"},
|
|
|
|
{ItemType::CurrencyItem, "currency"},
|
|
|
|
{ItemType::MiningTool, "miningtool"},
|
|
|
|
{ItemType::Flashlight, "flashlight"},
|
|
|
|
{ItemType::WireTool, "wiretool"},
|
|
|
|
{ItemType::BeamMiningTool, "beamminingtool"},
|
|
|
|
{ItemType::HarvestingTool, "harvestingtool"},
|
|
|
|
{ItemType::TillingTool, "tillingtool"},
|
|
|
|
{ItemType::PaintingBeamTool, "paintingbeamtool"},
|
|
|
|
{ItemType::HeadArmor, "headarmor"},
|
|
|
|
{ItemType::ChestArmor, "chestarmor"},
|
|
|
|
{ItemType::LegsArmor, "legsarmor"},
|
|
|
|
{ItemType::BackArmor, "backarmor"},
|
|
|
|
{ItemType::Consumable, "consumable"},
|
|
|
|
{ItemType::Blueprint, "blueprint"},
|
|
|
|
{ItemType::Codex, "codex"},
|
|
|
|
{ItemType::InspectionTool, "inspectiontool"},
|
|
|
|
{ItemType::InstrumentItem, "instrument"},
|
|
|
|
{ItemType::ThrownItem, "thrownitem"},
|
|
|
|
{ItemType::UnlockItem, "unlockitem"},
|
|
|
|
{ItemType::ActiveItem, "activeitem"},
|
|
|
|
{ItemType::AugmentItem, "augmentitem"}
|
|
|
|
};
|
|
|
|
|
|
|
|
uint64_t ItemDatabase::getCountOfItem(List<ItemPtr> const& bag, ItemDescriptor const& item, bool exactMatch) {
|
|
|
|
auto normalizedBag = normalizeBag(bag);
|
|
|
|
return getCountOfItem(normalizedBag, item, exactMatch);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t ItemDatabase::getCountOfItem(HashMap<ItemDescriptor, uint64_t> const& bag, ItemDescriptor const& item, bool exactMatch) {
|
|
|
|
ItemDescriptor matchItem = exactMatch ? item.singular() : ItemDescriptor(item.name(), 1);
|
|
|
|
if (!bag.contains(matchItem)) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return bag.get(matchItem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
HashMap<ItemDescriptor, uint64_t> ItemDatabase::normalizeBag(List<ItemPtr> const& bag) {
|
|
|
|
HashMap<ItemDescriptor, uint64_t> normalizedBag;
|
|
|
|
for (auto const& item : bag) {
|
|
|
|
if (!item)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
normalizedBag[ItemDescriptor(item->name(), 1)] += item->count();
|
|
|
|
|
|
|
|
if (!item->parameters().toObject().empty())
|
|
|
|
normalizedBag[ItemDescriptor(item->name(), 1, item->parameters())] += item->count();
|
|
|
|
}
|
|
|
|
|
|
|
|
return normalizedBag;
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::recipesFromSubset(HashMap<ItemDescriptor, uint64_t> const& normalizedBag, StringMap<uint64_t> const& availableCurrencies, HashSet<ItemRecipe> const& subset) {
|
|
|
|
HashSet<ItemRecipe> res;
|
|
|
|
for (auto const& recipe : subset) {
|
|
|
|
// add this recipe if we can make it.
|
|
|
|
if (canMakeRecipe(recipe, normalizedBag, availableCurrencies))
|
|
|
|
res.add(recipe);
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::recipesFromSubset(HashMap<ItemDescriptor, uint64_t> const& normalizedBag, StringMap<uint64_t> const& availableCurrencies,
|
|
|
|
HashSet<ItemRecipe> const& subset, StringSet const& allowedTypes) {
|
|
|
|
HashSet<ItemRecipe> res;
|
|
|
|
for (auto const& recipe : subset) {
|
|
|
|
// is it the right kind of recipe for this check ?
|
|
|
|
if (recipe.groups.hasIntersection(allowedTypes) || allowedTypes.empty() || recipe.groups.empty()) {
|
|
|
|
// do we have the ingredients to make it.
|
|
|
|
if (canMakeRecipe(recipe, normalizedBag, availableCurrencies)) {
|
|
|
|
res.add(recipe);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
String ItemDatabase::guiFilterString(ItemPtr const& item) {
|
|
|
|
return (item->name() + item->friendlyName() + item->description()).toLower().splitAny(" ,.?*\\+/|\t").join("");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ItemDatabase::canMakeRecipe(ItemRecipe const& recipe, HashMap<ItemDescriptor, uint64_t> const& availableIngredients, StringMap<uint64_t> const& availableCurrencies) {
|
|
|
|
for (auto const& p : recipe.currencyInputs) {
|
|
|
|
if (availableCurrencies.value(p.first, 0) < p.second)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto const& input : recipe.inputs) {
|
|
|
|
ItemDescriptor matchInput = recipe.matchInputParameters ? input.singular() : ItemDescriptor(input.name(), 1);
|
|
|
|
if (availableIngredients.value(matchInput) < input.count())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemDatabase::ItemDatabase()
|
|
|
|
: m_luaRoot(make_shared<LuaRoot>()) {
|
|
|
|
scanItems();
|
|
|
|
addObjectItems();
|
|
|
|
addCodexes();
|
|
|
|
scanRecipes();
|
|
|
|
addBlueprints();
|
|
|
|
}
|
|
|
|
|
2023-07-23 22:44:02 +10:00
|
|
|
void ItemDatabase::cleanup() {
|
|
|
|
{
|
|
|
|
MutexLocker locker(m_cacheMutex);
|
|
|
|
m_itemCache.cleanup([](ItemCacheEntry const&, ItemPtr const& item) {
|
|
|
|
return !item.unique();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-20 14:33:09 +10:00
|
|
|
ItemPtr ItemDatabase::diskLoad(Json const& diskStore) const {
|
|
|
|
if (diskStore) {
|
|
|
|
return item(ItemDescriptor::loadStore(diskStore));
|
|
|
|
} else {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemPtr ItemDatabase::fromJson(Json const& spec) const {
|
|
|
|
return item(ItemDescriptor(spec));
|
|
|
|
}
|
|
|
|
|
|
|
|
Json ItemDatabase::diskStore(ItemConstPtr const& itemPtr) const {
|
|
|
|
if (itemPtr)
|
|
|
|
return itemPtr->descriptor().diskStore();
|
|
|
|
else
|
|
|
|
return Json();
|
|
|
|
}
|
|
|
|
|
|
|
|
Json ItemDatabase::toJson(ItemConstPtr const& itemPtr) const {
|
|
|
|
if (itemPtr)
|
|
|
|
return itemPtr->descriptor().toJson();
|
|
|
|
else
|
|
|
|
return Json();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ItemDatabase::hasItem(String const& itemName) const {
|
|
|
|
return m_items.contains(itemName);
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemType ItemDatabase::itemType(String const& itemName) const {
|
|
|
|
return itemData(itemName).type;
|
|
|
|
}
|
|
|
|
|
|
|
|
String ItemDatabase::itemFriendlyName(String const& itemName) const {
|
|
|
|
return itemData(itemName).friendlyName;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringSet ItemDatabase::itemTags(String const& itemName) const {
|
|
|
|
return itemData(itemName).itemTags;
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemDatabase::ItemConfig ItemDatabase::itemConfig(String const& itemName, Json parameters, Maybe<float> level, Maybe<uint64_t> seed) const {
|
|
|
|
auto const& data = itemData(itemName);
|
|
|
|
|
|
|
|
ItemConfig itemConfig;
|
|
|
|
if (data.assetsConfig)
|
|
|
|
itemConfig.config = Root::singleton().assets()->json(*data.assetsConfig);
|
|
|
|
itemConfig.directory = data.directory;
|
|
|
|
itemConfig.config = jsonMerge(itemConfig.config, data.customConfig);
|
|
|
|
itemConfig.parameters = parameters;
|
|
|
|
|
|
|
|
if (auto builder = itemConfig.config.optString("builder")) {
|
|
|
|
RecursiveMutexLocker locker(m_luaMutex);
|
|
|
|
auto context = m_luaRoot->createContext(*builder);
|
|
|
|
context.setCallbacks("root", LuaBindings::makeRootCallbacks());
|
|
|
|
context.setCallbacks("sb", LuaBindings::makeUtilityCallbacks());
|
|
|
|
luaTie(itemConfig.config, itemConfig.parameters) = context.invokePath<LuaTupleReturn<Json, Json>>(
|
|
|
|
"build", itemConfig.directory, itemConfig.config, itemConfig.parameters, level, seed);
|
|
|
|
}
|
|
|
|
|
|
|
|
return itemConfig;
|
|
|
|
}
|
|
|
|
|
2023-07-23 22:44:02 +10:00
|
|
|
ItemPtr ItemDatabase::itemShared(ItemDescriptor descriptor, Maybe<float> level, Maybe<uint64_t> seed) const {
|
2023-06-20 14:33:09 +10:00
|
|
|
if (!descriptor)
|
|
|
|
return {};
|
|
|
|
|
2023-07-23 22:44:02 +10:00
|
|
|
ItemCacheEntry entry{ descriptor, level, seed };
|
|
|
|
MutexLocker locker(m_cacheMutex);
|
|
|
|
if (ItemPtr* cached = m_itemCache.ptr(entry))
|
|
|
|
return *cached;
|
|
|
|
else {
|
|
|
|
locker.unlock();
|
|
|
|
|
|
|
|
ItemPtr item = tryCreateItem(descriptor, level, seed);
|
|
|
|
get<2>(entry) = item->parameters().optUInt("seed"); // Seed could've been changed by the buildscript
|
|
|
|
|
|
|
|
locker.lock();
|
2024-02-19 16:55:19 +01:00
|
|
|
return m_itemCache.get(entry, [&](ItemCacheEntry const&) -> ItemPtr { return std::move(item); });
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
2023-07-23 22:44:02 +10:00
|
|
|
}
|
2023-06-20 14:33:09 +10:00
|
|
|
|
2024-03-08 18:14:40 -05:00
|
|
|
ItemPtr ItemDatabase::item(ItemDescriptor descriptor, Maybe<float> level, Maybe<uint64_t> seed, bool ignoreInvalid) const {
|
2023-07-23 22:44:02 +10:00
|
|
|
if (!descriptor)
|
|
|
|
return {};
|
|
|
|
else
|
2024-03-08 18:14:40 -05:00
|
|
|
return tryCreateItem(descriptor, level, seed, ignoreInvalid);
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ItemDatabase::hasRecipeToMake(ItemDescriptor const& item) const {
|
|
|
|
auto si = item.singular();
|
|
|
|
for (auto const& recipe : m_recipes)
|
|
|
|
if (recipe.output.singular() == si)
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ItemDatabase::hasRecipeToMake(ItemDescriptor const& item, StringSet const& allowedTypes) const {
|
|
|
|
auto si = item.singular();
|
|
|
|
for (auto const& recipe : m_recipes)
|
|
|
|
if (recipe.output.singular() == si)
|
|
|
|
for (auto allowedType : allowedTypes)
|
|
|
|
if (recipe.groups.contains(allowedType))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::recipesForOutputItem(String itemName) const {
|
|
|
|
HashSet<ItemRecipe> result;
|
|
|
|
for (auto const& recipe : m_recipes)
|
|
|
|
if (recipe.output.name() == itemName)
|
|
|
|
result.add(recipe);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::recipesFromBagContents(List<ItemPtr> const& bag, StringMap<uint64_t> const& availableCurrencies) const {
|
|
|
|
auto normalizedBag = normalizeBag(bag);
|
|
|
|
return recipesFromBagContents(normalizedBag, availableCurrencies);
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::recipesFromBagContents(HashMap<ItemDescriptor, uint64_t> const& bag, StringMap<uint64_t> const& availableCurrencies) const {
|
|
|
|
return recipesFromSubset(bag, availableCurrencies, m_recipes);
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::recipesFromBagContents(List<ItemPtr> const& bag, StringMap<uint64_t> const& availableCurrencies, StringSet const& allowedTypes) const {
|
|
|
|
auto normalizedBag = normalizeBag(bag);
|
|
|
|
return recipesFromBagContents(normalizedBag, availableCurrencies, allowedTypes);
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::recipesFromBagContents(HashMap<ItemDescriptor, uint64_t> const& bag, StringMap<uint64_t> const& availableCurrencies, StringSet const& allowedTypes) const {
|
|
|
|
return recipesFromSubset(bag, availableCurrencies, m_recipes, allowedTypes);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t ItemDatabase::maxCraftableInBag(List<ItemPtr> const& bag, StringMap<uint64_t> const& availableCurrencies, ItemRecipe const& recipe) const {
|
|
|
|
auto normalizedBag = normalizeBag(bag);
|
|
|
|
|
|
|
|
return maxCraftableInBag(normalizedBag, availableCurrencies, recipe);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t ItemDatabase::maxCraftableInBag(HashMap<ItemDescriptor, uint64_t> const& bag, StringMap<uint64_t> const& availableCurrencies, ItemRecipe const& recipe) const {
|
|
|
|
uint64_t res = highest<uint64_t>();
|
|
|
|
|
|
|
|
for (auto const& p : recipe.currencyInputs) {
|
|
|
|
uint64_t available = availableCurrencies.value(p.first, 0);
|
|
|
|
if (available == 0)
|
|
|
|
return 0;
|
|
|
|
else if (p.second > 0)
|
|
|
|
res = min(available / p.second, res);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto const& input : recipe.inputs) {
|
|
|
|
if (!bag.contains(input.singular()))
|
|
|
|
return 0;
|
|
|
|
else if (input.count() > 0)
|
|
|
|
res = min(bag.get(input.singular()) / input.count(), res);
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemRecipe ItemDatabase::getPreciseRecipeForMaterials(String const& group, List<ItemPtr> const& bag, StringMap<uint64_t> const& availableCurrencies) const {
|
|
|
|
// picks the recipe that:
|
|
|
|
// * can be crafted (duh)
|
|
|
|
// * uses all the input material types
|
|
|
|
// * uses the most materials (if recipes exist with the same input materials)
|
|
|
|
|
|
|
|
auto options = recipesFromBagContents(bag, availableCurrencies);
|
|
|
|
ItemRecipe result;
|
|
|
|
int ingredientsCount = 0;
|
|
|
|
for (auto const& recipe : options) {
|
|
|
|
if (!recipe.groups.contains(group))
|
|
|
|
continue;
|
|
|
|
bool usesAllItemTypes = true;
|
|
|
|
for (auto const& item : bag) {
|
|
|
|
bool match = false;
|
|
|
|
for (auto const& input : recipe.inputs)
|
|
|
|
if (item->matches(input, recipe.matchInputParameters))
|
|
|
|
match = true;
|
|
|
|
if (!match)
|
|
|
|
usesAllItemTypes = false;
|
|
|
|
}
|
|
|
|
if (!usesAllItemTypes)
|
|
|
|
continue;
|
|
|
|
int count = 0;
|
|
|
|
for (auto const& input : recipe.inputs)
|
|
|
|
count += input.count();
|
|
|
|
if (count > ingredientsCount)
|
|
|
|
result = recipe;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemRecipe ItemDatabase::parseRecipe(Json const& config) const {
|
|
|
|
ItemRecipe res;
|
|
|
|
try {
|
|
|
|
res.currencyInputs = jsonToMapV<StringMap<uint64_t>>(config.get("currencyInputs", JsonObject()), mem_fn(&Json::toUInt));
|
|
|
|
|
|
|
|
// parse currency items into currency inputs
|
|
|
|
for (auto input : config.getArray("input")) {
|
|
|
|
auto id = ItemDescriptor(input);
|
|
|
|
if (itemType(id.name()) == ItemType::CurrencyItem) {
|
2023-07-23 22:44:02 +10:00
|
|
|
auto currencyItem = as<CurrencyItem>(itemShared(id));
|
2023-06-20 14:33:09 +10:00
|
|
|
res.currencyInputs[currencyItem->currencyType()] += currencyItem->totalValue();
|
|
|
|
} else {
|
|
|
|
res.inputs.push_back(id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res.output = ItemDescriptor(config.get("output"));
|
|
|
|
res.duration = config.getFloat("duration", Root::singleton().assets()->json("/items/defaultParameters.config:defaultCraftDuration").toFloat());
|
|
|
|
res.groups = StringSet::from(jsonToStringList(config.get("groups", JsonArray())));
|
2023-07-23 22:44:02 +10:00
|
|
|
if (auto item = ItemDatabase::itemShared(res.output)) {
|
2023-06-20 14:33:09 +10:00
|
|
|
res.outputRarity = item->rarity();
|
|
|
|
res.guiFilterString = guiFilterString(item);
|
|
|
|
}
|
|
|
|
res.collectables = jsonToMapV<StringMap<String>>(config.get("collectables", JsonObject()), mem_fn(&Json::toString));
|
|
|
|
res.matchInputParameters = config.getBool("matchInputParameters", false);
|
|
|
|
|
|
|
|
} catch (JsonException const& e) {
|
2023-06-27 20:23:44 +10:00
|
|
|
throw RecipeException(strf("Recipe missing required ingredient: {}", outputException(e, false)));
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::allRecipes() const {
|
|
|
|
return m_recipes;
|
|
|
|
}
|
|
|
|
|
|
|
|
HashSet<ItemRecipe> ItemDatabase::allRecipes(StringSet const& types) const {
|
|
|
|
HashSet<ItemRecipe> res;
|
|
|
|
for (auto const& i : m_recipes) {
|
|
|
|
if (i.groups.hasIntersection(types))
|
|
|
|
res.add(i);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemPtr ItemDatabase::applyAugment(ItemPtr const item, AugmentItem* augment) const {
|
|
|
|
if (item) {
|
|
|
|
RecursiveMutexLocker locker(m_luaMutex);
|
|
|
|
LuaBaseComponent script;
|
|
|
|
script.setLuaRoot(m_luaRoot);
|
|
|
|
script.setScripts(augment->augmentScripts());
|
|
|
|
script.addCallbacks("item", LuaBindings::makeItemCallbacks(augment));
|
|
|
|
script.addCallbacks("config", LuaBindings::makeConfigCallbacks(bind(&Item::instanceValue, augment, _1, _2)));
|
|
|
|
script.init();
|
|
|
|
auto luaResult = script.invoke<LuaTupleReturn<Json, Maybe<uint64_t>>>("apply", item->descriptor().toJson());
|
|
|
|
script.uninit();
|
|
|
|
locker.unlock();
|
|
|
|
|
|
|
|
if (luaResult) {
|
|
|
|
if (!get<0>(*luaResult).isNull()) {
|
|
|
|
augment->take(get<1>(*luaResult).value(1));
|
|
|
|
return ItemDatabase::item(ItemDescriptor(get<0>(*luaResult)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ItemDatabase::ageItem(ItemPtr& item, double aging) const {
|
|
|
|
if (!item)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
auto const& itemData = ItemDatabase::itemData(item->name());
|
|
|
|
if (itemData.agingScripts.empty())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ItemDescriptor original = item->descriptor();
|
|
|
|
|
|
|
|
RecursiveMutexLocker locker(m_luaMutex);
|
|
|
|
LuaBaseComponent script;
|
|
|
|
script.setLuaRoot(m_luaRoot);
|
|
|
|
script.setScripts(itemData.agingScripts);
|
|
|
|
script.init();
|
|
|
|
auto aged = script.invoke<Json>("ageItem", original.toJson(), aging).apply(construct<ItemDescriptor>());
|
|
|
|
script.uninit();
|
|
|
|
locker.unlock();
|
|
|
|
|
|
|
|
if (aged && *aged != original) {
|
|
|
|
item = ItemDatabase::item(*aged);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
List<String> ItemDatabase::allItems() const {
|
|
|
|
return m_items.keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemPtr ItemDatabase::createItem(ItemType type, ItemConfig const& config) {
|
|
|
|
if (type == ItemType::Generic) {
|
|
|
|
return make_shared<GenericItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::LiquidItem) {
|
|
|
|
return make_shared<LiquidItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::MaterialItem) {
|
|
|
|
return make_shared<MaterialItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::ObjectItem) {
|
|
|
|
return make_shared<ObjectItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::CurrencyItem) {
|
|
|
|
return make_shared<CurrencyItem>(config.config, config.directory);
|
|
|
|
} else if (type == ItemType::MiningTool) {
|
|
|
|
return make_shared<MiningTool>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::Flashlight) {
|
|
|
|
return make_shared<Flashlight>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::WireTool) {
|
|
|
|
return make_shared<WireTool>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::BeamMiningTool) {
|
|
|
|
return make_shared<BeamMiningTool>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::PaintingBeamTool) {
|
|
|
|
return make_shared<PaintingBeamTool>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::TillingTool) {
|
|
|
|
return make_shared<TillingTool>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::HarvestingTool) {
|
|
|
|
return make_shared<HarvestingTool>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::HeadArmor) {
|
|
|
|
return make_shared<HeadArmor>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::ChestArmor) {
|
|
|
|
return make_shared<ChestArmor>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::LegsArmor) {
|
|
|
|
return make_shared<LegsArmor>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::BackArmor) {
|
|
|
|
return make_shared<BackArmor>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::Consumable) {
|
|
|
|
return make_shared<ConsumableItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::Blueprint) {
|
|
|
|
return make_shared<BlueprintItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::Codex) {
|
|
|
|
return make_shared<CodexItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::InspectionTool) {
|
|
|
|
return make_shared<InspectionTool>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::InstrumentItem) {
|
|
|
|
return make_shared<InstrumentItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::ThrownItem) {
|
|
|
|
return make_shared<ThrownItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::UnlockItem) {
|
|
|
|
return make_shared<UnlockItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::ActiveItem) {
|
|
|
|
return make_shared<ActiveItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else if (type == ItemType::AugmentItem) {
|
|
|
|
return make_shared<AugmentItem>(config.config, config.directory, config.parameters);
|
|
|
|
} else {
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ItemException(strf("Unknown item type {}", (int)type));
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-08 18:14:40 -05:00
|
|
|
ItemPtr ItemDatabase::tryCreateItem(ItemDescriptor const& descriptor, Maybe<float> level, Maybe<uint64_t> seed, bool ignoreInvalid) const {
|
2023-07-23 22:44:02 +10:00
|
|
|
ItemPtr result;
|
2024-03-08 18:14:40 -05:00
|
|
|
String name = descriptor.name();
|
|
|
|
Json parameters = descriptor.parameters();
|
|
|
|
|
2023-07-23 22:44:02 +10:00
|
|
|
try {
|
2024-03-08 18:14:40 -05:00
|
|
|
if ((name == "perfectlygenericitem") && parameters.contains("genericItemStorage")) {
|
|
|
|
Json storage = parameters.get("genericItemStorage");
|
|
|
|
name = storage.getString("name");
|
|
|
|
parameters = storage.get("parameters");
|
|
|
|
}
|
|
|
|
result = createItem(m_items.get(name).type, itemConfig(name, parameters, level, seed));
|
2023-07-23 22:44:02 +10:00
|
|
|
}
|
|
|
|
catch (std::exception const& e) {
|
2024-03-08 18:14:40 -05:00
|
|
|
if (descriptor.name() == "perfectlygenericitem") {
|
|
|
|
Logger::error("Could not re-instantiate item '{}'. {}", descriptor, outputException(e, false));
|
|
|
|
result = createItem(m_items.get("perfectlygenericitem").type, itemConfig("perfectlygenericitem", descriptor.parameters(), level, seed));
|
|
|
|
} else if (!ignoreInvalid) {
|
|
|
|
Logger::error("Could not instantiate item '{}'. {}", descriptor, outputException(e, false));
|
|
|
|
result = createItem(m_items.get("perfectlygenericitem").type, itemConfig("perfectlygenericitem", JsonObject({
|
|
|
|
{"genericItemStorage", descriptor.toJson()},
|
|
|
|
{"shortdescription", descriptor.name()},
|
|
|
|
{"description", "Reinstall the parent mod to return this item to normal\n^red;(to retain data, do not place as object)"}
|
|
|
|
}), {}, {}));
|
|
|
|
} else
|
|
|
|
throw e;
|
2023-07-23 22:44:02 +10:00
|
|
|
}
|
|
|
|
result->setCount(descriptor.count());
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-06-20 14:33:09 +10:00
|
|
|
ItemDatabase::ItemData const& ItemDatabase::itemData(String const& name) const {
|
|
|
|
if (auto p = m_items.ptr(name))
|
|
|
|
return *p;
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ItemException::format("No such item '{}'", name);
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
ItemRecipe ItemDatabase::makeRecipe(List<ItemDescriptor> inputs, ItemDescriptor output, float duration, StringSet groups) const {
|
|
|
|
ItemRecipe res;
|
2024-02-19 16:55:19 +01:00
|
|
|
res.inputs = std::move(inputs);
|
|
|
|
res.output = std::move(output);
|
2023-06-20 14:33:09 +10:00
|
|
|
res.duration = duration;
|
2024-02-19 16:55:19 +01:00
|
|
|
res.groups = std::move(groups);
|
2023-07-23 22:44:02 +10:00
|
|
|
if (auto item = ItemDatabase::itemShared(res.output)) {
|
2023-06-20 14:33:09 +10:00
|
|
|
res.outputRarity = item->rarity();
|
|
|
|
res.guiFilterString = guiFilterString(item);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ItemDatabase::addItemSet(ItemType type, String const& extension) {
|
|
|
|
auto assets = Root::singleton().assets();
|
2024-03-15 21:28:11 +11:00
|
|
|
for (auto& file : assets->scanExtension(extension)) {
|
2023-06-20 14:33:09 +10:00
|
|
|
ItemData data;
|
|
|
|
try {
|
|
|
|
auto config = assets->json(file);
|
|
|
|
data.type = type;
|
|
|
|
data.assetsConfig = file;
|
|
|
|
data.name = config.get("itemName").toString();
|
|
|
|
data.friendlyName = config.getString("shortdescription", {});
|
|
|
|
data.itemTags = config.opt("itemTags").apply(jsonToStringSet).value();
|
|
|
|
data.agingScripts = config.opt("itemAgingScripts").apply(jsonToStringList).value();
|
|
|
|
data.directory = AssetPath::directory(file);
|
|
|
|
|
|
|
|
data.agingScripts = data.agingScripts.transformed(bind(&AssetPath::relativeTo, data.directory, _1));
|
|
|
|
} catch (std::exception const& e) {
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ItemException(strf("Could not load item asset {}", file), e);
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
if (m_items.contains(data.name))
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ItemException(strf("Duplicate item name '{}' found", data.name));
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
m_items[data.name] = data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ItemDatabase::addObjectDropItem(String const& objectPath, Json const& objectConfig) {
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
|
|
|
|
ItemData data;
|
|
|
|
data.type = ItemType::ObjectItem;
|
|
|
|
data.name = objectConfig.get("objectName").toString();
|
|
|
|
data.friendlyName = objectConfig.getString("shortdescription", {});
|
|
|
|
data.itemTags = objectConfig.opt("itemTags").apply(jsonToStringSet).value();
|
|
|
|
data.agingScripts = objectConfig.opt("itemAgingScripts").apply(jsonToStringList).value();
|
|
|
|
data.directory = AssetPath::directory(objectPath);
|
|
|
|
JsonObject customConfig = objectConfig.toObject();
|
|
|
|
if (!customConfig.contains("inventoryIcon")) {
|
|
|
|
customConfig["inventoryIcon"] = assets->json("/objects/defaultParameters.config:missingIcon");
|
2023-06-27 20:23:44 +10:00
|
|
|
Logger::warn(strf("Missing inventoryIcon for {}, using default", data.name).c_str());
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
customConfig["itemName"] = data.name;
|
|
|
|
if (!customConfig.contains("tooltipKind"))
|
|
|
|
customConfig["tooltipKind"] = "object";
|
|
|
|
|
|
|
|
if (!customConfig.contains("printable"))
|
|
|
|
customConfig["printable"] = customConfig.contains("price");
|
|
|
|
|
|
|
|
// Don't inherit object scripts. this is kind of a crappy solution to prevent
|
|
|
|
// ObjectItems (which are firable and therefore scripted) from trying to
|
|
|
|
// execute scripts intended for objects
|
|
|
|
customConfig.remove("scripts");
|
|
|
|
|
2024-02-19 16:55:19 +01:00
|
|
|
data.customConfig = std::move(customConfig);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
if (m_items.contains(data.name))
|
2023-06-27 20:23:44 +10:00
|
|
|
throw ItemException(strf("Object drop '{}' shares name with existing item", data.name));
|
2023-06-20 14:33:09 +10:00
|
|
|
|
2024-02-19 16:55:19 +01:00
|
|
|
m_items[data.name] = std::move(data);
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
void ItemDatabase::scanItems() {
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
|
|
|
|
List<std::pair<ItemType, String>> itemSets;
|
|
|
|
auto scanItemType = [&itemSets, assets](ItemType type, String const& extension) {
|
|
|
|
itemSets.append(make_pair(type, extension));
|
|
|
|
assets->queueJsons(assets->scanExtension(extension));
|
|
|
|
};
|
|
|
|
|
|
|
|
scanItemType(ItemType::Generic, "item");
|
|
|
|
scanItemType(ItemType::LiquidItem, "liqitem");
|
|
|
|
scanItemType(ItemType::MaterialItem, "matitem");
|
|
|
|
scanItemType(ItemType::MiningTool, "miningtool");
|
|
|
|
scanItemType(ItemType::Flashlight, "flashlight");
|
|
|
|
scanItemType(ItemType::WireTool, "wiretool");
|
|
|
|
scanItemType(ItemType::BeamMiningTool, "beamaxe");
|
|
|
|
scanItemType(ItemType::TillingTool, "tillingtool");
|
|
|
|
scanItemType(ItemType::PaintingBeamTool, "painttool");
|
|
|
|
scanItemType(ItemType::HarvestingTool, "harvestingtool");
|
|
|
|
scanItemType(ItemType::HeadArmor, "head");
|
|
|
|
scanItemType(ItemType::ChestArmor, "chest");
|
|
|
|
scanItemType(ItemType::LegsArmor, "legs");
|
|
|
|
scanItemType(ItemType::BackArmor, "back");
|
|
|
|
scanItemType(ItemType::CurrencyItem, "currency");
|
|
|
|
scanItemType(ItemType::Consumable, "consumable");
|
|
|
|
scanItemType(ItemType::Blueprint, "blueprint");
|
|
|
|
scanItemType(ItemType::InspectionTool, "inspectiontool");
|
|
|
|
scanItemType(ItemType::InstrumentItem, "instrument");
|
|
|
|
scanItemType(ItemType::ThrownItem, "thrownitem");
|
|
|
|
scanItemType(ItemType::UnlockItem, "unlock");
|
|
|
|
scanItemType(ItemType::ActiveItem, "activeitem");
|
|
|
|
scanItemType(ItemType::AugmentItem, "augment");
|
|
|
|
|
|
|
|
for (auto const& itemset : itemSets)
|
|
|
|
addItemSet(itemset.first, itemset.second);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ItemDatabase::addObjectItems() {
|
|
|
|
auto objectDatabase = Root::singleton().objectDatabase();
|
|
|
|
|
|
|
|
for (auto const& objectName : objectDatabase->allObjects()) {
|
|
|
|
auto objectConfig = objectDatabase->getConfig(objectName);
|
|
|
|
|
|
|
|
if (objectConfig->hasObjectItem)
|
|
|
|
addObjectDropItem(objectConfig->path, objectConfig->config);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ItemDatabase::scanRecipes() {
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
|
2024-03-15 21:28:11 +11:00
|
|
|
auto& files = assets->scanExtension("recipe");
|
2023-06-20 14:33:09 +10:00
|
|
|
assets->queueJsons(files);
|
2024-03-15 21:28:11 +11:00
|
|
|
for (auto& file : files) {
|
2023-06-20 14:33:09 +10:00
|
|
|
try {
|
|
|
|
m_recipes.add(parseRecipe(assets->json(file)));
|
|
|
|
} catch (std::exception const& e) {
|
2023-06-27 20:23:44 +10:00
|
|
|
Logger::error("Could not load recipe {}: {}", file, outputException(e, false));
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ItemDatabase::addBlueprints() {
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
|
|
|
|
for (auto const& recipe : m_recipes) {
|
|
|
|
auto baseDesc = recipe.output;
|
2023-07-23 22:44:02 +10:00
|
|
|
auto baseItem = itemShared(baseDesc);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
2023-06-27 20:23:44 +10:00
|
|
|
String blueprintName = strf("{}-recipe", baseItem->name());
|
2023-06-20 14:33:09 +10:00
|
|
|
if (m_items.contains(blueprintName))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
try {
|
|
|
|
ItemData blueprintData;
|
|
|
|
|
|
|
|
blueprintData.type = ItemType::Blueprint;
|
|
|
|
JsonObject configInfo;
|
|
|
|
configInfo["recipe"] = baseDesc.singular().toJson();
|
|
|
|
|
|
|
|
String description = assets->json("/blueprint.config:description").toString();
|
|
|
|
description = description.replace("<item>", baseItem->friendlyName());
|
|
|
|
configInfo["description"] = Json(description);
|
|
|
|
|
|
|
|
String shortDesc = assets->json("/blueprint.config:shortdescription").toString();
|
|
|
|
shortDesc = shortDesc.replace("<item>", baseItem->friendlyName());
|
|
|
|
configInfo["shortdescription"] = Json(shortDesc);
|
|
|
|
|
|
|
|
configInfo["category"] = assets->json("/blueprint.config:category").toString();
|
|
|
|
|
|
|
|
blueprintData.name = blueprintName;
|
|
|
|
blueprintData.friendlyName = shortDesc;
|
|
|
|
configInfo["itemName"] = blueprintData.name;
|
|
|
|
|
|
|
|
if (baseItem->instanceValue("inventoryIcon", false))
|
|
|
|
configInfo["inventoryIcon"] = baseItem->instanceValue("inventoryIcon");
|
|
|
|
|
|
|
|
configInfo["rarity"] = RarityNames.getRight(baseItem->rarity());
|
|
|
|
|
|
|
|
configInfo["price"] = baseItem->price();
|
|
|
|
|
2024-02-19 16:55:19 +01:00
|
|
|
blueprintData.customConfig = std::move(configInfo);
|
2023-06-20 14:33:09 +10:00
|
|
|
blueprintData.directory = itemData(baseDesc.name()).directory;
|
|
|
|
|
|
|
|
m_items[blueprintData.name] = blueprintData;
|
|
|
|
} catch (std::exception const& e) {
|
2023-06-27 20:23:44 +10:00
|
|
|
Logger::error("Could not create blueprint item from recipe: {}", outputException(e, false));
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ItemDatabase::addCodexes() {
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
auto codexConfig = assets->json("/codex.config");
|
|
|
|
|
|
|
|
auto codexDatabase = Root::singleton().codexDatabase();
|
|
|
|
for (auto const& codexPair : codexDatabase->codexes()) {
|
2023-06-27 20:23:44 +10:00
|
|
|
String codexItemName = strf("{}-codex", codexPair.second->id());
|
2023-06-20 14:33:09 +10:00
|
|
|
if (m_items.contains(codexItemName)) {
|
2023-06-27 20:23:44 +10:00
|
|
|
Logger::warn("Couldn't create codex item {} because an item with that name is already defined", codexItemName);
|
2023-06-20 14:33:09 +10:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
ItemData codexItemData;
|
|
|
|
|
|
|
|
codexItemData.type = ItemType::Codex;
|
|
|
|
codexItemData.name = codexItemName;
|
|
|
|
codexItemData.friendlyName = codexPair.second->title();
|
|
|
|
codexItemData.directory = codexPair.second->directory();
|
|
|
|
|
|
|
|
auto customConfig = jsonMerge(codexConfig.get("defaultItemConfig"), codexPair.second->itemConfig()).toObject();
|
|
|
|
customConfig["itemName"] = codexItemName;
|
|
|
|
customConfig["codexId"] = codexPair.second->id();
|
|
|
|
customConfig["shortdescription"] = codexPair.second->title();
|
|
|
|
customConfig["description"] = codexPair.second->description();
|
|
|
|
customConfig["codexIcon"] = codexPair.second->icon();
|
|
|
|
codexItemData.customConfig = customConfig;
|
|
|
|
|
|
|
|
m_items[codexItemName] = codexItemData;
|
|
|
|
} catch (std::exception const& e) {
|
2023-06-27 20:23:44 +10:00
|
|
|
Logger::error("Could not create item for codex {}: {}", codexPair.second->id(), outputException(e, false));
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|