mirror of
https://github.com/DarkflameUniverse/DarkflameServer.git
synced 2025-01-11 15:27:07 +00:00
1250 lines
38 KiB
C++
1250 lines
38 KiB
C++
#include "PetComponent.h"
|
|
#include "GameMessages.h"
|
|
#include "BrickDatabase.h"
|
|
#include "CDClientDatabase.h"
|
|
#include "CDPetAbilitiesTable.h"
|
|
#include "CDPetComponentTable.h"
|
|
#include "ChatPackets.h"
|
|
#include "EntityManager.h"
|
|
#include "Character.h"
|
|
#include "CharacterComponent.h"
|
|
#include "InventoryComponent.h"
|
|
#include "Item.h"
|
|
#include "MissionComponent.h"
|
|
#include "SwitchComponent.h"
|
|
#include "DestroyableComponent.h"
|
|
#include "dpWorld.h"
|
|
#include "PetDigServer.h"
|
|
#include "ObjectIDManager.h"
|
|
#include "eUnequippableActiveType.h"
|
|
#include "eTerminateType.h"
|
|
#include "ePetTamingNotifyType.h"
|
|
#include "ePetAbilityType.h"
|
|
#include "eUseItemResponse.h"
|
|
#include "ePlayerFlag.h"
|
|
#include "eHelpType.h"
|
|
|
|
#include "Game.h"
|
|
#include "dConfig.h"
|
|
#include "dChatFilter.h"
|
|
#include "dZoneManager.h"
|
|
#include "Database.h"
|
|
#include "EntityInfo.h"
|
|
#include "eMissionTaskType.h"
|
|
#include "RenderComponent.h"
|
|
#include "eObjectBits.h"
|
|
#include "eGameMasterLevel.h"
|
|
#include "eMissionState.h"
|
|
#include "dNavMesh.h"
|
|
|
|
std::unordered_map<LOT, PetComponent::PuzzleData> PetComponent::buildCache{};
|
|
std::unordered_map<LWOOBJID, LWOOBJID> PetComponent::currentActivities{};
|
|
std::unordered_map<LWOOBJID, LWOOBJID> PetComponent::activePets{};
|
|
|
|
/**
|
|
* Maps all the pet lots to a flag indicating that the player has caught it. All basic pets have been guessed by ObjID
|
|
* while the faction ones could be checked using their respective missions.
|
|
*/
|
|
const std::map<LOT, int32_t> PetComponent::petFlags{
|
|
{ 3050, 801 }, // Elephant
|
|
{ 3054, 803 }, // Cat
|
|
{ 3195, 806 }, // Triceratops
|
|
{ 3254, 807 }, // Terrier
|
|
{ 3261, 811 }, // Skunk
|
|
{ 3672, 813 }, // Bunny
|
|
{ 3994, 814 }, // Crocodile
|
|
{ 5635, 815 }, // Doberman
|
|
{ 5636, 816 }, // Buffalo
|
|
{ 5637, 818 }, // Robot Dog
|
|
{ 5639, 819 }, // Red Dragon
|
|
{ 5640, 820 }, // Tortoise
|
|
{ 5641, 821 }, // Green Dragon
|
|
{ 5643, 822 }, // Panda, see mission 786
|
|
{ 5642, 823 }, // Mantis
|
|
{ 6720, 824 }, // Warthog
|
|
{ 3520, 825 }, // Lion, see mission 1318
|
|
{ 7638, 826 }, // Goat
|
|
{ 7694, 827 }, // Crab
|
|
{ 12294, 829 }, // Reindeer
|
|
{ 12431, 830 }, // Stegosaurus, see mission 1386
|
|
{ 12432, 831 }, // Saber cat, see mission 1389
|
|
{ 12433, 832 }, // Gryphon, see mission 1392
|
|
{ 12434, 833 }, // Alien, see mission 1188
|
|
// 834: unknown?, see mission 506, 688
|
|
{ 16210, 836 }, // Ninjago Earth Dragon, see mission 1836
|
|
{ 13067, 838 }, // Skeleton dragon
|
|
};
|
|
|
|
PetComponent::PetComponent(Entity* parentEntity, uint32_t componentId)
|
|
: Component{ parentEntity }
|
|
, m_Flags{ PetFlag::SPAWNING }
|
|
{
|
|
m_PetInfo = CDClientManager::GetTable<CDPetComponentTable>()->GetByID(componentId); // TODO: Make reference when safe
|
|
m_ComponentId = componentId;
|
|
m_StartPosition = m_Parent->GetPosition();
|
|
m_MovementAI = nullptr;
|
|
|
|
m_ReadyToInteract = false;
|
|
m_State = PetAiState::spawn;
|
|
SetIsHandlingInteraction(false);
|
|
|
|
std::string checkPreconditions = GeneralUtils::UTF16ToWTF8(parentEntity->GetVar<std::u16string>(u"CheckPrecondition"));
|
|
m_Preconditions =
|
|
checkPreconditions.empty() ? std::nullopt : std::make_optional(PreconditionExpression(checkPreconditions));
|
|
|
|
m_FollowRadius = 8.0f; //Game::zoneManager->GetPetFollowRadius(); // TODO: FIX THIS TO LOAD DYNAMICALLY
|
|
}
|
|
|
|
void PetComponent::Serialize(RakNet::BitStream& outBitStream, bool bIsInitialUpdate) {
|
|
const bool tamed = m_Owner != LWOOBJID_EMPTY;
|
|
|
|
constexpr bool isDirty = true;
|
|
outBitStream.Write(isDirty); // Always serialize as dirty for now
|
|
|
|
outBitStream.Write(m_Flags.Value());
|
|
outBitStream.Write(tamed ? m_Interaction.ability : ePetAbilityType::Invalid); // Something with the overhead icon?
|
|
|
|
const bool interacting = m_Interaction.obj != LWOOBJID_EMPTY;
|
|
|
|
outBitStream.Write(interacting);
|
|
if (interacting) {
|
|
outBitStream.Write(m_Interaction.obj);
|
|
}
|
|
|
|
outBitStream.Write(tamed);
|
|
if (tamed) {
|
|
outBitStream.Write(m_Owner);
|
|
}
|
|
|
|
if (bIsInitialUpdate) {
|
|
outBitStream.Write(tamed);
|
|
if (tamed) {
|
|
outBitStream.Write(m_ModerationStatus);
|
|
|
|
const auto nameData = GeneralUtils::UTF8ToUTF16(m_Name);
|
|
const auto ownerNameData = GeneralUtils::UTF8ToUTF16(m_OwnerName);
|
|
|
|
outBitStream.Write<uint8_t>(nameData.size());
|
|
for (const auto c : nameData) {
|
|
outBitStream.Write(c);
|
|
}
|
|
|
|
outBitStream.Write<uint8_t>(ownerNameData.size());
|
|
for (const auto c : ownerNameData) {
|
|
outBitStream.Write(c);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void PetComponent::OnUse(Entity* originator) {
|
|
LOG_DEBUG("PET USE!");
|
|
|
|
if (IsReadyToInteract()) {
|
|
switch (m_Interaction.ability) {
|
|
case ePetAbilityType::DigAtPosition: // Treasure dig
|
|
StartInteractTreasureDig();
|
|
break;
|
|
|
|
case ePetAbilityType::JumpOnObject: // Bouncer
|
|
StartInteractBouncer();
|
|
break;
|
|
}
|
|
} else {
|
|
StartTamingMinigame(originator);
|
|
}
|
|
}
|
|
|
|
void PetComponent::Update(float deltaTime) {
|
|
// Update timers
|
|
m_TimerBounce -= deltaTime;
|
|
|
|
if (m_Timer > 0) {
|
|
m_Timer -= deltaTime;
|
|
return;
|
|
}
|
|
|
|
// Remove "left behind" pets and handle failing pet taming minigame
|
|
if (m_Owner != LWOOBJID_EMPTY) {
|
|
const Entity* const owner = GetOwner();
|
|
if (!owner) {
|
|
m_Parent->Kill();
|
|
return;
|
|
}
|
|
} else {
|
|
ClientFailTamingMinigame(); // TODO: This is not despawning the built model correctly
|
|
}
|
|
|
|
if (m_Flags.Has<PetFlag::SPAWNING>()) OnSpawn();
|
|
|
|
// Handle pet AI states
|
|
switch (m_State) {
|
|
case PetAiState::idle:
|
|
Wander();
|
|
break;
|
|
|
|
case PetAiState::follow:
|
|
OnFollow(deltaTime);
|
|
break;
|
|
|
|
case PetAiState::goToObj:
|
|
if (m_MovementAI->AtFinalWaypoint()) {
|
|
LOG_DEBUG("Reached object!");
|
|
m_MovementAI->Stop();
|
|
SetPetAiState(PetAiState::interact);
|
|
} else {
|
|
m_Timer += 0.5f;
|
|
}
|
|
break;
|
|
|
|
case PetAiState::interact:
|
|
OnInteract();
|
|
break;
|
|
|
|
default:
|
|
LOG_DEBUG("Unknown state: %d!", m_Flags);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void PetComponent::StartTamingMinigame(Entity* originator) {
|
|
// The minigame logic beneath this comment should be rewritten... eventually
|
|
if (m_Owner != LWOOBJID_EMPTY) return;
|
|
|
|
if (m_Tamer != LWOOBJID_EMPTY) {
|
|
const auto* const tamer = Game::entityManager->GetEntity(m_Tamer);
|
|
|
|
if (tamer != nullptr) return;
|
|
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
}
|
|
|
|
auto* const inventoryComponent = originator->GetComponent<InventoryComponent>();
|
|
|
|
if (!inventoryComponent) return;
|
|
|
|
if (m_Preconditions.has_value() && !m_Preconditions->Check(originator, true)) return;
|
|
|
|
auto* const movementAIComponent = m_Parent->GetComponent<MovementAIComponent>();
|
|
|
|
if (movementAIComponent != nullptr) {
|
|
movementAIComponent->Stop();
|
|
}
|
|
|
|
inventoryComponent->DespawnPet();
|
|
|
|
const auto& cached = buildCache.find(m_Parent->GetLOT());
|
|
int32_t imaginationCost = 0;
|
|
|
|
std::string buildFile;
|
|
|
|
// It may make sense to move this minigame-specific logic into another file
|
|
if (cached == buildCache.end()) {
|
|
auto query = CDClientDatabase::CreatePreppedStmt(
|
|
"SELECT ValidPiecesLXF, PuzzleModelLot, Timelimit, NumValidPieces, imagCostPerBuild FROM TamingBuildPuzzles WHERE NPCLot = ?;");
|
|
query.bind(1, static_cast<int>(m_Parent->GetLOT()));
|
|
|
|
auto result = query.execQuery();
|
|
|
|
if (result.eof()) {
|
|
ChatPackets::SendSystemMessage(originator->GetSystemAddress(), u"Failed to find the puzzle minigame for this pet.");
|
|
|
|
return;
|
|
}
|
|
|
|
if (result.fieldIsNull("ValidPiecesLXF")) {
|
|
result.finalize();
|
|
|
|
return;
|
|
}
|
|
|
|
buildFile = std::string(result.getStringField("ValidPiecesLXF"));
|
|
|
|
PuzzleData data{
|
|
.puzzleModelLot = result.getIntField("PuzzleModelLot"),
|
|
.buildFile = buildFile,
|
|
.timeLimit = static_cast<float>(result.getFloatField("Timelimit")),
|
|
.imaginationCost = result.getIntField("imagCostPerBuild"),
|
|
.numValidPieces = result.getIntField("NumValidPieces")
|
|
};
|
|
if (data.timeLimit <= 0) data.timeLimit = 60;
|
|
imaginationCost = data.imaginationCost;
|
|
|
|
buildCache[m_Parent->GetLOT()] = data;
|
|
|
|
result.finalize();
|
|
} else {
|
|
buildFile = cached->second.buildFile;
|
|
imaginationCost = cached->second.imaginationCost;
|
|
}
|
|
|
|
const auto* const destroyableComponent = originator->GetComponent<DestroyableComponent>();
|
|
|
|
if (!destroyableComponent) return;
|
|
|
|
const auto imagination = destroyableComponent->GetImagination();
|
|
|
|
if (imagination < imaginationCost) return;
|
|
|
|
const auto& bricks = BrickDatabase::GetBricks(buildFile);
|
|
|
|
if (bricks.empty()) {
|
|
ChatPackets::SendSystemMessage(originator->GetSystemAddress(), u"Failed to load the puzzle minigame for this pet.");
|
|
LOG("Couldn't find %s for minigame!", buildFile.c_str());
|
|
|
|
return;
|
|
}
|
|
|
|
const auto petPosition = m_Parent->GetPosition();
|
|
|
|
const auto originatorPosition = originator->GetPosition();
|
|
|
|
m_Parent->SetRotation(NiQuaternion::LookAt(petPosition, originatorPosition));
|
|
|
|
float interactionDistance = m_Parent->GetVar<float>(u"interaction_distance");
|
|
|
|
if (interactionDistance <= 0) {
|
|
interactionDistance = 15;
|
|
}
|
|
|
|
auto position = originatorPosition;
|
|
|
|
NiPoint3 forward = NiQuaternion::LookAt(m_Parent->GetPosition(), originator->GetPosition()).GetForwardVector();
|
|
forward.y = 0;
|
|
|
|
if (dpWorld::IsLoaded()) {
|
|
NiPoint3 attempt = petPosition + forward * interactionDistance;
|
|
|
|
float y = dpWorld::GetNavMesh()->GetHeightAtPoint(attempt);
|
|
|
|
while (std::abs(y - petPosition.y) > 4 && interactionDistance > 10) {
|
|
const NiPoint3 forward = m_Parent->GetRotation().GetForwardVector();
|
|
|
|
attempt = originatorPosition + forward * interactionDistance;
|
|
|
|
y = dpWorld::GetNavMesh()->GetHeightAtPoint(attempt);
|
|
|
|
interactionDistance -= 0.5f;
|
|
}
|
|
|
|
position = attempt;
|
|
} else {
|
|
position = petPosition + forward * interactionDistance;
|
|
}
|
|
|
|
|
|
const auto rotation = NiQuaternion::LookAt(position, petPosition);
|
|
|
|
GameMessages::SendNotifyPetTamingMinigame(
|
|
originator->GetObjectID(),
|
|
m_Parent->GetObjectID(),
|
|
LWOOBJID_EMPTY,
|
|
true,
|
|
ePetTamingNotifyType::BEGIN,
|
|
petPosition,
|
|
position,
|
|
rotation,
|
|
UNASSIGNED_SYSTEM_ADDRESS
|
|
);
|
|
|
|
GameMessages::SendNotifyPetTamingMinigame(
|
|
m_Parent->GetObjectID(),
|
|
LWOOBJID_EMPTY,
|
|
originator->GetObjectID(),
|
|
true,
|
|
ePetTamingNotifyType::BEGIN,
|
|
petPosition,
|
|
position,
|
|
rotation,
|
|
UNASSIGNED_SYSTEM_ADDRESS
|
|
);
|
|
|
|
GameMessages::SendNotifyPetTamingPuzzleSelected(originator->GetObjectID(), bricks, originator->GetSystemAddress());
|
|
|
|
m_Tamer = originator->GetObjectID();
|
|
m_Flags.Set<PetFlag::IDLE, PetFlag::UNKNOWN4>();
|
|
|
|
currentActivities.insert_or_assign(m_Tamer, m_Parent->GetObjectID());
|
|
|
|
// Notify the start of a pet taming minigame
|
|
m_Parent->GetScript()->OnNotifyPetTamingMinigame(m_Parent, originator, ePetTamingNotifyType::BEGIN);
|
|
}
|
|
|
|
void PetComponent::TryBuild(uint32_t numBricks, bool clientFailed) {
|
|
if (m_Tamer == LWOOBJID_EMPTY) return;
|
|
|
|
auto* const tamer = Game::entityManager->GetEntity(m_Tamer);
|
|
|
|
if (!tamer) {
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
|
|
return;
|
|
}
|
|
|
|
const auto& cached = buildCache.find(m_Parent->GetLOT());
|
|
|
|
if (cached == buildCache.end()) return;
|
|
|
|
auto* const destroyableComponent = tamer->GetComponent<DestroyableComponent>();
|
|
|
|
if (!destroyableComponent) return;
|
|
|
|
auto imagination = destroyableComponent->GetImagination();
|
|
|
|
imagination -= cached->second.imaginationCost;
|
|
|
|
destroyableComponent->SetImagination(imagination);
|
|
|
|
Game::entityManager->SerializeEntity(tamer);
|
|
|
|
if (clientFailed) {
|
|
if (imagination < cached->second.imaginationCost) {
|
|
ClientFailTamingMinigame();
|
|
}
|
|
} else {
|
|
m_Timer = 0;
|
|
}
|
|
|
|
if (numBricks == 0) return;
|
|
|
|
GameMessages::SendPetTamingTryBuildResult(m_Tamer, !clientFailed, numBricks, tamer->GetSystemAddress());
|
|
}
|
|
|
|
void PetComponent::NotifyTamingBuildSuccess(const NiPoint3 position) {
|
|
if (m_Tamer == LWOOBJID_EMPTY) return;
|
|
|
|
auto* const tamer = Game::entityManager->GetEntity(m_Tamer);
|
|
|
|
if (!tamer) {
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
|
|
return;
|
|
}
|
|
|
|
const auto& cached = buildCache.find(m_Parent->GetLOT());
|
|
|
|
if (cached == buildCache.end()) {
|
|
return;
|
|
}
|
|
|
|
GameMessages::SendPlayFXEffect(tamer, -1, u"petceleb", "", LWOOBJID_EMPTY, 1, 1, true);
|
|
RenderComponent::PlayAnimation(tamer, u"rebuild-celebrate");
|
|
|
|
EntityInfo info{};
|
|
info.lot = cached->second.puzzleModelLot;
|
|
info.pos = position;
|
|
info.rot = NiQuaternionConstant::IDENTITY;
|
|
info.spawnerID = tamer->GetObjectID();
|
|
|
|
auto* const modelEntity = Game::entityManager->CreateEntity(info);
|
|
|
|
m_ModelId = modelEntity->GetObjectID();
|
|
|
|
Game::entityManager->ConstructEntity(modelEntity);
|
|
|
|
GameMessages::SendNotifyTamingModelLoadedOnServer(m_Tamer, tamer->GetSystemAddress());
|
|
|
|
GameMessages::SendPetResponse(m_Tamer, m_Parent->GetObjectID(), 0, 10, 0, tamer->GetSystemAddress());
|
|
|
|
auto* const inventoryComponent = tamer->GetComponent<InventoryComponent>();
|
|
if (!inventoryComponent) return;
|
|
|
|
LWOOBJID petSubKey = ObjectIDManager::GenerateRandomObjectID();
|
|
|
|
GeneralUtils::SetBit(petSubKey, eObjectBits::CHARACTER);
|
|
GeneralUtils::SetBit(petSubKey, eObjectBits::PERSISTENT);
|
|
|
|
m_DatabaseId = petSubKey;
|
|
|
|
std::string petName = tamer->GetCharacter()->GetName();
|
|
petName += "'s Pet";
|
|
|
|
GameMessages::SendAddPetToPlayer(m_Tamer, 0, GeneralUtils::UTF8ToUTF16(petName), petSubKey, m_Parent->GetLOT(), tamer->GetSystemAddress());
|
|
|
|
GameMessages::SendRegisterPetID(m_Tamer, m_Parent->GetObjectID(), tamer->GetSystemAddress());
|
|
|
|
GameMessages::SendRegisterPetDBID(m_Tamer, petSubKey, tamer->GetSystemAddress());
|
|
|
|
inventoryComponent->AddItem(m_Parent->GetLOT(), 1, eLootSourceType::ACTIVITY, eInventoryType::MODELS, {}, LWOOBJID_EMPTY, true, false, petSubKey);
|
|
|
|
auto* const item = inventoryComponent->FindItemBySubKey(petSubKey, MODELS);
|
|
if (!item) return;
|
|
|
|
DatabasePet databasePet{};
|
|
|
|
databasePet.lot = m_Parent->GetLOT();
|
|
databasePet.moderationState = 1;
|
|
databasePet.name = petName;
|
|
|
|
inventoryComponent->SetDatabasePet(petSubKey, databasePet);
|
|
|
|
Activate(item, false, true);
|
|
|
|
m_Timer = 0;
|
|
|
|
GameMessages::SendNotifyPetTamingMinigame(
|
|
m_Tamer,
|
|
LWOOBJID_EMPTY,
|
|
LWOOBJID_EMPTY,
|
|
false,
|
|
ePetTamingNotifyType::NAMINGPET,
|
|
NiPoint3Constant::ZERO,
|
|
NiPoint3Constant::ZERO,
|
|
NiQuaternionConstant::IDENTITY,
|
|
UNASSIGNED_SYSTEM_ADDRESS
|
|
);
|
|
|
|
// Triggers the catch a pet missions
|
|
if (petFlags.find(m_Parent->GetLOT()) != petFlags.end()) {
|
|
tamer->GetCharacter()->SetPlayerFlag(petFlags.at(m_Parent->GetLOT()), true);
|
|
}
|
|
|
|
auto* const missionComponent = tamer->GetComponent<MissionComponent>();
|
|
|
|
if (missionComponent != nullptr) {
|
|
missionComponent->Progress(eMissionTaskType::PET_TAMING, m_Parent->GetLOT());
|
|
}
|
|
|
|
m_Flags.Reset<PetFlag::IDLE>();
|
|
|
|
auto* const characterComponent = tamer->GetComponent<CharacterComponent>();
|
|
if (characterComponent != nullptr) {
|
|
characterComponent->UpdatePlayerStatistic(PetsTamed);
|
|
}
|
|
}
|
|
|
|
void PetComponent::RequestSetPetName(const std::u16string& name) {
|
|
if (m_Tamer == LWOOBJID_EMPTY) {
|
|
if (m_Owner != LWOOBJID_EMPTY) {
|
|
auto* owner = GetOwner();
|
|
|
|
m_ModerationStatus = 1; // Pending
|
|
m_Name = "";
|
|
|
|
//Save our pet's new name to the db:
|
|
SetPetNameForModeration(GeneralUtils::UTF16ToWTF8(name));
|
|
|
|
GameMessages::SendSetPetName(m_Owner, GeneralUtils::UTF8ToUTF16(m_Name), m_DatabaseId, owner->GetSystemAddress());
|
|
GameMessages::SendSetPetNameModerated(m_Owner, m_DatabaseId, m_ModerationStatus, owner->GetSystemAddress());
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
auto* tamer = Game::entityManager->GetEntity(m_Tamer);
|
|
|
|
if (tamer == nullptr) {
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
|
|
return;
|
|
}
|
|
|
|
LOG("Got set pet name (%s)", GeneralUtils::UTF16ToWTF8(name).c_str());
|
|
|
|
auto* inventoryComponent = tamer->GetComponent<InventoryComponent>();
|
|
if (!inventoryComponent) return;
|
|
|
|
m_ModerationStatus = 1; // Pending
|
|
m_Name = "";
|
|
|
|
//Save our pet's new name to the db:
|
|
SetPetNameForModeration(GeneralUtils::UTF16ToWTF8(name));
|
|
|
|
Game::entityManager->SerializeEntity(m_Parent);
|
|
|
|
std::u16string u16name = GeneralUtils::UTF8ToUTF16(m_Name);
|
|
std::u16string u16ownerName = GeneralUtils::UTF8ToUTF16(m_OwnerName);
|
|
GameMessages::SendSetPetName(m_Tamer, u16name, m_DatabaseId, tamer->GetSystemAddress());
|
|
GameMessages::SendSetPetName(m_Tamer, u16name, LWOOBJID_EMPTY, tamer->GetSystemAddress());
|
|
GameMessages::SendPetNameChanged(m_Parent->GetObjectID(), m_ModerationStatus, u16name, u16ownerName, UNASSIGNED_SYSTEM_ADDRESS);
|
|
GameMessages::SendSetPetNameModerated(m_Tamer, m_DatabaseId, m_ModerationStatus, tamer->GetSystemAddress());
|
|
|
|
GameMessages::SendNotifyPetTamingMinigame(
|
|
m_Tamer,
|
|
m_Parent->GetObjectID(),
|
|
m_Tamer,
|
|
false,
|
|
ePetTamingNotifyType::SUCCESS,
|
|
NiPoint3Constant::ZERO,
|
|
NiPoint3Constant::ZERO,
|
|
NiQuaternionConstant::IDENTITY,
|
|
UNASSIGNED_SYSTEM_ADDRESS
|
|
);
|
|
|
|
GameMessages::SendTerminateInteraction(m_Tamer, eTerminateType::FROM_INTERACTION, m_Parent->GetObjectID());
|
|
|
|
auto* const modelEntity = Game::entityManager->GetEntity(m_ModelId);
|
|
|
|
if (modelEntity != nullptr) {
|
|
modelEntity->Smash(m_Tamer);
|
|
}
|
|
|
|
currentActivities.erase(m_Tamer);
|
|
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
|
|
// Notify the end of a pet taming minigame
|
|
m_Parent->GetScript()->OnNotifyPetTamingMinigame(m_Parent, tamer, ePetTamingNotifyType::SUCCESS);
|
|
}
|
|
|
|
void PetComponent::ClientExitTamingMinigame(bool voluntaryExit) {
|
|
if (m_Tamer == LWOOBJID_EMPTY) return;
|
|
|
|
auto* const tamer = Game::entityManager->GetEntity(m_Tamer);
|
|
|
|
if (!tamer) {
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
|
|
return;
|
|
}
|
|
|
|
GameMessages::SendNotifyPetTamingMinigame(
|
|
m_Tamer,
|
|
m_Parent->GetObjectID(),
|
|
m_Tamer,
|
|
false,
|
|
ePetTamingNotifyType::QUIT,
|
|
NiPoint3Constant::ZERO,
|
|
NiPoint3Constant::ZERO,
|
|
NiQuaternionConstant::IDENTITY,
|
|
UNASSIGNED_SYSTEM_ADDRESS
|
|
);
|
|
|
|
GameMessages::SendNotifyTamingModelLoadedOnServer(m_Tamer, tamer->GetSystemAddress());
|
|
|
|
GameMessages::SendTerminateInteraction(m_Tamer, eTerminateType::FROM_INTERACTION, m_Parent->GetObjectID());
|
|
|
|
currentActivities.erase(m_Tamer);
|
|
|
|
m_Flags.Reset<PetFlag::TAMEABLE>();
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
m_Timer = 0;
|
|
|
|
Game::entityManager->SerializeEntity(m_Parent);
|
|
|
|
// Notify the end of a pet taming minigame
|
|
m_Parent->GetScript()->OnNotifyPetTamingMinigame(m_Parent, tamer, ePetTamingNotifyType::QUIT);
|
|
}
|
|
|
|
void PetComponent::StartTimer() {
|
|
const auto& cached = buildCache.find(m_Parent->GetLOT());
|
|
|
|
if (cached == buildCache.end()) {
|
|
return;
|
|
}
|
|
|
|
m_Timer = cached->second.timeLimit;
|
|
}
|
|
|
|
void PetComponent::ClientFailTamingMinigame() {
|
|
if (m_Tamer == LWOOBJID_EMPTY) return;
|
|
|
|
auto* const tamer = Game::entityManager->GetEntity(m_Tamer);
|
|
|
|
if (!tamer) {
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
|
|
return;
|
|
}
|
|
|
|
GameMessages::SendNotifyPetTamingMinigame(
|
|
m_Tamer,
|
|
m_Parent->GetObjectID(),
|
|
m_Tamer,
|
|
false,
|
|
ePetTamingNotifyType::FAILED,
|
|
NiPoint3Constant::ZERO,
|
|
NiPoint3Constant::ZERO,
|
|
NiQuaternionConstant::IDENTITY,
|
|
UNASSIGNED_SYSTEM_ADDRESS
|
|
);
|
|
|
|
GameMessages::SendNotifyTamingModelLoadedOnServer(m_Tamer, tamer->GetSystemAddress());
|
|
|
|
GameMessages::SendTerminateInteraction(m_Tamer, eTerminateType::FROM_INTERACTION, m_Parent->GetObjectID());
|
|
|
|
currentActivities.erase(m_Tamer);
|
|
|
|
m_Flags.Reset<PetFlag::TAMEABLE>();
|
|
m_Tamer = LWOOBJID_EMPTY;
|
|
m_Timer = 0;
|
|
|
|
Game::entityManager->SerializeEntity(m_Parent);
|
|
|
|
// Notify the end of a pet taming minigame
|
|
m_Parent->GetScript()->OnNotifyPetTamingMinigame(m_Parent, tamer, ePetTamingNotifyType::FAILED);
|
|
}
|
|
|
|
void PetComponent::Wander() {
|
|
if (!m_MovementAI->AtFinalWaypoint()) return;
|
|
|
|
m_MovementAI->SetHaltDistance(0);
|
|
|
|
const auto& info = m_MovementAI->GetInfo();
|
|
|
|
const auto div = static_cast<int>(info.wanderDelayMax);
|
|
m_Timer = (div == 0 ? 0 : GeneralUtils::GenerateRandomNumber<int>(0, div)) + info.wanderDelayMin; //set a random timer to stay put.
|
|
|
|
const float radius = info.wanderRadius * sqrt(static_cast<double>(GeneralUtils::GenerateRandomNumber<float>(0, 1))); //our wander radius + a bit of random range
|
|
const float theta = ((static_cast<double>(GeneralUtils::GenerateRandomNumber<float>(0, 1)) * 2 * PI));
|
|
|
|
const NiPoint3 delta =
|
|
{
|
|
radius * cos(theta),
|
|
0,
|
|
radius * sin(theta)
|
|
};
|
|
|
|
auto destination = m_StartPosition + delta;
|
|
|
|
if (dpWorld::IsLoaded()) {
|
|
destination.y = dpWorld::GetNavMesh()->GetHeightAtPoint(destination);
|
|
}
|
|
|
|
if (Vector3::DistanceSquared(destination, m_MovementAI->GetParent()->GetPosition()) < 2 * 2) {
|
|
m_MovementAI->Stop();
|
|
|
|
return;
|
|
}
|
|
|
|
m_MovementAI->SetMaxSpeed(m_PetInfo.sprintSpeed);
|
|
|
|
m_MovementAI->SetDestination(destination);
|
|
|
|
m_Timer += (m_MovementAI->GetParent()->GetPosition().x - destination.x) / m_PetInfo.sprintSpeed;
|
|
}
|
|
|
|
void PetComponent::OnSpawn() {
|
|
m_MovementAI = m_Parent->GetComponent<MovementAIComponent>();
|
|
|
|
if (m_StartPosition == NiPoint3Constant::ZERO) {
|
|
m_StartPosition = m_Parent->GetPosition();
|
|
}
|
|
|
|
if (m_Owner != LWOOBJID_EMPTY) {
|
|
m_Parent->SetOwnerOverride(m_Owner);
|
|
m_MovementAI->SetMaxSpeed(m_PetInfo.sprintSpeed);
|
|
m_MovementAI->SetHaltDistance(m_FollowRadius);
|
|
SetPetAiState(PetAiState::follow);
|
|
} else {
|
|
m_Flags.Set<PetFlag::TAMEABLE>();
|
|
SetPetAiState(PetAiState::idle);
|
|
}
|
|
|
|
m_Flags.Set<PetFlag::IDLE>();
|
|
m_Flags.Unset<PetFlag::SPAWNING>();
|
|
Game::entityManager->SerializeEntity(m_Parent);
|
|
}
|
|
|
|
void PetComponent::OnFollow(const float deltaTime) {
|
|
Entity* owner = GetOwner();
|
|
if (!owner) return;
|
|
|
|
const NiPoint3 ownerPos = owner->GetPosition();
|
|
|
|
// Find interactions
|
|
SwitchComponent* closestSwitch = SwitchComponent::GetClosestSwitch(ownerPos);
|
|
if (closestSwitch != nullptr && !closestSwitch->GetActive()) {
|
|
const NiPoint3 switchPos = closestSwitch->GetParentEntity()->GetPosition();
|
|
const LWOOBJID switchID = closestSwitch->GetParentEntity()->GetObjectID();
|
|
const float distance = Vector3::DistanceSquared(ownerPos, switchPos);
|
|
if (distance < 16 * 16) {
|
|
StartInteract(switchPos, PetInteractType::bouncer, switchID);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Determine if the "Lost Tags" mission has been completed and digging has been unlocked
|
|
const auto* const missionComponent = owner->GetComponent<MissionComponent>();
|
|
if (!missionComponent) return;
|
|
const bool digUnlocked = missionComponent->GetMissionState(842) == eMissionState::COMPLETE;
|
|
|
|
const Entity* closestTreasure = PetDigServer::GetClosestTresure(ownerPos);
|
|
const bool nonDragonForBone = closestTreasure->GetLOT() == 12192 && m_Parent->GetLOT() != 13067;
|
|
if (!nonDragonForBone && closestTreasure != nullptr && digUnlocked) {
|
|
const NiPoint3 treasurePos = closestTreasure->GetPosition();
|
|
const float distance = Vector3::DistanceSquared(ownerPos, treasurePos);
|
|
if (distance < 16 * 16) {
|
|
StartInteract(treasurePos, PetInteractType::treasure, m_Owner);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Handle actual following logic
|
|
const NiPoint3 currentPos = m_MovementAI->GetParent()->GetPosition();
|
|
const float distanceToOwner = Vector3::DistanceSquared(currentPos, ownerPos);
|
|
|
|
// If the player's position is within range, stop moving
|
|
if (distanceToOwner <= m_FollowRadius * m_FollowRadius) {
|
|
m_MovementAI->Stop();
|
|
} else { // Chase the player's new position
|
|
m_MovementAI->SetDestination(ownerPos);
|
|
}
|
|
|
|
// Teleporting logic
|
|
if (distanceToOwner > 50 * 50 || m_TimerAway > 5) {
|
|
m_MovementAI->Warp(ownerPos);
|
|
|
|
m_Timer = 1;
|
|
m_TimerAway = 0;
|
|
|
|
return;
|
|
} else if (distanceToOwner > 15 * 15 || std::abs(ownerPos.y - currentPos.y) >= 3) {
|
|
m_TimerAway += deltaTime;
|
|
}
|
|
|
|
m_Timer += 0.5f;
|
|
}
|
|
|
|
void PetComponent::OnInteract() {
|
|
Entity* owner = GetOwner();
|
|
if (!owner) return;
|
|
|
|
const NiPoint3 ownerPos = owner->GetPosition();
|
|
const NiPoint3 currentPos = m_MovementAI->GetParent()->GetPosition();
|
|
const float distanceFromOwner = Vector3::DistanceSquared(ownerPos, currentPos);
|
|
|
|
if (distanceFromOwner > 25 * 25) {
|
|
LOG_DEBUG("Disengaging from object interaction due to player distance.");
|
|
StopInteract();
|
|
return;
|
|
}
|
|
|
|
switch (m_Interaction.type) {
|
|
case PetInteractType::bouncer:
|
|
if (IsReadyToInteract()) HandleInteractBouncer();
|
|
else SetupInteractBouncer();
|
|
break;
|
|
|
|
case PetInteractType::treasure:
|
|
if (IsReadyToInteract()) HandleInteractTreasureDig();
|
|
else SetupInteractTreasureDig();
|
|
break;
|
|
|
|
default:
|
|
LOG_DEBUG("INTERACT = NONE! RETURNING!");
|
|
StopInteract();
|
|
m_Timer += 0.5f;
|
|
break;
|
|
}
|
|
}
|
|
|
|
void PetComponent::StartInteract(const NiPoint3 position, const PetInteractType interactionType, const LWOOBJID interactID) {
|
|
m_Interaction.obj = interactID; // TODO: Check if this should be serialized for goToObj
|
|
m_Interaction.type = interactionType;
|
|
m_Interaction.ability = ePetAbilityType::GoToObject;
|
|
SetPetAiState(PetAiState::goToObj);
|
|
m_MovementAI->SetMaxSpeed(m_PetInfo.runSpeed);
|
|
m_MovementAI->SetHaltDistance(0.0f);
|
|
m_MovementAI->SetDestination(position);
|
|
LOG_DEBUG("Starting interaction!");
|
|
Game::entityManager->SerializeEntity(m_Parent);
|
|
}
|
|
|
|
void PetComponent::StopInteract(bool bDontSerialize) {
|
|
Entity* const owner = GetOwner();
|
|
if (!owner) return;
|
|
|
|
constexpr auto petAbility = ePetAbilityType::Invalid;
|
|
|
|
m_Interaction.obj = LWOOBJID_EMPTY;
|
|
m_Interaction.type = PetInteractType::none;
|
|
m_Interaction.ability = petAbility;
|
|
SetPetAiState(PetAiState::follow);
|
|
m_Flags.Reset<PetFlag::IDLE>();
|
|
SetIsReadyToInteract(false);
|
|
SetIsHandlingInteraction(false); // Needed?
|
|
m_MovementAI->SetMaxSpeed(m_PetInfo.sprintSpeed);
|
|
m_MovementAI->SetHaltDistance(m_FollowRadius);
|
|
LOG_DEBUG("Stopping interaction!");
|
|
|
|
if (!bDontSerialize) {
|
|
Game::entityManager->SerializeEntity(m_Parent);
|
|
}
|
|
|
|
GameMessages::SendShowPetActionButton(m_Owner, petAbility, false, owner->GetSystemAddress()); // Needed?
|
|
}
|
|
|
|
void PetComponent::SetupInteractBouncer() {
|
|
const auto* const owner = GetOwner();
|
|
if (!owner) return;
|
|
|
|
LOG_DEBUG("Setting up bouncer interaction!");
|
|
SetIsReadyToInteract(true);
|
|
constexpr auto petAbility = ePetAbilityType::JumpOnObject;
|
|
|
|
m_Interaction.ability = petAbility;
|
|
m_Flags.Unset<PetFlag::IDLE>();
|
|
m_Flags.Set<PetFlag::ON_SWITCH, PetFlag::NOT_WAITING>(); // TODO: Double-check this is the right flag being set
|
|
LOG_DEBUG("m_Flags = %d", m_Flags);
|
|
Game::entityManager->SerializeEntity(m_Parent); // TODO: Double-check pet packet captures
|
|
|
|
const auto sysAddr = owner->GetSystemAddress();
|
|
GameMessages::SendHelp(m_Owner, eHelpType::PR_BOUNCER_TUTORIAL_03, sysAddr);
|
|
GameMessages::SendShowPetActionButton(m_Owner, petAbility, true, sysAddr);
|
|
|
|
SwitchComponent* closestSwitch = SwitchComponent::GetClosestSwitch(m_MovementAI->GetDestination()); // TODO: Find a better way to do this
|
|
closestSwitch->EntityEnter(m_Parent);
|
|
|
|
m_Timer += 0.5f;
|
|
}
|
|
|
|
void PetComponent::StartInteractBouncer() {
|
|
Entity* const user = GetOwner();
|
|
if (IsHandlingInteraction() || !user) return;
|
|
|
|
auto* const destroyableComponent = user->GetComponent<DestroyableComponent>();
|
|
if (!destroyableComponent) return;
|
|
|
|
auto imagination = destroyableComponent->GetImagination();
|
|
const auto imaginationCost =
|
|
CDClientManager::GetTable<CDPetAbilitiesTable>()->GetByID(ePetAbilityType::JumpOnObject).imaginationCost;
|
|
|
|
if (imagination < imaginationCost) {
|
|
//GameMessages::SendHelp(user->GetObjectID(), eHelpType::PR_NEED_IMAGINATION, user->GetSystemAddress()); // Check if right message!
|
|
return;
|
|
}
|
|
GameMessages::SendHelp(user->GetObjectID(), eHelpType::PR_TOOLTIP_1ST_PET_JUMPED_ON_SWITCH, user->GetSystemAddress());
|
|
|
|
GameMessages::SendShowPetActionButton(m_Owner, ePetAbilityType::Invalid, false, user->GetSystemAddress());
|
|
|
|
imagination -= imaginationCost;
|
|
destroyableComponent->SetImagination(imagination);
|
|
Game::entityManager->SerializeEntity(user);
|
|
|
|
// THIS IS ALL BAD, BAD, BAD! FIX IT, ME! >:(
|
|
SetIsHandlingInteraction(true);
|
|
SwitchComponent* closestSwitch = SwitchComponent::GetClosestSwitch(m_MovementAI->GetDestination()); // TODO: Find a better way to do this
|
|
closestSwitch->EntityEnter(m_Parent);
|
|
}
|
|
|
|
void PetComponent::HandleInteractBouncer() {
|
|
if (IsHandlingInteraction()) {
|
|
auto* const petSwitch = SwitchComponent::GetClosestSwitch(m_MovementAI->GetDestination()); // TODO: Find a better way to do this
|
|
if (!petSwitch) return;
|
|
|
|
auto* const petSwitchEntity = petSwitch->GetParentEntity();
|
|
if (!petSwitchEntity) return;
|
|
|
|
m_Parent->AddCallbackTimer(1.0f, [this, petSwitch, petSwitchEntity]() {
|
|
auto* const bouncerComp = petSwitch->GetPetBouncer();
|
|
const auto bouncerCompPos = bouncerComp->GetParentEntity()->GetPosition();
|
|
const auto bouncerId = bouncerComp->GetParentEntity()->GetObjectID();
|
|
|
|
bouncerComp->SetPetBouncerEnabled(true);
|
|
GameMessages::SendRequestClientBounce(bouncerId, this->GetOwnerId(), NiPoint3Constant::ZERO, NiPoint3Constant::ZERO, bouncerId, true, false, UNASSIGNED_SYSTEM_ADDRESS); //TODO: Check packet captures!!
|
|
bouncerComp->SetPetBouncerEnabled(false);
|
|
RenderComponent::PlayAnimation(petSwitchEntity, u"up");
|
|
});
|
|
|
|
RenderComponent::PlayAnimation(petSwitchEntity, u"launch"); //u"engaged"); //TODO: Check if the timing on this is right
|
|
// TODO: Need to freeze player movement until the bounce begins!
|
|
|
|
Command(NiPoint3Constant::ZERO, LWOOBJID_EMPTY, 1, GeneralUtils::ToUnderlying(PetEmote::ActivateSwitch), true); // Plays 'jump on switch' animation
|
|
StopInteract();
|
|
}
|
|
m_Timer += 0.5f;
|
|
}
|
|
|
|
void PetComponent::SetupInteractTreasureDig() {
|
|
const auto* owner = GetOwner();
|
|
if (!owner) return;
|
|
|
|
LOG_DEBUG("Setting up dig interaction!");
|
|
SetIsReadyToInteract(true);
|
|
constexpr auto petAbility = ePetAbilityType::DigAtPosition;
|
|
|
|
m_Interaction.ability = petAbility;
|
|
m_Flags.Unset<PetFlag::IDLE>();
|
|
m_Flags.Set<PetFlag::ON_SWITCH, PetFlag::NOT_WAITING>(); // TODO: Double-check this is the right flag being set
|
|
LOG_DEBUG("m_Flags = %d", m_Flags);
|
|
Game::entityManager->SerializeEntity(m_Parent); // TODO: Double-check pet packet captures
|
|
|
|
const auto sysAddr = owner->GetSystemAddress();
|
|
GameMessages::SendHelp(m_Owner, eHelpType::PR_DIG_TUTORIAL_01, sysAddr);
|
|
GameMessages::SendShowPetActionButton(m_Owner, petAbility, true, sysAddr);
|
|
|
|
m_Timer += 0.5f;
|
|
}
|
|
|
|
void PetComponent::StartInteractTreasureDig() {
|
|
Entity* const user = GetOwner();
|
|
if (IsHandlingInteraction() || !user) return;
|
|
|
|
auto* const destroyableComponent = user->GetComponent<DestroyableComponent>();
|
|
if (!destroyableComponent) return;
|
|
|
|
auto imagination = destroyableComponent->GetImagination();
|
|
const auto imaginationCost =
|
|
CDClientManager::GetTable<CDPetAbilitiesTable>()->GetByID(ePetAbilityType::DigAtPosition).imaginationCost;
|
|
|
|
if (imagination < imaginationCost) {
|
|
//GameMessages::SendHelp(user->GetObjectID(), eHelpType::PR_NEED_IMAGINATION, user->GetSystemAddress()); // Check if right message!
|
|
return;
|
|
}
|
|
|
|
GameMessages::SendShowPetActionButton(m_Owner, ePetAbilityType::Invalid, false, user->GetSystemAddress());
|
|
|
|
imagination -= imaginationCost;
|
|
destroyableComponent->SetImagination(imagination);
|
|
Game::entityManager->SerializeEntity(user);
|
|
|
|
SetIsHandlingInteraction(true);
|
|
m_Flags.Unset<PetFlag::ON_SWITCH, PetFlag::NOT_WAITING>(); // TODO: FIND THE CORRECT STATUS TO USE HERE
|
|
m_Flags.Set<PetFlag::IDLE>();
|
|
LOG_DEBUG("StartInteractTreasureDig() m_Flags = %d", m_Flags);
|
|
Game::entityManager->SerializeEntity(m_Parent);
|
|
|
|
Command(NiPoint3Constant::ZERO, LWOOBJID_EMPTY, 1, GeneralUtils::ToUnderlying(PetEmote::DigTreasure), true); // Plays 'dig' animation
|
|
m_Timer = 2.0f;
|
|
}
|
|
|
|
void PetComponent::HandleInteractTreasureDig() {
|
|
if (IsHandlingInteraction()) {
|
|
auto* const owner = GetOwner();
|
|
if (!owner) return;
|
|
|
|
auto* const treasure = PetDigServer::GetClosestTresure(m_MovementAI->GetDestination()); // TODO: Find a better way to do this
|
|
treasure->Smash(m_Parent->GetObjectID());
|
|
|
|
GameMessages::SendHelp(m_Owner, eHelpType::PR_DIG_TUTORIAL_03, owner->GetSystemAddress());
|
|
|
|
LOG_DEBUG("Pet dig completed!");
|
|
StopInteract(true); //TODO: This may not be totally consistent with live behavior, where the pet seems to stay near the dig and not immediately follow
|
|
|
|
return;
|
|
}
|
|
|
|
if (m_TimerBounce <= 0.0f) {
|
|
Command(NiPoint3Constant::ZERO, LWOOBJID_EMPTY, 1, GeneralUtils::ToUnderlying(PetEmote::Bounce), true); // Plays 'bounce' animation
|
|
m_TimerBounce = 1.0f;
|
|
}
|
|
|
|
m_Timer += 0.5f;
|
|
}
|
|
|
|
void PetComponent::Activate(Item* item, bool registerPet, bool fromTaming) { // TODO: Offset spawn position so it's not on top of player char
|
|
AddDrainImaginationTimer(item, fromTaming);
|
|
|
|
m_ItemId = item->GetId();
|
|
m_DatabaseId = item->GetSubKey();
|
|
|
|
auto* const inventoryComponent = item->GetInventory()->GetComponent();
|
|
|
|
if (!inventoryComponent) return;
|
|
|
|
inventoryComponent->DespawnPet();
|
|
|
|
m_Owner = inventoryComponent->GetParent()->GetObjectID();
|
|
|
|
auto* const owner = GetOwner();
|
|
|
|
if (!owner) return;
|
|
m_Flags.Set<PetFlag::SPAWNING>();
|
|
|
|
auto databaseData = inventoryComponent->GetDatabasePet(m_DatabaseId);
|
|
|
|
m_ModerationStatus = databaseData.moderationState;
|
|
|
|
bool updatedModerationStatus = false;
|
|
|
|
//Load mod status from db:
|
|
if (m_ModerationStatus != 2) {
|
|
LoadPetNameFromModeration();
|
|
|
|
databaseData.name = m_Name;
|
|
databaseData.moderationState = m_ModerationStatus;
|
|
|
|
inventoryComponent->SetDatabasePet(m_DatabaseId, databaseData);
|
|
|
|
updatedModerationStatus = true;
|
|
} else {
|
|
m_Name = databaseData.name;
|
|
}
|
|
|
|
m_OwnerName = owner->GetCharacter()->GetName();
|
|
|
|
if (updatedModerationStatus) {
|
|
GameMessages::SendSetPetName(m_Owner, GeneralUtils::UTF8ToUTF16(m_Name), m_DatabaseId, owner->GetSystemAddress());
|
|
GameMessages::SendSetPetNameModerated(m_Owner, m_DatabaseId, m_ModerationStatus, owner->GetSystemAddress());
|
|
}
|
|
|
|
GameMessages::SendMarkInventoryItemAsActive(m_Owner, true, eUnequippableActiveType::PET, m_ItemId, GetOwner()->GetSystemAddress());
|
|
|
|
activePets.emplace(m_Owner, m_Parent->GetObjectID());
|
|
|
|
Game::entityManager->SerializeEntity(m_Parent);
|
|
|
|
owner->GetCharacter()->SetPlayerFlag(ePlayerFlag::FIRST_MANUAL_PET_HIBERNATE, true);
|
|
|
|
if (registerPet) {
|
|
GameMessages::SendAddPetToPlayer(m_Owner, 0, GeneralUtils::UTF8ToUTF16(m_Name), m_DatabaseId, m_Parent->GetLOT(), owner->GetSystemAddress());
|
|
|
|
GameMessages::SendRegisterPetID(m_Owner, m_Parent->GetObjectID(), owner->GetSystemAddress());
|
|
|
|
GameMessages::SendRegisterPetDBID(m_Owner, m_DatabaseId, owner->GetSystemAddress());
|
|
}
|
|
}
|
|
|
|
void PetComponent::AddDrainImaginationTimer(Item* item, bool fromTaming) {
|
|
if (Game::config->GetValue("pets_take_imagination") != "1") return;
|
|
|
|
const auto* const playerInventory = item->GetInventory();
|
|
if (!playerInventory) return;
|
|
|
|
const auto* const playerInventoryComponent = playerInventory->GetComponent();
|
|
if (!playerInventoryComponent) return;
|
|
|
|
const auto* const playerEntity = playerInventoryComponent->GetParent();
|
|
if (!playerEntity) return;
|
|
|
|
auto* const playerDestroyableComponent = playerEntity->GetComponent<DestroyableComponent>();
|
|
if (!playerDestroyableComponent) return;
|
|
|
|
// Drain by 1 when you summon pet or when this method is called, but not when we have just tamed this pet.
|
|
if (!fromTaming) playerDestroyableComponent->Imagine(-1);
|
|
|
|
// Set this to a variable so when this is called back from the player the timer doesn't fire off.
|
|
m_Parent->AddCallbackTimer(m_PetInfo.imaginationDrainRate, [playerDestroyableComponent, this, item]() {
|
|
if (!playerDestroyableComponent) {
|
|
LOG("No petComponent and/or no playerDestroyableComponent");
|
|
return;
|
|
}
|
|
|
|
// If we are out of imagination despawn the pet.
|
|
if (playerDestroyableComponent->GetImagination() < 1) {
|
|
this->Deactivate(eHelpType::PR_NO_IMAGINATION_HIBERNATE);
|
|
auto playerEntity = playerDestroyableComponent->GetParent();
|
|
if (!playerEntity) return;
|
|
}
|
|
|
|
this->AddDrainImaginationTimer(item);
|
|
});
|
|
}
|
|
|
|
void PetComponent::Deactivate(const eHelpType msg) {
|
|
if (msg != eHelpType::NONE) {
|
|
GameMessages::SendHelp(m_Parent->GetObjectID(), msg, m_Parent->GetSystemAddress());
|
|
}
|
|
|
|
GameMessages::SendPlayFXEffect(m_Parent->GetObjectID(), -1, u"despawn", "", LWOOBJID_EMPTY, 1, 1, true);
|
|
|
|
GameMessages::SendMarkInventoryItemAsActive(m_Owner, false, eUnequippableActiveType::PET, m_ItemId, GetOwner()->GetSystemAddress());
|
|
|
|
activePets.erase(m_Owner);
|
|
|
|
m_Parent->Kill();
|
|
|
|
const auto* const owner = GetOwner();
|
|
|
|
if (!owner) return;
|
|
|
|
GameMessages::SendAddPetToPlayer(m_Owner, 0, u"", LWOOBJID_EMPTY, LOT_NULL, owner->GetSystemAddress());
|
|
|
|
GameMessages::SendRegisterPetID(m_Owner, LWOOBJID_EMPTY, owner->GetSystemAddress());
|
|
|
|
GameMessages::SendRegisterPetDBID(m_Owner, LWOOBJID_EMPTY, owner->GetSystemAddress());
|
|
|
|
GameMessages::SendShowPetActionButton(m_Owner, ePetAbilityType::Invalid, false, owner->GetSystemAddress());
|
|
}
|
|
|
|
void PetComponent::Release() {
|
|
auto* const inventoryComponent = GetOwner()->GetComponent<InventoryComponent>();
|
|
|
|
if (!inventoryComponent) return;
|
|
|
|
Deactivate();
|
|
|
|
inventoryComponent->RemoveDatabasePet(m_DatabaseId);
|
|
|
|
auto* const item = inventoryComponent->FindItemBySubKey(m_DatabaseId);
|
|
|
|
item->SetCount(0, false, false);
|
|
}
|
|
|
|
void PetComponent::Command(const NiPoint3& position, const LWOOBJID source, const int32_t commandType, const int32_t typeId, const bool overrideObey) {
|
|
auto* const owner = GetOwner();
|
|
if (!owner) return;
|
|
|
|
if (commandType == 1) {
|
|
// Emotes
|
|
GameMessages::SendPlayEmote(m_Parent->GetObjectID(), typeId, owner->GetObjectID(), UNASSIGNED_SYSTEM_ADDRESS);
|
|
} else if (commandType == 3) {
|
|
StopInteract(); // TODO: Verify this is necessary
|
|
SetPetAiState(PetAiState::follow);
|
|
} else if (commandType == 6) {
|
|
// TODO: Go to player
|
|
}
|
|
|
|
if (owner->GetGMLevel() >= eGameMasterLevel::DEVELOPER) {
|
|
ChatPackets::SendSystemMessage(owner->GetSystemAddress(), u"Commmand Type: " + (GeneralUtils::to_u16string(commandType)) + u" - Type Id: " + (GeneralUtils::to_u16string(typeId)));
|
|
}
|
|
|
|
// Add movement functionality
|
|
if (position != NiPoint3Constant::ZERO) {
|
|
m_MovementAI->SetDestination(position);
|
|
}
|
|
}
|
|
|
|
Entity* PetComponent::GetOwner() const {
|
|
return Game::entityManager->GetEntity(m_Owner);
|
|
}
|
|
|
|
PetComponent* PetComponent::GetTamingPet(const LWOOBJID tamer) {
|
|
const auto& pair = currentActivities.find(tamer);
|
|
|
|
if (pair == currentActivities.end()) {
|
|
return nullptr;
|
|
}
|
|
|
|
auto* const entity = Game::entityManager->GetEntity(pair->second);
|
|
|
|
if (!entity) {
|
|
currentActivities.erase(tamer);
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
return entity->GetComponent<PetComponent>();
|
|
}
|
|
|
|
PetComponent* PetComponent::GetActivePet(const LWOOBJID owner) {
|
|
const auto& pair = activePets.find(owner);
|
|
|
|
if (pair == activePets.end()) {
|
|
return nullptr;
|
|
}
|
|
|
|
auto* const entity = Game::entityManager->GetEntity(pair->second);
|
|
|
|
if (!entity) {
|
|
activePets.erase(owner);
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
return entity->GetComponent<PetComponent>();
|
|
}
|
|
|
|
void PetComponent::SetPetNameForModeration(const std::string& petName) {
|
|
int approved = 1; //default, in mod
|
|
|
|
//Make sure that the name isn't already auto-approved:
|
|
if (Game::chatFilter->IsSentenceOkay(petName, eGameMasterLevel::CIVILIAN).empty()) {
|
|
approved = 2; //approved
|
|
}
|
|
|
|
//Save to db:
|
|
Database::Get()->SetPetNameModerationStatus(m_DatabaseId, IPetNames::Info{ petName, approved });
|
|
}
|
|
|
|
void PetComponent::LoadPetNameFromModeration() {
|
|
const auto petNameInfo = Database::Get()->GetPetNameInfo(m_DatabaseId);
|
|
if (petNameInfo) {
|
|
m_ModerationStatus = petNameInfo->approvalStatus;
|
|
if (m_ModerationStatus == 2) {
|
|
m_Name = petNameInfo->petName;
|
|
}
|
|
}
|
|
}
|