2010-07-02 07:38:22 +00:00
|
|
|
|
|
|
|
#include "world.hpp"
|
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
#include <cmath>
|
2010-07-03 13:04:00 +00:00
|
|
|
#include <iostream>
|
|
|
|
|
2010-07-27 12:05:05 +00:00
|
|
|
#include <components/bsa/bsa_archive.hpp>
|
2010-07-02 07:38:22 +00:00
|
|
|
|
2010-07-27 12:05:05 +00:00
|
|
|
#include "../mwrender/sky.hpp"
|
|
|
|
#include "../mwrender/interior.hpp"
|
2010-08-20 11:33:03 +00:00
|
|
|
#include "../mwrender/exterior.hpp"
|
2010-07-27 12:05:05 +00:00
|
|
|
|
|
|
|
#include "../mwmechanics/mechanicsmanager.hpp"
|
2010-07-02 07:38:22 +00:00
|
|
|
|
2010-08-14 07:29:38 +00:00
|
|
|
#include "../mwsound/soundmanager.hpp"
|
|
|
|
|
2010-07-09 14:07:03 +00:00
|
|
|
#include "ptr.hpp"
|
2010-07-27 12:05:05 +00:00
|
|
|
#include "environment.hpp"
|
2010-08-14 09:27:13 +00:00
|
|
|
#include "class.hpp"
|
2010-07-09 14:07:03 +00:00
|
|
|
|
2010-07-02 14:18:25 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
template<typename T>
|
2010-07-02 15:21:27 +00:00
|
|
|
void listCellScripts (const ESMS::ESMStore& store,
|
2010-07-10 11:19:04 +00:00
|
|
|
ESMS::CellRefList<T, MWWorld::RefData>& cellRefList, MWWorld::World::ScriptList& scriptList,
|
|
|
|
MWWorld::Ptr::CellStore *cell)
|
2010-07-02 14:18:25 +00:00
|
|
|
{
|
2010-07-03 13:41:20 +00:00
|
|
|
for (typename ESMS::CellRefList<T, MWWorld::RefData>::List::iterator iter (
|
2010-07-02 14:18:25 +00:00
|
|
|
cellRefList.list.begin());
|
|
|
|
iter!=cellRefList.list.end(); ++iter)
|
|
|
|
{
|
2010-08-04 12:04:22 +00:00
|
|
|
if (!iter->base->script.empty() && iter->mData.getCount())
|
2010-07-02 15:21:27 +00:00
|
|
|
{
|
|
|
|
if (const ESM::Script *script = store.scripts.find (iter->base->script))
|
2010-08-03 18:01:52 +00:00
|
|
|
{
|
2010-07-02 15:21:27 +00:00
|
|
|
iter->mData.setLocals (*script);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-02 15:21:27 +00:00
|
|
|
scriptList.push_back (
|
2010-07-10 11:19:04 +00:00
|
|
|
std::make_pair (iter->base->script, MWWorld::Ptr (&*iter, cell)));
|
2010-07-02 15:21:27 +00:00
|
|
|
}
|
|
|
|
}
|
2010-07-02 14:18:25 +00:00
|
|
|
}
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
ESMS::LiveCellRef<T, MWWorld::RefData> *searchViaHandle (const std::string& handle,
|
|
|
|
ESMS::CellRefList<T, MWWorld::RefData>& refList)
|
|
|
|
{
|
2010-08-03 20:06:48 +00:00
|
|
|
typedef typename ESMS::CellRefList<T, MWWorld::RefData>::List::iterator iterator;
|
|
|
|
|
|
|
|
for (iterator iter (refList.list.begin()); iter!=refList.list.end(); ++iter)
|
|
|
|
{
|
|
|
|
if (iter->mData.getHandle()==handle)
|
|
|
|
{
|
|
|
|
return &*iter;
|
|
|
|
}
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2010-07-02 14:18:25 +00:00
|
|
|
}
|
|
|
|
|
2010-07-03 13:41:20 +00:00
|
|
|
namespace MWWorld
|
2010-07-02 07:38:22 +00:00
|
|
|
{
|
2010-07-03 13:41:20 +00:00
|
|
|
void World::insertInteriorScripts (ESMS::CellStore<RefData>& cell)
|
2010-07-02 14:18:25 +00:00
|
|
|
{
|
2010-07-10 11:19:04 +00:00
|
|
|
listCellScripts (mStore, cell.activators, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.potions, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.appas, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.armors, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.books, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.clothes, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.containers, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.creatures, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.doors, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.ingreds, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.lights, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.lockpicks, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.miscItems, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.npcs, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.probes, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.repairs, mLocalScripts, &cell);
|
|
|
|
listCellScripts (mStore, cell.weapons, mLocalScripts, &cell);
|
2010-07-02 14:18:25 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-10 11:19:04 +00:00
|
|
|
Ptr World::getPtr (const std::string& name, Ptr::CellStore& cell)
|
2010-07-05 10:09:04 +00:00
|
|
|
{
|
|
|
|
if (ESMS::LiveCellRef<ESM::Activator, RefData> *ref = cell.activators.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-07-05 10:09:04 +00:00
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Potion, RefData> *ref = cell.potions.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Apparatus, RefData> *ref = cell.appas.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Armor, RefData> *ref = cell.armors.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Book, RefData> *ref = cell.books.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Clothing, RefData> *ref = cell.clothes.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Container, RefData> *ref = cell.containers.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Creature, RefData> *ref = cell.creatures.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Door, RefData> *ref = cell.doors.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Ingredient, RefData> *ref = cell.ingreds.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::CreatureLevList, RefData> *ref = cell.creatureLists.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::ItemLevList, RefData> *ref = cell.itemLists.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Light, RefData> *ref = cell.lights.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Tool, RefData> *ref = cell.lockpicks.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Misc, RefData> *ref = cell.miscItems.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::NPC, RefData> *ref = cell.npcs.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-08-03 13:24:44 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Probe, RefData> *ref = cell.probes.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-08-03 13:24:44 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Repair, RefData> *ref = cell.repairs.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Static, RefData> *ref = cell.statics.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Weapon, RefData> *ref = cell.weapons.find (name))
|
2010-07-10 11:19:04 +00:00
|
|
|
return Ptr (ref, &cell);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
return Ptr();
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
|
|
|
Ptr World::getPtrViaHandle (const std::string& handle, Ptr::CellStore& cell)
|
|
|
|
{
|
|
|
|
if (ESMS::LiveCellRef<ESM::Activator, RefData> *ref =
|
|
|
|
searchViaHandle (handle, cell.activators))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Potion, RefData> *ref = searchViaHandle (handle, cell.potions))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Apparatus, RefData> *ref = searchViaHandle (handle, cell.appas))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Armor, RefData> *ref = searchViaHandle (handle, cell.armors))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Book, RefData> *ref = searchViaHandle (handle, cell.books))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Clothing, RefData> *ref = searchViaHandle (handle, cell.clothes))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Container, RefData> *ref =
|
|
|
|
searchViaHandle (handle, cell.containers))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Creature, RefData> *ref =
|
|
|
|
searchViaHandle (handle, cell.creatures))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Door, RefData> *ref = searchViaHandle (handle, cell.doors))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Ingredient, RefData> *ref =
|
|
|
|
searchViaHandle (handle, cell.ingreds))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Light, RefData> *ref = searchViaHandle (handle, cell.lights))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Tool, RefData> *ref = searchViaHandle (handle, cell.lockpicks))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Misc, RefData> *ref = searchViaHandle (handle, cell.miscItems))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::NPC, RefData> *ref = searchViaHandle (handle, cell.npcs))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
2010-08-03 19:43:05 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Probe, RefData> *ref = searchViaHandle (handle, cell.probes))
|
2010-08-03 18:01:52 +00:00
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
2010-08-03 19:43:05 +00:00
|
|
|
if (ESMS::LiveCellRef<ESM::Repair, RefData> *ref = searchViaHandle (handle, cell.repairs))
|
2010-08-03 18:01:52 +00:00
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Static, RefData> *ref = searchViaHandle (handle, cell.statics))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
if (ESMS::LiveCellRef<ESM::Weapon, RefData> *ref = searchViaHandle (handle, cell.weapons))
|
|
|
|
return Ptr (ref, &cell);
|
|
|
|
|
|
|
|
return Ptr();
|
|
|
|
}
|
|
|
|
|
2010-07-10 11:19:04 +00:00
|
|
|
MWRender::CellRender *World::searchRender (Ptr::CellStore *store)
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
|
|
|
CellRenderCollection::iterator iter = mActiveCells.find (store);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-09 14:07:03 +00:00
|
|
|
if (iter!=mActiveCells.end())
|
|
|
|
{
|
|
|
|
return iter->second;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
iter = mBufferedCells.find (store);
|
|
|
|
if (iter!=mBufferedCells.end())
|
|
|
|
return iter->second;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-09 14:07:03 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
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-21 09:43:07 +00:00
|
|
|
void World::removeScripts (Ptr::CellStore *cell)
|
|
|
|
{
|
|
|
|
ScriptList::iterator iter = mLocalScripts.begin();
|
|
|
|
|
|
|
|
while (iter!=mLocalScripts.end())
|
|
|
|
{
|
|
|
|
if (iter->second.getCell()==cell)
|
|
|
|
mLocalScripts.erase (iter++);
|
|
|
|
else
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void World::unloadCell (CellRenderCollection::iterator iter)
|
|
|
|
{
|
|
|
|
removeScripts (iter->first);
|
|
|
|
mEnvironment.mMechanicsManager->dropActors (iter->first);
|
|
|
|
iter->second->destroy();
|
|
|
|
mEnvironment.mSoundManager->stopSound (iter->first);
|
|
|
|
delete iter->second;
|
|
|
|
mActiveCells.erase (iter);
|
|
|
|
}
|
|
|
|
|
2010-08-22 18:55:22 +00:00
|
|
|
void World::loadCell (Ptr::CellStore *cell, MWRender::CellRender *render)
|
|
|
|
{
|
|
|
|
// register local scripts
|
|
|
|
insertInteriorScripts (*cell);
|
|
|
|
|
|
|
|
// This connects the cell data with the rendering scene.
|
|
|
|
std::pair<CellRenderCollection::iterator, bool> result =
|
|
|
|
mActiveCells.insert (std::make_pair (cell, render));
|
|
|
|
|
|
|
|
if (result.second)
|
|
|
|
{
|
|
|
|
// Load the cell and insert it into the renderer
|
|
|
|
result.first->second->show();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void World::playerCellChange (Ptr::CellStore *cell, const ESM::Position& position)
|
|
|
|
{
|
|
|
|
mPlayerPos->setPos (position.pos[0], position.pos[1], position.pos[2], true);
|
|
|
|
mPlayerPos->setCell (cell);
|
|
|
|
// TODO orientation
|
|
|
|
|
|
|
|
mEnvironment.mMechanicsManager->addActor (mPlayerPos->getPlayer());
|
|
|
|
mEnvironment.mMechanicsManager->watchActor (mPlayerPos->getPlayer());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void World::adjustSky()
|
|
|
|
{
|
|
|
|
if (mSky)
|
|
|
|
{
|
|
|
|
toggleSky();
|
|
|
|
// TODO set weather
|
|
|
|
toggleSky();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-16 12:26:46 +00:00
|
|
|
World::World (OEngine::Render::OgreRenderer& renderer, const boost::filesystem::path& dataDir,
|
2010-07-27 12:05:05 +00:00
|
|
|
const std::string& master, bool newGame, Environment& environment)
|
2010-07-18 17:48:02 +00:00
|
|
|
: mSkyManager (0), mScene (renderer), mPlayerPos (0), mCurrentCell (0), mGlobalVariables (0),
|
2010-07-27 12:05:05 +00:00
|
|
|
mSky (false), mCellChanged (false), mEnvironment (environment)
|
2010-08-03 18:01:52 +00:00
|
|
|
{
|
2010-07-02 07:38:22 +00:00
|
|
|
boost::filesystem::path masterPath (dataDir);
|
|
|
|
masterPath /= master;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-02 07:38:22 +00:00
|
|
|
std::cout << "Loading ESM " << masterPath.string() << "\n";
|
|
|
|
|
|
|
|
// This parses the ESM file and loads a sample cell
|
|
|
|
mEsm.open (masterPath.file_string());
|
|
|
|
mStore.load (mEsm);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-08-21 10:31:04 +00:00
|
|
|
mPlayerPos = new MWRender::PlayerPos (mScene.getCamera(), mStore.npcs.find ("player"), *this);
|
2010-07-05 10:09:04 +00:00
|
|
|
|
2010-07-04 11:33:33 +00:00
|
|
|
// global variables
|
2010-07-18 14:48:01 +00:00
|
|
|
mGlobalVariables = new Globals (mStore);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-04 14:00:32 +00:00
|
|
|
if (newGame)
|
2010-08-03 18:01:52 +00:00
|
|
|
{
|
2010-07-04 14:00:32 +00:00
|
|
|
// set new game mark
|
2010-07-18 14:48:01 +00:00
|
|
|
mGlobalVariables->setInt ("chargenstate", 1);
|
2010-07-04 14:00:32 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
mSkyManager =
|
2010-08-03 18:01:52 +00:00
|
|
|
MWRender::SkyManager::create(renderer.getWindow(), mScene.getCamera());
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-02 07:38:22 +00:00
|
|
|
World::~World()
|
|
|
|
{
|
|
|
|
for (CellRenderCollection::iterator iter (mActiveCells.begin());
|
|
|
|
iter!=mActiveCells.end(); ++iter)
|
|
|
|
delete iter->second;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-02 07:38:22 +00:00
|
|
|
for (CellRenderCollection::iterator iter (mBufferedCells.begin());
|
|
|
|
iter!=mBufferedCells.end(); ++iter)
|
|
|
|
delete iter->second;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
delete mPlayerPos;
|
2010-07-02 07:38:22 +00:00
|
|
|
delete mSkyManager;
|
2010-07-18 14:48:01 +00:00
|
|
|
delete mGlobalVariables;
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-02 07:38:22 +00:00
|
|
|
MWRender::PlayerPos& World::getPlayerPos()
|
|
|
|
{
|
2010-07-05 10:09:04 +00:00
|
|
|
return *mPlayerPos;
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-02 14:18:25 +00:00
|
|
|
ESMS::ESMStore& World::getStore()
|
|
|
|
{
|
|
|
|
return mStore;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-02 14:18:25 +00:00
|
|
|
const World::ScriptList& World::getLocalScripts() const
|
|
|
|
{
|
|
|
|
return mLocalScripts;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-03 10:12:13 +00:00
|
|
|
bool World::hasCellChanged() const
|
|
|
|
{
|
2010-07-22 10:29:23 +00:00
|
|
|
return mCellChanged;
|
2010-07-03 10:12:13 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 14:48:01 +00:00
|
|
|
Globals::Data& World::getGlobalVariable (const std::string& name)
|
2010-07-04 11:33:33 +00:00
|
|
|
{
|
2010-07-18 14:48:01 +00:00
|
|
|
return (*mGlobalVariables)[name];
|
2010-07-04 11:33:33 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-21 13:01:35 +00:00
|
|
|
char World::getGlobalVariableType (const std::string& name) const
|
|
|
|
{
|
|
|
|
return mGlobalVariables->getType (name);
|
2010-08-03 18:01:52 +00:00
|
|
|
}
|
|
|
|
|
2010-07-10 11:19:04 +00:00
|
|
|
Ptr World::getPtr (const std::string& name, bool activeOnly)
|
2010-07-05 10:09:04 +00:00
|
|
|
{
|
|
|
|
// the player is always in an active cell.
|
|
|
|
if (name=="player")
|
|
|
|
{
|
2010-07-26 11:09:44 +00:00
|
|
|
return mPlayerPos->getPlayer();
|
2010-07-05 10:09:04 +00:00
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
// active cells
|
|
|
|
for (CellRenderCollection::iterator iter (mActiveCells.begin());
|
|
|
|
iter!=mActiveCells.end(); ++iter)
|
|
|
|
{
|
|
|
|
Ptr ptr = getPtr (name, *iter->first);
|
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
|
|
|
|
2010-07-05 10:09:04 +00:00
|
|
|
if (!activeOnly)
|
|
|
|
{
|
|
|
|
// TODO: inactive cells
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
// TODO player
|
|
|
|
|
|
|
|
for (CellRenderCollection::iterator iter (mActiveCells.begin());
|
|
|
|
iter!=mActiveCells.end(); ++iter)
|
|
|
|
{
|
|
|
|
Ptr ptr = getPtrViaHandle (handle, *iter->first);
|
|
|
|
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw std::runtime_error ("unknown Ogre handle: " + handle);
|
|
|
|
}
|
|
|
|
|
2010-07-10 11:19:04 +00:00
|
|
|
void World::enable (Ptr reference)
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
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();
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-10 11:19:04 +00:00
|
|
|
if (MWRender::CellRender *render = searchRender (reference.getCell()))
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
2010-07-10 11:19:04 +00:00
|
|
|
render->enable (reference.getRefData().getHandle());
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-08-14 09:27:13 +00:00
|
|
|
if (mActiveCells.find (reference.getCell())!=mActiveCells.end())
|
|
|
|
Class::get (reference).enable (reference, mEnvironment);
|
2010-07-09 14:07:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-10 11:19:04 +00:00
|
|
|
void World::disable (Ptr reference)
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
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();
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-10 11:19:04 +00:00
|
|
|
if (MWRender::CellRender *render = searchRender (reference.getCell()))
|
2010-07-09 14:07:03 +00:00
|
|
|
{
|
2010-07-10 11:19:04 +00:00
|
|
|
render->disable (reference.getRefData().getHandle());
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-08-14 09:27:13 +00:00
|
|
|
if (mActiveCells.find (reference.getCell())!=mActiveCells.end())
|
2010-08-14 09:39:32 +00:00
|
|
|
{
|
2010-08-14 09:27:13 +00:00
|
|
|
Class::get (reference).disable (reference, mEnvironment);
|
2010-08-14 09:39:32 +00:00
|
|
|
mEnvironment.mSoundManager->stopSound3D (reference);
|
|
|
|
}
|
2010-07-09 14:07:03 +00:00
|
|
|
}
|
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)
|
|
|
|
{
|
|
|
|
hours += mGlobalVariables->getFloat ("gamehour");
|
2010-07-22 09:48:27 +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)
|
|
|
|
mGlobalVariables->setInt ("dayspassed", days + mGlobalVariables->getInt ("dayspassed"));
|
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
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
mGlobalVariables->setFloat ("gamehour", hour);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
mSkyManager->setHour (hour);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 16:29:16 +00:00
|
|
|
if (days>0)
|
2010-07-22 09:48:27 +00:00
|
|
|
setDay (days + mGlobalVariables->getInt ("day"));
|
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)
|
|
|
|
{
|
|
|
|
if (day<0)
|
|
|
|
day = 0;
|
2010-07-22 09:48:27 +00:00
|
|
|
|
|
|
|
int month = mGlobalVariables->getInt ("month");
|
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);
|
2010-07-22 09:48:27 +00:00
|
|
|
if (day<days)
|
|
|
|
break;
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
if (month<11)
|
|
|
|
{
|
|
|
|
++month;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
month = 0;
|
|
|
|
mGlobalVariables->setInt ("year", mGlobalVariables->getInt ("year")+1);
|
|
|
|
}
|
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
|
|
|
}
|
|
|
|
|
|
|
|
mGlobalVariables->setInt ("day", day);
|
2010-07-22 09:48:27 +00:00
|
|
|
mGlobalVariables->setInt ("month", month);
|
|
|
|
|
|
|
|
mSkyManager->setDate (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
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
if (mGlobalVariables->getInt ("day")>=days)
|
|
|
|
mGlobalVariables->setInt ("day", days-1);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
mGlobalVariables->setInt ("month", month);
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 09:48:27 +00:00
|
|
|
if (years>0)
|
|
|
|
mGlobalVariables->setInt ("year", years+mGlobalVariables->getInt ("year"));
|
|
|
|
|
|
|
|
mSkyManager->setDate (mGlobalVariables->getInt ("day"), month);
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
void World::toggleSky()
|
|
|
|
{
|
|
|
|
if (mSky)
|
|
|
|
{
|
|
|
|
mSky = false;
|
|
|
|
mSkyManager->disable();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mSky = true;
|
|
|
|
// TODO check for extorior or interior with sky.
|
|
|
|
mSkyManager->setHour (mGlobalVariables->getFloat ("gamehour"));
|
2010-07-22 09:48:27 +00:00
|
|
|
mSkyManager->setDate (mGlobalVariables->getInt ("day"),
|
|
|
|
mGlobalVariables->getInt ("month"));
|
2010-07-18 17:48:02 +00:00
|
|
|
mSkyManager->enable();
|
|
|
|
}
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
int World::getMasserPhase() const
|
|
|
|
{
|
|
|
|
return mSkyManager->getMasserPhase();
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
int World::getSecundaPhase() const
|
|
|
|
{
|
|
|
|
return mSkyManager->getSecundaPhase();
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-18 17:48:02 +00:00
|
|
|
void World::setMoonColour (bool red)
|
|
|
|
{
|
2010-08-03 18:01:52 +00:00
|
|
|
mSkyManager->setMoonColour (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
|
|
|
|
{
|
|
|
|
return mGlobalVariables->getInt ("timescale");
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 10:29:23 +00:00
|
|
|
void World::changeCell (const std::string& cellName, const ESM::Position& position)
|
|
|
|
{
|
|
|
|
// remove active
|
|
|
|
CellRenderCollection::iterator active = mActiveCells.begin();
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 10:29:23 +00:00
|
|
|
if (active!=mActiveCells.end())
|
|
|
|
{
|
2010-08-21 09:43:07 +00:00
|
|
|
unloadCell (active);
|
2010-07-22 10:29:23 +00:00
|
|
|
}
|
|
|
|
|
2010-08-22 18:55:22 +00:00
|
|
|
// Load cell.
|
|
|
|
mInteriors[cellName].loadInt (cellName, mStore, mEsm);
|
|
|
|
Ptr::CellStore *cell = &mInteriors[cellName];
|
2010-07-22 10:29:23 +00:00
|
|
|
|
2010-08-22 18:55:22 +00:00
|
|
|
loadCell (cell, new MWRender::InteriorCellRender (*cell, mEnvironment, mScene));
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-08-22 18:55:22 +00:00
|
|
|
// adjust player
|
|
|
|
playerCellChange (cell, position);
|
2010-07-27 12:05:05 +00:00
|
|
|
|
|
|
|
// Sky system
|
2010-08-22 18:55:22 +00:00
|
|
|
adjustSky();
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-07-22 10:29:23 +00:00
|
|
|
mCellChanged = true;
|
|
|
|
}
|
2010-08-03 18:01:52 +00:00
|
|
|
|
2010-08-20 11:33:03 +00:00
|
|
|
void World::changeCell (int X, int Y, const ESM::Position& position)
|
|
|
|
{
|
|
|
|
// remove active
|
|
|
|
CellRenderCollection::iterator active = mActiveCells.begin();
|
|
|
|
|
|
|
|
if (active!=mActiveCells.end())
|
|
|
|
{
|
2010-08-21 09:43:07 +00:00
|
|
|
unloadCell (active);
|
2010-08-20 11:33:03 +00:00
|
|
|
}
|
|
|
|
|
2010-08-22 18:55:22 +00:00
|
|
|
// Load cell.
|
|
|
|
mExteriors[std::make_pair (X, Y)].loadExt (X, Y, mStore, mEsm);
|
|
|
|
Ptr::CellStore *cell = &mExteriors[std::make_pair (X, Y)];
|
2010-08-20 11:33:03 +00:00
|
|
|
|
2010-08-22 18:55:22 +00:00
|
|
|
loadCell (cell, new MWRender::ExteriorCellRender (*cell, mEnvironment, mScene));
|
2010-08-20 11:33:03 +00:00
|
|
|
|
2010-08-22 18:55:22 +00:00
|
|
|
// adjust player
|
|
|
|
playerCellChange (cell, position);
|
2010-08-20 11:33:03 +00:00
|
|
|
|
|
|
|
// Sky system
|
2010-08-22 18:55:22 +00:00
|
|
|
adjustSky();
|
2010-08-20 11:33:03 +00:00
|
|
|
|
|
|
|
mCellChanged = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void World::changeToExteriorCell (const ESM::Position& position)
|
|
|
|
{
|
2010-08-22 19:30:48 +00:00
|
|
|
int x = 0;
|
|
|
|
int y = 0;
|
2010-08-20 11:33:03 +00:00
|
|
|
|
2010-08-22 19:30:48 +00:00
|
|
|
positionToIndex (position.pos[0], position.pos[1], x, y);
|
2010-08-20 11:33:03 +00:00
|
|
|
|
|
|
|
changeCell (x, y, position);
|
|
|
|
}
|
|
|
|
|
2010-07-22 10:29:23 +00:00
|
|
|
void World::markCellAsUnchanged()
|
|
|
|
{
|
2010-08-03 18:01:52 +00:00
|
|
|
mCellChanged = false;
|
2010-07-22 10:29:23 +00:00
|
|
|
}
|
2010-08-05 11:29:49 +00:00
|
|
|
|
|
|
|
std::string World::getFacedHandle()
|
|
|
|
{
|
2010-08-25 07:19:15 +00:00
|
|
|
std::pair<std::string, float> result = mScene.getFacedHandle (*this);
|
2010-08-05 11:29:49 +00:00
|
|
|
|
|
|
|
if (result.first.empty() ||
|
|
|
|
result.second>getStore().gameSettings.find ("iMaxActivateDist")->i)
|
|
|
|
return "";
|
|
|
|
|
|
|
|
return result.first;
|
|
|
|
}
|
2010-08-07 18:25:17 +00:00
|
|
|
|
|
|
|
void World::deleteObject (Ptr ptr)
|
|
|
|
{
|
|
|
|
if (ptr.getRefData().getCount()>0)
|
|
|
|
{
|
|
|
|
ptr.getRefData().setCount (0);
|
|
|
|
|
|
|
|
if (MWRender::CellRender *render = searchRender (ptr.getCell()))
|
|
|
|
{
|
|
|
|
render->deleteObject (ptr.getRefData().getHandle());
|
|
|
|
ptr.getRefData().setHandle ("");
|
|
|
|
|
2010-08-14 09:27:13 +00:00
|
|
|
if (mActiveCells.find (ptr.getCell())!=mActiveCells.end())
|
2010-08-14 09:39:32 +00:00
|
|
|
{
|
2010-08-14 09:27:13 +00:00
|
|
|
Class::get (ptr).disable (ptr, mEnvironment);
|
2010-08-14 09:39:32 +00:00
|
|
|
mEnvironment.mSoundManager->stopSound3D (ptr);
|
|
|
|
}
|
2010-08-07 18:25:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-08-21 10:31:04 +00:00
|
|
|
|
|
|
|
void World::moveObject (Ptr ptr, float x, float y, float z)
|
|
|
|
{
|
|
|
|
ptr.getCellRef().pos.pos[0] = x;
|
|
|
|
ptr.getCellRef().pos.pos[1] = y;
|
|
|
|
ptr.getCellRef().pos.pos[2] = z;
|
|
|
|
|
2010-08-21 10:41:59 +00:00
|
|
|
if (ptr==mPlayerPos->getPlayer())
|
|
|
|
{
|
|
|
|
CellRenderCollection::iterator active = mActiveCells.begin();
|
|
|
|
|
|
|
|
if (active!=mActiveCells.end())
|
|
|
|
{
|
|
|
|
if (!(active->first->cell->data.flags & ESM::Cell::Interior))
|
|
|
|
{
|
|
|
|
// exterior -> adjust loaded cells
|
2010-08-22 19:30:48 +00:00
|
|
|
int cellX = 0;
|
|
|
|
int cellY = 0;
|
2010-08-21 10:41:59 +00:00
|
|
|
|
2010-08-22 19:30:48 +00:00
|
|
|
positionToIndex (x, y, cellX, cellY);
|
2010-08-21 10:41:59 +00:00
|
|
|
|
|
|
|
if (active->first->cell->data.gridX!=cellX || active->first->cell->data.gridY!=cellY)
|
|
|
|
{
|
|
|
|
changeCell (cellX, cellY, mPlayerPos->getPlayer().getCellRef().pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-21 10:31:04 +00:00
|
|
|
// TODO cell change for non-player ref
|
|
|
|
}
|
2010-08-22 19:30:48 +00:00
|
|
|
|
|
|
|
void World::indexToPosition (int cellX, int cellY, float &x, float &y) const
|
|
|
|
{
|
|
|
|
const int cellSize = 8192;
|
|
|
|
|
|
|
|
x = cellSize * cellX;
|
|
|
|
y = cellSize * cellY;
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2010-07-02 07:38:22 +00:00
|
|
|
}
|