2006-07-27 06:53:23 +00:00
|
|
|
/*
|
|
|
|
SDL - Simple DirectMedia Layer
|
2011-02-11 22:37:15 -08:00
|
|
|
Copyright (C) 1997-2011 Sam Lantinga
|
2006-07-27 06:53:23 +00:00
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Lesser General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Lesser General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
|
|
License along with this library; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
|
|
|
|
Sam Lantinga
|
|
|
|
slouken@libsdl.org
|
|
|
|
*/
|
|
|
|
#include "SDL_config.h"
|
|
|
|
|
|
|
|
#include "../SDL_sysvideo.h"
|
2010-05-09 20:47:22 -07:00
|
|
|
#include "../SDL_pixels_c.h"
|
2006-07-27 06:53:23 +00:00
|
|
|
#include "../../events/SDL_keyboard_c.h"
|
2009-01-01 07:59:08 +00:00
|
|
|
#include "../../events/SDL_mouse_c.h"
|
2006-07-27 06:53:23 +00:00
|
|
|
|
|
|
|
#include "SDL_x11video.h"
|
2009-01-01 07:59:08 +00:00
|
|
|
#include "SDL_x11mouse.h"
|
2010-06-30 16:19:44 -04:00
|
|
|
#include "SDL_x11shape.h"
|
2006-07-27 06:53:23 +00:00
|
|
|
|
2009-05-31 11:53:12 +00:00
|
|
|
#ifdef SDL_VIDEO_DRIVER_PANDORA
|
|
|
|
#include "SDL_x11opengles.h"
|
|
|
|
#endif
|
|
|
|
|
2010-05-09 20:47:22 -07:00
|
|
|
#include "SDL_timer.h"
|
2009-09-05 07:33:54 +00:00
|
|
|
#include "SDL_syswm.h"
|
|
|
|
|
2009-02-19 06:48:25 +00:00
|
|
|
#define _NET_WM_STATE_REMOVE 0l
|
|
|
|
#define _NET_WM_STATE_ADD 1l
|
|
|
|
#define _NET_WM_STATE_TOGGLE 2l
|
|
|
|
|
2010-07-13 23:11:10 -07:00
|
|
|
static SDL_bool
|
2010-07-14 00:56:08 -07:00
|
|
|
X11_IsWindowOldFullscreen(_THIS, SDL_Window * window)
|
2010-07-13 23:11:10 -07:00
|
|
|
{
|
2010-07-14 00:56:08 -07:00
|
|
|
SDL_VideoData *videodata = (SDL_VideoData *) _this->driverdata;
|
2010-07-13 23:11:10 -07:00
|
|
|
|
|
|
|
/* ICCCM2.0-compliant window managers can handle fullscreen windows */
|
|
|
|
if ((window->flags & SDL_WINDOW_FULLSCREEN) && !videodata->net_wm) {
|
|
|
|
return SDL_TRUE;
|
|
|
|
} else {
|
|
|
|
return SDL_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-14 00:56:08 -07:00
|
|
|
static SDL_bool
|
|
|
|
X11_IsWindowMapped(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
SDL_VideoData *videodata = (SDL_VideoData *) _this->driverdata;
|
|
|
|
XWindowAttributes attr;
|
|
|
|
|
|
|
|
XGetWindowAttributes(videodata->display, data->xwindow, &attr);
|
|
|
|
if (attr.map_state != IsUnmapped) {
|
|
|
|
return SDL_TRUE;
|
|
|
|
} else {
|
|
|
|
return SDL_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
X11_GetWMStateProperty(_THIS, SDL_Window * window, Atom atoms[3])
|
|
|
|
{
|
|
|
|
SDL_VideoData *data = (SDL_VideoData *) _this->driverdata;
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
if (window->flags & SDL_WINDOW_FULLSCREEN) {
|
|
|
|
atoms[count++] = data->_NET_WM_STATE_FULLSCREEN;
|
|
|
|
}
|
|
|
|
if (window->flags & SDL_WINDOW_MAXIMIZED) {
|
|
|
|
atoms[count++] = data->_NET_WM_STATE_MAXIMIZED_VERT;
|
|
|
|
atoms[count++] = data->_NET_WM_STATE_MAXIMIZED_HORZ;
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2008-12-30 17:09:42 +00:00
|
|
|
static void
|
|
|
|
X11_GetDisplaySize(_THIS, SDL_Window * window, int *w, int *h)
|
|
|
|
{
|
|
|
|
SDL_VideoData *data = (SDL_VideoData *) _this->driverdata;
|
|
|
|
SDL_DisplayData *displaydata =
|
2011-02-10 14:44:25 -08:00
|
|
|
(SDL_DisplayData *) SDL_GetDisplayForWindow(window)->driverdata;
|
2008-12-30 17:09:42 +00:00
|
|
|
XWindowAttributes attr;
|
|
|
|
|
2010-07-30 00:55:00 -07:00
|
|
|
XGetWindowAttributes(data->display, RootWindow(data->display, displaydata->screen), &attr);
|
|
|
|
if (window->flags & SDL_WINDOW_FULLSCREEN) {
|
|
|
|
/* The bounds when this window is visible is the fullscreen mode */
|
|
|
|
SDL_DisplayMode fullscreen_mode;
|
|
|
|
if (SDL_GetWindowDisplayMode(window, &fullscreen_mode) == 0) {
|
|
|
|
attr.width = fullscreen_mode.w;
|
|
|
|
attr.height = fullscreen_mode.h;
|
|
|
|
}
|
|
|
|
}
|
2008-12-30 17:09:42 +00:00
|
|
|
if (w) {
|
|
|
|
*w = attr.width;
|
|
|
|
}
|
|
|
|
if (h) {
|
|
|
|
*h = attr.height;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
static int
|
|
|
|
SetupWindowData(_THIS, SDL_Window * window, Window w, BOOL created)
|
|
|
|
{
|
|
|
|
SDL_VideoData *videodata = (SDL_VideoData *) _this->driverdata;
|
|
|
|
SDL_WindowData *data;
|
|
|
|
int numwindows = videodata->numwindows;
|
2008-03-07 17:20:37 +00:00
|
|
|
int windowlistlength = videodata->windowlistlength;
|
2006-07-27 06:53:23 +00:00
|
|
|
SDL_WindowData **windowlist = videodata->windowlist;
|
|
|
|
|
|
|
|
/* Allocate the window data */
|
2008-03-06 23:07:02 +00:00
|
|
|
data = (SDL_WindowData *) SDL_calloc(1, sizeof(*data));
|
2006-07-27 06:53:23 +00:00
|
|
|
if (!data) {
|
|
|
|
SDL_OutOfMemory();
|
|
|
|
return -1;
|
|
|
|
}
|
2010-01-21 06:21:52 +00:00
|
|
|
data->window = window;
|
|
|
|
data->xwindow = w;
|
2006-07-27 06:53:23 +00:00
|
|
|
#ifdef X_HAVE_UTF8_STRING
|
|
|
|
if (SDL_X11_HAVE_UTF8) {
|
|
|
|
data->ic =
|
|
|
|
pXCreateIC(videodata->im, XNClientWindow, w, XNFocusWindow, w,
|
|
|
|
XNInputStyle, XIMPreeditNothing | XIMStatusNothing,
|
|
|
|
XNResourceName, videodata->classname, XNResourceClass,
|
|
|
|
videodata->classname, NULL);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
data->created = created;
|
|
|
|
data->videodata = videodata;
|
|
|
|
|
2008-03-06 23:07:02 +00:00
|
|
|
/* Associate the data with the window */
|
|
|
|
|
2008-03-07 17:20:37 +00:00
|
|
|
if (numwindows < windowlistlength) {
|
|
|
|
windowlist[numwindows] = data;
|
|
|
|
videodata->numwindows++;
|
2008-03-06 23:07:02 +00:00
|
|
|
} else {
|
2008-03-07 17:20:37 +00:00
|
|
|
windowlist =
|
|
|
|
(SDL_WindowData **) SDL_realloc(windowlist,
|
|
|
|
(numwindows +
|
|
|
|
1) * sizeof(*windowlist));
|
|
|
|
if (!windowlist) {
|
|
|
|
SDL_OutOfMemory();
|
|
|
|
SDL_free(data);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
windowlist[numwindows] = data;
|
|
|
|
videodata->numwindows++;
|
|
|
|
videodata->windowlistlength++;
|
|
|
|
videodata->windowlist = windowlist;
|
2008-03-06 23:07:02 +00:00
|
|
|
}
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
/* Fill in the SDL window with the window data */
|
|
|
|
{
|
|
|
|
XWindowAttributes attrib;
|
|
|
|
|
|
|
|
XGetWindowAttributes(data->videodata->display, w, &attrib);
|
|
|
|
window->x = attrib.x;
|
|
|
|
window->y = attrib.y;
|
|
|
|
window->w = attrib.width;
|
|
|
|
window->h = attrib.height;
|
|
|
|
if (attrib.map_state != IsUnmapped) {
|
|
|
|
window->flags |= SDL_WINDOW_SHOWN;
|
|
|
|
} else {
|
|
|
|
window->flags &= ~SDL_WINDOW_SHOWN;
|
|
|
|
}
|
2011-02-04 18:05:20 -08:00
|
|
|
data->visual = attrib.visual;
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
2009-02-19 06:48:25 +00:00
|
|
|
|
|
|
|
{
|
2010-07-13 23:11:10 -07:00
|
|
|
Atom _NET_WM_STATE = data->videodata->_NET_WM_STATE;
|
|
|
|
Atom _NET_WM_STATE_MAXIMIZED_VERT = data->videodata->_NET_WM_STATE_MAXIMIZED_VERT;
|
|
|
|
Atom _NET_WM_STATE_MAXIMIZED_HORZ = data->videodata->_NET_WM_STATE_MAXIMIZED_HORZ;
|
|
|
|
Atom _NET_WM_STATE_FULLSCREEN = data->videodata->_NET_WM_STATE_FULLSCREEN;
|
2009-02-19 06:48:25 +00:00
|
|
|
Atom actualType;
|
|
|
|
int actualFormat;
|
|
|
|
unsigned long i, numItems, bytesAfter;
|
|
|
|
unsigned char *propertyValue = NULL;
|
|
|
|
long maxLength = 1024;
|
|
|
|
|
|
|
|
if (XGetWindowProperty(data->videodata->display, w, _NET_WM_STATE,
|
2009-02-19 06:49:59 +00:00
|
|
|
0l, maxLength, False, XA_ATOM, &actualType,
|
|
|
|
&actualFormat, &numItems, &bytesAfter,
|
|
|
|
&propertyValue) == Success) {
|
|
|
|
Atom *atoms = (Atom *) propertyValue;
|
2009-02-19 06:48:25 +00:00
|
|
|
int maximized = 0;
|
2010-07-13 23:11:10 -07:00
|
|
|
int fullscreen = 0;
|
2009-02-19 06:48:25 +00:00
|
|
|
|
|
|
|
for (i = 0; i < numItems; ++i) {
|
|
|
|
if (atoms[i] == _NET_WM_STATE_MAXIMIZED_VERT) {
|
|
|
|
maximized |= 1;
|
|
|
|
} else if (atoms[i] == _NET_WM_STATE_MAXIMIZED_HORZ) {
|
|
|
|
maximized |= 2;
|
2010-07-13 23:11:10 -07:00
|
|
|
} else if ( atoms[i] == _NET_WM_STATE_FULLSCREEN) {
|
|
|
|
fullscreen = 1;
|
2009-02-19 06:48:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (maximized == 3) {
|
|
|
|
window->flags |= SDL_WINDOW_MAXIMIZED;
|
2010-07-13 23:11:10 -07:00
|
|
|
} else if (fullscreen == 1) {
|
|
|
|
window->flags |= SDL_WINDOW_FULLSCREEN;
|
2009-02-19 06:48:25 +00:00
|
|
|
}
|
|
|
|
XFree(propertyValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
/* FIXME: How can I tell?
|
|
|
|
{
|
|
|
|
DWORD style = GetWindowLong(hwnd, GWL_STYLE);
|
|
|
|
if (style & WS_VISIBLE) {
|
|
|
|
if (style & (WS_BORDER | WS_THICKFRAME)) {
|
|
|
|
window->flags &= ~SDL_WINDOW_BORDERLESS;
|
|
|
|
} else {
|
|
|
|
window->flags |= SDL_WINDOW_BORDERLESS;
|
|
|
|
}
|
|
|
|
if (style & WS_THICKFRAME) {
|
|
|
|
window->flags |= SDL_WINDOW_RESIZABLE;
|
|
|
|
} else {
|
|
|
|
window->flags &= ~SDL_WINDOW_RESIZABLE;
|
|
|
|
}
|
|
|
|
if (style & WS_MINIMIZE) {
|
|
|
|
window->flags |= SDL_WINDOW_MINIMIZED;
|
|
|
|
} else {
|
|
|
|
window->flags &= ~SDL_WINDOW_MINIMIZED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (GetFocus() == hwnd) {
|
|
|
|
int index = data->videodata->keyboard;
|
|
|
|
window->flags |= SDL_WINDOW_INPUT_FOCUS;
|
2010-01-21 06:21:52 +00:00
|
|
|
SDL_SetKeyboardFocus(index, data->window);
|
2006-07-27 06:53:23 +00:00
|
|
|
|
|
|
|
if (window->flags & SDL_WINDOW_INPUT_GRABBED) {
|
|
|
|
RECT rect;
|
|
|
|
GetClientRect(hwnd, &rect);
|
|
|
|
ClientToScreen(hwnd, (LPPOINT) & rect);
|
|
|
|
ClientToScreen(hwnd, (LPPOINT) & rect + 1);
|
|
|
|
ClipCursor(&rect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* All done! */
|
|
|
|
window->driverdata = data;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
X11_CreateWindow(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_VideoData *data = (SDL_VideoData *) _this->driverdata;
|
|
|
|
SDL_DisplayData *displaydata =
|
2011-02-10 14:44:25 -08:00
|
|
|
(SDL_DisplayData *) SDL_GetDisplayForWindow(window)->driverdata;
|
2010-07-14 00:28:15 -07:00
|
|
|
Display *display = data->display;
|
|
|
|
int screen = displaydata->screen;
|
2006-07-27 06:53:23 +00:00
|
|
|
Visual *visual;
|
|
|
|
int depth;
|
|
|
|
XSetWindowAttributes xattr;
|
|
|
|
Window w;
|
|
|
|
XSizeHints *sizehints;
|
|
|
|
XWMHints *wmhints;
|
|
|
|
XClassHint *classhints;
|
2010-07-14 00:28:15 -07:00
|
|
|
Atom _NET_WM_WINDOW_TYPE;
|
|
|
|
Atom _NET_WM_WINDOW_TYPE_NORMAL;
|
2010-07-14 00:56:08 -07:00
|
|
|
int wmstate_count;
|
|
|
|
Atom wmstate_atoms[3];
|
2010-07-13 23:11:10 -07:00
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
#if SDL_VIDEO_DRIVER_X11_XINERAMA
|
|
|
|
/* FIXME
|
|
|
|
if ( use_xinerama ) {
|
|
|
|
x = xinerama_info.x_org;
|
|
|
|
y = xinerama_info.y_org;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
#endif
|
2011-01-24 17:38:18 -08:00
|
|
|
#if SDL_VIDEO_OPENGL_GLX
|
2006-07-27 06:53:23 +00:00
|
|
|
if (window->flags & SDL_WINDOW_OPENGL) {
|
2006-07-28 08:43:17 +00:00
|
|
|
XVisualInfo *vinfo;
|
|
|
|
|
2010-07-14 00:28:15 -07:00
|
|
|
vinfo = X11_GL_GetVisual(_this, display, screen);
|
2006-07-28 08:43:17 +00:00
|
|
|
if (!vinfo) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
visual = vinfo->visual;
|
|
|
|
depth = vinfo->depth;
|
|
|
|
XFree(vinfo);
|
|
|
|
} else
|
2009-05-31 11:53:12 +00:00
|
|
|
#endif
|
|
|
|
#ifdef SDL_VIDEO_DRIVER_PANDORA
|
|
|
|
if (window->flags & SDL_WINDOW_OPENGL) {
|
|
|
|
XVisualInfo *vinfo;
|
|
|
|
|
2010-07-14 00:28:15 -07:00
|
|
|
vinfo = X11_GLES_GetVisual(_this, display, screen);
|
2009-05-31 11:53:12 +00:00
|
|
|
if (!vinfo) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
visual = vinfo->visual;
|
|
|
|
depth = vinfo->depth;
|
|
|
|
XFree(vinfo);
|
|
|
|
} else
|
2006-07-28 08:43:17 +00:00
|
|
|
#endif
|
|
|
|
{
|
2006-07-27 06:53:23 +00:00
|
|
|
visual = displaydata->visual;
|
|
|
|
depth = displaydata->depth;
|
|
|
|
}
|
|
|
|
|
2011-02-21 22:52:31 -08:00
|
|
|
xattr.override_redirect = False;
|
2006-07-27 06:53:23 +00:00
|
|
|
xattr.background_pixel = 0;
|
|
|
|
xattr.border_pixel = 0;
|
2011-02-10 11:39:08 -08:00
|
|
|
xattr.colormap = XCreateColormap(display, RootWindow(display, screen), visual, AllocNone);
|
2006-07-27 06:53:23 +00:00
|
|
|
|
2011-02-21 22:52:31 -08:00
|
|
|
w = XCreateWindow(display, RootWindow(display, screen),
|
|
|
|
window->x, window->y, window->w, window->h,
|
|
|
|
0, depth, InputOutput, visual,
|
2006-07-27 06:53:23 +00:00
|
|
|
(CWOverrideRedirect | CWBackPixel | CWBorderPixel |
|
|
|
|
CWColormap), &xattr);
|
2006-07-28 08:43:17 +00:00
|
|
|
if (!w) {
|
|
|
|
SDL_SetError("Couldn't create window");
|
|
|
|
return -1;
|
|
|
|
}
|
2009-05-31 11:53:12 +00:00
|
|
|
#if SDL_VIDEO_DRIVER_PANDORA
|
|
|
|
/* Create the GLES window surface */
|
|
|
|
_this->gles_data->egl_surface =
|
|
|
|
_this->gles_data->eglCreateWindowSurface(_this->gles_data->
|
|
|
|
egl_display,
|
|
|
|
_this->gles_data->egl_config,
|
|
|
|
(NativeWindowType) w, NULL);
|
|
|
|
|
|
|
|
if (_this->gles_data->egl_surface == EGL_NO_SURFACE) {
|
|
|
|
SDL_SetError("Could not create GLES window surface");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
2006-07-27 06:53:23 +00:00
|
|
|
|
|
|
|
sizehints = XAllocSizeHints();
|
|
|
|
if (sizehints) {
|
2011-02-21 22:52:31 -08:00
|
|
|
if (!(window->flags & SDL_WINDOW_RESIZABLE)) {
|
2006-07-27 06:53:23 +00:00
|
|
|
sizehints->min_width = sizehints->max_width = window->w;
|
|
|
|
sizehints->min_height = sizehints->max_height = window->h;
|
2009-08-02 14:27:48 +00:00
|
|
|
sizehints->flags = PMaxSize | PMinSize;
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
2011-02-21 22:52:31 -08:00
|
|
|
sizehints->x = window->x;
|
|
|
|
sizehints->y = window->y;
|
|
|
|
sizehints->flags |= USPosition;
|
2010-07-14 00:28:15 -07:00
|
|
|
XSetWMNormalHints(display, w, sizehints);
|
2006-07-27 06:53:23 +00:00
|
|
|
XFree(sizehints);
|
|
|
|
}
|
|
|
|
|
2011-02-21 22:52:31 -08:00
|
|
|
if (window->flags & SDL_WINDOW_BORDERLESS) {
|
2006-07-27 06:53:23 +00:00
|
|
|
SDL_bool set;
|
|
|
|
Atom WM_HINTS;
|
|
|
|
|
|
|
|
/* We haven't modified the window manager hints yet */
|
|
|
|
set = SDL_FALSE;
|
|
|
|
|
|
|
|
/* First try to set MWM hints */
|
2010-07-14 00:28:15 -07:00
|
|
|
WM_HINTS = XInternAtom(display, "_MOTIF_WM_HINTS", True);
|
2006-07-27 06:53:23 +00:00
|
|
|
if (WM_HINTS != None) {
|
|
|
|
/* Hints used by Motif compliant window managers */
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
unsigned long functions;
|
|
|
|
unsigned long decorations;
|
|
|
|
long input_mode;
|
|
|
|
unsigned long status;
|
|
|
|
} MWMHints = {
|
|
|
|
(1L << 1), 0, 0, 0, 0};
|
|
|
|
|
2010-07-14 00:28:15 -07:00
|
|
|
XChangeProperty(display, w, WM_HINTS, WM_HINTS, 32,
|
2006-07-27 06:53:23 +00:00
|
|
|
PropModeReplace, (unsigned char *) &MWMHints,
|
2010-07-14 00:28:15 -07:00
|
|
|
sizeof(MWMHints) / 4);
|
2006-07-27 06:53:23 +00:00
|
|
|
set = SDL_TRUE;
|
|
|
|
}
|
|
|
|
/* Now try to set KWM hints */
|
2010-07-14 00:28:15 -07:00
|
|
|
WM_HINTS = XInternAtom(display, "KWM_WIN_DECORATION", True);
|
2006-07-27 06:53:23 +00:00
|
|
|
if (WM_HINTS != None) {
|
|
|
|
long KWMHints = 0;
|
|
|
|
|
2010-07-14 00:28:15 -07:00
|
|
|
XChangeProperty(display, w, WM_HINTS, WM_HINTS, 32,
|
2006-07-27 06:53:23 +00:00
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *) &KWMHints,
|
2010-07-14 00:28:15 -07:00
|
|
|
sizeof(KWMHints) / 4);
|
2006-07-27 06:53:23 +00:00
|
|
|
set = SDL_TRUE;
|
|
|
|
}
|
|
|
|
/* Now try to set GNOME hints */
|
2010-07-14 00:28:15 -07:00
|
|
|
WM_HINTS = XInternAtom(display, "_WIN_HINTS", True);
|
2006-07-27 06:53:23 +00:00
|
|
|
if (WM_HINTS != None) {
|
|
|
|
long GNOMEHints = 0;
|
|
|
|
|
2010-07-14 00:28:15 -07:00
|
|
|
XChangeProperty(display, w, WM_HINTS, WM_HINTS, 32,
|
2006-07-27 06:53:23 +00:00
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *) &GNOMEHints,
|
2010-07-14 00:28:15 -07:00
|
|
|
sizeof(GNOMEHints) / 4);
|
2006-07-27 06:53:23 +00:00
|
|
|
set = SDL_TRUE;
|
|
|
|
}
|
|
|
|
/* Finally set the transient hints if necessary */
|
|
|
|
if (!set) {
|
2010-07-14 00:28:15 -07:00
|
|
|
XSetTransientForHint(display, w, RootWindow(display, screen));
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SDL_bool set;
|
|
|
|
Atom WM_HINTS;
|
|
|
|
|
|
|
|
/* We haven't modified the window manager hints yet */
|
|
|
|
set = SDL_FALSE;
|
|
|
|
|
|
|
|
/* First try to unset MWM hints */
|
2010-07-14 00:28:15 -07:00
|
|
|
WM_HINTS = XInternAtom(display, "_MOTIF_WM_HINTS", True);
|
2006-07-27 06:53:23 +00:00
|
|
|
if (WM_HINTS != None) {
|
2010-07-14 00:28:15 -07:00
|
|
|
XDeleteProperty(display, w, WM_HINTS);
|
2006-07-27 06:53:23 +00:00
|
|
|
set = SDL_TRUE;
|
|
|
|
}
|
|
|
|
/* Now try to unset KWM hints */
|
2010-07-14 00:28:15 -07:00
|
|
|
WM_HINTS = XInternAtom(display, "KWM_WIN_DECORATION", True);
|
2006-07-27 06:53:23 +00:00
|
|
|
if (WM_HINTS != None) {
|
2010-07-14 00:28:15 -07:00
|
|
|
XDeleteProperty(display, w, WM_HINTS);
|
2006-07-27 06:53:23 +00:00
|
|
|
set = SDL_TRUE;
|
|
|
|
}
|
|
|
|
/* Now try to unset GNOME hints */
|
2010-07-14 00:28:15 -07:00
|
|
|
WM_HINTS = XInternAtom(display, "_WIN_HINTS", True);
|
2006-07-27 06:53:23 +00:00
|
|
|
if (WM_HINTS != None) {
|
2010-07-14 00:28:15 -07:00
|
|
|
XDeleteProperty(display, w, WM_HINTS);
|
2006-07-27 06:53:23 +00:00
|
|
|
set = SDL_TRUE;
|
|
|
|
}
|
|
|
|
/* Finally unset the transient hints if necessary */
|
|
|
|
if (!set) {
|
2010-07-18 11:37:26 -07:00
|
|
|
XDeleteProperty(display, w, XA_WM_TRANSIENT_FOR);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the input hints so we get keyboard input */
|
|
|
|
wmhints = XAllocWMHints();
|
|
|
|
if (wmhints) {
|
|
|
|
wmhints->input = True;
|
2006-07-29 21:51:00 +00:00
|
|
|
wmhints->flags = InputHint;
|
2010-07-14 00:28:15 -07:00
|
|
|
XSetWMHints(display, w, wmhints);
|
2006-07-27 06:53:23 +00:00
|
|
|
XFree(wmhints);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the class hints so we can get an icon (AfterStep) */
|
|
|
|
classhints = XAllocClassHint();
|
|
|
|
if (classhints != NULL) {
|
|
|
|
classhints->res_name = data->classname;
|
|
|
|
classhints->res_class = data->classname;
|
2010-07-14 00:28:15 -07:00
|
|
|
XSetClassHint(display, w, classhints);
|
2006-07-27 06:53:23 +00:00
|
|
|
XFree(classhints);
|
|
|
|
}
|
|
|
|
|
2010-07-14 00:56:08 -07:00
|
|
|
/* Set the window manager state */
|
|
|
|
wmstate_count = X11_GetWMStateProperty(_this, window, wmstate_atoms);
|
|
|
|
if (wmstate_count > 0) {
|
|
|
|
XChangeProperty(display, w, data->_NET_WM_STATE, XA_ATOM, 32,
|
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *)wmstate_atoms, wmstate_count);
|
|
|
|
} else {
|
|
|
|
XDeleteProperty(display, w, data->_NET_WM_STATE);
|
2010-07-13 23:11:10 -07:00
|
|
|
}
|
|
|
|
|
2010-07-14 00:28:15 -07:00
|
|
|
/* Let the window manager know we're a "normal" window */
|
|
|
|
_NET_WM_WINDOW_TYPE = XInternAtom(display, "_NET_WM_WINDOW_TYPE", False);
|
|
|
|
_NET_WM_WINDOW_TYPE_NORMAL = XInternAtom(display, "_NET_WM_WINDOW_TYPE_NORMAL", False);
|
|
|
|
XChangeProperty(display, w, _NET_WM_WINDOW_TYPE, XA_ATOM, 32,
|
|
|
|
PropModeReplace,
|
|
|
|
(unsigned char *)&_NET_WM_WINDOW_TYPE_NORMAL, 1);
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
/* Allow the window to be deleted by the window manager */
|
2010-07-14 00:28:15 -07:00
|
|
|
XSetWMProtocols(display, w, &data->WM_DELETE_WINDOW, 1);
|
2006-07-27 06:53:23 +00:00
|
|
|
|
|
|
|
if (SetupWindowData(_this, window, w, SDL_TRUE) < 0) {
|
2010-07-14 00:28:15 -07:00
|
|
|
XDestroyWindow(display, w);
|
2006-07-27 06:53:23 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2011-02-21 22:52:31 -08:00
|
|
|
|
2008-03-07 20:54:11 +00:00
|
|
|
#ifdef X_HAVE_UTF8_STRING
|
|
|
|
{
|
|
|
|
Uint32 fevent = 0;
|
|
|
|
pXGetICValues(((SDL_WindowData *) window->driverdata)->ic,
|
|
|
|
XNFilterEvents, &fevent, NULL);
|
2010-07-14 00:28:15 -07:00
|
|
|
XSelectInput(display, w,
|
2008-03-07 20:54:11 +00:00
|
|
|
(FocusChangeMask | EnterWindowMask | LeaveWindowMask |
|
|
|
|
ExposureMask | ButtonPressMask | ButtonReleaseMask |
|
|
|
|
PointerMotionMask | KeyPressMask | KeyReleaseMask |
|
|
|
|
PropertyChangeMask | StructureNotifyMask |
|
|
|
|
KeymapStateMask | fevent));
|
|
|
|
}
|
|
|
|
#else
|
2008-08-25 18:02:14 +00:00
|
|
|
{
|
2010-07-14 00:28:15 -07:00
|
|
|
XSelectInput(display, w,
|
2008-08-26 05:57:41 +00:00
|
|
|
(FocusChangeMask | EnterWindowMask | LeaveWindowMask |
|
|
|
|
ExposureMask | ButtonPressMask | ButtonReleaseMask |
|
|
|
|
PointerMotionMask | KeyPressMask | KeyReleaseMask |
|
|
|
|
PropertyChangeMask | StructureNotifyMask |
|
|
|
|
KeymapStateMask));
|
2008-08-25 18:02:14 +00:00
|
|
|
}
|
2008-03-07 20:54:11 +00:00
|
|
|
#endif
|
|
|
|
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2008-03-07 17:20:37 +00:00
|
|
|
X11_CreateWindowFrom(_THIS, SDL_Window * window, const void *data)
|
2006-07-27 06:53:23 +00:00
|
|
|
{
|
|
|
|
Window w = (Window) data;
|
|
|
|
|
2010-07-13 23:11:10 -07:00
|
|
|
window->title = X11_GetWindowTitle(_this, w);
|
2006-07-27 06:53:23 +00:00
|
|
|
|
|
|
|
if (SetupWindowData(_this, window, w, SDL_FALSE) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-07-13 23:11:10 -07:00
|
|
|
char *
|
|
|
|
X11_GetWindowTitle(_THIS, Window xwindow)
|
|
|
|
{
|
|
|
|
SDL_VideoData *data = (SDL_VideoData *) _this->driverdata;
|
|
|
|
Display *display = data->display;
|
|
|
|
int status, real_format;
|
|
|
|
Atom real_type;
|
|
|
|
unsigned long items_read, items_left;
|
|
|
|
unsigned char *propdata;
|
|
|
|
char *title = NULL;
|
|
|
|
|
|
|
|
status = XGetWindowProperty(display, xwindow, data->_NET_WM_NAME,
|
|
|
|
0L, 8192L, False, data->UTF8_STRING, &real_type, &real_format,
|
|
|
|
&items_read, &items_left, &propdata);
|
|
|
|
if (status == Success) {
|
|
|
|
title = SDL_strdup(SDL_static_cast(char*, propdata));
|
|
|
|
XFree(propdata);
|
|
|
|
} else {
|
|
|
|
status = XGetWindowProperty(display, xwindow, XA_WM_NAME,
|
|
|
|
0L, 8192L, False, XA_STRING, &real_type, &real_format,
|
|
|
|
&items_read, &items_left, &propdata);
|
|
|
|
if (status == Success) {
|
|
|
|
title = SDL_iconv_string("UTF-8", "", SDL_static_cast(char*, propdata), items_read+1);
|
|
|
|
} else {
|
|
|
|
title = SDL_strdup("");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return title;
|
|
|
|
}
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
void
|
|
|
|
X11_SetWindowTitle(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
|
|
|
XTextProperty titleprop, iconprop;
|
|
|
|
Status status;
|
|
|
|
const char *title = window->title;
|
|
|
|
const char *icon = NULL;
|
|
|
|
|
|
|
|
#ifdef X_HAVE_UTF8_STRING
|
2010-07-13 23:11:10 -07:00
|
|
|
Atom _NET_WM_NAME = data->videodata->_NET_WM_NAME;
|
|
|
|
Atom _NET_WM_ICON_NAME = data->videodata->_NET_WM_ICON_NAME;
|
2006-07-27 06:53:23 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
if (title != NULL) {
|
2007-07-04 08:01:04 +00:00
|
|
|
char *title_locale = SDL_iconv_utf8_locale(title);
|
|
|
|
if (!title_locale) {
|
2006-07-27 06:53:23 +00:00
|
|
|
SDL_OutOfMemory();
|
|
|
|
return;
|
|
|
|
}
|
2007-07-04 08:01:04 +00:00
|
|
|
status = XStringListToTextProperty(&title_locale, 1, &titleprop);
|
|
|
|
SDL_free(title_locale);
|
2006-07-27 06:53:23 +00:00
|
|
|
if (status) {
|
2010-01-21 06:21:52 +00:00
|
|
|
XSetTextProperty(display, data->xwindow, &titleprop, XA_WM_NAME);
|
2006-07-27 06:53:23 +00:00
|
|
|
XFree(titleprop.value);
|
|
|
|
}
|
|
|
|
#ifdef X_HAVE_UTF8_STRING
|
|
|
|
if (SDL_X11_HAVE_UTF8) {
|
|
|
|
status =
|
|
|
|
Xutf8TextListToTextProperty(display, (char **) &title, 1,
|
|
|
|
XUTF8StringStyle, &titleprop);
|
|
|
|
if (status == Success) {
|
2010-01-21 06:21:52 +00:00
|
|
|
XSetTextProperty(display, data->xwindow, &titleprop,
|
2006-07-27 06:53:23 +00:00
|
|
|
_NET_WM_NAME);
|
|
|
|
XFree(titleprop.value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
if (icon != NULL) {
|
2007-07-04 08:01:04 +00:00
|
|
|
char *icon_locale = SDL_iconv_utf8_locale(icon);
|
|
|
|
if (!icon_locale) {
|
2006-07-27 06:53:23 +00:00
|
|
|
SDL_OutOfMemory();
|
|
|
|
return;
|
|
|
|
}
|
2007-07-04 08:01:04 +00:00
|
|
|
status = XStringListToTextProperty(&icon_locale, 1, &iconprop);
|
|
|
|
SDL_free(icon_locale);
|
2006-07-27 06:53:23 +00:00
|
|
|
if (status) {
|
2010-01-21 06:21:52 +00:00
|
|
|
XSetTextProperty(display, data->xwindow, &iconprop,
|
2006-07-27 06:53:23 +00:00
|
|
|
XA_WM_ICON_NAME);
|
|
|
|
XFree(iconprop.value);
|
|
|
|
}
|
|
|
|
#ifdef X_HAVE_UTF8_STRING
|
|
|
|
if (SDL_X11_HAVE_UTF8) {
|
|
|
|
status =
|
|
|
|
Xutf8TextListToTextProperty(display, (char **) &icon, 1,
|
|
|
|
XUTF8StringStyle, &iconprop);
|
|
|
|
if (status == Success) {
|
2010-01-21 06:21:52 +00:00
|
|
|
XSetTextProperty(display, data->xwindow, &iconprop,
|
2006-07-27 06:53:23 +00:00
|
|
|
_NET_WM_ICON_NAME);
|
|
|
|
XFree(iconprop.value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
2009-01-02 17:39:48 +00:00
|
|
|
void
|
|
|
|
X11_SetWindowIcon(_THIS, SDL_Window * window, SDL_Surface * icon)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
2010-07-13 23:11:10 -07:00
|
|
|
Atom _NET_WM_ICON = data->videodata->_NET_WM_ICON;
|
2009-01-02 17:39:48 +00:00
|
|
|
|
|
|
|
if (icon) {
|
|
|
|
SDL_PixelFormat format;
|
|
|
|
SDL_Surface *surface;
|
|
|
|
int propsize;
|
2010-07-17 18:44:34 -07:00
|
|
|
long *propdata;
|
2009-01-02 17:39:48 +00:00
|
|
|
|
|
|
|
/* Convert the icon to ARGB for modern window managers */
|
2011-02-13 13:46:10 -08:00
|
|
|
SDL_InitFormat(&format, SDL_PIXELFORMAT_ARGB8888);
|
2009-01-02 17:39:48 +00:00
|
|
|
surface = SDL_ConvertSurface(icon, &format, 0);
|
|
|
|
if (!surface) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the _NET_WM_ICON property */
|
2009-01-04 17:14:27 +00:00
|
|
|
propsize = 2 + (icon->w * icon->h);
|
2010-07-21 21:53:41 -07:00
|
|
|
propdata = SDL_malloc(propsize * sizeof(long));
|
2009-01-02 17:39:48 +00:00
|
|
|
if (propdata) {
|
2010-07-17 18:44:34 -07:00
|
|
|
int x, y;
|
|
|
|
Uint32 *src;
|
|
|
|
long *dst;
|
|
|
|
|
2009-01-02 17:39:48 +00:00
|
|
|
propdata[0] = icon->w;
|
|
|
|
propdata[1] = icon->h;
|
2010-07-17 18:44:34 -07:00
|
|
|
dst = &propdata[2];
|
|
|
|
for (y = 0; y < icon->h; ++y) {
|
|
|
|
src = (Uint32*)((Uint8*)surface->pixels + y * surface->pitch);
|
|
|
|
for (x = 0; x < icon->w; ++x) {
|
|
|
|
*dst++ = *src++;
|
|
|
|
}
|
|
|
|
}
|
2010-01-21 06:21:52 +00:00
|
|
|
XChangeProperty(display, data->xwindow, _NET_WM_ICON, XA_CARDINAL,
|
2009-01-04 17:14:27 +00:00
|
|
|
32, PropModeReplace, (unsigned char *) propdata,
|
|
|
|
propsize);
|
2009-01-02 17:39:48 +00:00
|
|
|
}
|
|
|
|
SDL_FreeSurface(surface);
|
|
|
|
} else {
|
2010-01-21 06:21:52 +00:00
|
|
|
XDeleteProperty(display, data->xwindow, _NET_WM_ICON);
|
2009-01-02 17:39:48 +00:00
|
|
|
}
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2009-01-02 17:39:48 +00:00
|
|
|
}
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
void
|
|
|
|
X11_SetWindowPosition(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
2010-07-13 23:11:10 -07:00
|
|
|
SDL_bool oldstyle_fullscreen;
|
2008-12-17 07:17:54 +00:00
|
|
|
int x, y;
|
2006-07-27 06:53:23 +00:00
|
|
|
|
2010-07-13 23:11:10 -07:00
|
|
|
/* ICCCM2.0-compliant window managers can handle fullscreen windows */
|
2010-07-14 00:56:08 -07:00
|
|
|
oldstyle_fullscreen = X11_IsWindowOldFullscreen(_this, window);
|
2010-07-13 23:11:10 -07:00
|
|
|
|
|
|
|
if (oldstyle_fullscreen
|
2011-02-10 14:44:25 -08:00
|
|
|
|| SDL_WINDOWPOS_ISCENTERED(window->x)) {
|
2008-12-30 17:09:42 +00:00
|
|
|
X11_GetDisplaySize(_this, window, &x, NULL);
|
|
|
|
x = (x - window->w) / 2;
|
2008-12-17 07:17:54 +00:00
|
|
|
} else {
|
|
|
|
x = window->x;
|
|
|
|
}
|
2010-07-13 23:11:10 -07:00
|
|
|
if (oldstyle_fullscreen
|
2011-02-10 14:44:25 -08:00
|
|
|
|| SDL_WINDOWPOS_ISCENTERED(window->y)) {
|
2008-12-30 17:09:42 +00:00
|
|
|
X11_GetDisplaySize(_this, window, NULL, &y);
|
|
|
|
y = (y - window->h) / 2;
|
2008-12-17 07:17:54 +00:00
|
|
|
} else {
|
|
|
|
y = window->y;
|
|
|
|
}
|
2010-01-21 06:21:52 +00:00
|
|
|
XMoveWindow(display, data->xwindow, x, y);
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
X11_SetWindowSize(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
|
|
|
|
2010-12-31 08:50:06 -08:00
|
|
|
if (SDL_IsShapedWindow(window))
|
2010-06-30 16:19:44 -04:00
|
|
|
X11_ResizeWindowShape(window);
|
2010-01-21 06:21:52 +00:00
|
|
|
XResizeWindow(display, data->xwindow, window->w, window->h);
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
X11_ShowWindow(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
|
|
|
|
2010-01-21 06:21:52 +00:00
|
|
|
XMapRaised(display, data->xwindow);
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
X11_HideWindow(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
|
|
|
|
2010-01-21 06:21:52 +00:00
|
|
|
XUnmapWindow(display, data->xwindow);
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
X11_RaiseWindow(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
|
|
|
|
2010-01-21 06:21:52 +00:00
|
|
|
XRaiseWindow(display, data->xwindow);
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
2009-02-19 06:48:25 +00:00
|
|
|
static void
|
2011-02-15 17:23:02 -08:00
|
|
|
SetWindowMaximized(_THIS, SDL_Window * window, SDL_bool maximized)
|
2009-02-19 06:48:25 +00:00
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
SDL_DisplayData *displaydata =
|
2011-02-10 14:44:25 -08:00
|
|
|
(SDL_DisplayData *) SDL_GetDisplayForWindow(window)->driverdata;
|
2009-02-19 06:48:25 +00:00
|
|
|
Display *display = data->videodata->display;
|
2010-07-13 23:11:10 -07:00
|
|
|
Atom _NET_WM_STATE = data->videodata->_NET_WM_STATE;
|
|
|
|
Atom _NET_WM_STATE_MAXIMIZED_VERT = data->videodata->_NET_WM_STATE_MAXIMIZED_VERT;
|
|
|
|
Atom _NET_WM_STATE_MAXIMIZED_HORZ = data->videodata->_NET_WM_STATE_MAXIMIZED_HORZ;
|
2010-07-14 00:56:08 -07:00
|
|
|
Atom _NET_WM_STATE_FULLSCREEN = data->videodata->_NET_WM_STATE_FULLSCREEN;
|
|
|
|
|
|
|
|
if (X11_IsWindowMapped(_this, window)) {
|
|
|
|
XEvent e;
|
|
|
|
|
2010-07-14 07:25:07 -07:00
|
|
|
SDL_zero(e);
|
2010-07-14 00:56:08 -07:00
|
|
|
e.xany.type = ClientMessage;
|
|
|
|
e.xclient.message_type = _NET_WM_STATE;
|
|
|
|
e.xclient.format = 32;
|
|
|
|
e.xclient.window = data->xwindow;
|
|
|
|
e.xclient.data.l[0] =
|
|
|
|
maximized ? _NET_WM_STATE_ADD : _NET_WM_STATE_REMOVE;
|
|
|
|
e.xclient.data.l[1] = _NET_WM_STATE_MAXIMIZED_VERT;
|
|
|
|
e.xclient.data.l[2] = _NET_WM_STATE_MAXIMIZED_HORZ;
|
|
|
|
e.xclient.data.l[3] = 0l;
|
|
|
|
|
|
|
|
XSendEvent(display, RootWindow(display, displaydata->screen), 0,
|
|
|
|
SubstructureNotifyMask | SubstructureRedirectMask, &e);
|
|
|
|
} else {
|
|
|
|
int count = 0;
|
|
|
|
Atom atoms[3];
|
|
|
|
|
|
|
|
if (window->flags & SDL_WINDOW_FULLSCREEN) {
|
|
|
|
atoms[count++] = _NET_WM_STATE_FULLSCREEN;
|
|
|
|
}
|
|
|
|
if (maximized) {
|
|
|
|
atoms[count++] = _NET_WM_STATE_MAXIMIZED_VERT;
|
|
|
|
atoms[count++] = _NET_WM_STATE_MAXIMIZED_HORZ;
|
|
|
|
}
|
|
|
|
if (count > 0) {
|
|
|
|
XChangeProperty(display, data->xwindow, _NET_WM_STATE, XA_ATOM, 32,
|
|
|
|
PropModeReplace, (unsigned char *)atoms, count);
|
|
|
|
} else {
|
|
|
|
XDeleteProperty(display, data->xwindow, _NET_WM_STATE);
|
|
|
|
}
|
|
|
|
}
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2009-02-19 06:48:25 +00:00
|
|
|
}
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
void
|
|
|
|
X11_MaximizeWindow(_THIS, SDL_Window * window)
|
|
|
|
{
|
2011-02-15 17:23:02 -08:00
|
|
|
SetWindowMaximized(_this, window, SDL_TRUE);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
X11_MinimizeWindow(_THIS, SDL_Window * window)
|
|
|
|
{
|
2010-07-13 23:11:10 -07:00
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
SDL_DisplayData *displaydata =
|
2011-02-10 14:44:25 -08:00
|
|
|
(SDL_DisplayData *) SDL_GetDisplayForWindow(window)->driverdata;
|
2010-07-13 23:11:10 -07:00
|
|
|
Display *display = data->videodata->display;
|
|
|
|
|
|
|
|
XIconifyWindow(display, data->xwindow, displaydata->screen);
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
X11_RestoreWindow(_THIS, SDL_Window * window)
|
|
|
|
{
|
2011-02-15 17:23:02 -08:00
|
|
|
SetWindowMaximized(_this, window, SDL_FALSE);
|
2006-07-27 06:53:23 +00:00
|
|
|
X11_ShowWindow(_this, window);
|
|
|
|
}
|
|
|
|
|
2011-02-15 23:07:14 -08:00
|
|
|
void
|
|
|
|
X11_SetWindowFullscreen(_THIS, SDL_Window * window, SDL_VideoDisplay * _display, SDL_bool fullscreen)
|
2011-02-15 17:23:02 -08:00
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
2011-02-15 23:07:14 -08:00
|
|
|
SDL_DisplayData *displaydata = (SDL_DisplayData *) _display->driverdata;
|
2011-02-15 17:23:02 -08:00
|
|
|
Display *display = data->videodata->display;
|
|
|
|
Atom _NET_WM_STATE = data->videodata->_NET_WM_STATE;
|
|
|
|
Atom _NET_WM_STATE_MAXIMIZED_VERT = data->videodata->_NET_WM_STATE_MAXIMIZED_VERT;
|
|
|
|
Atom _NET_WM_STATE_MAXIMIZED_HORZ = data->videodata->_NET_WM_STATE_MAXIMIZED_HORZ;
|
|
|
|
Atom _NET_WM_STATE_FULLSCREEN = data->videodata->_NET_WM_STATE_FULLSCREEN;
|
|
|
|
|
|
|
|
if (X11_IsWindowMapped(_this, window)) {
|
|
|
|
XEvent e;
|
|
|
|
|
|
|
|
SDL_zero(e);
|
|
|
|
e.xany.type = ClientMessage;
|
|
|
|
e.xclient.message_type = _NET_WM_STATE;
|
|
|
|
e.xclient.format = 32;
|
|
|
|
e.xclient.window = data->xwindow;
|
|
|
|
e.xclient.data.l[0] =
|
|
|
|
fullscreen ? _NET_WM_STATE_ADD : _NET_WM_STATE_REMOVE;
|
|
|
|
e.xclient.data.l[1] = _NET_WM_STATE_FULLSCREEN;
|
|
|
|
e.xclient.data.l[3] = 0l;
|
|
|
|
|
|
|
|
XSendEvent(display, RootWindow(display, displaydata->screen), 0,
|
|
|
|
SubstructureNotifyMask | SubstructureRedirectMask, &e);
|
|
|
|
} else {
|
|
|
|
int count = 0;
|
|
|
|
Atom atoms[3];
|
|
|
|
|
|
|
|
if (fullscreen) {
|
|
|
|
atoms[count++] = _NET_WM_STATE_FULLSCREEN;
|
|
|
|
}
|
|
|
|
if (window->flags & SDL_WINDOW_MAXIMIZED) {
|
|
|
|
atoms[count++] = _NET_WM_STATE_MAXIMIZED_VERT;
|
|
|
|
atoms[count++] = _NET_WM_STATE_MAXIMIZED_HORZ;
|
|
|
|
}
|
|
|
|
if (count > 0) {
|
|
|
|
XChangeProperty(display, data->xwindow, _NET_WM_STATE, XA_ATOM, 32,
|
|
|
|
PropModeReplace, (unsigned char *)atoms, count);
|
|
|
|
} else {
|
|
|
|
XDeleteProperty(display, data->xwindow, _NET_WM_STATE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
XFlush(display);
|
|
|
|
}
|
|
|
|
|
2006-07-27 06:53:23 +00:00
|
|
|
void
|
|
|
|
X11_SetWindowGrab(_THIS, SDL_Window * window)
|
|
|
|
{
|
2008-12-17 07:17:54 +00:00
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
2010-07-13 23:11:10 -07:00
|
|
|
SDL_bool oldstyle_fullscreen;
|
|
|
|
|
|
|
|
/* ICCCM2.0-compliant window managers can handle fullscreen windows */
|
2010-07-14 00:56:08 -07:00
|
|
|
oldstyle_fullscreen = X11_IsWindowOldFullscreen(_this, window);
|
2008-12-17 07:17:54 +00:00
|
|
|
|
2010-07-13 23:11:10 -07:00
|
|
|
if (((window->flags & SDL_WINDOW_INPUT_GRABBED) || oldstyle_fullscreen)
|
2008-12-17 07:19:55 +00:00
|
|
|
&& (window->flags & SDL_WINDOW_INPUT_FOCUS)) {
|
2008-12-17 07:17:54 +00:00
|
|
|
/* Try to grab the mouse */
|
2008-12-17 07:19:55 +00:00
|
|
|
for (;;) {
|
|
|
|
int result =
|
2010-01-21 06:21:52 +00:00
|
|
|
XGrabPointer(display, data->xwindow, True, 0, GrabModeAsync,
|
|
|
|
GrabModeAsync, data->xwindow, None, CurrentTime);
|
2008-12-17 07:19:55 +00:00
|
|
|
if (result == GrabSuccess) {
|
2008-12-17 07:17:54 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
SDL_Delay(100);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Raise the window if we grab the mouse */
|
2010-01-21 06:21:52 +00:00
|
|
|
XRaiseWindow(display, data->xwindow);
|
2008-12-17 07:17:54 +00:00
|
|
|
|
|
|
|
/* Now grab the keyboard */
|
2010-01-21 06:21:52 +00:00
|
|
|
XGrabKeyboard(display, data->xwindow, True, GrabModeAsync,
|
2008-12-17 07:19:55 +00:00
|
|
|
GrabModeAsync, CurrentTime);
|
2008-12-17 07:17:54 +00:00
|
|
|
} else {
|
|
|
|
XUngrabPointer(display, CurrentTime);
|
|
|
|
XUngrabKeyboard(display, CurrentTime);
|
|
|
|
}
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
X11_DestroyWindow(_THIS, SDL_Window * window)
|
|
|
|
{
|
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
2008-03-06 23:07:02 +00:00
|
|
|
window->driverdata = NULL;
|
2006-07-27 06:53:23 +00:00
|
|
|
|
|
|
|
if (data) {
|
2008-03-06 23:07:02 +00:00
|
|
|
SDL_VideoData *videodata = (SDL_VideoData *) data->videodata;
|
|
|
|
Display *display = videodata->display;
|
|
|
|
int numwindows = videodata->numwindows;
|
|
|
|
SDL_WindowData **windowlist = videodata->windowlist;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (windowlist) {
|
2008-03-07 17:20:37 +00:00
|
|
|
for (i = 0; i < numwindows; ++i) {
|
2010-01-21 06:21:52 +00:00
|
|
|
if (windowlist[i] && (windowlist[i]->window == window)) {
|
2008-03-07 17:20:37 +00:00
|
|
|
windowlist[i] = windowlist[numwindows - 1];
|
|
|
|
windowlist[numwindows - 1] = NULL;
|
|
|
|
videodata->numwindows--;
|
|
|
|
break;
|
|
|
|
}
|
2008-03-06 23:07:02 +00:00
|
|
|
}
|
|
|
|
}
|
2006-07-27 06:53:23 +00:00
|
|
|
#ifdef X_HAVE_UTF8_STRING
|
|
|
|
if (data->ic) {
|
|
|
|
XDestroyIC(data->ic);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (data->created) {
|
2010-01-21 06:21:52 +00:00
|
|
|
XDestroyWindow(display, data->xwindow);
|
2011-01-19 10:14:11 -08:00
|
|
|
XFlush(display);
|
2006-07-27 06:53:23 +00:00
|
|
|
}
|
|
|
|
SDL_free(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_bool
|
|
|
|
X11_GetWindowWMInfo(_THIS, SDL_Window * window, SDL_SysWMinfo * info)
|
|
|
|
{
|
2010-07-12 01:20:57 -07:00
|
|
|
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
|
|
|
|
Display *display = data->videodata->display;
|
|
|
|
|
|
|
|
if (info->version.major == SDL_MAJOR_VERSION &&
|
|
|
|
info->version.minor == SDL_MINOR_VERSION) {
|
|
|
|
info->subsystem = SDL_SYSWM_X11;
|
2011-01-20 16:05:59 -08:00
|
|
|
info->info.x11.display = display;
|
|
|
|
info->info.x11.window = data->xwindow;
|
2006-07-27 06:53:23 +00:00
|
|
|
return SDL_TRUE;
|
|
|
|
} else {
|
|
|
|
SDL_SetError("Application not compiled with SDL %d.%d\n",
|
|
|
|
SDL_MAJOR_VERSION, SDL_MINOR_VERSION);
|
|
|
|
return SDL_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vi: set ts=4 sw=4 expandtab: */
|