/* * The ManaPlus Client * 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 "net/eathena/vendingrecv.h" #include "actormanager.h" #include "itemcolormanager.h" #include "notifymanager.h" #include "const/resources/currency.h" #include "being/localplayer.h" #include "being/playerinfo.h" #include "const/net/inventory.h" #include "enums/resources/notifytypes.h" #include "gui/windows/buydialog.h" #include "gui/widgets/createwidget.h" #include "listeners/vendingmodelistener.h" #include "listeners/vendingslotslistener.h" #include "net/messagein.h" #include "resources/iteminfo.h" #include "resources/db/unitsdb.h" #include "resources/inventory/inventory.h" #include "resources/item/itemoptionslist.h" #include "resources/item/shopitem.h" #include "utils/gettext.h" #include "utils/stringutils.h" #include "debug.h" extern int packetVersion; extern int serverVersion; extern int itemIdLen; namespace EAthena { namespace VendingRecv { BuyDialog *mBuyDialog = nullptr; } // namespace VendingRecv void VendingRecv::processOpenReq(Net::MessageIn &msg) { VendingSlotsListener::distributeEvent(msg.readInt16("slots allowed")); } void VendingRecv::processShowBoard(Net::MessageIn &msg) { if (actorManager == nullptr) return; const BeingId id = msg.readBeingId("owner id"); Being *const dstBeing = actorManager->findBeing(id); if (dstBeing != nullptr) { dstBeing->setSellBoard(msg.readString(80, "shop name")); } else { msg.readString(80, "shop name"); } } void VendingRecv::processHideBoard(Net::MessageIn &msg) { if (actorManager == nullptr) return; const BeingId id = msg.readBeingId("owner id"); Being *const dstBeing = actorManager->findBeing(id); if (dstBeing != nullptr) dstBeing->setSellBoard(std::string()); if (dstBeing == localPlayer) { PlayerInfo::enableVending(false); VendingModeListener::distributeEvent(false); } } void VendingRecv::processItemsList(Net::MessageIn &msg) { if (actorManager == nullptr) return; int packetLen = 22; if (msg.getVersion() >= 20160921) packetLen = 53; else if (msg.getVersion() >= 20150226) packetLen = 47; if (itemIdLen == 4) packetLen += 10; int offset = 8; if (msg.getVersion() >= 20100105) offset += 4; const int count = (msg.readInt16("len") - offset) / packetLen; const BeingId id = msg.readBeingId("id"); const Being *const being = actorManager->findBeing(id); if (being == nullptr) return; int cards[maxCards]; CREATEWIDGETV(mBuyDialog, BuyDialog, being, DEFAULT_CURRENCY); mBuyDialog->setMoney(PlayerInfo::getAttribute(Attributes::MONEY)); if (msg.getVersion() >= 20100105) msg.readInt32("vender id"); for (int f = 0; f < count; f ++) { const int value = msg.readInt32("price"); const int amount = msg.readInt16("amount"); const int index = msg.readInt16("inv index"); const ItemTypeT type = static_cast( msg.readUInt8("item type")); const int itemId = msg.readItemId("item id"); msg.readUInt8("identify"); msg.readUInt8("attribute"); msg.readUInt8("refine"); for (int d = 0; d < maxCards; d ++) cards[d] = msg.readItemId("card"); ItemOptionsList *options = nullptr; if (msg.getVersion() >= 20150226) { options = new ItemOptionsList; for (int d = 0; d < 5; d ++) { const uint16_t idx = msg.readInt16("option index"); const uint16_t val = msg.readInt16("option value"); msg.readUInt8("option param"); options->add(idx, val); } } if (msg.getVersion() >= 20160921) { msg.readInt32("equip type?"); msg.readInt16("look"); } const ItemColor color = ItemColorManager::getColorFromCards(&cards[0]); ShopItem *const item = mBuyDialog->addItem(itemId, type, color, amount, value); if (item != nullptr) { item->setInvIndex(index); item->setOptions(options); } delete options; } mBuyDialog->sort(); } void VendingRecv::processBuyAck(Net::MessageIn &msg) { msg.readInt16("inv index"); msg.readInt16("amount"); const int flag = msg.readUInt8("flag"); switch (flag) { case 0: break; case 1: NotifyManager::notify(NotifyTypes::BUY_FAILED_NO_MONEY); break; case 2: NotifyManager::notify(NotifyTypes::BUY_FAILED_OVERWEIGHT); break; case 4: NotifyManager::notify(NotifyTypes::BUY_FAILED_TOO_MANY_ITEMS); break; case 5: NotifyManager::notify(NotifyTypes::BUY_TRADE_FAILED); break; case 6: // +++ probably need show exact error messages? case 7: NotifyManager::notify(NotifyTypes::BUY_FAILED); break; default: NotifyManager::notify(NotifyTypes::BUY_FAILED); UNIMPLEMENTEDPACKETFIELD(flag); break; } } void VendingRecv::processOpen(Net::MessageIn &msg) { int packetLen = 22; if (msg.getVersion() >= 20150226) packetLen += 25; if (itemIdLen == 4) packetLen += 10; const int count = (msg.readInt16("len") - 8) / packetLen; msg.readInt32("id"); for (int f = 0; f < count; f ++) { msg.readInt32("price"); msg.readInt16("inv index"); msg.readInt16("amount"); msg.readUInt8("item type"); msg.readItemId("item id"); msg.readUInt8("identify"); msg.readUInt8("attribute"); msg.readUInt8("refine"); for (int d = 0; d < maxCards; d ++) msg.readItemId("card"); if (msg.getVersion() >= 20150226) { for (int d = 0; d < 5; d ++) { msg.readInt16("option index"); msg.readInt16("option value"); msg.readUInt8("option param"); } } } PlayerInfo::enableVending(true); VendingModeListener::distributeEvent(true); } void VendingRecv::processReport(Net::MessageIn &msg) { const int index = msg.readInt16("inv index") - INVENTORY_OFFSET; const int amount = msg.readInt16("amount"); int money = 0; if (msg.getVersion() >= 20141016) { msg.readInt32("char id"); msg.readInt32("time"); money = msg.readInt32("zeny"); } const Inventory *const inventory = PlayerInfo::getCartInventory(); if (inventory == nullptr) return; const Item *const item = inventory->getItem(index); if (item == nullptr) return; const ItemInfo &info = item->getInfo(); std::string str; if (money != 0) { // TRANSLATORS: vending sold item message str = strprintf(_("Sold item %s amount %d. You got: %s"), info.getLink().c_str(), amount, UnitsDb::formatCurrency(money).c_str()); } else { // TRANSLATORS: vending sold item message str = strprintf(_("Sold item %s amount %d"), info.getLink().c_str(), amount); } NotifyManager::notify(NotifyTypes::VENDING_SOLD_ITEM, str); } void VendingRecv::processOpenStatus(Net::MessageIn &msg) { UNIMPLEMENTEDPACKET; msg.readUInt8("result"); } } // namespace EAthena