| 1 |
/* |
| 2 |
* TinyJS |
| 3 |
* |
| 4 |
* A single-file Javascript-alike engine |
| 5 |
* |
| 6 |
* Authored By Gordon Williams <gw@pur3.co.uk> |
| 7 |
* |
| 8 |
* Copyright (C) 2009 Pur3 Ltd |
| 9 |
* |
| 10 |
|
| 11 |
* 42TinyJS |
| 12 |
* |
| 13 |
* A fork of TinyJS with the goal to makes a more JavaScript/ECMA compliant engine |
| 14 |
* |
| 15 |
* Authored / Changed By Armin Diedering <armin@diedering.de> |
| 16 |
* |
| 17 |
* Copyright (C) 2010-2013 ardisoft |
| 18 |
* |
| 19 |
* |
| 20 |
* Permission is hereby granted, free of charge, to any person obtaining a copy of |
| 21 |
* this software and associated documentation files (the "Software"), to deal in |
| 22 |
* the Software without restriction, including without limitation the rights to |
| 23 |
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies |
| 24 |
* of the Software, and to permit persons to whom the Software is furnished to do |
| 25 |
* so, subject to the following conditions: |
| 26 |
|
| 27 |
* The above copyright notice and this permission notice shall be included in all |
| 28 |
* copies or substantial portions of the Software. |
| 29 |
|
| 30 |
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| 31 |
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| 32 |
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
| 33 |
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| 34 |
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
| 35 |
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
| 36 |
* SOFTWARE. |
| 37 |
*/ |
| 38 |
|
| 39 |
|
| 40 |
#include <math.h> |
| 41 |
#include <cstdlib> |
| 42 |
#include <sstream> |
| 43 |
#include <time.h> |
| 44 |
#include "TinyJS.h" |
| 45 |
|
| 46 |
using namespace std; |
| 47 |
// ----------------------------------------------- Actual Functions |
| 48 |
|
| 49 |
static void scTrace(const CFunctionsScopePtr &c, void * userdata) { |
| 50 |
CTinyJS *js = (CTinyJS*)userdata; |
| 51 |
if(c->getArgumentsLength()) |
| 52 |
c->getArgument(0)->trace(); |
| 53 |
else |
| 54 |
js->getRoot()->trace("root"); |
| 55 |
} |
| 56 |
|
| 57 |
static void scObjectDump(const CFunctionsScopePtr &c, void *) { |
| 58 |
c->getArgument("this")->trace("> "); |
| 59 |
} |
| 60 |
|
| 61 |
static void scObjectClone(const CFunctionsScopePtr &c, void *) { |
| 62 |
CScriptVarPtr obj = c->getArgument("this"); |
| 63 |
c->setReturnVar(obj->clone()); |
| 64 |
} |
| 65 |
|
| 66 |
static void scIntegerValueOf(const CFunctionsScopePtr &c, void *) { |
| 67 |
string str = c->getArgument("str")->toString(); |
| 68 |
|
| 69 |
int val = 0; |
| 70 |
if (str.length()==1) |
| 71 |
val = str.operator[](0); |
| 72 |
c->setReturnVar(c->newScriptVar(val)); |
| 73 |
} |
| 74 |
|
| 75 |
static void scJSONStringify(const CFunctionsScopePtr &c, void *) { |
| 76 |
uint32_t UniqueID = c->getContext()->getUniqueID(); |
| 77 |
bool hasRecursion=false; |
| 78 |
c->setReturnVar(c->newScriptVar(c->getArgument("obj")->getParsableString("", " ", UniqueID, hasRecursion))); |
| 79 |
if(hasRecursion) c->throwError(TypeError, "cyclic object value"); |
| 80 |
} |
| 81 |
|
| 82 |
static void scArrayContains(const CFunctionsScopePtr &c, void *data) { |
| 83 |
CScriptVarPtr obj = c->getArgument("obj"); |
| 84 |
CScriptVarPtr arr = c->getArgument("this"); |
| 85 |
|
| 86 |
int l = arr->getArrayLength(); |
| 87 |
CScriptVarPtr equal = c->constScriptVar(Undefined); |
| 88 |
for (int i=0;i<l;i++) { |
| 89 |
equal = obj->mathsOp(arr->getArrayIndex(i), LEX_EQUAL); |
| 90 |
if(equal->toBoolean()) { |
| 91 |
c->setReturnVar(c->constScriptVar(true)); |
| 92 |
return; |
| 93 |
} |
| 94 |
} |
| 95 |
c->setReturnVar(c->constScriptVar(false)); |
| 96 |
} |
| 97 |
|
| 98 |
static void scArrayRemove(const CFunctionsScopePtr &c, void *data) { |
| 99 |
CScriptVarPtr obj = c->getArgument("obj"); |
| 100 |
CScriptVarPtr arr = c->getArgument("this"); |
| 101 |
int i; |
| 102 |
vector<int> removedIndices; |
| 103 |
|
| 104 |
int l = arr->getArrayLength(); |
| 105 |
CScriptVarPtr equal = c->constScriptVar(Undefined); |
| 106 |
for (i=0;i<l;i++) { |
| 107 |
equal = obj->mathsOp(arr->getArrayIndex(i), LEX_EQUAL); |
| 108 |
if(equal->toBoolean()) { |
| 109 |
removedIndices.push_back(i); |
| 110 |
} |
| 111 |
} |
| 112 |
if(removedIndices.size()) { |
| 113 |
vector<int>::iterator remove_it = removedIndices.begin(); |
| 114 |
int next_remove = *remove_it; |
| 115 |
int next_insert = *remove_it++; |
| 116 |
for (i=next_remove;i<l;i++) { |
| 117 |
|
| 118 |
CScriptVarLinkPtr link = arr->findChild(int2string(i)); |
| 119 |
if(i == next_remove) { |
| 120 |
if(link) arr->removeLink(link); |
| 121 |
if(remove_it != removedIndices.end()) |
| 122 |
next_remove = *remove_it++; |
| 123 |
} else { |
| 124 |
if(link) { |
| 125 |
arr->setArrayIndex(next_insert++, link); |
| 126 |
arr->removeLink(link); |
| 127 |
} |
| 128 |
} |
| 129 |
} |
| 130 |
} |
| 131 |
} |
| 132 |
|
| 133 |
static void scArrayJoin(const CFunctionsScopePtr &c, void *data) { |
| 134 |
string sep = c->getArgument("separator")->toString(); |
| 135 |
CScriptVarPtr arr = c->getArgument("this"); |
| 136 |
|
| 137 |
ostringstream sstr; |
| 138 |
int l = arr->getArrayLength(); |
| 139 |
for (int i=0;i<l;i++) { |
| 140 |
if (i>0) sstr << sep; |
| 141 |
sstr << arr->getArrayIndex(i)->toString(); |
| 142 |
} |
| 143 |
|
| 144 |
c->setReturnVar(c->newScriptVar(sstr.str())); |
| 145 |
} |
| 146 |
|
| 147 |
// ----------------------------------------------- Register Functions |
| 148 |
void registerFunctions(CTinyJS *tinyJS) { |
| 149 |
} |
| 150 |
extern "C" void _registerFunctions(CTinyJS *tinyJS) { |
| 151 |
tinyJS->addNative("function trace()", scTrace, tinyJS, SCRIPTVARLINK_BUILDINDEFAULT); |
| 152 |
tinyJS->addNative("function Object.prototype.dump()", scObjectDump, 0, SCRIPTVARLINK_BUILDINDEFAULT); |
| 153 |
tinyJS->addNative("function Object.prototype.clone()", scObjectClone, 0, SCRIPTVARLINK_BUILDINDEFAULT); |
| 154 |
|
| 155 |
tinyJS->addNative("function Integer.valueOf(str)", scIntegerValueOf, 0, SCRIPTVARLINK_BUILDINDEFAULT); // value of a single character |
| 156 |
tinyJS->addNative("function JSON.stringify(obj, replacer)", scJSONStringify, 0, SCRIPTVARLINK_BUILDINDEFAULT); // convert to JSON. replacer is ignored at the moment |
| 157 |
tinyJS->addNative("function Array.prototype.contains(obj)", scArrayContains, 0, SCRIPTVARLINK_BUILDINDEFAULT); |
| 158 |
tinyJS->addNative("function Array.prototype.remove(obj)", scArrayRemove, 0, SCRIPTVARLINK_BUILDINDEFAULT); |
| 159 |
tinyJS->addNative("function Array.prototype.join(separator)", scArrayJoin, 0, SCRIPTVARLINK_BUILDINDEFAULT); |
| 160 |
} |
| 161 |
|