/*
* The Mana Client
* Copyright (C) 2004-2009 The Mana World Development Team
* Copyright (C) 2009-2012 The Mana Developers
*
* This file is part of The Mana 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 "net/tmwa/inventoryhandler.h"
#include "configuration.h"
#include "equipment.h"
#include "event.h"
#include "game.h"
#include "inventory.h"
#include "item.h"
#include "itemshortcut.h"
#include "localplayer.h"
#include "log.h"
#include "gui/equipmentwindow.h"
#include "gui/widgets/chattab.h"
#include "net/tmwa/messagein.h"
#include "net/tmwa/messageout.h"
#include "net/tmwa/protocol.h"
#include "resources/iteminfo.h"
#include "utils/gettext.h"
#include "utils/stringutils.h"
extern Net::InventoryHandler *inventoryHandler;
namespace TmwAthena {
static const EquipmentSlot EQUIP_POINTS[EQUIP_VECTOR_END] = {
EQUIP_LEGS_SLOT,
EQUIP_FIGHT1_SLOT,
EQUIP_ARMS_SLOT,
EQUIP_NECKLACE_SLOT,
EQUIP_RING1_SLOT,
EQUIP_FIGHT2_SLOT,
EQUIP_FEET_SLOT,
EQUIP_RING2_SLOT,
EQUIP_HEAD_SLOT,
EQUIP_TORSO_SLOT,
EQUIP_PROJECTILE_SLOT
};
static int getSlot(int eAthenaSlot)
{
if (eAthenaSlot == 0)
{
return EQUIP_VECTOR_END;
}
if (eAthenaSlot & 0x8000)
return EQUIP_PROJECTILE_SLOT;
int mask = 1;
int position = 0;
while (!(eAthenaSlot & mask))
{
mask <<= 1;
position++;
}
return EQUIP_POINTS[position];
}
enum { debugInventory = 1 };
InventoryHandler::InventoryHandler()
{
static const Uint16 _messages[] = {
SMSG_PLAYER_INVENTORY,
SMSG_PLAYER_INVENTORY_ADD,
SMSG_PLAYER_INVENTORY_REMOVE,
SMSG_PLAYER_INVENTORY_USE,
SMSG_ITEM_USE_RESPONSE,
SMSG_PLAYER_STORAGE_ITEMS,
SMSG_PLAYER_STORAGE_EQUIP,
SMSG_PLAYER_STORAGE_STATUS,
SMSG_PLAYER_STORAGE_ADD,
SMSG_PLAYER_STORAGE_REMOVE,
SMSG_PLAYER_STORAGE_CLOSE,
SMSG_PLAYER_EQUIPMENT,
SMSG_PLAYER_EQUIP,
SMSG_PLAYER_UNEQUIP,
SMSG_PLAYER_ARROW_EQUIP,
SMSG_PLAYER_ATTACK_RANGE,
0
};
handledMessages = _messages;
inventoryHandler = this;
mStorage = nullptr;
mStorageWindow = nullptr;
listen(Event::ItemChannel);
}
InventoryHandler::~InventoryHandler()
{
if (mStorageWindow)
{
mStorageWindow->close();
mStorageWindow = nullptr;
}
delete mStorage;
}
void InventoryHandler::handleMessage(MessageIn &msg)
{
int number, flag;
int index, amount, itemId, equipType;
int identified, cards[4], itemType;
Inventory *inventory = PlayerInfo::getInventory();
PlayerInfo::getEquipment()->setBackend(&mEquips);
switch (msg.getId())
{
case SMSG_PLAYER_INVENTORY:
case SMSG_PLAYER_STORAGE_ITEMS:
if (msg.getId() == SMSG_PLAYER_INVENTORY)
{
// Clear inventory - this will be a complete refresh
mEquips.clear();
inventory->clear();
}
else
{
mInventoryItems.clear();
}
msg.readInt16(); // length
number = (msg.getLength() - 4) / 18;
for (int loop = 0; loop < number; loop++)
{
index = msg.readInt16();
itemId = msg.readInt16();
itemType = msg.readInt8();
identified = msg.readInt8();
amount = msg.readInt16();
msg.readInt16(); // Arrow
for (int i = 0; i < 4; i++)
cards[i] = msg.readInt16();
index -= (msg.getId() == SMSG_PLAYER_INVENTORY) ?
INVENTORY_OFFSET : STORAGE_OFFSET;
if (debugInventory)
{
logger->log("Index: %d, ID: %d, Type: %d, Identified: %d, "
"Qty: %d, Cards: %d, %d, %d, %d",
index, itemId, itemType, identified, amount,
cards[0], cards[1], cards[2], cards[3]);
}
if (msg.getId() == SMSG_PLAYER_INVENTORY)
inventory->setItem(index, itemId, amount);
else
mInventoryItems.push_back(InventoryItem(index, itemId,
amount, false));
}
break;
case SMSG_PLAYER_STORAGE_EQUIP:
msg.readInt16(); // length
number = (msg.getLength() - 4) / 20;
for (int loop = 0; loop < number; loop++)
{
index = msg.readInt16() - STORAGE_OFFSET;
itemId = msg.readInt16();
itemType = msg.readInt8();
identified = msg.readInt8();
amount = 1;
msg.readInt16(); // Equip Point?
msg.readInt16(); // Another Equip Point?
msg.readInt8(); // Attribute (broken)
msg.readInt8(); // Refine level
for (int i = 0; i < 4; i++)
cards[i] = msg.readInt16();
if (debugInventory)
{
logger->log("Index: %d, ID: %d, Type: %d, Identified: %d, "
"Qty: %d, Cards: %d, %d, %d, %d",
index, itemId, itemType, identified, amount,
cards[0], cards[1], cards[2], cards[3]);
}
mInventoryItems.push_back(InventoryItem(index, itemId, amount,
false));
}
break;
case SMSG_PLAYER_INVENTORY_ADD:
index = msg.readInt16() - INVENTORY_OFFSET;
amount = msg.readInt16();
itemId = msg.readInt16();
identified = msg.readInt8();
msg.readInt8(); // attribute
msg.readInt8(); // refine
for (int i = 0; i < 4; i++)
cards[i] = msg.readInt16();
msg.readInt16(); // EquipType
itemType = msg.readInt8();
{
const ItemInfo &itemInfo = itemDb->get(itemId);
unsigned char err = msg.readInt8();
if (err)
{
local_player->pickedUp(itemInfo, 0, err);
}
else
{
local_player->pickedUp(itemInfo, amount, PICKUP_OKAY);
Item *item = inventory->getItem(index);
if (item && item->getId() == itemId)
amount += inventory->getItem(index)->getQuantity();
inventory->setItem(index, itemId, amount);
}
inventoryWindow->updateButtons();
} break;
case SMSG_PLAYER_INVENTORY_REMOVE:
index = msg.readInt16() - INVENTORY_OFFSET;
amount = msg.readInt16();
if (Item *item = inventory->getItem(index))
{
item->increaseQuantity(-amount);
if (item->getQuantity() == 0)
inventory->removeItemAt(index);
inventoryWindow->updateButtons();
}
break;
case SMSG_PLAYER_INVENTORY_USE:
index = msg.readInt16() - INVENTORY_OFFSET;
msg.readInt16(); // item id
msg.readInt32(); // id
amount = msg.readInt16();
msg.readInt8(); // type
if (Item *item = inventory->getItem(index))
{
if (amount)
item->setQuantity(amount);
else
inventory->removeItemAt(index);
inventoryWindow->updateButtons();
}
break;
case SMSG_ITEM_USE_RESPONSE:
index = msg.readInt16() - INVENTORY_OFFSET;
amount = msg.readInt16();
if (msg.readInt8() == 0)
{
SERVER_NOTICE(_("Failed to use item."))
}
else
{
if (Item *item = inventory->getItem(index))
{
if (amount)
item->setQuantity(amount);
else
inventory->removeItemAt(index);
inventoryWindow->updateButtons();
}
}
break;
case SMSG_PLAYER_STORAGE_STATUS:
/*
* This is the closest we get to an "Open Storage" packet from the
* server. It always comes after the two SMSG_PLAYER_STORAGE_...
* packets that update storage contents.
*/
{
msg.readInt16(); // Used count
int size = msg.readInt16(); // Max size
if (!mStorage)
mStorage = new Inventory(Inventory::STORAGE, size);
auto it = mInventoryItems.begin();
auto it_end = mInventoryItems.end();
for (; it != it_end; it++)
mStorage->setItem((*it).slot, (*it).id, (*it).quantity);
mInventoryItems.clear();
if (!mStorageWindow)
mStorageWindow = new InventoryWindow(mStorage);
}
break;
case SMSG_PLAYER_STORAGE_ADD:
// Move an item into storage
index = msg.readInt16() - STORAGE_OFFSET;
amount = msg.readInt32();
itemId = msg.readInt16();
identified = msg.readInt8();
msg.readInt8(); // attribute
msg.readInt8(); // refine
for (int i = 0; i < 4; i++)
cards[i] = msg.readInt16();
if (Item *item = mStorage->getItem(index))
{
item->setId(itemId);
item->increaseQuantity(amount);
}
else
mStorage->setItem(index, itemId, amount);
break;
case SMSG_PLAYER_STORAGE_REMOVE:
// Move an item out of storage
index = msg.readInt16() - STORAGE_OFFSET;
amount = msg.readInt16();
if (Item *item = mStorage->getItem(index))
{
item->increaseQuantity(-amount);
if (item->getQuantity() == 0)
mStorage->removeItemAt(index);
}
break;
case SMSG_PLAYER_STORAGE_CLOSE:
// Storage access has been closed
// Storage window deletes itself
mStorageWindow = nullptr;
mStorage->clear();
delete mStorage;
mStorage = nullptr;
break;
case SMSG_PLAYER_EQUIPMENT:
msg.readInt16(); // length
number = (msg.getLength() - 4) / 20;
for (int loop = 0; loop < number; loop++)
{
index = msg.readInt16() - INVENTORY_OFFSET;
itemId = msg.readInt16();
msg.readInt8(); // type
msg.readInt8(); // identify flag
msg.readInt16(); // equip type
equipType = msg.readInt16();
msg.readInt8(); // attribute
msg.readInt8(); // refine
msg.skip(8); // card
inventory->setItem(index, itemId, 1);
if (equipType)
{
mEquips.setEquipment(getSlot(equipType), index);
}
// Load the equipment boxes
if (equipmentWindow)
equipmentWindow->loadEquipBoxes();
}
break;
case SMSG_PLAYER_EQUIP:
index = msg.readInt16() - INVENTORY_OFFSET;
equipType = msg.readInt16();
flag = msg.readInt8();
if (!flag)
SERVER_NOTICE(_("Unable to equip."))
else
mEquips.setEquipment(getSlot(equipType), index);
break;
case SMSG_PLAYER_UNEQUIP:
index = msg.readInt16() - INVENTORY_OFFSET;
equipType = msg.readInt16();
flag = msg.readInt8();
if (!flag)
{
SERVER_NOTICE(_("Unable to unequip."))
}
else
{
mEquips.setEquipment(getSlot(equipType), -1);
// Reset the attack range to unarmed.
local_player->setAttackRange(-1);
}
break;
case SMSG_PLAYER_ATTACK_RANGE:
{
// The range is in tiles, so we translate it back to pixels
Map *map = Game::instance()->getCurrentMap();
if (map)
{
local_player->setAttackRange(msg.readInt16()
* map->getTileWidth());
}
else
{
logger->log("Couldn't set attacke range due to the lack"
"of an initialized map.");
local_player->setAttackRange(-1);
}
}
break;
case SMSG_PLAYER_ARROW_EQUIP:
index = msg.readInt16();
if (index <= 1)
break;
index -= INVENTORY_OFFSET;
logger->log("Arrows equipped: %i", index);
mEquips.setEquipment(EQUIP_PROJECTILE_SLOT, index);
break;
}
}
void InventoryHandler::event(Event::Channel channel,
const Event &event)
{
if (channel == Event::ItemChannel)
{
if (event.getType() == Event::DoCloseInventory)
{
// No need to worry about type
MessageOut outMsg(CMSG_CLOSE_STORAGE);
}
else
{
Item *item = event.getItem("item");
if (!item)
return;
int index = item->getInvIndex() + INVENTORY_OFFSET;
if (event.getType() == Event::DoEquip)
{
MessageOut outMsg(CMSG_PLAYER_EQUIP);
outMsg.writeInt16(index);
outMsg.writeInt16(0);
}
else if (event.getType() == Event::DoUnequip)
{
MessageOut outMsg(CMSG_PLAYER_UNEQUIP);
outMsg.writeInt16(index);
}
else if (event.getType() == Event::DoUse)
{
MessageOut outMsg(CMSG_PLAYER_INVENTORY_USE);
outMsg.writeInt16(index);
outMsg.writeInt32(item->getId()); // unused
}
else if (event.getType() == Event::DoDrop)
{
int amount = event.getInt("amount", 1);
// TODO: Fix wrong coordinates of drops, serverside?
// (what's wrong here?)
MessageOut outMsg(CMSG_PLAYER_INVENTORY_DROP);
outMsg.writeInt16(index);
outMsg.writeInt16(amount);
}
else if (event.getType() == Event::DoMove)
{
int newIndex = event.getInt("newIndex", -1);
if (newIndex >= 0)
{
// Not implemented for tmwAthena (possible?)
}
else
{
int source = event.getInt("source");
int destination = event.getInt("destination");
int amount = event.getInt("amount", 1);
if (source == Inventory::INVENTORY
&& destination == Inventory::STORAGE)
{
MessageOut outMsg(CMSG_MOVE_TO_STORAGE);
outMsg.writeInt16(index);
outMsg.writeInt32(amount);
}
else if (source == Inventory::STORAGE
&& destination == Inventory::INVENTORY)
{
MessageOut outMsg(CMSG_MOVE_FROM_STORAGE);
outMsg.writeInt16(index - INVENTORY_OFFSET
+ STORAGE_OFFSET);
outMsg.writeInt32(amount);
}
}
}
}
}
}
bool InventoryHandler::canSplit(const Item *item)
{
return false;
}
size_t InventoryHandler::getSize(int type) const
{
switch (type)
{
case Inventory::INVENTORY:
return 100;
case Inventory::STORAGE:
return 0; // Comes from server after items
case Inventory::TRADE:
return 12;
case GUILD_STORAGE:
return 0; // Comes from server after items
default:
return 0;
}
}
} // namespace TmwAthena