/*
* The ManaPlus Client
* Copyright (C) 2004-2009 The Mana World Development Team
* Copyright (C) 2009-2010 The Mana Developers
* Copyright (C) 2011 The ManaPlus 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/tmwa/specialhandler.h"
#include "log.h"
#include "playerinfo.h"
#include "gui/skilldialog.h"
#include "gui/widgets/chattab.h"
#include "net/messagein.h"
#include "net/messageout.h"
#include "net/tmwa/protocol.h"
#include "utils/gettext.h"
#include "debug.h"
/** job dependend identifiers (?) */
#define SKILL_BASIC 0x0001
#define SKILL_WARP 0x001b
#define SKILL_STEAL 0x0032
#define SKILL_ENVENOM 0x0034
/** basic skills identifiers */
#define BSKILL_TRADE 0x0000
#define BSKILL_EMOTE 0x0001
#define BSKILL_SIT 0x0002
#define BSKILL_CREATECHAT 0x0003
#define BSKILL_JOINPARTY 0x0004
#define BSKILL_SHOUT 0x0005
#define BSKILL_PK 0x0006 // ??
#define BSKILL_SETALLIGN 0x0007 // ??
/** reasons why action failed */
#define RFAIL_SKILLDEP 0x00
#define RFAIL_INSUFSP 0x01
#define RFAIL_INSUFHP 0x02
#define RFAIL_NOMEMO 0x03
#define RFAIL_SKILLDELAY 0x04
#define RFAIL_ZENY 0x05
#define RFAIL_WEAPON 0x06
#define RFAIL_REDGEM 0x07
#define RFAIL_BLUEGEM 0x08
#define RFAIL_OVERWEIGHT 0x09
#define RFAIL_GENERIC 0x0a
/** should always be zero if failed */
#define SKILL_FAILED 0x00
extern Net::SpecialHandler *specialHandler;
namespace TmwAthena
{
SpecialHandler::SpecialHandler()
{
static const Uint16 _messages[] =
{
SMSG_PLAYER_SKILLS,
SMSG_SKILL_FAILED,
SMSG_PLAYER_SKILL_UP,
0
};
handledMessages = _messages;
specialHandler = this;
}
void SpecialHandler::handleMessage(Net::MessageIn &msg)
{
int skillCount;
int skillId;
switch (msg.getId())
{
case SMSG_PLAYER_SKILLS:
{
msg.readInt16(); // length
skillCount = (msg.getLength() - 4) / 37;
for (int k = 0; k < skillCount; k++)
{
skillId = msg.readInt16();
msg.readInt16(); // target type
msg.skip(2); // unused
int level = msg.readInt16();
msg.readInt16(); // sp
msg.readInt16(); // range
msg.skip(24); // unused
int up = msg.readInt8();
PlayerInfo::setStatBase(skillId, level);
if (skillDialog)
{
if (!skillDialog->setModifiable(skillId, up))
skillDialog->addSkill(skillId, level, up);
}
}
break;
}
case SMSG_PLAYER_SKILL_UP:
{
skillId = msg.readInt16();
int level = msg.readInt16();
msg.readInt16(); // sp
msg.readInt16(); // range
int up = msg.readInt8();
PlayerInfo::setStatBase(skillId, level);
if (skillDialog)
{
if (!skillDialog->setModifiable(skillId, up))
skillDialog->addSkill(skillId, level, up);
}
}
break;
case SMSG_SKILL_FAILED:
{
// Action failed (ex. sit because you have not reached the
// right level)
skillId = msg.readInt16();
short bskill = msg.readInt16();
msg.readInt16(); // btype
char success = msg.readInt8();
char reason = msg.readInt8();
if (success != SKILL_FAILED && bskill == BSKILL_EMOTE)
logger->log("Action: %d/%d", bskill, success);
std::string msg;
if (success == SKILL_FAILED && skillId == SKILL_BASIC)
{
switch (bskill)
{
case BSKILL_TRADE:
msg = _("Trade failed!");
break;
case BSKILL_EMOTE:
msg = _("Emote failed!");
break;
case BSKILL_SIT:
msg = _("Sit failed!");
break;
case BSKILL_CREATECHAT:
msg = _("Chat creating failed!");
break;
case BSKILL_JOINPARTY:
msg = _("Could not join party!");
break;
case BSKILL_SHOUT:
msg = _("Cannot shout!");
break;
default:
logger->log("QQQ SMSG_SKILL_FAILED: bskill "
+ toString(bskill));
break;
}
msg += " ";
switch (reason)
{
case RFAIL_SKILLDEP:
msg += _("You have not yet reached a high enough "
"lvl!");
break;
case RFAIL_INSUFHP:
msg += _("Insufficient HP!");
break;
case RFAIL_INSUFSP:
msg += _("Insufficient SP!");
break;
case RFAIL_NOMEMO:
msg += _("You have no memos!");
break;
case RFAIL_SKILLDELAY:
msg += _("You cannot do that right now!");
break;
case RFAIL_ZENY:
msg += _("Seems you need more money... ;-)");
break;
case RFAIL_WEAPON:
msg += _("You cannot use this skill with that "
"kind of weapon!");
break;
case RFAIL_REDGEM:
msg += _("You need another red gem!");
break;
case RFAIL_BLUEGEM:
msg += _("You need another blue gem!");
break;
case RFAIL_OVERWEIGHT:
msg += _("You're carrying to much to do this!");
break;
default:
msg += _("Huh? What's that?");
logger->log("QQQ SMSG_SKILL_FAILED: reason "
+ toString(reason));
break;
}
}
else
{
switch (skillId)
{
case SKILL_WARP :
msg = _("Warp failed...");
break;
case SKILL_STEAL :
msg = _("Could not steal anything...");
break;
case SKILL_ENVENOM :
msg = _("Poison had no effect...");
break;
default:
logger->log("QQQ SMSG_SKILL_FAILED: skillId "
+ toString(skillId));
break;
}
}
if (localChatTab)
localChatTab->chatLog(msg);
}
break;
default:
break;
}
}
void SpecialHandler::use(int id _UNUSED_)
{
// TODO
}
void SpecialHandler::use(int id, int level, int beingId)
{
MessageOut outMsg(CMSG_SKILL_USE_BEING);
outMsg.writeInt16(static_cast(level));
outMsg.writeInt16(static_cast(id));
outMsg.writeInt16(static_cast(beingId));
}
void SpecialHandler::use(int id, int level, int x, int y)
{
MessageOut outMsg(CMSG_SKILL_USE_POSITION);
outMsg.writeInt16(static_cast(level));
outMsg.writeInt16(static_cast(id));
outMsg.writeInt16(static_cast(x));
outMsg.writeInt16(static_cast(y));
}
void SpecialHandler::use(int id, const std::string &map)
{
MessageOut outMsg(CMSG_SKILL_USE_MAP);
outMsg.writeInt16(static_cast(id));
outMsg.writeString(map, 16);
}
} // namespace TmwAthena