Compare commits
1 commit
master
...
feature/wi
Author | SHA1 | Date | |
---|---|---|---|
![]() |
b301a7beb4 |
|
@ -35,6 +35,7 @@
|
|||
#define _WIN32_WINNT 0x0501
|
||||
#include <SFML/Window/Window.hpp>
|
||||
#include <SFML/Window/Win32/InputImpl.hpp>
|
||||
#include <SFML/Window/Win32/WindowImplWin32.hpp>
|
||||
#include <windows.h>
|
||||
|
||||
|
||||
|
@ -229,26 +230,34 @@ void InputImpl::setMousePosition(const Vector2i& position, const WindowBase& rel
|
|||
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
bool InputImpl::isTouchDown(unsigned int /*finger*/)
|
||||
bool InputImpl::isTouchDown(unsigned int finger)
|
||||
{
|
||||
// Not applicable
|
||||
return false;
|
||||
return WindowImplWin32::isTouchDown(finger);
|
||||
}
|
||||
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
Vector2i InputImpl::getTouchPosition(unsigned int /*finger*/)
|
||||
Vector2i InputImpl::getTouchPosition(unsigned int finger)
|
||||
{
|
||||
// Not applicable
|
||||
return Vector2i();
|
||||
return WindowImplWin32::getTouchPosition(finger);
|
||||
}
|
||||
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
Vector2i InputImpl::getTouchPosition(unsigned int /*finger*/, const WindowBase& /*relativeTo*/)
|
||||
Vector2i InputImpl::getTouchPosition(unsigned int finger, const WindowBase& relativeTo)
|
||||
{
|
||||
// Not applicable
|
||||
return Vector2i();
|
||||
WindowHandle handle = relativeTo.getSystemHandle();
|
||||
Vector2i pos;
|
||||
|
||||
if (handle && WindowImplWin32::isTouchDown(finger))
|
||||
{
|
||||
pos = WindowImplWin32::getTouchPosition(finger);
|
||||
POINT point = { pos.x, pos.y };
|
||||
ScreenToClient(handle, &point);
|
||||
pos.x = point.x;
|
||||
pos.y = point.y;
|
||||
}
|
||||
return pos;
|
||||
}
|
||||
|
||||
} // namespace priv
|
||||
|
|
|
@ -66,13 +66,93 @@ namespace
|
|||
unsigned int handleCount = 0; // All window handles
|
||||
const wchar_t* className = L"SFML_Window";
|
||||
sf::priv::WindowImplWin32* fullscreenWindow = NULL;
|
||||
HINSTANCE user32Dll = NULL;
|
||||
DWORD touchIDs[10];
|
||||
POINT touches[10];
|
||||
|
||||
#if WINVER < 0x0601
|
||||
// Define touch API that's available for more recent versions of Windows
|
||||
#define WM_TOUCH 0x0240
|
||||
|
||||
DECLARE_HANDLE(HTOUCHINPUT);
|
||||
|
||||
typedef struct tagTOUCHINPUT
|
||||
{
|
||||
LONG x;
|
||||
LONG y;
|
||||
HANDLE hSource;
|
||||
DWORD dwID;
|
||||
DWORD dwFlags;
|
||||
DWORD dwMask;
|
||||
DWORD dwTime;
|
||||
ULONG_PTR dwExtraInfo;
|
||||
DWORD cxContact;
|
||||
DWORD cyContact;
|
||||
} TOUCHINPUT, *PTOUCHINPUT;
|
||||
|
||||
typedef TOUCHINPUT const * PCTOUCHINPUT;
|
||||
|
||||
#define TOUCH_COORD_TO_PIXEL(l) ((l) / 100)
|
||||
|
||||
#define TOUCHEVENTF_MOVE 0x0001
|
||||
#define TOUCHEVENTF_DOWN 0x0002
|
||||
#define TOUCHEVENTF_UP 0x0004
|
||||
#define TOUCHEVENTF_INRANGE 0x0008
|
||||
#define TOUCHEVENTF_PRIMARY 0x0010
|
||||
#define TOUCHEVENTF_NOCOALESCE 0x0020
|
||||
#define TOUCHEVENTF_PEN 0x0040
|
||||
#define TOUCHEVENTF_PALM 0x0080
|
||||
|
||||
typedef BOOL(WINAPI* RegisterTouchWindowFuncType)(HWND, ULONG);
|
||||
typedef BOOL(WINAPI* CloseTouchInputHandleFuncType)(HTOUCHINPUT);
|
||||
typedef BOOL(WINAPI* GetTouchInputInfoFuncType)(HTOUCHINPUT, UINT, PTOUCHINPUT, int);
|
||||
|
||||
RegisterTouchWindowFuncType RegisterTouchWindow = NULL;
|
||||
CloseTouchInputHandleFuncType CloseTouchInputHandle = NULL;
|
||||
GetTouchInputInfoFuncType GetTouchInputInfo = NULL;
|
||||
bool touchEnabled = false;
|
||||
#else
|
||||
static const bool touchEnabled = true;
|
||||
#endif
|
||||
|
||||
// Convert a hardware dependent ID to a 0 based index we can use
|
||||
sf::Int8 getTouchID(DWORD id)
|
||||
{
|
||||
for (int i = 0; i < 10; ++i)
|
||||
{
|
||||
if (touchIDs[i] == id)
|
||||
return i;
|
||||
}
|
||||
for (int i = 0; i < 10; ++i)
|
||||
{
|
||||
if (touchIDs[i] == -1)
|
||||
{
|
||||
touchIDs[i] = id;
|
||||
return i;
|
||||
}
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
// Get a system error string from an error code
|
||||
std::string getErrorString(DWORD error)
|
||||
{
|
||||
PTCHAR buffer;
|
||||
|
||||
if (FormatMessage(FORMAT_MESSAGE_MAX_WIDTH_MASK | FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, error, 0, reinterpret_cast<PTCHAR>(&buffer), 0, NULL) == 0)
|
||||
return "Unknown error.";
|
||||
|
||||
sf::String message = buffer;
|
||||
LocalFree(buffer);
|
||||
return message.toAnsiString();
|
||||
}
|
||||
|
||||
const GUID GUID_DEVINTERFACE_HID = {0x4d1e55b2, 0xf16f, 0x11cf, {0x88, 0xcb, 0x00, 0x11, 0x11, 0x00, 0x00, 0x30}};
|
||||
|
||||
void setProcessDpiAware()
|
||||
{
|
||||
// Try SetProcessDpiAwareness first
|
||||
HINSTANCE shCoreDll = LoadLibrary(L"Shcore.dll");
|
||||
HINSTANCE shCoreDll = LoadLibraryA("Shcore.dll");
|
||||
|
||||
if (shCoreDll)
|
||||
{
|
||||
|
@ -107,8 +187,6 @@ namespace
|
|||
|
||||
// Fall back to SetProcessDPIAware if SetProcessDpiAwareness
|
||||
// is not available on this system
|
||||
HINSTANCE user32Dll = LoadLibrary(L"user32.dll");
|
||||
|
||||
if (user32Dll)
|
||||
{
|
||||
typedef BOOL (WINAPI* SetProcessDPIAwareFuncType)(void);
|
||||
|
@ -119,8 +197,6 @@ namespace
|
|||
if (!SetProcessDPIAwareFunc())
|
||||
sf::err() << "Failed to set process DPI awareness" << std::endl;
|
||||
}
|
||||
|
||||
FreeLibrary(user32Dll);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -144,12 +220,20 @@ m_mouseInside (false),
|
|||
m_fullscreen (false),
|
||||
m_cursorGrabbed (false)
|
||||
{
|
||||
// Set that this process is DPI aware and can handle DPI scaling
|
||||
setProcessDpiAware();
|
||||
// If we're the first window handle
|
||||
if (handleCount == 0)
|
||||
{
|
||||
// Ensure User32.dll is loaded
|
||||
if (!user32Dll)
|
||||
user32Dll = LoadLibraryA("User32.dll");
|
||||
|
||||
// Set that this process is DPI aware and can handle DPI scaling
|
||||
setProcessDpiAware();
|
||||
}
|
||||
|
||||
if (m_handle)
|
||||
{
|
||||
// If we're the first window handle, we only need to poll for joysticks when WM_DEVICECHANGE message is received
|
||||
// We only need to poll for joysticks when WM_DEVICECHANGE message is received
|
||||
if (handleCount == 0)
|
||||
JoystickImpl::setLazyUpdates(true);
|
||||
|
||||
|
@ -158,6 +242,9 @@ m_cursorGrabbed (false)
|
|||
// We change the event procedure of the control (it is important to save the old one)
|
||||
SetWindowLongPtrW(m_handle, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(this));
|
||||
m_callback = SetWindowLongPtrW(m_handle, GWLP_WNDPROC, reinterpret_cast<LONG_PTR>(&WindowImplWin32::globalOnEvent));
|
||||
|
||||
// Try to prepare touch events, if necessary
|
||||
prepareTouch();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -177,8 +264,16 @@ m_mouseInside (false),
|
|||
m_fullscreen ((style & Style::Fullscreen) != 0),
|
||||
m_cursorGrabbed (m_fullscreen)
|
||||
{
|
||||
// Set that this process is DPI aware and can handle DPI scaling
|
||||
setProcessDpiAware();
|
||||
// If we're the first window handle
|
||||
if (handleCount == 0)
|
||||
{
|
||||
// Ensure User32.dll is loaded
|
||||
if (!user32Dll)
|
||||
user32Dll = LoadLibraryA("User32.dll");
|
||||
|
||||
// Set that this process is DPI aware and can handle DPI scaling
|
||||
setProcessDpiAware();
|
||||
}
|
||||
|
||||
// Register the window class at first call
|
||||
if (windowCount == 0)
|
||||
|
@ -238,6 +333,9 @@ m_cursorGrabbed (m_fullscreen)
|
|||
if (m_fullscreen)
|
||||
switchToFullscreen(mode);
|
||||
|
||||
// Try to prepare touch events, if necessary
|
||||
prepareTouch();
|
||||
|
||||
// Increment window count
|
||||
windowCount++;
|
||||
}
|
||||
|
@ -257,8 +355,18 @@ WindowImplWin32::~WindowImplWin32()
|
|||
{
|
||||
--handleCount;
|
||||
|
||||
// This was the last handle
|
||||
if (handleCount == 0)
|
||||
{
|
||||
// Free User32.dll
|
||||
if (user32Dll)
|
||||
{
|
||||
FreeLibrary(user32Dll);
|
||||
user32Dll = NULL;
|
||||
}
|
||||
// Reenable automatic joystick polling
|
||||
JoystickImpl::setLazyUpdates(false);
|
||||
}
|
||||
}
|
||||
|
||||
if (!m_callback)
|
||||
|
@ -999,6 +1107,73 @@ void WindowImplWin32::processEvent(UINT message, WPARAM wParam, LPARAM lParam)
|
|||
|
||||
break;
|
||||
}
|
||||
case WM_TOUCH:
|
||||
{
|
||||
// Get the number of events
|
||||
Uint16 num = LOWORD(wParam);
|
||||
|
||||
// Reserve memory
|
||||
PTOUCHINPUT events = new TOUCHINPUT[num];
|
||||
|
||||
if (events)
|
||||
{
|
||||
if (GetTouchInputInfo(reinterpret_cast<HTOUCHINPUT>(lParam), num, events, sizeof(TOUCHINPUT)))
|
||||
{
|
||||
POINT point;
|
||||
for (int i = 0; i < num; ++i)
|
||||
{
|
||||
Event event;
|
||||
Int8 index = getTouchID(events[i].dwID);
|
||||
|
||||
// Out of Ids? Should never happen
|
||||
if (index == -1)
|
||||
continue;
|
||||
|
||||
event.touch.finger = index;
|
||||
point.x = TOUCH_COORD_TO_PIXEL(events[i].x);
|
||||
point.y = TOUCH_COORD_TO_PIXEL(events[i].y);
|
||||
|
||||
POINT cpoint = point;
|
||||
ScreenToClient(m_handle, &cpoint);
|
||||
event.touch.x = cpoint.x;
|
||||
event.touch.y = cpoint.y;
|
||||
|
||||
if (events[i].dwFlags & TOUCHEVENTF_DOWN)
|
||||
{
|
||||
event.type = Event::TouchBegan;
|
||||
pushEvent(event);
|
||||
|
||||
// Prevent initial move event
|
||||
touches[index] = point;
|
||||
}
|
||||
if (events[i].dwFlags & TOUCHEVENTF_UP)
|
||||
{
|
||||
event.type = Event::TouchEnded;
|
||||
pushEvent(event);
|
||||
|
||||
// Remove the stored ID
|
||||
touchIDs[index] = -1;
|
||||
}
|
||||
if (events[i].dwFlags & TOUCHEVENTF_MOVE) {
|
||||
// Only handle real movement
|
||||
if (touches[index].x != point.x || touches[index].y != point.y)
|
||||
{
|
||||
touches[index] = point;
|
||||
event.type = Event::TouchMoved;
|
||||
pushEvent(event);
|
||||
}
|
||||
}
|
||||
}
|
||||
CloseTouchInputHandle(reinterpret_cast<HTOUCHINPUT>(lParam));
|
||||
}
|
||||
else
|
||||
{
|
||||
err() << "Failed to get touch input info: " << getErrorString(GetLastError()) << std::endl;
|
||||
}
|
||||
delete[] events;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1160,6 +1335,48 @@ LRESULT CALLBACK WindowImplWin32::globalOnEvent(HWND handle, UINT message, WPARA
|
|||
return DefWindowProcW(handle, message, wParam, lParam);
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
void WindowImplWin32::prepareTouch()
|
||||
{
|
||||
static bool prepared = false;
|
||||
if (!prepared)
|
||||
{
|
||||
prepared = true;
|
||||
|
||||
#if WINVER < 0x0601
|
||||
RegisterTouchWindow = reinterpret_cast<RegisterTouchWindowFuncType>(GetProcAddress(user32Dll, "RegisterTouchWindow"));
|
||||
|
||||
touchEnabled = RegisterTouchWindow != NULL;
|
||||
|
||||
// If we've got touch support, load the other procs
|
||||
if (touchEnabled)
|
||||
{
|
||||
CloseTouchInputHandle = reinterpret_cast<CloseTouchInputHandleFuncType>(GetProcAddress(user32Dll, "CloseTouchInputHandle"));
|
||||
GetTouchInputInfo = reinterpret_cast<GetTouchInputInfoFuncType>(GetProcAddress(user32Dll, "GetTouchInputInfo"));
|
||||
|
||||
// Reset touch IDs
|
||||
for (int i = 0; i < 10; ++i)
|
||||
touchIDs[i] = -1;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
if (touchEnabled && m_handle)
|
||||
RegisterTouchWindow(m_handle, 0);
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
bool WindowImplWin32::isTouchDown(unsigned int finger)
|
||||
{
|
||||
return touchIDs[finger] != -1;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
Vector2i WindowImplWin32::getTouchPosition(unsigned int finger)
|
||||
{
|
||||
return Vector2i(touches[finger].x, touches[finger].y);
|
||||
}
|
||||
|
||||
} // namespace priv
|
||||
|
||||
} // namespace sf
|
||||
|
|
|
@ -184,6 +184,26 @@ public:
|
|||
////////////////////////////////////////////////////////////
|
||||
virtual bool hasFocus() const;
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
/// \brief Get global touch state for a finger
|
||||
///
|
||||
/// \param finger Finger index
|
||||
///
|
||||
/// \return True, if the finger is active
|
||||
///
|
||||
////////////////////////////////////////////////////////////
|
||||
static bool isTouchDown(unsigned int finger);
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
/// \brief Get global touch coordinates for a finger
|
||||
///
|
||||
/// \param finger Finger index
|
||||
///
|
||||
/// \return Touch position
|
||||
///
|
||||
////////////////////////////////////////////////////////////
|
||||
static Vector2i getTouchPosition(unsigned int finger);
|
||||
|
||||
protected:
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
|
@ -269,6 +289,12 @@ private:
|
|||
////////////////////////////////////////////////////////////
|
||||
static LRESULT CALLBACK globalOnEvent(HWND handle, UINT message, WPARAM wParam, LPARAM lParam);
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
/// \brief Helper function to prepare and enable touch handling
|
||||
///
|
||||
////////////////////////////////////////////////////////////
|
||||
void prepareTouch();
|
||||
|
||||
////////////////////////////////////////////////////////////
|
||||
// Member data
|
||||
////////////////////////////////////////////////////////////
|
||||
|
|
Loading…
Reference in a new issue