mirror of https://github.com/adamdruppe/arsd.git
starting clipboard code
This commit is contained in:
parent
e38ed8babf
commit
dbc6110d57
462
simpledisplay.d
462
simpledisplay.d
|
@ -3,7 +3,7 @@ module simpledisplay;
|
|||
// Note: if you are using Image on X, you might want to do:
|
||||
/*
|
||||
static if(UsingSimpledisplayX11) {
|
||||
if(Image.impl.xshmQueryCompleted && !Image.impl.xshmAvailable) {
|
||||
if(!Image.impl.xshmAvailable) {
|
||||
// the images will use the slower XPutImage, you might
|
||||
// want to consider an alternative method to get better speed
|
||||
}
|
||||
|
@ -83,6 +83,135 @@ version(X11)
|
|||
else
|
||||
enum bool UsingSimpledisplayX11 = false;
|
||||
|
||||
|
||||
// basic functions to access the clipboard
|
||||
/+
|
||||
|
||||
|
||||
http://msdn.microsoft.com/en-us/library/windows/desktop/ff729168%28v=vs.85%29.aspx
|
||||
http://msdn.microsoft.com/en-us/library/windows/desktop/ms649039%28v=vs.85%29.aspx
|
||||
http://msdn.microsoft.com/en-us/library/windows/desktop/ms649035%28v=vs.85%29.aspx
|
||||
http://msdn.microsoft.com/en-us/library/windows/desktop/ms649051%28v=vs.85%29.aspx
|
||||
http://msdn.microsoft.com/en-us/library/windows/desktop/ms649037%28v=vs.85%29.aspx
|
||||
http://msdn.microsoft.com/en-us/library/windows/desktop/ms649035%28v=vs.85%29.aspx
|
||||
http://msdn.microsoft.com/en-us/library/windows/desktop/ms649016%28v=vs.85%29.aspx
|
||||
|
||||
string getClipboardText(SimpleWindow clipboardOwner) {
|
||||
version(Windows) {
|
||||
HWND hwndOwner = clipboardOwner ? clipboardOwner.impl.hwnd : null;
|
||||
if(OpenClipboard(hwndOwner) == 0)
|
||||
throw new Exception("OpenClipboard");
|
||||
scope(exit)
|
||||
CloseClipboard();
|
||||
if(auto dataHandle = GetClipboardData(1 /*CF_TEXT*/)) {
|
||||
/*
|
||||
Text format. Each line ends with a carriage return/linefeed (CR-LF) combination. A null character signals the end of the data. Use this format for ANSI text.
|
||||
|
||||
CF_UNICODETEXT
|
||||
13
|
||||
Unicode text format. Each line ends with a carriage return/linefeed (CR-LF) combination. A null character signals the end of the data.
|
||||
*/
|
||||
|
||||
if(auto data = GlobalLock(dataHandle)) {
|
||||
scope(exit)
|
||||
GlobalUnlock(data);
|
||||
|
||||
// FIXME actually copy data
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
assert(0, "not implementeD");
|
||||
}
|
||||
|
||||
void setClipboardText(string text) {
|
||||
version(Windows) {
|
||||
OpenClipboard();
|
||||
EmptyClipboard();
|
||||
SetClipboardData();
|
||||
CloseClipboard();
|
||||
}
|
||||
}
|
||||
+/
|
||||
|
||||
// FIXME: functions for doing images would be nice too - CF_DIB and whatever it is on X would be ok if we took the MemoryImage from color.d, or an Image from here. hell it might even be a variadic template that sets all the formats in one call. that might be cool.
|
||||
|
||||
version(X11) {
|
||||
// and the PRIMARY on X, be sure to put these in static if(UsingSimpledisplayX11)
|
||||
|
||||
@property Atom GetAtom(string name, bool create = false)(Display* display) {
|
||||
static Atom a;
|
||||
if(!a) {
|
||||
a = XInternAtom(display, name, !create);
|
||||
}
|
||||
if(a == None)
|
||||
throw new Exception("XInternAtom");
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Asserts ownership of PRIMARY and copies the text into a buffer that clients can request later
|
||||
void setPrimarySelection(SimpleWindow window, string text) {
|
||||
assert(window !is null);
|
||||
|
||||
auto display = XDisplayConnection.get();
|
||||
XSetSelectionOwner(display, GetAtom!"PRIMARY"(display), window.impl.window, 0 /* CurrentTime */);
|
||||
window.impl.setSelectionHandler = (XEvent ev) {
|
||||
XSelectionRequestEvent* event = &ev.xselectionrequest;
|
||||
XSelectionEvent selectionEvent;
|
||||
selectionEvent.type = EventType.SelectionNotify;
|
||||
selectionEvent.display = event.display;
|
||||
selectionEvent.requestor = event.requestor;
|
||||
selectionEvent.selection = event.selection;
|
||||
selectionEvent.time = event.time;
|
||||
selectionEvent.target = event.target;
|
||||
|
||||
if(event.property == None)
|
||||
selectionEvent.property = event.target;
|
||||
if(event.target == XA_STRING) {
|
||||
selectionEvent.property = event.property;
|
||||
XChangeProperty (display,
|
||||
selectionEvent.requestor,
|
||||
selectionEvent.property,
|
||||
event.target,
|
||||
8 /* bits */, 0 /* PropModeReplace */,
|
||||
text.ptr, text.length);
|
||||
} else if(event.target == GetAtom!"UTF8_STRING"(display)) {
|
||||
selectionEvent.property = event.property;
|
||||
XChangeProperty (display,
|
||||
selectionEvent.requestor,
|
||||
selectionEvent.property,
|
||||
event.target,
|
||||
8 /* bits */, 0 /* PropModeReplace */,
|
||||
text.ptr, text.length);
|
||||
} else {
|
||||
selectionEvent.property = None; // I don't know how to handle this type...
|
||||
}
|
||||
|
||||
XSendEvent(display, selectionEvent.requestor, false, 0, cast(XEvent*) &selectionEvent);
|
||||
};
|
||||
}
|
||||
|
||||
void getPrimarySelection(SimpleWindow window, void delegate(string) handler) {
|
||||
assert(window !is null);
|
||||
|
||||
auto display = XDisplayConnection.get();
|
||||
auto atom = GetAtom!"PRIMARY"(display);
|
||||
|
||||
window.impl.getSelectionHandler = handler;
|
||||
|
||||
auto target = XA_STRING;
|
||||
// or auto target = GetAtom!"UTF8_STRING"(display)
|
||||
XConvertSelection(display, atom, target, GetAtom!("SDD_DATA", true)(display), window.impl.window, 0 /*CurrentTime*/);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
enum RasterOp {
|
||||
normal,
|
||||
xor,
|
||||
}
|
||||
|
||||
// being phobos-free keeps the size WAY down
|
||||
private const(char)* toStringz(string s) { return (s ~ '\0').ptr; }
|
||||
private string[] split(string a, char c) {
|
||||
|
@ -646,6 +775,94 @@ final class Image {
|
|||
impl.dispose();
|
||||
}
|
||||
|
||||
// these numbers are used for working with rawData itself, skipping putPixel and getPixel
|
||||
// if you do the math yourself you might be able to optimize it. Call these functions only once and cache the value.
|
||||
pure const @system nothrow {
|
||||
/*
|
||||
To use these to draw a blue rectangle with size WxH at position X,Y...
|
||||
|
||||
// make certain that it will fit before we proceed
|
||||
enforce(X + W <= img.width && Y + H <= img.height); // you could also adjust the size to clip it, but be sure not to run off since this here will do raw pointers with no bounds checks!
|
||||
|
||||
// gather all the values you'll need up front. These can be kept until the image changes size if you want
|
||||
// (though calculating them isn't really that expensive).
|
||||
auto nextLineAdjustment = img.adjustmentForNextLine();
|
||||
auto offR = img.redByteOffset();
|
||||
auto offB = img.blueByteOffset();
|
||||
auto offG = img.greenByteOffset();
|
||||
auto bpp = img.bytesPerPixel();
|
||||
|
||||
auto data = img.getDataPointer();
|
||||
|
||||
// figure out the starting byte offset
|
||||
auto offset = img.offsetForTopLeftPixel() + nextLineAdjustment*Y + bpp * X;
|
||||
|
||||
auto startOfLine = data + offset; // get our pointer lined up on the first pixel
|
||||
|
||||
// and now our drawing loop for the rectangle
|
||||
foreach(y; 0 .. H) {
|
||||
auto data = startOfLine; // we keep the start of line separately so moving to the next line is simple and portable
|
||||
foreach(x; 0 .. W) {
|
||||
// write our color
|
||||
data[offR] = 0;
|
||||
data[offG] = 0;
|
||||
data[offB] = 255;
|
||||
|
||||
data += bpp; // moving to the next pixel is just an addition...
|
||||
}
|
||||
startOfLine += nextLineAdjustment;
|
||||
}
|
||||
|
||||
|
||||
As you can see, the loop itself was very simple thanks to the calculations being moved outside.
|
||||
|
||||
FIXME: I wonder if I can make the pixel formats consistently 32 bit across platforms, so the color offsets
|
||||
can be made into a bitmask or something so we can write them as *uint...
|
||||
*/
|
||||
|
||||
int offsetForTopLeftPixel() {
|
||||
version(X11) {
|
||||
return 0;
|
||||
} else version(Windows) {
|
||||
return (((cast(int) width * 3 + 3) / 4) * 4) * (height - 1);
|
||||
} else static assert(0, "fill in this info for other OSes");
|
||||
}
|
||||
|
||||
int adjustmentForNextLine() {
|
||||
version(X11) {
|
||||
return width * 4;
|
||||
} else version(Windows) {
|
||||
// windows bmps are upside down, so the adjustment is actually negative
|
||||
return -((cast(int) width * 3 + 3) / 4) * 4;
|
||||
} else static assert(0, "fill in this info for other OSes");
|
||||
}
|
||||
|
||||
// once you have the position of a pixel, use these to get to the proper color
|
||||
int redByteOffset() {
|
||||
version(X11) {
|
||||
return 2;
|
||||
} else version(Windows) {
|
||||
return 2;
|
||||
} else static assert(0, "fill in this info for other OSes");
|
||||
}
|
||||
|
||||
int greenByteOffset() {
|
||||
version(X11) {
|
||||
return 1;
|
||||
} else version(Windows) {
|
||||
return 1;
|
||||
} else static assert(0, "fill in this info for other OSes");
|
||||
}
|
||||
|
||||
int blueByteOffset() {
|
||||
version(X11) {
|
||||
return 0;
|
||||
} else version(Windows) {
|
||||
return 0;
|
||||
} else static assert(0, "fill in this info for other OSes");
|
||||
}
|
||||
}
|
||||
|
||||
final void putPixel(int x, int y, Color c) {
|
||||
if(x < 0 || x >= width)
|
||||
return;
|
||||
|
@ -655,6 +872,15 @@ final class Image {
|
|||
impl.setPixel(x, y, c);
|
||||
}
|
||||
|
||||
final Color getPixel(int x, int y) {
|
||||
if(x < 0 || x >= width)
|
||||
return Color.transparent;
|
||||
if(y < 0 || y >= height)
|
||||
return Color.transparent;
|
||||
|
||||
return impl.getPixel(x, y);
|
||||
}
|
||||
|
||||
final void opIndexAssign(Color c, int x, int y) {
|
||||
putPixel(x, y, c);
|
||||
}
|
||||
|
@ -795,6 +1021,10 @@ struct ScreenPainter {
|
|||
impl.fillColor(c);
|
||||
}
|
||||
|
||||
@property void rasterOp(RasterOp op) {
|
||||
impl.rasterOp(op);
|
||||
}
|
||||
|
||||
void transform(ref Point p) {
|
||||
p.x += originX;
|
||||
p.y += originY;
|
||||
|
@ -817,9 +1047,18 @@ struct ScreenPainter {
|
|||
impl.drawPixmap(s, upperLeft.x, upperLeft.y);
|
||||
}
|
||||
|
||||
void drawImage(Point upperLeft, Image i) {
|
||||
void drawImage(Point upperLeft, Image i, Point upperLeftOfImage = Point(0, 0), int w = 0, int h = 0) {
|
||||
transform(upperLeft);
|
||||
impl.drawImage(upperLeft.x, upperLeft.y, i);
|
||||
if(w == 0 || w > i.width)
|
||||
w = i.width;
|
||||
if(h == 0 || h > i.height)
|
||||
h = i.height;
|
||||
if(upperLeftOfImage.x < 0)
|
||||
upperLeftOfImage.x = 0;
|
||||
if(upperLeftOfImage.y < 0)
|
||||
upperLeftOfImage.y = 0;
|
||||
|
||||
impl.drawImage(upperLeft.x, upperLeft.y, i, upperLeftOfImage.x, upperLeftOfImage.y, w, h);
|
||||
}
|
||||
|
||||
Size textSize(string text) {
|
||||
|
@ -1248,7 +1487,7 @@ version(Windows) {
|
|||
return DefWindowProc(hWnd, iMessage, wParam, lParam);
|
||||
}
|
||||
} catch (Exception e) {
|
||||
assert(false, "Exception caught in WndProc");
|
||||
assert(false, "Exception caught in WndProc " ~ e.toString());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1334,6 +1573,19 @@ version(Windows) {
|
|||
pen = _activePen;
|
||||
}
|
||||
|
||||
@property void rasterOp(RasterOp op) {
|
||||
int mode;
|
||||
final switch(op) {
|
||||
case RasterOp.normal:
|
||||
mode = R2_COPYPEN;
|
||||
break;
|
||||
case RasterOp.xor:
|
||||
mode = R2_XORPEN;
|
||||
break;
|
||||
}
|
||||
SetROP2(hdc, mode);
|
||||
}
|
||||
|
||||
HBRUSH originalBrush;
|
||||
HBRUSH currentBrush;
|
||||
@property void fillColor(Color c) {
|
||||
|
@ -1356,7 +1608,7 @@ version(Windows) {
|
|||
// SetBkColor(hdc, RGB(255, 255, 255));
|
||||
}
|
||||
|
||||
void drawImage(int x, int y, Image i) {
|
||||
void drawImage(int x, int y, Image i, int ix, int iy, int w, int h) {
|
||||
BITMAP bm;
|
||||
|
||||
HDC hdcMem = CreateCompatibleDC(hdc);
|
||||
|
@ -1364,7 +1616,7 @@ version(Windows) {
|
|||
|
||||
GetObject(i.handle, bm.sizeof, &bm);
|
||||
|
||||
BitBlt(hdc, x, y, bm.bmWidth, bm.bmHeight, hdcMem, 0, 0, SRCCOPY);
|
||||
BitBlt(hdc, x, y, w /* bm.bmWidth */, /*bm.bmHeight*/ h, hdcMem, ix, iy, SRCCOPY);
|
||||
|
||||
SelectObject(hdcMem, hbmOld);
|
||||
DeleteDC(hdcMem);
|
||||
|
@ -1391,6 +1643,7 @@ version(Windows) {
|
|||
}
|
||||
|
||||
void drawText(int x, int y, int x2, int y2, string text, uint alignment) {
|
||||
// FIXME: use the unicode function
|
||||
if(x2 == 0 && y2 == 0)
|
||||
TextOut(hdc, x, y, text.ptr, text.length);
|
||||
else {
|
||||
|
@ -1442,7 +1695,7 @@ version(Windows) {
|
|||
}
|
||||
|
||||
void drawRectangle(int x, int y, int width, int height) {
|
||||
Rectangle(hdc, x, y, x + width, y + height);
|
||||
Rectangle(hdc, x, y, x + width+1, y + height+1); // FIXME: I think it now matches the X version with +1 but I don't think this is right
|
||||
}
|
||||
|
||||
/// Arguments are the points of the bounding rectangle
|
||||
|
@ -1614,6 +1867,20 @@ version(Windows) {
|
|||
// FIXME
|
||||
// ev.hardwareCode
|
||||
// ev.modifierState =
|
||||
|
||||
/+
|
||||
// we always want to send the character too, so let's convert it
|
||||
ubyte[256] state;
|
||||
wchar[16] buffer;
|
||||
GetKeyboardState(state.ptr);
|
||||
ToUnicodeEx(wParam, lParam, state.ptr, buffer.ptr, buffer.length, 0, null);
|
||||
|
||||
foreach(dchar d; buffer) {
|
||||
ev.character = d;
|
||||
break;
|
||||
}
|
||||
+/
|
||||
|
||||
ev.window = wind;
|
||||
if(wind.handleKeyEvent)
|
||||
wind.handleKeyEvent(ev);
|
||||
|
@ -1789,7 +2056,7 @@ version(Windows) {
|
|||
|
||||
if(!done && handlePulse !is null)
|
||||
handlePulse();
|
||||
Thread.sleep(dur!"msecs"(pulseTimeout));
|
||||
SleepEx(cast(DWORD) pulseTimeout, true);
|
||||
}
|
||||
} else {
|
||||
while((ret = GetMessage(&message, null, 0, 0)) != 0) {
|
||||
|
@ -1797,6 +2064,8 @@ version(Windows) {
|
|||
throw new Exception("GetMessage failed");
|
||||
TranslateMessage(&message);
|
||||
DispatchMessage(&message);
|
||||
|
||||
SleepEx(0, true); // I call this to give it a chance to do stuff like async io, which apparently never happens when you just block in GetMessage
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1808,6 +2077,19 @@ version(Windows) {
|
|||
HBITMAP handle;
|
||||
ubyte* rawData;
|
||||
|
||||
Color getPixel(int x, int y) {
|
||||
auto itemsPerLine = ((cast(int) width * 3 + 3) / 4) * 4;
|
||||
// remember, bmps are upside down
|
||||
auto offset = itemsPerLine * (height - y - 1) + x * 3;
|
||||
|
||||
Color c;
|
||||
c.a = 255;
|
||||
c.b = rawData[offset + 0];
|
||||
c.g = rawData[offset + 1];
|
||||
c.r = rawData[offset + 2];
|
||||
return c;
|
||||
}
|
||||
|
||||
void setPixel(int x, int y, Color c) {
|
||||
auto itemsPerLine = ((cast(int) width * 3 + 3) / 4) * 4;
|
||||
// remember, bmps are upside down
|
||||
|
@ -1989,6 +2271,21 @@ version(X11) {
|
|||
pen = _pen;
|
||||
}
|
||||
|
||||
@property void rasterOp(RasterOp op) {
|
||||
int mode;
|
||||
final switch(op) {
|
||||
case RasterOp.normal:
|
||||
mode = GXcopy;
|
||||
break;
|
||||
case RasterOp.xor:
|
||||
mode = GXxor;
|
||||
break;
|
||||
}
|
||||
XSetFunction(display, gc, mode);
|
||||
}
|
||||
|
||||
|
||||
|
||||
@property void fillColor(Color c) {
|
||||
_fillColor = c;
|
||||
if(c.a == 0) {
|
||||
|
@ -2011,12 +2308,12 @@ version(X11) {
|
|||
outlineColor = tmp;
|
||||
}
|
||||
|
||||
void drawImage(int x, int y, Image i) {
|
||||
void drawImage(int x, int y, Image i, int ix, int iy, int w, int h) {
|
||||
// source x, source y
|
||||
if(i.xshmAvailable)
|
||||
XShmPutImage(display, d, gc, i.handle, 0, 0, x, y, i.width, i.height, false);
|
||||
XShmPutImage(display, d, gc, i.handle, ix, iy, x, y, w, h, false);
|
||||
else
|
||||
XPutImage(display, d, gc, i.handle, 0, 0, x, y, i.width, i.height);
|
||||
XPutImage(display, d, gc, i.handle, ix, iy, x, y, w, h);
|
||||
}
|
||||
|
||||
void drawPixmap(Sprite s, int x, int y) {
|
||||
|
@ -2047,7 +2344,16 @@ version(X11) {
|
|||
return Size(maxWidth, h);
|
||||
}
|
||||
|
||||
void drawText(in int x, in int y, in int x2, in int y2, in string text, in uint alignment) {
|
||||
void drawText(in int x, in int y, in int x2, in int y2, in string originalText, in uint alignment) {
|
||||
// FIXME: we should actually draw unicode.. but until then, I'm going to strip out multibyte chars
|
||||
string text;
|
||||
foreach(dchar ch; originalText)
|
||||
if(ch < 128)
|
||||
text ~= ch;
|
||||
if(text.length == 0)
|
||||
return;
|
||||
|
||||
|
||||
int textHeight = 12;
|
||||
|
||||
// FIXME: should we clip it to the bounding box?
|
||||
|
@ -2100,7 +2406,7 @@ version(X11) {
|
|||
px = pos;
|
||||
}
|
||||
|
||||
XDrawString(display, d, gc, px, py + lineHeight, line.ptr, cast(int) line.length);
|
||||
XDrawString(display, d, gc, px, py + (font ? font.max_bounds.ascent : lineHeight), line.ptr, cast(int) line.length);
|
||||
cy += lineHeight + 4;
|
||||
}
|
||||
}
|
||||
|
@ -2119,7 +2425,7 @@ version(X11) {
|
|||
void drawRectangle(int x, int y, int width, int height) {
|
||||
if(backgroundIsNotTransparent) {
|
||||
swapColors();
|
||||
XFillRectangle(display, d, gc, x, y, width, height);
|
||||
XFillRectangle(display, d, gc, x+1, y+1, width-1, height-1);
|
||||
swapColors();
|
||||
}
|
||||
if(foregroundIsNotTransparent)
|
||||
|
@ -2210,19 +2516,21 @@ version(X11) {
|
|||
XShmSegmentInfo shminfo;
|
||||
|
||||
static bool xshmQueryCompleted;
|
||||
static bool xshmAvailable;
|
||||
static bool _xshmAvailable;
|
||||
public static @property bool xshmAvailable() {
|
||||
if(!xshmQueryCompleted) {
|
||||
int i1, i2, i3;
|
||||
xshmQueryCompleted = true;
|
||||
_xshmAvailable = XQueryExtension(XDisplayConnection.get(), "MIT-SHM", &i1, &i2, &i3);
|
||||
}
|
||||
return _xshmAvailable;
|
||||
}
|
||||
|
||||
void createImage(int width, int height) {
|
||||
auto display = XDisplayConnection.get();
|
||||
assert(display !is null);
|
||||
auto screen = DefaultScreen(display);
|
||||
|
||||
if(!xshmQueryCompleted) {
|
||||
int i1, i2, i3;
|
||||
xshmQueryCompleted = true;
|
||||
xshmAvailable = XQueryExtension(display, "MIT-SHM", &i1, &i2, &i3);
|
||||
}
|
||||
|
||||
if(xshmAvailable) {
|
||||
handle = XShmCreateImage(
|
||||
display,
|
||||
|
@ -2234,6 +2542,7 @@ version(X11) {
|
|||
width, height);
|
||||
assert(handle !is null);
|
||||
|
||||
assert(handle.bytes_per_line == 4 * width);
|
||||
shminfo.shmid = shmget(IPC_PRIVATE, handle.bytes_per_line * height, IPC_CREAT | 511 /* 0777 */);
|
||||
assert(shminfo.shmid >= 0);
|
||||
handle.data = shminfo.shmaddr = rawData = cast(ubyte*) shmat(shminfo.shmid, null, 0);
|
||||
|
@ -2266,14 +2575,19 @@ version(X11) {
|
|||
XDestroyImage(handle);
|
||||
if(xshmAvailable)
|
||||
shmdt(shminfo.shmaddr);
|
||||
handle = null;
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
Color getPixel(int x, int y) {
|
||||
|
||||
auto offset = (y * width + x) * 4;
|
||||
Color c;
|
||||
c.a = 255;
|
||||
c.b = rawData[offset + 0];
|
||||
c.g = rawData[offset + 1];
|
||||
c.r = rawData[offset + 2];
|
||||
return c;
|
||||
}
|
||||
*/
|
||||
|
||||
void setPixel(int x, int y, Color c) {
|
||||
auto offset = (y * width + x) * 4;
|
||||
|
@ -2317,6 +2631,9 @@ version(X11) {
|
|||
|
||||
Pixmap buffer;
|
||||
|
||||
void delegate(XEvent) setSelectionHandler;
|
||||
void delegate(string) getSelectionHandler;
|
||||
|
||||
version(without_opengl) {} else
|
||||
GLXContext glc;
|
||||
|
||||
|
@ -2404,6 +2721,8 @@ version(X11) {
|
|||
EventMask.ExposureMask |
|
||||
EventMask.KeyPressMask |
|
||||
EventMask.KeyReleaseMask |
|
||||
EventMask.PropertyChangeMask |
|
||||
EventMask.FocusChangeMask |
|
||||
EventMask.StructureNotifyMask
|
||||
| EventMask.PointerMotionMask // FIXME: not efficient
|
||||
| EventMask.ButtonPressMask
|
||||
|
@ -2468,6 +2787,49 @@ version(X11) {
|
|||
}
|
||||
|
||||
switch(e.type) {
|
||||
case EventType.SelectionClear:
|
||||
if(auto win = e.xselectionclear.window in SimpleWindow.nativeMapping)
|
||||
{ /* FIXME??????? */ }
|
||||
break;
|
||||
case EventType.SelectionRequest:
|
||||
if(auto win = e.xselectionrequest.owner in SimpleWindow.nativeMapping)
|
||||
if(win.setSelectionHandler !is null) {
|
||||
win.setSelectionHandler(e);
|
||||
}
|
||||
break;
|
||||
case EventType.SelectionNotify:
|
||||
if(auto win = e.xselection.requestor in SimpleWindow.nativeMapping)
|
||||
if(win.getSelectionHandler !is null) {
|
||||
// FIXME: maybe we should call a different handler for PRIMARY vs CLIPBOARD
|
||||
if(e.xselection.property == None || e.xselection.property == GetAtom!"NULL"(e.xselection.display)) {
|
||||
win.getSelectionHandler(null);
|
||||
} else {
|
||||
Atom target;
|
||||
int format;
|
||||
arch_ulong bytesafter, length;
|
||||
char* value;
|
||||
XGetWindowProperty(
|
||||
e.xselection.display,
|
||||
e.xselection.requestor,
|
||||
e.xselection.property,
|
||||
0,
|
||||
100000 /* length */,
|
||||
false,
|
||||
0 /*AnyPropertyType*/,
|
||||
&target, &format, &length, &bytesafter, &value);
|
||||
|
||||
// FIXME: it might be sent in pieces...
|
||||
// FIXME: or be other formats...
|
||||
|
||||
win.getSelectionHandler(value[0 .. length].idup);
|
||||
XFree(value);
|
||||
XDeleteProperty(
|
||||
e.xselection.display,
|
||||
e.xselection.requestor,
|
||||
e.xselection.property);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case EventType.ConfigureNotify:
|
||||
auto event = e.xconfigure;
|
||||
if(auto win = event.window in SimpleWindow.nativeMapping) {
|
||||
|
@ -2484,6 +2846,7 @@ version(X11) {
|
|||
|
||||
// resize the internal buffer to match the window...
|
||||
auto newPixmap = XCreatePixmap(display, cast(Drawable) event.window, win.width, win.height, 24);
|
||||
XFillRectangle(display, newPixmap, (*win).gc, 0, 0, win.width, win.height);
|
||||
XCopyArea(display,
|
||||
cast(Drawable) (*win).buffer,
|
||||
cast(Drawable) newPixmap,
|
||||
|
@ -2729,6 +3092,8 @@ version(Windows) {
|
|||
extern(Windows) {
|
||||
// The included D headers are incomplete, finish them here
|
||||
// enough that this module works.
|
||||
|
||||
DWORD SleepEx(DWORD, BOOL);
|
||||
alias GetObjectA GetObject;
|
||||
alias GetMessageA GetMessage;
|
||||
alias PeekMessageA PeekMessage;
|
||||
|
@ -2742,6 +3107,20 @@ version(Windows) {
|
|||
alias DefWindowProcA DefWindowProc;
|
||||
alias DrawTextA DrawText;
|
||||
|
||||
|
||||
int ToUnicodeEx(
|
||||
UINT wVirtKey,
|
||||
UINT wScanCode,
|
||||
const BYTE *lpKeyState,
|
||||
LPWSTR pwszBuff,
|
||||
int cchBuff,
|
||||
UINT wFlags,
|
||||
HKL dwhkl
|
||||
);
|
||||
BOOL GetKeyboardState(
|
||||
PBYTE lpKeyState
|
||||
);
|
||||
|
||||
enum DT_BOTTOM = 8;
|
||||
enum DT_CALCRECT = 1024;
|
||||
enum DT_CENTER = 1;
|
||||
|
@ -2818,6 +3197,10 @@ nothrow:
|
|||
HBRUSH GetSysColorBrush(int nIndex);
|
||||
DWORD GetSysColor(int nIndex);
|
||||
|
||||
int SetROP2(HDC, int);
|
||||
enum R2_XORPEN = 7;
|
||||
enum R2_COPYPEN = 13;
|
||||
|
||||
bool Ellipse(HDC, int, int, int, int);
|
||||
bool Arc(HDC, int, int, int, int, int, int, int, int);
|
||||
bool Polygon(HDC, POINT*, int);
|
||||
|
@ -2870,6 +3253,33 @@ KeySym XKeycodeToKeysym(
|
|||
int /* index */
|
||||
);
|
||||
|
||||
|
||||
int XConvertSelection(Display *display, Atom selection, Atom target,
|
||||
Atom property, Window requestor, Time time);
|
||||
|
||||
int XFree(void*);
|
||||
int XDeleteProperty(Display *display, Window w, Atom property);
|
||||
|
||||
int XChangeProperty(Display *display, Window w, Atom property, Atom
|
||||
type, int format, int mode, in void *data, int nelements);
|
||||
|
||||
|
||||
|
||||
int XGetWindowProperty(Display *display, Window w, Atom property, arch_long
|
||||
long_offset, arch_long long_length, Bool del, Atom req_type, Atom
|
||||
*actual_type_return, int *actual_format_return, arch_ulong
|
||||
*nitems_return, arch_ulong *bytes_after_return, char
|
||||
**prop_return);
|
||||
|
||||
int XSetSelectionOwner(Display *display, Atom selection, Window owner,
|
||||
Time time);
|
||||
|
||||
Window XGetSelectionOwner(Display *display, Atom selection);
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
Display* XOpenDisplay(const char*);
|
||||
int XCloseDisplay(Display*);
|
||||
|
||||
|
@ -4099,6 +4509,10 @@ struct Visual
|
|||
int XSetForeground(Display*, GC, uint);
|
||||
int XSetBackground(Display*, GC, uint);
|
||||
|
||||
int XSetFunction(Display*, GC, int);
|
||||
enum GXcopy = 0x3;
|
||||
enum GXxor = 0x6;
|
||||
|
||||
GC XCreateGC(Display*, Drawable, uint, void*);
|
||||
int XCopyGC(Display*, GC, uint, GC);
|
||||
int XFreeGC(Display*, GC);
|
||||
|
|
Loading…
Reference in New Issue