diff --git a/bmp.d b/bmp.d index a8ecc0f..d5a0d59 100644 --- a/bmp.d +++ b/bmp.d @@ -1,16 +1,40 @@ -import core.stdc.stdio; +module arsd.bmp; import arsd.color; MemoryImage readBmp(string filename) { + import core.stdc.stdio; + FILE* fp = fopen((filename ~ "\0").ptr, "rb".ptr); if(fp is null) throw new Exception("can't open save file"); scope(exit) fclose(fp); - uint read4() { uint what; fread(&what, 4, 1, fp); return what; } - ushort read2(){ ushort what; fread(&what, 2, 1, fp); return what; } - ubyte read1() { return cast(ubyte) fgetc(fp); } + void specialFread(void* tgt, size_t size) { + fread(tgt, size, 1, fp); + } + + return readBmpIndirect(&specialFread); +} + +MemoryImage readBmp(in ubyte[] data) { + const(ubyte)[] current = data; + void specialFread(void* tgt, size_t size) { + while(size) { + *cast(ubyte*)(tgt) = current[0]; + current = current[1 .. $]; + tgt++; + size--; + } + } + + return readBmpIndirect(&specialFread); +} + +MemoryImage readBmpIndirect(void delegate(void*, size_t) fread) { + uint read4() { uint what; fread(&what, 4); return what; } + ushort read2(){ ushort what; fread(&what, 2); return what; } + ubyte read1(){ ubyte what; fread(&what, 1); return what; } void require1(ubyte t, size_t line = __LINE__) { if(read1() != t) @@ -234,6 +258,7 @@ MemoryImage readBmp(string filename) { } void writeBmp(MemoryImage img, string filename) { + import core.stdc.stdio; FILE* fp = fopen((filename ~ "\0").ptr, "wb".ptr); if(fp is null) throw new Exception("can't open save file"); @@ -351,13 +376,15 @@ void writeBmp(MemoryImage img, string filename) { } } -/* +/+ void main() { import simpledisplay; + //import std.file; + //auto img = readBmp(cast(ubyte[]) std.file.read("/home/me/test2.bmp")); auto img = readBmp("/home/me/test2.bmp"); import std.stdio; writeln((cast(Object)img).toString()); displayImage(Image.fromMemoryImage(img)); //img.writeBmp("/home/me/test2.bmp"); } -*/ ++/ diff --git a/jsvar.d b/jsvar.d index cd79378..d237c7f 100644 --- a/jsvar.d +++ b/jsvar.d @@ -116,7 +116,7 @@ void main() { } // the WrappedNativeObject is disgusting - // but works. + // but works. sort of. /* Foop foop2; @@ -169,6 +169,18 @@ void main() { a; }, globals)); + /* + globals.minigui = json!q{}; + import arsd.minigui; + globals.minigui.createWindow = { + var v; + auto mw = new MainWindow(); + v._object = new OpaqueNativeObject!(MainWindow)(mw); + v.loop = { mw.loop(); }; + return v; + }; + */ + repl(globals); writeln("BACK IN D!"); @@ -1146,6 +1158,7 @@ struct var { } } +// this doesn't really work class WrappedNativeObject(T, bool wrapData = true) : PrototypeObject { T nativeObject; @@ -1246,9 +1259,9 @@ class OpaqueNativeObject(T) : PrototypeObject { this.item = t; } - override string toString() const { - return item.toString(); - } + //override string toString() const { + //return item.toString(); + //} override OpaqueNativeObject!T copy() { auto n = new OpaqueNativeObject!T(item); diff --git a/minigui.d b/minigui.d index 26c31ea..63481d3 100644 --- a/minigui.d +++ b/minigui.d @@ -1,22 +1,136 @@ +// http://msdn.microsoft.com/en-us/library/windows/desktop/bb775498%28v=vs.85%29.aspx + +/// FOR BEST RESULTS: be sure to link with the appropriate subsystem command +/// -L/SUBSYSTEM:WINDOWS:5.0 +/// otherwise you'll get a console and other visual bugs. module arsd.minigui; +/* + STILL NEEDED: + * combo box. (this is diff than select because you can free-form edit too. more like a lineedit with autoselect) + * slider + * listbox + * spinner + * label? + * rich text +*/ + +abstract class ComboboxBase : Widget { + // if the user can enter arbitrary data, we want to use 2 == CBS_DROPDOWN + // or to always show the list, we want CBS_SIMPLE == 1 + version(win32_widgets) + this(uint style, Widget parent = null) { + super(parent); + parentWindow = parent.parentWindow; + createWin32Window(this, "ComboBox", null, style); + } + + private string[] options; + private int selection = -1; + + void addOption(string s) { + options ~= s; + version(win32_widgets) + SendMessageA(hwnd, 323 /*CB_ADDSTRING*/, 0, cast(LPARAM) toStringzInternal(s)); + } + + void setSelection(int idx) { + selection = idx; + version(win32_widgets) + SendMessageA(hwnd, 334 /*CB_SETCURSEL*/, idx, 0); + } + + version(win32_widgets) + override void handleWmCommand(ushort cmd, ushort id) { + selection = SendMessageA(hwnd, 327 /* CB_GETCURSEL */, 0, 0); + auto event = new Event("changed", this); + event.dispatch(); + } +} + +class DropDownSelection : ComboboxBase { + this(Widget parent = null) { + version(win32_widgets) + super(3 /* CBS_DROPDOWNLIST */, parent); + } +} + +class FreeEntrySelection : ComboboxBase { + this(Widget parent = null) { + version(win32_widgets) + super(2 /* CBS_DROPDOWN */, parent); + } +} + +class ComboBox : ComboboxBase { + this(Widget parent = null) { + version(win32_widgets) + super(1 /* CBS_SIMPLE */, parent); + } +} + + +/+ +class Spinner : Widget { + version(win32_widgets) + this(Widget parent = null) { + super(parent); + parentWindow = parent.parentWindow; + auto hlayout = new HorizontalLayout(this); + lineEdit = new LineEdit(hlayout); + upDownControl = new UpDownControl(hlayout); + } + + LineEdit lineEdit; + UpDownControl upDownControl; +} + +class UpDownControl : Widget { + version(win32_widgets) + this(Widget parent = null) { + super(parent); + parentWindow = parent.parentWindow; + createWin32Window(this, "msctls_updown32", null, 4/*UDS_ALIGNRIGHT*/| 2 /* UDS_SETBUDDYINT */ | 16 /* UDS_AUTOBUDDY */ | 32 /* UDS_ARROWKEYS */); + } + + override int minHeight() { return Window.lineHeight; } + override int maxHeight() { return Window.lineHeight * 3/2; } + + override int minWidth() { return Window.lineHeight * 3/2; } + override int maxWidth() { return Window.lineHeight * 3/2; } +} ++/ + +class DataView : Widget { + // this is the omnibus data viewer + // the internal data layout is something like: + // string[string][] but also each node can have parents +} + + // http://msdn.microsoft.com/en-us/library/windows/desktop/bb775491(v=vs.85).aspx#PROGRESS_CLASS +// http://svn.dsource.org/projects/bindings/trunk/win32/commctrl.d + // FIXME: menus should prolly capture the mouse. ugh i kno. -import simpledisplay; +public import simpledisplay; // this is a hack to call the original window procedure on native win32 widgets if our event listener thing prevents default. private bool lastDefaultPrevented; version(Windows) { // use native widgets when available unless specifically asked otherwise - version(custom_widgets) {} - else { + version(custom_widgets) { + enum bool UsingCustomWidgets = true; + } else { version = win32_widgets; + enum bool UsingCustomWidgets = false; } // and native theming when needed - version = win32_theming; + //version = win32_theming; +} else { + enum bool UsingCustomWidgets = true; } /* @@ -31,11 +145,30 @@ version(Windows) { connect(paste, &textEdit.insertTextAtCarat); would be nice. + + + + I kinda want an omnibus dataview that combines list, tree, + and table - it can be switched dynamically between them. + + Flattening policy: only show top level, show recursive, show grouped + List styles: plain list (e.g.