mirror of https://github.com/adamdruppe/arsd.git
web.d updates
This commit is contained in:
parent
7d7b37273d
commit
40fbab4e1f
7
dom.d
7
dom.d
|
@ -14,9 +14,11 @@
|
||||||
and adding spaces is easier than using LT macros everywhere.
|
and adding spaces is easier than using LT macros everywhere.
|
||||||
|
|
||||||
|
|
||||||
BTW: this file depends on arsd.characterencodings, so help it
|
BTW: this file optionally depends on arsd.characterencodings, to
|
||||||
correctly read files from the internet. You should be able to
|
help it correctly read files from the internet. You should be able to
|
||||||
get characterencodings.d from the same place you got this file.
|
get characterencodings.d from the same place you got this file.
|
||||||
|
|
||||||
|
If you want it to stand alone, just always use the `parseUtf8` function.
|
||||||
*/
|
*/
|
||||||
module arsd.dom;
|
module arsd.dom;
|
||||||
|
|
||||||
|
@ -38,6 +40,7 @@ bool isConvenientAttribute(string name) {
|
||||||
"src", "content", "pattern",
|
"src", "content", "pattern",
|
||||||
"placeholder", "required", "alt",
|
"placeholder", "required", "alt",
|
||||||
"rel",
|
"rel",
|
||||||
|
"method", "action", "enctype"
|
||||||
];
|
];
|
||||||
foreach(l; list)
|
foreach(l; list)
|
||||||
if(name == l) return true;
|
if(name == l) return true;
|
||||||
|
|
|
@ -271,7 +271,7 @@ version(exception_2_example) {
|
||||||
alias enforce = enforceBase!ExceptionBase;
|
alias enforce = enforceBase!ExceptionBase;
|
||||||
|
|
||||||
import core.stdc.stdio;
|
import core.stdc.stdio;
|
||||||
//auto fp = enforce!fopen("nofile.txt".ptr, "rb".ptr);
|
auto fp = enforce!fopen("nofile.txt".ptr, "rb".ptr);
|
||||||
|
|
||||||
|
|
||||||
// construct, passing it error details as data, not strings.
|
// construct, passing it error details as data, not strings.
|
||||||
|
|
|
@ -68,8 +68,12 @@ final class OpenGlTexture {
|
||||||
|
|
||||||
// For easy 2d drawing of it
|
// For easy 2d drawing of it
|
||||||
void draw(Point where, int width = 0, int height = 0, float rotation = 0.0, Color bg = Color.white) {
|
void draw(Point where, int width = 0, int height = 0, float rotation = 0.0, Color bg = Color.white) {
|
||||||
|
draw(where.x, where.y, width, height, rotation, bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw(float x, float y, int width = 0, int height = 0, float rotation = 0.0, Color bg = Color.white) {
|
||||||
glPushMatrix();
|
glPushMatrix();
|
||||||
glTranslatef(where.x, where.y, 0);
|
glTranslatef(x, y, 0);
|
||||||
glRotatef(rotation, 0,0, 1);
|
glRotatef(rotation, 0,0, 1);
|
||||||
|
|
||||||
if(width == 0)
|
if(width == 0)
|
||||||
|
|
4
jsvar.d
4
jsvar.d
|
@ -904,8 +904,7 @@ struct var {
|
||||||
return this.opEquals(var(t));
|
return this.opEquals(var(t));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool opEquals(T:var)(T t) const {
|
||||||
public bool opEquals(T:var)(T t) {
|
|
||||||
// FIXME: should this be == or === ?
|
// FIXME: should this be == or === ?
|
||||||
if(this._type != t._type)
|
if(this._type != t._type)
|
||||||
return false;
|
return false;
|
||||||
|
@ -1127,6 +1126,7 @@ struct var {
|
||||||
} else if(_type == Type.Object) {
|
} else if(_type == Type.Object) {
|
||||||
// objects might overload opIndex
|
// objects might overload opIndex
|
||||||
var* n = new var();
|
var* n = new var();
|
||||||
|
if("opIndex" in this)
|
||||||
*n = this["opIndex"](idx);
|
*n = this["opIndex"](idx);
|
||||||
return *n;
|
return *n;
|
||||||
}
|
}
|
||||||
|
|
31
minigui.d
31
minigui.d
|
@ -6,6 +6,37 @@
|
||||||
module arsd.minigui;
|
module arsd.minigui;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
||||||
|
The main goals of minigui.d are to:
|
||||||
|
1) Provide basic widgets that just work in a lightweight lib.
|
||||||
|
I basically want things comparable to a plain HTML form,
|
||||||
|
plus the easy and obvious things you expect from Windows
|
||||||
|
apps like a menu.
|
||||||
|
2) Use native things when possible for best functionality with
|
||||||
|
least library weight.
|
||||||
|
3) Give building blocks to provide easy extension for your
|
||||||
|
custom widgets, or hooking into additional native widgets
|
||||||
|
I didn't wrap.
|
||||||
|
4) Provide interfaces for easy interaction between third
|
||||||
|
party minigui extensions. (event model, perhaps
|
||||||
|
signals/slots, drop-in ease of use bits.)
|
||||||
|
5) Zero non-system dependencies, including Phobos as much as
|
||||||
|
I reasonably can. It must only import arsd.color and
|
||||||
|
my simpledisplay.d. If you need more, it will have to be
|
||||||
|
an extension module.
|
||||||
|
6) An easy layout system that generally works.
|
||||||
|
|
||||||
|
A stretch goal is to make it easy to make gui forms with code,
|
||||||
|
some kind of resource file (xml?) and even a wysiwyg designer.
|
||||||
|
|
||||||
|
Another stretch goal is to make it easy to hook data into the gui,
|
||||||
|
including from reflection. So like auto-generate a form from a
|
||||||
|
function signature or struct definition, or show a list from an
|
||||||
|
array that automatically updates as the array is changed. Then,
|
||||||
|
your program focuses on the data more than the gui interaction.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
STILL NEEDED:
|
STILL NEEDED:
|
||||||
* combo box. (this is diff than select because you can free-form edit too. more like a lineedit with autoselect)
|
* combo box. (this is diff than select because you can free-form edit too. more like a lineedit with autoselect)
|
||||||
* slider
|
* slider
|
||||||
|
|
|
@ -2457,6 +2457,7 @@ version(Windows) {
|
||||||
HWND hwnd;
|
HWND hwnd;
|
||||||
int oldWidth;
|
int oldWidth;
|
||||||
int oldHeight;
|
int oldHeight;
|
||||||
|
bool inSizeMove;
|
||||||
|
|
||||||
// the extern(Windows) wndproc should just forward to this
|
// the extern(Windows) wndproc should just forward to this
|
||||||
int windowProcedure(HWND hwnd, uint msg, WPARAM wParam, LPARAM lParam) {
|
int windowProcedure(HWND hwnd, uint msg, WPARAM wParam, LPARAM lParam) {
|
||||||
|
@ -2476,6 +2477,12 @@ version(Windows) {
|
||||||
case WM_SIZE:
|
case WM_SIZE:
|
||||||
width = LOWORD(lParam);
|
width = LOWORD(lParam);
|
||||||
height = HIWORD(lParam);
|
height = HIWORD(lParam);
|
||||||
|
|
||||||
|
// I want to avoid tearing in the windows (my code is inefficient
|
||||||
|
// so this is a hack around that) so while sizing, we don't trigger,
|
||||||
|
// but we do want to trigger on events like mazimize.
|
||||||
|
if(!inSizeMove)
|
||||||
|
goto size_changed;
|
||||||
break;
|
break;
|
||||||
// I don't like the tearing I get when redrawing on WM_SIZE
|
// I don't like the tearing I get when redrawing on WM_SIZE
|
||||||
// (I know there's other ways to fix that but I don't like that behavior anyway)
|
// (I know there's other ways to fix that but I don't like that behavior anyway)
|
||||||
|
@ -2483,12 +2490,16 @@ version(Windows) {
|
||||||
case 0x0231: /* WM_ENTERSIZEMOVE */
|
case 0x0231: /* WM_ENTERSIZEMOVE */
|
||||||
oldWidth = this.width;
|
oldWidth = this.width;
|
||||||
oldHeight = this.height;
|
oldHeight = this.height;
|
||||||
|
inSizeMove = true;
|
||||||
break;
|
break;
|
||||||
case 0x0232: /* WM_EXITSIZEMOVE */
|
case 0x0232: /* WM_EXITSIZEMOVE */
|
||||||
|
inSizeMove = false;
|
||||||
// nothing relevant changed, don't bother redrawing
|
// nothing relevant changed, don't bother redrawing
|
||||||
if(oldWidth == width && oldHeight == height)
|
if(oldWidth == width && oldHeight == height)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
size_changed:
|
||||||
|
|
||||||
// note: OpenGL windows don't use a backing bmp, so no need to change them
|
// note: OpenGL windows don't use a backing bmp, so no need to change them
|
||||||
// if resizability is anything other than allowResizing, it is meant to either stretch the one image or just do nothing
|
// if resizability is anything other than allowResizing, it is meant to either stretch the one image or just do nothing
|
||||||
if(openglMode == OpenGlOptions.no && resizability == Resizablity.allowResizing) {
|
if(openglMode == OpenGlOptions.no && resizability == Resizablity.allowResizing) {
|
||||||
|
|
139
terminal.d
139
terminal.d
|
@ -3104,6 +3104,145 @@ version(Windows) {
|
||||||
extern(Windows) HRESULT SHGetFolderPathA(HWND, int, HANDLE, DWORD, LPSTR);
|
extern(Windows) HRESULT SHGetFolderPathA(HWND, int, HANDLE, DWORD, LPSTR);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Like getting a line, printing a lot of lines is kinda important too, so I'm including
|
||||||
|
that widget here too. */
|
||||||
|
|
||||||
|
|
||||||
|
struct ScrollbackBuffer {
|
||||||
|
struct LineComponent {
|
||||||
|
string text;
|
||||||
|
int color = Color.DEFAULT;
|
||||||
|
int background = Color.DEFAULT;
|
||||||
|
void delegate() onclick;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Line {
|
||||||
|
LineComponent[] components;
|
||||||
|
int length() {
|
||||||
|
int l = 0;
|
||||||
|
foreach(c; components)
|
||||||
|
l += c.text.length;
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME: limit scrollback lines.length
|
||||||
|
|
||||||
|
Line[] lines;
|
||||||
|
string name;
|
||||||
|
|
||||||
|
int scrollbackPosition;
|
||||||
|
|
||||||
|
void drawInto(Terminal* terminal, in int x, in int y, in int width, in int height) {
|
||||||
|
if(lines.length == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* We need to figure out how much is going to fit
|
||||||
|
in a first pass, so we can figure out where to
|
||||||
|
start drawing */
|
||||||
|
|
||||||
|
int remaining = height + scrollbackPosition;
|
||||||
|
int start = lines.length;
|
||||||
|
int howMany = 0;
|
||||||
|
|
||||||
|
// we'll work backwards to figure out how much will fit...
|
||||||
|
// this will give accurate per-line things even with changing width and wrapping
|
||||||
|
// while being generally efficient - we usually want to show the end of the list
|
||||||
|
// anyway; actually using the scrollback is a bit of an exceptional case.
|
||||||
|
foreach_reverse(line; lines) {
|
||||||
|
int written = 0;
|
||||||
|
foreach(component; line.components) {
|
||||||
|
auto towrite = component.text;
|
||||||
|
written += towrite.length;
|
||||||
|
while(written > width) {
|
||||||
|
written -= width;
|
||||||
|
remaining--;
|
||||||
|
if(remaining <= 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
remaining--;
|
||||||
|
|
||||||
|
|
||||||
|
start--;
|
||||||
|
howMany++;
|
||||||
|
if(remaining <= 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// second pass: actually draw it
|
||||||
|
int linePos = remaining;
|
||||||
|
|
||||||
|
foreach(idx, line; lines[start .. start + howMany]) {
|
||||||
|
int written = 0;
|
||||||
|
|
||||||
|
terminal.moveTo(x, y + ((linePos >= 0) ? linePos : 0));
|
||||||
|
foreach(component; line.components) {
|
||||||
|
terminal.color(component.color, component.background);
|
||||||
|
auto towrite = component.text;
|
||||||
|
while(linePos < 0 && width < towrite.length) {
|
||||||
|
towrite = towrite[width .. $];
|
||||||
|
linePos++;
|
||||||
|
}
|
||||||
|
terminal.write(towrite);
|
||||||
|
written += towrite.length;
|
||||||
|
while(written > width) {
|
||||||
|
written -= width;
|
||||||
|
linePos++;
|
||||||
|
if(linePos >= height)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(written < width)
|
||||||
|
foreach(i; written .. width)
|
||||||
|
terminal.write(" ");
|
||||||
|
|
||||||
|
linePos++;
|
||||||
|
|
||||||
|
if(linePos >= height)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach(i; linePos .. height) {
|
||||||
|
if(i >= 0 && i < height) {
|
||||||
|
terminal.moveTo(x, y + i);
|
||||||
|
foreach(w; 0 .. width)
|
||||||
|
terminal.write(" ");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void addLine(string line) {
|
||||||
|
lines ~= Line([LineComponent(line)]);
|
||||||
|
}
|
||||||
|
|
||||||
|
void scrollUp(int lines = 1) {
|
||||||
|
scrollbackPosition += lines;
|
||||||
|
if(scrollbackPosition >= this.lines.length)
|
||||||
|
scrollbackPosition = this.lines.length - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void scrollDown(int lines = 1) {
|
||||||
|
scrollbackPosition -= lines;
|
||||||
|
if(scrollbackPosition < 0)
|
||||||
|
scrollbackPosition = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// does scrolling via wheel and keyboard and also clicks on content
|
||||||
|
void handleEvent() {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
||||||
// more efficient scrolling
|
// more efficient scrolling
|
||||||
|
|
6
web.d
6
web.d
|
@ -1843,12 +1843,12 @@ Form createAutomaticForm(Document document, string action, in Parameter[] parame
|
||||||
input.name = param.name;
|
input.name = param.name;
|
||||||
input.innerText = param.value;
|
input.innerText = param.value;
|
||||||
|
|
||||||
input.rows = "7";
|
input.attrs.rows = "7";
|
||||||
|
|
||||||
auto idx = type.indexOf("-");
|
auto idx = type.indexOf("-");
|
||||||
if(idx != -1) {
|
if(idx != -1) {
|
||||||
idx++;
|
idx++;
|
||||||
input.rows = type[idx .. $];
|
input.attrs.rows = type[idx .. $];
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
input = Element.make("input");
|
input = Element.make("input");
|
||||||
|
@ -4021,7 +4021,7 @@ void translateQsa(Document document, Cgi cgi, string logicalScriptName = null) {
|
||||||
string[][string] vars;
|
string[][string] vars;
|
||||||
foreach(k, v; cgi.getArray)
|
foreach(k, v; cgi.getArray)
|
||||||
vars[k] = cast(string[]) v;
|
vars[k] = cast(string[]) v;
|
||||||
foreach(k, v; decodeVariablesSingle(a.qsa)) {
|
foreach(k, v; decodeVariablesSingle(a.attrs.qsa)) {
|
||||||
if(k in cgi.get && cgi.get[k] == v)
|
if(k in cgi.get && cgi.get[k] == v)
|
||||||
matches++;
|
matches++;
|
||||||
possibilities++;
|
possibilities++;
|
||||||
|
|
Loading…
Reference in New Issue