/* * The ManaPlus Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * Copyright (C) 2011-2013 The ManaPlus Developers * * This file is part of The ManaPlus Client. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ #include "resources/itemdb.h" #include "client.h" #include "configuration.h" #include "logger.h" #include "resources/iteminfo.h" #include "resources/resourcemanager.h" #include "utils/dtor.h" #include "utils/gettext.h" #include "debug.h" namespace { ItemDB::ItemInfos mItemInfos; ItemDB::NamedItemInfos mNamedItemInfos; ItemInfo *mUnknown; bool mLoaded = false; StringVect mTagNames; std::map<std::string, int> mTags; } // Forward declarations static void loadSpriteRef(ItemInfo *const itemInfo, const XmlNodePtr node); static void loadSoundRef(ItemInfo *const itemInfo, const XmlNodePtr node); static void loadFloorSprite(SpriteDisplay *const display, const XmlNodePtr node); static void loadReplaceSprite(ItemInfo *const itemInfo, const XmlNodePtr replaceNode); static void loadOrderSprite(ItemInfo *const itemInfo, const XmlNodePtr node, const bool drawAfter); static int parseSpriteName(const std::string &name); static int parseDirectionName(const std::string &name); static const char *const fields[][2] = { // TRANSLATORS: item info label { "attack", N_("Attack %+d") }, // TRANSLATORS: item info label { "defense", N_("Defense %+d") }, // TRANSLATORS: item info label { "hp", N_("HP %+d") }, // TRANSLATORS: item info label { "mp", N_("MP %+d") } }; static std::vector<ItemDB::Stat> extraStats; void ItemDB::setStatsList(const std::vector<ItemDB::Stat> &stats) { extraStats = stats; } static ItemType itemTypeFromString(const std::string &name) { if (name == "generic" || name == "other") { return ITEM_UNUSABLE; } else if (name == "usable") { return ITEM_USABLE; } else if (name == "equip-1hand") { return ITEM_EQUIPMENT_ONE_HAND_WEAPON; } else if (name == "equip-2hand") { return ITEM_EQUIPMENT_TWO_HANDS_WEAPON; } else if (name == "equip-torso") { return ITEM_EQUIPMENT_TORSO; } else if (name == "equip-arms") { return ITEM_EQUIPMENT_ARMS; } else if (name == "equip-head") { return ITEM_EQUIPMENT_HEAD; } else if (name == "equip-legs") { return ITEM_EQUIPMENT_LEGS; } else if (name == "equip-shield") { return ITEM_EQUIPMENT_SHIELD; } else if (name == "equip-ring") { return ITEM_EQUIPMENT_RING; } else if (name == "equip-charm") { return ITEM_EQUIPMENT_CHARM; } else if (name == "equip-necklace" || name == "equip-neck") { return ITEM_EQUIPMENT_NECKLACE; } else if (name == "equip-feet") { return ITEM_EQUIPMENT_FEET; } else if (name == "equip-ammo") { return ITEM_EQUIPMENT_AMMO; } else if (name == "racesprite") { return ITEM_SPRITE_RACE; } else if (name == "hairsprite") { return ITEM_SPRITE_HAIR; } else { logger->log("Unknown item type: " + name); return ITEM_UNUSABLE; } } void ItemDB::load() { if (mLoaded) unload(); int tagNum = 0; logger->log1("Initializing item database..."); mTags.clear(); mTagNames.clear(); mTagNames.push_back("All"); mTagNames.push_back("Usable"); mTagNames.push_back("Unusable"); mTagNames.push_back("Equipment"); mTags["All"] = tagNum ++; mTags["Usable"] = tagNum ++; mTags["Unusable"] = tagNum ++; mTags["Equipment"] = tagNum ++; mUnknown = new ItemInfo; // TRANSLATORS: item name mUnknown->setName(_("Unknown item")); mUnknown->setDisplay(SpriteDisplay()); std::string errFile = paths.getStringValue("spriteErrorFile"); mUnknown->setSprite(errFile, GENDER_MALE, 0); mUnknown->setSprite(errFile, GENDER_FEMALE, 0); mUnknown->setSprite(errFile, GENDER_OTHER, 0); mUnknown->addTag(mTags["All"]); XML::Document doc("items.xml"); const XmlNodePtr rootNode = doc.rootNode(); if (!rootNode || !xmlNameEqual(rootNode, "items")) { logger->log("ItemDB: Error while loading items.xml!"); mLoaded = true; return; } for_each_xml_child_node(node, rootNode) { if (!xmlNameEqual(node, "item")) continue; const int id = XML::getProperty(node, "id", 0); if (id == 0) { logger->log1("ItemDB: Invalid or missing item ID in items.xml!"); continue; } else if (mItemInfos.find(id) != mItemInfos.end()) { logger->log("ItemDB: Redefinition of item ID %d", id); } const std::string typeStr = XML::getProperty(node, "type", "other"); const int weight = XML::getProperty(node, "weight", 0); const int view = XML::getProperty(node, "view", 0); std::string name = XML::langProperty(node, "name", ""); std::string image = XML::getProperty(node, "image", ""); std::string floor = XML::getProperty(node, "floor", ""); std::string description = XML::langProperty(node, "description", ""); std::string attackAction = XML::getProperty(node, "attack-action", ""); std::string drawBefore = XML::getProperty(node, "drawBefore", ""); std::string drawAfter = XML::getProperty(node, "drawAfter", ""); const int pet = XML::getProperty(node, "pet", 0); const int maxFloorOffset = XML::getIntProperty( node, "maxFloorOffset", 32, 0, 32); std::string colors; if (serverVersion >= 1) { colors = XML::getProperty(node, "colors", ""); // check for empty hair palete if (colors.empty() && id <= -1 && id > -100) colors = "hair"; } else { if (id <= -1 && id > -100) colors = "hair"; else colors.clear(); } std::string tags[3]; tags[0] = XML::getProperty(node, "tag", XML::getProperty(node, "tag1", "")); tags[1] = XML::getProperty(node, "tag2", ""); tags[2] = XML::getProperty(node, "tag3", ""); const int drawPriority = XML::getProperty(node, "drawPriority", 0); const int attackRange = XML::getProperty(node, "attack-range", 0); std::string missileParticle = XML::getProperty( node, "missile-particle", ""); const int hitEffectId = XML::getProperty(node, "hit-effect-id", paths.getIntValue("hitEffectId")); const int criticalEffectId = XML::getProperty( node, "critical-hit-effect-id", paths.getIntValue("criticalHitEffectId")); const int missEffectId = XML::getProperty(node, "miss-effect-id", paths.getIntValue("missEffectId")); SpriteDisplay display; display.image = image; if (floor != "") display.floor = floor; else display.floor = image; ItemInfo *const itemInfo = new ItemInfo; itemInfo->setId(id); // TRANSLATORS: item info name itemInfo->setName(name.empty() ? _("unnamed") : name); itemInfo->setDescription(description); itemInfo->setType(itemTypeFromString(typeStr)); itemInfo->addTag(mTags["All"]); itemInfo->setPet(pet); itemInfo->setProtected(XML::getBoolProperty( node, "sellProtected", false)); switch (itemInfo->getType()) { case ITEM_USABLE: itemInfo->addTag(mTags["Usable"]); break; case ITEM_UNUSABLE: itemInfo->addTag(mTags["Unusable"]); break; default: case ITEM_EQUIPMENT_ONE_HAND_WEAPON: case ITEM_EQUIPMENT_TWO_HANDS_WEAPON: case ITEM_EQUIPMENT_TORSO: case ITEM_EQUIPMENT_ARMS: case ITEM_EQUIPMENT_HEAD: case ITEM_EQUIPMENT_LEGS: case ITEM_EQUIPMENT_SHIELD: case ITEM_EQUIPMENT_RING: case ITEM_EQUIPMENT_NECKLACE: case ITEM_EQUIPMENT_FEET: case ITEM_EQUIPMENT_AMMO: case ITEM_EQUIPMENT_CHARM: case ITEM_SPRITE_RACE: case ITEM_SPRITE_HAIR: itemInfo->addTag(mTags["Equipment"]); break; } for (int f = 0; f < 3; f++) { if (tags[f] != "") { if (mTags.find(tags[f]) == mTags.end()) { mTagNames.push_back(tags[f]); mTags[tags[f]] = tagNum ++; } itemInfo->addTag(mTags[tags[f]]); } } itemInfo->setView(view); itemInfo->setWeight(weight); itemInfo->setAttackAction(attackAction); itemInfo->setAttackRange(attackRange); itemInfo->setMissileParticleFile(missileParticle); itemInfo->setHitEffectId(hitEffectId); itemInfo->setCriticalHitEffectId(criticalEffectId); itemInfo->setMissEffectId(missEffectId); itemInfo->setDrawBefore(-1, parseSpriteName(drawBefore)); itemInfo->setDrawAfter(-1, parseSpriteName(drawAfter)); itemInfo->setDrawPriority(-1, drawPriority); itemInfo->setColorsList(colors); itemInfo->setMaxFloorOffset(maxFloorOffset); itemInfo->setPickupCursor(XML::getProperty( node, "pickupCursor", "pickup")); std::string effect; for (size_t i = 0; i < sizeof(fields) / sizeof(fields[0]); ++ i) { const int value = XML::getProperty(node, fields[i][0], 0); if (!value) continue; if (!effect.empty()) effect.append(" / "); effect.append(strprintf(gettext(fields[i][1]), value)); } FOR_EACH (std::vector<Stat>::const_iterator, it, extraStats) { const int value = XML::getProperty(node, it->tag.c_str(), 0); if (!value) continue; if (!effect.empty()) effect.append(" / "); effect.append(strprintf(it->format.c_str(), value)); } std::string temp = XML::langProperty(node, "effect", ""); if (!effect.empty() && !temp.empty()) effect.append(" / "); effect.append(temp); itemInfo->setEffect(effect); for_each_xml_child_node(itemChild, node) { if (xmlNameEqual(itemChild, "sprite")) { std::string attackParticle = XML::getProperty( itemChild, "particle-effect", ""); itemInfo->setParticleEffect(attackParticle); loadSpriteRef(itemInfo, itemChild); } else if (xmlNameEqual(itemChild, "sound")) { loadSoundRef(itemInfo, itemChild); } else if (xmlNameEqual(itemChild, "floor")) { loadFloorSprite(&display, itemChild); } else if (xmlNameEqual(itemChild, "replace")) { loadReplaceSprite(itemInfo, itemChild); } else if (xmlNameEqual(itemChild, "drawAfter")) { loadOrderSprite(itemInfo, itemChild, true); } else if (xmlNameEqual(itemChild, "drawBefore")) { loadOrderSprite(itemInfo, itemChild, false); } } /* logger->log("start dump item: %d", id); if (itemInfo->isRemoveSprites()) { for (int f = 0; f < 10; f ++) { logger->log("dir: %d", f); SpriteToItemMap *const spriteToItems = itemInfo->getSpriteToItemReplaceMap(f); if (!spriteToItems) { logger->log("null"); continue; } for (SpriteToItemMapCIter itr = spriteToItems->begin(), itr_end = spriteToItems->end(); itr != itr_end; ++ itr) { const int remSprite = itr->first; const std::map<int, int> &itemReplacer = itr->second; logger->log("sprite: %d", remSprite); for (std::map<int, int>::const_iterator repIt = itemReplacer.begin(), repIt_end = itemReplacer.end(); repIt != repIt_end; ++ repIt) { logger->log("from %d to %d", repIt->first, repIt->second); } } } } logger->log("--------------------------------"); logger->log("end dump item"); */ itemInfo->setDisplay(display); mItemInfos[id] = itemInfo; if (!name.empty()) { temp = normalize(name); const NamedItemInfos::const_iterator itr = mNamedItemInfos.find(temp); if (itr == mNamedItemInfos.end()) { mNamedItemInfos[temp] = itemInfo; } else { logger->log("ItemDB: Duplicate name of item found item %d", id); } } if (!attackAction.empty()) { if (attackRange == 0) { logger->log("ItemDB: Missing attack range from weapon %i!", id); } } #define CHECK_PARAM(param, error_value) \ if (param == error_value) \ logger->log("ItemDB: Missing " #param " attribute for item %i!", \ id) if (id >= 0 && typeStr != "other") { CHECK_PARAM(name, ""); CHECK_PARAM(description, ""); CHECK_PARAM(image, ""); } #undef CHECK_PARAM } mLoaded = true; } const StringVect &ItemDB::getTags() { return mTagNames; } int ItemDB::getTagId(const std::string &tagName) { return mTags[tagName]; } void ItemDB::unload() { logger->log1("Unloading item database..."); delete mUnknown; mUnknown = nullptr; delete_all(mItemInfos); mItemInfos.clear(); mNamedItemInfos.clear(); mTags.clear(); mTagNames.clear(); mLoaded = false; } bool ItemDB::exists(const int id) { if (!mLoaded) return false; const ItemInfos::const_iterator i = mItemInfos.find(id); return i != mItemInfos.end(); } const ItemInfo &ItemDB::get(const int id) { if (!mLoaded) load(); const ItemInfos::const_iterator i = mItemInfos.find(id); if (i == mItemInfos.end()) { logger->log("ItemDB: Warning, unknown item ID# %d", id); return *mUnknown; } return *(i->second); } const ItemInfo &ItemDB::get(const std::string &name) { if (!mLoaded) load(); const NamedItemInfos::const_iterator i = mNamedItemInfos.find( normalize(name)); if (i == mNamedItemInfos.end()) { if (!name.empty()) { logger->log("ItemDB: Warning, unknown item name \"%s\"", name.c_str()); } return *mUnknown; } return *(i->second); } const std::map<int, ItemInfo*> &ItemDB::getItemInfos() { return mItemInfos; } int parseSpriteName(const std::string &name) { int id = -1; if (name == "shoes" || name == "boot" || name == "boots") { id = 1; } else if (name == "bottomclothes" || name == "bottom" || name == "pants") { id = 2; } else if (name == "topclothes" || name == "top" || name == "torso" || name == "body") { id = 3; } else if (name == "misc1") { id = 4; } else if (name == "misc2" || name == "scarf" || name == "scarfs") { id = 5; } else if (name == "hair") { id = 6; } else if (name == "hat" || name == "hats") { id = 7; } else if (name == "wings") { id = 8; } else if (name == "glove" || name == "gloves") { id = 9; } else if (name == "weapon" || name == "weapons") { id = 10; } else if (name == "shield" || name == "shields") { id = 11; } else if (name == "amulet" || name == "amulets") { id = 12; } else if (name == "ring" || name == "rings") { id = 13; } return id; } int parseDirectionName(const std::string &name) { int id = -1; if (name == "down") { if (serverVersion > 0) id = DIRECTION_DOWN; else id = -2; } else if (name == "downleft" || name == "leftdown") { id = DIRECTION_DOWNLEFT; } else if (name == "left") { id = DIRECTION_LEFT; } else if (name == "upleft" || name == "leftup") { id = DIRECTION_UPLEFT; } else if (name == "up") { if (serverVersion > 0) id = DIRECTION_UP; else id = -3; } else if (name == "upright" || name == "rightup") { id = DIRECTION_UPRIGHT; } else if (name == "right") { id = DIRECTION_RIGHT; } else if (name == "downright" || name == "rightdown") { id = DIRECTION_DOWNRIGHT; } else if (name == "downall") { id = -2; } else if (name == "upall") { id = -3; } // hack for died action. else if (name == "died") { id = 9; } return id; } void loadSpriteRef(ItemInfo *const itemInfo, const XmlNodePtr node) { const std::string gender = XML::getProperty(node, "gender", "unisex"); const std::string filename = reinterpret_cast<const char*>( node->xmlChildrenNode->content); const int race = XML::getProperty(node, "race", 0); if (gender == "male" || gender == "unisex") itemInfo->setSprite(filename, GENDER_MALE, race); if (gender == "female" || gender == "unisex") itemInfo->setSprite(filename, GENDER_FEMALE, race); if (gender == "other" || gender == "unisex") itemInfo->setSprite(filename, GENDER_OTHER, race); } void loadSoundRef(ItemInfo *const itemInfo, const XmlNodePtr node) { const std::string event = XML::getProperty(node, "event", ""); const std::string filename = reinterpret_cast<const char*>( node->xmlChildrenNode->content); const int delay = XML::getProperty(node, "delay", 0); if (event == "hit") { itemInfo->addSound(SOUND_EVENT_HIT, filename, delay); } else if (event == "strike" || event == "miss") { itemInfo->addSound(SOUND_EVENT_MISS, filename, delay); } else { logger->log("ItemDB: Ignoring unknown sound event '%s'", event.c_str()); } } void loadFloorSprite(SpriteDisplay *const display, const XmlNodePtr floorNode) { for_each_xml_child_node(spriteNode, floorNode) { if (xmlNameEqual(spriteNode, "sprite")) { SpriteReference *const currentSprite = new SpriteReference; currentSprite->sprite = reinterpret_cast<const char*>( spriteNode->xmlChildrenNode->content); currentSprite->variant = XML::getProperty(spriteNode, "variant", 0); display->sprites.push_back(currentSprite); } else if (xmlNameEqual(spriteNode, "particlefx")) { display->particles.push_back(reinterpret_cast<const char*>( spriteNode->xmlChildrenNode->content)); } } } void loadReplaceSprite(ItemInfo *const itemInfo, const XmlNodePtr replaceNode) { const std::string removeSprite = XML::getProperty( replaceNode, "sprite", ""); const int direction = parseDirectionName(XML::getProperty( replaceNode, "direction", "all")); itemInfo->setRemoveSprites(); switch (direction) { case -1: { for (int f = 0; f < 10; f ++) { std::map<int, int> *const mapList = itemInfo->addReplaceSprite( parseSpriteName(removeSprite), f); if (!mapList) continue; for_each_xml_child_node(itemNode, replaceNode) { if (xmlNameEqual(itemNode, "item")) { const int from = XML::getProperty(itemNode, "from", 0); const int to = XML::getProperty(itemNode, "to", 1); (*mapList)[from] = to; } } } break; } case -2: { itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_DOWN); itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_DOWNLEFT); itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_DOWNRIGHT); for_each_xml_child_node(itemNode, replaceNode) { if (xmlNameEqual(itemNode, "item")) { const int from = XML::getProperty(itemNode, "from", 0); const int to = XML::getProperty(itemNode, "to", 1); std::map<int, int> *mapList = itemInfo->addReplaceSprite( parseSpriteName(removeSprite), DIRECTION_DOWN); if (mapList) (*mapList)[from] = to; mapList = itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_DOWNLEFT); if (mapList) (*mapList)[from] = to; mapList = itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_DOWNRIGHT); if (mapList) (*mapList)[from] = to; } } break; } case -3: { itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_UP); itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_UPLEFT); itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_UPRIGHT); for_each_xml_child_node(itemNode, replaceNode) { if (xmlNameEqual(itemNode, "item")) { const int from = XML::getProperty(itemNode, "from", 0); const int to = XML::getProperty(itemNode, "to", 1); std::map<int, int> *mapList = itemInfo->addReplaceSprite( parseSpriteName(removeSprite), DIRECTION_UP); if (mapList) (*mapList)[from] = to; mapList = itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_UPLEFT); if (mapList) (*mapList)[from] = to; mapList = itemInfo->addReplaceSprite(parseSpriteName( removeSprite), DIRECTION_UPRIGHT); if (mapList) (*mapList)[from] = to; } } break; } default: { std::map<int, int> *const mapList = itemInfo->addReplaceSprite( parseSpriteName(removeSprite), direction); if (!mapList) return; for_each_xml_child_node(itemNode, replaceNode) { if (xmlNameEqual(itemNode, "item")) { const int from = XML::getProperty(itemNode, "from", 0); const int to = XML::getProperty(itemNode, "to", 1); (*mapList)[from] = to; } } break; } } } void loadOrderSprite(ItemInfo *const itemInfo, const XmlNodePtr node, const bool drawAfter) { const int sprite = parseSpriteName(XML::getProperty(node, "name", "")); const int priority = XML::getProperty(node, "priority", 0); const int direction = parseDirectionName(XML::getProperty( node, "direction", "all")); if (drawAfter) itemInfo->setDrawAfter(direction, sprite); else itemInfo->setDrawBefore(direction, sprite); itemInfo->setDrawPriority(direction, priority); }