2012-07-03 10:30:50 +00:00
|
|
|
#include "worldimp.hpp"
|
2013-09-29 07:11:57 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <boost/tr1/tr1/unordered_map>
|
|
|
|
#elif defined HAVE_UNORDERED_MAP
|
|
|
|
#include <unordered_map>
|
|
|
|
#else
|
|
|
|
#include <tr1/unordered_map>
|
|
|
|
#endif
|
2010-07-02 07:38:22 +00:00
|
|
|
|
2013-08-16 11:01:52 +00:00
|
|
|
#include <OgreSceneNode.h>
|
|
|
|
|
2014-04-13 04:53:36 +00:00
|
|
|
#include <libs/openengine/bullet/trace.h>
|
2013-02-07 05:44:58 +00:00
|
|
|
#include <libs/openengine/bullet/physic.hpp>
|
|
|
|
|
2010-07-27 12:05:05 +00:00
|
|
|
#include <components/bsa/bsa_archive.hpp>
|
2011-05-05 17:32:42 +00:00
|
|
|
#include <components/files/collections.hpp>
|
2013-01-31 18:45:32 +00:00
|
|
|
#include <components/compiler/locals.hpp>
|
2014-01-16 11:03:23 +00:00
|
|
|
#include <components/esm/cellid.hpp>
|
2010-07-02 07:38:22 +00:00
|
|
|
|
2013-04-30 07:43:32 +00:00
|
|
|
#include <boost/math/special_functions/sign.hpp>
|
|
|
|
|
2012-04-23 13:27:03 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
2012-08-09 12:33:21 +00:00
|
|
|
#include "../mwbase/soundmanager.hpp"
|
2012-08-11 15:30:55 +00:00
|
|
|
#include "../mwbase/mechanicsmanager.hpp"
|
2012-08-12 16:11:09 +00:00
|
|
|
#include "../mwbase/windowmanager.hpp"
|
2012-04-23 13:27:03 +00:00
|
|
|
|
2013-02-20 09:55:12 +00:00
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
2013-03-31 07:13:56 +00:00
|
|
|
#include "../mwmechanics/movement.hpp"
|
2013-08-05 21:23:39 +00:00
|
|
|
#include "../mwmechanics/npcstats.hpp"
|
2013-11-16 01:34:43 +00:00
|
|
|
#include "../mwmechanics/spellcasting.hpp"
|
2014-01-14 06:40:17 +00:00
|
|
|
#include "../mwmechanics/levelledlist.hpp"
|
2014-03-08 04:51:47 +00:00
|
|
|
#include "../mwmechanics/combat.hpp"
|
2014-05-13 07:58:32 +00:00
|
|
|
#include "../mwmechanics/aiavoiddoor.hpp" //Used to tell actors to avoid doors
|
2013-02-20 09:55:12 +00:00
|
|
|
|
2010-07-27 12:05:05 +00:00
|
|
|
#include "../mwrender/sky.hpp"
|
2013-07-25 19:58:43 +00:00
|
|
|
#include "../mwrender/animation.hpp"
|
2010-07-27 12:05:05 +00:00
|
|
|
|
2012-12-23 19:23:24 +00:00
|
|
|
#include "../mwclass/door.hpp"
|
|
|
|
|
2011-01-04 14:58:22 +00:00
|
|
|
#include "player.hpp"
|
2012-05-16 19:27:02 +00:00
|
|
|
#include "manualref.hpp"
|
2011-01-29 13:33:44 +00:00
|
|
|
#include "cellfunctors.hpp"
|
2013-01-27 11:03:48 +00:00
|
|
|
#include "containerstore.hpp"
|
2013-04-15 00:56:23 +00:00
|
|
|
#include "inventorystore.hpp"
|
2014-01-01 01:22:11 +00:00
|
|
|
#include "actionteleport.hpp"
|
2010-11-06 17:11:09 +00:00
|
|
|
|
2013-09-29 07:11:57 +00:00
|
|
|
#include "contentloader.hpp"
|
|
|
|
#include "esmloader.hpp"
|
|
|
|
#include "omwloader.hpp"
|
|
|
|
|
2012-02-24 15:12:43 +00:00
|
|
|
using namespace Ogre;
|
|
|
|
|
2010-07-03 13:41:20 +00:00
|
|
|
namespace MWWorld
|
2010-07-02 07:38:22 +00:00
|
|
|
{
|
2013-09-29 07:11:57 +00:00
|
|
|
struct GameContentLoader : public ContentLoader
|
|
|
|
{
|
|
|
|
GameContentLoader(Loading::Listener& listener)
|
|
|
|
: ContentLoader(listener)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool addLoader(const std::string& extension, ContentLoader* loader)
|
|
|
|
{
|
|
|
|
return mLoaders.insert(std::make_pair(extension, loader)).second;
|
|
|
|
}
|
|
|
|
|
|
|
|
void load(const boost::filesystem::path& filepath, int& index)
|
|
|
|
{
|
2013-11-30 13:25:29 +00:00
|
|
|
LoadersContainer::iterator it(mLoaders.find(Misc::StringUtils::lowerCase(filepath.extension().string())));
|
2013-09-29 07:11:57 +00:00
|
|
|
if (it != mLoaders.end())
|
|
|
|
{
|
|
|
|
it->second->load(filepath, index);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::string msg("Cannot load file: ");
|
|
|
|
msg += filepath.string();
|
|
|
|
throw std::runtime_error(msg.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef std::tr1::unordered_map<std::string, ContentLoader*> LoadersContainer;
|
|
|
|
LoadersContainer mLoaders;
|
|
|
|
};
|
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
int World::getDaysPerMonth (int month) const
|
|
|
|
{
|
|
|
|
switch (month)
|
|
|
|
{
|
|
|
|
case 0: return 31;
|
|
|
|
case 1: return 28;
|
|
|
|
case 2: return 31;
|
|
|
|
case 3: return 30;
|
|
|
|
case 4: return 31;
|
|
|
|
case 5: return 30;
|
|
|
|
case 6: return 31;
|
|
|
|
case 7: return 31;
|
|
|
|
case 8: return 30;
|
|
|
|
case 9: return 31;
|
|
|
|
case 10: return 30;
|
|
|
|
case 11: return 31;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
throw std::runtime_error ("month out of range");
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-08-22 18:55:22 +00:00
|
|
|
void World::adjustSky()
|
|
|
|
{
|
2012-02-26 17:21:11 +00:00
|
|
|
if (mSky && (isCellExterior() || isCellQuasiExterior()))
|
2010-08-22 18:55:22 +00:00
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
mRendering->skySetHour (mGlobalVariables["gamehour"].getFloat());
|
|
|
|
mRendering->skySetDate (mGlobalVariables["day"].getInteger(),
|
|
|
|
mGlobalVariables["month"].getInteger());
|
2012-02-26 17:21:11 +00:00
|
|
|
|
2012-04-01 13:07:41 +00:00
|
|
|
mRendering->skyEnable();
|
2010-08-22 18:55:22 +00:00
|
|
|
}
|
2012-02-26 17:21:11 +00:00
|
|
|
else
|
2012-04-01 13:07:41 +00:00
|
|
|
mRendering->skyDisable();
|
2010-08-22 18:55:22 +00:00
|
|
|
}
|
|
|
|
|
2012-01-29 18:51:20 +00:00
|
|
|
World::World (OEngine::Render::OgreRenderer& renderer,
|
2011-05-05 17:32:42 +00:00
|
|
|
const Files::Collections& fileCollections,
|
2013-09-29 07:11:57 +00:00
|
|
|
const std::vector<std::string>& contentFiles,
|
2013-05-15 15:54:18 +00:00
|
|
|
const boost::filesystem::path& resDir, const boost::filesystem::path& cacheDir,
|
2014-03-13 12:19:32 +00:00
|
|
|
ToUTF8::Utf8Encoder* encoder, const std::map<std::string,std::string>& fallbackMap,
|
|
|
|
int activationDistanceOverride, const std::string& startCell)
|
2013-12-10 14:09:58 +00:00
|
|
|
: mPlayer (0), mLocalScripts (mStore),
|
2012-12-18 18:09:27 +00:00
|
|
|
mSky (true), mCells (mStore, mEsm),
|
2014-03-13 12:19:32 +00:00
|
|
|
mActivationDistanceOverride (activationDistanceOverride),
|
2014-01-24 17:21:52 +00:00
|
|
|
mFallback(fallbackMap), mPlayIntro(0), mTeleportEnabled(true), mLevitationEnabled(true),
|
|
|
|
mFacedDistance(FLT_MAX), mGodMode(false), mContentFiles (contentFiles),
|
2014-03-13 12:19:32 +00:00
|
|
|
mGoToJail(false),
|
|
|
|
mStartCell (startCell)
|
2010-08-03 18:01:52 +00:00
|
|
|
{
|
2012-01-29 15:31:18 +00:00
|
|
|
mPhysics = new PhysicsSystem(renderer);
|
2012-01-29 18:51:20 +00:00
|
|
|
mPhysEngine = mPhysics->getEngine();
|
2012-03-15 15:01:41 +00:00
|
|
|
|
2013-03-15 09:17:30 +00:00
|
|
|
mRendering = new MWRender::RenderingManager(renderer, resDir, cacheDir, mPhysEngine,&mFallback);
|
2012-03-15 15:01:41 +00:00
|
|
|
|
2013-02-19 03:03:24 +00:00
|
|
|
mPhysEngine->setSceneManager(renderer.getScene());
|
|
|
|
|
2013-03-15 09:17:30 +00:00
|
|
|
mWeatherManager = new MWWorld::WeatherManager(mRendering,&mFallback);
|
2011-05-05 17:32:42 +00:00
|
|
|
|
2012-11-10 20:43:41 +00:00
|
|
|
// NOTE: We might need to reserve one more for the running game / save.
|
2013-09-29 07:11:57 +00:00
|
|
|
mEsm.resize(contentFiles.size());
|
2013-08-27 13:48:13 +00:00
|
|
|
Loading::Listener* listener = MWBase::Environment::get().getWindowManager()->getLoadingScreen();
|
|
|
|
listener->loadingOn();
|
2013-03-05 07:02:27 +00:00
|
|
|
|
2013-09-29 07:11:57 +00:00
|
|
|
GameContentLoader gameContentLoader(*listener);
|
|
|
|
EsmLoader esmLoader(mStore, mEsm, encoder, *listener);
|
|
|
|
OmwLoader omwLoader(*listener);
|
2012-10-07 18:00:55 +00:00
|
|
|
|
2013-09-29 07:11:57 +00:00
|
|
|
gameContentLoader.addLoader(".esm", &esmLoader);
|
|
|
|
gameContentLoader.addLoader(".esp", &esmLoader);
|
|
|
|
gameContentLoader.addLoader(".omwgame", &omwLoader);
|
|
|
|
gameContentLoader.addLoader(".omwaddon", &omwLoader);
|
2013-03-05 07:02:27 +00:00
|
|
|
|
2013-09-29 07:11:57 +00:00
|
|
|
loadContentFiles(fileCollections, contentFiles, gameContentLoader);
|
2012-10-07 18:00:55 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
listener->loadingOff();
|
2013-03-05 07:02:27 +00:00
|
|
|
|
2013-05-01 10:47:50 +00:00
|
|
|
// insert records that may not be present in all versions of MW
|
|
|
|
if (mEsm[0].getFormat() == 0)
|
|
|
|
ensureNeededRecords();
|
|
|
|
|
2012-11-25 18:07:16 +00:00
|
|
|
mStore.setUp();
|
2013-09-30 06:54:20 +00:00
|
|
|
mStore.movePlayerRecord();
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables.fill (mStore);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-05-15 15:54:18 +00:00
|
|
|
mWorldScene = new Scene(*mRendering, mPhysics);
|
|
|
|
}
|
|
|
|
|
2014-03-13 12:19:32 +00:00
|
|
|
void World::startNewGame (bool bypass)
|
2013-05-15 15:54:18 +00:00
|
|
|
{
|
2014-01-11 05:47:58 +00:00
|
|
|
mGoToJail = false;
|
|
|
|
mLevitationEnabled = true;
|
|
|
|
mTeleportEnabled = true;
|
2013-05-15 15:54:18 +00:00
|
|
|
|
|
|
|
// Rebuild player
|
|
|
|
setupPlayer();
|
|
|
|
|
2013-08-20 09:04:26 +00:00
|
|
|
renderPlayer();
|
2013-05-15 15:54:18 +00:00
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->updatePlayer();
|
|
|
|
|
2014-03-13 12:19:32 +00:00
|
|
|
if (bypass && !mStartCell.empty())
|
|
|
|
{
|
|
|
|
ESM::Position pos;
|
|
|
|
|
|
|
|
if (findExteriorPosition (mStartCell, pos))
|
|
|
|
{
|
|
|
|
changeToExteriorCell (pos);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
findInteriorPosition (mStartCell, pos);
|
|
|
|
changeToInteriorCell (mStartCell, pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// \todo if !bypass, do not add player location to global map for the duration of one
|
|
|
|
/// frame
|
|
|
|
ESM::Position pos;
|
|
|
|
const int cellSize = 8192;
|
|
|
|
pos.pos[0] = cellSize/2;
|
|
|
|
pos.pos[1] = cellSize/2;
|
|
|
|
pos.pos[2] = 0;
|
|
|
|
pos.rot[0] = 0;
|
|
|
|
pos.rot[1] = 0;
|
|
|
|
pos.rot[2] = 0;
|
|
|
|
mWorldScene->changeToExteriorCell(pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!bypass)
|
|
|
|
{
|
|
|
|
// FIXME: should be set to 1, but the sound manager won't pause newly started sounds
|
|
|
|
mPlayIntro = 2;
|
|
|
|
|
|
|
|
// set new game mark
|
|
|
|
mGlobalVariables["chargenstate"].setInteger (1);
|
|
|
|
mGlobalVariables["pcrace"].setInteger (3);
|
|
|
|
}
|
2012-12-04 09:58:43 +00:00
|
|
|
|
2013-05-15 15:54:18 +00:00
|
|
|
// we don't want old weather to persist on a new game
|
|
|
|
delete mWeatherManager;
|
2013-11-28 10:22:34 +00:00
|
|
|
mWeatherManager = 0;
|
2013-05-15 15:54:18 +00:00
|
|
|
mWeatherManager = new MWWorld::WeatherManager(mRendering,&mFallback);
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|
2012-04-04 14:08:46 +00:00
|
|
|
|
2013-12-05 11:49:25 +00:00
|
|
|
void World::clear()
|
|
|
|
{
|
2014-05-11 00:07:28 +00:00
|
|
|
mRendering->clear();
|
|
|
|
|
2013-12-05 11:49:25 +00:00
|
|
|
mLocalScripts.clear();
|
2014-01-11 14:34:32 +00:00
|
|
|
mPlayer->clear();
|
2013-12-05 11:49:25 +00:00
|
|
|
|
|
|
|
// enable collision
|
|
|
|
if (!mPhysics->toggleCollisionMode())
|
|
|
|
mPhysics->toggleCollisionMode();
|
|
|
|
|
|
|
|
mWorldScene->changeToVoid();
|
|
|
|
|
2014-01-18 13:53:25 +00:00
|
|
|
mStore.clearDynamic();
|
|
|
|
mStore.setUp();
|
|
|
|
|
2013-12-05 11:49:25 +00:00
|
|
|
if (mPlayer)
|
2013-12-07 12:17:28 +00:00
|
|
|
{
|
|
|
|
mPlayer->setCell (0);
|
2013-12-05 11:49:25 +00:00
|
|
|
mPlayer->getPlayer().getRefData() = RefData();
|
2014-01-18 13:53:25 +00:00
|
|
|
mPlayer->set (mStore.get<ESM::NPC>().find ("player"));
|
2013-12-07 12:17:28 +00:00
|
|
|
}
|
2013-12-05 11:49:25 +00:00
|
|
|
|
|
|
|
mCells.clear();
|
2013-12-05 13:18:43 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
mMagicBolts.clear();
|
2013-12-05 13:18:43 +00:00
|
|
|
mProjectiles.clear();
|
|
|
|
mDoorStates.clear();
|
|
|
|
|
|
|
|
mGodMode = false;
|
|
|
|
mSky = true;
|
|
|
|
mTeleportEnabled = true;
|
|
|
|
mPlayIntro = 0;
|
|
|
|
mFacedDistance = FLT_MAX;
|
2013-12-10 14:09:58 +00:00
|
|
|
|
|
|
|
mGlobalVariables.fill (mStore);
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-12-07 12:17:28 +00:00
|
|
|
int World::countSavedGameRecords() const
|
|
|
|
{
|
2013-12-12 11:19:25 +00:00
|
|
|
return
|
2014-04-28 09:29:57 +00:00
|
|
|
mCells.countSavedGameRecords()
|
|
|
|
+mStore.countSavedGameRecords()
|
2014-01-23 10:29:40 +00:00
|
|
|
+mGlobalVariables.countSavedGameRecords()
|
|
|
|
+1 // player record
|
2014-05-14 07:47:49 +00:00
|
|
|
+1 // weather record
|
|
|
|
+1; // actorId counter
|
2013-12-07 12:17:28 +00:00
|
|
|
}
|
|
|
|
|
2014-04-28 09:29:57 +00:00
|
|
|
void World::write (ESM::ESMWriter& writer, Loading::Listener& progress) const
|
2013-12-07 12:17:28 +00:00
|
|
|
{
|
2014-05-11 00:07:28 +00:00
|
|
|
// Active cells could have a dirty fog of war, sync it to the CellStore first
|
|
|
|
for (Scene::CellStoreCollection::const_iterator iter (mWorldScene->getActiveCells().begin());
|
|
|
|
iter!=mWorldScene->getActiveCells().end(); ++iter)
|
|
|
|
{
|
|
|
|
CellStore* cellstore = *iter;
|
|
|
|
mRendering->writeFog(cellstore);
|
|
|
|
}
|
|
|
|
|
2014-05-14 07:47:49 +00:00
|
|
|
MWMechanics::CreatureStats::writeActorIdCounter(writer);
|
|
|
|
progress.increaseProgress();
|
|
|
|
|
2014-04-28 09:29:57 +00:00
|
|
|
mCells.write (writer, progress);
|
|
|
|
mStore.write (writer, progress);
|
|
|
|
mGlobalVariables.write (writer, progress);
|
|
|
|
mPlayer->write (writer, progress);
|
|
|
|
mWeatherManager->write (writer, progress);
|
2013-12-07 12:17:28 +00:00
|
|
|
}
|
|
|
|
|
2014-01-27 12:27:42 +00:00
|
|
|
void World::readRecord (ESM::ESMReader& reader, int32_t type,
|
|
|
|
const std::map<int, int>& contentFileMap)
|
2013-12-07 12:17:28 +00:00
|
|
|
{
|
2014-05-14 07:47:49 +00:00
|
|
|
if (type == ESM::REC_ACTC)
|
|
|
|
{
|
|
|
|
MWMechanics::CreatureStats::readActorIdCounter(reader);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-12-12 11:19:25 +00:00
|
|
|
if (!mStore.readRecord (reader, type) &&
|
2014-01-16 11:03:23 +00:00
|
|
|
!mGlobalVariables.readRecord (reader, type) &&
|
2014-01-23 10:29:40 +00:00
|
|
|
!mPlayer->readRecord (reader, type) &&
|
2014-03-20 06:25:52 +00:00
|
|
|
!mWeatherManager->readRecord (reader, type) &&
|
2014-01-27 12:27:42 +00:00
|
|
|
!mCells.readRecord (reader, type, contentFileMap))
|
2013-12-07 12:17:28 +00:00
|
|
|
{
|
2013-12-12 11:19:25 +00:00
|
|
|
throw std::runtime_error ("unknown record in saved game");
|
2013-12-07 12:17:28 +00:00
|
|
|
}
|
|
|
|
}
|
2012-01-29 22:50:51 +00:00
|
|
|
|
2013-05-01 10:47:50 +00:00
|
|
|
void World::ensureNeededRecords()
|
|
|
|
{
|
|
|
|
if (!mStore.get<ESM::GameSetting>().search("sCompanionShare"))
|
|
|
|
{
|
|
|
|
ESM::GameSetting sCompanionShare;
|
|
|
|
sCompanionShare.mId = "sCompanionShare";
|
|
|
|
ESM::Variant value;
|
|
|
|
value.setType(ESM::VT_String);
|
|
|
|
value.setString("Companion Share");
|
|
|
|
sCompanionShare.mValue = value;
|
|
|
|
mStore.insertStatic(sCompanionShare);
|
|
|
|
}
|
|
|
|
if (!mStore.get<ESM::Global>().search("dayspassed"))
|
|
|
|
{
|
|
|
|
// vanilla Morrowind does not define dayspassed.
|
|
|
|
ESM::Global dayspassed;
|
|
|
|
dayspassed.mId = "dayspassed";
|
|
|
|
ESM::Variant value;
|
|
|
|
value.setType(ESM::VT_Long);
|
|
|
|
value.setInteger(1); // but the addons start counting at 1 :(
|
|
|
|
dayspassed.mValue = value;
|
|
|
|
mStore.insertStatic(dayspassed);
|
|
|
|
}
|
|
|
|
if (!mStore.get<ESM::GameSetting>().search("fWereWolfRunMult"))
|
|
|
|
{
|
|
|
|
ESM::GameSetting fWereWolfRunMult;
|
|
|
|
fWereWolfRunMult.mId = "fWereWolfRunMult";
|
|
|
|
ESM::Variant value;
|
|
|
|
value.setType(ESM::VT_Float);
|
|
|
|
value.setFloat(1.f);
|
|
|
|
fWereWolfRunMult.mValue = value;
|
|
|
|
mStore.insertStatic(fWereWolfRunMult);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-02 07:38:22 +00:00
|
|
|
World::~World()
|
|
|
|
{
|
2012-02-21 19:22:46 +00:00
|
|
|
delete mWeatherManager;
|
2011-08-09 07:56:09 +00:00
|
|
|
delete mWorldScene;
|
2012-01-29 18:51:20 +00:00
|
|
|
delete mRendering;
|
2011-08-01 13:55:36 +00:00
|
|
|
delete mPhysics;
|
2011-10-29 07:55:22 +00:00
|
|
|
|
|
|
|
delete mPlayer;
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|
2011-08-01 12:41:15 +00:00
|
|
|
|
|
|
|
const ESM::Cell *World::getExterior (const std::string& cellName) const
|
|
|
|
{
|
|
|
|
// first try named cells
|
2012-11-06 08:36:21 +00:00
|
|
|
const ESM::Cell *cell = mStore.get<ESM::Cell>().searchExtByName (cellName);
|
|
|
|
if (cell != 0) {
|
2011-08-01 12:41:15 +00:00
|
|
|
return cell;
|
2012-11-06 08:36:21 +00:00
|
|
|
}
|
2011-08-01 12:41:15 +00:00
|
|
|
|
|
|
|
// didn't work -> now check for regions
|
2012-11-06 08:36:21 +00:00
|
|
|
const MWWorld::Store<ESM::Region> ®ions = mStore.get<ESM::Region>();
|
|
|
|
MWWorld::Store<ESM::Region>::iterator it = regions.begin();
|
|
|
|
for (; it != regions.end(); ++it)
|
2011-08-01 12:41:15 +00:00
|
|
|
{
|
2012-12-23 19:23:24 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(cellName, it->mName))
|
2011-08-01 12:41:15 +00:00
|
|
|
{
|
2012-11-06 08:36:21 +00:00
|
|
|
return mStore.get<ESM::Cell>().searchExtByRegion(it->mId);
|
2011-08-01 12:41:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2011-08-09 07:56:09 +00:00
|
|
|
|
2013-03-18 07:29:40 +00:00
|
|
|
const MWWorld::Fallback *World::getFallback() const
|
2013-03-17 12:50:15 +00:00
|
|
|
{
|
|
|
|
return &mFallback;
|
|
|
|
}
|
|
|
|
|
2013-12-05 12:21:26 +00:00
|
|
|
CellStore *World::getExterior (int x, int y)
|
2011-08-01 12:34:50 +00:00
|
|
|
{
|
2011-09-08 09:02:55 +00:00
|
|
|
return mCells.getExterior (x, y);
|
2011-08-01 12:34:50 +00:00
|
|
|
}
|
2011-08-09 07:56:09 +00:00
|
|
|
|
2013-12-05 12:21:26 +00:00
|
|
|
CellStore *World::getInterior (const std::string& name)
|
2011-08-01 12:34:50 +00:00
|
|
|
{
|
2011-09-08 09:02:55 +00:00
|
|
|
return mCells.getInterior (name);
|
2011-08-01 12:34:50 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2014-01-16 11:03:23 +00:00
|
|
|
CellStore *World::getCell (const ESM::CellId& id)
|
|
|
|
{
|
|
|
|
if (id.mPaged)
|
|
|
|
return getExterior (id.mIndex.mX, id.mIndex.mY);
|
|
|
|
else
|
|
|
|
return getInterior (id.mWorldspace);
|
|
|
|
}
|
|
|
|
|
2013-12-16 13:40:47 +00:00
|
|
|
void World::useDeathCamera()
|
2013-12-15 16:50:25 +00:00
|
|
|
{
|
2013-12-16 13:40:47 +00:00
|
|
|
if(mRendering->getCamera()->isVanityOrPreviewModeEnabled() )
|
|
|
|
{
|
|
|
|
mRendering->getCamera()->togglePreviewMode(false);
|
|
|
|
mRendering->getCamera()->toggleVanityMode(false);
|
|
|
|
}
|
|
|
|
if(mRendering->getCamera()->isFirstPerson())
|
|
|
|
togglePOV();
|
2013-12-15 16:50:25 +00:00
|
|
|
}
|
|
|
|
|
2011-01-04 14:58:22 +00:00
|
|
|
MWWorld::Player& World::getPlayer()
|
2010-07-02 07:38:22 +00:00
|
|
|
{
|
2011-01-04 14:58:22 +00:00
|
|
|
return *mPlayer;
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2012-10-01 15:17:04 +00:00
|
|
|
const MWWorld::ESMStore& World::getStore() const
|
2010-07-02 14:18:25 +00:00
|
|
|
{
|
|
|
|
return mStore;
|
|
|
|
}
|
2011-08-09 07:56:09 +00:00
|
|
|
|
2012-11-10 20:43:41 +00:00
|
|
|
std::vector<ESM::ESMReader>& World::getEsmReader()
|
2011-07-31 15:07:11 +00:00
|
|
|
{
|
|
|
|
return mEsm;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2011-10-06 10:29:59 +00:00
|
|
|
LocalScripts& World::getLocalScripts()
|
2010-07-02 14:18:25 +00:00
|
|
|
{
|
|
|
|
return mLocalScripts;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-03 10:12:13 +00:00
|
|
|
bool World::hasCellChanged() const
|
|
|
|
{
|
2011-08-01 02:06:38 +00:00
|
|
|
return mWorldScene->hasCellChanged();
|
2010-07-03 10:12:13 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-11-28 08:10:38 +00:00
|
|
|
void World::setGlobalInt (const std::string& name, int value)
|
2010-07-04 11:33:33 +00:00
|
|
|
{
|
2013-11-28 08:13:54 +00:00
|
|
|
if (name=="gamehour")
|
|
|
|
setHour (value);
|
|
|
|
else if (name=="day")
|
|
|
|
setDay (value);
|
|
|
|
else if (name=="month")
|
|
|
|
setMonth (value);
|
|
|
|
else
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables[name].setInteger (value);
|
2010-07-04 11:33:33 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-11-28 08:10:38 +00:00
|
|
|
void World::setGlobalFloat (const std::string& name, float value)
|
2011-04-21 09:00:00 +00:00
|
|
|
{
|
2013-11-28 08:13:54 +00:00
|
|
|
if (name=="gamehour")
|
|
|
|
setHour (value);
|
|
|
|
else if (name=="day")
|
|
|
|
setDay (value);
|
|
|
|
else if (name=="month")
|
|
|
|
setMonth (value);
|
|
|
|
else
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables[name].setFloat (value);
|
2011-04-21 09:00:00 +00:00
|
|
|
}
|
|
|
|
|
2013-11-28 08:10:38 +00:00
|
|
|
int World::getGlobalInt (const std::string& name) const
|
2010-07-21 13:01:35 +00:00
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
return mGlobalVariables[name].getInteger();
|
2010-08-03 18:01:52 +00:00
|
|
|
}
|
|
|
|
|
2013-11-28 08:10:38 +00:00
|
|
|
float World::getGlobalFloat (const std::string& name) const
|
2012-12-20 23:16:34 +00:00
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
return mGlobalVariables[name].getFloat();
|
2012-12-20 23:16:34 +00:00
|
|
|
}
|
2013-02-05 11:19:06 +00:00
|
|
|
|
2010-07-21 13:01:35 +00:00
|
|
|
char World::getGlobalVariableType (const std::string& name) const
|
2012-12-21 18:09:31 +00:00
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
return mGlobalVariables.getType (name);
|
2010-08-03 18:01:52 +00:00
|
|
|
}
|
2012-12-21 18:09:31 +00:00
|
|
|
|
2013-11-26 10:39:58 +00:00
|
|
|
std::string World::getCellName (const MWWorld::CellStore *cell) const
|
2012-12-21 18:09:31 +00:00
|
|
|
{
|
2013-11-26 10:39:58 +00:00
|
|
|
if (!cell)
|
|
|
|
cell = mWorldScene->getCurrentCell();
|
2012-12-21 18:09:31 +00:00
|
|
|
|
2014-02-21 10:35:46 +00:00
|
|
|
if (!cell->getCell()->isExterior() || !cell->getCell()->mName.empty())
|
|
|
|
return cell->getCell()->mName;
|
2012-12-21 18:09:31 +00:00
|
|
|
|
2014-02-21 10:35:46 +00:00
|
|
|
if (const ESM::Region* region = getStore().get<ESM::Region>().search (cell->getCell()->mRegion))
|
2013-11-26 10:39:58 +00:00
|
|
|
return region->mName;
|
2013-02-05 11:19:06 +00:00
|
|
|
|
2013-11-26 11:47:30 +00:00
|
|
|
return getStore().get<ESM::GameSetting>().find ("sDefaultCellname")->mValue.getString();
|
2012-12-21 18:09:31 +00:00
|
|
|
}
|
2012-12-20 23:16:34 +00:00
|
|
|
|
2013-01-13 17:05:12 +00:00
|
|
|
void World::removeRefScript (MWWorld::RefData *ref)
|
|
|
|
{
|
|
|
|
mLocalScripts.remove (ref);
|
|
|
|
}
|
|
|
|
|
2014-01-11 02:33:17 +00:00
|
|
|
Ptr World::searchPtr (const std::string& name, bool activeOnly)
|
2010-07-05 10:09:04 +00:00
|
|
|
{
|
2014-01-11 02:33:17 +00:00
|
|
|
Ptr ret;
|
2010-07-05 10:09:04 +00:00
|
|
|
// the player is always in an active cell.
|
|
|
|
if (name=="player")
|
|
|
|
{
|
2011-01-04 14:58:22 +00:00
|
|
|
return mPlayer->getPlayer();
|
2010-07-05 10:09:04 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2014-01-01 01:22:11 +00:00
|
|
|
std::string lowerCaseName = Misc::StringUtils::lowerCase(name);
|
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
// active cells
|
2011-11-16 04:31:18 +00:00
|
|
|
for (Scene::CellStoreCollection::const_iterator iter (mWorldScene->getActiveCells().begin());
|
2011-08-01 02:06:38 +00:00
|
|
|
iter!=mWorldScene->getActiveCells().end(); ++iter)
|
2010-07-05 10:09:04 +00:00
|
|
|
{
|
2013-12-05 12:21:26 +00:00
|
|
|
CellStore* cellstore = *iter;
|
2014-01-01 01:22:11 +00:00
|
|
|
Ptr ptr = mCells.getPtr (lowerCaseName, *cellstore, true);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (!ptr.isEmpty())
|
2010-07-10 11:19:04 +00:00
|
|
|
return ptr;
|
2010-07-05 10:09:04 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2014-02-13 21:58:12 +00:00
|
|
|
Ptr ptr = Class::get (mPlayer->getPlayer()).
|
|
|
|
getContainerStore (mPlayer->getPlayer()).search (lowerCaseName);
|
|
|
|
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (!activeOnly)
|
|
|
|
{
|
2014-01-11 02:33:17 +00:00
|
|
|
ret = mCells.getPtr (lowerCaseName);
|
2010-07-05 10:09:04 +00:00
|
|
|
}
|
2014-01-11 02:33:17 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2014-01-11 02:33:17 +00:00
|
|
|
Ptr World::getPtr (const std::string& name, bool activeOnly)
|
|
|
|
{
|
|
|
|
Ptr ret = searchPtr(name, activeOnly);
|
|
|
|
if (!ret.isEmpty())
|
|
|
|
return ret;
|
2010-07-05 10:09:04 +00:00
|
|
|
throw std::runtime_error ("unknown ID: " + name);
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
|
|
|
Ptr World::getPtrViaHandle (const std::string& handle)
|
2012-10-19 17:48:02 +00:00
|
|
|
{
|
|
|
|
Ptr res = searchPtrViaHandle (handle);
|
|
|
|
if (res.isEmpty ())
|
|
|
|
throw std::runtime_error ("unknown Ogre handle: " + handle);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
Ptr World::searchPtrViaHandle (const std::string& handle)
|
2010-08-03 18:01:52 +00:00
|
|
|
{
|
2011-02-10 12:32:34 +00:00
|
|
|
if (mPlayer->getPlayer().getRefData().getHandle()==handle)
|
|
|
|
return mPlayer->getPlayer();
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2014-04-29 13:32:00 +00:00
|
|
|
return mWorldScene->searchPtrViaHandle (handle);
|
2010-08-03 18:01:52 +00:00
|
|
|
}
|
|
|
|
|
2014-04-29 13:27:49 +00:00
|
|
|
Ptr World::searchPtrViaActorId (int actorId)
|
|
|
|
{
|
2014-05-15 00:36:30 +00:00
|
|
|
// The player is not registered in any CellStore so must be checked manually
|
|
|
|
if (actorId == getPlayerPtr().getClass().getCreatureStats(getPlayerPtr()).getActorId())
|
|
|
|
return getPlayerPtr();
|
|
|
|
// Now search cells
|
2014-04-29 13:27:49 +00:00
|
|
|
return mWorldScene->searchPtrViaActorId (actorId);
|
2010-08-03 18:01:52 +00:00
|
|
|
}
|
|
|
|
|
2013-12-05 12:21:26 +00:00
|
|
|
void World::addContainerScripts(const Ptr& reference, CellStore * cell)
|
2013-01-20 16:38:56 +00:00
|
|
|
{
|
|
|
|
if( reference.getTypeName()==typeid (ESM::Container).name() ||
|
|
|
|
reference.getTypeName()==typeid (ESM::NPC).name() ||
|
|
|
|
reference.getTypeName()==typeid (ESM::Creature).name())
|
|
|
|
{
|
|
|
|
MWWorld::ContainerStore& container = MWWorld::Class::get(reference).getContainerStore(reference);
|
|
|
|
for(MWWorld::ContainerStoreIterator it = container.begin(); it != container.end(); ++it)
|
|
|
|
{
|
|
|
|
std::string script = MWWorld::Class::get(*it).getScript(*it);
|
|
|
|
if(script != "")
|
|
|
|
{
|
|
|
|
MWWorld::Ptr item = *it;
|
|
|
|
item.mCell = cell;
|
|
|
|
mLocalScripts.add (script, item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-02 23:06:38 +00:00
|
|
|
void World::enable (const Ptr& reference)
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
2014-02-13 19:24:27 +00:00
|
|
|
// enable is a no-op for items in containers
|
|
|
|
if (!reference.isInCell())
|
|
|
|
return;
|
|
|
|
|
2010-07-10 11:19:04 +00:00
|
|
|
if (!reference.getRefData().isEnabled())
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
2010-07-10 11:19:04 +00:00
|
|
|
reference.getRefData().enable();
|
2013-02-05 11:19:06 +00:00
|
|
|
|
2012-05-25 15:28:27 +00:00
|
|
|
if(mWorldScene->getActiveCells().find (reference.getCell()) != mWorldScene->getActiveCells().end() && reference.getRefData().getCount())
|
|
|
|
mWorldScene->addObjectToScene (reference);
|
2010-07-09 14:07:03 +00:00
|
|
|
}
|
|
|
|
}
|
2013-02-05 11:19:06 +00:00
|
|
|
|
2013-01-20 16:38:56 +00:00
|
|
|
void World::removeContainerScripts(const Ptr& reference)
|
|
|
|
{
|
|
|
|
if( reference.getTypeName()==typeid (ESM::Container).name() ||
|
|
|
|
reference.getTypeName()==typeid (ESM::NPC).name() ||
|
|
|
|
reference.getTypeName()==typeid (ESM::Creature).name())
|
|
|
|
{
|
|
|
|
MWWorld::ContainerStore& container = MWWorld::Class::get(reference).getContainerStore(reference);
|
|
|
|
for(MWWorld::ContainerStoreIterator it = container.begin(); it != container.end(); ++it)
|
2013-01-20 15:43:52 +00:00
|
|
|
{
|
2013-01-20 16:38:56 +00:00
|
|
|
std::string script = MWWorld::Class::get(*it).getScript(*it);
|
|
|
|
if(script != "")
|
2013-01-20 15:43:52 +00:00
|
|
|
{
|
2013-01-20 16:38:56 +00:00
|
|
|
MWWorld::Ptr item = *it;
|
|
|
|
mLocalScripts.remove (item);
|
2013-01-20 15:43:52 +00:00
|
|
|
}
|
|
|
|
}
|
2010-07-09 14:07:03 +00:00
|
|
|
}
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2012-07-02 23:06:38 +00:00
|
|
|
void World::disable (const Ptr& reference)
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
2014-02-13 19:24:27 +00:00
|
|
|
// disable is a no-op for items in containers
|
|
|
|
if (!reference.isInCell())
|
|
|
|
return;
|
|
|
|
|
2010-08-07 18:29:10 +00:00
|
|
|
if (reference.getRefData().isEnabled())
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
2010-08-07 18:29:10 +00:00
|
|
|
reference.getRefData().disable();
|
2013-02-05 11:19:06 +00:00
|
|
|
|
2012-05-25 15:28:27 +00:00
|
|
|
if(mWorldScene->getActiveCells().find (reference.getCell())!=mWorldScene->getActiveCells().end() && reference.getRefData().getCount())
|
|
|
|
mWorldScene->removeObjectFromScene (reference);
|
2010-08-03 18:01:52 +00:00
|
|
|
}
|
2010-07-09 14:07:03 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
void World::advanceTime (double hours)
|
|
|
|
{
|
2013-11-19 15:42:24 +00:00
|
|
|
MWBase::Environment::get().getMechanicsManager()->advanceTime(hours*3600);
|
|
|
|
|
2012-09-25 19:28:25 +00:00
|
|
|
mWeatherManager->advanceTime (hours);
|
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
hours += mGlobalVariables["gamehour"].getFloat();
|
2012-01-23 13:33:06 +00:00
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
setHour (hours);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
int days = hours / 24;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
if (days>0)
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables["dayspassed"].setInteger (
|
|
|
|
days + mGlobalVariables["dayspassed"].getInteger());
|
2010-07-18 16:29:16 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
void World::setHour (double hour)
|
|
|
|
{
|
|
|
|
if (hour<0)
|
|
|
|
hour = 0;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
int days = hour / 24;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
hour = std::fmod (hour, 24);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables["gamehour"].setFloat (hour);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2012-01-29 18:51:20 +00:00
|
|
|
mRendering->skySetHour (hour);
|
2012-03-15 15:01:41 +00:00
|
|
|
|
2012-02-22 19:12:08 +00:00
|
|
|
mWeatherManager->setHour (hour);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
if (days>0)
|
2013-12-10 14:09:58 +00:00
|
|
|
setDay (days + mGlobalVariables["day"].getInteger());
|
2010-07-18 16:29:16 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
void World::setDay (int day)
|
|
|
|
{
|
2012-11-05 17:01:50 +00:00
|
|
|
if (day<1)
|
|
|
|
day = 1;
|
2010-07-22 09:48:27 +00:00
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
int month = mGlobalVariables["month"].getInteger();
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
while (true)
|
2010-07-18 16:29:16 +00:00
|
|
|
{
|
2010-08-03 18:01:52 +00:00
|
|
|
int days = getDaysPerMonth (month);
|
2012-11-05 17:01:50 +00:00
|
|
|
if (day<=days)
|
2010-07-22 09:48:27 +00:00
|
|
|
break;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
if (month<11)
|
|
|
|
{
|
|
|
|
++month;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
month = 0;
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables["year"].setInteger (mGlobalVariables["year"].getInteger()+1);
|
2010-07-22 09:48:27 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
day -= days;
|
2010-08-03 18:01:52 +00:00
|
|
|
}
|
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables["day"].setInteger (day);
|
|
|
|
mGlobalVariables["month"].setInteger (month);
|
2010-07-22 09:48:27 +00:00
|
|
|
|
2012-01-29 18:51:20 +00:00
|
|
|
mRendering->skySetDate (day, month);
|
2010-08-03 18:01:52 +00:00
|
|
|
}
|
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
void World::setMonth (int month)
|
|
|
|
{
|
|
|
|
if (month<0)
|
|
|
|
month = 0;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
int years = month / 12;
|
|
|
|
month = month % 12;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
int days = getDaysPerMonth (month);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
if (mGlobalVariables["day"].getInteger()>days)
|
|
|
|
mGlobalVariables["day"].setInteger (days);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables["month"].setInteger (month);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
if (years>0)
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables["year"].setInteger (years+mGlobalVariables["year"].getInteger());
|
2010-07-22 09:48:27 +00:00
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
mRendering->skySetDate (mGlobalVariables["day"].getInteger(), month);
|
2010-07-22 09:48:27 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-11-26 11:47:30 +00:00
|
|
|
int World::getDay() const
|
2012-09-18 18:53:32 +00:00
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
return mGlobalVariables["day"].getInteger();
|
2012-09-18 18:53:32 +00:00
|
|
|
}
|
2010-07-22 09:48:27 +00:00
|
|
|
|
2013-11-26 11:47:30 +00:00
|
|
|
int World::getMonth() const
|
2012-09-18 18:53:32 +00:00
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
return mGlobalVariables["month"].getInteger();
|
2010-07-22 09:48:27 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2013-11-28 08:27:10 +00:00
|
|
|
int World::getYear() const
|
2012-09-18 18:53:32 +00:00
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
return mGlobalVariables["year"].getInteger();
|
2012-09-18 18:53:32 +00:00
|
|
|
}
|
|
|
|
|
2013-11-26 11:47:30 +00:00
|
|
|
std::string World::getMonthName (int month) const
|
2012-09-18 18:53:32 +00:00
|
|
|
{
|
2013-11-26 11:47:30 +00:00
|
|
|
if (month==-1)
|
|
|
|
month = getMonth();
|
|
|
|
|
|
|
|
const int months = 12;
|
|
|
|
|
|
|
|
if (month<0 || month>=months)
|
|
|
|
return "";
|
|
|
|
|
|
|
|
static const char *monthNames[months] =
|
|
|
|
{
|
|
|
|
"sMonthMorningstar", "sMonthSunsdawn", "sMonthFirstseed", "sMonthRainshand",
|
|
|
|
"sMonthSecondseed", "sMonthMidyear", "sMonthSunsheight", "sMonthLastseed",
|
|
|
|
"sMonthHeartfire", "sMonthFrostfall", "sMonthSunsdusk", "sMonthEveningstar"
|
|
|
|
};
|
|
|
|
|
|
|
|
return getStore().get<ESM::GameSetting>().find (monthNames[month])->mValue.getString();
|
2012-09-18 18:53:32 +00:00
|
|
|
}
|
|
|
|
|
2012-05-19 07:31:45 +00:00
|
|
|
TimeStamp World::getTimeStamp() const
|
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
return TimeStamp (mGlobalVariables["gamehour"].getFloat(),
|
|
|
|
mGlobalVariables["dayspassed"].getInteger());
|
2012-05-19 07:31:45 +00:00
|
|
|
}
|
|
|
|
|
2011-04-26 19:38:21 +00:00
|
|
|
bool World::toggleSky()
|
2010-07-18 17:48:02 +00:00
|
|
|
{
|
|
|
|
if (mSky)
|
|
|
|
{
|
|
|
|
mSky = false;
|
2012-01-29 18:51:20 +00:00
|
|
|
mRendering->skyDisable();
|
2011-04-26 19:38:21 +00:00
|
|
|
return false;
|
2010-07-18 17:48:02 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mSky = true;
|
2012-01-29 18:51:20 +00:00
|
|
|
mRendering->skyEnable();
|
2011-04-26 19:38:21 +00:00
|
|
|
return true;
|
2010-07-18 17:48:02 +00:00
|
|
|
}
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
int World::getMasserPhase() const
|
|
|
|
{
|
2012-01-29 18:51:20 +00:00
|
|
|
return mRendering->skyGetMasserPhase();
|
2010-07-18 17:48:02 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
int World::getSecundaPhase() const
|
|
|
|
{
|
2012-01-29 18:51:20 +00:00
|
|
|
return mRendering->skyGetSecundaPhase();
|
2010-07-18 17:48:02 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
void World::setMoonColour (bool red)
|
|
|
|
{
|
2012-01-29 18:51:20 +00:00
|
|
|
mRendering->skySetMoonColour (red);
|
2010-07-18 17:48:02 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-21 12:10:52 +00:00
|
|
|
float World::getTimeScaleFactor() const
|
|
|
|
{
|
2013-12-10 14:09:58 +00:00
|
|
|
return mGlobalVariables["timescale"].getFloat();
|
2010-07-21 12:10:52 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2011-02-10 09:38:45 +00:00
|
|
|
void World::changeToInteriorCell (const std::string& cellName, const ESM::Position& position)
|
2010-07-22 10:29:23 +00:00
|
|
|
{
|
2014-01-08 17:39:44 +00:00
|
|
|
removeContainerScripts(getPlayerPtr());
|
2013-11-22 01:12:37 +00:00
|
|
|
mWorldScene->changeToInteriorCell(cellName, position);
|
2014-01-08 17:39:44 +00:00
|
|
|
addContainerScripts(getPlayerPtr(), getPlayerPtr().getCell());
|
2010-07-22 10:29:23 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2011-02-10 09:38:45 +00:00
|
|
|
void World::changeToExteriorCell (const ESM::Position& position)
|
2010-08-20 11:33:03 +00:00
|
|
|
{
|
2014-01-08 17:39:44 +00:00
|
|
|
removeContainerScripts(getPlayerPtr());
|
2013-11-22 01:12:37 +00:00
|
|
|
mWorldScene->changeToExteriorCell(position);
|
2014-01-08 17:39:44 +00:00
|
|
|
addContainerScripts(getPlayerPtr(), getPlayerPtr().getCell());
|
2010-08-20 11:33:03 +00:00
|
|
|
}
|
|
|
|
|
2014-01-16 11:03:23 +00:00
|
|
|
void World::changeToCell (const ESM::CellId& cellId, const ESM::Position& position)
|
|
|
|
{
|
|
|
|
if (cellId.mPaged)
|
|
|
|
changeToExteriorCell (position);
|
|
|
|
else
|
|
|
|
changeToInteriorCell (cellId.mWorldspace, position);
|
|
|
|
}
|
|
|
|
|
2010-07-22 10:29:23 +00:00
|
|
|
void World::markCellAsUnchanged()
|
|
|
|
{
|
2011-08-01 02:06:38 +00:00
|
|
|
return mWorldScene->markCellAsUnchanged();
|
2010-07-22 10:29:23 +00:00
|
|
|
}
|
2010-08-05 11:29:49 +00:00
|
|
|
|
2013-01-08 07:40:17 +00:00
|
|
|
float World::getMaxActivationDistance ()
|
2010-08-05 11:29:49 +00:00
|
|
|
{
|
2013-01-09 03:52:18 +00:00
|
|
|
if (mActivationDistanceOverride >= 0)
|
|
|
|
return mActivationDistanceOverride;
|
|
|
|
|
2013-01-08 07:40:17 +00:00
|
|
|
return (std::max) (getNpcActivationDistance (), getObjectActivationDistance ());
|
|
|
|
}
|
|
|
|
|
|
|
|
float World::getNpcActivationDistance ()
|
|
|
|
{
|
2013-01-09 03:52:18 +00:00
|
|
|
if (mActivationDistanceOverride >= 0)
|
|
|
|
return mActivationDistanceOverride;
|
|
|
|
|
2013-01-08 07:40:17 +00:00
|
|
|
return getStore().get<ESM::GameSetting>().find ("iMaxActivateDist")->getInt()*5/4;
|
|
|
|
}
|
|
|
|
|
|
|
|
float World::getObjectActivationDistance ()
|
|
|
|
{
|
2013-01-09 03:52:18 +00:00
|
|
|
if (mActivationDistanceOverride >= 0)
|
|
|
|
return mActivationDistanceOverride;
|
|
|
|
|
2013-01-08 07:40:17 +00:00
|
|
|
return getStore().get<ESM::GameSetting>().find ("iMaxActivateDist")->getInt();
|
|
|
|
}
|
|
|
|
|
2013-01-08 07:27:37 +00:00
|
|
|
MWWorld::Ptr World::getFacedObject()
|
2010-08-05 11:29:49 +00:00
|
|
|
{
|
2014-01-01 22:30:58 +00:00
|
|
|
if (mFacedHandle.empty())
|
|
|
|
return MWWorld::Ptr();
|
|
|
|
return searchPtrViaHandle(mFacedHandle);
|
2010-08-05 11:29:49 +00:00
|
|
|
}
|
2010-08-07 18:25:17 +00:00
|
|
|
|
2013-08-23 14:01:30 +00:00
|
|
|
std::pair<MWWorld::Ptr,Ogre::Vector3> World::getHitContact(const MWWorld::Ptr &ptr, float distance)
|
2013-07-25 19:58:43 +00:00
|
|
|
{
|
|
|
|
const ESM::Position &posdata = ptr.getRefData().getPosition();
|
|
|
|
Ogre::Vector3 pos(posdata.pos);
|
|
|
|
Ogre::Quaternion rot = Ogre::Quaternion(Ogre::Radian(posdata.rot[2]), Ogre::Vector3::NEGATIVE_UNIT_Z) *
|
2014-03-07 05:11:00 +00:00
|
|
|
Ogre::Quaternion(Ogre::Radian(posdata.rot[0]), Ogre::Vector3::NEGATIVE_UNIT_X);
|
2013-07-25 19:58:43 +00:00
|
|
|
|
|
|
|
MWRender::Animation *anim = mRendering->getAnimation(ptr);
|
|
|
|
if(anim != NULL)
|
|
|
|
{
|
|
|
|
Ogre::Node *node = anim->getNode("Head");
|
2014-01-19 13:02:39 +00:00
|
|
|
if (node == NULL)
|
|
|
|
node = anim->getNode("Bip01 Head");
|
2013-07-25 19:58:43 +00:00
|
|
|
if(node != NULL)
|
|
|
|
pos += node->_getDerivedPosition();
|
|
|
|
}
|
|
|
|
|
2013-08-23 14:01:30 +00:00
|
|
|
std::pair<std::string,Ogre::Vector3> result = mPhysics->getHitContact(ptr.getRefData().getHandle(),
|
|
|
|
pos, rot, distance);
|
2013-07-25 19:58:43 +00:00
|
|
|
if(result.first.empty())
|
2013-08-23 14:01:30 +00:00
|
|
|
return std::make_pair(MWWorld::Ptr(), Ogre::Vector3(0.0f));
|
2013-07-25 19:58:43 +00:00
|
|
|
|
2013-08-23 14:01:30 +00:00
|
|
|
return std::make_pair(searchPtrViaHandle(result.first), result.second);
|
2013-07-25 19:58:43 +00:00
|
|
|
}
|
|
|
|
|
2012-07-02 23:06:38 +00:00
|
|
|
void World::deleteObject (const Ptr& ptr)
|
2010-08-07 18:25:17 +00:00
|
|
|
{
|
2013-10-16 16:39:29 +00:00
|
|
|
if (ptr.getRefData().getCount() > 0)
|
2010-08-07 18:25:17 +00:00
|
|
|
{
|
2013-10-16 16:39:29 +00:00
|
|
|
ptr.getRefData().setCount(0);
|
2010-08-07 18:25:17 +00:00
|
|
|
|
2013-10-16 16:39:29 +00:00
|
|
|
if (ptr.isInCell()
|
|
|
|
&& mWorldScene->getActiveCells().find(ptr.getCell()) != mWorldScene->getActiveCells().end()
|
|
|
|
&& ptr.getRefData().isEnabled())
|
2012-05-24 11:26:07 +00:00
|
|
|
{
|
2012-05-25 15:28:27 +00:00
|
|
|
mWorldScene->removeObjectFromScene (ptr);
|
2012-05-24 11:26:07 +00:00
|
|
|
mLocalScripts.remove (ptr);
|
2013-01-20 17:01:30 +00:00
|
|
|
removeContainerScripts (ptr);
|
2012-05-24 11:26:07 +00:00
|
|
|
}
|
2010-08-07 18:25:17 +00:00
|
|
|
}
|
|
|
|
}
|
2010-08-21 10:31:04 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
void World::moveObject(const Ptr &ptr, CellStore* newCell, float x, float y, float z)
|
2010-08-21 10:31:04 +00:00
|
|
|
{
|
2012-07-30 19:28:14 +00:00
|
|
|
ESM::Position &pos = ptr.getRefData().getPosition();
|
2013-04-04 14:51:22 +00:00
|
|
|
|
2013-02-05 11:19:06 +00:00
|
|
|
pos.pos[0] = x;
|
|
|
|
pos.pos[1] = y;
|
|
|
|
pos.pos[2] = z;
|
2013-04-03 21:55:57 +00:00
|
|
|
|
2012-07-30 19:28:14 +00:00
|
|
|
Ogre::Vector3 vec(x, y, z);
|
|
|
|
|
2012-08-08 10:51:33 +00:00
|
|
|
CellStore *currCell = ptr.getCell();
|
|
|
|
bool isPlayer = ptr == mPlayer->getPlayer();
|
2013-05-15 15:54:18 +00:00
|
|
|
bool haveToMove = isPlayer || mWorldScene->isCellActive(*currCell);
|
2012-11-07 18:34:28 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
if (currCell != newCell)
|
2012-11-05 18:40:02 +00:00
|
|
|
{
|
2013-04-04 14:51:22 +00:00
|
|
|
removeContainerScripts(ptr);
|
2013-01-31 00:21:04 +00:00
|
|
|
|
2012-11-05 18:40:02 +00:00
|
|
|
if (isPlayer)
|
2013-01-29 07:39:11 +00:00
|
|
|
{
|
2014-03-07 05:11:00 +00:00
|
|
|
if (!newCell->isExterior())
|
|
|
|
changeToInteriorCell(Misc::StringUtils::lowerCase(newCell->getCell()->mName), pos);
|
2012-11-05 18:40:02 +00:00
|
|
|
else
|
|
|
|
{
|
2014-03-07 05:11:00 +00:00
|
|
|
int cellX = newCell->getCell()->getGridX();
|
|
|
|
int cellY = newCell->getCell()->getGridY();
|
2012-08-11 13:01:55 +00:00
|
|
|
mWorldScene->changeCell(cellX, cellY, pos, false);
|
2012-08-08 10:51:33 +00:00
|
|
|
}
|
2014-03-07 05:11:00 +00:00
|
|
|
addContainerScripts (getPlayerPtr(), newCell);
|
2013-01-29 07:39:11 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-11-05 18:40:02 +00:00
|
|
|
if (!mWorldScene->isCellActive(*currCell))
|
2014-03-07 05:11:00 +00:00
|
|
|
ptr.getClass().copyToCell(ptr, *newCell, pos);
|
|
|
|
else if (!mWorldScene->isCellActive(*newCell))
|
2012-11-05 18:40:02 +00:00
|
|
|
{
|
2012-08-08 10:51:33 +00:00
|
|
|
mWorldScene->removeObjectFromScene(ptr);
|
|
|
|
mLocalScripts.remove(ptr);
|
2013-02-05 11:19:06 +00:00
|
|
|
removeContainerScripts (ptr);
|
2012-08-08 10:51:33 +00:00
|
|
|
haveToMove = false;
|
2013-11-28 16:31:17 +00:00
|
|
|
|
|
|
|
MWWorld::Ptr newPtr = MWWorld::Class::get(ptr)
|
2014-03-07 05:11:00 +00:00
|
|
|
.copyToCell(ptr, *newCell);
|
2013-11-28 16:31:17 +00:00
|
|
|
newPtr.getRefData().setBaseNode(0);
|
|
|
|
|
|
|
|
objectLeftActiveCell(ptr, newPtr);
|
2012-11-05 18:40:02 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-08-08 10:51:33 +00:00
|
|
|
MWWorld::Ptr copy =
|
2014-03-07 05:11:00 +00:00
|
|
|
MWWorld::Class::get(ptr).copyToCell(ptr, *newCell, pos);
|
2012-08-08 10:51:33 +00:00
|
|
|
|
2013-02-24 10:59:21 +00:00
|
|
|
mRendering->updateObjectCell(ptr, copy);
|
2014-01-20 11:05:13 +00:00
|
|
|
MWBase::Environment::get().getSoundManager()->updatePtr (ptr, copy);
|
2013-02-24 10:59:21 +00:00
|
|
|
|
2013-01-29 08:19:24 +00:00
|
|
|
MWBase::MechanicsManager *mechMgr = MWBase::Environment::get().getMechanicsManager();
|
2013-02-25 17:57:34 +00:00
|
|
|
mechMgr->updateCell(ptr, copy);
|
2012-08-08 10:51:33 +00:00
|
|
|
|
2013-02-05 11:19:06 +00:00
|
|
|
std::string script =
|
|
|
|
MWWorld::Class::get(ptr).getScript(ptr);
|
|
|
|
if (!script.empty())
|
2012-11-05 18:40:02 +00:00
|
|
|
{
|
2013-01-29 08:19:24 +00:00
|
|
|
mLocalScripts.remove(ptr);
|
2013-02-05 11:19:06 +00:00
|
|
|
removeContainerScripts (ptr);
|
2013-01-29 08:19:24 +00:00
|
|
|
mLocalScripts.add(script, copy);
|
2014-03-07 05:11:00 +00:00
|
|
|
addContainerScripts (copy, newCell);
|
2010-08-21 10:41:59 +00:00
|
|
|
}
|
|
|
|
}
|
2012-08-08 10:51:33 +00:00
|
|
|
ptr.getRefData().setCount(0);
|
2010-08-21 10:41:59 +00:00
|
|
|
}
|
|
|
|
}
|
2014-01-02 01:27:48 +00:00
|
|
|
if (haveToMove && ptr.getRefData().getBaseNode())
|
2012-11-05 18:40:02 +00:00
|
|
|
{
|
2012-07-30 19:28:14 +00:00
|
|
|
mRendering->moveObject(ptr, vec);
|
2012-11-05 18:40:02 +00:00
|
|
|
mPhysics->moveObject (ptr);
|
2012-07-30 19:28:14 +00:00
|
|
|
}
|
2012-08-08 10:51:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool World::moveObjectImp(const Ptr& ptr, float x, float y, float z)
|
|
|
|
{
|
|
|
|
CellStore *cell = ptr.getCell();
|
2013-04-04 14:51:22 +00:00
|
|
|
|
2012-08-08 10:51:33 +00:00
|
|
|
if (cell->isExterior()) {
|
|
|
|
int cellX, cellY;
|
|
|
|
positionToIndex(x, y, cellX, cellY);
|
2012-08-11 15:30:55 +00:00
|
|
|
|
2012-08-08 10:51:33 +00:00
|
|
|
cell = getExterior(cellX, cellY);
|
|
|
|
}
|
2013-04-04 14:51:22 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
moveObject(ptr, cell, x, y, z);
|
2012-08-11 15:30:55 +00:00
|
|
|
|
2012-08-08 10:51:33 +00:00
|
|
|
return cell != ptr.getCell();
|
2011-08-02 16:50:31 +00:00
|
|
|
}
|
|
|
|
|
2012-07-02 23:06:38 +00:00
|
|
|
void World::moveObject (const Ptr& ptr, float x, float y, float z)
|
2011-08-02 16:50:31 +00:00
|
|
|
{
|
|
|
|
moveObjectImp(ptr, x, y, z);
|
2010-08-21 10:31:04 +00:00
|
|
|
}
|
2010-08-22 19:30:48 +00:00
|
|
|
|
2012-07-09 16:47:59 +00:00
|
|
|
void World::scaleObject (const Ptr& ptr, float scale)
|
2012-05-25 16:23:06 +00:00
|
|
|
{
|
2012-09-17 07:37:50 +00:00
|
|
|
ptr.getCellRef().mScale = scale;
|
2013-04-12 22:13:56 +00:00
|
|
|
MWWorld::Class::get(ptr).adjustScale(ptr,scale);
|
2012-12-04 09:58:43 +00:00
|
|
|
|
2012-11-05 19:48:07 +00:00
|
|
|
if(ptr.getRefData().getBaseNode() == 0)
|
|
|
|
return;
|
|
|
|
mRendering->scaleObject(ptr, Vector3(scale,scale,scale));
|
2012-11-05 18:40:02 +00:00
|
|
|
mPhysics->scaleObject(ptr);
|
2012-05-25 16:23:06 +00:00
|
|
|
}
|
|
|
|
|
2013-03-31 09:24:44 +00:00
|
|
|
void World::rotateObjectImp (const Ptr& ptr, Ogre::Vector3 rot, bool adjust)
|
2012-05-29 08:15:29 +00:00
|
|
|
{
|
2013-04-28 08:14:58 +00:00
|
|
|
const float two_pi = Ogre::Math::TWO_PI;
|
|
|
|
const float pi = Ogre::Math::PI;
|
|
|
|
|
|
|
|
float *objRot = ptr.getRefData().getPosition().rot;
|
|
|
|
if(adjust)
|
|
|
|
{
|
|
|
|
objRot[0] += rot.x;
|
|
|
|
objRot[1] += rot.y;
|
|
|
|
objRot[2] += rot.z;
|
|
|
|
}
|
|
|
|
else
|
2012-11-05 19:48:07 +00:00
|
|
|
{
|
2013-03-31 00:31:16 +00:00
|
|
|
objRot[0] = rot.x;
|
|
|
|
objRot[1] = rot.y;
|
|
|
|
objRot[2] = rot.z;
|
2013-04-28 08:14:58 +00:00
|
|
|
}
|
2012-09-01 21:28:12 +00:00
|
|
|
|
2013-04-28 08:14:58 +00:00
|
|
|
if(Class::get(ptr).isActor())
|
|
|
|
{
|
|
|
|
/* HACK? Actors shouldn't really be rotating around X (or Y), but
|
|
|
|
* currently it's done so for rotating the camera, which needs
|
|
|
|
* clamping.
|
|
|
|
*/
|
|
|
|
const float half_pi = Ogre::Math::HALF_PI;
|
|
|
|
|
|
|
|
if(objRot[0] < -half_pi) objRot[0] = -half_pi;
|
|
|
|
else if(objRot[0] > half_pi) objRot[0] = half_pi;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while(objRot[0] < -pi) objRot[0] += two_pi;
|
|
|
|
while(objRot[0] > pi) objRot[0] -= two_pi;
|
|
|
|
}
|
2013-04-25 18:52:38 +00:00
|
|
|
|
2013-04-28 08:14:58 +00:00
|
|
|
while(objRot[1] < -pi) objRot[1] += two_pi;
|
|
|
|
while(objRot[1] > pi) objRot[1] -= two_pi;
|
|
|
|
|
|
|
|
while(objRot[2] < -pi) objRot[2] += two_pi;
|
|
|
|
while(objRot[2] > pi) objRot[2] -= two_pi;
|
|
|
|
|
|
|
|
if(ptr.getRefData().getBaseNode() != 0)
|
|
|
|
{
|
|
|
|
mRendering->rotateObject(ptr);
|
|
|
|
mPhysics->rotateObject(ptr);
|
2013-02-25 18:00:50 +00:00
|
|
|
}
|
2012-05-29 08:15:29 +00:00
|
|
|
}
|
|
|
|
|
2013-04-25 17:14:10 +00:00
|
|
|
void World::localRotateObject (const Ptr& ptr, float x, float y, float z)
|
2013-04-15 14:45:53 +00:00
|
|
|
{
|
2013-04-16 19:17:19 +00:00
|
|
|
if (ptr.getRefData().getBaseNode() != 0) {
|
2013-04-24 19:42:04 +00:00
|
|
|
|
2013-04-25 17:14:10 +00:00
|
|
|
ptr.getRefData().getLocalRotation().rot[0]=Ogre::Degree(x).valueRadians();
|
|
|
|
ptr.getRefData().getLocalRotation().rot[1]=Ogre::Degree(y).valueRadians();
|
|
|
|
ptr.getRefData().getLocalRotation().rot[2]=Ogre::Degree(z).valueRadians();
|
2013-04-24 19:42:04 +00:00
|
|
|
|
2013-04-25 18:52:38 +00:00
|
|
|
float fullRotateRad=Ogre::Degree(360).valueRadians();
|
|
|
|
|
|
|
|
while(ptr.getRefData().getLocalRotation().rot[0]>=fullRotateRad)
|
|
|
|
ptr.getRefData().getLocalRotation().rot[0]-=fullRotateRad;
|
|
|
|
while(ptr.getRefData().getLocalRotation().rot[1]>=fullRotateRad)
|
|
|
|
ptr.getRefData().getLocalRotation().rot[1]-=fullRotateRad;
|
|
|
|
while(ptr.getRefData().getLocalRotation().rot[2]>=fullRotateRad)
|
|
|
|
ptr.getRefData().getLocalRotation().rot[2]-=fullRotateRad;
|
|
|
|
|
|
|
|
while(ptr.getRefData().getLocalRotation().rot[0]<=-fullRotateRad)
|
|
|
|
ptr.getRefData().getLocalRotation().rot[0]+=fullRotateRad;
|
|
|
|
while(ptr.getRefData().getLocalRotation().rot[1]<=-fullRotateRad)
|
|
|
|
ptr.getRefData().getLocalRotation().rot[1]+=fullRotateRad;
|
|
|
|
while(ptr.getRefData().getLocalRotation().rot[2]<=-fullRotateRad)
|
|
|
|
ptr.getRefData().getLocalRotation().rot[2]+=fullRotateRad;
|
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
Ogre::Quaternion worldRotQuat(Ogre::Quaternion(Ogre::Radian(ptr.getRefData().getPosition().rot[0]), Ogre::Vector3::NEGATIVE_UNIT_X)*
|
|
|
|
Ogre::Quaternion(Ogre::Radian(ptr.getRefData().getPosition().rot[1]), Ogre::Vector3::NEGATIVE_UNIT_Y)*
|
|
|
|
Ogre::Quaternion(Ogre::Radian(ptr.getRefData().getPosition().rot[2]), Ogre::Vector3::NEGATIVE_UNIT_Z));
|
2013-04-25 17:14:10 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
Ogre::Quaternion rot(Ogre::Quaternion(Ogre::Degree(x), Ogre::Vector3::NEGATIVE_UNIT_X)*
|
|
|
|
Ogre::Quaternion(Ogre::Degree(y), Ogre::Vector3::NEGATIVE_UNIT_Y)*
|
|
|
|
Ogre::Quaternion(Ogre::Degree(z), Ogre::Vector3::NEGATIVE_UNIT_Z));
|
2013-04-25 17:14:10 +00:00
|
|
|
|
|
|
|
ptr.getRefData().getBaseNode()->setOrientation(worldRotQuat*rot);
|
2013-04-16 19:17:19 +00:00
|
|
|
mPhysics->rotateObject(ptr);
|
|
|
|
}
|
2013-04-15 14:45:53 +00:00
|
|
|
}
|
|
|
|
|
2013-04-03 21:55:57 +00:00
|
|
|
void World::adjustPosition(const Ptr &ptr)
|
|
|
|
{
|
2014-01-01 21:46:10 +00:00
|
|
|
Ogre::Vector3 pos (ptr.getRefData().getPosition().pos);
|
2013-04-03 21:55:57 +00:00
|
|
|
|
2013-04-04 14:51:22 +00:00
|
|
|
if(!ptr.getRefData().getBaseNode())
|
|
|
|
{
|
|
|
|
// will be adjusted when Ptr's cell becomes active
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-08-20 10:08:46 +00:00
|
|
|
float terrainHeight = mRendering->getTerrainHeightAt(pos);
|
2013-04-04 14:51:22 +00:00
|
|
|
|
2013-08-20 10:08:46 +00:00
|
|
|
if (pos.z < terrainHeight)
|
|
|
|
pos.z = terrainHeight;
|
2013-04-04 14:51:22 +00:00
|
|
|
|
2013-05-15 15:54:18 +00:00
|
|
|
ptr.getRefData().getPosition().pos[2] = pos.z + 20; // place slightly above. will snap down to ground with code below
|
2013-04-04 14:51:22 +00:00
|
|
|
|
2013-04-03 21:55:57 +00:00
|
|
|
if (!isFlying(ptr))
|
|
|
|
{
|
|
|
|
Ogre::Vector3 traced = mPhysics->traceDown(ptr);
|
|
|
|
if (traced.z < pos.z)
|
|
|
|
pos.z = traced.z;
|
|
|
|
}
|
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
moveObject(ptr, ptr.getCell(), pos.x, pos.y, pos.z);
|
2013-04-03 21:55:57 +00:00
|
|
|
}
|
|
|
|
|
2013-03-31 09:24:44 +00:00
|
|
|
void World::rotateObject (const Ptr& ptr,float x,float y,float z, bool adjust)
|
|
|
|
{
|
|
|
|
rotateObjectImp(ptr, Ogre::Vector3(Ogre::Degree(x).valueRadians(),
|
|
|
|
Ogre::Degree(y).valueRadians(),
|
|
|
|
Ogre::Degree(z).valueRadians()),
|
|
|
|
adjust);
|
|
|
|
}
|
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
MWWorld::Ptr World::safePlaceObject(const MWWorld::Ptr& ptr, MWWorld::CellStore* cell, ESM::Position pos)
|
2012-09-17 11:36:48 +00:00
|
|
|
{
|
2014-03-07 05:11:00 +00:00
|
|
|
return copyObjectToCell(ptr,cell,pos,false);
|
2012-09-17 11:36:48 +00:00
|
|
|
}
|
|
|
|
|
2010-09-11 10:21:55 +00:00
|
|
|
void World::indexToPosition (int cellX, int cellY, float &x, float &y, bool centre) const
|
2010-08-22 19:30:48 +00:00
|
|
|
{
|
|
|
|
const int cellSize = 8192;
|
|
|
|
|
|
|
|
x = cellSize * cellX;
|
|
|
|
y = cellSize * cellY;
|
2010-09-11 10:21:55 +00:00
|
|
|
|
|
|
|
if (centre)
|
|
|
|
{
|
|
|
|
x += cellSize/2;
|
|
|
|
y += cellSize/2;
|
|
|
|
}
|
2010-08-22 19:30:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void World::positionToIndex (float x, float y, int &cellX, int &cellY) const
|
|
|
|
{
|
|
|
|
const int cellSize = 8192;
|
|
|
|
|
2013-08-19 18:30:22 +00:00
|
|
|
cellX = std::floor(x/cellSize);
|
|
|
|
cellY = std::floor(y/cellSize);
|
2010-08-22 19:30:48 +00:00
|
|
|
}
|
2011-01-29 16:39:34 +00:00
|
|
|
|
2013-08-18 05:34:38 +00:00
|
|
|
void World::queueMovement(const Ptr &ptr, const Vector3 &velocity)
|
|
|
|
{
|
|
|
|
mPhysics->queueObjectMovement(ptr, velocity);
|
|
|
|
}
|
|
|
|
|
|
|
|
void World::doPhysics(float duration)
|
2011-01-29 16:39:34 +00:00
|
|
|
{
|
2013-04-30 18:26:59 +00:00
|
|
|
processDoors(duration);
|
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
moveMagicBolts(duration);
|
2013-11-13 13:02:15 +00:00
|
|
|
moveProjectiles(duration);
|
|
|
|
|
2013-08-18 05:34:38 +00:00
|
|
|
const PtrVelocityList &results = mPhysics->applyQueuedMovement(duration);
|
|
|
|
PtrVelocityList::const_iterator player(results.end());
|
|
|
|
for(PtrVelocityList::const_iterator iter(results.begin());iter != results.end();iter++)
|
2013-02-05 20:45:10 +00:00
|
|
|
{
|
|
|
|
if(iter->first.getRefData().getHandle() == "player")
|
|
|
|
{
|
|
|
|
/* Handle player last, in case a cell transition occurs */
|
|
|
|
player = iter;
|
|
|
|
continue;
|
|
|
|
}
|
2013-08-18 05:34:38 +00:00
|
|
|
moveObjectImp(iter->first, iter->second.x, iter->second.y, iter->second.z);
|
2013-02-05 20:45:10 +00:00
|
|
|
}
|
2013-08-18 05:34:38 +00:00
|
|
|
if(player != results.end())
|
|
|
|
moveObjectImp(player->first, player->second.x, player->second.y, player->second.z);
|
2013-04-28 12:59:15 +00:00
|
|
|
|
2013-08-18 05:34:38 +00:00
|
|
|
mPhysEngine->stepSimulation(duration);
|
2013-04-28 13:36:37 +00:00
|
|
|
}
|
|
|
|
|
2013-05-05 23:46:50 +00:00
|
|
|
bool World::castRay (float x1, float y1, float z1, float x2, float y2, float z2)
|
|
|
|
{
|
2013-07-31 16:46:32 +00:00
|
|
|
Ogre::Vector3 a(x1,y1,z1);
|
2013-05-05 23:46:50 +00:00
|
|
|
Ogre::Vector3 b(x2,y2,z2);
|
2013-05-07 18:42:21 +00:00
|
|
|
return mPhysics->castRay(a,b,false,true);
|
2013-05-05 23:46:50 +00:00
|
|
|
}
|
|
|
|
|
2013-04-28 13:36:37 +00:00
|
|
|
void World::processDoors(float duration)
|
|
|
|
{
|
2013-04-28 13:59:59 +00:00
|
|
|
std::map<MWWorld::Ptr, int>::iterator it = mDoorStates.begin();
|
|
|
|
while (it != mDoorStates.end())
|
2013-04-28 12:59:15 +00:00
|
|
|
{
|
2014-01-28 18:03:20 +00:00
|
|
|
if (!mWorldScene->isCellActive(*it->first.getCell()) || !it->first.getRefData().getBaseNode())
|
2014-05-14 23:58:44 +00:00
|
|
|
{
|
|
|
|
// The door is no longer in an active cell, or it was disabled.
|
|
|
|
// Erase from mDoorStates, since we no longer need to move it.
|
|
|
|
// Once we load the door's cell again (or re-enable the door), Door::insertObject will reinsert to mDoorStates.
|
2013-04-28 13:36:37 +00:00
|
|
|
mDoorStates.erase(it++);
|
2014-05-14 23:58:44 +00:00
|
|
|
}
|
2013-04-28 12:59:15 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
float oldRot = Ogre::Radian(it->first.getRefData().getLocalRotation().rot[2]).valueDegrees();
|
2013-04-28 13:59:59 +00:00
|
|
|
float diff = duration * 90;
|
2014-05-14 23:58:44 +00:00
|
|
|
float targetRot = std::min(std::max(0.f, oldRot + diff * (it->second == 1 ? 1 : -1)), 90.f);
|
2013-04-28 12:59:15 +00:00
|
|
|
localRotateObject(it->first, 0, 0, targetRot);
|
|
|
|
|
2014-05-14 23:58:44 +00:00
|
|
|
bool reached = (targetRot == 90.f && it->second) || targetRot == 0.f;
|
|
|
|
|
2013-04-30 18:26:59 +00:00
|
|
|
/// \todo should use convexSweepTest here
|
2013-04-28 12:59:15 +00:00
|
|
|
std::vector<std::string> collisions = mPhysics->getCollisions(it->first);
|
|
|
|
for (std::vector<std::string>::iterator cit = collisions.begin(); cit != collisions.end(); ++cit)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = getPtrViaHandle(*cit);
|
2014-05-14 23:58:44 +00:00
|
|
|
if (ptr.getClass().isActor())
|
2013-04-28 12:59:15 +00:00
|
|
|
{
|
2014-05-13 07:58:32 +00:00
|
|
|
// Collided with actor, ask actor to try to avoid door
|
2014-05-14 17:38:10 +00:00
|
|
|
if(ptr != MWBase::Environment::get().getWorld()->getPlayerPtr() ) {
|
2014-05-14 23:58:44 +00:00
|
|
|
MWMechanics::AiSequence& seq = ptr.getClass().getCreatureStats(ptr).getAiSequence();
|
2014-05-14 17:38:10 +00:00
|
|
|
if(seq.getTypeId() != MWMechanics::AiPackage::TypeIdAvoidDoor) //Only add it once
|
|
|
|
seq.stack(MWMechanics::AiAvoidDoor(it->first),ptr);
|
|
|
|
}
|
2014-05-14 08:05:18 +00:00
|
|
|
|
2014-05-13 07:58:32 +00:00
|
|
|
// we need to undo the rotation
|
2013-07-07 01:52:17 +00:00
|
|
|
localRotateObject(it->first, 0, 0, oldRot);
|
2014-05-14 23:58:44 +00:00
|
|
|
reached = false;
|
2014-05-14 17:38:10 +00:00
|
|
|
//break; //Removed in case multiple actors are touching
|
2013-04-28 12:59:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-14 23:58:44 +00:00
|
|
|
if (reached)
|
|
|
|
{
|
|
|
|
// Mark as non-moving
|
|
|
|
it->first.getClass().setDoorState(it->first, 0);
|
2013-04-28 13:59:59 +00:00
|
|
|
mDoorStates.erase(it++);
|
2014-05-14 23:58:44 +00:00
|
|
|
}
|
2013-04-28 13:59:59 +00:00
|
|
|
else
|
|
|
|
++it;
|
2013-04-28 12:59:15 +00:00
|
|
|
}
|
|
|
|
}
|
2011-01-29 16:39:34 +00:00
|
|
|
}
|
2011-03-13 21:33:55 +00:00
|
|
|
|
2011-04-26 19:38:21 +00:00
|
|
|
bool World::toggleCollisionMode()
|
2011-03-13 21:33:55 +00:00
|
|
|
{
|
2013-08-25 01:19:12 +00:00
|
|
|
return mPhysics->toggleCollisionMode();
|
2011-03-13 21:33:55 +00:00
|
|
|
}
|
2011-03-16 08:09:45 +00:00
|
|
|
|
2011-04-26 19:38:21 +00:00
|
|
|
bool World::toggleRenderMode (RenderMode mode)
|
2011-03-16 08:09:45 +00:00
|
|
|
{
|
2012-01-29 18:51:20 +00:00
|
|
|
return mRendering->toggleRenderMode (mode);
|
2011-03-16 08:09:45 +00:00
|
|
|
}
|
2011-07-22 13:56:54 +00:00
|
|
|
|
2012-11-07 21:52:34 +00:00
|
|
|
const ESM::Potion *World::createRecord (const ESM::Potion& record)
|
2011-07-22 13:56:54 +00:00
|
|
|
{
|
2012-12-04 09:58:43 +00:00
|
|
|
return mStore.insert(record);
|
2011-07-22 13:56:54 +00:00
|
|
|
}
|
|
|
|
|
2012-11-07 21:52:34 +00:00
|
|
|
const ESM::Class *World::createRecord (const ESM::Class& record)
|
2011-07-22 13:56:54 +00:00
|
|
|
{
|
2012-11-07 21:52:34 +00:00
|
|
|
return mStore.insert(record);
|
2011-07-22 13:56:54 +00:00
|
|
|
}
|
|
|
|
|
2012-11-07 21:52:34 +00:00
|
|
|
const ESM::Spell *World::createRecord (const ESM::Spell& record)
|
2012-10-15 19:54:19 +00:00
|
|
|
{
|
2012-11-07 21:52:34 +00:00
|
|
|
return mStore.insert(record);
|
2011-07-22 13:56:54 +00:00
|
|
|
}
|
2011-09-26 09:11:30 +00:00
|
|
|
|
|
|
|
const ESM::Cell *World::createRecord (const ESM::Cell& record)
|
|
|
|
{
|
2012-11-06 13:51:38 +00:00
|
|
|
return mStore.insert(record);
|
2011-09-26 09:11:30 +00:00
|
|
|
}
|
|
|
|
|
2012-11-08 12:37:57 +00:00
|
|
|
const ESM::NPC *World::createRecord(const ESM::NPC &record)
|
|
|
|
{
|
2012-11-10 07:51:48 +00:00
|
|
|
bool update = false;
|
2012-12-04 09:58:43 +00:00
|
|
|
|
2012-12-23 19:23:24 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(record.mId, "player"))
|
2012-12-04 09:58:43 +00:00
|
|
|
{
|
2013-03-12 01:44:03 +00:00
|
|
|
std::vector<std::string> ids;
|
|
|
|
getStore().get<ESM::Race>().listIdentifier(ids);
|
2012-12-04 09:58:43 +00:00
|
|
|
|
2013-03-12 01:44:03 +00:00
|
|
|
unsigned int i=0;
|
2012-12-04 09:58:43 +00:00
|
|
|
|
2013-03-12 01:44:03 +00:00
|
|
|
for (; i<ids.size(); ++i)
|
|
|
|
if (Misc::StringUtils::ciEqual (ids[i], record.mRace))
|
2012-12-04 09:58:43 +00:00
|
|
|
break;
|
|
|
|
|
2013-12-10 14:09:58 +00:00
|
|
|
mGlobalVariables["pcrace"].setInteger (i == ids.size() ? 0 : i+1);
|
2012-12-04 09:58:43 +00:00
|
|
|
|
2012-11-10 07:51:48 +00:00
|
|
|
const ESM::NPC *player =
|
|
|
|
mPlayer->getPlayer().get<ESM::NPC>()->mBase;
|
2011-09-26 09:11:30 +00:00
|
|
|
|
2012-11-10 07:51:48 +00:00
|
|
|
update = record.isMale() != player->isMale() ||
|
2012-12-23 19:23:24 +00:00
|
|
|
!Misc::StringUtils::ciEqual(record.mRace, player->mRace) ||
|
|
|
|
!Misc::StringUtils::ciEqual(record.mHead, player->mHead) ||
|
|
|
|
!Misc::StringUtils::ciEqual(record.mHair, player->mHair);
|
2011-09-26 09:11:30 +00:00
|
|
|
}
|
2012-11-10 07:51:48 +00:00
|
|
|
const ESM::NPC *ret = mStore.insert(record);
|
|
|
|
if (update) {
|
|
|
|
mRendering->renderPlayer(mPlayer->getPlayer());
|
|
|
|
}
|
|
|
|
return ret;
|
2011-09-26 09:11:30 +00:00
|
|
|
}
|
2011-11-21 13:08:44 +00:00
|
|
|
|
2013-03-28 16:41:00 +00:00
|
|
|
const ESM::Armor *World::createRecord (const ESM::Armor& record)
|
|
|
|
{
|
|
|
|
return mStore.insert(record);
|
|
|
|
}
|
|
|
|
|
|
|
|
const ESM::Weapon *World::createRecord (const ESM::Weapon& record)
|
|
|
|
{
|
|
|
|
return mStore.insert(record);
|
|
|
|
}
|
|
|
|
|
|
|
|
const ESM::Clothing *World::createRecord (const ESM::Clothing& record)
|
|
|
|
{
|
|
|
|
return mStore.insert(record);
|
|
|
|
}
|
|
|
|
|
|
|
|
const ESM::Enchantment *World::createRecord (const ESM::Enchantment& record)
|
|
|
|
{
|
|
|
|
return mStore.insert(record);
|
|
|
|
}
|
|
|
|
|
|
|
|
const ESM::Book *World::createRecord (const ESM::Book& record)
|
|
|
|
{
|
|
|
|
return mStore.insert(record);
|
|
|
|
}
|
|
|
|
|
2012-11-03 19:29:55 +00:00
|
|
|
void World::update (float duration, bool paused)
|
2012-01-23 13:33:06 +00:00
|
|
|
{
|
2013-05-15 15:54:18 +00:00
|
|
|
if (mPlayIntro)
|
|
|
|
{
|
|
|
|
--mPlayIntro;
|
|
|
|
if (mPlayIntro == 0)
|
2014-03-27 18:10:15 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->playVideo(mFallback.getFallbackString("Movies_New_Game"), true);
|
2013-05-15 15:54:18 +00:00
|
|
|
}
|
|
|
|
|
2014-01-11 05:47:58 +00:00
|
|
|
if (mGoToJail && !paused)
|
|
|
|
goToJail();
|
|
|
|
|
2013-12-29 11:47:44 +00:00
|
|
|
updateWeather(duration);
|
2013-03-03 14:10:40 +00:00
|
|
|
|
2012-11-03 19:29:55 +00:00
|
|
|
mWorldScene->update (duration, paused);
|
2012-12-04 09:58:43 +00:00
|
|
|
|
2013-11-28 16:31:17 +00:00
|
|
|
if (!paused)
|
|
|
|
doPhysics (duration);
|
2013-08-18 05:34:38 +00:00
|
|
|
|
2013-01-08 06:48:24 +00:00
|
|
|
performUpdateSceneQueries ();
|
|
|
|
|
|
|
|
updateWindowManager ();
|
2013-12-31 23:12:31 +00:00
|
|
|
|
2014-01-21 13:50:58 +00:00
|
|
|
if (!paused && mPlayer->getPlayer().getCell()->isExterior())
|
2013-12-31 23:12:31 +00:00
|
|
|
{
|
|
|
|
ESM::Position pos = mPlayer->getPlayer().getRefData().getPosition();
|
2014-01-01 21:46:10 +00:00
|
|
|
mPlayer->setLastKnownExteriorPosition(Ogre::Vector3(pos.pos));
|
2013-12-31 23:12:31 +00:00
|
|
|
}
|
2013-01-08 06:48:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void World::updateWindowManager ()
|
|
|
|
{
|
2012-04-15 15:10:08 +00:00
|
|
|
// inform the GUI about focused object
|
2013-01-08 07:27:37 +00:00
|
|
|
MWWorld::Ptr object = getFacedObject ();
|
2012-05-14 19:37:43 +00:00
|
|
|
|
2012-10-18 20:21:39 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->setFocusObject(object);
|
2012-05-14 19:37:43 +00:00
|
|
|
|
2012-10-18 20:21:39 +00:00
|
|
|
// retrieve object dimensions so we know where to place the floating label
|
|
|
|
if (!object.isEmpty ())
|
|
|
|
{
|
2012-05-14 19:37:43 +00:00
|
|
|
Ogre::SceneNode* node = object.getRefData().getBaseNode();
|
2013-03-07 14:25:22 +00:00
|
|
|
Ogre::AxisAlignedBox bounds = node->_getWorldAABB();
|
2012-05-14 19:37:43 +00:00
|
|
|
if (bounds.isFinite())
|
|
|
|
{
|
|
|
|
Vector4 screenCoords = mRendering->boundingBoxToScreen(bounds);
|
|
|
|
MWBase::Environment::get().getWindowManager()->setFocusObjectScreenCoords(
|
|
|
|
screenCoords[0], screenCoords[1], screenCoords[2], screenCoords[3]);
|
|
|
|
}
|
2012-04-15 15:10:08 +00:00
|
|
|
}
|
2013-01-08 06:48:24 +00:00
|
|
|
}
|
2012-04-15 15:10:08 +00:00
|
|
|
|
2013-01-08 06:48:24 +00:00
|
|
|
void World::performUpdateSceneQueries ()
|
|
|
|
{
|
2012-03-24 16:59:26 +00:00
|
|
|
if (!mRendering->occlusionQuerySupported())
|
|
|
|
{
|
|
|
|
// cast a ray from player to sun to detect if the sun is visible
|
|
|
|
// this is temporary until we find a better place to put this code
|
|
|
|
// currently its here because we need to access the physics system
|
|
|
|
float* p = mPlayer->getPlayer().getRefData().getPosition().pos;
|
|
|
|
Vector3 sun = mRendering->getSkyManager()->getRealSunPos();
|
|
|
|
mRendering->getSkyManager()->setGlare(!mPhysics->castRay(Ogre::Vector3(p[0], p[1], p[2]), sun));
|
|
|
|
}
|
2012-03-25 18:52:56 +00:00
|
|
|
|
2013-03-08 22:46:25 +00:00
|
|
|
updateFacedHandle ();
|
2013-01-08 06:48:24 +00:00
|
|
|
}
|
2012-03-25 18:52:56 +00:00
|
|
|
|
2013-03-08 22:46:25 +00:00
|
|
|
void World::updateFacedHandle ()
|
2013-01-08 06:48:24 +00:00
|
|
|
{
|
2014-01-01 22:30:58 +00:00
|
|
|
float telekinesisRangeBonus =
|
|
|
|
mPlayer->getPlayer().getClass().getCreatureStats(mPlayer->getPlayer()).getMagicEffects()
|
2014-01-01 22:34:18 +00:00
|
|
|
.get(ESM::MagicEffect::Telekinesis).mMagnitude;
|
|
|
|
telekinesisRangeBonus = feetToGameUnits(telekinesisRangeBonus);
|
2014-01-01 22:30:58 +00:00
|
|
|
|
|
|
|
float activationDistance = getMaxActivationDistance() + telekinesisRangeBonus;
|
2014-01-01 22:59:17 +00:00
|
|
|
activationDistance += mRendering->getCameraDistance();
|
2014-01-01 22:30:58 +00:00
|
|
|
|
2013-01-08 06:48:24 +00:00
|
|
|
// send new query
|
|
|
|
// figure out which object we want to test against
|
|
|
|
std::vector < std::pair < float, std::string > > results;
|
|
|
|
if (MWBase::Environment::get().getWindowManager()->isGuiMode())
|
|
|
|
{
|
|
|
|
float x, y;
|
|
|
|
MWBase::Environment::get().getWindowManager()->getMousePosition(x, y);
|
2014-01-01 22:30:58 +00:00
|
|
|
results = mPhysics->getFacedHandles(x, y, activationDistance);
|
2013-03-18 20:33:12 +00:00
|
|
|
if (MWBase::Environment::get().getWindowManager()->isConsoleMode())
|
|
|
|
results = mPhysics->getFacedHandles(x, y, getMaxActivationDistance ()*50);
|
2013-01-08 06:48:24 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-01-01 22:30:58 +00:00
|
|
|
results = mPhysics->getFacedHandles(activationDistance);
|
2013-01-08 06:48:24 +00:00
|
|
|
}
|
2012-03-27 20:36:02 +00:00
|
|
|
|
2013-01-08 06:48:24 +00:00
|
|
|
// ignore the player and other things we're not interested in
|
|
|
|
std::vector < std::pair < float, std::string > >::iterator it = results.begin();
|
|
|
|
while (it != results.end())
|
|
|
|
{
|
|
|
|
if ( (*it).second.find("HeightField") != std::string::npos // not interested in terrain
|
|
|
|
|| getPtrViaHandle((*it).second) == mPlayer->getPlayer() ) // not interested in player (unless you want to talk to yourself)
|
|
|
|
{
|
|
|
|
it = results.erase(it);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
2012-03-25 18:52:56 +00:00
|
|
|
|
2013-07-31 16:46:32 +00:00
|
|
|
if (results.empty())
|
2013-01-08 06:48:24 +00:00
|
|
|
{
|
2013-03-08 22:46:25 +00:00
|
|
|
mFacedHandle = "";
|
|
|
|
mFacedDistance = FLT_MAX;
|
2012-03-25 18:52:56 +00:00
|
|
|
}
|
2013-01-08 06:48:24 +00:00
|
|
|
else
|
|
|
|
{
|
2013-03-08 22:46:25 +00:00
|
|
|
mFacedHandle = results.front().second;
|
|
|
|
mFacedDistance = results.front().first;
|
2013-01-08 06:48:24 +00:00
|
|
|
}
|
2012-01-23 13:33:06 +00:00
|
|
|
}
|
2012-03-15 15:01:41 +00:00
|
|
|
|
2012-02-23 20:12:06 +00:00
|
|
|
bool World::isCellExterior() const
|
|
|
|
{
|
2013-12-05 12:21:26 +00:00
|
|
|
CellStore *currentCell = mWorldScene->getCurrentCell();
|
2012-02-23 20:12:06 +00:00
|
|
|
if (currentCell)
|
|
|
|
{
|
2014-02-21 10:35:46 +00:00
|
|
|
return currentCell->getCell()->isExterior();
|
2012-02-23 20:12:06 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2012-03-15 15:01:41 +00:00
|
|
|
|
2012-02-23 20:12:06 +00:00
|
|
|
bool World::isCellQuasiExterior() const
|
|
|
|
{
|
2013-12-05 12:21:26 +00:00
|
|
|
CellStore *currentCell = mWorldScene->getCurrentCell();
|
2012-02-23 20:12:06 +00:00
|
|
|
if (currentCell)
|
|
|
|
{
|
2014-02-21 10:35:46 +00:00
|
|
|
if (!(currentCell->getCell()->mData.mFlags & ESM::Cell::QuasiEx))
|
2012-02-23 20:12:06 +00:00
|
|
|
return false;
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2012-03-15 15:01:41 +00:00
|
|
|
|
2012-02-25 20:34:38 +00:00
|
|
|
int World::getCurrentWeather() const
|
|
|
|
{
|
|
|
|
return mWeatherManager->getWeatherID();
|
|
|
|
}
|
2012-03-15 15:01:41 +00:00
|
|
|
|
2012-02-26 10:51:02 +00:00
|
|
|
void World::changeWeather(const std::string& region, const unsigned int id)
|
|
|
|
{
|
|
|
|
mWeatherManager->changeWeather(region, id);
|
|
|
|
}
|
2012-03-15 15:01:41 +00:00
|
|
|
|
2013-07-27 14:10:18 +00:00
|
|
|
void World::modRegion(const std::string ®ionid, const std::vector<char> &chances)
|
|
|
|
{
|
|
|
|
mWeatherManager->modRegion(regionid, chances);
|
|
|
|
}
|
|
|
|
|
2012-02-18 17:25:28 +00:00
|
|
|
OEngine::Render::Fader* World::getFader()
|
|
|
|
{
|
|
|
|
return mRendering->getFader();
|
|
|
|
}
|
2012-03-29 13:49:24 +00:00
|
|
|
|
2012-07-02 23:06:38 +00:00
|
|
|
Ogre::Vector2 World::getNorthVector (CellStore* cell)
|
2012-03-28 01:15:10 +00:00
|
|
|
{
|
2014-02-23 20:21:27 +00:00
|
|
|
MWWorld::CellRefList<ESM::Static>& statics = cell->get<ESM::Static>();
|
2012-06-29 16:54:23 +00:00
|
|
|
MWWorld::LiveCellRef<ESM::Static>* ref = statics.find("northmarker");
|
2012-03-28 01:15:10 +00:00
|
|
|
if (!ref)
|
|
|
|
return Vector2(0, 1);
|
|
|
|
Ogre::SceneNode* node = ref->mData.getBaseNode();
|
2013-02-26 12:39:10 +00:00
|
|
|
Vector3 dir = node->_getDerivedOrientation() * Ogre::Vector3(0,1,0);
|
|
|
|
Vector2 d = Vector2(dir.x, dir.y);
|
2012-03-28 01:15:10 +00:00
|
|
|
return d;
|
|
|
|
}
|
2012-03-30 13:31:07 +00:00
|
|
|
|
2014-01-01 21:37:52 +00:00
|
|
|
void World::getDoorMarkers (CellStore* cell, std::vector<World::DoorMarker>& out)
|
2012-08-28 15:30:34 +00:00
|
|
|
{
|
2014-02-23 20:21:27 +00:00
|
|
|
MWWorld::CellRefList<ESM::Door>& doors = cell->get<ESM::Door>();
|
2012-11-25 16:19:29 +00:00
|
|
|
CellRefList<ESM::Door>::List& refList = doors.mList;
|
|
|
|
for (CellRefList<ESM::Door>::List::iterator it = refList.begin(); it != refList.end(); ++it)
|
2012-08-28 15:30:34 +00:00
|
|
|
{
|
2012-08-28 16:23:01 +00:00
|
|
|
MWWorld::LiveCellRef<ESM::Door>& ref = *it;
|
2012-08-28 15:30:34 +00:00
|
|
|
|
2014-05-11 13:22:46 +00:00
|
|
|
if (!ref.mData.isEnabled())
|
|
|
|
continue;
|
|
|
|
|
2012-11-05 12:07:59 +00:00
|
|
|
if (ref.mRef.mTeleport)
|
2012-08-28 15:30:34 +00:00
|
|
|
{
|
|
|
|
World::DoorMarker newMarker;
|
2012-12-23 19:23:24 +00:00
|
|
|
newMarker.name = MWClass::Door::getDestination(ref);
|
2012-08-28 15:30:34 +00:00
|
|
|
|
|
|
|
ESM::Position pos = ref.mData.getPosition ();
|
|
|
|
|
|
|
|
newMarker.x = pos.pos[0];
|
|
|
|
newMarker.y = pos.pos[1];
|
2014-01-01 21:37:52 +00:00
|
|
|
out.push_back(newMarker);
|
2012-08-28 15:30:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void World::getInteriorMapPosition (Ogre::Vector2 position, float& nX, float& nY, int &x, int& y)
|
|
|
|
{
|
|
|
|
mRendering->getInteriorMapPosition(position, nX, nY, x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool World::isPositionExplored (float nX, float nY, int x, int y, bool interior)
|
|
|
|
{
|
|
|
|
return mRendering->isPositionExplored(nX, nY, x, y, interior);
|
|
|
|
}
|
|
|
|
|
2012-03-29 13:49:24 +00:00
|
|
|
void World::setWaterHeight(const float height)
|
|
|
|
{
|
|
|
|
mRendering->setWaterHeight(height);
|
|
|
|
}
|
|
|
|
|
2014-05-16 07:21:08 +00:00
|
|
|
bool World::toggleWater()
|
2012-03-29 16:33:08 +00:00
|
|
|
{
|
2014-05-16 07:21:08 +00:00
|
|
|
return mRendering->toggleWater();
|
2012-03-29 16:33:08 +00:00
|
|
|
}
|
|
|
|
|
2013-01-31 18:45:32 +00:00
|
|
|
void World::PCDropped (const Ptr& item)
|
|
|
|
{
|
|
|
|
std::string script = MWWorld::Class::get(item).getScript(item);
|
2013-02-06 09:12:40 +00:00
|
|
|
|
|
|
|
// Set OnPCDrop Variable on item's script, if it has a script with that variable declared
|
2013-01-31 18:45:32 +00:00
|
|
|
if(script != "")
|
2013-08-15 00:05:42 +00:00
|
|
|
item.getRefData().getLocals().setVarByInt(script, "onpcdrop", 1);
|
2013-01-31 18:45:32 +00:00
|
|
|
}
|
|
|
|
|
2014-05-16 01:19:38 +00:00
|
|
|
MWWorld::Ptr World::placeObject (const MWWorld::Ptr& object, float cursorX, float cursorY, int amount)
|
2012-05-14 15:41:17 +00:00
|
|
|
{
|
2012-05-15 14:47:23 +00:00
|
|
|
std::pair<bool, Ogre::Vector3> result = mPhysics->castRay(cursorX, cursorY);
|
|
|
|
|
|
|
|
if (!result.first)
|
2014-05-16 01:19:38 +00:00
|
|
|
return MWWorld::Ptr();
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
CellStore* cell = getPlayerPtr().getCell();
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2014-01-08 17:39:44 +00:00
|
|
|
ESM::Position pos = getPlayerPtr().getRefData().getPosition();
|
2012-05-15 14:47:23 +00:00
|
|
|
pos.pos[0] = result.second[0];
|
2013-02-26 13:28:35 +00:00
|
|
|
pos.pos[1] = result.second[1];
|
|
|
|
pos.pos[2] = result.second[2];
|
2013-03-09 12:27:24 +00:00
|
|
|
// We want only the Z part of the player's rotation
|
|
|
|
pos.rot[0] = 0;
|
|
|
|
pos.rot[1] = 0;
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2013-08-12 23:19:33 +00:00
|
|
|
// copy the object and set its count
|
|
|
|
int origCount = object.getRefData().getCount();
|
|
|
|
object.getRefData().setCount(amount);
|
2014-03-07 05:11:00 +00:00
|
|
|
Ptr dropped = copyObjectToCell(object, cell, pos, true);
|
2013-08-12 23:19:33 +00:00
|
|
|
object.getRefData().setCount(origCount);
|
|
|
|
|
|
|
|
// only the player place items in the world, so no need to check actor
|
2013-01-31 18:45:32 +00:00
|
|
|
PCDropped(dropped);
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2014-05-16 01:19:38 +00:00
|
|
|
return dropped;
|
2012-05-14 15:41:17 +00:00
|
|
|
}
|
|
|
|
|
2012-05-15 14:47:23 +00:00
|
|
|
bool World::canPlaceObject(float cursorX, float cursorY)
|
|
|
|
{
|
|
|
|
std::pair<bool, Ogre::Vector3> result = mPhysics->castRay(cursorX, cursorY);
|
|
|
|
|
|
|
|
/// \todo also check if the wanted position is on a flat surface, and not e.g. against a vertical wall!
|
|
|
|
|
|
|
|
if (!result.first)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-02-06 09:12:40 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
Ptr World::copyObjectToCell(const Ptr &object, CellStore* cell, ESM::Position pos, bool adjustPos)
|
2012-07-25 06:47:59 +00:00
|
|
|
{
|
2013-12-18 23:11:14 +00:00
|
|
|
if (object.getClass().isActor() || adjustPos)
|
2013-11-30 07:29:22 +00:00
|
|
|
{
|
|
|
|
Ogre::Vector3 min, max;
|
|
|
|
if (mPhysics->getObjectAABB(object, min, max)) {
|
2014-01-14 06:40:17 +00:00
|
|
|
pos.pos[0] -= (min.x + max.x) / 2;
|
|
|
|
pos.pos[1] -= (min.y + max.y) / 2;
|
|
|
|
pos.pos[2] -= min.z;
|
2013-11-30 07:29:22 +00:00
|
|
|
}
|
2012-07-26 12:14:11 +00:00
|
|
|
}
|
2012-07-25 13:18:17 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
if (cell->isExterior())
|
2014-01-14 06:40:17 +00:00
|
|
|
{
|
|
|
|
int cellX, cellY;
|
|
|
|
positionToIndex(pos.pos[0], pos.pos[1], cellX, cellY);
|
2014-03-07 05:11:00 +00:00
|
|
|
cell = mCells.getExterior(cellX, cellY);
|
2014-01-14 06:40:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MWWorld::Ptr dropped =
|
2014-03-07 05:11:00 +00:00
|
|
|
MWWorld::Class::get(object).copyToCell(object, *cell, pos);
|
2014-01-14 06:40:17 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
if (mWorldScene->isCellActive(*cell)) {
|
2012-07-26 15:06:48 +00:00
|
|
|
if (dropped.getRefData().isEnabled()) {
|
|
|
|
mWorldScene->addObjectToScene(dropped);
|
|
|
|
}
|
2012-07-26 12:14:11 +00:00
|
|
|
std::string script = MWWorld::Class::get(dropped).getScript(dropped);
|
|
|
|
if (!script.empty()) {
|
|
|
|
mLocalScripts.add(script, dropped);
|
|
|
|
}
|
2014-03-07 05:11:00 +00:00
|
|
|
addContainerScripts(dropped, cell);
|
2012-07-25 06:47:59 +00:00
|
|
|
}
|
2013-01-31 18:45:32 +00:00
|
|
|
|
|
|
|
return dropped;
|
2012-07-25 06:47:59 +00:00
|
|
|
}
|
|
|
|
|
2014-05-16 01:19:38 +00:00
|
|
|
MWWorld::Ptr World::dropObjectOnGround (const Ptr& actor, const Ptr& object, int amount)
|
2012-05-15 14:47:23 +00:00
|
|
|
{
|
2013-12-05 12:21:26 +00:00
|
|
|
MWWorld::CellStore* cell = actor.getCell();
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2012-07-25 16:25:53 +00:00
|
|
|
ESM::Position pos =
|
2013-01-09 21:16:45 +00:00
|
|
|
actor.getRefData().getPosition();
|
2013-03-09 12:27:24 +00:00
|
|
|
// We want only the Z part of the actor's rotation
|
|
|
|
pos.rot[0] = 0;
|
|
|
|
pos.rot[1] = 0;
|
2012-05-15 14:47:23 +00:00
|
|
|
|
2012-07-25 16:25:53 +00:00
|
|
|
Ogre::Vector3 orig =
|
2014-01-01 21:46:10 +00:00
|
|
|
Ogre::Vector3(pos.pos);
|
2012-07-25 16:25:53 +00:00
|
|
|
Ogre::Vector3 dir = Ogre::Vector3(0, 0, -1);
|
2012-08-04 07:34:49 +00:00
|
|
|
|
2012-07-25 16:25:53 +00:00
|
|
|
float len = (pos.pos[2] >= 0) ? pos.pos[2] : -pos.pos[2];
|
|
|
|
len += 100.0;
|
|
|
|
|
|
|
|
std::pair<bool, Ogre::Vector3> hit =
|
|
|
|
mPhysics->castRay(orig, dir, len);
|
|
|
|
pos.pos[2] = hit.second.z;
|
|
|
|
|
2013-08-12 23:19:33 +00:00
|
|
|
// copy the object and set its count
|
|
|
|
int origCount = object.getRefData().getCount();
|
|
|
|
object.getRefData().setCount(amount);
|
2014-03-07 05:11:00 +00:00
|
|
|
Ptr dropped = copyObjectToCell(object, cell, pos);
|
2013-08-12 23:19:33 +00:00
|
|
|
object.getRefData().setCount(origCount);
|
|
|
|
|
2013-01-31 18:45:32 +00:00
|
|
|
if(actor == mPlayer->getPlayer()) // Only call if dropped by player
|
|
|
|
PCDropped(dropped);
|
2014-05-16 01:19:38 +00:00
|
|
|
return dropped;
|
2012-05-15 14:47:23 +00:00
|
|
|
}
|
2012-05-22 23:32:36 +00:00
|
|
|
|
|
|
|
void World::processChangedSettings(const Settings::CategorySettingVector& settings)
|
|
|
|
{
|
|
|
|
mRendering->processChangedSettings(settings);
|
|
|
|
}
|
2012-06-22 10:56:04 +00:00
|
|
|
|
|
|
|
void World::getTriangleBatchCount(unsigned int &triangles, unsigned int &batches)
|
|
|
|
{
|
|
|
|
mRendering->getTriangleBatchCount(triangles, batches);
|
|
|
|
}
|
2012-08-03 10:42:09 +00:00
|
|
|
|
2013-02-19 06:39:43 +00:00
|
|
|
bool
|
|
|
|
World::isFlying(const MWWorld::Ptr &ptr) const
|
|
|
|
{
|
2013-08-21 14:24:02 +00:00
|
|
|
if(!ptr.getClass().isActor())
|
|
|
|
return false;
|
|
|
|
|
2014-01-05 14:38:12 +00:00
|
|
|
if (ptr.getClass().getCreatureStats(ptr).isDead())
|
|
|
|
return false;
|
|
|
|
|
2014-03-15 21:09:14 +00:00
|
|
|
if (ptr.getClass().canFly(ptr))
|
2014-01-05 14:38:12 +00:00
|
|
|
return true;
|
|
|
|
|
2013-08-21 14:24:02 +00:00
|
|
|
const MWMechanics::CreatureStats &stats = ptr.getClass().getCreatureStats(ptr);
|
2014-01-02 20:49:04 +00:00
|
|
|
if(stats.getMagicEffects().get(ESM::MagicEffect::Levitate).mMagnitude > 0
|
2014-01-02 15:38:23 +00:00
|
|
|
&& isLevitationEnabled())
|
2013-08-21 14:24:02 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
const OEngine::Physic::PhysicActor *actor = mPhysEngine->getCharacter(ptr.getRefData().getHandle());
|
|
|
|
if(!actor || !actor->getCollisionMode())
|
2013-02-20 09:55:12 +00:00
|
|
|
return true;
|
2013-08-21 14:24:02 +00:00
|
|
|
|
2013-02-20 09:55:12 +00:00
|
|
|
return false;
|
2013-02-19 06:39:43 +00:00
|
|
|
}
|
|
|
|
|
2013-10-02 20:53:29 +00:00
|
|
|
bool
|
|
|
|
World::isSlowFalling(const MWWorld::Ptr &ptr) const
|
|
|
|
{
|
|
|
|
if(!ptr.getClass().isActor())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const MWMechanics::CreatureStats &stats = ptr.getClass().getCreatureStats(ptr);
|
2014-01-02 20:49:04 +00:00
|
|
|
if(stats.getMagicEffects().get(ESM::MagicEffect::SlowFall).mMagnitude > 0)
|
2013-10-02 20:53:29 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-08-07 13:34:11 +00:00
|
|
|
bool World::isSubmerged(const MWWorld::Ptr &object) const
|
|
|
|
{
|
|
|
|
float *fpos = object.getRefData().getPosition().pos;
|
|
|
|
Ogre::Vector3 pos(fpos[0], fpos[1], fpos[2]);
|
|
|
|
|
|
|
|
const OEngine::Physic::PhysicActor *actor = mPhysEngine->getCharacter(object.getRefData().getHandle());
|
|
|
|
if(actor) pos.z += 1.85*actor->getHalfExtents().z;
|
|
|
|
|
|
|
|
return isUnderwater(object.getCell(), pos);
|
|
|
|
}
|
|
|
|
|
2012-08-03 10:42:09 +00:00
|
|
|
bool
|
2013-02-15 06:35:15 +00:00
|
|
|
World::isSwimming(const MWWorld::Ptr &object) const
|
2012-08-03 10:42:09 +00:00
|
|
|
{
|
|
|
|
/// \todo add check ifActor() - only actors can swim
|
|
|
|
float *fpos = object.getRefData().getPosition().pos;
|
|
|
|
Ogre::Vector3 pos(fpos[0], fpos[1], fpos[2]);
|
|
|
|
|
2013-02-06 20:56:30 +00:00
|
|
|
/// \fixme 3/4ths submerged?
|
|
|
|
const OEngine::Physic::PhysicActor *actor = mPhysEngine->getCharacter(object.getRefData().getHandle());
|
|
|
|
if(actor) pos.z += actor->getHalfExtents().z * 1.5;
|
2012-08-03 10:42:09 +00:00
|
|
|
|
2013-02-25 16:27:50 +00:00
|
|
|
return isUnderwater(object.getCell(), pos);
|
2012-08-03 10:42:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2013-12-05 12:21:26 +00:00
|
|
|
World::isUnderwater(const MWWorld::CellStore* cell, const Ogre::Vector3 &pos) const
|
2012-08-03 10:42:09 +00:00
|
|
|
{
|
2014-02-21 10:35:46 +00:00
|
|
|
if (!(cell->getCell()->mData.mFlags & ESM::Cell::HasWater)) {
|
2012-08-03 10:42:09 +00:00
|
|
|
return false;
|
|
|
|
}
|
2014-02-23 16:34:18 +00:00
|
|
|
return pos.z < cell->getWaterLevel();
|
2012-08-03 10:42:09 +00:00
|
|
|
}
|
2012-08-14 16:33:29 +00:00
|
|
|
|
2014-04-13 04:53:36 +00:00
|
|
|
// physactor->getOnGround() is not a reliable indicator of whether the actor
|
|
|
|
// is on the ground (defaults to false, which means code blocks such as
|
|
|
|
// CharacterController::update() may falsely detect "falling").
|
|
|
|
//
|
|
|
|
// Also, collisions can move z position slightly off zero, giving a false
|
|
|
|
// indication. In order to reduce false detection of jumping, small distance
|
|
|
|
// below the actor is detected and ignored. A value of 1.5 is used here, but
|
|
|
|
// something larger may be more suitable. This change should resolve Bug#1271.
|
|
|
|
//
|
2014-04-13 08:34:08 +00:00
|
|
|
// TODO: There might be better places to update PhysicActor::mOnGround.
|
2013-02-15 06:35:15 +00:00
|
|
|
bool World::isOnGround(const MWWorld::Ptr &ptr) const
|
|
|
|
{
|
|
|
|
RefData &refdata = ptr.getRefData();
|
|
|
|
const OEngine::Physic::PhysicActor *physactor = mPhysEngine->getCharacter(refdata.getHandle());
|
2014-04-13 08:34:08 +00:00
|
|
|
|
|
|
|
if(!physactor)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if(physactor->getOnGround())
|
|
|
|
return true;
|
|
|
|
else
|
2014-04-13 04:53:36 +00:00
|
|
|
{
|
|
|
|
Ogre::Vector3 pos(ptr.getRefData().getPosition().pos);
|
|
|
|
OEngine::Physic::ActorTracer tracer;
|
|
|
|
// a small distance above collision object is considered "on ground"
|
|
|
|
tracer.findGround(physactor->getCollisionBody(),
|
|
|
|
pos,
|
|
|
|
pos - Ogre::Vector3(0, 0, 1.5f), // trace a small amount down
|
|
|
|
mPhysEngine);
|
|
|
|
if(tracer.mFraction < 1.0f) // collision, must be close to something below
|
2014-04-13 08:34:08 +00:00
|
|
|
{
|
|
|
|
const_cast<OEngine::Physic::PhysicActor *> (physactor)->setOnGround(true);
|
|
|
|
return true;
|
|
|
|
}
|
2014-04-13 04:53:36 +00:00
|
|
|
else
|
2014-04-13 08:34:08 +00:00
|
|
|
return false;
|
2014-04-13 04:53:36 +00:00
|
|
|
}
|
2013-02-15 06:35:15 +00:00
|
|
|
}
|
|
|
|
|
2013-04-09 18:24:41 +00:00
|
|
|
bool World::vanityRotateCamera(float * rot)
|
|
|
|
{
|
|
|
|
return mRendering->vanityRotateCamera(rot);
|
|
|
|
}
|
|
|
|
|
2013-12-05 12:21:26 +00:00
|
|
|
void World::setCameraDistance(float dist, bool adjust, bool override_)
|
2013-07-16 16:38:18 +00:00
|
|
|
{
|
2013-12-05 12:21:26 +00:00
|
|
|
return mRendering->setCameraDistance(dist, adjust, override_);
|
2013-07-16 16:38:18 +00:00
|
|
|
}
|
|
|
|
|
2013-04-29 15:56:16 +00:00
|
|
|
void World::setupPlayer()
|
2013-04-15 00:56:23 +00:00
|
|
|
{
|
2013-04-29 06:44:44 +00:00
|
|
|
const ESM::NPC *player = mStore.get<ESM::NPC>().find("player");
|
2013-05-15 15:54:18 +00:00
|
|
|
if (!mPlayer)
|
|
|
|
mPlayer = new MWWorld::Player(player, *this);
|
|
|
|
else
|
|
|
|
mPlayer->set(player);
|
2013-04-29 06:44:44 +00:00
|
|
|
|
|
|
|
Ptr ptr = mPlayer->getPlayer();
|
|
|
|
mRendering->setupPlayer(ptr);
|
2013-04-15 00:56:23 +00:00
|
|
|
}
|
|
|
|
|
2012-08-14 16:33:29 +00:00
|
|
|
void World::renderPlayer()
|
|
|
|
{
|
|
|
|
mRendering->renderPlayer(mPlayer->getPlayer());
|
2014-01-25 15:13:45 +00:00
|
|
|
|
|
|
|
// At this point the Animation object is live, and the CharacterController associated with it must be created.
|
|
|
|
// It has to be done at this point: resetCamera below does animation->setViewMode -> CharacterController::forceStateUpdate
|
|
|
|
// so we should make sure not to use a "stale" controller for that.
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->add(mPlayer->getPlayer());
|
|
|
|
|
2013-03-12 01:50:23 +00:00
|
|
|
mPhysics->addActor(mPlayer->getPlayer());
|
2013-12-05 13:18:43 +00:00
|
|
|
mRendering->resetCamera();
|
2012-08-14 16:33:29 +00:00
|
|
|
}
|
2012-09-12 22:54:32 +00:00
|
|
|
|
2012-09-19 00:53:06 +00:00
|
|
|
int World::canRest ()
|
2012-09-18 18:53:32 +00:00
|
|
|
{
|
2013-12-05 12:21:26 +00:00
|
|
|
CellStore *currentCell = mWorldScene->getCurrentCell();
|
2012-09-19 00:53:06 +00:00
|
|
|
|
2013-08-09 01:30:47 +00:00
|
|
|
Ptr player = mPlayer->getPlayer();
|
|
|
|
RefData &refdata = player.getRefData();
|
2013-02-07 05:44:58 +00:00
|
|
|
Ogre::Vector3 playerPos(refdata.getPosition().pos);
|
2012-09-19 00:53:06 +00:00
|
|
|
|
2013-02-15 04:10:04 +00:00
|
|
|
const OEngine::Physic::PhysicActor *physactor = mPhysEngine->getCharacter(refdata.getHandle());
|
2013-04-27 18:57:44 +00:00
|
|
|
if((!physactor->getOnGround()&&physactor->getCollisionMode()) || isUnderwater(currentCell, playerPos))
|
2012-09-19 00:53:06 +00:00
|
|
|
return 2;
|
2014-02-21 10:35:46 +00:00
|
|
|
if((currentCell->getCell()->mData.mFlags&ESM::Cell::NoSleep) ||
|
2013-08-09 01:30:47 +00:00
|
|
|
Class::get(player).getNpcStats(player).isWerewolf())
|
2012-09-19 00:53:06 +00:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
2012-09-18 18:53:32 +00:00
|
|
|
}
|
2012-09-25 00:35:50 +00:00
|
|
|
|
2013-01-16 17:59:19 +00:00
|
|
|
MWRender::Animation* World::getAnimation(const MWWorld::Ptr &ptr)
|
|
|
|
{
|
|
|
|
return mRendering->getAnimation(ptr);
|
|
|
|
}
|
|
|
|
|
2013-08-19 18:30:22 +00:00
|
|
|
void World::frameStarted (float dt, bool paused)
|
2013-03-05 13:24:29 +00:00
|
|
|
{
|
2013-08-19 18:30:22 +00:00
|
|
|
mRendering->frameStarted(dt, paused);
|
2013-03-05 13:24:29 +00:00
|
|
|
}
|
2013-04-28 12:59:15 +00:00
|
|
|
|
2014-01-24 16:49:16 +00:00
|
|
|
void World::screenshot(Ogre::Image &image, int w, int h)
|
|
|
|
{
|
|
|
|
mRendering->screenshot(image, w, h);
|
|
|
|
}
|
|
|
|
|
2013-04-28 12:59:15 +00:00
|
|
|
void World::activateDoor(const MWWorld::Ptr& door)
|
|
|
|
{
|
2014-05-14 23:58:44 +00:00
|
|
|
int state = door.getClass().getDoorState(door);
|
|
|
|
switch (state)
|
2013-04-28 12:59:15 +00:00
|
|
|
{
|
2014-05-14 23:58:44 +00:00
|
|
|
case 0:
|
2013-04-28 12:59:15 +00:00
|
|
|
if (door.getRefData().getLocalRotation().rot[2] == 0)
|
2014-05-14 23:58:44 +00:00
|
|
|
state = 1; // if closed, then open
|
2013-04-28 12:59:15 +00:00
|
|
|
else
|
2014-05-14 23:58:44 +00:00
|
|
|
state = 2; // if open, then close
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
state = 1; // if closing, then open
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
default:
|
|
|
|
state = 2; // if opening, then close
|
|
|
|
break;
|
2013-04-28 12:59:15 +00:00
|
|
|
}
|
2014-05-14 23:58:44 +00:00
|
|
|
door.getClass().setDoorState(door, state);
|
|
|
|
mDoorStates[door] = state;
|
2013-04-28 12:59:15 +00:00
|
|
|
}
|
2013-04-29 09:00:15 +00:00
|
|
|
|
2014-05-14 23:58:44 +00:00
|
|
|
void World::activateDoor(const Ptr &door, bool open)
|
2014-05-13 07:58:32 +00:00
|
|
|
{
|
2014-05-14 23:58:44 +00:00
|
|
|
int state = open ? 1 : 2;
|
|
|
|
door.getClass().setDoorState(door, state);
|
|
|
|
mDoorStates[door] = state;
|
2014-05-13 07:58:32 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 09:15:43 +00:00
|
|
|
bool World::getPlayerStandingOn (const MWWorld::Ptr& object)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr player = mPlayer->getPlayer();
|
|
|
|
if (!mPhysEngine->getCharacter("player")->getOnGround())
|
|
|
|
return false;
|
|
|
|
btVector3 from (player.getRefData().getPosition().pos[0], player.getRefData().getPosition().pos[1], player.getRefData().getPosition().pos[2]);
|
|
|
|
btVector3 to = from - btVector3(0,0,5);
|
|
|
|
std::pair<std::string, float> result = mPhysEngine->rayTest(from, to);
|
|
|
|
return result.first == object.getRefData().getBaseNode()->getName();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool World::getActorStandingOn (const MWWorld::Ptr& object)
|
|
|
|
{
|
|
|
|
return mPhysEngine->isAnyActorStandingOn(object.getRefData().getBaseNode()->getName());
|
|
|
|
}
|
2013-05-01 09:42:24 +00:00
|
|
|
|
|
|
|
float World::getWindSpeed()
|
|
|
|
{
|
|
|
|
if (isCellExterior() || isCellQuasiExterior())
|
|
|
|
return mWeatherManager->getWindSpeed();
|
|
|
|
else
|
|
|
|
return 0.f;
|
|
|
|
}
|
2013-05-11 16:38:27 +00:00
|
|
|
|
|
|
|
void World::getContainersOwnedBy (const MWWorld::Ptr& npc, std::vector<MWWorld::Ptr>& out)
|
|
|
|
{
|
|
|
|
const Scene::CellStoreCollection& collection = mWorldScene->getActiveCells();
|
|
|
|
for (Scene::CellStoreCollection::const_iterator cellIt = collection.begin(); cellIt != collection.end(); ++cellIt)
|
|
|
|
{
|
2014-02-23 20:21:27 +00:00
|
|
|
MWWorld::CellRefList<ESM::Container>& containers = (*cellIt)->get<ESM::Container>();
|
2013-05-11 16:38:27 +00:00
|
|
|
CellRefList<ESM::Container>::List& refList = containers.mList;
|
|
|
|
for (CellRefList<ESM::Container>::List::iterator container = refList.begin(); container != refList.end(); ++container)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr (&*container, *cellIt);
|
|
|
|
if (Misc::StringUtils::ciEqual(ptr.getCellRef().mOwner, npc.getCellRef().mRefID))
|
|
|
|
out.push_back(ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-05-27 00:18:36 +00:00
|
|
|
|
|
|
|
struct ListHandlesFunctor
|
|
|
|
{
|
|
|
|
std::vector<std::string> mHandles;
|
|
|
|
|
2014-01-01 21:37:52 +00:00
|
|
|
bool operator() (Ptr ptr)
|
2013-05-27 00:18:36 +00:00
|
|
|
{
|
2014-01-01 21:37:52 +00:00
|
|
|
Ogre::SceneNode* handle = ptr.getRefData().getBaseNode();
|
2013-05-27 00:18:36 +00:00
|
|
|
if (handle)
|
|
|
|
mHandles.push_back(handle->getName());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void World::getItemsOwnedBy (const MWWorld::Ptr& npc, std::vector<MWWorld::Ptr>& out)
|
|
|
|
{
|
|
|
|
const Scene::CellStoreCollection& collection = mWorldScene->getActiveCells();
|
|
|
|
for (Scene::CellStoreCollection::const_iterator cellIt = collection.begin(); cellIt != collection.end(); ++cellIt)
|
|
|
|
{
|
|
|
|
ListHandlesFunctor functor;
|
|
|
|
(*cellIt)->forEach<ListHandlesFunctor>(functor);
|
|
|
|
|
|
|
|
for (std::vector<std::string>::iterator it = functor.mHandles.begin(); it != functor.mHandles.end(); ++it)
|
2013-08-15 00:05:42 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(searchPtrViaHandle(*it).getCellRef().mOwner, npc.getCellRef().mRefID))
|
2013-05-27 00:18:36 +00:00
|
|
|
out.push_back(searchPtrViaHandle(*it));
|
|
|
|
}
|
|
|
|
}
|
2014-04-29 13:27:49 +00:00
|
|
|
|
2013-09-10 14:16:13 +00:00
|
|
|
bool World::getLOS(const MWWorld::Ptr& npc,const MWWorld::Ptr& targetNpc)
|
|
|
|
{
|
2014-03-07 08:33:46 +00:00
|
|
|
if (!targetNpc.getRefData().isEnabled() || !npc.getRefData().isEnabled())
|
|
|
|
return false; // cannot get LOS unless both NPC's are enabled
|
2013-10-28 17:01:40 +00:00
|
|
|
Ogre::Vector3 halfExt1 = mPhysEngine->getCharacter(npc.getRefData().getHandle())->getHalfExtents();
|
|
|
|
float* pos1 = npc.getRefData().getPosition().pos;
|
|
|
|
Ogre::Vector3 halfExt2 = mPhysEngine->getCharacter(targetNpc.getRefData().getHandle())->getHalfExtents();
|
|
|
|
float* pos2 = targetNpc.getRefData().getPosition().pos;
|
|
|
|
|
|
|
|
btVector3 from(pos1[0],pos1[1],pos1[2]+halfExt1.z);
|
|
|
|
btVector3 to(pos2[0],pos2[1],pos2[2]+halfExt2.z);
|
|
|
|
|
|
|
|
std::pair<std::string, float> result = mPhysEngine->rayTest(from, to,false);
|
|
|
|
if(result.first == "") return true;
|
2013-09-10 14:16:13 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-04-20 16:35:07 +00:00
|
|
|
float World::getDistToNearestRayHit(const Ogre::Vector3& from, const Ogre::Vector3& dir, float maxDist)
|
|
|
|
{
|
|
|
|
btVector3 btFrom(from.x, from.y, from.z);
|
|
|
|
btVector3 btTo = btVector3(dir.x, dir.y, dir.z);
|
|
|
|
btTo.normalize();
|
|
|
|
btTo = btFrom + btTo * maxDist;
|
|
|
|
|
|
|
|
std::pair<std::string, float> result = mPhysEngine->rayTest(btFrom, btTo, false);
|
|
|
|
|
|
|
|
if(result.second == -1) return maxDist;
|
|
|
|
else return result.second*(btTo-btFrom).length();
|
|
|
|
}
|
|
|
|
|
2013-06-27 21:11:20 +00:00
|
|
|
void World::enableActorCollision(const MWWorld::Ptr& actor, bool enable)
|
|
|
|
{
|
|
|
|
OEngine::Physic::PhysicActor *physicActor = mPhysEngine->getCharacter(actor.getRefData().getHandle());
|
2013-08-15 13:53:50 +00:00
|
|
|
|
2014-05-12 23:43:52 +00:00
|
|
|
physicActor->enableCollisionBody(enable);
|
2013-06-27 21:11:20 +00:00
|
|
|
}
|
2013-07-07 11:03:06 +00:00
|
|
|
|
|
|
|
bool World::findInteriorPosition(const std::string &name, ESM::Position &pos)
|
|
|
|
{
|
|
|
|
typedef MWWorld::CellRefList<ESM::Door>::List DoorList;
|
|
|
|
|
|
|
|
pos.rot[0] = pos.rot[1] = pos.rot[2] = 0;
|
|
|
|
pos.pos[0] = pos.pos[1] = pos.pos[2] = 0;
|
|
|
|
|
|
|
|
MWWorld::CellStore *cellStore = getInterior(name);
|
|
|
|
|
|
|
|
if (0 == cellStore) {
|
|
|
|
return false;
|
|
|
|
}
|
2014-02-23 20:21:27 +00:00
|
|
|
const DoorList &doors = cellStore->get<ESM::Door>().mList;
|
2013-07-07 11:03:06 +00:00
|
|
|
for (DoorList::const_iterator it = doors.begin(); it != doors.end(); ++it) {
|
|
|
|
if (!it->mRef.mTeleport) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
MWWorld::CellStore *source = 0;
|
|
|
|
|
|
|
|
// door to exterior
|
|
|
|
if (it->mRef.mDestCell.empty()) {
|
|
|
|
int x, y;
|
|
|
|
const float *pos = it->mRef.mDoorDest.pos;
|
|
|
|
positionToIndex(pos[0], pos[1], x, y);
|
|
|
|
source = getExterior(x, y);
|
|
|
|
}
|
|
|
|
// door to interior
|
|
|
|
else {
|
|
|
|
source = getInterior(it->mRef.mDestCell);
|
|
|
|
}
|
|
|
|
if (0 != source) {
|
|
|
|
// Find door leading to our current teleport door
|
|
|
|
// and use it destination to position inside cell.
|
2014-02-23 20:21:27 +00:00
|
|
|
const DoorList &doors = source->get<ESM::Door>().mList;
|
2013-07-07 11:03:06 +00:00
|
|
|
for (DoorList::const_iterator jt = doors.begin(); jt != doors.end(); ++jt) {
|
|
|
|
if (it->mRef.mTeleport &&
|
|
|
|
Misc::StringUtils::ciEqual(name, jt->mRef.mDestCell))
|
|
|
|
{
|
|
|
|
/// \note Using _any_ door pointed to the interior,
|
|
|
|
/// not the one pointed to current door.
|
|
|
|
pos = jt->mRef.mDoorDest;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool World::findExteriorPosition(const std::string &name, ESM::Position &pos)
|
|
|
|
{
|
|
|
|
pos.rot[0] = pos.rot[1] = pos.rot[2] = 0;
|
|
|
|
|
|
|
|
if (const ESM::Cell *ext = getExterior(name)) {
|
|
|
|
int x = ext->getGridX();
|
|
|
|
int y = ext->getGridY();
|
|
|
|
indexToPosition(x, y, pos.pos[0], pos.pos[1], true);
|
|
|
|
|
2014-01-01 17:05:28 +00:00
|
|
|
// Note: Z pos will be adjusted by adjustPosition later
|
|
|
|
pos.pos[2] = 0;
|
2013-07-07 11:03:06 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2013-07-27 07:14:55 +00:00
|
|
|
|
|
|
|
void World::enableTeleporting(bool enable)
|
|
|
|
{
|
|
|
|
mTeleportEnabled = enable;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool World::isTeleportingEnabled() const
|
|
|
|
{
|
|
|
|
return mTeleportEnabled;
|
|
|
|
}
|
|
|
|
|
2013-10-02 13:12:41 +00:00
|
|
|
void World::enableLevitation(bool enable)
|
|
|
|
{
|
|
|
|
mLevitationEnabled = enable;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool World::isLevitationEnabled() const
|
|
|
|
{
|
|
|
|
return mLevitationEnabled;
|
|
|
|
}
|
|
|
|
|
2013-08-05 21:23:39 +00:00
|
|
|
void World::setWerewolf(const MWWorld::Ptr& actor, bool werewolf)
|
|
|
|
{
|
|
|
|
MWMechanics::NpcStats& npcStats = Class::get(actor).getNpcStats(actor);
|
|
|
|
|
|
|
|
// The actor does not have to change state
|
|
|
|
if (npcStats.isWerewolf() == werewolf)
|
|
|
|
return;
|
|
|
|
|
|
|
|
npcStats.setWerewolf(werewolf);
|
|
|
|
|
2014-01-04 19:43:57 +00:00
|
|
|
// This is a bit dangerous. Equipped items other than WerewolfRobe may reference
|
|
|
|
// bones that do not even exist with the werewolf object root.
|
|
|
|
// Therefore, make sure to unequip everything at once, and only fire the change event
|
|
|
|
// (which will rebuild the animation parts) afterwards. unequipAll will do this for us.
|
2013-08-05 21:23:39 +00:00
|
|
|
MWWorld::InventoryStore& invStore = MWWorld::Class::get(actor).getInventoryStore(actor);
|
2013-08-08 11:11:34 +00:00
|
|
|
invStore.unequipAll(actor);
|
2013-08-05 21:23:39 +00:00
|
|
|
|
2013-08-08 11:11:34 +00:00
|
|
|
if(werewolf)
|
|
|
|
{
|
2013-11-21 05:27:19 +00:00
|
|
|
InventoryStore &inv = actor.getClass().getInventoryStore(actor);
|
2013-08-08 11:11:34 +00:00
|
|
|
|
2014-01-04 03:39:06 +00:00
|
|
|
inv.equip(InventoryStore::Slot_Robe, inv.ContainerStore::add("werewolfrobe", 1, actor), actor);
|
2013-08-08 11:11:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-01-04 03:39:06 +00:00
|
|
|
actor.getClass().getContainerStore(actor).remove("werewolfrobe", 1, actor);
|
2013-08-08 11:11:34 +00:00
|
|
|
}
|
|
|
|
|
2014-01-04 19:43:57 +00:00
|
|
|
// NpcAnimation::updateParts will already rebuild the animation when it detects change of Npc type.
|
|
|
|
// the following is just for reattaching the camera properly.
|
|
|
|
mRendering->rebuildPtr(actor);
|
|
|
|
|
2013-08-08 11:11:34 +00:00
|
|
|
if(actor.getRefData().getHandle() == "player")
|
2013-08-05 21:23:39 +00:00
|
|
|
{
|
|
|
|
// Update the GUI only when called on the player
|
|
|
|
MWBase::WindowManager* windowManager = MWBase::Environment::get().getWindowManager();
|
2013-08-07 18:38:03 +00:00
|
|
|
|
2013-08-05 21:23:39 +00:00
|
|
|
if (werewolf)
|
|
|
|
{
|
|
|
|
windowManager->forceHide(MWGui::GW_Inventory);
|
|
|
|
windowManager->forceHide(MWGui::GW_Magic);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
windowManager->unsetForceHide(MWGui::GW_Inventory);
|
|
|
|
windowManager->unsetForceHide(MWGui::GW_Magic);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-13 07:25:39 +00:00
|
|
|
void World::applyWerewolfAcrobatics(const Ptr &actor)
|
|
|
|
{
|
|
|
|
const Store<ESM::GameSetting> &gmst = getStore().get<ESM::GameSetting>();
|
|
|
|
MWMechanics::NpcStats &stats = Class::get(actor).getNpcStats(actor);
|
|
|
|
|
2014-01-03 02:46:30 +00:00
|
|
|
stats.getSkill(ESM::Skill::Acrobatics).setBase(gmst.find("fWerewolfAcrobatics")->getFloat());
|
2013-08-13 07:25:39 +00:00
|
|
|
}
|
|
|
|
|
2013-09-12 12:30:00 +00:00
|
|
|
bool World::getGodModeState()
|
|
|
|
{
|
|
|
|
return mGodMode;
|
|
|
|
}
|
|
|
|
|
2013-08-25 01:19:12 +00:00
|
|
|
bool World::toggleGodMode()
|
|
|
|
{
|
2013-09-15 19:48:32 +00:00
|
|
|
mGodMode = !mGodMode;
|
2013-08-30 00:25:36 +00:00
|
|
|
|
|
|
|
return mGodMode;
|
2013-08-25 01:19:12 +00:00
|
|
|
}
|
|
|
|
|
2013-09-29 07:11:57 +00:00
|
|
|
void World::loadContentFiles(const Files::Collections& fileCollections,
|
|
|
|
const std::vector<std::string>& content, ContentLoader& contentLoader)
|
|
|
|
{
|
|
|
|
std::vector<std::string>::const_iterator it(content.begin());
|
|
|
|
std::vector<std::string>::const_iterator end(content.end());
|
|
|
|
for (int idx = 0; it != end; ++it, ++idx)
|
|
|
|
{
|
|
|
|
boost::filesystem::path filename(*it);
|
|
|
|
const Files::MultiDirCollection& col = fileCollections.getCollection(filename.extension().string());
|
|
|
|
if (col.doesExist(*it))
|
|
|
|
{
|
|
|
|
contentLoader.load(col.getPath(*it), idx);
|
|
|
|
}
|
2014-04-08 18:19:09 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
std::stringstream msg;
|
|
|
|
msg << "Failed loading " << *it << ": the content file does not exist";
|
|
|
|
throw std::runtime_error(msg.str());
|
|
|
|
}
|
2013-09-29 07:11:57 +00:00
|
|
|
}
|
|
|
|
}
|
2013-11-09 09:54:51 +00:00
|
|
|
|
2013-12-26 21:06:13 +00:00
|
|
|
bool World::startSpellCast(const Ptr &actor)
|
|
|
|
{
|
|
|
|
MWMechanics::CreatureStats& stats = actor.getClass().getCreatureStats(actor);
|
|
|
|
|
|
|
|
std::string message;
|
|
|
|
bool fail = false;
|
2014-01-08 17:39:44 +00:00
|
|
|
bool isPlayer = (actor == getPlayerPtr());
|
2013-12-26 21:06:13 +00:00
|
|
|
|
|
|
|
std::string selectedSpell = stats.getSpells().getSelectedSpell();
|
|
|
|
|
|
|
|
if (!selectedSpell.empty())
|
|
|
|
{
|
|
|
|
const ESM::Spell* spell = getStore().get<ESM::Spell>().search(selectedSpell);
|
|
|
|
|
|
|
|
// Check mana
|
|
|
|
MWMechanics::DynamicStat<float> magicka = stats.getMagicka();
|
|
|
|
if (magicka.getCurrent() < spell->mData.mCost)
|
|
|
|
{
|
|
|
|
message = "#{sMagicInsufficientSP}";
|
|
|
|
fail = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this is a power, check if it was already used in the last 24h
|
2013-12-27 17:29:15 +00:00
|
|
|
if (!fail && spell->mData.mType == ESM::Spell::ST_Power)
|
2013-12-26 21:06:13 +00:00
|
|
|
{
|
2014-05-12 19:04:02 +00:00
|
|
|
if (stats.getSpells().canUsePower(spell->mId))
|
|
|
|
stats.getSpells().usePower(spell->mId);
|
2013-12-26 21:06:13 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
message = "#{sPowerAlreadyUsed}";
|
|
|
|
fail = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reduce mana
|
2014-01-14 02:34:11 +00:00
|
|
|
if (!fail)
|
|
|
|
{
|
|
|
|
magicka.setCurrent(magicka.getCurrent() - spell->mData.mCost);
|
|
|
|
stats.setMagicka(magicka);
|
|
|
|
}
|
2013-12-26 21:06:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (isPlayer && fail)
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox(message);
|
|
|
|
|
|
|
|
return !fail;
|
|
|
|
}
|
|
|
|
|
2013-11-09 06:51:46 +00:00
|
|
|
void World::castSpell(const Ptr &actor)
|
|
|
|
{
|
|
|
|
MWMechanics::CreatureStats& stats = actor.getClass().getCreatureStats(actor);
|
2013-11-17 22:15:57 +00:00
|
|
|
|
|
|
|
MWWorld::Ptr target = getFacedObject();
|
2013-11-11 22:43:28 +00:00
|
|
|
|
2013-11-09 06:51:46 +00:00
|
|
|
std::string selectedSpell = stats.getSpells().getSelectedSpell();
|
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
MWMechanics::CastSpell cast(actor, target);
|
2014-01-20 14:48:06 +00:00
|
|
|
if (!target.isEmpty())
|
|
|
|
cast.mHitPosition = Ogre::Vector3(target.getRefData().getPosition().pos);
|
2013-11-09 06:51:46 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
if (!selectedSpell.empty())
|
2013-11-09 06:51:46 +00:00
|
|
|
{
|
2013-11-17 22:15:57 +00:00
|
|
|
const ESM::Spell* spell = getStore().get<ESM::Spell>().search(selectedSpell);
|
2013-11-12 00:26:16 +00:00
|
|
|
|
2013-11-17 22:15:57 +00:00
|
|
|
cast.cast(spell);
|
2013-11-10 14:40:31 +00:00
|
|
|
}
|
2014-01-19 10:42:58 +00:00
|
|
|
else if (actor.getClass().hasInventoryStore(actor))
|
2013-11-10 14:40:31 +00:00
|
|
|
{
|
2014-01-19 10:42:58 +00:00
|
|
|
MWWorld::InventoryStore& inv = actor.getClass().getInventoryStore(actor);
|
|
|
|
if (inv.getSelectedEnchantItem() != inv.end())
|
|
|
|
cast.cast(*inv.getSelectedEnchantItem());
|
2013-11-09 06:51:46 +00:00
|
|
|
}
|
2013-11-13 13:02:15 +00:00
|
|
|
}
|
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
void World::launchProjectile (MWWorld::Ptr actor, MWWorld::Ptr projectile,
|
|
|
|
const Ogre::Vector3& worldPos, const Ogre::Quaternion& orient, MWWorld::Ptr bow, float speed)
|
|
|
|
{
|
|
|
|
ProjectileState state;
|
2014-05-14 07:48:16 +00:00
|
|
|
state.mActorId = actor.getClass().getCreatureStats(actor).getActorId();
|
2014-03-07 05:11:00 +00:00
|
|
|
state.mBow = bow;
|
2014-03-08 04:51:47 +00:00
|
|
|
state.mVelocity = orient.yAxis() * speed;
|
2014-03-07 05:11:00 +00:00
|
|
|
|
|
|
|
MWWorld::ManualRef ref(getStore(), projectile.getCellRef().mRefID);
|
|
|
|
|
|
|
|
ESM::Position pos;
|
|
|
|
pos.pos[0] = worldPos.x;
|
|
|
|
pos.pos[1] = worldPos.y;
|
|
|
|
pos.pos[2] = worldPos.z;
|
|
|
|
|
|
|
|
// Do NOT copy actor rotation! actors use a different rotation order, and this will not produce the same facing direction.
|
|
|
|
Ogre::Matrix3 mat;
|
|
|
|
orient.ToRotationMatrix(mat);
|
|
|
|
Ogre::Radian xr,yr,zr;
|
|
|
|
mat.ToEulerAnglesXYZ(xr, yr, zr);
|
|
|
|
pos.rot[0] = -xr.valueRadians();
|
|
|
|
pos.rot[1] = -yr.valueRadians();
|
|
|
|
pos.rot[2] = -zr.valueRadians();
|
|
|
|
|
|
|
|
MWWorld::Ptr ptr = copyObjectToCell(ref.getPtr(), actor.getCell(), pos, false);
|
|
|
|
ptr.getRefData().setCount(1);
|
|
|
|
|
|
|
|
mProjectiles[ptr] = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
void World::launchMagicBolt (const std::string& id, bool stack, const ESM::EffectList& effects,
|
2013-11-13 13:02:15 +00:00
|
|
|
const MWWorld::Ptr& actor, const std::string& sourceName)
|
|
|
|
{
|
|
|
|
std::string projectileModel;
|
|
|
|
std::string sound;
|
2013-11-28 16:31:17 +00:00
|
|
|
float speed = 0;
|
2013-11-13 13:02:15 +00:00
|
|
|
for (std::vector<ESM::ENAMstruct>::const_iterator iter (effects.mList.begin());
|
|
|
|
iter!=effects.mList.end(); ++iter)
|
|
|
|
{
|
|
|
|
if (iter->mRange != ESM::RT_Target)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const ESM::MagicEffect *magicEffect = getStore().get<ESM::MagicEffect>().find (
|
|
|
|
iter->mEffectID);
|
|
|
|
|
|
|
|
projectileModel = magicEffect->mBolt;
|
2013-11-29 09:39:37 +00:00
|
|
|
if (projectileModel.empty())
|
|
|
|
projectileModel = "VFX_DefaultBolt";
|
2013-11-13 13:02:15 +00:00
|
|
|
|
|
|
|
static const std::string schools[] = {
|
|
|
|
"alteration", "conjuration", "destruction", "illusion", "mysticism", "restoration"
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!magicEffect->mBoltSound.empty())
|
|
|
|
sound = magicEffect->mBoltSound;
|
|
|
|
else
|
|
|
|
sound = schools[magicEffect->mData.mSchool] + " bolt";
|
|
|
|
|
2013-11-28 16:31:17 +00:00
|
|
|
speed = magicEffect->mData.mSpeed;
|
2013-11-13 13:02:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (projectileModel.empty())
|
|
|
|
return;
|
|
|
|
|
2013-11-28 16:31:17 +00:00
|
|
|
// Spawn at 0.75 * ActorHeight
|
|
|
|
float height = mPhysEngine->getCharacter(actor.getRefData().getHandle())->getHalfExtents().z * 2 * 0.75;
|
2013-11-14 13:45:08 +00:00
|
|
|
|
2013-11-13 13:02:15 +00:00
|
|
|
MWWorld::ManualRef ref(getStore(), projectileModel);
|
|
|
|
ESM::Position pos;
|
|
|
|
pos.pos[0] = actor.getRefData().getPosition().pos[0];
|
|
|
|
pos.pos[1] = actor.getRefData().getPosition().pos[1];
|
2013-11-28 16:31:17 +00:00
|
|
|
pos.pos[2] = actor.getRefData().getPosition().pos[2] + height;
|
2014-03-07 05:11:00 +00:00
|
|
|
|
|
|
|
// Do NOT copy rotation directly! actors use a different rotation order, and this will not produce the same facing direction.
|
|
|
|
Ogre::Quaternion orient = Ogre::Quaternion(Ogre::Radian(actor.getRefData().getPosition().rot[2]), Ogre::Vector3::NEGATIVE_UNIT_Z) *
|
|
|
|
Ogre::Quaternion(Ogre::Radian(actor.getRefData().getPosition().rot[0]), Ogre::Vector3::NEGATIVE_UNIT_X);
|
|
|
|
Ogre::Matrix3 mat;
|
|
|
|
orient.ToRotationMatrix(mat);
|
|
|
|
Ogre::Radian xr,yr,zr;
|
|
|
|
mat.ToEulerAnglesXYZ(xr, yr, zr);
|
|
|
|
pos.rot[0] = -xr.valueRadians();
|
|
|
|
pos.rot[1] = -yr.valueRadians();
|
|
|
|
pos.rot[2] = -zr.valueRadians();
|
|
|
|
|
2013-11-13 13:02:15 +00:00
|
|
|
ref.getPtr().getCellRef().mPos = pos;
|
2014-03-07 05:11:00 +00:00
|
|
|
CellStore* cell = actor.getCell();
|
|
|
|
MWWorld::Ptr ptr = copyObjectToCell(ref.getPtr(), cell, pos);
|
2013-11-13 13:02:15 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
MagicBoltState state;
|
2013-11-13 13:02:15 +00:00
|
|
|
state.mSourceName = sourceName;
|
|
|
|
state.mId = id;
|
2014-05-14 07:48:16 +00:00
|
|
|
state.mActorId = actor.getClass().getCreatureStats(actor).getActorId();
|
2013-11-28 16:31:17 +00:00
|
|
|
state.mSpeed = speed;
|
|
|
|
state.mStack = stack;
|
2013-11-13 13:02:15 +00:00
|
|
|
|
2014-01-20 12:00:43 +00:00
|
|
|
// Only interested in "on target" effects
|
|
|
|
for (std::vector<ESM::ENAMstruct>::const_iterator iter (effects.mList.begin());
|
|
|
|
iter!=effects.mList.end(); ++iter)
|
|
|
|
{
|
|
|
|
if (iter->mRange == ESM::RT_Target)
|
|
|
|
state.mEffects.mList.push_back(*iter);
|
|
|
|
}
|
|
|
|
|
2013-11-13 13:02:15 +00:00
|
|
|
MWBase::SoundManager *sndMgr = MWBase::Environment::get().getSoundManager();
|
2014-01-20 10:50:52 +00:00
|
|
|
sndMgr->playSound3D(ptr, sound, 1.0f, 1.0f, MWBase::SoundManager::Play_TypeSfx, MWBase::SoundManager::Play_Loop);
|
2013-11-13 13:02:15 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
mMagicBolts[ptr] = state;
|
2013-11-09 06:51:46 +00:00
|
|
|
}
|
|
|
|
|
2013-11-13 13:02:15 +00:00
|
|
|
void World::moveProjectiles(float duration)
|
|
|
|
{
|
2013-11-28 16:31:17 +00:00
|
|
|
std::map<std::string, ProjectileState> moved;
|
2013-11-13 13:02:15 +00:00
|
|
|
for (std::map<MWWorld::Ptr, ProjectileState>::iterator it = mProjectiles.begin(); it != mProjectiles.end();)
|
|
|
|
{
|
|
|
|
if (!mWorldScene->isCellActive(*it->first.getCell()))
|
|
|
|
{
|
2014-01-24 12:22:30 +00:00
|
|
|
deleteObject(it->first);
|
2013-11-13 13:02:15 +00:00
|
|
|
mProjectiles.erase(it++);
|
|
|
|
continue;
|
|
|
|
}
|
2013-11-28 16:31:17 +00:00
|
|
|
|
|
|
|
MWWorld::Ptr ptr = it->first;
|
|
|
|
|
2014-03-08 04:51:47 +00:00
|
|
|
// gravity constant - must be way lower than the gravity affecting actors, since we're not
|
|
|
|
// simulating aerodynamics at all
|
|
|
|
it->second.mVelocity -= Ogre::Vector3(0, 0, 627.2f * 0.1f) * duration;
|
2013-11-28 16:31:17 +00:00
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
Ogre::Vector3 pos(ptr.getRefData().getPosition().pos);
|
2014-03-08 04:51:47 +00:00
|
|
|
Ogre::Vector3 newPos = pos + it->second.mVelocity * duration;
|
|
|
|
|
|
|
|
Ogre::Quaternion orient = Ogre::Vector3::UNIT_Y.getRotationTo(it->second.mVelocity);
|
|
|
|
Ogre::Matrix3 mat;
|
|
|
|
orient.ToRotationMatrix(mat);
|
|
|
|
Ogre::Radian xr,yr,zr;
|
|
|
|
mat.ToEulerAnglesXYZ(xr, yr, zr);
|
|
|
|
rotateObject(ptr, -xr.valueDegrees(), -yr.valueDegrees(), -zr.valueDegrees());
|
2014-03-07 05:11:00 +00:00
|
|
|
|
|
|
|
// Check for impact
|
|
|
|
btVector3 from(pos.x, pos.y, pos.z);
|
|
|
|
btVector3 to(newPos.x, newPos.y, newPos.z);
|
|
|
|
std::vector<std::pair<float, std::string> > collisions = mPhysEngine->rayTest2(from, to);
|
|
|
|
bool hit=false;
|
|
|
|
|
|
|
|
// HACK: query against the shape as well, since the ray does not take the volume into account
|
|
|
|
// really, this should be a convex cast, but the whole physics system needs a rewrite
|
|
|
|
std::vector<std::string> col2 = mPhysEngine->getCollisions(ptr.getRefData().getHandle());
|
|
|
|
for (std::vector<std::string>::iterator ci = col2.begin(); ci != col2.end(); ++ci)
|
|
|
|
collisions.push_back(std::make_pair(0.f,*ci));
|
|
|
|
|
|
|
|
for (std::vector<std::pair<float, std::string> >::iterator cIt = collisions.begin(); cIt != collisions.end() && !hit; ++cIt)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr obstacle = searchPtrViaHandle(cIt->second);
|
|
|
|
if (obstacle == ptr)
|
|
|
|
continue;
|
2014-01-10 21:51:42 +00:00
|
|
|
|
2014-05-14 07:48:16 +00:00
|
|
|
MWWorld::Ptr caster = searchPtrViaActorId(it->second.mActorId);
|
2014-03-07 05:11:00 +00:00
|
|
|
|
|
|
|
// Arrow intersects with player immediately after shooting :/
|
|
|
|
if (obstacle == caster)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (caster.isEmpty())
|
|
|
|
caster = obstacle;
|
|
|
|
|
|
|
|
if (obstacle.isEmpty())
|
|
|
|
{
|
|
|
|
// Terrain
|
|
|
|
}
|
|
|
|
else if (obstacle.getClass().isActor())
|
|
|
|
{
|
2014-03-08 04:51:47 +00:00
|
|
|
MWMechanics::projectileHit(caster, obstacle, it->second.mBow, ptr, pos + (newPos - pos) * cIt->first);
|
2014-03-07 05:11:00 +00:00
|
|
|
}
|
|
|
|
hit = true;
|
|
|
|
}
|
|
|
|
if (hit)
|
|
|
|
{
|
2014-03-08 04:51:47 +00:00
|
|
|
deleteObject(ptr);
|
2014-03-07 05:11:00 +00:00
|
|
|
mProjectiles.erase(it++);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string handle = ptr.getRefData().getHandle();
|
|
|
|
|
|
|
|
moveObject(ptr, newPos.x, newPos.y, newPos.z);
|
|
|
|
|
|
|
|
// HACK: Re-fetch Ptrs if necessary, since the cell might have changed
|
|
|
|
if (!ptr.getRefData().getCount())
|
|
|
|
{
|
|
|
|
moved[handle] = it->second;
|
|
|
|
mProjectiles.erase(it++);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
// HACK: Re-fetch Ptrs if necessary, since the cell might have changed
|
|
|
|
for (std::map<std::string, ProjectileState>::iterator it = moved.begin(); it != moved.end(); ++it)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr newPtr = searchPtrViaHandle(it->first);
|
|
|
|
if (newPtr.isEmpty()) // The projectile went into an inactive cell and was deleted
|
|
|
|
continue;
|
|
|
|
mProjectiles[getPtrViaHandle(it->first)] = it->second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void World::moveMagicBolts(float duration)
|
|
|
|
{
|
|
|
|
std::map<std::string, MagicBoltState> moved;
|
|
|
|
for (std::map<MWWorld::Ptr, MagicBoltState>::iterator it = mMagicBolts.begin(); it != mMagicBolts.end();)
|
|
|
|
{
|
|
|
|
if (!mWorldScene->isCellActive(*it->first.getCell()))
|
|
|
|
{
|
|
|
|
deleteObject(it->first);
|
|
|
|
mMagicBolts.erase(it++);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
MWWorld::Ptr ptr = it->first;
|
|
|
|
|
|
|
|
Ogre::Vector3 rot(ptr.getRefData().getPosition().rot);
|
|
|
|
|
|
|
|
Ogre::Quaternion orient = ptr.getRefData().getBaseNode()->getOrientation();
|
2014-01-10 21:51:42 +00:00
|
|
|
static float fTargetSpellMaxSpeed = getStore().get<ESM::GameSetting>().find("fTargetSpellMaxSpeed")->getFloat();
|
|
|
|
float speed = fTargetSpellMaxSpeed * it->second.mSpeed;
|
2013-11-28 16:31:17 +00:00
|
|
|
|
|
|
|
Ogre::Vector3 direction = orient.yAxis();
|
|
|
|
direction.normalise();
|
|
|
|
Ogre::Vector3 pos(ptr.getRefData().getPosition().pos);
|
|
|
|
Ogre::Vector3 newPos = pos + direction * duration * speed;
|
|
|
|
|
|
|
|
// Check for impact
|
|
|
|
btVector3 from(pos.x, pos.y, pos.z);
|
|
|
|
btVector3 to(newPos.x, newPos.y, newPos.z);
|
|
|
|
std::vector<std::pair<float, std::string> > collisions = mPhysEngine->rayTest2(from, to);
|
2013-11-28 20:49:15 +00:00
|
|
|
bool explode = false;
|
|
|
|
for (std::vector<std::pair<float, std::string> >::iterator cIt = collisions.begin(); cIt != collisions.end() && !explode; ++cIt)
|
2013-11-28 16:31:17 +00:00
|
|
|
{
|
|
|
|
MWWorld::Ptr obstacle = searchPtrViaHandle(cIt->second);
|
|
|
|
if (obstacle == ptr)
|
|
|
|
continue;
|
2013-11-28 20:49:15 +00:00
|
|
|
|
2014-05-14 07:48:16 +00:00
|
|
|
MWWorld::Ptr caster = searchPtrViaActorId(it->second.mActorId);
|
2014-01-20 12:43:21 +00:00
|
|
|
if (caster.isEmpty())
|
|
|
|
caster = obstacle;
|
|
|
|
|
|
|
|
if (obstacle.isEmpty())
|
|
|
|
{
|
|
|
|
// Terrain
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MWMechanics::CastSpell cast(caster, obstacle);
|
2014-01-20 14:48:06 +00:00
|
|
|
cast.mHitPosition = pos;
|
2014-01-20 12:43:21 +00:00
|
|
|
cast.mId = it->second.mId;
|
|
|
|
cast.mSourceName = it->second.mSourceName;
|
|
|
|
cast.mStack = it->second.mStack;
|
2014-01-20 14:48:06 +00:00
|
|
|
cast.inflict(obstacle, caster, it->second.mEffects, ESM::RT_Target, false, true);
|
2014-01-20 12:43:21 +00:00
|
|
|
}
|
|
|
|
|
2013-11-28 20:49:15 +00:00
|
|
|
explode = true;
|
2014-01-20 12:00:43 +00:00
|
|
|
}
|
2013-11-28 20:49:15 +00:00
|
|
|
|
2014-01-20 12:00:43 +00:00
|
|
|
if (explode)
|
|
|
|
{
|
2014-05-14 07:48:16 +00:00
|
|
|
MWWorld::Ptr caster = searchPtrViaActorId(it->second.mActorId);
|
2014-01-20 14:48:06 +00:00
|
|
|
explodeSpell(Ogre::Vector3(ptr.getRefData().getPosition().pos), ptr, it->second.mEffects, caster, it->second.mId, it->second.mSourceName);
|
2013-11-28 16:31:17 +00:00
|
|
|
|
|
|
|
deleteObject(ptr);
|
2014-03-07 05:11:00 +00:00
|
|
|
mMagicBolts.erase(it++);
|
2013-11-28 16:31:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string handle = ptr.getRefData().getHandle();
|
|
|
|
|
|
|
|
moveObject(ptr, newPos.x, newPos.y, newPos.z);
|
|
|
|
|
|
|
|
// HACK: Re-fetch Ptrs if necessary, since the cell might have changed
|
|
|
|
if (!ptr.getRefData().getCount())
|
|
|
|
{
|
|
|
|
moved[handle] = it->second;
|
2014-03-07 05:11:00 +00:00
|
|
|
mMagicBolts.erase(it++);
|
2013-11-28 16:31:17 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
// HACK: Re-fetch Ptrs if necessary, since the cell might have changed
|
2014-03-07 05:11:00 +00:00
|
|
|
for (std::map<std::string, MagicBoltState>::iterator it = moved.begin(); it != moved.end(); ++it)
|
2013-11-28 16:31:17 +00:00
|
|
|
{
|
|
|
|
MWWorld::Ptr newPtr = searchPtrViaHandle(it->first);
|
|
|
|
if (newPtr.isEmpty()) // The projectile went into an inactive cell and was deleted
|
|
|
|
continue;
|
2014-03-07 05:11:00 +00:00
|
|
|
mMagicBolts[getPtrViaHandle(it->first)] = it->second;
|
2013-11-28 16:31:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void World::objectLeftActiveCell(Ptr object, Ptr movedPtr)
|
|
|
|
{
|
|
|
|
// For now, projectiles moved to an inactive cell are just deleted, because there's no reliable way to hold on to the meta information
|
2014-03-07 05:11:00 +00:00
|
|
|
if (mMagicBolts.find(object) != mMagicBolts.end())
|
|
|
|
deleteObject(movedPtr);
|
2013-11-28 16:31:17 +00:00
|
|
|
if (mProjectiles.find(object) != mProjectiles.end())
|
|
|
|
deleteObject(movedPtr);
|
2013-11-13 13:02:15 +00:00
|
|
|
}
|
2013-12-08 22:36:37 +00:00
|
|
|
|
2013-11-25 12:00:05 +00:00
|
|
|
const std::vector<std::string>& World::getContentFiles() const
|
|
|
|
{
|
|
|
|
return mContentFiles;
|
|
|
|
}
|
2014-01-05 23:02:03 +00:00
|
|
|
|
2013-12-08 22:36:37 +00:00
|
|
|
void World::breakInvisibility(const Ptr &actor)
|
|
|
|
{
|
|
|
|
actor.getClass().getCreatureStats(actor).getActiveSpells().purgeEffect(ESM::MagicEffect::Invisibility);
|
2014-01-19 10:42:58 +00:00
|
|
|
if (actor.getClass().hasInventoryStore(actor))
|
2014-01-17 15:31:27 +00:00
|
|
|
actor.getClass().getInventoryStore(actor).purgeEffect(ESM::MagicEffect::Invisibility);
|
2013-12-08 22:36:37 +00:00
|
|
|
}
|
2013-12-10 22:48:49 +00:00
|
|
|
|
2013-12-30 16:53:02 +00:00
|
|
|
bool World::isDark() const
|
2013-12-10 22:48:49 +00:00
|
|
|
{
|
2014-04-28 16:33:42 +00:00
|
|
|
MWWorld::CellStore* cell = mPlayer->getPlayer().getCell();
|
|
|
|
if (cell->isExterior())
|
|
|
|
return mWeatherManager->isDark();
|
|
|
|
else
|
|
|
|
{
|
|
|
|
uint32_t ambient = cell->getCell()->mAmbi.mAmbient;
|
|
|
|
int ambientTotal = (ambient & 0xff)
|
|
|
|
+ ((ambient>>8) & 0xff)
|
|
|
|
+ ((ambient>>16) & 0xff);
|
|
|
|
return !(cell->getCell()->mData.mFlags & ESM::Cell::NoSleep) && ambientTotal <= 201;
|
|
|
|
}
|
2013-12-31 17:35:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool World::findInteriorPositionInWorldSpace(MWWorld::CellStore* cell, Ogre::Vector3& result)
|
|
|
|
{
|
2014-01-09 00:49:58 +00:00
|
|
|
if (cell->isExterior())
|
|
|
|
return false;
|
2014-02-23 20:21:27 +00:00
|
|
|
MWWorld::CellRefList<ESM::Door>& doors = cell->get<ESM::Door>();
|
2013-12-31 17:35:46 +00:00
|
|
|
CellRefList<ESM::Door>::List& refList = doors.mList;
|
|
|
|
|
|
|
|
// Check if any door in the cell leads to an exterior directly
|
|
|
|
for (CellRefList<ESM::Door>::List::iterator it = refList.begin(); it != refList.end(); ++it)
|
|
|
|
{
|
|
|
|
MWWorld::LiveCellRef<ESM::Door>& ref = *it;
|
|
|
|
if (ref.mRef.mTeleport && ref.mRef.mDestCell.empty())
|
|
|
|
{
|
|
|
|
ESM::Position pos = ref.mRef.mDoorDest;
|
2014-01-01 21:46:10 +00:00
|
|
|
result = Ogre::Vector3(pos.pos);
|
2013-12-31 17:35:46 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// No luck :(
|
|
|
|
return false;
|
2013-12-10 22:48:49 +00:00
|
|
|
}
|
2014-01-01 01:22:11 +00:00
|
|
|
|
|
|
|
void World::teleportToClosestMarker (const MWWorld::Ptr& ptr,
|
2014-01-09 00:49:58 +00:00
|
|
|
const std::string& id)
|
2014-01-01 01:22:11 +00:00
|
|
|
{
|
2014-01-09 00:49:58 +00:00
|
|
|
Ogre::Vector3 worldPos;
|
|
|
|
if (!findInteriorPositionInWorldSpace(ptr.getCell(), worldPos))
|
|
|
|
worldPos = mPlayer->getLastKnownExteriorPosition();
|
|
|
|
|
2014-01-01 01:22:11 +00:00
|
|
|
MWWorld::Ptr closestMarker;
|
|
|
|
float closestDistance = FLT_MAX;
|
|
|
|
|
|
|
|
std::vector<MWWorld::Ptr> markers;
|
|
|
|
mCells.getExteriorPtrs(id, markers);
|
|
|
|
|
|
|
|
for (std::vector<MWWorld::Ptr>::iterator it = markers.begin(); it != markers.end(); ++it)
|
|
|
|
{
|
|
|
|
ESM::Position pos = it->getRefData().getPosition();
|
2014-01-01 21:46:10 +00:00
|
|
|
Ogre::Vector3 markerPos = Ogre::Vector3(pos.pos);
|
2014-01-01 01:22:11 +00:00
|
|
|
float distance = worldPos.squaredDistance(markerPos);
|
|
|
|
if (distance < closestDistance)
|
|
|
|
{
|
|
|
|
closestDistance = distance;
|
|
|
|
closestMarker = *it;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
MWWorld::ActionTeleport action("", closestMarker.getRefData().getPosition());
|
|
|
|
action.execute(ptr);
|
2013-12-10 22:48:49 +00:00
|
|
|
}
|
2014-01-01 13:57:14 +00:00
|
|
|
|
2013-12-29 11:47:44 +00:00
|
|
|
void World::updateWeather(float duration)
|
|
|
|
{
|
2013-12-31 19:40:23 +00:00
|
|
|
if (mPlayer->wasTeleported())
|
|
|
|
{
|
|
|
|
mPlayer->setTeleported(false);
|
|
|
|
mWeatherManager->switchToNextWeather(true);
|
|
|
|
}
|
2013-12-31 20:18:10 +00:00
|
|
|
|
|
|
|
mWeatherManager->update(duration);
|
2013-12-29 11:47:44 +00:00
|
|
|
}
|
2014-01-01 21:37:52 +00:00
|
|
|
|
|
|
|
struct AddDetectedReference
|
|
|
|
{
|
|
|
|
AddDetectedReference(std::vector<Ptr>& out, Ptr detector, World::DetectionType type, float squaredDist)
|
|
|
|
: mOut(out), mDetector(detector), mType(type), mSquaredDist(squaredDist)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<Ptr>& mOut;
|
|
|
|
Ptr mDetector;
|
|
|
|
float mSquaredDist;
|
|
|
|
World::DetectionType mType;
|
|
|
|
bool operator() (MWWorld::Ptr ptr)
|
|
|
|
{
|
|
|
|
if (Ogre::Vector3(ptr.getRefData().getPosition().pos).squaredDistance(
|
|
|
|
Ogre::Vector3(mDetector.getRefData().getPosition().pos)) >= mSquaredDist)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!ptr.getRefData().isEnabled())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Consider references inside containers as well
|
|
|
|
if (ptr.getClass().isActor() || ptr.getClass().getTypeName() == typeid(ESM::Container).name())
|
|
|
|
{
|
|
|
|
MWWorld::ContainerStore& store = ptr.getClass().getContainerStore(ptr);
|
|
|
|
{
|
|
|
|
for (MWWorld::ContainerStoreIterator it = store.begin(); it != store.end(); ++it)
|
|
|
|
{
|
|
|
|
if (needToAdd(*it))
|
|
|
|
{
|
|
|
|
mOut.push_back(ptr);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (needToAdd(ptr))
|
|
|
|
mOut.push_back(ptr);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool needToAdd (MWWorld::Ptr ptr)
|
|
|
|
{
|
|
|
|
if (mType == World::Detect_Creature && ptr.getClass().getTypeName() != typeid(ESM::Creature).name())
|
|
|
|
return false;
|
|
|
|
if (mType == World::Detect_Key && !ptr.getClass().isKey(ptr))
|
|
|
|
return false;
|
|
|
|
if (mType == World::Detect_Enchantment && ptr.getClass().getEnchantment(ptr).empty())
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void World::listDetectedReferences(const Ptr &ptr, std::vector<Ptr> &out, DetectionType type)
|
|
|
|
{
|
|
|
|
const MWMechanics::MagicEffects& effects = ptr.getClass().getCreatureStats(ptr).getMagicEffects();
|
|
|
|
float dist=0;
|
|
|
|
if (type == World::Detect_Creature)
|
2014-01-02 20:49:04 +00:00
|
|
|
dist = effects.get(ESM::MagicEffect::DetectAnimal).mMagnitude;
|
2014-01-01 21:37:52 +00:00
|
|
|
else if (type == World::Detect_Key)
|
2014-01-02 20:49:04 +00:00
|
|
|
dist = effects.get(ESM::MagicEffect::DetectKey).mMagnitude;
|
2014-01-01 21:37:52 +00:00
|
|
|
else if (type == World::Detect_Enchantment)
|
2014-01-02 20:49:04 +00:00
|
|
|
dist = effects.get(ESM::MagicEffect::DetectEnchantment).mMagnitude;
|
2014-01-01 21:37:52 +00:00
|
|
|
|
|
|
|
if (!dist)
|
|
|
|
return;
|
|
|
|
|
2014-01-01 22:34:18 +00:00
|
|
|
dist = feetToGameUnits(dist);
|
2014-01-01 21:37:52 +00:00
|
|
|
|
|
|
|
AddDetectedReference functor (out, ptr, type, dist*dist);
|
|
|
|
|
|
|
|
const Scene::CellStoreCollection& active = mWorldScene->getActiveCells();
|
|
|
|
for (Scene::CellStoreCollection::const_iterator it = active.begin(); it != active.end(); ++it)
|
|
|
|
{
|
|
|
|
MWWorld::CellStore* cellStore = *it;
|
|
|
|
cellStore->forEach(functor);
|
|
|
|
}
|
|
|
|
}
|
2014-01-01 22:34:18 +00:00
|
|
|
|
|
|
|
float World::feetToGameUnits(float feet)
|
|
|
|
{
|
|
|
|
// Looks like there is no GMST for this. This factor was determined in experiments
|
|
|
|
// with the Telekinesis effect.
|
|
|
|
return feet * 22;
|
|
|
|
}
|
2014-01-08 17:39:44 +00:00
|
|
|
|
|
|
|
MWWorld::Ptr World::getPlayerPtr()
|
|
|
|
{
|
|
|
|
return mPlayer->getPlayer();
|
|
|
|
}
|
2014-01-08 23:40:25 +00:00
|
|
|
|
|
|
|
void World::updateDialogueGlobals()
|
|
|
|
{
|
|
|
|
MWWorld::Ptr player = getPlayerPtr();
|
|
|
|
int bounty = player.getClass().getNpcStats(player).getBounty();
|
|
|
|
int playerGold = player.getClass().getContainerStore(player).count(ContainerStore::sGoldId);
|
|
|
|
|
|
|
|
float fCrimeGoldDiscountMult = getStore().get<ESM::GameSetting>().find("fCrimeGoldDiscountMult")->getFloat();
|
|
|
|
float fCrimeGoldTurnInMult = getStore().get<ESM::GameSetting>().find("fCrimeGoldTurnInMult")->getFloat();
|
|
|
|
|
|
|
|
int discount = bounty*fCrimeGoldDiscountMult;
|
|
|
|
int turnIn = bounty * fCrimeGoldTurnInMult;
|
|
|
|
|
2014-01-24 17:21:52 +00:00
|
|
|
mGlobalVariables["pchascrimegold"].setInteger((bounty <= playerGold) ? 1 : 0);
|
2014-01-08 23:40:25 +00:00
|
|
|
|
2014-01-24 17:21:52 +00:00
|
|
|
mGlobalVariables["pchasgolddiscount"].setInteger((discount <= playerGold) ? 1 : 0);
|
|
|
|
mGlobalVariables["crimegolddiscount"].setInteger(discount);
|
2014-01-08 23:40:25 +00:00
|
|
|
|
2014-01-24 17:21:52 +00:00
|
|
|
mGlobalVariables["crimegoldturnin"].setInteger(turnIn);
|
|
|
|
mGlobalVariables["pchasturnin"].setInteger((turnIn <= playerGold) ? 1 : 0);
|
2014-01-08 23:40:25 +00:00
|
|
|
}
|
2014-01-11 02:29:41 +00:00
|
|
|
|
|
|
|
void World::confiscateStolenItems(const Ptr &ptr)
|
|
|
|
{
|
|
|
|
Ogre::Vector3 playerPos;
|
|
|
|
if (!findInteriorPositionInWorldSpace(ptr.getCell(), playerPos))
|
|
|
|
playerPos = mPlayer->getLastKnownExteriorPosition();
|
|
|
|
|
|
|
|
MWWorld::Ptr closestChest;
|
|
|
|
float closestDistance = FLT_MAX;
|
|
|
|
|
2014-04-23 17:20:43 +00:00
|
|
|
//Find closest stolen_goods chest
|
2014-01-11 02:29:41 +00:00
|
|
|
std::vector<MWWorld::Ptr> chests;
|
|
|
|
mCells.getInteriorPtrs("stolen_goods", chests);
|
|
|
|
|
|
|
|
Ogre::Vector3 chestPos;
|
|
|
|
for (std::vector<MWWorld::Ptr>::iterator it = chests.begin(); it != chests.end(); ++it)
|
|
|
|
{
|
|
|
|
if (!findInteriorPositionInWorldSpace(it->getCell(), chestPos))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
float distance = playerPos.squaredDistance(chestPos);
|
|
|
|
if (distance < closestDistance)
|
|
|
|
{
|
|
|
|
closestDistance = distance;
|
|
|
|
closestChest = *it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-23 17:20:43 +00:00
|
|
|
if (!closestChest.isEmpty()) //Found a close chest
|
2014-01-11 02:29:41 +00:00
|
|
|
{
|
|
|
|
ContainerStore& store = ptr.getClass().getContainerStore(ptr);
|
2014-04-23 17:20:43 +00:00
|
|
|
for (ContainerStoreIterator it = store.begin(); it != store.end(); ++it) //Move all stolen stuff into chest
|
2014-01-11 02:29:41 +00:00
|
|
|
{
|
2014-04-23 17:20:43 +00:00
|
|
|
if (!it->getCellRef().mOwner.empty() && it->getCellRef().mOwner != "player") //Not owned by no one/player?
|
2014-01-11 02:29:41 +00:00
|
|
|
{
|
|
|
|
closestChest.getClass().getContainerStore(closestChest).add(*it, it->getRefData().getCount(), closestChest);
|
|
|
|
store.remove(*it, it->getRefData().getCount(), ptr);
|
|
|
|
}
|
|
|
|
}
|
2014-04-23 17:20:43 +00:00
|
|
|
closestChest.getCellRef().mLockLevel = abs(closestChest.getCellRef().mLockLevel);
|
2014-01-11 02:29:41 +00:00
|
|
|
}
|
|
|
|
}
|
2014-01-11 05:47:58 +00:00
|
|
|
|
|
|
|
void World::goToJail()
|
|
|
|
{
|
|
|
|
if (!mGoToJail)
|
|
|
|
{
|
|
|
|
// Save for next update, since the player should be able to read the dialog text first
|
|
|
|
mGoToJail = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mGoToJail = false;
|
|
|
|
|
|
|
|
MWWorld::Ptr player = getPlayerPtr();
|
|
|
|
teleportToClosestMarker(player, "prisonmarker");
|
|
|
|
int bounty = player.getClass().getNpcStats(player).getBounty();
|
|
|
|
player.getClass().getNpcStats(player).setBounty(0);
|
|
|
|
confiscateStolenItems(player);
|
|
|
|
|
|
|
|
int iDaysinPrisonMod = getStore().get<ESM::GameSetting>().find("iDaysinPrisonMod")->getInt();
|
|
|
|
int days = std::max(1, bounty / iDaysinPrisonMod);
|
|
|
|
|
|
|
|
advanceTime(days * 24);
|
|
|
|
|
|
|
|
std::set<int> skills;
|
|
|
|
for (int day=0; day<days; ++day)
|
|
|
|
{
|
|
|
|
int skill = std::rand()/ (static_cast<double> (RAND_MAX) + 1) * ESM::Skill::Length;
|
|
|
|
skills.insert(skill);
|
|
|
|
|
|
|
|
MWMechanics::SkillValue& value = player.getClass().getNpcStats(player).getSkill(skill);
|
|
|
|
if (skill == ESM::Skill::Security || skill == ESM::Skill::Sneak)
|
|
|
|
value.setBase(std::min(100, value.getBase()+1));
|
|
|
|
else
|
|
|
|
value.setBase(value.getBase()-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
const Store<ESM::GameSetting>& gmst = getStore().get<ESM::GameSetting>();
|
|
|
|
|
|
|
|
std::string message;
|
|
|
|
if (days == 1)
|
|
|
|
message = gmst.find("sNotifyMessage42")->getString();
|
|
|
|
else
|
|
|
|
message = gmst.find("sNotifyMessage43")->getString();
|
|
|
|
|
|
|
|
std::stringstream dayStr;
|
|
|
|
dayStr << days;
|
|
|
|
if (message.find("%d") != std::string::npos)
|
|
|
|
message.replace(message.find("%d"), 2, dayStr.str());
|
|
|
|
|
|
|
|
for (std::set<int>::iterator it = skills.begin(); it != skills.end(); ++it)
|
|
|
|
{
|
|
|
|
std::string skillName = gmst.find(ESM::Skill::sSkillNameIds[*it])->getString();
|
|
|
|
std::stringstream skillValue;
|
|
|
|
skillValue << player.getClass().getNpcStats(player).getSkill(*it).getBase();
|
|
|
|
std::string skillMsg = gmst.find("sNotifyMessage44")->getString();
|
|
|
|
if (*it == ESM::Skill::Sneak || *it == ESM::Skill::Security)
|
|
|
|
skillMsg = gmst.find("sNotifyMessage39")->getString();
|
|
|
|
|
|
|
|
if (skillMsg.find("%s") != std::string::npos)
|
|
|
|
skillMsg.replace(skillMsg.find("%s"), 2, skillName);
|
|
|
|
if (skillMsg.find("%d") != std::string::npos)
|
|
|
|
skillMsg.replace(skillMsg.find("%d"), 2, skillValue.str());
|
|
|
|
message += "\n" + skillMsg;
|
|
|
|
}
|
|
|
|
|
2014-04-23 09:54:18 +00:00
|
|
|
// TODO: Sleep the player
|
2014-04-02 04:18:22 +00:00
|
|
|
|
2014-01-11 05:47:58 +00:00
|
|
|
std::vector<std::string> buttons;
|
|
|
|
buttons.push_back("#{sOk}");
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox(message, buttons);
|
|
|
|
}
|
|
|
|
}
|
2014-01-14 06:40:17 +00:00
|
|
|
|
|
|
|
void World::spawnRandomCreature(const std::string &creatureList)
|
|
|
|
{
|
|
|
|
const ESM::CreatureLevList* list = getStore().get<ESM::CreatureLevList>().find(creatureList);
|
|
|
|
|
|
|
|
int iNumberCreatures = getStore().get<ESM::GameSetting>().find("iNumberCreatures")->getInt();
|
|
|
|
int numCreatures = 1 + std::rand()/ (static_cast<double> (RAND_MAX) + 1) * iNumberCreatures; // [1, iNumberCreatures]
|
|
|
|
|
|
|
|
for (int i=0; i<numCreatures; ++i)
|
|
|
|
{
|
|
|
|
std::string selectedCreature = MWMechanics::getLevelledItem(list, true);
|
|
|
|
if (selectedCreature.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
ESM::Position ipos = mPlayer->getPlayer().getRefData().getPosition();
|
|
|
|
Ogre::Vector3 pos(ipos.pos);
|
|
|
|
Ogre::Quaternion rot(Ogre::Radian(-ipos.rot[2]), Ogre::Vector3::UNIT_Z);
|
|
|
|
const float distance = 50;
|
|
|
|
pos = pos + distance*rot.yAxis();
|
|
|
|
ipos.pos[0] = pos.x;
|
|
|
|
ipos.pos[1] = pos.y;
|
|
|
|
ipos.pos[2] = pos.z;
|
|
|
|
ipos.rot[0] = 0;
|
|
|
|
ipos.rot[1] = 0;
|
|
|
|
ipos.rot[2] = 0;
|
|
|
|
|
|
|
|
MWWorld::CellStore* cell = mPlayer->getPlayer().getCell();
|
|
|
|
MWWorld::ManualRef ref(getStore(), selectedCreature, 1);
|
|
|
|
ref.getPtr().getCellRef().mPos = ipos;
|
|
|
|
|
2014-03-07 05:11:00 +00:00
|
|
|
safePlaceObject(ref.getPtr(), cell, ipos);
|
2014-01-14 06:40:17 +00:00
|
|
|
}
|
|
|
|
}
|
2014-01-17 09:52:44 +00:00
|
|
|
|
|
|
|
void World::spawnBloodEffect(const Ptr &ptr, const Vector3 &worldPosition)
|
|
|
|
{
|
|
|
|
int type = ptr.getClass().getBloodTexture(ptr);
|
|
|
|
std::string texture;
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case 2:
|
|
|
|
texture = getFallback()->getFallbackString("Blood_Texture_2");
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
texture = getFallback()->getFallbackString("Blood_Texture_1");
|
|
|
|
break;
|
|
|
|
case 0:
|
|
|
|
default:
|
|
|
|
texture = getFallback()->getFallbackString("Blood_Texture_0");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::stringstream modelName;
|
|
|
|
modelName << "Blood_Model_";
|
|
|
|
int roll = std::rand()/ (static_cast<double> (RAND_MAX) + 1) * 3; // [0, 2]
|
|
|
|
modelName << roll;
|
|
|
|
std::string model = "meshes\\" + getFallback()->getFallbackString(modelName.str());
|
|
|
|
|
|
|
|
mRendering->spawnEffect(model, texture, worldPosition);
|
|
|
|
}
|
2014-01-20 14:48:06 +00:00
|
|
|
|
|
|
|
void World::explodeSpell(const Vector3 &origin, const MWWorld::Ptr& object, const ESM::EffectList &effects, const Ptr &caster,
|
|
|
|
const std::string& id, const std::string& sourceName)
|
|
|
|
{
|
|
|
|
std::map<MWWorld::Ptr, std::vector<ESM::ENAMstruct> > toApply;
|
|
|
|
for (std::vector<ESM::ENAMstruct>::const_iterator effectIt = effects.mList.begin();
|
|
|
|
effectIt != effects.mList.end(); ++effectIt)
|
|
|
|
{
|
|
|
|
const ESM::MagicEffect* effect = getStore().get<ESM::MagicEffect>().find(effectIt->mEffectID);
|
|
|
|
|
|
|
|
if (effectIt->mArea <= 0)
|
|
|
|
continue; // Not an area effect
|
|
|
|
|
|
|
|
// Spawn the explosion orb effect
|
|
|
|
const ESM::Static* areaStatic;
|
|
|
|
if (!effect->mCasting.empty())
|
|
|
|
areaStatic = getStore().get<ESM::Static>().find (effect->mArea);
|
|
|
|
else
|
|
|
|
areaStatic = getStore().get<ESM::Static>().find ("VFX_DefaultArea");
|
|
|
|
|
|
|
|
mRendering->spawnEffect("meshes\\" + areaStatic->mModel, "", origin, effectIt->mArea);
|
|
|
|
|
|
|
|
// Play explosion sound (make sure to use NoTrack, since we will delete the projectile now)
|
|
|
|
static const std::string schools[] = {
|
|
|
|
"alteration", "conjuration", "destruction", "illusion", "mysticism", "restoration"
|
|
|
|
};
|
|
|
|
MWBase::SoundManager *sndMgr = MWBase::Environment::get().getSoundManager();
|
|
|
|
if(!effect->mAreaSound.empty())
|
|
|
|
sndMgr->playSound3D(object, effect->mAreaSound, 1.0f, 1.0f, MWBase::SoundManager::Play_TypeSfx, MWBase::SoundManager::Play_NoTrack);
|
|
|
|
else
|
|
|
|
sndMgr->playSound3D(object, schools[effect->mData.mSchool]+" area", 1.0f, 1.0f, MWBase::SoundManager::Play_TypeSfx, MWBase::SoundManager::Play_NoTrack);
|
|
|
|
|
|
|
|
// Get the actors in range of the effect
|
|
|
|
std::vector<MWWorld::Ptr> objects;
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->getObjectsInRange(
|
|
|
|
origin, feetToGameUnits(effectIt->mArea), objects);
|
|
|
|
|
|
|
|
for (std::vector<MWWorld::Ptr>::iterator affected = objects.begin(); affected != objects.end(); ++affected)
|
|
|
|
toApply[*affected].push_back(*effectIt);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now apply the appropriate effects to each actor in range
|
|
|
|
for (std::map<MWWorld::Ptr, std::vector<ESM::ENAMstruct> >::iterator apply = toApply.begin(); apply != toApply.end(); ++apply)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr source = caster;
|
|
|
|
// Vanilla-compatible behaviour of never applying the spell to the caster
|
|
|
|
// (could be changed by mods later)
|
|
|
|
if (apply->first == caster)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (source.isEmpty())
|
|
|
|
source = apply->first;
|
|
|
|
|
|
|
|
MWMechanics::CastSpell cast(source, apply->first);
|
|
|
|
cast.mHitPosition = origin;
|
|
|
|
cast.mId = id;
|
|
|
|
cast.mSourceName = sourceName;
|
|
|
|
cast.mStack = false;
|
|
|
|
ESM::EffectList effects;
|
|
|
|
effects.mList = apply->second;
|
|
|
|
cast.inflict(apply->first, caster, effects, ESM::RT_Target, false, true);
|
|
|
|
}
|
|
|
|
}
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|