// Copyright (C) 2002-2012 Nikolaus Gebhardt // Copyright (C) 2014-2015 Dawid Gan // This file is part of the "Irrlicht Engine". // For conditions of distribution and use, see copyright notice in irrlicht.h extern bool GLContextDebugBit; #include "CIrrDeviceLinux.h" #ifdef _IRR_COMPILE_WITH_X11_DEVICE_ #include #include #include #include #include #include #include "IEventReceiver.h" #include "ISceneManager.h" #include "IGUIEnvironment.h" #include "os.h" #include "CTimer.h" #include "irrString.h" #include "Keycodes.h" #include "COSOperator.h" #include "CColorConverter.h" #include "SIrrCreationParameters.h" #include "IGUISpriteBank.h" #ifdef _IRR_COMPILE_WITH_X11_ #include #include #endif #ifdef _IRR_COMPILE_WITH_OPENGL_ #include #ifdef _IRR_OPENGL_USE_EXTPOINTER_ #define GLX_GLXEXT_PROTOTYPES #include "glxext.h" #endif #endif #ifdef _IRR_LINUX_XCURSOR_ #include #endif #if defined _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ #include #include #ifdef __FreeBSD__ #include #elif defined(__linux__) // linux/joystick.h includes linux/input.h, which #defines values for various KEY_FOO keys. // These override the irr::KEY_FOO equivalents, which stops key handling from working. // As a workaround, defining _INPUT_H stops linux/input.h from being included; it // doesn't actually seem to be necessary except to pull in sys/ioctl.h. #define _INPUT_H #include // Would normally be included in linux/input.h #include #undef _INPUT_H #endif #endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ #define XRANDR_ROTATION_LEFT (1 << 1) #define XRANDR_ROTATION_RIGHT (1 << 3) namespace irr { namespace video { extern bool useCoreContext; IVideoDriver* createOpenGLDriver(const SIrrlichtCreationParameters& params, io::IFileSystem* io, CIrrDeviceLinux* device); IVideoDriver* createOGLES2Driver(const SIrrlichtCreationParameters& params, video::SExposedVideoData& data, io::IFileSystem* io); } } // end namespace irr #if defined(_IRR_COMPILE_WITH_X11_) namespace { Atom X_ATOM_CLIPBOARD; Atom X_ATOM_TARGETS; Atom X_ATOM_UTF8_STRING; }; #endif namespace irr { const char* wmDeleteWindow = "WM_DELETE_WINDOW"; //! constructor CIrrDeviceLinux::CIrrDeviceLinux(const SIrrlichtCreationParameters& param) : CIrrDeviceStub(param), #ifdef _IRR_COMPILE_WITH_X11_ display(0), visual(0), screennr(0), window(0), StdHints(0), SoftwareImage(0), XInputMethod(0), XInputContext(0), m_font_set(0), numlock_mask(0), m_ime_enabled(false), #ifdef _IRR_COMPILE_WITH_OPENGL_ glxWin(0), Context(0), #endif #endif Width(param.WindowSize.Width), Height(param.WindowSize.Height), WindowHasFocus(false), WindowMinimized(false), UseXVidMode(false), UseXRandR(false), UseGLXWindow(false), ExternalWindow(false), AutorepeatSupport(0), SupportsNetWM(false), NeedsGrabPointer(false) { #ifdef _DEBUG setDebugName("CIrrDeviceLinux"); #endif // print version, distribution etc. // thx to LynxLuna for pointing me to the uname function core::stringc linuxversion; struct utsname LinuxInfo; uname(&LinuxInfo); linuxversion += LinuxInfo.sysname; linuxversion += " "; linuxversion += LinuxInfo.release; linuxversion += " "; linuxversion += LinuxInfo.version; linuxversion += " "; linuxversion += LinuxInfo.machine; Operator = new COSOperator(linuxversion, this); os::Printer::log(linuxversion.c_str(), ELL_INFORMATION); // create keymap createKeyMap(); // create window if (CreationParams.DriverType != video::EDT_NULL) { // create the window, only if we do not use the null device if (!createWindow()) return; } // create cursor control CursorControl = new CCursorControl(this, CreationParams.DriverType == video::EDT_NULL); // create driver createDriver(); if (!VideoDriver) return; #ifdef _IRR_COMPILE_WITH_X11_ m_ime_position.x = 0; m_ime_position.y = 0; createInputContext(); numlock_mask = getNumlockMask(display); // Get maximum 16 characters from input method m_ime_char_holder.reallocate(16); #endif createGUIAndScene(); } //! destructor CIrrDeviceLinux::~CIrrDeviceLinux() { #ifdef _IRR_COMPILE_WITH_X11_ if (StdHints) XFree(StdHints); // Disable cursor (it is drop'ed in stub) if (CursorControl) { CursorControl->setVisible(false); static_cast(CursorControl)->clearCursors(); } // Must free OpenGL textures etc before destroying context, so can't wait for stub destructor if ( GUIEnvironment ) { GUIEnvironment->drop(); GUIEnvironment = NULL; } if ( SceneManager ) { SceneManager->drop(); SceneManager = NULL; } if ( VideoDriver ) { VideoDriver->drop(); VideoDriver = NULL; } destroyInputContext(); if (display) { #ifdef _IRR_COMPILE_WITH_OPENGL_ if (Context) { if (glxWin) { if (!glXMakeContextCurrent(display, None, None, NULL)) os::Printer::log("Could not release glx context.", ELL_WARNING); } else { if (!glXMakeCurrent(display, None, NULL)) os::Printer::log("Could not release glx context.", ELL_WARNING); } glXDestroyContext(display, Context); if (glxWin) glXDestroyWindow(display, glxWin); } #endif // #ifdef _IRR_COMPILE_WITH_OPENGL_ if (SoftwareImage) XDestroyImage(SoftwareImage); if (!ExternalWindow) { XDestroyWindow(display,window); } // Reset fullscreen resolution change restoreResolution(); if (!ExternalWindow) { XCloseDisplay(display); } } if (visual) XFree(visual); #endif // #ifdef _IRR_COMPILE_WITH_X11_ #if defined(_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) for (u32 joystick = 0; joystick < ActiveJoysticks.size(); ++joystick) { if (ActiveJoysticks[joystick].fd >= 0) { close(ActiveJoysticks[joystick].fd); } } #endif } #if defined(_IRR_COMPILE_WITH_X11_) static bool XErrorSignaled = false; int IrrPrintXError(Display *display, XErrorEvent *event) { char msg[256]; char msg2[256]; XErrorSignaled = true; snprintf(msg, 256, "%d", event->request_code); XGetErrorDatabaseText(display, "XRequest", msg, "unknown", msg2, 256); XGetErrorText(display, event->error_code, msg, 256); os::Printer::log("X Error", msg, ELL_WARNING); os::Printer::log("From call ", msg2, ELL_WARNING); return 0; } #endif bool CIrrDeviceLinux::restoreResolution() { if (!CreationParams.Fullscreen) return true; #ifdef _IRR_LINUX_X11_VIDMODE_ if (UseXVidMode && CreationParams.Fullscreen) { XF86VidModeSwitchToMode(display, screennr, &oldVideoMode); XF86VidModeSetViewPort(display, screennr, 0, 0); } #endif #ifdef _IRR_LINUX_X11_RANDR_ if (UseXRandR && CreationParams.Fullscreen && old_mode != 0) { XRRScreenResources* res = XRRGetScreenResources(display, DefaultRootWindow(display)); if (!res) return false; XRROutputInfo* output = XRRGetOutputInfo(display, res, output_id); if (!output || !output->crtc || output->connection == RR_Disconnected) { XRRFreeScreenResources(res); XRRFreeOutputInfo(output); return false; } XRRCrtcInfo* crtc = XRRGetCrtcInfo(display, res, output->crtc); if (!crtc) { XRRFreeScreenResources(res); XRRFreeOutputInfo(output); return false; } Status s = XRRSetCrtcConfig(display, res, output->crtc, CurrentTime, crtc->x, crtc->y, old_mode, crtc->rotation, &output_id, 1); XRRFreeOutputInfo(output); XRRFreeCrtcInfo(crtc); XRRFreeScreenResources(res); if (s != Success) return false; } #endif return true; } bool CIrrDeviceLinux::changeResolution() { if (!CreationParams.Fullscreen) return true; getVideoModeList(); #if defined(_IRR_LINUX_X11_VIDMODE_) || defined(_IRR_LINUX_X11_RANDR_) s32 eventbase, errorbase; s32 bestMode = -1; #endif #ifdef _IRR_LINUX_X11_VIDMODE_ if (XF86VidModeQueryExtension(display, &eventbase, &errorbase)) { // enumerate video modes s32 modeCount; XF86VidModeModeInfo** modes; float refresh_rate, refresh_rate_old; XF86VidModeGetAllModeLines(display, screennr, &modeCount, &modes); // find fitting mode for (s32 i = 0; ihdisplay >= Width && modes[i]->vdisplay >= Height) { bestMode = i; } else if (bestMode!=-1 && modes[i]->hdisplay == modes[bestMode]->hdisplay && modes[i]->vdisplay == modes[bestMode]->vdisplay) { refresh_rate_old = (modes[bestMode]->dotclock * 1000.0) / (modes[bestMode]->htotal * modes[bestMode]->vtotal); refresh_rate = (modes[i]->dotclock * 1000.0) / (modes[i]->htotal * modes[i]->vtotal); if (refresh_rate > refresh_rate_old) { bestMode = i; } } else if (bestMode!=-1 && modes[i]->hdisplay >= Width && modes[i]->vdisplay >= Height && modes[i]->hdisplay <= modes[bestMode]->hdisplay && modes[i]->vdisplay <= modes[bestMode]->vdisplay) { bestMode = i; } } if (bestMode != -1) { os::Printer::log("Starting vidmode fullscreen mode...", ELL_INFORMATION); os::Printer::log("hdisplay: ", core::stringc(modes[bestMode]->hdisplay).c_str(), ELL_INFORMATION); os::Printer::log("vdisplay: ", core::stringc(modes[bestMode]->vdisplay).c_str(), ELL_INFORMATION); XF86VidModeSwitchToMode(display, screennr, modes[bestMode]); XF86VidModeSetViewPort(display, screennr, 0, 0); UseXVidMode=true; } else { os::Printer::log("Could not find specified video mode, running windowed.", ELL_WARNING); CreationParams.Fullscreen = false; } XFree(modes); } #endif #ifdef _IRR_LINUX_X11_RANDR_ while (XRRQueryExtension(display, &eventbase, &errorbase)) { if (output_id == 0) break; XRRScreenResources* res = XRRGetScreenResources(display, DefaultRootWindow(display)); if (!res) break; XRROutputInfo* output = XRRGetOutputInfo(display, res, output_id); if (!output || !output->crtc || output->connection == RR_Disconnected) { XRRFreeOutputInfo(output); XRRFreeScreenResources(res); break; } XRRCrtcInfo* crtc = XRRGetCrtcInfo(display, res, output->crtc); if (!crtc) { XRRFreeOutputInfo(output); XRRFreeScreenResources(res); break; } float refresh_rate, refresh_rate_new; core::dimension2d mode0_size = core::dimension2d(0, 0); for (int i = 0; i < res->nmode; i++) { const XRRModeInfo* mode = &res->modes[i]; core::dimension2d size; if (crtc->rotation & (XRANDR_ROTATION_LEFT|XRANDR_ROTATION_RIGHT)) { size = core::dimension2d(mode->height, mode->width); } else { size = core::dimension2d(mode->width, mode->height); } if (bestMode == -1 && mode->id == output->modes[0]) { mode0_size = size; } if (bestMode == -1 && size.Width == Width && size.Height == Height) { for (int j = 0; j < output->nmode; j++) { if (mode->id == output->modes[j]) { bestMode = j; refresh_rate = (mode->dotClock * 1000.0) / (mode->hTotal * mode->vTotal); break; } } } else if (bestMode != -1 && size.Width == Width && size.Height == Height) { refresh_rate_new = (mode->dotClock * 1000.0) / (mode->hTotal * mode->vTotal); if (refresh_rate_new <= refresh_rate) break; for (int j = 0; j < output->nmode; j++) { if (mode->id == output->modes[j]) { bestMode = j; refresh_rate = refresh_rate_new; break; } } } } // If video mode not found, try to use first available if (bestMode == -1) { bestMode = 0; Width = mode0_size.Width; Height = mode0_size.Height; } Status s = XRRSetCrtcConfig(display, res, output->crtc, CurrentTime, crtc->x, crtc->y, output->modes[bestMode], crtc->rotation, &output_id, 1); if (s == Success) UseXRandR = true; if (UseXRandR && SupportsNetWM) { XRRPanning* panning = XRRGetPanning(display, res, output->crtc); if ((panning->width != Width && panning->width != 0) || (panning->height != Height && panning->height != 0)) { NeedsGrabPointer = true; } XRRFreePanning(panning); } XRRFreeCrtcInfo(crtc); XRRFreeOutputInfo(output); XRRFreeScreenResources(res); break; } if (UseXRandR == false) { os::Printer::log("Could not get video output. Try to run in windowed mode.", ELL_WARNING); CreationParams.Fullscreen = false; } #endif return CreationParams.Fullscreen; } #if defined(_IRR_COMPILE_WITH_X11_) void IrrPrintXGrabError(int grabResult, const c8 * grabCommand ) { if ( grabResult == GrabSuccess ) { // os::Printer::log(grabCommand, ": GrabSuccess", ELL_INFORMATION); return; } switch ( grabResult ) { case AlreadyGrabbed: os::Printer::log(grabCommand, ": AlreadyGrabbed", ELL_WARNING); break; case GrabNotViewable: os::Printer::log(grabCommand, ": GrabNotViewable", ELL_WARNING); break; case GrabFrozen: os::Printer::log(grabCommand, ": GrabFrozen", ELL_WARNING); break; case GrabInvalidTime: os::Printer::log(grabCommand, ": GrabInvalidTime", ELL_WARNING); break; default: os::Printer::log(grabCommand, ": grab failed with unknown problem", ELL_WARNING); break; } } #endif #ifdef _IRR_COMPILE_WITH_OPENGL_ #ifdef _IRR_COMPILE_WITH_X11_ static GLXContext getMeAGLContext(Display *display, GLXFBConfig glxFBConfig, bool force_legacy_context) { GLXContext Context; irr::video::useCoreContext = true; int core43ctxdebug[] = { GLX_CONTEXT_MAJOR_VERSION_ARB, 4, GLX_CONTEXT_MINOR_VERSION_ARB, 3, GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB, GLX_CONTEXT_FLAGS_ARB, GLX_CONTEXT_DEBUG_BIT_ARB, None }; int core43ctx[] = { GLX_CONTEXT_MAJOR_VERSION_ARB, 4, GLX_CONTEXT_MINOR_VERSION_ARB, 3, GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB, None }; int core33ctxdebug[] = { GLX_CONTEXT_MAJOR_VERSION_ARB, 3, GLX_CONTEXT_MINOR_VERSION_ARB, 3, GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB, GLX_CONTEXT_FLAGS_ARB, GLX_CONTEXT_DEBUG_BIT_ARB, None }; int core33ctx[] = { GLX_CONTEXT_MAJOR_VERSION_ARB, 3, GLX_CONTEXT_MINOR_VERSION_ARB, 3, GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB, None }; int core31ctxdebug[] = { GLX_CONTEXT_MAJOR_VERSION_ARB, 3, GLX_CONTEXT_MINOR_VERSION_ARB, 1, GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB, GLX_CONTEXT_FLAGS_ARB, GLX_CONTEXT_DEBUG_BIT_ARB, None }; int core31ctx[] = { GLX_CONTEXT_MAJOR_VERSION_ARB, 3, GLX_CONTEXT_MINOR_VERSION_ARB, 1, GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB, None }; PFNGLXCREATECONTEXTATTRIBSARBPROC glXCreateContextAttribsARB = 0; glXCreateContextAttribsARB = (PFNGLXCREATECONTEXTATTRIBSARBPROC) glXGetProcAddressARB( (const GLubyte *) "glXCreateContextAttribsARB" ); if(!force_legacy_context) { // create core 4.3 context os::Printer::log("Creating OpenGL 4.3 context...", ELL_INFORMATION); Context = glXCreateContextAttribsARB(display, glxFBConfig, 0, True, GLContextDebugBit ? core43ctxdebug : core43ctx); if (!XErrorSignaled) return Context; XErrorSignaled = false; // create core 3.3 context os::Printer::log("Creating OpenGL 3.3 context...", ELL_INFORMATION); Context = glXCreateContextAttribsARB(display, glxFBConfig, 0, True, GLContextDebugBit ? core33ctxdebug : core33ctx); if (!XErrorSignaled) return Context; XErrorSignaled = false; // create core 3.1 context (for older mesa) os::Printer::log("Creating OpenGL 3.1 context...", ELL_INFORMATION); Context = glXCreateContextAttribsARB(display, glxFBConfig, 0, True, GLContextDebugBit ? core31ctxdebug : core31ctx); if (!XErrorSignaled) return Context; } // if(force_legacy_context) XErrorSignaled = false; irr::video::useCoreContext = false; // fall back to legacy context os::Printer::log("Creating legacy OpenGL 2.1 context...", ELL_INFORMATION); Context = glXCreateNewContext(display, glxFBConfig, GLX_RGBA_TYPE, NULL, True); return Context; } #endif #endif void CIrrDeviceLinux::grabPointer(bool grab) { #ifdef _IRR_COMPILE_WITH_X11_ if (grab) { int result = 0; for (int i = 0; i < 500; i++) { const unsigned int mask = ButtonPressMask | ButtonReleaseMask | PointerMotionMask | FocusChangeMask; int result = XGrabPointer(display, window, True, mask, GrabModeAsync, GrabModeAsync, window, None, CurrentTime); if (result == GrabSuccess) break; usleep(1000); } if (result != GrabSuccess) { os::Printer::log("Couldn't grab pointer.", ELL_WARNING); } } else { XUngrabPointer(display, CurrentTime); } #endif } bool CIrrDeviceLinux::createWindow() { #ifdef _IRR_COMPILE_WITH_X11_ os::Printer::log("Creating X window...", ELL_INFORMATION); XSetErrorHandler(IrrPrintXError); display = XOpenDisplay(0); if (!display) { os::Printer::log("Error: Need running XServer to start Irrlicht Engine.", ELL_ERROR); if (XDisplayName(0)[0]) os::Printer::log("Could not open display", XDisplayName(0), ELL_ERROR); else os::Printer::log("Could not open display, set DISPLAY variable", ELL_ERROR); return false; } screennr = DefaultScreen(display); Atom *list; Atom type; int form; unsigned long remain, len; Atom WMCheck = XInternAtom(display, "_NET_SUPPORTING_WM_CHECK", false); Status s = XGetWindowProperty(display, DefaultRootWindow(display), WMCheck, 0L, 1L, False, XA_WINDOW, &type, &form, &len, &remain, (unsigned char **)&list); if (s == Success) { XFree(list); SupportsNetWM = (len > 0); } changeResolution(); #ifdef _IRR_COMPILE_WITH_OPENGL_ GLXFBConfig glxFBConfig = NULL; int major, minor; bool isAvailableGLX=false; if (CreationParams.DriverType==video::EDT_OPENGL) { isAvailableGLX=glXQueryExtension(display,&major,&minor); if (isAvailableGLX && glXQueryVersion(display, &major, &minor)) { #ifdef GLX_VERSION_1_3 typedef GLXFBConfig * ( * PFNGLXCHOOSEFBCONFIGPROC) (Display *dpy, int screen, const int *attrib_list, int *nelements); #ifdef _IRR_OPENGL_USE_EXTPOINTER_ PFNGLXCHOOSEFBCONFIGPROC glxChooseFBConfig = (PFNGLXCHOOSEFBCONFIGPROC)glXGetProcAddress(reinterpret_cast("glXChooseFBConfig")); #else PFNGLXCHOOSEFBCONFIGPROC glxChooseFBConfig=glXChooseFBConfig; #endif if (major==1 && minor>2 && glxChooseFBConfig) { // attribute array for the draw buffer int visualAttrBuffer[] = { GLX_RENDER_TYPE, GLX_RGBA_BIT, GLX_RED_SIZE, 4, GLX_GREEN_SIZE, 4, GLX_BLUE_SIZE, 4, GLX_ALPHA_SIZE, CreationParams.WithAlphaChannel?1:0, GLX_DEPTH_SIZE, CreationParams.ZBufferBits, //10,11 GLX_DOUBLEBUFFER, CreationParams.Doublebuffer?True:False, GLX_STENCIL_SIZE, CreationParams.Stencilbuffer?1:0, #if defined(GLX_VERSION_1_4) && defined(GLX_SAMPLE_BUFFERS) // we need to check the extension string! GLX_SAMPLE_BUFFERS, 1, GLX_SAMPLES, CreationParams.AntiAlias, // 18,19 #elif defined(GLX_ARB_multisample) GLX_SAMPLE_BUFFERS_ARB, 1, GLX_SAMPLES_ARB, CreationParams.AntiAlias, // 18,19 #elif defined(GLX_SGIS_multisample) GLX_SAMPLE_BUFFERS_SGIS, 1, GLX_SAMPLES_SGIS, CreationParams.AntiAlias, // 18,19 #else #error #endif #ifdef GLX_ARB_framebuffer_sRGB GLX_FRAMEBUFFER_SRGB_CAPABLE_ARB, CreationParams.HandleSRGB, #elif defined(GLX_EXT_framebuffer_sRGB) GLX_FRAMEBUFFER_SRGB_CAPABLE_EXT, CreationParams.HandleSRGB, #else #error #endif GLX_STEREO, CreationParams.Stereobuffer?True:False, None }; GLXFBConfig *configList=0; int nitems=0; if (CreationParams.AntiAlias<2) { visualAttrBuffer[17] = 0; visualAttrBuffer[19] = 0; } if (CreationParams.HandleSRGB == false) { visualAttrBuffer[21] = GLX_DONT_CARE; } // first round with unchanged values { configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); if (!configList && CreationParams.AntiAlias) { while (!configList && (visualAttrBuffer[19]>1)) { visualAttrBuffer[19] -= 1; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); } if (!configList) { visualAttrBuffer[17] = 0; visualAttrBuffer[19] = 0; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); if (configList) { os::Printer::log("No FSAA available.", ELL_WARNING); CreationParams.AntiAlias=0; } else { //reenable multisampling visualAttrBuffer[17] = 1; visualAttrBuffer[19] = CreationParams.AntiAlias; } } } } // Try to disable sRGB framebuffer if (!configList && CreationParams.HandleSRGB) { os::Printer::log("No sRGB framebuffer available.", ELL_WARNING); CreationParams.HandleSRGB=false; visualAttrBuffer[21] = GLX_DONT_CARE; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); if (!configList && CreationParams.AntiAlias) { while (!configList && (visualAttrBuffer[19]>1)) { visualAttrBuffer[19] -= 1; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); } if (!configList) { visualAttrBuffer[17] = 0; visualAttrBuffer[19] = 0; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); if (configList) { os::Printer::log("No FSAA available.", ELL_WARNING); CreationParams.AntiAlias=0; } else { //reenable multisampling visualAttrBuffer[17] = 1; visualAttrBuffer[19] = CreationParams.AntiAlias; } } } } // Next try with flipped stencil buffer value // If the first round was with stencil flag it's now without // Other way round also makes sense because some configs // only have depth buffer combined with stencil buffer if (!configList) { if (CreationParams.Stencilbuffer) os::Printer::log("No stencilbuffer available, disabling stencil shadows.", ELL_WARNING); CreationParams.Stencilbuffer = !CreationParams.Stencilbuffer; visualAttrBuffer[15]=CreationParams.Stencilbuffer?1:0; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); if (!configList && CreationParams.AntiAlias) { while (!configList && (visualAttrBuffer[19]>1)) { visualAttrBuffer[19] -= 1; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); } if (!configList) { visualAttrBuffer[17] = 0; visualAttrBuffer[19] = 0; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); if (configList) { os::Printer::log("No FSAA available.", ELL_WARNING); CreationParams.AntiAlias=0; } else { //reenable multisampling visualAttrBuffer[17] = 1; visualAttrBuffer[19] = CreationParams.AntiAlias; } } } } // Next try without double buffer if (!configList && CreationParams.Doublebuffer) { os::Printer::log("No doublebuffering available.", ELL_WARNING); CreationParams.Doublebuffer=false; visualAttrBuffer[13] = GLX_DONT_CARE; CreationParams.Stencilbuffer = false; visualAttrBuffer[15]=0; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); if (!configList && CreationParams.AntiAlias) { while (!configList && (visualAttrBuffer[19]>1)) { visualAttrBuffer[19] -= 1; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); } if (!configList) { visualAttrBuffer[17] = 0; visualAttrBuffer[19] = 0; configList=glxChooseFBConfig(display, screennr, visualAttrBuffer,&nitems); if (configList) { os::Printer::log("No FSAA available.", ELL_WARNING); CreationParams.AntiAlias=0; } else { //reenable multisampling visualAttrBuffer[17] = 1; visualAttrBuffer[19] = CreationParams.AntiAlias; } } } } if (configList) { glxFBConfig=configList[0]; XFree(configList); UseGLXWindow=true; #ifdef _IRR_OPENGL_USE_EXTPOINTER_ typedef XVisualInfo * ( * PFNGLXGETVISUALFROMFBCONFIGPROC) (Display *dpy, GLXFBConfig config); PFNGLXGETVISUALFROMFBCONFIGPROC glxGetVisualFromFBConfig= (PFNGLXGETVISUALFROMFBCONFIGPROC)glXGetProcAddress(reinterpret_cast("glXGetVisualFromFBConfig")); if (glxGetVisualFromFBConfig) visual = glxGetVisualFromFBConfig(display,glxFBConfig); #else visual = glXGetVisualFromFBConfig(display,glxFBConfig); #endif } } else #endif { // attribute array for the draw buffer int visualAttrBuffer[] = { GLX_RGBA, GLX_USE_GL, GLX_RED_SIZE, 4, GLX_GREEN_SIZE, 4, GLX_BLUE_SIZE, 4, GLX_ALPHA_SIZE, CreationParams.WithAlphaChannel?1:0, GLX_DEPTH_SIZE, CreationParams.ZBufferBits, GLX_STENCIL_SIZE, CreationParams.Stencilbuffer?1:0, // 12,13 // The following attributes have no flags, but are // either present or not. As a no-op we use // GLX_USE_GL, which is silently ignored by glXChooseVisual CreationParams.Doublebuffer?GLX_DOUBLEBUFFER:GLX_USE_GL, // 14 CreationParams.Stereobuffer?GLX_STEREO:GLX_USE_GL, // 15 #ifdef GLX_ARB_framebuffer_sRGB CreationParams.HandleSRGB?GLX_FRAMEBUFFER_SRGB_CAPABLE_ARB:GLX_USE_GL, #elif defined(GLX_EXT_framebuffer_sRGB) CreationParams.HandleSRGB?GLX_FRAMEBUFFER_SRGB_CAPABLE_EXT:GLX_USE_GL, #endif None }; visual=glXChooseVisual(display, screennr, visualAttrBuffer); if (!visual) { if (CreationParams.Stencilbuffer) os::Printer::log("No stencilbuffer available, disabling.", ELL_WARNING); CreationParams.Stencilbuffer = !CreationParams.Stencilbuffer; visualAttrBuffer[13]=CreationParams.Stencilbuffer?1:0; visual=glXChooseVisual(display, screennr, visualAttrBuffer); if (!visual && CreationParams.Doublebuffer) { os::Printer::log("No doublebuffering available.", ELL_WARNING); CreationParams.Doublebuffer=false; visualAttrBuffer[14] = GLX_USE_GL; visual=glXChooseVisual(display, screennr, visualAttrBuffer); } } } } else os::Printer::log("No GLX support available. OpenGL driver will not work.", ELL_WARNING); } // don't use the XVisual with OpenGL, because it ignores all requested // properties of the CreationParams else if (!visual) #endif // _IRR_COMPILE_WITH_OPENGL_ // create visual with standard X methods { os::Printer::log("Using plain X visual"); XVisualInfo visTempl; //Template to hold requested values int visNumber; // Return value of available visuals visTempl.screen = screennr; // ARGB visuals should be avoided for usual applications visTempl.depth = CreationParams.WithAlphaChannel?32:24; while ((!visual) && (visTempl.depth>=16)) { visual = XGetVisualInfo(display, VisualScreenMask|VisualDepthMask, &visTempl, &visNumber); visTempl.depth -= 8; } if (!visual && !CreationParams.WithAlphaChannel) { visTempl.depth = 32; visual = XGetVisualInfo(display, VisualScreenMask|VisualDepthMask, &visTempl, &visNumber); } } if (!visual) { os::Printer::log("Fatal error, could not get visual.", ELL_ERROR); XCloseDisplay(display); display=0; return false; } #ifdef _DEBUG else os::Printer::log("Visual chosen: ", core::stringc(static_cast(visual->visualid)).c_str(), ELL_DEBUG); #endif // create color map Colormap colormap; colormap = XCreateColormap(display, RootWindow(display, visual->screen), visual->visual, AllocNone); attributes.colormap = colormap; attributes.border_pixel = 0; attributes.event_mask = StructureNotifyMask | FocusChangeMask | ExposureMask; if (!CreationParams.IgnoreInput) attributes.event_mask |= PointerMotionMask | ButtonPressMask | KeyPressMask | ButtonReleaseMask | KeyReleaseMask; if (!CreationParams.WindowId) { attributes.override_redirect = !SupportsNetWM && CreationParams.Fullscreen; // create new Window window = XCreateWindow(display, RootWindow(display, visual->screen), 0, 0, Width, Height, 0, visual->depth, InputOutput, visual->visual, CWBorderPixel | CWColormap | CWEventMask | CWOverrideRedirect, &attributes); XMapRaised(display, window); CreationParams.WindowId = (void*)window; Atom wmDelete; wmDelete = XInternAtom(display, wmDeleteWindow, True); XSetWMProtocols(display, window, &wmDelete, 1); if (CreationParams.Fullscreen) { // Some window managers don't respect values from XCreateWindow and // place window in random position. This may cause that fullscreen // window is showed in wrong screen. It doesn't matter for vidmode // which displays cloned image in all devices. #ifdef _IRR_LINUX_X11_RANDR_ XMoveResizeWindow(display, window, crtc_x, crtc_y, Width, Height); XRaiseWindow(display, window); XFlush(display); #endif } unsigned int display_width = XDisplayWidth(display, screennr); unsigned int display_height = XDisplayHeight(display, screennr); bool has_display_size = (Width == display_width && Height == display_height); if (SupportsNetWM && (CreationParams.Fullscreen || has_display_size)) { Atom WMStateAtom = XInternAtom(display, "_NET_WM_STATE", true); Atom WMStateAtom1 = None; Atom WMStateAtom2 = None; if (CreationParams.Fullscreen) { WMStateAtom1 = XInternAtom(display, "_NET_WM_STATE_FULLSCREEN", true); } else { WMStateAtom1 = XInternAtom(display, "_NET_WM_STATE_MAXIMIZED_HORZ", true); WMStateAtom2 = XInternAtom(display, "_NET_WM_STATE_MAXIMIZED_VERT", true); } XEvent xev = {0}; // The event should be filled with zeros before setting its attributes xev.type = ClientMessage; xev.xclient.window = window; xev.xclient.message_type = WMStateAtom; xev.xclient.format = 32; xev.xclient.data.l[0] = 1; xev.xclient.data.l[1] = WMStateAtom1; xev.xclient.data.l[2] = WMStateAtom2; XSendEvent(display, RootWindow(display, visual->screen), false, SubstructureRedirectMask | SubstructureNotifyMask, &xev); XFlush(display); // Wait until window state is already changed bool changed = false; for (int i = 0; i < 500; i++) { Atom type; int format; unsigned long numItems, bytesAfter; Atom* atoms = NULL; int s = XGetWindowProperty(display, window, WMStateAtom, 0l, 1024, False, XA_ATOM, &type, &format, &numItems, &bytesAfter, (unsigned char**)&atoms); if (s == Success) { for (unsigned int i = 0; i < numItems; ++i) { if (atoms[i] == WMStateAtom1) { changed = true; break; } } XFree(atoms); } if (changed == true) break; usleep(1000); } if (!changed) { os::Printer::log("Warning! Got timeout when changing window state", ELL_WARNING); } if (NeedsGrabPointer) { grabPointer(true); } } if (!SupportsNetWM && CreationParams.Fullscreen) { XSetInputFocus(display, window, RevertToParent, CurrentTime); int grabKb = XGrabKeyboard(display, window, True, GrabModeAsync, GrabModeAsync, CurrentTime); IrrPrintXGrabError(grabKb, "XGrabKeyboard"); int grabPointer = XGrabPointer(display, window, True, ButtonPressMask, GrabModeAsync, GrabModeAsync, window, None, CurrentTime); IrrPrintXGrabError(grabPointer, "XGrabPointer"); XWarpPointer(display, None, window, 0, 0, 0, 0, 0, 0); } } else { // attach external window window = (Window)CreationParams.WindowId; if (!CreationParams.IgnoreInput) { XCreateWindow(display, window, 0, 0, Width, Height, 0, visual->depth, InputOutput, visual->visual, CWBorderPixel | CWColormap | CWEventMask, &attributes); } XWindowAttributes wa; XGetWindowAttributes(display, window, &wa); CreationParams.WindowSize.Width = wa.width; CreationParams.WindowSize.Height = wa.height; CreationParams.Fullscreen = false; ExternalWindow = true; } WindowMinimized=false; // Currently broken in X, see Bug ID 2795321 // XkbSetDetectableAutoRepeat(display, True, &AutorepeatSupport); #ifdef _IRR_COMPILE_WITH_OPENGL_ // connect glx context to window Context=0; if (isAvailableGLX && CreationParams.DriverType==video::EDT_OPENGL) { if (UseGLXWindow) { glxWin=glXCreateWindow(display,glxFBConfig,window,NULL); if (glxWin) { getLogger()->setLogLevel(ELL_NONE); Context = getMeAGLContext(display, glxFBConfig, CreationParams.ForceLegacyDevice); getLogger()->setLogLevel(CreationParams.LoggingLevel); if (Context) { if (!glXMakeContextCurrent(display, glxWin, glxWin, Context)) { os::Printer::log("Could not make context current.", ELL_WARNING); glXDestroyContext(display, Context); } } else { os::Printer::log("Could not create GLX rendering context.", ELL_WARNING); } } else { os::Printer::log("Could not create GLX window.", ELL_WARNING); } } else { Context = glXCreateContext(display, visual, NULL, True); if (Context) { if (!glXMakeCurrent(display, window, Context)) { os::Printer::log("Could not make context current.", ELL_WARNING); glXDestroyContext(display, Context); } } else { os::Printer::log("Could not create GLX rendering context.", ELL_WARNING); } } } #endif // _IRR_COMPILE_WITH_OPENGL_ Window tmp; u32 borderWidth; int x,y; unsigned int bits; XGetGeometry(display, window, &tmp, &x, &y, &Width, &Height, &borderWidth, &bits); CreationParams.Bits = bits; CreationParams.WindowSize.Width = Width; CreationParams.WindowSize.Height = Height; if (SupportsNetWM == true) { Atom opaque_region = XInternAtom(display, "_NET_WM_OPAQUE_REGION", true); if (opaque_region != None) { unsigned long window_rect[4] = {0, 0, Width, Height}; XChangeProperty(display, window, opaque_region, XA_CARDINAL, 32, PropModeReplace, (unsigned char*)window_rect, 4); } } StdHints = XAllocSizeHints(); long num; XGetWMNormalHints(display, window, StdHints, &num); // create an XImage for the software renderer //(thx to Nadav for some clues on how to do that!) if (CreationParams.DriverType == video::EDT_SOFTWARE || CreationParams.DriverType == video::EDT_BURNINGSVIDEO) { SoftwareImage = XCreateImage(display, visual->visual, visual->depth, ZPixmap, 0, 0, Width, Height, BitmapPad(display), 0); // use malloc because X will free it later on if (SoftwareImage) SoftwareImage->data = (char*) malloc(SoftwareImage->bytes_per_line * SoftwareImage->height * sizeof(char)); } initXAtoms(); #endif // #ifdef _IRR_COMPILE_WITH_X11_ return true; } //! create the driver void CIrrDeviceLinux::createDriver() { switch(CreationParams.DriverType) { #ifdef _IRR_COMPILE_WITH_X11_ case video::EDT_SOFTWARE: #ifdef _IRR_COMPILE_WITH_SOFTWARE_ VideoDriver = video::createSoftwareDriver(CreationParams.WindowSize, CreationParams.Fullscreen, FileSystem, this); #else os::Printer::log("No Software driver support compiled in.", ELL_ERROR); #endif break; case video::EDT_BURNINGSVIDEO: #ifdef _IRR_COMPILE_WITH_BURNINGSVIDEO_ VideoDriver = video::createBurningVideoDriver(CreationParams, FileSystem, this); #else os::Printer::log("Burning's video driver was not compiled in.", ELL_ERROR); #endif break; case video::EDT_OPENGL: #ifdef _IRR_COMPILE_WITH_OPENGL_ if (Context) VideoDriver = video::createOpenGLDriver(CreationParams, FileSystem, this); #else os::Printer::log("No OpenGL support compiled in.", ELL_ERROR); #endif break; case video::EDT_OGLES2: { #ifdef _IRR_COMPILE_WITH_OGLES2_ video::SExposedVideoData data; data.OpenGLLinux.X11Window = window; data.OpenGLLinux.X11Display = display; VideoDriver = video::createOGLES2Driver(CreationParams, data, FileSystem); #else os::Printer::log("No OpenGL ES 2.0 support compiled in.", ELL_ERROR); #endif break; } case video::EDT_DIRECT3D8: case video::EDT_DIRECT3D9: os::Printer::log("This driver is not available in Linux. Try OpenGL or Software renderer.", ELL_ERROR); break; case video::EDT_NULL: VideoDriver = video::createNullDriver(FileSystem, CreationParams.WindowSize); break; default: os::Printer::log("Unable to create video driver of unknown type.", ELL_ERROR); break; #else case video::EDT_NULL: VideoDriver = video::createNullDriver(FileSystem, CreationParams.WindowSize); break; default: os::Printer::log("No X11 support compiled in. Only Null driver available.", ELL_ERROR); break; #endif } } #ifdef _IRR_COMPILE_WITH_X11_ bool CIrrDeviceLinux::createInputContext() { if (!display) return false; // One one side it would be nicer to let users do that - on the other hand // not setting the environment locale will not work when using i18n X11 functions. // So users would have to call it always or their input is broken badly. // We can restore immediately - so won't mess with anything in users apps. core::stringc oldLocale(setlocale(LC_CTYPE, NULL)); setlocale(LC_CTYPE, ""); // use environment locale if ( !XSupportsLocale() ) { os::Printer::log("Locale not supported. Falling back to non-i18n input.", ELL_WARNING); setlocale(LC_CTYPE, oldLocale.c_str()); return false; } // It's showed as memory leak, but we shouldn't delete it. From the xlib // documentation: "The returned modifiers string is owned by Xlib and // should not be modified or freed by the client." char* p = XSetLocaleModifiers(""); if (p == NULL) { os::Printer::log("Could not set locale modifiers. Falling back to non-i18n input.", ELL_WARNING); setlocale(LC_CTYPE, oldLocale.c_str()); return false; } XInputMethod = XOpenIM(display, NULL, NULL, NULL); if ( !XInputMethod ) { os::Printer::log("XOpenIM failed to create an input method. Falling back to non-i18n input.", ELL_WARNING); setlocale(LC_CTYPE, oldLocale.c_str()); return false; } XIMStyles *im_supported_styles = NULL; XGetIMValues(XInputMethod, XNQueryInputStyle, &im_supported_styles, (void*)NULL); if (!im_supported_styles) { os::Printer::log("XGetIMValues failed to get supported styles. Falling back to non-i18n input.", ELL_WARNING); setlocale(LC_CTYPE, oldLocale.c_str()); return false; } // Only OverTheSpot and Root pre-edit type are implemented for now core::array supported_style; //supported_style.push_back(XIMPreeditPosition | XIMStatusNothing); supported_style.push_back(XIMPreeditNothing | XIMStatusNothing); XIMStyle best_style = 0; bool found = false; int supported_style_start = -1; while (!found && supported_style_start < (int)supported_style.size()) { supported_style_start++; for (int i = 0; i < im_supported_styles->count_styles; i++) { XIMStyle cur_style = im_supported_styles->supported_styles[i]; if (cur_style == supported_style[supported_style_start]) { best_style = cur_style; found = true; break; } } } XFree(im_supported_styles); if (!found) { XDestroyIC(XInputContext); XInputContext = 0; os::Printer::log("XInputMethod has no input style we can use. Falling back to non-i18n input.", ELL_WARNING); setlocale(LC_CTYPE, oldLocale.c_str()); return false; } // Only use root preedit type for now /*if (best_style != (XIMPreeditNothing | XIMStatusNothing)) { char **list = NULL; int count = 0; m_font_set = XCreateFontSet(display, "fixed", &list, &count, NULL); if (!m_font_set) { os::Printer::log("XInputContext failed to create font set. Falling back to non-i18n input.", ELL_WARNING); setlocale(LC_CTYPE, oldLocale.c_str()); return false; } if (count > 0) { XFreeStringList(list); } XPoint spot = {0, 0}; XVaNestedList p_list = XVaCreateNestedList(0, XNSpotLocation, &spot, XNFontSet, m_font_set, (void*)NULL); XInputContext = XCreateIC(XInputMethod, XNInputStyle, best_style, XNClientWindow, window, XNFocusWindow, window, XNPreeditAttributes, p_list, (void*)NULL); XFree(p_list); } else*/ { XInputContext = XCreateIC(XInputMethod, XNInputStyle, best_style, XNClientWindow, window, XNFocusWindow, window, (void*)NULL); } if (!XInputContext ) { os::Printer::log("XInputContext failed to create an input context. Falling back to non-i18n input.", ELL_WARNING); setlocale(LC_CTYPE, oldLocale.c_str()); return false; } XSetICFocus(XInputContext); setlocale(LC_CTYPE, oldLocale.c_str()); return true; } void CIrrDeviceLinux::destroyInputContext() { if ( XInputContext ) { XUnsetICFocus(XInputContext); XDestroyIC(XInputContext); XInputContext = 0; } if ( XInputMethod ) { XCloseIM(XInputMethod); XInputMethod = 0; } if (display && m_font_set) { XFreeFontSet(display, m_font_set); m_font_set = 0; } } void CIrrDeviceLinux::setIMELocation(const irr::core::position2di& pos) { if (!XInputContext || !m_ime_enabled) return; m_ime_position.x = pos.X; m_ime_position.y = pos.Y; updateIMELocation(); } void CIrrDeviceLinux::updateIMELocation() { if (!XInputContext || !m_ime_enabled) return; XVaNestedList list; list = XVaCreateNestedList(0, XNSpotLocation, &m_ime_position, (void*)NULL); XSetICValues(XInputContext, XNPreeditAttributes, list, (void*)NULL); XFree(list); } void CIrrDeviceLinux::setIMEEnable(bool enable) { if (!XInputContext) return; m_ime_enabled = enable; } int CIrrDeviceLinux::getNumlockMask(Display* display) { int mask_table[8] = {ShiftMask, LockMask, ControlMask, Mod1Mask, Mod2Mask, Mod3Mask, Mod4Mask, Mod5Mask}; if (!display) return 0; KeyCode numlock_keycode = XKeysymToKeycode(display, XK_Num_Lock); if (numlock_keycode == NoSymbol) return 0; XModifierKeymap* map = XGetModifierMapping(display); if (!map) return 0; int mask = 0; for (int i = 0; i < 8 * map->max_keypermod; i++) { if (map->modifiermap[i] != numlock_keycode) continue; mask = mask_table[i/map->max_keypermod]; break; } XFreeModifiermap(map); return mask; } EKEY_CODE CIrrDeviceLinux::getKeyCode(XEvent &event) { int keyCode = 0; #ifdef _IRR_COMPILE_WITH_X11_ SKeyMap mp; // First check for numpad keys bool is_numpad_key = false; if (event.xkey.state & numlock_mask) { mp.X11Key = XkbKeycodeToKeysym(display, event.xkey.keycode, 0, 1); if ((mp.X11Key >=XK_KP_0 && mp.X11Key <= XK_KP_9 ) || mp.X11Key == XK_KP_Decimal) is_numpad_key = true; } // If it's not numpad key, then get keycode in typical way if (!is_numpad_key) { mp.X11Key = XkbKeycodeToKeysym(display, event.xkey.keycode, 0, 0); } const s32 idx = KeyMap.binary_search(mp); if (idx != -1) { keyCode = KeyMap[idx].Win32Key; } if (keyCode == 0) { // Any value is better than none, that allows at least using the keys. // Worst case is that some keys will be identical, still better than _all_ // unknown keys being identical. if (mp.X11Key) { keyCode = (int)IRR_KEY_CODES_COUNT + mp.X11Key; } else { keyCode = (int)IRR_KEY_CODES_COUNT + event.xkey.keycode; } os::Printer::log("EKEY_CODE is 0, fallback keycode", core::stringc(keyCode).c_str(), ELL_INFORMATION); } #endif return (EKEY_CODE)keyCode; } #endif //! runs the device. Returns false if device wants to be deleted bool CIrrDeviceLinux::run() { os::Timer::tick(); #ifdef _IRR_COMPILE_WITH_X11_ if ( CursorControl ) static_cast(CursorControl)->update(); if ((CreationParams.DriverType != video::EDT_NULL) && display) { SEvent irrevent; irrevent.MouseInput.ButtonStates = 0xffffffff; while (XPending(display) > 0 && !Close) { if (!m_ime_char_holder.empty()) { irrevent.KeyInput.Char = m_ime_char_holder[0]; irrevent.EventType = irr::EET_KEY_INPUT_EVENT; irrevent.KeyInput.PressedDown = true; irrevent.KeyInput.Control = false; irrevent.KeyInput.Shift = false; irrevent.KeyInput.Key = (irr::EKEY_CODE)0; postEventFromUser(irrevent); m_ime_char_holder.erase(0); continue; } XEvent event; XNextEvent(display, &event); if (m_ime_enabled && XFilterEvent(&event, None)) { updateIMELocation(); continue; } switch (event.type) { case ConfigureNotify: updateIMELocation(); // check for changed window size if ((event.xconfigure.width != (int) Width) || (event.xconfigure.height != (int) Height)) { Width = event.xconfigure.width; Height = event.xconfigure.height; // resize image data if (SoftwareImage) { XDestroyImage(SoftwareImage); SoftwareImage = XCreateImage(display, visual->visual, visual->depth, ZPixmap, 0, 0, Width, Height, BitmapPad(display), 0); // use malloc because X will free it later on if (SoftwareImage) SoftwareImage->data = (char*) malloc(SoftwareImage->bytes_per_line * SoftwareImage->height * sizeof(char)); } if (VideoDriver) VideoDriver->OnResize(core::dimension2d(Width, Height)); } break; case MapNotify: WindowMinimized=false; break; case UnmapNotify: WindowMinimized=true; break; case FocusIn: if (NeedsGrabPointer) { grabPointer(true); } WindowHasFocus=true; break; case FocusOut: if (NeedsGrabPointer) { grabPointer(false); } WindowHasFocus=false; break; case MotionNotify: irrevent.EventType = irr::EET_MOUSE_INPUT_EVENT; irrevent.MouseInput.Event = irr::EMIE_MOUSE_MOVED; irrevent.MouseInput.X = event.xbutton.x; irrevent.MouseInput.Y = event.xbutton.y; irrevent.MouseInput.Control = (event.xkey.state & ControlMask) != 0; irrevent.MouseInput.Shift = (event.xkey.state & ShiftMask) != 0; // mouse button states irrevent.MouseInput.ButtonStates = (event.xbutton.state & Button1Mask) ? irr::EMBSM_LEFT : 0; irrevent.MouseInput.ButtonStates |= (event.xbutton.state & Button3Mask) ? irr::EMBSM_RIGHT : 0; irrevent.MouseInput.ButtonStates |= (event.xbutton.state & Button2Mask) ? irr::EMBSM_MIDDLE : 0; postEventFromUser(irrevent); break; case ButtonPress: case ButtonRelease: irrevent.EventType = irr::EET_MOUSE_INPUT_EVENT; irrevent.MouseInput.X = event.xbutton.x; irrevent.MouseInput.Y = event.xbutton.y; irrevent.MouseInput.Control = (event.xkey.state & ControlMask) != 0; irrevent.MouseInput.Shift = (event.xkey.state & ShiftMask) != 0; // mouse button states // This sets the state which the buttons had _prior_ to the event. // So unlike on Windows the button which just got changed has still the old state here. // We handle that below by flipping the corresponding bit later. irrevent.MouseInput.ButtonStates = (event.xbutton.state & Button1Mask) ? irr::EMBSM_LEFT : 0; irrevent.MouseInput.ButtonStates |= (event.xbutton.state & Button3Mask) ? irr::EMBSM_RIGHT : 0; irrevent.MouseInput.ButtonStates |= (event.xbutton.state & Button2Mask) ? irr::EMBSM_MIDDLE : 0; irrevent.MouseInput.Event = irr::EMIE_COUNT; switch(event.xbutton.button) { case Button1: irrevent.MouseInput.Event = (event.type == ButtonPress) ? irr::EMIE_LMOUSE_PRESSED_DOWN : irr::EMIE_LMOUSE_LEFT_UP; irrevent.MouseInput.ButtonStates ^= irr::EMBSM_LEFT; break; case Button3: irrevent.MouseInput.Event = (event.type == ButtonPress) ? irr::EMIE_RMOUSE_PRESSED_DOWN : irr::EMIE_RMOUSE_LEFT_UP; irrevent.MouseInput.ButtonStates ^= irr::EMBSM_RIGHT; break; case Button2: irrevent.MouseInput.Event = (event.type == ButtonPress) ? irr::EMIE_MMOUSE_PRESSED_DOWN : irr::EMIE_MMOUSE_LEFT_UP; irrevent.MouseInput.ButtonStates ^= irr::EMBSM_MIDDLE; break; case Button4: if (event.type == ButtonPress) { irrevent.MouseInput.Event = EMIE_MOUSE_WHEEL; irrevent.MouseInput.Wheel = 1.0f; } break; case Button5: if (event.type == ButtonPress) { irrevent.MouseInput.Event = EMIE_MOUSE_WHEEL; irrevent.MouseInput.Wheel = -1.0f; } break; } if (irrevent.MouseInput.Event != irr::EMIE_COUNT) { postEventFromUser(irrevent); if ( irrevent.MouseInput.Event >= EMIE_LMOUSE_PRESSED_DOWN && irrevent.MouseInput.Event <= EMIE_MMOUSE_PRESSED_DOWN ) { u32 clicks = checkSuccessiveClicks(irrevent.MouseInput.X, irrevent.MouseInput.Y, irrevent.MouseInput.Event); if ( clicks == 2 ) { irrevent.MouseInput.Event = (EMOUSE_INPUT_EVENT)(EMIE_LMOUSE_DOUBLE_CLICK + irrevent.MouseInput.Event-EMIE_LMOUSE_PRESSED_DOWN); postEventFromUser(irrevent); } else if ( clicks == 3 ) { irrevent.MouseInput.Event = (EMOUSE_INPUT_EVENT)(EMIE_LMOUSE_TRIPLE_CLICK + irrevent.MouseInput.Event-EMIE_LMOUSE_PRESSED_DOWN); postEventFromUser(irrevent); } } } break; case MappingNotify: XRefreshKeyboardMapping (&event.xmapping) ; break; case KeyRelease: if (0 == AutorepeatSupport && (XPending( display ) > 0) ) { // check for Autorepeat manually // We'll do the same as Windows does: Only send KeyPressed // So every KeyRelease is a real release XEvent next_event; XPeekEvent (event.xkey.display, &next_event); if ((next_event.type == KeyPress) && (next_event.xkey.keycode == event.xkey.keycode) && (next_event.xkey.time - event.xkey.time) < 2) // usually same time, but on some systems a difference of 1 is possible { // Ignore the key release event break; } } irrevent.EventType = irr::EET_KEY_INPUT_EVENT; irrevent.KeyInput.PressedDown = false; irrevent.KeyInput.Char = 0; // on release that's undefined irrevent.KeyInput.Control = (event.xkey.state & ControlMask) != 0; irrevent.KeyInput.Shift = (event.xkey.state & ShiftMask) != 0; irrevent.KeyInput.Key = getKeyCode(event); postEventFromUser(irrevent); break; case KeyPress: { SKeyMap mp; if ( XInputContext ) { Status status; int strLen = XwcLookupString(XInputContext, &event.xkey, m_ime_char_holder.pointer(), 16 * sizeof(wchar_t), &mp.X11Key, &status); if ( status == XBufferOverflow ) { os::Printer::log("XwcLookupString needs a larger buffer", ELL_INFORMATION); } if ( strLen > 0 && (status == XLookupChars || status == XLookupBoth) ) { if ( strLen > 1 ) { m_ime_char_holder.set_used(strLen > 16 ? 16 : strLen); continue; } else { irrevent.KeyInput.Char = m_ime_char_holder[0]; } } else { irrevent.KeyInput.Char = 0; } } else // Old version without InputContext. Does not support i18n, but good to have as fallback. { union { char buf[8]; wchar_t wbuf[2]; } tmp = {{0}}; XLookupString(&event.xkey, tmp.buf, sizeof(tmp.buf), &mp.X11Key, NULL); irrevent.KeyInput.Char = tmp.wbuf[0]; } irrevent.EventType = irr::EET_KEY_INPUT_EVENT; irrevent.KeyInput.PressedDown = true; irrevent.KeyInput.Control = (event.xkey.state & ControlMask) != 0; irrevent.KeyInput.Shift = (event.xkey.state & ShiftMask) != 0; irrevent.KeyInput.Key = getKeyCode(event); postEventFromUser(irrevent); } break; case ClientMessage: { char* atom = XGetAtomName(display, event.xclient.message_type); if (atom != NULL) { if (strcmp(atom, "WM_PROTOCOLS") == 0) { os::Printer::log("Quit message received.", ELL_INFORMATION); Close = true; } XFree(atom); } if (!Close) { // we assume it's a user message irrevent.EventType = irr::EET_USER_EVENT; irrevent.UserEvent.UserData1 = (s32)event.xclient.data.l[0]; irrevent.UserEvent.UserData2 = (s32)event.xclient.data.l[1]; postEventFromUser(irrevent); } } break; case SelectionRequest: { XEvent respond; XSelectionRequestEvent *req = &(event.xselectionrequest); if ( req->target == X_ATOM_UTF8_STRING) { XChangeProperty (display, req->requestor, req->property, req->target, 8, // format PropModeReplace, (unsigned char*) Clipboard.c_str(), Clipboard.size()); respond.xselection.property = req->property; } else if ( req->target == X_ATOM_TARGETS ) { long data[1]; data[0] = X_ATOM_UTF8_STRING; XChangeProperty (display, req->requestor, req->property, XA_ATOM, 32, PropModeReplace, (unsigned char *) &data, sizeof (data)); respond.xselection.property = req->property; } else { respond.xselection.property= None; } respond.xselection.type= SelectionNotify; respond.xselection.display= req->display; respond.xselection.requestor= req->requestor; respond.xselection.selection=req->selection; respond.xselection.target= req->target; respond.xselection.time = req->time; XSendEvent (display, req->requestor,0,0,&respond); XFlush (display); } break; default: break; } // end switch } // end while } #endif //_IRR_COMPILE_WITH_X11_ if (!Close) pollJoysticks(); return !Close; } //! Pause the current process for the minimum time allowed only to allow other processes to execute void CIrrDeviceLinux::yield() { struct timespec ts = {0,1}; nanosleep(&ts, NULL); } //! Pause execution and let other processes to run for a specified amount of time. void CIrrDeviceLinux::sleep(u32 timeMs, bool pauseTimer=false) { const bool wasStopped = Timer ? Timer->isStopped() : true; struct timespec ts; ts.tv_sec = (time_t) (timeMs / 1000); ts.tv_nsec = (long) (timeMs % 1000) * 1000000; if (pauseTimer && !wasStopped) Timer->stop(); nanosleep(&ts, NULL); if (pauseTimer && !wasStopped) Timer->start(); } //! sets the caption of the window void CIrrDeviceLinux::setWindowCaption(const wchar_t* text) { #ifdef _IRR_COMPILE_WITH_X11_ if (CreationParams.DriverType == video::EDT_NULL) return; XTextProperty txt; if (Success==XwcTextListToTextProperty(display, const_cast(&text), 1, XStdICCTextStyle, &txt)) { XSetWMName(display, window, &txt); XSetWMIconName(display, window, &txt); XFree(txt.value); } #endif } //! sets the class of the window void CIrrDeviceLinux::setWindowClass(const char* text) { #ifdef _IRR_COMPILE_WITH_X11_ if (CreationParams.DriverType == video::EDT_NULL) return; // Set class hints on Linux, used by Window Managers. XClassHint* classhint = XAllocClassHint(); classhint->res_name = (char*)text; classhint->res_class = (char*)text; XSetClassHint(display, window, classhint); XFree(classhint); #endif } //! presents a surface in the client area bool CIrrDeviceLinux::present(video::IImage* image, void* windowId, core::rect* srcRect) { #ifdef _IRR_COMPILE_WITH_X11_ // this is only necessary for software drivers. if (!SoftwareImage) return true; // thx to Nadav, who send me some clues of how to display the image // to the X Server. const u32 destwidth = SoftwareImage->width; const u32 minWidth = core::min_(image->getDimension().Width, destwidth); const u32 destPitch = SoftwareImage->bytes_per_line; video::ECOLOR_FORMAT destColor; switch (SoftwareImage->bits_per_pixel) { case 16: if (SoftwareImage->depth==16) destColor = video::ECF_R5G6B5; else destColor = video::ECF_A1R5G5B5; break; case 24: destColor = video::ECF_R8G8B8; break; case 32: destColor = video::ECF_A8R8G8B8; break; default: os::Printer::log("Unsupported screen depth."); return false; } u8* srcdata = reinterpret_cast(image->lock()); u8* destData = reinterpret_cast(SoftwareImage->data); const u32 destheight = SoftwareImage->height; const u32 srcheight = core::min_(image->getDimension().Height, destheight); const u32 srcPitch = image->getPitch(); for (u32 y=0; y!=srcheight; ++y) { video::CColorConverter::convert_viaFormat(srcdata,image->getColorFormat(), minWidth, destData, destColor); srcdata+=srcPitch; destData+=destPitch; } image->unlock(); GC gc = DefaultGC(display, DefaultScreen(display)); Window myWindow=window; if (windowId) myWindow = reinterpret_cast(windowId); XPutImage(display, myWindow, gc, SoftwareImage, 0, 0, 0, 0, destwidth, destheight); #endif return true; } //! notifies the device that it should close itself void CIrrDeviceLinux::closeDevice() { Close = true; } //! returns if window is active. if not, nothing need to be drawn bool CIrrDeviceLinux::isWindowActive() const { return (WindowHasFocus && !WindowMinimized); } //! returns if window has focus. bool CIrrDeviceLinux::isWindowFocused() const { return WindowHasFocus; } //! returns if window is minimized. bool CIrrDeviceLinux::isWindowMinimized() const { return WindowMinimized; } //! returns color format of the window. video::ECOLOR_FORMAT CIrrDeviceLinux::getColorFormat() const { #ifdef _IRR_COMPILE_WITH_X11_ if (visual && (visual->depth != 16)) return video::ECF_R8G8B8; else #endif return video::ECF_R5G6B5; } //! Sets if the window should be resizable in windowed mode. void CIrrDeviceLinux::setResizable(bool resize) { #ifdef _IRR_COMPILE_WITH_X11_ if (CreationParams.DriverType == video::EDT_NULL || CreationParams.Fullscreen ) return; XUnmapWindow(display, window); if ( !resize ) { // Must be heap memory because data size depends on X Server XSizeHints *hints = XAllocSizeHints(); hints->flags=PSize|PMinSize|PMaxSize; hints->min_width=hints->max_width=hints->base_width=Width; hints->min_height=hints->max_height=hints->base_height=Height; XSetWMNormalHints(display, window, hints); XFree(hints); } else { XSetWMNormalHints(display, window, StdHints); } XMapWindow(display, window); XFlush(display); #endif // #ifdef _IRR_COMPILE_WITH_X11_ } //! Return pointer to a list with all video modes supported by the gfx adapter. video::IVideoModeList* CIrrDeviceLinux::getVideoModeList() { #ifdef _IRR_COMPILE_WITH_X11_ if (!VideoModeList.getVideoModeCount()) { bool temporaryDisplay = false; if (!display) { display = XOpenDisplay(0); temporaryDisplay=true; } if (display) { #if defined(_IRR_LINUX_X11_VIDMODE_) || defined(_IRR_LINUX_X11_RANDR_) s32 eventbase, errorbase; s32 defaultDepth=DefaultDepth(display,screennr); #endif #ifdef _IRR_LINUX_X11_VIDMODE_ if (XF86VidModeQueryExtension(display, &eventbase, &errorbase)) { // enumerate video modes int modeCount; XF86VidModeModeInfo** modes; XF86VidModeGetAllModeLines(display, screennr, &modeCount, &modes); // save current video mode oldVideoMode = *modes[0]; // find fitting mode VideoModeList.setDesktop(defaultDepth, core::dimension2d( modes[0]->hdisplay, modes[0]->vdisplay)); for (int i = 0; i( modes[i]->hdisplay, modes[i]->vdisplay), defaultDepth); } XFree(modes); } #endif #ifdef _IRR_LINUX_X11_RANDR_ output_id = 0; old_mode = 0; while (XRRQueryExtension(display, &eventbase, &errorbase)) { XRROutputInfo* output = NULL; XRRCrtcInfo* crtc = NULL; crtc_x = crtc_y = -1; XRRScreenResources* res = XRRGetScreenResources(display, DefaultRootWindow(display)); if (!res) break; const char* output_name = getenv("IRR_VIDEO_OUTPUT"); RROutput primary_id = XRRGetOutputPrimary(display, DefaultRootWindow(display)); for (int i = 0; i < res->noutput; i++) { XRROutputInfo* output_tmp = XRRGetOutputInfo(display, res, res->outputs[i]); if (!output_tmp || !output_tmp->crtc || output_tmp->connection == RR_Disconnected) { XRRFreeOutputInfo(output_tmp); continue; } XRRCrtcInfo* crtc_tmp = XRRGetCrtcInfo(display, res, output_tmp->crtc); if (!crtc_tmp) { XRRFreeOutputInfo(output_tmp); continue; } bool is_primary = false; if (output_name != NULL) { is_primary = (strcmp(output_name, output_tmp->name) == 0); } else { is_primary = (res->outputs[i] == primary_id); } if (is_primary || output_id == 0 || crtc_tmp->x < crtc->x || (crtc_tmp->x == crtc->x && crtc_tmp->y < crtc->y)) { XRRFreeCrtcInfo(crtc); XRRFreeOutputInfo(output); output = output_tmp; crtc = crtc_tmp; output_id = res->outputs[i]; } else { XRRFreeCrtcInfo(crtc_tmp); XRRFreeOutputInfo(output_tmp); } if (is_primary) break; } if (output_id == 0) { os::Printer::log("Could not get video output.", ELL_WARNING); break; } crtc_x = crtc->x; crtc_y = crtc->y; for (int i = 0; i < res->nmode; i++) { const XRRModeInfo* mode = &res->modes[i]; core::dimension2d size; if (crtc->rotation & (XRANDR_ROTATION_LEFT|XRANDR_ROTATION_RIGHT)) { size = core::dimension2d(mode->height, mode->width); } else { size = core::dimension2d(mode->width, mode->height); } for (int j = 0; j < output->nmode; j++) { if (mode->id == output->modes[j]) { VideoModeList.addMode(size, defaultDepth); break; } } if (mode->id == crtc->mode) { old_mode = crtc->mode; VideoModeList.setDesktop(defaultDepth, size); } } XRRFreeCrtcInfo(crtc); XRRFreeOutputInfo(output); XRRFreeScreenResources(res); break; } #endif } if (display && temporaryDisplay) { XCloseDisplay(display); display=0; } } #endif return &VideoModeList; } //! Minimize window void CIrrDeviceLinux::minimizeWindow() { #ifdef _IRR_COMPILE_WITH_X11_ XIconifyWindow(display, window, screennr); #endif } //! Maximize window void CIrrDeviceLinux::maximizeWindow() { #ifdef _IRR_COMPILE_WITH_X11_ XMapWindow(display, window); #endif } //! Restore original window size void CIrrDeviceLinux::restoreWindow() { #ifdef _IRR_COMPILE_WITH_X11_ XMapWindow(display, window); #endif } /* Returns the parent window of "window" (i.e. the ancestor of window that is a direct child of the root, or window itself if it is a direct child). If window is the root window, returns window. */ #ifdef _IRR_COMPILE_WITH_X11_ bool get_toplevel_parent(Display* display, Window window, Window* tp_window) { Window current_window = window; Window parent; Window root; Window* children; unsigned int num_children; while (true) { bool success = XQueryTree(display, current_window, &root, &parent, &children, &num_children); if (!success) { os::Printer::log("XQueryTree error", ELL_ERROR); return false; } if (children) { XFree(children); } if (current_window == root || parent == root) { *tp_window = current_window; return true; } else { current_window = parent; } } return false; } #endif //! Move window to requested position bool CIrrDeviceLinux::moveWindow(int x, int y) { #ifdef _IRR_COMPILE_WITH_X11_ if (CreationParams.DriverType == video::EDT_NULL || CreationParams.Fullscreen) return false; int display_width = XDisplayWidth(display, screennr); int display_height = XDisplayHeight(display, screennr); core::min_(x, display_width - (int)Width); core::min_(y, display_height - (int)Height); XMoveWindow(display, window, x, y); return true; #endif return false; } //! Get current window position. bool CIrrDeviceLinux::getWindowPosition(int* x, int* y) { #ifdef _IRR_COMPILE_WITH_X11_ if (CreationParams.DriverType == video::EDT_NULL || CreationParams.Fullscreen) return false; Window tp_window; bool success = get_toplevel_parent(display, window, &tp_window); if (!success) return false; XWindowAttributes xwa; success = XGetWindowAttributes(display, tp_window, &xwa); if (!success) return false; *x = xwa.x; *y = xwa.y; return true; #endif return false; } void CIrrDeviceLinux::createKeyMap() { // I don't know if this is the best method to create // the lookuptable, but I'll leave it like that until // I find a better version. #ifdef _IRR_COMPILE_WITH_X11_ KeyMap.reallocate(190); KeyMap.push_back(SKeyMap(XK_BackSpace, IRR_KEY_BACK)); KeyMap.push_back(SKeyMap(XK_Tab, IRR_KEY_TAB)); KeyMap.push_back(SKeyMap(XK_ISO_Left_Tab, IRR_KEY_TAB)); KeyMap.push_back(SKeyMap(XK_Linefeed, 0)); // ??? KeyMap.push_back(SKeyMap(XK_Clear, IRR_KEY_CLEAR)); KeyMap.push_back(SKeyMap(XK_Return, IRR_KEY_RETURN)); KeyMap.push_back(SKeyMap(XK_Pause, IRR_KEY_PAUSE)); KeyMap.push_back(SKeyMap(XK_Scroll_Lock, IRR_KEY_SCROLL)); KeyMap.push_back(SKeyMap(XK_Sys_Req, 0)); // ??? KeyMap.push_back(SKeyMap(XK_Escape, IRR_KEY_ESCAPE)); KeyMap.push_back(SKeyMap(XK_Insert, IRR_KEY_INSERT)); KeyMap.push_back(SKeyMap(XK_Delete, IRR_KEY_DELETE)); KeyMap.push_back(SKeyMap(XK_Home, IRR_KEY_HOME)); KeyMap.push_back(SKeyMap(XK_Left, IRR_KEY_LEFT)); KeyMap.push_back(SKeyMap(XK_Up, IRR_KEY_UP)); KeyMap.push_back(SKeyMap(XK_Right, IRR_KEY_RIGHT)); KeyMap.push_back(SKeyMap(XK_Down, IRR_KEY_DOWN)); KeyMap.push_back(SKeyMap(XK_Prior, IRR_KEY_PRIOR)); KeyMap.push_back(SKeyMap(XK_Page_Up, IRR_KEY_PRIOR)); KeyMap.push_back(SKeyMap(XK_Next, IRR_KEY_NEXT)); KeyMap.push_back(SKeyMap(XK_Page_Down, IRR_KEY_NEXT)); KeyMap.push_back(SKeyMap(XK_End, IRR_KEY_END)); KeyMap.push_back(SKeyMap(XK_Begin, IRR_KEY_HOME)); KeyMap.push_back(SKeyMap(XK_Num_Lock, IRR_KEY_NUMLOCK)); KeyMap.push_back(SKeyMap(XK_KP_Space, IRR_KEY_SPACE)); KeyMap.push_back(SKeyMap(XK_KP_Tab, IRR_KEY_TAB)); KeyMap.push_back(SKeyMap(XK_KP_Enter, IRR_KEY_RETURN)); KeyMap.push_back(SKeyMap(XK_KP_F1, IRR_KEY_F1)); KeyMap.push_back(SKeyMap(XK_KP_F2, IRR_KEY_F2)); KeyMap.push_back(SKeyMap(XK_KP_F3, IRR_KEY_F3)); KeyMap.push_back(SKeyMap(XK_KP_F4, IRR_KEY_F4)); KeyMap.push_back(SKeyMap(XK_KP_Home, IRR_KEY_HOME)); KeyMap.push_back(SKeyMap(XK_KP_Left, IRR_KEY_LEFT)); KeyMap.push_back(SKeyMap(XK_KP_Up, IRR_KEY_UP)); KeyMap.push_back(SKeyMap(XK_KP_Right, IRR_KEY_RIGHT)); KeyMap.push_back(SKeyMap(XK_KP_Down, IRR_KEY_DOWN)); KeyMap.push_back(SKeyMap(XK_Print, IRR_KEY_PRINT)); KeyMap.push_back(SKeyMap(XK_KP_Prior, IRR_KEY_PRIOR)); KeyMap.push_back(SKeyMap(XK_KP_Page_Up, IRR_KEY_PRIOR)); KeyMap.push_back(SKeyMap(XK_KP_Next, IRR_KEY_NEXT)); KeyMap.push_back(SKeyMap(XK_KP_Page_Down, IRR_KEY_NEXT)); KeyMap.push_back(SKeyMap(XK_KP_End, IRR_KEY_END)); KeyMap.push_back(SKeyMap(XK_KP_Begin, IRR_KEY_HOME)); KeyMap.push_back(SKeyMap(XK_KP_Insert, IRR_KEY_INSERT)); KeyMap.push_back(SKeyMap(XK_KP_Delete, IRR_KEY_DELETE)); KeyMap.push_back(SKeyMap(XK_KP_Equal, 0)); // ??? KeyMap.push_back(SKeyMap(XK_KP_Multiply, IRR_KEY_MULTIPLY)); KeyMap.push_back(SKeyMap(XK_KP_Add, IRR_KEY_ADD)); KeyMap.push_back(SKeyMap(XK_KP_Separator, IRR_KEY_SEPARATOR)); KeyMap.push_back(SKeyMap(XK_KP_Subtract, IRR_KEY_SUBTRACT)); KeyMap.push_back(SKeyMap(XK_KP_Decimal, IRR_KEY_DECIMAL)); KeyMap.push_back(SKeyMap(XK_KP_Divide, IRR_KEY_DIVIDE)); KeyMap.push_back(SKeyMap(XK_KP_0, IRR_KEY_NUMPAD0)); KeyMap.push_back(SKeyMap(XK_KP_1, IRR_KEY_NUMPAD1)); KeyMap.push_back(SKeyMap(XK_KP_2, IRR_KEY_NUMPAD2)); KeyMap.push_back(SKeyMap(XK_KP_3, IRR_KEY_NUMPAD3)); KeyMap.push_back(SKeyMap(XK_KP_4, IRR_KEY_NUMPAD4)); KeyMap.push_back(SKeyMap(XK_KP_5, IRR_KEY_NUMPAD5)); KeyMap.push_back(SKeyMap(XK_KP_6, IRR_KEY_NUMPAD6)); KeyMap.push_back(SKeyMap(XK_KP_7, IRR_KEY_NUMPAD7)); KeyMap.push_back(SKeyMap(XK_KP_8, IRR_KEY_NUMPAD8)); KeyMap.push_back(SKeyMap(XK_KP_9, IRR_KEY_NUMPAD9)); KeyMap.push_back(SKeyMap(XK_F1, IRR_KEY_F1)); KeyMap.push_back(SKeyMap(XK_F2, IRR_KEY_F2)); KeyMap.push_back(SKeyMap(XK_F3, IRR_KEY_F3)); KeyMap.push_back(SKeyMap(XK_F4, IRR_KEY_F4)); KeyMap.push_back(SKeyMap(XK_F5, IRR_KEY_F5)); KeyMap.push_back(SKeyMap(XK_F6, IRR_KEY_F6)); KeyMap.push_back(SKeyMap(XK_F7, IRR_KEY_F7)); KeyMap.push_back(SKeyMap(XK_F8, IRR_KEY_F8)); KeyMap.push_back(SKeyMap(XK_F9, IRR_KEY_F9)); KeyMap.push_back(SKeyMap(XK_F10, IRR_KEY_F10)); KeyMap.push_back(SKeyMap(XK_F11, IRR_KEY_F11)); KeyMap.push_back(SKeyMap(XK_F12, IRR_KEY_F12)); KeyMap.push_back(SKeyMap(XK_Shift_L, IRR_KEY_LSHIFT)); KeyMap.push_back(SKeyMap(XK_Shift_R, IRR_KEY_RSHIFT)); KeyMap.push_back(SKeyMap(XK_Control_L, IRR_KEY_LCONTROL)); KeyMap.push_back(SKeyMap(XK_Control_R, IRR_KEY_RCONTROL)); KeyMap.push_back(SKeyMap(XK_Caps_Lock, IRR_KEY_CAPITAL)); KeyMap.push_back(SKeyMap(XK_Shift_Lock, IRR_KEY_CAPITAL)); KeyMap.push_back(SKeyMap(XK_Meta_L, IRR_KEY_LWIN)); KeyMap.push_back(SKeyMap(XK_Meta_R, IRR_KEY_RWIN)); KeyMap.push_back(SKeyMap(XK_Alt_L, IRR_KEY_LMENU)); KeyMap.push_back(SKeyMap(XK_Alt_R, IRR_KEY_RMENU)); KeyMap.push_back(SKeyMap(XK_ISO_Level3_Shift, IRR_KEY_RMENU)); KeyMap.push_back(SKeyMap(XK_Menu, IRR_KEY_MENU)); KeyMap.push_back(SKeyMap(XK_space, IRR_KEY_SPACE)); KeyMap.push_back(SKeyMap(XK_exclam, IRR_KEY_EXCLAM)); KeyMap.push_back(SKeyMap(XK_quotedbl, IRR_KEY_QUOTEDBL)); KeyMap.push_back(SKeyMap(XK_section, IRR_KEY_SECTION)); //? KeyMap.push_back(SKeyMap(XK_numbersign, IRR_KEY_OEM_2)); KeyMap.push_back(SKeyMap(XK_dollar, IRR_KEY_DOLLAR)); KeyMap.push_back(SKeyMap(XK_percent, 0)); //? KeyMap.push_back(SKeyMap(XK_ampersand, IRR_KEY_AMPERSAND)); KeyMap.push_back(SKeyMap(XK_apostrophe, IRR_KEY_OEM_7)); KeyMap.push_back(SKeyMap(XK_parenleft, IRR_KEY_PARENLEFT)); KeyMap.push_back(SKeyMap(XK_parenright, IRR_KEY_PARENRIGHT)); KeyMap.push_back(SKeyMap(XK_asterisk, 0)); //? KeyMap.push_back(SKeyMap(XK_plus, IRR_KEY_PLUS)); //? KeyMap.push_back(SKeyMap(XK_comma, IRR_KEY_COMMA)); //? KeyMap.push_back(SKeyMap(XK_minus, IRR_KEY_MINUS)); //? KeyMap.push_back(SKeyMap(XK_period, IRR_KEY_PERIOD)); //? KeyMap.push_back(SKeyMap(XK_slash, IRR_KEY_OEM_2)); //? KeyMap.push_back(SKeyMap(XK_0, IRR_KEY_0)); KeyMap.push_back(SKeyMap(XK_1, IRR_KEY_1)); KeyMap.push_back(SKeyMap(XK_2, IRR_KEY_2)); KeyMap.push_back(SKeyMap(XK_3, IRR_KEY_3)); KeyMap.push_back(SKeyMap(XK_4, IRR_KEY_4)); KeyMap.push_back(SKeyMap(XK_5, IRR_KEY_5)); KeyMap.push_back(SKeyMap(XK_6, IRR_KEY_6)); KeyMap.push_back(SKeyMap(XK_7, IRR_KEY_7)); KeyMap.push_back(SKeyMap(XK_8, IRR_KEY_8)); KeyMap.push_back(SKeyMap(XK_9, IRR_KEY_9)); KeyMap.push_back(SKeyMap(XK_colon, IRR_KEY_COLON)); KeyMap.push_back(SKeyMap(XK_semicolon, IRR_KEY_OEM_1)); KeyMap.push_back(SKeyMap(XK_less, IRR_KEY_OEM_102)); KeyMap.push_back(SKeyMap(XK_equal, IRR_KEY_PLUS)); KeyMap.push_back(SKeyMap(XK_greater, 0)); //? KeyMap.push_back(SKeyMap(XK_question, 0)); //? KeyMap.push_back(SKeyMap(XK_at, IRR_KEY_2)); //? KeyMap.push_back(SKeyMap(XK_mu, IRR_KEY_MU)); //? KeyMap.push_back(SKeyMap(XK_EuroSign, 0)); //? KeyMap.push_back(SKeyMap(XK_A, IRR_KEY_A)); KeyMap.push_back(SKeyMap(XK_B, IRR_KEY_B)); KeyMap.push_back(SKeyMap(XK_C, IRR_KEY_C)); KeyMap.push_back(SKeyMap(XK_D, IRR_KEY_D)); KeyMap.push_back(SKeyMap(XK_E, IRR_KEY_E)); KeyMap.push_back(SKeyMap(XK_F, IRR_KEY_F)); KeyMap.push_back(SKeyMap(XK_G, IRR_KEY_G)); KeyMap.push_back(SKeyMap(XK_H, IRR_KEY_H)); KeyMap.push_back(SKeyMap(XK_I, IRR_KEY_I)); KeyMap.push_back(SKeyMap(XK_J, IRR_KEY_J)); KeyMap.push_back(SKeyMap(XK_K, IRR_KEY_K)); KeyMap.push_back(SKeyMap(XK_L, IRR_KEY_L)); KeyMap.push_back(SKeyMap(XK_M, IRR_KEY_M)); KeyMap.push_back(SKeyMap(XK_N, IRR_KEY_N)); KeyMap.push_back(SKeyMap(XK_O, IRR_KEY_O)); KeyMap.push_back(SKeyMap(XK_P, IRR_KEY_P)); KeyMap.push_back(SKeyMap(XK_Q, IRR_KEY_Q)); KeyMap.push_back(SKeyMap(XK_R, IRR_KEY_R)); KeyMap.push_back(SKeyMap(XK_S, IRR_KEY_S)); KeyMap.push_back(SKeyMap(XK_T, IRR_KEY_T)); KeyMap.push_back(SKeyMap(XK_U, IRR_KEY_U)); KeyMap.push_back(SKeyMap(XK_V, IRR_KEY_V)); KeyMap.push_back(SKeyMap(XK_W, IRR_KEY_W)); KeyMap.push_back(SKeyMap(XK_X, IRR_KEY_X)); KeyMap.push_back(SKeyMap(XK_Y, IRR_KEY_Y)); KeyMap.push_back(SKeyMap(XK_Z, IRR_KEY_Z)); KeyMap.push_back(SKeyMap(XK_bracketleft, IRR_KEY_OEM_4)); KeyMap.push_back(SKeyMap(XK_backslash, IRR_KEY_OEM_5)); KeyMap.push_back(SKeyMap(XK_bracketright, IRR_KEY_OEM_6)); KeyMap.push_back(SKeyMap(XK_asciicircum, IRR_KEY_OEM_5)); KeyMap.push_back(SKeyMap(XK_degree, 0)); //? KeyMap.push_back(SKeyMap(XK_underscore, IRR_KEY_MINUS)); //? KeyMap.push_back(SKeyMap(XK_grave, IRR_KEY_OEM_3)); KeyMap.push_back(SKeyMap(XK_acute, IRR_KEY_OEM_6)); KeyMap.push_back(SKeyMap(XK_a, IRR_KEY_A)); KeyMap.push_back(SKeyMap(XK_b, IRR_KEY_B)); KeyMap.push_back(SKeyMap(XK_c, IRR_KEY_C)); KeyMap.push_back(SKeyMap(XK_d, IRR_KEY_D)); KeyMap.push_back(SKeyMap(XK_e, IRR_KEY_E)); KeyMap.push_back(SKeyMap(XK_f, IRR_KEY_F)); KeyMap.push_back(SKeyMap(XK_g, IRR_KEY_G)); KeyMap.push_back(SKeyMap(XK_h, IRR_KEY_H)); KeyMap.push_back(SKeyMap(XK_i, IRR_KEY_I)); KeyMap.push_back(SKeyMap(XK_j, IRR_KEY_J)); KeyMap.push_back(SKeyMap(XK_k, IRR_KEY_K)); KeyMap.push_back(SKeyMap(XK_l, IRR_KEY_L)); KeyMap.push_back(SKeyMap(XK_m, IRR_KEY_M)); KeyMap.push_back(SKeyMap(XK_n, IRR_KEY_N)); KeyMap.push_back(SKeyMap(XK_o, IRR_KEY_O)); KeyMap.push_back(SKeyMap(XK_p, IRR_KEY_P)); KeyMap.push_back(SKeyMap(XK_q, IRR_KEY_Q)); KeyMap.push_back(SKeyMap(XK_r, IRR_KEY_R)); KeyMap.push_back(SKeyMap(XK_s, IRR_KEY_S)); KeyMap.push_back(SKeyMap(XK_t, IRR_KEY_T)); KeyMap.push_back(SKeyMap(XK_u, IRR_KEY_U)); KeyMap.push_back(SKeyMap(XK_v, IRR_KEY_V)); KeyMap.push_back(SKeyMap(XK_w, IRR_KEY_W)); KeyMap.push_back(SKeyMap(XK_x, IRR_KEY_X)); KeyMap.push_back(SKeyMap(XK_y, IRR_KEY_Y)); KeyMap.push_back(SKeyMap(XK_z, IRR_KEY_Z)); KeyMap.push_back(SKeyMap(XK_ssharp, IRR_KEY_OEM_4)); KeyMap.push_back(SKeyMap(XK_adiaeresis, IRR_KEY_OEM_7)); KeyMap.push_back(SKeyMap(XK_odiaeresis, IRR_KEY_OEM_3)); KeyMap.push_back(SKeyMap(XK_udiaeresis, IRR_KEY_OEM_1)); KeyMap.push_back(SKeyMap(XK_Super_L, IRR_KEY_LWIN)); KeyMap.push_back(SKeyMap(XK_Super_R, IRR_KEY_RWIN)); KeyMap.push_back(SKeyMap(XK_agrave, IRR_KEY_AGRAVE)); KeyMap.push_back(SKeyMap(XK_ccedilla, IRR_KEY_CCEDILLA )); KeyMap.push_back(SKeyMap(XK_eacute, IRR_KEY_EACUTE)); KeyMap.push_back(SKeyMap(XK_egrave, IRR_KEY_EGRAVE)); KeyMap.push_back(SKeyMap(XK_ugrave, IRR_KEY_UGRAVE)); KeyMap.push_back(SKeyMap(XK_twosuperior, IRR_KEY_TWOSUPERIOR)); KeyMap.sort(); #endif } bool CIrrDeviceLinux::activateJoysticks(core::array & joystickInfo) { #if defined (_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) joystickInfo.clear(); u32 joystick; for (joystick = 0; joystick < 32; ++joystick) { // The joystick device could be here... core::stringc devName = "/dev/js"; devName += joystick; SJoystickInfo returnInfo; JoystickInfo info; info.fd = open(devName.c_str(), O_RDONLY); if (-1 == info.fd) { // ...but Ubuntu and possibly other distros // create the devices in /dev/input devName = "/dev/input/js"; devName += joystick; info.fd = open(devName.c_str(), O_RDONLY); if (-1 == info.fd) { // and BSD here devName = "/dev/joy"; devName += joystick; info.fd = open(devName.c_str(), O_RDONLY); } } if (-1 == info.fd) continue; #ifdef __FreeBSD__ info.axes=2; info.buttons=2; #elif defined(__linux__) ioctl( info.fd, JSIOCGAXES, &(info.axes) ); ioctl( info.fd, JSIOCGBUTTONS, &(info.buttons) ); fcntl( info.fd, F_SETFL, O_NONBLOCK ); #endif (void)memset(&info.persistentData, 0, sizeof(info.persistentData)); info.persistentData.EventType = irr::EET_JOYSTICK_INPUT_EVENT; info.persistentData.JoystickEvent.Joystick = ActiveJoysticks.size(); // There's no obvious way to determine which (if any) axes represent a POV // hat, so we'll just set it to "not used" and forget about it. info.persistentData.JoystickEvent.POV = 65535; ActiveJoysticks.push_back(info); returnInfo.HasGenericName = false; returnInfo.Joystick = joystick; returnInfo.PovHat = SJoystickInfo::POV_HAT_UNKNOWN; returnInfo.Axes = info.axes; returnInfo.Buttons = info.buttons; #if !defined(__FreeBSD__) && !defined(__OpenBSD__) && !defined(__NetBSD__) char name[80]; ioctl( info.fd, JSIOCGNAME(80), name); returnInfo.Name = name; #endif joystickInfo.push_back(returnInfo); } for (joystick = 0; joystick < joystickInfo.size(); ++joystick) { char logString[256]; (void)sprintf(logString, "Found joystick %u, %u axes, %u buttons '%s'", joystick, joystickInfo[joystick].Axes, joystickInfo[joystick].Buttons, joystickInfo[joystick].Name.c_str()); os::Printer::log(logString, ELL_INFORMATION); } return true; #else return false; #endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ } void CIrrDeviceLinux::pollJoysticks() { #if defined (_IRR_COMPILE_WITH_JOYSTICK_EVENTS_) if (0 == ActiveJoysticks.size()) return; for (u32 j= 0; j< ActiveJoysticks.size(); ++j) { JoystickInfo & info = ActiveJoysticks[j]; bool event_received = false; #ifdef __FreeBSD__ struct joystick js; if (read(info.fd, &js, sizeof(js)) == sizeof(js)) { event_received = true; info.persistentData.JoystickEvent.ButtonStates = js.b1 | (js.b2 << 1); /* should be a two-bit field */ info.persistentData.JoystickEvent.Axis[0] = js.x; /* X axis */ info.persistentData.JoystickEvent.Axis[1] = js.y; /* Y axis */ } #elif defined(__linux__) struct js_event event; while (sizeof(event) == read(info.fd, &event, sizeof(event))) { switch(event.type & ~JS_EVENT_INIT) { case JS_EVENT_BUTTON: if (event.value) { event_received = true; info.persistentData.JoystickEvent.ButtonStates |= (1 << event.number); } else { event_received = true; info.persistentData.JoystickEvent.ButtonStates &= ~(1 << event.number); } break; case JS_EVENT_AXIS: if (event.number < SEvent::SJoystickEvent::NUMBER_OF_AXES) { event_received = true; info.persistentData.JoystickEvent.Axis[event.number] = event.value; } break; default: break; } } #endif // Send an irrlicht joystick event once per ::run() even if no new data were received. (void)postEventFromUser(info.persistentData); #ifdef _IRR_COMPILE_WITH_X11_ if (event_received) { XResetScreenSaver(display); } #endif } #endif // _IRR_COMPILE_WITH_JOYSTICK_EVENTS_ } //! Set the current Gamma Value for the Display bool CIrrDeviceLinux::setGammaRamp( f32 red, f32 green, f32 blue, f32 brightness, f32 contrast ) { #if defined(_IRR_LINUX_X11_VIDMODE_) || defined(_IRR_LINUX_X11_RANDR_) s32 eventbase, errorbase; #ifdef _IRR_LINUX_X11_VIDMODE_ if (XF86VidModeQueryExtension(display, &eventbase, &errorbase)) { XF86VidModeGamma gamma; gamma.red=red; gamma.green=green; gamma.blue=blue; XF86VidModeSetGamma(display, screennr, &gamma); return true; } #endif #if defined(_IRR_LINUX_X11_VIDMODE_) && defined(_IRR_LINUX_X11_RANDR_) else #endif #ifdef _IRR_LINUX_X11_RANDR_ if (XRRQueryExtension(display, &eventbase, &errorbase)) { XRRQueryVersion(display, &eventbase, &errorbase); // major, minor if (eventbase>=1 && errorbase>1) { #if (RANDR_MAJOR>1 || RANDR_MINOR>1) XRRCrtcGamma *gamma = XRRGetCrtcGamma(display, screennr); if (gamma) { *gamma->red=(u16)red; *gamma->green=(u16)green; *gamma->blue=(u16)blue; XRRSetCrtcGamma(display, screennr, gamma); XRRFreeGamma(gamma); return true; } #endif } } #endif #endif return false; } //! Get the current Gamma Value for the Display bool CIrrDeviceLinux::getGammaRamp( f32 &red, f32 &green, f32 &blue, f32 &brightness, f32 &contrast ) { brightness = 0.f; contrast = 0.f; #if defined(_IRR_LINUX_X11_VIDMODE_) || defined(_IRR_LINUX_X11_RANDR_) s32 eventbase, errorbase; #ifdef _IRR_LINUX_X11_VIDMODE_ if (XF86VidModeQueryExtension(display, &eventbase, &errorbase)) { XF86VidModeGamma gamma; XF86VidModeGetGamma(display, screennr, &gamma); red = gamma.red; green = gamma.green; blue = gamma.blue; return true; } #endif #if defined(_IRR_LINUX_X11_VIDMODE_) && defined(_IRR_LINUX_X11_RANDR_) else #endif #ifdef _IRR_LINUX_X11_RANDR_ if (XRRQueryExtension(display, &eventbase, &errorbase)) { XRRQueryVersion(display, &eventbase, &errorbase); // major, minor if (eventbase>=1 && errorbase>1) { #if (RANDR_MAJOR>1 || RANDR_MINOR>1) XRRCrtcGamma *gamma = XRRGetCrtcGamma(display, screennr); if (gamma) { red = *gamma->red; green = *gamma->green; blue= *gamma->blue; XRRFreeGamma(gamma); return true; } #endif } } #endif #endif return false; } //! gets text from the clipboard //! \return Returns 0 if no string is in there. const c8* CIrrDeviceLinux::getTextFromClipboard() const { #if defined(_IRR_COMPILE_WITH_X11_) if (X_ATOM_CLIPBOARD == None) { os::Printer::log("Couldn't access X clipboard", ELL_WARNING); return 0; } Window ownerWindow = XGetSelectionOwner(display, X_ATOM_CLIPBOARD); if (ownerWindow == window) { return Clipboard.c_str(); } Clipboard = ""; if (ownerWindow == None) return 0; Atom selection = XInternAtom(display, "IRR_SELECTION", False); XConvertSelection(display, X_ATOM_CLIPBOARD, X_ATOM_UTF8_STRING, selection, window, CurrentTime); const int SELECTION_RETRIES = 500; int i = 0; for (i = 0; i < SELECTION_RETRIES; i++) { XEvent xevent; bool res = XCheckTypedWindowEvent(display, window, SelectionNotify, &xevent); if (res && xevent.xselection.selection == X_ATOM_CLIPBOARD) break; usleep(1000); } if (i == SELECTION_RETRIES) { os::Printer::log("Timed out waiting for SelectionNotify event", ELL_WARNING); return 0; } Atom type; int format; unsigned long numItems, dummy; unsigned char *data; int result = XGetWindowProperty(display, window, selection, 0, INT_MAX/4, False, AnyPropertyType, &type, &format, &numItems, &dummy, &data); if (result == Success) { Clipboard = (irr::c8*)data; XFree(data); } return Clipboard.c_str(); #else return 0; #endif } //! copies text to the clipboard void CIrrDeviceLinux::copyToClipboard(const c8* text) const { #if defined(_IRR_COMPILE_WITH_X11_) // Actually there is no clipboard on X but applications just say they own the clipboard and return text when asked. // Which btw. also means that on X you lose clipboard content when closing applications. Clipboard = text; XSetSelectionOwner (display, X_ATOM_CLIPBOARD, window, CurrentTime); XFlush (display); #endif } #ifdef _IRR_COMPILE_WITH_X11_ // return true if the passed event has the type passed in parameter arg Bool PredicateIsEventType(Display *display, XEvent *event, XPointer arg) { if ( event && event->type == *(int*)arg ) { // os::Printer::log("remove event:", core::stringc((int)arg).c_str(), ELL_INFORMATION); return True; } return False; } #endif //_IRR_COMPILE_WITH_X11_ //! Remove all messages pending in the system message loop void CIrrDeviceLinux::clearSystemMessages() { #ifdef _IRR_COMPILE_WITH_X11_ if (CreationParams.DriverType != video::EDT_NULL) { XEvent event; int usrArg = ButtonPress; while ( XCheckIfEvent(display, &event, PredicateIsEventType, XPointer(&usrArg)) == True ) {} usrArg = ButtonRelease; while ( XCheckIfEvent(display, &event, PredicateIsEventType, XPointer(&usrArg)) == True ) {} usrArg = MotionNotify; while ( XCheckIfEvent(display, &event, PredicateIsEventType, XPointer(&usrArg)) == True ) {} usrArg = KeyRelease; while ( XCheckIfEvent(display, &event, PredicateIsEventType, XPointer(&usrArg)) == True ) {} usrArg = KeyPress; while ( XCheckIfEvent(display, &event, PredicateIsEventType, XPointer(&usrArg)) == True ) {} } #endif //_IRR_COMPILE_WITH_X11_ } void CIrrDeviceLinux::initXAtoms() { #ifdef _IRR_COMPILE_WITH_X11_ X_ATOM_CLIPBOARD = XInternAtom(display, "CLIPBOARD", False); X_ATOM_TARGETS = XInternAtom(display, "TARGETS", False); X_ATOM_UTF8_STRING = XInternAtom (display, "UTF8_STRING", False); #endif } #ifdef _IRR_COMPILE_WITH_X11_ Cursor CIrrDeviceLinux::TextureToMonochromeCursor(irr::video::ITexture * tex, const core::rect& sourceRect, const core::position2d &hotspot) { XImage * sourceImage = XCreateImage(display, visual->visual, 1, // depth, ZPixmap, // XYBitmap (depth=1), ZPixmap(depth=x) 0, 0, sourceRect.getWidth(), sourceRect.getHeight(), 32, // bitmap_pad, 0// bytes_per_line (0 means continuos in memory) ); sourceImage->data = new char[sourceImage->height * sourceImage->bytes_per_line]; XImage * maskImage = XCreateImage(display, visual->visual, 1, // depth, ZPixmap, 0, 0, sourceRect.getWidth(), sourceRect.getHeight(), 32, // bitmap_pad, 0 // bytes_per_line ); maskImage->data = new char[maskImage->height * maskImage->bytes_per_line]; // write texture into XImage video::ECOLOR_FORMAT format = tex->getColorFormat(); u32 bytesPerPixel = video::IImage::getBitsPerPixelFromFormat(format) / 8; u32 bytesLeftGap = sourceRect.UpperLeftCorner.X * bytesPerPixel; u32 bytesRightGap = tex->getPitch() - sourceRect.LowerRightCorner.X * bytesPerPixel; const u8* data = (const u8*)tex->lock(video::ETLM_READ_ONLY, 0); data += sourceRect.UpperLeftCorner.Y*tex->getPitch(); for ( s32 y = 0; y < sourceRect.getHeight(); ++y ) { data += bytesLeftGap; for ( s32 x = 0; x < sourceRect.getWidth(); ++x ) { video::SColor pixelCol; pixelCol.setData((const void*)data, format); data += bytesPerPixel; if ( pixelCol.getAlpha() == 0 ) // transparent { XPutPixel(maskImage, x, y, 0); XPutPixel(sourceImage, x, y, 0); } else // color { if ( pixelCol.getAverage() >= 127 ) XPutPixel(sourceImage, x, y, 1); else XPutPixel(sourceImage, x, y, 0); XPutPixel(maskImage, x, y, 1); } } data += bytesRightGap; } tex->unlock(); Pixmap sourcePixmap = XCreatePixmap(display, window, sourceImage->width, sourceImage->height, sourceImage->depth); Pixmap maskPixmap = XCreatePixmap(display, window, maskImage->width, maskImage->height, maskImage->depth); XGCValues values; values.foreground = 1; values.background = 1; GC gc = XCreateGC( display, sourcePixmap, GCForeground | GCBackground, &values ); XPutImage(display, sourcePixmap, gc, sourceImage, 0, 0, 0, 0, sourceImage->width, sourceImage->height); XPutImage(display, maskPixmap, gc, maskImage, 0, 0, 0, 0, maskImage->width, maskImage->height); XFreeGC(display, gc); XDestroyImage(sourceImage); XDestroyImage(maskImage); Cursor cursorResult = 0; XColor foreground, background; foreground.red = 65535; foreground.green = 65535; foreground.blue = 65535; foreground.flags = DoRed | DoGreen | DoBlue; background.red = 0; background.green = 0; background.blue = 0; background.flags = DoRed | DoGreen | DoBlue; cursorResult = XCreatePixmapCursor(display, sourcePixmap, maskPixmap, &foreground, &background, hotspot.X, hotspot.Y); XFreePixmap(display, sourcePixmap); XFreePixmap(display, maskPixmap); return cursorResult; } #ifdef _IRR_LINUX_XCURSOR_ Cursor CIrrDeviceLinux::TextureToARGBCursor(irr::video::ITexture * tex, const core::rect& sourceRect, const core::position2d &hotspot) { XcursorImage * image = XcursorImageCreate (sourceRect.getWidth(), sourceRect.getHeight()); image->xhot = hotspot.X; image->yhot = hotspot.Y; // write texture into XcursorImage video::ECOLOR_FORMAT format = tex->getColorFormat(); u32 bytesPerPixel = video::IImage::getBitsPerPixelFromFormat(format) / 8; u32 bytesLeftGap = sourceRect.UpperLeftCorner.X * bytesPerPixel; u32 bytesRightGap = tex->getPitch() - sourceRect.LowerRightCorner.X * bytesPerPixel; XcursorPixel* target = image->pixels; const u8* data = (const u8*)tex->lock(ETLM_READ_ONLY, 0); data += sourceRect.UpperLeftCorner.Y*tex->getPitch(); for ( s32 y = 0; y < sourceRect.getHeight(); ++y ) { data += bytesLeftGap; for ( s32 x = 0; x < sourceRect.getWidth(); ++x ) { video::SColor pixelCol; pixelCol.setData((const void*)data, format); data += bytesPerPixel; *target = (XcursorPixel)pixelCol.color; ++target; } data += bytesRightGap; } tex->unlock(); Cursor cursorResult=XcursorImageLoadCursor(display, image); XcursorImageDestroy(image); return cursorResult; } #endif // #ifdef _IRR_LINUX_XCURSOR_ Cursor CIrrDeviceLinux::TextureToCursor(irr::video::ITexture * tex, const core::rect& sourceRect, const core::position2d &hotspot) { #ifdef _IRR_LINUX_XCURSOR_ return TextureToARGBCursor( tex, sourceRect, hotspot ); #else return TextureToMonochromeCursor( tex, sourceRect, hotspot ); #endif } #endif // _IRR_COMPILE_WITH_X11_ CIrrDeviceLinux::CCursorControl::CCursorControl(CIrrDeviceLinux* dev, bool null) : Device(dev) #ifdef _IRR_COMPILE_WITH_X11_ , PlatformBehavior(gui::ECPB_NONE), lastQuery(0) #endif , IsVisible(true), Null(null), UseReferenceRect(false) , ActiveIcon(gui::ECI_NORMAL), ActiveIconStartTime(0) { #ifdef _IRR_COMPILE_WITH_X11_ if (!Null) { XGCValues values; unsigned long valuemask = 0; XColor fg, bg; // this code, for making the cursor invisible was sent in by // Sirshane, thank your very much! Pixmap invisBitmap = XCreatePixmap(Device->display, Device->window, 32, 32, 1); Pixmap maskBitmap = XCreatePixmap(Device->display, Device->window, 32, 32, 1); Colormap screen_colormap = DefaultColormap( Device->display, DefaultScreen( Device->display ) ); XAllocNamedColor( Device->display, screen_colormap, "black", &fg, &fg ); XAllocNamedColor( Device->display, screen_colormap, "white", &bg, &bg ); GC gc = XCreateGC( Device->display, invisBitmap, valuemask, &values ); XSetForeground( Device->display, gc, BlackPixel( Device->display, DefaultScreen( Device->display ) ) ); XFillRectangle( Device->display, invisBitmap, gc, 0, 0, 32, 32 ); XFillRectangle( Device->display, maskBitmap, gc, 0, 0, 32, 32 ); invisCursor = XCreatePixmapCursor( Device->display, invisBitmap, maskBitmap, &fg, &bg, 1, 1 ); XFreeGC(Device->display, gc); XFreePixmap(Device->display, invisBitmap); XFreePixmap(Device->display, maskBitmap); initCursors(); } #endif } CIrrDeviceLinux::CCursorControl::~CCursorControl() { // Do not clearCursors here as the display is already closed // TODO (cutealien): droping cursorcontrol earlier might work, not sure about reason why that's done in stub currently. } #ifdef _IRR_COMPILE_WITH_X11_ void CIrrDeviceLinux::CCursorControl::clearCursors() { if (!Null) XFreeCursor(Device->display, invisCursor); for ( u32 i=0; i < Cursors.size(); ++i ) { for ( u32 f=0; f < Cursors[i].Frames.size(); ++f ) { XFreeCursor(Device->display, Cursors[i].Frames[f].IconHW); } } } void CIrrDeviceLinux::CCursorControl::initCursors() { Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_top_left_arrow)) ); // (or XC_arrow?) Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_crosshair)) ); Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_hand2)) ); // (or XC_hand1? ) Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_question_arrow)) ); Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_xterm)) ); Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_X_cursor)) ); // (or XC_pirate?) Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_watch)) ); // (or XC_clock?) Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_fleur)) ); Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_top_right_corner)) ); // NESW not available in X11 Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_top_left_corner)) ); // NWSE not available in X11 Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_sb_v_double_arrow)) ); Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_sb_h_double_arrow)) ); Cursors.push_back( CursorX11(XCreateFontCursor(Device->display, XC_sb_up_arrow)) ); // (or XC_center_ptr?) } void CIrrDeviceLinux::CCursorControl::update() { if ( (u32)ActiveIcon < Cursors.size() && !Cursors[ActiveIcon].Frames.empty() && Cursors[ActiveIcon].FrameTime ) { // update animated cursors. This could also be done by X11 in case someone wants to figure that out (this way was just easier to implement) u32 now = Device->getTimer()->getRealTime(); u32 frame = ((now - ActiveIconStartTime) / Cursors[ActiveIcon].FrameTime) % Cursors[ActiveIcon].Frames.size(); XDefineCursor(Device->display, Device->window, Cursors[ActiveIcon].Frames[frame].IconHW); } } #endif //! Sets the active cursor icon void CIrrDeviceLinux::CCursorControl::setActiveIcon(gui::ECURSOR_ICON iconId) { #ifdef _IRR_COMPILE_WITH_X11_ if ( iconId >= (s32)Cursors.size() ) return; if ( Cursors[iconId].Frames.size() ) XDefineCursor(Device->display, Device->window, Cursors[iconId].Frames[0].IconHW); ActiveIconStartTime = Device->getTimer()->getRealTime(); ActiveIcon = iconId; #endif } //! Add a custom sprite as cursor icon. gui::ECURSOR_ICON CIrrDeviceLinux::CCursorControl::addIcon(const gui::SCursorSprite& icon) { #ifdef _IRR_COMPILE_WITH_X11_ if ( icon.SpriteId >= 0 ) { CursorX11 cX11; cX11.FrameTime = icon.SpriteBank->getSprites()[icon.SpriteId].frameTime; for ( u32 i=0; i < icon.SpriteBank->getSprites()[icon.SpriteId].Frames.size(); ++i ) { irr::u32 texId = icon.SpriteBank->getSprites()[icon.SpriteId].Frames[i].textureNumber; irr::u32 rectId = icon.SpriteBank->getSprites()[icon.SpriteId].Frames[i].rectNumber; irr::core::rect rectIcon = icon.SpriteBank->getPositions()[rectId]; Cursor cursor = Device->TextureToCursor(icon.SpriteBank->getTexture(texId), rectIcon, icon.HotSpot); cX11.Frames.push_back( CursorFrameX11(cursor) ); } Cursors.push_back( cX11 ); return (gui::ECURSOR_ICON)(Cursors.size() - 1); } #endif return gui::ECI_NORMAL; } //! replace the given cursor icon. void CIrrDeviceLinux::CCursorControl::changeIcon(gui::ECURSOR_ICON iconId, const gui::SCursorSprite& icon) { #ifdef _IRR_COMPILE_WITH_X11_ if ( iconId >= (s32)Cursors.size() ) return; for ( u32 i=0; i < Cursors[iconId].Frames.size(); ++i ) XFreeCursor(Device->display, Cursors[iconId].Frames[i].IconHW); if ( icon.SpriteId >= 0 ) { CursorX11 cX11; cX11.FrameTime = icon.SpriteBank->getSprites()[icon.SpriteId].frameTime; for ( u32 i=0; i < icon.SpriteBank->getSprites()[icon.SpriteId].Frames.size(); ++i ) { irr::u32 texId = icon.SpriteBank->getSprites()[icon.SpriteId].Frames[i].textureNumber; irr::u32 rectId = icon.SpriteBank->getSprites()[icon.SpriteId].Frames[i].rectNumber; irr::core::rect rectIcon = icon.SpriteBank->getPositions()[rectId]; Cursor cursor = Device->TextureToCursor(icon.SpriteBank->getTexture(texId), rectIcon, icon.HotSpot); cX11.Frames.push_back( CursorFrameX11(cursor) ); } Cursors[iconId] = cX11; } #endif } irr::core::dimension2di CIrrDeviceLinux::CCursorControl::getSupportedIconSize() const { // this returns the closest match that is smaller or same size, so we just pass a value which should be large enough for cursors unsigned int width=0, height=0; #ifdef _IRR_COMPILE_WITH_X11_ XQueryBestCursor(Device->display, Device->window, 64, 64, &width, &height); #endif return core::dimension2di(width, height); } } // end namespace #endif // _IRR_COMPILE_WITH_X11_DEVICE_