#include "worldimp.hpp" #ifdef _WIN32 #include #elif defined HAVE_UNORDERED_MAP #include #else #include #endif #include #include #include #include #include #include #include "../mwbase/environment.hpp" #include "../mwbase/soundmanager.hpp" #include "../mwbase/mechanicsmanager.hpp" #include "../mwbase/windowmanager.hpp" #include "../mwbase/scriptmanager.hpp" #include "../mwmechanics/creaturestats.hpp" #include "../mwmechanics/movement.hpp" #include "../mwmechanics/npcstats.hpp" #include "../mwmechanics/spellcasting.hpp" #include "../mwmechanics/levelledlist.hpp" #include "../mwrender/sky.hpp" #include "../mwrender/animation.hpp" #include "../mwclass/door.hpp" #include "player.hpp" #include "manualref.hpp" #include "cellfunctors.hpp" #include "containerstore.hpp" #include "inventorystore.hpp" #include "actionteleport.hpp" #include "contentloader.hpp" #include "esmloader.hpp" #include "omwloader.hpp" using namespace Ogre; namespace { /* // NOTE this code is never instantiated (proper copy in localscripts.cpp), // so this commented out to not produce syntactic errors template void listCellScripts (const MWWorld::ESMStore& store, MWWorld::CellRefList& cellRefList, MWWorld::LocalScripts& localScripts, MWWorld::Ptr::CellStore *cell) { for (typename MWWorld::CellRefList::List::iterator iter ( cellRefList.mList.begin()); iter!=cellRefList.mList.end(); ++iter) { if (!iter->mBase->mScript.empty() && iter->mData.getCount()) { if (const ESM::Script *script = store.get().find (iter->mBase->mScript)) { iter->mData.setLocals (*script); localScripts.add (iter->mBase->mScript, MWWorld::Ptr (&*iter, cell)); } } } } */ template MWWorld::LiveCellRef *searchViaHandle (const std::string& handle, MWWorld::CellRefList& refList) { typedef typename MWWorld::CellRefList::List::iterator iterator; for (iterator iter (refList.mList.begin()); iter!=refList.mList.end(); ++iter) { if (iter->mData.getCount() > 0 && iter->mData.getBaseNode()){ if (iter->mData.getHandle()==handle) { return &*iter; } } } return 0; } } namespace MWWorld { 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) { LoadersContainer::iterator it(mLoaders.find(Misc::StringUtils::lowerCase(filepath.extension().string()))); 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 LoadersContainer; LoadersContainer mLoaders; }; Ptr World::getPtrViaHandle (const std::string& handle, Ptr::CellStore& cell) { if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mActivators)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mPotions)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mAppas)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mArmors)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mBooks)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mClothes)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mContainers)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mCreatures)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mDoors)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mIngreds)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mLights)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mLockpicks)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mMiscItems)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mNpcs)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mProbes)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mRepairs)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mStatics)) return Ptr (ref, &cell); if (MWWorld::LiveCellRef *ref = searchViaHandle (handle, cell.mWeapons)) return Ptr (ref, &cell); return Ptr(); } 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; } throw std::runtime_error ("month out of range"); } void World::adjustSky() { if (mSky && (isCellExterior() || isCellQuasiExterior())) { mRendering->skySetHour (mGlobalVariables->getFloat ("gamehour")); mRendering->skySetDate (mGlobalVariables->getInt ("day"), mGlobalVariables->getInt ("month")); mRendering->skyEnable(); } else mRendering->skyDisable(); } World::World (OEngine::Render::OgreRenderer& renderer, const Files::Collections& fileCollections, const std::vector& contentFiles, const boost::filesystem::path& resDir, const boost::filesystem::path& cacheDir, ToUTF8::Utf8Encoder* encoder, const std::map& fallbackMap, int mActivationDistanceOverride) : mPlayer (0), mLocalScripts (mStore), mGlobalVariables (0), mSky (true), mCells (mStore, mEsm), mActivationDistanceOverride (mActivationDistanceOverride), mFallback(fallbackMap), mPlayIntro(0), mTeleportEnabled(true), mLevitationEnabled(false), mFacedDistance(FLT_MAX), mGodMode(false), mGoToJail(false) { mPhysics = new PhysicsSystem(renderer); mPhysEngine = mPhysics->getEngine(); mRendering = new MWRender::RenderingManager(renderer, resDir, cacheDir, mPhysEngine,&mFallback); mPhysEngine->setSceneManager(renderer.getScene()); mWeatherManager = new MWWorld::WeatherManager(mRendering,&mFallback); // NOTE: We might need to reserve one more for the running game / save. mEsm.resize(contentFiles.size()); Loading::Listener* listener = MWBase::Environment::get().getWindowManager()->getLoadingScreen(); listener->loadingOn(); GameContentLoader gameContentLoader(*listener); EsmLoader esmLoader(mStore, mEsm, encoder, *listener); OmwLoader omwLoader(*listener); gameContentLoader.addLoader(".esm", &esmLoader); gameContentLoader.addLoader(".esp", &esmLoader); gameContentLoader.addLoader(".omwgame", &omwLoader); gameContentLoader.addLoader(".omwaddon", &omwLoader); loadContentFiles(fileCollections, contentFiles, gameContentLoader); listener->loadingOff(); // insert records that may not be present in all versions of MW if (mEsm[0].getFormat() == 0) ensureNeededRecords(); mStore.setUp(); mStore.movePlayerRecord(); mGlobalVariables = new Globals (mStore); mWorldScene = new Scene(*mRendering, mPhysics); } void World::startNewGame() { mGoToJail = false; mLevitationEnabled = true; mTeleportEnabled = true; mWorldScene->changeToVoid(); mStore.clearDynamic(); mStore.setUp(); mCells.clear(); // Rebuild player setupPlayer(); mPlayer->setCell(NULL); MWWorld::Ptr player = mPlayer->getPlayer(); // removes NpcStats, ContainerStore etc player.getRefData().setCustomData(NULL); renderPlayer(); mRendering->resetCamera(); // make sure to do this so that local scripts from items that were in the players inventory are removed mLocalScripts.clear(); MWBase::Environment::get().getWindowManager()->updatePlayer(); 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); // enable collision if(!mPhysics->toggleCollisionMode()) mPhysics->toggleCollisionMode(); // FIXME: should be set to 1, but the sound manager won't pause newly started sounds mPlayIntro = 2; // global variables delete mGlobalVariables; mGlobalVariables = new Globals (mStore); // set new game mark mGlobalVariables->setInt ("chargenstate", 1); mGlobalVariables->setInt ("pcrace", 3); // we don't want old weather to persist on a new game delete mWeatherManager; mWeatherManager = new MWWorld::WeatherManager(mRendering,&mFallback); MWBase::Environment::get().getScriptManager()->resetGlobalScripts(); } void World::ensureNeededRecords() { if (!mStore.get().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().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().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); } } World::~World() { delete mWeatherManager; delete mWorldScene; delete mGlobalVariables; delete mRendering; delete mPhysics; delete mPlayer; } const ESM::Cell *World::getExterior (const std::string& cellName) const { // first try named cells const ESM::Cell *cell = mStore.get().searchExtByName (cellName); if (cell != 0) { return cell; } // didn't work -> now check for regions const MWWorld::Store ®ions = mStore.get(); MWWorld::Store::iterator it = regions.begin(); for (; it != regions.end(); ++it) { if (Misc::StringUtils::ciEqual(cellName, it->mName)) { return mStore.get().searchExtByRegion(it->mId); } } return 0; } const MWWorld::Fallback *World::getFallback() const { return &mFallback; } Ptr::CellStore *World::getExterior (int x, int y) { return mCells.getExterior (x, y); } Ptr::CellStore *World::getInterior (const std::string& name) { return mCells.getInterior (name); } MWWorld::Player& World::getPlayer() { return *mPlayer; } const MWWorld::ESMStore& World::getStore() const { return mStore; } std::vector& World::getEsmReader() { return mEsm; } LocalScripts& World::getLocalScripts() { return mLocalScripts; } bool World::hasCellChanged() const { return mWorldScene->hasCellChanged(); } Globals::Data& World::getGlobalVariable (const std::string& name) { return (*mGlobalVariables)[name]; } Globals::Data World::getGlobalVariable (const std::string& name) const { return (*mGlobalVariables)[name]; } char World::getGlobalVariableType (const std::string& name) const { return mGlobalVariables->getType (name); } std::vector World::getGlobals () const { return mGlobalVariables->getGlobals(); } std::string World::getCurrentCellName () const { std::string name; Ptr::CellStore *cell = mWorldScene->getCurrentCell(); if (cell->mCell->isExterior()) { if (cell->mCell->mName != "") { name = cell->mCell->mName; } else { const ESM::Region* region = MWBase::Environment::get().getWorld()->getStore().get().search(cell->mCell->mRegion); if (region) name = region->mName; else { const ESM::GameSetting *setting = MWBase::Environment::get().getWorld()->getStore().get().search("sDefaultCellname"); if (setting && setting->mValue.getType()==ESM::VT_String) name = setting->mValue.getString(); } } } else { name = cell->mCell->mName; } return name; } void World::removeRefScript (MWWorld::RefData *ref) { mLocalScripts.remove (ref); } Ptr World::searchPtr (const std::string& name, bool activeOnly) { Ptr ret; // the player is always in an active cell. if (name=="player") { return mPlayer->getPlayer(); } Ptr ptr = Class::get (mPlayer->getPlayer()). getContainerStore (mPlayer->getPlayer()).search (name); if (!ptr.isEmpty()) return ptr; std::string lowerCaseName = Misc::StringUtils::lowerCase(name); // active cells for (Scene::CellStoreCollection::const_iterator iter (mWorldScene->getActiveCells().begin()); iter!=mWorldScene->getActiveCells().end(); ++iter) { Ptr::CellStore* cellstore = *iter; Ptr ptr = mCells.getPtr (lowerCaseName, *cellstore, true); if (!ptr.isEmpty()) return ptr; } if (!activeOnly) { ret = mCells.getPtr (lowerCaseName); } return ret; } Ptr World::getPtr (const std::string& name, bool activeOnly) { Ptr ret = searchPtr(name, activeOnly); if (!ret.isEmpty()) return ret; throw std::runtime_error ("unknown ID: " + name); } Ptr World::getPtrViaHandle (const std::string& handle) { Ptr res = searchPtrViaHandle (handle); if (res.isEmpty ()) throw std::runtime_error ("unknown Ogre handle: " + handle); return res; } Ptr World::searchPtrViaHandle (const std::string& handle) { if (mPlayer->getPlayer().getRefData().getHandle()==handle) return mPlayer->getPlayer(); for (Scene::CellStoreCollection::const_iterator iter (mWorldScene->getActiveCells().begin()); iter!=mWorldScene->getActiveCells().end(); ++iter) { Ptr::CellStore* cellstore = *iter; Ptr ptr = getPtrViaHandle (handle, *cellstore); if (!ptr.isEmpty()) return ptr; } return MWWorld::Ptr(); } void World::addContainerScripts(const Ptr& reference, Ptr::CellStore * cell) { 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); } } } } void World::enable (const Ptr& reference) { if (!reference.getRefData().isEnabled()) { reference.getRefData().enable(); if(mWorldScene->getActiveCells().find (reference.getCell()) != mWorldScene->getActiveCells().end() && reference.getRefData().getCount()) mWorldScene->addObjectToScene (reference); } } 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) { std::string script = MWWorld::Class::get(*it).getScript(*it); if(script != "") { MWWorld::Ptr item = *it; mLocalScripts.remove (item); } } } } void World::disable (const Ptr& reference) { if (reference.getRefData().isEnabled()) { reference.getRefData().disable(); if(mWorldScene->getActiveCells().find (reference.getCell())!=mWorldScene->getActiveCells().end() && reference.getRefData().getCount()) mWorldScene->removeObjectFromScene (reference); } } void World::advanceTime (double hours) { MWBase::Environment::get().getMechanicsManager()->advanceTime(hours*3600); mWeatherManager->advanceTime (hours); hours += mGlobalVariables->getFloat ("gamehour"); setHour (hours); int days = hours / 24; if (days>0) mGlobalVariables->setInt ("dayspassed", days + mGlobalVariables->getInt ("dayspassed")); } void World::setHour (double hour) { if (hour<0) hour = 0; int days = hour / 24; hour = std::fmod (hour, 24); mGlobalVariables->setFloat ("gamehour", hour); mRendering->skySetHour (hour); mWeatherManager->setHour (hour); if (days>0) setDay (days + mGlobalVariables->getInt ("day")); } void World::setDay (int day) { if (day<1) day = 1; int month = mGlobalVariables->getInt ("month"); while (true) { int days = getDaysPerMonth (month); if (day<=days) break; if (month<11) { ++month; } else { month = 0; mGlobalVariables->setInt ("year", mGlobalVariables->getInt ("year")+1); } day -= days; } mGlobalVariables->setInt ("day", day); mGlobalVariables->setInt ("month", month); mRendering->skySetDate (day, month); mWeatherManager->setDate (day, month); } void World::setMonth (int month) { if (month<0) month = 0; int years = month / 12; month = month % 12; int days = getDaysPerMonth (month); if (mGlobalVariables->getInt ("day")>days) mGlobalVariables->setInt ("day", days); mGlobalVariables->setInt ("month", month); if (years>0) mGlobalVariables->setInt ("year", years+mGlobalVariables->getInt ("year")); mRendering->skySetDate (mGlobalVariables->getInt ("day"), month); } int World::getDay() { return mGlobalVariables->getInt("day"); } int World::getMonth() { return mGlobalVariables->getInt("month"); } TimeStamp World::getTimeStamp() const { return TimeStamp (mGlobalVariables->getFloat ("gamehour"), mGlobalVariables->getInt ("dayspassed")); } bool World::toggleSky() { if (mSky) { mSky = false; mRendering->skyDisable(); return false; } else { mSky = true; mRendering->skyEnable(); return true; } } int World::getMasserPhase() const { return mRendering->skyGetMasserPhase(); } int World::getSecundaPhase() const { return mRendering->skyGetSecundaPhase(); } void World::setMoonColour (bool red) { mRendering->skySetMoonColour (red); } float World::getTimeScaleFactor() const { return mGlobalVariables->getFloat ("timescale"); } void World::changeToInteriorCell (const std::string& cellName, const ESM::Position& position) { removeContainerScripts(getPlayerPtr()); mWorldScene->changeToInteriorCell(cellName, position); addContainerScripts(getPlayerPtr(), getPlayerPtr().getCell()); } void World::changeToExteriorCell (const ESM::Position& position) { removeContainerScripts(getPlayerPtr()); mWorldScene->changeToExteriorCell(position); addContainerScripts(getPlayerPtr(), getPlayerPtr().getCell()); } void World::markCellAsUnchanged() { return mWorldScene->markCellAsUnchanged(); } float World::getMaxActivationDistance () { if (mActivationDistanceOverride >= 0) return mActivationDistanceOverride; return (std::max) (getNpcActivationDistance (), getObjectActivationDistance ()); } float World::getNpcActivationDistance () { if (mActivationDistanceOverride >= 0) return mActivationDistanceOverride; return getStore().get().find ("iMaxActivateDist")->getInt()*5/4; } float World::getObjectActivationDistance () { if (mActivationDistanceOverride >= 0) return mActivationDistanceOverride; return getStore().get().find ("iMaxActivateDist")->getInt(); } MWWorld::Ptr World::getFacedObject() { if (mFacedHandle.empty()) return MWWorld::Ptr(); return searchPtrViaHandle(mFacedHandle); } std::pair World::getHitContact(const MWWorld::Ptr &ptr, float distance) { 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) * Ogre::Quaternion(Ogre::Radian(posdata.rot[0]), Ogre::Vector3::UNIT_X); MWRender::Animation *anim = mRendering->getAnimation(ptr); if(anim != NULL) { Ogre::Node *node = anim->getNode("Head"); if (node == NULL) node = anim->getNode("Bip01 Head"); if(node != NULL) pos += node->_getDerivedPosition(); } std::pair result = mPhysics->getHitContact(ptr.getRefData().getHandle(), pos, rot, distance); if(result.first.empty()) return std::make_pair(MWWorld::Ptr(), Ogre::Vector3(0.0f)); return std::make_pair(searchPtrViaHandle(result.first), result.second); } void World::deleteObject (const Ptr& ptr) { if (ptr.getRefData().getCount() > 0) { ptr.getRefData().setCount(0); if (ptr.isInCell() && mWorldScene->getActiveCells().find(ptr.getCell()) != mWorldScene->getActiveCells().end() && ptr.getRefData().isEnabled()) { mWorldScene->removeObjectFromScene (ptr); mLocalScripts.remove (ptr); removeContainerScripts (ptr); } } } void World::moveObject(const Ptr &ptr, CellStore &newCell, float x, float y, float z) { ESM::Position &pos = ptr.getRefData().getPosition(); pos.pos[0] = x; pos.pos[1] = y; pos.pos[2] = z; Ogre::Vector3 vec(x, y, z); CellStore *currCell = ptr.getCell(); bool isPlayer = ptr == mPlayer->getPlayer(); bool haveToMove = isPlayer || mWorldScene->isCellActive(*currCell); if (*currCell != newCell) { removeContainerScripts(ptr); if (isPlayer) { if (!newCell.isExterior()) changeToInteriorCell(Misc::StringUtils::lowerCase(newCell.mCell->mName), pos); else { int cellX = newCell.mCell->getGridX(); int cellY = newCell.mCell->getGridY(); mWorldScene->changeCell(cellX, cellY, pos, false); } addContainerScripts (getPlayerPtr(), &newCell); } else { if (!mWorldScene->isCellActive(*currCell)) copyObjectToCell(ptr, newCell, pos); else if (!mWorldScene->isCellActive(newCell)) { mWorldScene->removeObjectFromScene(ptr); mLocalScripts.remove(ptr); removeContainerScripts (ptr); haveToMove = false; MWWorld::Ptr newPtr = MWWorld::Class::get(ptr) .copyToCell(ptr, newCell); newPtr.getRefData().setBaseNode(0); objectLeftActiveCell(ptr, newPtr); } else { MWWorld::Ptr copy = MWWorld::Class::get(ptr).copyToCell(ptr, newCell, pos); mRendering->updateObjectCell(ptr, copy); MWBase::MechanicsManager *mechMgr = MWBase::Environment::get().getMechanicsManager(); mechMgr->updateCell(ptr, copy); std::string script = MWWorld::Class::get(ptr).getScript(ptr); if (!script.empty()) { mLocalScripts.remove(ptr); removeContainerScripts (ptr); mLocalScripts.add(script, copy); addContainerScripts (copy, &newCell); } } ptr.getRefData().setCount(0); } } if (haveToMove && ptr.getRefData().getBaseNode()) { mRendering->moveObject(ptr, vec); mPhysics->moveObject (ptr); } } 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); cell = getExterior(cellX, cellY); } moveObject(ptr, *cell, x, y, z); return cell != ptr.getCell(); } void World::moveObject (const Ptr& ptr, float x, float y, float z) { moveObjectImp(ptr, x, y, z); } void World::scaleObject (const Ptr& ptr, float scale) { ptr.getCellRef().mScale = scale; MWWorld::Class::get(ptr).adjustScale(ptr,scale); if(ptr.getRefData().getBaseNode() == 0) return; mRendering->scaleObject(ptr, Vector3(scale,scale,scale)); mPhysics->scaleObject(ptr); } void World::rotateObjectImp (const Ptr& ptr, Ogre::Vector3 rot, bool adjust) { 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 { objRot[0] = rot.x; objRot[1] = rot.y; objRot[2] = rot.z; } 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; } 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); } } void World::localRotateObject (const Ptr& ptr, float x, float y, float z) { if (ptr.getRefData().getBaseNode() != 0) { 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(); 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; float *worldRot = ptr.getRefData().getPosition().rot; Ogre::Quaternion worldRotQuat(Ogre::Quaternion(Ogre::Radian(-worldRot[0]), Ogre::Vector3::UNIT_X)* Ogre::Quaternion(Ogre::Radian(-worldRot[1]), Ogre::Vector3::UNIT_Y)* Ogre::Quaternion(Ogre::Radian(-worldRot[2]), Ogre::Vector3::UNIT_Z)); Ogre::Quaternion rot(Ogre::Quaternion(Ogre::Radian(Ogre::Degree(-x).valueRadians()), Ogre::Vector3::UNIT_X)* Ogre::Quaternion(Ogre::Radian(Ogre::Degree(-y).valueRadians()), Ogre::Vector3::UNIT_Y)* Ogre::Quaternion(Ogre::Radian(Ogre::Degree(-z).valueRadians()), Ogre::Vector3::UNIT_Z)); ptr.getRefData().getBaseNode()->setOrientation(worldRotQuat*rot); mPhysics->rotateObject(ptr); } } void World::adjustPosition(const Ptr &ptr) { Ogre::Vector3 pos (ptr.getRefData().getPosition().pos); if(!ptr.getRefData().getBaseNode()) { // will be adjusted when Ptr's cell becomes active return; } float terrainHeight = mRendering->getTerrainHeightAt(pos); if (pos.z < terrainHeight) pos.z = terrainHeight; ptr.getRefData().getPosition().pos[2] = pos.z + 20; // place slightly above. will snap down to ground with code below if (!isFlying(ptr)) { Ogre::Vector3 traced = mPhysics->traceDown(ptr); if (traced.z < pos.z) pos.z = traced.z; } moveObject(ptr, *ptr.getCell(), pos.x, pos.y, pos.z); } 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); } MWWorld::Ptr World::safePlaceObject(const MWWorld::Ptr& ptr,MWWorld::CellStore &Cell,ESM::Position pos) { return copyObjectToCell(ptr,Cell,pos,false); } void World::indexToPosition (int cellX, int cellY, float &x, float &y, bool centre) const { const int cellSize = 8192; x = cellSize * cellX; y = cellSize * cellY; if (centre) { x += cellSize/2; y += cellSize/2; } } void World::positionToIndex (float x, float y, int &cellX, int &cellY) const { const int cellSize = 8192; cellX = std::floor(x/cellSize); cellY = std::floor(y/cellSize); } void World::queueMovement(const Ptr &ptr, const Vector3 &velocity) { mPhysics->queueObjectMovement(ptr, velocity); } void World::doPhysics(float duration) { processDoors(duration); moveProjectiles(duration); const PtrVelocityList &results = mPhysics->applyQueuedMovement(duration); PtrVelocityList::const_iterator player(results.end()); for(PtrVelocityList::const_iterator iter(results.begin());iter != results.end();iter++) { if(iter->first.getRefData().getHandle() == "player") { /* Handle player last, in case a cell transition occurs */ player = iter; continue; } moveObjectImp(iter->first, iter->second.x, iter->second.y, iter->second.z); } if(player != results.end()) moveObjectImp(player->first, player->second.x, player->second.y, player->second.z); mPhysEngine->stepSimulation(duration); } bool World::castRay (float x1, float y1, float z1, float x2, float y2, float z2) { Ogre::Vector3 a(x1,y1,z1); Ogre::Vector3 b(x2,y2,z2); return mPhysics->castRay(a,b,false,true); } void World::processDoors(float duration) { std::map::iterator it = mDoorStates.begin(); while (it != mDoorStates.end()) { if (!mWorldScene->isCellActive(*it->first.getCell())) mDoorStates.erase(it++); else { float oldRot = Ogre::Radian(it->first.getRefData().getLocalRotation().rot[2]).valueDegrees(); float diff = duration * 90; float targetRot = std::min(std::max(0.f, oldRot + diff * (it->second ? 1 : -1)), 90.f); localRotateObject(it->first, 0, 0, targetRot); /// \todo should use convexSweepTest here std::vector collisions = mPhysics->getCollisions(it->first); for (std::vector::iterator cit = collisions.begin(); cit != collisions.end(); ++cit) { MWWorld::Ptr ptr = getPtrViaHandle(*cit); if (MWWorld::Class::get(ptr).isActor()) { // we collided with an actor, we need to undo the rotation localRotateObject(it->first, 0, 0, oldRot); break; } } if ((targetRot == 90.f && it->second) || targetRot == 0.f) mDoorStates.erase(it++); else ++it; } } } bool World::toggleCollisionMode() { return mPhysics->toggleCollisionMode(); } bool World::toggleRenderMode (RenderMode mode) { return mRendering->toggleRenderMode (mode); } const ESM::Potion *World::createRecord (const ESM::Potion& record) { return mStore.insert(record); } const ESM::Class *World::createRecord (const ESM::Class& record) { return mStore.insert(record); } const ESM::Spell *World::createRecord (const ESM::Spell& record) { return mStore.insert(record); } const ESM::Cell *World::createRecord (const ESM::Cell& record) { return mStore.insert(record); } const ESM::NPC *World::createRecord(const ESM::NPC &record) { bool update = false; if (Misc::StringUtils::ciEqual(record.mId, "player")) { std::vector ids; getStore().get().listIdentifier(ids); unsigned int i=0; for (; isetInt ("pcrace", (i == ids.size()) ? 0 : i+1); const ESM::NPC *player = mPlayer->getPlayer().get()->mBase; update = record.isMale() != player->isMale() || !Misc::StringUtils::ciEqual(record.mRace, player->mRace) || !Misc::StringUtils::ciEqual(record.mHead, player->mHead) || !Misc::StringUtils::ciEqual(record.mHair, player->mHair); } const ESM::NPC *ret = mStore.insert(record); if (update) { mRendering->renderPlayer(mPlayer->getPlayer()); } return ret; } 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); } void World::update (float duration, bool paused) { if (mPlayIntro) { --mPlayIntro; if (mPlayIntro == 0) mRendering->playVideo(mFallback.getFallbackString("Movies_New_Game"), true); } if (mGoToJail && !paused) goToJail(); updateWeather(duration); mWorldScene->update (duration, paused); if (!paused) doPhysics (duration); performUpdateSceneQueries (); updateWindowManager (); if (mPlayer->getPlayer().getCell()->isExterior()) { ESM::Position pos = mPlayer->getPlayer().getRefData().getPosition(); mPlayer->setLastKnownExteriorPosition(Ogre::Vector3(pos.pos)); } } void World::updateWindowManager () { // inform the GUI about focused object MWWorld::Ptr object = getFacedObject (); MWBase::Environment::get().getWindowManager()->setFocusObject(object); // retrieve object dimensions so we know where to place the floating label if (!object.isEmpty ()) { Ogre::SceneNode* node = object.getRefData().getBaseNode(); Ogre::AxisAlignedBox bounds = node->_getWorldAABB(); if (bounds.isFinite()) { Vector4 screenCoords = mRendering->boundingBoxToScreen(bounds); MWBase::Environment::get().getWindowManager()->setFocusObjectScreenCoords( screenCoords[0], screenCoords[1], screenCoords[2], screenCoords[3]); } } } void World::performUpdateSceneQueries () { 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)); } updateFacedHandle (); } void World::updateFacedHandle () { float telekinesisRangeBonus = mPlayer->getPlayer().getClass().getCreatureStats(mPlayer->getPlayer()).getMagicEffects() .get(ESM::MagicEffect::Telekinesis).mMagnitude; telekinesisRangeBonus = feetToGameUnits(telekinesisRangeBonus); float activationDistance = getMaxActivationDistance() + telekinesisRangeBonus; activationDistance += mRendering->getCameraDistance(); // 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); results = mPhysics->getFacedHandles(x, y, activationDistance); if (MWBase::Environment::get().getWindowManager()->isConsoleMode()) results = mPhysics->getFacedHandles(x, y, getMaxActivationDistance ()*50); } else { results = mPhysics->getFacedHandles(activationDistance); } // 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; } if (results.empty()) { mFacedHandle = ""; mFacedDistance = FLT_MAX; } else { mFacedHandle = results.front().second; mFacedDistance = results.front().first; } } bool World::isCellExterior() const { Ptr::CellStore *currentCell = mWorldScene->getCurrentCell(); if (currentCell) { return currentCell->mCell->isExterior(); } return false; } bool World::isCellQuasiExterior() const { Ptr::CellStore *currentCell = mWorldScene->getCurrentCell(); if (currentCell) { if (!(currentCell->mCell->mData.mFlags & ESM::Cell::QuasiEx)) return false; else return true; } return false; } int World::getCurrentWeather() const { return mWeatherManager->getWeatherID(); } void World::changeWeather(const std::string& region, const unsigned int id) { mWeatherManager->changeWeather(region, id); } void World::modRegion(const std::string ®ionid, const std::vector &chances) { mWeatherManager->modRegion(regionid, chances); } OEngine::Render::Fader* World::getFader() { return mRendering->getFader(); } Ogre::Vector2 World::getNorthVector (CellStore* cell) { MWWorld::CellRefList& statics = cell->mStatics; MWWorld::LiveCellRef* ref = statics.find("northmarker"); if (!ref) return Vector2(0, 1); Ogre::SceneNode* node = ref->mData.getBaseNode(); Vector3 dir = node->_getDerivedOrientation() * Ogre::Vector3(0,1,0); Vector2 d = Vector2(dir.x, dir.y); return d; } void World::getDoorMarkers (CellStore* cell, std::vector& out) { MWWorld::CellRefList& doors = cell->mDoors; CellRefList::List& refList = doors.mList; for (CellRefList::List::iterator it = refList.begin(); it != refList.end(); ++it) { MWWorld::LiveCellRef& ref = *it; if (ref.mRef.mTeleport) { World::DoorMarker newMarker; newMarker.name = MWClass::Door::getDestination(ref); ESM::Position pos = ref.mData.getPosition (); newMarker.x = pos.pos[0]; newMarker.y = pos.pos[1]; out.push_back(newMarker); } } } 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); } void World::setWaterHeight(const float height) { mRendering->setWaterHeight(height); } void World::toggleWater() { mRendering->toggleWater(); } void World::PCDropped (const Ptr& item) { std::string script = MWWorld::Class::get(item).getScript(item); // Set OnPCDrop Variable on item's script, if it has a script with that variable declared if(script != "") item.getRefData().getLocals().setVarByInt(script, "onpcdrop", 1); } bool World::placeObject (const MWWorld::Ptr& object, float cursorX, float cursorY, int amount) { std::pair result = mPhysics->castRay(cursorX, cursorY); if (!result.first) return false; CellStore* cell; if (isCellExterior()) { int cellX, cellY; positionToIndex(result.second[0], result.second[1], cellX, cellY); cell = mCells.getExterior(cellX, cellY); } else cell = getPlayerPtr().getCell(); ESM::Position pos = getPlayerPtr().getRefData().getPosition(); pos.pos[0] = result.second[0]; pos.pos[1] = result.second[1]; pos.pos[2] = result.second[2]; // We want only the Z part of the player's rotation pos.rot[0] = 0; pos.rot[1] = 0; // copy the object and set its count int origCount = object.getRefData().getCount(); object.getRefData().setCount(amount); Ptr dropped = copyObjectToCell(object, *cell, pos, true); object.getRefData().setCount(origCount); // only the player place items in the world, so no need to check actor PCDropped(dropped); return true; } bool World::canPlaceObject(float cursorX, float cursorY) { std::pair 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; } Ptr World::copyObjectToCell(const Ptr &object, CellStore &cell, ESM::Position pos, bool adjustPos) { if (object.getClass().isActor() || adjustPos) { Ogre::Vector3 min, max; if (mPhysics->getObjectAABB(object, min, max)) { pos.pos[0] -= (min.x + max.x) / 2; pos.pos[1] -= (min.y + max.y) / 2; pos.pos[2] -= min.z; } } if (cell.isExterior()) { int cellX, cellY; positionToIndex(pos.pos[0], pos.pos[1], cellX, cellY); cell = *mCells.getExterior(cellX, cellY); } MWWorld::Ptr dropped = MWWorld::Class::get(object).copyToCell(object, cell, pos); if (mWorldScene->isCellActive(cell)) { if (dropped.getRefData().isEnabled()) { mWorldScene->addObjectToScene(dropped); } std::string script = MWWorld::Class::get(dropped).getScript(dropped); if (!script.empty()) { mLocalScripts.add(script, dropped); } addContainerScripts(dropped, &cell); } return dropped; } void World::dropObjectOnGround (const Ptr& actor, const Ptr& object, int amount) { MWWorld::Ptr::CellStore* cell = actor.getCell(); ESM::Position pos = actor.getRefData().getPosition(); // We want only the Z part of the actor's rotation pos.rot[0] = 0; pos.rot[1] = 0; Ogre::Vector3 orig = Ogre::Vector3(pos.pos); Ogre::Vector3 dir = Ogre::Vector3(0, 0, -1); float len = (pos.pos[2] >= 0) ? pos.pos[2] : -pos.pos[2]; len += 100.0; std::pair hit = mPhysics->castRay(orig, dir, len); pos.pos[2] = hit.second.z; // copy the object and set its count int origCount = object.getRefData().getCount(); object.getRefData().setCount(amount); Ptr dropped = copyObjectToCell(object, *cell, pos); object.getRefData().setCount(origCount); if(actor == mPlayer->getPlayer()) // Only call if dropped by player PCDropped(dropped); } void World::processChangedSettings(const Settings::CategorySettingVector& settings) { mRendering->processChangedSettings(settings); } void World::getTriangleBatchCount(unsigned int &triangles, unsigned int &batches) { mRendering->getTriangleBatchCount(triangles, batches); } bool World::isFlying(const MWWorld::Ptr &ptr) const { if(!ptr.getClass().isActor()) return false; if (ptr.getClass().getCreatureStats(ptr).isDead()) return false; if (ptr.getClass().isFlying(ptr)) return true; const MWMechanics::CreatureStats &stats = ptr.getClass().getCreatureStats(ptr); if(stats.getMagicEffects().get(ESM::MagicEffect::Levitate).mMagnitude > 0 && isLevitationEnabled()) return true; const OEngine::Physic::PhysicActor *actor = mPhysEngine->getCharacter(ptr.getRefData().getHandle()); if(!actor || !actor->getCollisionMode()) return true; return false; } bool World::isSlowFalling(const MWWorld::Ptr &ptr) const { if(!ptr.getClass().isActor()) return false; const MWMechanics::CreatureStats &stats = ptr.getClass().getCreatureStats(ptr); if(stats.getMagicEffects().get(ESM::MagicEffect::SlowFall).mMagnitude > 0) return true; return false; } 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); } bool World::isSwimming(const MWWorld::Ptr &object) const { /// \todo add check ifActor() - only actors can swim float *fpos = object.getRefData().getPosition().pos; Ogre::Vector3 pos(fpos[0], fpos[1], fpos[2]); /// \fixme 3/4ths submerged? const OEngine::Physic::PhysicActor *actor = mPhysEngine->getCharacter(object.getRefData().getHandle()); if(actor) pos.z += actor->getHalfExtents().z * 1.5; return isUnderwater(object.getCell(), pos); } bool World::isUnderwater(const MWWorld::Ptr::CellStore* cell, const Ogre::Vector3 &pos) const { if (!(cell->mCell->mData.mFlags & ESM::Cell::HasWater)) { return false; } return pos.z < cell->mWaterLevel; } bool World::isOnGround(const MWWorld::Ptr &ptr) const { RefData &refdata = ptr.getRefData(); const OEngine::Physic::PhysicActor *physactor = mPhysEngine->getCharacter(refdata.getHandle()); return physactor && physactor->getOnGround(); } bool World::vanityRotateCamera(float * rot) { return mRendering->vanityRotateCamera(rot); } void World::setCameraDistance(float dist, bool adjust, bool override) { return mRendering->setCameraDistance(dist, adjust, override);; } void World::setupPlayer() { const ESM::NPC *player = mStore.get().find("player"); if (!mPlayer) mPlayer = new MWWorld::Player(player, *this); else mPlayer->set(player); Ptr ptr = mPlayer->getPlayer(); mRendering->setupPlayer(ptr); } void World::renderPlayer() { mRendering->renderPlayer(mPlayer->getPlayer()); mPhysics->addActor(mPlayer->getPlayer()); } int World::canRest () { Ptr::CellStore *currentCell = mWorldScene->getCurrentCell(); Ptr player = mPlayer->getPlayer(); RefData &refdata = player.getRefData(); Ogre::Vector3 playerPos(refdata.getPosition().pos); const OEngine::Physic::PhysicActor *physactor = mPhysEngine->getCharacter(refdata.getHandle()); if((!physactor->getOnGround()&&physactor->getCollisionMode()) || isUnderwater(currentCell, playerPos)) return 2; if((currentCell->mCell->mData.mFlags&ESM::Cell::NoSleep) || Class::get(player).getNpcStats(player).isWerewolf()) return 1; return 0; } MWRender::Animation* World::getAnimation(const MWWorld::Ptr &ptr) { return mRendering->getAnimation(ptr); } void World::playVideo (const std::string &name, bool allowSkipping) { mRendering->playVideo(name, allowSkipping); } void World::stopVideo () { mRendering->stopVideo(); } void World::frameStarted (float dt, bool paused) { mRendering->frameStarted(dt, paused); } void World::activateDoor(const MWWorld::Ptr& door) { if (mDoorStates.find(door) != mDoorStates.end()) { // if currently opening, then close, if closing, then open mDoorStates[door] = !mDoorStates[door]; } else { if (door.getRefData().getLocalRotation().rot[2] == 0) mDoorStates[door] = 1; // open else mDoorStates[door] = 0; // close } } bool World::getOpenOrCloseDoor(const Ptr &door) { if (mDoorStates.find(door) != mDoorStates.end()) return !mDoorStates[door]; // if currently opening or closing, then do the opposite return door.getRefData().getLocalRotation().rot[2] == 0; } 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 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()); } float World::getWindSpeed() { if (isCellExterior() || isCellQuasiExterior()) return mWeatherManager->getWindSpeed(); else return 0.f; } void World::getContainersOwnedBy (const MWWorld::Ptr& npc, std::vector& out) { const Scene::CellStoreCollection& collection = mWorldScene->getActiveCells(); for (Scene::CellStoreCollection::const_iterator cellIt = collection.begin(); cellIt != collection.end(); ++cellIt) { MWWorld::CellRefList& containers = (*cellIt)->mContainers; CellRefList::List& refList = containers.mList; for (CellRefList::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); } } } struct ListHandlesFunctor { std::vector mHandles; bool operator() (Ptr ptr) { Ogre::SceneNode* handle = ptr.getRefData().getBaseNode(); if (handle) mHandles.push_back(handle->getName()); return true; } }; void World::getItemsOwnedBy (const MWWorld::Ptr& npc, std::vector& out) { const Scene::CellStoreCollection& collection = mWorldScene->getActiveCells(); for (Scene::CellStoreCollection::const_iterator cellIt = collection.begin(); cellIt != collection.end(); ++cellIt) { ListHandlesFunctor functor; (*cellIt)->forEach(functor); for (std::vector::iterator it = functor.mHandles.begin(); it != functor.mHandles.end(); ++it) if (Misc::StringUtils::ciEqual(searchPtrViaHandle(*it).getCellRef().mOwner, npc.getCellRef().mRefID)) out.push_back(searchPtrViaHandle(*it)); } } bool World::getLOS(const MWWorld::Ptr& npc,const MWWorld::Ptr& targetNpc) { 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 result = mPhysEngine->rayTest(from, to,false); if(result.first == "") return true; return false; } void World::enableActorCollision(const MWWorld::Ptr& actor, bool enable) { OEngine::Physic::PhysicActor *physicActor = mPhysEngine->getCharacter(actor.getRefData().getHandle()); physicActor->enableCollisions(enable); } bool World::findInteriorPosition(const std::string &name, ESM::Position &pos) { typedef MWWorld::CellRefList::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; } const DoorList &doors = cellStore->mDoors.mList; 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. const DoorList &doors = source->mDoors.mList; 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); // Note: Z pos will be adjusted by adjustPosition later pos.pos[2] = 0; return true; } return false; } void World::enableTeleporting(bool enable) { mTeleportEnabled = enable; } bool World::isTeleportingEnabled() const { return mTeleportEnabled; } void World::enableLevitation(bool enable) { mLevitationEnabled = enable; } bool World::isLevitationEnabled() const { return mLevitationEnabled; } 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); // 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. MWWorld::InventoryStore& invStore = MWWorld::Class::get(actor).getInventoryStore(actor); invStore.unequipAll(actor); if(werewolf) { InventoryStore &inv = actor.getClass().getInventoryStore(actor); inv.equip(InventoryStore::Slot_Robe, inv.ContainerStore::add("werewolfrobe", 1, actor), actor); } else { actor.getClass().getContainerStore(actor).remove("werewolfrobe", 1, actor); } // 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); if(actor.getRefData().getHandle() == "player") { // Update the GUI only when called on the player MWBase::WindowManager* windowManager = MWBase::Environment::get().getWindowManager(); windowManager->unsetSelectedWeapon(); if (werewolf) { windowManager->forceHide(MWGui::GW_Inventory); windowManager->forceHide(MWGui::GW_Magic); } else { windowManager->unsetForceHide(MWGui::GW_Inventory); windowManager->unsetForceHide(MWGui::GW_Magic); } } } void World::applyWerewolfAcrobatics(const Ptr &actor) { const Store &gmst = getStore().get(); MWMechanics::NpcStats &stats = Class::get(actor).getNpcStats(actor); stats.getSkill(ESM::Skill::Acrobatics).setBase(gmst.find("fWerewolfAcrobatics")->getFloat()); } bool World::getGodModeState() { return mGodMode; } bool World::toggleGodMode() { mGodMode = !mGodMode; return mGodMode; } void World::loadContentFiles(const Files::Collections& fileCollections, const std::vector& content, ContentLoader& contentLoader) { std::vector::const_iterator it(content.begin()); std::vector::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); } } } bool World::startSpellCast(const Ptr &actor) { MWMechanics::CreatureStats& stats = actor.getClass().getCreatureStats(actor); std::string message; bool fail = false; bool isPlayer = (actor == getPlayerPtr()); std::string selectedSpell = stats.getSpells().getSelectedSpell(); if (!selectedSpell.empty()) { const ESM::Spell* spell = getStore().get().search(selectedSpell); // Check mana MWMechanics::DynamicStat 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 if (!fail && spell->mData.mType == ESM::Spell::ST_Power) { if (stats.canUsePower(spell->mId)) stats.usePower(spell->mId); else { message = "#{sPowerAlreadyUsed}"; fail = true; } } // Reduce mana if (!fail) { magicka.setCurrent(magicka.getCurrent() - spell->mData.mCost); stats.setMagicka(magicka); } } if (isPlayer && fail) MWBase::Environment::get().getWindowManager()->messageBox(message); return !fail; } void World::castSpell(const Ptr &actor) { MWMechanics::CreatureStats& stats = actor.getClass().getCreatureStats(actor); MWWorld::Ptr target = getFacedObject(); std::string selectedSpell = stats.getSpells().getSelectedSpell(); MWMechanics::CastSpell cast(actor, target); if (!selectedSpell.empty()) { const ESM::Spell* spell = getStore().get().search(selectedSpell); cast.cast(spell); } else if (actor.getClass().hasInventoryStore(actor)) { MWWorld::InventoryStore& inv = actor.getClass().getInventoryStore(actor); if (inv.getSelectedEnchantItem() != inv.end()) cast.cast(*inv.getSelectedEnchantItem()); } } void World::launchProjectile (const std::string& id, bool stack, const ESM::EffectList& effects, const MWWorld::Ptr& actor, const std::string& sourceName) { std::string projectileModel; std::string sound; float speed = 0; for (std::vector::const_iterator iter (effects.mList.begin()); iter!=effects.mList.end(); ++iter) { if (iter->mRange != ESM::RT_Target) continue; const ESM::MagicEffect *magicEffect = getStore().get().find ( iter->mEffectID); projectileModel = magicEffect->mBolt; if (projectileModel.empty()) projectileModel = "VFX_DefaultBolt"; 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"; speed = magicEffect->mData.mSpeed; break; } if (projectileModel.empty()) return; // Spawn at 0.75 * ActorHeight float height = mPhysEngine->getCharacter(actor.getRefData().getHandle())->getHalfExtents().z * 2 * 0.75; MWWorld::ManualRef ref(getStore(), projectileModel); ESM::Position pos; pos.pos[0] = actor.getRefData().getPosition().pos[0]; pos.pos[1] = actor.getRefData().getPosition().pos[1]; pos.pos[2] = actor.getRefData().getPosition().pos[2] + height; pos.rot[0] = actor.getRefData().getPosition().rot[0]; pos.rot[1] = actor.getRefData().getPosition().rot[1]; pos.rot[2] = actor.getRefData().getPosition().rot[2]; ref.getPtr().getCellRef().mPos = pos; MWWorld::Ptr ptr = copyObjectToCell(ref.getPtr(), *actor.getCell(), pos); ProjectileState state; state.mSourceName = sourceName; state.mId = id; state.mActorHandle = actor.getRefData().getHandle(); state.mSpeed = speed; state.mEffects = effects; state.mStack = stack; MWBase::SoundManager *sndMgr = MWBase::Environment::get().getSoundManager(); sndMgr->playSound3D(ptr, sound, 1.0f, 1.0f); mProjectiles[ptr] = state; } void World::moveProjectiles(float duration) { std::map moved; for (std::map::iterator it = mProjectiles.begin(); it != mProjectiles.end();) { if (!mWorldScene->isCellActive(*it->first.getCell())) { mProjectiles.erase(it++); continue; } MWWorld::Ptr ptr = it->first; Ogre::Vector3 rot(ptr.getRefData().getPosition().rot); // TODO: Why -rot.z, but not -rot.x? (note: same issue in MovementSolver::move) Ogre::Quaternion orient = Ogre::Quaternion(Ogre::Radian(-rot.z), Ogre::Vector3::UNIT_Z); orient = orient * Ogre::Quaternion(Ogre::Radian(rot.x), Ogre::Vector3::UNIT_X); static float fTargetSpellMaxSpeed = getStore().get().find("fTargetSpellMaxSpeed")->getFloat(); float speed = fTargetSpellMaxSpeed * it->second.mSpeed; 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 > collisions = mPhysEngine->rayTest2(from, to); bool explode = false; for (std::vector >::iterator cIt = collisions.begin(); cIt != collisions.end() && !explode; ++cIt) { MWWorld::Ptr obstacle = searchPtrViaHandle(cIt->second); if (obstacle == ptr) continue; explode = true; MWWorld::Ptr caster = searchPtrViaHandle(it->second.mActorHandle); if (caster.isEmpty()) caster = obstacle; if (obstacle.isEmpty()) { // Terrain } else { MWMechanics::CastSpell cast(caster, obstacle); cast.mStack = it->second.mStack; cast.mId = it->second.mId; cast.mSourceName = it->second.mSourceName; cast.inflict(obstacle, caster, it->second.mEffects, ESM::RT_Target, false); } deleteObject(ptr); mProjectiles.erase(it++); } if (explode) { // TODO: Explode 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::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::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 if (mProjectiles.find(object) != mProjectiles.end()) { deleteObject(movedPtr); } } void World::breakInvisibility(const Ptr &actor) { actor.getClass().getCreatureStats(actor).getActiveSpells().purgeEffect(ESM::MagicEffect::Invisibility); if (actor.getClass().hasInventoryStore(actor)) actor.getClass().getInventoryStore(actor).purgeEffect(ESM::MagicEffect::Invisibility); } bool World::isDark() const { return mWeatherManager->isDark(); } bool World::findInteriorPositionInWorldSpace(MWWorld::CellStore* cell, Ogre::Vector3& result) { if (cell->isExterior()) return false; MWWorld::CellRefList& doors = cell->mDoors; CellRefList::List& refList = doors.mList; // Check if any door in the cell leads to an exterior directly for (CellRefList::List::iterator it = refList.begin(); it != refList.end(); ++it) { MWWorld::LiveCellRef& ref = *it; if (ref.mRef.mTeleport && ref.mRef.mDestCell.empty()) { ESM::Position pos = ref.mRef.mDoorDest; result = Ogre::Vector3(pos.pos); return true; } } // No luck :( return false; } void World::teleportToClosestMarker (const MWWorld::Ptr& ptr, const std::string& id) { Ogre::Vector3 worldPos; if (!findInteriorPositionInWorldSpace(ptr.getCell(), worldPos)) worldPos = mPlayer->getLastKnownExteriorPosition(); MWWorld::Ptr closestMarker; float closestDistance = FLT_MAX; std::vector markers; mCells.getExteriorPtrs(id, markers); for (std::vector::iterator it = markers.begin(); it != markers.end(); ++it) { ESM::Position pos = it->getRefData().getPosition(); Ogre::Vector3 markerPos = Ogre::Vector3(pos.pos); float distance = worldPos.squaredDistance(markerPos); if (distance < closestDistance) { closestDistance = distance; closestMarker = *it; } } MWWorld::ActionTeleport action("", closestMarker.getRefData().getPosition()); action.execute(ptr); } void World::updateWeather(float duration) { if (mPlayer->wasTeleported()) { mPlayer->setTeleported(false); mWeatherManager->switchToNextWeather(true); } mWeatherManager->update(duration); } struct AddDetectedReference { AddDetectedReference(std::vector& out, Ptr detector, World::DetectionType type, float squaredDist) : mOut(out), mDetector(detector), mType(type), mSquaredDist(squaredDist) { } std::vector& 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 &out, DetectionType type) { const MWMechanics::MagicEffects& effects = ptr.getClass().getCreatureStats(ptr).getMagicEffects(); float dist=0; if (type == World::Detect_Creature) dist = effects.get(ESM::MagicEffect::DetectAnimal).mMagnitude; else if (type == World::Detect_Key) dist = effects.get(ESM::MagicEffect::DetectKey).mMagnitude; else if (type == World::Detect_Enchantment) dist = effects.get(ESM::MagicEffect::DetectEnchantment).mMagnitude; if (!dist) return; dist = feetToGameUnits(dist); 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); } } 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; } MWWorld::Ptr World::getPlayerPtr() { return mPlayer->getPlayer(); } 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().find("fCrimeGoldDiscountMult")->getFloat(); float fCrimeGoldTurnInMult = getStore().get().find("fCrimeGoldTurnInMult")->getFloat(); int discount = bounty*fCrimeGoldDiscountMult; int turnIn = bounty * fCrimeGoldTurnInMult; mGlobalVariables->setInt("pchascrimegold", (bounty <= playerGold) ? 1 : 0); mGlobalVariables->setInt("pchasgolddiscount", (discount <= playerGold) ? 1 : 0); mGlobalVariables->setInt("crimegolddiscount", discount); mGlobalVariables->setInt("crimegoldturnin", turnIn); mGlobalVariables->setInt("pchasturnin", (turnIn <= playerGold) ? 1 : 0); } void World::confiscateStolenItems(const Ptr &ptr) { Ogre::Vector3 playerPos; if (!findInteriorPositionInWorldSpace(ptr.getCell(), playerPos)) playerPos = mPlayer->getLastKnownExteriorPosition(); MWWorld::Ptr closestChest; float closestDistance = FLT_MAX; std::vector chests; mCells.getInteriorPtrs("stolen_goods", chests); Ogre::Vector3 chestPos; for (std::vector::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; } } if (!closestChest.isEmpty()) { ContainerStore& store = ptr.getClass().getContainerStore(ptr); for (ContainerStoreIterator it = store.begin(); it != store.end(); ++it) { if (!it->getCellRef().mOwner.empty() && it->getCellRef().mOwner != "player") { closestChest.getClass().getContainerStore(closestChest).add(*it, it->getRefData().getCount(), closestChest); store.remove(*it, it->getRefData().getCount(), ptr); } } } } 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().find("iDaysinPrisonMod")->getInt(); int days = std::max(1, bounty / iDaysinPrisonMod); advanceTime(days * 24); std::set skills; for (int day=0; day (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& gmst = getStore().get(); 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::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; } std::vector buttons; buttons.push_back("#{sOk}"); MWBase::Environment::get().getWindowManager()->messageBox(message, buttons); } } void World::spawnRandomCreature(const std::string &creatureList) { const ESM::CreatureLevList* list = getStore().get().find(creatureList); int iNumberCreatures = getStore().get().find("iNumberCreatures")->getInt(); int numCreatures = 1 + std::rand()/ (static_cast (RAND_MAX) + 1) * iNumberCreatures; // [1, iNumberCreatures] for (int i=0; igetPlayer().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; safePlaceObject(ref.getPtr(),*cell,ipos); } } 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 (RAND_MAX) + 1) * 3; // [0, 2] modelName << roll; std::string model = "meshes\\" + getFallback()->getFallbackString(modelName.str()); mRendering->spawnEffect(model, texture, worldPosition); } }