mirror of
https://github.com/OpenMW/openmw.git
synced 2025-10-24 03:26:40 +00:00
453 lines
16 KiB
C++
453 lines
16 KiB
C++
#include "sdlinputwrapper.hpp"
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
#include <components/settings/values.hpp>
|
|
|
|
#include <osgViewer/Viewer>
|
|
|
|
namespace SDLUtil
|
|
{
|
|
|
|
InputWrapper::InputWrapper(SDL_Window* window, osg::ref_ptr<osgViewer::Viewer> viewer, bool grab)
|
|
: mSDLWindow(window)
|
|
, mViewer(std::move(viewer))
|
|
, mMouseListener(nullptr)
|
|
, mSensorListener(nullptr)
|
|
, mKeyboardListener(nullptr)
|
|
, mWindowListener(nullptr)
|
|
, mConListener(nullptr)
|
|
, mWarpX(0)
|
|
, mWarpY(0)
|
|
, mWarpCompensate(false)
|
|
, mWrapPointer(false)
|
|
, mAllowGrab(grab)
|
|
, mWantMouseVisible(false)
|
|
, mWantGrab(false)
|
|
, mWantRelative(false)
|
|
, mGrabPointer(false)
|
|
, mMouseRelative(false)
|
|
, mFirstMouseMove(true)
|
|
, mMouseZ(0)
|
|
, mMouseX(0)
|
|
, mMouseY(0)
|
|
, mWindowHasFocus(true)
|
|
, mMouseInWindow(true)
|
|
{
|
|
Uint32 flags = SDL_GetWindowFlags(mSDLWindow);
|
|
mWindowHasFocus = (flags & SDL_WINDOW_INPUT_FOCUS);
|
|
mMouseInWindow = (flags & SDL_WINDOW_MOUSE_FOCUS);
|
|
_setWindowScale();
|
|
}
|
|
|
|
InputWrapper::~InputWrapper() {}
|
|
|
|
void InputWrapper::_setWindowScale()
|
|
{
|
|
int w, h;
|
|
SDL_GetWindowSize(mSDLWindow, &w, &h);
|
|
int dw, dh;
|
|
SDL_GL_GetDrawableSize(mSDLWindow, &dw, &dh);
|
|
mScaleX = dw / w;
|
|
mScaleY = dh / h;
|
|
}
|
|
|
|
void InputWrapper::capture(bool windowEventsOnly)
|
|
{
|
|
mViewer->getEventQueue()->frame(0.f);
|
|
|
|
SDL_PumpEvents();
|
|
|
|
SDL_Event evt;
|
|
|
|
if (windowEventsOnly)
|
|
{
|
|
// During loading, handle window events, discard button presses and mouse movement and keep others for later
|
|
while (SDL_PeepEvents(&evt, 1, SDL_GETEVENT, SDL_WINDOWEVENT, SDL_WINDOWEVENT) > 0)
|
|
handleWindowEvent(evt);
|
|
|
|
SDL_FlushEvent(SDL_KEYDOWN);
|
|
SDL_FlushEvent(SDL_CONTROLLERBUTTONDOWN);
|
|
SDL_FlushEvent(SDL_MOUSEBUTTONDOWN);
|
|
SDL_FlushEvent(SDL_MOUSEMOTION);
|
|
SDL_FlushEvent(SDL_MOUSEWHEEL);
|
|
|
|
return;
|
|
}
|
|
|
|
while (SDL_PollEvent(&evt))
|
|
{
|
|
#if SDL_VERSION_ATLEAST(2, 30, 50)
|
|
// SDL2-compat may pass us SDL3 display and window events alongside the SDL2 events for funsies
|
|
// Until we are ready to move to SDL3, we'll want to prevent the noise
|
|
|
|
// Silence 0x151 to 0x1FF range
|
|
if (evt.type > SDL_DISPLAYEVENT && evt.type < SDL_WINDOWEVENT)
|
|
continue;
|
|
|
|
// Silence 0x202 to 0x2FF range
|
|
if (evt.type > SDL_SYSWMEVENT && evt.type < SDL_KEYDOWN)
|
|
continue;
|
|
#endif
|
|
switch (evt.type)
|
|
{
|
|
case SDL_MOUSEMOTION:
|
|
// Ignore this if it happened due to a warp
|
|
if (!_handleWarpMotion(evt.motion))
|
|
{
|
|
// If in relative mode, don't trigger events unless window has focus
|
|
if (!mWantRelative || mWindowHasFocus)
|
|
mMouseListener->mouseMoved(_packageMouseMotion(evt));
|
|
|
|
// Try to keep the mouse inside the window
|
|
if (mWindowHasFocus)
|
|
_wrapMousePointer(evt.motion);
|
|
}
|
|
break;
|
|
case SDL_MOUSEWHEEL:
|
|
mMouseListener->mouseMoved(_packageMouseMotion(evt));
|
|
mMouseListener->mouseWheelMoved(evt.wheel);
|
|
break;
|
|
case SDL_SENSORUPDATE:
|
|
mSensorListener->sensorUpdated(evt.sensor);
|
|
break;
|
|
case SDL_MOUSEBUTTONDOWN:
|
|
mMouseListener->mousePressed(evt.button, evt.button.button);
|
|
break;
|
|
case SDL_MOUSEBUTTONUP:
|
|
mMouseListener->mouseReleased(evt.button, evt.button.button);
|
|
break;
|
|
case SDL_KEYDOWN:
|
|
mKeyboardListener->keyPressed(evt.key);
|
|
|
|
if (!isModifierHeld(KMOD_ALT) && evt.key.keysym.sym >= SDLK_F1 && evt.key.keysym.sym <= SDLK_F12)
|
|
{
|
|
mViewer->getEventQueue()->keyPress(
|
|
osgGA::GUIEventAdapter::KEY_F1 + (evt.key.keysym.sym - SDLK_F1));
|
|
}
|
|
|
|
break;
|
|
case SDL_KEYUP:
|
|
if (!evt.key.repeat)
|
|
{
|
|
mKeyboardListener->keyReleased(evt.key);
|
|
|
|
if (!isModifierHeld(KMOD_ALT) && evt.key.keysym.sym >= SDLK_F1
|
|
&& evt.key.keysym.sym <= SDLK_F12)
|
|
mViewer->getEventQueue()->keyRelease(
|
|
osgGA::GUIEventAdapter::KEY_F1 + (evt.key.keysym.sym - SDLK_F1));
|
|
}
|
|
|
|
break;
|
|
case SDL_TEXTEDITING:
|
|
break;
|
|
case SDL_TEXTINPUT:
|
|
mKeyboardListener->textInput(evt.text);
|
|
break;
|
|
case SDL_KEYMAPCHANGED:
|
|
break;
|
|
case SDL_JOYHATMOTION: // As we manage everything with GameController, don't even bother with these.
|
|
case SDL_JOYAXISMOTION:
|
|
case SDL_JOYBUTTONDOWN:
|
|
case SDL_JOYBUTTONUP:
|
|
case SDL_JOYDEVICEADDED:
|
|
case SDL_JOYDEVICEREMOVED:
|
|
break;
|
|
case SDL_CONTROLLERDEVICEADDED:
|
|
if (mConListener)
|
|
mConListener->controllerAdded(
|
|
1, evt.cdevice); // We only support one joystick, so give everything a generic deviceID
|
|
break;
|
|
case SDL_CONTROLLERDEVICEREMOVED:
|
|
if (mConListener)
|
|
mConListener->controllerRemoved(evt.cdevice);
|
|
break;
|
|
case SDL_CONTROLLERBUTTONDOWN:
|
|
if (mConListener)
|
|
mConListener->buttonPressed(1, evt.cbutton);
|
|
break;
|
|
case SDL_CONTROLLERBUTTONUP:
|
|
if (mConListener)
|
|
mConListener->buttonReleased(1, evt.cbutton);
|
|
break;
|
|
case SDL_CONTROLLERAXISMOTION:
|
|
if (mConListener)
|
|
mConListener->axisMoved(1, evt.caxis);
|
|
break;
|
|
case SDL_CONTROLLERSENSORUPDATE:
|
|
// controller sensor data is received on demand
|
|
break;
|
|
case SDL_CONTROLLERTOUCHPADDOWN:
|
|
mConListener->touchpadPressed(1, TouchEvent(evt.ctouchpad));
|
|
break;
|
|
case SDL_CONTROLLERTOUCHPADMOTION:
|
|
mConListener->touchpadMoved(1, TouchEvent(evt.ctouchpad));
|
|
break;
|
|
case SDL_CONTROLLERTOUCHPADUP:
|
|
mConListener->touchpadReleased(1, TouchEvent(evt.ctouchpad));
|
|
break;
|
|
case SDL_WINDOWEVENT:
|
|
handleWindowEvent(evt);
|
|
break;
|
|
case SDL_QUIT:
|
|
if (mWindowListener)
|
|
mWindowListener->windowClosed();
|
|
break;
|
|
case SDL_DISPLAYEVENT:
|
|
switch (evt.display.event)
|
|
{
|
|
case SDL_DISPLAYEVENT_ORIENTATION:
|
|
if (mSensorListener
|
|
&& evt.display.display == static_cast<Uint32>(Settings::video().mScreen))
|
|
{
|
|
mSensorListener->displayOrientationChanged();
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
case SDL_CLIPBOARDUPDATE:
|
|
break; // We don't need this event, clipboard is retrieved on demand
|
|
|
|
case SDL_FINGERDOWN:
|
|
case SDL_FINGERUP:
|
|
case SDL_FINGERMOTION:
|
|
case SDL_DOLLARGESTURE:
|
|
case SDL_DOLLARRECORD:
|
|
case SDL_MULTIGESTURE:
|
|
// No use for touch & gesture events
|
|
break;
|
|
|
|
case SDL_APP_WILLENTERBACKGROUND:
|
|
case SDL_APP_WILLENTERFOREGROUND:
|
|
case SDL_APP_DIDENTERBACKGROUND:
|
|
case SDL_APP_DIDENTERFOREGROUND:
|
|
// We do not need background/foreground switch event for mobile devices so far
|
|
break;
|
|
|
|
case SDL_APP_TERMINATING:
|
|
// There is nothing we can do here.
|
|
break;
|
|
|
|
case SDL_APP_LOWMEMORY:
|
|
Log(Debug::Warning) << "System reports that free RAM on device is running low. You may encounter "
|
|
"an unexpected behaviour.";
|
|
break;
|
|
|
|
default:
|
|
Log(Debug::Info) << "Unhandled SDL event of type 0x" << std::hex << evt.type;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void InputWrapper::handleWindowEvent(const SDL_Event& evt)
|
|
{
|
|
switch (evt.window.event)
|
|
{
|
|
case SDL_WINDOWEVENT_ENTER:
|
|
mMouseInWindow = true;
|
|
updateMouseSettings();
|
|
break;
|
|
case SDL_WINDOWEVENT_LEAVE:
|
|
mMouseInWindow = false;
|
|
updateMouseSettings();
|
|
break;
|
|
case SDL_WINDOWEVENT_MOVED:
|
|
// I'm not sure what OSG is using the window position for, but I don't think it's needed,
|
|
// so we ignore window moved events (improves window movement performance)
|
|
break;
|
|
case SDL_WINDOWEVENT_SIZE_CHANGED:
|
|
int w, h;
|
|
SDL_GL_GetDrawableSize(mSDLWindow, &w, &h);
|
|
int x, y;
|
|
SDL_GetWindowPosition(mSDLWindow, &x, &y);
|
|
|
|
// Happens when you Alt-Tab out of game
|
|
if (w == 0 && h == 0)
|
|
return;
|
|
|
|
mViewer->getCamera()->getGraphicsContext()->resized(x, y, w, h);
|
|
|
|
mViewer->getEventQueue()->windowResize(x, y, w, h);
|
|
|
|
if (mWindowListener)
|
|
mWindowListener->windowResized(w, h);
|
|
|
|
_setWindowScale();
|
|
|
|
break;
|
|
|
|
case SDL_WINDOWEVENT_RESIZED:
|
|
// This should also fire SIZE_CHANGED, so no need to handle
|
|
break;
|
|
|
|
case SDL_WINDOWEVENT_FOCUS_GAINED:
|
|
mWindowHasFocus = true;
|
|
updateMouseSettings();
|
|
break;
|
|
case SDL_WINDOWEVENT_FOCUS_LOST:
|
|
mWindowHasFocus = false;
|
|
updateMouseSettings();
|
|
break;
|
|
case SDL_WINDOWEVENT_CLOSE:
|
|
break;
|
|
case SDL_WINDOWEVENT_SHOWN:
|
|
case SDL_WINDOWEVENT_RESTORED:
|
|
if (mWindowListener)
|
|
mWindowListener->windowVisibilityChange(true);
|
|
break;
|
|
case SDL_WINDOWEVENT_HIDDEN:
|
|
case SDL_WINDOWEVENT_MINIMIZED:
|
|
if (mWindowListener)
|
|
mWindowListener->windowVisibilityChange(false);
|
|
break;
|
|
}
|
|
}
|
|
|
|
bool InputWrapper::isModifierHeld(int mod)
|
|
{
|
|
return (SDL_GetModState() & mod) != 0;
|
|
}
|
|
|
|
bool InputWrapper::isKeyDown(SDL_Scancode key)
|
|
{
|
|
return (SDL_GetKeyboardState(nullptr)[key]) != 0;
|
|
}
|
|
|
|
/// \brief Moves the mouse to the specified point within the viewport
|
|
void InputWrapper::warpMouse(int x, int y)
|
|
{
|
|
SDL_WarpMouseInWindow(mSDLWindow, x, y);
|
|
mWarpCompensate = true;
|
|
mWarpX = x;
|
|
mWarpY = y;
|
|
}
|
|
|
|
/// \brief Locks the pointer to the window
|
|
void InputWrapper::setGrabPointer(bool grab)
|
|
{
|
|
mWantGrab = grab;
|
|
updateMouseSettings();
|
|
}
|
|
|
|
/// \brief Set the mouse to relative positioning. Doesn't move the cursor
|
|
/// and disables mouse acceleration.
|
|
void InputWrapper::setMouseRelative(bool relative)
|
|
{
|
|
mWantRelative = relative;
|
|
updateMouseSettings();
|
|
}
|
|
|
|
void InputWrapper::setMouseVisible(bool visible)
|
|
{
|
|
mWantMouseVisible = visible;
|
|
updateMouseSettings();
|
|
}
|
|
|
|
void InputWrapper::updateMouseSettings()
|
|
{
|
|
mGrabPointer = mWantGrab && mMouseInWindow && mWindowHasFocus;
|
|
SDL_SetWindowGrab(mSDLWindow, mGrabPointer && mAllowGrab ? SDL_TRUE : SDL_FALSE);
|
|
|
|
SDL_ShowCursor(mWantMouseVisible || !mWindowHasFocus);
|
|
|
|
bool relative = mWantRelative && mMouseInWindow && mWindowHasFocus;
|
|
if (mMouseRelative == relative)
|
|
return;
|
|
|
|
mMouseRelative = relative;
|
|
|
|
mWrapPointer = false;
|
|
|
|
// eep, wrap the pointer manually if the input driver doesn't support
|
|
// relative positioning natively
|
|
// also use wrapping if no-grab was specified in options (SDL_SetRelativeMouseMode
|
|
// appears to eat the mouse cursor when pausing in a debugger)
|
|
bool success = mAllowGrab && SDL_SetRelativeMouseMode(relative ? SDL_TRUE : SDL_FALSE) == 0;
|
|
if (relative && !success)
|
|
mWrapPointer = true;
|
|
|
|
// now remove all mouse events using the old setting from the queue
|
|
SDL_PumpEvents();
|
|
SDL_FlushEvent(SDL_MOUSEMOTION);
|
|
}
|
|
|
|
/// \brief Internal method for ignoring relative motions as a side effect
|
|
/// of warpMouse()
|
|
bool InputWrapper::_handleWarpMotion(const SDL_MouseMotionEvent& evt)
|
|
{
|
|
if (!mWarpCompensate)
|
|
return false;
|
|
|
|
// this was a warp event, signal the caller to eat it.
|
|
if (evt.x == mWarpX && evt.y == mWarpY)
|
|
{
|
|
mWarpCompensate = false;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// \brief Wrap the mouse to the viewport
|
|
void InputWrapper::_wrapMousePointer(const SDL_MouseMotionEvent& evt)
|
|
{
|
|
// don't wrap if we don't want relative movements, support relative
|
|
// movements natively, or aren't grabbing anyways
|
|
if (!mMouseRelative || !mWrapPointer || !mGrabPointer)
|
|
return;
|
|
|
|
int width = 0;
|
|
int height = 0;
|
|
|
|
SDL_GetWindowSize(mSDLWindow, &width, &height);
|
|
|
|
const int fudgeFactorX = width / 4;
|
|
const int fudgeFactorY = height / 4;
|
|
|
|
// warp the mouse if it's about to go outside the window
|
|
if (evt.x - fudgeFactorX < 0 || evt.x + fudgeFactorX > width || evt.y - fudgeFactorY < 0
|
|
|| evt.y + fudgeFactorY > height)
|
|
{
|
|
warpMouse(width / 2, height / 2);
|
|
}
|
|
}
|
|
|
|
/// \brief Package mouse and mousewheel motions into a single event
|
|
MouseMotionEvent InputWrapper::_packageMouseMotion(const SDL_Event& evt)
|
|
{
|
|
MouseMotionEvent packEvt = {};
|
|
packEvt.x = mMouseX * mScaleX;
|
|
packEvt.y = mMouseY * mScaleY;
|
|
packEvt.z = mMouseZ;
|
|
|
|
if (evt.type == SDL_MOUSEMOTION)
|
|
{
|
|
packEvt.x = mMouseX = evt.motion.x * mScaleX;
|
|
packEvt.y = mMouseY = evt.motion.y * mScaleY;
|
|
packEvt.xrel = evt.motion.xrel * mScaleX;
|
|
packEvt.yrel = evt.motion.yrel * mScaleY;
|
|
packEvt.type = SDL_MOUSEMOTION;
|
|
if (mFirstMouseMove)
|
|
{
|
|
// first event should be treated as non-relative, since there's no point of reference
|
|
// SDL then (incorrectly) uses (0,0) as point of reference, on Linux at least...
|
|
packEvt.xrel = packEvt.yrel = 0;
|
|
mFirstMouseMove = false;
|
|
}
|
|
}
|
|
else if (evt.type == SDL_MOUSEWHEEL)
|
|
{
|
|
mMouseZ += packEvt.zrel = (evt.wheel.y * 120);
|
|
packEvt.z = mMouseZ;
|
|
packEvt.type = SDL_MOUSEWHEEL;
|
|
}
|
|
else
|
|
{
|
|
throw std::runtime_error("Tried to package non-motion event!");
|
|
}
|
|
|
|
return packEvt;
|
|
}
|
|
}
|