/* * 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 . */ #include "net/tmwa/inventoryhandler.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 "net/tmwa/messagein.h" #include "net/tmwa/messageout.h" #include "net/tmwa/protocol.h" #include "resources/iteminfo.h" #include "utils/gettext.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; 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(); 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 &card : cards) card = 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 &card : cards) card = 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 &card : cards) card = 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); for (auto &item : mInventoryItems) mStorage->setItem(item.slot, item.id, item.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 &card : cards) card = 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); } } 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