// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/> // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see <http://www.gnu.org/licenses/>. #include "stdpch.h" #include "npc_icon.h" #include "ingame_database_manager.h" #include "game_share/generic_xml_msg_mngr.h" #include "entities.h" #include "net_manager.h" using namespace std; using namespace NLMISC; CNPCIconCache* CNPCIconCache::_Instance = NULL; // Time after which the state of a NPC is considered obsolete and must be refreshed (because it's in gamecycle, actual time increases if server slows down, to avoid server congestion) NLMISC::TGameCycle CNPCIconCache::_CacheRefreshTimerDelay = NPC_ICON::DefaultClientNPCIconRefreshTimerDelayGC; // Time between updates of the "catchall timer" NLMISC::TGameCycle CNPCIconCache::_CatchallTimerPeriod = NPC_ICON::DefaultClientNPCIconRefreshTimerDelayGC; extern CEntityManager EntitiesMngr; extern CGenericXmlMsgHeaderManager GenericMsgHeaderMngr; extern CNetManager NetMngr; // #pragma optimize ("", off) CNPCIconCache::CNPCIconCache() : _LastRequestTimestamp(0), _LastTimerUpdateTimestamp(0), _Enabled(true) { _Icons[NPC_ICON::IconNone].init("", ""); _Icons[NPC_ICON::IconNotAMissionGiver].init("", ""); _Icons[NPC_ICON::IconListHasOutOfReachMissions].init("mission_available.tga", ""); //"MP_Blood.tga" _Icons[NPC_ICON::IconListHasAlreadyTakenMissions].init("ICO_Task_Generic.tga", "r2ed_tool_redo"); _Icons[NPC_ICON::IconListHasAvailableMission].init("mission_available.tga", "", CViewRadar::MissionList); //"MP_Wood.tga" _Icons[NPC_ICON::IconAutoHasUnavailableMissions].init("spe_com.tga", ""); _Icons[NPC_ICON::IconAutoHasAvailableMission].init("spe_com.tga", "", CViewRadar::MissionAuto); //"MP_Oil.tga" _Icons[NPC_ICON::IconStepMission].init("mission_step.tga", "", CViewRadar::MissionStep); //"MP_Shell.tga" _DescriptionsToRequest.reserve(256); } void CNPCIconCache::release() { if (_Instance) { delete _Instance; _Instance = NULL; } } const CNPCIconCache::CNPCIconDesc& CNPCIconCache::getNPCIcon(const CEntityCL *entity, bool bypassEnabled) { // Not applicable? Most entities (creatures, characters) have a null key here. BOMB_IF(!entity, "NULL entity in getNPCIcon", return _Icons[NPC_ICON::IconNone]); TNPCIconCacheKey npcIconCacheKey = CNPCIconCache::entityToKey(entity); if (npcIconCacheKey == 0) return _Icons[NPC_ICON::IconNone]; // Is system disabled? if ((!enabled()) && !bypassEnabled) return _Icons[NPC_ICON::IconNone]; // This method must be reasonably fast, because it constantly gets called by the radar view H_AUTO(GetNPCIconWithKey); // Not applicable (more checks)? if (!entity->canHaveMissionIcon()) return _Icons[NPC_ICON::IconNone]; if (!entity->isFriend()) // to display icons in the radar, we need the Contextual property to be received as soon as possible return _Icons[NPC_ICON::IconNone]; // Temporarily not shown if the player is in interaction with the NPC if (UserEntity->interlocutor() != CLFECOMMON::INVALID_SLOT) { CEntityCL *interlocutorEntity = EntitiesMngr.entity(UserEntity->interlocutor()); if (interlocutorEntity && (entityToKey(interlocutorEntity) == npcIconCacheKey)) return _Icons[NPC_ICON::IconNone]; } if (UserEntity->trader() != CLFECOMMON::INVALID_SLOT) { CEntityCL *traderEntity = EntitiesMngr.entity(UserEntity->trader()); if (traderEntity && (entityToKey(traderEntity) == npcIconCacheKey)) return _Icons[NPC_ICON::IconNone]; } // 1. Test if the NPC is involved in a current goal if (isNPCaCurrentGoal(npcIconCacheKey)) return _Icons[NPC_ICON::IconStepMission]; // 2. Compute "has mission to take": take from cache, or query the server H_AUTO(GetNPCIcon_GIVER); CMissionGiverMap::iterator img = _MissionGivers.find(npcIconCacheKey); if (img != _MissionGivers.end()) { CNPCMissionGiverDesc& giver = (*img).second; if (giver.getState() != NPC_ICON::AwaitingFirstData) { // Ask the server to refresh the state if the information is old // but only known mission givers that have a chance to propose new missions if ((giver.getState() != NPC_ICON::NotAMissionGiver) && // (giver.getState() != NPC_ICON::ListHasAlreadyTakenMissions) && // commented out because it would not refresh in case an auto mission become available (!giver.isDescTransient())) { NLMISC::TGameCycle informationAge = NetMngr.getCurrentServerTick() - giver.getLastUpdateTimestamp(); if (informationAge > _CacheRefreshTimerDelay) { queryMissionGiverData(npcIconCacheKey); giver.setDescTransient(); } } // Return the icon depending on the state in the cache return _Icons[giver.getState()]; // TNPCIconId maps TNPCMissionGiverState } } else { // Create mission giver entry and query the server CNPCMissionGiverDesc giver; CMissionGiverMap::iterator itg = _MissionGivers.insert(make_pair(npcIconCacheKey, giver)).first; queryMissionGiverData(npcIconCacheKey); //(*itg).second.setDescTransient(); // already made transient by constructor } return _Icons[NPC_ICON::IconNone]; } #define getArraySize(a) (sizeof(a)/sizeof(a[0])) void CNPCIconCache::addObservers() { // Disabled? if (!enabled()) return; // Mission Journal static const char *missionStartStopLeavesToMonitor [2] = {"TITLE", "FINISHED"}; IngameDbMngr.addBranchObserver( IngameDbMngr.getNodePtr(), "MISSIONS", MissionStartStopObserver, missionStartStopLeavesToMonitor, getArraySize(missionStartStopLeavesToMonitor)); static const char *missionNpcAliasLeavesToMonitor [1] = {"NPC_ALIAS"}; IngameDbMngr.addBranchObserver( IngameDbMngr.getNodePtr(), "MISSIONS", MissionNpcAliasObserver, missionNpcAliasLeavesToMonitor, getArraySize(missionNpcAliasLeavesToMonitor)); // Skills static const char *skillLeavesToMonitor [2] = {"SKILL", "BaseSKILL"}; IngameDbMngr.addBranchObserver( IngameDbMngr.getNodePtr(), "CHARACTER_INFO:SKILLS", MissionPrerequisitEventObserver, skillLeavesToMonitor, getArraySize(skillLeavesToMonitor)); // Owned Items static const char *bagLeavesToMonitor [1] = {"SHEET"}; // just saves 2000 bytes or so (500 * observer pointer entry in vector) compared to one observer per bag slot IngameDbMngr.addBranchObserver( IngameDbMngr.getNodePtr(), "INVENTORY:BAG", MissionPrerequisitEventObserver, bagLeavesToMonitor, getArraySize(bagLeavesToMonitor)); // Worn Items IngameDbMngr.addBranchObserver( "INVENTORY:HAND", &MissionPrerequisitEventObserver); IngameDbMngr.addBranchObserver( "INVENTORY:EQUIP", &MissionPrerequisitEventObserver); // Known Bricks IngameDbMngr.addBranchObserver( "BRICK_FAMILY", &MissionPrerequisitEventObserver); // For other events, search for calls of onEventForMissionAvailabilityForThisChar() } void CNPCIconCache::removeObservers() { // Disabled? if (!enabled()) return; // Mission Journal IngameDbMngr.getNodePtr()->removeBranchObserver("MISSIONS", MissionStartStopObserver); IngameDbMngr.getNodePtr()->removeBranchObserver("MISSIONS", MissionNpcAliasObserver); // Skills IngameDbMngr.getNodePtr()->removeBranchObserver("CHARACTER_INFO:SKILLS", MissionPrerequisitEventObserver); // Owned Items IngameDbMngr.getNodePtr()->removeBranchObserver("INVENTORY:BAG", MissionPrerequisitEventObserver); // Worn Items IngameDbMngr.getNodePtr()->removeBranchObserver("INVENTORY:HAND", MissionPrerequisitEventObserver); IngameDbMngr.getNodePtr()->removeBranchObserver("INVENTORY:EQUIP", MissionPrerequisitEventObserver); // Known Bricks IngameDbMngr.getNodePtr()->removeBranchObserver("BRICK_FAMILY", MissionPrerequisitEventObserver); } void CNPCIconCache::CMissionStartStopObserver::update(ICDBNode* node) { // Every time a mission in progress is started or stopped, refresh the icon for visible NPCs (including mission giver information) CNPCIconCache::getInstance().onEventForMissionInProgress(); } void CNPCIconCache::CMissionNpcAliasObserver::update(ICDBNode* node) { CNPCIconCache::getInstance().onNpcAliasChangedInMissionGoals(); } void CNPCIconCache::CMissionPrerequisitEventObserver::update(ICDBNode* node) { // Every time a mission in progress changes, refresh the icon for the related npc CNPCIconCache::getInstance().onEventForMissionAvailabilityForThisChar(); } void CNPCIconCache::onEventForMissionAvailabilityForThisChar() { // Disabled? if (!enabled()) return; queryAllVisibleMissionGiverData(0); } void CNPCIconCache::queryMissionGiverData(TNPCIconCacheKey npcIconCacheKey) { _DescriptionsToRequest.push_back(npcIconCacheKey); //static set<TNPCIconCacheKey> requests1; //requests1.insert(npcIconCacheKey); //nldebug("%u: queryMissionGiverData %u (total %u)", NetMngr.getCurrentServerTick(), npcIconCacheKey, requests1.size()); } void CNPCIconCache::queryAllVisibleMissionGiverData(NLMISC::TGameCycle olderThan) { // Request an update for all npcs (qualifying, i.e. that have missions) in vision for (uint i=0; i<EntitiesMngr.entities().size(); ++i) { CEntityCL *entity = EntitiesMngr.entity(i); if (!entity || !(entity->canHaveMissionIcon() && entity->isFriend())) continue; TNPCIconCacheKey npcIconCacheKey = CNPCIconCache::entityToKey(entity); CMissionGiverMap::iterator img = _MissionGivers.find(npcIconCacheKey); if (img == _MissionGivers.end()) continue; // if the NPC does not have an entry yet, it will be created by getNPCIcon() // Refresh only known mission givers that have a chance to propose new missions CNPCMissionGiverDesc& giver = (*img).second; if (giver.getState() == NPC_ICON::NotAMissionGiver) continue; // if (giver.getState() == NPC_ICON::ListHasAlreadyTakenMissions) // continue; // commented out because it would not refresh in case an auto mission becomes available if (olderThan != 0) { // Don't refresh desscriptions already awaiting an update if (giver.isDescTransient()) continue; // Don't refresh NPCs having data more recent than specified NLMISC::TGameCycle informationAge = NetMngr.getCurrentServerTick() - giver.getLastUpdateTimestamp(); if (informationAge <= olderThan) continue; // Don't refresh NPC being involved in a mission goal (the step icon has higher priority over the giver icon) // If later the NPC is no more involved before the information is considered old, it will show // the same giver state until the information is considered old. That's why we let refresh // the NPC when triggered by an event (olderThan == 0). if (isNPCaCurrentGoal(npcIconCacheKey)) continue; } _DescriptionsToRequest.push_back(npcIconCacheKey); giver.setDescTransient(); //static set<TNPCIconCacheKey> requests2; //requests2.insert(npcIconCacheKey); //nldebug("%u: queryAllVisibleMissionGiverData %u (total %u)", NetMngr.getCurrentServerTick(), npcIconCacheKey, requests2.size()); } } void CNPCIconCache::update() { // Every CatchallTimerPeriod, browse visible entities and refresh the ones with outdated state // (e.g. the ones not displayed in radar). if (NetMngr.getCurrentServerTick() > _LastTimerUpdateTimestamp + _CatchallTimerPeriod) { _LastTimerUpdateTimestamp = NetMngr.getCurrentServerTick(); // Disabled? if (!enabled()) return; queryAllVisibleMissionGiverData(_CacheRefreshTimerDelay); } // Every tick update at most (2 cycles actually, cf. server<->client communication frequency), // send all pending requests in a single message. if (NetMngr.getCurrentServerTick() > _LastRequestTimestamp) { if (!_DescriptionsToRequest.empty()) { CBitMemStream out; GenericMsgHeaderMngr.pushNameToStream("NPC_ICON:GET_DESC", out); uint8 nb8 = uint8(_DescriptionsToRequest.size() & 0xff); // up to vision size (255 i.e. 256 minus user) out.serial(nb8); for (CSmallKeyList::const_iterator ikl=_DescriptionsToRequest.begin(); ikl!=_DescriptionsToRequest.end(); ++ikl) { TNPCIconCacheKey key = *ikl; out.serial(key); } NetMngr.push(out); //nldebug("%u: Pushing %hu NPC desc requests", NetMngr.getCurrentServerTick(), nb8); _DescriptionsToRequest.clear(); } _LastRequestTimestamp = NetMngr.getCurrentServerTick(); } } void CNPCIconCache::onEventForMissionInProgress() { // Disabled? if (!enabled()) return; // Immediately reflect the mission journal (Step icons) refreshIconsOfScene(true); // Ask the server to update availability status (will refresh icons if there is at least one change) onEventForMissionAvailabilityForThisChar(); } void CNPCIconCache::onNpcAliasChangedInMissionGoals() { // Disabled? if (!enabled()) return; // Update the storage of keys having a current mission goal. storeKeysOfCurrentGoals(); // Immediately reflect the mission journal (Step icons) refreshIconsOfScene(true); } bool CNPCIconCache::isNPCaCurrentGoal(TNPCIconCacheKey npcIconCacheKey) const { // There aren't many simultaneous goals, we can safely browse the vector for (CSmallKeyList::const_iterator ikl=_KeysOfCurrentGoals.begin(); ikl!=_KeysOfCurrentGoals.end(); ++ikl) { if ((*ikl) == npcIconCacheKey) return true; } return false; } void CNPCIconCache::storeKeysOfCurrentGoals() { // This event is very unfrequent, and the number of elements of _KeysOfCurrentGoals is usually very small // (typically 0 to 3, while theoretical max is 15*20) so we don't mind rebuilding the list. _KeysOfCurrentGoals.clear(); CCDBNodeBranch *missionNode = safe_cast<CCDBNodeBranch*>(IngameDbMngr.getNodePtr()->getNode(ICDBNode::CTextId("MISSIONS"))); BOMB_IF (!missionNode, "MISSIONS node missing in DB", return); uint nbCurrentMissionSlots = missionNode->getNbNodes(); for (uint i=0; i!=nbCurrentMissionSlots; ++i) { ICDBNode *missionEntry = missionNode->getNode((uint16)i); ICDBNode::CTextId titleNode("TITLE"); if (missionEntry->getProp(titleNode) == 0) continue; CCDBNodeBranch *stepsToDoNode = safe_cast<CCDBNodeBranch*>(missionEntry->getNode(ICDBNode::CTextId("GOALS"))); BOMB_IF(!stepsToDoNode, "GOALS node missing in MISSIONS DB", return); uint nbGoals = stepsToDoNode->getNbNodes(); for (uint j=0; j!=nbGoals; ++j) { ICDBNode *stepNode = stepsToDoNode->getNode((uint16)j); CCDBNodeLeaf *aliasNode = safe_cast<CCDBNodeLeaf*>(stepNode->getNode(ICDBNode::CTextId("NPC_ALIAS"))); BOMB_IF(!aliasNode, "NPC_ALIAS node missing in MISSIONS DB", return); TNPCIconCacheKey npcIconCacheKey = (TNPCIconCacheKey)aliasNode->getValue32(); if (npcIconCacheKey != 0) _KeysOfCurrentGoals.push_back(npcIconCacheKey); } } } void CNPCIconCache::refreshIconsOfScene(bool force) { // Browse all NPCs in vision, and refresh their inscene interface for (uint i=0; i<EntitiesMngr.entities().size(); ++i) { CEntityCL *entity = EntitiesMngr.entity(i); if (!entity) continue; CMissionGiverMap::iterator it = _MissionGivers.find(CNPCIconCache::entityToKey(entity)); if ((it!=_MissionGivers.end()) && ((*it).second.hasChanged() || force)) { EntitiesMngr.refreshInsceneInterfaceOfFriendNPC(i); (*it).second.setChanged(false); } } } bool CNPCIconCache::onReceiveMissionAvailabilityForThisChar(TNPCIconCacheKey npcIconCacheKey, NPC_ICON::TNPCMissionGiverState state) { CMissionGiverMap::iterator img = _MissionGivers.find(npcIconCacheKey); BOMB_IF(img == _MissionGivers.end(), "Mission Giver " << npcIconCacheKey << "not found", return false); //if (state != NPC_ICON::NotAMissionGiver) //{ // static set<TNPCIconCacheKey> qualifs; // qualifs.insert(npcIconCacheKey); // nldebug("NPC %u qualifies (total=%u)", npcIconCacheKey, qualifs.size()); //} return (*img).second.updateMissionAvailabilityForThisChar(state); } bool CNPCMissionGiverDesc::updateMissionAvailabilityForThisChar(NPC_ICON::TNPCMissionGiverState state) { _HasChanged = (state != _MissionGiverState); _MissionGiverState = state; _LastUpdateTimestamp = NetMngr.getCurrentServerTick(); _IsDescTransient = false; return _HasChanged; } void CNPCIconCache::setMissionGiverTimer(NLMISC::TGameCycle delay) { _CacheRefreshTimerDelay = delay; _CatchallTimerPeriod = delay; } std::string CNPCIconCache::getDump() const { string s = toString("System %s\nCurrent timers: %u %u\n", _Enabled?"enabled":"disabled", _CacheRefreshTimerDelay, _CatchallTimerPeriod); s += toString("%u NPCs in mission giver map:\n", _MissionGivers.size()); for (CMissionGiverMap::const_iterator img=_MissionGivers.begin(); img!=_MissionGivers.end(); ++img) { const CNPCMissionGiverDesc& giver = (*img).second; s += toString("NPC %u: ", (*img).first) + giver.getDump() + "\n"; } s += "Current NPC goals:\n"; for (CSmallKeyList::const_iterator ikl=_KeysOfCurrentGoals.begin(); ikl!=_KeysOfCurrentGoals.end(); ++ikl) { s += toString("NPC %u", (*ikl)); } return s; } std::string CNPCMissionGiverDesc::getDump() const { return toString("%u [%u s ago]", _MissionGiverState, (NetMngr.getCurrentServerTick()-_LastUpdateTimestamp)/10); } void CNPCIconCache::setEnabled(bool b) { if (!_Enabled && b) { _Enabled = b; addObservers(); // with _Enabled true storeKeysOfCurrentGoals(); // import from the DB refreshIconsOfScene(true); } else if (_Enabled && !b) { removeObservers(); // with _Enabled true _Enabled = b; refreshIconsOfScene(true); } } #ifndef FINAL_VERSION #error FINAL_VERSION should be defined (0 or 1) #endif #if !FINAL_VERSION NLMISC_COMMAND(dumpNPCIconCache, "Display descriptions of NPCs", "") { log.displayNL(CNPCIconCache::getInstance().getDump().c_str()); return true; } NLMISC_COMMAND(queryMissionGiverData, "Query mission giver data for the specified alias", "<alias>") { if (args.size() == 0) return false; uint32 alias; NLMISC::fromString(args[0], alias); CNPCIconCache::getInstance().queryMissionGiverData(alias); //giver.setDescTransient(); return true; } #endif //#pragma optimize ("", on)