#include "world.hpp" #include #include "components/bsa/bsa_archive.hpp" #include "components/engine/ogre/renderer.hpp" #include "apps/openmw/mwrender/sky.hpp" #include "apps/openmw/mwrender/interior.hpp" #include "ptr.hpp" namespace { template void listCellScripts (const ESMS::ESMStore& store, ESMS::CellRefList& cellRefList, MWWorld::World::ScriptList& scriptList) { for (typename ESMS::CellRefList::List::iterator iter ( cellRefList.list.begin()); iter!=cellRefList.list.end(); ++iter) { if (!iter->base->script.empty()) { if (const ESM::Script *script = store.scripts.find (iter->base->script)) { iter->mData.setLocals (*script); scriptList.push_back ( std::make_pair (iter->base->script, MWWorld::Ptr (&*iter))); } } } } template bool hasReference (ESMS::CellRefList& cellRefList, const MWWorld::Ptr& ptr) { for (typename ESMS::CellRefList::List::iterator iter ( cellRefList.list.begin()); iter!=cellRefList.list.end(); ++iter) { if (&iter->mData==&ptr.getRefData()) return true; } return false; } } namespace MWWorld { void World::insertInteriorScripts (ESMS::CellStore& cell) { listCellScripts (mStore, cell.activators, mLocalScripts); listCellScripts (mStore, cell.potions, mLocalScripts); listCellScripts (mStore, cell.appas, mLocalScripts); listCellScripts (mStore, cell.armors, mLocalScripts); listCellScripts (mStore, cell.books, mLocalScripts); listCellScripts (mStore, cell.clothes, mLocalScripts); listCellScripts (mStore, cell.containers, mLocalScripts); listCellScripts (mStore, cell.creatures, mLocalScripts); listCellScripts (mStore, cell.doors, mLocalScripts); listCellScripts (mStore, cell.ingreds, mLocalScripts); listCellScripts (mStore, cell.lights, mLocalScripts); listCellScripts (mStore, cell.lockpicks, mLocalScripts); listCellScripts (mStore, cell.miscItems, mLocalScripts); listCellScripts (mStore, cell.npcs, mLocalScripts); listCellScripts (mStore, cell.probes, mLocalScripts); listCellScripts (mStore, cell.repairs, mLocalScripts); listCellScripts (mStore, cell.weapons, mLocalScripts); } Ptr World::getPtr (const std::string& name, CellStore& cell) { if (ESMS::LiveCellRef *ref = cell.activators.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.potions.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.appas.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.armors.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.books.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.clothes.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.containers.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.creatures.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.doors.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.ingreds.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.creatureLists.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.itemLists.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.lights.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.lockpicks.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.miscItems.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.npcs.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.probes.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.repairs.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.statics.find (name)) return ref; if (ESMS::LiveCellRef *ref = cell.weapons.find (name)) return ref; return Ptr(); } MWRender::CellRender *World::searchRender (CellStore *store) { CellRenderCollection::iterator iter = mActiveCells.find (store); if (iter!=mActiveCells.end()) { return iter->second; } else { iter = mBufferedCells.find (store); if (iter!=mBufferedCells.end()) return iter->second; } return 0; } World::World (Render::OgreRenderer& renderer, const boost::filesystem::path& dataDir, const std::string& master, const std::string& startCell, bool newGame) : mSkyManager (0), mScene (renderer), mPlayerPos (0) { boost::filesystem::path masterPath (dataDir); masterPath /= master; 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); mInteriors[startCell].loadInt (startCell, mStore, mEsm); insertInteriorScripts (mInteriors[startCell]); mPlayerPos = new MWRender::PlayerPos (mScene.getCamera(), mStore.npcs.find ("player")); // global variables for (ESMS::RecListT::MapType::const_iterator iter (mStore.globals.list.begin()); iter != mStore.globals.list.end(); ++iter) mGlobalVariables.insert (std::make_pair (iter->first, iter->second.value)); if (newGame) { // set new game mark float newGameState = 1; mGlobalVariables["chargenstate"] = *reinterpret_cast (&newGameState); } std::cout << "\nSetting up cell rendering\n"; // This connects the cell data with the rendering scene. mActiveCells.insert (std::make_pair (&mInteriors[startCell], new MWRender::InteriorCellRender (mInteriors[startCell], mScene))); // Load the cell and insert it into the renderer for (CellRenderCollection::iterator iter (mActiveCells.begin()); iter!=mActiveCells.end(); ++iter) iter->second->show(); // Optionally enable the sky ///\todo FIXME if (false) mSkyManager = MWRender::SkyManager::create(renderer.getWindow(), mScene.getCamera()); } World::~World() { for (CellRenderCollection::iterator iter (mActiveCells.begin()); iter!=mActiveCells.end(); ++iter) delete iter->second; for (CellRenderCollection::iterator iter (mBufferedCells.begin()); iter!=mBufferedCells.end(); ++iter) delete iter->second; delete mPlayerPos; delete mSkyManager; } MWRender::PlayerPos& World::getPlayerPos() { return *mPlayerPos; } ESMS::ESMStore& World::getStore() { return mStore; } const World::ScriptList& World::getLocalScripts() const { return mLocalScripts; } bool World::hasCellChanged() const { // Cell change not implemented yet. return false; } Interpreter::Type_Data& World::getGlobalVariable (const std::string& name) { std::map::iterator iter = mGlobalVariables.find (name); if (iter==mGlobalVariables.end()) throw std::runtime_error ("unknown global variable: " + name); return iter->second; } std::pair World::getPtr (const std::string& name, bool activeOnly) { // the player is always in an active cell. if (name=="player") { // TODO: find real cell (might need to be stored in playerPos). For now we // use the first active cell. This will fail the moment we move into an // exterior cell. return std::make_pair (mPlayerPos->getPlayer(), mActiveCells.begin()->first); } // active cells for (CellRenderCollection::iterator iter (mActiveCells.begin()); iter!=mActiveCells.end(); ++iter) { Ptr ptr = getPtr (name, *iter->first); if (!ptr.isEmpty()) return std::make_pair (ptr, iter->first); } if (!activeOnly) { // TODO: inactive cells } throw std::runtime_error ("unknown ID: " + name); } void World::enable (std::pair& reference) { if (!reference.first.getRefData().isEnabled()) { reference.first.getRefData().enable(); if (MWRender::CellRender *render = searchRender (reference.second)) { render->enable (reference.first.getRefData().getHandle()); } } } void World::disable (std::pair& reference) { if (!reference.first.getRefData().isEnabled()) { reference.first.getRefData().enable(); if (MWRender::CellRender *render = searchRender (reference.second)) { render->disable (reference.first.getRefData().getHandle()); } } } World::CellStore *World::find (const Ptr& ptr) { for (CellRenderCollection::iterator iter (mActiveCells.begin()); iter!=mActiveCells.end(); ++iter) { if ( hasReference (iter->first->activators, ptr) || hasReference (iter->first->potions, ptr) || hasReference (iter->first->appas, ptr) || hasReference (iter->first->armors, ptr) || hasReference (iter->first->books, ptr) || hasReference (iter->first->clothes, ptr) || hasReference (iter->first->containers, ptr) || hasReference (iter->first->creatures, ptr) || hasReference (iter->first->doors, ptr) || hasReference (iter->first->ingreds, ptr) || hasReference (iter->first->lights, ptr) || hasReference (iter->first->lockpicks, ptr) || hasReference (iter->first->miscItems, ptr) || hasReference (iter->first->npcs, ptr) || hasReference (iter->first->probes, ptr) || hasReference (iter->first->repairs, ptr) || hasReference (iter->first->weapons, ptr)) return iter->first; } throw std::runtime_error ("failed to locate reference in active cell"); } }