diff --git a/dom.d b/dom.d
index 210ba00..187eed1 100644
--- a/dom.d
+++ b/dom.d
@@ -477,7 +477,7 @@ class Element {
}
///.
- final SomeElementType requireElementById(SomeElementType = Element)(string id)
+ final SomeElementType requireElementById(SomeElementType = Element, string file = __FILE__, int line = __LINE__)(string id)
if(
is(SomeElementType : Element)
)
@@ -487,12 +487,12 @@ class Element {
body {
auto e = cast(SomeElementType) getElementById(id);
if(e is null)
- throw new ElementNotFoundException(SomeElementType.stringof, "id=" ~ id);
+ throw new ElementNotFoundException(SomeElementType.stringof, "id=" ~ id, file, line);
return e;
}
///.
- final SomeElementType requireSelector(SomeElementType = Element)(string selector)
+ final SomeElementType requireSelector(SomeElementType = Element, string file = __FILE__, int line = __LINE__)(string selector)
if(
is(SomeElementType : Element)
)
@@ -502,7 +502,7 @@ class Element {
body {
auto e = cast(SomeElementType) querySelector(selector);
if(e is null)
- throw new ElementNotFoundException(SomeElementType.stringof, selector);
+ throw new ElementNotFoundException(SomeElementType.stringof, selector, file, line);
return e;
}
@@ -689,7 +689,7 @@ class Element {
string where = a.href; // same as a.getAttribute("href");
*/
// name != "popFront" is so duck typing doesn't think it's a range
- string opDispatch(string name)(string v = null) if(name != "popFront") {
+ @property string opDispatch(string name)(string v = null) if(name != "popFront") {
if(v !is null)
setAttribute(name, v);
return getAttribute(name);
@@ -707,12 +707,18 @@ class Element {
return children;
}
-
+ /// get all the classes on this element
+ @property string[] classes() {
+ return className.split(" ");
+ }
/// Adds a string to the class attribute. The class attribute is used a lot in CSS.
Element addClass(string c) {
+ if(hasClass(c))
+ return this; // don't add it twice
+
string cn = getAttribute("class");
- if(cn is null) {
+ if(cn.length == 0) {
setAttribute("class", c);
return this;
} else {
@@ -724,10 +730,18 @@ class Element {
/// Removes a particular class name.
Element removeClass(string c) {
- auto cn = className;
+ if(!hasClass(c))
+ return this;
+ string n;
+ foreach(name; classes) {
+ if(c == name)
+ continue; // cut it out
+ if(n.length)
+ n ~= " ";
+ n ~= name;
+ }
- // FIXME: this is actually wrong!
- className = cn.replace(c, "").strip;
+ className = n.strip;
return this;
}
@@ -1680,7 +1694,7 @@ class Element {
/**
Returns a lazy range of all its children, recursively.
*/
- ElementStream tree() {
+ @property ElementStream tree() {
return new ElementStream(this);
}
}
@@ -3286,11 +3300,11 @@ class Document : FileResource {
}
/// ditto
- final SomeElementType requireSelector(SomeElementType = Element)(string selector)
+ final SomeElementType requireSelector(SomeElementType = Element, string file = __FILE__, int line = __LINE__)(string selector)
if( is(SomeElementType : Element))
out(ret) { assert(ret !is null); }
body {
- return root.requireSelector!(SomeElementType)(selector);
+ return root.requireSelector!(SomeElementType, file, line)(selector);
}
@@ -4521,7 +4535,7 @@ final class Stack(T) {
final class ElementStream {
///.
- Element front() {
+ @property Element front() {
return current.element;
}
@@ -4572,7 +4586,7 @@ final class ElementStream {
}
///.
- bool empty() {
+ @property bool empty() {
return isEmpty;
}
diff --git a/web.d b/web.d
index ec088a5..e765a4a 100644
--- a/web.d
+++ b/web.d
@@ -324,16 +324,14 @@ class ApiProvider : WebDotDBaseType {
protected void addCsrfTokens(Document document) {
if(document is null)
return;
- if(!csrfTokenAddedToScript) {
+ auto bod = document.mainBody;
+ if(!bod.hasAttribute("data-csrf-key")) {
auto tokenInfo = _getCsrfInfo();
if(tokenInfo is null)
return;
-
- auto bod = document.mainBody;
if(bod !is null) {
bod.setAttribute("data-csrf-key", tokenInfo["key"]);
bod.setAttribute("data-csrf-token", tokenInfo["token"]);
- csrfTokenAddedToScript = true;
}
addCsrfTokens(document.root);
@@ -346,29 +344,22 @@ class ApiProvider : WebDotDBaseType {
super._postProcess(document);
}
- private bool csrfTokenAddedToScript;
- //private bool csrfTokenAddedToForms;
-
/// This adds CSRF tokens to all forms in the tree
protected void addCsrfTokens(Element element) {
if(element is null)
return;
- //if(!csrfTokenAddedToForms) {
- auto tokenInfo = _getCsrfInfo();
- if(tokenInfo is null)
- return;
+ auto tokenInfo = _getCsrfInfo();
+ if(tokenInfo is null)
+ return;
- foreach(formElement; element.getElementsByTagName("form")) {
- if(formElement.method != "POST" && formElement.method != "post")
- continue;
- auto form = cast(Form) formElement;
- assert(form !is null);
+ foreach(formElement; element.getElementsByTagName("form")) {
+ if(formElement.method != "POST" && formElement.method != "post")
+ continue;
+ auto form = cast(Form) formElement;
+ assert(form !is null);
- form.setValue(tokenInfo["key"], tokenInfo["token"]);
- }
-
- //csrfTokenAddedToForms = true;
- //}
+ form.setValue(tokenInfo["key"], tokenInfo["token"]);
+ }
}
// and added to ajax forms..
@@ -477,7 +468,7 @@ class ApiProvider : WebDotDBaseType {
assert(ret !is null);
}
body {
- auto document = new Document("
", true, true);
+ auto document = new Document("", true, true);
if(this.reflection !is null)
document.title = this.reflection.name;
auto container = document.getElementById("body");
@@ -499,12 +490,16 @@ class ApiProvider : WebDotDBaseType {
private string _errorMessageForCatchAll;
private FileResource _catchallEntry(string path, string funName, string errorMessage) {
if(!errorMessage.length) {
+ /*
string allFuncs, allObjs;
foreach(n, f; reflection.functions)
allFuncs ~= n ~ "\n";
foreach(n, f; reflection.objects)
allObjs ~= n ~ "\n";
errorMessage = "no such function " ~ funName ~ "\n functions are:\n" ~ allFuncs ~ "\n\nObjects are:\n" ~ allObjs;
+ */
+
+ errorMessage = "No such page: " ~ funName;
}
_errorMessageForCatchAll = errorMessage;
@@ -906,6 +901,7 @@ Parameter reflectParam(param)() {
struct CallInfo {
string objectIdentifier;
immutable(FunctionInfo)* func;
+ void delegate(Document)[] postProcessors;
}
class NonCanonicalUrlException : Exception {
@@ -961,6 +957,9 @@ CallInfo parseUrl(in ReflectionInfo* reflection, string url, string defaultFunct
name = "/"; // should call _defaultPage
}
+ if(reflection.instantiation !is null)
+ info.postProcessors ~= &(reflection.instantiation._postProcess);
+
if(name in reflection.functions) {
info.func = reflection.functions[name];
@@ -1058,13 +1057,6 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
if(cgi.pathInfo.indexOf("builtin.") != -1 && instantiation.builtInFunctions !is null)
base = instantiation.builtInFunctions;
- if(instantiator.length) {
- assert(fun !is null);
- assert(fun.parentObject !is null);
- assert(fun.parentObject.instantiate !is null);
- realObject = fun.parentObject.instantiate(instantiator);
- }
-
try {
if(fun is null) {
auto d = instantiation._catchallEntry(
@@ -1092,6 +1084,15 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
assert(fun.dispatcher !is null);
assert(cgi !is null);
+
+ if(instantiator.length) {
+ assert(fun !is null);
+ assert(fun.parentObject !is null);
+ assert(fun.parentObject.instantiate !is null);
+ realObject = fun.parentObject.instantiate(instantiator);
+ }
+
+
result.type = fun.returnType;
string format = cgi.request("format", reflection.defaultOutputFormat);
@@ -1179,6 +1180,10 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
result.result.str = form.toString();
} else {
+ auto fourOhFour = cast(NoSuchPageException) e;
+ if(fourOhFour !is null)
+ cgi.setResponseStatus("404 File Not Found");
+
if(instantiation._errorFunction !is null) {
auto document = instantiation._errorFunction(e);
if(document is null)
@@ -1186,8 +1191,10 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
result.result.str = (document.toString());
} else {
gotnull:
- auto code = Element.make("pre");
- code.innerText = e.toString();
+ auto code = Element.make("div");
+ code.addClass("exception-error-message");
+ code.addChild("p", e.msg);
+ debug code.addChild("pre", e.toString());
result.result.str = (code.toString());
}
@@ -1235,9 +1242,9 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
if(result.result.type == JSON_TYPE.STRING) {
auto returned = result.result.str;
- if((fun !is null) && envelopeFormat != "html") {
+ if(envelopeFormat != "html") {
Document document;
- if(result.success && fun.returnTypeIsDocument && returned.length) {
+ if(result.success && fun !is null && fun.returnTypeIsDocument && returned.length) {
// probably not super efficient...
document = new TemplatedDocument(returned);
} else {
@@ -1253,9 +1260,18 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
if(envelopeFormat == "document") {
// forming a nice chain here...
// FIXME: this isn't actually a nice chain!
+ bool[void delegate(Document)] run;
+
+ auto postProcessors = info.postProcessors;
if(base !is instantiation)
- instantiation._postProcess(document);
- base._postProcess(document);
+ postProcessors ~= &(instantiation._postProcess);
+ postProcessors ~= &(base._postProcess);
+ foreach(pp; postProcessors) {
+ if(pp in run)
+ continue;
+ run[pp] = true;
+ pp(document);
+ }
}
returned = document.toString;
@@ -2645,10 +2661,50 @@ immutable(string[]) weekdayNames = [
];
+// this might be temporary
+struct TemplateFilters {
+ string date(string replacement, string[], in Element, string) {
+ auto date = to!long(replacement);
+
+ import std.date;
+
+ auto day = dateFromTime(date);
+ auto year = yearFromTime(date);
+ auto month = monthNames[monthFromTime(date)];
+ replacement = format("%s %d, %d", month, day, year);
+
+ return replacement;
+ }
+
+ string uri(string replacement, string[], in Element, string) {
+ return std.uri.encodeComponent(replacement);
+ }
+
+ string js(string replacement, string[], in Element, string) {
+ return toJson(replacement);
+ }
+
+ static auto defaultThings() {
+ string delegate(string, string[], in Element, string)[string] pipeFunctions;
+ TemplateFilters filters;
+
+ if("date" !in pipeFunctions)
+ pipeFunctions["date"] = &filters.date;
+ if("uri" !in pipeFunctions)
+ pipeFunctions["uri"] = &filters.uri;
+ if("js" !in pipeFunctions)
+ pipeFunctions["js"] = &filters.js;
+ return pipeFunctions;
+ }
+}
+void applyTemplateToElement(
+ Element e,
+ in string[string] vars,
+ in string delegate(string, string[], in Element, string)[string] pipeFunctions = TemplateFilters.defaultThings())
+{
-void applyTemplateToElement(Element e, in string[string] vars, in string delegate(string, string[], in Element, string)[string] pipeFunctions = null) {
foreach(ele; e.tree) {
auto tc = cast(TextNode) ele;
if(tc !is null) {
@@ -2689,6 +2745,7 @@ string htmlTemplateWithData(in string text, in string[string] vars, in string de
size_t lastAppend = 0;
string name = null;
+ bool replacementPresent = false;
string replacement = null;
string currentPipe = null;
@@ -2702,26 +2759,19 @@ string htmlTemplateWithData(in string text, in string[string] vars, in string de
nameStart = i + 1;
auto it = name in vars;
- if(it !is null)
+ if(it !is null) {
replacement = *it;
+ replacementPresent = true;
+ }
}
void pipeHandler() {
if(currentPipe is null || replacement is null)
return;
- switch(currentPipe) {
- case "date":
- auto date = to!long(replacement);
-
- import std.date;
-
- auto day = dateFromTime(date);
- auto year = yearFromTime(date);
- auto month = monthNames[monthFromTime(date)];
- replacement = format("%s %d, %d", month, day, year);
- break;
- default:
+ if(currentPipe in pipeFunctions) {
+ replacement = pipeFunctions[currentPipe](replacement, null, null, null); // FIXME context
+ // string, string[], in Element, string
}
currentPipe = null;
@@ -2733,6 +2783,7 @@ string htmlTemplateWithData(in string text, in string[string] vars, in string de
if(c == '{') {
replacementStart = i;
state++;
+ replacementPresent = false;
}
break;
case 1:
@@ -2760,7 +2811,7 @@ string htmlTemplateWithData(in string text, in string[string] vars, in string de
pipeHandler(); // anything that was there
stepHandler(); // might make a new pipe if the first...
pipeHandler(); // new names/pipes since this is the last go
- if(name !is null && replacement !is null) {
+ if(name !is null && replacementPresent /*&& replacement !is null*/) {
newText ~= text[lastAppend .. replacementStart];
if(useHtml)
replacement = htmlEntitiesEncode(replacement).replace("\n", "
");
@@ -3267,6 +3318,7 @@ enum string javascriptBaseImpl = q{
xmlHttp.send(a);
if(!async && callback) {
+ xmlHttp.timeout = 500;
return callback(xmlHttp.responseText, xmlHttp.responseXML);
}
return xmlHttp;