diff --git a/cgi.d b/cgi.d index 242e4ec..9468dab 100644 --- a/cgi.d +++ b/cgi.d @@ -1,3 +1,4 @@ +// FIXME: if an exception is thrown, we shouldn't necessarily cache... /++ Provides a uniform server-side API for CGI, FastCGI, SCGI, and HTTP web applications. @@ -1581,6 +1582,26 @@ class Cgi { //RequestMethod _requestMethod; } +/// use this for testing or other isolated things +Cgi dummyCgi(Cgi.RequestMethod method = Cgi.RequestMethod.GET, string url = null, in ubyte[] data = null, void delegate(const(ubyte)[]) outputSink = null) { + // we want to ignore, not use stdout + if(outputSink is null) + outputSink = delegate void(const(ubyte)[]) { }; + + string[string] env; + env["REQUEST_METHOD"] = to!string(method); + env["CONTENT_LENGTH"] = to!string(data.length); + + auto cgi = new Cgi( + 5_000_000, + env, + { return data; }, + outputSink, + null); + + return cgi; +} + // should this be a separate module? Probably, but that's a hassle. @@ -2504,6 +2525,10 @@ long sysTimeToDTime(in SysTime sysTime) { return convert!("hnsecs", "msecs")(sysTime.stdTime - 621355968000000000L); } +long dateTimeToDTime(in DateTime dt) { + return sysTimeToDTime(cast(SysTime) dt); +} + long getUtcTime() { // renamed primarily to avoid conflict with std.date itself return sysTimeToDTime(Clock.currTime(UTC())); } diff --git a/color.d b/color.d index 7fe4634..95ce208 100644 --- a/color.d +++ b/color.d @@ -1,11 +1,9 @@ module arsd.color; -// NOTE: this is obsolete. use color.d instead. - -import std.stdio; import std.math; import std.conv; import std.algorithm; +import std.string : strip, split; struct Color { ubyte r; @@ -20,16 +18,28 @@ struct Color { this.a = cast(ubyte) alpha; } - static Color transparent() { - return Color(0, 0, 0, 0); - } + static Color transparent() { return Color(0, 0, 0, 0); } + static Color white() { return Color(255, 255, 255); } + static Color black() { return Color(0, 0, 0); } + static Color red() { return Color(255, 0, 0); } + static Color green() { return Color(0, 255, 0); } + static Color blue() { return Color(0, 0, 255); } + static Color yellow() { return Color(255, 255, 0); } + static Color teal() { return Color(0, 255, 255); } + static Color purple() { return Color(255, 0, 255); } - static Color white() { - return Color(255, 255, 255); + /* + ubyte[4] toRgbaArray() { + return [r,g,b,a]; } + */ - static Color black() { - return Color(0, 0, 0); + string toCssString() { + import std.string; + if(a == 255) + return format("#%02x%02x%02x", r, g, b); + else + return format("rgba(%d, %d, %d, %f)", r, g, b, cast(real)a / 255.0); } string toString() { @@ -39,6 +49,141 @@ struct Color { else return format("%02x%02x%02x%02x", r, g, b, a); } + + static Color fromNameString(string s) { + Color c; + foreach(member; __traits(allMembers, Color)) { + static if(__traits(compiles, c = __traits(getMember, Color, member))) { + if(s == member) + return __traits(getMember, Color, member); + } + } + throw new Exception("Unknown color " ~ s); + } + + static Color fromString(string s) { + s = s.strip(); + + Color c; + c.a = 255; + + // trying named colors via the static no-arg methods here + foreach(member; __traits(allMembers, Color)) { + static if(__traits(compiles, c = __traits(getMember, Color, member))) { + if(s == member) + return __traits(getMember, Color, member); + } + } + + // try various notations borrowed from CSS (though a little extended) + + // hsl(h,s,l,a) where h is degrees and s,l,a are 0 >= x <= 1.0 + if(s.startsWith("hsl(") || s.startsWith("hsla(")) { + assert(s[$-1] == ')'); + s = s[s.startsWith("hsl(") ? 4 : 5 .. $ - 1]; // the closing paren + + real[3] hsl; + ubyte a = 255; + + auto parts = s.split(","); + foreach(i, part; parts) { + if(i < 3) + hsl[i] = to!real(part.strip); + else + a = cast(ubyte) (to!real(part.strip) * 255); + } + + c = .fromHsl(hsl); + c.a = a; + + return c; + } + + // rgb(r,g,b,a) where r,g,b are 0-255 and a is 0-1.0 + if(s.startsWith("rgb(") || s.startsWith("rgba(")) { + assert(s[$-1] == ')'); + s = s[s.startsWith("rgb(") ? 4 : 5 .. $ - 1]; // the closing paren + + auto parts = s.split(","); + foreach(i, part; parts) { + // lol the loop-switch pattern + auto v = to!real(part.strip); + switch(i) { + case 0: // red + c.r = cast(ubyte) v; + break; + case 1: + c.g = cast(ubyte) v; + break; + case 2: + c.b = cast(ubyte) v; + break; + case 3: + c.a = cast(ubyte) (v * 255); + break; + default: // ignore + } + } + + return c; + } + + + + + // otherwise let's try it as a hex string, really loosely + + if(s.length && s[0] == '#') + s = s[1 .. $]; + + // not a built in... do it as a hex string + if(s.length >= 2) { + c.r = fromHexInternal(s[0 .. 2]); + s = s[2 .. $]; + } + if(s.length >= 2) { + c.g = fromHexInternal(s[0 .. 2]); + s = s[2 .. $]; + } + if(s.length >= 2) { + c.b = fromHexInternal(s[0 .. 2]); + s = s[2 .. $]; + } + if(s.length >= 2) { + c.a = fromHexInternal(s[0 .. 2]); + s = s[2 .. $]; + } + + return c; + } + + static Color fromHsl(real h, real s, real l) { + return .fromHsl(h, s, l); + } +} + + +private ubyte fromHexInternal(string s) { + import std.range; + int result = 0; + + int exp = 1; + //foreach(c; retro(s)) { // FIXME: retro doesn't work right in dtojs + foreach_reverse(c; s) { + if(c >= 'A' && c <= 'F') + result += exp * (c - 'A' + 10); + else if(c >= 'a' && c <= 'f') + result += exp * (c - 'a' + 10); + else if(c >= '0' && c <= '9') + result += exp * (c - '0'); + else + // throw new Exception("invalid hex character: " ~ cast(char) c); + return 0; + + exp *= 16; + } + + return cast(ubyte) result; } @@ -46,7 +191,7 @@ Color fromHsl(real[3] hsl) { return fromHsl(hsl[0], hsl[1], hsl[2]); } -Color fromHsl(real h, real s, real l) { +Color fromHsl(real h, real s, real l, real a = 255) { h = h % 360; real C = (1 - abs(2 * l - 1)) * s; @@ -95,7 +240,7 @@ Color fromHsl(real h, real s, real l) { cast(ubyte)(r * 255), cast(ubyte)(g * 255), cast(ubyte)(b * 255), - 255); + cast(ubyte)(a)); } real[3] toHsl(Color c, bool useWeightedLightness = false) { @@ -198,7 +343,7 @@ Color oppositeLightness(Color c) { /// Try to determine a text color - either white or black - based on the input Color makeTextColor(Color c) { auto hsl = toHsl(c, true); // give green a bonus for contrast - if(hsl[2] > 0.5) + if(hsl[2] > 0.71) return Color(0, 0, 0); else return Color(255, 255, 255); @@ -321,7 +466,8 @@ int fromHex(string s) { int result = 0; int exp = 1; - foreach(c; retro(s)) { + // foreach(c; retro(s)) { + foreach_reverse(c; s) { if(c >= 'A' && c <= 'F') result += exp * (c - 'A' + 10); else if(c >= 'a' && c <= 'f') @@ -356,3 +502,26 @@ Color colorFromString(string s) { return c; } + +/* +import browser.window; +import std.conv; +void main() { + import browser.document; + foreach(ele; document.querySelectorAll("input")) { + ele.addEventListener("change", { + auto h = to!real(document.querySelector("input[name=h]").value); + auto s = to!real(document.querySelector("input[name=s]").value); + auto l = to!real(document.querySelector("input[name=l]").value); + + Color c = Color.fromHsl(h, s, l); + + auto e = document.getElementById("example"); + e.style.backgroundColor = c.toCssString(); + + // JSElement __js_this; + // __js_this.style.backgroundColor = c.toCssString(); + }, false); + } +} +*/ diff --git a/database.d b/database.d index a487d20..5e44341 100644 --- a/database.d +++ b/database.d @@ -48,6 +48,14 @@ interface Database { } import std.stdio; +Ret queryOneColumn(Ret, T...)(Database db, string sql, T t) { + auto res = db.query(sql, t); + if(res.empty) + throw new Exception("no row in result"); + auto row = res.front; + return to!Ret(row[0]); +} + struct Query { ResultSet result; static Query opCall(T...)(Database db, string sql, T t) { diff --git a/dom.d b/dom.d index 16ceee2..10dbea5 100644 --- a/dom.d +++ b/dom.d @@ -20,6 +20,18 @@ */ module arsd.dom; +// FIXME: it would be kinda cool to have some support for internal DTDs +// and maybe XPath as well, to some extent +/* + we could do + meh this sux + + auto xpath = XPath(element); + + // get the first p + xpath.p[0].a["href"] +*/ + // public import arsd.domconvenience; // merged for now /* domconvenience follows { */ @@ -190,6 +202,13 @@ mixin template DomConvenienceFunctions() { return parentNode.insertAfter(this, e); } + Element addSibling(Element e) { + return parentNode.insertAfter(this, e); + } + + Element addChild(Element e) { + return this.appendChild(e); + } /// Convenience function to append text intermixed with other children. /// For example: div.addChildren("You can visit my website by ", new Link("mysite.com", "clicking here"), "."); @@ -207,7 +226,7 @@ mixin template DomConvenienceFunctions() { } ///. - Element addChild(string tagName, Element firstChild) + Element addChild(string tagName, Element firstChild, string info2 = null) in { assert(firstChild !is null); } @@ -220,13 +239,13 @@ mixin template DomConvenienceFunctions() { //assert(firstChild.parentDocument is this.parentDocument); } body { - auto e = Element.make(tagName); + auto e = Element.make(tagName, "", info2); e.appendChild(firstChild); this.appendChild(e); return e; } - Element addChild(string tagName, Html innerHtml) + Element addChild(string tagName, in Html innerHtml, string info2 = null) in { } out(ret) { @@ -235,7 +254,7 @@ mixin template DomConvenienceFunctions() { assert(ret.parentDocument is this.parentDocument); } body { - auto e = Element.make(tagName); + auto e = Element.make(tagName, "", info2); this.appendChild(e); e.innerHTML = innerHtml.source; return e; @@ -419,6 +438,10 @@ struct ElementCollection { elements = [e]; } + this(Element e, string selector) { + elements = e.querySelectorAll(selector); + } + this(Element[] e) { elements = e; } @@ -518,7 +541,10 @@ struct ElementStyle { string set(string name, string value) { if(name.length == 0) return value; - name = unCamelCase(name); + if(name == "cssFloat") + name = "float"; + else + name = unCamelCase(name); auto r = rules(); r[name] = value; @@ -536,7 +562,10 @@ struct ElementStyle { return value; } string get(string name) const { - name = unCamelCase(name); + if(name == "cssFloat") + name = "float"; + else + name = unCamelCase(name); auto r = rules(); if(name in r) return r[name]; @@ -829,7 +858,8 @@ class Element { return e; } - static Element make(string tagName, Html innerHtml, string childInfo2 = null) { + static Element make(string tagName, in Html innerHtml, string childInfo2 = null) { + // FIXME: childInfo2 is ignored when info1 is null auto m = Element.make(tagName, cast(string) null, childInfo2); m.innerHTML = innerHtml.source; return m; @@ -1172,6 +1202,16 @@ class Element { return getAttribute(name); } + /* + // this would be nice for convenience, but it broke the getter above. + @property void opDispatch(string name)(bool boolean) if(name != "popFront") { + if(boolean) + setAttribute(name, name); + else + removeAttribute(name); + } + */ + /** Returns the element's children. */ @@ -1904,6 +1944,59 @@ class Element { @property ElementStream tree() { return new ElementStream(this); } + + + // I moved these from Form because they are generally useful. + // Ideally, I'd put them in arsd.html and use UFCS, but that doesn't work with the opDispatch here. + /// Tags: HTML, HTML5 + Element addField(string label, string name, string type = "text", FormFieldOptions fieldOptions = FormFieldOptions.none) { + auto fs = this; + auto i = fs.addChild("label"); + i.addChild("span", label); + Element input; + if(type == "textarea") + input = i.addChild("textarea"). + setAttribute("name", name). + setAttribute("rows", "6"); + else + input = i.addChild("input"). + setAttribute("name", name). + setAttribute("type", type); + + // these are html 5 attributes; you'll have to implement fallbacks elsewhere. In Javascript or maybe I'll add a magic thing to html.d later. + fieldOptions.applyToElement(input); + return i; + } + + Element addField(string label, string name, FormFieldOptions fieldOptions) { + return addField(label, name, "text", fieldOptions); + } + + Element addField(string label, string name, string[string] options, FormFieldOptions fieldOptions = FormFieldOptions.none) { + auto fs = this; + auto i = fs.addChild("label"); + i.addChild("span", label); + auto sel = i.addChild("select").setAttribute("name", name); + + foreach(k, opt; options) + sel.addChild("option", opt, k); + + // FIXME: implement requirements somehow + + return i; + } + + Element addSubmitButton(string label = null) { + auto t = this; + auto holder = t.addChild("div"); + holder.addClass("submit-holder"); + auto i = holder.addChild("input"); + i.type = "submit"; + if(label.length) + i.value = label; + return holder; + } + } ///. @@ -2069,6 +2162,7 @@ dchar parseEntity(in dchar[] entity) { } import std.utf; +import std.stdio; /// This takes a string of raw HTML and decodes the entities into a nice D utf-8 string. /// By default, it uses loose mode - it will try to return a useful string from garbage input too. @@ -2098,7 +2192,11 @@ string htmlEntitiesDecode(string data, bool strict = false) { // if not strict, let's try to parse both. - a ~= buffer[0.. std.utf.encode(buffer, parseEntity(entityBeingTried))]; + if(entityBeingTried == "&&") + a ~= "&"; // double amp means keep the first one, still try to parse the next one + else + a ~= buffer[0.. std.utf.encode(buffer, parseEntity(entityBeingTried))]; + // tryingEntity is still true entityBeingTried = entityBeingTried[0 .. 1]; // keep the & entityAttemptIndex = 0; // restarting o this @@ -2106,6 +2204,14 @@ string htmlEntitiesDecode(string data, bool strict = false) { if(ch == ';') { tryingEntity = false; a ~= buffer[0.. std.utf.encode(buffer, parseEntity(entityBeingTried))]; + } else if(ch == ' ') { + // e.g. you & i + if(strict) + throw new Exception("unterminated entity at " ~ to!string(entityBeingTried)); + else { + tryingEntity = false; + a ~= to!(char[])(entityBeingTried); + } } else { if(entityAttemptIndex >= 9) { if(strict) @@ -2128,6 +2234,15 @@ string htmlEntitiesDecode(string data, bool strict = false) { } } + if(tryingEntity) { + if(strict) + throw new Exception("unterminated entity at " ~ to!string(entityBeingTried)); + + // otherwise, let's try to recover, at least so we don't drop any data + a ~= to!string(entityBeingTried); + // FIXME: what if we have "cool &"? should we try to parse it? + } + return cast(string) a; // assumeUnique is actually kinda slow, lol } @@ -2376,44 +2491,29 @@ class Form : Element { tagName = "form"; } - Element addField(string label, string name, string type = "text") { - auto fs = this.querySelector("fieldset div"); - if(fs is null) fs = this; - auto i = fs.addChild("label"); - i.addChild("span", label); - if(type == "textarea") - i.addChild("textarea"). - setAttribute("name", name). - setAttribute("rows", "6"); + override Element addField(string label, string name, string type = "text", FormFieldOptions fieldOptions = FormFieldOptions.none) { + auto t = this.querySelector("fieldset div"); + if(t is null) + return super.addField(label, name, type, fieldOptions); else - i.addChild("input"). - setAttribute("name", name). - setAttribute("type", type); - - return i; + return t.addField(label, name, type, fieldOptions); } - Element addField(string label, string name, string[string] options) { - auto fs = this.querySelector("fieldset div"); - if(fs is null) fs = this; - auto i = fs.addChild("label"); - i.addChild("span", label); - auto sel = i.addChild("select").setAttribute("name", name); - - foreach(k, opt; options) - sel.addChild("option", opt, k); - - return i; + override Element addField(string label, string name, FormFieldOptions fieldOptions) { + auto type = "text"; + auto t = this.querySelector("fieldset div"); + if(t is null) + return super.addField(label, name, type, fieldOptions); + else + return t.addField(label, name, type, fieldOptions); } - Element addSubmitButton(string label = null) { - auto holder = this.addChild("div"); - holder.addClass("submit-holder"); - auto i = holder.addChild("input"); - i.type = "submit"; - if(label.length) - i.value = label; - return holder; + override Element addField(string label, string name, string[string] options, FormFieldOptions fieldOptions = FormFieldOptions.none) { + auto t = this.querySelector("fieldset div"); + if(t is null) + return super.addField(label, name, options, fieldOptions); + else + return t.addField(label, name, options, fieldOptions); } // FIXME: doesn't handle arrays; multiple fields can have the same name @@ -2680,6 +2780,16 @@ class Table : Element { return appendRowInternal("td", "tbody", t); } + void addColumnClasses(string[] classes...) { + auto grid = getGrid(); + foreach(row; grid) + foreach(i, cl; classes) { + if(cl.length) + if(i < row.length) + row[i].addClass(cl); + } + } + private Element appendRowInternal(T...)(string innerType, string findType, T t) { Element row = Element.make("tr"); @@ -3293,6 +3403,8 @@ class Document : FileResource { pos += 7; // FIXME: major malfunction possible here auto cdataStart = pos; + + // cdata isn't allowed to nest, so this should be generally ok, as long as it is found auto cdataEnd = pos + data[pos .. $].indexOf("]]>"); pos = cdataEnd + 3; @@ -5076,6 +5188,59 @@ class Event { } } +struct FormFieldOptions { + // usable for any + + /// this is a regex pattern used to validate the field + string pattern; + /// must the field be filled in? Even with a regex, it can be submitted blank if this is false. + bool isRequired; + /// this is displayed as an example to the user + string placeholder; + + // usable for numeric ones + + + // convenience methods to quickly get some options + static FormFieldOptions none() { + FormFieldOptions f; + return f; + } + + static FormFieldOptions required() { + FormFieldOptions f; + f.isRequired = true; + return f; + } + + static FormFieldOptions regex(string pattern, bool required = false) { + FormFieldOptions f; + f.pattern = pattern; + f.isRequired = required; + return f; + } + + static FormFieldOptions fromElement(Element e) { + FormFieldOptions f; + if(e.hasAttribute("required")) + f.isRequired = true; + if(e.hasAttribute("pattern")) + f.pattern = e.pattern; + if(e.hasAttribute("placeholder")) + f.placeholder = e.placeholder; + return f; + } + + Element applyToElement(Element e) { + if(this.isRequired) + e.required = "required"; + if(this.pattern.length) + e.pattern = this.pattern; + if(this.placeholder.length) + e.placeholder = this.placeholder; + return e; + } +} /* Copyright: Adam D. Ruppe, 2010 - 2012 diff --git a/html.d b/html.d index b0d3734..eed508e 100644 --- a/html.d +++ b/html.d @@ -259,7 +259,6 @@ Element checkbox(string name, string value, string label, bool checked = false) return lbl; } - /++ Convenience function to create a small