mirror of
				https://github.com/DarkflameUniverse/DarkflameServer.git
				synced 2025-10-25 08:48:12 +00:00 
			
		
		
		
	 94f8a99fba
			
		
	
	94f8a99fba
	
	
	
		
			
			This reverts commit d7e16ab589697fd1a0a270a02c622b8fa752638f. Fixes an issue where the incorrect mission is marked as being offerable and this overwrote the original mission offered by wisp lee and forced him to offer a different, unacceptable mission. Tested that completing the mission twice from a new daily state and a repeatable state both completed the chain correctly. Unsure what the original bug was, but it does not appear to be present. On top of that, there is no pre-requisite for mission 1883 anywhere in the cdclient, so at best, this check was always false, but the correct behavior is exact equivalence for the mission state.
		
			
				
	
	
		
			179 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			179 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "MissionPrerequisites.h"
 | |
| 
 | |
| #include <sstream>
 | |
| #include <ctime>
 | |
| 
 | |
| #include "CDClientManager.h"
 | |
| #include "dLogger.h"
 | |
| 
 | |
| 
 | |
| PrerequisiteExpression::PrerequisiteExpression(const std::string& str) {
 | |
| 	std::stringstream a;
 | |
| 	std::stringstream b;
 | |
| 	std::stringstream s;
 | |
| 
 | |
| 	auto bor = false;
 | |
| 
 | |
| 	auto sub = false;
 | |
| 
 | |
| 	auto done = false;
 | |
| 
 | |
| 	for (auto i = 0u; i < str.size(); ++i) {
 | |
| 		if (done) {
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		const auto character = str[i];
 | |
| 
 | |
| 		switch (character) {
 | |
| 		case '|':
 | |
| 			bor = true;
 | |
| 			b << str.substr(i + 1);
 | |
| 			done = true;
 | |
| 			break;
 | |
| 		case ' ':
 | |
| 		case ')':
 | |
| 			break;
 | |
| 		case ',':
 | |
| 		case '&':
 | |
| 		case '(':
 | |
| 			b << str.substr(i + 1);
 | |
| 			done = true;
 | |
| 			break;
 | |
| 		case '0':
 | |
| 		case '1':
 | |
| 		case '2':
 | |
| 		case '3':
 | |
| 		case '4':
 | |
| 		case '5':
 | |
| 		case '6':
 | |
| 		case '7':
 | |
| 		case '8':
 | |
| 		case '9':
 | |
| 			if (sub) {
 | |
| 				s << character;
 | |
| 			} else {
 | |
| 				a << character;
 | |
| 			}
 | |
| 			break;
 | |
| 		case ':':
 | |
| 			sub = true;
 | |
| 			break;
 | |
| 		default:
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	this->m_or = bor;
 | |
| 
 | |
| 	const auto aString = a.str();
 | |
| 
 | |
| 	if (!aString.empty()) {
 | |
| 		this->a = std::stoul(a.str());
 | |
| 	} else {
 | |
| 		this->a = 0;
 | |
| 	}
 | |
| 
 | |
| 	const auto subString = s.str();
 | |
| 
 | |
| 	if (!subString.empty()) {
 | |
| 		this->sub = std::stoul(s.str());
 | |
| 	} else {
 | |
| 		this->sub = 0;
 | |
| 	}
 | |
| 
 | |
| 	const auto bString = b.str();
 | |
| 
 | |
| 	if (!bString.empty()) {
 | |
| 		this->b = new PrerequisiteExpression(bString);
 | |
| 	} else {
 | |
| 		this->b = nullptr;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| bool PrerequisiteExpression::Execute(const std::unordered_map<uint32_t, Mission*>& missions) const {
 | |
| 	auto a = this->a == 0;
 | |
| 
 | |
| 	auto b = this->b == nullptr;
 | |
| 
 | |
| 	if (!a) {
 | |
| 		const auto index = missions.find(this->a);
 | |
| 
 | |
| 		if (index != missions.end()) {
 | |
| 			const auto* mission = index->second;
 | |
| 
 | |
| 			if (this->sub != 0) {
 | |
| 				// Special case for one Wisp Lee repeatable mission.
 | |
| 				a = mission->GetMissionState() == static_cast<eMissionState>(this->sub);
 | |
| 			} else if (mission->IsComplete()) {
 | |
| 				a = true;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (!b) {
 | |
| 		b = this->b->Execute(missions);
 | |
| 	}
 | |
| 
 | |
| 	if (this->m_or) {
 | |
| 		return a || b;
 | |
| 	}
 | |
| 
 | |
| 	return a && b;
 | |
| }
 | |
| 
 | |
| 
 | |
| PrerequisiteExpression::~PrerequisiteExpression() {
 | |
| 	delete b;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool MissionPrerequisites::CanAccept(const uint32_t missionId, const std::unordered_map<uint32_t, Mission*>& missions) {
 | |
| 	const auto& missionIndex = missions.find(missionId);
 | |
| 
 | |
| 	if (missionIndex != missions.end()) {
 | |
| 		auto* mission = missionIndex->second;
 | |
| 		const auto& info = mission->GetClientInfo();
 | |
| 
 | |
| 		if (info.repeatable) {
 | |
| 			const auto prerequisitesMet = CheckPrerequisites(missionId, missions);
 | |
| 
 | |
| 			// Checked by client
 | |
| 			const time_t time = std::time(nullptr);
 | |
| 			const time_t lock = mission->GetTimestamp() + info.cooldownTime * 60;
 | |
| 
 | |
| 			// If there's no time limit, just check the prerequisites, otherwise make sure both conditions are met
 | |
| 			return (info.cooldownTime == -1 ? prerequisitesMet : (lock - time < 0)) && prerequisitesMet;
 | |
| 		}
 | |
| 
 | |
| 		// Mission is already accepted and cannot be repeatedly accepted
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	// Mission is not yet accepted, check the prerequisites
 | |
| 	return CheckPrerequisites(missionId, missions);
 | |
| }
 | |
| 
 | |
| bool MissionPrerequisites::CheckPrerequisites(uint32_t missionId, const std::unordered_map<uint32_t, Mission*>& missions) {
 | |
| 	const auto& index = expressions.find(missionId);
 | |
| 	if (index != expressions.end()) {
 | |
| 		return index->second->Execute(missions);
 | |
| 	}
 | |
| 
 | |
| 	auto* missionsTable = CDClientManager::Instance().GetTable<CDMissionsTable>();
 | |
| 	const auto missionEntries = missionsTable->Query([=](const CDMissions& entry) {
 | |
| 		return entry.id == static_cast<int>(missionId);
 | |
| 		});
 | |
| 
 | |
| 	if (missionEntries.empty())
 | |
| 		return false;
 | |
| 
 | |
| 	auto* expression = new PrerequisiteExpression(missionEntries[0].prereqMissionID);
 | |
| 	expressions.insert_or_assign(missionId, expression);
 | |
| 
 | |
| 	return expression->Execute(missions);
 | |
| }
 | |
| 
 | |
| std::unordered_map<uint32_t, PrerequisiteExpression*> MissionPrerequisites::expressions = {};
 |