2012-05-13 16:14:03 +00:00
|
|
|
#include "hud.hpp"
|
2010-09-17 23:30:23 +00:00
|
|
|
|
2015-01-10 01:50:43 +00:00
|
|
|
#include <OgreMath.h>
|
|
|
|
|
|
|
|
#include <MyGUI_RenderManager.h>
|
|
|
|
#include <MyGUI_ProgressBar.h>
|
|
|
|
#include <MyGUI_Button.h>
|
|
|
|
#include <MyGUI_InputManager.h>
|
|
|
|
#include <MyGUI_ImageBox.h>
|
|
|
|
#include <MyGUI_ScrollView.h>
|
|
|
|
|
2014-08-16 16:12:18 +00:00
|
|
|
#include <components/misc/resourcehelpers.hpp>
|
2015-01-31 22:27:34 +00:00
|
|
|
#include <components/settings/settings.hpp>
|
2014-08-16 16:12:18 +00:00
|
|
|
|
2012-05-13 16:35:56 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
2012-08-09 12:33:21 +00:00
|
|
|
#include "../mwbase/soundmanager.hpp"
|
2012-08-12 16:11:09 +00:00
|
|
|
#include "../mwbase/windowmanager.hpp"
|
2014-12-19 10:26:54 +00:00
|
|
|
#include "../mwbase/world.hpp"
|
2012-07-03 10:30:50 +00:00
|
|
|
|
2013-05-11 16:38:27 +00:00
|
|
|
#include "../mwworld/class.hpp"
|
2014-12-19 10:26:54 +00:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2012-05-13 16:35:56 +00:00
|
|
|
|
2013-07-30 04:00:20 +00:00
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
2013-08-05 22:07:24 +00:00
|
|
|
#include "../mwmechanics/npcstats.hpp"
|
2013-07-30 04:00:20 +00:00
|
|
|
|
2012-06-02 12:19:02 +00:00
|
|
|
#include "inventorywindow.hpp"
|
2012-06-02 10:25:24 +00:00
|
|
|
#include "console.hpp"
|
2013-03-03 11:01:19 +00:00
|
|
|
#include "spellicons.hpp"
|
2013-05-11 16:38:27 +00:00
|
|
|
#include "itemmodel.hpp"
|
2015-01-10 00:21:17 +00:00
|
|
|
#include "draganddrop.hpp"
|
2010-09-17 23:44:40 +00:00
|
|
|
|
2014-05-16 01:19:38 +00:00
|
|
|
#include "itemmodel.hpp"
|
2014-06-05 20:13:18 +00:00
|
|
|
#include "itemwidget.hpp"
|
2014-05-16 01:19:38 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
namespace MWGui
|
2010-09-18 19:24:05 +00:00
|
|
|
{
|
|
|
|
|
2014-05-16 01:19:38 +00:00
|
|
|
/**
|
|
|
|
* Makes it possible to use ItemModel::moveItem to move an item from an inventory to the world.
|
|
|
|
*/
|
|
|
|
class WorldItemModel : public ItemModel
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
WorldItemModel(float left, float top) : mLeft(left), mTop(top) {}
|
|
|
|
virtual ~WorldItemModel() {}
|
|
|
|
virtual MWWorld::Ptr copyItem (const ItemStack& item, size_t count, bool setNewOwner=false)
|
|
|
|
{
|
|
|
|
MWBase::World* world = MWBase::Environment::get().getWorld();
|
|
|
|
|
|
|
|
MWWorld::Ptr dropped;
|
|
|
|
if (world->canPlaceObject(mLeft, mTop))
|
|
|
|
dropped = world->placeObject(item.mBase, mLeft, mTop, count);
|
|
|
|
else
|
|
|
|
dropped = world->dropObjectOnGround(world->getPlayerPtr(), item.mBase, count);
|
|
|
|
if (setNewOwner)
|
2014-05-25 12:13:07 +00:00
|
|
|
dropped.getCellRef().setOwner("");
|
2014-05-16 01:19:38 +00:00
|
|
|
|
|
|
|
return dropped;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void removeItem (const ItemStack& item, size_t count) { throw std::runtime_error("removeItem not implemented"); }
|
|
|
|
virtual ModelIndex getIndex (ItemStack item) { throw std::runtime_error("getIndex not implemented"); }
|
|
|
|
virtual void update() {}
|
|
|
|
virtual size_t getItemCount() { return 0; }
|
|
|
|
virtual ItemStack getItem (ModelIndex index) { throw std::runtime_error("getItem not implemented"); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Where to drop the item
|
|
|
|
float mLeft;
|
|
|
|
float mTop;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-08-17 03:45:35 +00:00
|
|
|
HUD::HUD(CustomMarkerCollection &customMarkers, int fpsLevel, DragAndDrop* dragAndDrop)
|
2013-04-17 22:56:48 +00:00
|
|
|
: Layout("openmw_hud.layout")
|
2014-08-17 03:45:35 +00:00
|
|
|
, LocalMapBase(customMarkers)
|
2013-04-17 22:56:48 +00:00
|
|
|
, mHealth(NULL)
|
|
|
|
, mMagicka(NULL)
|
|
|
|
, mStamina(NULL)
|
2013-08-07 13:34:11 +00:00
|
|
|
, mDrowning(NULL)
|
|
|
|
, mDrowningFrame(NULL)
|
2013-10-27 08:05:01 +00:00
|
|
|
, mDrowningFlash(NULL)
|
2013-04-17 22:56:48 +00:00
|
|
|
, mWeapImage(NULL)
|
|
|
|
, mSpellImage(NULL)
|
|
|
|
, mWeapStatus(NULL)
|
|
|
|
, mSpellStatus(NULL)
|
|
|
|
, mEffectBox(NULL)
|
|
|
|
, mMinimap(NULL)
|
|
|
|
, mCompass(NULL)
|
|
|
|
, mCrosshair(NULL)
|
|
|
|
, mFpsBox(NULL)
|
|
|
|
, mFpsCounter(NULL)
|
|
|
|
, mTriangleCounter(NULL)
|
|
|
|
, mBatchCounter(NULL)
|
|
|
|
, mHealthManaStaminaBaseLeft(0)
|
|
|
|
, mWeapBoxBaseLeft(0)
|
|
|
|
, mSpellBoxBaseLeft(0)
|
|
|
|
, mEffectBoxBaseRight(0)
|
|
|
|
, mMinimapBoxBaseRight(0)
|
|
|
|
, mDragAndDrop(dragAndDrop)
|
|
|
|
, mCellNameTimer(0.0f)
|
|
|
|
, mCellNameBox(NULL)
|
|
|
|
, mMapVisible(true)
|
|
|
|
, mWeaponVisible(true)
|
|
|
|
, mSpellVisible(true)
|
|
|
|
, mWorldMouseOver(false)
|
2014-04-23 17:49:09 +00:00
|
|
|
, mEnemyHealthTimer(-1)
|
2014-07-23 20:04:18 +00:00
|
|
|
, mEnemyActorId(-1)
|
2013-10-27 08:05:01 +00:00
|
|
|
, mIsDrowning(false)
|
2013-12-30 22:08:53 +00:00
|
|
|
, mWeaponSpellTimer(0.f)
|
|
|
|
, mDrowningFlashTheta(0.f)
|
2013-04-17 22:56:48 +00:00
|
|
|
{
|
2014-08-01 15:15:28 +00:00
|
|
|
mMainWidget->setSize(MyGUI::RenderManager::getInstance().getViewSize());
|
2012-05-26 19:09:21 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
// Energy bars
|
|
|
|
getWidget(mHealthFrame, "HealthFrame");
|
|
|
|
getWidget(mHealth, "Health");
|
|
|
|
getWidget(mMagicka, "Magicka");
|
|
|
|
getWidget(mStamina, "Stamina");
|
2013-07-30 04:00:20 +00:00
|
|
|
getWidget(mEnemyHealth, "EnemyHealth");
|
2013-04-17 22:56:48 +00:00
|
|
|
mHealthManaStaminaBaseLeft = mHealthFrame->getLeft();
|
2012-05-29 13:36:48 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
MyGUI::Widget *healthFrame, *magickaFrame, *fatigueFrame;
|
|
|
|
getWidget(healthFrame, "HealthFrame");
|
|
|
|
getWidget(magickaFrame, "MagickaFrame");
|
|
|
|
getWidget(fatigueFrame, "FatigueFrame");
|
|
|
|
healthFrame->eventMouseButtonClick += MyGUI::newDelegate(this, &HUD::onHMSClicked);
|
|
|
|
magickaFrame->eventMouseButtonClick += MyGUI::newDelegate(this, &HUD::onHMSClicked);
|
|
|
|
fatigueFrame->eventMouseButtonClick += MyGUI::newDelegate(this, &HUD::onHMSClicked);
|
2012-07-13 10:51:58 +00:00
|
|
|
|
2013-08-07 13:34:11 +00:00
|
|
|
//Drowning bar
|
|
|
|
getWidget(mDrowningFrame, "DrowningFrame");
|
|
|
|
getWidget(mDrowning, "Drowning");
|
2013-10-27 08:05:01 +00:00
|
|
|
getWidget(mDrowningFlash, "Flash");
|
2013-08-07 13:34:11 +00:00
|
|
|
mDrowning->setProgressRange(200);
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
const MyGUI::IntSize& viewSize = MyGUI::RenderManager::getInstance().getViewSize();
|
2012-07-13 10:51:58 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
// Item and spell images and status bars
|
|
|
|
getWidget(mWeapBox, "WeapBox");
|
|
|
|
getWidget(mWeapImage, "WeapImage");
|
|
|
|
getWidget(mWeapStatus, "WeapStatus");
|
|
|
|
mWeapBoxBaseLeft = mWeapBox->getLeft();
|
|
|
|
mWeapBox->eventMouseButtonClick += MyGUI::newDelegate(this, &HUD::onWeaponClicked);
|
2012-07-13 10:51:58 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
getWidget(mSpellBox, "SpellBox");
|
|
|
|
getWidget(mSpellImage, "SpellImage");
|
|
|
|
getWidget(mSpellStatus, "SpellStatus");
|
|
|
|
mSpellBoxBaseLeft = mSpellBox->getLeft();
|
|
|
|
mSpellBox->eventMouseButtonClick += MyGUI::newDelegate(this, &HUD::onMagicClicked);
|
2010-09-18 19:24:05 +00:00
|
|
|
|
2013-08-03 00:43:56 +00:00
|
|
|
getWidget(mSneakBox, "SneakBox");
|
|
|
|
mSneakBoxBaseLeft = mSneakBox->getLeft();
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
getWidget(mEffectBox, "EffectBox");
|
|
|
|
mEffectBoxBaseRight = viewSize.width - mEffectBox->getRight();
|
2012-05-26 21:06:15 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
getWidget(mMinimapBox, "MiniMapBox");
|
|
|
|
mMinimapBoxBaseRight = viewSize.width - mMinimapBox->getRight();
|
|
|
|
getWidget(mMinimap, "MiniMap");
|
|
|
|
getWidget(mCompass, "Compass");
|
|
|
|
getWidget(mMinimapButton, "MiniMapButton");
|
|
|
|
mMinimapButton->eventMouseButtonClick += MyGUI::newDelegate(this, &HUD::onMapClicked);
|
2010-09-18 19:24:05 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
getWidget(mCellNameBox, "CellName");
|
|
|
|
getWidget(mWeaponSpellBox, "WeaponSpellName");
|
2012-04-13 11:17:50 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
getWidget(mCrosshair, "Crosshair");
|
2010-09-18 19:24:05 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
setFpsLevel(fpsLevel);
|
2012-05-13 16:35:56 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
getWidget(mTriangleCounter, "TriangleCounter");
|
|
|
|
getWidget(mBatchCounter, "BatchCounter");
|
2013-03-03 11:01:19 +00:00
|
|
|
|
2014-12-23 01:33:14 +00:00
|
|
|
LocalMapBase::init(mMinimap, mCompass, Settings::Manager::getInt("local map hud widget size", "Map"));
|
2013-03-03 11:01:19 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mMainWidget->eventMouseButtonClick += MyGUI::newDelegate(this, &HUD::onWorldClicked);
|
|
|
|
mMainWidget->eventMouseMove += MyGUI::newDelegate(this, &HUD::onWorldMouseOver);
|
|
|
|
mMainWidget->eventMouseLostFocus += MyGUI::newDelegate(this, &HUD::onWorldMouseLostFocus);
|
2010-09-18 19:24:05 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mSpellIcons = new SpellIcons();
|
|
|
|
}
|
2012-04-13 11:17:50 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
HUD::~HUD()
|
2012-04-13 11:17:50 +00:00
|
|
|
{
|
2015-01-31 15:26:15 +00:00
|
|
|
mMainWidget->eventMouseLostFocus.clear();
|
|
|
|
mMainWidget->eventMouseMove.clear();
|
|
|
|
mMainWidget->eventMouseButtonClick.clear();
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
delete mSpellIcons;
|
2012-04-13 11:17:50 +00:00
|
|
|
}
|
2013-04-17 22:56:48 +00:00
|
|
|
|
|
|
|
void HUD::setFpsLevel(int level)
|
2012-04-13 11:17:50 +00:00
|
|
|
{
|
2013-04-17 22:56:48 +00:00
|
|
|
mFpsCounter = 0;
|
2012-04-13 11:17:50 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
MyGUI::Widget* fps;
|
|
|
|
getWidget(fps, "FPSBoxAdv");
|
|
|
|
fps->setVisible(false);
|
|
|
|
getWidget(fps, "FPSBox");
|
|
|
|
fps->setVisible(false);
|
2010-09-18 19:34:49 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
if (level == 2)
|
|
|
|
{
|
|
|
|
getWidget(mFpsBox, "FPSBoxAdv");
|
|
|
|
mFpsBox->setVisible(true);
|
|
|
|
getWidget(mFpsCounter, "FPSCounterAdv");
|
|
|
|
}
|
|
|
|
else if (level == 1)
|
|
|
|
{
|
|
|
|
getWidget(mFpsBox, "FPSBox");
|
|
|
|
mFpsBox->setVisible(true);
|
|
|
|
getWidget(mFpsCounter, "FPSCounter");
|
|
|
|
}
|
|
|
|
}
|
2012-01-16 15:33:21 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setFPS(float fps)
|
|
|
|
{
|
|
|
|
if (mFpsCounter)
|
2015-01-10 02:01:01 +00:00
|
|
|
mFpsCounter->setCaption(MyGUI::utility::toString((int)fps));
|
2013-04-17 22:56:48 +00:00
|
|
|
}
|
2011-01-02 16:24:57 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setTriangleCount(unsigned int count)
|
2010-09-18 16:04:53 +00:00
|
|
|
{
|
2015-01-10 02:01:01 +00:00
|
|
|
mTriangleCounter->setCaption(MyGUI::utility::toString(count));
|
2013-04-17 22:56:48 +00:00
|
|
|
}
|
2011-01-02 16:24:57 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setBatchCount(unsigned int count)
|
|
|
|
{
|
2015-01-10 02:01:01 +00:00
|
|
|
mBatchCounter->setCaption(MyGUI::utility::toString(count));
|
2013-04-17 22:56:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void HUD::setValue(const std::string& id, const MWMechanics::DynamicStat<float>& value)
|
|
|
|
{
|
2013-12-09 23:41:36 +00:00
|
|
|
int current = std::max(0, static_cast<int>(value.getCurrent()));
|
|
|
|
int modified = static_cast<int>(value.getModified());
|
|
|
|
|
|
|
|
MyGUI::Widget* w;
|
2015-01-10 02:01:01 +00:00
|
|
|
std::string valStr = MyGUI::utility::toString(current) + "/" + MyGUI::utility::toString(modified);
|
2013-12-09 23:41:36 +00:00
|
|
|
if (id == "HBar")
|
|
|
|
{
|
|
|
|
mHealth->setProgressRange(modified);
|
|
|
|
mHealth->setProgressPosition(current);
|
|
|
|
getWidget(w, "HealthFrame");
|
|
|
|
w->setUserString("Caption_HealthDescription", "#{sHealthDesc}\n" + valStr);
|
|
|
|
}
|
|
|
|
else if (id == "MBar")
|
2010-09-18 16:04:53 +00:00
|
|
|
{
|
2013-12-09 23:41:36 +00:00
|
|
|
mMagicka->setProgressRange (modified);
|
|
|
|
mMagicka->setProgressPosition (current);
|
|
|
|
getWidget(w, "MagickaFrame");
|
|
|
|
w->setUserString("Caption_HealthDescription", "#{sIntDesc}\n" + valStr);
|
|
|
|
}
|
|
|
|
else if (id == "FBar")
|
|
|
|
{
|
|
|
|
mStamina->setProgressRange (modified);
|
|
|
|
mStamina->setProgressPosition (current);
|
|
|
|
getWidget(w, "FatigueFrame");
|
|
|
|
w->setUserString("Caption_HealthDescription", "#{sFatDesc}\n" + valStr);
|
|
|
|
}
|
2013-04-17 22:56:48 +00:00
|
|
|
}
|
2012-08-12 23:26:15 +00:00
|
|
|
|
2014-04-27 02:27:26 +00:00
|
|
|
void HUD::setDrowningTimeLeft(float time, float maxTime)
|
2013-08-07 13:34:11 +00:00
|
|
|
{
|
2014-04-27 02:27:26 +00:00
|
|
|
size_t progress = time/maxTime*200.0;
|
2013-10-27 08:05:01 +00:00
|
|
|
mDrowning->setProgressPosition(progress);
|
|
|
|
|
|
|
|
bool isDrowning = (progress == 0);
|
|
|
|
if (isDrowning && !mIsDrowning) // Just started drowning
|
|
|
|
mDrowningFlashTheta = 0.0f; // Start out on bright red every time.
|
|
|
|
|
|
|
|
mDrowningFlash->setVisible(isDrowning);
|
|
|
|
mIsDrowning = isDrowning;
|
2013-08-07 13:34:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void HUD::setDrowningBarVisible(bool visible)
|
|
|
|
{
|
|
|
|
mDrowningFrame->setVisible(visible);
|
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::onWorldClicked(MyGUI::Widget* _sender)
|
2012-05-13 16:35:56 +00:00
|
|
|
{
|
2013-04-17 22:56:48 +00:00
|
|
|
if (!MWBase::Environment::get().getWindowManager ()->isGuiMode ())
|
|
|
|
return;
|
2012-05-13 16:35:56 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
if (mDragAndDrop->mIsOnDragAndDrop)
|
|
|
|
{
|
|
|
|
// drop item into the gameworld
|
2014-05-12 00:20:56 +00:00
|
|
|
MWBase::Environment::get().getWorld()->breakInvisibility(
|
|
|
|
MWBase::Environment::get().getWorld()->getPlayerPtr());
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
MyGUI::IntSize viewSize = MyGUI::RenderManager::getInstance().getViewSize();
|
|
|
|
MyGUI::IntPoint cursorPosition = MyGUI::InputManager::getInstance().getMousePosition();
|
|
|
|
float mouseX = cursorPosition.left / float(viewSize.width);
|
|
|
|
float mouseY = cursorPosition.top / float(viewSize.height);
|
2012-05-15 21:28:04 +00:00
|
|
|
|
2014-05-16 01:19:38 +00:00
|
|
|
WorldItemModel drop (mouseX, mouseY);
|
|
|
|
mDragAndDrop->drop(&drop, NULL);
|
2012-05-14 15:41:17 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->changePointer("arrow");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GuiMode mode = MWBase::Environment::get().getWindowManager()->getMode();
|
2012-06-02 10:25:24 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
if ( (mode != GM_Console) && (mode != GM_Container) && (mode != GM_Inventory) )
|
|
|
|
return;
|
2012-06-02 10:25:24 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
MWWorld::Ptr object = MWBase::Environment::get().getWorld()->getFacedObject();
|
|
|
|
|
|
|
|
if (mode == GM_Console)
|
|
|
|
MWBase::Environment::get().getWindowManager()->getConsole()->setSelectedObject(object);
|
|
|
|
else if ((mode == GM_Container) || (mode == GM_Inventory))
|
|
|
|
{
|
|
|
|
// pick up object
|
2014-01-17 14:27:59 +00:00
|
|
|
if (!object.isEmpty())
|
|
|
|
MWBase::Environment::get().getWindowManager()->getInventoryWindow()->pickUpObject(object);
|
2013-04-17 22:56:48 +00:00
|
|
|
}
|
2012-06-02 10:25:24 +00:00
|
|
|
}
|
|
|
|
}
|
2012-05-13 16:35:56 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::onWorldMouseOver(MyGUI::Widget* _sender, int x, int y)
|
2012-05-15 14:47:23 +00:00
|
|
|
{
|
2013-04-17 22:56:48 +00:00
|
|
|
if (mDragAndDrop->mIsOnDragAndDrop)
|
|
|
|
{
|
|
|
|
mWorldMouseOver = false;
|
|
|
|
|
|
|
|
MyGUI::IntSize viewSize = MyGUI::RenderManager::getInstance().getViewSize();
|
|
|
|
MyGUI::IntPoint cursorPosition = MyGUI::InputManager::getInstance().getMousePosition();
|
|
|
|
float mouseX = cursorPosition.left / float(viewSize.width);
|
|
|
|
float mouseY = cursorPosition.top / float(viewSize.height);
|
2012-06-02 10:25:24 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
MWBase::World* world = MWBase::Environment::get().getWorld();
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
// if we can't drop the object at the wanted position, show the "drop on ground" cursor.
|
|
|
|
bool canDrop = world->canPlaceObject(mouseX, mouseY);
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
if (!canDrop)
|
|
|
|
MWBase::Environment::get().getWindowManager()->changePointer("drop_ground");
|
|
|
|
else
|
|
|
|
MWBase::Environment::get().getWindowManager()->changePointer("arrow");
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
}
|
2012-05-15 14:47:23 +00:00
|
|
|
else
|
2013-04-17 22:56:48 +00:00
|
|
|
{
|
2013-03-07 11:46:26 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->changePointer("arrow");
|
2013-04-17 22:56:48 +00:00
|
|
|
mWorldMouseOver = true;
|
|
|
|
}
|
2012-05-15 14:47:23 +00:00
|
|
|
}
|
2013-04-17 22:56:48 +00:00
|
|
|
|
|
|
|
void HUD::onWorldMouseLostFocus(MyGUI::Widget* _sender, MyGUI::Widget* _new)
|
2012-05-15 14:47:23 +00:00
|
|
|
{
|
2013-03-07 11:46:26 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->changePointer("arrow");
|
2013-04-17 22:56:48 +00:00
|
|
|
mWorldMouseOver = false;
|
2012-05-15 14:47:23 +00:00
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::onHMSClicked(MyGUI::Widget* _sender)
|
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWindowManager()->toggleVisible(GW_Stats);
|
|
|
|
}
|
2012-05-26 19:09:21 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::onMapClicked(MyGUI::Widget* _sender)
|
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWindowManager()->toggleVisible(GW_Map);
|
|
|
|
}
|
2012-05-26 21:06:15 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::onWeaponClicked(MyGUI::Widget* _sender)
|
2012-05-26 21:06:15 +00:00
|
|
|
{
|
2014-01-08 17:39:44 +00:00
|
|
|
const MWWorld::Ptr& player = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
2014-05-22 18:37:22 +00:00
|
|
|
if (player.getClass().getNpcStats(player).isWerewolf())
|
2013-08-12 13:36:16 +00:00
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sWerewolfRefusal}");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->toggleVisible(GW_Inventory);
|
|
|
|
}
|
2012-05-26 21:06:15 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::onMagicClicked(MyGUI::Widget* _sender)
|
|
|
|
{
|
2014-01-08 17:39:44 +00:00
|
|
|
const MWWorld::Ptr& player = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
2014-05-22 18:37:22 +00:00
|
|
|
if (player.getClass().getNpcStats(player).isWerewolf())
|
2013-08-12 13:36:16 +00:00
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sWerewolfRefusal}");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->toggleVisible(GW_Magic);
|
2012-05-26 21:06:15 +00:00
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setCellName(const std::string& cellName)
|
|
|
|
{
|
|
|
|
if (mCellName != cellName)
|
|
|
|
{
|
|
|
|
mCellNameTimer = 5.0f;
|
|
|
|
mCellName = cellName;
|
2012-05-28 07:19:25 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mCellNameBox->setCaptionWithReplacing("#{sCell=" + mCellName + "}");
|
|
|
|
mCellNameBox->setVisible(mMapVisible);
|
|
|
|
}
|
|
|
|
}
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::onFrame(float dt)
|
|
|
|
{
|
2014-06-17 14:27:33 +00:00
|
|
|
LocalMapBase::onFrame(dt);
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mCellNameTimer -= dt;
|
|
|
|
mWeaponSpellTimer -= dt;
|
|
|
|
if (mCellNameTimer < 0)
|
|
|
|
mCellNameBox->setVisible(false);
|
|
|
|
if (mWeaponSpellTimer < 0)
|
|
|
|
mWeaponSpellBox->setVisible(false);
|
2013-07-30 04:00:20 +00:00
|
|
|
|
|
|
|
mEnemyHealthTimer -= dt;
|
|
|
|
if (mEnemyHealth->getVisible() && mEnemyHealthTimer < 0)
|
|
|
|
{
|
|
|
|
mEnemyHealth->setVisible(false);
|
|
|
|
mWeaponSpellBox->setPosition(mWeaponSpellBox->getPosition() + MyGUI::IntPoint(0,20));
|
|
|
|
}
|
2013-10-27 08:05:01 +00:00
|
|
|
|
|
|
|
if (mIsDrowning)
|
|
|
|
mDrowningFlashTheta += dt * Ogre::Math::TWO_PI;
|
2013-04-17 22:56:48 +00:00
|
|
|
}
|
2012-11-05 19:41:26 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setSelectedSpell(const std::string& spellId, int successChancePercent)
|
|
|
|
{
|
|
|
|
const ESM::Spell* spell =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().find(spellId);
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
std::string spellName = spell->mName;
|
|
|
|
if (spellName != mSpellName && mSpellVisible)
|
|
|
|
{
|
|
|
|
mWeaponSpellTimer = 5.0f;
|
|
|
|
mSpellName = spellName;
|
|
|
|
mWeaponSpellBox->setCaption(mSpellName);
|
|
|
|
mWeaponSpellBox->setVisible(true);
|
|
|
|
}
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mSpellStatus->setProgressRange(100);
|
|
|
|
mSpellStatus->setProgressPosition(successChancePercent);
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mSpellBox->setUserString("ToolTipType", "Spell");
|
|
|
|
mSpellBox->setUserString("Spell", spellId);
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
// use the icon of the first effect
|
|
|
|
const ESM::MagicEffect* effect =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::MagicEffect>().find(spell->mEffects.mList.front().mEffectID);
|
|
|
|
|
|
|
|
std::string icon = effect->mIcon;
|
2014-08-16 16:12:18 +00:00
|
|
|
int slashPos = icon.rfind('\\');
|
2013-04-17 22:56:48 +00:00
|
|
|
icon.insert(slashPos+1, "b_");
|
2014-08-16 16:12:18 +00:00
|
|
|
icon = Misc::ResourceHelpers::correctIconPath(icon);
|
2014-06-05 20:13:18 +00:00
|
|
|
|
|
|
|
mSpellImage->setItem(MWWorld::Ptr());
|
|
|
|
mSpellImage->setIcon(icon);
|
2012-05-29 16:33:01 +00:00
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setSelectedEnchantItem(const MWWorld::Ptr& item, int chargePercent)
|
|
|
|
{
|
2014-05-22 18:37:22 +00:00
|
|
|
std::string itemName = item.getClass().getName(item);
|
2013-04-17 22:56:48 +00:00
|
|
|
if (itemName != mSpellName && mSpellVisible)
|
|
|
|
{
|
|
|
|
mWeaponSpellTimer = 5.0f;
|
|
|
|
mSpellName = itemName;
|
|
|
|
mWeaponSpellBox->setCaption(mSpellName);
|
|
|
|
mWeaponSpellBox->setVisible(true);
|
|
|
|
}
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mSpellStatus->setProgressRange(100);
|
|
|
|
mSpellStatus->setProgressPosition(chargePercent);
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mSpellBox->setUserString("ToolTipType", "ItemPtr");
|
|
|
|
mSpellBox->setUserData(item);
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2014-06-05 20:13:18 +00:00
|
|
|
mSpellImage->setItem(item);
|
2012-05-29 16:33:01 +00:00
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setSelectedWeapon(const MWWorld::Ptr& item, int durabilityPercent)
|
|
|
|
{
|
2014-05-22 18:37:22 +00:00
|
|
|
std::string itemName = item.getClass().getName(item);
|
2013-04-17 22:56:48 +00:00
|
|
|
if (itemName != mWeaponName && mWeaponVisible)
|
|
|
|
{
|
|
|
|
mWeaponSpellTimer = 5.0f;
|
|
|
|
mWeaponName = itemName;
|
|
|
|
mWeaponSpellBox->setCaption(mWeaponName);
|
|
|
|
mWeaponSpellBox->setVisible(true);
|
|
|
|
}
|
|
|
|
|
2014-08-25 01:15:28 +00:00
|
|
|
mWeapBox->clearUserStrings();
|
2013-04-17 22:56:48 +00:00
|
|
|
mWeapBox->setUserString("ToolTipType", "ItemPtr");
|
|
|
|
mWeapBox->setUserData(item);
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mWeapStatus->setProgressRange(100);
|
|
|
|
mWeapStatus->setProgressPosition(durabilityPercent);
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2014-06-05 20:13:18 +00:00
|
|
|
mWeapImage->setItem(item);
|
2012-05-29 16:33:01 +00:00
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::unsetSelectedSpell()
|
2012-05-29 16:33:01 +00:00
|
|
|
{
|
2013-04-17 22:56:48 +00:00
|
|
|
std::string spellName = "#{sNone}";
|
|
|
|
if (spellName != mSpellName && mSpellVisible)
|
|
|
|
{
|
|
|
|
mWeaponSpellTimer = 5.0f;
|
|
|
|
mSpellName = spellName;
|
|
|
|
mWeaponSpellBox->setCaptionWithReplacing(mSpellName);
|
|
|
|
mWeaponSpellBox->setVisible(true);
|
|
|
|
}
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mSpellStatus->setProgressRange(100);
|
|
|
|
mSpellStatus->setProgressPosition(0);
|
2014-06-05 20:13:18 +00:00
|
|
|
mSpellImage->setItem(MWWorld::Ptr());
|
2013-04-17 22:56:48 +00:00
|
|
|
mSpellBox->clearUserStrings();
|
|
|
|
}
|
2012-05-29 16:33:01 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::unsetSelectedWeapon()
|
2012-05-29 16:33:01 +00:00
|
|
|
{
|
2013-04-17 22:56:48 +00:00
|
|
|
std::string itemName = "#{sSkillHandtohand}";
|
|
|
|
if (itemName != mWeaponName && mWeaponVisible)
|
|
|
|
{
|
|
|
|
mWeaponSpellTimer = 5.0f;
|
|
|
|
mWeaponName = itemName;
|
|
|
|
mWeaponSpellBox->setCaptionWithReplacing(mWeaponName);
|
|
|
|
mWeaponSpellBox->setVisible(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
mWeapStatus->setProgressRange(100);
|
|
|
|
mWeapStatus->setProgressPosition(0);
|
2013-08-05 22:07:24 +00:00
|
|
|
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2014-01-08 17:39:44 +00:00
|
|
|
MWWorld::Ptr player = world->getPlayerPtr();
|
2014-06-05 20:13:18 +00:00
|
|
|
|
|
|
|
mWeapImage->setItem(MWWorld::Ptr());
|
2014-08-25 01:15:28 +00:00
|
|
|
std::string icon = (player.getClass().getNpcStats(player).isWerewolf()) ? "icons\\k\\tx_werewolf_hand.dds" : "icons\\k\\stealth_handtohand.dds";
|
|
|
|
mWeapImage->setIcon(icon);
|
2013-08-05 22:07:24 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mWeapBox->clearUserStrings();
|
2014-08-25 01:15:28 +00:00
|
|
|
mWeapBox->setUserString("ToolTipType", "Layout");
|
|
|
|
mWeapBox->setUserString("ToolTipLayout", "HandToHandToolTip");
|
|
|
|
mWeapBox->setUserString("Caption_HandToHandText", itemName);
|
|
|
|
mWeapBox->setUserString("ImageTexture_HandToHandImage", icon);
|
2012-05-29 16:33:01 +00:00
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setCrosshairVisible(bool visible)
|
|
|
|
{
|
|
|
|
mCrosshair->setVisible (visible);
|
|
|
|
}
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setHmsVisible(bool visible)
|
|
|
|
{
|
|
|
|
mHealth->setVisible(visible);
|
|
|
|
mMagicka->setVisible(visible);
|
|
|
|
mStamina->setVisible(visible);
|
|
|
|
updatePositions();
|
|
|
|
}
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setWeapVisible(bool visible)
|
|
|
|
{
|
|
|
|
mWeapBox->setVisible(visible);
|
|
|
|
updatePositions();
|
|
|
|
}
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setSpellVisible(bool visible)
|
|
|
|
{
|
|
|
|
mSpellBox->setVisible(visible);
|
|
|
|
updatePositions();
|
|
|
|
}
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-08-03 00:43:56 +00:00
|
|
|
void HUD::setSneakVisible(bool visible)
|
|
|
|
{
|
|
|
|
mSneakBox->setVisible(visible);
|
|
|
|
updatePositions();
|
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setEffectVisible(bool visible)
|
|
|
|
{
|
|
|
|
mEffectBox->setVisible (visible);
|
|
|
|
updatePositions();
|
|
|
|
}
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::setMinimapVisible(bool visible)
|
|
|
|
{
|
|
|
|
mMinimapBox->setVisible (visible);
|
|
|
|
updatePositions();
|
|
|
|
}
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::updatePositions()
|
|
|
|
{
|
2013-08-03 00:43:56 +00:00
|
|
|
int weapDx = 0, spellDx = 0, sneakDx = 0;
|
2013-04-17 22:56:48 +00:00
|
|
|
if (!mHealth->getVisible())
|
2013-08-03 00:43:56 +00:00
|
|
|
sneakDx = spellDx = weapDx = mWeapBoxBaseLeft - mHealthManaStaminaBaseLeft;
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
if (!mWeapBox->getVisible())
|
2013-08-03 00:43:56 +00:00
|
|
|
{
|
2013-04-17 22:56:48 +00:00
|
|
|
spellDx += mSpellBoxBaseLeft - mWeapBoxBaseLeft;
|
2013-08-03 00:43:56 +00:00
|
|
|
sneakDx = spellDx;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!mSpellBox->getVisible())
|
|
|
|
sneakDx += mSneakBoxBaseLeft - mSpellBoxBaseLeft;
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mWeaponVisible = mWeapBox->getVisible();
|
|
|
|
mSpellVisible = mSpellBox->getVisible();
|
|
|
|
if (!mWeaponVisible && !mSpellVisible)
|
|
|
|
mWeaponSpellBox->setVisible(false);
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mWeapBox->setPosition(mWeapBoxBaseLeft - weapDx, mWeapBox->getTop());
|
|
|
|
mSpellBox->setPosition(mSpellBoxBaseLeft - spellDx, mSpellBox->getTop());
|
2013-08-03 00:43:56 +00:00
|
|
|
mSneakBox->setPosition(mSneakBoxBaseLeft - sneakDx, mSneakBox->getTop());
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
const MyGUI::IntSize& viewSize = MyGUI::RenderManager::getInstance().getViewSize();
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
// effect box can have variable width -> variable left coordinate
|
|
|
|
int effectsDx = 0;
|
|
|
|
if (!mMinimapBox->getVisible ())
|
|
|
|
effectsDx = (viewSize.width - mMinimapBoxBaseRight) - (viewSize.width - mEffectBoxBaseRight);
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mMapVisible = mMinimapBox->getVisible ();
|
2014-05-20 07:45:39 +00:00
|
|
|
if (!mMapVisible)
|
|
|
|
mCellNameBox->setVisible(false);
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
mEffectBox->setPosition((viewSize.width - mEffectBoxBaseRight) - mEffectBox->getWidth() + effectsDx, mEffectBox->getTop());
|
|
|
|
}
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2014-03-15 19:48:54 +00:00
|
|
|
void HUD::updateEnemyHealthBar()
|
|
|
|
{
|
2014-07-23 20:04:18 +00:00
|
|
|
MWWorld::Ptr enemy = MWBase::Environment::get().getWorld()->searchPtrViaActorId(mEnemyActorId);
|
|
|
|
if (enemy.isEmpty())
|
|
|
|
return;
|
|
|
|
MWMechanics::CreatureStats& stats = enemy.getClass().getCreatureStats(enemy);
|
2014-03-15 19:48:54 +00:00
|
|
|
mEnemyHealth->setProgressRange(100);
|
|
|
|
// Health is usually cast to int before displaying. Actors die whenever they are < 1 health.
|
|
|
|
// Therefore any value < 1 should show as an empty health bar. We do the same in statswindow :)
|
|
|
|
mEnemyHealth->setProgressPosition(int(stats.getHealth().getCurrent()) / stats.getHealth().getModified() * 100);
|
2014-12-19 18:51:36 +00:00
|
|
|
|
|
|
|
static const float fNPCHealthBarFade = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fNPCHealthBarFade")->getFloat();
|
|
|
|
if (fNPCHealthBarFade > 0.f)
|
|
|
|
mEnemyHealth->setAlpha(std::max(0.f, std::min(1.f, mEnemyHealthTimer/fNPCHealthBarFade)));
|
|
|
|
|
2014-03-15 19:48:54 +00:00
|
|
|
}
|
|
|
|
|
2013-04-17 22:56:48 +00:00
|
|
|
void HUD::update()
|
|
|
|
{
|
|
|
|
mSpellIcons->updateWidgets(mEffectBox, true);
|
2013-07-30 04:00:20 +00:00
|
|
|
|
2014-07-23 20:04:18 +00:00
|
|
|
if (mEnemyActorId != -1 && mEnemyHealth->getVisible())
|
2013-07-30 04:00:20 +00:00
|
|
|
{
|
2014-03-15 19:48:54 +00:00
|
|
|
updateEnemyHealthBar();
|
2013-07-30 04:00:20 +00:00
|
|
|
}
|
2013-10-27 08:05:01 +00:00
|
|
|
|
|
|
|
if (mIsDrowning)
|
|
|
|
{
|
|
|
|
float intensity = (cos(mDrowningFlashTheta) + 1.0f) / 2.0f;
|
2014-04-27 07:14:14 +00:00
|
|
|
mDrowningFlash->setColour(MyGUI::Colour(intensity, 0, 0));
|
2013-10-27 08:05:01 +00:00
|
|
|
}
|
2013-07-30 04:00:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void HUD::setEnemy(const MWWorld::Ptr &enemy)
|
|
|
|
{
|
2014-07-23 20:04:18 +00:00
|
|
|
mEnemyActorId = enemy.getClass().getCreatureStats(enemy).getActorId();
|
2014-12-19 18:51:36 +00:00
|
|
|
mEnemyHealthTimer = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fNPCHealthBarTime")->getFloat();
|
2013-07-30 04:00:20 +00:00
|
|
|
if (!mEnemyHealth->getVisible())
|
|
|
|
mWeaponSpellBox->setPosition(mWeaponSpellBox->getPosition() - MyGUI::IntPoint(0,20));
|
|
|
|
mEnemyHealth->setVisible(true);
|
2014-03-15 19:48:54 +00:00
|
|
|
updateEnemyHealthBar();
|
2013-04-17 22:56:48 +00:00
|
|
|
}
|
2013-03-03 11:01:19 +00:00
|
|
|
|
2014-04-23 17:49:09 +00:00
|
|
|
void HUD::resetEnemy()
|
|
|
|
{
|
2014-07-23 20:04:18 +00:00
|
|
|
mEnemyActorId = -1;
|
2014-04-23 17:49:09 +00:00
|
|
|
mEnemyHealthTimer = -1;
|
|
|
|
}
|
|
|
|
|
2013-03-03 11:01:19 +00:00
|
|
|
}
|