2021-12-05 17:54:36 +00:00
|
|
|
/*
|
|
|
|
* Darkflame Universe
|
|
|
|
* Copyright 2019
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sstream>
|
|
|
|
#include "MissionOfferComponent.h"
|
|
|
|
#include "CDClientManager.h"
|
|
|
|
#include "CDMissionsTable.h"
|
|
|
|
#include "CDMissionNPCComponentTable.h"
|
|
|
|
#include "GameMessages.h"
|
|
|
|
#include "Entity.h"
|
|
|
|
#include "MissionComponent.h"
|
|
|
|
#include "dLogger.h"
|
|
|
|
#include "Game.h"
|
|
|
|
#include "MissionPrerequisites.h"
|
2023-01-22 23:38:47 +00:00
|
|
|
#include "eMissionState.h"
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
OfferedMission::OfferedMission(const uint32_t missionId, const bool offersMission, const bool acceptsMission) {
|
|
|
|
this->missionId = missionId;
|
|
|
|
this->offersMission = offersMission;
|
|
|
|
this->acceptsMission = acceptsMission;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
uint32_t OfferedMission::GetMissionId() const {
|
|
|
|
return this->missionId;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
bool OfferedMission::GetOfferMission() const {
|
|
|
|
return this->offersMission;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
bool OfferedMission::GetAcceptMission() const {
|
|
|
|
return this->acceptsMission;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------ MissionOfferComponent below ------------------------
|
|
|
|
|
|
|
|
MissionOfferComponent::MissionOfferComponent(Entity* parent, const LOT parentLot) : Component(parent) {
|
2022-07-28 13:39:57 +00:00
|
|
|
auto* compRegistryTable = CDClientManager::Instance()->GetTable<CDComponentsRegistryTable>("ComponentsRegistry");
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2023-03-04 07:16:37 +00:00
|
|
|
auto value = compRegistryTable->GetByIDAndType(parentLot, eReplicaComponentType::MISSION_OFFER, -1);
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
if (value != -1) {
|
|
|
|
const uint32_t componentId = value;
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
// Now lookup the missions in the MissionNPCComponent table
|
|
|
|
auto* missionNpcComponentTable = CDClientManager::Instance()->GetTable<CDMissionNPCComponentTable>("MissionNPCComponent");
|
2022-07-25 02:26:51 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
auto missions = missionNpcComponentTable->Query([=](const CDMissionNPCComponent& entry) {
|
|
|
|
return entry.id == static_cast<unsigned>(componentId);
|
|
|
|
});
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
for (auto& mission : missions) {
|
|
|
|
auto* offeredMission = new OfferedMission(mission.missionID, mission.offersMission, mission.acceptsMission);
|
|
|
|
this->offeredMissions.push_back(offeredMission);
|
|
|
|
}
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
MissionOfferComponent::~MissionOfferComponent() {
|
|
|
|
for (auto* mission : this->offeredMissions) {
|
|
|
|
if (mission) {
|
|
|
|
delete mission;
|
|
|
|
mission = nullptr;
|
|
|
|
}
|
|
|
|
}
|
2022-07-25 02:26:51 +00:00
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
offeredMissions.clear();
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
void MissionOfferComponent::OnUse(Entity* originator) {
|
|
|
|
OfferMissions(originator);
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
|
|
|
|
2022-07-28 13:39:57 +00:00
|
|
|
void MissionOfferComponent::OfferMissions(Entity* entity, const uint32_t specifiedMissionId) {
|
|
|
|
// First, get the entity's MissionComponent. If there is not one, then we cannot offer missions to this entity.
|
2023-03-04 07:16:37 +00:00
|
|
|
auto* missionComponent = static_cast<MissionComponent*>(entity->GetComponent(eReplicaComponentType::MISSION));
|
2022-07-28 13:39:57 +00:00
|
|
|
|
|
|
|
if (!missionComponent) {
|
|
|
|
Game::logger->Log("MissionOfferComponent", "Unable to get mission component for Entity %llu", entity->GetObjectID());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<uint32_t> offered{};
|
|
|
|
|
|
|
|
CDMissions info{};
|
|
|
|
|
|
|
|
if (specifiedMissionId > 0 && !Mission::IsValidMission(specifiedMissionId, info)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto* offeredMission : this->offeredMissions) {
|
|
|
|
if (specifiedMissionId > 0) {
|
|
|
|
if (offeredMission->GetMissionId() != specifiedMissionId && !info.isRandom) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// First, check if we already have the mission
|
|
|
|
const auto missionId = offeredMission->GetMissionId();
|
|
|
|
|
|
|
|
auto* mission = missionComponent->GetMission(missionId);
|
|
|
|
|
|
|
|
if (mission != nullptr) {
|
|
|
|
if (specifiedMissionId <= 0) {
|
|
|
|
// Handles the odd case where the offer object should not display the mission again
|
|
|
|
if (!mission->IsComplete() && mission->GetClientInfo().offer_objectID == m_Parent->GetLOT() && mission->GetClientInfo().target_objectID != m_Parent->GetLOT() && mission->IsFetchMission()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have the mission, if it is not complete, offer it
|
|
|
|
if (mission->IsActive() || mission->IsReadyToComplete()) {
|
|
|
|
GameMessages::SendOfferMission(entity->GetObjectID(), entity->GetSystemAddress(), missionId, m_Parent->GetObjectID());
|
|
|
|
|
|
|
|
offered.push_back(missionId);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto canAccept = MissionPrerequisites::CanAccept(missionId, missionComponent->GetMissions());
|
|
|
|
|
|
|
|
// Mission has not yet been accepted - check the prereqs
|
|
|
|
if (!canAccept)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!Mission::IsValidMission(missionId, info)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto& randomPool = info.randomPool;
|
|
|
|
const auto isRandom = info.isRandom;
|
|
|
|
|
|
|
|
if (isRandom && randomPool.empty()) // This means the mission is part of a random pool of missions.
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isRandom && !randomPool.empty()) {
|
|
|
|
std::istringstream stream(randomPool);
|
|
|
|
std::string token;
|
|
|
|
|
|
|
|
std::vector<uint32_t> randomMissionPool;
|
|
|
|
|
|
|
|
while (std::getline(stream, token, ',')) {
|
|
|
|
try {
|
|
|
|
const auto value = std::stoul(token);
|
|
|
|
|
|
|
|
randomMissionPool.push_back(value);
|
|
|
|
} catch (std::invalid_argument& exception) {
|
|
|
|
Game::logger->Log("MissionOfferComponent", "Failed to parse value (%s): (%s)!", token.c_str(), exception.what());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (specifiedMissionId > 0) {
|
|
|
|
const auto& iter = std::find(randomMissionPool.begin(), randomMissionPool.end(), specifiedMissionId);
|
|
|
|
|
|
|
|
if (iter != randomMissionPool.end() && MissionPrerequisites::CanAccept(specifiedMissionId, missionComponent->GetMissions())) {
|
|
|
|
GameMessages::SendOfferMission(entity->GetObjectID(), entity->GetSystemAddress(), specifiedMissionId, m_Parent->GetObjectID());
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<uint32_t> canAcceptPool;
|
|
|
|
|
|
|
|
for (const auto sample : randomMissionPool) {
|
|
|
|
const auto state = missionComponent->GetMissionState(sample);
|
|
|
|
|
2023-01-22 23:38:47 +00:00
|
|
|
if (state == eMissionState::ACTIVE ||
|
|
|
|
state == eMissionState::COMPLETE_ACTIVE ||
|
|
|
|
state == eMissionState::READY_TO_COMPLETE ||
|
|
|
|
state == eMissionState::COMPLETE_READY_TO_COMPLETE ||
|
2022-07-28 13:39:57 +00:00
|
|
|
sample == specifiedMissionId) {
|
|
|
|
mission = missionComponent->GetMission(sample);
|
|
|
|
|
|
|
|
if (mission == nullptr || mission->IsAchievement()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
GameMessages::SendOfferMission(entity->GetObjectID(), entity->GetSystemAddress(), sample, m_Parent->GetObjectID());
|
|
|
|
|
|
|
|
canAcceptPool.clear();
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (std::find(offered.begin(), offered.end(), sample) == offered.end() && MissionPrerequisites::CanAccept(sample, missionComponent->GetMissions())) {
|
|
|
|
canAcceptPool.push_back(sample);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the mission is already active or we already completed one of them today
|
|
|
|
if (canAcceptPool.empty())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const auto selected = canAcceptPool[GeneralUtils::GenerateRandomNumber<int>(0, canAcceptPool.size() - 1)];
|
|
|
|
|
|
|
|
GameMessages::SendOfferMission(entity->GetObjectID(), entity->GetSystemAddress(), selected, m_Parent->GetObjectID());
|
|
|
|
} else if (std::find(offered.begin(), offered.end(), missionId) == offered.end() && offeredMission->GetOfferMission()) {
|
|
|
|
GameMessages::SendOfferMission(entity->GetObjectID(), entity->GetSystemAddress(), missionId, m_Parent->GetObjectID());
|
|
|
|
}
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|