mirror of https://github.com/adamdruppe/arsd.git
this is becoming not half bad
This commit is contained in:
parent
394c2ce938
commit
d0aec8e606
6
jsvar.d
6
jsvar.d
|
@ -795,6 +795,8 @@ struct var {
|
||||||
public var opBinary(string op, T)(T t) {
|
public var opBinary(string op, T)(T t) {
|
||||||
var n;
|
var n;
|
||||||
if(payloadType() == Type.Object) {
|
if(payloadType() == Type.Object) {
|
||||||
|
if(this._payload._object is null)
|
||||||
|
return var(null);
|
||||||
var* operator = this._payload._object._peekMember("opBinary", true);
|
var* operator = this._payload._object._peekMember("opBinary", true);
|
||||||
if(operator !is null && operator._type == Type.Function) {
|
if(operator !is null && operator._type == Type.Function) {
|
||||||
return operator.call(this, op, t);
|
return operator.call(this, op, t);
|
||||||
|
@ -1945,7 +1947,7 @@ var subclassable(T)() if(is(T == class) || is(T == interface)) {
|
||||||
//import std.stdio; writeln("calling ", T.stringof, ".", memberName, " - ", methodOverriddenByScript(memberName), "/", _next_devirtualized, " on ", cast(size_t) cast(void*) this);
|
//import std.stdio; writeln("calling ", T.stringof, ".", memberName, " - ", methodOverriddenByScript(memberName), "/", _next_devirtualized, " on ", cast(size_t) cast(void*) this);
|
||||||
if(_next_devirtualized || !methodOverriddenByScript(memberName))
|
if(_next_devirtualized || !methodOverriddenByScript(memberName))
|
||||||
return __traits(getMember, super, memberName)(p);
|
return __traits(getMember, super, memberName)(p);
|
||||||
return _this[memberName](p).get!(typeof(return));
|
return _this[memberName].call(_this, p).get!(typeof(return));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -2169,7 +2171,7 @@ WrappedNativeObject wrapNativeObject(Class, bool special = false)(Class obj) if(
|
||||||
|
|
||||||
static if(special) {
|
static if(special) {
|
||||||
Class obj;
|
Class obj;
|
||||||
if(vthis_.payloadType() != var.Type.Object) { import std.stdio; writeln("getwno on ", vthis_); }
|
//if(vthis_.payloadType() != var.Type.Object) { import std.stdio; writeln("getwno on ", vthis_); }
|
||||||
while(vthis_ != null) {
|
while(vthis_ != null) {
|
||||||
obj = vthis_.getWno!Class;
|
obj = vthis_.getWno!Class;
|
||||||
if(obj !is null)
|
if(obj !is null)
|
||||||
|
|
71
minigui.d
71
minigui.d
|
@ -953,7 +953,7 @@ void recomputeChildLayout(string relevantMeasure)(Widget parent) {
|
||||||
|
|
||||||
if(mixin("child." ~ relevantMeasure) >= maximum) {
|
if(mixin("child." ~ relevantMeasure) >= maximum) {
|
||||||
auto adj = mixin("child." ~ relevantMeasure) - maximum;
|
auto adj = mixin("child." ~ relevantMeasure) - maximum;
|
||||||
mixin("child." ~ relevantMeasure) -= adj;
|
mixin("child._" ~ relevantMeasure) -= adj;
|
||||||
spaceRemaining += adj;
|
spaceRemaining += adj;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -961,13 +961,13 @@ void recomputeChildLayout(string relevantMeasure)(Widget parent) {
|
||||||
if(s <= 0)
|
if(s <= 0)
|
||||||
continue;
|
continue;
|
||||||
auto spaceAdjustment = spacePerChild * (spreadEvenly ? 1 : s);
|
auto spaceAdjustment = spacePerChild * (spreadEvenly ? 1 : s);
|
||||||
mixin("child." ~ relevantMeasure) += spaceAdjustment;
|
mixin("child._" ~ relevantMeasure) += spaceAdjustment;
|
||||||
spaceRemaining -= spaceAdjustment;
|
spaceRemaining -= spaceAdjustment;
|
||||||
if(mixin("child." ~ relevantMeasure) > maximum) {
|
if(mixin("child." ~ relevantMeasure) > maximum) {
|
||||||
auto diff = mixin("child." ~ relevantMeasure) - maximum;
|
auto diff = mixin("child." ~ relevantMeasure) - maximum;
|
||||||
mixin("child." ~ relevantMeasure) -= diff;
|
mixin("child._" ~ relevantMeasure) -= diff;
|
||||||
spaceRemaining += diff;
|
spaceRemaining += diff;
|
||||||
} else if(mixin("child." ~ relevantMeasure) < maximum) {
|
} else if(mixin("child._" ~ relevantMeasure) < maximum) {
|
||||||
stretchinessSum += mixin("child." ~ relevantMeasure ~ "Stretchiness()");
|
stretchinessSum += mixin("child." ~ relevantMeasure ~ "Stretchiness()");
|
||||||
if(mostStretchy is null || s >= mostStretchyS) {
|
if(mostStretchy is null || s >= mostStretchyS) {
|
||||||
mostStretchy = child;
|
mostStretchy = child;
|
||||||
|
@ -985,11 +985,11 @@ void recomputeChildLayout(string relevantMeasure)(Widget parent) {
|
||||||
else
|
else
|
||||||
auto maximum = child.maxWidth();
|
auto maximum = child.maxWidth();
|
||||||
|
|
||||||
mixin("child." ~ relevantMeasure) += spaceAdjustment;
|
mixin("child._" ~ relevantMeasure) += spaceAdjustment;
|
||||||
spaceRemaining -= spaceAdjustment;
|
spaceRemaining -= spaceAdjustment;
|
||||||
if(mixin("child." ~ relevantMeasure) > maximum) {
|
if(mixin("child._" ~ relevantMeasure) > maximum) {
|
||||||
auto diff = mixin("child." ~ relevantMeasure) - maximum;
|
auto diff = mixin("child." ~ relevantMeasure) - maximum;
|
||||||
mixin("child." ~ relevantMeasure) -= diff;
|
mixin("child._" ~ relevantMeasure) -= diff;
|
||||||
spaceRemaining += diff;
|
spaceRemaining += diff;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1184,8 +1184,47 @@ struct WidgetPainter {
|
||||||
|
|
||||||
/++
|
/++
|
||||||
This is the list of rectangles that actually need to be redrawn.
|
This is the list of rectangles that actually need to be redrawn.
|
||||||
|
|
||||||
|
Not actually implemented yet.
|
||||||
+/
|
+/
|
||||||
Rectangle[] invalidatedRectangles;
|
Rectangle[] invalidatedRectangles;
|
||||||
|
|
||||||
|
|
||||||
|
// all this stuff is a dangerous experiment....
|
||||||
|
static class ScriptableVersion {
|
||||||
|
ScreenPainterImplementation* p;
|
||||||
|
int originX, originY;
|
||||||
|
|
||||||
|
@scriptable:
|
||||||
|
void drawRectangle(int x, int y, int width, int height) {
|
||||||
|
p.drawRectangle(x + originX, y + originY, width, height);
|
||||||
|
}
|
||||||
|
void drawLine(int x1, int y1, int x2, int y2) {
|
||||||
|
p.drawLine(x1 + originX, y1 + originY, x2 + originX, y2 + originY);
|
||||||
|
}
|
||||||
|
void drawText(int x, int y, string text) {
|
||||||
|
p.drawText(x + originX, y + originY, 100000, 100000, text, 0);
|
||||||
|
}
|
||||||
|
void setOutlineColor(int r, int g, int b) {
|
||||||
|
p.pen = Pen(Color(r,g,b), 1);
|
||||||
|
}
|
||||||
|
void setFillColor(int r, int g, int b) {
|
||||||
|
p.fillColor = Color(r,g,b);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ScriptableVersion toArsdJsvar() {
|
||||||
|
auto sv = new ScriptableVersion;
|
||||||
|
sv.p = this.screenPainter.impl;
|
||||||
|
sv.originX = this.screenPainter.originX;
|
||||||
|
sv.originY = this.screenPainter.originY;
|
||||||
|
return sv;
|
||||||
|
}
|
||||||
|
|
||||||
|
static WidgetPainter fromJsVar(T)(T t) {
|
||||||
|
return WidgetPainter.init;
|
||||||
|
}
|
||||||
|
// done..........
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1211,9 +1250,6 @@ class Widget {
|
||||||
event.sendDirectly();
|
event.sendDirectly();
|
||||||
}
|
}
|
||||||
|
|
||||||
deprecated("Change ScreenPainter to WidgetPainter")
|
|
||||||
final void paint(ScreenPainter) { assert(0, "Change ScreenPainter to WidgetPainter and recompile your code"); }
|
|
||||||
|
|
||||||
Menu contextMenu(int x, int y) { return null; }
|
Menu contextMenu(int x, int y) { return null; }
|
||||||
|
|
||||||
final bool showContextMenu(int x, int y, int screenX = -2, int screenY = -2) {
|
final bool showContextMenu(int x, int y, int screenX = -2, int screenY = -2) {
|
||||||
|
@ -1498,11 +1534,17 @@ class Widget {
|
||||||
|
|
||||||
int x; // relative to the parent's origin
|
int x; // relative to the parent's origin
|
||||||
int y; // relative to the parent's origin
|
int y; // relative to the parent's origin
|
||||||
int width;
|
int _width;
|
||||||
int height;
|
int _height;
|
||||||
Widget[] children;
|
Widget[] children;
|
||||||
Widget parent;
|
Widget parent;
|
||||||
|
|
||||||
|
public @property int width() { return _width; }
|
||||||
|
public @property int height() { return _height; }
|
||||||
|
|
||||||
|
protected @property int width(int a) { return _width = a; }
|
||||||
|
protected @property int height(int a) { return _height = a; }
|
||||||
|
|
||||||
protected
|
protected
|
||||||
void registerMovement() {
|
void registerMovement() {
|
||||||
version(win32_widgets) {
|
version(win32_widgets) {
|
||||||
|
@ -1642,6 +1684,9 @@ class Widget {
|
||||||
///
|
///
|
||||||
void paint(WidgetPainter painter) {}
|
void paint(WidgetPainter painter) {}
|
||||||
|
|
||||||
|
deprecated("Change ScreenPainter to WidgetPainter")
|
||||||
|
final void paint(ScreenPainter) { assert(0, "Change ScreenPainter to WidgetPainter and recompile your code"); }
|
||||||
|
|
||||||
/// I don't actually like the name of this
|
/// I don't actually like the name of this
|
||||||
/// this draws a background on it
|
/// this draws a background on it
|
||||||
void erase(WidgetPainter painter) {
|
void erase(WidgetPainter painter) {
|
||||||
|
@ -5813,7 +5858,9 @@ class Button : MouseActivatedWidget {
|
||||||
|
|
||||||
private string label_;
|
private string label_;
|
||||||
|
|
||||||
|
///
|
||||||
string label() { return label_; }
|
string label() { return label_; }
|
||||||
|
///
|
||||||
void label(string l) {
|
void label(string l) {
|
||||||
label_ = l;
|
label_ = l;
|
||||||
version(win32_widgets) {
|
version(win32_widgets) {
|
||||||
|
|
216
script.d
216
script.d
|
@ -19,6 +19,7 @@
|
||||||
and the => operator too
|
and the => operator too
|
||||||
|
|
||||||
I kinda like the javascript foo`blargh` template literals too.
|
I kinda like the javascript foo`blargh` template literals too.
|
||||||
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/++
|
/++
|
||||||
|
@ -124,7 +125,7 @@
|
||||||
* variables must start with A-Z, a-z, _, or $, then must be [A-Za-z0-9_]*.
|
* 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.).
|
(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.
|
Variable names that start with __ are reserved and you shouldn't use them.
|
||||||
* int, float, string, array, bool, and json!q{} literals
|
* int, float, string, array, bool, and `#{}` (previously known as `json!q{}` aka object) literals
|
||||||
* var.prototype, var.typeof. prototype works more like Mozilla's __proto__ than standard javascript prototype.
|
* var.prototype, var.typeof. prototype works more like Mozilla's __proto__ than standard javascript prototype.
|
||||||
* the `|>` pipeline operator
|
* the `|>` pipeline operator
|
||||||
* classes:
|
* classes:
|
||||||
|
@ -200,8 +201,6 @@ make sure superclass ctors are called
|
||||||
|
|
||||||
FIXME: add easy to use premade packages for the global object.
|
FIXME: add easy to use premade packages for the global object.
|
||||||
|
|
||||||
FIXME: maybe simplify the json!q{ } thing a bit.
|
|
||||||
|
|
||||||
FIXME: the debugger statement from javascript might be cool to throw in too.
|
FIXME: the debugger statement from javascript might be cool to throw in too.
|
||||||
|
|
||||||
FIXME: add continuations or something too - actually doing it with fibers works pretty well
|
FIXME: add continuations or something too - actually doing it with fibers works pretty well
|
||||||
|
@ -217,6 +216,8 @@ make sure superclass ctors are called
|
||||||
|
|
||||||
|
|
||||||
History:
|
History:
|
||||||
|
April 28, 2020: added `#{}` as an alternative to the `json!q{}` syntax for object literals. Also fixed unary `!` operator.
|
||||||
|
|
||||||
April 26, 2020: added `switch`, fixed precedence bug, fixed doc issues and added some unittests
|
April 26, 2020: added `switch`, fixed precedence bug, fixed doc issues and added some unittests
|
||||||
|
|
||||||
Started writing it in July 2013. Yes, a basic precedence issue was there for almost SEVEN YEARS. You can use this as a toy but please don't use it for anything too serious, it really is very poorly written and not intelligently designed at all.
|
Started writing it in July 2013. Yes, a basic precedence issue was there for almost SEVEN YEARS. You can use this as a toy but please don't use it for anything too serious, it really is very poorly written and not intelligently designed at all.
|
||||||
|
@ -454,33 +455,56 @@ class NonScriptCatchableException : Exception {
|
||||||
|
|
||||||
/// Thrown on script syntax errors and the sort.
|
/// Thrown on script syntax errors and the sort.
|
||||||
class ScriptCompileException : Exception {
|
class ScriptCompileException : Exception {
|
||||||
this(string msg, int lineNumber, string file = __FILE__, size_t line = __LINE__) {
|
string s;
|
||||||
|
int lineNumber;
|
||||||
|
this(string msg, string s, int lineNumber, string file = __FILE__, size_t line = __LINE__) {
|
||||||
|
this.s = s;
|
||||||
|
this.lineNumber = lineNumber;
|
||||||
super(to!string(lineNumber) ~ ": " ~ msg, file, line);
|
super(to!string(lineNumber) ~ ": " ~ msg, file, line);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Thrown on things like interpretation failures.
|
/// Thrown on things like interpretation failures.
|
||||||
class ScriptRuntimeException : Exception {
|
class ScriptRuntimeException : Exception {
|
||||||
this(string msg, int lineNumber, string file = __FILE__, size_t line = __LINE__) {
|
string s;
|
||||||
|
int lineNumber;
|
||||||
|
this(string msg, string s, int lineNumber, string file = __FILE__, size_t line = __LINE__) {
|
||||||
|
this.s = s;
|
||||||
|
this.lineNumber = lineNumber;
|
||||||
super(to!string(lineNumber) ~ ": " ~ msg, file, line);
|
super(to!string(lineNumber) ~ ": " ~ msg, file, line);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
|
struct ScriptLocation {
|
||||||
|
string scriptFilename; ///
|
||||||
|
int lineNumber; ///
|
||||||
|
}
|
||||||
|
|
||||||
/// This represents an exception thrown by `throw x;` inside the script as it is interpreted.
|
/// This represents an exception thrown by `throw x;` inside the script as it is interpreted.
|
||||||
class ScriptException : Exception {
|
class ScriptException : Exception {
|
||||||
///
|
///
|
||||||
var payload;
|
var payload;
|
||||||
///
|
///
|
||||||
int lineNumber;
|
ScriptLocation loc;
|
||||||
this(var payload, int lineNumber, string file = __FILE__, size_t line = __LINE__) {
|
///
|
||||||
|
ScriptLocation[] callStack;
|
||||||
|
this(var payload, ScriptLocation loc, string file = __FILE__, size_t line = __LINE__) {
|
||||||
this.payload = payload;
|
this.payload = payload;
|
||||||
this.lineNumber = lineNumber;
|
if(loc.scriptFilename.length == 0)
|
||||||
super("script@" ~ to!string(lineNumber) ~ ": " ~ to!string(payload), file, line);
|
loc.scriptFilename = "user_script";
|
||||||
|
this.loc = loc;
|
||||||
|
super(loc.scriptFilename ~ "@" ~ to!string(loc.lineNumber) ~ ": " ~ to!string(payload), file, line);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
override string toString() {
|
override string toString() {
|
||||||
return "script@" ~ to!string(lineNumber) ~ ": " ~ payload.get!string;
|
return loc.scriptFilename ~ "@" ~ to!string(loc.lineNumber) ~ ": " ~ payload.get!string ~ to!string(callStack);
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
// might be nice to take a D exception and put a script stack trace in there too......
|
||||||
|
// also need toString to show the callStack
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ScriptToken {
|
struct ScriptToken {
|
||||||
|
@ -508,10 +532,12 @@ private enum string[] keywords = [
|
||||||
"if", "do",
|
"if", "do",
|
||||||
];
|
];
|
||||||
private enum string[] symbols = [
|
private enum string[] symbols = [
|
||||||
|
">>>", // FIXME
|
||||||
"//", "/*", "/+",
|
"//", "/*", "/+",
|
||||||
"&&", "||",
|
"&&", "||",
|
||||||
"+=", "-=", "*=", "/=", "~=", "==", "<=", ">=","!=", "%=",
|
"+=", "-=", "*=", "/=", "~=", "==", "<=", ">=","!=", "%=",
|
||||||
"&=", "|=", "^=",
|
"&=", "|=", "^=",
|
||||||
|
"#{",
|
||||||
"..",
|
"..",
|
||||||
"<<", ">>", // FIXME
|
"<<", ">>", // FIXME
|
||||||
"|>",
|
"|>",
|
||||||
|
@ -736,7 +762,7 @@ class TokenStream(TextStream) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if(pos == text.length && (escaped || inInterpolate || !atEnd()))
|
if(pos == text.length && (escaped || inInterpolate || !atEnd()))
|
||||||
throw new ScriptCompileException("Unclosed string literal", token.lineNumber);
|
throw new ScriptCompileException("Unclosed string literal", token.scriptFilename, token.lineNumber);
|
||||||
|
|
||||||
if(mustCopy) {
|
if(mustCopy) {
|
||||||
// there must be something escaped in there, so we need
|
// there must be something escaped in there, so we need
|
||||||
|
@ -758,7 +784,7 @@ class TokenStream(TextStream) {
|
||||||
case '"': copy ~= "\""; break;
|
case '"': copy ~= "\""; break;
|
||||||
case '\'': copy ~= "'"; break;
|
case '\'': copy ~= "'"; break;
|
||||||
default:
|
default:
|
||||||
throw new ScriptCompileException("Unknown escape char " ~ cast(char) ch, token.lineNumber);
|
throw new ScriptCompileException("Unknown escape char " ~ cast(char) ch, token.scriptFilename, token.lineNumber);
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
} else if(ch == '\\') {
|
} else if(ch == '\\') {
|
||||||
|
@ -794,13 +820,32 @@ class TokenStream(TextStream) {
|
||||||
pos++;
|
pos++;
|
||||||
|
|
||||||
if(pos + 1 == text.length)
|
if(pos + 1 == text.length)
|
||||||
throw new ScriptCompileException("unclosed /* */ comment", lineNumber);
|
throw new ScriptCompileException("unclosed /* */ comment", token.scriptFilename, lineNumber);
|
||||||
|
|
||||||
advance(pos + 2);
|
advance(pos + 2);
|
||||||
continue mainLoop;
|
continue mainLoop;
|
||||||
|
|
||||||
} else if(symbol == "/+") {
|
} else if(symbol == "/+") {
|
||||||
// FIXME: nesting comment
|
int open = 0;
|
||||||
|
int pos = 0;
|
||||||
|
while(pos + 1 < text.length) {
|
||||||
|
if(text[pos..pos+2] == "/+") {
|
||||||
|
open++;
|
||||||
|
pos++;
|
||||||
|
} else if(text[pos..pos+2] == "+/") {
|
||||||
|
open--;
|
||||||
|
pos++;
|
||||||
|
if(open == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
pos++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(pos + 1 == text.length)
|
||||||
|
throw new ScriptCompileException("unclosed /+ +/ comment", token.scriptFilename, lineNumber);
|
||||||
|
|
||||||
|
advance(pos + 1);
|
||||||
|
continue mainLoop;
|
||||||
}
|
}
|
||||||
// FIXME: documentation comments
|
// FIXME: documentation comments
|
||||||
|
|
||||||
|
@ -813,7 +858,7 @@ class TokenStream(TextStream) {
|
||||||
|
|
||||||
if(!found) {
|
if(!found) {
|
||||||
// FIXME: make sure this gives a valid utf-8 sequence
|
// FIXME: make sure this gives a valid utf-8 sequence
|
||||||
throw new ScriptCompileException("unknown token " ~ text[0], lineNumber);
|
throw new ScriptCompileException("unknown token " ~ text[0], token.scriptFilename, lineNumber);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -990,7 +1035,7 @@ class StringLiteralExpression : Expression {
|
||||||
idx++;
|
idx++;
|
||||||
}
|
}
|
||||||
if(open != 0)
|
if(open != 0)
|
||||||
throw new ScriptRuntimeException("Unclosed interpolation thing", token.lineNumber);
|
throw new ScriptRuntimeException("Unclosed interpolation thing", token.scriptFilename, token.lineNumber);
|
||||||
auto code = c[0 .. idx];
|
auto code = c[0 .. idx];
|
||||||
|
|
||||||
var result = .interpret(code, sc);
|
var result = .interpret(code, sc);
|
||||||
|
@ -1071,6 +1116,28 @@ class NegationExpression : Expression {
|
||||||
return InterpretResult(-n, sc);
|
return InterpretResult(-n, sc);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
class NotExpression : Expression {
|
||||||
|
Expression e;
|
||||||
|
this(Expression e) { this.e = e;}
|
||||||
|
override string toString() { return "!" ~ e.toString(); }
|
||||||
|
|
||||||
|
override InterpretResult interpret(PrototypeObject sc) {
|
||||||
|
var n = e.interpret(sc).value;
|
||||||
|
return InterpretResult(var(!n), sc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
class BitFlipExpression : Expression {
|
||||||
|
Expression e;
|
||||||
|
this(Expression e) { this.e = e;}
|
||||||
|
override string toString() { return "~" ~ e.toString(); }
|
||||||
|
|
||||||
|
override InterpretResult interpret(PrototypeObject sc) {
|
||||||
|
var n = e.interpret(sc).value;
|
||||||
|
// possible FIXME given the size. but it is fuzzy when dynamic..
|
||||||
|
return InterpretResult(var(~(n.get!long)), sc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class ArrayLiteralExpression : Expression {
|
class ArrayLiteralExpression : Expression {
|
||||||
this() {}
|
this() {}
|
||||||
|
|
||||||
|
@ -1096,7 +1163,7 @@ class ObjectLiteralExpression : Expression {
|
||||||
Expression[string] elements;
|
Expression[string] elements;
|
||||||
|
|
||||||
override string toString() {
|
override string toString() {
|
||||||
string s = "json!q{";
|
string s = "#{";
|
||||||
bool first = true;
|
bool first = true;
|
||||||
foreach(k, e; elements) {
|
foreach(k, e; elements) {
|
||||||
if(first)
|
if(first)
|
||||||
|
@ -1300,7 +1367,7 @@ class BinaryExpression : Expression {
|
||||||
sw: switch(op) {
|
sw: switch(op) {
|
||||||
// I would actually kinda prefer this to be static foreach, but normal
|
// I would actually kinda prefer this to be static foreach, but normal
|
||||||
// tuple foreach here has broaded compiler compatibility.
|
// tuple foreach here has broaded compiler compatibility.
|
||||||
foreach(ctOp; CtList!("+", "-", "*", "/", "==", "!=", "<=", ">=", ">", "<", "~", "&&", "||", "&", "|", "^", "%"))
|
foreach(ctOp; CtList!("+", "-", "*", "/", "==", "!=", "<=", ">=", ">", "<", "~", "&&", "||", "&", "|", "^", "%")) //, ">>", "<<", ">>>")) // FIXME
|
||||||
case ctOp: {
|
case ctOp: {
|
||||||
n = mixin("left "~ctOp~" right");
|
n = mixin("left "~ctOp~" right");
|
||||||
break sw;
|
break sw;
|
||||||
|
@ -1332,7 +1399,7 @@ class OpAssignExpression : Expression {
|
||||||
|
|
||||||
auto v = cast(VariableExpression) e1;
|
auto v = cast(VariableExpression) e1;
|
||||||
if(v is null)
|
if(v is null)
|
||||||
throw new ScriptRuntimeException("not an lvalue", 0 /* FIXME */);
|
throw new ScriptRuntimeException("not an lvalue", null, 0 /* FIXME */);
|
||||||
|
|
||||||
var right = e2.interpret(sc).value;
|
var right = e2.interpret(sc).value;
|
||||||
|
|
||||||
|
@ -1354,16 +1421,18 @@ class PipelineExpression : Expression {
|
||||||
Expression e1;
|
Expression e1;
|
||||||
Expression e2;
|
Expression e2;
|
||||||
CallExpression ce;
|
CallExpression ce;
|
||||||
|
ScriptLocation loc;
|
||||||
|
|
||||||
this(Expression e1, Expression e2) {
|
this(ScriptLocation loc, Expression e1, Expression e2) {
|
||||||
|
this.loc = loc;
|
||||||
this.e1 = e1;
|
this.e1 = e1;
|
||||||
this.e2 = e2;
|
this.e2 = e2;
|
||||||
|
|
||||||
if(auto ce = cast(CallExpression) e2) {
|
if(auto ce = cast(CallExpression) e2) {
|
||||||
this.ce = new CallExpression(ce.func);
|
this.ce = new CallExpression(loc, ce.func);
|
||||||
this.ce.arguments = [e1] ~ ce.arguments;
|
this.ce.arguments = [e1] ~ ce.arguments;
|
||||||
} else {
|
} else {
|
||||||
this.ce = new CallExpression(e2);
|
this.ce = new CallExpression(loc, e2);
|
||||||
this.ce.arguments ~= e1;
|
this.ce.arguments ~= e1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1391,25 +1460,13 @@ class AssignExpression : Expression {
|
||||||
override InterpretResult interpret(PrototypeObject sc) {
|
override InterpretResult interpret(PrototypeObject sc) {
|
||||||
auto v = cast(VariableExpression) e1;
|
auto v = cast(VariableExpression) e1;
|
||||||
if(v is null)
|
if(v is null)
|
||||||
throw new ScriptRuntimeException("not an lvalue", 0 /* FIXME */);
|
throw new ScriptRuntimeException("not an lvalue", null, 0 /* FIXME */);
|
||||||
|
|
||||||
auto ret = v.setVar(sc, e2.interpret(sc).value, false, suppressOverloading);
|
auto ret = v.setVar(sc, e2.interpret(sc).value, false, suppressOverloading);
|
||||||
|
|
||||||
return InterpretResult(ret, sc);
|
return InterpretResult(ret, sc);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class UnaryExpression : Expression {
|
|
||||||
string op;
|
|
||||||
Expression e;
|
|
||||||
// FIXME
|
|
||||||
|
|
||||||
override InterpretResult interpret(PrototypeObject sc) {
|
|
||||||
return InterpretResult();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class VariableExpression : Expression {
|
class VariableExpression : Expression {
|
||||||
string identifier;
|
string identifier;
|
||||||
|
|
||||||
|
@ -2034,7 +2091,7 @@ class ShallowCopyExpression : Expression {
|
||||||
override InterpretResult interpret(PrototypeObject sc) {
|
override InterpretResult interpret(PrototypeObject sc) {
|
||||||
auto v = cast(VariableExpression) e1;
|
auto v = cast(VariableExpression) e1;
|
||||||
if(v is null)
|
if(v is null)
|
||||||
throw new ScriptRuntimeException("not an lvalue", 0 /* FIXME */);
|
throw new ScriptRuntimeException("not an lvalue", null, 0 /* FIXME */);
|
||||||
|
|
||||||
v.getVar(sc, false)._object.copyPropertiesFrom(e2.interpret(sc).value._object);
|
v.getVar(sc, false)._object.copyPropertiesFrom(e2.interpret(sc).value._object);
|
||||||
|
|
||||||
|
@ -2080,7 +2137,7 @@ class ThrowExpression : Expression {
|
||||||
|
|
||||||
override InterpretResult interpret(PrototypeObject sc) {
|
override InterpretResult interpret(PrototypeObject sc) {
|
||||||
assert(whatToThrow !is null);
|
assert(whatToThrow !is null);
|
||||||
throw new ScriptException(whatToThrow.interpret(sc).value, where.lineNumber);
|
throw new ScriptException(whatToThrow.interpret(sc).value, ScriptLocation(where.scriptFilename, where.lineNumber));
|
||||||
assert(0);
|
assert(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2177,6 +2234,7 @@ PrototypeObject DefaultArgumentDummyObject;
|
||||||
class CallExpression : Expression {
|
class CallExpression : Expression {
|
||||||
Expression func;
|
Expression func;
|
||||||
Expression[] arguments;
|
Expression[] arguments;
|
||||||
|
ScriptLocation loc;
|
||||||
|
|
||||||
override string toString() {
|
override string toString() {
|
||||||
string s = func.toString() ~ "(";
|
string s = func.toString() ~ "(";
|
||||||
|
@ -2189,7 +2247,8 @@ class CallExpression : Expression {
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
this(Expression func) {
|
this(ScriptLocation loc, Expression func) {
|
||||||
|
this.loc = loc;
|
||||||
this.func = func;
|
this.func = func;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2209,7 +2268,7 @@ class CallExpression : Expression {
|
||||||
if(!v)
|
if(!v)
|
||||||
throw new ScriptException(
|
throw new ScriptException(
|
||||||
var(this.toString() ~ " failed, got: " ~ assertExpression.toInterpretedString(sc)),
|
var(this.toString() ~ " failed, got: " ~ assertExpression.toInterpretedString(sc)),
|
||||||
asrt.token.lineNumber);
|
ScriptLocation(asrt.token.scriptFilename, asrt.token.lineNumber));
|
||||||
|
|
||||||
return InterpretResult(v, sc);
|
return InterpretResult(v, sc);
|
||||||
}
|
}
|
||||||
|
@ -2244,16 +2303,21 @@ class CallExpression : Expression {
|
||||||
_this = sc._getMember("this", true, true);
|
_this = sc._getMember("this", true, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
return InterpretResult(f.apply(_this, args), sc);
|
try {
|
||||||
|
return InterpretResult(f.apply(_this, args), sc);
|
||||||
|
} catch(ScriptException se) {
|
||||||
|
se.callStack ~= loc;
|
||||||
|
throw se;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ScriptToken requireNextToken(MyTokenStreamHere)(ref MyTokenStreamHere tokens, ScriptToken.Type type, string str = null, string file = __FILE__, size_t line = __LINE__) {
|
ScriptToken requireNextToken(MyTokenStreamHere)(ref MyTokenStreamHere tokens, ScriptToken.Type type, string str = null, string file = __FILE__, size_t line = __LINE__) {
|
||||||
if(tokens.empty)
|
if(tokens.empty)
|
||||||
throw new ScriptCompileException("script ended prematurely", 0, file, line);
|
throw new ScriptCompileException("script ended prematurely", null, 0, file, line);
|
||||||
auto next = tokens.front;
|
auto next = tokens.front;
|
||||||
if(next.type != type || (str !is null && next.str != str))
|
if(next.type != type || (str !is null && next.str != str))
|
||||||
throw new ScriptCompileException("unexpected '"~next.str~"' while expecting " ~ to!string(type) ~ " " ~ str, next.lineNumber, file, line);
|
throw new ScriptCompileException("unexpected '"~next.str~"' while expecting " ~ to!string(type) ~ " " ~ str, next.scriptFilename, next.lineNumber, file, line);
|
||||||
|
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
return next;
|
return next;
|
||||||
|
@ -2275,7 +2339,7 @@ VariableExpression parseVariableName(MyTokenStreamHere)(ref MyTokenStreamHere to
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
return new VariableExpression(token.str);
|
return new VariableExpression(token.str);
|
||||||
}
|
}
|
||||||
throw new ScriptCompileException("Found "~token.str~" when expecting identifier", token.lineNumber);
|
throw new ScriptCompileException("Found "~token.str~" when expecting identifier", token.scriptFilename, token.lineNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
|
@ -2295,13 +2359,17 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
}
|
}
|
||||||
else if(token.type == ScriptToken.Type.identifier)
|
else if(token.type == ScriptToken.Type.identifier)
|
||||||
e = parseVariableName(tokens);
|
e = parseVariableName(tokens);
|
||||||
else if(token.type == ScriptToken.Type.symbol && (token.str == "-" || token.str == "+")) {
|
else if(token.type == ScriptToken.Type.symbol && (token.str == "-" || token.str == "+" || token.str == "!" || token.str == "~")) {
|
||||||
auto op = token.str;
|
auto op = token.str;
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
|
|
||||||
e = parsePart(tokens);
|
e = parsePart(tokens);
|
||||||
if(op == "-")
|
if(op == "-")
|
||||||
e = new NegationExpression(e);
|
e = new NegationExpression(e);
|
||||||
|
else if(op == "!")
|
||||||
|
e = new NotExpression(e);
|
||||||
|
else if(op == "~")
|
||||||
|
e = new BitFlipExpression(e);
|
||||||
} else {
|
} else {
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
|
|
||||||
|
@ -2333,7 +2401,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
bool first = true;
|
bool first = true;
|
||||||
moreElements:
|
moreElements:
|
||||||
if(tokens.empty)
|
if(tokens.empty)
|
||||||
throw new ScriptCompileException("unexpected end of file when reading array literal", token.lineNumber);
|
throw new ScriptCompileException("unexpected end of file when reading array literal", token.scriptFilename, token.lineNumber);
|
||||||
|
|
||||||
auto peek = tokens.front;
|
auto peek = tokens.front;
|
||||||
if(peek.type == ScriptToken.Type.symbol && peek.str == "]") {
|
if(peek.type == ScriptToken.Type.symbol && peek.str == "]") {
|
||||||
|
@ -2350,6 +2418,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
|
|
||||||
goto moreElements;
|
goto moreElements;
|
||||||
case "json!q{":
|
case "json!q{":
|
||||||
|
case "#{":
|
||||||
// json object literal
|
// json object literal
|
||||||
auto obj = new ObjectLiteralExpression();
|
auto obj = new ObjectLiteralExpression();
|
||||||
/*
|
/*
|
||||||
|
@ -2365,7 +2434,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if(tokens.empty)
|
if(tokens.empty)
|
||||||
throw new ScriptCompileException("unexpected end of file when reading object literal", token.lineNumber);
|
throw new ScriptCompileException("unexpected end of file when reading object literal", token.scriptFilename, token.lineNumber);
|
||||||
|
|
||||||
moreKeys:
|
moreKeys:
|
||||||
auto key = tokens.front;
|
auto key = tokens.front;
|
||||||
|
@ -2376,7 +2445,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if(key.type != ScriptToken.Type.string && key.type != ScriptToken.Type.identifier) {
|
if(key.type != ScriptToken.Type.string && key.type != ScriptToken.Type.identifier) {
|
||||||
throw new ScriptCompileException("unexpected '"~key.str~"' when reading object literal", key.lineNumber);
|
throw new ScriptCompileException("unexpected '"~key.str~"' when reading object literal", key.scriptFilename, key.lineNumber);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2384,7 +2453,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
|
|
||||||
auto value = parseExpression(tokens);
|
auto value = parseExpression(tokens);
|
||||||
if(tokens.empty)
|
if(tokens.empty)
|
||||||
throw new ScriptCompileException("unclosed object literal", key.lineNumber);
|
throw new ScriptCompileException("unclosed object literal", key.scriptFilename, key.lineNumber);
|
||||||
|
|
||||||
if(tokens.peekNextToken(ScriptToken.Type.symbol, ","))
|
if(tokens.peekNextToken(ScriptToken.Type.symbol, ","))
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
|
@ -2421,12 +2490,10 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
unknown:
|
unknown:
|
||||||
throw new ScriptCompileException("unexpected '"~token.str~"' when reading ident", token.lineNumber);
|
throw new ScriptCompileException("unexpected '"~token.str~"' when reading ident", token.scriptFilename, token.lineNumber);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: unary ! doesn't work right
|
|
||||||
|
|
||||||
funcLoop: while(!tokens.empty) {
|
funcLoop: while(!tokens.empty) {
|
||||||
auto peek = tokens.front;
|
auto peek = tokens.front;
|
||||||
if(peek.type == ScriptToken.Type.symbol) {
|
if(peek.type == ScriptToken.Type.symbol) {
|
||||||
|
@ -2456,7 +2523,8 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
}
|
}
|
||||||
return e;
|
return e;
|
||||||
}
|
}
|
||||||
assert(0, to!string(tokens));
|
|
||||||
|
throw new ScriptCompileException("Ran out of tokens when trying to parsePart", null, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
Expression parseArguments(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expression exp, ref Expression[] where) {
|
Expression parseArguments(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expression exp, ref Expression[] where) {
|
||||||
|
@ -2477,7 +2545,7 @@ Expression parseArguments(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expre
|
||||||
}
|
}
|
||||||
|
|
||||||
if(tokens.empty)
|
if(tokens.empty)
|
||||||
throw new ScriptCompileException("unexpected end of file when parsing call expression", peek.lineNumber);
|
throw new ScriptCompileException("unexpected end of file when parsing call expression", peek.scriptFilename, peek.lineNumber);
|
||||||
peek = tokens.front;
|
peek = tokens.front;
|
||||||
if(peek.type == ScriptToken.Type.symbol && peek.str == ",") {
|
if(peek.type == ScriptToken.Type.symbol && peek.str == ",") {
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
|
@ -2486,17 +2554,17 @@ Expression parseArguments(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expre
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
return exp;
|
return exp;
|
||||||
} else
|
} else
|
||||||
throw new ScriptCompileException("unexpected '"~peek.str~"' when reading argument list", peek.lineNumber);
|
throw new ScriptCompileException("unexpected '"~peek.str~"' when reading argument list", peek.scriptFilename, peek.lineNumber);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Expression parseFunctionCall(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expression e) {
|
Expression parseFunctionCall(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expression e) {
|
||||||
assert(!tokens.empty);
|
assert(!tokens.empty);
|
||||||
auto peek = tokens.front;
|
auto peek = tokens.front;
|
||||||
auto exp = new CallExpression(e);
|
auto exp = new CallExpression(ScriptLocation(peek.scriptFilename, peek.lineNumber), e);
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
if(tokens.empty)
|
if(tokens.empty)
|
||||||
throw new ScriptCompileException("unexpected end of file when parsing call expression", peek.lineNumber);
|
throw new ScriptCompileException("unexpected end of file when parsing call expression", peek.scriptFilename, peek.lineNumber);
|
||||||
return parseArguments(tokens, exp, exp.arguments);
|
return parseArguments(tokens, exp, exp.arguments);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2540,7 +2608,7 @@ Expression parseAddend(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
|
|
||||||
case "|>":
|
case "|>":
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
e1 = new PipelineExpression(e1, parseFactor(tokens));
|
e1 = new PipelineExpression(ScriptLocation(peek.scriptFilename, peek.lineNumber), e1, parseFactor(tokens));
|
||||||
break;
|
break;
|
||||||
case ".":
|
case ".":
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
|
@ -2595,12 +2663,12 @@ Expression parseAddend(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
return new OpAssignExpression(peek.str[0..1], e1, parseExpression(tokens));
|
return new OpAssignExpression(peek.str[0..1], e1, parseExpression(tokens));
|
||||||
default:
|
default:
|
||||||
throw new ScriptCompileException("Parse error, unexpected " ~ peek.str ~ " when looking for operator", peek.lineNumber);
|
throw new ScriptCompileException("Parse error, unexpected " ~ peek.str ~ " when looking for operator", peek.scriptFilename, peek.lineNumber);
|
||||||
}
|
}
|
||||||
//} else if(peek.type == ScriptToken.Type.identifier || peek.type == ScriptToken.Type.number) {
|
//} else if(peek.type == ScriptToken.Type.identifier || peek.type == ScriptToken.Type.number) {
|
||||||
//return parseFactor(tokens);
|
//return parseFactor(tokens);
|
||||||
} else
|
} else
|
||||||
throw new ScriptCompileException("Parse error, unexpected '" ~ peek.str ~ "'", peek.lineNumber);
|
throw new ScriptCompileException("Parse error, unexpected '" ~ peek.str ~ "'", peek.scriptFilename, peek.lineNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
return e1;
|
return e1;
|
||||||
|
@ -2635,7 +2703,7 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool
|
||||||
case "exit":
|
case "exit":
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new ScriptCompileException("unexpected " ~ ident.str ~ ". valid scope(idents) are success, failure, and exit", ident.lineNumber);
|
throw new ScriptCompileException("unexpected " ~ ident.str ~ ". valid scope(idents) are success, failure, and exit", ident.scriptFilename, ident.lineNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
tokens.requireNextToken(ScriptToken.Type.symbol, ")");
|
tokens.requireNextToken(ScriptToken.Type.symbol, ")");
|
||||||
|
@ -2790,7 +2858,7 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool
|
||||||
new VariableExpression(ident.str),
|
new VariableExpression(ident.str),
|
||||||
false),
|
false),
|
||||||
new FunctionLiteralExpression(args, bod, staticScopeBacking));
|
new FunctionLiteralExpression(args, bod, staticScopeBacking));
|
||||||
} else throw new ScriptCompileException("Unexpected " ~ tokens.front.str ~ " when reading class decl", tokens.front.lineNumber);
|
} else throw new ScriptCompileException("Unexpected " ~ tokens.front.str ~ " when reading class decl", tokens.front.scriptFilename, tokens.front.lineNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
tokens.requireNextToken(ScriptToken.Type.symbol, "}");
|
tokens.requireNextToken(ScriptToken.Type.symbol, "}");
|
||||||
|
@ -2854,7 +2922,7 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool
|
||||||
|
|
||||||
e.cases ~= c;
|
e.cases ~= c;
|
||||||
e.default_ = c;
|
e.default_ = c;
|
||||||
} else throw new ScriptCompileException("A switch statement must consists of cases and a default, nothing else ", tokens.front.lineNumber);
|
} else throw new ScriptCompileException("A switch statement must consists of cases and a default, nothing else ", tokens.front.scriptFilename, tokens.front.lineNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
tokens.requireNextToken(ScriptToken.Type.symbol, "}");
|
tokens.requireNextToken(ScriptToken.Type.symbol, "}");
|
||||||
|
@ -2938,7 +3006,7 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool
|
||||||
bool hadSomething = false;
|
bool hadSomething = false;
|
||||||
while(tokens.peekNextToken(ScriptToken.Type.keyword, "catch")) {
|
while(tokens.peekNextToken(ScriptToken.Type.keyword, "catch")) {
|
||||||
if(hadSomething)
|
if(hadSomething)
|
||||||
throw new ScriptCompileException("Only one catch block is allowed currently ", tokens.front.lineNumber);
|
throw new ScriptCompileException("Only one catch block is allowed currently ", tokens.front.scriptFilename, tokens.front.lineNumber);
|
||||||
hadSomething = true;
|
hadSomething = true;
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
tokens.requireNextToken(ScriptToken.Type.symbol, "(");
|
tokens.requireNextToken(ScriptToken.Type.symbol, "(");
|
||||||
|
@ -2964,13 +3032,13 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool
|
||||||
} else {
|
} else {
|
||||||
//assert(0);
|
//assert(0);
|
||||||
// return null;
|
// return null;
|
||||||
throw new ScriptCompileException("Parse error, unexpected end of input when reading expression", 0);//token.lineNumber);
|
throw new ScriptCompileException("Parse error, unexpected end of input when reading expression", null, 0);//token.lineNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
//writeln("parsed expression ", ret.toString());
|
//writeln("parsed expression ", ret.toString());
|
||||||
|
|
||||||
if(expectedEnd.length && tokens.empty && consumeEnd) // going loose on final ; at the end of input for repl convenience
|
if(expectedEnd.length && tokens.empty && consumeEnd) // going loose on final ; at the end of input for repl convenience
|
||||||
throw new ScriptCompileException("Parse error, unexpected end of input when reading expression, expecting " ~ expectedEnd, first.lineNumber);
|
throw new ScriptCompileException("Parse error, unexpected end of input when reading expression, expecting " ~ expectedEnd, first.scriptFilename, first.lineNumber);
|
||||||
|
|
||||||
if(expectedEnd.length && consumeEnd) {
|
if(expectedEnd.length && consumeEnd) {
|
||||||
if(tokens.peekNextToken(ScriptToken.Type.symbol, expectedEnd))
|
if(tokens.peekNextToken(ScriptToken.Type.symbol, expectedEnd))
|
||||||
|
@ -2998,24 +3066,24 @@ VariableDeclaration parseVariableDeclaration(MyTokenStreamHere)(ref MyTokenStrea
|
||||||
|
|
||||||
equalOk= true;
|
equalOk= true;
|
||||||
if(tokens.empty)
|
if(tokens.empty)
|
||||||
throw new ScriptCompileException("Parse error, dangling var at end of file", firstToken.lineNumber);
|
throw new ScriptCompileException("Parse error, dangling var at end of file", firstToken.scriptFilename, firstToken.lineNumber);
|
||||||
|
|
||||||
Expression initializer;
|
Expression initializer;
|
||||||
auto identifier = tokens.front;
|
auto identifier = tokens.front;
|
||||||
if(identifier.type != ScriptToken.Type.identifier)
|
if(identifier.type != ScriptToken.Type.identifier)
|
||||||
throw new ScriptCompileException("Parse error, found '"~identifier.str~"' when expecting var identifier", identifier.lineNumber);
|
throw new ScriptCompileException("Parse error, found '"~identifier.str~"' when expecting var identifier", identifier.scriptFilename, identifier.lineNumber);
|
||||||
|
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
|
|
||||||
tryTermination:
|
tryTermination:
|
||||||
if(tokens.empty)
|
if(tokens.empty)
|
||||||
throw new ScriptCompileException("Parse error, missing ; after var declaration at end of file", firstToken.lineNumber);
|
throw new ScriptCompileException("Parse error, missing ; after var declaration at end of file", firstToken.scriptFilename, firstToken.lineNumber);
|
||||||
|
|
||||||
auto peek = tokens.front;
|
auto peek = tokens.front;
|
||||||
if(peek.type == ScriptToken.Type.symbol) {
|
if(peek.type == ScriptToken.Type.symbol) {
|
||||||
if(peek.str == "=") {
|
if(peek.str == "=") {
|
||||||
if(!equalOk)
|
if(!equalOk)
|
||||||
throw new ScriptCompileException("Parse error, unexpected '"~identifier.str~"' after reading var initializer", peek.lineNumber);
|
throw new ScriptCompileException("Parse error, unexpected '"~identifier.str~"' after reading var initializer", peek.scriptFilename, peek.lineNumber);
|
||||||
equalOk = false;
|
equalOk = false;
|
||||||
tokens.popFront();
|
tokens.popFront();
|
||||||
initializer = parseExpression(tokens);
|
initializer = parseExpression(tokens);
|
||||||
|
@ -3031,9 +3099,9 @@ VariableDeclaration parseVariableDeclaration(MyTokenStreamHere)(ref MyTokenStrea
|
||||||
//tokens = tokens[1 .. $];
|
//tokens = tokens[1 .. $];
|
||||||
// we're done!
|
// we're done!
|
||||||
} else
|
} else
|
||||||
throw new ScriptCompileException("Parse error, unexpected '"~peek.str~"' when reading var declaration", peek.lineNumber);
|
throw new ScriptCompileException("Parse error, unexpected '"~peek.str~"' when reading var declaration", peek.scriptFilename, peek.lineNumber);
|
||||||
} else
|
} else
|
||||||
throw new ScriptCompileException("Parse error, unexpected '"~peek.str~"' when reading var declaration", peek.lineNumber);
|
throw new ScriptCompileException("Parse error, unexpected '"~peek.str~"' when reading var declaration", peek.scriptFilename, peek.lineNumber);
|
||||||
|
|
||||||
return decl;
|
return decl;
|
||||||
}
|
}
|
||||||
|
@ -3103,6 +3171,7 @@ Expression parseStatement(MyTokenStreamHere)(ref MyTokenStreamHere tokens, strin
|
||||||
goto case; // handle it like any other expression
|
goto case; // handle it like any other expression
|
||||||
}
|
}
|
||||||
case "json!{":
|
case "json!{":
|
||||||
|
case "#{":
|
||||||
case "[":
|
case "[":
|
||||||
case "(":
|
case "(":
|
||||||
case "null":
|
case "null":
|
||||||
|
@ -3143,13 +3212,14 @@ Expression parseStatement(MyTokenStreamHere)(ref MyTokenStreamHere tokens, strin
|
||||||
case "!":
|
case "!":
|
||||||
case "~":
|
case "~":
|
||||||
case "-":
|
case "-":
|
||||||
|
return parseExpression(tokens);
|
||||||
|
|
||||||
// BTW add custom object operator overloading to struct var
|
// BTW add custom object operator overloading to struct var
|
||||||
// and custom property overloading to PrototypeObject
|
// and custom property overloading to PrototypeObject
|
||||||
|
|
||||||
default:
|
default:
|
||||||
// whatever else keyword or operator related is actually illegal here
|
// whatever else keyword or operator related is actually illegal here
|
||||||
throw new ScriptCompileException("Parse error, unexpected " ~ token.str, token.lineNumber);
|
throw new ScriptCompileException("Parse error, unexpected " ~ token.str, token.scriptFilename, token.lineNumber);
|
||||||
}
|
}
|
||||||
// break;
|
// break;
|
||||||
case ScriptToken.Type.identifier:
|
case ScriptToken.Type.identifier:
|
||||||
|
@ -3196,7 +3266,7 @@ struct CompoundStatementRange(MyTokenStreamHere) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if(tokens.empty && terminatingSymbol !is null) {
|
if(tokens.empty && terminatingSymbol !is null) {
|
||||||
throw new ScriptCompileException("Reached end of file while trying to reach matching " ~ terminatingSymbol, startingLine);
|
throw new ScriptCompileException("Reached end of file while trying to reach matching " ~ terminatingSymbol, null, startingLine);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(terminatingSymbol !is null) {
|
if(terminatingSymbol !is null) {
|
||||||
|
|
|
@ -6574,6 +6574,7 @@ struct ScreenPainter {
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
|
@scriptable
|
||||||
@property void outlineColor(Color c) {
|
@property void outlineColor(Color c) {
|
||||||
if(impl is null) return;
|
if(impl is null) return;
|
||||||
if(activePen.color == c)
|
if(activePen.color == c)
|
||||||
|
@ -6583,6 +6584,7 @@ struct ScreenPainter {
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
|
@scriptable
|
||||||
@property void fillColor(Color c) {
|
@property void fillColor(Color c) {
|
||||||
if(impl is null) return;
|
if(impl is null) return;
|
||||||
impl.fillColor(c);
|
impl.fillColor(c);
|
||||||
|
@ -6662,6 +6664,7 @@ struct ScreenPainter {
|
||||||
}
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
|
@scriptable
|
||||||
void drawText(Point upperLeft, in char[] text, Point lowerRight = Point(0, 0), uint alignment = 0) {
|
void drawText(Point upperLeft, in char[] text, Point lowerRight = Point(0, 0), uint alignment = 0) {
|
||||||
if(impl is null) return;
|
if(impl is null) return;
|
||||||
if(lowerRight.x != 0 || lowerRight.y != 0) {
|
if(lowerRight.x != 0 || lowerRight.y != 0) {
|
||||||
|
@ -6734,6 +6737,7 @@ struct ScreenPainter {
|
||||||
|
|
||||||
|
|
||||||
/// Draws a pen using the current pen / outlineColor
|
/// Draws a pen using the current pen / outlineColor
|
||||||
|
@scriptable
|
||||||
void drawLine(Point starting, Point ending) {
|
void drawLine(Point starting, Point ending) {
|
||||||
if(impl is null) return;
|
if(impl is null) return;
|
||||||
if(isClipped(starting, ending)) return;
|
if(isClipped(starting, ending)) return;
|
||||||
|
@ -6745,6 +6749,7 @@ struct ScreenPainter {
|
||||||
/// Draws a rectangle using the current pen/outline color for the border and brush/fill color for the insides
|
/// Draws a rectangle using the current pen/outline color for the border and brush/fill color for the insides
|
||||||
/// The outer lines, inclusive of x = 0, y = 0, x = width - 1, and y = height - 1 are drawn with the outlineColor
|
/// The outer lines, inclusive of x = 0, y = 0, x = width - 1, and y = height - 1 are drawn with the outlineColor
|
||||||
/// The rest of the pixels are drawn with the fillColor. If fillColor is transparent, those pixels are not drawn.
|
/// The rest of the pixels are drawn with the fillColor. If fillColor is transparent, those pixels are not drawn.
|
||||||
|
@scriptable
|
||||||
void drawRectangle(Point upperLeft, int width, int height) {
|
void drawRectangle(Point upperLeft, int width, int height) {
|
||||||
if(impl is null) return;
|
if(impl is null) return;
|
||||||
if(isClipped(upperLeft, width, height)) return;
|
if(isClipped(upperLeft, width, height)) return;
|
||||||
|
|
Loading…
Reference in New Issue