mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 10:56:40 +00:00 
			
		
		
		
	Add a separate stats watcher for GUI
This commit is contained in:
		
							parent
							
								
									85b4b84342
								
							
						
					
					
						commit
						27f1a3376c
					
				
					 15 changed files with 340 additions and 272 deletions
				
			
		|  | @ -41,7 +41,7 @@ add_openmw_dir (mwgui | ||||||
|     itemmodel containeritemmodel inventoryitemmodel sortfilteritemmodel itemview |     itemmodel containeritemmodel inventoryitemmodel sortfilteritemmodel itemview | ||||||
|     tradeitemmodel companionitemmodel pickpocketitemmodel controllers savegamedialog |     tradeitemmodel companionitemmodel pickpocketitemmodel controllers savegamedialog | ||||||
|     recharge mode videowidget backgroundimage itemwidget screenfader debugwindow spellmodel spellview |     recharge mode videowidget backgroundimage itemwidget screenfader debugwindow spellmodel spellview | ||||||
|     draganddrop timeadvancer jailscreen itemchargeview keyboardnavigation textcolours |     draganddrop timeadvancer jailscreen itemchargeview keyboardnavigation textcolours statswatcher | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
| add_openmw_dir (mwdialogue | add_openmw_dir (mwdialogue | ||||||
|  |  | ||||||
|  | @ -67,10 +67,6 @@ namespace MWBase | ||||||
|             virtual void drop (const MWWorld::CellStore *cellStore) = 0; |             virtual void drop (const MWWorld::CellStore *cellStore) = 0; | ||||||
|             ///< Deregister all objects in the given cell.
 |             ///< Deregister all objects in the given cell.
 | ||||||
| 
 | 
 | ||||||
|             virtual void watchActor (const MWWorld::Ptr& ptr) = 0; |  | ||||||
|             ///< On each update look for changes in a previously registered actor and update the
 |  | ||||||
|             /// GUI accordingly.
 |  | ||||||
| 
 |  | ||||||
|             virtual void update (float duration, bool paused) = 0; |             virtual void update (float duration, bool paused) = 0; | ||||||
|             ///< Update objects
 |             ///< Update objects
 | ||||||
|             ///
 |             ///
 | ||||||
|  |  | ||||||
|  | @ -154,27 +154,11 @@ namespace MWBase | ||||||
| 
 | 
 | ||||||
|             virtual void setConsoleSelectedObject(const MWWorld::Ptr& object) = 0; |             virtual void setConsoleSelectedObject(const MWWorld::Ptr& object) = 0; | ||||||
| 
 | 
 | ||||||
|             /// Set value for the given ID.
 |  | ||||||
|             virtual void setValue (const std::string& id, const MWMechanics::AttributeValue& value) = 0; |  | ||||||
|             virtual void setValue (int parSkill, const MWMechanics::SkillValue& value) = 0; |  | ||||||
|             virtual void setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value) = 0; |  | ||||||
|             virtual void setValue (const std::string& id, const std::string& value) = 0; |  | ||||||
|             virtual void setValue (const std::string& id, int value) = 0; |  | ||||||
| 
 |  | ||||||
|             /// Set time left for the player to start drowning (update the drowning bar)
 |             /// Set time left for the player to start drowning (update the drowning bar)
 | ||||||
|             /// @param time time left to start drowning
 |             /// @param time time left to start drowning
 | ||||||
|             /// @param maxTime how long we can be underwater (in total) until drowning starts
 |             /// @param maxTime how long we can be underwater (in total) until drowning starts
 | ||||||
|             virtual void setDrowningTimeLeft (float time, float maxTime) = 0; |             virtual void setDrowningTimeLeft (float time, float maxTime) = 0; | ||||||
| 
 | 
 | ||||||
|             virtual void setPlayerClass (const ESM::Class &class_) = 0; |  | ||||||
|             ///< set current class of player
 |  | ||||||
| 
 |  | ||||||
|             virtual void configureSkills (const SkillList& major, const SkillList& minor) = 0; |  | ||||||
|             ///< configure skill groups, each set contains the skill ID for that group.
 |  | ||||||
| 
 |  | ||||||
|             virtual void updateSkillArea() = 0; |  | ||||||
|             ///< update display of skills, factions, birth sign, reputation and bounty
 |  | ||||||
| 
 |  | ||||||
|             virtual void changeCell(const MWWorld::CellStore* cell) = 0; |             virtual void changeCell(const MWWorld::CellStore* cell) = 0; | ||||||
|             ///< change the active cell
 |             ///< change the active cell
 | ||||||
| 
 | 
 | ||||||
|  | @ -359,6 +343,9 @@ namespace MWBase | ||||||
|             virtual void windowResized(int x, int y) = 0; |             virtual void windowResized(int x, int y) = 0; | ||||||
|             virtual void windowClosed() = 0; |             virtual void windowClosed() = 0; | ||||||
|             virtual bool isWindowVisible() = 0; |             virtual bool isWindowVisible() = 0; | ||||||
|  | 
 | ||||||
|  |             virtual void watchActor(const MWWorld::Ptr& ptr) = 0; | ||||||
|  |             virtual MWWorld::Ptr getWatchedActor() const = 0; | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -368,7 +368,6 @@ namespace MWGui | ||||||
|             if (klass) |             if (klass) | ||||||
|             { |             { | ||||||
|                 mPlayerClass = *klass; |                 mPlayerClass = *klass; | ||||||
|                 MWBase::Environment::get().getWindowManager()->setPlayerClass(mPlayerClass); |  | ||||||
|             } |             } | ||||||
|             MWBase::Environment::get().getWindowManager()->removeDialog(mPickClassDialog); |             MWBase::Environment::get().getWindowManager()->removeDialog(mPickClassDialog); | ||||||
|             mPickClassDialog = 0; |             mPickClassDialog = 0; | ||||||
|  | @ -422,7 +421,6 @@ namespace MWGui | ||||||
|         if (mNameDialog) |         if (mNameDialog) | ||||||
|         { |         { | ||||||
|             mPlayerName = mNameDialog->getTextInput(); |             mPlayerName = mNameDialog->getTextInput(); | ||||||
|             MWBase::Environment::get().getWindowManager()->setValue("name", mPlayerName); |  | ||||||
|             MWBase::Environment::get().getMechanicsManager()->setPlayerName(mPlayerName); |             MWBase::Environment::get().getMechanicsManager()->setPlayerName(mPlayerName); | ||||||
|             MWBase::Environment::get().getWindowManager()->removeDialog(mNameDialog); |             MWBase::Environment::get().getWindowManager()->removeDialog(mNameDialog); | ||||||
|             mNameDialog = 0; |             mNameDialog = 0; | ||||||
|  | @ -525,7 +523,6 @@ namespace MWGui | ||||||
| 
 | 
 | ||||||
|             MWBase::Environment::get().getMechanicsManager()->setPlayerClass(klass); |             MWBase::Environment::get().getMechanicsManager()->setPlayerClass(klass); | ||||||
|             mPlayerClass = klass; |             mPlayerClass = klass; | ||||||
|             MWBase::Environment::get().getWindowManager()->setPlayerClass(klass); |  | ||||||
| 
 | 
 | ||||||
|             // Do not delete dialog, so that choices are remembered in case we want to go back and adjust them later
 |             // Do not delete dialog, so that choices are remembered in case we want to go back and adjust them later
 | ||||||
|             mCreateClassDialog->setVisible(false); |             mCreateClassDialog->setVisible(false); | ||||||
|  | @ -722,7 +719,6 @@ namespace MWGui | ||||||
|             MWBase::Environment::get().getWorld()->getStore().get<ESM::Class>().find(mGenerateClass); |             MWBase::Environment::get().getWorld()->getStore().get<ESM::Class>().find(mGenerateClass); | ||||||
| 
 | 
 | ||||||
|         mPlayerClass = *klass; |         mPlayerClass = *klass; | ||||||
|         MWBase::Environment::get().getWindowManager()->setPlayerClass(mPlayerClass); |  | ||||||
| 
 | 
 | ||||||
|         updatePlayerHealth(); |         updatePlayerHealth(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -1,13 +1,12 @@ | ||||||
| #ifndef CHARACTER_CREATION_HPP | #ifndef CHARACTER_CREATION_HPP | ||||||
| #define CHARACTER_CREATION_HPP | #define CHARACTER_CREATION_HPP | ||||||
| 
 | 
 | ||||||
| #include <components/esm/loadskil.hpp> |  | ||||||
| #include <components/esm/loadclas.hpp> | #include <components/esm/loadclas.hpp> | ||||||
| 
 | 
 | ||||||
| #include <map> | #include <map> | ||||||
| #include <vector> | #include <vector> | ||||||
| 
 | 
 | ||||||
| #include "../mwmechanics/stat.hpp" | #include "statswatcher.hpp" | ||||||
| 
 | 
 | ||||||
| namespace osg | namespace osg | ||||||
| { | { | ||||||
|  | @ -35,21 +34,21 @@ namespace MWGui | ||||||
|     class ReviewDialog; |     class ReviewDialog; | ||||||
|     class MessageBoxManager; |     class MessageBoxManager; | ||||||
| 
 | 
 | ||||||
|     class CharacterCreation |     class CharacterCreation : public StatsListener | ||||||
|     { |     { | ||||||
|     public: |     public: | ||||||
|     typedef std::vector<int> SkillList; |     typedef std::vector<int> SkillList; | ||||||
| 
 | 
 | ||||||
|     CharacterCreation(osg::Group* parent, Resource::ResourceSystem* resourceSystem); |     CharacterCreation(osg::Group* parent, Resource::ResourceSystem* resourceSystem); | ||||||
|     ~CharacterCreation(); |     virtual ~CharacterCreation(); | ||||||
| 
 | 
 | ||||||
|     //Show a dialog
 |     //Show a dialog
 | ||||||
|     void spawnDialog(const char id); |     void spawnDialog(const char id); | ||||||
| 
 | 
 | ||||||
|     void setValue (const std::string& id, const MWMechanics::AttributeValue& value); |     void setValue (const std::string& id, const MWMechanics::AttributeValue& value) override; | ||||||
|     void setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value); |     void setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value) override; | ||||||
|     void setValue(const ESM::Skill::SkillEnum parSkill, const MWMechanics::SkillValue& value); |     void setValue(const ESM::Skill::SkillEnum parSkill, const MWMechanics::SkillValue& value) override; | ||||||
|     void configureSkills (const SkillList& major, const SkillList& minor); |     void configureSkills(const SkillList& major, const SkillList& minor) override; | ||||||
| 
 | 
 | ||||||
|     void onFrame(float duration); |     void onFrame(float duration); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2,8 +2,7 @@ | ||||||
| #define OPENMW_GAME_MWGUI_HUD_H | #define OPENMW_GAME_MWGUI_HUD_H | ||||||
| 
 | 
 | ||||||
| #include "mapwindow.hpp" | #include "mapwindow.hpp" | ||||||
| 
 | #include "statswatcher.hpp" | ||||||
| #include "../mwmechanics/stat.hpp" |  | ||||||
| 
 | 
 | ||||||
| namespace MWWorld | namespace MWWorld | ||||||
| { | { | ||||||
|  | @ -17,12 +16,12 @@ namespace MWGui | ||||||
|     class ItemWidget; |     class ItemWidget; | ||||||
|     class SpellWidget; |     class SpellWidget; | ||||||
| 
 | 
 | ||||||
|     class HUD : public WindowBase, public LocalMapBase |     class HUD : public WindowBase, public LocalMapBase, public StatsListener | ||||||
|     { |     { | ||||||
|     public: |     public: | ||||||
|         HUD(CustomMarkerCollection& customMarkers, DragAndDrop* dragAndDrop, MWRender::LocalMap* localMapRender); |         HUD(CustomMarkerCollection& customMarkers, DragAndDrop* dragAndDrop, MWRender::LocalMap* localMapRender); | ||||||
|         virtual ~HUD(); |         virtual ~HUD(); | ||||||
|         void setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value); |         void setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value) override; | ||||||
| 
 | 
 | ||||||
|         /// Set time left for the player to start drowning
 |         /// Set time left for the player to start drowning
 | ||||||
|         /// @param time time left to start drowning
 |         /// @param time time left to start drowning
 | ||||||
|  |  | ||||||
							
								
								
									
										209
									
								
								apps/openmw/mwgui/statswatcher.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										209
									
								
								apps/openmw/mwgui/statswatcher.cpp
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,209 @@ | ||||||
|  | #include "statswatcher.hpp" | ||||||
|  | 
 | ||||||
|  | #include "../mwbase/environment.hpp" | ||||||
|  | #include "../mwbase/windowmanager.hpp" | ||||||
|  | #include "../mwbase/world.hpp" | ||||||
|  | 
 | ||||||
|  | #include "../mwmechanics/npcstats.hpp" | ||||||
|  | #include "../mwmechanics/spellutil.hpp" | ||||||
|  | 
 | ||||||
|  | #include "../mwworld/class.hpp" | ||||||
|  | #include "../mwworld/esmstore.hpp" | ||||||
|  | #include "../mwworld/inventorystore.hpp" | ||||||
|  | 
 | ||||||
|  | namespace MWGui | ||||||
|  | { | ||||||
|  |     // mWatchedTimeToStartDrowning = -1 for correct drowning state check,
 | ||||||
|  |     // if stats.getTimeToStartDrowning() == 0 already on game start
 | ||||||
|  |     StatsWatcher::StatsWatcher() | ||||||
|  |       : mWatchedLevel(-1), mWatchedTimeToStartDrowning(-1), mWatchedStatsEmpty(true) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::watchActor(const MWWorld::Ptr& ptr) | ||||||
|  |     { | ||||||
|  |         mWatched = ptr; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::update() | ||||||
|  |     { | ||||||
|  |         if (mWatched.isEmpty()) | ||||||
|  |             return; | ||||||
|  | 
 | ||||||
|  |         MWBase::WindowManager *winMgr = MWBase::Environment::get().getWindowManager(); | ||||||
|  |         const MWMechanics::NpcStats &stats = mWatched.getClass().getNpcStats(mWatched); | ||||||
|  |         for (int i = 0;i < ESM::Attribute::Length;++i) | ||||||
|  |         { | ||||||
|  |             if (stats.getAttribute(i) != mWatchedAttributes[i] || mWatchedStatsEmpty) | ||||||
|  |             { | ||||||
|  |                 std::stringstream attrname; | ||||||
|  |                 attrname << "AttribVal"<<(i+1); | ||||||
|  | 
 | ||||||
|  |                 mWatchedAttributes[i] = stats.getAttribute(i); | ||||||
|  |                 setValue(attrname.str(), stats.getAttribute(i)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (stats.getHealth() != mWatchedHealth || mWatchedStatsEmpty) | ||||||
|  |         { | ||||||
|  |             static const std::string hbar("HBar"); | ||||||
|  |             mWatchedHealth = stats.getHealth(); | ||||||
|  |             setValue(hbar, stats.getHealth()); | ||||||
|  |         } | ||||||
|  |         if (stats.getMagicka() != mWatchedMagicka || mWatchedStatsEmpty) | ||||||
|  |         { | ||||||
|  |             static const std::string mbar("MBar"); | ||||||
|  |             mWatchedMagicka = stats.getMagicka(); | ||||||
|  |             setValue(mbar, stats.getMagicka()); | ||||||
|  |         } | ||||||
|  |         if (stats.getFatigue() != mWatchedFatigue || mWatchedStatsEmpty) | ||||||
|  |         { | ||||||
|  |             static const std::string fbar("FBar"); | ||||||
|  |             mWatchedFatigue = stats.getFatigue(); | ||||||
|  |             setValue(fbar, stats.getFatigue()); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         float timeToDrown = stats.getTimeToStartDrowning(); | ||||||
|  | 
 | ||||||
|  |         if (timeToDrown != mWatchedTimeToStartDrowning) | ||||||
|  |         { | ||||||
|  |             static const float fHoldBreathTime = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>() | ||||||
|  |                     .find("fHoldBreathTime")->mValue.getFloat(); | ||||||
|  | 
 | ||||||
|  |             mWatchedTimeToStartDrowning = timeToDrown; | ||||||
|  | 
 | ||||||
|  |             if(timeToDrown >= fHoldBreathTime || timeToDrown == -1.0) // -1.0 is a special value during initialization
 | ||||||
|  |                 winMgr->setDrowningBarVisibility(false); | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 winMgr->setDrowningBarVisibility(true); | ||||||
|  |                 winMgr->setDrowningTimeLeft(stats.getTimeToStartDrowning(), fHoldBreathTime); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         //Loop over ESM::Skill::SkillEnum
 | ||||||
|  |         for (int i = 0; i < ESM::Skill::Length; ++i) | ||||||
|  |         { | ||||||
|  |             if(stats.getSkill(i) != mWatchedSkills[i] || mWatchedStatsEmpty) | ||||||
|  |             { | ||||||
|  |                 mWatchedSkills[i] = stats.getSkill(i); | ||||||
|  |                 setValue((ESM::Skill::SkillEnum)i, stats.getSkill(i)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (stats.getLevel() != mWatchedLevel || mWatchedStatsEmpty) | ||||||
|  |         { | ||||||
|  |             mWatchedLevel = stats.getLevel(); | ||||||
|  |             setValue("level", mWatchedLevel); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (mWatched.getClass().isNpc()) | ||||||
|  |         { | ||||||
|  |             const ESM::NPC *watchedRecord = mWatched.get<ESM::NPC>()->mBase; | ||||||
|  | 
 | ||||||
|  |             if (watchedRecord->mName != mWatchedName || mWatchedStatsEmpty) | ||||||
|  |             { | ||||||
|  |                 mWatchedName = watchedRecord->mName; | ||||||
|  |                 setValue("name", watchedRecord->mName); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (watchedRecord->mRace != mWatchedRace || mWatchedStatsEmpty) | ||||||
|  |             { | ||||||
|  |                 mWatchedRace = watchedRecord->mRace; | ||||||
|  |                 const ESM::Race *race = MWBase::Environment::get().getWorld()->getStore() | ||||||
|  |                     .get<ESM::Race>().find(watchedRecord->mRace); | ||||||
|  |                 setValue("race", race->mName); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (watchedRecord->mClass != mWatchedClass || mWatchedStatsEmpty) | ||||||
|  |             { | ||||||
|  |                 mWatchedClass = watchedRecord->mClass; | ||||||
|  |                 const ESM::Class *cls = MWBase::Environment::get().getWorld()->getStore() | ||||||
|  |                     .get<ESM::Class>().find(watchedRecord->mClass); | ||||||
|  |                 setValue("class", cls->mName); | ||||||
|  | 
 | ||||||
|  |                 MWBase::WindowManager::SkillList majorSkills (5); | ||||||
|  |                 MWBase::WindowManager::SkillList minorSkills (5); | ||||||
|  | 
 | ||||||
|  |                 for (int i=0; i<5; ++i) | ||||||
|  |                 { | ||||||
|  |                     minorSkills[i] = cls->mData.mSkills[i][0]; | ||||||
|  |                     majorSkills[i] = cls->mData.mSkills[i][1]; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 configureSkills(majorSkills, minorSkills); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         mWatchedStatsEmpty = false; | ||||||
|  | 
 | ||||||
|  |         // Update the equipped weapon icon
 | ||||||
|  |         MWWorld::InventoryStore& inv = mWatched.getClass().getInventoryStore(mWatched); | ||||||
|  |         MWWorld::ContainerStoreIterator weapon = inv.getSlot(MWWorld::InventoryStore::Slot_CarriedRight); | ||||||
|  |         if (weapon == inv.end()) | ||||||
|  |             winMgr->unsetSelectedWeapon(); | ||||||
|  |         else | ||||||
|  |             winMgr->setSelectedWeapon(*weapon); | ||||||
|  | 
 | ||||||
|  |         // Update the selected spell icon
 | ||||||
|  |         MWWorld::ContainerStoreIterator enchantItem = inv.getSelectedEnchantItem(); | ||||||
|  |         if (enchantItem != inv.end()) | ||||||
|  |             winMgr->setSelectedEnchantItem(*enchantItem); | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             const std::string& spell = winMgr->getSelectedSpell(); | ||||||
|  |             if (!spell.empty()) | ||||||
|  |                 winMgr->setSelectedSpell(spell, int(MWMechanics::getSpellSuccessChance(spell, mWatched))); | ||||||
|  |             else | ||||||
|  |                 winMgr->unsetSelectedSpell(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::addListener(StatsListener* listener) | ||||||
|  |     { | ||||||
|  |         mListeners.insert(listener); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::removeListener(StatsListener* listener) | ||||||
|  |     { | ||||||
|  |         mListeners.erase(listener); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::setValue(const std::string& id, const MWMechanics::AttributeValue& value) | ||||||
|  |     { | ||||||
|  |         for (StatsListener* listener : mListeners) | ||||||
|  |             listener->setValue(id, value); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::setValue(ESM::Skill::SkillEnum parSkill, const MWMechanics::SkillValue& value) | ||||||
|  |     { | ||||||
|  |         /// \todo Don't use the skill enum as a parameter type (we will have to drop it anyway, once we
 | ||||||
|  |         /// allow custom skills.
 | ||||||
|  |         for (StatsListener* listener : mListeners) | ||||||
|  |             listener->setValue(parSkill, value); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::setValue(const std::string& id, const MWMechanics::DynamicStat<float>& value) | ||||||
|  |     { | ||||||
|  |         for (StatsListener* listener : mListeners) | ||||||
|  |             listener->setValue(id, value); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::setValue(const std::string& id, const std::string& value) | ||||||
|  |     { | ||||||
|  |         for (StatsListener* listener : mListeners) | ||||||
|  |             listener->setValue(id, value); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::setValue(const std::string& id, int value) | ||||||
|  |     { | ||||||
|  |         for (StatsListener* listener : mListeners) | ||||||
|  |             listener->setValue(id, value); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void StatsWatcher::configureSkills(const std::vector<int>& major, const std::vector<int>& minor) | ||||||
|  |     { | ||||||
|  |         for (StatsListener* listener : mListeners) | ||||||
|  |             listener->configureSkills(major, minor); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										69
									
								
								apps/openmw/mwgui/statswatcher.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								apps/openmw/mwgui/statswatcher.hpp
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,69 @@ | ||||||
|  | #ifndef MWGUI_STATSWATCHER_H | ||||||
|  | #define MWGUI_STATSWATCHER_H | ||||||
|  | 
 | ||||||
|  | #include <set> | ||||||
|  | 
 | ||||||
|  | #include <components/esm/attr.hpp> | ||||||
|  | #include <components/esm/loadskil.hpp> | ||||||
|  | 
 | ||||||
|  | #include "../mwmechanics/stat.hpp" | ||||||
|  | 
 | ||||||
|  | #include "../mwworld/ptr.hpp" | ||||||
|  | 
 | ||||||
|  | namespace MWGui | ||||||
|  | { | ||||||
|  |     class StatsListener | ||||||
|  |     { | ||||||
|  |     public: | ||||||
|  |         /// Set value for the given ID.
 | ||||||
|  |         virtual void setValue(const std::string& id, const MWMechanics::AttributeValue& value) {} | ||||||
|  |         virtual void setValue(const std::string& id, const MWMechanics::DynamicStat<float>& value) {} | ||||||
|  |         virtual void setValue(const std::string& id, const std::string& value) {} | ||||||
|  |         virtual void setValue(const std::string& id, int value) {} | ||||||
|  |         virtual void setValue(const ESM::Skill::SkillEnum parSkill, const MWMechanics::SkillValue& value) {} | ||||||
|  |         virtual void configureSkills(const std::vector<int>& major, const std::vector<int>& minor) {} | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     class StatsWatcher | ||||||
|  |     { | ||||||
|  |         MWWorld::Ptr mWatched; | ||||||
|  | 
 | ||||||
|  |         MWMechanics::AttributeValue mWatchedAttributes[ESM::Attribute::Length]; | ||||||
|  |         MWMechanics::SkillValue mWatchedSkills[ESM::Skill::Length]; | ||||||
|  | 
 | ||||||
|  |         MWMechanics::DynamicStat<float> mWatchedHealth; | ||||||
|  |         MWMechanics::DynamicStat<float> mWatchedMagicka; | ||||||
|  |         MWMechanics::DynamicStat<float> mWatchedFatigue; | ||||||
|  | 
 | ||||||
|  |         std::string mWatchedName; | ||||||
|  |         std::string mWatchedRace; | ||||||
|  |         std::string mWatchedClass; | ||||||
|  | 
 | ||||||
|  |         int mWatchedLevel; | ||||||
|  | 
 | ||||||
|  |         float mWatchedTimeToStartDrowning; | ||||||
|  | 
 | ||||||
|  |         bool mWatchedStatsEmpty; | ||||||
|  | 
 | ||||||
|  |         std::set<StatsListener*> mListeners; | ||||||
|  | 
 | ||||||
|  |         void setValue(const std::string& id, const MWMechanics::AttributeValue& value); | ||||||
|  |         void setValue(const std::string& id, const MWMechanics::DynamicStat<float>& value); | ||||||
|  |         void setValue(const std::string& id, const std::string& value); | ||||||
|  |         void setValue(const std::string& id, int value); | ||||||
|  |         void setValue(const ESM::Skill::SkillEnum parSkill, const MWMechanics::SkillValue& value); | ||||||
|  |         void configureSkills(const std::vector<int>& major, const std::vector<int>& minor); | ||||||
|  | 
 | ||||||
|  |     public: | ||||||
|  |         StatsWatcher(); | ||||||
|  | 
 | ||||||
|  |         void update(); | ||||||
|  |         void addListener(StatsListener* listener); | ||||||
|  |         void removeListener(StatsListener* listener); | ||||||
|  | 
 | ||||||
|  |         void watchActor(const MWWorld::Ptr& ptr); | ||||||
|  |         virtual MWWorld::Ptr getWatchedActor() const { return mWatched; } | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
|  | @ -1,16 +1,14 @@ | ||||||
| #ifndef MWGUI_STATS_WINDOW_H | #ifndef MWGUI_STATS_WINDOW_H | ||||||
| #define MWGUI_STATS_WINDOW_H | #define MWGUI_STATS_WINDOW_H | ||||||
| 
 | 
 | ||||||
| #include "../mwmechanics/stat.hpp" | #include "statswatcher.hpp" | ||||||
| #include "windowpinnablebase.hpp" | #include "windowpinnablebase.hpp" | ||||||
| 
 | 
 | ||||||
| #include <components/esm/loadskil.hpp> |  | ||||||
| 
 |  | ||||||
| namespace MWGui | namespace MWGui | ||||||
| { | { | ||||||
|     class WindowManager; |     class WindowManager; | ||||||
| 
 | 
 | ||||||
|     class StatsWindow : public WindowPinnableBase, public NoDrop |     class StatsWindow : public WindowPinnableBase, public NoDrop, public StatsListener | ||||||
|     { |     { | ||||||
|         public: |         public: | ||||||
|             typedef std::map<std::string, int> FactionList; |             typedef std::map<std::string, int> FactionList; | ||||||
|  | @ -26,13 +24,13 @@ namespace MWGui | ||||||
|             void setPlayerName(const std::string& playerName); |             void setPlayerName(const std::string& playerName); | ||||||
| 
 | 
 | ||||||
|             /// Set value for the given ID.
 |             /// Set value for the given ID.
 | ||||||
|             void setValue (const std::string& id, const MWMechanics::AttributeValue& value); |             void setValue (const std::string& id, const MWMechanics::AttributeValue& value) override; | ||||||
|             void setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value); |             void setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value) override; | ||||||
|             void setValue (const std::string& id, const std::string& value); |             void setValue (const std::string& id, const std::string& value) override; | ||||||
|             void setValue (const std::string& id, int value); |             void setValue (const std::string& id, int value) override; | ||||||
|             void setValue(const ESM::Skill::SkillEnum parSkill, const MWMechanics::SkillValue& value); |             void setValue(const ESM::Skill::SkillEnum parSkill, const MWMechanics::SkillValue& value) override; | ||||||
|  |             void configureSkills(const SkillList& major, const SkillList& minor) override; | ||||||
| 
 | 
 | ||||||
|             void configureSkills (const SkillList& major, const SkillList& minor); |  | ||||||
|             void setReputation (int reputation) { if (reputation != mReputation) mChanged = true; this->mReputation = reputation; } |             void setReputation (int reputation) { if (reputation != mReputation) mChanged = true; this->mReputation = reputation; } | ||||||
|             void setBounty (int bounty) { if (bounty != mBounty) mChanged = true; this->mBounty = bounty; } |             void setBounty (int bounty) { if (bounty != mBounty) mChanged = true; this->mBounty = bounty; } | ||||||
|             void updateSkillArea(); |             void updateSkillArea(); | ||||||
|  |  | ||||||
|  | @ -272,6 +272,8 @@ namespace MWGui | ||||||
|         mVideoWrapper = new SDLUtil::VideoWrapper(window, viewer); |         mVideoWrapper = new SDLUtil::VideoWrapper(window, viewer); | ||||||
|         mVideoWrapper->setGammaContrast(Settings::Manager::getFloat("gamma", "Video"), |         mVideoWrapper->setGammaContrast(Settings::Manager::getFloat("gamma", "Video"), | ||||||
|                                         Settings::Manager::getFloat("contrast", "Video")); |                                         Settings::Manager::getFloat("contrast", "Video")); | ||||||
|  | 
 | ||||||
|  |         mStatsWatcher.reset(new StatsWatcher()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void WindowManager::loadUserFonts() |     void WindowManager::loadUserFonts() | ||||||
|  | @ -466,6 +468,10 @@ namespace MWGui | ||||||
| 
 | 
 | ||||||
|         // Set up visibility
 |         // Set up visibility
 | ||||||
|         updateVisible(); |         updateVisible(); | ||||||
|  | 
 | ||||||
|  |         mStatsWatcher->addListener(mHud); | ||||||
|  |         mStatsWatcher->addListener(mStatsWindow); | ||||||
|  |         mStatsWatcher->addListener(mCharGen); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     int WindowManager::getFontHeight() const |     int WindowManager::getFontHeight() const | ||||||
|  | @ -478,8 +484,11 @@ namespace MWGui | ||||||
|         if (newgame) |         if (newgame) | ||||||
|         { |         { | ||||||
|             disallowAll(); |             disallowAll(); | ||||||
|  | 
 | ||||||
|  |             mStatsWatcher->removeListener(mCharGen); | ||||||
|             delete mCharGen; |             delete mCharGen; | ||||||
|             mCharGen = new CharacterCreation(mViewer->getSceneData()->asGroup(), mResourceSystem); |             mCharGen = new CharacterCreation(mViewer->getSceneData()->asGroup(), mResourceSystem); | ||||||
|  |             mStatsWatcher->addListener(mCharGen); | ||||||
|         } |         } | ||||||
|         else |         else | ||||||
|             allow(GW_ALL); |             allow(GW_ALL); | ||||||
|  | @ -489,6 +498,8 @@ namespace MWGui | ||||||
|     { |     { | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|  |             mStatsWatcher.reset(); | ||||||
|  | 
 | ||||||
|             mKeyboardNavigation.reset(); |             mKeyboardNavigation.reset(); | ||||||
| 
 | 
 | ||||||
|             MyGUI::LanguageManager::getInstance().eventRequestTag.clear(); |             MyGUI::LanguageManager::getInstance().eventRequestTag.clear(); | ||||||
|  | @ -649,58 +660,11 @@ namespace MWGui | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void WindowManager::setValue (const std::string& id, const MWMechanics::AttributeValue& value) |  | ||||||
|     { |  | ||||||
|         mStatsWindow->setValue (id, value); |  | ||||||
|         mCharGen->setValue(id, value); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void WindowManager::setValue (int parSkill, const MWMechanics::SkillValue& value) |  | ||||||
|     { |  | ||||||
|         /// \todo Don't use the skill enum as a parameter type (we will have to drop it anyway, once we
 |  | ||||||
|         /// allow custom skills.
 |  | ||||||
|         mStatsWindow->setValue(static_cast<ESM::Skill::SkillEnum> (parSkill), value); |  | ||||||
|         mCharGen->setValue(static_cast<ESM::Skill::SkillEnum> (parSkill), value); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void WindowManager::setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value) |  | ||||||
|     { |  | ||||||
|         mStatsWindow->setValue (id, value); |  | ||||||
|         mHud->setValue (id, value); |  | ||||||
|         mCharGen->setValue(id, value); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void WindowManager::setValue (const std::string& id, const std::string& value) |  | ||||||
|     { |  | ||||||
|         mStatsWindow->setValue (id, value); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void WindowManager::setValue (const std::string& id, int value) |  | ||||||
|     { |  | ||||||
|         mStatsWindow->setValue (id, value); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void WindowManager::setDrowningTimeLeft (float time, float maxTime) |     void WindowManager::setDrowningTimeLeft (float time, float maxTime) | ||||||
|     { |     { | ||||||
|         mHud->setDrowningTimeLeft(time, maxTime); |         mHud->setDrowningTimeLeft(time, maxTime); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void WindowManager::setPlayerClass (const ESM::Class &class_) |  | ||||||
|     { |  | ||||||
|         mStatsWindow->setValue("class", class_.mName); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void WindowManager::configureSkills (const SkillList& major, const SkillList& minor) |  | ||||||
|     { |  | ||||||
|         mStatsWindow->configureSkills (major, minor); |  | ||||||
|         mCharGen->configureSkills(major, minor); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void WindowManager::updateSkillArea() |  | ||||||
|     { |  | ||||||
|         mStatsWindow->updateSkillArea(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void WindowManager::removeDialog(Layout*dialog) |     void WindowManager::removeDialog(Layout*dialog) | ||||||
|     { |     { | ||||||
|         if (!dialog) |         if (!dialog) | ||||||
|  | @ -906,6 +870,8 @@ namespace MWGui | ||||||
| 
 | 
 | ||||||
|         updateActivatedQuickKey(); |         updateActivatedQuickKey(); | ||||||
| 
 | 
 | ||||||
|  |         mStatsWatcher->update(); | ||||||
|  | 
 | ||||||
|         cleanupGarbage(); |         cleanupGarbage(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -2210,4 +2176,14 @@ namespace MWGui | ||||||
|         for (unsigned int i=0; i<mWindows.size(); ++i) |         for (unsigned int i=0; i<mWindows.size(); ++i) | ||||||
|             mWindows[i]->setVisible(visible); |             mWindows[i]->setVisible(visible); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     void WindowManager::watchActor(const MWWorld::Ptr& ptr) | ||||||
|  |     { | ||||||
|  |         mStatsWatcher->watchActor(ptr); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     MWWorld::Ptr WindowManager::getWatchedActor() const | ||||||
|  |     { | ||||||
|  |         return mStatsWatcher->getWatchedActor(); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -13,13 +13,12 @@ | ||||||
| 
 | 
 | ||||||
| #include "../mwbase/windowmanager.hpp" | #include "../mwbase/windowmanager.hpp" | ||||||
| 
 | 
 | ||||||
| #include "../mwworld/ptr.hpp" |  | ||||||
| 
 |  | ||||||
| #include <components/sdlutil/events.hpp> | #include <components/sdlutil/events.hpp> | ||||||
| #include <components/settings/settings.hpp> | #include <components/settings/settings.hpp> | ||||||
| #include <components/to_utf8/to_utf8.hpp> | #include <components/to_utf8/to_utf8.hpp> | ||||||
| 
 | 
 | ||||||
| #include "mapwindow.hpp" | #include "mapwindow.hpp" | ||||||
|  | #include "statswatcher.hpp" | ||||||
| #include "textcolours.hpp" | #include "textcolours.hpp" | ||||||
| 
 | 
 | ||||||
| #include <MyGUI_KeyCode.h> | #include <MyGUI_KeyCode.h> | ||||||
|  | @ -195,22 +194,11 @@ namespace MWGui | ||||||
| 
 | 
 | ||||||
|     virtual void setConsoleSelectedObject(const MWWorld::Ptr& object); |     virtual void setConsoleSelectedObject(const MWWorld::Ptr& object); | ||||||
| 
 | 
 | ||||||
|     ///< Set value for the given ID.
 |  | ||||||
|     virtual void setValue (const std::string& id, const MWMechanics::AttributeValue& value); |  | ||||||
|     virtual void setValue (int parSkill, const MWMechanics::SkillValue& value); |  | ||||||
|     virtual void setValue (const std::string& id, const MWMechanics::DynamicStat<float>& value); |  | ||||||
|     virtual void setValue (const std::string& id, const std::string& value); |  | ||||||
|     virtual void setValue (const std::string& id, int value); |  | ||||||
| 
 |  | ||||||
|     /// Set time left for the player to start drowning (update the drowning bar)
 |     /// Set time left for the player to start drowning (update the drowning bar)
 | ||||||
|     /// @param time time left to start drowning
 |     /// @param time time left to start drowning
 | ||||||
|     /// @param maxTime how long we can be underwater (in total) until drowning starts
 |     /// @param maxTime how long we can be underwater (in total) until drowning starts
 | ||||||
|     virtual void setDrowningTimeLeft (float time, float maxTime); |     virtual void setDrowningTimeLeft (float time, float maxTime); | ||||||
| 
 | 
 | ||||||
|     virtual void setPlayerClass (const ESM::Class &class_);                        ///< set current class of player
 |  | ||||||
|     virtual void configureSkills (const SkillList& major, const SkillList& minor); ///< configure skill groups, each set contains the skill ID for that group.
 |  | ||||||
|     virtual void updateSkillArea();                                                ///< update display of skills, factions, birth sign, reputation and bounty
 |  | ||||||
| 
 |  | ||||||
|     virtual void changeCell(const MWWorld::CellStore* cell); ///< change the active cell
 |     virtual void changeCell(const MWWorld::CellStore* cell); ///< change the active cell
 | ||||||
| 
 | 
 | ||||||
|     virtual void setFocusObject(const MWWorld::Ptr& focus); |     virtual void setFocusObject(const MWWorld::Ptr& focus); | ||||||
|  | @ -298,6 +286,9 @@ namespace MWGui | ||||||
|     virtual void windowClosed(); |     virtual void windowClosed(); | ||||||
|     virtual bool isWindowVisible(); |     virtual bool isWindowVisible(); | ||||||
| 
 | 
 | ||||||
|  |     virtual void watchActor(const MWWorld::Ptr& ptr); | ||||||
|  |     virtual MWWorld::Ptr getWatchedActor() const; | ||||||
|  | 
 | ||||||
|     virtual void executeInConsole (const std::string& path); |     virtual void executeInConsole (const std::string& path); | ||||||
| 
 | 
 | ||||||
|     virtual void enableRest() { mRestAllowed = true; } |     virtual void enableRest() { mRestAllowed = true; } | ||||||
|  | @ -400,6 +391,7 @@ namespace MWGui | ||||||
|     osgViewer::Viewer* mViewer; |     osgViewer::Viewer* mViewer; | ||||||
| 
 | 
 | ||||||
|     std::unique_ptr<Gui::FontLoader> mFontLoader; |     std::unique_ptr<Gui::FontLoader> mFontLoader; | ||||||
|  |     std::unique_ptr<StatsWatcher> mStatsWatcher; | ||||||
| 
 | 
 | ||||||
|     bool mConsoleOnlyScripts; |     bool mConsoleOnlyScripts; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -236,10 +236,8 @@ namespace MWMechanics | ||||||
|         invStore.autoEquip(ptr); |         invStore.autoEquip(ptr); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // mWatchedTimeToStartDrowning = -1 for correct drowning state check,
 |  | ||||||
|     // if stats.getTimeToStartDrowning() == 0 already on game start
 |  | ||||||
|     MechanicsManager::MechanicsManager() |     MechanicsManager::MechanicsManager() | ||||||
|     : mWatchedLevel(-1), mWatchedTimeToStartDrowning(-1), mWatchedStatsEmpty (true), mUpdatePlayer (true), mClassSelected (false), |     : mUpdatePlayer (true), mClassSelected (false), | ||||||
|       mRaceSelected (false), mAI(true) |       mRaceSelected (false), mAI(true) | ||||||
|     { |     { | ||||||
|         //buildPlayer no longer here, needs to be done explicitly after all subsystems are up and running
 |         //buildPlayer no longer here, needs to be done explicitly after all subsystems are up and running
 | ||||||
|  | @ -261,16 +259,16 @@ namespace MWMechanics | ||||||
| 
 | 
 | ||||||
|     void MechanicsManager::remove(const MWWorld::Ptr& ptr) |     void MechanicsManager::remove(const MWWorld::Ptr& ptr) | ||||||
|     { |     { | ||||||
|         if(ptr == mWatched) |         if(ptr == MWBase::Environment::get().getWindowManager()->getWatchedActor()) | ||||||
|             mWatched = MWWorld::Ptr(); |             MWBase::Environment::get().getWindowManager()->watchActor(MWWorld::Ptr()); | ||||||
|         mActors.removeActor(ptr); |         mActors.removeActor(ptr); | ||||||
|         mObjects.removeObject(ptr); |         mObjects.removeObject(ptr); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void MechanicsManager::updateCell(const MWWorld::Ptr &old, const MWWorld::Ptr &ptr) |     void MechanicsManager::updateCell(const MWWorld::Ptr &old, const MWWorld::Ptr &ptr) | ||||||
|     { |     { | ||||||
|         if(old == mWatched) |         if(old == MWBase::Environment::get().getWindowManager()->getWatchedActor()) | ||||||
|             mWatched = ptr; |             MWBase::Environment::get().getWindowManager()->watchActor(ptr); | ||||||
| 
 | 
 | ||||||
|         if(ptr.getClass().isActor()) |         if(ptr.getClass().isActor()) | ||||||
|             mActors.updateActor(old, ptr); |             mActors.updateActor(old, ptr); | ||||||
|  | @ -278,19 +276,12 @@ namespace MWMechanics | ||||||
|             mObjects.updateObject(old, ptr); |             mObjects.updateObject(old, ptr); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     void MechanicsManager::drop(const MWWorld::CellStore *cellStore) |     void MechanicsManager::drop(const MWWorld::CellStore *cellStore) | ||||||
|     { |     { | ||||||
|         mActors.dropActors(cellStore, mWatched); |         mActors.dropActors(cellStore, getPlayer()); | ||||||
|         mObjects.dropObjects(cellStore); |         mObjects.dropObjects(cellStore); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     void MechanicsManager::watchActor(const MWWorld::Ptr& ptr) |  | ||||||
|     { |  | ||||||
|         mWatched = ptr; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void MechanicsManager::restoreStatsAfterCorprus(const MWWorld::Ptr& actor, const std::string& sourceId) |     void MechanicsManager::restoreStatsAfterCorprus(const MWWorld::Ptr& actor, const std::string& sourceId) | ||||||
|     { |     { | ||||||
|         auto& stats = actor.getClass().getCreatureStats (actor); |         auto& stats = actor.getClass().getCreatureStats (actor); | ||||||
|  | @ -311,133 +302,10 @@ namespace MWMechanics | ||||||
| 
 | 
 | ||||||
|     void MechanicsManager::update(float duration, bool paused) |     void MechanicsManager::update(float duration, bool paused) | ||||||
|     { |     { | ||||||
|         if(!mWatched.isEmpty()) |  | ||||||
|         { |  | ||||||
|             MWBase::WindowManager *winMgr = MWBase::Environment::get().getWindowManager(); |  | ||||||
|             const MWMechanics::NpcStats &stats = mWatched.getClass().getNpcStats(mWatched); |  | ||||||
|             for(int i = 0;i < ESM::Attribute::Length;++i) |  | ||||||
|             { |  | ||||||
|                 if(stats.getAttribute(i) != mWatchedAttributes[i] || mWatchedStatsEmpty) |  | ||||||
|                 { |  | ||||||
|                     std::stringstream attrname; |  | ||||||
|                     attrname << "AttribVal"<<(i+1); |  | ||||||
| 
 |  | ||||||
|                     mWatchedAttributes[i] = stats.getAttribute(i); |  | ||||||
|                     winMgr->setValue(attrname.str(), stats.getAttribute(i)); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if(stats.getHealth() != mWatchedHealth || mWatchedStatsEmpty) |  | ||||||
|             { |  | ||||||
|                 static const std::string hbar("HBar"); |  | ||||||
|                 mWatchedHealth = stats.getHealth(); |  | ||||||
|                 winMgr->setValue(hbar, stats.getHealth()); |  | ||||||
|             } |  | ||||||
|             if(stats.getMagicka() != mWatchedMagicka || mWatchedStatsEmpty) |  | ||||||
|             { |  | ||||||
|                 static const std::string mbar("MBar"); |  | ||||||
|                 mWatchedMagicka = stats.getMagicka(); |  | ||||||
|                 winMgr->setValue(mbar, stats.getMagicka()); |  | ||||||
|             } |  | ||||||
|             if(stats.getFatigue() != mWatchedFatigue || mWatchedStatsEmpty) |  | ||||||
|             { |  | ||||||
|                 static const std::string fbar("FBar"); |  | ||||||
|                 mWatchedFatigue = stats.getFatigue(); |  | ||||||
|                 winMgr->setValue(fbar, stats.getFatigue()); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             float timeToDrown = stats.getTimeToStartDrowning(); |  | ||||||
| 
 |  | ||||||
|             if(timeToDrown != mWatchedTimeToStartDrowning) |  | ||||||
|             { |  | ||||||
|                 static const float fHoldBreathTime = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>() |  | ||||||
|                         .find("fHoldBreathTime")->mValue.getFloat(); |  | ||||||
| 
 |  | ||||||
|                 mWatchedTimeToStartDrowning = timeToDrown; |  | ||||||
| 
 |  | ||||||
|                 if(timeToDrown >= fHoldBreathTime || timeToDrown == -1.0) // -1.0 is a special value during initialization
 |  | ||||||
|                     winMgr->setDrowningBarVisibility(false); |  | ||||||
|                 else |  | ||||||
|                 { |  | ||||||
|                     winMgr->setDrowningBarVisibility(true); |  | ||||||
|                     winMgr->setDrowningTimeLeft(stats.getTimeToStartDrowning(), fHoldBreathTime); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             //Loop over ESM::Skill::SkillEnum
 |  | ||||||
|             for(int i = 0; i < ESM::Skill::Length; ++i) |  | ||||||
|             { |  | ||||||
|                 if(stats.getSkill(i) != mWatchedSkills[i] || mWatchedStatsEmpty) |  | ||||||
|                 { |  | ||||||
|                     mWatchedSkills[i] = stats.getSkill(i); |  | ||||||
|                     winMgr->setValue((ESM::Skill::SkillEnum)i, stats.getSkill(i)); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if(stats.getLevel() != mWatchedLevel) |  | ||||||
|             { |  | ||||||
|                 mWatchedLevel = stats.getLevel(); |  | ||||||
|                 winMgr->setValue("level", mWatchedLevel); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             mWatchedStatsEmpty = false; |  | ||||||
| 
 |  | ||||||
|             // Update the equipped weapon icon
 |  | ||||||
|             MWWorld::InventoryStore& inv = mWatched.getClass().getInventoryStore(mWatched); |  | ||||||
|             MWWorld::ContainerStoreIterator weapon = inv.getSlot(MWWorld::InventoryStore::Slot_CarriedRight); |  | ||||||
|             if (weapon == inv.end()) |  | ||||||
|                 winMgr->unsetSelectedWeapon(); |  | ||||||
|             else |  | ||||||
|                 winMgr->setSelectedWeapon(*weapon); |  | ||||||
| 
 |  | ||||||
|             // Update the selected spell icon
 |  | ||||||
|             MWWorld::ContainerStoreIterator enchantItem = inv.getSelectedEnchantItem(); |  | ||||||
|             if (enchantItem != inv.end()) |  | ||||||
|                 winMgr->setSelectedEnchantItem(*enchantItem); |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 const std::string& spell = winMgr->getSelectedSpell(); |  | ||||||
|                 if (!spell.empty()) |  | ||||||
|                     winMgr->setSelectedSpell(spell, int(getSpellSuccessChance(spell, mWatched))); |  | ||||||
|                 else |  | ||||||
|                     winMgr->unsetSelectedSpell(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (mUpdatePlayer) |         if (mUpdatePlayer) | ||||||
|         { |         { | ||||||
|             MWBase::World *world = MWBase::Environment::get().getWorld(); |  | ||||||
| 
 |  | ||||||
|             // basic player profile; should not change anymore after the creation phase is finished.
 |  | ||||||
|             MWBase::WindowManager *winMgr = |  | ||||||
|                 MWBase::Environment::get().getWindowManager(); |  | ||||||
| 
 |  | ||||||
|             const ESM::NPC *player = |  | ||||||
|                 world->getPlayerPtr().get<ESM::NPC>()->mBase; |  | ||||||
| 
 |  | ||||||
|             const ESM::Race *race = |  | ||||||
|                 world->getStore().get<ESM::Race>().find(player->mRace); |  | ||||||
|             const ESM::Class *cls = |  | ||||||
|                 world->getStore().get<ESM::Class>().find(player->mClass); |  | ||||||
| 
 |  | ||||||
|             winMgr->setValue ("name", player->mName); |  | ||||||
|             winMgr->setValue ("race", race->mName); |  | ||||||
|             winMgr->setValue ("class", cls->mName); |  | ||||||
| 
 |  | ||||||
|             mUpdatePlayer = false; |             mUpdatePlayer = false; | ||||||
| 
 | 
 | ||||||
|             MWBase::WindowManager::SkillList majorSkills (5); |  | ||||||
|             MWBase::WindowManager::SkillList minorSkills (5); |  | ||||||
| 
 |  | ||||||
|             for (int i=0; i<5; ++i) |  | ||||||
|             { |  | ||||||
|                 minorSkills[i] = cls->mData.mSkills[i][0]; |  | ||||||
|                 majorSkills[i] = cls->mData.mSkills[i][1]; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             winMgr->configureSkills (majorSkills, minorSkills); |  | ||||||
| 
 |  | ||||||
|             // HACK? The player has been changed, so a new Animation object may
 |             // HACK? The player has been changed, so a new Animation object may
 | ||||||
|             // have been made for them. Make sure they're properly updated.
 |             // have been made for them. Make sure they're properly updated.
 | ||||||
|             MWWorld::Ptr ptr = getPlayer(); |             MWWorld::Ptr ptr = getPlayer(); | ||||||
|  | @ -517,7 +385,7 @@ namespace MWMechanics | ||||||
| 
 | 
 | ||||||
|     int MechanicsManager::getHoursToRest() const |     int MechanicsManager::getHoursToRest() const | ||||||
|     { |     { | ||||||
|         return mActors.getHoursToRest(mWatched); |         return mActors.getHoursToRest(getPlayer()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void MechanicsManager::setPlayerName (const std::string& name) |     void MechanicsManager::setPlayerName (const std::string& name) | ||||||
|  |  | ||||||
|  | @ -21,20 +21,6 @@ namespace MWMechanics | ||||||
| { | { | ||||||
|     class MechanicsManager : public MWBase::MechanicsManager |     class MechanicsManager : public MWBase::MechanicsManager | ||||||
|     { |     { | ||||||
|             MWWorld::Ptr mWatched; |  | ||||||
| 
 |  | ||||||
|             AttributeValue mWatchedAttributes[8]; |  | ||||||
|             SkillValue mWatchedSkills[27]; |  | ||||||
| 
 |  | ||||||
|             DynamicStat<float> mWatchedHealth; |  | ||||||
|             DynamicStat<float> mWatchedMagicka; |  | ||||||
|             DynamicStat<float> mWatchedFatigue; |  | ||||||
| 
 |  | ||||||
|             int mWatchedLevel; |  | ||||||
| 
 |  | ||||||
|             float mWatchedTimeToStartDrowning; |  | ||||||
| 
 |  | ||||||
|             bool mWatchedStatsEmpty; |  | ||||||
|             bool mUpdatePlayer; |             bool mUpdatePlayer; | ||||||
|             bool mClassSelected; |             bool mClassSelected; | ||||||
|             bool mRaceSelected; |             bool mRaceSelected; | ||||||
|  | @ -68,10 +54,6 @@ namespace MWMechanics | ||||||
|             virtual void drop(const MWWorld::CellStore *cellStore) override; |             virtual void drop(const MWWorld::CellStore *cellStore) override; | ||||||
|             ///< Deregister all objects in the given cell.
 |             ///< Deregister all objects in the given cell.
 | ||||||
| 
 | 
 | ||||||
|             virtual void watchActor(const MWWorld::Ptr& ptr) override; |  | ||||||
|             ///< On each update look for changes in a previously registered actor and update the
 |  | ||||||
|             /// GUI accordingly.
 |  | ||||||
| 
 |  | ||||||
|             virtual void update (float duration, bool paused) override; |             virtual void update (float duration, bool paused) override; | ||||||
|             ///< Update objects
 |             ///< Update objects
 | ||||||
|             ///
 |             ///
 | ||||||
|  |  | ||||||
|  | @ -741,15 +741,12 @@ namespace MWWorld | ||||||
|             player.getClass().adjustPosition(player, true); |             player.getClass().adjustPosition(player, true); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         MWBase::MechanicsManager *mechMgr = |         MWBase::Environment::get().getMechanicsManager()->updateCell(old, player); | ||||||
|             MWBase::Environment::get().getMechanicsManager(); |         MWBase::Environment::get().getWindowManager()->watchActor(player); | ||||||
| 
 |  | ||||||
|         mechMgr->updateCell(old, player); |  | ||||||
|         mechMgr->watchActor(player); |  | ||||||
| 
 | 
 | ||||||
|         mPhysics->updatePtr(old, player); |         mPhysics->updatePtr(old, player); | ||||||
| 
 | 
 | ||||||
|         MWBase::Environment::get().getWorld()->adjustSky(); |         world->adjustSky(); | ||||||
| 
 | 
 | ||||||
|         mLastPlayerPos = player.getRefData().getPosition().asVec3(); |         mLastPlayerPos = player.getRefData().getPosition().asVec3(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -2453,7 +2453,7 @@ namespace MWWorld | ||||||
|         rotateObject(player, 0.f, 0.f, 0.f, MWBase::RotationFlag_inverseOrder | MWBase::RotationFlag_adjust); |         rotateObject(player, 0.f, 0.f, 0.f, MWBase::RotationFlag_inverseOrder | MWBase::RotationFlag_adjust); | ||||||
| 
 | 
 | ||||||
|         MWBase::Environment::get().getMechanicsManager()->add(getPlayerPtr()); |         MWBase::Environment::get().getMechanicsManager()->add(getPlayerPtr()); | ||||||
|         MWBase::Environment::get().getMechanicsManager()->watchActor(getPlayerPtr()); |         MWBase::Environment::get().getWindowManager()->watchActor(getPlayerPtr()); | ||||||
| 
 | 
 | ||||||
|         std::string model = getPlayerPtr().getClass().getModel(getPlayerPtr()); |         std::string model = getPlayerPtr().getClass().getModel(getPlayerPtr()); | ||||||
|         model = Misc::ResourceHelpers::correctActorModelPath(model, mResourceSystem->getVFS()); |         model = Misc::ResourceHelpers::correctActorModelPath(model, mResourceSystem->getVFS()); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue