TDME2 1.9.121
MiniScript.h
Go to the documentation of this file.
1#pragma once
2
3#include <algorithm>
4#include <array>
5#include <stack>
6#include <string>
7#include <string_view>
8#include <unordered_map>
9#include <vector>
10
11#include <tdme/tdme.h>
13#include <tdme/math/Vector3.h>
18#include <tdme/utilities/Time.h>
19
20using std::array;
21using std::remove;
22using std::stack;
23using std::string;
24using std::string_view;
25using std::to_string;
26using std::unordered_map;
27using std::vector;
28
36
37/**
38 * Miniscript
39 */
41public:
44 // priority 3
46 // priority 5
49 OPERATOR_REMAINDER, // TODO: not yet
50 // priority 6
53 // priority 9
58 // priority 10
61 // priority 14
63 // priority 15
65 // priority 16
67 //
69 };
70
76 };
77
79 int line;
81 string statement;
83 };
84
93 };
94
96 private:
100 bool booleanValue { false };
101 int64_t integerValue { 0 };
102 float floatValue { 0.0f };
104
105 public:
106 /**
107 * Constructor
108 */
109 inline ScriptVariable() {
110 }
111
112 /**
113 * Constructor
114 * @param value value
115 */
116 inline ScriptVariable(bool value) {
117 setValue(value);
118 }
119
120 /**
121 * Constructor
122 * @param value value
123 */
124 inline ScriptVariable(int64_t value) {
125 setValue(value);
126 }
127
128 /**
129 * Constructor
130 * @param value value
131 */
132 inline ScriptVariable(float value) {
133 setValue(value);
134 }
135
136 /**
137 * Constructor
138 * @param value value
139 */
140 inline ScriptVariable(const string& value) {
141 setValue(value);
142 }
143
144 /**
145 * Constructor
146 * @param value value
147 */
148 inline ScriptVariable(const Vector3& value) {
149 setValue(value);
150 }
151
152 /**
153 * Constructor
154 * @param value value
155 */
156 inline ScriptVariable(const Transformations& value) {
157 setValue(value);
158 }
159
160 /**
161 * @return type
162 */
164 return type;
165 }
166
167 /**
168 * Get boolean value from given variable
169 * @param value value
170 * @param optional optionalfalse
171 * @return success
172 */
173 inline bool getBooleanValue(bool& value, bool optional = false) const {
174 switch(type) {
175 case TYPE_VOID:
176 return optional;
177 case TYPE_BOOLEAN:
178 value = booleanValue;
179 return true;
180 break;
181 case TYPE_INTEGER:
182 value = integerValue != 0;
183 return true;
184 case TYPE_FLOAT:
185 value = floatValue != 0.0f;
186 return true;
187 case TYPE_STRING:
188 {
189 auto lowerCaseString = StringTools::toLowerCase(stringValue);
190 if (lowerCaseString != "false" && lowerCaseString != "true" && lowerCaseString != "1" && lowerCaseString != "0") return optional;
191 value = lowerCaseString == "true" || lowerCaseString == "1";
192 return true;
193 }
194 case TYPE_VECTOR3:
195 return false;
197 return false;
198 break;
199 }
200 return false;
201 }
202
203 /**
204 * Get integer value from given variable
205 * @param value value
206 * @param optional optional
207 * @return success
208 */
209 inline bool getIntegerValue(int64_t& value, bool optional = false) const {
210 switch(type) {
211 case TYPE_VOID:
212 return optional;
213 case TYPE_BOOLEAN:
214 value = booleanValue == true?1:0;
215 return true;
216 break;
217 case TYPE_INTEGER:
218 value = integerValue;
219 return true;
220 case TYPE_FLOAT:
221 Console::println("MiniScript::getIntegerValue(): converting float to integer: precision loss");
222 value = floatValue;
223 return true;
224 case TYPE_STRING:
225 if (Integer::is(stringValue) == true) {
227 return true;
228 } else
229 if (Float::is(stringValue) == true) {
230 Console::println("MiniScript::getIntegerValue(): converting float to integer: precision loss");
231 value = static_cast<int64_t>(Float::parse(stringValue));
232 return true;
233 } else {
234 return optional;
235 }
236 case TYPE_VECTOR3:
237 return optional;
239 return optional;
240 }
241 return false;
242 }
243
244 /**
245 * Get float value from given variable
246 * @param value value
247 * @param optional optional
248 * @return success
249 */
250 inline bool getFloatValue(float& value, bool optional = false) const {
251 switch(type) {
252 case TYPE_VOID:
253 return optional;
254 case TYPE_BOOLEAN:
255 value = booleanValue == true?1.0f:0.0f;
256 return true;
257 break;
258 case TYPE_INTEGER:
259 value = integerValue;
260 return true;
261 case TYPE_FLOAT:
262 value = floatValue;
263 return true;
264 case TYPE_STRING:
265 if (Float::is(stringValue) == false) return optional;
266 value = Float::parse(stringValue);
267 return true;
268 case TYPE_VECTOR3:
269 return optional;
271 return optional;
272 }
273 return false;
274 }
275
276 /**
277 * Get string value from given variable
278 * @param value value
279 * @param optional optional
280 * @return success
281 */
282 inline bool getStringValue(string& value, bool optional = false) const {
283 switch(type) {
284 case TYPE_VOID:
285 return optional;
286 case TYPE_BOOLEAN:
287 value = booleanValue == true?"true":"false";
288 return true;
289 case TYPE_INTEGER:
290 value = to_string(integerValue);
291 return true;
292 case TYPE_FLOAT:
293 value = to_string(floatValue);
294 return true;
295 case TYPE_STRING:
296 value = stringValue;
297 return true;
298 case TYPE_VECTOR3:
299 return false;
301 return false;
302 }
303 return false;
304 }
305
306 /**
307 * Get vector3 value from given variable
308 * @param value value
309 * @param optional optional
310 * @return success
311 */
312 inline bool getVector3Value(Vector3& value, bool optional = false) const {
313 switch(type) {
314 case TYPE_VOID:
315 return optional;
316 case TYPE_BOOLEAN:
317 return optional;
318 case TYPE_INTEGER:
319 return optional;
320 case TYPE_FLOAT:
321 return optional;
322 case TYPE_STRING:
323 return optional;
324 case TYPE_VECTOR3:
325 value = vector3Value;
326 return true;
328 return optional;
329 }
330 return false;
331 }
332
333 /**
334 * Get transformations value from given variable
335 * @param value value
336 * @param optional optional
337 * @return success
338 */
339 inline bool getTransformationsValue(Transformations& value, bool optional = false) const {
340 switch(type) {
341 case TYPE_VOID:
342 return optional;
343 case TYPE_BOOLEAN:
344 return optional;
345 case TYPE_INTEGER:
346 return optional;
347 case TYPE_FLOAT:
348 return optional;
349 case TYPE_STRING:
350 return optional;
351 case TYPE_VECTOR3:
352 return optional;
354 value = transformationsValue;
355 return true;
356 }
357 return false;
358 }
359
360 /**
361 * Set boolean value from given value into variable
362 * @param value value
363 */
364 inline void setValue(bool value) {
366 booleanValue = value;
367 }
368
369 /**
370 * Set integer value from given value into variable
371 * @param value value
372 */
373 inline void setValue(int64_t value) {
375 integerValue = value;
376 }
377
378 /**
379 * Set float value from given value into variable
380 * @param value value
381 */
382 inline void setValue(float value) {
384 floatValue = value;
385 }
386
387 /**
388 * Set string value from given value into variable
389 * @param value value
390 */
391 inline void setValue(const string& value) {
393 stringValue = value;
394 }
395
396 /**
397 * Set vector3 value from given value into variable
398 * @param value value
399 */
400 inline void setValue(const Vector3& value) {
402 vector3Value = value;
403 }
404
405 /**
406 * Set transformations value from given value into variable
407 * @param value value
408 */
409 inline void setValue(const Transformations& value) {
411 transformationsValue = value;
412 }
413
414 /**
415 * Set implicit typed value given by value string
416 * @param value value
417 */
418 inline void setImplicitTypedValue(const string& value) {
419 if (value == "true") {
421 booleanValue = true;
422 } else
423 if (value == "false") {
425 booleanValue = false;
426 } else
427 if (Integer::is(value) == true) {
430 } else
431 if (Float::is(value) == true) {
433 floatValue = Float::parse(value);
434 } else {
436 stringValue = value;
437 }
438 }
439
440 /**
441 * Set implicit typed value given by value string
442 * @param value value
443 */
444 inline void setImplicitTypedValueFromStringView(const string_view& value) {
445 if (value == "true") {
447 booleanValue = true;
448 } else
449 if (value == "false") {
451 booleanValue = false;
452 } else
453 if (Integer::viewIs(value) == true) {
456 } else
457 if (Float::viewIs(value) == true) {
460 } else {
462 stringValue = string(value);
463 }
464 }
465
466 /**
467 * @return string representation of script variable type
468 */
469 inline static const string getTypeAsString(ScriptVariableType type) {
470 switch(type) {
471 case TYPE_VOID: return "Void";
472 case TYPE_BOOLEAN: return "Boolean";
473 case TYPE_INTEGER: return "Integer";
474 case TYPE_FLOAT: return "Float";
475 case TYPE_STRING: return "String";
476 case TYPE_VECTOR3: return "Vector3";
477 case TYPE_TRANSFORMATIONS: return "Transformations";
478 }
479 return string();
480 }
481
482 /**
483 * @return string representation of script variable type
484 */
485 inline const string getTypeAsString() {
486 return getTypeAsString(type);
487 }
488
489 /**
490 * @return string representation of script variable type
491 */
492 inline const string getAsString() {
493 string result;
494 result+= getTypeAsString();
495 result+= "(";
496 result+= getValueString();
497 result+= ")";
498 return result;
499 }
500
501 /**
502 * @return string representation of script variable type
503 */
504 inline const string getValueString() const {
505 string result;
506 switch (type) {
507 case TYPE_VOID:
508 break;
509 case TYPE_BOOLEAN:
510 result+= booleanValue == true?"1":"0";
511 break;
512 case TYPE_INTEGER:
513 result+= to_string(integerValue);
514 break;
515 case TYPE_FLOAT:
516 result+= to_string(floatValue);
517 break;
518 case TYPE_STRING:
519 result+= stringValue;
520 break;
521 case TYPE_VECTOR3:
522 result+=
523 "Vector3(" +
524 to_string(vector3Value.getX()) + ", " +
525 to_string(vector3Value.getY()) + ", " +
526 to_string(vector3Value.getZ()) + ")";
527 break;
529 result+=
530 "Transformations(translation = Vector3(" +
531 to_string(transformationsValue.getTranslation().getX()) + ", " +
532 to_string(transformationsValue.getTranslation().getY()) + ", " +
533 to_string(transformationsValue.getTranslation().getZ()) + "), " +
534 "scale = Vector3(" +
535 to_string(transformationsValue.getScale().getX()) + ", " +
536 to_string(transformationsValue.getScale().getY()) + ", " +
537 to_string(transformationsValue.getScale().getZ()) + ")";
538 for (auto i = 0; i < transformationsValue.getRotationCount(); i++) {
539 result+= ", rotations = (axis = Vector3(" +
540 to_string(transformationsValue.getRotationAxis(i).getX()) + ", " +
541 to_string(transformationsValue.getRotationAxis(i).getY()) + ", " +
542 to_string(transformationsValue.getRotationAxis(i).getZ()) + "), angle = " +
543 to_string(transformationsValue.getRotationAngle(i)) + ")";
544 }
545 result+= ")";
546 }
547 return result;
548 }
549 };
550
551 /**
552 * Script State Machine State
553 */
555 public:
556 /**
557 * Constructor
558 */
560
561 /**
562 * Destructor
563 */
565
566 /**
567 * @return name
568 */
569 virtual const string getName() = 0;
570
571 /**
572 * @return id
573 */
574 virtual int getId() = 0;
575
576 /**
577 * Execute script state machine state
578 */
579 virtual void execute() = 0;
580 };
581
582 /**
583 * Script
584 */
585 struct Script {
588 int line;
589 string condition;
590 string name;
592 vector<ScriptStatement> statements;
593 };
594
595 /**
596 * Script method
597 */
599 public:
602 string name;
604 };
605
606 /**
607 * Constructor
608 * @param argumentTypes argument types
609 * @param returnValueType return value type
610 */
611 ScriptMethod(const vector<ArgumentType>& argumentTypes = {}, ScriptVariableType returnValueType = ScriptVariableType::TYPE_VOID): returnValueType(returnValueType), argumentTypes(argumentTypes) {}
612
613 /**
614 * Destructor
615 */
616 virtual ~ScriptMethod() {}
617
618 /**
619 * @return script method name
620 */
621 virtual const string getMethodName() = 0;
622
623 /**
624 * Execute script method
625 * @param argumentValues argument values
626 * @param returnValue return value
627 * @param statement statement
628 */
629 virtual void executeMethod(const vector<ScriptVariable>& argumentValues, ScriptVariable& returnValue, const ScriptStatement& statement) = 0;
630
631 /**
632 * @return arguments
633 */
634 const vector<ArgumentType>& getArgumentTypes() {
635 return argumentTypes;
636 }
637
638 /**
639 * @return return value type
640 */
642 return returnValueType;
643 }
644
645 /**
646 * @return if variadic method
647 */
648 virtual bool isVariadic() {
649 return false;
650 }
651
652 /**
653 * @return if mixed return value
654 */
655 virtual bool isMixedReturnValue() {
656 return false;
657 }
658
659 /**
660 * @return operator
661 */
663 return OPERATOR_NONE;
664 }
665
666 /**
667 * @return description
668 */
669 virtual string getDescription() {
670 return "No description.";
671 }
672
673 private:
674 vector<ArgumentType> argumentTypes;
676 };
677
678protected:
679 struct ScriptState {
684 };
686 int state { -1 };
687 int lastState { -1 };
689 };
690 bool running { false };
691 int scriptIdx { -1 };
692 int statementIdx { -1 };
693 int64_t timeWaitStarted { -1LL };
694 int64_t timeWaitTime { -1LL };
695 string id;
696 unordered_map<string, ScriptVariable*> variables;
697 unordered_map<int, int64_t> forTimeStarted;
698 stack<bool> conditionStack;
699 stack<EndType> endTypeStack;
703 };
704
705 //
706 string hash;
707 bool native;
708 vector<Script> scripts;
709 vector<Script> nativeScripts;
711
712 /**
713 * Initialize native mini script
714 */
715 virtual void initializeNative();
716
717 /**
718 * @return if this script was compiled to C++ and is executed nativly
719 */
720 inline bool getNative() {
721 return native;
722 }
723
724 /**
725 * Set native
726 * @param native native
727 */
728 inline void setNative(bool native) {
729 this->native = native;
730 }
731
732 /**
733 * Set hash
734 * @param hash hash
735 */
736 inline void setHash(const string& hash) {
737 this->hash = hash;
738 }
739
740 /**
741 * Goto statement from given statements goto statement
742 * @param statement statement
743 */
744 void gotoStatementGoto(const ScriptStatement& statement) {
746 }
747
748 /**
749 * @return native script
750 */
751 inline vector<Script> getNativeScripts() {
752 return nativeScripts;
753 }
754
755 /**
756 * Set native scripts
757 * @param native native scripts
758 */
759 inline void setNativeScripts(const vector<Script>& nativeScripts) {
760 this->nativeScripts = nativeScripts;
761 }
762
763 /**
764 * Execute state machine
765 */
766 void executeStateMachine();
767
768 /**
769 * Reset script execution state
770 * @param scriptIdx script index
771 * @param stateMachineState state machine state
772 */
773 inline void resetScriptExecutationState(int scriptIdx, StateMachineState stateMachineState) {
776 while (scriptState.conditionStack.empty() == false) scriptState.conditionStack.pop();
777 while (scriptState.endTypeStack.empty() == false) scriptState.endTypeStack.pop();
778 scriptState.id.clear();
779 scriptState.scriptIdx = scriptIdx;
783 setScriptState(stateMachineState);
784 }
785
786 /**
787 * Stop script execution
788 */
789 inline void stopScriptExecutation() {
790 //
791 scriptState.running = false;
792 for (auto& scriptVariableIt: scriptState.variables) delete scriptVariableIt.second;
793 scriptState.variables.clear();
796 }
797
798 /**
799 * Set script state machine state
800 * @param state state
801 */
802 inline void setScriptState(int state) {
803 if (scriptState.state.state == state) return;
804 scriptState.state.state = state;
807 }
808
809 /**
810 * Determine script index to start
811 * @return script index or -1 if no script to start
812 */
813 virtual int determineScriptIdxToStart();
814
815 /**
816 * Determine named script index to start
817 * @return script index or -1 if no script to start
818 */
819 virtual int determineNamedScriptIdxToStart();
820
821private:
822 static constexpr bool VERBOSE { false };
823
824 //
826
827 //
828 unordered_map<string, ScriptMethod*> scriptMethods;
829 unordered_map<int, ScriptStateMachineState*> scriptStateMachineStates;
830 unordered_map<uint8_t, ScriptMethod*> scriptOperators;
833 bool scriptValid { false };
834
835 //
837 int idx { -1 };
839 };
840
841 /**
842 * Returns arguments as string placed in a vector of string_views
843 * @param arguments arguments
844 * @return arguments as string
845 */
846 inline const string getArgumentsAsString(const vector<string_view>& arguments) {
847 string argumentsString;
848 for (auto& argument: arguments) argumentsString+= (argumentsString.empty() == false?", ":"") + string("'") + string(argument) + string("'");
849 return argumentsString;
850 }
851
852 /**
853 * Execute a single script line
854 */
855 void executeScriptLine();
856
857 /**
858 * Parse a script statement
859 * @param statement statement
860 * @param method method
861 * @param arguments arguments
862 * @return success
863 */
864 bool parseScriptStatement(const string_view& statement, string_view& method, vector<string_view>& arguments);
865
866 /**
867 * Execute a script statement
868 * @param method method
869 * @param arguments arguments
870 * @param statement statement
871 * @return return value as script variablle
872 */
873 ScriptVariable executeScriptStatement(const string_view& method, const vector<string_view>& arguments, const ScriptStatement& statement);
874
875 /**
876 * Returns if char is operator char
877 * @param c char to test
878 * @return if char is operator char
879 */
880 static inline const bool isOperatorChar(char c) {
881 return OPERATOR_CHARS.find(c) != string::npos;
882 }
883
884 /**
885 * Determine next not substituted operator in statement
886 * @param statement statement
887 * @param nextOperator next operator
888 */
889 bool getNextStatementOperator(const string& statement, ScriptStatementOperator& nextOperator);
890
891 /**
892 * Trim argument and remove unnessessary parenthesis
893 * @param argument argument
894 * @return processed argument
895 */
896 const string trimArgument(const string& argument);
897
898 /**
899 * Find right argument in statement beginning from position
900 * @param statement statement
901 * @param position position
902 * @param length argument length
903 */
904 const string findRightArgument(const string statement, int position, int& length);
905
906 /**
907 * Find left argument in statement beginning from position
908 * @param statement statement
909 * @param position position
910 * @param length argument length
911 */
912 const string findLeftArgument(const string statement, int position, int& length);
913
914 /**
915 * Do statement pre processing, 1) replace operators with corresponding methods
916 * @param statement statement
917 */
918 const string doStatementPreProcessing(const string& statement);
919
920 /**
921 * Transpile script statement
922 * @param generatedCode generated code
923 * @param method method
924 * @param arguments arguments
925 * @param statement script statement
926 * @param scriptIdx script index
927 * @param statementIdx statement index
928 * @param methodCodeMap method code map
929 * @param scriptStateChanged script could have state changed
930 * @param scriptStopped script could have been stopped
931 * @param enabledNamedConditions enabled named conditions
932 * @param depth depth
933 * @param argumentIdx argument index
934 * @param parentArgumentIdx parent argument index
935 * @param returnValue return value
936 * @param injectCode code to additionally inject
937 * @param additionalIndent additional indent
938 */
939 bool transpileScriptStatement(string& generatedCode, const string_view& method, const vector<string_view>& arguments, const ScriptStatement& statement, int scriptIdx, int& statementIdx, const unordered_map<string, vector<string>>& methodCodeMap, bool& scriptStateChanged, bool& scriptStopped, vector<string>& enabledNamedConditions, int depth = 0, int argumentIdx = -1, int parentArgumentIdx = -1, const string& returnValue = string(), const string& injectCode = string(), int additionalIndent = 0);
940
941public:
942 /**
943 * Default constructor
944 */
945 MiniScript();
946
947 /**
948 * Destructor
949 */
950 virtual ~MiniScript();
951
952 /**
953 * @return script hash
954 */
955 inline const string& getHash() {
956 return hash;
957 }
958
959 /**
960 * @return scripts
961 */
962 const vector<Script>& getScripts() {
963 return scripts;
964 }
965
966 /**
967 * @return script state machine state
968 */
969 inline int getScriptState() {
970 return scriptState.state.state;
971 }
972
973 /**
974 * Start error script
975 */
976 inline void startErrorScript() {
977 emit("error");
978 }
979
980 /**
981 * Register state machine states
982 */
983 virtual void registerStateMachineStates();
984
985 /**
986 * Register methods
987 */
988 virtual void registerMethods();
989
990 /**
991 * Register variables
992 */
993 virtual void registerVariables();
994
995 /**
996 * Get operator as string
997 * @param scriptOperator script operator
998 * @return script operator as string
999 */
1000 inline static string getOperatorAsString(ScriptOperator scriptOperator) {
1001 switch(scriptOperator) {
1002 case(OPERATOR_NONE): return "NONE";
1003 case(OPERATOR_NOT): return "!";
1004 case(OPERATOR_MULTIPLICATION): return "*";
1005 case(OPERATOR_DIVISION): return "/";
1006 case(OPERATOR_REMAINDER): return "%";
1007 case(OPERATOR_ADDITION): return "+";
1008 case(OPERATOR_SUBTRACTION): return "-";
1009 case(OPERATOR_LESSER): return "<";
1010 case(OPERATOR_LESSEREQUALS): return "<=";
1011 case(OPERATOR_GREATER): return ">";
1012 case(OPERATOR_GREATEREQUALS): return ">=";
1013 case(OPERATOR_EQUALS): return "==";
1014 case(OPERATOR_NOTEQUAL): return "!=";
1015 case(OPERATOR_AND): return "&&";
1016 case(OPERATOR_OR): return "||";
1017 case(OPERATOR_SET): return "=";
1018 case(OPERATOR_MAX): return "MAX";
1019 default: return "INVALID";
1020 }
1021 }
1022
1023 /**
1024 * Check if arguments contain argument with given type
1025 * @param arguments arguments
1026 * @param type type
1027 * @return has type
1028 */
1029 inline static bool hasType(const vector<ScriptVariable>& arguments, ScriptVariableType type) {
1030 for (auto& argument: arguments) if (argument.getType() == type) return true;
1031 return false;
1032 }
1033
1034 /**
1035 * Check if arguments contain argument with given type
1036 * @param arguments arguments
1037 * @param type type
1038 * @return has type
1039 */
1040 template<std::size_t SIZE>
1041 inline static bool hasType(const array<ScriptVariable, SIZE>& arguments, ScriptVariableType type) {
1042 for (auto& argument: arguments) if (argument.getType() == type) return true;
1043 return false;
1044 }
1045
1046 /**
1047 * Get boolean value from given variable
1048 * @param arguments arguments
1049 * @param idx argument index
1050 * @param value value
1051 * @param optional optionalfalse
1052 * @return success
1053 */
1054 inline static bool getBooleanValue(const vector<ScriptVariable>& arguments, int idx, bool& value, bool optional = false) {
1055 if (idx >= arguments.size()) return optional;
1056 auto& argument = arguments[idx];
1057 return argument.getBooleanValue(value, optional);
1058 }
1059
1060 /**
1061 * Get boolean value from given variable
1062 * @param arguments arguments
1063 * @param idx argument index
1064 * @param value value
1065 * @param optional optionalfalse
1066 * @return success
1067 */
1068 template<std::size_t SIZE>
1069 inline static bool getBooleanValue(const array<ScriptVariable, SIZE>& arguments, int idx, bool& value, bool optional = false) {
1070 if (idx >= arguments.size()) return optional;
1071 auto& argument = arguments[idx];
1072 return argument.getBooleanValue(value, optional);
1073 }
1074
1075 /**
1076 * Get integer value from given variable
1077 * @param arguments arguments
1078 * @param idx argument index
1079 * @param value value
1080 * @param optional optional
1081 * @return success
1082 */
1083 inline static bool getIntegerValue(const vector<ScriptVariable>& arguments, int idx, int64_t& value, bool optional = false) {
1084 if (idx >= arguments.size()) return optional;
1085 auto& argument = arguments[idx];
1086 return argument.getIntegerValue(value, optional);
1087 }
1088
1089 /**
1090 * Get integer value from given variable
1091 * @param arguments arguments
1092 * @param idx argument index
1093 * @param value value
1094 * @param optional optional
1095 * @return success
1096 */
1097 template<std::size_t SIZE>
1098 inline static bool getIntegerValue(const array<ScriptVariable, SIZE>& arguments, int idx, int64_t& value, bool optional = false) {
1099 if (idx >= arguments.size()) return optional;
1100 auto& argument = arguments[idx];
1101 return argument.getIntegerValue(value, optional);
1102 }
1103
1104 /**
1105 * Get float value from given variable
1106 * @param arguments arguments
1107 * @param idx argument index
1108 * @param value value
1109 * @param optional optional
1110 * @return success
1111 */
1112 inline static bool getFloatValue(const vector<ScriptVariable>& arguments, int idx, float& value, bool optional = false) {
1113 if (idx >= arguments.size()) return optional;
1114 auto& argument = arguments[idx];
1115 return argument.getFloatValue(value, optional);
1116 }
1117
1118 /**
1119 * Get float value from given variable
1120 * @param arguments arguments
1121 * @param idx argument index
1122 * @param value value
1123 * @param optional optional
1124 * @return success
1125 */
1126 template<std::size_t SIZE>
1127 inline static bool getFloatValue(const array<ScriptVariable, SIZE>& arguments, int idx, float& value, bool optional = false) {
1128 if (idx >= arguments.size()) return optional;
1129 auto& argument = arguments[idx];
1130 return argument.getFloatValue(value, optional);
1131 }
1132
1133 /**
1134 * Get string value from given variable
1135 * @param arguments arguments
1136 * @param idx argument index
1137 * @param value value
1138 * @param optional optional
1139 * @return success
1140 */
1141 inline static bool getStringValue(const vector<ScriptVariable>& arguments, int idx, string& value, bool optional = false) {
1142 if (idx >= arguments.size()) return optional;
1143 auto& argument = arguments[idx];
1144 return argument.getStringValue(value, optional);
1145 }
1146
1147 /**
1148 * Get string value from given variable
1149 * @param arguments arguments
1150 * @param idx argument index
1151 * @param value value
1152 * @param optional optional
1153 * @return success
1154 */
1155 template<std::size_t SIZE>
1156 inline static bool getStringValue(const array<ScriptVariable, SIZE>& arguments, int idx, string& value, bool optional = false) {
1157 if (idx >= arguments.size()) return optional;
1158 auto& argument = arguments[idx];
1159 return argument.getStringValue(value, optional);
1160 }
1161
1162 /**
1163 * Get vector3 value from given variable
1164 * @param arguments arguments
1165 * @param idx argument index
1166 * @param value value
1167 * @param optional optional
1168 * @return success
1169 */
1170 inline static bool getVector3Value(const vector<ScriptVariable>& arguments, int idx, Vector3& value, bool optional = false) {
1171 if (idx >= arguments.size()) return optional;
1172 auto& argument = arguments[idx];
1173 return argument.getVector3Value(value, optional);
1174 }
1175
1176 /**
1177 * Get vector3 value from given variable
1178 * @param arguments arguments
1179 * @param idx argument index
1180 * @param value value
1181 * @param optional optional
1182 * @return success
1183 */
1184 template<std::size_t SIZE>
1185 inline static bool getVector3Value(const array<ScriptVariable, SIZE>& arguments, int idx, Vector3& value, bool optional = false) {
1186 if (idx >= arguments.size()) return optional;
1187 auto& argument = arguments[idx];
1188 return argument.getVector3Value(value, optional);
1189 }
1190
1191 /**
1192 * Get transformations value from given variable
1193 * @param arguments arguments
1194 * @param idx argument index
1195 * @param value value
1196 * @param optional optional
1197 * @return success
1198 */
1199 inline static bool getTransformationsValue(const vector<ScriptVariable>& arguments, int idx, Transformations& value, bool optional = false) {
1200 if (idx >= arguments.size()) return optional;
1201 auto& argument = arguments[idx];
1202 return argument.getTransformationsValue(value, optional);
1203 }
1204
1205 /**
1206 * Get transformations value from given variable
1207 * @param arguments arguments
1208 * @param idx argument index
1209 * @param value value
1210 * @param optional optional
1211 * @return success
1212 */
1213 template<std::size_t SIZE>
1214 inline static bool getTransformationsValue(const array<ScriptVariable, SIZE>& arguments, int idx, Transformations& value, bool optional = false) {
1215 if (idx >= arguments.size()) return optional;
1216 auto& argument = arguments[idx];
1217 return argument.getTransformationsValue(value, optional);
1218 }
1219
1220 /**
1221 * Register script state machine state
1222 * @param state state
1223 */
1224 void registerStateMachineState(ScriptStateMachineState* state);
1225
1226 /**
1227 * Register script method
1228 * @param method method
1229 */
1230 void registerMethod(ScriptMethod* method);
1231
1232 /**
1233 * Returns variable with given name
1234 * @param name name
1235 * @return variable
1236 */
1237 inline const ScriptVariable getVariable(const string& name) {
1238 if (StringTools::startsWith(name, "$") == false) {
1239 Console::println("MiniScript::getVariable(): '" + scriptFileName + "': variable with name '" + name + "' does not exist: variable names must start with '$'");
1240 return ScriptVariable();
1241 }
1242 auto scriptVariableIt = scriptState.variables.find(name);
1243 if (scriptVariableIt == scriptState.variables.end()) {
1244 Console::println("MiniScript::getVariable(): '" + scriptFileName + "': variable with name '" + name + "' does not exist.");
1245 return ScriptVariable();
1246 }
1247 return *scriptVariableIt->second;
1248 }
1249
1250 /**
1251 * Set script variable
1252 * @param name name
1253 * @param variable variable
1254 */
1255 inline void setVariable(const string& name, const ScriptVariable& variable) {
1256 if (StringTools::startsWith(name, "$") == false) {
1257 Console::println("MiniScript::setVariable(): '" + scriptFileName + "': variable name '" + name + "': variable names must start with '$'");
1258 return;
1259 }
1260 auto scriptVariableIt = scriptState.variables.find(name);
1261 if (scriptVariableIt != scriptState.variables.end()) {
1262 *scriptVariableIt->second = variable;
1263 return;
1264 } else {
1265 auto scriptVariable = new ScriptVariable();
1266 *scriptVariable = variable;
1267 scriptState.variables[name] = scriptVariable;
1268 }
1269 }
1270
1271 /**
1272 * Unset script variable
1273 * @param name name
1274 */
1275 inline void unsetVariable(const string& name) {
1276 if (StringTools::startsWith(name, "$") == false) {
1277 Console::println("MiniScript::unsetVariable(): '" + scriptFileName + "': variable name '" + name + "': variable names must start with '$'");
1278 return;
1279 }
1280 auto scriptVariableIt = scriptState.variables.find(name);
1281 if (scriptVariableIt == scriptState.variables.end()) return;
1282 delete scriptVariableIt->second;
1283 scriptState.variables.erase(scriptVariableIt);
1284 }
1285
1286 /**
1287 * Load script
1288 * @param pathName path name
1289 * @param fileName file name
1290 */
1291 void loadScript(const string& pathName, const string& fileName);
1292
1293 /**
1294 * Start script
1295 */
1296 virtual void startScript();
1297
1298 /**
1299 * Emit
1300 * @param condition condition
1301 */
1302 virtual void emit(const string& condition);
1303
1304 /**
1305 * Execute
1306 */
1307 virtual void execute();
1308
1309 /**
1310 * @return is running
1311 */
1312 inline bool isRunning() {
1313 return scriptState.running;
1314 }
1315
1316 /**
1317 * Get miniscript instance information
1318 */
1319 const string getInformation();
1320
1321 /**
1322 * Transpile a script statement
1323 * @param generatedCode generated code
1324 * @param scriptIdx script index
1325 * @param methodCodeMap method code map
1326 * @return success
1327 */
1328 bool transpile(string& generatedCode, int scriptIdx, const unordered_map<string, vector<string>>& methodCodeMap);
1329
1330 /**
1331 * Transpile a script condition
1332 * @param generatedCode generated code
1333 * @param scriptIdx script index
1334 * @param methodCodeMap method code map
1335 * @param returnValue return value
1336 * @param injectCode inject code
1337 * @return success
1338 */
1339 bool transpileScriptCondition(string& generatedCode, int scriptIdx, const unordered_map<string, vector<string>>& methodCodeMap, const string& returnValue, const string& injectCode, int depth = 0);
1340
1341};
Transformations which contain scale, rotations and translation.
const Vector3 & getScale() const
const Vector3 & getRotationAxis(const int idx) const
const int getRotationCount() const
const Vector3 & getTranslation() const
const float getRotationAngle(const int idx) const
3D vector 3 class
Definition: Vector3.h:22
float getY() const
Definition: Vector3.h:119
float getX() const
Definition: Vector3.h:103
float getZ() const
Definition: Vector3.h:136
Console class.
Definition: Console.h:26
static void println()
Print new line to console.
Definition: Console.cpp:78
Float class.
Definition: Float.h:23
static bool viewIs(const string_view &str)
Check if given string is a float string.
Definition: Float.cpp:33
static float viewParse(const string_view &str)
Parse float.
Definition: Float.cpp:52
static float parse(const string &str)
Parse float.
Definition: Float.cpp:41
static bool is(const string &str)
Check if given string is a float string.
Definition: Float.cpp:25
Integer class.
Definition: Integer.h:26
static bool viewIs(const string_view &str)
Check if given string is a integer string.
Definition: Integer.cpp:32
static int parse(const string &str)
Parse integer.
Definition: Integer.cpp:38
static int viewParse(const string_view &str)
Parse integer.
Definition: Integer.cpp:45
static bool is(const string &str)
Check if given string is a integer string.
Definition: Integer.cpp:26
ScriptMethod(const vector< ArgumentType > &argumentTypes={}, ScriptVariableType returnValueType=ScriptVariableType::TYPE_VOID)
Constructor.
Definition: MiniScript.h:611
const ScriptVariableType & getReturnValueType()
Definition: MiniScript.h:641
virtual void executeMethod(const vector< ScriptVariable > &argumentValues, ScriptVariable &returnValue, const ScriptStatement &statement)=0
Execute script method.
virtual ScriptOperator getOperator()
Definition: MiniScript.h:662
const vector< ArgumentType > & getArgumentTypes()
Definition: MiniScript.h:634
virtual const string getMethodName()=0
vector< ArgumentType > argumentTypes
Definition: MiniScript.h:674
virtual void execute()=0
Execute script state machine state.
bool getVector3Value(Vector3 &value, bool optional=false) const
Get vector3 value from given variable.
Definition: MiniScript.h:312
static const string getTypeAsString(ScriptVariableType type)
Definition: MiniScript.h:469
bool getIntegerValue(int64_t &value, bool optional=false) const
Get integer value from given variable.
Definition: MiniScript.h:209
void setValue(const string &value)
Set string value from given value into variable.
Definition: MiniScript.h:391
ScriptVariable(bool value)
Constructor.
Definition: MiniScript.h:116
ScriptVariable(const string &value)
Constructor.
Definition: MiniScript.h:140
ScriptVariable(float value)
Constructor.
Definition: MiniScript.h:132
bool getStringValue(string &value, bool optional=false) const
Get string value from given variable.
Definition: MiniScript.h:282
void setValue(const Transformations &value)
Set transformations value from given value into variable.
Definition: MiniScript.h:409
ScriptVariableType getType() const
Definition: MiniScript.h:163
void setValue(float value)
Set float value from given value into variable.
Definition: MiniScript.h:382
ScriptVariable(const Vector3 &value)
Constructor.
Definition: MiniScript.h:148
bool getTransformationsValue(Transformations &value, bool optional=false) const
Get transformations value from given variable.
Definition: MiniScript.h:339
ScriptVariable(int64_t value)
Constructor.
Definition: MiniScript.h:124
void setValue(int64_t value)
Set integer value from given value into variable.
Definition: MiniScript.h:373
bool getFloatValue(float &value, bool optional=false) const
Get float value from given variable.
Definition: MiniScript.h:250
bool getBooleanValue(bool &value, bool optional=false) const
Get boolean value from given variable.
Definition: MiniScript.h:173
void setValue(const Vector3 &value)
Set vector3 value from given value into variable.
Definition: MiniScript.h:400
void setImplicitTypedValueFromStringView(const string_view &value)
Set implicit typed value given by value string.
Definition: MiniScript.h:444
void setValue(bool value)
Set boolean value from given value into variable.
Definition: MiniScript.h:364
ScriptVariable(const Transformations &value)
Constructor.
Definition: MiniScript.h:156
void setImplicitTypedValue(const string &value)
Set implicit typed value given by value string.
Definition: MiniScript.h:418
vector< Script > scripts
Definition: MiniScript.h:708
static bool getFloatValue(const array< ScriptVariable, SIZE > &arguments, int idx, float &value, bool optional=false)
Get float value from given variable.
Definition: MiniScript.h:1127
bool transpile(string &generatedCode, int scriptIdx, const unordered_map< string, vector< string > > &methodCodeMap)
Transpile a script statement.
void setHash(const string &hash)
Set hash.
Definition: MiniScript.h:736
void resetScriptExecutationState(int scriptIdx, StateMachineState stateMachineState)
Reset script execution state.
Definition: MiniScript.h:773
static bool getBooleanValue(const vector< ScriptVariable > &arguments, int idx, bool &value, bool optional=false)
Get boolean value from given variable.
Definition: MiniScript.h:1054
static bool getVector3Value(const vector< ScriptVariable > &arguments, int idx, Vector3 &value, bool optional=false)
Get vector3 value from given variable.
Definition: MiniScript.h:1170
void loadScript(const string &pathName, const string &fileName)
Load script.
Definition: MiniScript.cpp:481
void registerStateMachineState(ScriptStateMachineState *state)
Register script state machine state.
Definition: MiniScript.cpp:75
const string & getHash()
Definition: MiniScript.h:955
vector< Script > getNativeScripts()
Definition: MiniScript.h:751
static string getOperatorAsString(ScriptOperator scriptOperator)
Get operator as string.
Definition: MiniScript.h:1000
unordered_map< int, ScriptStateMachineState * > scriptStateMachineStates
Definition: MiniScript.h:829
virtual void startScript()
Start script.
Definition: MiniScript.cpp:743
const string findLeftArgument(const string statement, int position, int &length)
Find left argument in statement beginning from position.
Definition: MiniScript.cpp:974
static bool getIntegerValue(const array< ScriptVariable, SIZE > &arguments, int idx, int64_t &value, bool optional=false)
Get integer value from given variable.
Definition: MiniScript.h:1098
static bool getFloatValue(const vector< ScriptVariable > &arguments, int idx, float &value, bool optional=false)
Get float value from given variable.
Definition: MiniScript.h:1112
unordered_map< string, ScriptMethod * > scriptMethods
Definition: MiniScript.h:828
ScriptVariable executeScriptStatement(const string_view &method, const vector< string_view > &arguments, const ScriptStatement &statement)
Execute a script statement.
Definition: MiniScript.cpp:254
static bool hasType(const array< ScriptVariable, SIZE > &arguments, ScriptVariableType type)
Check if arguments contain argument with given type.
Definition: MiniScript.h:1041
virtual void execute()
Execute.
Definition: MiniScript.cpp:462
virtual void registerStateMachineStates()
Register state machine states.
virtual void initializeNative()
Initialize native mini script.
Definition: MiniScript.cpp:84
void setVariable(const string &name, const ScriptVariable &variable)
Set script variable.
Definition: MiniScript.h:1255
virtual void registerVariables()
Register variables.
static constexpr bool VERBOSE
Definition: MiniScript.h:822
bool transpileScriptStatement(string &generatedCode, const string_view &method, const vector< string_view > &arguments, const ScriptStatement &statement, int scriptIdx, int &statementIdx, const unordered_map< string, vector< string > > &methodCodeMap, bool &scriptStateChanged, bool &scriptStopped, vector< string > &enabledNamedConditions, int depth=0, int argumentIdx=-1, int parentArgumentIdx=-1, const string &returnValue=string(), const string &injectCode=string(), int additionalIndent=0)
Transpile script statement.
virtual void registerMethods()
Register methods.
virtual void emit(const string &condition)
Emit.
Definition: MiniScript.cpp:389
static bool getIntegerValue(const vector< ScriptVariable > &arguments, int idx, int64_t &value, bool optional=false)
Get integer value from given variable.
Definition: MiniScript.h:1083
static const bool isOperatorChar(char c)
Returns if char is operator char.
Definition: MiniScript.h:880
static STATIC_DLL_IMPEXT string OPERATOR_CHARS
Definition: MiniScript.h:825
static bool hasType(const vector< ScriptVariable > &arguments, ScriptVariableType type)
Check if arguments contain argument with given type.
Definition: MiniScript.h:1029
virtual ~MiniScript()
Destructor.
Definition: MiniScript.cpp:69
bool getNextStatementOperator(const string &statement, ScriptStatementOperator &nextOperator)
Determine next not substituted operator in statement.
Definition: MiniScript.cpp:872
bool transpileScriptCondition(string &generatedCode, int scriptIdx, const unordered_map< string, vector< string > > &methodCodeMap, const string &returnValue, const string &injectCode, int depth=0)
Transpile a script condition.
void stopScriptExecutation()
Stop script execution.
Definition: MiniScript.h:789
void setNative(bool native)
Set native.
Definition: MiniScript.h:728
void setNativeScripts(const vector< Script > &nativeScripts)
Set native scripts.
Definition: MiniScript.h:759
static bool getTransformationsValue(const array< ScriptVariable, SIZE > &arguments, int idx, Transformations &value, bool optional=false)
Get transformations value from given variable.
Definition: MiniScript.h:1214
const string getInformation()
Get miniscript instance information.
static bool getVector3Value(const array< ScriptVariable, SIZE > &arguments, int idx, Vector3 &value, bool optional=false)
Get vector3 value from given variable.
Definition: MiniScript.h:1185
const string getArgumentsAsString(const vector< string_view > &arguments)
Returns arguments as string placed in a vector of string_views.
Definition: MiniScript.h:846
MiniScript()
Default constructor.
Definition: MiniScript.cpp:65
const vector< Script > & getScripts()
Definition: MiniScript.h:962
const string doStatementPreProcessing(const string &statement)
Do statement pre processing, 1) replace operators with corresponding methods.
static bool getTransformationsValue(const vector< ScriptVariable > &arguments, int idx, Transformations &value, bool optional=false)
Get transformations value from given variable.
Definition: MiniScript.h:1199
const ScriptVariable getVariable(const string &name)
Returns variable with given name.
Definition: MiniScript.h:1237
void startErrorScript()
Start error script.
Definition: MiniScript.h:976
const string trimArgument(const string &argument)
Trim argument and remove unnessessary parenthesis.
Definition: MiniScript.cpp:929
void unsetVariable(const string &name)
Unset script variable.
Definition: MiniScript.h:1275
void executeStateMachine()
Execute state machine.
Definition: MiniScript.cpp:416
static bool getBooleanValue(const array< ScriptVariable, SIZE > &arguments, int idx, bool &value, bool optional=false)
Get boolean value from given variable.
Definition: MiniScript.h:1069
void registerMethod(ScriptMethod *method)
Register script method.
Definition: MiniScript.cpp:87
vector< Script > nativeScripts
Definition: MiniScript.h:709
const string findRightArgument(const string statement, int position, int &length)
Find right argument in statement beginning from position.
Definition: MiniScript.cpp:937
static bool getStringValue(const array< ScriptVariable, SIZE > &arguments, int idx, string &value, bool optional=false)
Get string value from given variable.
Definition: MiniScript.h:1156
bool parseScriptStatement(const string_view &statement, string_view &method, vector< string_view > &arguments)
Parse a script statement.
Definition: MiniScript.cpp:120
virtual int determineNamedScriptIdxToStart()
Determine named script index to start.
Definition: MiniScript.cpp:816
void gotoStatementGoto(const ScriptStatement &statement)
Goto statement from given statements goto statement.
Definition: MiniScript.h:744
static bool getStringValue(const vector< ScriptVariable > &arguments, int idx, string &value, bool optional=false)
Get string value from given variable.
Definition: MiniScript.h:1141
virtual int determineScriptIdxToStart()
Determine script index to start.
Definition: MiniScript.cpp:756
void setScriptState(int state)
Set script state machine state.
Definition: MiniScript.h:802
unordered_map< uint8_t, ScriptMethod * > scriptOperators
Definition: MiniScript.h:830
void executeScriptLine()
Execute a single script line.
Definition: MiniScript.cpp:96
String tools class.
Definition: StringTools.h:20
static const bool startsWith(const string &src, const string &prefix)
Checks if string starts with prefix.
Definition: StringTools.h:29
static const string toLowerCase(const string &src)
Transform string to lower case.
Definition: StringTools.cpp:88
Time utility class.
Definition: Time.h:21
static int64_t getCurrentMillis()
Retrieve current time in milliseconds.
Definition: Time.h:28
unordered_map< string, ScriptVariable * > variables
Definition: MiniScript.h:696
unordered_map< int, int64_t > forTimeStarted
Definition: MiniScript.h:697
vector< ScriptStatement > statements
Definition: MiniScript.h:592
#define STATIC_DLL_IMPEXT
Definition: tdme.h:11