/*
* The ManaPlus Client
* Copyright (C) 2004-2009 The Mana World Development Team
* Copyright (C) 2009-2010 The Mana Developers
* Copyright (C) 2011-2020 The ManaPlus Developers
* Copyright (C) 2020-2023 The ManaVerse 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 .
*/
#include "resources/iteminfo.h"
#include "const/resources/spriteaction.h"
#include "const/resources/map/map.h"
#include "enums/resources/spritedirection.h"
#include "resources/itemmenuitem.h"
#include "resources/db/colordb.h"
#include "resources/db/itemdb.h"
#include "utils/checkutils.h"
#include "utils/dtor.h"
#include "utils/stringutils.h"
#include "debug.h"
ItemInfo::ItemInfo() :
mDisplay(),
mMissile(),
mName(),
mNameEn(),
mDescription(),
mEffect(),
mUseButton(),
mUseButton2(),
mType(ItemDbType::UNUSABLE),
mWeight(0),
mView(0),
mId(0),
mIsRemoveSprites(false),
mSpriteToItemReplaceList(),
mAttackAction(SpriteAction::INVALID),
mSkyAttackAction(SpriteAction::INVALID),
mWaterAttackAction(SpriteAction::INVALID),
mRideAttackAction(SpriteAction::INVALID),
mAttackRange(0),
mMissileParticle(),
mAnimationFiles(),
mSounds(),
mTags(),
mColorsList(nullptr),
mIconColorsList(nullptr),
mInventoryMenu(),
mStorageMenu(),
mCartMenu(),
mColorsListName(),
mIconColorsListName(),
mCardColor(ItemColor_zero),
mHitEffectId(-1),
mCriticalHitEffectId(-1),
mMissEffectId(-1),
maxFloorOffsetX(mapTileSize),
maxFloorOffsetY(mapTileSize),
mPickupCursor(Cursor::CURSOR_POINTER),
mProtected(false)
{
for (int f = 0; f < 10; f ++)
{
mSpriteToItemReplaceMap[f] = nullptr;
mDrawBefore[f] = -1;
mDrawAfter[f] = -1;
mDrawPriority[f] = 0;
}
}
ItemInfo::~ItemInfo()
{
delete_all(mSpriteToItemReplaceList);
mSpriteToItemReplaceList.clear();
for (int f = 0; f < 10; f ++)
mSpriteToItemReplaceMap[f] = nullptr;
}
const std::string &ItemInfo::getSprite(const GenderT gender,
const BeingTypeId race) const
{
if (mView != 0)
{
// Forward the request to the item defining how to view this item
return ItemDB::get(mView).getSprite(gender, race);
}
static const std::string empty;
std::map::const_iterator i =
mAnimationFiles.find(CAST_S32(gender) +
toInt(race, int) * 4);
if (i != mAnimationFiles.end())
return i->second;
i = mAnimationFiles.find(CAST_S32(gender));
if (i != mAnimationFiles.end())
return i->second;
return empty;
}
void ItemInfo::setAttackAction(const std::string &attackAction)
{
if (attackAction.empty())
mAttackAction = SpriteAction::ATTACK; // (Equal to unarmed animation)
else
mAttackAction = attackAction;
}
void ItemInfo::setSkyAttackAction(const std::string &attackAction)
{
if (attackAction.empty())
mSkyAttackAction = SpriteAction::ATTACKSKY;
else
mSkyAttackAction = attackAction;
}
void ItemInfo::setWaterAttackAction(const std::string &attackAction)
{
if (attackAction.empty())
mWaterAttackAction = SpriteAction::ATTACKWATER;
else
mWaterAttackAction = attackAction;
}
void ItemInfo::setRideAttackAction(const std::string &attackAction)
{
if (attackAction.empty())
mRideAttackAction = SpriteAction::ATTACKRIDE;
else
mRideAttackAction = attackAction;
}
void ItemInfo::addSound(const ItemSoundEvent::Type event,
const std::string &filename, const int delay)
{
mSounds[event].push_back(SoundInfo(
filename,
delay));
}
const SoundInfo &ItemInfo::getSound(const ItemSoundEvent::Type event) const
{
static const SoundInfo empty("", 0);
std::map::const_iterator i;
i = mSounds.find(event);
if (i == mSounds.end())
return empty;
return (!i->second.empty()) ? i->second[CAST_U32(rand())
% i->second.size()] : empty;
}
IntMap *ItemInfo::addReplaceSprite(const int sprite,
const int direction)
{
if (direction < 0 || direction >= 10)
return nullptr;
SpriteToItemMap *spMap = mSpriteToItemReplaceMap[direction];
if (spMap == nullptr)
{
spMap = new SpriteToItemMap;
mSpriteToItemReplaceMap[direction] = spMap;
mSpriteToItemReplaceList.push_back(spMap);
}
SpriteToItemMap::iterator it = spMap->find(sprite);
if (it == spMap->end())
{
IntMap tmp;
(*mSpriteToItemReplaceMap[direction])[sprite] = tmp;
it = mSpriteToItemReplaceMap[direction]->find(sprite);
}
return &it->second;
}
void ItemInfo::setColorsList(const std::string &name)
{
if (name.empty())
{
mColorsList = nullptr;
mColorsListName.clear();
}
else
{
mColorsList = ColorDB::getColorsList(name);
mColorsListName = name;
}
}
void ItemInfo::setIconColorsList(const std::string &name)
{
if (name.empty())
{
mIconColorsList = nullptr;
mIconColorsListName.clear();
}
else
{
mIconColorsList = ColorDB::getColorsList(name);
mIconColorsListName = name;
}
}
std::string ItemInfo::getDyeColorsString(const ItemColor color) const
{
if ((mColorsList == nullptr) || mColorsListName.empty())
return "";
const std::map ::const_iterator
it = mColorsList->find(color);
if (it == mColorsList->end())
return "";
return it->second.color;
}
std::string ItemInfo::getDyeIconColorsString(const ItemColor color) const
{
if ((mIconColorsList == nullptr) || mIconColorsListName.empty())
return "";
const std::map ::const_iterator
it = mIconColorsList->find(color);
if (it == mIconColorsList->end())
return "";
return it->second.color;
}
const std::string ItemInfo::getDescription(const ItemColor color) const
{
return replaceColors(mDescription, color);
}
const std::string ItemInfo::getName(const ItemColor color) const
{
return replaceColors(mName, color);
}
const std::string ItemInfo::getNameEn(const ItemColor color) const
{
return replaceColors(mNameEn, color);
}
const std::string ItemInfo::replaceColors(std::string str,
const ItemColor color) const
{
std::string name;
if ((mColorsList != nullptr) && !mColorsListName.empty())
{
const std::map ::const_iterator
it = mColorsList->find(color);
if (it == mColorsList->end())
name = "unknown";
else
name = it->second.name;
}
else
{
name = "unknown";
}
str = replaceAll(str, "%color%", name);
if (!name.empty())
name[0] = CAST_S8(toupper(name[0]));
return replaceAll(str, "%Color%", name);
}
const SpriteToItemMap *ItemInfo::getSpriteToItemReplaceMap(const int direction)
const
{
if (direction < 0 || direction >= 10)
return nullptr;
const SpriteToItemMap *const spMap = mSpriteToItemReplaceMap[direction];
if (spMap != nullptr)
return spMap;
if (direction == SpriteDirection::UPLEFT
|| direction == SpriteDirection::UPRIGHT)
{
return mSpriteToItemReplaceMap[SpriteDirection::UP];
}
if (direction == SpriteDirection::DOWNLEFT
|| direction == SpriteDirection::DOWNRIGHT)
{
return mSpriteToItemReplaceMap[SpriteDirection::DOWN];
}
return nullptr;
}
void ItemInfo::setSpriteOrder(int *const ptr,
const int direction,
const int n,
const int def)
{
switch (direction)
{
case -1:
{
for (int f = 0; f < 10; f ++)
{
if (ptr[f] == def)
ptr[f] = n;
}
return;
}
case -2:
{
ptr[SpriteDirection::DOWN] = n;
ptr[SpriteDirection::DOWNLEFT] = n;
ptr[SpriteDirection::DOWNRIGHT] = n;
return;
}
case -3:
{
ptr[SpriteDirection::UP] = n;
ptr[SpriteDirection::UPLEFT] = n;
ptr[SpriteDirection::UPRIGHT] = n;
return;
}
default:
break;
}
if (direction < 0 || direction >= 9)
return;
if (direction == SpriteDirection::UP)
{
if (ptr[SpriteDirection::UPLEFT] == def)
ptr[SpriteDirection::UPLEFT] = n;
if (ptr[SpriteDirection::UPRIGHT] == def)
ptr[SpriteDirection::UPRIGHT] = n;
}
else if (direction == SpriteDirection::DOWN)
{
if (ptr[SpriteDirection::DOWNLEFT] == def)
ptr[SpriteDirection::DOWNLEFT] = n;
if (ptr[SpriteDirection::DOWNRIGHT] == def)
ptr[SpriteDirection::DOWNRIGHT] = n;
}
ptr[direction] = n;
}
void ItemInfo::setDrawBefore(const int direction, const int n)
{
setSpriteOrder(&mDrawBefore[0], direction, n, -1);
}
void ItemInfo::setDrawAfter(const int direction, const int n)
{
setSpriteOrder(&mDrawAfter[0], direction, n, -1);
}
void ItemInfo::setDrawPriority(const int direction, const int n)
{
setSpriteOrder(&mDrawPriority[0], direction, n, 0);
}
int ItemInfo::getDrawBefore(const int direction) const
{
if (direction < 0 || direction >= 10)
return -1;
return mDrawBefore[direction];
}
int ItemInfo::getDrawAfter(const int direction) const
{
if (direction < 0 || direction >= 10)
return -1;
return mDrawAfter[direction];
}
int ItemInfo::getDrawPriority(const int direction) const
{
if (direction < 0 || direction >= 10)
return 0;
return mDrawPriority[direction];
}
void ItemInfo::setSprite(const std::string &animationFile,
const GenderT gender,
const int race)
{
mAnimationFiles[CAST_S32(gender) + race * 4] = animationFile;
}
std::string ItemInfo::getColorName(const ItemColor idx) const
{
if (mColorsList == nullptr)
return std::string();
const std::map ::const_iterator
it = mColorsList->find(idx);
if (it == mColorsList->end())
{
reportAlways("Color %d for palette %s not found",
CAST_S32(idx),
mColorsListName.c_str())
return std::string();
}
return it->second.name;
}
std::string ItemInfo::getColor(const ItemColor idx) const
{
if (mColorsList == nullptr)
return std::string();
const std::map ::const_iterator
it = mColorsList->find(idx);
if (it == mColorsList->end())
{
reportAlways("Color %d for palette %s not found",
CAST_S32(idx),
mColorsListName.c_str())
return std::string();
}
return it->second.color;
}
std::string ItemInfo::getIconColorName(const ItemColor idx) const
{
if (mIconColorsList == nullptr)
return std::string();
const std::map ::const_iterator
it = mIconColorsList->find(idx);
if (it == mIconColorsList->end())
{
reportAlways("Color %d for palette %s not found",
CAST_S32(idx),
mColorsListName.c_str())
return std::string();
}
return it->second.name;
}
std::string ItemInfo::getIconColor(const ItemColor idx) const
{
if (mIconColorsList == nullptr)
return std::string();
const std::map ::const_iterator
it = mIconColorsList->find(idx);
if (it == mIconColorsList->end())
{
reportAlways("Color %d for palette %s not found",
CAST_S32(idx),
mColorsListName.c_str())
return std::string();
}
return it->second.color;
}
const std::string ItemInfo::getLink() const
{
return strprintf("[@@%d|%s@@]", mId, mName.c_str());
}