// Ryzom - MMORPG Framework // 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 . #ifndef RY_SPHRASE_MANAGER_H #define RY_SPHRASE_MANAGER_H #include "nel/misc/types_nl.h" #include "game_share/sphrase_com.h" #include "game_share/skills.h" #include "game_share/brick_types.h" #include "game_share/resistance_type.h" #include "req_skill_formula.h" #include "brick_learned_callback.h" #include "skill_change_callback.h" #include "trade_common.h" #include "nel/gui/interface_element.h" #include "../time_client.h" // *************************************************************************** const std::string PHRASE_DB_BOOK="UI:PHRASE:BOOK"; const std::string PHRASE_DB_PROGRESSION[2]= {"UI:PHRASE:PROGRESS_ACTIONS", "UI:PHRASE:PROGRESS_UPGRADES"}; const std::string PHRASE_DB_MEMORY="UI:PHRASE:MEMORY"; const std::string PHRASE_DB_MEMORY_ALT="UI:PHRASE:MEMORY_ALT"; const std::string PHRASE_DB_EXECUTE_NEXT="UI:PHRASE:EXECUTE_NEXT:PHRASE"; const std::string PHRASE_DB_EXECUTE_NEXT_IS_CYCLIC="UI:PHRASE:EXECUTE_NEXT:ISCYCLIC"; const std::string PHRASE_DB_BOTCHAT="LOCAL:TRADING"; #define PHRASE_MAX_BOOK_SLOT 512 #define PHRASE_MAX_PROGRESSION_SLOT 512 #define PHRASE_MAX_MEMORY_SLOT 20 #define PHRASE_MAX_BOTCHAT_SLOT TRADE_MAX_ENTRIES // u16 only for client/server com. #define PHRASE_MAX_ID 65535 // For phrase execution counter #define PHRASE_EXECUTE_COUNTER_MASK 255 const std::string PHRASE_DB_COUNTER_NEXT="SERVER:EXECUTE_PHRASE:NEXT_COUNTER"; const std::string PHRASE_DB_COUNTER_CYCLE="SERVER:EXECUTE_PHRASE:CYCLE_COUNTER"; // TODO_OPTIM: too big test for the list_sheet* each frame with 512 entries!!! class CSuccessTableSheet; namespace NLMISC{ class CCDBNodeLeaf; } /** Special helper class for lua export : enclose a phrase into an object accessible to lua * This way we can setup the phrase tooltip in lua code instead of hardcoded C++ stuff letting * the possibility to the user to customize it to his liking. */ class CSPhraseComAdpater : public CReflectableRefPtrTarget { public: CSPhraseCom Phrase; REFLECT_EXPORT_START(CSPhraseComAdpater, CInterfaceElement) REFLECT_LUA_METHOD("getCastTime", luaGetCastTime) REFLECT_LUA_METHOD("getCastRange", luaGetCastRange) REFLECT_LUA_METHOD("getHpCost", luaGetHpCost) REFLECT_LUA_METHOD("getSapCost", luaGetSapCost) REFLECT_LUA_METHOD("getFocusCost", luaGetFocusCost) REFLECT_LUA_METHOD("getStaCost", luaGetStaCost) REFLECT_LUA_METHOD("getName", luaGetName) REFLECT_LUA_METHOD("getDesc", luaGetDesc) REFLECT_LUA_METHOD("getSuccessRate", luaGetSuccessRate) REFLECT_LUA_METHOD("isMagicPhrase", luaIsMagicPhrase) REFLECT_LUA_METHOD("isCombatPhrase", luaIsCombatPhrase) REFLECT_LUA_METHOD("isPowerPhrase", luaIsPowerPhrase) REFLECT_LUA_METHOD("getRegenTime", luaGetRegenTime) REFLECT_LUA_METHOD("getTotalRegenTime", luaGetTotalRegenTime) REFLECT_LUA_METHOD("getPowerDisableTime", luaGetPowerDisableTime) REFLECT_EXPORT_END int luaGetCastTime(CLuaState &ls); int luaGetCastRange(CLuaState &ls); int luaGetHpCost(CLuaState &ls); int luaGetSapCost(CLuaState &ls); int luaGetSuccessRate(CLuaState &ls); int luaGetFocusCost(CLuaState &ls); int luaGetStaCost(CLuaState &ls); int luaGetName(CLuaState &ls); int luaGetDesc(CLuaState &ls); int luaIsMagicPhrase(CLuaState &ls); int luaIsCombatPhrase(CLuaState &ls); int luaIsPowerPhrase(CLuaState &ls); int luaGetRegenTime(CLuaState &ls); int luaGetTotalRegenTime(CLuaState &ls); int luaGetPowerDisableTime(CLuaState &ls); }; // *************************************************************************** /** * Singleton to Get/Set Sabrina Phrase in SpellBook / Memory. * NB: you MUST create it (getInstance()), before loading of ingame.xmls. * \author Lionel Berenguier * \author Nevrax France * \date 2003 */ class CSPhraseManager { public: static CSPhraseManager *getInstance() { if(!_Instance) _Instance= new CSPhraseManager; return _Instance; } // release singleton static void releaseInstance(); // The Slot 0 of the Book indicate "NoPhrase". // The Slot 1 indicate the Edited Phrase for phrase composition, and is not "really" a part of the book. enum {EditionSlot= 1, BookStartSlot= 2}; /// To call when The DB inGame is setup. Else, no write is made to it before. (NB: DB is updated here) void initInGame(); /// Reset void reset(); /// Call juste once when interface is built void updateMemoryBar(); /// Call once per pass void update(); // \name Book Edition // @{ /** set a phrase on a slot. * \param lock special for BotChat phrase buy. when true, the phrase is never written in the book DB * the server has to unlock it (or delete if not confirmed) * NO MSG SENT */ void setPhrase(uint32 slot, const CSPhraseCom &phrase, bool lock= false); /** Same as setPhrase() but do not update the DB of book now * User can use this if lot of phrase must be chaned in one time. * User must call updateBookDB() after setting all those phrases. * Nb: suppose lock==false. */ void setPhraseNoUpdateDB(uint32 slot, const CSPhraseCom &phrase); /// update Action Book Database. You need only to use it in conjunction with setPhraseNoUpdateDB() void updateBookDB(); /// erase a phrase on a slot. NO MSG SENT void erasePhrase(uint32 slot); /// get a phrase for a slot. Empty Phrase returned if don't exist. O(logN) const CSPhraseCom &getPhrase(uint32 slot) const; /// get the number of memory lines. uint32 getNbMemoryLines() const { return (uint32)_Memories.size(); } /** Allocate a Free Slot (for NEW phrase setup). 0 is not a free slot * NB: the slot is not still correctly filled(), and getPhrase() will return Empty. use setPhrase() just after * \return 0 if not possible (eg too big) */ uint32 allocatePhraseSlot(); /// true if a free slot is available (don't allocate) bool hasFreeSlot() const; /// get a phrase version for a slot. -1 if phrase not present. O(1) sint32 getPhraseVersion(uint32 slot) const; // Receive a BotChat phrase Buy confirmation: NB: memorize it (and server send) if possible void receiveBotChatConfirmBuy(uint16 phraseId, bool confirm); // test if the given phrase is known bool isPhraseKnown(const CSPhraseCom &phrase) const; /** Filter the BOOK DB with this brickType or skill. * NB: if brickTypeFilter is UNKNOWN then filtered by skill * il filterSkill is unknown, then not filtered (all match). This is the default. */ void setBookFilter(BRICK_TYPE::EBrickType brickTypeFilter, SKILLS::ESkills skillFilter); /// get the book skill filter SKILLS::ESkills getBookSkillFilter() const {return _BookSkillFitler;} /// Send the PHRASE:DELETE message to the server void sendDeleteToServer(uint32 slot); /// Send the PHRASE:LEARN message to the server void sendLearnToServer(uint32 slot); /// This is a clean full delete: forget all phrase before, erase the phrase, send msg to server void fullDelete(uint32 slot); /** Do it only if the phrase id is only used in this memory slot * This is a clean full delete: forget all phrase before, erase the phrase, send msg to server */ void fullDeletePhraseIfLast(uint32 memoryLine, uint32 memorySlot); // @} // \name Memory Edition // @{ /// Memorize a phrase (no MSG send) void memorizePhrase(uint32 memoryLine, uint32 memorySlot, uint32 slot); /// Forget a phrase (no MSG send). NB: no-op if it is a macro void forgetPhrase(uint32 memoryLine, uint32 memorySlot); /// Get the phrase memorized. 0 if not found or if it is a macro uint32 getMemorizedPhrase(uint32 memoryLine, uint32 memorySlot) const; /// Memorize a macro (no MSG send). if was a phrase, forget first (client and server) void memorizeMacro(uint32 memoryLine, uint32 memorySlot, uint32 macroId); /// Forget a macro. NB: no-op if it is a phrase void forgetMacro(uint32 memoryLine, uint32 memorySlot); /// Get the phrase memorized. 0 if not found or if it is a macro uint32 getMemorizedMacro(uint32 memoryLine, uint32 memorySlot) const; /// return true if it is a memorized macro. false if empty or if it is a phrase bool isMemorizedMacro(uint32 memoryLine, uint32 memorySlot) const; /// To Be called when the macro visual changes void updateMacroShortcuts(sint32 macroId); /// To be called when the Id changes void deleteMacroShortcuts(sint32 macroId); /// Only one memory line is displayed in the Memory DB. if -1, erased. void selectMemoryLineDB(sint32 memoryLine); void selectMemoryLineDBalt(sint32 memoryLine); /// get the selected memory line. sint32 getSelectedMemoryLineDB() const {return _SelectedMemoryDB;} sint32 getSelectedMemoryAltLineDB() const {return _SelectedMemoryDBalt;} /// Common Method to send the Memorize msg to server void sendMemorizeToServer(uint32 memoryLine, uint32 memorySlot, uint32 phraseId); /// Common Method to send the Forget msg to server void sendForgetToServer(uint32 memoryLine, uint32 memorySlot); /// count all memories that use this phrase uint countAllThatUsePhrase(uint32 slot); /// Forget all Memories that use this phrase slot, AND send appropriate message to server. if sendMsgOnly==true, only the msg is sent to server (no modification to local) void forgetAllThatUsePhrase(uint32 slot, bool sendMsgOnly= false); /// Rememorize all memories that use this slot void rememorizeAllThatUsePhrase(uint32 slot); /// Update all Ctrl State of the action Bar, according to macro/itemInRightHand/available states void updateAllMemoryCtrlState(); /// Load or save the macro void serialMacroMemory(NLMISC::IStream &f); /// is this phrase sheet can be memorized / casted? bool isPhraseCastable(uint32 sheetId) const; bool isPhraseCastable(class CSPhraseSheet *phraseSheet) const; bool isPhraseCharacBuying(uint32 sheetId) const; bool isPhraseCharacBuying(class CSPhraseSheet *phraseSheet) const; /** return getMemorizedPhrase(memoryLine, memoryIndex), only if this phrase is only used in this memory slot * else return allocatePhraseSlot() */ uint32 getMemorizedPhraseIfLastOrNewSlot(uint32 memoryLine, uint32 memoryIndex); // @} // \name Phrase composition // @{ // The phrase edited/created. If 0, it may be a phrase newly created, but not still assigned. uint32 CompositionPhraseId; // For phrase created only. If not -1, then at creation, will be auto-memorized to this slot sint32 CompositionPhraseMemoryLineDest; sint32 CompositionPhraseMemorySlotDest; /// For BotChat learning, build a phrase from .sphrase sheetId void buildPhraseFromSheet(CSPhraseCom &phrase, sint32 sheetId); /** For a given phrase, see if there's a matching sheet, or 0 else. * NB : O(1) algo with respect to the number of phrase sheets */ sint32 getSheetFromPhrase(const CSPhraseCom &phrase) const; // @} // \name Phrase Execution // @{ /// Start execution from Client. Only the bitmaps are displayed. A sendExecuteToServer() or a cancelClientExecution() must follow for each clientExecute void clientExecute(uint32 memoryLine, uint32 memorySlot, bool cyclic); /// Common Method to send the execution msg to server void sendExecuteToServer(uint32 memoryLine, uint32 memorySlot, bool cyclic); /// Cancel the client execution (case when used with Move in combat for instance) void cancelClientExecute(bool cyclic); /// Execute a craft action (after selecting the plan and MPs), on both client and server side void executeCraft(uint32 memoryLine, uint32 memorySlot, uint32 planSheetId, std::vector &mpItemPartList, std::vector specificItemList); /// Execute a cristalize action on both client and server side void executeCristalize(uint32 memoryLine, uint32 memorySlot); /** Execute a default attack, on both client and server side. * NB: try first to launch a standard action "default attack" if found, * calling clientExecute() and sendExecuteToServer() */ void executeDefaultAttack(); /// Search the default attack like on client. false if not found. NB: find preferly in current memoryLine bool findDefaultAttack(uint32 &memoryLine, uint32 &memorySlot); uint getPhraseNextExecuteCounter() const {return _PhraseNextExecuteCounter;} uint getPhraseCycleExecuteCounter() const {return _PhraseCycleExecuteCounter;} /// Server Execution ACK void receiveAckExecuteFromServer(bool cyclic, uint counterValue, bool ok); /// get the Next PhraseId Executed (0 if none) uint32 getNextExecutePhraseId() const {return _CurrentExecutePhraseIdNext;} /// get the Cycle PhraseId Executed (0 if none) uint32 getCycleExecutePhraseId() const {return _CurrentExecutePhraseIdCycle;} // @} // \name Misc // @{ /** Build the Text description of a phrase (spell). NB: this is a TagFormated text * \phraseSheetId not 0 if comes from a .sphrase Sheet => used to show progression info, and display phrase description * \specialPhraseFormat if empty, format is auto selected. if "composition", same but the text is cut under the %compostart tag. * else take directly this format. */ void buildPhraseDesc(ucstring &text, const CSPhraseCom &phrase, uint32 phraseSheetId, bool wantRequirement, const std::string &specialPhraseFormat= std::string()); // Get the Phrase Success Rate % sint getPhraseSuccessRate(const CSPhraseCom &phrase); // Get the Phrase Success Rate %. Manually gives the Skill to do the comparison (for craft) sint getCraftPhraseSuccessRate(const CSPhraseCom &phrase, SKILLS::ESkills skill, uint minMpLevel); // Get the Phrase Success Rate %. Manually gives the Skill to do the comparison (for Forage Extraction) sint getForageExtractionPhraseSuccessRate(const CSPhraseCom &phrase, SKILLS::ESkills skill); // return the fmt according to forage terrain specializing ucstring getForageExtractionPhraseEcotypeFmt(const CSPhraseCom &phrase); // Get the Phrase Sap Cost void getPhraseSapCost(const CSPhraseCom &phrase, uint32 totalActionMalus, sint &cost, sint &costMalus); // Get the Phrase Sta Cost void getPhraseStaCost(const CSPhraseCom &phrase, uint32 totalActionMalus, sint &cost, sint &costMalus); // Get the Phrase Focus Cost void getPhraseFocusCost(const CSPhraseCom &phrase, uint32 totalActionMalus, sint &cost, sint &costMalus); // Get the Phrase Hp Cost void getPhraseHpCost(const CSPhraseCom &phrase, uint32 totalActionMalus, sint &cost, sint &costMalus); // Get the Phrase Cast Time void getPhraseCastTime(const CSPhraseCom &phrase, uint32 totalActionMalus, float &castTime, float &castTimeMalus); // Get the Phrase Range void getPhraseMagicRange(const CSPhraseCom &phrase, uint32 totalActionMalus, sint &range, sint &rangeMalus); // For all bricks, get the sum of the propId property float getPhraseSumBrickProp(const CSPhraseCom &phrase, uint propId, bool doAbs=false); /// If the temporary inventory is opened so we cant execute action bool isExecutionInvalidCauseTempInv() const; /// build a list of bricks that are needed in order to learn this phrase. NB: any brick already contained in the phrase are removed void buildPhraseBrickRequirement(uint32 phraseSheetId, std::vector &bricks); /// check if all bricks of phrase meet fame requirement bool fameRequirementOk(uint32 phraseSheetId); /// true if interesting to list the bricks of this phrase in help bool allowListBrickInHelp(const CSPhraseCom &phrase) const; /// return the combat restriction text (empty if not combat) void getCombatWeaponRestriction(ucstring &text, const CSPhraseCom &phrase, bool& usableWithMelee, bool& usableWithRange); void getCombatWeaponRestriction(ucstring &text, sint32 phraseSheetId, bool& usableWithMelee, bool& usableWithRange); // return true if any of the Bricks contains AvoidCyclic==true (the phrase cannot be cyclic) bool avoidCyclicForPhrase(const CSPhraseCom &phrase) const; bool avoidCyclicForPhrase(sint32 phraseSheetId) const; // Get total regeneration time for a power/aura phrase, in tick (0 if not a power/aura) NLMISC::TGameCycle getPowerDisableTime(const CSPhraseCom &phrase) const; // Get current time left for a power/aura phrase, in ticks (0 if not a power/aura) NLMISC::TGameCycle getRegenTime(const CSPhraseCom &phrase) const; // Get total time span necessary for a power/aura phrase to be available again (in ticks) NLMISC::TGameCycle getTotalRegenTime(const CSPhraseCom &phrase) const; /// Lookup a SuccessTable and return chance enum TSuccessTable { STCombat= 0, STOffensiveMagic, STCurativeMagic, STCraft, STExtract, STResistMagic, STResistMagicLink, STDodgeParry, NumSuccessTable }; sint getSuccessRate(TSuccessTable st, sint level, bool partialSuccess= false); // @} // \name Phrase Execution invalidation cause of Equip // @{ void setEquipInvalidation(sint64 serverTick, sint invalidTickTime); void updateEquipInvalidation(sint64 serverTick); bool isExecutionInvalidCauseEquip() const; // @} // \name BotChat special // @{ // get the current skill point price of a phrase sheet uint32 getCurrentPhraseSheetPrice(uint32 phraseSheetId); // get the base skill point price of a phrase sheet (ie don't check if bricks learned or not) uint32 getBasePhraseSheetPrice(uint32 phraseSheetId); // update bot chat prices in the range [start, end[ void updateBotChatPhrasePrice(uint start, uint end); // From a phrase required level, retrieve its section (for interface listing) uint32 getPhraseSectionFromLevel(uint32 level); // inverse void getPhraseLevelFromSection(uint32 section, uint32 &minLevel, uint32 &maxLevel); // For book display of section void getPhraseSectionBoundFromSkillFilter(sint &minSectionId, sint &maxSectionId); // Client Side Rolemaster phrase generation. return the number of db node filled uint fillRoleMasterGenericPhrase(uint32 rmFlags, uint32 rmRace); // @} /// true if the compareSkill is compatible with this combat phrase, eg more restrictive for instance bool skillCompatibleWithCombatPhrase(SKILLS::ESkills compareSkill, const std::vector &phraseBricks) const; bool skillCompatibleWithSpecialPowerPhrase(SKILLS::ESkills compareSkill, const std::vector &phraseBricks) const; /// true if phrase is usable with empty hands bool phraseUsableWithEmptyHands(const std::vector &phraseBricks) const; // get total action malus uint32 getTotalActionMalus(const CSPhraseCom &phrase) const; void updateAllActionRegen(); void touchRegenTickRangeFlag() { _RegenTickRangeTouched = true; } static uint64 getPhraseRequiredFlags(const CSPhraseCom &phrase); // **************** private: /// Constructor CSPhraseManager(); ~CSPhraseManager(); static CSPhraseManager *_Instance; typedef std::map TPhraseMap; typedef TPhraseMap::iterator ItPhraseMap; typedef std::map TPhraseToSheet; // map each phrase to its sheet id // Map of All Phrase. Contains the Book + some system phrase (1: the Edition Phrase) TPhraseMap _PhraseMap; TPhraseToSheet _PhraseToSheet; std::set _RegenPhrases; // timeouts for the power / auras bricks CTickRange _BrickRegenRange[64]; // Shortcut To Phrases Leaves std::vector _BookDbLeaves; std::vector _MemoryDbLeaves; std::vector _MemoryAltDbLeaves; NLMISC::CCDBNodeLeaf *_NextExecuteLeaf; NLMISC::CCDBNodeLeaf *_NextExecuteIsCyclicLeaf; // extra client data struct CPhraseClient { sint32 Version; bool Lock; CPhraseClient() { Version= 0; Lock= false; } }; std::vector _PhraseClient; uint32 _MaxSlotSet; std::vector _FreeSlots; bool _RegenTickRangeTouched; // Memory struct CMemorySlot { // Is this a macro bool IsMacro; // Is his visual dirty? bool IsMacroVisualDirty; // Macro or PhraseId uint32 Id; CMemorySlot() { IsMacro= false; IsMacroVisualDirty= false; Id= 0; } // suppose Id==0 is valid for a macro, but not for a phrase bool isEmpty() const { return Id==0 && IsMacro==false; } // Macro possible only if IsMacro==true bool isMacro() const { return IsMacro; } // phrase only if not macro, and Id!=0 bool isPhrase() const { return !IsMacro && Id!=0; } }; struct CMemoryLine { CMemorySlot Slot[PHRASE_MAX_MEMORY_SLOT]; }; std::vector _Memories; sint32 _SelectedMemoryDB; sint32 _SelectedMemoryDBalt; void updateMemoryDBAll(); void updateMemoryDBSlot(uint32 memorySlot); // The number of entries setuped to not 0 uint _LastBookNumDbFill; // The Book filter. SKILLS::ESkills _BookSkillFitler; BRICK_TYPE::EBrickType _BookBrickTypeFilter; // true if the phrase is "compatbile" with _BookSkillFitler bool matchBookSkillFilter(const CSPhraseCom &phrase) const; uint32 _PhraseNextExecuteCounter; uint32 _PhraseCycleExecuteCounter; bool _InitInGameDone; /// unlock a phrase (for BotChat Phrase buy confirmation) void unlockPhrase(uint32 slot); // Phrase compute related CSuccessTableSheet *_SuccessTableSheet[NumSuccessTable]; void loadSuccessTable(); // The user was indoor bool _UserIndoor; // called by getPhraseSuccessRate() public sint getPhraseSuccessRate(TSuccessTable st, const CSPhraseCom &phrase, sint skillValue, uint minMpLevel); // real stuff void setPhraseInternal(uint32 slot, const CSPhraseCom &phrase, bool lock, bool updateDB); // If the local counter and server counter are same, then view will be hid. bool isPhraseNextExecuteCounterSync() const; bool isPhraseCycleExecuteCounterSync() const; // setup by AH. -1 if not activated sint32 _CurrentExecuteLineNext; sint32 _CurrentExecuteSlotNext; uint32 _CurrentExecutePhraseIdNext; sint32 _CurrentExecuteLineCycle; sint32 _CurrentExecuteSlotCycle; uint32 _CurrentExecutePhraseIdCycle; // For debug only. setuped by sendExecuteToServer() in local mode only sint64 _PhraseDebugEndNextAction; sint64 _PhraseDebugEndCyclicAction; friend class CHandlerPhraseCounterUpdate; friend class CHandlerPhraseDebugClient; // Equip invalidation sint64 _EquipInvalidationEnd; sint64 _CurrentServerTick; // update the execution views void updateExecutionDisplay(); // update the memory ctrl void updateMemoryCtrlState(uint memorySlot, class CDBCtrlSheet *ctrl, SKILLS::ESkills itemSkill); // update the ith memory ctrl in the action bar void updateMemoryCtrlState(uint memorySlot); // Shortcut To PhraseSheets Leaves in BotChat std::vector _BotChatPhraseSheetLeaves; std::vector _BotChatPhrasePriceLeaves; // For phrase compatibility with enchant weapon special power NLMISC::CSheetId _EnchantWeaponMainBrick; // \name Phrase sheet progression // @{ enum TProgressType { ActionProgress= 0, UpgradeProgress, NumProgressType }; std::vector _ProgressionDbSheets[NumProgressType]; std::vector _ProgressionDbLevels[NumProgressType]; std::vector _ProgressionDbLocks[NumProgressType]; // For each skill, which phrase are learned when the skill is gained class CPhraseProgressInfo { public: uint32 SheetId; uint32 Level; }; friend class CPhraseSortEntry; class CPhraseProgression { public: // All the phrase that can be learned at this level, anbd the required level of the skill std::vector Phrases; }; CPhraseProgression _ProgressionPhrases[SKILLS::NUM_SKILLS]; // For each phrase SheetId, gives the required Skill formula (for help) typedef std::map TPhraseReqSkillMap; TPhraseReqSkillMap _PhraseReqSkillMap; // For db fill uint _LastProgressionNumDbFill[NumProgressType]; // For db update when a brick is learned class CProgressionUpdate : public IBrickLearnedCallback, public ISkillChangeCallback { public: virtual void onBrickLearned() { CSPhraseManager *pPM= CSPhraseManager::getInstance(); pPM->updatePhraseProgressionDB(); } virtual void onSkillChange() { CSPhraseManager *pPM= CSPhraseManager::getInstance(); pPM->updatePhraseProgressionDB(); } }; friend class CProgressionUpdate; CProgressionUpdate _ProgressionUpdate; // methods void updatePhraseProgressionDB(); // called at initInGame void computePhraseProgression(); void insertProgressionSkillRecurs(SKILLS::ESkills skill, uint32 value, sint *skillReqLevel, std::vector &skillsToInsert); // @} /// return the skill of the root SKILLS::ESkills getPhraseRootSkill(const std::vector &phraseBricks) const; /// return true if all the requirement in order to learn this prhase are met. bool phraseRequirementOk(uint32 phraseSheetId); /// Execution ACK reception // @{ struct CAckExecuteEntry { enum TState { WaitAck= 0, OK, KO }; TState State; uint32 Counter; sint32 MemoryLine; sint32 MemorySlot; uint32 PhraseId; }; typedef std::deque TAckExecuteList; TAckExecuteList _AckExecuteCycleList; TAckExecuteList _AckExecuteNextList; void appendCurrentToAckExecute(bool cyclic); void resetAckExecuteList(bool cyclic); // @} ucstring formatMalus(sint base, sint malus); ucstring formatMalus(float base, float malus); std::string formatBonusMalus(sint32 base, sint32 mod); // Special for combat: Build the "phrase skill compatible" formula // NB: Use a ReqSkillFormula, but don't use the value part, cause no requirement is made on it CReqSkillFormula buildCombatPhraseSkillFormula(const std::vector &phraseBricks) const; // For Magic and Magic Staff uint32 getSpellLevel(const std::vector &phraseBricks) const; // For Magic void getResistMagic(bool resistMagic[RESISTANCE_TYPE::NB_RESISTANCE_TYPE], const std::vector &phraseBricks); void updateAllMemoryCtrlRegenTickRange(); void updateMemoryCtrlRegenTickRange(uint memorySlot, CDBCtrlSheet *ctrl); void updateMemoryCtrlRegenTickRange(uint memorySlot); CDBCtrlSheet *getMemorySlotCtrl(uint memorySlot); CDBCtrlSheet *getMemoryAltSlotCtrl(uint memorySlot); CTickRange getRegenTickRange(const CSPhraseCom &phrase) const; NLMISC::CCDBNodeLeaf *getRegenTickRangeDbLeaf(uint powerIndex) const; // get regen tick range for a specific power, from the database CTickRange getRegenTickRange(uint powerIndex) const; public: // tmp for test : set regen tick range locally void setRegenTickRange(uint powerIndex, const CTickRange &tickRange); }; #endif // NL_SPHRASE_MANAGER_H /* End of sphrase_manager.h */