/* * Copyright (C) 2011 The Android Open Source Project * * 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 "X11Windowing.h" #include <stdio.h> #include <stdlib.h> #include <X11/Xlib.h> #include <X11/Xutil.h> #define DEBUG 0 #if DEBUG # define D(...) printf(__VA_ARGS__), printf("\n") #else # define D(...) ((void)0) #endif /* Try to remember the window position between creates/destroys */ static int X11_wmXPos = 100; static int X11_wmYPos = 100; static int X11_wmXAdjust = 0; static int X11_wmYAdjust = 0; static void get_window_pos( Display *disp, Window win, int *px, int *py ) { Window child; XTranslateCoordinates( disp, win, DefaultRootWindow(disp), 0, 0, px, py, &child ); } static void set_window_pos(Display *disp, Window win, int x, int y) { int xNew, yNew; int xAdjust = X11_wmXAdjust; int yAdjust = X11_wmYAdjust; /* this code is tricky because some window managers, but not all, * will translate the final window position by a given offset * corresponding to the frame decoration. * * so we first try to move the window, get the position that the * window manager has set, and if they are different, re-position the * window again with an adjustment. * * this causes a slight flicker since the window 'jumps' very * quickly from one position to the other. */ D("%s: move to [%d,%d] adjusted to [%d,%d]", __FUNCTION__, x, y, x+xAdjust, y+yAdjust); XMoveWindow(disp, win, x + xAdjust, y + yAdjust); XSync(disp, True); get_window_pos(disp, win, &xNew, &yNew); if (xNew != x || yNew != y) { X11_wmXAdjust = xAdjust = x - xNew; X11_wmYAdjust = yAdjust = y - yNew; D("%s: read pos [%d,%d], recomputing adjust=[%d,%d] moving to [%d,%d]\n", __FUNCTION__, xNew, yNew, xAdjust, yAdjust, x+xAdjust, y+yAdjust); XMoveWindow(disp, win, x + xAdjust, y + yAdjust ); } XSync(disp, False); } NativeDisplayType X11Windowing::getNativeDisplay() { Display *dpy = XOpenDisplay(NULL); return (NativeDisplayType)dpy; } NativeWindowType X11Windowing::createNativeWindow(NativeDisplayType _dpy, int width, int height) { Display *dpy = (Display *) _dpy; long defaultScreen = DefaultScreen( dpy ); Window rootWindow = RootWindow(dpy, defaultScreen); int depth = DefaultDepth(dpy, defaultScreen); XVisualInfo *visualInfo = new XVisualInfo; XMatchVisualInfo(dpy, defaultScreen, depth, TrueColor, visualInfo); if (visualInfo == NULL) { fprintf(stderr, "couldn't find matching visual\n"); return NULL; } Colormap x11Colormap = XCreateColormap(dpy, rootWindow, visualInfo->visual, AllocNone); XSetWindowAttributes sWA; sWA.colormap = x11Colormap; sWA.event_mask = StructureNotifyMask | ExposureMask; sWA.background_pixel = 0; sWA.border_pixel = 0; unsigned int attributes_mask = CWBackPixel | CWBorderPixel | CWEventMask | CWColormap; Window win = XCreateWindow( dpy, rootWindow, X11_wmXPos, X11_wmYPos, width, height, 0, CopyFromParent, InputOutput, CopyFromParent, attributes_mask, &sWA); XMapWindow(dpy, win); XFlush(dpy); set_window_pos(dpy, win, X11_wmXPos, X11_wmYPos); return NativeWindowType(win); } int X11Windowing::destroyNativeWindow(NativeDisplayType _dpy, NativeWindowType _win) { Display *dpy = (Display *)_dpy; Window win = (Window)_win; get_window_pos(dpy, win, &X11_wmXPos, &X11_wmYPos); D("%s: Saved window position [%d, %d]\n", __FUNCTION__, X11_wmXPos, X11_wmYPos); XDestroyWindow(dpy, win); XFlush(dpy); return 0; }