mirror of https://github.com/adamdruppe/arsd.git
fixes
This commit is contained in:
parent
41f8150149
commit
dafa15243d
|
@ -21,6 +21,10 @@ struct Tagged(alias field) {}
|
||||||
auto Tag(T)(T t) {
|
auto Tag(T)(T t) {
|
||||||
return TagStruct!T(t);
|
return TagStruct!T(t);
|
||||||
}
|
}
|
||||||
|
/// For example `@presentIf("version >= 2") int addedInVersion2;`
|
||||||
|
struct presentIf { string code; }
|
||||||
|
|
||||||
|
|
||||||
struct TagStruct(T) { T t; }
|
struct TagStruct(T) { T t; }
|
||||||
struct MustBeStruct(T) { T t; }
|
struct MustBeStruct(T) { T t; }
|
||||||
/// The marked field is not in the actual file
|
/// The marked field is not in the actual file
|
||||||
|
@ -66,14 +70,25 @@ union N(ty) {
|
||||||
ubyte[ty.sizeof] bytes;
|
ubyte[ty.sizeof] bytes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool fieldPresent(alias field, T)(T t) {
|
||||||
|
bool p = true;
|
||||||
|
static foreach(attr; __traits(getAttributes, field)) {
|
||||||
|
static if(is(typeof(attr) == presentIf)) {
|
||||||
|
bool p2 = false;
|
||||||
|
with(t) p2 = mixin(attr.code);
|
||||||
|
p = p && p2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
/// input range of ubytes...
|
/// input range of ubytes...
|
||||||
int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false) {
|
int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false) {
|
||||||
int bytesConsumed;
|
int bytesConsumed;
|
||||||
string currentItem;
|
string currentItem;
|
||||||
|
|
||||||
import std.conv;
|
import std.conv;
|
||||||
scope(failure)
|
try {
|
||||||
throw new Exception(T.stringof ~ "." ~ currentItem ~ " trouble " ~ to!string(t));
|
|
||||||
|
|
||||||
ubyte next() {
|
ubyte next() {
|
||||||
if(r.empty)
|
if(r.empty)
|
||||||
|
@ -90,7 +105,8 @@ int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false)
|
||||||
static if(is(typeof(__traits(getMember, T, memberName)))) {
|
static if(is(typeof(__traits(getMember, T, memberName)))) {
|
||||||
alias f = __traits(getMember, T, memberName);
|
alias f = __traits(getMember, T, memberName);
|
||||||
alias ty = typeof(f);
|
alias ty = typeof(f);
|
||||||
static if(fieldSaved!f) {
|
static if(fieldSaved!f)
|
||||||
|
if(fieldPresent!f(t)) {
|
||||||
endianness = bigEndian!f(endianness);
|
endianness = bigEndian!f(endianness);
|
||||||
// FIXME VariableLength
|
// FIXME VariableLength
|
||||||
static if(is(ty : ulong) || is(ty : double)) {
|
static if(is(ty : ulong) || is(ty : double)) {
|
||||||
|
@ -200,5 +216,141 @@ int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false)
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
} catch(Exception e) {
|
||||||
|
throw new Exception(T.stringof ~ "." ~ currentItem ~ " trouble " ~ to!string(t), e.file, e.line, e);
|
||||||
|
}
|
||||||
|
|
||||||
return bytesConsumed;
|
return bytesConsumed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int saveTo(T, Range)(ref T t, ref Range r, bool assumeBigEndian = false) {
|
||||||
|
int bytesWritten;
|
||||||
|
string currentItem;
|
||||||
|
|
||||||
|
import std.conv;
|
||||||
|
try {
|
||||||
|
|
||||||
|
void write(ubyte b) {
|
||||||
|
bytesWritten++;
|
||||||
|
static if(is(Range == ubyte[]))
|
||||||
|
r ~= b;
|
||||||
|
else
|
||||||
|
r.put(b);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool endianness = bigEndian!T(assumeBigEndian);
|
||||||
|
static foreach(memberName; __traits(allMembers, T)) {{
|
||||||
|
currentItem = memberName;
|
||||||
|
static if(is(typeof(__traits(getMember, T, memberName)))) {
|
||||||
|
alias f = __traits(getMember, T, memberName);
|
||||||
|
alias ty = typeof(f);
|
||||||
|
static if(fieldSaved!f)
|
||||||
|
if(fieldPresent!f(t)) {
|
||||||
|
endianness = bigEndian!f(endianness);
|
||||||
|
// FIXME VariableLength
|
||||||
|
static if(is(ty : ulong) || is(ty : double)) {
|
||||||
|
N!ty n;
|
||||||
|
n.member = __traits(getMember, t, memberName);
|
||||||
|
if(endianness) {
|
||||||
|
foreach(i; 0 .. ty.sizeof) {
|
||||||
|
version(BigEndian)
|
||||||
|
write(n.bytes[i]);
|
||||||
|
else
|
||||||
|
write(n.bytes[$ - 1 - i]);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
foreach(i; 0 .. ty.sizeof) {
|
||||||
|
version(BigEndian)
|
||||||
|
write(n.bytes[$ - 1 - i]);
|
||||||
|
else
|
||||||
|
write(n.bytes[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME: MustBe
|
||||||
|
} else static if(is(ty == struct)) {
|
||||||
|
bytesWritten += saveTo(__traits(getMember, t, memberName), r, endianness);
|
||||||
|
} else static if(is(ty == union)) {
|
||||||
|
static foreach(attr; __traits(getAttributes, ty))
|
||||||
|
static if(is(attr == Tagged!Field, alias Field))
|
||||||
|
enum tagField = __traits(identifier, Field);
|
||||||
|
static assert(is(typeof(tagField)), "Unions need a Tagged UDA on the union type (not the member) indicating the field that identifies the union");
|
||||||
|
|
||||||
|
auto tag = __traits(getMember, t, tagField);
|
||||||
|
// find the child of the union matching the tag...
|
||||||
|
bool found = false;
|
||||||
|
static foreach(um; __traits(allMembers, ty)) {
|
||||||
|
if(tag == getTag!(__traits(getMember, ty, um))) {
|
||||||
|
bytesWritten += saveTo(__traits(getMember, __traits(getMember, t, memberName), um), r, endianness);
|
||||||
|
found = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(!found) {
|
||||||
|
import std.format;
|
||||||
|
throw new Exception(format("found unknown union tag %s at %s", tag, t));
|
||||||
|
}
|
||||||
|
} else static if(is(ty == E[], E)) {
|
||||||
|
|
||||||
|
// the numBytesRemaining / numElementsRemaining thing here ASSUMING the
|
||||||
|
// arrays are already the correct size. the struct itself could invariant that maybe
|
||||||
|
|
||||||
|
foreach(item; __traits(getMember, t, memberName)) {
|
||||||
|
static if(is(typeof(item) == struct)) {
|
||||||
|
bytesWritten += saveTo(item, r, endianness);
|
||||||
|
} else {
|
||||||
|
static struct dummy {
|
||||||
|
typeof(item) i;
|
||||||
|
}
|
||||||
|
dummy d = dummy(item);
|
||||||
|
bytesWritten += saveTo(d, r, endianness);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} else static assert(0, ty.stringof);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
|
||||||
|
} catch(Exception e) {
|
||||||
|
throw new Exception(T.stringof ~ "." ~ currentItem ~ " save trouble " ~ to!string(t), e.file, e.line, e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return bytesWritten;
|
||||||
|
}
|
||||||
|
|
||||||
|
unittest {
|
||||||
|
static struct A {
|
||||||
|
int a;
|
||||||
|
@presentIf("a > 5") int b;
|
||||||
|
int c;
|
||||||
|
@NumElements!c ubyte[] d;
|
||||||
|
}
|
||||||
|
|
||||||
|
A a;
|
||||||
|
a.loadFrom(cast(ubyte[]) [1, 1, 0, 0, 7, 0, 0, 0, 3, 0, 0, 0, 6, 7, 8]);
|
||||||
|
|
||||||
|
assert(a.a == 257);
|
||||||
|
assert(a.b == 7);
|
||||||
|
assert(a.c == 3);
|
||||||
|
assert(a.d == [6,7,8]);
|
||||||
|
|
||||||
|
a = A.init;
|
||||||
|
|
||||||
|
a.loadFrom(cast(ubyte[]) [0, 0, 0, 0, 7, 0, 0, 0,1,2,3,4,5,6,7]);
|
||||||
|
assert(a.b == 0);
|
||||||
|
assert(a.c == 7);
|
||||||
|
assert(a.d == [1,2,3,4,5,6,7]);
|
||||||
|
|
||||||
|
a.a = 44;
|
||||||
|
a.c = 3;
|
||||||
|
a.d = [5,4,3];
|
||||||
|
|
||||||
|
ubyte[] saved;
|
||||||
|
|
||||||
|
a.saveTo(saved);
|
||||||
|
|
||||||
|
A b;
|
||||||
|
b.loadFrom(saved);
|
||||||
|
|
||||||
|
assert(a == b);
|
||||||
|
}
|
||||||
|
|
|
@ -234,10 +234,10 @@ void runGame(T : GameHelperBase)(T game, int maxUpdateRate = 20, int maxRedrawRa
|
||||||
if(update.buttonWasJustPressed(l1)) game.snes[L] = true;
|
if(update.buttonWasJustPressed(l1)) game.snes[L] = true;
|
||||||
if(update.buttonWasJustPressed(r1)) game.snes[R] = true;
|
if(update.buttonWasJustPressed(r1)) game.snes[R] = true;
|
||||||
// note: no need to check analog stick here cuz joystick.d already does it for us (per old playstation tradition)
|
// note: no need to check analog stick here cuz joystick.d already does it for us (per old playstation tradition)
|
||||||
if(update.axisChange(Axis.horizontalDpad) < 0 && update.axisPosition(Axis.horizontalDpad) < -20000) game.snes[Left] = true;
|
if(update.axisChange(Axis.horizontalDpad) < 0 && update.axisPosition(Axis.horizontalDpad) < -8) game.snes[Left] = true;
|
||||||
if(update.axisChange(Axis.horizontalDpad) > 0 && update.axisPosition(Axis.horizontalDpad) > 20000) game.snes[Right] = true;
|
if(update.axisChange(Axis.horizontalDpad) > 0 && update.axisPosition(Axis.horizontalDpad) > 8) game.snes[Right] = true;
|
||||||
if(update.axisChange(Axis.verticalDpad) < 0 && update.axisPosition(Axis.verticalDpad) < -20000) game.snes[Up] = true;
|
if(update.axisChange(Axis.verticalDpad) < 0 && update.axisPosition(Axis.verticalDpad) < -8) game.snes[Up] = true;
|
||||||
if(update.axisChange(Axis.verticalDpad) > 0 && update.axisPosition(Axis.verticalDpad) > 20000) game.snes[Down] = true;
|
if(update.axisChange(Axis.verticalDpad) > 0 && update.axisPosition(Axis.verticalDpad) > 8) game.snes[Down] = true;
|
||||||
|
|
||||||
if(update.buttonWasJustReleased(square)) game.snes[Y] = false;
|
if(update.buttonWasJustReleased(square)) game.snes[Y] = false;
|
||||||
if(update.buttonWasJustReleased(triangle)) game.snes[X] = false;
|
if(update.buttonWasJustReleased(triangle)) game.snes[X] = false;
|
||||||
|
@ -247,14 +247,15 @@ void runGame(T : GameHelperBase)(T game, int maxUpdateRate = 20, int maxRedrawRa
|
||||||
if(update.buttonWasJustReleased(start)) game.snes[Start] = false;
|
if(update.buttonWasJustReleased(start)) game.snes[Start] = false;
|
||||||
if(update.buttonWasJustReleased(l1)) game.snes[L] = false;
|
if(update.buttonWasJustReleased(l1)) game.snes[L] = false;
|
||||||
if(update.buttonWasJustReleased(r1)) game.snes[R] = false;
|
if(update.buttonWasJustReleased(r1)) game.snes[R] = false;
|
||||||
if(update.axisChange(Axis.horizontalDpad) > 0 && update.axisPosition(Axis.horizontalDpad) > -20000) game.snes[Left] = false;
|
if(update.axisChange(Axis.horizontalDpad) > 0 && update.axisPosition(Axis.horizontalDpad) > -8) game.snes[Left] = false;
|
||||||
if(update.axisChange(Axis.horizontalDpad) < 0 && update.axisPosition(Axis.horizontalDpad) < 20000) game.snes[Right] = false;
|
if(update.axisChange(Axis.horizontalDpad) < 0 && update.axisPosition(Axis.horizontalDpad) < 8) game.snes[Right] = false;
|
||||||
if(update.axisChange(Axis.verticalDpad) > 0 && update.axisPosition(Axis.verticalDpad) > -20000) game.snes[Up] = false;
|
if(update.axisChange(Axis.verticalDpad) > 0 && update.axisPosition(Axis.verticalDpad) > -8) game.snes[Up] = false;
|
||||||
if(update.axisChange(Axis.verticalDpad) < 0 && update.axisPosition(Axis.verticalDpad) < 20000) game.snes[Down] = false;
|
if(update.axisChange(Axis.verticalDpad) < 0 && update.axisPosition(Axis.verticalDpad) < 8) game.snes[Down] = false;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} else static if(__traits(isSame, Button, XBox360Buttons)) {
|
} else static if(__traits(isSame, Button, XBox360Buttons)) {
|
||||||
|
static assert(0);
|
||||||
// XBox style mapping
|
// XBox style mapping
|
||||||
// the reason this exists is if the programmer wants to use the xbox details, but
|
// the reason this exists is if the programmer wants to use the xbox details, but
|
||||||
// might also want the basic controller in here. joystick.d already does translations
|
// might also want the basic controller in here. joystick.d already does translations
|
||||||
|
|
|
@ -117,6 +117,10 @@ version(Windows) {
|
||||||
WindowsXInput wxi;
|
WindowsXInput wxi;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
version(OSX) {
|
||||||
|
struct JoystickState {}
|
||||||
|
}
|
||||||
|
|
||||||
JoystickState[4] joystickState;
|
JoystickState[4] joystickState;
|
||||||
|
|
||||||
version(linux) {
|
version(linux) {
|
||||||
|
|
21
jsvar.d
21
jsvar.d
|
@ -617,6 +617,12 @@ struct var {
|
||||||
// if it is var, we'll just blit it over
|
// if it is var, we'll just blit it over
|
||||||
public var opAssign(T)(T t) if(!is(T == var)) {
|
public var opAssign(T)(T t) if(!is(T == var)) {
|
||||||
static if(__traits(compiles, this = t.toArsdJsvar())) {
|
static if(__traits(compiles, this = t.toArsdJsvar())) {
|
||||||
|
static if(__traits(compiles, t is null)) {
|
||||||
|
if(t is null)
|
||||||
|
this = null;
|
||||||
|
else
|
||||||
|
this = t.toArsdJsvar();
|
||||||
|
} else
|
||||||
this = t.toArsdJsvar();
|
this = t.toArsdJsvar();
|
||||||
} else static if(isFloatingPoint!T) {
|
} else static if(isFloatingPoint!T) {
|
||||||
this._type = Type.Floating;
|
this._type = Type.Floating;
|
||||||
|
@ -943,7 +949,15 @@ struct var {
|
||||||
auto pl = this._payload._array;
|
auto pl = this._payload._array;
|
||||||
static if(isSomeString!T) {
|
static if(isSomeString!T) {
|
||||||
return to!string(pl);
|
return to!string(pl);
|
||||||
} else static if(isArray!T) {
|
} else static if(is(T == E[N], E, size_t N)) {
|
||||||
|
T ret;
|
||||||
|
foreach(i; 0 .. N) {
|
||||||
|
if(i >= pl.length)
|
||||||
|
break;
|
||||||
|
ret[i] = pl[i].get!E;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
} else static if(is(T == E[], E)) {
|
||||||
T ret;
|
T ret;
|
||||||
static if(is(ElementType!T == void)) {
|
static if(is(ElementType!T == void)) {
|
||||||
static assert(0, "try wrapping the function to get rid of void[] args");
|
static assert(0, "try wrapping the function to get rid of void[] args");
|
||||||
|
@ -1366,6 +1380,11 @@ struct var {
|
||||||
return var.fromJsonValue(decoded);
|
return var.fromJsonValue(decoded);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static var fromJsonFile(string filename) {
|
||||||
|
import std.file;
|
||||||
|
return var.fromJson(readText(filename));
|
||||||
|
}
|
||||||
|
|
||||||
static var fromJsonValue(JSONValue v) {
|
static var fromJsonValue(JSONValue v) {
|
||||||
var ret;
|
var ret;
|
||||||
|
|
||||||
|
|
141
minigui.d
141
minigui.d
|
@ -1,5 +1,7 @@
|
||||||
// http://msdn.microsoft.com/en-us/library/windows/desktop/bb775498%28v=vs.85%29.aspx
|
// http://msdn.microsoft.com/en-us/library/windows/desktop/bb775498%28v=vs.85%29.aspx
|
||||||
|
|
||||||
|
// osx style menu search.
|
||||||
|
|
||||||
// would be cool for a scroll bar to have marking capabilities
|
// would be cool for a scroll bar to have marking capabilities
|
||||||
// kinda like vim's marks just on clicks etc and visual representation
|
// kinda like vim's marks just on clicks etc and visual representation
|
||||||
// generically. may be cool to add an up arrow to the bottom too
|
// generically. may be cool to add an up arrow to the bottom too
|
||||||
|
@ -301,13 +303,19 @@ abstract class ComboboxBase : Widget {
|
||||||
event.dispatch();
|
event.dispatch();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
version(win32_widgets) {
|
||||||
|
override int minHeight() { return Window.lineHeight + 6; }
|
||||||
|
override int maxHeight() { return Window.lineHeight + 6; }
|
||||||
|
} else {
|
||||||
override int minHeight() { return Window.lineHeight + 4; }
|
override int minHeight() { return Window.lineHeight + 4; }
|
||||||
override int maxHeight() { return Window.lineHeight + 4; }
|
override int maxHeight() { return Window.lineHeight + 4; }
|
||||||
|
}
|
||||||
|
|
||||||
version(custom_widgets) {
|
version(custom_widgets) {
|
||||||
SimpleWindow dropDown;
|
SimpleWindow dropDown;
|
||||||
void popup() {
|
void popup() {
|
||||||
auto w = width;
|
auto w = width;
|
||||||
|
// FIXME: suggestedDropdownHeight see below
|
||||||
auto h = cast(int) this.options.length * Window.lineHeight + 8;
|
auto h = cast(int) this.options.length * Window.lineHeight + 8;
|
||||||
|
|
||||||
auto coord = this.globalCoordinates();
|
auto coord = this.globalCoordinates();
|
||||||
|
@ -396,7 +404,21 @@ class DropDownSelection : ComboboxBase {
|
||||||
painter.pen = Pen(Color.black, 1, Pen.Style.Solid);
|
painter.pen = Pen(Color.black, 1, Pen.Style.Solid);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
version(win32_widgets)
|
||||||
|
override void registerMovement() {
|
||||||
|
version(win32_widgets) {
|
||||||
|
if(hwnd) {
|
||||||
|
auto pos = getChildPositionRelativeToParentHwnd(this);
|
||||||
|
// the height given to this from Windows' perspective is supposed
|
||||||
|
// to include the drop down's height. so I add to it to give some
|
||||||
|
// room for that.
|
||||||
|
// FIXME: maybe make the subclass provide a suggestedDropdownHeight thing
|
||||||
|
MoveWindow(hwnd, pos[0], pos[1], width, height + 200, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sendResizeEvent();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -448,7 +470,7 @@ class FreeEntrySelection : ComboboxBase {
|
||||||
class ComboBox : ComboboxBase {
|
class ComboBox : ComboboxBase {
|
||||||
this(Widget parent = null) {
|
this(Widget parent = null) {
|
||||||
version(win32_widgets)
|
version(win32_widgets)
|
||||||
super(1 /* CBS_SIMPLE */, parent);
|
super(1 /* CBS_SIMPLE */ | CBS_NOINTEGRALHEIGHT, parent);
|
||||||
else version(custom_widgets) {
|
else version(custom_widgets) {
|
||||||
super(parent);
|
super(parent);
|
||||||
lineEdit = new LineEdit(this);
|
lineEdit = new LineEdit(this);
|
||||||
|
@ -1054,15 +1076,44 @@ version(win32_widgets) {
|
||||||
//assert(0, to!string(hWnd) ~ " :: " ~ to!string(TextEdit.nativeMapping)); // not supposed to happen
|
//assert(0, to!string(hWnd) ~ " :: " ~ to!string(TextEdit.nativeMapping)); // not supposed to happen
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern(Windows)
|
||||||
|
private
|
||||||
|
int HookedWndProcBSGROUPBOX_HACK(HWND hWnd, UINT iMessage, WPARAM wParam, LPARAM lParam) nothrow {
|
||||||
|
if(iMessage == WM_ERASEBKGND) {
|
||||||
|
auto dc = GetDC(hWnd);
|
||||||
|
auto b = SelectObject(dc, GetSysColorBrush(COLOR_3DFACE));
|
||||||
|
auto p = SelectObject(dc, GetStockObject(NULL_PEN));
|
||||||
|
RECT r;
|
||||||
|
GetWindowRect(hWnd, &r);
|
||||||
|
// since the pen is null, to fill the whole space, we need the +1 on both.
|
||||||
|
gdi.Rectangle(dc, 0, 0, r.right - r.left + 1, r.bottom - r.top + 1);
|
||||||
|
SelectObject(dc, p);
|
||||||
|
SelectObject(dc, b);
|
||||||
|
ReleaseDC(hWnd, dc);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return HookedWndProc(hWnd, iMessage, wParam, lParam);
|
||||||
|
}
|
||||||
|
|
||||||
// className MUST be a string literal
|
// className MUST be a string literal
|
||||||
void createWin32Window(Widget p, const(wchar)[] className, string windowText, DWORD style, DWORD extStyle = 0) {
|
void createWin32Window(Widget p, const(wchar)[] className, string windowText, DWORD style, DWORD extStyle = 0) {
|
||||||
assert(p.parentWindow !is null);
|
assert(p.parentWindow !is null);
|
||||||
assert(p.parentWindow.win.impl.hwnd !is null);
|
assert(p.parentWindow.win.impl.hwnd !is null);
|
||||||
|
|
||||||
|
auto bsgroupbox = style == BS_GROUPBOX;
|
||||||
|
|
||||||
HWND phwnd;
|
HWND phwnd;
|
||||||
if(p.parent !is null && p.parent.hwnd !is null)
|
|
||||||
phwnd = p.parent.hwnd;
|
auto wtf = p.parent;
|
||||||
else
|
while(wtf) {
|
||||||
|
if(wtf.hwnd !is null) {
|
||||||
|
phwnd = wtf.hwnd;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
wtf = wtf.parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(phwnd is null)
|
||||||
phwnd = p.parentWindow.win.impl.hwnd;
|
phwnd = p.parentWindow.win.impl.hwnd;
|
||||||
|
|
||||||
assert(phwnd !is null);
|
assert(phwnd !is null);
|
||||||
|
@ -1070,6 +1121,8 @@ version(win32_widgets) {
|
||||||
WCharzBuffer wt = WCharzBuffer(windowText);
|
WCharzBuffer wt = WCharzBuffer(windowText);
|
||||||
|
|
||||||
style |= WS_VISIBLE | WS_CHILD;
|
style |= WS_VISIBLE | WS_CHILD;
|
||||||
|
//if(className != WC_TABCONTROL)
|
||||||
|
style |= WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
|
||||||
p.hwnd = CreateWindowExW(extStyle, className.ptr, wt.ptr, style,
|
p.hwnd = CreateWindowExW(extStyle, className.ptr, wt.ptr, style,
|
||||||
CW_USEDEFAULT, CW_USEDEFAULT, 100, 100,
|
CW_USEDEFAULT, CW_USEDEFAULT, 100, 100,
|
||||||
phwnd, null, cast(HINSTANCE) GetModuleHandle(null), null);
|
phwnd, null, cast(HINSTANCE) GetModuleHandle(null), null);
|
||||||
|
@ -1093,6 +1146,9 @@ version(win32_widgets) {
|
||||||
p.simpleWindowWrappingHwnd.beingOpenKeepsAppOpen = false;
|
p.simpleWindowWrappingHwnd.beingOpenKeepsAppOpen = false;
|
||||||
Widget.nativeMapping[p.hwnd] = p;
|
Widget.nativeMapping[p.hwnd] = p;
|
||||||
|
|
||||||
|
if(bsgroupbox)
|
||||||
|
p.originalWindowProcedure = cast(WNDPROC) SetWindowLongPtr(p.hwnd, GWL_WNDPROC, cast(size_t) &HookedWndProcBSGROUPBOX_HACK);
|
||||||
|
else
|
||||||
p.originalWindowProcedure = cast(WNDPROC) SetWindowLongPtr(p.hwnd, GWL_WNDPROC, cast(size_t) &HookedWndProc);
|
p.originalWindowProcedure = cast(WNDPROC) SetWindowLongPtr(p.hwnd, GWL_WNDPROC, cast(size_t) &HookedWndProc);
|
||||||
|
|
||||||
EnumChildWindows(p.hwnd, &childHandler, cast(LPARAM) cast(void*) p);
|
EnumChildWindows(p.hwnd, &childHandler, cast(LPARAM) cast(void*) p);
|
||||||
|
@ -1150,6 +1206,11 @@ struct WidgetPainter {
|
||||||
class Widget {
|
class Widget {
|
||||||
mixin LayoutInfo!();
|
mixin LayoutInfo!();
|
||||||
|
|
||||||
|
protected void sendResizeEvent() {
|
||||||
|
auto event = new Event("resize", this);
|
||||||
|
event.sendDirectly();
|
||||||
|
}
|
||||||
|
|
||||||
deprecated("Change ScreenPainter to WidgetPainter")
|
deprecated("Change ScreenPainter to WidgetPainter")
|
||||||
final void paint(ScreenPainter) { assert(0, "Change ScreenPainter to WidgetPainter and recompile your code"); }
|
final void paint(ScreenPainter) { assert(0, "Change ScreenPainter to WidgetPainter and recompile your code"); }
|
||||||
|
|
||||||
|
@ -1450,6 +1511,7 @@ class Widget {
|
||||||
MoveWindow(hwnd, pos[0], pos[1], width, height, true);
|
MoveWindow(hwnd, pos[0], pos[1], width, height, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
sendResizeEvent();
|
||||||
}
|
}
|
||||||
|
|
||||||
Window parentWindow;
|
Window parentWindow;
|
||||||
|
@ -1741,9 +1803,35 @@ class OpenGlWidget : Widget {
|
||||||
///
|
///
|
||||||
this(Widget parent) {
|
this(Widget parent) {
|
||||||
this.parentWindow = parent.parentWindow;
|
this.parentWindow = parent.parentWindow;
|
||||||
win = new SimpleWindow(640, 480, null, OpenGlOptions.yes, Resizability.automaticallyScaleIfPossible, WindowTypes.nestedChild, WindowFlags.normal, this.parentWindow.win);
|
|
||||||
|
SimpleWindow pwin = this.parentWindow.win;
|
||||||
|
|
||||||
|
|
||||||
|
version(win32_widgets) {
|
||||||
|
HWND phwnd;
|
||||||
|
auto wtf = parent;
|
||||||
|
while(wtf) {
|
||||||
|
if(wtf.hwnd) {
|
||||||
|
phwnd = wtf.hwnd;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
wtf = wtf.parent;
|
||||||
|
}
|
||||||
|
// kinda a hack here just because the ctor below just needs a SimpleWindow wrapper....
|
||||||
|
if(phwnd)
|
||||||
|
pwin = new SimpleWindow(phwnd);
|
||||||
|
}
|
||||||
|
|
||||||
|
win = new SimpleWindow(640, 480, null, OpenGlOptions.yes, Resizability.automaticallyScaleIfPossible, WindowTypes.nestedChild, WindowFlags.normal, pwin);
|
||||||
super(parent);
|
super(parent);
|
||||||
|
|
||||||
|
/*
|
||||||
|
win.onFocusChange = (bool getting) {
|
||||||
|
if(getting)
|
||||||
|
this.focus();
|
||||||
|
};
|
||||||
|
*/
|
||||||
|
|
||||||
version(win32_widgets) {
|
version(win32_widgets) {
|
||||||
Widget.nativeMapping[win.hwnd] = this;
|
Widget.nativeMapping[win.hwnd] = this;
|
||||||
this.originalWindowProcedure = cast(WNDPROC) SetWindowLongPtr(win.hwnd, GWL_WNDPROC, cast(size_t) &HookedWndProc);
|
this.originalWindowProcedure = cast(WNDPROC) SetWindowLongPtr(win.hwnd, GWL_WNDPROC, cast(size_t) &HookedWndProc);
|
||||||
|
@ -1799,6 +1887,9 @@ class OpenGlWidget : Widget {
|
||||||
else
|
else
|
||||||
auto pos = getChildPositionRelativeToParentOrigin(this);
|
auto pos = getChildPositionRelativeToParentOrigin(this);
|
||||||
win.moveResize(pos[0], pos[1], width, height);
|
win.moveResize(pos[0], pos[1], width, height);
|
||||||
|
|
||||||
|
win.setAsCurrentOpenGlContext();
|
||||||
|
sendResizeEvent();
|
||||||
}
|
}
|
||||||
|
|
||||||
//void delegate() drawFrame;
|
//void delegate() drawFrame;
|
||||||
|
@ -3118,16 +3209,8 @@ class TabWidget : Widget {
|
||||||
}
|
}
|
||||||
|
|
||||||
override void recomputeChildLayout() {
|
override void recomputeChildLayout() {
|
||||||
this.registerMovement();
|
|
||||||
version(win32_widgets) {
|
version(win32_widgets) {
|
||||||
|
this.registerMovement();
|
||||||
// Windows doesn't actually parent widgets to the
|
|
||||||
// tab control, so we will temporarily pretend this isn't
|
|
||||||
// a native widget as we do the changes. A bit of a filthy
|
|
||||||
// hack, but a functional one.
|
|
||||||
auto hwnd = this.hwnd;
|
|
||||||
this.hwnd = null;
|
|
||||||
scope(exit) this.hwnd = hwnd;
|
|
||||||
|
|
||||||
RECT rect;
|
RECT rect;
|
||||||
GetWindowRect(hwnd, &rect);
|
GetWindowRect(hwnd, &rect);
|
||||||
|
@ -3144,6 +3227,7 @@ class TabWidget : Widget {
|
||||||
child.recomputeChildLayout();
|
child.recomputeChildLayout();
|
||||||
}
|
}
|
||||||
} else version(custom_widgets) {
|
} else version(custom_widgets) {
|
||||||
|
this.registerMovement();
|
||||||
foreach(child; children) {
|
foreach(child; children) {
|
||||||
child.x = 2;
|
child.x = 2;
|
||||||
child.y = tabBarHeight + 2; // for the border
|
child.y = tabBarHeight + 2; // for the border
|
||||||
|
@ -3350,7 +3434,7 @@ class TabWidgetPage : Widget {
|
||||||
this.title = title;
|
this.title = title;
|
||||||
super(parent);
|
super(parent);
|
||||||
|
|
||||||
/*
|
///*
|
||||||
version(win32_widgets) {
|
version(win32_widgets) {
|
||||||
static bool classRegistered = false;
|
static bool classRegistered = false;
|
||||||
if(!classRegistered) {
|
if(!classRegistered) {
|
||||||
|
@ -3358,6 +3442,7 @@ class TabWidgetPage : Widget {
|
||||||
WNDCLASSEX wc;
|
WNDCLASSEX wc;
|
||||||
wc.cbSize = wc.sizeof;
|
wc.cbSize = wc.sizeof;
|
||||||
wc.hInstance = hInstance;
|
wc.hInstance = hInstance;
|
||||||
|
wc.hbrBackground = cast(HBRUSH) (COLOR_3DFACE+1); // GetStockObject(WHITE_BRUSH);
|
||||||
wc.lpfnWndProc = &DefWindowProc;
|
wc.lpfnWndProc = &DefWindowProc;
|
||||||
wc.lpszClassName = "arsd_minigui_TabWidgetPage"w.ptr;
|
wc.lpszClassName = "arsd_minigui_TabWidgetPage"w.ptr;
|
||||||
if(!RegisterClassExW(&wc))
|
if(!RegisterClassExW(&wc))
|
||||||
|
@ -3368,7 +3453,7 @@ class TabWidgetPage : Widget {
|
||||||
|
|
||||||
createWin32Window(this, "arsd_minigui_TabWidgetPage"w, "", 0);
|
createWin32Window(this, "arsd_minigui_TabWidgetPage"w, "", 0);
|
||||||
}
|
}
|
||||||
*/
|
//*/
|
||||||
}
|
}
|
||||||
|
|
||||||
override int minHeight() {
|
override int minHeight() {
|
||||||
|
@ -3502,6 +3587,17 @@ class ScrollMessageWidget : Widget {
|
||||||
magic = true;
|
magic = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
|
void scrollUp() {
|
||||||
|
vsb.setPosition(vsb.position - 1);
|
||||||
|
notify();
|
||||||
|
}
|
||||||
|
/// Ditto
|
||||||
|
void scrollDown() {
|
||||||
|
vsb.setPosition(vsb.position + 1);
|
||||||
|
notify();
|
||||||
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
VerticalScrollbar verticalScrollBar() { return vsb; }
|
VerticalScrollbar verticalScrollBar() { return vsb; }
|
||||||
///
|
///
|
||||||
|
@ -4039,6 +4135,8 @@ class Window : Widget {
|
||||||
event.key = ev.key;
|
event.key = ev.key;
|
||||||
event.state = ev.modifierState;
|
event.state = ev.modifierState;
|
||||||
event.shiftKey = (ev.modifierState & ModifierState.shift) ? true : false;
|
event.shiftKey = (ev.modifierState & ModifierState.shift) ? true : false;
|
||||||
|
event.altKey = (ev.modifierState & ModifierState.alt) ? true : false;
|
||||||
|
event.ctrlKey = (ev.modifierState & ModifierState.ctrl) ? true : false;
|
||||||
event.dispatch();
|
event.dispatch();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -4095,6 +4193,7 @@ class Window : Widget {
|
||||||
event = new Event("click", ele);
|
event = new Event("click", ele);
|
||||||
event.clientX = eleR.x;
|
event.clientX = eleR.x;
|
||||||
event.clientY = eleR.y;
|
event.clientY = eleR.y;
|
||||||
|
event.state = ev.modifierState;
|
||||||
event.button = ev.button;
|
event.button = ev.button;
|
||||||
event.buttonLinear = ev.buttonLinear;
|
event.buttonLinear = ev.buttonLinear;
|
||||||
event.dispatch();
|
event.dispatch();
|
||||||
|
@ -6607,6 +6706,16 @@ class Event {
|
||||||
string stringValue; ///
|
string stringValue; ///
|
||||||
|
|
||||||
bool shiftKey; ///
|
bool shiftKey; ///
|
||||||
|
/++
|
||||||
|
NOTE: only set on key events right now
|
||||||
|
|
||||||
|
History:
|
||||||
|
Added April 15, 2020
|
||||||
|
+/
|
||||||
|
bool ctrlKey;
|
||||||
|
|
||||||
|
/// ditto
|
||||||
|
bool altKey;
|
||||||
|
|
||||||
private bool isBubbling;
|
private bool isBubbling;
|
||||||
|
|
||||||
|
|
|
@ -101,7 +101,7 @@ class ColorPickerDialog : Dialog {
|
||||||
override int minHeight() { return hslImage ? hslImage.height : 4; }
|
override int minHeight() { return hslImage ? hslImage.height : 4; }
|
||||||
override int maxHeight() { return hslImage ? hslImage.height : 4; }
|
override int maxHeight() { return hslImage ? hslImage.height : 4; }
|
||||||
override int marginBottom() { return 4; }
|
override int marginBottom() { return 4; }
|
||||||
override void paint(ScreenPainter painter) {
|
override void paint(WidgetPainter painter) {
|
||||||
if(hslImage)
|
if(hslImage)
|
||||||
hslImage.drawAt(painter, Point(0, 0));
|
hslImage.drawAt(painter, Point(0, 0));
|
||||||
}
|
}
|
||||||
|
@ -223,7 +223,7 @@ class ColorPickerDialog : Dialog {
|
||||||
super(s);
|
super(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
override void paint(ScreenPainter painter) {
|
override void paint(WidgetPainter painter) {
|
||||||
auto c = currentColor();
|
auto c = currentColor();
|
||||||
|
|
||||||
auto c1 = alphaBlend(c, Color(64, 64, 64));
|
auto c1 = alphaBlend(c, Color(64, 64, 64));
|
||||||
|
|
2
mvd.d
2
mvd.d
|
@ -89,5 +89,7 @@ unittest {
|
||||||
assert(mvd!foo(new DerivedClass, new DerivedClass) == 3);
|
assert(mvd!foo(new DerivedClass, new DerivedClass) == 3);
|
||||||
assert(mvd!foo(new OtherClass, new OtherClass) == 1);
|
assert(mvd!foo(new OtherClass, new OtherClass) == 1);
|
||||||
assert(mvd!foo(new OtherClass, new MyClass) == 1);
|
assert(mvd!foo(new OtherClass, new MyClass) == 1);
|
||||||
|
assert(mvd!foo(new DerivedClass, new DerivedClass) == 3);
|
||||||
|
assert(mvd!foo(new OtherClass, new MyClass) == 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -78,7 +78,7 @@
|
||||||
On Win32, you can pass `-L/subsystem:windows` if you don't want a
|
On Win32, you can pass `-L/subsystem:windows` if you don't want a
|
||||||
console to be automatically allocated.
|
console to be automatically allocated.
|
||||||
|
|
||||||
On Mac, when compiling with X11, you need XQuartz and -L-L/usr/X11R6/lib passed to dmd. If using the Cocoa implementation on Mac, you need to pass `-L-framework -LCocoa` to dmd.
|
On Mac, when compiling with X11, you need XQuartz and -L-L/usr/X11R6/lib passed to dmd. If using the Cocoa implementation on Mac, you need to pass `-L-framework -LCocoa` to dmd. For OpenGL, add `-L-framework -LOpenGL` to the build command.
|
||||||
|
|
||||||
On Ubuntu, you might need to install X11 development libraries to
|
On Ubuntu, you might need to install X11 development libraries to
|
||||||
successfully link.
|
successfully link.
|
||||||
|
@ -1408,6 +1408,9 @@ class SimpleWindow : CapableOfHandlingNativeEvent, CapableOfBeingDrawnUpon {
|
||||||
_width = 1;
|
_width = 1;
|
||||||
_height = 1;
|
_height = 1;
|
||||||
nativeMapping[nativeWindow] = this;
|
nativeMapping[nativeWindow] = this;
|
||||||
|
|
||||||
|
beingOpenKeepsAppOpen = false;
|
||||||
|
|
||||||
CapableOfHandlingNativeEvent.nativeHandleMapping[nativeWindow] = this;
|
CapableOfHandlingNativeEvent.nativeHandleMapping[nativeWindow] = this;
|
||||||
_suppressDestruction = true; // so it doesn't try to close
|
_suppressDestruction = true; // so it doesn't try to close
|
||||||
}
|
}
|
||||||
|
@ -8326,6 +8329,7 @@ version(Windows) {
|
||||||
}
|
}
|
||||||
|
|
||||||
int style;
|
int style;
|
||||||
|
uint flags = WS_EX_ACCEPTFILES; // accept drag-drop files
|
||||||
|
|
||||||
// FIXME: windowType and customizationFlags
|
// FIXME: windowType and customizationFlags
|
||||||
final switch(windowType) {
|
final switch(windowType) {
|
||||||
|
@ -8342,17 +8346,17 @@ version(Windows) {
|
||||||
case WindowTypes.popupMenu:
|
case WindowTypes.popupMenu:
|
||||||
case WindowTypes.notification:
|
case WindowTypes.notification:
|
||||||
style = WS_POPUP;
|
style = WS_POPUP;
|
||||||
|
flags |= WS_EX_NOACTIVATE;
|
||||||
break;
|
break;
|
||||||
case WindowTypes.nestedChild:
|
case WindowTypes.nestedChild:
|
||||||
style = WS_CHILD;
|
style = WS_CHILD;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint flags = WS_EX_ACCEPTFILES; // accept drag-drop files
|
|
||||||
if ((customizationFlags & WindowFlags.extraComposite) != 0)
|
if ((customizationFlags & WindowFlags.extraComposite) != 0)
|
||||||
flags |= WS_EX_LAYERED; // composite window for better performance and effects support
|
flags |= WS_EX_LAYERED; // composite window for better performance and effects support
|
||||||
|
|
||||||
hwnd = CreateWindowEx(flags, cn.ptr, toWStringz(title), style | WS_CLIPCHILDREN, // the clip children helps avoid flickering in minigui and doesn't seem to harm other use (mostly, sdpy is no child windows anyway) sooo i think it is ok
|
hwnd = CreateWindowEx(flags, cn.ptr, toWStringz(title), style | WS_CLIPCHILDREN | WS_CLIPSIBLINGS, // the clip children helps avoid flickering in minigui and doesn't seem to harm other use (mostly, sdpy is no child windows anyway) sooo i think it is ok
|
||||||
CW_USEDEFAULT, CW_USEDEFAULT, width, height,
|
CW_USEDEFAULT, CW_USEDEFAULT, width, height,
|
||||||
parent is null ? null : parent.impl.hwnd, null, hInstance, null);
|
parent is null ? null : parent.impl.hwnd, null, hInstance, null);
|
||||||
|
|
||||||
|
@ -8488,7 +8492,7 @@ version(Windows) {
|
||||||
static int triggerEvents(HWND hwnd, uint msg, WPARAM wParam, LPARAM lParam, int offsetX, int offsetY, SimpleWindow wind) {
|
static int triggerEvents(HWND hwnd, uint msg, WPARAM wParam, LPARAM lParam, int offsetX, int offsetY, SimpleWindow wind) {
|
||||||
MouseEvent mouse;
|
MouseEvent mouse;
|
||||||
|
|
||||||
void mouseEvent(bool isScreen = false) {
|
void mouseEvent(bool isScreen, ulong mods) {
|
||||||
auto x = LOWORD(lParam);
|
auto x = LOWORD(lParam);
|
||||||
auto y = HIWORD(lParam);
|
auto y = HIWORD(lParam);
|
||||||
if(isScreen) {
|
if(isScreen) {
|
||||||
|
@ -8503,7 +8507,7 @@ version(Windows) {
|
||||||
mouse.y = y + offsetY;
|
mouse.y = y + offsetY;
|
||||||
|
|
||||||
wind.mdx(mouse);
|
wind.mdx(mouse);
|
||||||
mouse.modifierState = cast(int) wParam;
|
mouse.modifierState = cast(int) mods;
|
||||||
mouse.window = wind;
|
mouse.window = wind;
|
||||||
|
|
||||||
if(wind.handleMouseEvent)
|
if(wind.handleMouseEvent)
|
||||||
|
@ -8597,61 +8601,67 @@ version(Windows) {
|
||||||
wind.handleKeyEvent(ev);
|
wind.handleKeyEvent(ev);
|
||||||
break;
|
break;
|
||||||
case 0x020a /*WM_MOUSEWHEEL*/:
|
case 0x020a /*WM_MOUSEWHEEL*/:
|
||||||
|
// send click
|
||||||
mouse.type = cast(MouseEventType) 1;
|
mouse.type = cast(MouseEventType) 1;
|
||||||
mouse.button = ((HIWORD(wParam) > 120) ? MouseButton.wheelDown : MouseButton.wheelUp);
|
mouse.button = ((HIWORD(wParam) > 120) ? MouseButton.wheelDown : MouseButton.wheelUp);
|
||||||
mouseEvent(true);
|
mouseEvent(true, LOWORD(wParam));
|
||||||
|
|
||||||
|
// also send release
|
||||||
|
mouse.type = cast(MouseEventType) 2;
|
||||||
|
mouse.button = ((HIWORD(wParam) > 120) ? MouseButton.wheelDown : MouseButton.wheelUp);
|
||||||
|
mouseEvent(true, LOWORD(wParam));
|
||||||
break;
|
break;
|
||||||
case WM_MOUSEMOVE:
|
case WM_MOUSEMOVE:
|
||||||
mouse.type = cast(MouseEventType) 0;
|
mouse.type = cast(MouseEventType) 0;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
break;
|
break;
|
||||||
case WM_LBUTTONDOWN:
|
case WM_LBUTTONDOWN:
|
||||||
case WM_LBUTTONDBLCLK:
|
case WM_LBUTTONDBLCLK:
|
||||||
mouse.type = cast(MouseEventType) 1;
|
mouse.type = cast(MouseEventType) 1;
|
||||||
mouse.button = MouseButton.left;
|
mouse.button = MouseButton.left;
|
||||||
mouse.doubleClick = msg == WM_LBUTTONDBLCLK;
|
mouse.doubleClick = msg == WM_LBUTTONDBLCLK;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
break;
|
break;
|
||||||
case WM_LBUTTONUP:
|
case WM_LBUTTONUP:
|
||||||
mouse.type = cast(MouseEventType) 2;
|
mouse.type = cast(MouseEventType) 2;
|
||||||
mouse.button = MouseButton.left;
|
mouse.button = MouseButton.left;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
break;
|
break;
|
||||||
case WM_RBUTTONDOWN:
|
case WM_RBUTTONDOWN:
|
||||||
case WM_RBUTTONDBLCLK:
|
case WM_RBUTTONDBLCLK:
|
||||||
mouse.type = cast(MouseEventType) 1;
|
mouse.type = cast(MouseEventType) 1;
|
||||||
mouse.button = MouseButton.right;
|
mouse.button = MouseButton.right;
|
||||||
mouse.doubleClick = msg == WM_RBUTTONDBLCLK;
|
mouse.doubleClick = msg == WM_RBUTTONDBLCLK;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
break;
|
break;
|
||||||
case WM_RBUTTONUP:
|
case WM_RBUTTONUP:
|
||||||
mouse.type = cast(MouseEventType) 2;
|
mouse.type = cast(MouseEventType) 2;
|
||||||
mouse.button = MouseButton.right;
|
mouse.button = MouseButton.right;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
break;
|
break;
|
||||||
case WM_MBUTTONDOWN:
|
case WM_MBUTTONDOWN:
|
||||||
case WM_MBUTTONDBLCLK:
|
case WM_MBUTTONDBLCLK:
|
||||||
mouse.type = cast(MouseEventType) 1;
|
mouse.type = cast(MouseEventType) 1;
|
||||||
mouse.button = MouseButton.middle;
|
mouse.button = MouseButton.middle;
|
||||||
mouse.doubleClick = msg == WM_MBUTTONDBLCLK;
|
mouse.doubleClick = msg == WM_MBUTTONDBLCLK;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
break;
|
break;
|
||||||
case WM_MBUTTONUP:
|
case WM_MBUTTONUP:
|
||||||
mouse.type = cast(MouseEventType) 2;
|
mouse.type = cast(MouseEventType) 2;
|
||||||
mouse.button = MouseButton.middle;
|
mouse.button = MouseButton.middle;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
break;
|
break;
|
||||||
case WM_XBUTTONDOWN:
|
case WM_XBUTTONDOWN:
|
||||||
case WM_XBUTTONDBLCLK:
|
case WM_XBUTTONDBLCLK:
|
||||||
mouse.type = cast(MouseEventType) 1;
|
mouse.type = cast(MouseEventType) 1;
|
||||||
mouse.button = HIWORD(wParam) == 1 ? MouseButton.backButton : MouseButton.forwardButton;
|
mouse.button = HIWORD(wParam) == 1 ? MouseButton.backButton : MouseButton.forwardButton;
|
||||||
mouse.doubleClick = msg == WM_XBUTTONDBLCLK;
|
mouse.doubleClick = msg == WM_XBUTTONDBLCLK;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
return 1; // MSDN says special treatment here, return TRUE to bypass simulation programs
|
return 1; // MSDN says special treatment here, return TRUE to bypass simulation programs
|
||||||
case WM_XBUTTONUP:
|
case WM_XBUTTONUP:
|
||||||
mouse.type = cast(MouseEventType) 2;
|
mouse.type = cast(MouseEventType) 2;
|
||||||
mouse.button = HIWORD(wParam) == 1 ? MouseButton.backButton : MouseButton.forwardButton;
|
mouse.button = HIWORD(wParam) == 1 ? MouseButton.backButton : MouseButton.forwardButton;
|
||||||
mouseEvent();
|
mouseEvent(false, wParam);
|
||||||
return 1; // see: https://msdn.microsoft.com/en-us/library/windows/desktop/ms646246(v=vs.85).aspx
|
return 1; // see: https://msdn.microsoft.com/en-us/library/windows/desktop/ms646246(v=vs.85).aspx
|
||||||
|
|
||||||
default: return 1;
|
default: return 1;
|
||||||
|
|
Loading…
Reference in New Issue