1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-22 00:53:50 +00:00
openmw-tes3mp/components/interpreter/miscopcodes.hpp

284 lines
8 KiB
C++
Raw Normal View History

#ifndef INTERPRETER_MISCOPCODES_H_INCLUDED
#define INTERPRETER_MISCOPCODES_H_INCLUDED
#include <stdexcept>
#include <vector>
#include <string>
2010-07-01 14:40:03 +00:00
#include <sstream>
#include <algorithm>
#include "opcodes.hpp"
#include "runtime.hpp"
#include "defines.hpp"
2015-04-22 15:58:55 +00:00
#include <components/misc/rng.hpp>
#include <components/misc/messageformatparser.hpp>
namespace Interpreter
{
class RuntimeMessageFormatter : public Misc::MessageFormatParser
{
private:
std::string mFormattedMessage;
Runtime& mRuntime;
protected:
virtual void visitedPlaceholder(Placeholder placeholder, char padding, int width, int precision, Notation notation)
{
std::ostringstream out;
out.fill(padding);
if (width != -1)
out.width(width);
if (precision != -1)
out.precision(precision);
switch (placeholder)
{
case StringPlaceholder:
{
int index = mRuntime[0].mInteger;
mRuntime.pop();
out << mRuntime.getStringLiteral(index);
mFormattedMessage += out.str();
}
break;
case IntegerPlaceholder:
{
Type_Integer value = mRuntime[0].mInteger;
mRuntime.pop();
out << value;
mFormattedMessage += out.str();
}
break;
case FloatPlaceholder:
{
float value = mRuntime[0].mFloat;
mRuntime.pop();
if (notation == FixedNotation)
2018-08-13 19:31:11 +00:00
{
out << std::fixed << value;
2018-08-13 19:31:11 +00:00
mFormattedMessage += out.str();
}
else if (notation == ShortestNotation)
{
2018-09-14 15:37:04 +00:00
out << value;
std::string standard = out.str();
2018-08-13 19:31:11 +00:00
out.str(std::string());
out.clear();
2018-09-14 15:37:04 +00:00
out << std::scientific << value;
std::string scientific = out.str();
2018-08-13 19:31:11 +00:00
2018-09-14 15:37:04 +00:00
mFormattedMessage += standard.length() < scientific.length() ? standard : scientific;
2018-08-13 19:31:11 +00:00
}
2018-09-14 15:37:04 +00:00
else
2018-08-13 19:31:11 +00:00
{
out << std::scientific << value;
mFormattedMessage += out.str();
}
}
break;
default:
break;
}
}
virtual void visitedCharacter(char c)
{
mFormattedMessage += c;
}
public:
RuntimeMessageFormatter(Runtime& runtime)
: mRuntime(runtime)
{
}
virtual void process(const std::string& message)
{
mFormattedMessage.clear();
MessageFormatParser::process(message);
}
std::string getFormattedMessage() const
{
return mFormattedMessage;
}
};
inline std::string formatMessage (const std::string& message, Runtime& runtime)
{
RuntimeMessageFormatter formatter(runtime);
formatter.process(message);
std::string formattedMessage = formatter.getFormattedMessage();
formattedMessage = fixDefinesMsgBox(formattedMessage, runtime.getContext());
return formattedMessage;
}
class OpMessageBox : public Opcode1
{
public:
virtual void execute (Runtime& runtime, unsigned int arg0)
{
2010-08-22 10:56:35 +00:00
// message
int index = runtime[0].mInteger;
runtime.pop();
std::string message = runtime.getStringLiteral (index);
// buttons
std::vector<std::string> buttons;
for (std::size_t i=0; i<arg0; ++i)
{
2016-08-29 10:20:00 +00:00
index = runtime[0].mInteger;
runtime.pop();
buttons.push_back (runtime.getStringLiteral (index));
}
std::reverse (buttons.begin(), buttons.end());
// handle additional parameters
std::string formattedMessage = formatMessage (message, runtime);
runtime.getContext().messageBox (formattedMessage, buttons);
}
};
class OpReport : public Opcode0
{
public:
virtual void execute (Runtime& runtime)
{
// message
int index = runtime[0].mInteger;
runtime.pop();
std::string message = runtime.getStringLiteral (index);
// handle additional parameters
std::string formattedMessage = formatMessage (message, runtime);
runtime.getContext().report (formattedMessage);
}
};
class OpMenuMode : public Opcode0
{
public:
virtual void execute (Runtime& runtime)
{
runtime.push (runtime.getContext().menuMode());
}
};
class OpRandom : public Opcode0
{
public:
virtual void execute (Runtime& runtime)
{
Type_Integer limit = runtime[0].mInteger;
if (limit<0)
throw std::runtime_error (
"random: argument out of range (Don't be so negative!)");
runtime[0].mFloat = static_cast<Type_Float>(Misc::Rng::rollDice(limit)); // [o, limit)
}
};
2010-07-06 10:06:50 +00:00
class OpGetSecondsPassed : public Opcode0
{
public:
2010-07-06 10:06:50 +00:00
virtual void execute (Runtime& runtime)
{
Type_Float duration = runtime.getContext().getSecondsPassed();
runtime.push (duration);
}
2010-07-06 10:06:50 +00:00
};
2010-07-09 14:07:03 +00:00
class OpEnable : public Opcode0
{
public:
2010-07-09 14:07:03 +00:00
virtual void execute (Runtime& runtime)
{
runtime.getContext().enable();
}
};
2010-07-09 14:07:03 +00:00
class OpDisable : public Opcode0
{
public:
2010-07-09 14:07:03 +00:00
virtual void execute (Runtime& runtime)
{
runtime.getContext().disable();
}
};
2010-07-09 14:07:03 +00:00
class OpGetDisabled : public Opcode0
{
public:
2010-07-09 14:07:03 +00:00
virtual void execute (Runtime& runtime)
{
runtime.push (runtime.getContext().isDisabled());
}
};
class OpEnableExplicit : public Opcode0
{
public:
virtual void execute (Runtime& runtime)
{
int index = runtime[0].mInteger;
runtime.pop();
std::string id = runtime.getStringLiteral (index);
runtime.getContext().enable (id);
}
};
class OpDisableExplicit : public Opcode0
{
public:
virtual void execute (Runtime& runtime)
{
int index = runtime[0].mInteger;
runtime.pop();
std::string id = runtime.getStringLiteral (index);
runtime.getContext().disable (id);
}
};
class OpGetDisabledExplicit : public Opcode0
{
public:
virtual void execute (Runtime& runtime)
{
int index = runtime[0].mInteger;
runtime.pop();
std::string id = runtime.getStringLiteral (index);
runtime.push (runtime.getContext().isDisabled (id));
}
};
}
#endif