mirror of https://github.com/adamdruppe/arsd.git
good stuff
This commit is contained in:
parent
1debdf7cc7
commit
fe84ef7707
25
cgi.d
25
cgi.d
|
@ -6845,11 +6845,14 @@ void freeIoOp(ref IoOp* ptr) {
|
||||||
version(Posix)
|
version(Posix)
|
||||||
version(with_addon_servers_connections)
|
version(with_addon_servers_connections)
|
||||||
void nonBlockingWrite(EventIoServer eis, int connection, const void[] data) {
|
void nonBlockingWrite(EventIoServer eis, int connection, const void[] data) {
|
||||||
|
|
||||||
|
//import std.stdio : writeln; writeln(cast(string) data);
|
||||||
|
|
||||||
import core.sys.posix.unistd;
|
import core.sys.posix.unistd;
|
||||||
|
|
||||||
auto ret = write(connection, data.ptr, data.length);
|
auto ret = write(connection, data.ptr, data.length);
|
||||||
if(ret != data.length) {
|
if(ret != data.length) {
|
||||||
if(ret == 0 || errno == EPIPE) {
|
if(ret == 0 || (ret == -1 && (errno == EPIPE || errno == ETIMEDOUT))) {
|
||||||
// the file is closed, remove it
|
// the file is closed, remove it
|
||||||
eis.fileClosed(connection);
|
eis.fileClosed(connection);
|
||||||
} else
|
} else
|
||||||
|
@ -7875,7 +7878,9 @@ final class EventSourceServerImplementation : EventSourceServer, EventIoServer {
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
void handleLocalConnectionClose(IoOp* op) {}
|
void handleLocalConnectionClose(IoOp* op) {
|
||||||
|
fileClosed(op.fd);
|
||||||
|
}
|
||||||
void handleLocalConnectionComplete(IoOp* op) {}
|
void handleLocalConnectionComplete(IoOp* op) {}
|
||||||
|
|
||||||
void wait_timeout() {
|
void wait_timeout() {
|
||||||
|
@ -7883,9 +7888,9 @@ final class EventSourceServerImplementation : EventSourceServer, EventIoServer {
|
||||||
foreach(url, connections; eventConnectionsByUrl)
|
foreach(url, connections; eventConnectionsByUrl)
|
||||||
foreach(connection; connections)
|
foreach(connection; connections)
|
||||||
if(connection.needsChunking)
|
if(connection.needsChunking)
|
||||||
nonBlockingWrite(this, connection.fd, "2\r\n:\n\r\n");
|
nonBlockingWrite(this, connection.fd, "1b\r\nevent: keepalive\ndata: ok\n\n\r\n");
|
||||||
else
|
else
|
||||||
nonBlockingWrite(this, connection.fd, ":\n\r\n");
|
nonBlockingWrite(this, connection.fd, "event: keepalive\ndata: ok\n\n\r\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
void fileClosed(int fd) {
|
void fileClosed(int fd) {
|
||||||
|
@ -8151,11 +8156,20 @@ void runAddonServer(EIS)(string localListenerName, EIS eis) if(is(EIS : EventIoS
|
||||||
ioops[sock] = acceptOp;
|
ioops[sock] = acceptOp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
import core.time : MonoTime, seconds;
|
||||||
|
|
||||||
|
MonoTime timeout = MonoTime.currTime + 15.seconds;
|
||||||
|
|
||||||
while(true) {
|
while(true) {
|
||||||
|
|
||||||
// FIXME: it should actually do a timerfd that runs on any thing that hasn't been run recently
|
// FIXME: it should actually do a timerfd that runs on any thing that hasn't been run recently
|
||||||
|
|
||||||
int timeout_milliseconds = 15000; // -1; // infinite
|
int timeout_milliseconds = 0; // -1; // infinite
|
||||||
|
|
||||||
|
timeout_milliseconds = cast(int) (timeout - MonoTime.currTime).total!"msecs";
|
||||||
|
if(timeout_milliseconds < 0)
|
||||||
|
timeout_milliseconds = 0;
|
||||||
|
|
||||||
//writeln("waiting for ", name);
|
//writeln("waiting for ", name);
|
||||||
|
|
||||||
version(linux) {
|
version(linux) {
|
||||||
|
@ -8172,6 +8186,7 @@ void runAddonServer(EIS)(string localListenerName, EIS eis) if(is(EIS : EventIoS
|
||||||
|
|
||||||
if(nfds == 0) {
|
if(nfds == 0) {
|
||||||
eis.wait_timeout();
|
eis.wait_timeout();
|
||||||
|
timeout += 15.seconds;
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach(idx; 0 .. nfds) {
|
foreach(idx; 0 .. nfds) {
|
||||||
|
|
14
dom.d
14
dom.d
|
@ -3400,8 +3400,8 @@ class Element : DomParent {
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
// at the top we don't have anything to really do
|
// at the top we don't have anything to really do
|
||||||
if(parent_ is null)
|
//if(parent_ is null)
|
||||||
return null;
|
//return null;
|
||||||
|
|
||||||
// I've used isEmpty before but this other check seems better....
|
// I've used isEmpty before but this other check seems better....
|
||||||
//|| this.isEmpty())
|
//|| this.isEmpty())
|
||||||
|
@ -8115,9 +8115,17 @@ unittest {
|
||||||
</html>`);
|
</html>`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
foreach(test; [
|
||||||
|
"<a att=\"http://ele\"><b><ele1>Hello</ele1>\n <c>\n <d>\n <ele2>How are you?</ele2>\n </d>\n <e>\n <ele3>Good & you?</ele3>\n </e>\n </c>\n </b>\n</a>",
|
||||||
|
"<a att=\"http://ele\"><b><ele1>Hello</ele1><c><d><ele2>How are you?</ele2></d><e><ele3>Good & you?</ele3></e></c></b></a>",
|
||||||
|
] )
|
||||||
{
|
{
|
||||||
auto document = new XmlDocument("<a att=\"http://ele\"><b><ele1>Hello</ele1>\n <c>\n <d>\n <ele2>How are you?</ele2>\n </d>\n <e>\n <ele3>Good & you?</ele3>\n </e>\n </c>\n </b>\n</a>");
|
auto document = new XmlDocument(test);
|
||||||
assert(document.root.toPrettyString(false, 0, " ") == "<a att=\"http://ele\">\n <b>\n <ele1>Hello</ele1>\n <c>\n <d>\n <ele2>How are you?</ele2>\n </d>\n <e>\n <ele3>Good & you?</ele3>\n </e>\n </c>\n </b>\n</a>");
|
assert(document.root.toPrettyString(false, 0, " ") == "<a att=\"http://ele\">\n <b>\n <ele1>Hello</ele1>\n <c>\n <d>\n <ele2>How are you?</ele2>\n </d>\n <e>\n <ele3>Good & you?</ele3>\n </e>\n </c>\n </b>\n</a>");
|
||||||
|
assert(document.toPrettyString(false, 0, " ") == "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<a att=\"http://ele\">\n <b>\n <ele1>Hello</ele1>\n <c>\n <d>\n <ele2>How are you?</ele2>\n </d>\n <e>\n <ele3>Good & you?</ele3>\n </e>\n </c>\n </b>\n</a>");
|
||||||
|
auto omg = document.root;
|
||||||
|
omg.parent_ = null;
|
||||||
|
assert(omg.toPrettyString(false, 0, " ") == "<a att=\"http://ele\">\n <b>\n <ele1>Hello</ele1>\n <c>\n <d>\n <ele2>How are you?</ele2>\n </d>\n <e>\n <ele3>Good & you?</ele3>\n </e>\n </c>\n </b>\n</a>");
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
|
|
10
jsvar.d
10
jsvar.d
|
@ -444,7 +444,7 @@ private var _op(alias _this, alias this2, string op, T)(T t) if(op != "~") {
|
||||||
_this._payload._integral = l;
|
_this._payload._integral = l;
|
||||||
return _this;
|
return _this;
|
||||||
} else static if(isFloatingPoint!T) {
|
} else static if(isFloatingPoint!T) {
|
||||||
static if(op == "&" || op == "|" || op == "^") {
|
static if(op == "&" || op == "|" || op == "^" || op == "<<" || op == ">>" || op == ">>>") {
|
||||||
this2._type = var.Type.Integral;
|
this2._type = var.Type.Integral;
|
||||||
long f = l;
|
long f = l;
|
||||||
mixin("f "~op~"= cast(long) t;");
|
mixin("f "~op~"= cast(long) t;");
|
||||||
|
@ -465,7 +465,7 @@ private var _op(alias _this, alias this2, string op, T)(T t) if(op != "~") {
|
||||||
_this._type = var.Type.Integral;
|
_this._type = var.Type.Integral;
|
||||||
_this._payload._integral = l;
|
_this._payload._integral = l;
|
||||||
} else{
|
} else{
|
||||||
static if(op == "&" || op == "|" || op == "^") {
|
static if(op == "&" || op == "|" || op == "^" || op == "<<" || op == ">>" || op == ">>>") {
|
||||||
long f = l;
|
long f = l;
|
||||||
mixin("f "~op~"= cast(long) rhs;");
|
mixin("f "~op~"= cast(long) rhs;");
|
||||||
_this._type = var.Type.Integral;
|
_this._type = var.Type.Integral;
|
||||||
|
@ -484,7 +484,7 @@ private var _op(alias _this, alias this2, string op, T)(T t) if(op != "~") {
|
||||||
auto f = this._payload._floating;
|
auto f = this._payload._floating;
|
||||||
|
|
||||||
static if(isIntegral!T || isFloatingPoint!T) {
|
static if(isIntegral!T || isFloatingPoint!T) {
|
||||||
static if(op == "&" || op == "|" || op == "^") {
|
static if(op == "&" || op == "|" || op == "^" || op == "<<" || op == ">>" || op == ">>>") {
|
||||||
long argh = cast(long) f;
|
long argh = cast(long) f;
|
||||||
mixin("argh "~op~"= cast(long) t;");
|
mixin("argh "~op~"= cast(long) t;");
|
||||||
_this._type = var.Type.Integral;
|
_this._type = var.Type.Integral;
|
||||||
|
@ -498,7 +498,7 @@ private var _op(alias _this, alias this2, string op, T)(T t) if(op != "~") {
|
||||||
} else static if(isSomeString!T) {
|
} else static if(isSomeString!T) {
|
||||||
auto rhs = stringToNumber(t);
|
auto rhs = stringToNumber(t);
|
||||||
|
|
||||||
static if(op == "&" || op == "|" || op == "^") {
|
static if(op == "&" || op == "|" || op == "^" || op == "<<" || op == ">>" || op == ">>>") {
|
||||||
long pain = cast(long) f;
|
long pain = cast(long) f;
|
||||||
mixin("pain "~op~"= cast(long) rhs;");
|
mixin("pain "~op~"= cast(long) rhs;");
|
||||||
_this._type = var.Type.Integral;
|
_this._type = var.Type.Integral;
|
||||||
|
@ -511,7 +511,7 @@ private var _op(alias _this, alias this2, string op, T)(T t) if(op != "~") {
|
||||||
return _this;
|
return _this;
|
||||||
} else static assert(0);
|
} else static assert(0);
|
||||||
} else if(this2.payloadType() == var.Type.String) {
|
} else if(this2.payloadType() == var.Type.String) {
|
||||||
static if(op == "&" || op == "|" || op == "^") {
|
static if(op == "&" || op == "|" || op == "^" || op == "<<" || op == ">>" || op == ">>>") {
|
||||||
long r = cast(long) stringToNumber(this2._payload._string);
|
long r = cast(long) stringToNumber(this2._payload._string);
|
||||||
long rhs;
|
long rhs;
|
||||||
} else {
|
} else {
|
||||||
|
|
142
minigui.d
142
minigui.d
|
@ -4227,6 +4227,148 @@ class OpenGlWidget : NestedChildWindowWidget {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
This demo shows how to draw text in an opengl scene.
|
||||||
|
+/
|
||||||
|
unittest {
|
||||||
|
import arsd.minigui;
|
||||||
|
import arsd.ttf;
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
auto window = new Window();
|
||||||
|
|
||||||
|
auto widget = new OpenGlWidget(window);
|
||||||
|
|
||||||
|
// old means non-shader code so compatible with glBegin etc.
|
||||||
|
// tbh I haven't implemented new one in font yet...
|
||||||
|
// anyway, declaring here, will construct soon.
|
||||||
|
OpenGlLimitedFont!(OpenGlFontGLVersion.old) glfont;
|
||||||
|
|
||||||
|
// this is a little bit awkward, calling some methods through
|
||||||
|
// the underlying SimpleWindow `win` method, and you can't do this
|
||||||
|
// on a nanovega widget due to conflicts so I should probably fix
|
||||||
|
// the api to be a bit easier. But here it will work.
|
||||||
|
//
|
||||||
|
// Alternatively, you could load the font on the first draw, inside
|
||||||
|
// the redrawOpenGlScene, and keep a flag so you don't do it every
|
||||||
|
// time. That'd be a bit easier since the lib sets up the context
|
||||||
|
// by then guaranteed.
|
||||||
|
//
|
||||||
|
// But still, I wanna show this.
|
||||||
|
widget.win.visibleForTheFirstTime = delegate {
|
||||||
|
// must set the opengl context
|
||||||
|
widget.win.setAsCurrentOpenGlContext();
|
||||||
|
|
||||||
|
// if you were doing a OpenGL 3+ shader, this
|
||||||
|
// gets especially important to do in order. With
|
||||||
|
// old-style opengl, I think you can even do it
|
||||||
|
// in main(), but meh, let's show it more correctly.
|
||||||
|
|
||||||
|
// Anyway, now it is time to load the font from the
|
||||||
|
// OS (you can alternatively load one from a .ttf file
|
||||||
|
// you bundle with the application), then load the
|
||||||
|
// font into texture for drawing.
|
||||||
|
|
||||||
|
auto osfont = new OperatingSystemFont("DejaVu Sans", 18);
|
||||||
|
|
||||||
|
assert(!osfont.isNull()); // make sure it actually loaded
|
||||||
|
|
||||||
|
// using typeof to avoid repeating the long name lol
|
||||||
|
glfont = new typeof(glfont)(
|
||||||
|
// get the raw data from the font for loading in here
|
||||||
|
// since it doesn't use the OS function to draw the
|
||||||
|
// text, we gotta treat it more as a file than as
|
||||||
|
// a drawing api.
|
||||||
|
osfont.getTtfBytes(),
|
||||||
|
18, // need to respecify size since opengl world is different coordinate system
|
||||||
|
|
||||||
|
// these last two numbers are why it is called
|
||||||
|
// "Limited" font. It only loads the characters
|
||||||
|
// in the given range, since the texture atlas
|
||||||
|
// it references is all a big image generated ahead
|
||||||
|
// of time. You could maybe do the whole thing but
|
||||||
|
// idk how much memory that is.
|
||||||
|
//
|
||||||
|
// But here, 0-128 represents the ASCII range, so
|
||||||
|
// good enough for most English things, numeric labels,
|
||||||
|
// etc.
|
||||||
|
0,
|
||||||
|
128
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
widget.redrawOpenGlScene = () {
|
||||||
|
// now we can use the glfont's drawString function
|
||||||
|
|
||||||
|
// first some opengl setup. You can do this in one place
|
||||||
|
// on window first visible too in many cases, just showing
|
||||||
|
// here cuz it is easier for me.
|
||||||
|
|
||||||
|
// gonna need some alpha blending or it just looks awful
|
||||||
|
glEnable(GL_BLEND);
|
||||||
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
||||||
|
glClearColor(0,0,0,0);
|
||||||
|
glDepthFunc(GL_LEQUAL);
|
||||||
|
|
||||||
|
// Also need to enable 2d textures, since it draws the
|
||||||
|
// font characters as images baked in
|
||||||
|
glMatrixMode(GL_MODELVIEW);
|
||||||
|
glLoadIdentity();
|
||||||
|
glDisable(GL_DEPTH_TEST);
|
||||||
|
glEnable(GL_TEXTURE_2D);
|
||||||
|
|
||||||
|
// the orthographic matrix is best for 2d things like text
|
||||||
|
// so let's set that up. This matrix makes the coordinates
|
||||||
|
// in the opengl scene be one-to-one with the actual pixels
|
||||||
|
// on screen. (Not necessarily best, you may wish to scale
|
||||||
|
// things, but it does help keep fonts looking normal.)
|
||||||
|
glMatrixMode(GL_PROJECTION);
|
||||||
|
glLoadIdentity();
|
||||||
|
glOrtho(0, widget.width, widget.height, 0, 0, 1);
|
||||||
|
|
||||||
|
// you can do other glScale, glRotate, glTranslate, etc
|
||||||
|
// to the matrix here of course if you want.
|
||||||
|
|
||||||
|
// note the x,y coordinates here are for the text baseline
|
||||||
|
// NOT the upper-left corner. The baseline is like the line
|
||||||
|
// in the notebook you write on. Most the letters are actually
|
||||||
|
// above it, but some, like p and q, dip a bit below it.
|
||||||
|
//
|
||||||
|
// So if you're used to the upper left coordinate like the
|
||||||
|
// rest of simpledisplay/minigui usually do, do the
|
||||||
|
// y + glfont.ascent to bring it down a little. So this
|
||||||
|
// example puts the string in the upper left of the window.
|
||||||
|
glfont.drawString(0, 0 + glfont.ascent, "Hello!!", Color.green);
|
||||||
|
|
||||||
|
// re color btw: the function sets a solid color internally,
|
||||||
|
// but you actually COULD do your own thing for rainbow effects
|
||||||
|
// and the sort if you wanted too, by pulling its guts out.
|
||||||
|
// Just view its source for an idea of how it actually draws:
|
||||||
|
// http://arsd-official.dpldocs.info/source/arsd.ttf.d.html#L332
|
||||||
|
|
||||||
|
// it gets a bit complicated with the character positioning,
|
||||||
|
// but the opengl parts are fairly simple: bind a texture,
|
||||||
|
// set the color, draw a quad for each letter.
|
||||||
|
|
||||||
|
|
||||||
|
// the last optional argument there btw is a bounding box
|
||||||
|
// it will/ use to word wrap and return an object you can
|
||||||
|
// use to implement scrolling or pagination; it tells how
|
||||||
|
// much of the string didn't fit in the box. But for simple
|
||||||
|
// labels we can just ignore that.
|
||||||
|
|
||||||
|
|
||||||
|
// I'd suggest drawing text as the last step, after you
|
||||||
|
// do your other drawing. You might use the push/pop matrix
|
||||||
|
// stuff to keep your place. You, in theory, should be able
|
||||||
|
// to do text in a 3d space but I've never actually tried
|
||||||
|
// that....
|
||||||
|
};
|
||||||
|
|
||||||
|
window.loop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
version(custom_widgets)
|
version(custom_widgets)
|
||||||
private alias ListWidgetBase = ScrollableWidget;
|
private alias ListWidgetBase = ScrollableWidget;
|
||||||
else
|
else
|
||||||
|
|
26
script.d
26
script.d
|
@ -1462,7 +1462,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!("+", "-", "*", "/", "==", "!=", "<=", ">=", ">", "<", "~", "&&", "||", "&", "|", "^", "%")) //, ">>", "<<", ">>>")) // FIXME
|
foreach(ctOp; CtList!("+", "-", "*", "/", "==", "!=", "<=", ">=", ">", "<", "~", "&&", "||", "&", "|", "^", "%", ">>", "<<", ">>>")) // FIXME
|
||||||
case ctOp: {
|
case ctOp: {
|
||||||
n = mixin("left "~ctOp~" right");
|
n = mixin("left "~ctOp~" right");
|
||||||
break sw;
|
break sw;
|
||||||
|
@ -2824,6 +2824,9 @@ Expression parseFactor(MyTokenStreamHere)(ref MyTokenStreamHere tokens) {
|
||||||
|
|
||||||
if(peek.type == ScriptToken.Type.symbol) {
|
if(peek.type == ScriptToken.Type.symbol) {
|
||||||
switch(peek.str) {
|
switch(peek.str) {
|
||||||
|
case "<<":
|
||||||
|
case ">>":
|
||||||
|
case ">>>":
|
||||||
case "*":
|
case "*":
|
||||||
case "/":
|
case "/":
|
||||||
case "%":
|
case "%":
|
||||||
|
@ -3510,7 +3513,6 @@ Expression parseStatement(MyTokenStreamHere)(ref MyTokenStreamHere tokens, strin
|
||||||
case "continue":
|
case "continue":
|
||||||
case "break":
|
case "break":
|
||||||
case "return":
|
case "return":
|
||||||
|
|
||||||
return parseExpression(tokens);
|
return parseExpression(tokens);
|
||||||
// unary prefix operators
|
// unary prefix operators
|
||||||
case "!":
|
case "!":
|
||||||
|
@ -3536,6 +3538,26 @@ Expression parseStatement(MyTokenStreamHere)(ref MyTokenStreamHere tokens, strin
|
||||||
assert(0);
|
assert(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FIXME someday this should work, my parser is so bad
|
||||||
|
// until then put parens around your == stuff.
|
||||||
|
version(none)
|
||||||
|
unittest {
|
||||||
|
interpret(q{
|
||||||
|
var a = 5;
|
||||||
|
var b = false;
|
||||||
|
assert(a == 5 || b);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
version(none)
|
||||||
|
unittest {
|
||||||
|
interpret(q{
|
||||||
|
var a = 5;
|
||||||
|
var b = false;
|
||||||
|
assert(((a == 5) || b));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
struct CompoundStatementRange(MyTokenStreamHere) {
|
struct CompoundStatementRange(MyTokenStreamHere) {
|
||||||
// FIXME: if MyTokenStreamHere is not a class, this fails!
|
// FIXME: if MyTokenStreamHere is not a class, this fails!
|
||||||
MyTokenStreamHere tokens;
|
MyTokenStreamHere tokens;
|
||||||
|
|
3
sqlite.d
3
sqlite.d
|
@ -372,7 +372,8 @@ struct Statement {
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case SQLITE_NULL:
|
case SQLITE_NULL:
|
||||||
v = null;
|
string n = null;
|
||||||
|
v = n;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
whatever == false
|
whatever == false
|
||||||
</or-else>
|
</or-else>
|
||||||
|
|
||||||
<for-each over="some_array" as="item">
|
<for-each over="some_array" as="item" index="idx">
|
||||||
<%= item %>
|
<%= item %>
|
||||||
</for-each>
|
</for-each>
|
||||||
<or-else>
|
<or-else>
|
||||||
|
@ -34,7 +34,7 @@
|
||||||
```
|
```
|
||||||
|
|
||||||
Functions available:
|
Functions available:
|
||||||
`encodeURIComponent`, `formatDate`, `dayOfWeek`, `formatTime`
|
`encodeURIComponent`, `formatDate`, `dayOfWeek`, `formatTime`, `filterKeys`
|
||||||
|
|
||||||
History:
|
History:
|
||||||
Things inside script tag were added on January 7, 2022.
|
Things inside script tag were added on January 7, 2022.
|
||||||
|
@ -65,6 +65,39 @@ void addDefaultFunctions(var context) {
|
||||||
import std.conv;
|
import std.conv;
|
||||||
// FIXME: I prolly want it to just set the prototype or something
|
// FIXME: I prolly want it to just set the prototype or something
|
||||||
|
|
||||||
|
/+
|
||||||
|
foo |> filterKeys(["foo", "bar"]);
|
||||||
|
|
||||||
|
It needs to match the filter, then if it is -pattern, it is removed and if it is +pattern, it is retained.
|
||||||
|
|
||||||
|
First one that matches applies to the key, so the last one in the list is your default.
|
||||||
|
|
||||||
|
Default is to reject. Putting a "*" at the end will keep everything not removed though.
|
||||||
|
|
||||||
|
["-foo", "*"] // keep everything except foo
|
||||||
|
+/
|
||||||
|
context.filterKeys = function var(var f, string[] filters) {
|
||||||
|
import std.path;
|
||||||
|
var o = var.emptyObject;
|
||||||
|
foreach(k, v; f) {
|
||||||
|
bool keep = false;
|
||||||
|
foreach(filter; filters) {
|
||||||
|
if(filter.length == 0)
|
||||||
|
throw new Exception("invalid filter");
|
||||||
|
bool filterOff = filter[0] == '-';
|
||||||
|
if(filterOff)
|
||||||
|
filter = filter[1 .. $];
|
||||||
|
if(globMatch(k.get!string, filter)) {
|
||||||
|
keep = !filterOff;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(keep)
|
||||||
|
o[k] = v;
|
||||||
|
}
|
||||||
|
return o;
|
||||||
|
};
|
||||||
|
|
||||||
context.encodeURIComponent = function string(var f) {
|
context.encodeURIComponent = function string(var f) {
|
||||||
import std.uri;
|
import std.uri;
|
||||||
return encodeComponent(f.get!string);
|
return encodeComponent(f.get!string);
|
||||||
|
@ -213,9 +246,11 @@ void expandTemplate(Element root, var context) {
|
||||||
var nc = var.emptyObject(context);
|
var nc = var.emptyObject(context);
|
||||||
lastBoolResult = false;
|
lastBoolResult = false;
|
||||||
auto got = interpret(ele.attrs.over, context);
|
auto got = interpret(ele.attrs.over, context);
|
||||||
foreach(item; got) {
|
foreach(k, item; got) {
|
||||||
lastBoolResult = true;
|
lastBoolResult = true;
|
||||||
nc[ele.attrs.as] = item;
|
nc[ele.attrs.as] = item;
|
||||||
|
if(ele.attrs.index.length)
|
||||||
|
nc[ele.attrs.index] = k;
|
||||||
auto clone = ele.cloneNode(true);
|
auto clone = ele.cloneNode(true);
|
||||||
clone.tagName = "root"; // it certainly isn't a for-each anymore!
|
clone.tagName = "root"; // it certainly isn't a for-each anymore!
|
||||||
expandTemplate(clone, nc);
|
expandTemplate(clone, nc);
|
||||||
|
@ -274,7 +309,7 @@ void expandTemplate(Element root, var context) {
|
||||||
check_more:
|
check_more:
|
||||||
auto idx = source.indexOf("<%=");
|
auto idx = source.indexOf("<%=");
|
||||||
if(idx != -1) {
|
if(idx != -1) {
|
||||||
newCode = source[0 .. idx];
|
newCode ~= source[0 .. idx];
|
||||||
auto remaining = source[idx + 3 .. $];
|
auto remaining = source[idx + 3 .. $];
|
||||||
idx = remaining.indexOf("%>");
|
idx = remaining.indexOf("%>");
|
||||||
if(idx == -1)
|
if(idx == -1)
|
||||||
|
|
Loading…
Reference in New Issue