1
Fork 0
mirror of https://github.com/SapphireServer/Sapphire.git synced 2025-04-28 07:07:45 +00:00
sapphire/src/world/Manager/ActionMgr.cpp

169 lines
4.9 KiB
C++
Raw Normal View History

#include "ActionMgr.h"
#include "Action/Action.h"
#include "Script/ScriptMgr.h"
#include "Actor/Player.h"
#include <Exd/ExdDataGenerated.h>
2019-02-09 18:02:11 +11:00
#include "Framework.h"
2019-02-09 21:48:42 +11:00
#include <Network/PacketWrappers/EffectPacket.h>
using namespace Sapphire;
World::Manager::ActionMgr::ActionMgr( Sapphire::FrameworkPtr pFw ) :
BaseManager( pFw )
{
}
2019-02-09 19:26:31 +11:00
void World::Manager::ActionMgr::handleAoEPlayerAction( Entity::Player& player, uint32_t actionId,
Data::ActionPtr actionData, Common::FFXIVARR_POSITION3 pos )
{
player.sendDebug( "got aoe act: {0}", actionData->name );
2019-02-09 18:02:11 +11:00
2019-02-09 18:02:11 +11:00
auto action = Action::make_Action( player.getAsPlayer(), actionId, actionData, framework() );
action->setPos( pos );
if( !actionData->targetArea )
{
// not an action that has an aoe, cancel it
action->castInterrupt();
return;
}
bootstrapAction( player, action, *actionData );
}
2019-02-09 19:26:31 +11:00
void World::Manager::ActionMgr::handleTargetedPlayerAction( Entity::Player& player, uint32_t actionId,
Data::ActionPtr actionData, uint64_t targetId )
{
auto action = Action::make_Action( player.getAsPlayer(), actionId, actionData, framework() );
// cancel any aoe actions casted with this packet
if( actionData->targetArea )
{
action->castInterrupt();
return;
}
2019-02-09 23:54:49 +11:00
if( targetId != player.getId() )
{
auto target = player.lookupTargetById( targetId );
2019-02-10 22:13:47 +11:00
if( !target )
{
// an eobj?
player.sendDebug( "Unable to find actor for targetId#{0}, passing through to event scripts...", targetId );
action->setResidentTargetId( targetId );
}
else if( auto chara = target->getAsChara() )
{
2019-02-09 23:54:49 +11:00
action->setTargetChara( chara );
2019-02-10 22:13:47 +11:00
}
2019-02-09 23:54:49 +11:00
}
bootstrapAction( player, action, *actionData );
}
void World::Manager::ActionMgr::handleItemAction( Sapphire::Entity::Player& player, uint32_t itemId,
Data::ItemActionPtr itemActionData,
uint16_t itemSourceSlot, uint16_t itemSourceContainer )
2019-02-09 19:26:31 +11:00
{
player.sendDebug( "got item act: {0}, slot: {1}, container: {2}", itemId, itemSourceSlot, itemSourceContainer );
// todo: check we have item & remove item from inventory
2019-02-09 21:48:42 +11:00
switch( itemActionData->type )
{
default:
2019-02-09 22:45:29 +11:00
{
player.sendDebug( "ItemAction type {0} not supported.", itemActionData->type );
break;
}
2019-02-09 21:48:42 +11:00
case Common::ItemActionType::ItemActionVFX:
{
handleItemActionVFX( player, itemId, itemActionData->data[ 0 ] );
break;
}
}
2019-02-09 19:26:31 +11:00
}
void World::Manager::ActionMgr::bootstrapAction( Entity::Player& player,
Action::ActionPtr currentAction,
Data::Action& actionData )
{
if( !canPlayerUseAction( player, *currentAction, actionData ) )
{
// forcefully interrupt the action and reset the cooldown
currentAction->castInterrupt();
return;
}
// if we have a cast time we want to associate the action with the player so update is called
if( currentAction->hasCastTime() )
{
player.setCurrentAction( currentAction );
}
// todo: what do in cases of swiftcast/etc? script callback?
currentAction->castStart();
}
bool World::Manager::ActionMgr::canPlayerUseAction( Entity::Player& player,
Action::Action& currentAction,
Data::Action& actionData )
{
// lol
if( !player.isAlive() )
return false;
// npc actions/non player actions
if( actionData.classJob == -1 )
return false;
2019-02-09 18:02:11 +11:00
if( player.getLevel() < actionData.classJobLevel )
return false;
if( player.getClass() != static_cast< Common::ClassJob >( actionData.classJob ) )
{
// check if not a base class action
auto exdData = framework()->get< Data::ExdDataGenerated >();
assert( exdData );
auto classJob = exdData->get< Data::ClassJob >( static_cast< uint8_t >( player.getClass() ) );
if( !classJob )
return false;
if( classJob->classJobParent != actionData.classJob )
return false;
}
// validate range
2019-02-09 23:54:49 +11:00
auto& actionCost = currentAction.getCostArray();
for( uint8_t i = 0; i < actionCost.size(); ++i )
{
// todo: validate costs/conditionals here
}
return true;
2019-02-09 21:48:42 +11:00
}
void World::Manager::ActionMgr::handleItemActionVFX( Sapphire::Entity::Player& player, uint32_t itemId, uint16_t vfxId )
{
Common::EffectEntry effect{};
effect.effectType = Common::ActionEffectType::VFX;
effect.value = vfxId;
auto effectPacket = std::make_shared< Network::Packets::Server::EffectPacket >( player.getId(), player.getId(), itemId );
effectPacket->setTargetActor( player.getId() );
effectPacket->setAnimationId( Common::ItemActionType::ItemActionVFX );
effectPacket->setDisplayType( Common::ActionEffectDisplayType::ShowItemName );
effectPacket->addEffect( effect );
player.sendToInRangeSet( effectPacket, true );
}