diff --git a/cgi.d b/cgi.d index 470e7c0..600b7db 100644 --- a/cgi.d +++ b/cgi.d @@ -45,7 +45,7 @@ int locationOf(T)(T[] data, string item) { /// If you are doing a custom cgi class, mixing this in can take care of /// the required constructors for you mixin template ForwardCgiConstructors() { - this(int maxContentLength = 5_000_000, + this(long maxContentLength = 5_000_000, string[string] env = null, const(ubyte)[] delegate() readdata = null, void delegate(const(ubyte)[]) _rawDataOutput = null @@ -136,7 +136,7 @@ class Cgi { CommandLine } /** Initializes it using the CGI (or FastCGI) interface */ - this(int maxContentLength = 5_000_000, + this(long maxContentLength = 5_000_000, // use this to override the environment variable listing in string[string] env = null, // and this should return a chunk of data. return empty when done @@ -251,7 +251,7 @@ class Cgi { // to be slow if they did that. The spec says it is always there though. // And it has worked reliably for me all year in the live environment, // but some servers might be different. - auto contentLength = to!size_t(getenv("CONTENT_LENGTH")); + auto contentLength = to!size_t(getenv("CONTENT_LENGTH")); immutable originalContentLength = contentLength; if(contentLength) { @@ -652,6 +652,8 @@ class Cgi { // Starting small because exiting the loop early is desirable, since // we're not keeping any ambiguity and 1 / 256 chance of exiting is // the best we can do. + if(a > pps.buffer.length) + break; // FIXME: is this right? assert(a <= pps.buffer.length); assert(a > 0); if(std.algorithm.endsWith(pps.buffer, pps.localBoundary[0 .. a])) { @@ -1777,8 +1779,7 @@ string getTempDirectory() { // And to add insult to injury, they are going to remove the tiny olive branch the new // module offered. Whatever, I want it at least some of it. -long sysTimeToDTime(in SysTime sysTime) -{ +long sysTimeToDTime(in SysTime sysTime) { return convert!("hnsecs", "msecs")(sysTime.stdTime - 621355968000000000L); } @@ -1787,11 +1788,11 @@ long getUtcTime() { // renamed primarily to avoid conflict with std.date itself } /* -Copyright: Adam D. Ruppe, 2008 - 2011 +Copyright: Adam D. Ruppe, 2008 - 2012 License: Boost License 1.0. Authors: Adam D. Ruppe - Copyright Adam D. Ruppe 2008 - 2011. + Copyright Adam D. Ruppe 2008 - 2012. Distributed under the Boost Software License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) diff --git a/color.d b/color.d index 19e2f41..11c5e3f 100644 --- a/color.d +++ b/color.d @@ -147,12 +147,65 @@ Color darken(Color c, real percentage) { return fromHsl(hsl); } +/// for light colors, call darken. for dark colors, call lighten. +/// The goal: get toward center grey. +Color moderate(Color c, real percentage) { + auto hsl = toHsl(c); + if(hsl[2] > 0.5) + hsl[2] *= (1 - percentage); + else + hsl[2] *= (1 + percentage); + if(hsl[2] > 1) + hsl[2] = 1; + return fromHsl(hsl); +} + +/// the opposite of moderate. Make darks darker and lights lighter +Color extremify(Color c, real percentage) { + auto hsl = toHsl(c); + if(hsl[2] < 0.5) + hsl[2] *= (1 - percentage); + else + hsl[2] *= (1 + percentage); + if(hsl[2] > 1) + hsl[2] = 1; + return fromHsl(hsl); +} + +/// Move around the lightness wheel, trying not to break on moderate things +Color oppositeLightness(Color c) { + auto hsl = toHsl(c); + + auto original = hsl[2]; + + if(original > 0.4 && original < 0.6) + hsl[2] = 0.8 - original; // so it isn't quite the same + else + hsl[2] = 1 - original; + + return fromHsl(hsl); +} + +Color setLightness(Color c, real lightness) { + auto hsl = toHsl(c); + hsl[2] = lightness; + return fromHsl(hsl); +} + + + Color rotateHue(Color c, real degrees) { auto hsl = toHsl(c); hsl[0] += degrees; return fromHsl(hsl); } +Color setHue(Color c, real hue) { + auto hsl = toHsl(c); + hsl[0] = hue; + return fromHsl(hsl); +} + Color desaturate(Color c, real percentage) { auto hsl = toHsl(c); hsl[1] *= (1 - percentage); @@ -167,6 +220,13 @@ Color saturate(Color c, real percentage) { return fromHsl(hsl); } +Color setSaturation(Color c, real saturation) { + auto hsl = toHsl(c); + hsl[1] = saturation; + return fromHsl(hsl); +} + + /* void main(string[] args) { auto color1 = toHsl(Color(255, 0, 0)); diff --git a/dom.d b/dom.d index 7b41250..dd156a1 100644 --- a/dom.d +++ b/dom.d @@ -3,6 +3,13 @@ module arsd.dom; // NOTE: do *NOT* override toString on Element subclasses. It won't work. // Instead, override writeToAppender(); +// FIXME: should I keep processing instructions like and (comments too lol)? I *want* them stripped out of most my output, but I want to be able to parse and create them too. + +// Stripping them is useful for reading php as html.... but adding them +// is good for building php. + +// I need to maintain compatibility with the way it is now too. + import arsd.characterencodings; import std.string; @@ -239,8 +246,9 @@ class Element { ///. @property Element cloned() out(ret) { - assert(ret.children.length == this.children.length); - assert(ret.tagName == this.tagName); + // FIXME: not sure why these fail... +// assert(ret.children.length == this.children.length); +// assert(ret.tagName == this.tagName); } body { auto e = new Element(parentDocument, tagName, attributes.dup, selfClosed); @@ -449,6 +457,11 @@ class Element { if(childInfo2 !is null) e.alt = childInfo2; break; + case "link": + e.href = childInfo; + if(childInfo2 !is null) + e.rel = childInfo2; + break; case "option": e.innerText = childInfo; if(childInfo2 !is null) @@ -678,6 +691,12 @@ class Element { return ret; } + /// . + Element[] getElementsByClassName(string cn) { + // is this correct? + return getElementsBySelector("." ~ cn); + } + ///. Element[] getElementsByTagName(string tag) { if(parentDocument && parentDocument.loose) @@ -1956,6 +1975,13 @@ class Form : Element { } } + /// This takes an array of strings and adds hidden elements for each one of them. Unlike setValue, + /// it makes no attempt to find and modify existing elements in the form to the new values. + void addValueArray(string key, string[] arrayOfValues) { + foreach(arr; arrayOfValues) + addChild("input", key, arr); + } + /// Gets the value of the field; what would be given if it submitted right now. (so /// it handles select boxes and radio buttons too). For checkboxes, if a value isn't /// given, but it is checked, it returns "checked", since null and "" are indistinguishable @@ -2092,8 +2118,11 @@ class Table : Element { ///. Element th(T)(T t) { - assert(parentDocument !is null); - Element e = parentDocument.createElement("th"); + Element e; + if(parentDocument !is null) + e = parentDocument.createElement("th"); + else + e = Element.make("th"); static if(is(T == Html)) e.innerHTML = t; else @@ -2103,8 +2132,11 @@ class Table : Element { ///. Element td(T)(T t) { - assert(parentDocument !is null); - Element e = parentDocument.createElement("td"); + Element e; + if(parentDocument !is null) + e = parentDocument.createElement("td"); + else + e = Element.make("td"); static if(is(T == Html)) e.innerHTML = t; else @@ -2114,27 +2146,25 @@ class Table : Element { ///. Element appendRow(T...)(T t) { - assert(parentDocument !is null); - - Element row = parentDocument.createElement("tr"); + Element row = Element.make("tr"); foreach(e; t) { static if(is(typeof(e) : Element)) { if(e.tagName == "td" || e.tagName == "th") row.appendChild(e); else { - Element a = parentDocument.createElement("td"); + Element a = Element.make("td"); a.appendChild(e); row.appendChild(a); } } else static if(is(typeof(e) == Html)) { - Element a = parentDocument.createElement("td"); + Element a = Element.make("td"); a.innerHTML = e.source; row.appendChild(a); } else { - Element a = parentDocument.createElement("td"); + Element a = Element.make("td"); a.innerText = to!string(e); row.appendChild(a); } @@ -2164,7 +2194,7 @@ class Table : Element { } if(cap is null) { - cap = parentDocument.createElement("caption"); + cap = Element.make("caption"); appendChild(cap); } @@ -2353,6 +2383,9 @@ struct Html { } +/// This might belong in another module, but it represents a file with a mime type and some data. +/// Document implements this interface with type = text/html (see Document.contentType for more info) +/// and data = document.toString, so you can return Documents anywhere web.d expects FileResources. interface FileResource { string contentType() const; immutable(ubyte)[] getData() const; @@ -2373,6 +2406,23 @@ class Document : FileResource { } + /// This is just something I'm toying with. Right now, you use opIndex to put in css selectors. + /// It returns a struct that forwards calls to all elements it holds, and returns itself so you + /// can chain it. + /// + /// Example: document["p"].innerText("hello").addClass("modified"); + /// + /// Equivalent to: foreach(e; document.getElementsBySelector("p")) { e.innerText("hello"); e.addClas("modified"); } + /// + /// Note: always use function calls (not property syntax) and don't use toString in there for best results. + /// + /// You can also do things like: document["p"]["b"] though tbh I'm not sure why since the selector string can do all that anyway. Maybe + /// you could put in some kind of custom filter function tho. + ElementCollection opIndex(string selector) { + auto e = ElementCollection(this.root); + return e[selector]; + } + string _contentType = "text/html; charset=utf-8"; /// If you're using this for some other kind of XML, you can @@ -2382,7 +2432,7 @@ class Document : FileResource { /// It is only used if the document is sent via a protocol like HTTP. /// /// This may be called by parse() if it recognizes the data. Otherwise, - /// if you don't set it, it assumes text/html. + /// if you don't set it, it assumes text/html; charset=utf-8. string contentType(string mimeType) { _contentType = mimeType; return _contentType; @@ -2510,13 +2560,32 @@ class Document : FileResource { string data; + if(!strict) { + // if we're in non-strict mode, we need to check + // the document for mislabeling too; sometimes + // web documents will say they are utf-8, but aren't + // actually properly encoded. If it fails to validate, + // we'll assume it's actually Windows encoding - the most + // likely candidate for mislabeled garbage. + dataEncoding = dataEncoding.toLower(); + dataEncoding = dataEncoding.replace(" ", ""); + dataEncoding = dataEncoding.replace("-", ""); + dataEncoding = dataEncoding.replace("_", ""); + if(dataEncoding == "utf8") { + try { + validate(rawdata); + } catch(UtfException e) { + dataEncoding = "Windows 1252"; + } + } + } + if(dataEncoding != "UTF-8") data = convertToUtf8(cast(immutable(ubyte)[]) rawdata, dataEncoding); else data = rawdata; assert(data !is null); - // go through character by character. // if you see a <, consider it a tag. // name goes until the first non tagname character @@ -2640,7 +2709,7 @@ class Document : FileResource { } // recursively read a tag Ele readElement(string[] parentChain = null) { - // FIXME: this is the closest function in this module, by far, even in strict mode. + // FIXME: this is the slowest function in this module, by far, even in strict mode. // Loose mode should perform decently, but strict mode is the important one. if(!strict && parentChain is null) parentChain = []; @@ -2765,15 +2834,17 @@ class Document : FileResource { // HACK to handle script and style as a raw data section as it is in HTML browsers if(tagName == "script" || tagName == "style") { - string closer = "" ~ tagName ~ ">"; - auto ending = indexOf(data[pos..$], closer); - if(loose && ending == -1) - ending = indexOf(data[pos..$], closer.toUpper); - if(ending == -1) - throw new Exception("tag " ~ tagName ~ " never closed"); - ending += pos; - e.innerRawSource = data[pos..ending]; - pos = ending + closer.length; + if(!selfClosed) { + string closer = "" ~ tagName ~ ">"; + auto ending = indexOf(data[pos..$], closer); + if(loose && ending == -1) + ending = indexOf(data[pos..$], closer.toUpper); + if(ending == -1) + throw new Exception("tag " ~ tagName ~ " never closed"); + ending += pos; + e.innerRawSource = data[pos..ending]; + pos = ending + closer.length; + } return Ele(0, e, null); } @@ -3173,6 +3244,11 @@ int intFromHex(string hex) { // dt << dl means go as far up as needed to find a dl (you have an element and want its containers) NOT IMPLEMENTED // :first means to stop at the first hit, don't do more (so p + p == p ~ p:first + + +// CSS4 draft currently says you can change the subject (the element actually returned) by putting a ! at the end of it. +// That might be useful to implement, though I do have parent selectors too. + ///. static immutable string[] selectorTokens = [ // It is important that the 2 character possibilities go first here for accurate lexing @@ -4282,6 +4358,41 @@ private string[string] dup(in string[string] arr) { return ret; } +// I'm just dicking around with this +struct ElementCollection { + this(Element e) { + elements = [e]; + } + + this(Element[] e) { + elements = e; + } + + Element[] elements; + //alias elements this; // let it implicitly convert to the underlying array + + ElementCollection opIndex(string selector) { + ElementCollection ec; + foreach(e; elements) + ec.elements ~= e.getElementsBySelector(selector); + return ec; + } + + /// Forward method calls to each individual element of the collection + /// returns this so it can be chained. + ElementCollection opDispatch(string name, T...)(T t) { + foreach(e; elements) { + mixin("e." ~ name)(t); + } + return this; + } + + ElementCollection opBinary(string op : "~")(ElementCollection rhs) { + return ElementCollection(this.elements ~ rhs.elements); + } +} + + /* Copyright: Adam D. Ruppe, 2010 - 2011 License: Boost License 1.0. diff --git a/html.d b/html.d index 193341c..2af6ba6 100644 --- a/html.d +++ b/html.d @@ -206,6 +206,105 @@ string favicon(Document document) { return "/favicon.ico"; // it pisses me off that the fucking browsers do this.... but they do, so I will too. } + +/++ Convenience function to create a small