2012-07-03 12:30:50 +02:00
|
|
|
#include "worldimp.hpp"
|
2010-07-02 09:38:22 +02:00
|
|
|
|
2010-07-27 14:05:05 +02:00
|
|
|
#include <components/bsa/bsa_archive.hpp>
|
2011-05-05 19:32:42 +02:00
|
|
|
#include <components/files/collections.hpp>
|
2010-07-02 09:38:22 +02:00
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
#include "../mwbase/environment.hpp"
|
2012-08-09 14:33:21 +02:00
|
|
|
#include "../mwbase/soundmanager.hpp"
|
2012-08-11 17:30:55 +02:00
|
|
|
#include "../mwbase/mechanicsmanager.hpp"
|
2012-08-12 18:11:09 +02:00
|
|
|
#include "../mwbase/windowmanager.hpp"
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2010-07-27 14:05:05 +02:00
|
|
|
#include "../mwrender/sky.hpp"
|
2011-11-01 13:47:36 -04:00
|
|
|
#include "../mwrender/player.hpp"
|
2010-07-27 14:05:05 +02:00
|
|
|
|
2011-01-04 15:58:22 +01:00
|
|
|
#include "player.hpp"
|
2012-05-16 21:27:02 +02:00
|
|
|
#include "manualref.hpp"
|
2011-01-29 14:33:44 +01:00
|
|
|
#include "cellfunctors.hpp"
|
2010-11-06 13:11:09 -04:00
|
|
|
|
2012-02-24 16:12:43 +01:00
|
|
|
using namespace Ogre;
|
|
|
|
|
2010-07-02 16:18:25 +02:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
template<typename T>
|
2010-07-02 17:21:27 +02:00
|
|
|
void listCellScripts (const ESMS::ESMStore& store,
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::CellRefList<T>& cellRefList, MWWorld::LocalScripts& localScripts,
|
2010-07-10 13:19:04 +02:00
|
|
|
MWWorld::Ptr::CellStore *cell)
|
2010-07-02 16:18:25 +02:00
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
for (typename MWWorld::CellRefList<T>::List::iterator iter (
|
2010-07-02 16:18:25 +02:00
|
|
|
cellRefList.list.begin());
|
|
|
|
iter!=cellRefList.list.end(); ++iter)
|
|
|
|
{
|
2010-08-04 14:04:22 +02:00
|
|
|
if (!iter->base->script.empty() && iter->mData.getCount())
|
2010-07-02 17:21:27 +02:00
|
|
|
{
|
|
|
|
if (const ESM::Script *script = store.scripts.find (iter->base->script))
|
2010-08-03 20:01:52 +02:00
|
|
|
{
|
2010-07-02 17:21:27 +02:00
|
|
|
iter->mData.setLocals (*script);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2011-10-06 12:29:59 +02:00
|
|
|
localScripts.add (iter->base->script, MWWorld::Ptr (&*iter, cell));
|
2010-07-02 17:21:27 +02:00
|
|
|
}
|
|
|
|
}
|
2010-07-02 16:18:25 +02:00
|
|
|
}
|
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
|
|
|
template<typename T>
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<T> *searchViaHandle (const std::string& handle,
|
|
|
|
MWWorld::CellRefList<T>& refList)
|
2010-08-03 20:01:52 +02:00
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
typedef typename MWWorld::CellRefList<T>::List::iterator iterator;
|
2010-08-03 22:06:48 +02:00
|
|
|
|
|
|
|
for (iterator iter (refList.list.begin()); iter!=refList.list.end(); ++iter)
|
|
|
|
{
|
2012-08-07 22:21:46 +04:00
|
|
|
if(iter->mData.getCount() > 0 && iter->mData.getBaseNode()){
|
2010-08-03 22:06:48 +02:00
|
|
|
if (iter->mData.getHandle()==handle)
|
|
|
|
{
|
|
|
|
return &*iter;
|
|
|
|
}
|
2011-11-19 17:45:05 -05:00
|
|
|
}
|
2010-08-03 22:06:48 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2010-07-02 16:18:25 +02:00
|
|
|
}
|
|
|
|
|
2010-07-03 15:41:20 +02:00
|
|
|
namespace MWWorld
|
2010-07-02 09:38:22 +02:00
|
|
|
{
|
2010-08-03 20:01:52 +02:00
|
|
|
Ptr World::getPtrViaHandle (const std::string& handle, Ptr::CellStore& cell)
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Activator> *ref =
|
2010-08-03 20:01:52 +02:00
|
|
|
searchViaHandle (handle, cell.activators))
|
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Potion> *ref = searchViaHandle (handle, cell.potions))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Apparatus> *ref = searchViaHandle (handle, cell.appas))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Armor> *ref = searchViaHandle (handle, cell.armors))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Book> *ref = searchViaHandle (handle, cell.books))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Clothing> *ref = searchViaHandle (handle, cell.clothes))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Container> *ref =
|
2010-08-03 20:01:52 +02:00
|
|
|
searchViaHandle (handle, cell.containers))
|
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Creature> *ref =
|
2010-08-03 20:01:52 +02:00
|
|
|
searchViaHandle (handle, cell.creatures))
|
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Door> *ref = searchViaHandle (handle, cell.doors))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Ingredient> *ref =
|
2010-08-03 20:01:52 +02:00
|
|
|
searchViaHandle (handle, cell.ingreds))
|
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Light> *ref = searchViaHandle (handle, cell.lights))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Tool> *ref = searchViaHandle (handle, cell.lockpicks))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Miscellaneous> *ref = searchViaHandle (handle, cell.miscItems))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::NPC> *ref = searchViaHandle (handle, cell.npcs))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Probe> *ref = searchViaHandle (handle, cell.probes))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Repair> *ref = searchViaHandle (handle, cell.repairs))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Static> *ref = searchViaHandle (handle, cell.statics))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
2012-06-29 18:54:23 +02:00
|
|
|
if (MWWorld::LiveCellRef<ESM::Weapon> *ref = searchViaHandle (handle, cell.weapons))
|
2010-08-03 20:01:52 +02:00
|
|
|
return Ptr (ref, &cell);
|
|
|
|
return Ptr();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-22 11:48:27 +02: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 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
throw std::runtime_error ("month out of range");
|
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-08-22 20:55:22 +02:00
|
|
|
void World::adjustSky()
|
|
|
|
{
|
2012-02-26 18:21:11 +01:00
|
|
|
if (mSky && (isCellExterior() || isCellQuasiExterior()))
|
2010-08-22 20:55:22 +02:00
|
|
|
{
|
2012-02-24 18:30:16 +01:00
|
|
|
mRendering->skySetHour (mGlobalVariables->getFloat ("gamehour"));
|
|
|
|
mRendering->skySetDate (mGlobalVariables->getInt ("day"),
|
|
|
|
mGlobalVariables->getInt ("month"));
|
2012-02-26 18:21:11 +01:00
|
|
|
|
2012-04-01 15:07:41 +02:00
|
|
|
mRendering->skyEnable();
|
2010-08-22 20:55:22 +02:00
|
|
|
}
|
2012-02-26 18:21:11 +01:00
|
|
|
else
|
2012-04-01 15:07:41 +02:00
|
|
|
mRendering->skyDisable();
|
2010-08-22 20:55:22 +02:00
|
|
|
}
|
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
void World::setFallbackValues (const std::map<std::string,std::string>& fallbackMap)
|
2012-04-02 20:47:09 +02:00
|
|
|
{
|
2012-04-03 02:14:39 +02:00
|
|
|
mFallback = fallbackMap;
|
2012-04-02 20:47:09 +02:00
|
|
|
}
|
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
std::string World::getFallback (const std::string& key) const
|
2012-04-04 16:15:15 +02:00
|
|
|
{
|
|
|
|
return getFallback(key, "");
|
|
|
|
}
|
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
std::string World::getFallback (const std::string& key, const std::string& def) const
|
2012-04-02 20:47:09 +02:00
|
|
|
{
|
2012-07-03 01:06:38 +02:00
|
|
|
std::map<std::string,std::string>::const_iterator it;
|
2012-04-02 20:47:09 +02:00
|
|
|
if((it = mFallback.find(key)) == mFallback.end())
|
|
|
|
{
|
2012-04-04 16:15:15 +02:00
|
|
|
return def;
|
2012-04-02 20:47:09 +02:00
|
|
|
}
|
|
|
|
return it->second;
|
|
|
|
}
|
|
|
|
|
2012-01-29 20:51:20 +02:00
|
|
|
World::World (OEngine::Render::OgreRenderer& renderer,
|
2011-05-05 19:32:42 +02:00
|
|
|
const Files::Collections& fileCollections,
|
2012-04-23 15:27:03 +02:00
|
|
|
const std::string& master, const boost::filesystem::path& resDir, bool newGame,
|
|
|
|
const std::string& encoding, std::map<std::string,std::string> fallbackMap)
|
2012-01-29 20:51:20 +02:00
|
|
|
: mPlayer (0), mLocalScripts (mStore), mGlobalVariables (0),
|
2012-07-03 12:30:50 +02:00
|
|
|
mSky (true), mNextDynamicRecord (0), mCells (mStore, mEsm),
|
2012-08-10 16:21:53 +02:00
|
|
|
mNumFacing(0)
|
2010-08-03 20:01:52 +02:00
|
|
|
{
|
2012-01-29 17:31:18 +02:00
|
|
|
mPhysics = new PhysicsSystem(renderer);
|
2012-01-29 20:51:20 +02:00
|
|
|
mPhysEngine = mPhysics->getEngine();
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
mRendering = new MWRender::RenderingManager(renderer, resDir, mPhysEngine);
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
mWeatherManager = new MWWorld::WeatherManager(mRendering);
|
2011-05-05 19:32:42 +02:00
|
|
|
|
|
|
|
boost::filesystem::path masterPath (fileCollections.getCollection (".esm").getPath (master));
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-02 09:38:22 +02:00
|
|
|
std::cout << "Loading ESM " << masterPath.string() << "\n";
|
|
|
|
|
|
|
|
// This parses the ESM file and loads a sample cell
|
Added new command line option: "encoding"
Added new command line option: "encoding" which allow to
change font encoding used in game messages.
Currently there are three evailable encodings:
win1250 - Central and Eastern European (languages
that use Latin script, such as Polish,
Czech, Slovak, Hungarian, Slovene, Bosnian,
Croatian, Serbian (Latin script),
Romanian and Albanian)
win1251 - languages that use the Cyrillic alphabet
such as Russian, Bulgarian, Serbian Cyrillic
and others
win1252 - Western European (Latin) - default
Signed-off-by: Lukasz Gromanowski <lgromanowski@gmail.com>
2011-07-17 22:16:50 +02:00
|
|
|
mEsm.setEncoding(encoding);
|
2011-03-28 17:36:05 +02:00
|
|
|
mEsm.open (masterPath.string());
|
2010-07-02 09:38:22 +02:00
|
|
|
mStore.load (mEsm);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2012-08-09 00:15:52 +04:00
|
|
|
mPlayer = new MWWorld::Player (mStore.npcs.find ("player"), *this);
|
|
|
|
mRendering->attachCameraTo(mPlayer->getPlayer());
|
|
|
|
|
2011-08-01 15:55:36 +02:00
|
|
|
mPhysics->addActor (mPlayer->getPlayer().getRefData().getHandle(), "", Ogre::Vector3 (0, 0, 0));
|
2010-07-05 12:09:04 +02:00
|
|
|
|
2010-07-04 13:33:33 +02:00
|
|
|
// global variables
|
2010-07-18 16:48:01 +02:00
|
|
|
mGlobalVariables = new Globals (mStore);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-04 16:00:32 +02:00
|
|
|
if (newGame)
|
2010-08-03 20:01:52 +02:00
|
|
|
{
|
2010-07-04 16:00:32 +02:00
|
|
|
// set new game mark
|
2010-07-18 16:48:01 +02:00
|
|
|
mGlobalVariables->setInt ("chargenstate", 1);
|
2010-07-04 16:00:32 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2012-07-03 12:30:50 +02:00
|
|
|
mWorldScene = new Scene(*mRendering, mPhysics);
|
2012-04-04 16:08:46 +02:00
|
|
|
|
2012-04-03 02:14:39 +02:00
|
|
|
setFallbackValues(fallbackMap);
|
2011-10-29 09:55:22 +02:00
|
|
|
|
2012-04-30 23:55:22 +02:00
|
|
|
lastTick = mTimer.getMilliseconds();
|
2010-07-02 09:38:22 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2012-01-29 17:50:51 -05:00
|
|
|
|
2010-07-02 09:38:22 +02:00
|
|
|
World::~World()
|
|
|
|
{
|
2012-02-21 20:22:46 +01:00
|
|
|
delete mWeatherManager;
|
2011-08-09 09:56:09 +02:00
|
|
|
delete mWorldScene;
|
2010-07-18 16:48:01 +02:00
|
|
|
delete mGlobalVariables;
|
2012-01-29 20:51:20 +02:00
|
|
|
delete mRendering;
|
2011-08-01 15:55:36 +02:00
|
|
|
delete mPhysics;
|
2011-10-29 09:55:22 +02:00
|
|
|
|
|
|
|
delete mPlayer;
|
2010-07-02 09:38:22 +02:00
|
|
|
}
|
2011-08-01 14:41:15 +02:00
|
|
|
|
|
|
|
const ESM::Cell *World::getExterior (const std::string& cellName) const
|
|
|
|
{
|
|
|
|
// first try named cells
|
|
|
|
if (const ESM::Cell *cell = mStore.cells.searchExtByName (cellName))
|
|
|
|
return cell;
|
|
|
|
|
|
|
|
// didn't work -> now check for regions
|
|
|
|
std::string cellName2 = ESMS::RecListT<ESM::Region>::toLower (cellName);
|
|
|
|
|
|
|
|
for (ESMS::RecListT<ESM::Region>::MapType::const_iterator iter (mStore.regions.list.begin());
|
|
|
|
iter!=mStore.regions.list.end(); ++iter)
|
|
|
|
{
|
|
|
|
if (ESMS::RecListT<ESM::Region>::toLower (iter->second.name)==cellName2)
|
|
|
|
{
|
|
|
|
if (const ESM::Cell *cell = mStore.cells.searchExtByRegion (iter->first))
|
|
|
|
return cell;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2011-08-09 09:56:09 +02:00
|
|
|
|
2011-08-01 14:34:50 +02:00
|
|
|
Ptr::CellStore *World::getExterior (int x, int y)
|
|
|
|
{
|
2011-09-08 11:02:55 +02:00
|
|
|
return mCells.getExterior (x, y);
|
2011-08-01 14:34:50 +02:00
|
|
|
}
|
2011-08-09 09:56:09 +02:00
|
|
|
|
2011-09-08 11:02:55 +02:00
|
|
|
Ptr::CellStore *World::getInterior (const std::string& name)
|
2011-08-01 14:34:50 +02:00
|
|
|
{
|
2011-09-08 11:02:55 +02:00
|
|
|
return mCells.getInterior (name);
|
2011-08-01 14:34:50 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2011-01-04 15:58:22 +01:00
|
|
|
MWWorld::Player& World::getPlayer()
|
2010-07-02 09:38:22 +02:00
|
|
|
{
|
2011-01-04 15:58:22 +01:00
|
|
|
return *mPlayer;
|
2010-07-02 09:38:22 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2011-04-21 10:49:45 +02:00
|
|
|
const ESMS::ESMStore& World::getStore() const
|
2010-07-02 16:18:25 +02:00
|
|
|
{
|
|
|
|
return mStore;
|
|
|
|
}
|
2011-08-09 09:56:09 +02:00
|
|
|
|
2011-07-31 17:07:11 +02:00
|
|
|
ESM::ESMReader& World::getEsmReader()
|
|
|
|
{
|
|
|
|
return mEsm;
|
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2011-10-06 12:29:59 +02:00
|
|
|
LocalScripts& World::getLocalScripts()
|
2010-07-02 16:18:25 +02:00
|
|
|
{
|
|
|
|
return mLocalScripts;
|
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-03 12:12:13 +02:00
|
|
|
bool World::hasCellChanged() const
|
|
|
|
{
|
2011-08-01 04:06:38 +02:00
|
|
|
return mWorldScene->hasCellChanged();
|
2010-07-03 12:12:13 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 16:48:01 +02:00
|
|
|
Globals::Data& World::getGlobalVariable (const std::string& name)
|
2010-07-04 13:33:33 +02:00
|
|
|
{
|
2010-07-18 16:48:01 +02:00
|
|
|
return (*mGlobalVariables)[name];
|
2010-07-04 13:33:33 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2011-04-21 11:00:00 +02:00
|
|
|
Globals::Data World::getGlobalVariable (const std::string& name) const
|
|
|
|
{
|
|
|
|
return (*mGlobalVariables)[name];
|
|
|
|
}
|
|
|
|
|
2010-07-21 15:01:35 +02:00
|
|
|
char World::getGlobalVariableType (const std::string& name) const
|
|
|
|
{
|
|
|
|
return mGlobalVariables->getType (name);
|
2010-08-03 20:01:52 +02:00
|
|
|
}
|
|
|
|
|
2010-07-10 13:19:04 +02:00
|
|
|
Ptr World::getPtr (const std::string& name, bool activeOnly)
|
2010-07-05 12:09:04 +02:00
|
|
|
{
|
|
|
|
// the player is always in an active cell.
|
|
|
|
if (name=="player")
|
|
|
|
{
|
2011-01-04 15:58:22 +01:00
|
|
|
return mPlayer->getPlayer();
|
2010-07-05 12:09:04 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-05 12:09:04 +02:00
|
|
|
// active cells
|
2011-11-15 23:31:18 -05:00
|
|
|
for (Scene::CellStoreCollection::const_iterator iter (mWorldScene->getActiveCells().begin());
|
2011-08-01 04:06:38 +02:00
|
|
|
iter!=mWorldScene->getActiveCells().end(); ++iter)
|
2010-07-05 12:09:04 +02:00
|
|
|
{
|
2011-11-13 23:14:18 -05:00
|
|
|
Ptr::CellStore* cellstore = *iter;
|
|
|
|
Ptr ptr = mCells.getPtr (name, *cellstore);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-05 12:09:04 +02:00
|
|
|
if (!ptr.isEmpty())
|
2010-07-10 13:19:04 +02:00
|
|
|
return ptr;
|
2010-07-05 12:09:04 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-05 12:09:04 +02:00
|
|
|
if (!activeOnly)
|
|
|
|
{
|
2011-09-22 12:44:17 +02:00
|
|
|
Ptr ptr = mCells.getPtr (name);
|
|
|
|
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
2010-07-05 12:09:04 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-05 12:09:04 +02:00
|
|
|
throw std::runtime_error ("unknown ID: " + name);
|
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
|
|
|
Ptr World::getPtrViaHandle (const std::string& handle)
|
|
|
|
{
|
2011-02-10 13:32:34 +01:00
|
|
|
if (mPlayer->getPlayer().getRefData().getHandle()==handle)
|
|
|
|
return mPlayer->getPlayer();
|
2011-11-15 23:31:18 -05:00
|
|
|
for (Scene::CellStoreCollection::const_iterator iter (mWorldScene->getActiveCells().begin());
|
2011-08-01 04:06:38 +02:00
|
|
|
iter!=mWorldScene->getActiveCells().end(); ++iter)
|
2010-08-03 20:01:52 +02:00
|
|
|
{
|
2011-11-13 23:14:18 -05:00
|
|
|
Ptr::CellStore* cellstore = *iter;
|
|
|
|
Ptr ptr = getPtrViaHandle (handle, *cellstore);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw std::runtime_error ("unknown Ogre handle: " + handle);
|
|
|
|
}
|
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
void World::enable (const Ptr& reference)
|
2010-07-09 16:07:03 +02:00
|
|
|
{
|
2010-07-10 13:19:04 +02:00
|
|
|
if (!reference.getRefData().isEnabled())
|
2010-07-09 16:07:03 +02:00
|
|
|
{
|
2010-07-10 13:19:04 +02:00
|
|
|
reference.getRefData().enable();
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2012-05-25 17:28:27 +02:00
|
|
|
if(mWorldScene->getActiveCells().find (reference.getCell()) != mWorldScene->getActiveCells().end() && reference.getRefData().getCount())
|
|
|
|
mWorldScene->addObjectToScene (reference);
|
2010-07-09 16:07:03 +02:00
|
|
|
}
|
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
void World::disable (const Ptr& reference)
|
2010-07-09 16:07:03 +02:00
|
|
|
{
|
2010-08-07 20:29:10 +02:00
|
|
|
if (reference.getRefData().isEnabled())
|
2010-07-09 16:07:03 +02:00
|
|
|
{
|
2010-08-07 20:29:10 +02:00
|
|
|
reference.getRefData().disable();
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2012-05-25 17:28:27 +02:00
|
|
|
if(mWorldScene->getActiveCells().find (reference.getCell())!=mWorldScene->getActiveCells().end() && reference.getRefData().getCount())
|
|
|
|
mWorldScene->removeObjectFromScene (reference);
|
2010-08-03 20:01:52 +02:00
|
|
|
}
|
2010-07-09 16:07:03 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 18:29:16 +02:00
|
|
|
void World::advanceTime (double hours)
|
|
|
|
{
|
|
|
|
hours += mGlobalVariables->getFloat ("gamehour");
|
2012-01-23 14:33:06 +01:00
|
|
|
|
2010-07-18 18:29:16 +02:00
|
|
|
setHour (hours);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
int days = hours / 24;
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
if (days>0)
|
|
|
|
mGlobalVariables->setInt ("dayspassed", days + mGlobalVariables->getInt ("dayspassed"));
|
2010-07-18 18:29:16 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 18:29:16 +02:00
|
|
|
void World::setHour (double hour)
|
|
|
|
{
|
|
|
|
if (hour<0)
|
|
|
|
hour = 0;
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 18:29:16 +02:00
|
|
|
int days = hour / 24;
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 18:29:16 +02:00
|
|
|
hour = std::fmod (hour, 24);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 18:29:16 +02:00
|
|
|
mGlobalVariables->setFloat ("gamehour", hour);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2012-01-29 20:51:20 +02:00
|
|
|
mRendering->skySetHour (hour);
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-02-22 20:12:08 +01:00
|
|
|
mWeatherManager->setHour (hour);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 18:29:16 +02:00
|
|
|
if (days>0)
|
2010-07-22 11:48:27 +02:00
|
|
|
setDay (days + mGlobalVariables->getInt ("day"));
|
2010-07-18 18:29:16 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 18:29:16 +02:00
|
|
|
void World::setDay (int day)
|
|
|
|
{
|
|
|
|
if (day<0)
|
|
|
|
day = 0;
|
2010-07-22 11:48:27 +02:00
|
|
|
|
|
|
|
int month = mGlobalVariables->getInt ("month");
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
while (true)
|
2010-07-18 18:29:16 +02:00
|
|
|
{
|
2010-08-03 20:01:52 +02:00
|
|
|
int days = getDaysPerMonth (month);
|
2010-07-22 11:48:27 +02:00
|
|
|
if (day<days)
|
|
|
|
break;
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
if (month<11)
|
|
|
|
{
|
|
|
|
++month;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
month = 0;
|
|
|
|
mGlobalVariables->setInt ("year", mGlobalVariables->getInt ("year")+1);
|
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
day -= days;
|
2010-08-03 20:01:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
mGlobalVariables->setInt ("day", day);
|
2010-07-22 11:48:27 +02:00
|
|
|
mGlobalVariables->setInt ("month", month);
|
|
|
|
|
2012-01-29 20:51:20 +02:00
|
|
|
mRendering->skySetDate (day, month);
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-02-22 20:12:08 +01:00
|
|
|
mWeatherManager->setDate (day, month);
|
2012-03-15 16:01:41 +01:00
|
|
|
|
|
|
|
|
2010-08-03 20:01:52 +02:00
|
|
|
}
|
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
void World::setMonth (int month)
|
|
|
|
{
|
|
|
|
if (month<0)
|
|
|
|
month = 0;
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
int years = month / 12;
|
|
|
|
month = month % 12;
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
int days = getDaysPerMonth (month);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
if (mGlobalVariables->getInt ("day")>=days)
|
|
|
|
mGlobalVariables->setInt ("day", days-1);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
mGlobalVariables->setInt ("month", month);
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-22 11:48:27 +02:00
|
|
|
if (years>0)
|
|
|
|
mGlobalVariables->setInt ("year", years+mGlobalVariables->getInt ("year"));
|
|
|
|
|
2012-01-29 20:51:20 +02:00
|
|
|
mRendering->skySetDate (mGlobalVariables->getInt ("day"), month);
|
2010-07-22 11:48:27 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2012-05-19 09:31:45 +02:00
|
|
|
TimeStamp World::getTimeStamp() const
|
|
|
|
{
|
|
|
|
return TimeStamp (mGlobalVariables->getFloat ("gamehour"),
|
|
|
|
mGlobalVariables->getInt ("dayspassed"));
|
|
|
|
}
|
|
|
|
|
2011-04-26 21:38:21 +02:00
|
|
|
bool World::toggleSky()
|
2010-07-18 19:48:02 +02:00
|
|
|
{
|
|
|
|
if (mSky)
|
|
|
|
{
|
|
|
|
mSky = false;
|
2012-01-29 20:51:20 +02:00
|
|
|
mRendering->skyDisable();
|
2011-04-26 21:38:21 +02:00
|
|
|
return false;
|
2010-07-18 19:48:02 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mSky = true;
|
2012-01-29 20:51:20 +02:00
|
|
|
mRendering->skyEnable();
|
2011-04-26 21:38:21 +02:00
|
|
|
return true;
|
2010-07-18 19:48:02 +02:00
|
|
|
}
|
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 19:48:02 +02:00
|
|
|
int World::getMasserPhase() const
|
|
|
|
{
|
2012-01-29 20:51:20 +02:00
|
|
|
return mRendering->skyGetMasserPhase();
|
2010-07-18 19:48:02 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 19:48:02 +02:00
|
|
|
int World::getSecundaPhase() const
|
|
|
|
{
|
2012-01-29 20:51:20 +02:00
|
|
|
return mRendering->skyGetSecundaPhase();
|
2010-07-18 19:48:02 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-18 19:48:02 +02:00
|
|
|
void World::setMoonColour (bool red)
|
|
|
|
{
|
2012-01-29 20:51:20 +02:00
|
|
|
mRendering->skySetMoonColour (red);
|
2010-07-18 19:48:02 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2010-07-21 14:10:52 +02:00
|
|
|
float World::getTimeScaleFactor() const
|
|
|
|
{
|
2012-03-16 19:02:33 +01:00
|
|
|
return mGlobalVariables->getFloat ("timescale");
|
2010-07-21 14:10:52 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2011-02-10 10:38:45 +01:00
|
|
|
void World::changeToInteriorCell (const std::string& cellName, const ESM::Position& position)
|
2010-07-22 12:29:23 +02:00
|
|
|
{
|
2011-08-01 04:06:38 +02:00
|
|
|
return mWorldScene->changeToInteriorCell(cellName, position);
|
2010-07-22 12:29:23 +02:00
|
|
|
}
|
2010-08-03 20:01:52 +02:00
|
|
|
|
2011-02-10 10:38:45 +01:00
|
|
|
void World::changeToExteriorCell (const ESM::Position& position)
|
2010-08-20 13:33:03 +02:00
|
|
|
{
|
2011-08-01 04:06:38 +02:00
|
|
|
return mWorldScene->changeToExteriorCell(position);
|
2010-08-20 13:33:03 +02:00
|
|
|
}
|
|
|
|
|
2010-07-22 12:29:23 +02:00
|
|
|
void World::markCellAsUnchanged()
|
|
|
|
{
|
2011-08-01 04:06:38 +02:00
|
|
|
return mWorldScene->markCellAsUnchanged();
|
2010-07-22 12:29:23 +02:00
|
|
|
}
|
2010-08-05 13:29:49 +02:00
|
|
|
|
|
|
|
std::string World::getFacedHandle()
|
|
|
|
{
|
2012-03-25 20:52:56 +02:00
|
|
|
if (!mRendering->occlusionQuerySupported())
|
|
|
|
{
|
|
|
|
std::pair<std::string, float> result = mPhysics->getFacedHandle (*this);
|
2010-08-05 13:29:49 +02:00
|
|
|
|
2012-03-25 20:52:56 +02:00
|
|
|
if (result.first.empty() ||
|
|
|
|
result.second>getStore().gameSettings.find ("iMaxActivateDist")->i)
|
|
|
|
return "";
|
2010-08-05 13:29:49 +02:00
|
|
|
|
2012-03-25 20:52:56 +02:00
|
|
|
return result.first;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// updated every few frames in update()
|
|
|
|
return mFacedHandle;
|
|
|
|
}
|
2010-08-05 13:29:49 +02:00
|
|
|
}
|
2010-08-07 20:25:17 +02:00
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
void World::deleteObject (const Ptr& ptr)
|
2010-08-07 20:25:17 +02:00
|
|
|
{
|
|
|
|
if (ptr.getRefData().getCount()>0)
|
|
|
|
{
|
|
|
|
ptr.getRefData().setCount (0);
|
|
|
|
|
2012-05-25 17:28:27 +02:00
|
|
|
if (mWorldScene->getActiveCells().find (ptr.getCell())!=mWorldScene->getActiveCells().end() &&
|
|
|
|
ptr.getRefData().isEnabled())
|
2012-05-24 13:26:07 +02:00
|
|
|
{
|
2012-05-25 17:28:27 +02:00
|
|
|
mWorldScene->removeObjectFromScene (ptr);
|
2012-05-24 13:26:07 +02:00
|
|
|
mLocalScripts.remove (ptr);
|
|
|
|
}
|
2010-08-07 20:25:17 +02:00
|
|
|
}
|
|
|
|
}
|
2010-08-21 12:31:04 +02:00
|
|
|
|
2012-08-08 14:51:33 +04:00
|
|
|
void World::moveObject(const Ptr &ptr, CellStore &newCell, float x, float y, float z)
|
2010-08-21 12:31:04 +02:00
|
|
|
{
|
2012-07-30 23:28:14 +04:00
|
|
|
ESM::Position &pos = ptr.getRefData().getPosition();
|
|
|
|
pos.pos[0] = x, pos.pos[1] = y, pos.pos[2] = z;
|
|
|
|
Ogre::Vector3 vec(x, y, z);
|
|
|
|
|
2012-08-08 14:51:33 +04:00
|
|
|
CellStore *currCell = ptr.getCell();
|
|
|
|
bool isPlayer = ptr == mPlayer->getPlayer();
|
|
|
|
bool haveToMove = mWorldScene->isCellActive(*currCell) || isPlayer;
|
2010-08-21 12:41:59 +02:00
|
|
|
|
2012-08-08 14:51:33 +04:00
|
|
|
if (*currCell != newCell) {
|
|
|
|
if (isPlayer) {
|
|
|
|
if (!newCell.isExterior()) {
|
|
|
|
changeToInteriorCell(newCell.cell->name, pos);
|
|
|
|
} else {
|
2012-08-11 17:01:55 +04:00
|
|
|
int cellX = newCell.cell->data.gridX;
|
|
|
|
int cellY = newCell.cell->data.gridY;
|
|
|
|
mWorldScene->changeCell(cellX, cellY, pos, false);
|
2012-08-08 14:51:33 +04:00
|
|
|
}
|
|
|
|
} else {
|
2012-08-11 18:28:37 +04:00
|
|
|
if (!mWorldScene->isCellActive(*currCell)) {
|
2012-08-08 14:51:33 +04:00
|
|
|
copyObjectToCell(ptr, newCell, pos);
|
2012-08-11 18:28:37 +04:00
|
|
|
} else if (!mWorldScene->isCellActive(newCell)) {
|
2012-08-08 14:51:33 +04:00
|
|
|
MWWorld::Class::get(ptr).copyToCell(ptr, newCell);
|
|
|
|
mWorldScene->removeObjectFromScene(ptr);
|
|
|
|
mLocalScripts.remove(ptr);
|
|
|
|
haveToMove = false;
|
|
|
|
} else {
|
|
|
|
MWWorld::Ptr copy =
|
|
|
|
MWWorld::Class::get(ptr).copyToCell(ptr, newCell);
|
|
|
|
|
|
|
|
mRendering->moveObjectToCell(copy, vec, currCell);
|
|
|
|
|
|
|
|
if (MWWorld::Class::get(ptr).isActor()) {
|
2012-08-11 17:30:55 +02:00
|
|
|
MWBase::MechanicsManager *mechMgr =
|
2012-08-08 14:51:33 +04:00
|
|
|
MWBase::Environment::get().getMechanicsManager();
|
|
|
|
|
|
|
|
mechMgr->removeActor(ptr);
|
|
|
|
mechMgr->addActor(copy);
|
2012-07-30 23:28:14 +04:00
|
|
|
} else {
|
2012-08-08 14:51:33 +04:00
|
|
|
std::string script =
|
|
|
|
MWWorld::Class::get(ptr).getScript(ptr);
|
|
|
|
if (!script.empty()) {
|
2012-07-30 23:28:14 +04:00
|
|
|
mLocalScripts.remove(ptr);
|
2012-08-08 14:51:33 +04:00
|
|
|
mLocalScripts.add(script, copy);
|
2012-07-30 23:28:14 +04:00
|
|
|
}
|
2010-08-21 12:41:59 +02:00
|
|
|
}
|
|
|
|
}
|
2012-08-08 14:51:33 +04:00
|
|
|
ptr.getRefData().setCount(0);
|
2010-08-21 12:41:59 +02:00
|
|
|
}
|
|
|
|
}
|
2012-07-30 23:28:14 +04:00
|
|
|
if (haveToMove) {
|
|
|
|
mRendering->moveObject(ptr, vec);
|
|
|
|
mPhysics->moveObject(ptr.getRefData().getHandle(), vec);
|
|
|
|
}
|
2012-08-08 14:51:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool World::moveObjectImp(const Ptr& ptr, float x, float y, float z)
|
|
|
|
{
|
|
|
|
CellStore *cell = ptr.getCell();
|
|
|
|
if (cell->isExterior()) {
|
|
|
|
int cellX, cellY;
|
|
|
|
positionToIndex(x, y, cellX, cellY);
|
2012-08-11 17:30:55 +02:00
|
|
|
|
2012-08-08 14:51:33 +04:00
|
|
|
cell = getExterior(cellX, cellY);
|
|
|
|
}
|
|
|
|
moveObject(ptr, *cell, x, y, z);
|
2012-08-11 17:30:55 +02:00
|
|
|
|
2012-08-08 14:51:33 +04:00
|
|
|
return cell != ptr.getCell();
|
2011-08-02 18:50:31 +02:00
|
|
|
}
|
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
void World::moveObject (const Ptr& ptr, float x, float y, float z)
|
2011-08-02 18:50:31 +02:00
|
|
|
{
|
|
|
|
moveObjectImp(ptr, x, y, z);
|
2010-08-21 12:31:04 +02:00
|
|
|
}
|
2010-08-22 21:30:48 +02:00
|
|
|
|
2012-07-09 18:47:59 +02:00
|
|
|
void World::scaleObject (const Ptr& ptr, float scale)
|
2012-05-25 18:23:06 +02:00
|
|
|
{
|
|
|
|
MWWorld::Class::get(ptr).adjustScale(ptr,scale);
|
|
|
|
|
|
|
|
ptr.getCellRef().scale = scale;
|
2012-07-10 11:15:46 +02:00
|
|
|
//scale = scale/ptr.getRefData().getBaseNode()->getScale().x;
|
2012-05-25 18:23:06 +02:00
|
|
|
ptr.getRefData().getBaseNode()->setScale(scale,scale,scale);
|
2012-07-09 15:41:19 +02:00
|
|
|
mPhysics->scaleObject( ptr.getRefData().getHandle(), scale );
|
2012-05-25 18:23:06 +02:00
|
|
|
}
|
|
|
|
|
2012-08-09 00:15:52 +04:00
|
|
|
void World::rotateObject (const Ptr& ptr,float x,float y,float z, bool adjust)
|
2012-05-29 10:15:29 +02:00
|
|
|
{
|
2012-08-11 13:23:54 +04:00
|
|
|
Ogre::Vector3 rot;
|
|
|
|
rot.x = Ogre::Degree(x).valueRadians();
|
|
|
|
rot.y = Ogre::Degree(y).valueRadians();
|
|
|
|
rot.z = Ogre::Degree(z).valueRadians();
|
|
|
|
|
2012-08-09 00:15:52 +04:00
|
|
|
if (mRendering->rotateObject(ptr, rot, adjust)) {
|
|
|
|
float *objRot = ptr.getRefData().getPosition().rot;
|
2012-08-11 13:23:54 +04:00
|
|
|
objRot[0] = rot.x, objRot[1] = rot.y, objRot[2] = rot.z;
|
2012-05-29 10:15:29 +02:00
|
|
|
|
2012-08-11 18:28:37 +04:00
|
|
|
if (ptr.getRefData().getBaseNode() != 0) {
|
|
|
|
mPhysics->rotateObject(
|
|
|
|
ptr.getRefData().getHandle(),
|
|
|
|
ptr.getRefData().getBaseNode()->getOrientation()
|
|
|
|
);
|
|
|
|
}
|
2012-08-09 00:15:52 +04:00
|
|
|
}
|
2012-05-29 10:15:29 +02:00
|
|
|
}
|
|
|
|
|
2010-09-11 12:21:55 +02:00
|
|
|
void World::indexToPosition (int cellX, int cellY, float &x, float &y, bool centre) const
|
2010-08-22 21:30:48 +02:00
|
|
|
{
|
|
|
|
const int cellSize = 8192;
|
|
|
|
|
|
|
|
x = cellSize * cellX;
|
|
|
|
y = cellSize * cellY;
|
2010-09-11 12:21:55 +02:00
|
|
|
|
|
|
|
if (centre)
|
|
|
|
{
|
|
|
|
x += cellSize/2;
|
|
|
|
y += cellSize/2;
|
|
|
|
}
|
2010-08-22 21:30:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void World::positionToIndex (float x, float y, int &cellX, int &cellY) const
|
|
|
|
{
|
|
|
|
const int cellSize = 8192;
|
|
|
|
|
|
|
|
cellX = static_cast<int> (x/cellSize);
|
|
|
|
|
|
|
|
if (x<0)
|
|
|
|
--cellX;
|
|
|
|
|
|
|
|
cellY = static_cast<int> (y/cellSize);
|
|
|
|
|
|
|
|
if (y<0)
|
|
|
|
--cellY;
|
|
|
|
}
|
2011-01-29 17:39:34 +01:00
|
|
|
|
2011-02-01 10:11:41 +01:00
|
|
|
void World::doPhysics (const std::vector<std::pair<std::string, Ogre::Vector3> >& actors,
|
|
|
|
float duration)
|
2011-01-29 17:39:34 +01:00
|
|
|
{
|
2012-04-30 23:55:22 +02:00
|
|
|
mPhysics->doPhysics(duration, actors);
|
2011-09-03 10:26:31 +02:00
|
|
|
|
2012-04-30 23:55:22 +02:00
|
|
|
const int tick = 16; // 16 ms ^= 60 Hz
|
2011-08-27 10:30:38 +02:00
|
|
|
|
2012-04-30 23:55:22 +02:00
|
|
|
// Game clock part of the loop, contains everything that has to be executed in a fixed timestep
|
|
|
|
long long dt = mTimer.getMilliseconds() - lastTick;
|
2012-05-01 20:51:32 +02:00
|
|
|
if (dt >= 100)
|
|
|
|
{
|
|
|
|
// throw away wall clock time if necessary to keep the framerate above the minimum of 10 fps
|
|
|
|
lastTick += (dt - 100);
|
|
|
|
dt = 100;
|
|
|
|
}
|
2012-04-30 23:55:22 +02:00
|
|
|
while (dt >= tick)
|
2011-08-27 10:30:38 +02:00
|
|
|
{
|
2012-04-30 23:55:22 +02:00
|
|
|
dt -= tick;
|
|
|
|
lastTick += tick;
|
|
|
|
|
|
|
|
std::vector< std::pair<std::string, Ogre::Vector3> > vectors = mPhysics->doPhysicsFixed (actors);
|
|
|
|
|
|
|
|
std::vector< std::pair<std::string, Ogre::Vector3> >::iterator player = vectors.end();
|
|
|
|
|
|
|
|
for (std::vector< std::pair<std::string, Ogre::Vector3> >::iterator it = vectors.begin();
|
|
|
|
it!= vectors.end(); ++it)
|
2011-08-27 10:30:38 +02:00
|
|
|
{
|
2012-04-30 23:55:22 +02:00
|
|
|
if (it->first=="player")
|
|
|
|
{
|
|
|
|
player = it;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = getPtrViaHandle (it->first);
|
|
|
|
moveObjectImp (ptr, it->second.x, it->second.y, it->second.z);
|
|
|
|
}
|
2011-08-27 10:30:38 +02:00
|
|
|
}
|
2012-04-30 23:55:22 +02:00
|
|
|
|
|
|
|
// Make sure player is moved last (otherwise the cell might change in the middle of an update
|
|
|
|
// loop)
|
|
|
|
if (player!=vectors.end())
|
2011-08-27 10:30:38 +02:00
|
|
|
{
|
2012-04-30 23:55:22 +02:00
|
|
|
if (moveObjectImp (getPtrViaHandle (player->first),
|
|
|
|
player->second.x, player->second.y, player->second.z) == true)
|
|
|
|
return; // abort the current loop if the cell has changed
|
2011-08-27 10:30:38 +02:00
|
|
|
}
|
2011-08-02 18:16:39 +02:00
|
|
|
}
|
2011-01-29 17:39:34 +01:00
|
|
|
}
|
2011-03-13 22:33:55 +01:00
|
|
|
|
2011-04-26 21:38:21 +02:00
|
|
|
bool World::toggleCollisionMode()
|
2011-03-13 22:33:55 +01:00
|
|
|
{
|
2012-02-20 14:07:16 +01:00
|
|
|
return mPhysics->toggleCollisionMode();;
|
2011-03-13 22:33:55 +01:00
|
|
|
}
|
2011-03-16 09:09:45 +01:00
|
|
|
|
2011-04-26 21:38:21 +02:00
|
|
|
bool World::toggleRenderMode (RenderMode mode)
|
2011-03-16 09:09:45 +01:00
|
|
|
{
|
2012-01-29 20:51:20 +02:00
|
|
|
return mRendering->toggleRenderMode (mode);
|
2011-03-16 09:09:45 +01:00
|
|
|
}
|
2011-07-22 15:56:54 +02:00
|
|
|
|
|
|
|
std::pair<std::string, const ESM::Potion *> World::createRecord (const ESM::Potion& record)
|
|
|
|
{
|
|
|
|
/// \todo Rewrite the ESMStore so that a dynamic 2nd ESMStore can be attached to it.
|
|
|
|
/// This function should then insert the record into the 2nd store (the code for this
|
|
|
|
/// should also be moved to the ESMStore class). It might be a good idea to review
|
|
|
|
/// the STL-container usage of the ESMStore before the rewrite.
|
|
|
|
|
|
|
|
std::ostringstream stream;
|
2011-07-23 11:56:01 +02:00
|
|
|
stream << "$dynamic" << mNextDynamicRecord++;
|
2011-07-22 15:56:54 +02:00
|
|
|
|
2012-08-28 09:32:38 +02:00
|
|
|
ESM::Potion record2 (record);
|
|
|
|
record2.mId = stream.str();
|
|
|
|
|
2011-07-22 15:56:54 +02:00
|
|
|
const ESM::Potion *created =
|
2012-08-28 09:32:38 +02:00
|
|
|
&mStore.potions.list.insert (std::make_pair (stream.str(), record2)).first->second;
|
2011-07-22 15:56:54 +02:00
|
|
|
|
|
|
|
mStore.all.insert (std::make_pair (stream.str(), ESM::REC_ALCH));
|
|
|
|
|
|
|
|
return std::make_pair (stream.str(), created);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<std::string, const ESM::Class *> World::createRecord (const ESM::Class& record)
|
|
|
|
{
|
|
|
|
/// \todo See function above.
|
|
|
|
std::ostringstream stream;
|
2011-07-23 11:56:01 +02:00
|
|
|
stream << "$dynamic" << mNextDynamicRecord++;
|
2011-07-22 15:56:54 +02:00
|
|
|
|
|
|
|
const ESM::Class *created =
|
|
|
|
&mStore.classes.list.insert (std::make_pair (stream.str(), record)).first->second;
|
|
|
|
|
|
|
|
mStore.all.insert (std::make_pair (stream.str(), ESM::REC_CLAS));
|
|
|
|
|
|
|
|
return std::make_pair (stream.str(), created);
|
|
|
|
}
|
2011-09-26 11:11:30 +02:00
|
|
|
|
|
|
|
const ESM::Cell *World::createRecord (const ESM::Cell& record)
|
|
|
|
{
|
|
|
|
if (record.data.flags & ESM::Cell::Interior)
|
|
|
|
{
|
|
|
|
if (mStore.cells.searchInt (record.name))
|
|
|
|
throw std::runtime_error ("failed creating interior cell");
|
|
|
|
|
|
|
|
ESM::Cell *cell = new ESM::Cell (record);
|
|
|
|
mStore.cells.intCells.insert (std::make_pair (record.name, cell));
|
|
|
|
return cell;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (mStore.cells.searchExt (record.data.gridX, record.data.gridY))
|
|
|
|
throw std::runtime_error ("failed creating exterior cell");
|
|
|
|
|
|
|
|
ESM::Cell *cell = new ESM::Cell (record);
|
|
|
|
mStore.cells.extCells.insert (
|
|
|
|
std::make_pair (std::make_pair (record.data.gridX, record.data.gridY), cell));
|
|
|
|
return cell;
|
|
|
|
}
|
|
|
|
}
|
2011-11-21 14:08:44 +01:00
|
|
|
|
|
|
|
void World::playAnimationGroup (const MWWorld::Ptr& ptr, const std::string& groupName, int mode,
|
|
|
|
int number)
|
|
|
|
{
|
2012-01-29 20:51:20 +02:00
|
|
|
mRendering->playAnimationGroup (ptr, groupName, mode, number);
|
2011-11-21 14:08:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void World::skipAnimation (const MWWorld::Ptr& ptr)
|
|
|
|
{
|
2012-01-29 20:51:20 +02:00
|
|
|
mRendering->skipAnimation (ptr);
|
2011-11-21 14:08:44 +01:00
|
|
|
}
|
2012-01-23 14:33:06 +01:00
|
|
|
|
|
|
|
void World::update (float duration)
|
|
|
|
{
|
2012-07-03 01:06:38 +02:00
|
|
|
/// \todo split this function up into subfunctions
|
|
|
|
|
2012-01-23 14:33:06 +01:00
|
|
|
mWorldScene->update (duration);
|
2012-08-17 10:10:37 +04:00
|
|
|
|
|
|
|
float pitch, yaw;
|
|
|
|
Ogre::Vector3 eyepos;
|
|
|
|
mRendering->getPlayerData(eyepos, pitch, yaw);
|
|
|
|
mPhysics->updatePlayerData(eyepos, pitch, yaw);
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-02-22 19:17:37 +01:00
|
|
|
mWeatherManager->update (duration);
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-04-15 17:10:08 +02:00
|
|
|
// inform the GUI about focused object
|
|
|
|
try
|
|
|
|
{
|
2012-05-14 21:37:43 +02:00
|
|
|
MWWorld::Ptr object = getPtrViaHandle(mFacedHandle);
|
|
|
|
MWBase::Environment::get().getWindowManager()->setFocusObject(object);
|
|
|
|
|
|
|
|
// retrieve object dimensions so we know where to place the floating label
|
|
|
|
Ogre::SceneNode* node = object.getRefData().getBaseNode();
|
|
|
|
Ogre::AxisAlignedBox bounds;
|
|
|
|
int i;
|
|
|
|
for (i=0; i<node->numAttachedObjects(); ++i)
|
|
|
|
{
|
|
|
|
Ogre::MovableObject* ob = node->getAttachedObject(i);
|
|
|
|
bounds.merge(ob->getWorldBoundingBox());
|
|
|
|
}
|
|
|
|
if (bounds.isFinite())
|
|
|
|
{
|
|
|
|
Vector4 screenCoords = mRendering->boundingBoxToScreen(bounds);
|
|
|
|
MWBase::Environment::get().getWindowManager()->setFocusObjectScreenCoords(
|
|
|
|
screenCoords[0], screenCoords[1], screenCoords[2], screenCoords[3]);
|
|
|
|
}
|
2012-04-15 17:10:08 +02:00
|
|
|
}
|
|
|
|
catch (std::runtime_error&)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr null;
|
2012-04-24 02:02:03 +02:00
|
|
|
MWBase::Environment::get().getWindowManager()->setFocusObject(null);
|
2012-04-15 17:10:08 +02:00
|
|
|
}
|
|
|
|
|
2012-03-24 17:59:26 +01: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();
|
|
|
|
sun = Vector3(sun.x, -sun.z, sun.y);
|
|
|
|
mRendering->getSkyManager()->setGlare(!mPhysics->castRay(Ogre::Vector3(p[0], p[1], p[2]), sun));
|
|
|
|
}
|
2012-03-25 20:52:56 +02:00
|
|
|
|
|
|
|
// update faced handle (object the player is looking at)
|
|
|
|
// this uses a mixture of raycasts and occlusion queries.
|
|
|
|
else // if (mRendering->occlusionQuerySupported())
|
|
|
|
{
|
|
|
|
MWRender::OcclusionQuery* query = mRendering->getOcclusionQuery();
|
|
|
|
if (!query->occlusionTestPending())
|
|
|
|
{
|
|
|
|
// get result of last query
|
|
|
|
if (mNumFacing == 0) mFacedHandle = "";
|
|
|
|
else if (mNumFacing == 1)
|
|
|
|
{
|
|
|
|
bool result = query->getTestResult();
|
|
|
|
mFacedHandle = result ? mFaced1Name : "";
|
|
|
|
}
|
|
|
|
else if (mNumFacing == 2)
|
|
|
|
{
|
|
|
|
bool result = query->getTestResult();
|
|
|
|
mFacedHandle = result ? mFaced2Name : mFaced1Name;
|
|
|
|
}
|
|
|
|
|
|
|
|
// send new query
|
|
|
|
// figure out which object we want to test against
|
2012-06-02 12:25:24 +02:00
|
|
|
std::vector < std::pair < float, std::string > > results;
|
|
|
|
if (MWBase::Environment::get().getWindowManager()->isGuiMode())
|
|
|
|
{
|
|
|
|
float x, y;
|
|
|
|
MWBase::Environment::get().getWindowManager()->getMousePosition(x, y);
|
|
|
|
results = mPhysics->getFacedObjects(x, y);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
results = mPhysics->getFacedObjects();
|
2012-03-25 20:52:56 +02:00
|
|
|
|
2012-04-01 14:22:04 +02: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())
|
2012-03-27 22:36:02 +02:00
|
|
|
{
|
2012-04-02 19:57:58 +02:00
|
|
|
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)
|
2012-03-27 22:36:02 +02:00
|
|
|
{
|
2012-04-01 14:22:04 +02:00
|
|
|
it = results.erase(it);
|
2012-03-27 22:36:02 +02:00
|
|
|
}
|
2012-04-01 14:22:04 +02:00
|
|
|
else
|
|
|
|
++it;
|
2012-03-27 22:36:02 +02:00
|
|
|
}
|
|
|
|
|
2012-03-25 20:52:56 +02:00
|
|
|
if (results.size() == 0)
|
|
|
|
{
|
|
|
|
mNumFacing = 0;
|
|
|
|
}
|
|
|
|
else if (results.size() == 1)
|
|
|
|
{
|
|
|
|
mFaced1 = getPtrViaHandle(results.front().second);
|
|
|
|
mFaced1Name = results.front().second;
|
|
|
|
mNumFacing = 1;
|
|
|
|
|
2012-06-02 12:25:24 +02:00
|
|
|
btVector3 p;
|
|
|
|
if (MWBase::Environment::get().getWindowManager()->isGuiMode())
|
|
|
|
{
|
|
|
|
float x, y;
|
|
|
|
MWBase::Environment::get().getWindowManager()->getMousePosition(x, y);
|
|
|
|
p = mPhysics->getRayPoint(results.front().first, x, y);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
p = mPhysics->getRayPoint(results.front().first);
|
2012-03-25 20:52:56 +02:00
|
|
|
Ogre::Vector3 pos(p.x(), p.z(), -p.y());
|
|
|
|
Ogre::SceneNode* node = mFaced1.getRefData().getBaseNode();
|
2012-04-01 14:22:04 +02:00
|
|
|
|
2012-04-01 14:53:41 +02:00
|
|
|
//std::cout << "Num facing 1 : " << mFaced1Name << std::endl;
|
|
|
|
//std::cout << "Type 1 " << mFaced1.getTypeName() << std::endl;
|
|
|
|
|
2012-03-25 20:52:56 +02:00
|
|
|
query->occlusionTest(pos, node);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mFaced1Name = results.front().second;
|
|
|
|
mFaced2Name = results[1].second;
|
|
|
|
mFaced1 = getPtrViaHandle(results.front().second);
|
|
|
|
mFaced2 = getPtrViaHandle(results[1].second);
|
|
|
|
mNumFacing = 2;
|
|
|
|
|
2012-06-02 12:25:24 +02:00
|
|
|
btVector3 p;
|
|
|
|
if (MWBase::Environment::get().getWindowManager()->isGuiMode())
|
|
|
|
{
|
|
|
|
float x, y;
|
|
|
|
MWBase::Environment::get().getWindowManager()->getMousePosition(x, y);
|
|
|
|
p = mPhysics->getRayPoint(results[1].first, x, y);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
p = mPhysics->getRayPoint(results[1].first);
|
2012-03-25 20:52:56 +02:00
|
|
|
Ogre::Vector3 pos(p.x(), p.z(), -p.y());
|
2012-04-01 14:22:04 +02:00
|
|
|
Ogre::SceneNode* node1 = mFaced1.getRefData().getBaseNode();
|
|
|
|
Ogre::SceneNode* node2 = mFaced2.getRefData().getBaseNode();
|
|
|
|
|
|
|
|
// no need to test if the first node is not occluder
|
|
|
|
if (!query->isPotentialOccluder(node1) && (mFaced1.getTypeName().find("Static") == std::string::npos))
|
|
|
|
{
|
|
|
|
mFacedHandle = mFaced1Name;
|
|
|
|
//std::cout << "node1 Not an occluder" << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// no need to test if the second object is static (thus cannot be activated)
|
|
|
|
if (mFaced2.getTypeName().find("Static") != std::string::npos)
|
|
|
|
{
|
|
|
|
mFacedHandle = mFaced1Name;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// work around door problems
|
|
|
|
if (mFaced1.getTypeName().find("Static") != std::string::npos
|
|
|
|
&& mFaced2.getTypeName().find("Door") != std::string::npos)
|
|
|
|
{
|
|
|
|
mFacedHandle = mFaced2Name;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
query->occlusionTest(pos, node2);
|
2012-03-25 20:52:56 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-01-23 14:33:06 +01:00
|
|
|
}
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-02-23 21:12:06 +01:00
|
|
|
bool World::isCellExterior() const
|
|
|
|
{
|
|
|
|
Ptr::CellStore *currentCell = mWorldScene->getCurrentCell();
|
|
|
|
if (currentCell)
|
|
|
|
{
|
|
|
|
if (!(currentCell->cell->data.flags & ESM::Cell::Interior))
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-02-23 21:12:06 +01:00
|
|
|
bool World::isCellQuasiExterior() const
|
|
|
|
{
|
|
|
|
Ptr::CellStore *currentCell = mWorldScene->getCurrentCell();
|
|
|
|
if (currentCell)
|
|
|
|
{
|
|
|
|
if (!(currentCell->cell->data.flags & ESM::Cell::QuasiEx))
|
|
|
|
return false;
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-02-25 21:34:38 +01:00
|
|
|
int World::getCurrentWeather() const
|
|
|
|
{
|
|
|
|
return mWeatherManager->getWeatherID();
|
|
|
|
}
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-02-26 11:51:02 +01:00
|
|
|
void World::changeWeather(const std::string& region, const unsigned int id)
|
|
|
|
{
|
|
|
|
mWeatherManager->changeWeather(region, id);
|
|
|
|
}
|
2012-03-15 16:01:41 +01:00
|
|
|
|
2012-02-18 18:25:28 +01:00
|
|
|
OEngine::Render::Fader* World::getFader()
|
|
|
|
{
|
|
|
|
return mRendering->getFader();
|
|
|
|
}
|
2012-03-29 15:49:24 +02:00
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
Ogre::Vector2 World::getNorthVector (CellStore* cell)
|
2012-03-28 03:15:10 +02:00
|
|
|
{
|
2012-08-28 18:23:59 +02:00
|
|
|
MWWorld::CellRefList<ESM::Static>& statics = cell->statics;
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Static>* ref = statics.find("northmarker");
|
2012-03-28 03:15:10 +02:00
|
|
|
if (!ref)
|
|
|
|
return Vector2(0, 1);
|
|
|
|
Ogre::SceneNode* node = ref->mData.getBaseNode();
|
2012-03-29 17:07:59 +02:00
|
|
|
Vector3 dir = node->_getDerivedOrientation().yAxis();
|
2012-03-28 03:15:10 +02:00
|
|
|
Vector2 d = Vector2(dir.x, dir.z);
|
|
|
|
return d;
|
|
|
|
}
|
2012-03-30 15:31:07 +02:00
|
|
|
|
2012-08-28 17:30:34 +02:00
|
|
|
std::vector<World::DoorMarker> World::getDoorMarkers (CellStore* cell)
|
|
|
|
{
|
|
|
|
std::vector<World::DoorMarker> result;
|
|
|
|
|
2012-08-28 18:23:01 +02:00
|
|
|
MWWorld::CellRefList<ESM::Door>& doors = cell->doors;
|
|
|
|
std::list< MWWorld::LiveCellRef<ESM::Door> >& refList = doors.list;
|
2012-08-28 17:30:34 +02:00
|
|
|
for (std::list< MWWorld::LiveCellRef<ESM::Door> >::iterator it = refList.begin(); it != refList.end(); ++it)
|
|
|
|
{
|
2012-08-28 18:23:01 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Door>& ref = *it;
|
2012-08-28 17:30:34 +02:00
|
|
|
|
|
|
|
if (ref.ref.teleport)
|
|
|
|
{
|
|
|
|
World::DoorMarker newMarker;
|
|
|
|
|
|
|
|
std::string dest;
|
|
|
|
if (ref.ref.destCell != "")
|
|
|
|
{
|
|
|
|
// door leads to an interior, use interior name
|
|
|
|
dest = ref.ref.destCell;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// door leads to exterior, use cell name (if any), otherwise translated region name
|
|
|
|
int x,y;
|
|
|
|
positionToIndex (ref.ref.doorDest.pos[0], ref.ref.doorDest.pos[1], x, y);
|
|
|
|
const ESM::Cell* cell = mStore.cells.findExt(x,y);
|
|
|
|
if (cell->name != "")
|
|
|
|
dest = cell->name;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const ESM::Region* region = mStore.regions.search(cell->region);
|
|
|
|
dest = region->name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
newMarker.name = dest;
|
|
|
|
|
|
|
|
ESM::Position pos = ref.mData.getPosition ();
|
|
|
|
|
|
|
|
newMarker.x = pos.pos[0];
|
|
|
|
newMarker.y = pos.pos[1];
|
|
|
|
result.push_back(newMarker);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
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 15:49:24 +02:00
|
|
|
void World::setWaterHeight(const float height)
|
|
|
|
{
|
|
|
|
mRendering->setWaterHeight(height);
|
|
|
|
}
|
|
|
|
|
2012-03-29 18:33:08 +02:00
|
|
|
void World::toggleWater()
|
|
|
|
{
|
|
|
|
mRendering->toggleWater();
|
|
|
|
}
|
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
bool World::placeObject (const Ptr& object, float cursorX, float cursorY)
|
2012-05-14 17:41:17 +02:00
|
|
|
{
|
2012-05-15 16:47:23 +02:00
|
|
|
std::pair<bool, Ogre::Vector3> result = mPhysics->castRay(cursorX, cursorY);
|
|
|
|
|
|
|
|
if (!result.first)
|
|
|
|
return false;
|
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
CellStore* cell;
|
2012-05-15 16:47:23 +02:00
|
|
|
if (isCellExterior())
|
|
|
|
{
|
|
|
|
int cellX, cellY;
|
|
|
|
positionToIndex(result.second[0], -result.second[2], cellX, cellY);
|
|
|
|
cell = mCells.getExterior(cellX, cellY);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
cell = getPlayer().getPlayer().getCell();
|
|
|
|
|
2012-07-24 12:30:59 +04:00
|
|
|
ESM::Position pos = getPlayer().getPlayer().getRefData().getPosition();
|
2012-05-15 16:47:23 +02:00
|
|
|
pos.pos[0] = result.second[0];
|
|
|
|
pos.pos[1] = -result.second[2];
|
|
|
|
pos.pos[2] = result.second[1];
|
|
|
|
|
2012-08-08 14:51:33 +04:00
|
|
|
copyObjectToCell(object, *cell, pos);
|
2012-07-26 16:14:11 +04:00
|
|
|
object.getRefData().setCount(0);
|
2012-05-15 16:47:23 +02:00
|
|
|
|
|
|
|
return true;
|
2012-05-14 17:41:17 +02:00
|
|
|
}
|
|
|
|
|
2012-05-15 16:47:23 +02: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;
|
|
|
|
}
|
|
|
|
|
2012-07-25 10:47:59 +04:00
|
|
|
void
|
2012-08-08 14:51:33 +04:00
|
|
|
World::copyObjectToCell(const Ptr &object, CellStore &cell, const ESM::Position &pos)
|
2012-07-25 10:47:59 +04:00
|
|
|
{
|
2012-07-26 16:14:11 +04:00
|
|
|
/// \todo add searching correct cell for position specified
|
2012-07-25 17:18:17 +04:00
|
|
|
MWWorld::Ptr dropped =
|
2012-07-26 16:14:11 +04:00
|
|
|
MWWorld::Class::get(object).copyToCell(object, cell, pos);
|
|
|
|
|
|
|
|
Ogre::Vector3 min, max;
|
|
|
|
if (mPhysics->getObjectAABB(object, min, max)) {
|
|
|
|
float *pos = dropped.getRefData().getPosition().pos;
|
|
|
|
pos[0] -= (min.x + max.x) / 2;
|
|
|
|
pos[1] -= (min.y + max.y) / 2;
|
|
|
|
pos[2] -= min.z;
|
|
|
|
}
|
2012-07-25 17:18:17 +04:00
|
|
|
|
2012-07-26 16:14:11 +04:00
|
|
|
if (mWorldScene->isCellActive(cell)) {
|
2012-07-26 19:06:48 +04:00
|
|
|
if (dropped.getRefData().isEnabled()) {
|
|
|
|
mWorldScene->addObjectToScene(dropped);
|
|
|
|
}
|
2012-07-26 16:14:11 +04:00
|
|
|
std::string script = MWWorld::Class::get(dropped).getScript(dropped);
|
|
|
|
if (!script.empty()) {
|
|
|
|
mLocalScripts.add(script, dropped);
|
|
|
|
}
|
2012-07-25 10:47:59 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-03 01:06:38 +02:00
|
|
|
void World::dropObjectOnGround (const Ptr& object)
|
2012-05-15 16:47:23 +02:00
|
|
|
{
|
|
|
|
MWWorld::Ptr::CellStore* cell = getPlayer().getPlayer().getCell();
|
|
|
|
|
2012-07-25 20:25:53 +04:00
|
|
|
ESM::Position pos =
|
2012-07-24 12:30:59 +04:00
|
|
|
getPlayer().getPlayer().getRefData().getPosition();
|
2012-05-15 16:47:23 +02:00
|
|
|
|
2012-07-25 20:25:53 +04:00
|
|
|
Ogre::Vector3 orig =
|
|
|
|
Ogre::Vector3(pos.pos[0], pos.pos[1], pos.pos[2]);
|
|
|
|
Ogre::Vector3 dir = Ogre::Vector3(0, 0, -1);
|
2012-08-04 09:34:49 +02:00
|
|
|
|
2012-07-25 20:25:53 +04: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;
|
|
|
|
|
2012-08-08 14:51:33 +04:00
|
|
|
copyObjectToCell(object, *cell, pos);
|
2012-07-26 16:14:11 +04:00
|
|
|
object.getRefData().setCount(0);
|
2012-05-15 16:47:23 +02:00
|
|
|
}
|
2012-05-23 01:32:36 +02:00
|
|
|
|
|
|
|
void World::processChangedSettings(const Settings::CategorySettingVector& settings)
|
|
|
|
{
|
|
|
|
mRendering->processChangedSettings(settings);
|
|
|
|
}
|
2012-06-22 12:56:04 +02:00
|
|
|
|
|
|
|
void World::getTriangleBatchCount(unsigned int &triangles, unsigned int &batches)
|
|
|
|
{
|
|
|
|
mRendering->getTriangleBatchCount(triangles, batches);
|
|
|
|
}
|
2012-08-03 14:42:09 +04:00
|
|
|
|
|
|
|
bool
|
|
|
|
World::isSwimming(const MWWorld::Ptr &object)
|
|
|
|
{
|
|
|
|
/// \todo add check ifActor() - only actors can swim
|
|
|
|
float *fpos = object.getRefData().getPosition().pos;
|
|
|
|
Ogre::Vector3 pos(fpos[0], fpos[1], fpos[2]);
|
|
|
|
|
|
|
|
/// \fixme should rely on object height
|
|
|
|
pos.z += 30;
|
|
|
|
|
|
|
|
return isUnderwater(*object.getCell()->cell, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
World::isUnderwater(const ESM::Cell &cell, const Ogre::Vector3 &pos)
|
|
|
|
{
|
2012-08-04 22:03:14 +02:00
|
|
|
if (!(cell.data.flags & ESM::Cell::HasWater)) {
|
2012-08-03 14:42:09 +04:00
|
|
|
return false;
|
|
|
|
}
|
2012-08-04 11:54:42 +04:00
|
|
|
return pos.z < cell.water;
|
2012-08-03 14:42:09 +04:00
|
|
|
}
|
2012-08-14 20:33:29 +04:00
|
|
|
|
|
|
|
void World::renderPlayer()
|
|
|
|
{
|
|
|
|
mRendering->renderPlayer(mPlayer->getPlayer());
|
|
|
|
}
|
2010-07-02 09:38:22 +02:00
|
|
|
}
|