From d0aec8e606a90c005b9cac6fcfb2047fb61b38fa Mon Sep 17 00:00:00 2001 From: "Adam D. Ruppe" Date: Tue, 28 Apr 2020 22:53:25 -0400 Subject: [PATCH] this is becoming not half bad --- jsvar.d | 6 +- minigui.d | 71 +++++++++++++--- script.d | 216 ++++++++++++++++++++++++++++++++---------------- simpledisplay.d | 5 ++ 4 files changed, 211 insertions(+), 87 deletions(-) diff --git a/jsvar.d b/jsvar.d index 2b6639b..6092cf3 100644 --- a/jsvar.d +++ b/jsvar.d @@ -795,6 +795,8 @@ struct var { public var opBinary(string op, T)(T t) { var n; if(payloadType() == Type.Object) { + if(this._payload._object is null) + return var(null); var* operator = this._payload._object._peekMember("opBinary", true); if(operator !is null && operator._type == Type.Function) { 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); if(_next_devirtualized || !methodOverriddenByScript(memberName)) 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) { 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) { obj = vthis_.getWno!Class; if(obj !is null) diff --git a/minigui.d b/minigui.d index 8846348..ff6fb4f 100644 --- a/minigui.d +++ b/minigui.d @@ -953,7 +953,7 @@ void recomputeChildLayout(string relevantMeasure)(Widget parent) { if(mixin("child." ~ relevantMeasure) >= maximum) { auto adj = mixin("child." ~ relevantMeasure) - maximum; - mixin("child." ~ relevantMeasure) -= adj; + mixin("child._" ~ relevantMeasure) -= adj; spaceRemaining += adj; continue; } @@ -961,13 +961,13 @@ void recomputeChildLayout(string relevantMeasure)(Widget parent) { if(s <= 0) continue; auto spaceAdjustment = spacePerChild * (spreadEvenly ? 1 : s); - mixin("child." ~ relevantMeasure) += spaceAdjustment; + mixin("child._" ~ relevantMeasure) += spaceAdjustment; spaceRemaining -= spaceAdjustment; if(mixin("child." ~ relevantMeasure) > maximum) { auto diff = mixin("child." ~ relevantMeasure) - maximum; - mixin("child." ~ relevantMeasure) -= diff; + mixin("child._" ~ relevantMeasure) -= diff; spaceRemaining += diff; - } else if(mixin("child." ~ relevantMeasure) < maximum) { + } else if(mixin("child._" ~ relevantMeasure) < maximum) { stretchinessSum += mixin("child." ~ relevantMeasure ~ "Stretchiness()"); if(mostStretchy is null || s >= mostStretchyS) { mostStretchy = child; @@ -985,11 +985,11 @@ void recomputeChildLayout(string relevantMeasure)(Widget parent) { else auto maximum = child.maxWidth(); - mixin("child." ~ relevantMeasure) += spaceAdjustment; + mixin("child._" ~ relevantMeasure) += spaceAdjustment; spaceRemaining -= spaceAdjustment; - if(mixin("child." ~ relevantMeasure) > maximum) { + if(mixin("child._" ~ relevantMeasure) > maximum) { auto diff = mixin("child." ~ relevantMeasure) - maximum; - mixin("child." ~ relevantMeasure) -= diff; + mixin("child._" ~ relevantMeasure) -= diff; spaceRemaining += diff; } } @@ -1184,8 +1184,47 @@ struct WidgetPainter { /++ This is the list of rectangles that actually need to be redrawn. + + Not actually implemented yet. +/ 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(); } - 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; } 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 y; // relative to the parent's origin - int width; - int height; + int _width; + int _height; Widget[] children; 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 void registerMovement() { version(win32_widgets) { @@ -1642,6 +1684,9 @@ class Widget { /// 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 /// this draws a background on it void erase(WidgetPainter painter) { @@ -5813,7 +5858,9 @@ class Button : MouseActivatedWidget { private string label_; + /// string label() { return label_; } + /// void label(string l) { label_ = l; version(win32_widgets) { diff --git a/script.d b/script.d index 3f9c4eb..2827060 100644 --- a/script.d +++ b/script.d @@ -19,6 +19,7 @@ and the => operator 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_]*. (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 + * 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. * the `|>` pipeline operator * classes: @@ -200,8 +201,6 @@ make sure superclass ctors are called 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: add continuations or something too - actually doing it with fibers works pretty well @@ -217,6 +216,8 @@ make sure superclass ctors are called 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 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. 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); } } /// Thrown on things like interpretation failures. 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); } } +/// +struct ScriptLocation { + string scriptFilename; /// + int lineNumber; /// +} + /// This represents an exception thrown by `throw x;` inside the script as it is interpreted. class ScriptException : Exception { /// var payload; /// - int lineNumber; - this(var payload, int lineNumber, string file = __FILE__, size_t line = __LINE__) { + ScriptLocation loc; + /// + ScriptLocation[] callStack; + this(var payload, ScriptLocation loc, string file = __FILE__, size_t line = __LINE__) { this.payload = payload; - this.lineNumber = lineNumber; - super("script@" ~ to!string(lineNumber) ~ ": " ~ to!string(payload), file, line); + if(loc.scriptFilename.length == 0) + loc.scriptFilename = "user_script"; + this.loc = loc; + super(loc.scriptFilename ~ "@" ~ to!string(loc.lineNumber) ~ ": " ~ to!string(payload), file, line); } + /* 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 { @@ -508,10 +532,12 @@ private enum string[] keywords = [ "if", "do", ]; private enum string[] symbols = [ + ">>>", // FIXME "//", "/*", "/+", "&&", "||", "+=", "-=", "*=", "/=", "~=", "==", "<=", ">=","!=", "%=", "&=", "|=", "^=", + "#{", "..", "<<", ">>", // FIXME "|>", @@ -736,7 +762,7 @@ class TokenStream(TextStream) { } 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) { // there must be something escaped in there, so we need @@ -758,7 +784,7 @@ class TokenStream(TextStream) { case '"': copy ~= "\""; break; case '\'': copy ~= "'"; break; 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; } else if(ch == '\\') { @@ -794,13 +820,32 @@ class TokenStream(TextStream) { pos++; if(pos + 1 == text.length) - throw new ScriptCompileException("unclosed /* */ comment", lineNumber); + throw new ScriptCompileException("unclosed /* */ comment", token.scriptFilename, lineNumber); advance(pos + 2); continue mainLoop; } 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 @@ -813,7 +858,7 @@ class TokenStream(TextStream) { if(!found) { // 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++; } 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]; var result = .interpret(code, sc); @@ -1071,6 +1116,28 @@ class NegationExpression : Expression { 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 { this() {} @@ -1096,7 +1163,7 @@ class ObjectLiteralExpression : Expression { Expression[string] elements; override string toString() { - string s = "json!q{"; + string s = "#{"; bool first = true; foreach(k, e; elements) { if(first) @@ -1300,7 +1367,7 @@ class BinaryExpression : Expression { sw: switch(op) { // I would actually kinda prefer this to be static foreach, but normal // tuple foreach here has broaded compiler compatibility. - foreach(ctOp; CtList!("+", "-", "*", "/", "==", "!=", "<=", ">=", ">", "<", "~", "&&", "||", "&", "|", "^", "%")) + foreach(ctOp; CtList!("+", "-", "*", "/", "==", "!=", "<=", ">=", ">", "<", "~", "&&", "||", "&", "|", "^", "%")) //, ">>", "<<", ">>>")) // FIXME case ctOp: { n = mixin("left "~ctOp~" right"); break sw; @@ -1332,7 +1399,7 @@ class OpAssignExpression : Expression { auto v = cast(VariableExpression) e1; 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; @@ -1354,16 +1421,18 @@ class PipelineExpression : Expression { Expression e1; Expression e2; CallExpression ce; + ScriptLocation loc; - this(Expression e1, Expression e2) { + this(ScriptLocation loc, Expression e1, Expression e2) { + this.loc = loc; this.e1 = e1; this.e2 = 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; } else { - this.ce = new CallExpression(e2); + this.ce = new CallExpression(loc, e2); this.ce.arguments ~= e1; } } @@ -1391,25 +1460,13 @@ class AssignExpression : Expression { override InterpretResult interpret(PrototypeObject sc) { auto v = cast(VariableExpression) e1; 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); return InterpretResult(ret, sc); } } - - -class UnaryExpression : Expression { - string op; - Expression e; - // FIXME - - override InterpretResult interpret(PrototypeObject sc) { - return InterpretResult(); - } -} - class VariableExpression : Expression { string identifier; @@ -2034,7 +2091,7 @@ class ShallowCopyExpression : Expression { override InterpretResult interpret(PrototypeObject sc) { auto v = cast(VariableExpression) e1; 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); @@ -2080,7 +2137,7 @@ class ThrowExpression : Expression { override InterpretResult interpret(PrototypeObject sc) { 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); } } @@ -2177,6 +2234,7 @@ PrototypeObject DefaultArgumentDummyObject; class CallExpression : Expression { Expression func; Expression[] arguments; + ScriptLocation loc; override string toString() { string s = func.toString() ~ "("; @@ -2189,7 +2247,8 @@ class CallExpression : Expression { return s; } - this(Expression func) { + this(ScriptLocation loc, Expression func) { + this.loc = loc; this.func = func; } @@ -2209,7 +2268,7 @@ class CallExpression : Expression { if(!v) throw new ScriptException( var(this.toString() ~ " failed, got: " ~ assertExpression.toInterpretedString(sc)), - asrt.token.lineNumber); + ScriptLocation(asrt.token.scriptFilename, asrt.token.lineNumber)); return InterpretResult(v, sc); } @@ -2244,16 +2303,21 @@ class CallExpression : Expression { _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__) { 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; 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(); return next; @@ -2275,7 +2339,7 @@ VariableExpression parseVariableName(MyTokenStreamHere)(ref MyTokenStreamHere to tokens.popFront(); 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) { @@ -2295,13 +2359,17 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { } else if(token.type == ScriptToken.Type.identifier) 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; tokens.popFront(); e = parsePart(tokens); if(op == "-") e = new NegationExpression(e); + else if(op == "!") + e = new NotExpression(e); + else if(op == "~") + e = new BitFlipExpression(e); } else { tokens.popFront(); @@ -2333,7 +2401,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { bool first = true; moreElements: 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; if(peek.type == ScriptToken.Type.symbol && peek.str == "]") { @@ -2350,6 +2418,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { goto moreElements; case "json!q{": + case "#{": // json object literal auto obj = new ObjectLiteralExpression(); /* @@ -2365,7 +2434,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { */ 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: auto key = tokens.front; @@ -2376,7 +2445,7 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { break; } 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); 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, ",")) tokens.popFront(); @@ -2421,12 +2490,10 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { } } else { 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) { auto peek = tokens.front; if(peek.type == ScriptToken.Type.symbol) { @@ -2456,7 +2523,8 @@ Expression parsePart(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { } 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) { @@ -2477,7 +2545,7 @@ Expression parseArguments(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expre } 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; if(peek.type == ScriptToken.Type.symbol && peek.str == ",") { tokens.popFront(); @@ -2486,17 +2554,17 @@ Expression parseArguments(MyTokenStreamHere)(ref MyTokenStreamHere tokens, Expre tokens.popFront(); return exp; } 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) { assert(!tokens.empty); auto peek = tokens.front; - auto exp = new CallExpression(e); + auto exp = new CallExpression(ScriptLocation(peek.scriptFilename, peek.lineNumber), e); tokens.popFront(); 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); } @@ -2540,7 +2608,7 @@ Expression parseAddend(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { case "|>": tokens.popFront(); - e1 = new PipelineExpression(e1, parseFactor(tokens)); + e1 = new PipelineExpression(ScriptLocation(peek.scriptFilename, peek.lineNumber), e1, parseFactor(tokens)); break; case ".": tokens.popFront(); @@ -2595,12 +2663,12 @@ Expression parseAddend(MyTokenStreamHere)(ref MyTokenStreamHere tokens) { tokens.popFront(); return new OpAssignExpression(peek.str[0..1], e1, parseExpression(tokens)); 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) { //return parseFactor(tokens); } 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; @@ -2635,7 +2703,7 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool case "exit": break; 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, ")"); @@ -2790,7 +2858,7 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool new VariableExpression(ident.str), false), 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, "}"); @@ -2854,7 +2922,7 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool e.cases ~= 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, "}"); @@ -2938,7 +3006,7 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool bool hadSomething = false; while(tokens.peekNextToken(ScriptToken.Type.keyword, "catch")) { 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; tokens.popFront(); tokens.requireNextToken(ScriptToken.Type.symbol, "("); @@ -2964,13 +3032,13 @@ Expression parseExpression(MyTokenStreamHere)(ref MyTokenStreamHere tokens, bool } else { //assert(0); // 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()); 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(tokens.peekNextToken(ScriptToken.Type.symbol, expectedEnd)) @@ -2998,24 +3066,24 @@ VariableDeclaration parseVariableDeclaration(MyTokenStreamHere)(ref MyTokenStrea equalOk= true; 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; auto identifier = tokens.front; 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(); tryTermination: 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; if(peek.type == ScriptToken.Type.symbol) { if(peek.str == "=") { 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; tokens.popFront(); initializer = parseExpression(tokens); @@ -3031,9 +3099,9 @@ VariableDeclaration parseVariableDeclaration(MyTokenStreamHere)(ref MyTokenStrea //tokens = tokens[1 .. $]; // we're done! } 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 - 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; } @@ -3103,6 +3171,7 @@ Expression parseStatement(MyTokenStreamHere)(ref MyTokenStreamHere tokens, strin goto case; // handle it like any other expression } case "json!{": + case "#{": case "[": case "(": case "null": @@ -3143,13 +3212,14 @@ Expression parseStatement(MyTokenStreamHere)(ref MyTokenStreamHere tokens, strin case "!": case "~": case "-": + return parseExpression(tokens); // BTW add custom object operator overloading to struct var // and custom property overloading to PrototypeObject default: // 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; case ScriptToken.Type.identifier: @@ -3196,7 +3266,7 @@ struct CompoundStatementRange(MyTokenStreamHere) { } 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) { diff --git a/simpledisplay.d b/simpledisplay.d index 67fe9b1..474300e 100644 --- a/simpledisplay.d +++ b/simpledisplay.d @@ -6574,6 +6574,7 @@ struct ScreenPainter { } /// + @scriptable @property void outlineColor(Color c) { if(impl is null) return; if(activePen.color == c) @@ -6583,6 +6584,7 @@ struct ScreenPainter { } /// + @scriptable @property void fillColor(Color c) { if(impl is null) return; 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) { if(impl is null) return; if(lowerRight.x != 0 || lowerRight.y != 0) { @@ -6734,6 +6737,7 @@ struct ScreenPainter { /// Draws a pen using the current pen / outlineColor + @scriptable void drawLine(Point starting, Point ending) { if(impl is null) 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 /// 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. + @scriptable void drawRectangle(Point upperLeft, int width, int height) { if(impl is null) return; if(isClipped(upperLeft, width, height)) return;