starting clipboard code

This commit is contained in:
Adam D. Ruppe 2013-09-27 15:33:15 -04:00
parent e38ed8babf
commit dbc6110d57
1 changed files with 438 additions and 24 deletions

View File

@ -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);