FunctionRegistry.java 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567
  1. package me.hammerle.snuviscript.code;
  2. import me.hammerle.snuviscript.inputprovider.InputProvider;
  3. import java.io.File;
  4. import java.nio.charset.StandardCharsets;
  5. import java.lang.reflect.Array;
  6. import java.nio.file.Files;
  7. import java.nio.file.Paths;
  8. import java.time.ZonedDateTime;
  9. import java.util.List;
  10. import java.util.ArrayList;
  11. import java.util.Arrays;
  12. import java.util.Collections;
  13. import java.util.Calendar;
  14. import java.util.GregorianCalendar;
  15. import java.util.HashMap;
  16. import java.util.Map;
  17. import java.util.HashSet;
  18. import java.util.Iterator;
  19. import java.util.Objects;
  20. import java.util.Set;
  21. import java.util.stream.Collectors;
  22. import me.hammerle.snuviscript.config.SnuviConfig;
  23. import me.hammerle.snuviscript.inputprovider.Variable;
  24. public class FunctionRegistry {
  25. private static final HashMap<String, Object> GLOBAL_VARS = new HashMap<>();
  26. private static final HashMap<String, NamedFunction> FUNCTIONS = new HashMap<>();
  27. protected static void registerFunction(String name, ExceptionBiFunction<Script, InputProvider[], Object> f) {
  28. FUNCTIONS.put(name, new NamedFunction(name, f));
  29. }
  30. protected static void registerConsumer(String name, ExceptionBiConsumer<Script, InputProvider[]> f) {
  31. FUNCTIONS.put(name, new NamedFunction(name, (sc, in) -> {
  32. f.apply(sc, in);
  33. return Void.TYPE;
  34. }));
  35. }
  36. protected static void registerAlias(String original, String alias) {
  37. FUNCTIONS.put(alias, FUNCTIONS.get(original));
  38. }
  39. public static NamedFunction getFunction(String f) {
  40. final String function = f.toLowerCase();
  41. return FUNCTIONS.getOrDefault(function, new NamedFunction(function, (sc, in) -> {
  42. sc.handleFunction(function, in);
  43. return Void.TYPE;
  44. }));
  45. }
  46. static {
  47. registerFunction("nothing", (sc, in) -> Void.TYPE);
  48. registerConsumer("error", (sc, in) -> sc.setStackTrace(in[0].getBoolean(sc)));
  49. registerConsumer("event.load", (sc, in) -> {
  50. String event = in[0].getString(sc);
  51. sc.loadEvent(event);
  52. sc.getScriptManager().loadEvent(event, sc);
  53. });
  54. registerConsumer("event.unload", (sc, in) -> {
  55. String event = in[0].getString(sc);
  56. sc.unloadEvent(in[0].getString(sc));
  57. sc.getScriptManager().unloadEvent(event, sc);
  58. });
  59. registerFunction("event.isloaded", (sc, in) -> sc.isEventLoaded(in[0].getString(sc)));
  60. registerFunction("script.get", (sc, in) -> {
  61. if(in.length == 0) {
  62. return sc;
  63. }
  64. String name = in[0].getString(sc);
  65. for(Script script : sc.getScriptManager().getScripts()) {
  66. if(script.getName().equals(name)) {
  67. return script;
  68. }
  69. }
  70. return null;
  71. });
  72. registerFunction("script.getfromid", (sc, in) -> sc.getScriptManager().getScript(in[0].getInt(sc)));
  73. registerFunction("script.getid", (sc, in) -> (double) ((Script) in[0].get(sc)).getId());
  74. registerFunction("script.getvar", (sc, in) -> {
  75. Script other = (Script) in[0].get(sc);
  76. Variable v = other.getVar(in[1].getString(sc));
  77. if(v == null) {
  78. return null;
  79. }
  80. return v.get(other);
  81. });
  82. registerConsumer("script.setvar", (sc, in) -> {
  83. Script other = (Script) in[0].get(sc);
  84. other.getVar(in[1].getString(sc)).set(other, in[2].get(sc));
  85. });
  86. registerFunction("script.getall", (sc, in) -> {
  87. String name = in[0].getString(sc);
  88. return sc.getScriptManager().getScripts().stream()
  89. .filter(script -> script.getName().equals(name))
  90. .collect(Collectors.toList());
  91. });
  92. registerConsumer("script.term", (sc, in) -> {
  93. Script other = (Script) in[0].get(sc);
  94. other.term();
  95. sc.getScriptManager().removeScript(other);
  96. });
  97. registerFunction(">>", (sc, in) -> (double) (in[0].getInt(sc) >> in[1].getInt(sc)));
  98. registerFunction("<<", (sc, in) -> (double) (in[0].getInt(sc) << in[1].getInt(sc)));
  99. registerFunction("&", (sc, in) -> (double) (in[0].getInt(sc) & in[1].getInt(sc)));
  100. registerFunction("|", (sc, in) -> (double) (in[0].getInt(sc) | in[1].getInt(sc)));
  101. registerFunction("^", (sc, in) -> (double) (in[0].getInt(sc) ^ in[1].getInt(sc)));
  102. registerFunction("~", (sc, in) -> (double) (~in[0].getInt(sc)));
  103. registerFunction("bit.set", (sc, in) -> (double) (in[0].getInt(sc) | (1 << (in[1].getInt(sc)))));
  104. registerFunction("bit.unset", (sc, in) -> (double) (in[0].getInt(sc) & (~(1 << (in[1].getInt(sc))))));
  105. registerFunction("bit.get", (sc, in) -> (in[0].getInt(sc) & (1 << (in[1].getInt(sc)))) != 0);
  106. registerFunction("%", (sc, in) -> (double) (in[0].getInt(sc) % in[1].getInt(sc)));
  107. registerAlias("%", "math.mod");
  108. registerFunction("math.abs", (sc, in) -> Math.abs(in[0].getDouble(sc)));
  109. registerFunction("math.pow", (sc, in) -> Math.pow(in[0].getDouble(sc), in[1].getDouble(sc)));
  110. registerFunction("math.root", (sc, in) -> Math.pow(in[0].getDouble(sc), 1.0 / in[1].getDouble(sc)));
  111. registerFunction("math.sqrt", (sc, in) -> Math.sqrt(in[0].getDouble(sc)));
  112. registerFunction("math.hypot", (sc, in) -> Math.hypot(in[0].getDouble(sc), in[1].getDouble(sc)));
  113. registerFunction("math.sin", (sc, in) -> Math.sin(in[0].getDouble(sc)));
  114. registerFunction("math.cos", (sc, in) -> Math.cos(in[0].getDouble(sc)));
  115. registerFunction("math.tan", (sc, in) -> Math.tan(in[0].getDouble(sc)));
  116. registerFunction("math.asin", (sc, in) -> Math.asin(in[0].getDouble(sc)));
  117. registerFunction("math.acos", (sc, in) -> Math.acos(in[0].getDouble(sc)));
  118. registerFunction("math.atan", (sc, in) -> Math.atan(in[0].getDouble(sc)));
  119. registerFunction("math.e", (sc, in) -> Math.E);
  120. registerFunction("math.pi", (sc, in) -> Math.PI);
  121. registerFunction("math.ln", (sc, in) -> Math.log(in[0].getDouble(sc)));
  122. registerFunction("math.log", (sc, in) -> Math.log10(in[0].getDouble(sc)));
  123. registerFunction("math.random", (sc, in) -> (double) SnuviUtils.randomInt(in[0].getInt(sc), in[1].getInt(sc)));
  124. registerFunction("math.round", (sc, in) -> (double) Math.round(in[0].getDouble(sc)));
  125. registerFunction("math.rounddown", (sc, in) -> Math.floor(in[0].getDouble(sc)));
  126. registerFunction("math.roundup", (sc, in) -> Math.ceil(in[0].getDouble(sc)));
  127. registerFunction("math.roundcomma", (sc, in) -> {
  128. double d = in[0].getDouble(sc);
  129. int factor = (int) Math.pow(10, in[1].getInt(sc));
  130. return (double) (((double) Math.round(d * factor)) / factor);
  131. });
  132. registerFunction("math.min", (sc, in) -> Math.min(in[0].getDouble(sc), in[1].getDouble(sc)));
  133. registerFunction("math.max", (sc, in) -> Math.max(in[0].getDouble(sc), in[1].getDouble(sc)));
  134. registerFunction("matrix.new", (sc, in) -> new Matrix());
  135. registerFunction("matrix.newrotationy", (sc, in) -> Matrix.getRotationY(in[0].getDouble(sc)));
  136. registerFunction("matrix.newrotationx", (sc, in) -> Matrix.getRotationX(in[0].getDouble(sc)));
  137. registerFunction("matrix.mul", (sc, in) -> ((Matrix) in[0].get(sc)).mul((Matrix) in[1].get(sc)));
  138. registerConsumer("matrix.mulvector", (sc, in) -> ((Matrix) in[0].get(sc)).mul((Vector) in[1].get(sc)));
  139. registerFunction("vector.new", (sc, in) -> new Vector(in[0].getDouble(sc), in[1].getDouble(sc), in[2].getDouble(sc)));
  140. registerConsumer("vector.set", (sc, in) -> ((Vector) in[0].get(sc)).set(in[1].getDouble(sc), in[2].getDouble(sc), in[3].getDouble(sc)));
  141. registerFunction("vector.getx", (sc, in) -> ((Vector) in[0].get(sc)).getX());
  142. registerFunction("vector.gety", (sc, in) -> ((Vector) in[0].get(sc)).getY());
  143. registerFunction("vector.getz", (sc, in) -> ((Vector) in[0].get(sc)).getZ());
  144. registerFunction("list.new", (sc, in) -> {
  145. if(in.length == 0) {
  146. return new ArrayList<>();
  147. }
  148. in[0].set(sc, new ArrayList<>());
  149. return Void.TYPE;
  150. });
  151. registerFunction("list.exists", (sc, in) -> in[0].get(sc) instanceof List);
  152. registerFunction("list.add", (sc, in) -> ((List) in[0].get(sc)).add(in[1].get(sc)));
  153. registerConsumer("list.addall", (sc, in) -> {
  154. List list = ((List) in[0].get(sc));
  155. for(int i = 1; i < in.length; i++) {
  156. list.add(in[i].get(sc));
  157. }
  158. });
  159. registerFunction("list.remove", (sc, in) -> ((List) in[0].get(sc)).remove(in[1].get(sc)));
  160. registerFunction("list.removeindex", (sc, in) -> ((List) in[0].get(sc)).remove(in[1].getInt(sc)));
  161. registerFunction("list.contains", (sc, in) -> ((List) in[0].get(sc)).contains(in[1].get(sc)));
  162. registerFunction("list.getsize", (sc, in) -> (double) ((List) in[0].get(sc)).size());
  163. registerFunction("list.getindex", (sc, in) -> ((List) in[0].get(sc)).get(in[1].getInt(sc)));
  164. registerAlias("list.getindex", "list.get");
  165. registerFunction("list.setindex", (sc, in) -> ((List) in[0].get(sc)).set(in[1].getInt(sc), in[2].get(sc)));
  166. registerConsumer("list.clear", (sc, in) -> ((List) in[0].get(sc)).clear());
  167. registerFunction("list.getindexof", (sc, in) -> (double) ((List) in[0].get(sc)).indexOf(in[1].get(sc)));
  168. registerConsumer("list.sort", (sc, in) -> {
  169. Collections.sort(((List<Object>) in[0].get(sc)), (o1, o2) -> ((Comparable) o1).compareTo(o2));
  170. });
  171. registerConsumer("list.reverse", (sc, in) -> Collections.reverse((List<Object>) in[0].get(sc)));
  172. registerConsumer("list.shuffle", (sc, in) -> Collections.shuffle((List<Object>) in[0].get(sc)));
  173. registerFunction("list.iterator", (sc, in) -> ((List) in[0].get(sc)).iterator());
  174. registerFunction("array.new", (sc, in) -> {
  175. if(in.length == 0) {
  176. throw new ArrayIndexOutOfBoundsException("missing array dimension");
  177. }
  178. int[] dim = new int[in.length];
  179. for(int i = 0; i < in.length; i++) {
  180. dim[i] = in[i].getInt(sc);
  181. }
  182. return Array.newInstance(Object.class, dim);
  183. });
  184. registerFunction("array.getsize", (sc, in) -> (double) Array.getLength(in[0].get(sc)));
  185. registerAlias("array.getsize", "array.length");
  186. registerFunction("map.new", (sc, in) -> {
  187. if(in.length == 0) {
  188. return new HashMap<>();
  189. }
  190. in[0].set(sc, new HashMap<>());
  191. return Void.TYPE;
  192. });
  193. registerFunction("map.exists", (sc, in) -> in[0].get(sc) instanceof Map);
  194. registerFunction("map.add", (sc, in) -> ((Map) in[0].get(sc)).put(in[1].get(sc), in[2].get(sc)));
  195. registerFunction("map.remove", (sc, in) -> ((Map) in[0].get(sc)).remove(in[1].get(sc)));
  196. registerFunction("map.contains", (sc, in) -> ((Map) in[0].get(sc)).containsKey(in[1].get(sc)));
  197. registerFunction("map.getsize", (sc, in) -> (double) ((Map) in[0].get(sc)).size());
  198. registerFunction("map.get", (sc, in) -> ((Map) in[0].get(sc)).get(in[1].get(sc)));
  199. registerFunction("map.getordefault", (sc, in) -> ((Map) in[0].get(sc)).getOrDefault(in[1].get(sc), in[2].get(sc)));
  200. registerConsumer("map.clear", (sc, in) -> ((Map) in[0].get(sc)).clear());
  201. registerFunction("map.iterator", (sc, in) -> ((Map) in[0].get(sc)).entrySet().iterator());
  202. registerFunction("map.getkey", (sc, in) -> ((Map.Entry) in[0].get(sc)).getKey());
  203. registerFunction("map.getvalue", (sc, in) -> ((Map.Entry) in[0].get(sc)).getValue());
  204. registerFunction("map.setvalue", (sc, in) -> ((Map.Entry) in[0].get(sc)).setValue(in[1].get(sc)));
  205. registerFunction("set.new", (sc, in) -> {
  206. if(in.length == 0) {
  207. return new HashSet<>();
  208. }
  209. in[0].set(sc, new HashSet<>());
  210. return Void.TYPE;
  211. });
  212. registerFunction("set.exists", (sc, in) -> in[0].get(sc) instanceof Set);
  213. registerFunction("set.add", (sc, in) -> ((Set) in[0].get(sc)).add(in[1].get(sc)));
  214. registerConsumer("set.addall", (sc, in) -> {
  215. Set set = ((Set) in[0].get(sc));
  216. for(int i = 1; i < in.length; i++) {
  217. set.add(in[i].get(sc));
  218. }
  219. });
  220. registerFunction("set.remove", (sc, in) -> ((Set) in[0].get(sc)).remove(in[1].get(sc)));
  221. registerFunction("set.contains", (sc, in) -> ((Set) in[0].get(sc)).contains(in[1].get(sc)));
  222. registerFunction("set.getsize", (sc, in) -> (double) ((Set) in[0].get(sc)).size());
  223. registerConsumer("set.clear", (sc, in) -> ((Set) in[0].get(sc)).clear());
  224. registerFunction("set.iterator", (sc, in) -> ((Set) in[0].get(sc)).iterator());
  225. registerFunction("time.new", (sc, in) -> {
  226. if(in.length <= 1) {
  227. GregorianCalendar cal = GregorianCalendar.from(ZonedDateTime.now());
  228. cal.setTimeInMillis(in[0].getLong(sc));
  229. return cal;
  230. } else {
  231. GregorianCalendar cal = GregorianCalendar.from(ZonedDateTime.now());
  232. cal.setTimeInMillis(in[1].getLong(sc));
  233. in[0].set(sc, cal);
  234. return Void.TYPE;
  235. }
  236. });
  237. registerFunction("time.getmillis", (sc, in) -> (double) System.currentTimeMillis());
  238. registerFunction("time.getnanos", (sc, in) -> (double) System.nanoTime());
  239. registerFunction("time.from", (sc, in) -> (double) ((GregorianCalendar) in[0].get(sc)).getTimeInMillis());
  240. registerConsumer("time.nextday", (sc, in) -> {
  241. GregorianCalendar cal = (GregorianCalendar) in[0].get(sc);
  242. cal.add(Calendar.DAY_OF_YEAR, 1);
  243. cal.set(Calendar.HOUR_OF_DAY, 0);
  244. cal.set(Calendar.SECOND, 0);
  245. cal.set(Calendar.MINUTE, 0);
  246. cal.set(Calendar.MILLISECOND, 0);
  247. });
  248. registerFunction("time.getyear", (sc, in) -> (double) ((GregorianCalendar) in[0].get(sc)).get(Calendar.YEAR));
  249. registerFunction("time.getmonth", (sc, in) -> (double) (((GregorianCalendar) in[0].get(sc)).get(Calendar.MONTH) + 1));
  250. registerFunction("time.getday", (sc, in) -> (double) (((GregorianCalendar) in[0].get(sc)).get(Calendar.DAY_OF_MONTH)));
  251. registerFunction("time.gethour", (sc, in) -> (double) ((GregorianCalendar) in[0].get(sc)).get(Calendar.HOUR_OF_DAY));
  252. registerFunction("time.getminute", (sc, in) -> (double) ((GregorianCalendar) in[0].get(sc)).get(Calendar.MINUTE));
  253. registerFunction("time.getsecond", (sc, in) -> (double) ((GregorianCalendar) in[0].get(sc)).get(Calendar.SECOND));
  254. registerFunction("text.matches", (sc, in) -> in[0].getString(sc).matches(in[1].getString(sc)));
  255. registerFunction("text.number", (sc, in) -> SnuviUtils.toString(in[0].getDouble(sc)));
  256. registerFunction("text.class", (sc, in) -> in[0].get(sc).getClass().getSimpleName());
  257. registerFunction("text.tolowercase", (sc, in) -> SnuviUtils.connect(sc, in, 0).toLowerCase());
  258. registerAlias("text.tolowercase", "tolowercase");
  259. registerFunction("text.touppercase", (sc, in) -> SnuviUtils.connect(sc, in, 0).toUpperCase());
  260. registerAlias("text.touppercase", "touppercase");
  261. registerFunction("text.split", (sc, in) -> {
  262. if(in.length <= 2) {
  263. String[] parts = in[1].getString(sc).split(in[0].getString(sc));
  264. ArrayList<Object> list = new ArrayList<>();
  265. for(String part : parts) {
  266. list.add(SnuviUtils.convert(part));
  267. }
  268. return list;
  269. }
  270. String[] parts = in[2].getString(sc).split(in[1].getString(sc));
  271. ArrayList<Object> list = new ArrayList<>();
  272. for(String part : parts) {
  273. list.add(SnuviUtils.convert(part));
  274. }
  275. in[0].set(sc, list);
  276. return Void.TYPE;
  277. });
  278. registerAlias("text.split", "split");
  279. registerFunction("text.concatlist", (sc, in) -> {
  280. StringBuilder sb = new StringBuilder();
  281. List<Object> list = (List<Object>) in[0].get(sc);
  282. String splitter = in[1].getString(sc);
  283. Iterator<Object> iter = list.iterator();
  284. int from = in[2].getInt(sc);
  285. int to = Math.min(in[3].getInt(sc), list.size() - 1);
  286. to -= from;
  287. while(iter.hasNext() && from > 0) {
  288. iter.next();
  289. from--;
  290. }
  291. while(iter.hasNext() && to > 0) {
  292. sb.append(iter.next());
  293. sb.append(splitter);
  294. to--;
  295. }
  296. if(iter.hasNext() && to == 0) {
  297. sb.append(iter.next());
  298. }
  299. return sb.toString();
  300. });
  301. registerAlias("text.concatlist", "concatlist");
  302. registerFunction("text.concat", (sc, in) -> SnuviUtils.connect(sc, in, 0));
  303. registerAlias("text.concat", "concat");
  304. registerFunction("text.concatspace", (sc, in) -> SnuviUtils.connect(sc, in, " ", 0));
  305. registerFunction("text", (sc, in) -> String.valueOf(in[0].get(sc)));
  306. registerFunction("text.substring", (sc, in) -> in[0].getString(sc).substring(in[1].getInt(sc), in[2].getInt(sc)));
  307. registerFunction("text.length", (sc, in) -> (double) in[0].getString(sc).length());
  308. registerFunction("text.startswith", (sc, in) -> in[0].getString(sc).startsWith(in[1].getString(sc), in[2].getInt(sc)));
  309. registerFunction("text.endswith", (sc, in) -> in[0].getString(sc).endsWith(in[1].getString(sc)));
  310. registerFunction("text.contains", (sc, in) -> in[0].getString(sc).contains(in[1].getString(sc)));
  311. registerFunction("text.indexof", (sc, in) -> (double) in[0].getString(sc).indexOf(in[1].getString(sc), in[2].getInt(sc)));
  312. registerFunction("text.lastindexof", (sc, in) -> (double) in[0].getString(sc).lastIndexOf(in[1].getString(sc), in[2].getInt(sc)));
  313. registerFunction("text.replace", (sc, in) -> in[0].getString(sc).replace(in[1].getString(sc), in[2].getString(sc)));
  314. registerFunction("text.trim", (sc, in) -> in[0].getString(sc).trim());
  315. registerFunction("text.charat", (sc, in) -> String.valueOf(in[0].getString(sc).charAt(in[1].getInt(sc))));
  316. registerFunction("text.charcode", (sc, in) -> (double) in[0].getString(sc).charAt(in[1].getInt(sc)));
  317. registerFunction("text.fromcode", (sc, in) -> String.valueOf((char) in[0].getInt(sc)));
  318. registerFunction("text.onlyletters", (sc, in) -> {
  319. for(char c : in[0].getString(sc).toCharArray()) {
  320. if(!Character.isLetter(c)) {
  321. return false;
  322. }
  323. }
  324. return true;
  325. });
  326. registerFunction("file.new", (sc, in) -> new File(in[0].getString(sc)));
  327. registerFunction("file.exists", (sc, in) -> ((File) in[0].get(sc)).exists());
  328. registerFunction("file.isfile", (sc, in) -> ((File) in[0].get(sc)).isFile());
  329. registerFunction("file.isdirectory", (sc, in) -> ((File) in[0].get(sc)).isDirectory());
  330. registerFunction("file.delete", (sc, in) -> ((File) in[0].get(sc)).delete());
  331. registerFunction("file.getname", (sc, in) -> ((File) in[0].get(sc)).getName());
  332. registerFunction("file.getlist", (sc, in) -> Arrays.asList(((File) in[0].get(sc)).listFiles()));
  333. registerFunction("file.read", (sc, in) -> Files.readAllLines(((File) in[0].get(sc)).toPath()));
  334. registerConsumer("file.write", (sc, in) -> {
  335. File f = (File) in[0].get(sc);
  336. if(f.getParentFile() != null) {
  337. f.getParentFile().mkdirs();
  338. }
  339. if(!f.exists()) {
  340. f.createNewFile();
  341. }
  342. Files.write(Paths.get(f.toURI()), ((List<Object>) in[1].get(sc))
  343. .stream().map(o -> String.valueOf(o)).collect(Collectors.toList()), StandardCharsets.UTF_8);
  344. });
  345. registerFunction("config.new", (sc, in) -> new SnuviConfig(in[0].getString(sc), in[1].getString(sc)));
  346. registerFunction("config.exists", (sc, in) -> ((SnuviConfig) in[0].get(sc)).exists());
  347. registerFunction("config.save", (sc, in) -> ((SnuviConfig) in[0].get(sc)).save(sc));
  348. registerConsumer("config.load", (sc, in) -> ((SnuviConfig) in[0].get(sc)).load(sc));
  349. registerFunction("config.delete", (sc, in) -> ((SnuviConfig) in[0].get(sc)).delete());
  350. registerConsumer("config.set", (sc, in) -> ((SnuviConfig) in[0].get(sc)).set(in[1].getString(sc), in[2].get(sc)));
  351. registerFunction("config.getbool", (sc, in) -> ((SnuviConfig) in[0].get(sc)).getBoolean(sc, in[1].getString(sc), in[2].getBoolean(sc)));
  352. registerFunction("config.getdouble", (sc, in) -> ((SnuviConfig) in[0].get(sc)).getDouble(sc, in[1].getString(sc), in[2].getDouble(sc)));
  353. registerFunction("config.getstring", (sc, in) -> ((SnuviConfig) in[0].get(sc)).getString(sc, in[1].getString(sc), in[2].getString(sc)));
  354. registerFunction("read.number", (sc, in) -> Double.parseDouble(in[0].getString(sc)));
  355. registerFunction("+", (sc, in) -> in[0].getDouble(sc) + in[1].getDouble(sc));
  356. registerAlias("+", "add");
  357. registerFunction("-", (sc, in) -> in.length == 1 ? -in[0].getDouble(sc) : in[0].getDouble(sc) - in[1].getDouble(sc));
  358. registerAlias("-", "sub");
  359. registerFunction("*", (sc, in) -> in[0].getDouble(sc) * in[1].getDouble(sc));
  360. registerAlias("*", "mul");
  361. registerFunction("/", (sc, in) -> in[0].getDouble(sc) / in[1].getDouble(sc));
  362. registerAlias("/", "div");
  363. registerFunction("=", (sc, in) -> {
  364. Object o = in[1].get(sc);
  365. in[0].set(sc, o);
  366. return o;
  367. });
  368. registerFunction("+=", (sc, in) -> {
  369. Object o = in[0].getDouble(sc) + in[1].getDouble(sc);
  370. in[0].set(sc, o);
  371. return o;
  372. });
  373. registerFunction("p++", (sc, in) -> {
  374. double d = in[0].getDouble(sc);
  375. in[0].set(sc, d + 1.0);
  376. return d;
  377. });
  378. registerAlias("p++", "inc");
  379. registerFunction("++", (sc, in) -> {
  380. double d = in[0].getDouble(sc) + 1.0;
  381. in[0].set(sc, d);
  382. return d;
  383. });
  384. registerFunction("-=", (sc, in) -> {
  385. Object o = in[0].getDouble(sc) - in[1].getDouble(sc);
  386. in[0].set(sc, o);
  387. return o;
  388. });
  389. registerFunction("p--", (sc, in) -> {
  390. double d = in[0].getDouble(sc);
  391. in[0].set(sc, d - 1.0);
  392. return d;
  393. });
  394. registerAlias("p--", "dec");
  395. registerFunction("--", (sc, in) -> {
  396. double d = in[0].getDouble(sc) - 1.0;
  397. in[0].set(sc, d);
  398. return d;
  399. });
  400. registerFunction("*=", (sc, in) -> {
  401. Object o = in[0].getDouble(sc) * in[1].getDouble(sc);
  402. in[0].set(sc, o);
  403. return o;
  404. });
  405. registerFunction("/=", (sc, in) -> {
  406. Object o = in[0].getDouble(sc) / in[1].getDouble(sc);
  407. in[0].set(sc, o);
  408. return o;
  409. });
  410. registerFunction("%=", (sc, in) -> {
  411. Object o = (double) (in[0].getInt(sc) % in[1].getInt(sc));
  412. in[0].set(sc, o);
  413. return o;
  414. });
  415. registerFunction("<<=", (sc, in) -> {
  416. Object o = (double) (in[0].getInt(sc) << in[1].getInt(sc));
  417. in[0].set(sc, o);
  418. return o;
  419. });
  420. registerFunction(">>=", (sc, in) -> {
  421. Object o = (double) (in[0].getInt(sc) >> in[1].getInt(sc));
  422. in[0].set(sc, o);
  423. return o;
  424. });
  425. registerFunction("&=", (sc, in) -> {
  426. Object o = (double) (in[0].getInt(sc) & in[1].getInt(sc));
  427. in[0].set(sc, o);
  428. return o;
  429. });
  430. registerFunction("^=", (sc, in) -> {
  431. Object o = (double) (in[0].getInt(sc) ^ in[1].getInt(sc));
  432. in[0].set(sc, o);
  433. return o;
  434. });
  435. registerFunction("|=", (sc, in) -> {
  436. Object o = (double) (in[0].getInt(sc) | in[1].getInt(sc));
  437. in[0].set(sc, o);
  438. return o;
  439. });
  440. registerFunction("getvar", (sc, in) -> sc.getVar(in[0].getString(sc)).get(sc));
  441. registerConsumer("setvar", (sc, in) -> sc.getVar(in[0].getString(sc)).set(sc, in[1].get(sc)));
  442. registerConsumer("removevar", (sc, in) -> sc.getVar(in[0].getString(sc)).set(sc, null));
  443. registerConsumer("wait", (sc, in) -> sc.setWaiting());
  444. registerConsumer("goto", (sc, in) -> sc.gotoLabel(in[0].getString(sc), true));
  445. registerConsumer("ignoregoto", (sc, in) -> sc.gotoLabel(in[0].getString(sc), false));
  446. registerAlias("ignoregoto", "igoto");
  447. registerConsumer("sgoto", (sc, in) -> {
  448. int time = in[0].getInt(sc);
  449. if(time < 0) {
  450. throw new IllegalArgumentException("time units can't be negative");
  451. }
  452. String label = in[1].getString(sc);
  453. sc.getScriptManager().getScheduler().scheduleTask(() -> {
  454. if(sc.shouldTerm() || sc.isHolded()) {
  455. return;
  456. }
  457. sc.gotoLabel(label, true, 1);
  458. sc.run();
  459. if(sc.shouldTerm()) {
  460. sc.getScriptManager().removeScript(sc);
  461. }
  462. }, time);
  463. });
  464. registerConsumer("gosub", (sc, in) -> sc.goSub(in[0].getString(sc)));
  465. registerFunction("==", (sc, in) -> Objects.equals(in[0].get(sc), in[1].get(sc)));
  466. registerAlias("==", "equal");
  467. registerAlias("==", "equals");
  468. registerFunction("!=", (sc, in) -> !Objects.equals(in[0].get(sc), in[1].get(sc)));
  469. registerAlias("!=", "notequal");
  470. registerFunction("<", (sc, in) -> ((Comparable) in[0].get(sc)).compareTo(in[1].get(sc)) < 0);
  471. registerAlias("<", "less");
  472. registerFunction(">", (sc, in) -> ((Comparable) in[0].get(sc)).compareTo(in[1].get(sc)) > 0);
  473. registerAlias(">", "greater");
  474. registerFunction("<=", (sc, in) -> ((Comparable) in[0].get(sc)).compareTo(in[1].get(sc)) <= 0);
  475. registerAlias("<=", "lessequal");
  476. registerFunction(">=", (sc, in) -> ((Comparable) in[0].get(sc)).compareTo(in[1].get(sc)) >= 0);
  477. registerAlias(">=", "greaterequal");
  478. registerFunction("!", (sc, in) -> !in[0].getBoolean(sc));
  479. registerAlias("!", "invert");
  480. registerFunction("&&", (sc, in) -> {
  481. for(InputProvider i : in) {
  482. if(!i.getBoolean(sc)) {
  483. return false;
  484. }
  485. }
  486. return true;
  487. });
  488. registerAlias("&&", "and");
  489. registerFunction("||", (sc, in) -> {
  490. for(InputProvider i : in) {
  491. if(i.getBoolean(sc)) {
  492. return true;
  493. }
  494. }
  495. return false;
  496. });
  497. registerAlias("||", "or");
  498. registerFunction("getscriptvar", (sc, in) -> GLOBAL_VARS.get(in[0].getString(sc)));
  499. registerFunction("setscriptvar", (sc, in) -> GLOBAL_VARS.put(in[0].getString(sc), in[1].get(sc)));
  500. registerFunction("delscriptvar", (sc, in) -> GLOBAL_VARS.remove(in[0].getString(sc)));
  501. registerConsumer("clearscriptvars", (sc, in) -> GLOBAL_VARS.clear());
  502. registerFunction("hasnext", (sc, in) -> ((Iterator) in[0].get(sc)).hasNext());
  503. registerFunction("next", (sc, in) -> ((Iterator) in[0].get(sc)).next());
  504. registerConsumer("remove", (sc, in) -> ((Iterator) in[0].get(sc)).remove());
  505. registerConsumer("swap", (sc, in) -> {
  506. Object o = in[0].get(sc);
  507. in[0].set(sc, in[1].get(sc));
  508. in[1].set(sc, o);
  509. });
  510. registerConsumer("print", (sc, in) -> {
  511. sc.getScriptManager().getLogger().print(SnuviUtils.connect(sc, in, 0), null, "print", sc.getName(), sc, sc.getStackTrace());
  512. });
  513. registerConsumer("rprint", (sc, in) -> System.out.println(SnuviUtils.connect(sc, in, 0)));
  514. registerConsumer("waitfor", (sc, in) -> {
  515. long l = in[0].getInt(sc);
  516. if(l < 0) {
  517. throw new IllegalArgumentException("time units can't be negative");
  518. }
  519. sc.setHolded(true);
  520. sc.setWaiting();
  521. sc.getScriptManager().getScheduler().scheduleTask(() -> {
  522. if(sc.shouldTerm()) {
  523. return;
  524. }
  525. sc.setHolded(false);
  526. sc.run();
  527. if(sc.shouldTerm()) {
  528. sc.getScriptManager().removeScript(sc);
  529. }
  530. }, l);
  531. });
  532. registerConsumer("term", (sc, in) -> {
  533. sc.term();
  534. sc.getScriptManager().removeScript(sc);
  535. });
  536. registerFunction("isbool", (sc, in) -> (in[0].get(sc) instanceof Boolean));
  537. registerFunction("isdouble", (sc, in) -> (in[0].get(sc) instanceof Double));
  538. registerFunction("islong", (sc, in) -> {
  539. Object o = in[0].get(sc);
  540. if(o instanceof Double) {
  541. double d = (Double) o;
  542. return d == (long) d;
  543. }
  544. return false;
  545. });
  546. registerConsumer("assert", (sc, in) -> {
  547. if(!in[0].getBoolean(sc)) {
  548. throw new IllegalArgumentException("assertion failed");
  549. }
  550. });
  551. registerFunction("class", (sc, in) -> in[0].get(sc).getClass());
  552. registerFunction("usedmemory", (sc, in) -> {
  553. Runtime runtime = Runtime.getRuntime();
  554. double usedMemory = (runtime.totalMemory() - runtime.freeMemory()) / 1048576;
  555. return usedMemory;
  556. });
  557. registerFunction("allocatedmemory", (sc, in) -> Runtime.getRuntime().totalMemory() / 1048576.0);
  558. }
  559. }