mirror of https://github.com/adamdruppe/arsd.git
stuff
This commit is contained in:
parent
c38f20fd7b
commit
d614f48642
3
cgi.d
3
cgi.d
|
@ -2191,6 +2191,9 @@ string toHexUpper(long num) {
|
|||
ret ~= d;
|
||||
}
|
||||
|
||||
if(ret.length == 1)
|
||||
ret ~= "0"; // url encoding requires two digits and that's what this function is used for...
|
||||
|
||||
return to!string(array(ret.retro));
|
||||
}
|
||||
|
||||
|
|
|
@ -46,7 +46,10 @@ import std.conv;
|
|||
/// Like convertToUtf8, but if the encoding is unknown, it just strips all chars > 127 and calls it done instead of throwing
|
||||
string convertToUtf8Lossy(immutable(ubyte)[] data, string dataCharacterEncoding) {
|
||||
try {
|
||||
return convertToUtf8(data, dataCharacterEncoding);
|
||||
auto ret = convertToUtf8(data, dataCharacterEncoding);
|
||||
import std.utf;
|
||||
validate(ret);
|
||||
return ret;
|
||||
} catch(Exception e) {
|
||||
string ret;
|
||||
foreach(b; data)
|
||||
|
|
8
dom.d
8
dom.d
|
@ -126,7 +126,7 @@ mixin template DomConvenienceFunctions() {
|
|||
|
||||
/// Returns whether the given class appears in this element.
|
||||
bool hasClass(string c) {
|
||||
auto cn = className;
|
||||
string cn = className;
|
||||
|
||||
auto idx = cn.indexOf(c);
|
||||
if(idx == -1)
|
||||
|
@ -1268,7 +1268,7 @@ class Element {
|
|||
/// This sets the style attribute with a string.
|
||||
@property ElementStyle style(string s) {
|
||||
this.setAttribute("style", s);
|
||||
return this.style();
|
||||
return this.style;
|
||||
}
|
||||
|
||||
private void parseAttributes(string[] whichOnes = null) {
|
||||
|
@ -1639,7 +1639,7 @@ class Element {
|
|||
|
||||
/// ditto
|
||||
@property Element innerHTML(Html html) {
|
||||
return this.innerHTML(html.source);
|
||||
return this.innerHTML = html.source;
|
||||
}
|
||||
|
||||
private void reparentTreeDocuments() {
|
||||
|
@ -1847,7 +1847,7 @@ class Element {
|
|||
Same result as innerText; the tag with all inner tags stripped out
|
||||
*/
|
||||
string outerText() const {
|
||||
return innerText();
|
||||
return innerText;
|
||||
}
|
||||
|
||||
|
||||
|
|
20
english.d
20
english.d
|
@ -1,5 +1,25 @@
|
|||
module arsd.english;
|
||||
|
||||
string plural(int count, string word, string pluralWord = null) {
|
||||
if(count == 1 || word.length == 0)
|
||||
return word; // it isn't actually plural
|
||||
|
||||
if(pluralWord !is null)
|
||||
return pluralWord;
|
||||
|
||||
switch(word[$ - 1]) {
|
||||
case 's':
|
||||
case 'a', 'e', 'i', 'o', 'u':
|
||||
return word ~ "es";
|
||||
case 'f':
|
||||
return word[0 .. $-1] ~ "ves";
|
||||
case 'y':
|
||||
return word[0 .. $-1] ~ "ies";
|
||||
default:
|
||||
return word ~ "s";
|
||||
}
|
||||
}
|
||||
|
||||
string numberToEnglish(long number) {
|
||||
string word;
|
||||
if(number == 0)
|
||||
|
|
2
html.d
2
html.d
|
@ -74,7 +74,7 @@ string[] htmlAttributeWhitelist = [
|
|||
|
||||
"href", "src", "type", "name",
|
||||
"id",
|
||||
"method", "enctype", // for forms only FIXME
|
||||
"method", "enctype", "value", "type", // for forms only FIXME
|
||||
|
||||
"align", "valign", "width", "height",
|
||||
];
|
||||
|
|
|
@ -0,0 +1,315 @@
|
|||
module mangle;
|
||||
|
||||
import std.conv;
|
||||
|
||||
static immutable string[23] primitives = [
|
||||
"char", // a
|
||||
"bool", // b
|
||||
"creal", // c
|
||||
"double", // d
|
||||
"real", // e
|
||||
"float", // f
|
||||
"byte", // g
|
||||
"ubyte", // h
|
||||
"int", // i
|
||||
"ireal", // j
|
||||
"uint", // k
|
||||
"long", // l
|
||||
"ulong", // m
|
||||
null, // n
|
||||
"ifloat", // o
|
||||
"idouble", // p
|
||||
"cfloat", // q
|
||||
"cdouble", // r
|
||||
"short", // s
|
||||
"ushort", // t
|
||||
"wchar", // u
|
||||
"void", // v
|
||||
"dchar", // w
|
||||
];
|
||||
|
||||
|
||||
|
||||
char manglePrimitive(in char[] t) {
|
||||
foreach(i, p; primitives)
|
||||
if(p == t)
|
||||
return cast(char) ('a' + i);
|
||||
return 0;
|
||||
}
|
||||
|
||||
import std.algorithm;
|
||||
import std.array;
|
||||
|
||||
bool isIdentifierChar(char c) {
|
||||
// FIXME: match the D spec
|
||||
return c == '_' || (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9');
|
||||
}
|
||||
|
||||
struct StackArray(Type, size_t capacity) {
|
||||
Type[capacity] buffer;
|
||||
size_t length;
|
||||
Type[] slice() { return buffer[0 .. length]; }
|
||||
void opOpAssign(string op : "~")(Type rhs, string file = __FILE__, size_t line = __LINE__) {
|
||||
if(length >= capacity) {
|
||||
throw new Error("no more room", file, line);
|
||||
}
|
||||
buffer[length] = rhs;
|
||||
length++;
|
||||
}
|
||||
}
|
||||
|
||||
char[] mangle(const(char)[] decl, char[] buffer) {
|
||||
|
||||
// FIXME: using this will allocate at *runtime*! Unbelievable.
|
||||
// it does that even if everything is enum
|
||||
auto dTokens = (sort!"a.length > b.length"(
|
||||
primitives ~
|
||||
[
|
||||
"(",
|
||||
")",
|
||||
".",
|
||||
",",
|
||||
"!",
|
||||
"[",
|
||||
"]",
|
||||
"*",
|
||||
"const",
|
||||
"immutable",
|
||||
"shared",
|
||||
"extern",
|
||||
]));
|
||||
|
||||
|
||||
|
||||
StackArray!(const(char)[], 128) tokensBuffer;
|
||||
main: while(decl.length) {
|
||||
if(decl[0] == ' ' || decl[0] == '\t' || decl[0] == '\n') {
|
||||
decl = decl[1 .. $];
|
||||
continue;
|
||||
}
|
||||
|
||||
foreach(token; dTokens) {
|
||||
if(token is null) continue;
|
||||
if(decl.length >= token.length && decl[0 .. token.length] == token) {
|
||||
// make sure this isn't an identifier that coincidentally starts with a keyword
|
||||
if(decl.length == token.length || !token[$ - 1].isIdentifierChar() || !decl[token.length].isIdentifierChar()) {
|
||||
tokensBuffer ~= token;
|
||||
decl = decl[token.length .. $];
|
||||
continue main;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// could be an identifier or literal
|
||||
|
||||
int pos = 0;
|
||||
while(pos < decl.length && decl[pos].isIdentifierChar)
|
||||
pos++;
|
||||
tokensBuffer ~= decl[0 .. pos];
|
||||
decl = decl[pos .. $];
|
||||
continue main;
|
||||
|
||||
// FIXME: literals should be handled too
|
||||
}
|
||||
|
||||
assert(decl.length == 0); // we should have consumed all the input into tokens
|
||||
|
||||
auto tokens = tokensBuffer.slice();
|
||||
|
||||
|
||||
char[64] returnTypeBuffer;
|
||||
auto returnType = parseAndMangleType(tokens, returnTypeBuffer);
|
||||
char[256] nameBuffer;
|
||||
auto name = parseName(tokens, nameBuffer[]);
|
||||
StackArray!(const(char)[], 64) arguments;
|
||||
// FIXME: templates and other types of thing should be handled
|
||||
assert(tokens[0] == "(", "other stuff not implemented " ~ tokens[0]);
|
||||
tokens = tokens[1 .. $];
|
||||
|
||||
char[64][32] argTypeBuffers;
|
||||
int i = 0;
|
||||
|
||||
while(tokens[0] != ")") {
|
||||
arguments ~= parseAndMangleType(tokens, argTypeBuffers[i]);
|
||||
i++;
|
||||
if(tokens[0] == ",")
|
||||
tokens = tokens[1 .. $];
|
||||
}
|
||||
|
||||
assert(tokens[0] == ")", "other stuff not implemented");
|
||||
|
||||
return mangleFunction(name, returnType, arguments.slice(), buffer);
|
||||
}
|
||||
|
||||
char[] parseName(ref const(char)[][] tokens, char[] nameBuffer) {
|
||||
size_t where = 0;
|
||||
more:
|
||||
nameBuffer[where .. where + tokens[0].length] = tokens[0][];
|
||||
where += tokens[0].length;
|
||||
tokens = tokens[1 .. $];
|
||||
if(tokens.length && tokens[0] == ".") {
|
||||
tokens = tokens[1 .. $];
|
||||
nameBuffer[where++] = '.';
|
||||
goto more;
|
||||
}
|
||||
|
||||
return nameBuffer[0 .. where];
|
||||
}
|
||||
|
||||
char[] intToString(int i, char[] buffer) {
|
||||
int pos = cast(int) buffer.length - 1;
|
||||
|
||||
if(i == 0) {
|
||||
buffer[pos] = '0';
|
||||
pos--;
|
||||
}
|
||||
|
||||
while(pos > 0 && i) {
|
||||
buffer[pos] = (i % 10) + '0';
|
||||
pos--;
|
||||
i /= 10;
|
||||
}
|
||||
|
||||
return buffer[pos + 1 .. $];
|
||||
}
|
||||
|
||||
|
||||
|
||||
char[] mangleName(in char[] name, char[] buffer) {
|
||||
import std.algorithm;
|
||||
import std.conv;
|
||||
|
||||
auto parts = name.splitter(".");
|
||||
|
||||
int bufferPos = 0;
|
||||
foreach(part; parts) {
|
||||
char[16] numberBuffer;
|
||||
auto number = intToString(cast(int) part.length, numberBuffer);
|
||||
|
||||
buffer[bufferPos .. bufferPos + number.length] = number[];
|
||||
bufferPos += number.length;
|
||||
|
||||
buffer[bufferPos .. bufferPos + part.length] = part[];
|
||||
bufferPos += part.length;
|
||||
}
|
||||
|
||||
return buffer[0 .. bufferPos];
|
||||
}
|
||||
|
||||
char[] mangleFunction(in char[] name, in char[] returnTypeMangled, in char[][] argumentsMangle, char[] buffer) {
|
||||
int bufferPos = 0;
|
||||
buffer[bufferPos++] = '_';
|
||||
buffer[bufferPos++] = 'D';
|
||||
|
||||
char[256] nameBuffer;
|
||||
auto mn = mangleName(name, nameBuffer);
|
||||
buffer[bufferPos .. bufferPos + mn.length] = mn[];
|
||||
bufferPos += mn.length;
|
||||
|
||||
buffer[bufferPos++] = 'F';
|
||||
foreach(arg; argumentsMangle) {
|
||||
buffer[bufferPos .. bufferPos + arg.length] = arg[];
|
||||
bufferPos += arg.length;
|
||||
}
|
||||
buffer[bufferPos++] = 'Z';
|
||||
buffer[bufferPos .. bufferPos + returnTypeMangled.length] = returnTypeMangled[];
|
||||
bufferPos += returnTypeMangled.length;
|
||||
|
||||
return buffer[0 .. bufferPos];
|
||||
}
|
||||
|
||||
char[] parseAndMangleType(ref const(char)[][] tokens, char[] buffer) {
|
||||
assert(tokens.length);
|
||||
|
||||
int bufferPos = 0;
|
||||
|
||||
void prepend(char p) {
|
||||
for(int i = bufferPos; i > 0; i--) {
|
||||
buffer[i] = buffer[i - 1];
|
||||
}
|
||||
buffer[0] = p;
|
||||
bufferPos++;
|
||||
}
|
||||
|
||||
// FIXME: handle all the random D type constructors
|
||||
if(tokens[0] == "const" || tokens[0] == "immutable") {
|
||||
if(tokens[0] == "const")
|
||||
buffer[bufferPos++] = 'x';
|
||||
else if(tokens[0] == "immutable")
|
||||
buffer[bufferPos++] = 'y';
|
||||
tokens = tokens[1 .. $];
|
||||
assert(tokens[0] == "(");
|
||||
tokens = tokens[1 .. $];
|
||||
auto next = parseAndMangleType(tokens, buffer[bufferPos .. $]);
|
||||
bufferPos += next.length;
|
||||
assert(tokens[0] == ")");
|
||||
tokens = tokens[1 .. $];
|
||||
} else {
|
||||
char primitive = manglePrimitive(tokens[0]);
|
||||
if(primitive) {
|
||||
buffer[bufferPos++] = primitive;
|
||||
tokens = tokens[1 .. $];
|
||||
} else {
|
||||
// probably a struct or something, parse it as an identifier
|
||||
// FIXME
|
||||
char[256] nameBuffer;
|
||||
auto name = parseName(tokens, nameBuffer[]);
|
||||
|
||||
char[256] mangledNameBuffer;
|
||||
auto mn = mangleName(name, mangledNameBuffer);
|
||||
|
||||
buffer[bufferPos++] = 'S';
|
||||
buffer[bufferPos .. bufferPos + mn.length] = mn[];
|
||||
bufferPos += mn.length;
|
||||
}
|
||||
}
|
||||
|
||||
while(tokens.length) {
|
||||
if(tokens[0] == "[") {
|
||||
tokens = tokens[1 .. $];
|
||||
prepend('A');
|
||||
assert(tokens[0] == "]", "other array not implemented");
|
||||
tokens = tokens[1 .. $];
|
||||
} else if(tokens[0] == "*") {
|
||||
prepend('P');
|
||||
tokens = tokens[1 .. $];
|
||||
} else break;
|
||||
}
|
||||
|
||||
return buffer[0 .. bufferPos];
|
||||
}
|
||||
|
||||
version(unittest) {
|
||||
int foo(int, string, int);
|
||||
string foo2(long, char[], int);
|
||||
struct S { int a; string b; }
|
||||
S foo3(S, S, string, long, int, S, int[], char[][]);
|
||||
long testcomplex(int, const(const(char)[]*)[], long);
|
||||
}
|
||||
unittest {
|
||||
import core.demangle;
|
||||
char[512] buffer;
|
||||
|
||||
import std.stdio;
|
||||
assert(mangle(demangle(foo.mangleof), buffer) == foo.mangleof);
|
||||
assert(mangle(demangle(foo2.mangleof), buffer) == foo2.mangleof);
|
||||
assert(mangle(demangle(foo3.mangleof), buffer) == foo3.mangleof);
|
||||
|
||||
assert(mangle(demangle(testcomplex.mangleof), buffer) == testcomplex.mangleof);
|
||||
// FIXME: these all fail if the functions are defined inside the unittest{} block
|
||||
// so still something wrong parsing those complex names or something
|
||||
}
|
||||
|
||||
// _D6test303fooFiAyaZi
|
||||
// _D6test303fooFiAyaZi
|
||||
|
||||
version(unittest)
|
||||
void main(string[] args) {
|
||||
|
||||
char[512] buffer;
|
||||
import std.stdio;
|
||||
if(args.length > 1)
|
||||
writeln(mangle(args[1], buffer));
|
||||
else
|
||||
writeln(mangle("int test30.foo(int, immutable(char)[])", buffer));
|
||||
}
|
|
@ -719,11 +719,9 @@ struct Terminal {
|
|||
*/
|
||||
|
||||
/// Writes to the terminal at the current cursor position.
|
||||
///
|
||||
/// Uses std.string.xformat for the format string handling.
|
||||
void writef(T...)(string f, T t) {
|
||||
import std.string;
|
||||
writePrintableString(xformat(f, t));
|
||||
writePrintableString(format(f, t));
|
||||
}
|
||||
|
||||
/// ditto
|
||||
|
@ -753,7 +751,7 @@ struct Terminal {
|
|||
/// Might give better performance than moveTo/writef because if the data to write matches the internal buffer, it skips sending anything (to override the buffer check, you can use moveTo and writePrintableString with ForceOption.alwaysSend)
|
||||
void writefAt(T...)(int x, int y, string f, T t) {
|
||||
import std.string;
|
||||
auto toWrite = xformat(f, t);
|
||||
auto toWrite = format(f, t);
|
||||
|
||||
auto oldX = _cursorX;
|
||||
auto oldY = _cursorY;
|
||||
|
|
Loading…
Reference in New Issue