2024-04-29 16:51:28 +01:00
|
|
|
#include <fstream>
|
|
|
|
#include <memory>
|
|
|
|
#include <map>
|
|
|
|
#include <optional>
|
|
|
|
#include <stack>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2024-05-06 01:45:42 +01:00
|
|
|
#include <queue>
|
2024-04-29 16:51:28 +01:00
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
#include <Common.h>
|
|
|
|
#include <Forwards.h>
|
2024-04-29 16:51:28 +01:00
|
|
|
#include <nlohmann/json.hpp>
|
|
|
|
|
|
|
|
namespace Sapphire
|
|
|
|
{
|
2024-05-05 16:15:05 +01:00
|
|
|
class EncounterTimeline
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
public:
|
2024-05-09 16:17:46 +01:00
|
|
|
// forwards
|
|
|
|
class TimelineActor;
|
2024-05-10 22:46:33 +01:00
|
|
|
class TimelinePack;
|
|
|
|
|
|
|
|
struct TimepointState
|
|
|
|
{
|
|
|
|
uint64_t m_startTime{ 0 };
|
|
|
|
uint64_t m_lastTick{ 0 };
|
|
|
|
bool m_finished{ false };
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ConditionState
|
|
|
|
{
|
|
|
|
uint64_t m_startTime{ 0 };
|
|
|
|
bool m_loop{ false };
|
|
|
|
bool m_completed{ false };
|
|
|
|
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
uint64_t m_startTime{ 0 };
|
|
|
|
uint64_t m_lastTimepointTime{ 0 };
|
|
|
|
uint32_t m_lastTimepointIndex{ 0 };
|
|
|
|
|
|
|
|
std::vector< TimepointState > m_timepointStates;
|
|
|
|
} m_phaseInfo;
|
|
|
|
};
|
2024-05-09 16:17:46 +01:00
|
|
|
|
2024-04-29 16:51:28 +01:00
|
|
|
// EncounterFight::OnTick() { switch EncounterTimepointConditionId }
|
2024-05-06 01:45:42 +01:00
|
|
|
enum class ConditionId : uint32_t
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
HpPctLessThan,
|
|
|
|
HpPctBetween,
|
2024-05-10 17:42:39 +01:00
|
|
|
|
2024-04-29 16:51:28 +01:00
|
|
|
DirectorVarEquals,
|
2024-05-06 01:45:42 +01:00
|
|
|
DirectorVarGreaterThan,
|
2024-05-06 20:13:53 +01:00
|
|
|
DirectorSeqEquals,
|
|
|
|
DirectorSeqGreaterThan,
|
|
|
|
DirectorFlagsEquals,
|
|
|
|
DirectorFlagsGreaterThan,
|
2024-05-10 17:42:39 +01:00
|
|
|
|
2024-05-09 17:27:51 +01:00
|
|
|
EncounterTimeElapsed,
|
2024-05-10 17:42:39 +01:00
|
|
|
CombatState,
|
|
|
|
BNpcHasFlags
|
2024-05-06 01:45:42 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
enum class DirectorOpId
|
|
|
|
{
|
|
|
|
SetDirectorVar,
|
|
|
|
SetDirectorVarLR,
|
|
|
|
SetDirectorSeq,
|
|
|
|
SetDirectorFlag,
|
2024-05-10 17:42:39 +01:00
|
|
|
ClearDirector
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
// TODO: what should this do?
|
2024-05-06 01:45:42 +01:00
|
|
|
enum class TimepointOverrideFlags : uint32_t
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
None,
|
|
|
|
Invulnerable
|
|
|
|
};
|
|
|
|
|
2024-05-06 01:45:42 +01:00
|
|
|
enum class TimepointDataType : uint32_t
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
Idle,
|
|
|
|
CastAction,
|
|
|
|
MoveTo,
|
2024-05-10 22:46:33 +01:00
|
|
|
|
2024-04-29 16:51:28 +01:00
|
|
|
LogMessage,
|
|
|
|
BattleTalk,
|
2024-05-10 22:46:33 +01:00
|
|
|
|
2024-04-29 16:51:28 +01:00
|
|
|
SetDirectorVar,
|
2024-05-10 15:48:42 +01:00
|
|
|
SetDirectorVarLR,
|
2024-05-06 01:45:42 +01:00
|
|
|
SetDirectorSeq,
|
|
|
|
SetDirectorFlag,
|
2024-05-10 22:46:33 +01:00
|
|
|
|
2024-04-29 16:51:28 +01:00
|
|
|
AddStatusEffect,
|
2024-05-10 17:42:39 +01:00
|
|
|
RemoveStatusEffect,
|
2024-05-10 22:46:33 +01:00
|
|
|
|
2024-05-10 17:42:39 +01:00
|
|
|
SetBNpcFlags,
|
|
|
|
SetEObjState,
|
|
|
|
SetBgm
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
2024-05-06 01:45:42 +01:00
|
|
|
enum class TimepointCallbackType : uint32_t
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
OnActionInit,
|
|
|
|
OnActionStart,
|
|
|
|
OnActionInterrupt,
|
|
|
|
OnActionExecute
|
|
|
|
};
|
|
|
|
|
2024-05-06 20:13:53 +01:00
|
|
|
enum class TargetSelectFilterId : uint32_t
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
Self,
|
|
|
|
Tank,
|
|
|
|
Healer,
|
|
|
|
Dps,
|
|
|
|
DpsMelee,
|
|
|
|
DpsRanged,
|
|
|
|
Furthest,
|
|
|
|
|
|
|
|
Aggro1,
|
|
|
|
Aggro2
|
|
|
|
};
|
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
enum class MoveType : uint32_t
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
WalkPath,
|
|
|
|
Teleport
|
|
|
|
};
|
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
enum class TimelineActorType : uint32_t
|
|
|
|
{
|
|
|
|
LayerSetObject,
|
|
|
|
BattleNpc
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class CombatStateType
|
|
|
|
{
|
|
|
|
Idle,
|
|
|
|
Combat,
|
|
|
|
Retreat,
|
|
|
|
Roaming,
|
|
|
|
JustDied,
|
|
|
|
Dead
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class TimelinePackType : uint32_t
|
|
|
|
{
|
|
|
|
Solo,
|
|
|
|
EncounterFight
|
|
|
|
};
|
|
|
|
|
2024-04-29 16:51:28 +01:00
|
|
|
struct TargetSelectFilter
|
|
|
|
{
|
2024-05-06 20:13:53 +01:00
|
|
|
TargetSelectFilterId m_flags;
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2024-05-10 15:48:42 +01:00
|
|
|
using TimepointCallbackFunc = std::function< void( InstanceContentPtr, uint64_t ) >;
|
2024-05-06 01:45:42 +01:00
|
|
|
// Timepoint Data Objects
|
|
|
|
struct TimepointCallbackData :
|
|
|
|
public std::enable_shared_from_this< TimepointCallbackData >
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
2024-05-06 01:45:42 +01:00
|
|
|
TimepointCallbackType m_type;
|
|
|
|
std::vector < TimepointCallbackFunc > m_callbacks;
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
2024-05-06 01:45:42 +01:00
|
|
|
using TimebackCallbackDataPtr = std::shared_ptr< TimepointCallbackData >;
|
|
|
|
using TimepointCallbacks = std::map< TimepointCallbackType, TimebackCallbackDataPtr >;
|
2024-04-29 16:51:28 +01:00
|
|
|
|
|
|
|
|
2024-05-06 01:45:42 +01:00
|
|
|
struct TimepointData :
|
|
|
|
public std::enable_shared_from_this< TimepointData >
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
2024-05-06 20:13:53 +01:00
|
|
|
TimepointData( TimepointDataType type ) : m_type( type ) {}
|
|
|
|
virtual ~TimepointData(){};
|
|
|
|
TimepointDataType m_type{ 0 };
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
2024-05-06 01:45:42 +01:00
|
|
|
using TimepointDataPtr = std::shared_ptr< TimepointData >;
|
2024-04-29 16:51:28 +01:00
|
|
|
|
2024-05-06 01:45:42 +01:00
|
|
|
struct TimepointDataIdle : public TimepointData
|
|
|
|
{
|
2024-05-10 15:48:42 +01:00
|
|
|
uint32_t m_layoutId;
|
2024-05-06 01:45:42 +01:00
|
|
|
uint64_t m_durationMs;
|
2024-05-06 20:13:53 +01:00
|
|
|
|
2024-05-10 15:48:42 +01:00
|
|
|
TimepointDataIdle( uint32_t layoutId, uint64_t durationMs ) :
|
2024-05-06 20:13:53 +01:00
|
|
|
TimepointData( TimepointDataType::Idle ),
|
2024-05-10 15:48:42 +01:00
|
|
|
m_layoutId( layoutId ),
|
2024-05-06 20:13:53 +01:00
|
|
|
m_durationMs( durationMs )
|
|
|
|
{
|
|
|
|
}
|
2024-05-06 01:45:42 +01:00
|
|
|
};
|
2024-04-29 16:51:28 +01:00
|
|
|
|
2024-05-06 20:13:53 +01:00
|
|
|
struct TimepointDataAddStatusEffect : public TimepointData
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
uint32_t m_statusEffectId;
|
|
|
|
TargetSelectFilter m_targetFilter;
|
|
|
|
uint32_t m_durationMs;
|
2024-05-06 20:13:53 +01:00
|
|
|
|
|
|
|
TimepointDataAddStatusEffect( uint32_t statusId, TargetSelectFilter targFilter, uint32_t durationMs ) :
|
|
|
|
TimepointData( TimepointDataType::AddStatusEffect ),
|
|
|
|
m_statusEffectId( statusId ),
|
|
|
|
m_targetFilter( targFilter ),
|
|
|
|
m_durationMs( durationMs )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TimepointDataRemoveStatusEffect : public TimepointData
|
|
|
|
{
|
|
|
|
uint32_t m_statusEffectId;
|
|
|
|
TargetSelectFilter m_targetFilter;
|
|
|
|
|
|
|
|
TimepointDataRemoveStatusEffect( uint32_t statusId, TargetSelectFilter targFilter ) :
|
|
|
|
TimepointData( TimepointDataType::RemoveStatusEffect ),
|
|
|
|
m_statusEffectId( statusId ),
|
|
|
|
m_targetFilter( targFilter )
|
|
|
|
{
|
|
|
|
}
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
2024-05-06 01:45:42 +01:00
|
|
|
struct TimepointDataAction : public TimepointData
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
2024-05-10 15:48:42 +01:00
|
|
|
uint32_t m_layoutId;
|
2024-04-29 16:51:28 +01:00
|
|
|
uint32_t m_actionId;
|
2024-05-06 01:45:42 +01:00
|
|
|
TimepointCallbacks m_callbacks;
|
2024-05-06 20:13:53 +01:00
|
|
|
|
2024-05-10 15:48:42 +01:00
|
|
|
TimepointDataAction( uint32_t layoutId, uint32_t actionId, TimepointCallbacks callbacks ) :
|
2024-05-06 20:13:53 +01:00
|
|
|
TimepointData( TimepointDataType::CastAction ),
|
2024-05-10 15:48:42 +01:00
|
|
|
m_layoutId( layoutId ),
|
2024-05-06 20:13:53 +01:00
|
|
|
m_actionId( actionId ),
|
|
|
|
m_callbacks( callbacks )
|
|
|
|
{
|
|
|
|
}
|
2024-05-06 01:45:42 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct TimepointDataMoveTo : public TimepointData
|
|
|
|
{
|
2024-05-10 15:48:42 +01:00
|
|
|
uint32_t m_layoutId;
|
2024-05-06 01:45:42 +01:00
|
|
|
MoveType m_moveType;
|
|
|
|
float m_x, m_y, m_z, m_rot;
|
2024-05-06 20:13:53 +01:00
|
|
|
|
2024-05-10 15:48:42 +01:00
|
|
|
TimepointDataMoveTo( uint32_t layoutId, MoveType moveType, float x, float y, float z, float rot ) :
|
2024-05-06 20:13:53 +01:00
|
|
|
TimepointData( TimepointDataType::MoveTo ),
|
2024-05-10 15:48:42 +01:00
|
|
|
m_layoutId( layoutId ),
|
2024-05-06 20:13:53 +01:00
|
|
|
m_moveType( moveType ),
|
|
|
|
m_x( x ), m_y( y ), m_z( z ), m_rot( rot )
|
|
|
|
{
|
|
|
|
}
|
2024-05-06 01:45:42 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct TimepointDataLogMessage : public TimepointData
|
|
|
|
{
|
2024-05-06 20:13:53 +01:00
|
|
|
uint32_t m_messageId;
|
|
|
|
uint32_t m_params[ 6 ]{ 0 };
|
|
|
|
|
|
|
|
TimepointDataLogMessage( uint32_t messageId, std::vector< uint32_t > params ) :
|
|
|
|
TimepointData( TimepointDataType::LogMessage ),
|
|
|
|
m_messageId( messageId )
|
|
|
|
{
|
|
|
|
for( auto i = 0; i < params.size(); ++i )
|
|
|
|
m_params[i] = params[i];
|
|
|
|
}
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
2024-05-06 01:45:42 +01:00
|
|
|
struct TimepointDataDirector : public TimepointData
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
2024-05-10 17:42:39 +01:00
|
|
|
DirectorOpId m_directorOp{ 0 };
|
2024-05-06 01:45:42 +01:00
|
|
|
union
|
|
|
|
{
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
uint8_t index;
|
|
|
|
union
|
|
|
|
{
|
|
|
|
uint8_t val;
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
uint8_t left, right;
|
|
|
|
};
|
|
|
|
} value;
|
|
|
|
};
|
|
|
|
uint8_t seq;
|
|
|
|
uint8_t flags;
|
2024-05-10 17:42:39 +01:00
|
|
|
} m_data{ 0 };
|
2024-05-06 20:13:53 +01:00
|
|
|
|
2024-05-10 17:42:39 +01:00
|
|
|
TimepointDataDirector( TimepointDataType type ) :
|
|
|
|
TimepointData( type )
|
|
|
|
{
|
|
|
|
switch( type )
|
|
|
|
{
|
|
|
|
case TimepointDataType::SetDirectorVar:
|
|
|
|
m_directorOp = DirectorOpId::SetDirectorVar;
|
|
|
|
break;
|
|
|
|
case TimepointDataType::SetDirectorVarLR:
|
|
|
|
m_directorOp = DirectorOpId::SetDirectorVarLR;
|
|
|
|
break;
|
|
|
|
case TimepointDataType::SetDirectorFlag:
|
|
|
|
m_directorOp = DirectorOpId::SetDirectorFlag;
|
|
|
|
break;
|
|
|
|
case TimepointDataType::SetDirectorSeq:
|
|
|
|
m_directorOp = DirectorOpId::SetDirectorSeq;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TimepointDataBNpcFlags : public TimepointData
|
|
|
|
{
|
|
|
|
uint32_t m_layoutId;
|
|
|
|
uint32_t m_flags;
|
|
|
|
|
|
|
|
TimepointDataBNpcFlags( uint32_t layoutId, uint32_t flags ) :
|
|
|
|
TimepointData( TimepointDataType::SetBNpcFlags ),
|
|
|
|
m_layoutId( layoutId ),
|
|
|
|
m_flags( flags )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TimepointDataEObjState : public TimepointData
|
|
|
|
{
|
|
|
|
uint32_t m_eobjId{ 0xE0000000 };
|
|
|
|
uint32_t m_state{ 0 };
|
|
|
|
|
|
|
|
TimepointDataEObjState( uint32_t eobjId, uint32_t state ) :
|
|
|
|
TimepointData( TimepointDataType::SetEObjState ),
|
|
|
|
m_eobjId( eobjId ),
|
|
|
|
m_state( state )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TimepointDataBGM : public TimepointData
|
|
|
|
{
|
|
|
|
uint32_t m_bgmId{ 0 };
|
|
|
|
|
|
|
|
TimepointDataBGM( uint32_t bgmId ):
|
|
|
|
TimepointData( TimepointDataType::SetBgm ),
|
|
|
|
m_bgmId( bgmId )
|
|
|
|
{
|
|
|
|
}
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
// todo: refactor all this to allow solo actor to use
|
2024-05-06 01:45:42 +01:00
|
|
|
class Timepoint :
|
|
|
|
public std::enable_shared_from_this< Timepoint >
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
public:
|
2024-05-06 01:45:42 +01:00
|
|
|
TimepointDataType m_type;
|
|
|
|
uint64_t m_duration{ 0 };
|
|
|
|
TimepointOverrideFlags m_overrideFlags;
|
|
|
|
TimepointDataPtr m_pData;
|
2024-04-29 16:51:28 +01:00
|
|
|
std::string m_description;
|
|
|
|
|
2024-05-06 01:45:42 +01:00
|
|
|
// todo: repeatable?
|
|
|
|
|
2024-05-06 20:13:53 +01:00
|
|
|
const TimepointDataPtr getData() const
|
|
|
|
{
|
|
|
|
return m_pData;
|
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
bool canExecute( TimepointState& state, uint64_t elapsed ) const
|
2024-05-06 01:45:42 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
return state.m_startTime == 0; // & &m_duration <= elapsed;
|
2024-05-06 01:45:42 +01:00
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
bool finished( TimepointState& state, uint64_t time ) const
|
2024-05-06 01:45:42 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
return state.m_startTime + m_duration <= time || state.m_finished;
|
2024-05-10 15:18:06 +01:00
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
void reset( TimepointState& state ) const
|
2024-05-10 15:18:06 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
state.m_startTime = 0;
|
|
|
|
state.m_lastTick = 0;
|
|
|
|
state.m_finished = false;
|
2024-05-06 01:45:42 +01:00
|
|
|
}
|
|
|
|
|
2024-05-09 17:27:51 +01:00
|
|
|
void from_json( const nlohmann::json& json, const std::unordered_map< std::string, TimelineActor >& actors, uint32_t selfLayoutId );
|
2024-05-10 15:18:06 +01:00
|
|
|
// todo: separate execute/update into onStart and onTick?
|
2024-05-10 22:46:33 +01:00
|
|
|
void update( TimepointState& state, InstanceContentPtr pInstance, uint64_t time ) const;
|
|
|
|
void execute( TimepointState& state, InstanceContentPtr pInstance, uint64_t time ) const;
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
2024-05-06 01:45:42 +01:00
|
|
|
class Phase :
|
|
|
|
public std::enable_shared_from_this< Phase >
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
public:
|
2024-05-06 01:45:42 +01:00
|
|
|
|
|
|
|
// todo: respect looping phases, allow callbacks to push timepoints
|
|
|
|
|
2024-04-29 16:51:28 +01:00
|
|
|
std::string m_name;
|
2024-05-09 16:17:46 +01:00
|
|
|
std::vector< Timepoint > m_timepoints;
|
2024-05-06 01:45:42 +01:00
|
|
|
|
|
|
|
// todo: i wrote this very sleep deprived, ensure it is actually sane
|
2024-05-10 22:46:33 +01:00
|
|
|
void execute( ConditionState& state, InstanceContentPtr pInstance, uint64_t time ) const
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
if( state.m_startTime == 0 )
|
|
|
|
state.m_startTime = time;
|
|
|
|
if( state.m_phaseInfo.m_startTime == 0 )
|
|
|
|
state.m_phaseInfo.m_startTime = time;
|
|
|
|
if( state.m_phaseInfo.m_lastTimepointTime == 0 )
|
|
|
|
state.m_phaseInfo.m_lastTimepointTime = time;
|
|
|
|
|
|
|
|
for( auto i = state.m_phaseInfo.m_lastTimepointIndex; i < m_timepoints.size(); )
|
2024-05-06 01:45:42 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
uint64_t phaseElapsed = time - state.m_phaseInfo.m_startTime;
|
|
|
|
uint64_t timepointElapsed = time - state.m_phaseInfo.m_lastTimepointTime;
|
|
|
|
|
|
|
|
auto& tpState = state.m_phaseInfo.m_timepointStates[ i ];
|
2024-05-09 16:17:46 +01:00
|
|
|
auto& timepoint = m_timepoints[ i ];
|
2024-05-10 22:46:33 +01:00
|
|
|
if( timepoint.canExecute( tpState, timepointElapsed ) )
|
2024-05-06 01:45:42 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
timepoint.execute( tpState, pInstance, time );
|
|
|
|
state.m_phaseInfo.m_lastTimepointTime = time;
|
2024-05-06 01:45:42 +01:00
|
|
|
}
|
2024-05-10 22:46:33 +01:00
|
|
|
else if( !timepoint.finished( tpState, timepointElapsed ) )
|
2024-05-10 15:18:06 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
timepoint.update( tpState, pInstance, time );
|
2024-05-10 15:18:06 +01:00
|
|
|
}
|
2024-05-09 16:17:46 +01:00
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
if( timepoint.finished( tpState, timepointElapsed ) )
|
2024-05-06 01:45:42 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
timepoint.reset( tpState );
|
2024-05-10 15:18:06 +01:00
|
|
|
// make sure this timepoint isnt run again unless phase loops
|
2024-05-09 16:17:46 +01:00
|
|
|
++i;
|
2024-05-10 22:46:33 +01:00
|
|
|
state.m_phaseInfo.m_lastTimepointIndex = i;
|
|
|
|
|
|
|
|
if( i == m_timepoints.size() )
|
|
|
|
{
|
|
|
|
state.m_phaseInfo.m_lastTimepointIndex++;
|
|
|
|
}
|
2024-05-09 16:17:46 +01:00
|
|
|
continue;
|
2024-05-06 01:45:42 +01:00
|
|
|
}
|
2024-05-09 16:17:46 +01:00
|
|
|
break;
|
2024-05-06 01:45:42 +01:00
|
|
|
}
|
2024-04-29 16:51:28 +01:00
|
|
|
}
|
2024-05-06 20:13:53 +01:00
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
void reset( ConditionState& state ) const
|
|
|
|
{
|
|
|
|
state.m_phaseInfo.m_startTime = 0;
|
|
|
|
state.m_phaseInfo.m_lastTimepointIndex = 0;
|
|
|
|
state.m_phaseInfo.m_lastTimepointTime = 0;
|
|
|
|
|
|
|
|
state.m_phaseInfo.m_timepointStates.clear();
|
|
|
|
for( auto i = 0; i < m_timepoints.size(); ++i )
|
|
|
|
state.m_phaseInfo.m_timepointStates.push_back( {} );
|
|
|
|
}
|
|
|
|
|
|
|
|
bool completed( ConditionState& state ) const
|
2024-05-06 20:13:53 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
return state.m_phaseInfo.m_lastTimepointIndex > m_timepoints.size();
|
2024-05-06 20:13:53 +01:00
|
|
|
}
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
2024-05-06 01:45:42 +01:00
|
|
|
using PhasePtr = std::shared_ptr< Phase >;
|
2024-04-29 16:51:28 +01:00
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
class PhaseCondition :
|
|
|
|
public std::enable_shared_from_this< PhaseCondition >
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
protected:
|
2024-05-06 01:45:42 +01:00
|
|
|
ConditionId m_conditionId{ 0 };
|
2024-05-06 20:13:53 +01:00
|
|
|
Phase m_phase;
|
2024-05-09 16:17:46 +01:00
|
|
|
std::string m_description;
|
2024-05-10 22:46:33 +01:00
|
|
|
uint32_t m_cooldown{ 0 };
|
|
|
|
bool m_loop{ false };
|
|
|
|
public:
|
2024-05-09 16:17:46 +01:00
|
|
|
PhaseCondition() {}
|
|
|
|
~PhaseCondition() {}
|
2024-04-29 16:51:28 +01:00
|
|
|
|
2024-05-10 17:42:39 +01:00
|
|
|
virtual void from_json( nlohmann::json& json, Phase& phase, ConditionId conditionId )
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
this->m_conditionId = conditionId;
|
|
|
|
this->m_loop = json.at( "loop" ).get< bool >();
|
2024-05-09 16:17:46 +01:00
|
|
|
//this->m_cooldown = json.at( "cooldown" ).get< uint32_t >();
|
2024-05-06 20:13:53 +01:00
|
|
|
this->m_phase = phase;
|
2024-05-10 22:46:33 +01:00
|
|
|
this->m_description = json.at( "description" ).get< std::string >();
|
2024-04-29 16:51:28 +01:00
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
void execute( ConditionState& state, InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time ) const
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
state.m_startTime = time;
|
|
|
|
m_phase.execute( state, pInstance, time );
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
void update( ConditionState& state, InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time ) const
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
m_phase.execute( state, pInstance, time );
|
2024-05-09 16:17:46 +01:00
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
void reset( ConditionState& state ) const
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
state.m_startTime = 0;
|
|
|
|
m_phase.reset( state );
|
2024-05-09 16:17:46 +01:00
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
bool inProgress( ConditionState& state ) const
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
return state.m_startTime != 0;
|
2024-05-09 16:17:46 +01:00
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
bool completed( ConditionState& state ) const
|
2024-05-06 20:13:53 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
return m_phase.completed( state );
|
2024-05-06 20:13:53 +01:00
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
bool isLoopable() const
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
|
|
|
return m_loop;
|
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
bool loopReady( ConditionState& state, uint64_t time ) const
|
2024-05-06 20:13:53 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
return m_phase.completed( state ) && m_loop && ( state.m_startTime + m_cooldown <= time );
|
2024-05-06 20:13:53 +01:00
|
|
|
}
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
virtual bool isConditionMet( ConditionState& state, InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time ) const
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
2024-05-09 16:17:46 +01:00
|
|
|
using PhaseConditionPtr = std::shared_ptr< PhaseCondition >;
|
2024-04-29 16:51:28 +01:00
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
class TimelineActor
|
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
protected:
|
|
|
|
std::vector< PhaseConditionPtr > m_phaseConditions;
|
|
|
|
std::queue< PhaseConditionPtr > m_phaseHistory;
|
|
|
|
std::vector< ConditionState > m_conditionStates;
|
2024-05-09 16:17:46 +01:00
|
|
|
public:
|
2024-05-09 17:27:51 +01:00
|
|
|
uint32_t m_layoutId{ 0 };
|
|
|
|
uint32_t m_hp{ 0 };
|
2024-05-09 16:17:46 +01:00
|
|
|
std::string m_name;
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
void addPhaseCondition( PhaseConditionPtr pCondition )
|
|
|
|
{
|
|
|
|
m_phaseConditions.push_back( pCondition );
|
|
|
|
m_conditionStates.push_back( {} );
|
|
|
|
}
|
2024-05-09 16:17:46 +01:00
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
// todo: make this sane and pass info down
|
|
|
|
void update( InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time )
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
|
|
|
// todo: handle auto attacks and make it so they dont fire during boss intermission phases
|
|
|
|
// todo: handle interrupts
|
2024-05-10 22:46:33 +01:00
|
|
|
for( auto i = 0; i < m_phaseConditions.size(); ++i )
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
const auto& pCondition = m_phaseConditions[ i ];
|
|
|
|
auto& state = m_conditionStates[ i ];
|
|
|
|
|
|
|
|
if( pCondition->completed( state ) )
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
|
|
|
if( pCondition->isLoopable() )
|
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
if( pCondition->loopReady( state, time ) )
|
|
|
|
pCondition->reset( state );
|
2024-05-09 16:17:46 +01:00
|
|
|
}
|
|
|
|
}
|
2024-05-10 22:46:33 +01:00
|
|
|
else if( pCondition->inProgress( state ) )
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
pCondition->update( state, pInstance, pack, time );
|
2024-05-09 16:17:46 +01:00
|
|
|
}
|
2024-05-10 22:46:33 +01:00
|
|
|
else if( pCondition->isConditionMet( state, pInstance, pack, time ) )
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
2024-05-10 22:46:33 +01:00
|
|
|
pCondition->execute( state, pInstance, pack, time );
|
2024-05-09 16:17:46 +01:00
|
|
|
m_phaseHistory.push( pCondition );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
// todo: actually handle solo stuff properly (or tie to zone director/content director at least)
|
2024-05-09 16:17:46 +01:00
|
|
|
class TimelinePack
|
|
|
|
{
|
|
|
|
TimelinePackType m_type{TimelinePackType::Solo};
|
|
|
|
std::vector< TimelineActor > m_actors;
|
|
|
|
std::string m_name;
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
uint64_t m_startTime{ 0 };
|
2024-05-09 16:17:46 +01:00
|
|
|
public:
|
|
|
|
TimelinePack() { }
|
|
|
|
TimelinePack( TimelinePackType type ) : m_type( type ) {}
|
2024-05-10 22:46:33 +01:00
|
|
|
|
|
|
|
void setStartTime( uint64_t time )
|
|
|
|
{
|
|
|
|
m_startTime = time;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t getStartTime() const
|
|
|
|
{
|
|
|
|
return m_startTime;
|
|
|
|
}
|
|
|
|
|
2024-05-10 15:48:42 +01:00
|
|
|
void update( InstanceContentPtr pInstance, uint64_t time )
|
2024-05-09 16:17:46 +01:00
|
|
|
{
|
|
|
|
for( auto& actor : m_actors )
|
2024-05-10 22:46:33 +01:00
|
|
|
actor.update( pInstance, *this, time );
|
2024-05-09 16:17:46 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
//
|
|
|
|
// Conditions
|
|
|
|
//
|
|
|
|
class ConditionHp : PhaseCondition
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
public:
|
2024-05-09 17:27:51 +01:00
|
|
|
uint32_t layoutId;
|
2024-04-29 16:51:28 +01:00
|
|
|
union
|
|
|
|
{
|
|
|
|
uint8_t val;
|
|
|
|
struct
|
|
|
|
{
|
2024-05-09 16:17:46 +01:00
|
|
|
uint8_t min, max;
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
} hp;
|
|
|
|
|
2024-05-10 17:42:39 +01:00
|
|
|
void from_json( nlohmann::json& json, Phase& phase, ConditionId conditionId,
|
2024-05-09 16:17:46 +01:00
|
|
|
const std::unordered_map< std::string, TimelineActor >& actors );
|
|
|
|
|
2024-05-10 22:46:33 +01:00
|
|
|
bool isConditionMet( ConditionState& state, InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time ) const override;
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
class ConditionDirectorVar : PhaseCondition
|
2024-04-29 16:51:28 +01:00
|
|
|
{
|
|
|
|
public:
|
2024-05-06 20:13:53 +01:00
|
|
|
union
|
|
|
|
{
|
|
|
|
struct
|
|
|
|
{
|
2024-05-10 17:42:39 +01:00
|
|
|
uint32_t index;
|
2024-05-06 20:13:53 +01:00
|
|
|
uint32_t value;
|
|
|
|
};
|
|
|
|
uint8_t seq;
|
2024-05-10 17:42:39 +01:00
|
|
|
uint8_t flags;
|
2024-05-06 20:13:53 +01:00
|
|
|
} param;
|
2024-05-09 16:17:46 +01:00
|
|
|
|
2024-05-10 17:42:39 +01:00
|
|
|
void from_json( nlohmann::json& json, Phase& phase, ConditionId conditionId );
|
2024-05-10 22:46:33 +01:00
|
|
|
bool isConditionMet( ConditionState& state, InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time ) const override;
|
2024-05-10 17:42:39 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
class ConditionEncounterTimeElapsed : PhaseCondition
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
uint64_t duration;
|
|
|
|
|
|
|
|
void from_json( nlohmann::json& json, Phase& phase, ConditionId conditionId );
|
2024-05-10 22:46:33 +01:00
|
|
|
bool isConditionMet( ConditionState& state, InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time ) const override;
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
class ConditionCombatState : PhaseCondition
|
|
|
|
{
|
|
|
|
public:
|
2024-05-09 17:27:51 +01:00
|
|
|
uint32_t layoutId;
|
|
|
|
CombatStateType combatState;
|
2024-05-09 16:17:46 +01:00
|
|
|
|
2024-05-10 17:42:39 +01:00
|
|
|
void from_json( nlohmann::json& json, Phase& phase, ConditionId conditionId, const std::unordered_map< std::string, TimelineActor >& actors );
|
2024-05-10 22:46:33 +01:00
|
|
|
bool isConditionMet( ConditionState& state, InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time ) const override;
|
2024-05-10 17:42:39 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
class ConditionBNpcFlags : PhaseCondition
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
uint32_t layoutId;
|
|
|
|
uint32_t flags;
|
|
|
|
|
|
|
|
void from_json( nlohmann::json& json, Phase& phase, ConditionId conditionId, const std::unordered_map< std::string, TimelineActor >& actors );
|
2024-05-10 22:46:33 +01:00
|
|
|
bool isConditionMet( ConditionState& state, InstanceContentPtr pInstance, TimelinePack& pack, uint64_t time ) const override;
|
2024-05-09 16:17:46 +01:00
|
|
|
};
|
2024-04-29 16:51:28 +01:00
|
|
|
|
|
|
|
public:
|
|
|
|
|
2024-05-09 16:17:46 +01:00
|
|
|
TimelinePack buildEncounterTimeline( uint32_t encounterId, bool reload = false );
|
2024-04-29 16:51:28 +01:00
|
|
|
};
|
|
|
|
}// namespace Sapphire
|