#include "inputmanager.hpp" #include #include #include #include #include #include #include "../mwgui/window_manager.hpp" #include #include #include #include "../engine.hpp" #include "../mwworld/player.hpp" #include "../mwrender/player.hpp" #include #include #include #include namespace MWInput { enum Actions { A_Quit, // Exit the program A_Screenshot, // Take a screenshot A_Inventory, // Toggle inventory screen A_Console, // Toggle console screen A_MoveLeft, // Move player left / right A_MoveRight, A_MoveUp, // Move up / down A_MoveDown, A_MoveForward, // Forward / Backward A_MoveBackward, A_Activate, A_Use, //Use weapon, spell, etc. A_Jump, A_AutoMove, //Toggle Auto-move forward A_Rest, //Rest A_Journal, //Journal A_Weapon, //Draw/Sheath weapon A_Spell, //Ready/Unready Casting A_AlwaysRun, //Toggle Always Run A_CycleSpellLeft, //cycling through spells A_CycleSpellRight, A_CycleWeaponLeft,//Cycling through weapons A_CycleWeaponRight, A_ToggleSneak, //Toggles Sneak, add Push-Sneak later A_ToggleWalk, //Toggle Walking/Running A_QuickSave, A_QuickLoad, A_QuickMenu, A_GameMenu, A_LAST // Marker for the last item }; // Class that handles all input and key bindings for OpenMW class InputImpl : public Ogre::FrameListener { OEngine::Input::DispatcherPtr disp; OEngine::Render::OgreRenderer &ogre; OEngine::Render::ExitListener exit; Mangle::Input::OISDriver input; OEngine::Input::Poller poller; OEngine::Render::MouseLookEventPtr mouse; OEngine::GUI::EventInjectorPtr guiEvents; MWWorld::Player &player; MWGui::WindowManager &windows; OMW::Engine& mEngine; // Count screenshots. int shotCount; /* InputImpl Methods */ // Write screenshot to file. void screenshot() { // Find the first unused filename with a do-while char buf[50]; do { snprintf(buf, 50, "screenshot%03d.png", shotCount++); } while (boost::filesystem::exists(buf)); ogre.screenshot(buf); } /* toggleInventory() is called when the user presses the button to toggle the inventory screen. */ void toggleInventory() { using namespace MWGui; GuiMode mode = windows.getMode(); // Toggle between game mode and inventory mode if(mode == GM_Game) setGuiMode(GM_Inventory); else if(mode == GM_Inventory) setGuiMode(GM_Game); // .. but don't touch any other mode. } // Toggle console void toggleConsole() { using namespace MWGui; GuiMode mode = windows.getMode(); // Switch to console mode no matter what mode we are currently // in, except of course if we are already in console mode if(mode == GM_Console) setGuiMode(GM_Game); else setGuiMode(GM_Console); } void activate() { mEngine.activate(); } void toggleAutoMove() { player.setmAutoMove(true); } void toggleWalking() { player.setmisWalking(true); } // Exit program now button (which is disabled in GUI mode) void exitNow() { if(!windows.isGuiMode()) exit.exitNow(); } public: InputImpl(OEngine::Render::OgreRenderer &_ogre, MWWorld::Player &_player, MWGui::WindowManager &_windows, bool debug, OMW::Engine& engine) : ogre(_ogre), exit(ogre.getWindow()), input(ogre.getWindow(), !debug), poller(input), player(_player), windows(_windows), mEngine (engine), shotCount(0) { using namespace OEngine::Input; using namespace OEngine::Render; using namespace OEngine::GUI; using namespace Mangle::Input; using namespace OIS; disp = DispatcherPtr(new Dispatcher(A_LAST)); // Bind MW-specific functions disp->funcs.bind(A_Quit, boost::bind(&InputImpl::exitNow, this), "Quit program"); disp->funcs.bind(A_Screenshot, boost::bind(&InputImpl::screenshot, this), "Screenshot"); disp->funcs.bind(A_Inventory, boost::bind(&InputImpl::toggleInventory, this), "Toggle inventory screen"); disp->funcs.bind(A_Console, boost::bind(&InputImpl::toggleConsole, this), "Toggle console"); disp->funcs.bind(A_Activate, boost::bind(&InputImpl::activate, this), "Activate"); disp->funcs.bind(A_AutoMove, boost::bind(&InputImpl::toggleAutoMove, this), "Auto Move"); disp->funcs.bind(A_ToggleWalk, boost::bind(&InputImpl::toggleWalking, this), "Toggle Walk/Run"); // Add the exit listener ogre.getRoot()->addFrameListener(&exit); // Add ourselves as a frame listener to catch movement keys ogre.getRoot()->addFrameListener(this); // Set up the mouse handler and tell it about the player camera mouse = MouseLookEventPtr(new MouseLookEvent(player.getRenderer()->getCamera())); // This event handler pumps events into MyGUI guiEvents = EventInjectorPtr(new EventInjector(windows.getGui())); // Hook 'mouse' and 'disp' up as event handlers into 'input' // (the OIS driver and event source.) We do this through an // EventList which dispatches the event to multiple handlers for // us. { EventList *lst = new EventList; input.setEvent(EventPtr(lst)); lst->add(mouse,Event::EV_MouseMove); lst->add(disp,Event::EV_KeyDown); lst->add(guiEvents,Event::EV_ALL); } // Start out in game mode setGuiMode(MWGui::GM_Game); /********************************** Key binding section The rest of this function has hard coded key bindings, and is intended to be replaced by user defined bindings later. **********************************/ // Key bindings for keypress events // NOTE: These keys do not require constant polling - use in conjuction with variables in loops. disp->bind(A_Quit, KC_Q); disp->bind(A_Quit, KC_ESCAPE); disp->bind(A_Screenshot, KC_SYSRQ); disp->bind(A_Inventory, KC_I); disp->bind(A_Console, KC_F1); disp->bind(A_Activate, KC_SPACE); disp->bind(A_AutoMove, KC_Z); disp->bind(A_ToggleSneak, KC_X); disp->bind(A_ToggleWalk, KC_C); // Key bindings for polled keys // NOTE: These keys are constantly being polled. Only add keys that must be checked each frame. // Arrow keys poller.bind(A_MoveLeft, KC_LEFT); poller.bind(A_MoveRight, KC_RIGHT); poller.bind(A_MoveForward, KC_UP); poller.bind(A_MoveBackward, KC_DOWN); // WASD keys poller.bind(A_MoveLeft, KC_A); poller.bind(A_MoveRight, KC_D); poller.bind(A_MoveForward, KC_W); poller.bind(A_MoveBackward, KC_S); // Use shift and ctrl for up and down poller.bind(A_MoveUp, KC_LSHIFT); poller.bind(A_MoveDown, KC_LCONTROL); } //NOTE: Used to check for movement keys bool frameStarted(const Ogre::FrameEvent &evt) { // Tell OIS to handle all input events input.capture(); // Update windows/gui as a result of input events // For instance this could mean opening a new window/dialog, // by doing this after the input events are handled we // ensure that window/gui changes appear quickly while // avoiding that window/gui changes does not happen in // event callbacks (which may crash) windows.update(); // Disable movement in Gui mode if(windows.isGuiMode()) return true; float speed = 300 * evt.timeSinceLastFrame; //placeholder player speed? float TESTwalkSpeed = 100 * evt.timeSinceLastFrame; //How about another? float moveX = 0, moveY = 0, moveZ = 0; //execute Automove - condition checked in function player.executeAutoMove((float)evt.timeSinceLastFrame); //or since last frame? //Poll and execute movement keys - will disable automove if pressed. if(poller.isDown(A_MoveLeft)) { if (player.getmisWalking() == false) { player.setmAutoMove(false); moveX -= speed; } else { player.setmAutoMove(false); moveX -= TESTwalkSpeed; } } if(poller.isDown(A_MoveRight)) { if (player.getmisWalking() == false) { player.setmAutoMove(false); moveX += speed; } else { player.setmAutoMove(false); moveX += TESTwalkSpeed; } } if(poller.isDown(A_MoveForward)) { if (player.getmisWalking() == false) { player.setmAutoMove(false); moveZ -= speed; } else { player.setmAutoMove(false); moveZ -= TESTwalkSpeed; } } if(poller.isDown(A_MoveBackward)) { if (player.getmisWalking() == false) { player.setmAutoMove(false); moveZ += speed; } else { player.setmAutoMove(false); moveZ += TESTwalkSpeed; } } // TODO: These should be enabled for floating modes (like // swimming and levitation) and disabled for everything else. if(poller.isDown(A_MoveUp)) moveY += speed; if(poller.isDown(A_MoveDown)) moveY -= speed; if(moveX != 0 || moveY != 0 || moveZ != 0) player.moveRel(moveX, moveY, moveZ); return true; } // Switch between gui modes. Besides controlling the Gui windows // this also makes sure input is directed to the right place void setGuiMode(MWGui::GuiMode mode) { // Tell the GUI what to show (this also takes care of the mouse // pointer) windows.setMode(mode); // Are we in GUI mode now? if(windows.isGuiMode()) { // Disable mouse look mouse->setCamera(NULL); // Enable GUI events guiEvents->enabled = true; } else { // Start mouse-looking again. TODO: This should also allow // for other ways to disable mouselook, like paralyzation. mouse->setCamera(player.getRenderer()->getCamera()); // Disable GUI events guiEvents->enabled = false; } } }; /***CONSTRUCTOR***/ MWInputManager::MWInputManager(OEngine::Render::OgreRenderer &ogre, MWWorld::Player &player, MWGui::WindowManager &windows, bool debug, OMW::Engine& engine) { impl = new InputImpl(ogre,player,windows,debug, engine); } /***DESTRUCTOR***/ MWInputManager::~MWInputManager() { delete impl; } void MWInputManager::setGuiMode(MWGui::GuiMode mode) { impl->setGuiMode(mode); } }