2023-06-20 14:33:09 +10:00
|
|
|
#include "StarQuestManager.hpp"
|
|
|
|
#include "StarRoot.hpp"
|
|
|
|
#include "StarPlayer.hpp"
|
|
|
|
#include "StarPlayerInventory.hpp"
|
|
|
|
#include "StarAssets.hpp"
|
|
|
|
#include "StarItemDatabase.hpp"
|
|
|
|
#include "StarRandom.hpp"
|
|
|
|
#include "StarJsonExtra.hpp"
|
|
|
|
#include "StarTime.hpp"
|
|
|
|
#include "StarClientContext.hpp"
|
|
|
|
#include "StarUniverseClient.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
|
|
|
QuestManager::QuestManager(Player* player) {
|
|
|
|
m_player = player;
|
|
|
|
m_world = nullptr;
|
|
|
|
m_trackOnWorldQuests = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QuestTemplatePtr getTemplate(String const& templateId) {
|
|
|
|
return Root::singleton().questTemplateDatabase()->questTemplate(templateId);
|
|
|
|
}
|
|
|
|
|
|
|
|
StringMap<QuestPtr> readQuests(Json const& json) {
|
|
|
|
auto versioningDatabase = Root::singleton().versioningDatabase();
|
|
|
|
auto questTemplateDatabase = Root::singleton().questTemplateDatabase();
|
|
|
|
|
|
|
|
auto validateArc = [questTemplateDatabase](QuestArcDescriptor const& arc) {
|
|
|
|
for (auto quest : arc.quests) {
|
|
|
|
if (!questTemplateDatabase->questTemplate(quest.templateId))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
StringMap<QuestPtr> result;
|
|
|
|
for (auto const& questPair : json.iterateObject()) {
|
|
|
|
// don't load the quest unless all quests in the arc exist
|
|
|
|
Json diskStore = versioningDatabase->loadVersionedJson(VersionedJson::fromJson(questPair.second), "Quest");
|
|
|
|
auto questArc = QuestArcDescriptor::diskLoad(diskStore.get("arc"));
|
|
|
|
if (validateArc(questArc))
|
|
|
|
result[questPair.first] = make_shared<Quest>(questPair.second);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
function<bool (QuestPtr const&)> questFilter(QuestState state) {
|
|
|
|
return [state](QuestPtr const& quest) {
|
|
|
|
return quest->state() == state;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::diskLoad(Json const& quests) {
|
|
|
|
m_quests = readQuests(quests.get("quests", JsonObject{}));
|
|
|
|
m_trackedQuestId = quests.optString("currentQuest");
|
|
|
|
}
|
|
|
|
|
|
|
|
Json QuestManager::diskStore() {
|
|
|
|
auto questPtrToJson = [](QuestPtr const& quest) { return quest->diskStore(); };
|
|
|
|
|
|
|
|
auto quests = m_quests.values();
|
|
|
|
|
|
|
|
return JsonObject{
|
|
|
|
{"quests", jsonFromMapV<StringMap<QuestPtr>>(m_quests, questPtrToJson)},
|
|
|
|
{"currentQuest", jsonFromMaybe(m_trackedQuestId)}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::setUniverseClient(UniverseClient* client) {
|
|
|
|
m_client = client;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::init(World* world) {
|
|
|
|
m_world = world;
|
2024-10-15 16:16:57 +11:00
|
|
|
for (auto& quest : m_quests) {
|
|
|
|
if (!questValidOnServer(quest.second))
|
|
|
|
continue;
|
|
|
|
quest.second->init(m_player, world, m_client);
|
|
|
|
}
|
2023-06-20 14:33:09 +10:00
|
|
|
m_trackOnWorldQuests = true;
|
|
|
|
|
|
|
|
// untrack tracked quest if it's not cross-server, and we're on a different server
|
|
|
|
if (auto tracked = m_trackedQuestId) {
|
|
|
|
if (auto maybeQuest = m_quests.maybe(*tracked)) {
|
|
|
|
auto quest = *maybeQuest;
|
|
|
|
if (quest->hideCrossServer() && quest->serverUuid() && *quest->serverUuid() != m_player->clientContext()->serverUuid())
|
|
|
|
m_trackedQuestId = {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::uninit() {
|
|
|
|
m_quests.values().exec([](QuestPtr const& quest) { quest->uninit(); });
|
|
|
|
m_world = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QuestManager::canStart(QuestArcDescriptor const& questArc) const {
|
|
|
|
if (questArc.quests.size() == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (auto questDesc : questArc.quests) {
|
|
|
|
auto questTemplate = getTemplate(questDesc.templateId);
|
|
|
|
if (!questTemplate)
|
|
|
|
return false;
|
|
|
|
if (auto quest = m_quests.maybe(questDesc.questId))
|
|
|
|
if ((*quest)->state() != QuestState::Failed)
|
|
|
|
return false;
|
|
|
|
for (auto const& prerequisiteQuestId : questTemplate->prerequisiteQuests) {
|
|
|
|
if (!hasCompleted(prerequisiteQuestId))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (auto item : questTemplate->requiredItems)
|
|
|
|
if (!m_player->inventory()->hasItem(item))
|
|
|
|
return false;
|
|
|
|
if (questTemplate->requiredShipLevel) {
|
2023-07-22 22:31:04 +10:00
|
|
|
if (m_player->shipUpgrades().shipLevel < *questTemplate->requiredShipLevel)
|
2023-06-20 14:33:09 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::offer(QuestPtr const& quest) {
|
|
|
|
m_quests[quest->questId()] = quest;
|
|
|
|
quest->init(m_player, m_world, m_client);
|
|
|
|
quest->offer();
|
|
|
|
}
|
|
|
|
|
2024-10-15 16:16:57 +11:00
|
|
|
StringMap<QuestPtr> QuestManager::quests() const {
|
|
|
|
return m_quests;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringMap<QuestPtr> QuestManager::serverQuests() const {
|
|
|
|
StringMap<QuestPtr> filtered;
|
|
|
|
for (auto& pair : m_quests) {
|
|
|
|
QuestPtr q = pair.second;
|
|
|
|
if (!questValidOnServer(q))
|
|
|
|
continue;
|
|
|
|
filtered.insert(pair.first, q);
|
|
|
|
}
|
|
|
|
return filtered;
|
|
|
|
}
|
|
|
|
|
2023-06-20 14:33:09 +10:00
|
|
|
QuestPtr QuestManager::getQuest(String const& questId) const {
|
|
|
|
return m_quests.get(questId);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QuestManager::hasQuest(String const& questId) const {
|
|
|
|
return m_quests.contains(questId);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QuestManager::hasAcceptedQuest(String const& questId) const {
|
|
|
|
return m_quests.contains(questId) && m_quests.get(questId)->state() != QuestState::New && m_quests.get(questId)->state() != QuestState::Offer;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QuestManager::isActive(String const& questId) const {
|
|
|
|
return m_quests.contains(questId) && m_quests.get(questId)->state() == QuestState::Active;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QuestManager::isCurrent(String const& questId) const {
|
|
|
|
return m_onWorldQuestId.orMaybe(m_trackedQuestId) == questId;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QuestManager::isTracked(String const& questId) const {
|
|
|
|
return m_trackedQuestId == questId;
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::setAsTracked(Maybe<String> const& questId) {
|
|
|
|
if (questId && isActive(*questId)) {
|
|
|
|
m_trackedQuestId = questId;
|
|
|
|
if (m_onWorldQuestId) {
|
|
|
|
// stop auto tracking quests on this world
|
|
|
|
m_onWorldQuestId = {};
|
|
|
|
m_trackOnWorldQuests = false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
m_trackedQuestId = {};
|
|
|
|
m_trackOnWorldQuests = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::markAsRead(String const& questId) {
|
|
|
|
if (!m_quests.contains(questId))
|
|
|
|
return;
|
|
|
|
|
|
|
|
getQuest(questId)->markAsRead();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QuestManager::hasCompleted(String const& questId) const {
|
|
|
|
if (auto quest = m_quests.maybe(questId))
|
|
|
|
if ((*quest)->state() == QuestState::Complete)
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool QuestManager::canTurnIn(String const& questId) const {
|
|
|
|
if (auto quest = m_quests.maybe(questId))
|
|
|
|
if ((*quest)->state() == QuestState::Active && (*quest)->canTurnIn())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<QuestPtr> QuestManager::getFirstNewQuest() {
|
2024-10-15 16:16:57 +11:00
|
|
|
for (auto& q : m_quests) {
|
|
|
|
if (questValidOnServer(q.second) && q.second->state() == QuestState::Offer)
|
|
|
|
return q.second;
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<QuestPtr> QuestManager::getFirstCompletableQuest() {
|
2024-10-15 16:16:57 +11:00
|
|
|
for (auto& q : m_quests) {
|
|
|
|
if (questValidOnServer(q.second) && q.second->state() == QuestState::Complete && q.second->showDialog())
|
|
|
|
return q.second;
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<QuestPtr> QuestManager::getFirstFailableQuest() {
|
2024-10-15 16:16:57 +11:00
|
|
|
for (auto& q : m_quests) {
|
|
|
|
if (questValidOnServer(q.second) && q.second->state() == QuestState::Failed && q.second->showDialog())
|
|
|
|
return q.second;
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<QuestPtr> QuestManager::getFirstMainQuest() {
|
2024-10-15 16:16:57 +11:00
|
|
|
for (auto& q : m_quests) {
|
|
|
|
if (questValidOnServer(q.second) && q.second->state() == QuestState::Active && q.second->mainQuest())
|
|
|
|
return q.second;
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
void sortQuests(List<QuestPtr>& quests) {
|
|
|
|
std::sort(quests.begin(),
|
|
|
|
quests.end(),
|
|
|
|
[](QuestPtr const& left, QuestPtr const& right) -> bool {
|
|
|
|
int64_t leftUpdated = left->lastUpdatedOn();
|
|
|
|
int64_t rightUpdated = right->lastUpdatedOn();
|
|
|
|
String leftQuestId = left->templateId();
|
|
|
|
String rightQuestId = right->templateId();
|
|
|
|
return std::tie(leftUpdated, leftQuestId) < std::tie(rightUpdated, rightQuestId);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
List<QuestPtr> QuestManager::listActiveQuests() const {
|
2024-10-15 16:16:57 +11:00
|
|
|
List<QuestPtr> result = serverQuests().values();
|
2023-06-20 14:33:09 +10:00
|
|
|
result.filter([&](QuestPtr quest) {
|
|
|
|
return quest->state() == QuestState::Active && quest->showInLog();
|
|
|
|
});
|
|
|
|
sortQuests(result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
List<QuestPtr> QuestManager::listCompletedQuests() const {
|
2024-10-15 16:16:57 +11:00
|
|
|
List<QuestPtr> result = serverQuests().values();
|
2023-06-20 14:33:09 +10:00
|
|
|
result.filter([](QuestPtr quest) {
|
|
|
|
return quest->state() == QuestState::Complete && quest->showInLog();
|
|
|
|
});
|
|
|
|
sortQuests(result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
List<QuestPtr> QuestManager::listFailedQuests() const {
|
2024-10-15 16:16:57 +11:00
|
|
|
List<QuestPtr> result = serverQuests().values();
|
2023-06-20 14:33:09 +10:00
|
|
|
result.filter([](QuestPtr quest) {
|
|
|
|
return quest->state() == QuestState::Failed && quest->showInLog();
|
|
|
|
});
|
|
|
|
sortQuests(result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<QuestPtr> QuestManager::currentQuest() const {
|
|
|
|
auto questId = m_onWorldQuestId.orMaybe(m_trackedQuestId);
|
|
|
|
if (questId && isActive(*questId)) {
|
|
|
|
auto current = getQuest(*questId);
|
|
|
|
if (current->showInLog())
|
|
|
|
return current;
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<QuestPtr> QuestManager::trackedQuest() const {
|
|
|
|
if (m_trackedQuestId && isActive(*m_trackedQuestId)) {
|
|
|
|
auto current = getQuest(*m_trackedQuestId);
|
|
|
|
if (current->showInLog())
|
|
|
|
return current;
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<QuestIndicator> QuestManager::getQuestIndicator(EntityPtr const& entity) const {
|
|
|
|
Maybe<String> indicatorType;
|
|
|
|
Vec2F indicatorPos = entity->position() + Vec2F(0, 2.75);
|
2023-07-01 08:01:29 +10:00
|
|
|
auto questGiver = as<InteractiveEntity>(entity);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
2023-07-01 08:01:29 +10:00
|
|
|
if (questGiver) {
|
2023-06-20 14:33:09 +10:00
|
|
|
if (!indicatorType) {
|
2023-07-01 08:01:29 +10:00
|
|
|
for (auto& questId : questGiver->turnInQuests()) {
|
2023-06-20 14:33:09 +10:00
|
|
|
if (!isActive(questId))
|
|
|
|
continue;
|
|
|
|
auto quest = getQuest(questId);
|
|
|
|
if (quest->canTurnIn()) {
|
|
|
|
indicatorType = quest->questReceiverIndicator();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!indicatorType) {
|
|
|
|
auto questTemplateDatabase = Root::singleton().questTemplateDatabase();
|
2023-07-01 08:01:29 +10:00
|
|
|
for (auto& questArc : questGiver->offeredQuests()) {
|
2023-06-20 14:33:09 +10:00
|
|
|
if (canStart(questArc) && questArc.quests.size() > 0) {
|
2023-07-01 08:01:29 +10:00
|
|
|
auto& questDesc = questArc.quests[0];
|
2023-06-20 14:33:09 +10:00
|
|
|
auto questTemplate = questTemplateDatabase->questTemplate(questDesc.templateId);
|
|
|
|
indicatorType = questTemplate->questGiverIndicator;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (indicatorType) {
|
|
|
|
Json indicators = Root::singleton().assets()->json("/quests/quests.config:indicators");
|
|
|
|
String indicatorImage = indicators.get(*indicatorType).getString("image");
|
2023-07-01 08:01:29 +10:00
|
|
|
if (questGiver)
|
|
|
|
indicatorPos = questGiver->questIndicatorPosition();
|
2023-06-20 14:33:09 +10:00
|
|
|
return QuestIndicator{indicatorImage, indicatorPos};
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& pair : m_quests) {
|
|
|
|
if (pair.second->state() == QuestState::Active) {
|
2023-07-01 08:01:29 +10:00
|
|
|
if (auto indicatorImage = pair.second->customIndicator(entity)) {
|
|
|
|
if (questGiver)
|
|
|
|
indicatorPos = questGiver->questIndicatorPosition();
|
2023-06-20 14:33:09 +10:00
|
|
|
return QuestIndicator{ *indicatorImage, indicatorPos };
|
2023-07-01 08:01:29 +10:00
|
|
|
}
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
StringSet QuestManager::interestingObjects() {
|
|
|
|
StringSet result;
|
|
|
|
m_quests.values().exec([&result](QuestPtr const& quest) {
|
|
|
|
if (auto questObjects = quest->receiveMessage("interestingObjects", true, JsonArray()))
|
|
|
|
result.addAll(jsonToStringSet(*questObjects));
|
|
|
|
});
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Maybe<Json> QuestManager::receiveMessage(String const& message, bool localMessage, JsonArray const& args) {
|
|
|
|
starAssert(m_world);
|
|
|
|
Maybe<Json> result;
|
|
|
|
m_quests.values().exec([&result, message, localMessage, args](
|
|
|
|
QuestPtr const& quest) { result = result.orMaybe(quest->receiveMessage(message, localMessage, args)); });
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-07-21 00:58:49 +10:00
|
|
|
void QuestManager::update(float dt) {
|
2023-06-20 14:33:09 +10:00
|
|
|
startInitialQuests();
|
|
|
|
|
|
|
|
if (m_trackedQuestId && !isActive(*m_trackedQuestId))
|
|
|
|
m_trackedQuestId = {};
|
|
|
|
|
|
|
|
if (m_onWorldQuestId) {
|
|
|
|
auto active = isActive(*m_onWorldQuestId);
|
|
|
|
if (active) {
|
|
|
|
auto worldId = getQuest(*m_onWorldQuestId)->worldId();
|
|
|
|
if (!worldId || m_player->clientContext()->playerWorldId() != *worldId)
|
|
|
|
active = false;
|
|
|
|
}
|
|
|
|
if (!active)
|
|
|
|
m_onWorldQuestId = {};
|
|
|
|
} else if (m_trackOnWorldQuests) {
|
|
|
|
auto playerWorldId = m_client->clientContext()->playerWorldId();
|
|
|
|
auto trackedWorld = currentQuest().sequence([](QuestPtr q) { return q->worldId(); });
|
|
|
|
if (!trackedWorld || playerWorldId != *trackedWorld) {
|
|
|
|
// the currently tracked quest is not on this world, track another quest on this world
|
|
|
|
for (auto quest : listActiveQuests()) {
|
|
|
|
if (auto questWorld = quest->worldId()) {
|
|
|
|
if (playerWorldId == *questWorld)
|
|
|
|
m_onWorldQuestId = quest->questId();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& entry : m_quests) {
|
|
|
|
auto quest = entry.second;
|
|
|
|
QuestState state = quest->state();
|
|
|
|
bool finished = state == QuestState::Complete || state == QuestState::Failed;
|
|
|
|
if (state == QuestState::New || (finished && quest->ephemeral() && !quest->showDialog())) {
|
|
|
|
quest->uninit();
|
|
|
|
m_quests.remove(entry.first);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-15 16:16:57 +11:00
|
|
|
for (auto& q : m_quests) {
|
|
|
|
if (questValidOnServer(q.second))
|
|
|
|
q.second->update(dt);
|
|
|
|
}
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::startInitialQuests() {
|
|
|
|
auto startingQuests =
|
2023-06-27 20:23:44 +10:00
|
|
|
Root::singleton().assets()->json(strf("/quests/quests.config:initialquests.{}", m_player->species())).toArray();
|
2023-06-20 14:33:09 +10:00
|
|
|
for (auto const& questArcJson : startingQuests) {
|
|
|
|
QuestArcDescriptor quest = QuestArcDescriptor::fromJson(questArcJson);
|
|
|
|
if (canStart(quest))
|
|
|
|
offer(make_shared<Quest>(quest, 0, m_player));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuestManager::setMostRecentQuestCurrent() {
|
|
|
|
List<QuestPtr> sortedActiveQuests = listActiveQuests();
|
|
|
|
if (sortedActiveQuests.size() > 0)
|
|
|
|
setAsTracked(sortedActiveQuests.last()->questId());
|
|
|
|
}
|
|
|
|
|
2024-10-15 16:16:57 +11:00
|
|
|
bool QuestManager::questValidOnServer(QuestPtr q) const {
|
|
|
|
return !(q->hideCrossServer() && q->serverUuid().isValid() && *q->serverUuid() != m_player->clientContext()->serverUuid());
|
|
|
|
}
|
|
|
|
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|