1
Fork 0
mirror of https://github.com/SapphireServer/Sapphire.git synced 2025-04-26 14:37:44 +00:00
sapphire/src/world/Network/Handlers/EventHandlers.cpp

287 lines
11 KiB
C++

#include <Common.h>
#include <Logging/Logger.h>
#include <Exd/ExdDataGenerated.h>
#include <Network/CommonNetwork.h>
#include <Network/GamePacket.h>
#include <Network/PacketContainer.h>
#include <Network/PacketDef/Zone/ServerZoneDef.h>
#include <Network/PacketDef/Zone/ClientZoneDef.h>
#include "Network/GameConnection.h"
#include "Network/PacketWrappers/ServerNoticePacket.h"
#include "Network/PacketWrappers/EventStartPacket.h"
#include "Network/PacketWrappers/EventFinishPacket.h"
#include "Network/PacketWrappers/PlayerStateFlagsPacket.h"
#include "Script/ScriptMgr.h"
#include <Util/Util.h>
#include "Event/EventHandler.h"
#include "Manager/EventMgr.h"
#include "Territory/InstanceContent.h"
#include "Territory/QuestBattle.h"
#include "Session.h"
#include "Framework.h"
using namespace Sapphire::Common;
using namespace Sapphire::Network::Packets;
using namespace Sapphire::Network::Packets::Server;
void Sapphire::Network::GameConnection::eventHandlerTalk( FrameworkPtr pFw,
const Packets::FFXIVARR_PACKET_RAW& inPacket,
Entity::Player& player )
{
auto pScriptMgr = pFw->get< Scripting::ScriptMgr >();
auto pExdData = pFw->get< Data::ExdDataGenerated >();
auto pEventMgr = pFw->get< World::Manager::EventMgr >();
const auto packet = WorldChannelPacket< Client::FFXIVIpcEventHandlerTalk >( inPacket );
const auto actorId = packet.data().actorId;
const auto eventId = packet.data().eventId;
auto eventType = static_cast< uint16_t >( eventId >> 16 );
std::string eventName = "onTalk";
std::string objName = pEventMgr->getEventName( eventId );
player.sendDebug( "Chara: {0} -> {1} \neventId: {2} ({3:08X})",
actorId, pEventMgr->mapEventActorToRealActor( static_cast< uint32_t >( actorId ) ),
eventId, eventId );
player.sendDebug( "Calling: {0}.{1}", objName, eventName );
player.eventStart( actorId, eventId, Event::EventHandler::Talk, 0, 0 );
if( auto instance = player.getCurrentInstance() )
{
instance->onTalk( player, eventId, actorId );
}
else if( !pScriptMgr->onTalk( player, actorId, eventId ) &&
eventType == Event::EventHandler::EventHandlerType::Quest )
{
auto questInfo = pExdData->get< Sapphire::Data::Quest >( eventId );
if( questInfo )
player.sendUrgent( "Quest not implemented: {0} ({1})", questInfo->name, questInfo->id );
}
player.checkEvent( eventId );
}
void Sapphire::Network::GameConnection::eventHandlerEmote( FrameworkPtr pFw,
const Packets::FFXIVARR_PACKET_RAW& inPacket,
Entity::Player& player )
{
auto pScriptMgr = pFw->get< Scripting::ScriptMgr >();
auto pExdData = pFw->get< Data::ExdDataGenerated >();
auto pEventMgr = pFw->get< World::Manager::EventMgr >();
const auto packet = WorldChannelPacket< Client::FFXIVIpcEventHandlerEmote >( inPacket );
const auto actorId = packet.data().actorId;
const auto eventId = packet.data().eventId;
const auto emoteId = packet.data().emoteId;
const auto eventType = static_cast< uint16_t >( eventId >> 16 );
std::string eventName = "onEmote";
std::string objName = pEventMgr->getEventName( eventId );
player.sendDebug( "Chara: {0} -> {1} \neventId: {2} ({3:08X})",
actorId, pEventMgr->mapEventActorToRealActor( static_cast< uint32_t >( actorId ) ),
eventId, eventId );
player.sendDebug( "Calling: {0}.{1}", objName, eventName );
player.eventStart( actorId, eventId, Event::EventHandler::Emote, 0, emoteId );
if( !pScriptMgr->onEmote( player, actorId, eventId, static_cast< uint8_t >( emoteId ) ) &&
eventType == Event::EventHandler::EventHandlerType::Quest )
{
auto questInfo = pExdData->get< Sapphire::Data::Quest >( eventId );
if( questInfo )
player.sendUrgent( "Quest not implemented: {0}", questInfo->name );
}
player.checkEvent( eventId );
}
void Sapphire::Network::GameConnection::eventHandlerWithinRange( FrameworkPtr pFw,
const Packets::FFXIVARR_PACKET_RAW& inPacket,
Entity::Player& player )
{
auto pScriptMgr = pFw->get< Scripting::ScriptMgr >();
auto pEventMgr = pFw->get< World::Manager::EventMgr >();
const auto packet = WorldChannelPacket< Client::FFXIVIpcEventHandlerWithinRange >( inPacket );
const auto eventId = packet.data().eventId;
const auto param1 = packet.data().param1;
const auto& pos = packet.data().position;
std::string eventName = "onWithinRange";
std::string objName = pEventMgr->getEventName( eventId );
player.sendDebug( "Calling: {0}.{1} - {2} p1: {3}", objName, eventName, eventId, param1 );
player.eventStart( player.getId(), eventId, Event::EventHandler::WithinRange, 1, param1 );
pScriptMgr->onWithinRange( player, eventId, param1, pos.x, pos.y, pos.z );
player.checkEvent( eventId );
}
void Sapphire::Network::GameConnection::eventHandlerOutsideRange( FrameworkPtr pFw,
const Packets::FFXIVARR_PACKET_RAW& inPacket,
Entity::Player& player )
{
auto pScriptMgr = pFw->get< Scripting::ScriptMgr >();
auto pEventMgr = pFw->get< World::Manager::EventMgr >();
const auto packet = WorldChannelPacket< Client::FFXIVIpcEventHandlerOutsideRange >( inPacket );
const auto eventId = packet.data().eventId;
const auto param1 = packet.data().param1;
const auto& pos = packet.data().position;
std::string eventName = "onOutsideRange";
std::string objName = pEventMgr->getEventName( eventId );
player.sendDebug( "Calling: {0}.{1} - {2} p1: {3}", objName, eventName, eventId, param1 );
player.eventStart( player.getId(), eventId, Event::EventHandler::WithinRange, 1, param1 );
pScriptMgr->onOutsideRange( player, eventId, param1, pos.x, pos.y, pos.z );
player.checkEvent( eventId );
}
void Sapphire::Network::GameConnection::eventHandlerEnterTerritory( FrameworkPtr pFw,
const Packets::FFXIVARR_PACKET_RAW& inPacket,
Entity::Player& player )
{
auto pScriptMgr = pFw->get< Scripting::ScriptMgr >();
auto pEventMgr = pFw->get< World::Manager::EventMgr >();
const auto packet = WorldChannelPacket< Client::FFXIVIpcEnterTerritoryHandler >( inPacket );
const auto eventId = packet.data().eventId;
const auto param1 = packet.data().param1;
const auto param2 = packet.data().param2;
std::string eventName = "onEnterTerritory";
std::string objName = pEventMgr->getEventName( eventId );
player.sendDebug( "Calling: {0}.{1} - {2}", objName, eventName, eventId & 0xFFFF );
if( auto instance = player.getCurrentInstance() )
{
player.eventStart( player.getId(), eventId, Event::EventHandler::EnterTerritory, 1, player.getZoneId() );
instance->onEnterTerritory( player, eventId, param1, param2 );
}
else if( auto instance = player.getCurrentQuestBattle() )
{
player.eventStart( player.getId(), eventId, Event::EventHandler::EnterTerritory, 1, player.getZoneId() );
instance->onEnterTerritory( player, eventId, param1, param2 );
}
else
{
player.eventStart( player.getId(), eventId, Event::EventHandler::EnterTerritory, 0, player.getZoneId() );
pScriptMgr->onEnterTerritory( player, eventId, param1, param2 );
}
player.checkEvent( eventId );
}
void Sapphire::Network::GameConnection::eventHandlerReturn( FrameworkPtr pFw,
const Packets::FFXIVARR_PACKET_RAW& inPacket,
Entity::Player& player )
{
auto pEventMgr = pFw->get< World::Manager::EventMgr >();
const auto packet = WorldChannelPacket< Client::FFXIVIpcEventHandlerReturn >( inPacket );
const auto eventId = packet.data().eventId;
const auto scene = packet.data().scene;
const auto param1 = packet.data().param1;
const auto param2 = packet.data().param2;
const auto param3 = packet.data().param3;
const auto param4 = packet.data().param4;
std::string eventName = pEventMgr->getEventName( eventId );
player.sendDebug( "eventId: {0} ({0:08X}) scene: {1}, p1: {2}, p2: {3}, p3: {4}",
eventId, scene, param1, param2, param3 );
auto pEvent = player.getEvent( eventId );
if( pEvent )
{
pEvent->setPlayedScene( false );
// try to retrieve a stored callback
auto eventCallback = pEvent->getEventReturnCallback();
// if there is one, proceed to call it
if( eventCallback )
{
Event::SceneResult result;
result.actorId = pEvent->getActorId();
result.eventId = eventId;
result.param1 = param1;
result.param2 = param2;
result.param3 = param3;
result.param4 = param4;
eventCallback( player, result );
}
// we might have a scene chain callback instead so check for that too
else if( auto chainCallback = pEvent->getSceneChainCallback() )
chainCallback( player );
}
player.checkEvent( eventId );
}
void Sapphire::Network::GameConnection::eventHandlerLinkshell( FrameworkPtr pFw,
const Packets::FFXIVARR_PACKET_RAW& inPacket,
Entity::Player& player )
{
const auto packet = WorldChannelPacket< Client::FFXIVIpcLinkshellEventHandler >( inPacket );
auto linkshellEvent = makeWorldPacket< Server::FFXIVIpcEventLinkshell >( player.getId() );
linkshellEvent->data().eventId = packet.data().eventId;
linkshellEvent->data().scene = static_cast< uint8_t >( packet.data().scene );
linkshellEvent->data().param3 = 1;
linkshellEvent->data().unknown1 = 0x15a;
player.queuePacket( linkshellEvent );
}
void Sapphire::Network::GameConnection::eventHandlerShop( FrameworkPtr pFw,
const Packets::FFXIVARR_PACKET_RAW& inPacket,
Entity::Player& player )
{
auto pEventMgr = pFw->get< World::Manager::EventMgr >();
const auto packet = WorldChannelPacket< Client::FFXIVIpcShopEventHandler >( inPacket );
auto pScriptMgr = pFw->get< Scripting::ScriptMgr >();
auto pExdData = pFw->get< Data::ExdDataGenerated >();
const auto eventId = packet.data().eventId;
auto eventType = static_cast< uint16_t >( eventId >> 16 );
std::string eventName = "onOpen";
std::string objName = pEventMgr->getEventName( eventId );
player.sendDebug( "EventId: {0} ({0:08X})", eventId );
player.sendDebug( "Calling: {0}.{1}", objName, eventName );
player.eventStart( player.getId(), eventId, Event::EventHandler::UI, 0, packet.data().param );
pScriptMgr->onTalk( player, player.getId(), eventId );
}