mirror of
				https://github.com/TES3MP/openmw-tes3mp.git
				synced 2025-10-31 01:56:52 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			266 lines
		
	
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			266 lines
		
	
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "cellextensions.hpp"
 | |
| 
 | |
| #include <limits>
 | |
| 
 | |
| #include "../mwworld/esmstore.hpp"
 | |
| 
 | |
| #include <components/compiler/opcodes.hpp>
 | |
| 
 | |
| #include <components/interpreter/interpreter.hpp>
 | |
| #include <components/interpreter/runtime.hpp>
 | |
| #include <components/interpreter/opcodes.hpp>
 | |
| 
 | |
| #include "../mwworld/actionteleport.hpp"
 | |
| #include "../mwworld/cellstore.hpp"
 | |
| #include "../mwbase/environment.hpp"
 | |
| #include "../mwworld/player.hpp"
 | |
| #include "../mwbase/statemanager.hpp"
 | |
| #include "../mwbase/windowmanager.hpp"
 | |
| #include "../mwbase/world.hpp"
 | |
| 
 | |
| #include "../mwmechanics/actorutil.hpp"
 | |
| 
 | |
| #include "interpretercontext.hpp"
 | |
| 
 | |
| namespace MWScript
 | |
| {
 | |
|     namespace Cell
 | |
|     {
 | |
|         class OpCellChanged : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     runtime.push (MWBase::Environment::get().getWorld()->hasCellChanged() ? 1 : 0);
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpTestCells : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     if (MWBase::Environment::get().getStateManager()->getState() != MWBase::StateManager::State_NoGame)
 | |
|                     {
 | |
|                         runtime.getContext().report("Use TestCells from the main menu, when there is no active game session.");
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     bool wasConsole = MWBase::Environment::get().getWindowManager()->isConsoleMode();
 | |
|                     if (wasConsole)
 | |
|                         MWBase::Environment::get().getWindowManager()->toggleConsole();
 | |
| 
 | |
|                     MWBase::Environment::get().getWorld()->testExteriorCells();
 | |
| 
 | |
|                     if (wasConsole)
 | |
|                         MWBase::Environment::get().getWindowManager()->toggleConsole();
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpTestInteriorCells : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     if (MWBase::Environment::get().getStateManager()->getState() != MWBase::StateManager::State_NoGame)
 | |
|                     {
 | |
|                         runtime.getContext().report("Use TestInteriorCells from the main menu, when there is no active game session.");
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     bool wasConsole = MWBase::Environment::get().getWindowManager()->isConsoleMode();
 | |
|                     if (wasConsole)
 | |
|                         MWBase::Environment::get().getWindowManager()->toggleConsole();
 | |
| 
 | |
|                     MWBase::Environment::get().getWorld()->testInteriorCells();
 | |
| 
 | |
|                     if (wasConsole)
 | |
|                         MWBase::Environment::get().getWindowManager()->toggleConsole();
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpCOC : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     std::string cell = runtime.getStringLiteral (runtime[0].mInteger);
 | |
|                     runtime.pop();
 | |
| 
 | |
|                     ESM::Position pos;
 | |
|                     MWBase::World *world = MWBase::Environment::get().getWorld();
 | |
|                     const MWWorld::Ptr playerPtr = world->getPlayerPtr();
 | |
| 
 | |
|                     if (world->findExteriorPosition(cell, pos))
 | |
|                     {
 | |
|                         MWWorld::ActionTeleport("", pos, false).execute(playerPtr);
 | |
|                         world->adjustPosition(playerPtr, false);
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         // Change to interior even if findInteriorPosition()
 | |
|                         // yields false. In this case position will be zero-point.
 | |
|                         world->findInteriorPosition(cell, pos);
 | |
|                         MWWorld::ActionTeleport(cell, pos, false).execute(playerPtr);
 | |
|                     }
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpCOE : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     Interpreter::Type_Integer x = runtime[0].mInteger;
 | |
|                     runtime.pop();
 | |
| 
 | |
|                     Interpreter::Type_Integer y = runtime[0].mInteger;
 | |
|                     runtime.pop();
 | |
| 
 | |
|                     ESM::Position pos;
 | |
|                     MWBase::World *world = MWBase::Environment::get().getWorld();
 | |
|                     const MWWorld::Ptr playerPtr = world->getPlayerPtr();
 | |
| 
 | |
|                     world->indexToPosition (x, y, pos.pos[0], pos.pos[1], true);
 | |
|                     pos.pos[2] = 0;
 | |
| 
 | |
|                     pos.rot[0] = pos.rot[1] = pos.rot[2] = 0;
 | |
| 
 | |
|                     MWWorld::ActionTeleport("", pos, false).execute(playerPtr);
 | |
|                     world->adjustPosition(playerPtr, false);
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpGetInterior : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     if (!MWMechanics::getPlayer().isInCell())
 | |
|                     {
 | |
|                         runtime.push (0);
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     bool interior =
 | |
|                         !MWMechanics::getPlayer().getCell()->getCell()->isExterior();
 | |
| 
 | |
|                     runtime.push (interior ? 1 : 0);
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpGetPCCell : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     std::string name = runtime.getStringLiteral (runtime[0].mInteger);
 | |
|                     runtime.pop();
 | |
| 
 | |
|                     if (!MWMechanics::getPlayer().isInCell())
 | |
|                     {
 | |
|                         runtime.push(0);
 | |
|                         return;
 | |
|                     }
 | |
|                     const MWWorld::CellStore *cell = MWMechanics::getPlayer().getCell();
 | |
| 
 | |
|                     std::string current = MWBase::Environment::get().getWorld()->getCellName(cell);
 | |
|                     Misc::StringUtils::lowerCaseInPlace(current);
 | |
| 
 | |
|                     bool match = current.length()>=name.length() &&
 | |
|                         current.substr (0, name.length())==name;
 | |
| 
 | |
|                     runtime.push (match ? 1 : 0);
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpGetWaterLevel : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     if (!MWMechanics::getPlayer().isInCell())
 | |
|                     {
 | |
|                         runtime.push(0.f);
 | |
|                         return;
 | |
|                     }
 | |
|                     MWWorld::CellStore *cell = MWMechanics::getPlayer().getCell();
 | |
|                     if (cell->isExterior())
 | |
|                         runtime.push(0.f); // vanilla oddity, return 0 even though water is actually at -1
 | |
|                     else if (cell->getCell()->hasWater())
 | |
|                         runtime.push (cell->getWaterLevel());
 | |
|                     else
 | |
|                         runtime.push (-std::numeric_limits<float>::max());
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpSetWaterLevel : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     Interpreter::Type_Float level = runtime[0].mFloat;
 | |
| 
 | |
|                     if (!MWMechanics::getPlayer().isInCell())
 | |
|                     {
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     MWWorld::CellStore *cell = MWMechanics::getPlayer().getCell();
 | |
| 
 | |
|                     if (cell->getCell()->isExterior())
 | |
|                         throw std::runtime_error("Can't set water level in exterior cell");
 | |
| 
 | |
|                     cell->setWaterLevel (level);
 | |
|                     MWBase::Environment::get().getWorld()->setWaterHeight (cell->getWaterLevel());
 | |
|                 }
 | |
|         };
 | |
| 
 | |
|         class OpModWaterLevel : public Interpreter::Opcode0
 | |
|         {
 | |
|             public:
 | |
| 
 | |
|                 void execute (Interpreter::Runtime& runtime) override
 | |
|                 {
 | |
|                     Interpreter::Type_Float level = runtime[0].mFloat;
 | |
| 
 | |
|                     if (!MWMechanics::getPlayer().isInCell())
 | |
|                     {
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     MWWorld::CellStore *cell = MWMechanics::getPlayer().getCell();
 | |
| 
 | |
|                     if (cell->getCell()->isExterior())
 | |
|                         throw std::runtime_error("Can't set water level in exterior cell");
 | |
| 
 | |
|                     cell->setWaterLevel (cell->getWaterLevel()+level);
 | |
|                     MWBase::Environment::get().getWorld()->setWaterHeight(cell->getWaterLevel());
 | |
|                 }
 | |
|         };
 | |
| 
 | |
| 
 | |
|         void installOpcodes (Interpreter::Interpreter& interpreter)
 | |
|         {
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeCellChanged, new OpCellChanged);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeTestCells, new OpTestCells);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeTestInteriorCells, new OpTestInteriorCells);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeCOC, new OpCOC);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeCOE, new OpCOE);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeGetInterior, new OpGetInterior);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeGetPCCell, new OpGetPCCell);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeGetWaterLevel, new OpGetWaterLevel);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeSetWaterLevel, new OpSetWaterLevel);
 | |
|             interpreter.installSegment5 (Compiler::Cell::opcodeModWaterLevel, new OpModWaterLevel);
 | |
|         }
 | |
|     }
 | |
| }
 |