/** * FreeRDP: A Remote Desktop Protocol Client * X11 Windows * * Copyright 2011 Marc-Andre Moreau * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include #include #ifdef WITH_XEXT #include #endif #include "xf_window.h" /* Extended Window Manager Hints: http://standards.freedesktop.org/wm-spec/wm-spec-1.3.html */ #define MWM_HINTS_DECORATIONS (1L << 1) #define PROP_MOTIF_WM_HINTS_ELEMENTS 5 struct _PropMotifWmHints { unsigned long flags; unsigned long functions; unsigned long decorations; long inputMode; unsigned long status; }; typedef struct _PropMotifWmHints PropMotifWmHints; void xf_SendClientMessage(xfInfo* xfi, xfWindow* window, Atom atom, long msg, long d1, long d2, long d3) { XEvent xevent; xevent.xclient.type = ClientMessage; xevent.xclient.message_type = atom; xevent.xclient.window = window->handle; xevent.xclient.format = 32; xevent.xclient.data.l[0] = CurrentTime; xevent.xclient.data.l[1] = msg; xevent.xclient.data.l[2] = d1; xevent.xclient.data.l[3] = d2; xevent.xclient.data.l[4] = d3; XSendEvent(xfi->display, window->handle, False, NoEventMask, &xevent); XSync(xfi->display, False); } void xf_SetWindowFullscreen(xfInfo* xfi, xfWindow* window, boolean fullscreen) { if (fullscreen) { if (window->decorations) xf_SetWindowDecorations(xfi, window, False); XMoveResizeWindow(xfi->display, window->handle, 0, 0, window->width, window->height); XMapRaised(xfi->display, window->handle); //XGrabPointer(xfi->display, window->handle, True, 0, GrabModeAsync, GrabModeAsync, window->handle, 0L, CurrentTime); //XGrabKeyboard(xfi->display, window->handle, False, GrabModeAsync, GrabModeAsync, CurrentTime); //XSetInputFocus(xfi->display, window->handle, RevertToParent, CurrentTime); window->fullscreen = True; } } /* http://tronche.com/gui/x/xlib/window-information/XGetWindowProperty.html */ boolean xf_GetWindowProperty(xfInfo* xfi, Window window, Atom property, int length, unsigned long* nitems, unsigned long* bytes, uint8** prop) { int status; Atom actual_type; int actual_format; if (property == None) return False; status = XGetWindowProperty(xfi->display, window, property, 0, length, False, AnyPropertyType, &actual_type, &actual_format, nitems, bytes, prop); if (status != Success) return False; return True; } boolean xf_GetCurrentDesktop(xfInfo* xfi) { boolean status; unsigned long nitems; unsigned long bytes; unsigned char* prop; status = xf_GetWindowProperty(xfi, DefaultRootWindow(xfi->display), xfi->_NET_CURRENT_DESKTOP, 1, &nitems, &bytes, &prop); if (status != True) return False; xfi->current_desktop = (int) *prop; xfree(prop); return True; } boolean xf_GetWorkArea(xfInfo* xfi) { long* plong; boolean status; unsigned long nitems; unsigned long bytes; unsigned char* prop; status = xf_GetWindowProperty(xfi, DefaultRootWindow(xfi->display), xfi->_NET_WORKAREA, 32 * 4, &nitems, &bytes, &prop); if (status != True) return False; xf_GetCurrentDesktop(xfi); plong = (long*) prop; xfi->workArea.x = plong[xfi->current_desktop * 4 + 0]; xfi->workArea.y = plong[xfi->current_desktop * 4 + 1]; xfi->workArea.width = plong[xfi->current_desktop * 4 + 2]; xfi->workArea.height = plong[xfi->current_desktop * 4 + 3]; xfree(prop); return True; } void xf_SetWindowDecorations(xfInfo* xfi, xfWindow* window, boolean show) { PropMotifWmHints hints; hints.decorations = show; hints.flags = MWM_HINTS_DECORATIONS; XChangeProperty(xfi->display, window->handle, xfi->_MOTIF_WM_HINTS, xfi->_MOTIF_WM_HINTS, 32, PropModeReplace, (uint8*) &hints, PROP_MOTIF_WM_HINTS_ELEMENTS); window->decorations = show; } void xf_SetWindowUnlisted(xfInfo* xfi, xfWindow* window) { Atom window_state[2]; window_state[0] = xfi->_NET_WM_STATE_SKIP_PAGER; window_state[1] = xfi->_NET_WM_STATE_SKIP_TASKBAR; XChangeProperty(xfi->display, window->handle, xfi->_NET_WM_STATE, XA_ATOM, 32, PropModeReplace, (uint8*) &window_state, 2); } void xf_SetWindowStyle(xfInfo* xfi, xfWindow* window, uint32 style, uint32 ex_style) { Atom window_type; window_type = xfi->_NET_WM_WINDOW_TYPE_NORMAL; if ((style & WS_POPUP) || (style & WS_DLGFRAME) || (ex_style & WS_EX_DLGMODALFRAME)) { window_type = xfi->_NET_WM_WINDOW_TYPE_DIALOG; } if (ex_style & WS_EX_TOOLWINDOW) { xf_SetWindowUnlisted(xfi, window); window_type = xfi->_NET_WM_WINDOW_TYPE_UTILITY; } XChangeProperty(xfi->display, window->handle, xfi->_NET_WM_WINDOW_TYPE, XA_ATOM, 32, PropModeReplace, (uint8*) &window_type, 1); } xfWindow* xf_CreateDesktopWindow(xfInfo* xfi, char* name, int width, int height) { xfWindow* window; window = (xfWindow*) xzalloc(sizeof(xfWindow)); if (window != NULL) { int input_mask; XSizeHints* size_hints; XClassHint* class_hints; window->width = width; window->height = height; window->decorations = True; window->fullscreen = True; window->handle = XCreateWindow(xfi->display, RootWindowOfScreen(xfi->screen), xfi->workArea.x, xfi->workArea.y, xfi->width, xfi->height, 0, xfi->depth, InputOutput, xfi->visual, CWBackPixel | CWBackingStore | CWOverrideRedirect | CWColormap | CWBorderPixel, &xfi->attribs); class_hints = XAllocClassHint(); if (class_hints != NULL) { if (name != NULL) class_hints->res_name = name; class_hints->res_class = "freerdp"; XSetClassHint(xfi->display, window->handle, class_hints); XFree(class_hints); } size_hints = XAllocSizeHints(); if (size_hints) { size_hints->flags = PMinSize | PMaxSize; size_hints->min_width = size_hints->max_width = xfi->width; size_hints->min_height = size_hints->max_height = xfi->height; XSetWMNormalHints(xfi->display, window->handle, size_hints); XFree(size_hints); } input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | PointerMotionMask | ExposureMask; XSelectInput(xfi->display, window->handle, input_mask); XMapWindow(xfi->display, window->handle); } return window; } void xf_FixWindowCoordinates(xfInfo* xfi, int* x, int* y, int* width, int* height) { int vscreen_width; int vscreen_height; vscreen_width = xfi->vscreen.area.right - xfi->vscreen.area.left + 1; vscreen_height = xfi->vscreen.area.bottom - xfi->vscreen.area.top + 1; if (*x < xfi->vscreen.area.left) { *width += *x; *x = xfi->vscreen.area.left; } if (*y < xfi->vscreen.area.top) { *height += *y; *y = xfi->vscreen.area.top; } if (*width > vscreen_width) { *width = vscreen_width; } if (*height > vscreen_height) { *height = vscreen_height; } } char rail_window_class[] = "RAIL:00000000"; xfWindow* xf_CreateWindow(xfInfo* xfi, xfWindow* parent, int x, int y, int width, int height, uint32 id) { xfWindow* window; window = (xfWindow*) xzalloc(sizeof(xfWindow)); if ((width * height) < 1) return NULL; xf_FixWindowCoordinates(xfi, &x, &y, &width, &height); window->left = x; window->top = y; window->right = x + width - 1; window->bottom = y + height - 1; window->width = width; window->height = height; if (window != NULL) { XGCValues gcv; int input_mask; XClassHint* class_hints; window->decorations = False; window->fullscreen = False; window->parent = parent; window->isLocalMoveSizeModeEnabled = False; window->handle = XCreateWindow(xfi->display, RootWindowOfScreen(xfi->screen), x, y, window->width, window->height, 0, xfi->depth, InputOutput, xfi->visual, CWBackPixel | CWBackingStore | CWOverrideRedirect | CWColormap | CWBorderPixel, &xfi->attribs); xf_SetWindowDecorations(xfi, window, window->decorations); class_hints = XAllocClassHint(); if (class_hints != NULL) { char* class; class = xmalloc(sizeof(rail_window_class)); snprintf(class, sizeof(rail_window_class), "RAIL:%08X", id); class_hints->res_name = "RAIL"; class_hints->res_class = class; XSetClassHint(xfi->display, window->handle, class_hints); XFree(class_hints); xfree(class); } input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask | VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | PointerMotionMask | ExposureMask; XSelectInput(xfi->display, window->handle, input_mask); XMapWindow(xfi->display, window->handle); memset(&gcv, 0, sizeof(gcv)); window->gc = XCreateGC(xfi->display, window->handle, GCGraphicsExposures, &gcv); window->surface = XCreatePixmap(xfi->display, window->handle, window->width, window->height, xfi->depth); xf_MoveWindow(xfi, window, x, y, width, height); } return window; } void xf_SetWindowMinMaxInfo(xfInfo* xfi, xfWindow* window, int maxWidth, int maxHeight, int maxPosX, int maxPosY, int minTrackWidth, int minTrackHeight, int maxTrackWidth, int maxTrackHeight) { XSizeHints* size_hints; size_hints = XAllocSizeHints(); if (size_hints) { size_hints->flags = PMinSize | PMaxSize | PResizeInc; size_hints->min_width = minTrackWidth; size_hints->min_height = minTrackHeight; size_hints->max_width = maxTrackWidth; size_hints->max_height = maxTrackHeight; /* to speedup window drawing we need to select optimal value for sizing step. */ size_hints->width_inc = size_hints->height_inc = 5; XSetWMNormalHints(xfi->display, window->handle, size_hints); XFree(size_hints); } } void xf_SendMoveResizeEvent(xfInfo* xfi, xfWindow* window, int direction, int x_root, int y_root) { // TODO: // - how to receive movesize canceling event? // - how to produce correct RAIL movesize finish? // - how to receive move/size window coordinates in process of local move/size? XEvent event; event.xclient.type = ClientMessage; event.xclient.window = window->handle; event.xclient.message_type = xfi->_NET_WM_MOVERESIZE; event.xclient.serial = 0; event.xclient.display = xfi->display; event.xclient.send_event = True; event.xclient.format = 32; event.xclient.data.l[0] = x_root; event.xclient.data.l[1] = y_root; event.xclient.data.l[2] = direction; event.xclient.data.l[3] = 1; /* button 1 */ event.xclient.data.l[4] = 0; XUngrabPointer(xfi->display, CurrentTime); XSendEvent(xfi->display, RootWindowOfScreen(xfi->screen), False, SubstructureNotifyMask, &event); } #define XF_NET_WM_MOVERESIZE_SIZE_TOPLEFT 0 #define XF_NET_WM_MOVERESIZE_SIZE_TOP 1 #define XF_NET_WM_MOVERESIZE_SIZE_TOPRIGHT 2 #define XF_NET_WM_MOVERESIZE_SIZE_RIGHT 3 #define XF_NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT 4 #define XF_NET_WM_MOVERESIZE_SIZE_BOTTOM 5 #define XF_NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT 6 #define XF_NET_WM_MOVERESIZE_SIZE_LEFT 7 #define XF_NET_WM_MOVERESIZE_MOVE 8 /* movement only */ #define XF_NET_WM_MOVERESIZE_SIZE_KEYBOARD 9 /* size via keyboard */ #define XF_NET_WM_MOVERESIZE_MOVE_KEYBOARD 10 /* move via keyboard */ #define XF_NET_WM_MOVERESIZE_CANCEL 11 /* cancel operation */ void xf_StartLocalMoveSize(xfInfo* xfi, xfWindow* window, uint16 moveSizeType, int posX, int posY) { struct { uint16 moveSizeType; int direction; } type2dir_table[] = { { RAIL_WMSZ_LEFT, XF_NET_WM_MOVERESIZE_SIZE_LEFT }, { RAIL_WMSZ_RIGHT, XF_NET_WM_MOVERESIZE_SIZE_RIGHT }, { RAIL_WMSZ_TOP, XF_NET_WM_MOVERESIZE_SIZE_TOP }, { RAIL_WMSZ_TOPLEFT, XF_NET_WM_MOVERESIZE_SIZE_TOPLEFT }, { RAIL_WMSZ_TOPRIGHT, XF_NET_WM_MOVERESIZE_SIZE_TOPRIGHT }, { RAIL_WMSZ_BOTTOM, XF_NET_WM_MOVERESIZE_SIZE_BOTTOM }, { RAIL_WMSZ_BOTTOMLEFT, XF_NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT }, { RAIL_WMSZ_BOTTOMRIGHT, XF_NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT }, { RAIL_WMSZ_MOVE, XF_NET_WM_MOVERESIZE_MOVE }, { RAIL_WMSZ_KEYMOVE, XF_NET_WM_MOVERESIZE_MOVE_KEYBOARD }, { RAIL_WMSZ_KEYSIZE, XF_NET_WM_MOVERESIZE_SIZE_KEYBOARD }, }; int i = 0; int x_root = 0; int y_root = 0; int direction = -1; window->isLocalMoveSizeModeEnabled = True; x_root = posX; y_root = posY; if (moveSizeType == RAIL_WMSZ_MOVE) { x_root += window->left; y_root += window->top; } for (i = 0; i < RAIL_ARRAY_SIZE(type2dir_table); i++) { if (type2dir_table[i].moveSizeType == moveSizeType) { direction = type2dir_table[i].direction; break; } } if (direction == -1) { printf("xf_StartLocalMoveSize: unknown moveSizeType. (window=0x%X moveSizeType=0x%X)\n", (uint32) window->handle, moveSizeType); return; } xf_SendMoveResizeEvent(xfi, window, direction, x_root, y_root); } void xf_StopLocalMoveSize(xfInfo* xfi, xfWindow* window, uint16 moveSizeType, int topLeftX, int topLeftY) { window->isLocalMoveSizeModeEnabled = False; if (moveSizeType == RAIL_WMSZ_MOVE) { xf_MoveWindow(xfi, window, topLeftX, topLeftY, window->width, window->height); } } void xf_MoveWindow(xfInfo* xfi, xfWindow* window, int x, int y, int width, int height) { Pixmap surface; if ((width * height) < 1) return; if (window->isLocalMoveSizeModeEnabled) return; xf_FixWindowCoordinates(xfi, &x, &y, &width, &height); XMoveResizeWindow(xfi->display, window->handle, x, y, width, height); surface = XCreatePixmap(xfi->display, window->handle, width, height, xfi->depth); XCopyArea(xfi->display, surface, window->surface, window->gc, 0, 0, window->width, window->height, 0, 0); XFreePixmap(xfi->display, window->surface); window->surface = surface; window->left = x; window->top = y; window->right = x + width - 1; window->bottom = y + height - 1; window->width = width; window->height = height; } void xf_ShowWindow(xfInfo* xfi, xfWindow* window, uint8 state) { switch (state) { case WINDOW_HIDE: XWithdrawWindow(xfi->display, window->handle, xfi->screen_number); break; case WINDOW_SHOW_MINIMIZED: XIconifyWindow(xfi->display, window->handle, xfi->screen_number); break; case WINDOW_SHOW_MAXIMIZED: XRaiseWindow(xfi->display, window->handle); break; case WINDOW_SHOW: XMapWindow(xfi->display, window->handle); break; } XFlush(xfi->display); } void xf_SetWindowIcon(xfInfo* xfi, xfWindow* window, rdpIcon* icon) { int x, y; int pixels; int propsize; long* propdata; long* dstp; uint32* srcp; if (icon->big != True) return; pixels = icon->entry->width * icon->entry->height; propsize = 2 + pixels; propdata = xmalloc(propsize * sizeof(long)); propdata[0] = icon->entry->width; propdata[1] = icon->entry->height; dstp = &(propdata[2]); srcp = (uint32*) icon->extra; for (y = 0; y < icon->entry->height; y++) { for (x = 0; x < icon->entry->width; x++) { *dstp++ = *srcp++; } } XChangeProperty(xfi->display, window->handle, xfi->_NET_WM_ICON, XA_CARDINAL, 32, PropModeReplace, (uint8*) propdata, propsize); XFlush(xfi->display); } void xf_SetWindowRects(xfInfo* xfi, xfWindow* window, RECTANGLE_16* rects, int nrects) { int i; XRectangle* xrects; xrects = xmalloc(sizeof(XRectangle) * nrects); for (i = 0; i < nrects; i++) { xrects[i].x = rects[i].left; xrects[i].y = rects[i].top; xrects[i].width = rects[i].right - rects[i].left; xrects[i].height = rects[i].bottom - rects[i].top; } #ifdef WITH_XEXT XShapeCombineRectangles(xfi->display, window->handle, ShapeBounding, 0, 0, xrects, nrects, ShapeSet, 0); #endif xfree(xrects); } void xf_SetWindowVisibilityRects(xfInfo* xfi, xfWindow* window, RECTANGLE_16* rects, int nrects) { int i; XRectangle* xrects; xrects = xmalloc(sizeof(XRectangle) * nrects); for (i = 0; i < nrects; i++) { xrects[i].x = rects[i].left; xrects[i].y = rects[i].top; xrects[i].width = rects[i].right - rects[i].left; xrects[i].height = rects[i].bottom - rects[i].top; } #ifdef WITH_XEXT //XShapeCombineRectangles(xfi->display, window->handle, ShapeBounding, 0, 0, xrects, nrects, ShapeSet, 0); #endif xfree(xrects); } void xf_DestroyWindow(xfInfo* xfi, xfWindow* window) { if (window->gc) XFreeGC(xfi->display, window->gc); if (window->surface) XFreePixmap(xfi->display, window->surface); if (window->handle) { XUnmapWindow(xfi->display, window->handle); XDestroyWindow(xfi->display, window->handle); } xfree(window); }