/* * 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/mailrecv.h" #include "itemcolormanager.h" #include "notifymanager.h" #include "const/resources/item/cards.h" #include "enums/resources/notifytypes.h" #include "gui/mailmessage.h" #include "gui/windows/mailviewwindow.h" #include "gui/windows/mailwindow.h" #include "net/mailhandler.h" #include "net/messagein.h" #include "utils/checkutils.h" #include "utils/gettext.h" #include "utils/stringutils.h" #include "resources/inventory/inventory.h" #include "debug.h" namespace EAthena { void MailRecv::processMailOpen(Net::MessageIn &msg) { const int flag = msg.readInt32("flag"); switch (flag) { case 0: // open window if (mailWindow != nullptr) { if (!mailWindow->isWindowVisible()) mailWindow->setVisible(Visible_true); mailWindow->requestMoveToTop(); } break; case 1: // close window if (mailWindow != nullptr) { if (mailWindow->isWindowVisible()) mailWindow->setVisible(Visible_false); } break; default: UNIMPLEMENTEDPACKETFIELD(flag); break; } } void MailRecv::processMailList(Net::MessageIn &msg) { const int count = (msg.readInt16("len") - 8) / 73; const int amount = msg.readInt32("amount"); if (count != amount) logger->log("error: wrong mails count"); mailWindow->clear(); for (int f = 0; f < count; f ++) { MailMessage *const mail = new MailMessage; mail->id = msg.readInt32("message id"); mail->title = msg.readString(40, "title"); mail->read = msg.readUInt8("unread flag") != 0U ? true : false; mail->sender = msg.readString(24, "sender name"); mail->time = msg.readInt32("time stamp"); mail->strTime = timeToStr(mail->time); mailWindow->addMail(mail); } } void MailRecv::processReadMail(Net::MessageIn &msg) { const int sz = msg.readInt16("len") - 101; MailMessage *const mail = new MailMessage; mail->id = msg.readInt32("message id"); mail->title = msg.readString(40, "title"); mail->sender = msg.readString(24, "sender name"); msg.readInt32("unused"); mail->money = msg.readInt32("money"); const int amount = msg.readInt32("item amount"); const int itemId = msg.readInt16("item id"); const ItemTypeT itemType = static_cast( msg.readInt16("item type")); const uint8_t identify = msg.readUInt8("identify"); const Damaged damaged = fromBool(msg.readUInt8("attribute"), Damaged); const uint8_t refine = msg.readUInt8("refine"); int cards[maxCards]; for (int d = 0; d < maxCards; d ++) cards[d] = msg.readUInt16("card"); const int msgLen = msg.readUInt8("msg len"); if (msgLen != sz) logger->log("error: wrong message size"); mail->text = msg.readString(sz, "message"); msg.readUInt8("zero"); mail->strTime = timeToStr(mail->time); if (mailWindow == nullptr) { reportAlways("Mail window not created") delete mail; return; } mailWindow->showMessage(mail, itemId != 0 ? 1 : 0); if (mailViewWindow == nullptr) { reportAlways("Mail view window not created") return; } Inventory *const inventory = mailViewWindow->getInventory(); if (inventory == nullptr) { reportAlways("Mail view window missing inventory") return; } const int slot = inventory->addItem(itemId, itemType, amount, refine, ItemColorManager::getColorFromCards(&cards[0]), fromBool(identify, Identified), damaged, Favorite_false, Equipm_false, Equipped_false); if (slot != -1) inventory->setCards(slot, cards, maxCards); mailViewWindow->updateItems(); } void MailRecv::processGetAttachment(Net::MessageIn &msg) { const uint8_t flag = msg.readUInt8("flag"); switch (flag) { case 0: NotifyManager::notify(NotifyTypes::MAIL_GET_ATTACH_OK); break; case 1: NotifyManager::notify(NotifyTypes::MAIL_GET_ATTACH_ERROR); break; case 2: NotifyManager::notify(NotifyTypes::MAIL_GET_ATTACH_TOO_MANY_ITEMS); break; default: UNIMPLEMENTEDPACKETFIELD(flag); break; } } void MailRecv::processSendMailAck(Net::MessageIn &msg) { const uint8_t flag = msg.readUInt8("fail flag"); switch (flag) { case 0: NotifyManager::notify(NotifyTypes::MAIL_SEND_OK); break; case 1: NotifyManager::notify(NotifyTypes::MAIL_SEND_ERROR); break; default: UNIMPLEMENTEDPACKETFIELD(flag); break; } } void MailRecv::processNewMail(Net::MessageIn &msg) { msg.readInt32("message id"); const std::string subj = msg.readString(40, "title"); const std::string sender = msg.readString(24, "sender name"); NotifyManager::notify(NotifyTypes::NEW_MAIL, // TRANSLATORS: mail message notification strprintf(_("You have new mail from %s with subject %s"), sender.c_str(), subj.c_str())); mailHandler->refresh(); } void MailRecv::processSetAttachmentAck(Net::MessageIn &msg) { const int index = msg.readInt16("index"); const int flag = msg.readUInt8("flag"); if (flag != 0) { if (index != 0) NotifyManager::notify(NotifyTypes::MAIL_ATTACH_ITEM_ERROR); else NotifyManager::notify(NotifyTypes::MAIL_ATTACH_MONEY_ERROR); } } void MailRecv::processDeleteAck(Net::MessageIn &msg) { const int mail = msg.readInt32("message id"); const int flag = msg.readInt16("fail flag"); if (flag != 0) { NotifyManager::notify(NotifyTypes::MAIL_DELETE_ERROR); } else { NotifyManager::notify(NotifyTypes::MAIL_DELETE_OK); mailWindow->removeMail(mail); } } void MailRecv::processMailReturn(Net::MessageIn &msg) { const int mail = msg.readInt32("message id"); const int flag = msg.readInt16("fail flag"); if (flag != 0) { NotifyManager::notify(NotifyTypes::MAIL_RETURN_ERROR); } else { NotifyManager::notify(NotifyTypes::MAIL_RETURN_OK); mailWindow->removeMail(mail); } } } // namespace EAthena