mirror of https://github.com/adamdruppe/arsd.git
more operators and stuff
This commit is contained in:
parent
eb4b47c66c
commit
c3789483cb
426
jsvar.d
426
jsvar.d
|
@ -47,6 +47,13 @@ import std.json;
|
|||
|
||||
it should consistently throw on missing semicolons
|
||||
|
||||
*) nesting comments, `` string literals
|
||||
*) properties???//
|
||||
a.prop on the rhs => a.prop()
|
||||
a.prop on the lhs => a.prop(rhs);
|
||||
if opAssign, it can just do a.prop(a.prop().opBinary!op(rhs));
|
||||
|
||||
But, how do we mark properties in var? Can we make them work this way in D too?
|
||||
0) add global functions to the object like assert()
|
||||
1) ensure operator precedence is sane
|
||||
2) a++ would prolly be nice, and def -a
|
||||
|
@ -55,10 +62,10 @@ import std.json;
|
|||
foreach(i; array) -> for(var i = 0; i < array.length; i++)
|
||||
foreach(i; object) -> for(var v = object.visit; !v.empty; v.popFront) { var i = v.front; / *...* / }
|
||||
4) switches?
|
||||
8) IMPORTANT FIXME: make sure return/break/continue/exit? work right
|
||||
15) perhaps new, which can work in the style of javascript i figure
|
||||
|
||||
6) explicit type conversions somehow
|
||||
10) __FILE__ and __LINE__ as default function arguments should work like in D
|
||||
16) stack traces on script exceptions
|
||||
17) an exception type that we can create in the script
|
||||
|
||||
14) import???????/ it could just attach a particular object to the local scope, and the module decl just giving the local scope a name
|
||||
there could be a super-global object that is the prototype of the "global" used here
|
||||
|
@ -69,9 +76,9 @@ import std.json;
|
|||
|
||||
though maybe to export vars there could be an explicit export namespace or something.
|
||||
|
||||
16) stack traces on script exceptions
|
||||
17) an exception type that we can create in the script
|
||||
5) gotos? labels?
|
||||
|
||||
6) gotos? labels? labeled break/continue?
|
||||
18) what about something like ruby's blocks or macros? parsing foo(arg) { code } is easy enough, but how would we use it?
|
||||
|
||||
try is considered a statement right now and this only works on top level surrounded by {}
|
||||
it should be usable anywhere
|
||||
|
@ -79,9 +86,10 @@ import std.json;
|
|||
var FIXME:
|
||||
|
||||
user defined operator overloading on objects, including opCall
|
||||
prototype objects for Array, String, and Function
|
||||
flesh out prototype objects for Array, String, and Function
|
||||
|
||||
opEquals and stricterOpEquals
|
||||
opDispatch overriding
|
||||
|
||||
it would be nice if delegates on native types could work
|
||||
*/
|
||||
|
@ -96,9 +104,30 @@ import std.json;
|
|||
There is no comma operator, but you can use a scope as an expression: a++, b++; can be written as {a++;b++;}
|
||||
*/
|
||||
|
||||
|
||||
version(test_script)
|
||||
struct Foop {
|
||||
int a = 12;
|
||||
string n = "hate";
|
||||
void speak() { writeln(n, " ", a); n = "love"; writeln(n, " is what it is now"); }
|
||||
void speak2() { writeln("speak2 ", n, " ", a); }
|
||||
}
|
||||
version(test_script)
|
||||
void main() {
|
||||
// the WrappedNativeObject is disgusting
|
||||
// but works.
|
||||
/*
|
||||
Foop foop2;
|
||||
|
||||
var foop;
|
||||
foop._object = new WrappedNativeObject!Foop(foop2);
|
||||
|
||||
foop.speak()();
|
||||
foop.a = 25;
|
||||
writeln(foop.n);
|
||||
foop.speak2()();
|
||||
return;
|
||||
*/
|
||||
|
||||
import arsd.script;
|
||||
struct Test {
|
||||
int a = 10;
|
||||
|
@ -111,7 +140,13 @@ void main() {
|
|||
|
||||
globals.arrtest = var.emptyArray;
|
||||
|
||||
globals.write = (var a) { writeln("script said: ", a); };
|
||||
globals.write._function = (var _this, var[] args) {
|
||||
string s;
|
||||
foreach(a; args)
|
||||
s ~= a.get!string;
|
||||
writeln("script said: ", s);
|
||||
return var(null);
|
||||
};
|
||||
|
||||
// call D defined functions in script
|
||||
globals.func = (var a, var b) { writeln("Hello, world! You are : ", a, " and ", b); };
|
||||
|
@ -266,6 +301,7 @@ private var _op(alias _this, alias this2, string op, T)(T t) if(op == "~") {
|
|||
|
||||
}
|
||||
|
||||
// FIXME: maybe the bitops should be moved out to another function like ~ is
|
||||
private var _op(alias _this, alias this2, string op, T)(T t) if(op != "~") {
|
||||
static if(is(T == var)) {
|
||||
if(t.payloadType() == var.Type.Integral)
|
||||
|
@ -284,23 +320,38 @@ private var _op(alias _this, alias this2, string op, T)(T t) if(op != "~") {
|
|||
_this._payload._integral = l;
|
||||
return _this;
|
||||
} else static if(isFloatingPoint!T) {
|
||||
this2._type = var.Type.Floating;
|
||||
real f = l;
|
||||
mixin("f "~op~"= t;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = f;
|
||||
static if(op == "&" || op == "|" || op == "^") {
|
||||
this2._type = var.Type.Integral;
|
||||
long f = l;
|
||||
mixin("f "~op~"= cast(long) t;");
|
||||
_this._type = var.Type.Integral;
|
||||
_this._payload._integral = f;
|
||||
} else {
|
||||
this2._type = var.Type.Floating;
|
||||
real f = l;
|
||||
mixin("f "~op~"= t;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = f;
|
||||
}
|
||||
return _this;
|
||||
} else static if(isSomeString!T) {
|
||||
auto rhs = stringToNumber(t);
|
||||
if(realIsInteger(rhs)) {
|
||||
mixin("l "~op~"= rhs;");
|
||||
mixin("l "~op~"= cast(long) rhs;");
|
||||
_this._type = var.Type.Integral;
|
||||
_this._payload._integral = l;
|
||||
} else{
|
||||
real f = l;
|
||||
mixin("f "~op~"= rhs;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = f;
|
||||
static if(op == "&" || op == "|" || op == "^") {
|
||||
long f = l;
|
||||
mixin("f "~op~"= cast(long) rhs;");
|
||||
_this._type = var.Type.Integral;
|
||||
_this._payload._integral = f;
|
||||
} else {
|
||||
real f = l;
|
||||
mixin("f "~op~"= rhs;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = f;
|
||||
}
|
||||
}
|
||||
return _this;
|
||||
|
||||
|
@ -309,30 +360,56 @@ private var _op(alias _this, alias this2, string op, T)(T t) if(op != "~") {
|
|||
auto f = this._payload._floating;
|
||||
|
||||
static if(isIntegral!T || isFloatingPoint!T) {
|
||||
mixin("f "~op~"= t;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = f;
|
||||
static if(op == "&" || op == "|" || op == "^") {
|
||||
long argh = cast(long) f;
|
||||
mixin("argh "~op~"= cast(long) t;");
|
||||
_this._type = var.Type.Integral;
|
||||
_this._payload._integral = argh;
|
||||
} else {
|
||||
mixin("f "~op~"= t;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = f;
|
||||
}
|
||||
return _this;
|
||||
} else static if(isSomeString!T) {
|
||||
auto rhs = stringToNumber(t);
|
||||
mixin("f "~op~"= rhs;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = f;
|
||||
|
||||
static if(op == "&" || op == "|" || op == "^") {
|
||||
long pain = cast(long) f;
|
||||
mixin("pain "~op~"= cast(long) rhs;");
|
||||
_this._type = var.Type.Integral;
|
||||
_this._payload._floating = pain;
|
||||
} else {
|
||||
mixin("f "~op~"= rhs;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = f;
|
||||
}
|
||||
return _this;
|
||||
} else assert(0);
|
||||
} else if(this2.payloadType() == var.Type.String) {
|
||||
real r = stringToNumber(this2._payload._string);
|
||||
real rhs;
|
||||
static if(op == "&" || op == "|" || op == "^") {
|
||||
long r = cast(long) stringToNumber(this2._payload._string);
|
||||
long rhs;
|
||||
} else {
|
||||
real r = stringToNumber(this2._payload._string);
|
||||
real rhs;
|
||||
}
|
||||
|
||||
static if(isSomeString!T) {
|
||||
rhs = stringToNumber(t);
|
||||
rhs = cast(typeof(rhs)) stringToNumber(t);
|
||||
} else {
|
||||
rhs = to!real(t);
|
||||
rhs = to!(typeof(rhs))(t);
|
||||
}
|
||||
|
||||
mixin("r " ~ op ~ "= rhs;");
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = r;
|
||||
|
||||
static if(is(typeof(r) == real)) {
|
||||
_this._type = var.Type.Floating;
|
||||
_this._payload._floating = r;
|
||||
} else static if(is(typeof(r) == long)) {
|
||||
_this._type = var.Type.Integral;
|
||||
_this._payload._integral = r;
|
||||
} else static assert(0);
|
||||
return _this;
|
||||
} else {
|
||||
// the operation is nonsensical, we should throw or ignore it
|
||||
|
@ -353,6 +430,28 @@ struct var {
|
|||
this.opAssign(t);
|
||||
}
|
||||
|
||||
public var _copy() {
|
||||
final switch(payloadType()) {
|
||||
case Type.Integral:
|
||||
case Type.Boolean:
|
||||
case Type.Floating:
|
||||
case Type.Function:
|
||||
case Type.String:
|
||||
// since strings are immutable, we can pretend they are value types too
|
||||
return this; // value types don't need anything special to be copied
|
||||
|
||||
case Type.Array:
|
||||
var cp;
|
||||
cp = this._payload._array[];
|
||||
return cp;
|
||||
case Type.Object:
|
||||
var cp;
|
||||
if(this._payload._object !is null)
|
||||
cp._object = this._payload._object.copy;
|
||||
return cp;
|
||||
}
|
||||
}
|
||||
|
||||
public bool opCast(T:bool)() {
|
||||
final switch(this._type) {
|
||||
case Type.Object:
|
||||
|
@ -449,11 +548,23 @@ struct var {
|
|||
}
|
||||
|
||||
public var opOpAssign(string op, T)(T t) {
|
||||
if(payloadType() == Type.Object) {
|
||||
var operator = this["opOpAssign"];
|
||||
if(operator._type == Type.Function)
|
||||
return operator.call(this, op, t);
|
||||
}
|
||||
|
||||
return _op!(this, this, op, T)(t);
|
||||
}
|
||||
|
||||
public var opBinary(string op, T)(T t) {
|
||||
var n;
|
||||
if(payloadType() == Type.Object) {
|
||||
var operator = this["opBinary"];
|
||||
if(operator._type == Type.Function) {
|
||||
return operator.call(this, op, t);
|
||||
}
|
||||
}
|
||||
return _op!(n, this, op, T)(t);
|
||||
}
|
||||
|
||||
|
@ -555,9 +666,11 @@ struct var {
|
|||
|
||||
return T.init;
|
||||
case Type.Function:
|
||||
static if(isSomeString!T)
|
||||
return "<function>";
|
||||
// FIXME: we just might be able to do better for both of these
|
||||
return T.init;
|
||||
break;
|
||||
//break;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -614,6 +727,36 @@ struct var {
|
|||
throw new DynamicTypeException(this, t, file, line);
|
||||
}
|
||||
|
||||
public var opSlice(var e1, var e2) {
|
||||
return this.opSlice(e1.get!int, e2.get!int);
|
||||
}
|
||||
|
||||
public var opSlice(int e1, int e2) {
|
||||
if(this.payloadType() == Type.Array) {
|
||||
if(e1 > _payload._array.length)
|
||||
e1 = _payload._array.length;
|
||||
if(e2 > _payload._array.length)
|
||||
e2 = _payload._array.length;
|
||||
return var(_payload._array[e1 .. e2]);
|
||||
}
|
||||
if(this.payloadType() == Type.String) {
|
||||
if(e1 > _payload._string.length)
|
||||
e1 = _payload._string.length;
|
||||
if(e2 > _payload._string.length)
|
||||
e2 = _payload._string.length;
|
||||
return var(_payload._string[e1 .. e2]);
|
||||
}
|
||||
if(this.payloadType() == Type.Object) {
|
||||
var operator = this["opSlice"];
|
||||
if(operator._type == Type.Function) {
|
||||
return operator.call(this, e1, e2);
|
||||
}
|
||||
}
|
||||
|
||||
// might be worth throwing here too
|
||||
return var(null);
|
||||
}
|
||||
|
||||
public @property ref var opDispatch(string name, string file = __FILE__, size_t line = __LINE__)() {
|
||||
return this[name];
|
||||
}
|
||||
|
@ -634,18 +777,48 @@ struct var {
|
|||
// if name is numeric, we should convert to int
|
||||
if(name.length && name[0] >= '0' && name[0] <= '9')
|
||||
return opIndex(to!size_t(name), file, line);
|
||||
_requireType(Type.Object); // FIXME
|
||||
if(_payload._object is null)
|
||||
|
||||
if(this.payloadType() != Type.Object && name == "prototype")
|
||||
return prototype();
|
||||
|
||||
if(name == "typeof") {
|
||||
var* tmp = new var;
|
||||
*tmp = to!string(this.payloadType());
|
||||
return *tmp;
|
||||
}
|
||||
|
||||
if(name == "length" && this.payloadType() == Type.String) {
|
||||
var* tmp = new var;
|
||||
*tmp = _payload._string.length;
|
||||
return *tmp;
|
||||
}
|
||||
if(name == "length" && this.payloadType() == Type.Array) {
|
||||
var* tmp = new var;
|
||||
*tmp = _payload._array.length;
|
||||
return *tmp;
|
||||
}
|
||||
|
||||
PrototypeObject from;
|
||||
if(this.payloadType() == Type.Object)
|
||||
from = _payload._object;
|
||||
else {
|
||||
var pt = this.prototype();
|
||||
assert(pt.payloadType() == Type.Object);
|
||||
from = pt._payload._object;
|
||||
}
|
||||
|
||||
if(from is null)
|
||||
throw new DynamicTypeException(var(null), Type.Object, file, line);
|
||||
return this._payload._object._getMember(name, true, false, file, line);
|
||||
return from._getMember(name, true, false, file, line);
|
||||
}
|
||||
|
||||
public ref var opIndexAssign(T)(T t, string name, string file = __FILE__, size_t line = __LINE__) {
|
||||
if(name.length && name[0] >= '0' && name[0] <= '9')
|
||||
return opIndexAssign(t, to!size_t(name), file, line);
|
||||
_requireType(Type.Object); // FIXME
|
||||
_requireType(Type.Object); // FIXME?
|
||||
if(_payload._object is null)
|
||||
throw new DynamicTypeException(var(null), Type.Object, file, line);
|
||||
|
||||
this._payload._object._getMember(name, false, false, file, line) = t;
|
||||
return this._payload._object._properties[name];
|
||||
}
|
||||
|
@ -672,6 +845,14 @@ struct var {
|
|||
return *n;
|
||||
}
|
||||
|
||||
ref var _getOwnProperty(string name, string file = __FILE__, size_t line = __LINE__) {
|
||||
if(_type == Type.Object) {
|
||||
if(_payload._object !is null)
|
||||
return this._payload._object._getMember(name, false, false, file, line);
|
||||
}
|
||||
var* n = new var();
|
||||
return *n;
|
||||
}
|
||||
|
||||
@property static var emptyObject(PrototypeObject prototype = null) {
|
||||
var v;
|
||||
|
@ -681,6 +862,54 @@ struct var {
|
|||
return v;
|
||||
}
|
||||
|
||||
// what I call prototype is more like what Mozilla calls __proto__, but tbh I think this is better so meh
|
||||
@property ref var prototype() {
|
||||
static var _arrayPrototype;
|
||||
static var _functionPrototype;
|
||||
static var _stringPrototype;
|
||||
|
||||
|
||||
final switch(payloadType()) {
|
||||
case Type.Array:
|
||||
assert(_arrayPrototype._type == Type.Object);
|
||||
if(_arrayPrototype._payload._object is null) {
|
||||
_arrayPrototype._object = new PrototypeObject();
|
||||
writeln("ctor on ", payloadType());
|
||||
}
|
||||
|
||||
return _arrayPrototype;
|
||||
break;
|
||||
case Type.Function:
|
||||
assert(_functionPrototype._type == Type.Object);
|
||||
if(_functionPrototype._payload._object is null) {
|
||||
_functionPrototype._object = new PrototypeObject();
|
||||
}
|
||||
|
||||
return _functionPrototype;
|
||||
break;
|
||||
case Type.String:
|
||||
assert(_stringPrototype._type == Type.Object);
|
||||
if(_stringPrototype._payload._object is null) {
|
||||
_stringPrototype._object = new PrototypeObject();
|
||||
}
|
||||
|
||||
return _stringPrototype;
|
||||
break;
|
||||
case Type.Object:
|
||||
if(_payload._object)
|
||||
return _payload._object._prototype;
|
||||
// FIXME: should we do a generic object prototype?
|
||||
break;
|
||||
case Type.Integral:
|
||||
case Type.Floating:
|
||||
case Type.Boolean:
|
||||
// these types don't have prototypes
|
||||
}
|
||||
|
||||
var* v = new var(null);
|
||||
return *v;
|
||||
}
|
||||
|
||||
@property static var emptyArray() {
|
||||
var v;
|
||||
v._type = Type.Array;
|
||||
|
@ -789,22 +1018,129 @@ struct var {
|
|||
}
|
||||
}
|
||||
|
||||
/*
|
||||
@PrototypeConstructor
|
||||
void foo(var _this) {
|
||||
class WrappedNativeObject(T, bool wrapData = true) : PrototypeObject {
|
||||
T nativeObject;
|
||||
|
||||
|
||||
auto makeWrapper(string member)() {
|
||||
return (var _this, var[] args) {
|
||||
auto func = &(__traits(getMember, nativeObject, member));
|
||||
var ret;
|
||||
|
||||
// this is a filthy hack and i hate it
|
||||
// the problem with overriding getMember though is we can't really control what happens when it is set, since that's all done through the ref, and we don't want to overload stuff there since it can be copied.
|
||||
// so instead on each method call, I'll copy the data from the prototype back out... and then afterward, copy from the object back to the prototype. gross.
|
||||
|
||||
// first we need to make sure that the native object is updated...
|
||||
static if(wrapData)
|
||||
updateNativeObject();
|
||||
|
||||
|
||||
|
||||
ParameterTypeTuple!(__traits(getMember, nativeObject, member)) fargs;
|
||||
foreach(idx, a; fargs) {
|
||||
if(idx == args.length)
|
||||
break;
|
||||
cast(Unqual!(typeof(a))) fargs[idx] = args[idx].get!(Unqual!(typeof(a)));
|
||||
}
|
||||
|
||||
static if(is(ReturnType!func == void)) {
|
||||
func(fargs);
|
||||
} else {
|
||||
ret = func(fargs);
|
||||
}
|
||||
|
||||
|
||||
// then transfer updates from it back here
|
||||
static if(wrapData)
|
||||
getUpdatesFromNativeObject();
|
||||
|
||||
return ret;
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
this(T t) {
|
||||
this.name = T.stringof;
|
||||
this.nativeObject = t;
|
||||
// this.prototype = new PrototypeObject();
|
||||
|
||||
foreach(member; __traits(allMembers, T)) {
|
||||
static if(__traits(compiles, __traits(getMember, nativeObject, member))) {
|
||||
static if(is(typeof(__traits(getMember, nativeObject, member)) == function)) {
|
||||
this._getMember(member, false, false)._function =
|
||||
makeWrapper!(member)();
|
||||
} else static if(wrapData)
|
||||
this._getMember(member, false, false) = __traits(getMember, nativeObject, member);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void updateNativeObject() {
|
||||
foreach(member; __traits(allMembers, T)) {
|
||||
static if(__traits(compiles, __traits(getMember, nativeObject, member))) {
|
||||
static if(is(typeof(__traits(getMember, nativeObject, member)) == function)) {
|
||||
// ignore, if these are overridden, we want it to stay that way
|
||||
} else {
|
||||
// if this doesn't compile, it is prolly cuz it is const or something
|
||||
static if(__traits(compiles, this._getMember(member, false, false).putInto(__traits(getMember, nativeObject, member))))
|
||||
this._getMember(member, false, false).putInto(__traits(getMember, nativeObject, member));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void getUpdatesFromNativeObject() {
|
||||
foreach(member; __traits(allMembers, T)) {
|
||||
static if(__traits(compiles, __traits(getMember, nativeObject, member))) {
|
||||
static if(is(typeof(__traits(getMember, nativeObject, member)) == function)) {
|
||||
// ignore, these won't change
|
||||
} else {
|
||||
this._getMember(member, false, false) = __traits(getMember, nativeObject, member);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
override WrappedNativeObject!T copy() {
|
||||
auto n = new WrappedNativeObject!T(nativeObject);
|
||||
// FIXME: what if nativeObject is a reference type?
|
||||
return n;
|
||||
}
|
||||
}
|
||||
|
||||
@PrototypeThis
|
||||
a.foo = function(var _this) {}
|
||||
*/
|
||||
|
||||
class PrototypeObject {
|
||||
string name;
|
||||
PrototypeObject prototype;
|
||||
var _prototype;
|
||||
|
||||
PrototypeObject prototype() {
|
||||
if(_prototype.payloadType() == var.Type.Object)
|
||||
return _prototype._payload._object;
|
||||
return null;
|
||||
}
|
||||
|
||||
PrototypeObject prototype(PrototypeObject set) {
|
||||
this._prototype._object = set;
|
||||
return set;
|
||||
}
|
||||
|
||||
var[string] _properties;
|
||||
|
||||
PrototypeObject copy() {
|
||||
auto n = new PrototypeObject();
|
||||
n.prototype = this.prototype;
|
||||
n.name = this.name;
|
||||
foreach(k, v; _properties) {
|
||||
n._properties[k] = v._copy;
|
||||
}
|
||||
return n;
|
||||
}
|
||||
|
||||
// FIXME: maybe throw something else
|
||||
package ref var _getMember(string name, bool recurse, bool throwOnFailure, string file = __FILE__, size_t line = __LINE__) {
|
||||
/*package*/ ref var _getMember(string name, bool recurse, bool throwOnFailure, string file = __FILE__, size_t line = __LINE__) {
|
||||
if(name == "prototype")
|
||||
return _prototype;
|
||||
|
||||
auto curr = this;
|
||||
do {
|
||||
auto prop = name in curr._properties;
|
||||
|
|
498
script.d
498
script.d
|
@ -1,30 +1,81 @@
|
|||
/**
|
||||
Script features:
|
||||
|
||||
* scope guards
|
||||
OVERVIEW
|
||||
* easy interop with D thanks to arsd.jsvar. When interpreting, pass a var object to use as globals.
|
||||
This object also contains the global state when interpretation is done.
|
||||
* mostly familiar syntax, hybrid of D and Javascript
|
||||
* simple implementation is moderately small and fairly easy to hack on (though it gets messier by the day), but it isn't made for speed.
|
||||
|
||||
SPECIFICS
|
||||
* mixin aka eval (does it at runtime, so more like eval than mixin, but I want it to look like D)
|
||||
* scope guards, like in D
|
||||
* try/catch/finally/throw
|
||||
* for/while
|
||||
* D style operators
|
||||
* for/while (foreach coming soon)
|
||||
* D style operators: +-/* on all numeric types, ~ on strings and arrays, |&^ on integers.
|
||||
Operators can coerce types as needed: 10 ~ "hey" == "10hey". 10 + "3" == 13.
|
||||
Any math, except bitwise math, with a floating point component returns a floating point component, but pure int math is done as ints (unlike Javascript btw).
|
||||
Any bitwise math coerces to int.
|
||||
|
||||
So you can do some type coercion like this:
|
||||
|
||||
a = a|0; // forces to int
|
||||
a = "" ~ a; // forces to string
|
||||
a = a+0.0; // coerces to float
|
||||
|
||||
But I'll add nicer type stuff later. (maybe cast() actually)
|
||||
* some operator overloading on objects, passing opBinary(op, rhs), length, and perhaps others through like they would be in D.
|
||||
* if/else
|
||||
* int, float, string, array, and json!q{} literals
|
||||
* __FILE__, __LINE__
|
||||
* array slicing, but note that slices are rvalues currently
|
||||
* variables must start with A-Z, a-z, _, or $, then must be [A-Za-z0-9_]*.
|
||||
(The $ can also stand alone, and this is a special thing when slicing, so you probably shouldn't use it at all.).
|
||||
Variable names that start with __ are reserved and you shouldn't use them.
|
||||
* int, float, string, array, bool, and json!q{} literals
|
||||
* var.prototype, var.typeof. prototype works more like Mozilla's __proto__ than standard javascript prototype.
|
||||
* classes:
|
||||
class Foo : bar {
|
||||
// constructors, D style
|
||||
this(var a) { ctor.... }
|
||||
|
||||
// static vars go on the auto created prototype
|
||||
static var b = 10;
|
||||
|
||||
// instance vars go on this instance itself
|
||||
var instance = 20;
|
||||
|
||||
// "virtual" functions can be overridden kinda like you expect in D, though there is no override keyword
|
||||
function virt() {
|
||||
// be sure to use this. as a prefix for any class defined variables in here
|
||||
}
|
||||
}
|
||||
|
||||
var foo = new Foo(12);
|
||||
|
||||
You can also use 'new' on another object to get a copy of it.
|
||||
* return, break, continue, but currently cannot do labeled breaks and continues
|
||||
* __FILE__, __LINE__, but currently not as default arguments for D behavior (they always evaluate at the definition point)
|
||||
* most everything are expressions, though note this is pretty buggy! But as a consequence:
|
||||
for(var a = 0, b = 0; a < 10; a+=1, b+=1) {}
|
||||
won't work but this will:
|
||||
for(var a = 0, b = 0; a < 10; {a+=1; b+=1}) {}
|
||||
|
||||
You can encase things in {} anywhere instead of a comma operator, and it works kinda similarly.
|
||||
|
||||
{} creates a new scope inside it and returns the last value evaluated.
|
||||
* functions:
|
||||
var fn = function(args...) expr;
|
||||
or
|
||||
function fn(args....) expr;
|
||||
|
||||
Special function things:
|
||||
_arguments
|
||||
_thisfunc
|
||||
this
|
||||
Special function local variables:
|
||||
_arguments = var[] of the arguments passed
|
||||
_thisfunc = reference to the function itself
|
||||
this = reference to the object on which it is being called - note this is like Javascript, not D.
|
||||
|
||||
args can say var if you want, but don't have to
|
||||
default arguments supported in any position
|
||||
when calling, you can use the default keyword to use the default value in any position
|
||||
|
||||
* easy interop with D thanks to arsd.jsvar. When interpreting, pass a var object to use as globals.
|
||||
This object also contains the global state when interpretation is done.
|
||||
|
||||
*/
|
||||
module arsd.script;
|
||||
|
||||
|
@ -79,23 +130,27 @@ struct ScriptToken {
|
|||
}
|
||||
|
||||
// these need to be ordered from longest to shortest
|
||||
// some of these aren't actually used, like struct and goto right now, but I want them reserved for later
|
||||
private enum string[] keywords = [
|
||||
"function",
|
||||
"function", "continue",
|
||||
"__FILE__", "__LINE__", // these two are special to the lexer
|
||||
"foreach", "json!q{", "default", "finally",
|
||||
"return",
|
||||
"while", "catch", "throw", "scope",
|
||||
"return", "static", "struct", "import", "module", "assert", "switch",
|
||||
"while", "catch", "throw", "scope", "break", "super", "class", "false", "mixin", "super",
|
||||
"auto", // provided as an alias for var right now, may change later
|
||||
"null", "else",
|
||||
"var", "for", "try",
|
||||
"if",
|
||||
"null", "else", "true", "eval", "goto", "enum", "case",
|
||||
"var", "for", "try", "new",
|
||||
"if", "do",
|
||||
];
|
||||
private enum string[] symbols = [
|
||||
"//", "/*", "/+",
|
||||
"&&", "||",
|
||||
"+=", "-=", "*=", "/=", "~=", "==", "<=", ">=","!=",
|
||||
"&=", "|=", "^=",
|
||||
"..",
|
||||
".",",",";",":",
|
||||
"[", "]", "{", "}", "(", ")",
|
||||
"&", "|", "^",
|
||||
"+", "-", "*", "/", "=", "<", ">","~","!",
|
||||
];
|
||||
|
||||
|
@ -108,6 +163,8 @@ class TokenStream(TextStream) {
|
|||
|
||||
void advance(int size) {
|
||||
foreach(i; 0 .. size) {
|
||||
if(text.empty)
|
||||
break;
|
||||
if(text[0] == '\n')
|
||||
lineNumber ++;
|
||||
text.popFront();
|
||||
|
@ -173,7 +230,7 @@ class TokenStream(TextStream) {
|
|||
token.type = sawDot ? ScriptToken.Type.float_number : ScriptToken.Type.int_number;
|
||||
token.str = text[0 .. pos];
|
||||
advance(pos);
|
||||
} else if((text[0] >= 'a' && text[0] <= 'z') || (text[0] == '_') || (text[0] >= 'A' && text[0] <= 'Z')) {
|
||||
} else if((text[0] >= 'a' && text[0] <= 'z') || (text[0] == '_') || (text[0] >= 'A' && text[0] <= 'Z') || text[0] == '$') {
|
||||
bool found = false;
|
||||
foreach(keyword; keywords)
|
||||
if(text.length >= keyword.length && text[0 .. keyword.length] == keyword &&
|
||||
|
@ -212,6 +269,9 @@ class TokenStream(TextStream) {
|
|||
if(!found) {
|
||||
token.type = ScriptToken.Type.identifier;
|
||||
int pos;
|
||||
if(text[0] == '$')
|
||||
pos++;
|
||||
|
||||
while(pos < text.length
|
||||
&& ((text[pos] >= 'a' && text[pos] <= 'z') ||
|
||||
(text[pos] == '_') ||
|
||||
|
@ -311,6 +371,17 @@ class Expression {
|
|||
abstract InterpretResult interpret(PrototypeObject sc);
|
||||
}
|
||||
|
||||
class MixinExpression : Expression {
|
||||
Expression e1;
|
||||
this(Expression e1) {
|
||||
this.e1 = e1;
|
||||
}
|
||||
|
||||
override InterpretResult interpret(PrototypeObject sc) {
|
||||
return InterpretResult(.interpret(e1.interpret(sc).value.get!string ~ ";", sc), sc);
|
||||
}
|
||||
}
|
||||
|
||||
class StringLiteralExpression : Expression {
|
||||
string literal;
|
||||
|
||||
|
@ -327,6 +398,20 @@ class StringLiteralExpression : Expression {
|
|||
return InterpretResult(var(literal), sc);
|
||||
}
|
||||
}
|
||||
|
||||
class BoolLiteralExpression : Expression {
|
||||
bool literal;
|
||||
this(string l) {
|
||||
literal = to!bool(l);
|
||||
}
|
||||
|
||||
override string toString() { return to!string(literal); }
|
||||
|
||||
override InterpretResult interpret(PrototypeObject sc) {
|
||||
return InterpretResult(var(literal), sc);
|
||||
}
|
||||
}
|
||||
|
||||
class IntLiteralExpression : Expression {
|
||||
long literal;
|
||||
|
||||
|
@ -387,6 +472,12 @@ class FunctionLiteralExpression : Expression {
|
|||
DefaultArgumentDummyObject = new PrototypeObject();
|
||||
}
|
||||
|
||||
this(VariableDeclaration args, Expression bod) {
|
||||
this();
|
||||
this.arguments = args;
|
||||
this.functionBody = bod;
|
||||
}
|
||||
|
||||
/*
|
||||
function identifier (arg list) expression
|
||||
|
||||
|
@ -478,7 +569,7 @@ class BinaryExpression : Expression {
|
|||
//writeln(left, " "~op~" ", right);
|
||||
|
||||
var n;
|
||||
foreach(ctOp; CtList!("+", "-", "*", "/", "==", "!=", "<=", ">=", ">", "<", "~"))
|
||||
foreach(ctOp; CtList!("+", "-", "*", "/", "==", "!=", "<=", ">=", ">", "<", "~", "&&", "||", "&", "|", "^"))
|
||||
if(ctOp == op) {
|
||||
n = mixin("left "~ctOp~" right");
|
||||
}
|
||||
|
@ -513,7 +604,7 @@ class OpAssignExpression : Expression {
|
|||
//writeln(left, " "~op~"= ", right);
|
||||
|
||||
var n;
|
||||
foreach(ctOp; CtList!("+=", "-=", "*=", "/=", "~="))
|
||||
foreach(ctOp; CtList!("+=", "-=", "*=", "/=", "~=", "&=", "|=", "^="))
|
||||
if(ctOp[0..1] == op)
|
||||
n = mixin("v.getVar(sc) "~ctOp~" right");
|
||||
|
||||
|
@ -538,7 +629,7 @@ class AssignExpression : Expression {
|
|||
if(v is null)
|
||||
throw new ScriptRuntimeException("not an lvalue", 0 /* FIXME */);
|
||||
|
||||
auto ret = v.getVar(sc) = e2.interpret(sc).value;
|
||||
auto ret = v.getVar(sc, false) = e2.interpret(sc).value;
|
||||
|
||||
return InterpretResult(ret, sc);
|
||||
}
|
||||
|
@ -566,8 +657,8 @@ class VariableExpression : Expression {
|
|||
return identifier;
|
||||
}
|
||||
|
||||
ref var getVar(PrototypeObject sc) {
|
||||
return sc._getMember(identifier, true, true);
|
||||
ref var getVar(PrototypeObject sc, bool recurse = true) {
|
||||
return sc._getMember(identifier, true /* FIXME: recurse?? */, true);
|
||||
}
|
||||
|
||||
ref var getVarFrom(PrototypeObject sc, ref var v) {
|
||||
|
@ -582,15 +673,17 @@ class VariableExpression : Expression {
|
|||
class DotVarExpression : VariableExpression {
|
||||
Expression e1;
|
||||
VariableExpression e2;
|
||||
bool recurse = true;
|
||||
|
||||
this(Expression e1) {
|
||||
this.e1 = e1;
|
||||
super(null);
|
||||
}
|
||||
|
||||
this(Expression e1, VariableExpression e2) {
|
||||
this(Expression e1, VariableExpression e2, bool recurse = true) {
|
||||
this.e1 = e1;
|
||||
this.e2 = e2;
|
||||
this.recurse = recurse;
|
||||
//assert(typeid(e2) == typeid(VariableExpression));
|
||||
super("<do not use>");//e1.identifier ~ "." ~ e2.identifier);
|
||||
}
|
||||
|
@ -599,9 +692,17 @@ class DotVarExpression : VariableExpression {
|
|||
return e1.toString() ~ "." ~ e2.toString();
|
||||
}
|
||||
|
||||
override ref var getVar(PrototypeObject sc) {
|
||||
override ref var getVar(PrototypeObject sc, bool recurse = true) {
|
||||
if(!this.recurse) {
|
||||
// this is a special hack...
|
||||
if(auto ve = cast(VariableExpression) e1) {
|
||||
return ve.getVar(sc)._getOwnProperty(e2.identifier);
|
||||
}
|
||||
assert(0);
|
||||
}
|
||||
|
||||
if(auto ve = cast(VariableExpression) e1)
|
||||
return this.getVarFrom(sc, ve.getVar(sc));
|
||||
return this.getVarFrom(sc, ve.getVar(sc, recurse));
|
||||
else {
|
||||
// make a temporary for the lhs
|
||||
auto v = new var();
|
||||
|
@ -629,9 +730,9 @@ class IndexExpression : VariableExpression {
|
|||
return e1.toString() ~ "[" ~ e2.toString() ~ "]";
|
||||
}
|
||||
|
||||
override ref var getVar(PrototypeObject sc) {
|
||||
override ref var getVar(PrototypeObject sc, bool recurse = true) {
|
||||
if(auto ve = cast(VariableExpression) e1)
|
||||
return ve.getVar(sc)[e2.interpret(sc).value];
|
||||
return ve.getVar(sc, recurse)[e2.interpret(sc).value];
|
||||
else {
|
||||
auto v = new var();
|
||||
*v = e1.interpret(sc).value;
|
||||
|
@ -640,10 +741,57 @@ class IndexExpression : VariableExpression {
|
|||
}
|
||||
}
|
||||
|
||||
class SliceExpression : Expression {
|
||||
// e1[e2 .. e3]
|
||||
Expression e1;
|
||||
Expression e2;
|
||||
Expression e3;
|
||||
|
||||
this(Expression e1, Expression e2, Expression e3) {
|
||||
this.e1 = e1;
|
||||
this.e2 = e2;
|
||||
this.e3 = e3;
|
||||
}
|
||||
|
||||
override string toString() {
|
||||
return e1.toString() ~ "[" ~ e2.toString() ~ " .. " ~ e3.toString() ~ "]";
|
||||
}
|
||||
|
||||
override InterpretResult interpret(PrototypeObject sc) {
|
||||
var lhs = e1.interpret(sc).value;
|
||||
|
||||
auto specialScope = new PrototypeObject();
|
||||
specialScope.prototype = sc;
|
||||
specialScope._getMember("$", false, false) = lhs.length;
|
||||
|
||||
return InterpretResult(lhs[e2.interpret(specialScope).value .. e3.interpret(specialScope).value], sc);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class LoopControlExpression : Expression {
|
||||
InterpretResult.FlowControl op;
|
||||
this(string op) {
|
||||
if(op == "continue")
|
||||
this.op = InterpretResult.FlowControl.Continue;
|
||||
else if(op == "break")
|
||||
this.op = InterpretResult.FlowControl.Break;
|
||||
else assert(0, op);
|
||||
}
|
||||
|
||||
override InterpretResult interpret(PrototypeObject sc) {
|
||||
return InterpretResult(var(null), sc, op);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class ReturnExpression : Expression {
|
||||
Expression value;
|
||||
|
||||
this(Expression v) {
|
||||
value = v;
|
||||
}
|
||||
|
||||
override InterpretResult interpret(PrototypeObject sc) {
|
||||
return InterpretResult(value.interpret(sc).value, sc, InterpretResult.FlowControl.Return);
|
||||
}
|
||||
|
@ -680,7 +828,10 @@ class ScopeExpression : Expression {
|
|||
}
|
||||
|
||||
foreach(expression; expressions) {
|
||||
ret = expression.interpret(innerScope).value;
|
||||
auto res = expression.interpret(innerScope);
|
||||
ret = res.value;
|
||||
if(res.flowControl != InterpretResult.FlowControl.Normal)
|
||||
return InterpretResult(ret, sc, res.flowControl);
|
||||
}
|
||||
return InterpretResult(ret, sc);
|
||||
}
|
||||
|
@ -704,23 +855,35 @@ class ForExpression : Expression {
|
|||
if(initialization !is null)
|
||||
initialization.interpret(loopScope);
|
||||
|
||||
void doLoopBody() {
|
||||
// FIXME: break, continue
|
||||
result = loopBody.interpret(loopScope).value;
|
||||
InterpretResult.FlowControl flowControl;
|
||||
|
||||
static string doLoopBody() { return q{
|
||||
auto res = loopBody.interpret(loopScope);
|
||||
result = res.value;
|
||||
flowControl = res.flowControl;
|
||||
if(flowControl == InterpretResult.FlowControl.Break)
|
||||
break;
|
||||
if(flowControl == InterpretResult.FlowControl.Return)
|
||||
break;
|
||||
//if(flowControl == InterpretResult.FlowControl.Continue)
|
||||
// this is fine, we still want to do the advancement
|
||||
if(advancement)
|
||||
advancement.interpret(loopScope);
|
||||
}
|
||||
};}
|
||||
|
||||
if(condition !is null) {
|
||||
while(condition.interpret(loopScope).value) {
|
||||
doLoopBody();
|
||||
mixin(doLoopBody());
|
||||
}
|
||||
} else
|
||||
while(true) {
|
||||
doLoopBody();
|
||||
mixin(doLoopBody());
|
||||
}
|
||||
|
||||
return InterpretResult(result, sc);
|
||||
if(flowControl != InterpretResult.FlowControl.Return)
|
||||
flowControl = InterpretResult.FlowControl.Normal;
|
||||
|
||||
return InterpretResult(result, sc, flowControl);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -732,7 +895,7 @@ class IfExpression : Expression {
|
|||
this() {}
|
||||
|
||||
override InterpretResult interpret(PrototypeObject sc) {
|
||||
var result;
|
||||
InterpretResult result;
|
||||
assert(condition !is null);
|
||||
|
||||
auto ifScope = new PrototypeObject();
|
||||
|
@ -740,12 +903,38 @@ class IfExpression : Expression {
|
|||
|
||||
if(condition.interpret(ifScope).value) {
|
||||
if(ifTrue !is null)
|
||||
result = ifTrue.interpret(ifScope).value;
|
||||
result = ifTrue.interpret(ifScope);
|
||||
} else {
|
||||
if(ifFalse !is null)
|
||||
result = ifFalse.interpret(ifScope).value;
|
||||
result = ifFalse.interpret(ifScope);
|
||||
}
|
||||
return InterpretResult(result, sc);
|
||||
return InterpretResult(result.value, sc, result.flowControl);
|
||||
}
|
||||
}
|
||||
|
||||
class NewExpression : Expression {
|
||||
Expression what;
|
||||
Expression[] args;
|
||||
this(Expression w) {
|
||||
what = w;
|
||||
}
|
||||
|
||||
override InterpretResult interpret(PrototypeObject sc) {
|
||||
assert(what !is null);
|
||||
|
||||
var[] args;
|
||||
foreach(arg; this.args)
|
||||
args ~= arg.interpret(sc).value;
|
||||
|
||||
var original = what.interpret(sc).value;
|
||||
var n = original._copy;
|
||||
if(n.payloadType() == var.Type.Object) {
|
||||
var ctor = original._getOwnProperty("__ctor");
|
||||
if(ctor)
|
||||
ctor.apply(n, args);
|
||||
}
|
||||
|
||||
return InterpretResult(n, sc);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -861,9 +1050,9 @@ class CallExpression : Expression {
|
|||
}
|
||||
|
||||
var _this;
|
||||
if(auto dve = cast(DotVarExpression) func)
|
||||
_this = dve.interpret(sc).value;
|
||||
else if(auto ide = cast(IndexExpression) func)
|
||||
if(auto dve = cast(DotVarExpression) func) {
|
||||
_this = dve.e1.interpret(sc).value;
|
||||
} else if(auto ide = cast(IndexExpression) func)
|
||||
_this = ide.interpret(sc).value;
|
||||
|
||||
return InterpretResult(f.apply(var(_this), args), sc);
|
||||
|
@ -918,6 +1107,19 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
e = new StringLiteralExpression(token.str);
|
||||
else if(token.type == ScriptToken.Type.symbol || token.type == ScriptToken.Type.keyword) {
|
||||
switch(token.str) {
|
||||
case "true":
|
||||
case "false":
|
||||
e = new BoolLiteralExpression(token.str);
|
||||
break;
|
||||
case "new":
|
||||
// FIXME: why is this needed here? maybe it should be here instead of parseExpression
|
||||
tokens.pushFront(token);
|
||||
return parseExpression(tokens);
|
||||
case "(":
|
||||
//tokens.popFront();
|
||||
auto parenthetical = new ParentheticalExpression(parseExpression(tokens));
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, ")");
|
||||
return parenthetical;
|
||||
case "[":
|
||||
// array literal
|
||||
auto arr = new ArrayLiteralExpression();
|
||||
|
@ -941,7 +1143,6 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
arr.elements ~= parseExpression(tokens);
|
||||
|
||||
goto moreElements;
|
||||
break;
|
||||
case "json!q{":
|
||||
// json object literal
|
||||
auto obj = new ObjectLiteralExpression();
|
||||
|
@ -1002,6 +1203,12 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
case "null":
|
||||
e = new NullLiteralExpression();
|
||||
break;
|
||||
case "mixin":
|
||||
case "eval":
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, "(");
|
||||
e = new MixinExpression(parseExpression(tokens));
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, ")");
|
||||
break;
|
||||
default:
|
||||
goto unknown;
|
||||
}
|
||||
|
@ -1020,7 +1227,13 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
break;
|
||||
case "[":
|
||||
tokens.popFront();
|
||||
e = new IndexExpression(e, parseExpression(tokens));
|
||||
auto e1 = parseExpression(tokens);
|
||||
if(tokens.peekNextToken(ScriptToken.Type.symbol, "..")) {
|
||||
tokens.popFront();
|
||||
e = new SliceExpression(e, e1, parseExpression(tokens));
|
||||
} else {
|
||||
e = new IndexExpression(e, e1);
|
||||
}
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, "]");
|
||||
break;
|
||||
case ".":
|
||||
|
@ -1037,15 +1250,9 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
assert(0, to!string(tokens));
|
||||
}
|
||||
|
||||
Expression parseFunctionCall(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expression e) {
|
||||
assert(!tokens.empty);
|
||||
auto peek = tokens.front;
|
||||
auto exp = new CallExpression(e);
|
||||
tokens.popFront();
|
||||
if(tokens.empty)
|
||||
throw new ScriptCompileException("unexpected end of file when parsing call expression", peek.lineNumber);
|
||||
Expression parseArguments(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expression exp, ref Expression[] where) {
|
||||
// arguments.
|
||||
peek = tokens.front;
|
||||
auto peek = tokens.front;
|
||||
if(peek.type == ScriptToken.Type.symbol && peek.str == ")") {
|
||||
tokens.popFront();
|
||||
return exp;
|
||||
|
@ -1055,9 +1262,9 @@ Expression parseFunctionCall(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Ex
|
|||
|
||||
if(tokens.peekNextToken(ScriptToken.Type.keyword, "default")) {
|
||||
tokens.popFront();
|
||||
exp.arguments ~= null;
|
||||
where ~= null;
|
||||
} else {
|
||||
exp.arguments ~= parseExpression(tokens);
|
||||
where ~= parseExpression(tokens);
|
||||
}
|
||||
|
||||
if(tokens.empty)
|
||||
|
@ -1071,6 +1278,17 @@ Expression parseFunctionCall(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Ex
|
|||
return exp;
|
||||
} else
|
||||
throw new ScriptCompileException("unexpected '"~peek.str~"' when reading argument list", peek.lineNumber);
|
||||
|
||||
}
|
||||
|
||||
Expression parseFunctionCall(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expression e) {
|
||||
assert(!tokens.empty);
|
||||
auto peek = tokens.front;
|
||||
auto exp = new CallExpression(e);
|
||||
tokens.popFront();
|
||||
if(tokens.empty)
|
||||
throw new ScriptCompileException("unexpected end of file when parsing call expression", peek.lineNumber);
|
||||
return parseArguments(tokens, exp, exp.arguments);
|
||||
}
|
||||
|
||||
Expression parseFactor(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||
|
@ -1101,6 +1319,7 @@ Expression parseAddend(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
|
||||
if(peek.type == ScriptToken.Type.symbol) {
|
||||
switch(peek.str) {
|
||||
case "..": // possible FIXME
|
||||
case ")": // possible FIXME
|
||||
case "]": // possible FIXME
|
||||
case "}": // possible FIXME
|
||||
|
@ -1111,6 +1330,19 @@ Expression parseAddend(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
tokens.popFront();
|
||||
return new AssignExpression(e1, parseAddend(tokens));
|
||||
case "~":
|
||||
// FIXME: make sure this has the right associativity
|
||||
|
||||
case "&&": // FIXME: precedence?
|
||||
case "||":
|
||||
|
||||
case "&":
|
||||
case "|":
|
||||
case "^":
|
||||
|
||||
case "&=":
|
||||
case "|=":
|
||||
case "^=":
|
||||
|
||||
case "+":
|
||||
case "-":
|
||||
|
||||
|
@ -1188,6 +1420,133 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
ret = parseFunctionCall(tokens, parenthetical);
|
||||
} else
|
||||
ret = parenthetical;
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "new")) {
|
||||
auto start = tokens.front;
|
||||
tokens.popFront();
|
||||
|
||||
auto expr = parseVariableName(tokens);
|
||||
auto ne = new NewExpression(expr);
|
||||
if(tokens.peekNextToken(ScriptToken.Type.symbol, "(")) {
|
||||
tokens.popFront();
|
||||
parseArguments(tokens, ne, ne.args);
|
||||
}
|
||||
|
||||
ret = ne;
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "class")) {
|
||||
auto start = tokens.front;
|
||||
tokens.popFront();
|
||||
|
||||
Expression[] expressions;
|
||||
|
||||
// the way classes work is they are actually object literals with a different syntax. new foo then just copies it
|
||||
/*
|
||||
we create a prototype object
|
||||
we create an object, with that prototype
|
||||
|
||||
set all functions and static stuff to the prototype
|
||||
the rest goes to the object
|
||||
|
||||
the expression returns the object we made
|
||||
*/
|
||||
|
||||
auto vars = new VariableDeclaration();
|
||||
vars.identifiers = ["__proto", "__obj"];
|
||||
vars.initializers = [new ObjectLiteralExpression(), new ObjectLiteralExpression()];
|
||||
expressions ~= vars;
|
||||
|
||||
// FIXME: operators need to have their this be bound somehow since it isn't passwed
|
||||
// OR the op rewrite could pass this
|
||||
|
||||
expressions ~= new AssignExpression(
|
||||
new DotVarExpression(new VariableExpression("__obj"), new VariableExpression("prototype")),
|
||||
new VariableExpression("__proto"));
|
||||
|
||||
auto classIdent = tokens.requireNextToken(ScriptToken.Type.identifier);
|
||||
|
||||
if(tokens.peekNextToken(ScriptToken.Type.symbol, ":")) {
|
||||
tokens.popFront();
|
||||
auto inheritFrom = tokens.requireNextToken(ScriptToken.Type.identifier);
|
||||
|
||||
expressions ~= new AssignExpression(
|
||||
new DotVarExpression(new VariableExpression("__proto"), new VariableExpression("prototype")),
|
||||
new VariableExpression(inheritFrom.str));
|
||||
}
|
||||
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, "{");
|
||||
|
||||
void addVarDecl(VariableDeclaration decl, string o) {
|
||||
foreach(i, ident; decl.identifiers) {
|
||||
// FIXME: make sure this goes on the instance, never the prototype!
|
||||
expressions ~= new AssignExpression(
|
||||
new DotVarExpression(
|
||||
new VariableExpression(o),
|
||||
new VariableExpression(ident),
|
||||
false),
|
||||
decl.initializers[i]
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// FIXME: we could actually add private vars and just put them in this scope. maybe
|
||||
|
||||
while(!tokens.peekNextToken(ScriptToken.Type.symbol, "}")) {
|
||||
if(tokens.peekNextToken(ScriptToken.Type.symbol, ";")) {
|
||||
tokens.popFront();
|
||||
continue;
|
||||
}
|
||||
|
||||
if(tokens.peekNextToken(ScriptToken.Type.identifier, "this")) {
|
||||
// ctor
|
||||
tokens.popFront();
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, "(");
|
||||
auto args = parseVariableDeclaration(tokens, ")");
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, ")");
|
||||
auto bod = parseExpression(tokens);
|
||||
|
||||
expressions ~= new AssignExpression(
|
||||
new DotVarExpression(
|
||||
new VariableExpression("__obj"),
|
||||
new VariableExpression("__ctor")),
|
||||
new FunctionLiteralExpression(args, bod));
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "var")) {
|
||||
// instance variable
|
||||
auto decl = parseVariableDeclaration(tokens, ";");
|
||||
addVarDecl(decl, "__obj");
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "static")) {
|
||||
// prototype var
|
||||
tokens.popFront();
|
||||
auto decl = parseVariableDeclaration(tokens, ";");
|
||||
addVarDecl(decl, "__proto");
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "function")) {
|
||||
// prototype function
|
||||
tokens.popFront();
|
||||
auto ident = tokens.requireNextToken(ScriptToken.Type.identifier);
|
||||
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, "(");
|
||||
auto args = parseVariableDeclaration(tokens, ")");
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, ")");
|
||||
auto bod = parseExpression(tokens);
|
||||
|
||||
expressions ~= new AssignExpression(
|
||||
new DotVarExpression(
|
||||
new VariableExpression("__proto"),
|
||||
new VariableExpression(ident.str),
|
||||
false),
|
||||
new FunctionLiteralExpression(args, bod));
|
||||
} else throw new ScriptCompileException("Unexpected " ~ tokens.front.str ~ " when reading class decl", tokens.front.lineNumber);
|
||||
}
|
||||
|
||||
tokens.requireNextToken(ScriptToken.Type.symbol, "}");
|
||||
|
||||
// returning he object from the scope...
|
||||
expressions ~= new VariableExpression("__obj");
|
||||
|
||||
auto scopeExpr = new ScopeExpression(expressions);
|
||||
auto classVarExpr = new VariableDeclaration();
|
||||
classVarExpr.identifiers = [classIdent.str];
|
||||
classVarExpr.initializers = [scopeExpr];
|
||||
|
||||
ret = classVarExpr;
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "if")) {
|
||||
tokens.popFront();
|
||||
auto e = new IfExpression();
|
||||
|
@ -1225,6 +1584,18 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
|||
e.loopBody = parseExpression(tokens);
|
||||
ret = e;
|
||||
expectedEnd = "";
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "break") || tokens.peekNextToken(ScriptToken.Type.keyword, "continue")) {
|
||||
auto token = tokens.front;
|
||||
tokens.popFront();
|
||||
ret = new LoopControlExpression(token.str);
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "return")) {
|
||||
tokens.popFront();
|
||||
Expression retVal;
|
||||
if(tokens.peekNextToken(ScriptToken.Type.symbol, ";"))
|
||||
retVal = new NullLiteralExpression();
|
||||
else
|
||||
retVal = parseExpression(tokens);
|
||||
ret = new ReturnExpression(retVal);
|
||||
} else if(tokens.peekNextToken(ScriptToken.Type.keyword, "throw")) {
|
||||
auto token = tokens.front;
|
||||
tokens.popFront();
|
||||
|
@ -1398,6 +1769,10 @@ Expression parseStatement(MyTokenStreamHere)(ref MyTokenStreamHere tokens, strin
|
|||
case "{":
|
||||
case "scope":
|
||||
|
||||
// classes
|
||||
case "class":
|
||||
case "new":
|
||||
|
||||
// flow control
|
||||
case "if":
|
||||
case "while":
|
||||
|
@ -1407,9 +1782,12 @@ Expression parseStatement(MyTokenStreamHere)(ref MyTokenStreamHere tokens, strin
|
|||
// exceptions
|
||||
case "try":
|
||||
case "throw":
|
||||
return parseExpression(tokens);
|
||||
case "return":
|
||||
|
||||
// flow
|
||||
case "continue":
|
||||
case "break":
|
||||
case "return":
|
||||
return parseExpression(tokens);
|
||||
// unary prefix operators
|
||||
case "!":
|
||||
case "~":
|
||||
|
@ -1539,11 +1917,13 @@ var interpretExpressions(ExpressionStream)(ExpressionStream expressions, Prototy
|
|||
|
||||
var interpretStream(MyTokenStreamHere)(MyTokenStreamHere tokens, PrototypeObject variables) if(is(ElementType!MyTokenStreamHere == ScriptToken)) {
|
||||
assert(variables !is null);
|
||||
// this is an entry point that all others lead to, right before getting to interpretExpressions...
|
||||
|
||||
return interpretExpressions(parseScript(tokens), variables);
|
||||
}
|
||||
|
||||
var interpretStream(MyTokenStreamHere)(MyTokenStreamHere tokens, var variables) if(is(ElementType!MyTokenStreamHere == ScriptToken)) {
|
||||
return interpretExpressions(parseScript(tokens),
|
||||
return interpretStream(tokens,
|
||||
(variables.payloadType() == var.Type.Object && variables._payload._object !is null) ? variables._payload._object : new PrototypeObject());
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue