2012-07-09 18:47:59 +02:00
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
|
|
|
|
#include <components/esm_store/store.hpp>
|
|
|
|
|
|
|
|
#include <components/compiler/extensions.hpp>
|
|
|
|
|
|
|
|
#include <components/interpreter/interpreter.hpp>
|
|
|
|
#include <components/interpreter/runtime.hpp>
|
|
|
|
#include <components/interpreter/opcodes.hpp>
|
|
|
|
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
|
|
|
|
#include "interpretercontext.hpp"
|
|
|
|
#include "ref.hpp"
|
2012-07-09 19:28:44 +02:00
|
|
|
#include "OgreSceneNode.h"
|
2012-07-09 18:47:59 +02:00
|
|
|
|
|
|
|
namespace MWScript
|
|
|
|
{
|
|
|
|
namespace Transformation
|
|
|
|
{
|
|
|
|
template<class R>
|
|
|
|
class OpSetScale : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
2012-07-09 19:28:44 +02:00
|
|
|
Interpreter::Type_Float scale = runtime[0].mFloat;
|
2012-07-09 18:47:59 +02:00
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
MWBase::Environment::get().getWorld()->scaleObject(ptr,scale);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-07-09 19:28:44 +02:00
|
|
|
template<class R>
|
|
|
|
class OpGetScale : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
runtime.push(ptr.getCellRef().scale);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-07-09 18:47:59 +02:00
|
|
|
template<class R>
|
|
|
|
class OpSetAngle : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string axis = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
2012-07-09 19:28:44 +02:00
|
|
|
Interpreter::Type_Float angle = runtime[0].mFloat;
|
2012-07-09 18:47:59 +02:00
|
|
|
runtime.pop();
|
|
|
|
|
2012-07-10 11:15:46 +02:00
|
|
|
float ax = Ogre::Radian(ptr.getRefData().getPosition().rot[0]).valueDegrees();
|
|
|
|
float ay = Ogre::Radian(ptr.getRefData().getPosition().rot[1]).valueDegrees();
|
|
|
|
float az = Ogre::Radian(ptr.getRefData().getPosition().rot[2]).valueDegrees();
|
|
|
|
|
2012-08-04 09:18:20 +02:00
|
|
|
if (axis == "x")
|
2012-07-09 18:47:59 +02:00
|
|
|
{
|
2012-07-10 11:15:46 +02:00
|
|
|
MWBase::Environment::get().getWorld()->rotateObject(ptr,angle,ay,az);
|
2012-07-09 18:47:59 +02:00
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else if (axis == "y")
|
2012-07-09 18:47:59 +02:00
|
|
|
{
|
2012-07-10 11:15:46 +02:00
|
|
|
MWBase::Environment::get().getWorld()->rotateObject(ptr,ax,angle,az);
|
2012-07-09 18:47:59 +02:00
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else if (axis == "z")
|
2012-07-09 18:47:59 +02:00
|
|
|
{
|
2012-07-10 11:15:46 +02:00
|
|
|
MWBase::Environment::get().getWorld()->rotateObject(ptr,ax,ay,angle);
|
2012-07-09 18:47:59 +02:00
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else
|
|
|
|
throw std::runtime_error ("invalid ration axis: " + axis);
|
2012-07-09 18:47:59 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-07-09 19:28:44 +02:00
|
|
|
template<class R>
|
2012-08-01 12:21:42 +02:00
|
|
|
class OpGetStartingAngle : public Interpreter::Opcode0
|
2012-07-09 19:28:44 +02:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string axis = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
2012-08-04 09:18:20 +02:00
|
|
|
if (axis == "x")
|
2012-07-09 19:28:44 +02:00
|
|
|
{
|
2012-07-10 11:15:46 +02:00
|
|
|
runtime.push(Ogre::Radian(ptr.getRefData().getPosition().rot[0]).valueDegrees());
|
2012-07-09 19:28:44 +02:00
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else if (axis == "y")
|
2012-07-09 19:28:44 +02:00
|
|
|
{
|
2012-07-10 11:15:46 +02:00
|
|
|
runtime.push(Ogre::Radian(ptr.getRefData().getPosition().rot[1]).valueDegrees());
|
2012-07-09 19:28:44 +02:00
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else if (axis == "z")
|
2012-07-09 19:28:44 +02:00
|
|
|
{
|
2012-07-10 11:15:46 +02:00
|
|
|
runtime.push(Ogre::Radian(ptr.getRefData().getPosition().rot[2]).valueDegrees());
|
2012-07-09 19:28:44 +02:00
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else
|
|
|
|
throw std::runtime_error ("invalid ration axis: " + axis);
|
2012-07-09 19:28:44 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-08-01 12:21:42 +02:00
|
|
|
template<class R>
|
|
|
|
class OpGetAngle : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string axis = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
2012-08-02 14:57:35 +02:00
|
|
|
|
2012-08-04 09:18:20 +02:00
|
|
|
if (axis=="x")
|
2012-08-01 12:21:42 +02:00
|
|
|
{
|
|
|
|
runtime.push(Ogre::Radian(ptr.getCellRef().pos.rot[0]).valueDegrees());
|
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else if (axis=="y")
|
2012-08-01 12:21:42 +02:00
|
|
|
{
|
|
|
|
runtime.push(Ogre::Radian(ptr.getCellRef().pos.rot[1]).valueDegrees());
|
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else if (axis=="z")
|
2012-08-01 12:21:42 +02:00
|
|
|
{
|
|
|
|
runtime.push(Ogre::Radian(ptr.getCellRef().pos.rot[2]).valueDegrees());
|
|
|
|
}
|
2012-08-04 09:18:20 +02:00
|
|
|
else
|
|
|
|
throw std::runtime_error ("invalid ration axis: " + axis);
|
2012-07-09 19:28:44 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-08-03 18:14:05 +02:00
|
|
|
template<class R>
|
|
|
|
class OpGetPos : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string axis = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
2012-08-05 16:21:53 +02:00
|
|
|
if(axis == "x")
|
2012-08-03 18:14:05 +02:00
|
|
|
{
|
|
|
|
runtime.push(ptr.getRefData().getPosition().pos[0]);
|
|
|
|
}
|
2012-08-09 23:50:44 +02:00
|
|
|
else if(axis == "y")
|
2012-08-03 18:14:05 +02:00
|
|
|
{
|
|
|
|
runtime.push(ptr.getRefData().getPosition().pos[1]);
|
|
|
|
}
|
2012-08-09 23:50:44 +02:00
|
|
|
else if(axis == "z")
|
2012-08-03 18:14:05 +02:00
|
|
|
{
|
|
|
|
runtime.push(ptr.getRefData().getPosition().pos[2]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class R>
|
|
|
|
class OpSetPos : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string axis = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
Interpreter::Type_Float pos = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
float ax = ptr.getRefData().getPosition().pos[0];
|
|
|
|
float ay = ptr.getRefData().getPosition().pos[1];
|
|
|
|
float az = ptr.getRefData().getPosition().pos[2];
|
|
|
|
std::cout << "setPos";
|
|
|
|
|
2012-08-05 16:21:53 +02:00
|
|
|
if(axis == "x")
|
2012-08-03 18:14:05 +02:00
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWorld()->moveObject(ptr,pos,ay,az);
|
|
|
|
}
|
2012-08-09 23:50:44 +02:00
|
|
|
else if(axis == "y")
|
2012-08-03 18:14:05 +02:00
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWorld()->moveObject(ptr,ax,pos,az);
|
|
|
|
}
|
2012-08-09 23:50:44 +02:00
|
|
|
else if(axis == "z")
|
2012-08-03 18:14:05 +02:00
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWorld()->moveObject(ptr,ax,ay,pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-08-03 18:20:51 +02:00
|
|
|
template<class R>
|
|
|
|
class OpGetStartingPos : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string axis = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
2012-08-05 16:21:53 +02:00
|
|
|
if(axis == "x")
|
2012-08-03 18:20:51 +02:00
|
|
|
{
|
|
|
|
runtime.push(ptr.getCellRef().pos.pos[0]);
|
|
|
|
}
|
2012-08-09 23:50:44 +02:00
|
|
|
else if(axis == "y")
|
2012-08-03 18:20:51 +02:00
|
|
|
{
|
|
|
|
runtime.push(ptr.getCellRef().pos.pos[1]);
|
|
|
|
}
|
2012-08-09 23:50:44 +02:00
|
|
|
else if(axis == "z")
|
2012-08-03 18:20:51 +02:00
|
|
|
{
|
|
|
|
runtime.push(ptr.getCellRef().pos.pos[2]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-08-09 23:45:06 +02:00
|
|
|
template<class R>
|
|
|
|
class OpPositionCell : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
Interpreter::Type_Float x = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
Interpreter::Type_Float z = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
Interpreter::Type_Float zRot = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
std::string cellID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
2012-08-11 22:08:04 +02:00
|
|
|
MWWorld::CellStore* store = 0;
|
2012-08-11 19:16:00 +02:00
|
|
|
try
|
|
|
|
{
|
|
|
|
MWWorld::CellStore* store = MWBase::Environment::get().getWorld()->getInterior(cellID);
|
|
|
|
}
|
|
|
|
catch(std::exception &e)
|
2012-08-11 09:52:49 +02:00
|
|
|
{
|
2012-08-11 14:06:58 +02:00
|
|
|
const ESM::Cell* cell = MWBase::Environment::get().getWorld()->getExterior(cellID);
|
2012-08-11 09:52:49 +02:00
|
|
|
if(cell)
|
|
|
|
{
|
2012-08-11 14:06:58 +02:00
|
|
|
store = MWBase::Environment::get().getWorld()->getExterior(cell->getGridX(),cell->getGridY());
|
2012-08-11 09:52:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if(store)
|
2012-08-09 23:45:06 +02:00
|
|
|
{
|
2012-08-11 14:06:58 +02:00
|
|
|
MWBase::Environment::get().getWorld()->moveObject(ptr,*store,x,y,z);
|
2012-08-11 09:52:49 +02:00
|
|
|
float ax = Ogre::Radian(ptr.getRefData().getPosition().rot[0]).valueDegrees();
|
|
|
|
float ay = Ogre::Radian(ptr.getRefData().getPosition().rot[1]).valueDegrees();
|
|
|
|
if(ptr.getTypeName() == "struct ESM::NPC")//some morrowind oddity
|
|
|
|
{
|
|
|
|
ax = ax/60.;
|
|
|
|
ay = ay/60.;
|
|
|
|
zRot = zRot/60.;
|
|
|
|
}
|
|
|
|
MWBase::Environment::get().getWorld()->rotateObject(ptr,ax,ay,zRot);
|
2012-08-09 23:45:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-08-05 16:44:56 +02:00
|
|
|
template<class R>
|
|
|
|
class OpPosition : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
Interpreter::Type_Float x = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
Interpreter::Type_Float z = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
Interpreter::Type_Float zRot = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
2012-08-11 18:07:20 +02:00
|
|
|
int cx,cy;
|
|
|
|
MWBase::Environment::get().getWorld()->positionToIndex(x,y,cx,cy);
|
2012-08-11 09:52:49 +02:00
|
|
|
MWBase::Environment::get().getWorld()->moveObject(ptr,
|
2012-08-11 18:07:20 +02:00
|
|
|
*MWBase::Environment::get().getWorld()->getExterior(cx,cy),x,y,z);
|
2012-08-05 16:44:56 +02:00
|
|
|
float ax = Ogre::Radian(ptr.getRefData().getPosition().rot[0]).valueDegrees();
|
|
|
|
float ay = Ogre::Radian(ptr.getRefData().getPosition().rot[1]).valueDegrees();
|
|
|
|
if(ptr.getTypeName() == "struct ESM::NPC")//some morrowind oddity
|
|
|
|
{
|
|
|
|
ax = ax/60.;
|
|
|
|
ay = ay/60.;
|
|
|
|
zRot = zRot/60.;
|
|
|
|
}
|
|
|
|
MWBase::Environment::get().getWorld()->rotateObject(ptr,ax,ay,zRot);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-07-09 18:47:59 +02:00
|
|
|
const int opcodeSetScale = 0x2000164;
|
|
|
|
const int opcodeSetScaleExplicit = 0x2000165;
|
|
|
|
const int opcodeSetAngle = 0x2000166;
|
|
|
|
const int opcodeSetAngleExplicit = 0x2000167;
|
2012-07-09 19:28:44 +02:00
|
|
|
const int opcodeGetScale = 0x2000168;
|
|
|
|
const int opcodeGetScaleExplicit = 0x2000169;
|
|
|
|
const int opcodeGetAngle = 0x200016a;
|
|
|
|
const int opcodeGetAngleExplicit = 0x200016b;
|
2012-08-03 18:14:05 +02:00
|
|
|
const int opcodeGetPos = 0x200016c;
|
|
|
|
const int opcodeGetPosExplicit = 0x200016d;
|
|
|
|
const int opcodeSetPos = 0x200016e;
|
|
|
|
const int opcodeSetPosExplicit = 0x200016f;
|
2012-08-03 18:20:51 +02:00
|
|
|
const int opcodeGetStartingPos = 0x2000170;
|
|
|
|
const int opcodeGetStartingPosExplicit = 0x2000171;
|
2012-08-05 16:44:56 +02:00
|
|
|
const int opcodePosition = 0x2000172;
|
|
|
|
const int opcodePositionExplicit = 0x2000173;
|
2012-08-09 23:45:06 +02:00
|
|
|
const int opcodePositionCell = 0x2000174;
|
|
|
|
const int opcodePositionCellExplicit = 0x2000175;
|
2012-07-09 18:47:59 +02:00
|
|
|
|
|
|
|
void registerExtensions (Compiler::Extensions& extensions)
|
|
|
|
{
|
|
|
|
extensions.registerInstruction("setscale","f",opcodeSetScale,opcodeSetScaleExplicit);
|
2012-07-09 19:28:44 +02:00
|
|
|
extensions.registerFunction("getscale",'f',"",opcodeGetScale,opcodeGetScaleExplicit);
|
|
|
|
extensions.registerInstruction("setangle","Sf",opcodeSetAngle,opcodeSetAngleExplicit);
|
|
|
|
extensions.registerFunction("getangle",'f',"S",opcodeGetAngle,opcodeGetAngleExplicit);
|
2012-08-05 16:21:53 +02:00
|
|
|
extensions.registerInstruction("setpos","cf",opcodeSetPos,opcodeSetPosExplicit);
|
|
|
|
extensions.registerFunction("getpos",'f',"c",opcodeGetPos,opcodeGetPosExplicit);
|
|
|
|
extensions.registerFunction("getstartingpos",'f',"c",opcodeGetStartingPos,opcodeGetStartingPosExplicit);
|
2012-08-05 16:44:56 +02:00
|
|
|
extensions.registerInstruction("position","ffff",opcodePosition,opcodePositionExplicit);
|
2012-08-09 23:45:06 +02:00
|
|
|
extensions.registerInstruction("positioncell","ffffS",opcodePositionCell,opcodePositionCellExplicit);
|
2012-07-09 18:47:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void installOpcodes (Interpreter::Interpreter& interpreter)
|
|
|
|
{
|
|
|
|
interpreter.installSegment5(opcodeSetScale,new OpSetScale<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeSetScaleExplicit,new OpSetScale<ExplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeSetAngle,new OpSetAngle<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeSetAngleExplicit,new OpSetAngle<ExplicitRef>);
|
2012-07-09 19:28:44 +02:00
|
|
|
interpreter.installSegment5(opcodeGetScale,new OpGetScale<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeGetScaleExplicit,new OpGetScale<ExplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeGetAngle,new OpGetAngle<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeGetAngleExplicit,new OpGetAngle<ExplicitRef>);
|
2012-08-03 18:14:05 +02:00
|
|
|
interpreter.installSegment5(opcodeGetPos,new OpGetPos<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeGetPosExplicit,new OpGetPos<ExplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeSetPos,new OpSetPos<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeSetPosExplicit,new OpSetPos<ExplicitRef>);
|
2012-08-03 18:20:51 +02:00
|
|
|
interpreter.installSegment5(opcodeGetStartingPos,new OpGetStartingPos<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodeGetStartingPosExplicit,new OpGetStartingPos<ExplicitRef>);
|
2012-08-05 16:44:56 +02:00
|
|
|
interpreter.installSegment5(opcodePosition,new OpPosition<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodePositionExplicit,new OpPosition<ExplicitRef>);
|
2012-08-09 23:45:06 +02:00
|
|
|
interpreter.installSegment5(opcodePositionCell,new OpPositionCell<ImplicitRef>);
|
|
|
|
interpreter.installSegment5(opcodePositionCellExplicit,new OpPositionCell<ExplicitRef>);
|
2012-07-09 18:47:59 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|