1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 15:35:23 +00:00
OpenMW/apps/openmw/mwinput/inputmanagerimp.cpp

1402 lines
58 KiB
C++
Raw Normal View History

#include "inputmanagerimp.hpp"
2010-07-17 19:58:15 +02:00
#include <osgViewer/ViewerEventHandlers>
2012-08-12 20:45:02 +02:00
#include <MyGUI_InputManager.h>
2012-08-27 10:01:53 +02:00
#include <MyGUI_Widget.h>
#include <MyGUI_Button.h>
#include <MyGUI_EditBox.h>
2010-07-17 19:58:15 +02:00
#include <SDL_version.h>
2018-08-14 23:05:43 +04:00
#include <components/debug/debuglog.hpp>
2015-05-13 16:50:47 +02:00
#include <components/sdlutil/sdlinputwrapper.hpp>
#include <components/sdlutil/sdlvideowrapper.hpp>
2016-10-20 02:12:01 +02:00
#include <components/esm/esmwriter.hpp>
#include <components/esm/esmreader.hpp>
#include <components/esm/controlsstate.hpp>
2015-05-13 16:50:47 +02:00
#include "../mwbase/world.hpp"
2012-08-12 20:45:02 +02:00
#include "../mwbase/windowmanager.hpp"
#include "../mwbase/statemanager.hpp"
#include "../mwbase/environment.hpp"
#include "../mwbase/mechanicsmanager.hpp"
2014-02-23 20:11:05 +01:00
#include "../mwworld/player.hpp"
#include "../mwworld/class.hpp"
#include "../mwworld/inventorystore.hpp"
#include "../mwworld/esmstore.hpp"
2014-06-17 17:18:30 +02:00
#include "../mwmechanics/npcstats.hpp"
2015-08-21 21:12:39 +12:00
#include "../mwmechanics/actorutil.hpp"
2010-07-17 19:58:15 +02:00
#include "actionmanager.hpp"
#include "mousemanager.hpp"
2020-04-08 11:43:45 +04:00
#include "sdlmappings.hpp"
#include "sensormanager.hpp"
2010-07-17 19:58:15 +02:00
namespace MWInput
{
2015-05-03 17:24:35 +02:00
InputManager::InputManager(
2015-05-13 16:50:47 +02:00
SDL_Window* window,
osg::ref_ptr<osgViewer::Viewer> viewer,
osg::ref_ptr<osgViewer::ScreenCaptureHandler> screenCaptureHandler,
2017-11-09 18:26:27 +01:00
osgViewer::ScreenCaptureHandler::CaptureOperation *screenCaptureOperation,
const std::string& userFile, bool userFileExists, const std::string& userControllerBindingsFile,
const std::string& controllerBindingsFile, bool grab)
: mWindow(window)
, mWindowVisible(true)
, mJoystickLastUsed(false)
2018-10-09 10:21:12 +04:00
, mInputManager(nullptr)
, mVideoWrapper(nullptr)
, mUserFile(userFile)
2015-04-25 13:37:42 -05:00
, mDragDrop(false)
, mGrabCursor (Settings::Manager::getBool("grab cursor", "Input"))
, mControlsDisabled(false)
, mJoystickEnabled (Settings::Manager::getBool("enable controller", "Input"))
2012-08-17 16:42:42 +04:00
, mPreviewPOVDelay(0.f)
2012-08-18 01:31:57 +04:00
, mTimeIdle(0.f)
2015-04-25 13:37:42 -05:00
, mGuiCursorEnabled(true)
, mGamepadGuiCursorEnabled(true)
2015-04-25 13:37:42 -05:00
, mDetectingKeyboard(false)
, mOverencumberedMessageDelay(0.f)
2019-02-27 13:29:48 +00:00
, mGamepadZoom(0)
2019-02-27 14:03:16 -08:00
, mSneakToggleShortcutTimer(0.f)
, mSneakGamepadShortcut(false)
, mAttemptJump(false)
, mGamepadCursorSpeed(Settings::Manager::getFloat("gamepad cursor speed", "Input"))
2015-05-23 22:44:00 +02:00
, mFakeDeviceID(1)
2010-07-17 19:58:15 +02:00
{
2015-05-13 16:50:47 +02:00
mInputManager = new SDLUtil::InputWrapper(window, viewer, grab);
mInputManager->setKeyboardEventCallback (this);
mInputManager->setWindowEventCallback(this);
2014-12-08 21:57:32 -06:00
mInputManager->setControllerEventCallback(this);
2010-07-17 19:58:15 +02:00
mVideoWrapper = new SDLUtil::VideoWrapper(window, viewer);
mVideoWrapper->setGammaContrast(Settings::Manager::getFloat("gamma", "Video"),
Settings::Manager::getFloat("contrast", "Video"));
std::string file = userFileExists ? userFile : "";
2018-10-09 10:21:12 +04:00
mInputBinder = new ICS::InputControlSystem(file, true, this, nullptr, A_Last);
2010-07-17 19:58:15 +02:00
loadKeyDefaults();
2014-12-08 21:57:32 -06:00
loadControllerDefaults();
2012-08-13 01:26:15 +02:00
for (int i = 0; i < A_Last; ++i)
2012-08-12 20:45:02 +02:00
{
mInputBinder->getChannel (i)->addListener (this);
2012-08-12 20:45:02 +02:00
}
2012-08-12 20:45:02 +02:00
mControlSwitch["playercontrols"] = true;
mControlSwitch["playerfighting"] = true;
mControlSwitch["playerjumping"] = true;
mControlSwitch["playerlooking"] = true;
mControlSwitch["playermagic"] = true;
mControlSwitch["playerviewswitch"] = true;
mControlSwitch["vanitymode"] = true;
2014-12-20 14:46:11 -06:00
/* Joystick Init */
2014-12-20 14:46:11 -06:00
// Load controller mappings
2019-08-03 19:55:58 +00:00
if(!controllerBindingsFile.empty())
2014-12-20 14:46:11 -06:00
{
SDL_GameControllerAddMappingsFromFile(controllerBindingsFile.c_str());
2014-12-20 14:46:11 -06:00
}
2019-08-03 19:55:58 +00:00
if(!userControllerBindingsFile.empty())
{
SDL_GameControllerAddMappingsFromFile(userControllerBindingsFile.c_str());
}
2014-12-20 14:46:11 -06:00
// Open all presently connected sticks
int numSticks = SDL_NumJoysticks();
for(int i = 0; i < numSticks; i++)
{
2014-12-20 14:46:11 -06:00
if(SDL_IsGameController(i))
{
SDL_ControllerDeviceEvent evt;
evt.which = i;
controllerAdded(mFakeDeviceID, evt);
2018-08-14 23:05:43 +04:00
Log(Debug::Info) << "Detected game controller: " << SDL_GameControllerNameForIndex(i);
2014-12-20 14:46:11 -06:00
}
else
{
2018-08-14 23:05:43 +04:00
Log(Debug::Info) << "Detected unusable controller: " << SDL_JoystickNameForIndex(i);
2014-12-20 14:46:11 -06:00
}
}
2015-05-15 00:41:21 +02:00
mSensorManager = new SensorManager();
mInputManager->setSensorEventCallback (mSensorManager);
mMouseManager = new MouseManager(mInputBinder, mInputManager, window);
mInputManager->setMouseEventCallback (mMouseManager);
mActionManager = new ActionManager(mInputBinder, screenCaptureOperation, viewer, screenCaptureHandler);
}
void InputManager::clear()
{
// Enable all controls
for (std::map<std::string, bool>::iterator it = mControlSwitch.begin(); it != mControlSwitch.end(); ++it)
it->second = true;
mActionManager->clear();
mSensorManager->clear();
mMouseManager->clear();
}
2012-08-12 20:45:02 +02:00
InputManager::~InputManager()
{
mInputBinder->save (mUserFile);
2011-01-18 15:20:36 +01:00
delete mActionManager;
delete mMouseManager;
delete mSensorManager;
delete mInputBinder;
2015-05-13 16:50:47 +02:00
delete mInputManager;
delete mVideoWrapper;
}
bool InputManager::isWindowVisible()
{
return mWindowVisible;
}
void InputManager::setPlayerControlsEnabled(bool enabled)
{
2017-09-24 13:49:35 +02:00
int playerChannels[] = {A_AutoMove, A_AlwaysRun, A_ToggleWeapon,
A_ToggleSpell, A_Rest, A_QuickKey1, A_QuickKey2,
A_QuickKey3, A_QuickKey4, A_QuickKey5, A_QuickKey6,
A_QuickKey7, A_QuickKey8, A_QuickKey9, A_QuickKey10,
2017-03-26 22:56:36 +04:00
A_Use, A_Journal};
2017-09-24 13:49:35 +02:00
for(size_t i = 0; i < sizeof(playerChannels)/sizeof(playerChannels[0]); i++) {
int pc = playerChannels[i];
mInputBinder->getChannel(pc)->setEnabled(enabled);
}
}
bool InputManager::gamepadToGuiControl(const SDL_ControllerButtonEvent &arg)
{
// Presumption of GUI mode will be removed in the future.
// MyGUI KeyCodes *may* change.
MyGUI::KeyCode key = MyGUI::KeyCode::None;
switch (arg.button)
{
case SDL_CONTROLLER_BUTTON_DPAD_UP:
key = MyGUI::KeyCode::ArrowUp;
break;
case SDL_CONTROLLER_BUTTON_DPAD_RIGHT:
key = MyGUI::KeyCode::ArrowRight;
break;
case SDL_CONTROLLER_BUTTON_DPAD_DOWN:
key = MyGUI::KeyCode::ArrowDown;
break;
case SDL_CONTROLLER_BUTTON_DPAD_LEFT:
key = MyGUI::KeyCode::ArrowLeft;
break;
case SDL_CONTROLLER_BUTTON_A:
// If we are using the joystick as a GUI mouse, A must be handled via mouse.
if (mGamepadGuiCursorEnabled)
return false;
key = MyGUI::KeyCode::Space;
break;
case SDL_CONTROLLER_BUTTON_B:
if (MyGUI::InputManager::getInstance().isModalAny())
MWBase::Environment::get().getWindowManager()->exitCurrentModal();
else
MWBase::Environment::get().getWindowManager()->exitCurrentGuiMode();
return true;
case SDL_CONTROLLER_BUTTON_X:
key = MyGUI::KeyCode::Semicolon;
break;
case SDL_CONTROLLER_BUTTON_Y:
key = MyGUI::KeyCode::Apostrophe;
break;
case SDL_CONTROLLER_BUTTON_LEFTSHOULDER:
key = MyGUI::KeyCode::Period;
break;
case SDL_CONTROLLER_BUTTON_RIGHTSHOULDER:
key = MyGUI::KeyCode::Slash;
break;
case SDL_CONTROLLER_BUTTON_LEFTSTICK:
mGamepadGuiCursorEnabled = !mGamepadGuiCursorEnabled;
MWBase::Environment::get().getWindowManager()->setCursorActive(mGamepadGuiCursorEnabled);
return true;
default:
return false;
}
// Some keys will work even when Text Input windows/modals are in focus.
if (SDL_IsTextInputActive())
return false;
MWBase::Environment::get().getWindowManager()->injectKeyPress(key, 0, false);
return true;
}
bool InputManager::gamepadToGuiControl(const SDL_ControllerAxisEvent &arg)
{
switch (arg.axis)
{
case SDL_CONTROLLER_AXIS_TRIGGERRIGHT:
if (arg.value == 32767) // Treat like a button.
MWBase::Environment::get().getWindowManager()->injectKeyPress(MyGUI::KeyCode::Minus, 0, false);
break;
case SDL_CONTROLLER_AXIS_TRIGGERLEFT:
if (arg.value == 32767) // Treat like a button.
MWBase::Environment::get().getWindowManager()->injectKeyPress(MyGUI::KeyCode::Equals, 0, false);
break;
case SDL_CONTROLLER_AXIS_LEFTX:
case SDL_CONTROLLER_AXIS_LEFTY:
case SDL_CONTROLLER_AXIS_RIGHTX:
case SDL_CONTROLLER_AXIS_RIGHTY:
// If we are using the joystick as a GUI mouse, process mouse movement elsewhere.
if (mGamepadGuiCursorEnabled)
return false;
break;
default:
return false;
}
return true;
}
2012-08-12 20:45:02 +02:00
void InputManager::channelChanged(ICS::Channel* channel, float currentValue, float previousValue)
2010-07-20 21:10:51 +02:00
{
resetIdleTime ();
2010-07-17 19:58:15 +02:00
2012-08-12 20:45:02 +02:00
int action = channel->getNumber();
if (mDragDrop && action != A_GameMenu && action != A_Inventory)
return;
if((previousValue == 1 || previousValue == 0) && (currentValue==1 || currentValue==0))
{
//Is a normal button press, so don't change it at all
}
//Otherwise only trigger button presses as they go through specific points
else if(previousValue >= .8 && currentValue < .8)
{
currentValue = 0.0;
previousValue = 1.0;
}
else if(previousValue <= .6 && currentValue > .6)
{
currentValue = 1.0;
previousValue = 0.0;
}
else
{
//If it's not switching between those values, ignore the channel change.
return;
}
2014-12-08 21:57:32 -06:00
if (mControlSwitch["playercontrols"])
{
if (action == A_Use)
{
2019-02-27 13:29:48 +00:00
if(mJoystickLastUsed && currentValue == 1.0 && actionIsActive(A_ToggleWeapon))
action = A_CycleWeaponRight;
else if (mJoystickLastUsed && currentValue == 1.0 && actionIsActive(A_ToggleSpell))
action = A_CycleSpellRight;
else
{
MWWorld::Player& player = MWBase::Environment::get().getWorld()->getPlayer();
MWMechanics::DrawState_ state = player.getDrawState();
player.setAttackingOrSpell(currentValue != 0 && state != MWMechanics::DrawState_Nothing);
2019-02-27 13:29:48 +00:00
}
}
else if (action == A_Jump)
2019-02-27 13:29:48 +00:00
{
if(mJoystickLastUsed && currentValue == 1.0 && actionIsActive(A_ToggleWeapon))
action = A_CycleWeaponLeft;
else if (mJoystickLastUsed && currentValue == 1.0 && actionIsActive(A_ToggleSpell))
action = A_CycleSpellLeft;
else
mAttemptJump = (currentValue == 1.0 && previousValue == 0.0);
}
}
2012-08-12 20:45:02 +02:00
if (currentValue == 1)
mActionManager->executeAction(action);
}
void InputManager::updateCursorMode()
2010-07-17 19:58:15 +02:00
{
2015-05-13 16:50:47 +02:00
bool grab = !MWBase::Environment::get().getWindowManager()->containsMode(MWGui::GM_MainMenu)
&& !MWBase::Environment::get().getWindowManager()->isConsoleMode();
2015-05-13 16:50:47 +02:00
bool was_relative = mInputManager->getMouseRelative();
bool is_relative = !MWBase::Environment::get().getWindowManager()->isGuiMode();
// don't keep the pointer away from the window edge in gui mode
// stop using raw mouse motions and switch to system cursor movements
2015-05-13 16:50:47 +02:00
mInputManager->setMouseRelative(is_relative);
//we let the mouse escape in the main menu
2015-05-13 16:50:47 +02:00
mInputManager->setGrabPointer(grab && (mGrabCursor || is_relative));
//we switched to non-relative mode, move our cursor to where the in-game
//cursor is
if(!is_relative && was_relative != is_relative)
{
mMouseManager->warpMouse();
}
}
void InputManager::update(float dt, bool disableControls, bool disableEvents)
{
mControlsDisabled = disableControls;
2015-05-13 16:50:47 +02:00
mInputManager->setMouseVisible(MWBase::Environment::get().getWindowManager()->getCursorVisible());
2015-05-13 16:50:47 +02:00
mInputManager->capture(disableEvents);
if (mControlsDisabled)
{
updateCursorMode();
return;
}
// update values of channels (as a result of pressed keys)
mInputBinder->update(dt);
updateCursorMode();
if (mGuiCursorEnabled && !(mJoystickLastUsed && !mGamepadGuiCursorEnabled))
2014-12-08 21:57:32 -06:00
{
float xAxis = mInputBinder->getChannel(A_MoveLeftRight)->getValue()*2.0f-1.0f;
float yAxis = mInputBinder->getChannel(A_MoveForwardBackward)->getValue()*2.0f-1.0f;
float zAxis = mInputBinder->getChannel(A_LookUpDown)->getValue()*2.0f-1.0f;
2014-12-08 21:57:32 -06:00
xAxis *= (1.5f - mInputBinder->getChannel(A_Use)->getValue());
yAxis *= (1.5f - mInputBinder->getChannel(A_Use)->getValue());
// We keep track of our own mouse position, so that moving the mouse while in
// game mode does not move the position of the GUI cursor
float xMove = xAxis * dt * 1500.0f * mInvUiScalingFactor * mGamepadCursorSpeed;
float yMove = yAxis * dt * 1500.0f * mInvUiScalingFactor * mGamepadCursorSpeed;
if (xMove != 0|| yMove != 0 || zAxis != 0)
{
int mouseWheelMove = static_cast<int>(-zAxis * dt * 1500.0f);
2014-12-08 21:57:32 -06:00
mMouseManager->injectMouseMove(xMove, yMove, mouseWheelMove);
mMouseManager->warpMouse();
MWBase::Environment::get().getWindowManager()->setCursorActive(true);
}
}
2014-12-08 21:57:32 -06:00
if (mMouseManager->update(dt, disableControls))
resetIdleTime();
2014-12-08 21:57:32 -06:00
if (mSensorManager->update(dt, mGuiCursorEnabled, mControlSwitch["playerlooking"]))
resetIdleTime();
2011-02-03 12:16:59 +01:00
// Disable movement in Gui mode
if (!(MWBase::Environment::get().getWindowManager()->isGuiMode()
|| MWBase::Environment::get().getStateManager()->getState() != MWBase::StateManager::State_Running))
2012-08-12 20:45:02 +02:00
{
// Configure player movement according to keyboard input. Actual movement will
// be done in the physics system.
if (mControlSwitch["playercontrols"])
{
MWWorld::Player& player = MWBase::Environment::get().getWorld()->getPlayer();
bool triedToMove = false;
2014-12-08 21:57:32 -06:00
bool isRunning = false;
bool alwaysRunAllowed = false;
// joystick movement
float xAxis = mInputBinder->getChannel(A_MoveLeftRight)->getValue();
float yAxis = mInputBinder->getChannel(A_MoveForwardBackward)->getValue();
2019-03-02 23:46:48 +00:00
if (xAxis != .5)
{
triedToMove = true;
player.setLeftRight((xAxis - 0.5f) * 2);
}
2019-03-02 23:46:48 +00:00
if (yAxis != .5)
{
triedToMove = true;
player.setAutoMove (false);
player.setForwardBackward((yAxis - 0.5f) * 2 * -1);
}
if (triedToMove)
mJoystickLastUsed = true;
2014-12-08 21:57:32 -06:00
// keyboard movement
isRunning = xAxis > .75 || xAxis < .25 || yAxis > .75 || yAxis < .25;
if(triedToMove) resetIdleTime();
if (actionIsActive(A_MoveLeft) != actionIsActive(A_MoveRight))
{
alwaysRunAllowed = true;
triedToMove = true;
player.setLeftRight (actionIsActive(A_MoveRight) ? 1 : -1);
}
if (actionIsActive(A_MoveForward) != actionIsActive(A_MoveBackward))
{
alwaysRunAllowed = true;
triedToMove = true;
player.setAutoMove (false);
player.setForwardBackward (actionIsActive(A_MoveForward) ? 1 : -1);
}
if (player.getAutoMove())
{
alwaysRunAllowed = true;
triedToMove = true;
player.setForwardBackward (1);
}
2013-04-07 18:04:30 +01:00
static const bool isToggleSneak = Settings::Manager::getBool("toggle sneak", "Input");
if (!isToggleSneak)
{
2019-02-27 14:03:16 -08:00
if(mJoystickLastUsed)
{
if(actionIsActive(A_Sneak))
{
if(mSneakToggleShortcutTimer) // New Sneak Button Press
{
if(mSneakToggleShortcutTimer <= 0.3f)
{
mSneakGamepadShortcut = true;
mActionManager->toggleSneaking();
2019-02-27 14:03:16 -08:00
}
else
mSneakGamepadShortcut = false;
}
if(!mActionManager->isSneaking())
mActionManager->toggleSneaking();
2019-02-27 14:03:16 -08:00
mSneakToggleShortcutTimer = 0.f;
}
else
{
if(!mSneakGamepadShortcut && mActionManager->isSneaking())
mActionManager->toggleSneaking();
2019-02-27 14:03:16 -08:00
if(mSneakToggleShortcutTimer <= 0.3f)
mSneakToggleShortcutTimer += dt;
}
}
else
player.setSneak(actionIsActive(A_Sneak));
}
if (mAttemptJump && mControlSwitch["playerjumping"])
{
player.setUpDown (1);
triedToMove = true;
mOverencumberedMessageDelay = 0.f;
}
2012-08-12 15:50:37 +04:00
if ((mActionManager->isAlwaysRunActive() && alwaysRunAllowed) || isRunning)
player.setRunState(!actionIsActive(A_Run));
else
player.setRunState(actionIsActive(A_Run));
2013-02-06 17:51:47 -08:00
// if player tried to start moving, but can't (due to being overencumbered), display a notification.
if (triedToMove)
{
MWWorld::Ptr playerPtr = MWBase::Environment::get().getWorld ()->getPlayerPtr();
mOverencumberedMessageDelay -= dt;
if (playerPtr.getClass().getEncumbrance(playerPtr) > playerPtr.getClass().getCapacity(playerPtr))
{
player.setAutoMove (false);
if (mOverencumberedMessageDelay <= 0)
{
MWBase::Environment::get().getWindowManager ()->messageBox("#{sNotifyMessage59}");
mOverencumberedMessageDelay = 1.0;
}
}
}
if (mControlSwitch["playerviewswitch"]) {
if (actionIsActive(A_TogglePOV)) {
if (mPreviewPOVDelay <= 0.5 &&
(mPreviewPOVDelay += dt) > 0.5)
{
mPreviewPOVDelay = 1.f;
MWBase::Environment::get().getWorld()->togglePreviewMode(true);
}
} else {
//disable preview mode
MWBase::Environment::get().getWorld()->togglePreviewMode(false);
if (mPreviewPOVDelay > 0.f && mPreviewPOVDelay <= 0.5) {
MWBase::Environment::get().getWorld()->togglePOV();
}
mPreviewPOVDelay = 0.f;
2019-02-27 13:29:48 +00:00
mGamepadZoom = 0;
}
if(mGamepadZoom)
{
MWBase::Environment::get().getWorld()->changeVanityModeScale(mGamepadZoom);
MWBase::Environment::get().getWorld()->setCameraDistance(mGamepadZoom, true, true);
}
}
}
if (actionIsActive(A_MoveForward) ||
actionIsActive(A_MoveBackward) ||
actionIsActive(A_MoveLeft) ||
actionIsActive(A_MoveRight) ||
actionIsActive(A_Jump) ||
actionIsActive(A_Sneak) ||
actionIsActive(A_TogglePOV) ||
actionIsActive(A_ZoomIn) ||
actionIsActive(A_ZoomOut) )
{
resetIdleTime();
} else {
updateIdleTime(dt);
}
}
2019-02-27 13:29:48 +00:00
else
mGamepadZoom = 0;
mAttemptJump = false; // Can only jump on first frame input is on
2012-08-12 20:45:02 +02:00
}
void InputManager::setDragDrop(bool dragDrop)
{
mDragDrop = dragDrop;
2010-07-17 19:58:15 +02:00
}
void InputManager::setGamepadGuiCursorEnabled(bool enabled)
{
mGamepadGuiCursorEnabled = enabled;
}
2012-08-12 20:45:02 +02:00
void InputManager::changeInputMode(bool guiMode)
{
mGuiCursorEnabled = guiMode;
mMouseManager->setGuiCursorEnabled(guiMode);
mMouseManager->setMouseLookEnabled(!guiMode);
if (guiMode)
MWBase::Environment::get().getWindowManager()->showCrosshair(false);
MWBase::Environment::get().getWindowManager()->setCursorVisible(guiMode && (!mJoystickLastUsed || mGamepadGuiCursorEnabled));
// if not in gui mode, the camera decides whether to show crosshair or not.
2012-08-12 20:45:02 +02:00
}
void InputManager::processChangedSettings(const Settings::CategorySettingVector& changed)
{
bool changeRes = false;
2012-08-12 20:45:02 +02:00
for (Settings::CategorySettingVector::const_iterator it = changed.begin();
it != changed.end(); ++it)
{
if (it->first == "Input" && it->second == "grab cursor")
mGrabCursor = Settings::Manager::getBool("grab cursor", "Input");
if (it->first == "Input" && it->second == "enable controller")
mJoystickEnabled = Settings::Manager::getBool("enable controller", "Input");
if (it->first == "Video" && (
it->second == "resolution x"
|| it->second == "resolution y"
|| it->second == "fullscreen"
|| it->second == "window border"))
changeRes = true;
if (it->first == "Video" && it->second == "vsync")
mVideoWrapper->setSyncToVBlank(Settings::Manager::getBool("vsync", "Video"));
if (it->first == "Video" && (it->second == "gamma" || it->second == "contrast"))
mVideoWrapper->setGammaContrast(Settings::Manager::getFloat("gamma", "Video"),
Settings::Manager::getFloat("contrast", "Video"));
}
if (changeRes)
{
mVideoWrapper->setVideoMode(Settings::Manager::getInt("resolution x", "Video"),
Settings::Manager::getInt("resolution y", "Video"),
Settings::Manager::getBool("fullscreen", "Video"),
Settings::Manager::getBool("window border", "Video"));
}
mMouseManager->processChangedSettings(changed);
mSensorManager->processChangedSettings(changed);
}
bool InputManager::getControlSwitch (const std::string& sw)
{
return mControlSwitch[sw];
}
2012-08-12 20:45:02 +02:00
void InputManager::toggleControlSwitch (const std::string& sw, bool value)
{
MWWorld::Player& player = MWBase::Environment::get().getWorld()->getPlayer();
/// \note 7 switches at all, if-else is relevant
if (sw == "playercontrols" && !value)
{
player.setLeftRight(0);
player.setForwardBackward(0);
player.setAutoMove(false);
player.setUpDown(0);
}
else if (sw == "playerjumping" && !value)
{
/// \fixme maybe crouching at this time
player.setUpDown(0);
}
else if (sw == "vanitymode")
{
MWBase::Environment::get().getWorld()->allowVanityMode(value);
}
else if (sw == "playerlooking" && !value)
{
MWBase::Environment::get().getWorld()->rotateObject(player.getPlayer(), 0.f, 0.f, 0.f);
}
mControlSwitch[sw] = value;
}
2014-02-13 15:08:40 +01:00
void InputManager::keyPressed( const SDL_KeyboardEvent &arg )
2012-08-12 20:45:02 +02:00
{
// HACK: to make Morrowind's default keybinding for the console work without printing an extra "^" upon closing
// This assumes that SDL_TextInput events always come *after* the key event
// (which is somewhat reasonable, and hopefully true for all SDL platforms)
OIS::KeyCode kc = mInputManager->sdl2OISKeyCode(arg.keysym.sym);
if (mInputBinder->getKeyBinding(mInputBinder->getControl(A_Console), ICS::Control::INCREASE)
2014-09-13 20:39:32 +02:00
== arg.keysym.scancode
&& MWBase::Environment::get().getWindowManager()->isConsoleMode())
SDL_StopTextInput();
bool consumed = false;
2017-09-27 21:30:12 +02:00
if (kc != OIS::KC_UNASSIGNED && !mInputBinder->detectingBindingState())
{
consumed = MWBase::Environment::get().getWindowManager()->injectKeyPress(MyGUI::KeyCode::Enum(kc), 0, arg.repeat);
if (SDL_IsTextInputActive() && // Little trick to check if key is printable
( !(SDLK_SCANCODE_MASK & arg.keysym.sym) && std::isprint(arg.keysym.sym)))
consumed = true;
setPlayerControlsEnabled(!consumed);
}
if (arg.repeat)
return;
if (!mControlsDisabled && !consumed)
mInputBinder->keyPressed (arg);
2014-12-08 21:57:32 -06:00
mJoystickLastUsed = false;
2012-08-12 20:45:02 +02:00
}
2013-06-16 19:43:59 +02:00
void InputManager::textInput(const SDL_TextInputEvent &arg)
{
MyGUI::UString ustring(&arg.text[0]);
MyGUI::UString::utf32string utf32string = ustring.asUTF32();
for (MyGUI::UString::utf32string::const_iterator it = utf32string.begin(); it != utf32string.end(); ++it)
2013-06-16 19:43:59 +02:00
MyGUI::InputManager::getInstance().injectKeyPress(MyGUI::KeyCode::None, *it);
}
2014-02-13 15:08:40 +01:00
void InputManager::keyReleased(const SDL_KeyboardEvent &arg )
{
2014-12-08 21:57:32 -06:00
mJoystickLastUsed = false;
OIS::KeyCode kc = mInputManager->sdl2OISKeyCode(arg.keysym.sym);
2012-08-12 20:45:02 +02:00
2017-09-27 21:30:12 +02:00
if (!mInputBinder->detectingBindingState())
setPlayerControlsEnabled(!MyGUI::InputManager::getInstance().injectKeyRelease(MyGUI::KeyCode::Enum(kc)));
mInputBinder->keyReleased (arg);
}
2012-08-12 20:45:02 +02:00
void InputManager::buttonPressed(int deviceID, const SDL_ControllerButtonEvent &arg )
2014-12-08 21:57:32 -06:00
{
if (!mJoystickEnabled || mInputBinder->detectingBindingState())
return;
2014-12-08 21:57:32 -06:00
mJoystickLastUsed = true;
if (MWBase::Environment::get().getWindowManager()->isGuiMode())
2014-12-08 21:57:32 -06:00
{
if (gamepadToGuiControl(arg))
return;
if (mGamepadGuiCursorEnabled)
2014-12-08 21:57:32 -06:00
{
// Temporary mouse binding until keyboard controls are available:
if (arg.button == SDL_CONTROLLER_BUTTON_A) // We'll pretend that A is left click.
2014-12-08 21:57:32 -06:00
{
bool mousePressSuccess = mMouseManager->injectMouseButtonPress(SDL_BUTTON_LEFT);
if (MyGUI::InputManager::getInstance().getMouseFocusWidget())
{
MyGUI::Button* b = MyGUI::InputManager::getInstance().getMouseFocusWidget()->castType<MyGUI::Button>(false);
if (b && b->getEnabled())
MWBase::Environment::get().getWindowManager()->playSound("Menu Click");
}
setPlayerControlsEnabled(!mousePressSuccess);
2014-12-08 21:57:32 -06:00
}
}
}
else
setPlayerControlsEnabled(true);
2014-12-08 21:57:32 -06:00
2015-03-03 11:23:50 +01:00
//esc, to leave initial movie screen
2015-05-13 16:50:47 +02:00
OIS::KeyCode kc = mInputManager->sdl2OISKeyCode(SDLK_ESCAPE);
setPlayerControlsEnabled(!MyGUI::InputManager::getInstance().injectKeyPress(MyGUI::KeyCode::Enum(kc), 0));
2014-12-08 21:57:32 -06:00
if (!mControlsDisabled)
mInputBinder->buttonPressed(deviceID, arg);
2014-12-08 21:57:32 -06:00
}
void InputManager::buttonReleased(int deviceID, const SDL_ControllerButtonEvent &arg )
2014-12-08 21:57:32 -06:00
{
if(mInputBinder->detectingBindingState())
{
mInputBinder->buttonReleased(deviceID, arg);
return;
}
if (!mJoystickEnabled || mControlsDisabled)
return;
2014-12-08 21:57:32 -06:00
mJoystickLastUsed = true;
if (MWBase::Environment::get().getWindowManager()->isGuiMode())
2014-12-08 21:57:32 -06:00
{
if (mGamepadGuiCursorEnabled)
{
// Temporary mouse binding until keyboard controls are available:
if (arg.button == SDL_CONTROLLER_BUTTON_A) // We'll pretend that A is left click.
{
bool mousePressSuccess = mMouseManager->injectMouseButtonRelease(SDL_BUTTON_LEFT);
if (mInputBinder->detectingBindingState()) // If the player just triggered binding, don't let button release bind.
return;
2014-12-08 21:57:32 -06:00
setPlayerControlsEnabled(!mousePressSuccess);
}
}
2014-12-08 21:57:32 -06:00
}
else
setPlayerControlsEnabled(true);
2014-12-08 21:57:32 -06:00
//esc, to leave initial movie screen
2015-05-13 16:50:47 +02:00
OIS::KeyCode kc = mInputManager->sdl2OISKeyCode(SDLK_ESCAPE);
setPlayerControlsEnabled(!MyGUI::InputManager::getInstance().injectKeyRelease(MyGUI::KeyCode::Enum(kc)));
mInputBinder->buttonReleased(deviceID, arg);
2014-12-08 21:57:32 -06:00
}
void InputManager::axisMoved(int deviceID, const SDL_ControllerAxisEvent &arg )
2014-12-08 21:57:32 -06:00
{
if(!mJoystickEnabled || mControlsDisabled)
return;
mJoystickLastUsed = true;
if (MWBase::Environment::get().getWindowManager()->isGuiMode())
{
gamepadToGuiControl(arg);
}
else
2019-02-27 13:29:48 +00:00
{
if(mPreviewPOVDelay == 1.f && arg.value) // Preview Mode Gamepad Zooming
2019-02-27 13:29:48 +00:00
{
if(arg.axis == SDL_CONTROLLER_AXIS_TRIGGERRIGHT)
2019-02-27 13:29:48 +00:00
{
2019-11-02 19:47:05 +03:00
mGamepadZoom = arg.value * 0.85f / 1000.f;
return; // Do not propagate event.
2019-02-27 13:29:48 +00:00
}
else if(arg.axis == SDL_CONTROLLER_AXIS_TRIGGERLEFT)
2019-02-27 13:29:48 +00:00
{
2019-11-02 19:47:05 +03:00
mGamepadZoom = -arg.value * 0.85f / 1000.f;
return; // Do not propagate event.
2019-02-27 13:29:48 +00:00
}
}
}
mInputBinder->axisMoved(deviceID, arg);
2014-12-08 21:57:32 -06:00
}
void InputManager::controllerAdded(int deviceID, const SDL_ControllerDeviceEvent &arg)
2014-12-08 21:57:32 -06:00
{
mInputBinder->controllerAdded(deviceID, arg);
2014-12-08 21:57:32 -06:00
}
void InputManager::controllerRemoved(const SDL_ControllerDeviceEvent &arg)
{
mInputBinder->controllerRemoved(arg);
}
2013-07-29 02:32:08 +02:00
void InputManager::windowFocusChange(bool have_focus)
{
}
2013-07-29 02:32:08 +02:00
void InputManager::windowVisibilityChange(bool visible)
{
mWindowVisible = visible;
2013-07-29 02:32:08 +02:00
}
void InputManager::windowResized(int x, int y)
{
// Note: this is a side effect of resolution change or window resize.
// There is no need to track these changes.
2015-05-13 16:50:47 +02:00
Settings::Manager::setInt("resolution x", "Video", x);
Settings::Manager::setInt("resolution y", "Video", y);
2019-05-04 21:38:36 +04:00
Settings::Manager::resetPendingChange("resolution x", "Video");
Settings::Manager::resetPendingChange("resolution y", "Video");
2015-05-13 16:50:47 +02:00
MWBase::Environment::get().getWindowManager()->windowResized(x, y);
// We should reload TrueType fonts to fit new resolution
MWBase::Environment::get().getWindowManager()->loadUserFonts();
}
void InputManager::windowClosed()
{
MWBase::Environment::get().getStateManager()->requestQuit();
}
2012-08-18 01:31:57 +04:00
void InputManager::resetIdleTime()
{
2013-04-27 01:24:36 -07:00
if (mTimeIdle < 0)
MWBase::Environment::get().getWorld()->toggleVanityMode(false);
2012-08-18 01:31:57 +04:00
mTimeIdle = 0.f;
}
void InputManager::updateIdleTime(float dt)
{
2014-01-02 00:13:23 +01:00
static const float vanityDelay = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>()
2018-08-29 18:38:12 +03:00
.find("fVanityDelay")->mValue.getFloat();
2013-04-27 01:24:36 -07:00
if (mTimeIdle >= 0.f)
2012-08-18 01:31:57 +04:00
mTimeIdle += dt;
2014-01-02 00:13:23 +01:00
if (mTimeIdle > vanityDelay) {
2013-04-27 01:24:36 -07:00
MWBase::Environment::get().getWorld()->toggleVanityMode(true);
2012-08-18 01:31:57 +04:00
mTimeIdle = -1.f;
}
}
2012-08-12 20:45:02 +02:00
bool InputManager::actionIsActive (int id)
{
2014-12-09 00:02:18 -06:00
return (mInputBinder->getChannel (id)->getValue ()==1.0);
2012-08-12 20:45:02 +02:00
}
void InputManager::loadKeyDefaults (bool force)
{
// using hardcoded key defaults is inevitable, if we want the configuration files to stay valid
// across different versions of OpenMW (in the case where another input action is added)
2014-09-13 20:39:32 +02:00
std::map<int, SDL_Scancode> defaultKeyBindings;
//Gets the Keyvalue from the Scancode; gives the button in the same place reguardless of keyboard format
2014-09-13 20:39:32 +02:00
defaultKeyBindings[A_Activate] = SDL_SCANCODE_SPACE;
defaultKeyBindings[A_MoveBackward] = SDL_SCANCODE_S;
defaultKeyBindings[A_MoveForward] = SDL_SCANCODE_W;
defaultKeyBindings[A_MoveLeft] = SDL_SCANCODE_A;
defaultKeyBindings[A_MoveRight] = SDL_SCANCODE_D;
defaultKeyBindings[A_ToggleWeapon] = SDL_SCANCODE_F;
defaultKeyBindings[A_ToggleSpell] = SDL_SCANCODE_R;
defaultKeyBindings[A_CycleSpellLeft] = SDL_SCANCODE_MINUS;
defaultKeyBindings[A_CycleSpellRight] = SDL_SCANCODE_EQUALS;
defaultKeyBindings[A_CycleWeaponLeft] = SDL_SCANCODE_LEFTBRACKET;
defaultKeyBindings[A_CycleWeaponRight] = SDL_SCANCODE_RIGHTBRACKET;
2014-09-13 20:39:32 +02:00
defaultKeyBindings[A_QuickKeysMenu] = SDL_SCANCODE_F1;
defaultKeyBindings[A_Console] = SDL_SCANCODE_GRAVE;
defaultKeyBindings[A_Run] = SDL_SCANCODE_LSHIFT;
defaultKeyBindings[A_Sneak] = SDL_SCANCODE_LCTRL;
defaultKeyBindings[A_AutoMove] = SDL_SCANCODE_Q;
defaultKeyBindings[A_Jump] = SDL_SCANCODE_E;
defaultKeyBindings[A_Journal] = SDL_SCANCODE_J;
defaultKeyBindings[A_Rest] = SDL_SCANCODE_T;
defaultKeyBindings[A_GameMenu] = SDL_SCANCODE_ESCAPE;
defaultKeyBindings[A_TogglePOV] = SDL_SCANCODE_TAB;
defaultKeyBindings[A_QuickKey1] = SDL_SCANCODE_1;
defaultKeyBindings[A_QuickKey2] = SDL_SCANCODE_2;
defaultKeyBindings[A_QuickKey3] = SDL_SCANCODE_3;
defaultKeyBindings[A_QuickKey4] = SDL_SCANCODE_4;
defaultKeyBindings[A_QuickKey5] = SDL_SCANCODE_5;
defaultKeyBindings[A_QuickKey6] = SDL_SCANCODE_6;
defaultKeyBindings[A_QuickKey7] = SDL_SCANCODE_7;
defaultKeyBindings[A_QuickKey8] = SDL_SCANCODE_8;
defaultKeyBindings[A_QuickKey9] = SDL_SCANCODE_9;
defaultKeyBindings[A_QuickKey10] = SDL_SCANCODE_0;
defaultKeyBindings[A_Screenshot] = SDL_SCANCODE_F12;
defaultKeyBindings[A_ToggleHUD] = SDL_SCANCODE_F11;
defaultKeyBindings[A_ToggleDebug] = SDL_SCANCODE_F10;
2014-09-13 20:39:32 +02:00
defaultKeyBindings[A_AlwaysRun] = SDL_SCANCODE_CAPSLOCK;
defaultKeyBindings[A_QuickSave] = SDL_SCANCODE_F5;
defaultKeyBindings[A_QuickLoad] = SDL_SCANCODE_F9;
std::map<int, int> defaultMouseButtonBindings;
defaultMouseButtonBindings[A_Inventory] = SDL_BUTTON_RIGHT;
defaultMouseButtonBindings[A_Use] = SDL_BUTTON_LEFT;
std::map<int, ICS::InputControlSystem::MouseWheelClick> defaultMouseWheelBindings;
defaultMouseWheelBindings[A_ZoomIn] = ICS::InputControlSystem::MouseWheelClick::UP;
defaultMouseWheelBindings[A_ZoomOut] = ICS::InputControlSystem::MouseWheelClick::DOWN;
2012-08-13 01:26:15 +02:00
for (int i = 0; i < A_Last; ++i)
{
ICS::Control* control;
bool controlExists = mInputBinder->getChannel(i)->getControlsCount () != 0;
if (!controlExists)
{
2017-05-06 23:05:13 +02:00
control = new ICS::Control(std::to_string(i), false, true, 0, ICS::ICS_MAX, ICS::ICS_MAX);
mInputBinder->addControl(control);
control->attachChannel(mInputBinder->getChannel(i), ICS::Channel::DIRECT);
}
else
{
control = mInputBinder->getChannel(i)->getAttachedControls ().front().control;
}
2012-08-27 15:51:01 +02:00
if (!controlExists || force ||
2014-09-13 20:39:32 +02:00
( mInputBinder->getKeyBinding (control, ICS::Control::INCREASE) == SDL_SCANCODE_UNKNOWN
&& mInputBinder->getMouseButtonBinding (control, ICS::Control::INCREASE) == ICS_MAX_DEVICE_BUTTONS
&& mInputBinder->getMouseWheelBinding(control, ICS::Control::INCREASE) == ICS::InputControlSystem::MouseWheelClick::UNASSIGNED
2012-08-27 15:51:01 +02:00
))
{
2014-12-08 21:57:32 -06:00
clearAllKeyBindings(control);
if (defaultKeyBindings.find(i) != defaultKeyBindings.end()
&& (force || !mInputBinder->isKeyBound(defaultKeyBindings[i])))
2014-12-08 21:57:32 -06:00
{
2014-12-09 11:16:17 -06:00
control->setInitialValue(0.0f);
2014-09-13 20:39:32 +02:00
mInputBinder->addKeyBinding(control, defaultKeyBindings[i], ICS::Control::INCREASE);
2014-12-08 21:57:32 -06:00
}
else if (defaultMouseButtonBindings.find(i) != defaultMouseButtonBindings.end()
&& (force || !mInputBinder->isMouseButtonBound(defaultMouseButtonBindings[i])))
2014-12-08 21:57:32 -06:00
{
2014-12-09 11:16:17 -06:00
control->setInitialValue(0.0f);
mInputBinder->addMouseButtonBinding (control, defaultMouseButtonBindings[i], ICS::Control::INCREASE);
2014-12-08 21:57:32 -06:00
}
else if (defaultMouseWheelBindings.find(i) != defaultMouseWheelBindings.end()
&& (force || !mInputBinder->isMouseWheelBound(defaultMouseWheelBindings[i])))
{
control->setInitialValue(0.f);
mInputBinder->addMouseWheelBinding(control, defaultMouseWheelBindings[i], ICS::Control::INCREASE);
}
2017-09-24 15:32:44 +02:00
if (i == A_LookLeftRight && !mInputBinder->isKeyBound(SDL_SCANCODE_KP_4) && !mInputBinder->isKeyBound(SDL_SCANCODE_KP_6))
{
mInputBinder->addKeyBinding(control, SDL_SCANCODE_KP_6, ICS::Control::INCREASE);
mInputBinder->addKeyBinding(control, SDL_SCANCODE_KP_4, ICS::Control::DECREASE);
}
if (i == A_LookUpDown && !mInputBinder->isKeyBound(SDL_SCANCODE_KP_8) && !mInputBinder->isKeyBound(SDL_SCANCODE_KP_2))
{
mInputBinder->addKeyBinding(control, SDL_SCANCODE_KP_2, ICS::Control::INCREASE);
mInputBinder->addKeyBinding(control, SDL_SCANCODE_KP_8, ICS::Control::DECREASE);
}
2014-12-08 21:57:32 -06:00
}
}
}
void InputManager::loadControllerDefaults(bool force)
{
// using hardcoded key defaults is inevitable, if we want the configuration files to stay valid
// across different versions of OpenMW (in the case where another input action is added)
std::map<int, int> defaultButtonBindings;
defaultButtonBindings[A_Activate] = SDL_CONTROLLER_BUTTON_A;
defaultButtonBindings[A_ToggleWeapon] = SDL_CONTROLLER_BUTTON_X;
2019-02-27 13:29:48 +00:00
defaultButtonBindings[A_ToggleSpell] = SDL_CONTROLLER_BUTTON_Y;
2014-12-08 21:57:32 -06:00
//defaultButtonBindings[A_QuickButtonsMenu] = SDL_GetButtonFromScancode(SDL_SCANCODE_F1); // Need to implement, should be ToggleSpell(5) AND Wait(9)
2019-02-27 13:29:48 +00:00
defaultButtonBindings[A_Sneak] = SDL_CONTROLLER_BUTTON_LEFTSTICK;
defaultButtonBindings[A_Journal] = SDL_CONTROLLER_BUTTON_LEFTSHOULDER;
defaultButtonBindings[A_Rest] = SDL_CONTROLLER_BUTTON_RIGHTSHOULDER;
defaultButtonBindings[A_TogglePOV] = SDL_CONTROLLER_BUTTON_RIGHTSTICK;
2014-12-08 21:57:32 -06:00
defaultButtonBindings[A_Inventory] = SDL_CONTROLLER_BUTTON_B;
defaultButtonBindings[A_GameMenu] = SDL_CONTROLLER_BUTTON_START;
defaultButtonBindings[A_QuickSave] = SDL_CONTROLLER_BUTTON_GUIDE;
defaultButtonBindings[A_MoveForward] = SDL_CONTROLLER_BUTTON_DPAD_UP;
defaultButtonBindings[A_MoveLeft] = SDL_CONTROLLER_BUTTON_DPAD_LEFT;
defaultButtonBindings[A_MoveBackward] = SDL_CONTROLLER_BUTTON_DPAD_DOWN;
defaultButtonBindings[A_MoveRight] = SDL_CONTROLLER_BUTTON_DPAD_RIGHT;
2014-12-08 21:57:32 -06:00
std::map<int, int> defaultAxisBindings;
defaultAxisBindings[A_MoveForwardBackward] = SDL_CONTROLLER_AXIS_LEFTY;
defaultAxisBindings[A_MoveLeftRight] = SDL_CONTROLLER_AXIS_LEFTX;
defaultAxisBindings[A_LookUpDown] = SDL_CONTROLLER_AXIS_RIGHTY;
defaultAxisBindings[A_LookLeftRight] = SDL_CONTROLLER_AXIS_RIGHTX;
defaultAxisBindings[A_Use] = SDL_CONTROLLER_AXIS_TRIGGERRIGHT;
2019-02-27 13:29:48 +00:00
defaultAxisBindings[A_Jump] = SDL_CONTROLLER_AXIS_TRIGGERLEFT;
2014-12-08 21:57:32 -06:00
for (int i = 0; i < A_Last; i++)
{
ICS::Control* control;
bool controlExists = mInputBinder->getChannel(i)->getControlsCount () != 0;
if (!controlExists)
{
2015-03-03 11:23:50 +01:00
float initial;
if (defaultAxisBindings.find(i) == defaultAxisBindings.end())
2015-03-03 11:23:50 +01:00
initial = 0.0f;
else initial = 0.5f;
2017-05-06 23:05:13 +02:00
control = new ICS::Control(std::to_string(i), false, true, initial, ICS::ICS_MAX, ICS::ICS_MAX);
2014-12-08 21:57:32 -06:00
mInputBinder->addControl(control);
control->attachChannel(mInputBinder->getChannel(i), ICS::Channel::DIRECT);
}
else
{
control = mInputBinder->getChannel(i)->getAttachedControls ().front().control;
}
if (!controlExists || force || ( mInputBinder->getJoystickAxisBinding (control, mFakeDeviceID, ICS::Control::INCREASE) == ICS::InputControlSystem::UNASSIGNED && mInputBinder->getJoystickButtonBinding (control, mFakeDeviceID, ICS::Control::INCREASE) == ICS_MAX_DEVICE_BUTTONS ))
2014-12-08 21:57:32 -06:00
{
clearAllControllerBindings(control);
if (defaultButtonBindings.find(i) != defaultButtonBindings.end()
&& (force || !mInputBinder->isJoystickButtonBound(mFakeDeviceID, defaultButtonBindings[i])))
2014-12-08 21:57:32 -06:00
{
2014-12-09 11:16:17 -06:00
control->setInitialValue(0.0f);
mInputBinder->addJoystickButtonBinding(control, mFakeDeviceID, defaultButtonBindings[i], ICS::Control::INCREASE);
2014-12-08 21:57:32 -06:00
}
else if (defaultAxisBindings.find(i) != defaultAxisBindings.end() && (force || !mInputBinder->isJoystickAxisBound(mFakeDeviceID, defaultAxisBindings[i])))
2014-12-08 21:57:32 -06:00
{
2014-12-09 11:16:17 -06:00
control->setValue(0.5f);
control->setInitialValue(0.5f);
mInputBinder->addJoystickAxisBinding(control, mFakeDeviceID, defaultAxisBindings[i], ICS::Control::INCREASE);
2014-12-08 21:57:32 -06:00
}
}
}
}
2012-08-13 01:26:15 +02:00
std::string InputManager::getActionDescription (int action)
{
std::map<int, std::string> descriptions;
if (action == A_Screenshot)
return "Screenshot";
else if (action == A_ZoomIn)
return "Zoom In";
else if (action == A_ZoomOut)
return "Zoom Out";
2020-01-10 14:29:04 +04:00
else if (action == A_ToggleHUD)
return "Toggle HUD";
2013-05-19 18:40:37 +02:00
descriptions[A_Use] = "sUse";
2012-08-13 01:26:15 +02:00
descriptions[A_Activate] = "sActivate";
descriptions[A_MoveBackward] = "sBack";
descriptions[A_MoveForward] = "sForward";
descriptions[A_MoveLeft] = "sLeft";
descriptions[A_MoveRight] = "sRight";
descriptions[A_ToggleWeapon] = "sReady_Weapon";
descriptions[A_ToggleSpell] = "sReady_Magic";
descriptions[A_CycleSpellLeft] = "sPrevSpell";
descriptions[A_CycleSpellRight] = "sNextSpell";
descriptions[A_CycleWeaponLeft] = "sPrevWeapon";
descriptions[A_CycleWeaponRight] = "sNextWeapon";
2012-08-13 01:26:15 +02:00
descriptions[A_Console] = "sConsoleTitle";
2013-02-06 17:51:47 -08:00
descriptions[A_Run] = "sRun";
2013-03-06 16:58:56 +01:00
descriptions[A_Sneak] = "sCrouch_Sneak";
2012-08-13 01:26:15 +02:00
descriptions[A_AutoMove] = "sAuto_Run";
descriptions[A_Jump] = "sJump";
descriptions[A_Journal] = "sJournal";
descriptions[A_Rest] = "sRestKey";
descriptions[A_Inventory] = "sInventory";
descriptions[A_TogglePOV] = "sTogglePOVCmd";
descriptions[A_QuickKeysMenu] = "sQuickMenu";
descriptions[A_QuickKey1] = "sQuick1Cmd";
descriptions[A_QuickKey2] = "sQuick2Cmd";
descriptions[A_QuickKey3] = "sQuick3Cmd";
descriptions[A_QuickKey4] = "sQuick4Cmd";
descriptions[A_QuickKey5] = "sQuick5Cmd";
descriptions[A_QuickKey6] = "sQuick6Cmd";
descriptions[A_QuickKey7] = "sQuick7Cmd";
descriptions[A_QuickKey8] = "sQuick8Cmd";
descriptions[A_QuickKey9] = "sQuick9Cmd";
descriptions[A_QuickKey10] = "sQuick10Cmd";
2013-03-14 20:27:16 +01:00
descriptions[A_AlwaysRun] = "sAlways_Run";
2014-04-23 21:02:09 -04:00
descriptions[A_QuickSave] = "sQuickSaveCmd";
descriptions[A_QuickLoad] = "sQuickLoadCmd";
2012-08-13 01:26:15 +02:00
if (descriptions[action] == "")
return ""; // not configurable
return "#{" + descriptions[action] + "}";
}
2014-12-08 21:57:32 -06:00
std::string InputManager::getActionKeyBindingName (int action)
2012-08-13 01:26:15 +02:00
{
if (mInputBinder->getChannel (action)->getControlsCount () == 0)
2012-08-13 01:26:15 +02:00
return "#{sNone}";
ICS::Control* c = mInputBinder->getChannel (action)->getAttachedControls ().front().control;
2012-08-13 01:26:15 +02:00
SDL_Scancode key = mInputBinder->getKeyBinding (c, ICS::Control::INCREASE);
unsigned int mouse = mInputBinder->getMouseButtonBinding (c, ICS::Control::INCREASE);
ICS::InputControlSystem::MouseWheelClick wheel = mInputBinder->getMouseWheelBinding(c, ICS::Control::INCREASE);
if (key != SDL_SCANCODE_UNKNOWN)
return MyGUI::TextIterator::toTagsString(mInputBinder->scancodeToString (key));
else if (mouse != ICS_MAX_DEVICE_BUTTONS)
2017-05-06 23:05:13 +02:00
return "#{sMouse} " + std::to_string(mouse);
else if (wheel != ICS::InputControlSystem::MouseWheelClick::UNASSIGNED)
switch (wheel)
{
case ICS::InputControlSystem::MouseWheelClick::UP:
return "Mouse Wheel Up";
case ICS::InputControlSystem::MouseWheelClick::DOWN:
return "Mouse Wheel Down";
case ICS::InputControlSystem::MouseWheelClick::RIGHT:
return "Mouse Wheel Right";
case ICS::InputControlSystem::MouseWheelClick::LEFT:
return "Mouse Wheel Left";
default:
return "#{sNone}";
}
2012-08-13 01:26:15 +02:00
else
return "#{sNone}";
}
2014-12-08 21:57:32 -06:00
std::string InputManager::getActionControllerBindingName (int action)
{
if (mInputBinder->getChannel (action)->getControlsCount () == 0)
return "#{sNone}";
ICS::Control* c = mInputBinder->getChannel (action)->getAttachedControls ().front().control;
if (mInputBinder->getJoystickAxisBinding (c, mFakeDeviceID, ICS::Control::INCREASE) != ICS::InputControlSystem::UNASSIGNED)
return sdlControllerAxisToString(mInputBinder->getJoystickAxisBinding (c, mFakeDeviceID, ICS::Control::INCREASE));
else if (mInputBinder->getJoystickButtonBinding (c, mFakeDeviceID, ICS::Control::INCREASE) != ICS_MAX_DEVICE_BUTTONS )
return sdlControllerButtonToString(mInputBinder->getJoystickButtonBinding (c, mFakeDeviceID, ICS::Control::INCREASE));
2014-12-08 21:57:32 -06:00
else
return "#{sNone}";
}
std::vector<int> InputManager::getActionKeySorting()
2012-08-13 01:26:15 +02:00
{
std::vector<int> ret;
ret.push_back(A_MoveForward);
ret.push_back(A_MoveBackward);
ret.push_back(A_MoveLeft);
ret.push_back(A_MoveRight);
ret.push_back(A_TogglePOV);
ret.push_back(A_ZoomIn);
ret.push_back(A_ZoomOut);
2013-02-06 17:51:47 -08:00
ret.push_back(A_Run);
2013-03-14 20:27:16 +01:00
ret.push_back(A_AlwaysRun);
2013-03-06 16:58:56 +01:00
ret.push_back(A_Sneak);
2012-08-13 01:26:15 +02:00
ret.push_back(A_Activate);
2013-05-19 18:40:37 +02:00
ret.push_back(A_Use);
2012-08-13 01:26:15 +02:00
ret.push_back(A_ToggleWeapon);
ret.push_back(A_ToggleSpell);
ret.push_back(A_CycleSpellLeft);
ret.push_back(A_CycleSpellRight);
ret.push_back(A_CycleWeaponLeft);
ret.push_back(A_CycleWeaponRight);
2012-08-13 01:26:15 +02:00
ret.push_back(A_AutoMove);
ret.push_back(A_Jump);
ret.push_back(A_Inventory);
ret.push_back(A_Journal);
ret.push_back(A_Rest);
ret.push_back(A_Console);
2014-04-23 21:02:09 -04:00
ret.push_back(A_QuickSave);
ret.push_back(A_QuickLoad);
2020-01-10 14:29:04 +04:00
ret.push_back(A_ToggleHUD);
ret.push_back(A_Screenshot);
ret.push_back(A_QuickKeysMenu);
ret.push_back(A_QuickKey1);
ret.push_back(A_QuickKey2);
ret.push_back(A_QuickKey3);
ret.push_back(A_QuickKey4);
ret.push_back(A_QuickKey5);
ret.push_back(A_QuickKey6);
ret.push_back(A_QuickKey7);
ret.push_back(A_QuickKey8);
ret.push_back(A_QuickKey9);
ret.push_back(A_QuickKey10);
2012-08-13 01:26:15 +02:00
return ret;
}
2014-12-08 21:57:32 -06:00
std::vector<int> InputManager::getActionControllerSorting()
{
std::vector<int> ret;
ret.push_back(A_TogglePOV);
ret.push_back(A_ZoomIn);
ret.push_back(A_ZoomOut);
2014-12-08 21:57:32 -06:00
ret.push_back(A_Sneak);
ret.push_back(A_Activate);
ret.push_back(A_Use);
ret.push_back(A_ToggleWeapon);
ret.push_back(A_ToggleSpell);
ret.push_back(A_AutoMove);
ret.push_back(A_Jump);
ret.push_back(A_Inventory);
ret.push_back(A_Journal);
ret.push_back(A_Rest);
ret.push_back(A_QuickSave);
ret.push_back(A_QuickLoad);
2020-01-10 14:29:04 +04:00
ret.push_back(A_ToggleHUD);
2014-12-08 21:57:32 -06:00
ret.push_back(A_Screenshot);
ret.push_back(A_QuickKeysMenu);
ret.push_back(A_QuickKey1);
ret.push_back(A_QuickKey2);
ret.push_back(A_QuickKey3);
ret.push_back(A_QuickKey4);
ret.push_back(A_QuickKey5);
ret.push_back(A_QuickKey6);
ret.push_back(A_QuickKey7);
ret.push_back(A_QuickKey8);
ret.push_back(A_QuickKey9);
ret.push_back(A_QuickKey10);
ret.push_back(A_CycleSpellLeft);
ret.push_back(A_CycleSpellRight);
ret.push_back(A_CycleWeaponLeft);
ret.push_back(A_CycleWeaponRight);
2014-12-08 21:57:32 -06:00
return ret;
}
2012-08-13 01:26:15 +02:00
2014-12-08 21:57:32 -06:00
void InputManager::enableDetectingBindingMode (int action, bool keyboard)
2012-08-13 01:26:15 +02:00
{
2014-12-08 21:57:32 -06:00
mDetectingKeyboard = keyboard;
ICS::Control* c = mInputBinder->getChannel (action)->getAttachedControls ().front().control;
mInputBinder->enableDetectingBindingState (c, ICS::Control::INCREASE);
2012-08-13 01:26:15 +02:00
}
void InputManager::keyBindingDetected(ICS::InputControlSystem* ICS, ICS::Control* control
2014-09-13 20:39:32 +02:00
, SDL_Scancode key, ICS::Control::ControlChangingDirection direction)
2012-08-13 01:26:15 +02:00
{
//Disallow binding escape key
2014-09-13 20:39:32 +02:00
if(key==SDL_SCANCODE_ESCAPE)
2014-12-08 21:57:32 -06:00
{
//Stop binding if esc pressed
mInputBinder->cancelDetectingBindingState();
MWBase::Environment::get().getWindowManager ()->notifyInputActionBound ();
return;
}
// Disallow binding reserved keys
2020-01-10 14:29:04 +04:00
if (key == SDL_SCANCODE_F3 || key == SDL_SCANCODE_F4 || key == SDL_SCANCODE_F10)
return;
#ifndef __APPLE__
// Disallow binding Windows/Meta keys
if (key == SDL_SCANCODE_LGUI || key == SDL_SCANCODE_RGUI)
return;
#endif
2014-12-08 21:57:32 -06:00
if(!mDetectingKeyboard)
return;
2013-05-03 12:44:27 +02:00
2014-12-08 21:57:32 -06:00
clearAllKeyBindings(control);
2014-12-09 11:16:17 -06:00
control->setInitialValue(0.0f);
2012-08-13 01:26:15 +02:00
ICS::DetectingBindingListener::keyBindingDetected (ICS, control, key, direction);
MWBase::Environment::get().getWindowManager ()->notifyInputActionBound ();
}
void InputManager::mouseAxisBindingDetected(ICS::InputControlSystem* ICS, ICS::Control* control
, ICS::InputControlSystem::NamedAxis axis, ICS::Control::ControlChangingDirection direction)
{
// we don't want mouse movement bindings
return;
}
2012-08-13 01:26:15 +02:00
void InputManager::mouseButtonBindingDetected(ICS::InputControlSystem* ICS, ICS::Control* control
, unsigned int button, ICS::Control::ControlChangingDirection direction)
{
2014-12-08 21:57:32 -06:00
if(!mDetectingKeyboard)
return;
clearAllKeyBindings(control);
2014-12-09 11:16:17 -06:00
control->setInitialValue(0.0f);
2012-08-13 01:26:15 +02:00
ICS::DetectingBindingListener::mouseButtonBindingDetected (ICS, control, button, direction);
MWBase::Environment::get().getWindowManager ()->notifyInputActionBound ();
}
void InputManager::mouseWheelBindingDetected(ICS::InputControlSystem* ICS, ICS::Control* control
, ICS::InputControlSystem::MouseWheelClick click, ICS::Control::ControlChangingDirection direction)
{
if(!mDetectingKeyboard)
return;
clearAllKeyBindings(control);
control->setInitialValue(0.0f);
ICS::DetectingBindingListener::mouseWheelBindingDetected(ICS, control, click, direction);
MWBase::Environment::get().getWindowManager()->notifyInputActionBound();
}
void InputManager::joystickAxisBindingDetected(ICS::InputControlSystem* ICS, int deviceID, ICS::Control* control
2014-12-08 21:57:32 -06:00
, int axis, ICS::Control::ControlChangingDirection direction)
2012-08-13 01:26:15 +02:00
{
2014-12-08 21:57:32 -06:00
//only allow binding to the trigers
if(axis != SDL_CONTROLLER_AXIS_TRIGGERLEFT && axis != SDL_CONTROLLER_AXIS_TRIGGERRIGHT)
return;
if(mDetectingKeyboard)
return;
2012-08-13 01:26:15 +02:00
2014-12-08 21:57:32 -06:00
clearAllControllerBindings(control);
control->setValue(0.5f); //axis bindings must start at 0.5
2014-12-09 11:16:17 -06:00
control->setInitialValue(0.5f);
ICS::DetectingBindingListener::joystickAxisBindingDetected (ICS, deviceID, control, axis, direction);
2012-08-13 01:26:15 +02:00
MWBase::Environment::get().getWindowManager ()->notifyInputActionBound ();
}
void InputManager::joystickButtonBindingDetected(ICS::InputControlSystem* ICS, int deviceID, ICS::Control* control
2014-12-08 21:57:32 -06:00
, unsigned int button, ICS::Control::ControlChangingDirection direction)
2012-08-13 01:26:15 +02:00
{
2014-12-08 21:57:32 -06:00
if(mDetectingKeyboard)
return;
clearAllControllerBindings(control);
2014-12-09 11:16:17 -06:00
control->setInitialValue(0.0f);
ICS::DetectingBindingListener::joystickButtonBindingDetected (ICS, deviceID, control, button, direction);
2012-08-13 01:26:15 +02:00
MWBase::Environment::get().getWindowManager ()->notifyInputActionBound ();
}
2014-12-08 21:57:32 -06:00
void InputManager::clearAllKeyBindings (ICS::Control* control)
2012-08-13 01:26:15 +02:00
{
// right now we don't really need multiple bindings for the same action, so remove all others first
2014-09-13 20:39:32 +02:00
if (mInputBinder->getKeyBinding (control, ICS::Control::INCREASE) != SDL_SCANCODE_UNKNOWN)
mInputBinder->removeKeyBinding (mInputBinder->getKeyBinding (control, ICS::Control::INCREASE));
if (mInputBinder->getMouseButtonBinding (control, ICS::Control::INCREASE) != ICS_MAX_DEVICE_BUTTONS)
mInputBinder->removeMouseButtonBinding (mInputBinder->getMouseButtonBinding (control, ICS::Control::INCREASE));
if (mInputBinder->getMouseWheelBinding (control, ICS::Control::INCREASE) != ICS::InputControlSystem::MouseWheelClick::UNASSIGNED)
mInputBinder->removeMouseWheelBinding (mInputBinder->getMouseWheelBinding(control, ICS::Control::INCREASE));
2014-12-08 21:57:32 -06:00
}
2012-08-13 01:26:15 +02:00
2014-12-08 21:57:32 -06:00
void InputManager::clearAllControllerBindings (ICS::Control* control)
{
// right now we don't really need multiple bindings for the same action, so remove all others first
if (mInputBinder->getJoystickAxisBinding (control, mFakeDeviceID, ICS::Control::INCREASE) != SDL_SCANCODE_UNKNOWN)
mInputBinder->removeJoystickAxisBinding (mFakeDeviceID, mInputBinder->getJoystickAxisBinding (control, mFakeDeviceID, ICS::Control::INCREASE));
if (mInputBinder->getJoystickButtonBinding (control, mFakeDeviceID, ICS::Control::INCREASE) != ICS_MAX_DEVICE_BUTTONS)
mInputBinder->removeJoystickButtonBinding (mFakeDeviceID, mInputBinder->getJoystickButtonBinding (control, mFakeDeviceID, ICS::Control::INCREASE));
2012-08-13 01:26:15 +02:00
}
2016-10-20 02:12:01 +02:00
int InputManager::countSavedGameRecords() const
{
return 1;
}
void InputManager::write(ESM::ESMWriter& writer, Loading::Listener& /*progress*/)
{
ESM::ControlsState controls;
controls.mViewSwitchDisabled = !getControlSwitch("playerviewswitch");
controls.mControlsDisabled = !getControlSwitch("playercontrols");
controls.mJumpingDisabled = !getControlSwitch("playerjumping");
controls.mLookingDisabled = !getControlSwitch("playerlooking");
controls.mVanityModeDisabled = !getControlSwitch("vanitymode");
controls.mWeaponDrawingDisabled = !getControlSwitch("playerfighting");
controls.mSpellDrawingDisabled = !getControlSwitch("playermagic");
writer.startRecord (ESM::REC_INPU);
controls.save(writer);
writer.endRecord (ESM::REC_INPU);
}
void InputManager::readRecord(ESM::ESMReader& reader, uint32_t type)
{
if (type == ESM::REC_INPU)
{
ESM::ControlsState controls;
controls.load(reader);
toggleControlSwitch("playerviewswitch", !controls.mViewSwitchDisabled);
toggleControlSwitch("playercontrols", !controls.mControlsDisabled);
toggleControlSwitch("playerjumping", !controls.mJumpingDisabled);
toggleControlSwitch("playerlooking", !controls.mLookingDisabled);
toggleControlSwitch("vanitymode", !controls.mVanityModeDisabled);
toggleControlSwitch("playerfighting", !controls.mWeaponDrawingDisabled);
toggleControlSwitch("playermagic", !controls.mSpellDrawingDisabled);
}
}
2014-12-08 21:57:32 -06:00
void InputManager::resetToDefaultKeyBindings()
{
loadKeyDefaults(true);
}
2014-12-08 21:57:32 -06:00
void InputManager::resetToDefaultControllerBindings()
{
loadControllerDefaults(true);
}
2010-07-17 19:58:15 +02:00
}