Enable fullscreen switching with resolution mode changing using Xrandr in linux.
This changes the dependency in linux from libxxf86vm-dev to libxrandr-dev. Use Alt-Return to toggle fullscreen mode (as in windows). git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@5065 8ced0084-cf51-0410-be5f-012b33b47a6e
This commit is contained in:
parent
9254a2ddb5
commit
7c76d51c67
|
@ -119,11 +119,91 @@ void UpdateFPSDisplay(const char *text)
|
||||||
OpenGL_SetWindowText(temp);
|
OpenGL_SetWindowText(temp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if defined(HAVE_X11) && HAVE_X11
|
||||||
|
void CreateXWindow (void)
|
||||||
|
{
|
||||||
|
Atom wmDelete;
|
||||||
|
int width, height;
|
||||||
|
|
||||||
|
if (GLWin.fs)
|
||||||
|
{
|
||||||
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
|
if (GLWin.fullSize >= 0)
|
||||||
|
XRRSetScreenConfig(GLWin.dpy, GLWin.screenConfig, RootWindow(GLWin.dpy, GLWin.screen),
|
||||||
|
GLWin.fullSize, GLWin.screenRotation, CurrentTime);
|
||||||
|
#endif
|
||||||
|
GLWin.attr.override_redirect = True;
|
||||||
|
width = GLWin.fullWidth;
|
||||||
|
height = GLWin.fullHeight;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
GLWin.attr.override_redirect = False;
|
||||||
|
width = GLWin.winWidth;
|
||||||
|
height = GLWin.winHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Control window size and picture scaling
|
||||||
|
s_backbuffer_width = width;
|
||||||
|
s_backbuffer_height = height;
|
||||||
|
|
||||||
|
// create the window
|
||||||
|
GLWin.attr.event_mask = ExposureMask | KeyPressMask | KeyReleaseMask |
|
||||||
|
StructureNotifyMask | ResizeRedirectMask;
|
||||||
|
GLWin.win = XCreateWindow(GLWin.dpy, RootWindow(GLWin.dpy, GLWin.vi->screen),
|
||||||
|
0, 0, width, height, 0, GLWin.vi->depth, InputOutput, GLWin.vi->visual,
|
||||||
|
CWBorderPixel | CWColormap | CWEventMask | CWOverrideRedirect, &GLWin.attr);
|
||||||
|
wmDelete = XInternAtom(GLWin.dpy, "WM_DELETE_WINDOW", True);
|
||||||
|
XSetWMProtocols(GLWin.dpy, GLWin.win, &wmDelete, 1);
|
||||||
|
XSetStandardProperties(GLWin.dpy, GLWin.win, "GPU", "GPU", None, NULL, 0, NULL);
|
||||||
|
XMapRaised(GLWin.dpy, GLWin.win);
|
||||||
|
if (GLWin.fs)
|
||||||
|
{
|
||||||
|
XGrabKeyboard(GLWin.dpy, GLWin.win, True, GrabModeAsync, GrabModeAsync, CurrentTime);
|
||||||
|
XGrabPointer(GLWin.dpy, GLWin.win, True, NULL,
|
||||||
|
GrabModeAsync, GrabModeAsync, GLWin.win, None, CurrentTime);
|
||||||
|
}
|
||||||
|
XSync(GLWin.dpy, True);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DestroyXWindow(void)
|
||||||
|
{
|
||||||
|
if( GLWin.ctx )
|
||||||
|
{
|
||||||
|
if( !glXMakeCurrent(GLWin.dpy, None, NULL))
|
||||||
|
{
|
||||||
|
printf("Could not release drawing context.\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* switch back to original desktop resolution if we were in fullscreen */
|
||||||
|
if( GLWin.fs )
|
||||||
|
{
|
||||||
|
XUngrabKeyboard (GLWin.dpy, CurrentTime);
|
||||||
|
XUngrabPointer (GLWin.dpy, CurrentTime);
|
||||||
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
|
if (GLWin.fullSize >= 0)
|
||||||
|
XRRSetScreenConfig(GLWin.dpy, GLWin.screenConfig, RootWindow(GLWin.dpy, GLWin.screen),
|
||||||
|
GLWin.deskSize, GLWin.screenRotation, CurrentTime);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
XUnmapWindow(GLWin.dpy, GLWin.win);
|
||||||
|
XSync(GLWin.dpy, True);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ToggleFullscreenMode (void)
|
||||||
|
{
|
||||||
|
DestroyXWindow();
|
||||||
|
GLWin.fs = !GLWin.fs;
|
||||||
|
CreateXWindow();
|
||||||
|
OpenGL_MakeCurrent();
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
// Create rendering window.
|
// Create rendering window.
|
||||||
// Call browser: Core.cpp:EmuThread() > main.cpp:Video_Initialize()
|
// Call browser: Core.cpp:EmuThread() > main.cpp:Video_Initialize()
|
||||||
bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _iwidth, int _iheight)
|
bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _iwidth, int _iheight)
|
||||||
{
|
{
|
||||||
|
#if !defined(HAVE_X11) || !HAVE_X11
|
||||||
// Check for fullscreen mode
|
// Check for fullscreen mode
|
||||||
int _twidth, _theight;
|
int _twidth, _theight;
|
||||||
if (g_Config.bFullscreen)
|
if (g_Config.bFullscreen)
|
||||||
|
@ -154,6 +234,7 @@ bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _iwidth, int _iheight
|
||||||
// Control window size and picture scaling
|
// Control window size and picture scaling
|
||||||
s_backbuffer_width = _twidth;
|
s_backbuffer_width = _twidth;
|
||||||
s_backbuffer_height = _theight;
|
s_backbuffer_height = _theight;
|
||||||
|
#endif
|
||||||
|
|
||||||
g_VideoInitialize.pPeekMessages = &Callback_PeekMessages;
|
g_VideoInitialize.pPeekMessages = &Callback_PeekMessages;
|
||||||
g_VideoInitialize.pUpdateFPSDisplay = &UpdateFPSDisplay;
|
g_VideoInitialize.pUpdateFPSDisplay = &UpdateFPSDisplay;
|
||||||
|
@ -269,12 +350,9 @@ bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _iwidth, int _iheight
|
||||||
// --------------------------------------
|
// --------------------------------------
|
||||||
|
|
||||||
#elif defined(HAVE_X11) && HAVE_X11
|
#elif defined(HAVE_X11) && HAVE_X11
|
||||||
XVisualInfo *vi;
|
|
||||||
Colormap cmap;
|
Colormap cmap;
|
||||||
int dpyWidth, dpyHeight;
|
|
||||||
int glxMajorVersion, glxMinorVersion;
|
int glxMajorVersion, glxMinorVersion;
|
||||||
int vidModeMajorVersion, vidModeMinorVersion;
|
int vidModeMajorVersion, vidModeMinorVersion;
|
||||||
Atom wmProtocols[3];
|
|
||||||
|
|
||||||
// attributes for a single buffered visual in RGBA format with at least
|
// attributes for a single buffered visual in RGBA format with at least
|
||||||
// 8 bits per color and a 24 bit depth buffer
|
// 8 bits per color and a 24 bit depth buffer
|
||||||
|
@ -292,6 +370,8 @@ bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _iwidth, int _iheight
|
||||||
GLX_BLUE_SIZE, 8,
|
GLX_BLUE_SIZE, 8,
|
||||||
GLX_DEPTH_SIZE, 24,
|
GLX_DEPTH_SIZE, 24,
|
||||||
GLX_SAMPLE_BUFFERS_ARB, g_Config.iMultisampleMode, GLX_SAMPLES_ARB, 1, None };
|
GLX_SAMPLE_BUFFERS_ARB, g_Config.iMultisampleMode, GLX_SAMPLES_ARB, 1, None };
|
||||||
|
|
||||||
|
|
||||||
GLWin.dpy = XOpenDisplay(0);
|
GLWin.dpy = XOpenDisplay(0);
|
||||||
g_VideoInitialize.pWindowHandle = (HWND)GLWin.dpy;
|
g_VideoInitialize.pWindowHandle = (HWND)GLWin.dpy;
|
||||||
GLWin.screen = DefaultScreen(GLWin.dpy);
|
GLWin.screen = DefaultScreen(GLWin.dpy);
|
||||||
|
@ -300,9 +380,9 @@ bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _iwidth, int _iheight
|
||||||
GLWin.fs = g_Config.bFullscreen; //Set to setting in Options
|
GLWin.fs = g_Config.bFullscreen; //Set to setting in Options
|
||||||
|
|
||||||
/* get an appropriate visual */
|
/* get an appropriate visual */
|
||||||
vi = glXChooseVisual(GLWin.dpy, GLWin.screen, attrListDbl);
|
GLWin.vi = glXChooseVisual(GLWin.dpy, GLWin.screen, attrListDbl);
|
||||||
if (vi == NULL) {
|
if (GLWin.vi == NULL) {
|
||||||
vi = glXChooseVisual(GLWin.dpy, GLWin.screen, attrListSgl);
|
GLWin.vi = glXChooseVisual(GLWin.dpy, GLWin.screen, attrListSgl);
|
||||||
GLWin.doubleBuffered = False;
|
GLWin.doubleBuffered = False;
|
||||||
ERROR_LOG(VIDEO, "Only Singlebuffered Visual!");
|
ERROR_LOG(VIDEO, "Only Singlebuffered Visual!");
|
||||||
}
|
}
|
||||||
|
@ -314,102 +394,78 @@ bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _iwidth, int _iheight
|
||||||
glXQueryVersion(GLWin.dpy, &glxMajorVersion, &glxMinorVersion);
|
glXQueryVersion(GLWin.dpy, &glxMajorVersion, &glxMinorVersion);
|
||||||
NOTICE_LOG(VIDEO, "glX-Version %d.%d", glxMajorVersion, glxMinorVersion);
|
NOTICE_LOG(VIDEO, "glX-Version %d.%d", glxMajorVersion, glxMinorVersion);
|
||||||
// Create a GLX context.
|
// Create a GLX context.
|
||||||
GLWin.ctx = glXCreateContext(GLWin.dpy, vi, 0, GL_TRUE);
|
GLWin.ctx = glXCreateContext(GLWin.dpy, GLWin.vi, 0, GL_TRUE);
|
||||||
if(!GLWin.ctx)
|
if(!GLWin.ctx)
|
||||||
{
|
{
|
||||||
PanicAlert("Couldn't Create GLX context.Quit");
|
PanicAlert("Couldn't Create GLX context.Quit");
|
||||||
exit(0); // TODO: Don't bring down entire Emu
|
exit(0); // TODO: Don't bring down entire Emu
|
||||||
}
|
}
|
||||||
// Create a color map.
|
// Create a color map.
|
||||||
cmap = XCreateColormap(GLWin.dpy, RootWindow(GLWin.dpy, vi->screen), vi->visual, AllocNone);
|
cmap = XCreateColormap(GLWin.dpy, RootWindow(GLWin.dpy, GLWin.vi->screen), GLWin.vi->visual, AllocNone);
|
||||||
GLWin.attr.colormap = cmap;
|
GLWin.attr.colormap = cmap;
|
||||||
GLWin.attr.border_pixel = 0;
|
GLWin.attr.border_pixel = 0;
|
||||||
XkbSetDetectableAutoRepeat(GLWin.dpy, True, NULL);
|
XkbSetDetectableAutoRepeat(GLWin.dpy, True, NULL);
|
||||||
|
|
||||||
#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
// Get the resolution setings for both fullscreen and windowed modes
|
||||||
// get a connection
|
if (strlen(g_Config.cFSResolution) > 1)
|
||||||
XF86VidModeQueryVersion(GLWin.dpy, &vidModeMajorVersion, &vidModeMinorVersion);
|
sscanf(g_Config.cFSResolution, "%dx%d", &GLWin.fullWidth, &GLWin.fullHeight);
|
||||||
|
else // No full screen reso set, fall back to desktop resolution
|
||||||
if (GLWin.fs) {
|
{
|
||||||
|
GLWin.fullWidth = DisplayWidth(GLWin.dpy, GLWin.screen);
|
||||||
XF86VidModeModeInfo **modes = NULL;
|
GLWin.fullHeight = DisplayHeight(GLWin.dpy, GLWin.screen);
|
||||||
int modeNum = 0;
|
|
||||||
int bestMode = 0;
|
|
||||||
|
|
||||||
// set best mode to current
|
|
||||||
bestMode = 0;
|
|
||||||
NOTICE_LOG(VIDEO, "XF86VidModeExtension-Version %d.%d", vidModeMajorVersion, vidModeMinorVersion);
|
|
||||||
XF86VidModeGetAllModeLines(GLWin.dpy, GLWin.screen, &modeNum, &modes);
|
|
||||||
|
|
||||||
if (modeNum > 0 && modes != NULL) {
|
|
||||||
/* save desktop-resolution before switching modes */
|
|
||||||
GLWin.deskMode = *modes[0];
|
|
||||||
/* look for mode with requested resolution */
|
|
||||||
for (int i = 0; i < modeNum; i++) {
|
|
||||||
if ((modes[i]->hdisplay == _twidth) && (modes[i]->vdisplay == _theight)) {
|
|
||||||
bestMode = i;
|
|
||||||
}
|
}
|
||||||
|
if (strlen(g_Config.cInternalRes) > 1)
|
||||||
|
sscanf(g_Config.cInternalRes, "%dx%d", &GLWin.winWidth, &GLWin.winHeight);
|
||||||
|
else // No Window resolution set, fall back to default
|
||||||
|
{
|
||||||
|
GLWin.winWidth = _iwidth;
|
||||||
|
GLWin.winHeight = _iheight;
|
||||||
}
|
}
|
||||||
|
|
||||||
XF86VidModeSwitchToMode(GLWin.dpy, GLWin.screen, modes[bestMode]);
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
XF86VidModeSetViewPort(GLWin.dpy, GLWin.screen, 0, 0);
|
XRRQueryVersion(GLWin.dpy, &vidModeMajorVersion, &vidModeMinorVersion);
|
||||||
dpyWidth = modes[bestMode]->hdisplay;
|
XRRScreenSize *sizes;
|
||||||
dpyHeight = modes[bestMode]->vdisplay;
|
int numSizes;
|
||||||
NOTICE_LOG(VIDEO, "Resolution %dx%d", dpyWidth, dpyHeight);
|
|
||||||
XFree(modes);
|
|
||||||
|
|
||||||
/* create a fullscreen window */
|
NOTICE_LOG(VIDEO, "XRRExtension-Version %d.%d", vidModeMajorVersion, vidModeMinorVersion);
|
||||||
GLWin.attr.override_redirect = True;
|
|
||||||
GLWin.attr.event_mask = ExposureMask | KeyPressMask | KeyReleaseMask | StructureNotifyMask;
|
GLWin.screenConfig = XRRGetScreenInfo(GLWin.dpy, RootWindow(GLWin.dpy, GLWin.screen));
|
||||||
GLWin.win = XCreateWindow(GLWin.dpy, RootWindow(GLWin.dpy, vi->screen),
|
|
||||||
0, 0, dpyWidth, dpyHeight, 0, vi->depth, InputOutput, vi->visual,
|
/* save desktop resolution */
|
||||||
CWBorderPixel | CWColormap | CWEventMask | CWOverrideRedirect,
|
GLWin.deskSize = XRRConfigCurrentConfiguration(GLWin.screenConfig, &GLWin.screenRotation);
|
||||||
&GLWin.attr);
|
/* Set the desktop resolution as the default */
|
||||||
XWarpPointer(GLWin.dpy, None, GLWin.win, 0, 0, 0, 0, 0, 0);
|
GLWin.fullSize = -1;
|
||||||
XMapRaised(GLWin.dpy, GLWin.win);
|
|
||||||
XGrabKeyboard(GLWin.dpy, GLWin.win, True, GrabModeAsync, GrabModeAsync, CurrentTime);
|
/* Find the index of the fullscreen resolution from config */
|
||||||
XGrabPointer(GLWin.dpy, GLWin.win, True, NULL,
|
sizes = XRRConfigSizes(GLWin.screenConfig, &numSizes);
|
||||||
GrabModeAsync, GrabModeAsync, GLWin.win, None, CurrentTime);
|
if (numSizes > 0 && sizes != NULL) {
|
||||||
|
for (int i = 0; i < numSizes; i++) {
|
||||||
|
if ((sizes[i].width == GLWin.fullWidth) && (sizes[i].height == GLWin.fullHeight)) {
|
||||||
|
GLWin.fullSize = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
NOTICE_LOG(VIDEO, "Fullscreen Resolution %dx%d", sizes[GLWin.fullSize].width, sizes[GLWin.fullSize].height);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
ERROR_LOG(VIDEO, "Failed to start fullscreen. If you received the "
|
ERROR_LOG(VIDEO, "Failed to obtain fullscreen sizes.\n"
|
||||||
"\"XFree86-VidModeExtension\" extension is missing, add\n"
|
"Using current desktop resolution for fullscreen.\n");
|
||||||
"Load \"extmod\"\n"
|
GLWin.fullWidth = DisplayWidth(GLWin.dpy, GLWin.screen);
|
||||||
"to your X configuration file (under the Module Section)\n");
|
GLWin.fullHeight = DisplayHeight(GLWin.dpy, GLWin.screen);
|
||||||
GLWin.fs = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
GLWin.fullWidth = DisplayWidth(GLWin.dpy, GLWin.screen);
|
||||||
|
GLWin.fullHeight = DisplayHeight(GLWin.dpy, GLWin.screen);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (!GLWin.fs) {
|
CreateXWindow();
|
||||||
|
|
||||||
//XRootWindow(dpy,screen)
|
|
||||||
//int X = (rcdesktop.right-rcdesktop.left)/2 - (rc.right-rc.left)/2;
|
|
||||||
//int Y = (rcdesktop.bottom-rcdesktop.top)/2 - (rc.bottom-rc.top)/2;
|
|
||||||
|
|
||||||
// create a window in window mode
|
|
||||||
GLWin.attr.event_mask = ExposureMask | KeyPressMask | KeyReleaseMask |
|
|
||||||
StructureNotifyMask | ResizeRedirectMask;
|
|
||||||
GLWin.win = XCreateWindow(GLWin.dpy, RootWindow(GLWin.dpy, vi->screen),
|
|
||||||
0, 0, _twidth, _theight, 0, vi->depth, InputOutput, vi->visual,
|
|
||||||
CWBorderPixel | CWColormap | CWEventMask, &GLWin.attr);
|
|
||||||
// only set window title and handle WM_PROTOCOLS if in windowed mode
|
|
||||||
wmProtocols[0] = XInternAtom(GLWin.dpy, "WM_DELETE_WINDOW", True);
|
|
||||||
wmProtocols[1] = XInternAtom(GLWin.dpy, "_NET_WM_STATE", False);
|
|
||||||
wmProtocols[2] = XInternAtom(GLWin.dpy, "_NET_WM_STATE_FULLSCREEN", False);
|
|
||||||
XSetWMProtocols(GLWin.dpy, GLWin.win, wmProtocols, 3);
|
|
||||||
XSetStandardProperties(GLWin.dpy, GLWin.win, "GPU",
|
|
||||||
"GPU", None, NULL, 0, NULL);
|
|
||||||
XMapRaised(GLWin.dpy, GLWin.win);
|
|
||||||
}
|
|
||||||
g_VideoInitialize.pXWindow = (Window *) &GLWin.win;
|
g_VideoInitialize.pXWindow = (Window *) &GLWin.win;
|
||||||
|
|
||||||
if (g_Config.bHideCursor)
|
if (g_Config.bHideCursor)
|
||||||
{
|
{
|
||||||
// make a blank cursor
|
// make a blank cursor
|
||||||
Pixmap Blank;
|
Pixmap Blank;
|
||||||
XColor DummyColor;
|
XColor DummyColor;
|
||||||
char ZeroData[1] = {0};
|
char ZeroData[1] = {0};
|
||||||
Cursor MouseCursor;
|
|
||||||
Blank = XCreateBitmapFromData (GLWin.dpy, GLWin.win, ZeroData, 1, 1);
|
Blank = XCreateBitmapFromData (GLWin.dpy, GLWin.win, ZeroData, 1, 1);
|
||||||
GLWin.blankCursor = XCreatePixmapCursor(GLWin.dpy, Blank, Blank, &DummyColor, &DummyColor, 0, 0);
|
GLWin.blankCursor = XCreatePixmapCursor(GLWin.dpy, Blank, Blank, &DummyColor, &DummyColor, 0, 0);
|
||||||
XFreePixmap (GLWin.dpy, Blank);
|
XFreePixmap (GLWin.dpy, Blank);
|
||||||
|
@ -418,37 +474,6 @@ bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _iwidth, int _iheight
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(HAVE_X11) && HAVE_X11
|
|
||||||
void X11_EWMH_Fullscreen(int action)
|
|
||||||
{
|
|
||||||
assert(action == _NET_WM_STATE_REMOVE ||
|
|
||||||
action == _NET_WM_STATE_ADD || action == _NET_WM_STATE_TOGGLE);
|
|
||||||
|
|
||||||
XEvent xev;
|
|
||||||
|
|
||||||
// Init X event structure for _NET_WM_STATE_FULLSCREEN client message
|
|
||||||
xev.xclient.type = ClientMessage;
|
|
||||||
xev.xclient.serial = 0;
|
|
||||||
xev.xclient.send_event = True;
|
|
||||||
xev.xclient.message_type = XInternAtom(GLWin.dpy, "_NET_WM_STATE", False);
|
|
||||||
xev.xclient.window = GLWin.win;
|
|
||||||
xev.xclient.format = 32;
|
|
||||||
xev.xclient.data.l[0] = action;
|
|
||||||
xev.xclient.data.l[1] = XInternAtom(GLWin.dpy, "_NET_WM_STATE_FULLSCREEN", False);
|
|
||||||
xev.xclient.data.l[2] = 0;
|
|
||||||
xev.xclient.data.l[3] = 0;
|
|
||||||
xev.xclient.data.l[4] = 0;
|
|
||||||
|
|
||||||
// Send the event
|
|
||||||
if (!XSendEvent(GLWin.dpy, DefaultRootWindow(GLWin.dpy), False,
|
|
||||||
SubstructureRedirectMask | SubstructureNotifyMask,
|
|
||||||
&xev))
|
|
||||||
{
|
|
||||||
ERROR_LOG(VIDEO, "Failed to switch fullscreen/windowed mode.\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
bool OpenGL_MakeCurrent()
|
bool OpenGL_MakeCurrent()
|
||||||
{
|
{
|
||||||
#if defined(USE_WX) && USE_WX
|
#if defined(USE_WX) && USE_WX
|
||||||
|
@ -561,17 +586,8 @@ void OpenGL_Update()
|
||||||
else
|
else
|
||||||
FKeyPressed = key - 0xff4e;
|
FKeyPressed = key - 0xff4e;
|
||||||
}
|
}
|
||||||
else if (key == XK_Escape)
|
else if (key == XK_Return && ((event.xkey.state & Mod1Mask) == Mod1Mask))
|
||||||
{
|
ToggleFullscreenMode();
|
||||||
if (!GLWin.fs)
|
|
||||||
{
|
|
||||||
X11_EWMH_Fullscreen(_NET_WM_STATE_TOGGLE);
|
|
||||||
XWithdrawWindow(GLWin.dpy,GLWin.win,GLWin.screen);
|
|
||||||
XMapRaised(GLWin.dpy,GLWin.win);
|
|
||||||
XRaiseWindow(GLWin.dpy,GLWin.win);
|
|
||||||
XFlush(GLWin.dpy);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
else {
|
||||||
if(key == XK_Shift_L || key == XK_Shift_R)
|
if(key == XK_Shift_L || key == XK_Shift_R)
|
||||||
ShiftPressed = true;
|
ShiftPressed = true;
|
||||||
|
@ -587,7 +603,7 @@ void OpenGL_Update()
|
||||||
XDefineCursor(GLWin.dpy, GLWin.win, GLWin.blankCursor);
|
XDefineCursor(GLWin.dpy, GLWin.win, GLWin.blankCursor);
|
||||||
break;
|
break;
|
||||||
case FocusOut:
|
case FocusOut:
|
||||||
if (g_Config.bHideCursor)
|
if (g_Config.bHideCursor && !GLWin.fs)
|
||||||
XUndefineCursor(GLWin.dpy, GLWin.win);
|
XUndefineCursor(GLWin.dpy, GLWin.win);
|
||||||
break;
|
break;
|
||||||
case ConfigureNotify:
|
case ConfigureNotify:
|
||||||
|
@ -597,6 +613,12 @@ void OpenGL_Update()
|
||||||
&GLWin.width, &GLWin.height, &borderDummy, &GLWin.depth);
|
&GLWin.width, &GLWin.height, &borderDummy, &GLWin.depth);
|
||||||
s_backbuffer_width = GLWin.width;
|
s_backbuffer_width = GLWin.width;
|
||||||
s_backbuffer_height = GLWin.height;
|
s_backbuffer_height = GLWin.height;
|
||||||
|
// Save windowed mode size for return from fullscreen
|
||||||
|
if (!GLWin.fs)
|
||||||
|
{
|
||||||
|
GLWin.winWidth = GLWin.width;
|
||||||
|
GLWin.winHeight = GLWin.height;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case ClientMessage:
|
case ClientMessage:
|
||||||
if ((ulong) event.xclient.data.l[0] == XInternAtom(GLWin.dpy, "WM_DELETE_WINDOW", False))
|
if ((ulong) event.xclient.data.l[0] == XInternAtom(GLWin.dpy, "WM_DELETE_WINDOW", False))
|
||||||
|
@ -650,23 +672,18 @@ void OpenGL_Shutdown()
|
||||||
hDC = NULL; // Set DC To NULL
|
hDC = NULL; // Set DC To NULL
|
||||||
}
|
}
|
||||||
#elif defined(HAVE_X11) && HAVE_X11
|
#elif defined(HAVE_X11) && HAVE_X11
|
||||||
#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
DestroyXWindow();
|
||||||
/* switch back to original desktop resolution if we were in fs */
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
if ((GLWin.dpy != NULL) && GLWin.fs) {
|
if (GLWin.fullSize >= 0)
|
||||||
XUngrabKeyboard (GLWin.dpy, CurrentTime);
|
XRRFreeScreenConfigInfo(GLWin.screenConfig);
|
||||||
XUngrabPointer (GLWin.dpy, CurrentTime);
|
|
||||||
XF86VidModeSwitchToMode(GLWin.dpy, GLWin.screen, &GLWin.deskMode);
|
|
||||||
XF86VidModeSetViewPort(GLWin.dpy, GLWin.screen, 0, 0);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
if (g_Config.bHideCursor) XUndefineCursor(GLWin.dpy, GLWin.win);
|
if (g_Config.bHideCursor)
|
||||||
|
{
|
||||||
|
XUndefineCursor(GLWin.dpy, GLWin.win);
|
||||||
|
XFreeCursor(GLWin.dpy, GLWin.blankCursor);
|
||||||
|
}
|
||||||
if (GLWin.ctx)
|
if (GLWin.ctx)
|
||||||
{
|
{
|
||||||
if (!glXMakeCurrent(GLWin.dpy, None, NULL))
|
|
||||||
{
|
|
||||||
ERROR_LOG(VIDEO, "Could not release drawing context.\n");
|
|
||||||
}
|
|
||||||
XUnmapWindow(GLWin.dpy, GLWin.win);
|
|
||||||
glXDestroyContext(GLWin.dpy, GLWin.ctx);
|
glXDestroyContext(GLWin.dpy, GLWin.ctx);
|
||||||
XCloseDisplay(GLWin.dpy);
|
XCloseDisplay(GLWin.dpy);
|
||||||
GLWin.ctx = NULL;
|
GLWin.ctx = NULL;
|
||||||
|
|
|
@ -70,15 +70,9 @@
|
||||||
#include <X11/Xutil.h>
|
#include <X11/Xutil.h>
|
||||||
#include <X11/keysym.h>
|
#include <X11/keysym.h>
|
||||||
|
|
||||||
// EWMH state actions, see
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
// http://freedesktop.org/wiki/Specifications/wm-spec?action=show&redirect=Standards%2Fwm-spec
|
#include <X11/extensions/Xrandr.h>
|
||||||
#define _NET_WM_STATE_REMOVE 0 /* remove/unset property */
|
#endif // XRANDR
|
||||||
#define _NET_WM_STATE_ADD 1 /* add/set property */
|
|
||||||
#define _NET_WM_STATE_TOGGLE 2 /* toggle property */
|
|
||||||
|
|
||||||
#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
|
||||||
#include <X11/extensions/xf86vmode.h>
|
|
||||||
#endif // XXF86VM
|
|
||||||
#endif // X11
|
#endif // X11
|
||||||
|
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
@ -92,14 +86,19 @@ typedef struct {
|
||||||
#elif defined(HAVE_X11) && HAVE_X11
|
#elif defined(HAVE_X11) && HAVE_X11
|
||||||
Window win;
|
Window win;
|
||||||
Display *dpy;
|
Display *dpy;
|
||||||
|
XVisualInfo *vi;
|
||||||
GLXContext ctx;
|
GLXContext ctx;
|
||||||
Cursor blankCursor;
|
Cursor blankCursor;
|
||||||
XSetWindowAttributes attr;
|
XSetWindowAttributes attr;
|
||||||
Bool fs;
|
Bool fs;
|
||||||
Bool doubleBuffered;
|
Bool doubleBuffered;
|
||||||
#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
int fullWidth, fullHeight;
|
||||||
XF86VidModeModeInfo deskMode;
|
int winWidth, winHeight;
|
||||||
#endif // XXF86VM
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
|
XRRScreenConfiguration *screenConfig;
|
||||||
|
Rotation screenRotation;
|
||||||
|
int deskSize, fullSize;
|
||||||
|
#endif // XRANDR
|
||||||
#endif // X11
|
#endif // X11
|
||||||
#if defined(USE_WX) && USE_WX
|
#if defined(USE_WX) && USE_WX
|
||||||
wxGLCanvas *glCanvas;
|
wxGLCanvas *glCanvas;
|
||||||
|
|
|
@ -100,11 +100,10 @@ if sys.platform == 'win32':
|
||||||
gfxenv['CPPPATH'] += libs
|
gfxenv['CPPPATH'] += libs
|
||||||
|
|
||||||
|
|
||||||
# check for xxf86vm
|
# check for Xrandr
|
||||||
|
gfxenv['HAVE_XRANDR'] = gfxenv['HAVE_X11'] and conf.CheckPKG('xrandr')
|
||||||
|
|
||||||
gfxenv['HAVE_XXF86VM'] = gfxenv['HAVE_X11'] and conf.CheckPKG('xxf86vm')
|
conf.Define('HAVE_XRANDR', gfxenv['HAVE_XRANDR'])
|
||||||
|
|
||||||
conf.Define('HAVE_XXF86VM', gfxenv['HAVE_XXF86VM'])
|
|
||||||
|
|
||||||
conf.Finish()
|
conf.Finish()
|
||||||
|
|
||||||
|
|
|
@ -207,45 +207,30 @@ void Win32AddResolutions() {
|
||||||
ZeroMemory(&dmi, sizeof(dmi));
|
ZeroMemory(&dmi, sizeof(dmi));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#elif defined(HAVE_X11) && HAVE_X11 && defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
#elif defined(HAVE_X11) && HAVE_X11 && defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
void X11AddResolutions() {
|
void X11AddResolutions() {
|
||||||
int glxMajorVersion, glxMinorVersion;
|
|
||||||
int vidModeMajorVersion, vidModeMinorVersion;
|
|
||||||
GLWin.dpy = XOpenDisplay(0);
|
GLWin.dpy = XOpenDisplay(0);
|
||||||
glXQueryVersion(GLWin.dpy, &glxMajorVersion, &glxMinorVersion);
|
|
||||||
XF86VidModeQueryVersion(GLWin.dpy, &vidModeMajorVersion, &vidModeMinorVersion);
|
|
||||||
//Get all full screen resos for the config dialog
|
//Get all full screen resos for the config dialog
|
||||||
XF86VidModeModeInfo **modes = NULL;
|
XRRScreenSize *sizes = NULL;
|
||||||
int modeNum = 0;
|
int modeNum = 0;
|
||||||
int bestMode = 0;
|
|
||||||
|
|
||||||
//set best mode to current
|
sizes = XRRSizes(GLWin.dpy, GLWin.screen, &modeNum);
|
||||||
bestMode = 0;
|
if (modeNum > 0 && sizes != NULL)
|
||||||
XF86VidModeGetAllModeLines(GLWin.dpy, GLWin.screen, &modeNum, &modes);
|
|
||||||
int px = 0, py = 0;
|
|
||||||
if (modeNum > 0 && modes != NULL)
|
|
||||||
{
|
{
|
||||||
for (int i = 0; i < modeNum; i++)
|
for (int i = 0; i < modeNum; i++)
|
||||||
{
|
|
||||||
if (px != modes[i]->hdisplay && py != modes[i]->vdisplay)
|
|
||||||
{
|
{
|
||||||
char temp[32];
|
char temp[32];
|
||||||
sprintf(temp,"%dx%d", modes[i]->hdisplay, modes[i]->vdisplay);
|
sprintf(temp,"%dx%d", sizes[i].width, sizes[i].height);
|
||||||
|
|
||||||
#if defined(HAVE_WX) && HAVE_WX
|
#if defined(HAVE_WX) && HAVE_WX
|
||||||
m_ConfigFrame->AddFSReso(temp);
|
m_ConfigFrame->AddFSReso(temp);
|
||||||
m_ConfigFrame->AddWindowReso(temp);//Add same to Window ones,
|
//Add same to window resolutions, since
|
||||||
//since they should be
|
//they should be nearly all that's needed
|
||||||
//nearly all that's needed
|
m_ConfigFrame->AddWindowReso(temp);
|
||||||
#endif
|
#endif
|
||||||
px = modes[i]->hdisplay;//Used to remove repeating from
|
|
||||||
//different screen depths
|
|
||||||
py = modes[i]->vdisplay;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
XFree(modes);
|
|
||||||
}
|
|
||||||
#elif defined(HAVE_COCOA) && HAVE_COCOA
|
#elif defined(HAVE_COCOA) && HAVE_COCOA
|
||||||
void CocoaAddResolutions() {
|
void CocoaAddResolutions() {
|
||||||
|
|
||||||
|
@ -304,7 +289,7 @@ void DllConfig(HWND _hParent)
|
||||||
|
|
||||||
#if defined(_WIN32)
|
#if defined(_WIN32)
|
||||||
Win32AddResolutions();
|
Win32AddResolutions();
|
||||||
#elif defined(HAVE_X11) && HAVE_X11 && defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
#elif defined(HAVE_X11) && HAVE_X11 && defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
X11AddResolutions();
|
X11AddResolutions();
|
||||||
#elif defined(HAVE_COCOA) && HAVE_COCOA
|
#elif defined(HAVE_COCOA) && HAVE_COCOA
|
||||||
CocoaAddResolutions();
|
CocoaAddResolutions();
|
||||||
|
|
|
@ -349,37 +349,39 @@ bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _twidth, int _theight
|
||||||
GLWin.attr.border_pixel = 0;
|
GLWin.attr.border_pixel = 0;
|
||||||
XkbSetDetectableAutoRepeat(GLWin.dpy, True, NULL);
|
XkbSetDetectableAutoRepeat(GLWin.dpy, True, NULL);
|
||||||
|
|
||||||
#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
// get a connection
|
// get a connection
|
||||||
XF86VidModeQueryVersion(GLWin.dpy, &vidModeMajorVersion, &vidModeMinorVersion);
|
XRRQueryVersion(GLWin.dpy, &vidModeMajorVersion, &vidModeMinorVersion);
|
||||||
|
|
||||||
if (GLWin.fs) {
|
if (GLWin.fs) {
|
||||||
|
|
||||||
XF86VidModeModeInfo **modes = NULL;
|
XRRScreenSize *sizes;
|
||||||
int modeNum = 0;
|
int numSizes;
|
||||||
int bestMode = 0;
|
int bestSize;
|
||||||
|
|
||||||
// set best mode to current
|
NOTICE_LOG(VIDEO, "XRRExtension-Version %d.%d", vidModeMajorVersion, vidModeMinorVersion);
|
||||||
bestMode = 0;
|
|
||||||
NOTICE_LOG(VIDEO, "XF86VidModeExtension-Version %d.%d", vidModeMajorVersion, vidModeMinorVersion);
|
GLWin.screenConfig = XRRGetScreenInfo(GLWin.dpy, RootWindow(GLWin.dpy, GLWin.screen));
|
||||||
XF86VidModeGetAllModeLines(GLWin.dpy, GLWin.screen, &modeNum, &modes);
|
|
||||||
|
|
||||||
if (modeNum > 0 && modes != NULL) {
|
|
||||||
/* save desktop-resolution before switching modes */
|
/* save desktop-resolution before switching modes */
|
||||||
GLWin.deskMode = *modes[0];
|
GLWin.deskSize = XRRConfigCurrentConfiguration(GLWin.screenConfig, &GLWin.screenRotation);
|
||||||
|
bestSize = GLWin.deskSize;
|
||||||
|
|
||||||
|
sizes = XRRConfigSizes(GLWin.screenConfig, &numSizes);
|
||||||
|
if (numSizes > 0 && sizes != NULL) {
|
||||||
/* look for mode with requested resolution */
|
/* look for mode with requested resolution */
|
||||||
for (int i = 0; i < modeNum; i++) {
|
for (int i = 0; i < numSizes; i++) {
|
||||||
if ((modes[i]->hdisplay == _twidth) && (modes[i]->vdisplay == _theight)) {
|
if ((sizes[i].width == _twidth) && (sizes[i].height == _theight)) {
|
||||||
bestMode = i;
|
bestSize = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
XF86VidModeSwitchToMode(GLWin.dpy, GLWin.screen, modes[bestMode]);
|
XRRSetScreenConfig(GLWin.dpy, GLWin.screenConfig, RootWindow(GLWin.dpy, GLWin.screen),
|
||||||
XF86VidModeSetViewPort(GLWin.dpy, GLWin.screen, 0, 0);
|
bestSize, GLWin.screenRotation, CurrentTime);
|
||||||
dpyWidth = modes[bestMode]->hdisplay;
|
|
||||||
dpyHeight = modes[bestMode]->vdisplay;
|
dpyWidth = sizes[bestSize].width;
|
||||||
|
dpyHeight = sizes[bestSize].height;
|
||||||
NOTICE_LOG(VIDEO, "Resolution %dx%d", dpyWidth, dpyHeight);
|
NOTICE_LOG(VIDEO, "Resolution %dx%d", dpyWidth, dpyHeight);
|
||||||
XFree(modes);
|
|
||||||
|
|
||||||
/* create a fullscreen window */
|
/* create a fullscreen window */
|
||||||
GLWin.attr.override_redirect = True;
|
GLWin.attr.override_redirect = True;
|
||||||
|
@ -606,6 +608,16 @@ void OpenGL_Shutdown()
|
||||||
hDC = NULL; // Set DC To NULL
|
hDC = NULL; // Set DC To NULL
|
||||||
}
|
}
|
||||||
#elif defined(HAVE_X11) && HAVE_X11
|
#elif defined(HAVE_X11) && HAVE_X11
|
||||||
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
|
/* switch back to original desktop resolution if we were in fs */
|
||||||
|
if ((GLWin.dpy != NULL) && GLWin.fs) {
|
||||||
|
XUngrabKeyboard (GLWin.dpy, CurrentTime);
|
||||||
|
XUngrabPointer (GLWin.dpy, CurrentTime);
|
||||||
|
XRRSetScreenConfig(GLWin.dpy, GLWin.screenConfig, RootWindow(GLWin.dpy, GLWin.screen),
|
||||||
|
GLWin.deskSize, GLWin.screenRotation, CurrentTime);
|
||||||
|
XRRFreeScreenConfigInfo(GLWin.screenConfig);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
if (GLWin.ctx)
|
if (GLWin.ctx)
|
||||||
{
|
{
|
||||||
if (!glXMakeCurrent(GLWin.dpy, None, NULL))
|
if (!glXMakeCurrent(GLWin.dpy, None, NULL))
|
||||||
|
@ -617,15 +629,6 @@ void OpenGL_Shutdown()
|
||||||
XCloseDisplay(GLWin.dpy);
|
XCloseDisplay(GLWin.dpy);
|
||||||
GLWin.ctx = NULL;
|
GLWin.ctx = NULL;
|
||||||
}
|
}
|
||||||
#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
|
||||||
/* switch back to original desktop resolution if we were in fs */
|
|
||||||
if (GLWin.dpy != NULL) {
|
|
||||||
if (GLWin.fs) {
|
|
||||||
XF86VidModeSwitchToMode(GLWin.dpy, GLWin.screen, &GLWin.deskMode);
|
|
||||||
XF86VidModeSetViewPort(GLWin.dpy, GLWin.screen, 0, 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -75,9 +75,9 @@
|
||||||
#include <X11/Xlib.h>
|
#include <X11/Xlib.h>
|
||||||
#include <X11/Xutil.h>
|
#include <X11/Xutil.h>
|
||||||
#include <X11/keysym.h>
|
#include <X11/keysym.h>
|
||||||
#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
#include <X11/extensions/xf86vmode.h>
|
#include <X11/extensions/Xrandr.h>
|
||||||
#endif // XXF86VM
|
#endif // XRANDR
|
||||||
#endif // X11
|
#endif // X11
|
||||||
|
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
@ -95,9 +95,11 @@ typedef struct {
|
||||||
XSetWindowAttributes attr;
|
XSetWindowAttributes attr;
|
||||||
Bool fs;
|
Bool fs;
|
||||||
Bool doubleBuffered;
|
Bool doubleBuffered;
|
||||||
#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
XF86VidModeModeInfo deskMode;
|
XRRScreenConfiguration *screenConfig;
|
||||||
#endif // XXF86VM
|
Rotation screenRotation;
|
||||||
|
int deskSize;
|
||||||
|
#endif // XRANDR
|
||||||
#endif // X11
|
#endif // X11
|
||||||
#if defined(USE_WX) && USE_WX
|
#if defined(USE_WX) && USE_WX
|
||||||
wxGLCanvas *glCanvas;
|
wxGLCanvas *glCanvas;
|
||||||
|
|
|
@ -88,11 +88,10 @@ if sys.platform == 'win32':
|
||||||
gfxenv['CPPPATH'] += libs
|
gfxenv['CPPPATH'] += libs
|
||||||
|
|
||||||
|
|
||||||
# check for xxf86vm
|
# check for Xrandr
|
||||||
|
gfxenv['HAVE_XRANDR'] = gfxenv['HAVE_X11'] and conf.CheckPKG('xrandr')
|
||||||
|
|
||||||
gfxenv['HAVE_XXF86VM'] = gfxenv['HAVE_X11'] and conf.CheckPKG('xxf86vm')
|
conf.Define('HAVE_XRANDR', gfxenv['HAVE_XRANDR'])
|
||||||
|
|
||||||
conf.Define('HAVE_XXF86VM', gfxenv['HAVE_XXF86VM'])
|
|
||||||
|
|
||||||
conf.Finish()
|
conf.Finish()
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue