mirror of
https://github.com/minetest/irrlicht.git
synced 2024-12-27 00:17:30 +01:00
9062ac91af
This changes the behaviour on Win32 somewhat when Windows returned a CURSOR_SUPPRESSED state (touch-screen input hiding cursor globally). Previously we set IsVisible it to false when CURSOR_SUPPRESSED was set. Also we handle the CURSOR_SUPPRESSED state slightly different now and still try to hide cursors once when requested. Reason for the change is that the old behaviour made it harder to recover from touch-screens hiding the cursor because Irrlicht didn't know anymore which state is _should_ have. This also unifies the behaviour on all drivers as the other drivers already returned the visible flag independent of the system being able to actually show the cursor. git-svn-id: svn://svn.code.sf.net/p/irrlicht/code/trunk@6109 dfc29bdd-3216-0410-991c-e03cc46cb475
448 lines
13 KiB
C++
448 lines
13 KiB
C++
// Copyright (C) 2002-2012 Nikolaus Gebhardt
|
|
// This file is part of the "Irrlicht Engine".
|
|
// For conditions of distribution and use, see copyright notice in irrlicht.h
|
|
|
|
#ifndef __C_IRR_DEVICE_WIN32_H_INCLUDED__
|
|
#define __C_IRR_DEVICE_WIN32_H_INCLUDED__
|
|
|
|
#include "IrrCompileConfig.h"
|
|
#ifdef _IRR_COMPILE_WITH_WINDOWS_DEVICE_
|
|
|
|
#include "CIrrDeviceStub.h"
|
|
#include "IrrlichtDevice.h"
|
|
#include "IImagePresenter.h"
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
#if !defined(_IRR_XBOX_PLATFORM_)
|
|
#include <windows.h>
|
|
#include <mmsystem.h> // For JOYCAPS
|
|
#include <windowsx.h>
|
|
#endif
|
|
#if !defined(GET_X_LPARAM)
|
|
#define GET_X_LPARAM(lp) ((int)(short)LOWORD(lp))
|
|
#define GET_Y_LPARAM(lp) ((int)(short)HIWORD(lp))
|
|
#endif
|
|
|
|
namespace irr
|
|
{
|
|
struct SJoystickWin32Control;
|
|
|
|
class CIrrDeviceWin32 : public CIrrDeviceStub, video::IImagePresenter
|
|
{
|
|
friend struct SJoystickWin32Control;
|
|
public:
|
|
|
|
//! constructor
|
|
CIrrDeviceWin32(const SIrrlichtCreationParameters& params);
|
|
|
|
//! destructor
|
|
virtual ~CIrrDeviceWin32();
|
|
|
|
//! runs the device. Returns false if device wants to be deleted
|
|
virtual bool run() _IRR_OVERRIDE_;
|
|
|
|
//! Cause the device to temporarily pause execution and let other processes to run
|
|
// This should bring down processor usage without major performance loss for Irrlicht
|
|
virtual void yield() _IRR_OVERRIDE_;
|
|
|
|
//! Pause execution and let other processes to run for a specified amount of time.
|
|
virtual void sleep(u32 timeMs, bool pauseTimer) _IRR_OVERRIDE_;
|
|
|
|
//! sets the caption of the window
|
|
virtual void setWindowCaption(const wchar_t* text) _IRR_OVERRIDE_;
|
|
|
|
//! returns if window is active. if not, nothing need to be drawn
|
|
virtual bool isWindowActive() const _IRR_OVERRIDE_;
|
|
|
|
//! returns if window has focus
|
|
virtual bool isWindowFocused() const _IRR_OVERRIDE_;
|
|
|
|
//! returns if window is minimized
|
|
virtual bool isWindowMinimized() const _IRR_OVERRIDE_;
|
|
|
|
//! presents a surface in the client area
|
|
virtual bool present(video::IImage* surface, void* windowId=0, core::rect<s32>* src=0) _IRR_OVERRIDE_;
|
|
|
|
//! notifies the device that it should close itself
|
|
virtual void closeDevice() _IRR_OVERRIDE_;
|
|
|
|
//! \return Returns a pointer to a list with all video modes
|
|
//! supported by the gfx adapter.
|
|
virtual video::IVideoModeList* getVideoModeList() _IRR_OVERRIDE_;
|
|
|
|
//! Notifies the device, that it has been resized
|
|
/** Must be publis as it is called from free function (event handler) */
|
|
void OnResized();
|
|
|
|
//! Sets if the window should be resizable in windowed mode.
|
|
virtual void setResizable(bool resize=false) _IRR_OVERRIDE_;
|
|
|
|
//! Resize the render window.
|
|
virtual void setWindowSize(const irr::core::dimension2d<u32>& size) _IRR_OVERRIDE_;
|
|
|
|
//! Minimizes the window.
|
|
virtual void minimizeWindow() _IRR_OVERRIDE_;
|
|
|
|
//! Maximizes the window.
|
|
virtual void maximizeWindow() _IRR_OVERRIDE_;
|
|
|
|
//! Restores the window size.
|
|
virtual void restoreWindow() _IRR_OVERRIDE_;
|
|
|
|
//! Get the position of the window on screen
|
|
virtual core::position2di getWindowPosition() _IRR_OVERRIDE_;
|
|
|
|
//! Activate any joysticks, and generate events for them.
|
|
virtual bool activateJoysticks(core::array<SJoystickInfo> & joystickInfo) _IRR_OVERRIDE_;
|
|
|
|
//! Set the current Gamma Value for the Display
|
|
virtual bool setGammaRamp( f32 red, f32 green, f32 blue, f32 brightness, f32 contrast ) _IRR_OVERRIDE_;
|
|
|
|
//! Get the current Gamma Value for the Display
|
|
virtual bool getGammaRamp( f32 &red, f32 &green, f32 &blue, f32 &brightness, f32 &contrast ) _IRR_OVERRIDE_;
|
|
|
|
//! Remove all messages pending in the system message loop
|
|
virtual void clearSystemMessages() _IRR_OVERRIDE_;
|
|
|
|
//! Get the device type
|
|
virtual E_DEVICE_TYPE getType() const _IRR_OVERRIDE_
|
|
{
|
|
return EIDT_WIN32;
|
|
}
|
|
|
|
//! Compares to the last call of this function to return double and triple clicks.
|
|
//! \return Returns only 1,2 or 3. A 4th click will start with 1 again.
|
|
virtual u32 checkSuccessiveClicks(s32 mouseX, s32 mouseY, EMOUSE_INPUT_EVENT inputEvent ) _IRR_OVERRIDE_
|
|
{
|
|
// we just have to make it public
|
|
return CIrrDeviceStub::checkSuccessiveClicks(mouseX, mouseY, inputEvent );
|
|
}
|
|
|
|
//! Switch to fullscreen
|
|
bool switchToFullScreen(bool reset=false);
|
|
|
|
//! Check for and show last Windows API error to help internal debugging.
|
|
//! Does call GetLastError and on errors formats the error text and displays it in a messagebox.
|
|
static void ReportLastWinApiError();
|
|
|
|
//! Same function Windows offers in VersionHelpers.h, but we can't use that as it's not available before SDK 8.1
|
|
static bool isWindowsVistaOrGreater();
|
|
|
|
// convert an Irrlicht texture to a windows cursor
|
|
HCURSOR TextureToCursor(HWND hwnd, irr::video::ITexture * tex, const core::rect<s32>& sourceRect, const core::position2d<s32> &hotspot);
|
|
|
|
//! Implementation of the win32 cursor control
|
|
class CCursorControl : public gui::ICursorControl
|
|
{
|
|
public:
|
|
|
|
CCursorControl(CIrrDeviceWin32* device, const core::dimension2d<u32>& wsize, HWND hwnd, bool fullscreen);
|
|
~CCursorControl();
|
|
|
|
//! Changes the visible state of the mouse cursor.
|
|
virtual void setVisible(bool visible) _IRR_OVERRIDE_
|
|
{
|
|
CURSORINFO info;
|
|
info.cbSize = sizeof(CURSORINFO);
|
|
BOOL gotCursorInfo = GetCursorInfo(&info);
|
|
while ( gotCursorInfo )
|
|
{
|
|
#ifdef CURSOR_SUPPRESSED
|
|
// Since Windows 8 the cursor can be suppressed by a touch interface
|
|
if (visible && info.flags == CURSOR_SUPPRESSED)
|
|
{
|
|
break;
|
|
}
|
|
#endif
|
|
if ( (visible && info.flags == CURSOR_SHOWING) || // visible
|
|
(!visible && info.flags == 0 ) ) // hidden
|
|
{
|
|
break;
|
|
}
|
|
// this only increases an internal
|
|
// display counter in windows, so it
|
|
// might have to be called some more
|
|
const int showResult = ShowCursor(visible);
|
|
// if result has correct sign we can
|
|
// stop here as well
|
|
if (( !visible && showResult < 0 ) ||
|
|
(visible && showResult >= 0))
|
|
break;
|
|
// yes, it really must be set each time
|
|
info.cbSize = sizeof(CURSORINFO);
|
|
gotCursorInfo = GetCursorInfo(&info);
|
|
|
|
#ifdef CURSOR_SUPPRESSED
|
|
// Not sure if a cursor which we tried to hide still can be suppressed.
|
|
// I have no touch-display for testing this and MSDN doesn't describe it.
|
|
// But adding this check shouldn't hurt and might prevent an endless loop.
|
|
if (!visible && info.flags == CURSOR_SUPPRESSED)
|
|
{
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
IsVisible = visible;
|
|
}
|
|
|
|
//! Returns if the cursor is currently visible.
|
|
virtual bool isVisible() const _IRR_OVERRIDE_
|
|
{
|
|
return IsVisible;
|
|
}
|
|
|
|
//! Sets the new position of the cursor.
|
|
virtual void setPosition(const core::position2d<f32> &pos) _IRR_OVERRIDE_
|
|
{
|
|
setPosition(pos.X, pos.Y);
|
|
}
|
|
|
|
//! Sets the new position of the cursor.
|
|
virtual void setPosition(f32 x, f32 y) _IRR_OVERRIDE_
|
|
{
|
|
if (!UseReferenceRect)
|
|
setPosition(core::round32(x*WindowSize.Width), core::round32(y*WindowSize.Height));
|
|
else
|
|
setPosition(core::round32(x*ReferenceRect.getWidth()), core::round32(y*ReferenceRect.getHeight()));
|
|
}
|
|
|
|
//! Sets the new position of the cursor.
|
|
virtual void setPosition(const core::position2d<s32> &pos) _IRR_OVERRIDE_
|
|
{
|
|
setPosition(pos.X, pos.Y);
|
|
}
|
|
|
|
//! Sets the new position of the cursor.
|
|
virtual void setPosition(s32 x, s32 y) _IRR_OVERRIDE_
|
|
{
|
|
if (UseReferenceRect)
|
|
{
|
|
SetCursorPos(ReferenceRect.UpperLeftCorner.X + x,
|
|
ReferenceRect.UpperLeftCorner.Y + y);
|
|
}
|
|
else
|
|
{
|
|
RECT rect;
|
|
if (GetWindowRect(HWnd, &rect))
|
|
SetCursorPos(x + rect.left + BorderX, y + rect.top + BorderY);
|
|
}
|
|
|
|
CursorPos.X = x;
|
|
CursorPos.Y = y;
|
|
}
|
|
|
|
//! Returns the current position of the mouse cursor.
|
|
virtual const core::position2d<s32>& getPosition(bool updateCursor) _IRR_OVERRIDE_
|
|
{
|
|
if ( updateCursor )
|
|
updateInternalCursorPosition();
|
|
return CursorPos;
|
|
}
|
|
|
|
//! Returns the current position of the mouse cursor.
|
|
virtual core::position2d<f32> getRelativePosition(bool updateCursor) _IRR_OVERRIDE_
|
|
{
|
|
if ( updateCursor )
|
|
updateInternalCursorPosition();
|
|
|
|
if (!UseReferenceRect)
|
|
{
|
|
return core::position2d<f32>(CursorPos.X * InvWindowSize.Width,
|
|
CursorPos.Y * InvWindowSize.Height);
|
|
}
|
|
|
|
return core::position2d<f32>(CursorPos.X / (f32)ReferenceRect.getWidth(),
|
|
CursorPos.Y / (f32)ReferenceRect.getHeight());
|
|
}
|
|
|
|
//! Sets an absolute reference rect for calculating the cursor position.
|
|
virtual void setReferenceRect(core::rect<s32>* rect=0) _IRR_OVERRIDE_
|
|
{
|
|
if (rect)
|
|
{
|
|
ReferenceRect = *rect;
|
|
UseReferenceRect = true;
|
|
|
|
// prevent division through zero and uneven sizes
|
|
|
|
if (!ReferenceRect.getHeight() || ReferenceRect.getHeight()%2)
|
|
ReferenceRect.LowerRightCorner.Y += 1;
|
|
|
|
if (!ReferenceRect.getWidth() || ReferenceRect.getWidth()%2)
|
|
ReferenceRect.LowerRightCorner.X += 1;
|
|
}
|
|
else
|
|
UseReferenceRect = false;
|
|
}
|
|
|
|
/** Used to notify the cursor that the window was resized. */
|
|
void OnResize(const core::dimension2d<u32>& size)
|
|
{
|
|
WindowSize = size;
|
|
if (size.Width!=0)
|
|
InvWindowSize.Width = 1.0f / size.Width;
|
|
else
|
|
InvWindowSize.Width = 0.f;
|
|
|
|
if (size.Height!=0)
|
|
InvWindowSize.Height = 1.0f / size.Height;
|
|
else
|
|
InvWindowSize.Height = 0.f;
|
|
}
|
|
|
|
/** Used to notify the cursor that the window resizable settings changed. */
|
|
void updateBorderSize(bool fullscreen, bool resizable)
|
|
{
|
|
if (!fullscreen)
|
|
{
|
|
s32 paddingBorder = 0;
|
|
#if defined (SM_CXPADDEDBORDER)
|
|
if (CIrrDeviceWin32::isWindowsVistaOrGreater())
|
|
paddingBorder = GetSystemMetrics(SM_CXPADDEDBORDER);
|
|
#endif
|
|
|
|
if (resizable)
|
|
{
|
|
BorderX = GetSystemMetrics(SM_CXSIZEFRAME) + paddingBorder;
|
|
BorderY = GetSystemMetrics(SM_CYCAPTION) + GetSystemMetrics(SM_CYSIZEFRAME) + paddingBorder;
|
|
}
|
|
else
|
|
{
|
|
BorderX = GetSystemMetrics(SM_CXDLGFRAME) + paddingBorder;
|
|
BorderY = GetSystemMetrics(SM_CYCAPTION) + GetSystemMetrics(SM_CYDLGFRAME) + paddingBorder;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
BorderX = BorderY = 0;
|
|
}
|
|
}
|
|
|
|
|
|
//! Sets the active cursor icon
|
|
virtual void setActiveIcon(gui::ECURSOR_ICON iconId) _IRR_OVERRIDE_;
|
|
|
|
//! Gets the currently active icon
|
|
virtual gui::ECURSOR_ICON getActiveIcon() const _IRR_OVERRIDE_
|
|
{
|
|
return ActiveIcon;
|
|
}
|
|
|
|
//! Add a custom sprite as cursor icon.
|
|
virtual gui::ECURSOR_ICON addIcon(const gui::SCursorSprite& icon) _IRR_OVERRIDE_;
|
|
|
|
//! replace the given cursor icon.
|
|
virtual void changeIcon(gui::ECURSOR_ICON iconId, const gui::SCursorSprite& icon) _IRR_OVERRIDE_;
|
|
|
|
//! Return a system-specific size which is supported for cursors. Larger icons will fail, smaller icons might work.
|
|
virtual core::dimension2di getSupportedIconSize() const _IRR_OVERRIDE_;
|
|
|
|
void update();
|
|
|
|
private:
|
|
|
|
//! Updates the internal cursor position
|
|
void updateInternalCursorPosition()
|
|
{
|
|
POINT p;
|
|
if (!GetCursorPos(&p))
|
|
{
|
|
DWORD xy = GetMessagePos();
|
|
p.x = GET_X_LPARAM(xy);
|
|
p.y = GET_Y_LPARAM(xy);
|
|
}
|
|
|
|
if (UseReferenceRect)
|
|
{
|
|
CursorPos.X = p.x - ReferenceRect.UpperLeftCorner.X;
|
|
CursorPos.Y = p.y - ReferenceRect.UpperLeftCorner.Y;
|
|
}
|
|
else
|
|
{
|
|
RECT rect;
|
|
if (GetWindowRect(HWnd, &rect))
|
|
{
|
|
CursorPos.X = p.x-rect.left-BorderX;
|
|
CursorPos.Y = p.y-rect.top-BorderY;
|
|
}
|
|
else
|
|
{
|
|
// window seems not to be existent, so set cursor to
|
|
// a negative value
|
|
CursorPos.X = -1;
|
|
CursorPos.Y = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
CIrrDeviceWin32* Device;
|
|
core::position2d<s32> CursorPos;
|
|
core::dimension2d<u32> WindowSize;
|
|
core::dimension2d<f32> InvWindowSize;
|
|
HWND HWnd;
|
|
|
|
s32 BorderX, BorderY;
|
|
core::rect<s32> ReferenceRect;
|
|
bool UseReferenceRect;
|
|
bool IsVisible;
|
|
|
|
|
|
struct CursorFrameW32
|
|
{
|
|
CursorFrameW32() : IconHW(0) {}
|
|
CursorFrameW32(HCURSOR icon) : IconHW(icon) {}
|
|
|
|
HCURSOR IconHW; // hardware cursor
|
|
};
|
|
|
|
struct CursorW32
|
|
{
|
|
CursorW32() {}
|
|
explicit CursorW32(HCURSOR iconHw, u32 frameTime=0) : FrameTime(frameTime)
|
|
{
|
|
Frames.push_back( CursorFrameW32(iconHw) );
|
|
}
|
|
core::array<CursorFrameW32> Frames;
|
|
u32 FrameTime;
|
|
};
|
|
|
|
core::array<CursorW32> Cursors;
|
|
gui::ECURSOR_ICON ActiveIcon;
|
|
u32 ActiveIconStartTime;
|
|
|
|
void initCursors();
|
|
};
|
|
|
|
//! returns the win32 cursor control
|
|
CCursorControl* getWin32CursorControl();
|
|
|
|
private:
|
|
|
|
//! create the driver
|
|
void createDriver();
|
|
|
|
//! Process system events
|
|
void handleSystemMessages();
|
|
|
|
void getWindowsVersion(core::stringc& version);
|
|
|
|
void resizeIfNecessary();
|
|
|
|
DWORD getWindowStyle(bool fullscreen, bool resizable) const;
|
|
|
|
HWND HWnd;
|
|
|
|
bool ChangedToFullScreen;
|
|
bool Resized;
|
|
bool ExternalWindow;
|
|
CCursorControl* Win32CursorControl;
|
|
DEVMODE DesktopMode;
|
|
|
|
SJoystickWin32Control* JoyControl;
|
|
};
|
|
|
|
} // end namespace irr
|
|
|
|
#endif // _IRR_COMPILE_WITH_WINDOWS_DEVICE_
|
|
#endif // __C_IRR_DEVICE_WIN32_H_INCLUDED__
|