scummvm/engines/sherlock/events.cpp

300 lines
6.7 KiB
C++
Raw Normal View History

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#include "common/scummsys.h"
#include "common/events.h"
#include "common/system.h"
#include "engines/util.h"
#include "graphics/cursorman.h"
#include "sherlock/sherlock.h"
#include "sherlock/events.h"
namespace Sherlock {
2015-03-21 20:25:15 -04:00
Events::Events(SherlockEngine *vm) {
_vm = vm;
_cursorImages = nullptr;
_cursorId = INVALID_CURSOR;
_frameCounter = 1;
_priorFrameTime = 0;
_mouseButtons = 0;
_pressed = _released = false;
_rightPressed = _rightReleased = false;
_oldButtons = _oldRightButton = false;
}
2015-03-21 20:25:15 -04:00
Events::~Events() {
delete _cursorImages;
}
/**
* Load a set of cursors from the specified file
*/
2015-03-21 20:25:15 -04:00
void Events::loadCursors(const Common::String &filename) {
hideCursor();
delete _cursorImages;
_cursorImages = new ImageFile(filename);
_cursorId = INVALID_CURSOR;
}
/**
* Set the cursor to show
*/
2015-03-21 20:25:15 -04:00
void Events::setCursor(CursorId cursorId) {
if (cursorId == _cursorId)
return;
_cursorId = cursorId;
// Set the cursor data
Graphics::Surface &s = (*_cursorImages)[cursorId];
2015-04-16 17:32:11 -05:00
setCursor(s);
}
/**
* Set the cursor to show from a passed frame
*/
void Events::setCursor(const Graphics::Surface &src) {
CursorMan.replaceCursor(src.getPixels(), src.w, src.h, 0, 0, 0xff);
2015-04-21 01:26:47 -05:00
showCursor();
}
/**
* Show the mouse cursor
*/
2015-03-21 20:25:15 -04:00
void Events::showCursor() {
CursorMan.showMouse(true);
}
/**
* Hide the mouse cursor
*/
2015-03-21 20:25:15 -04:00
void Events::hideCursor() {
CursorMan.showMouse(false);
}
2015-03-21 20:25:15 -04:00
/**
2015-05-07 19:33:44 +02:00
* Returns the cursor
2015-03-21 20:25:15 -04:00
*/
CursorId Events::getCursor() const {
return _cursorId;
}
/**
* Returns true if the mouse cursor is visible
*/
2015-03-21 20:25:15 -04:00
bool Events::isCursorVisible() const {
return CursorMan.isVisible();
}
/**
* Move the mouse
*/
void Events::moveMouse(const Common::Point &pt) {
g_system->warpMouse(pt.x, pt.y);
}
/**
* Check for any pending events
*/
2015-03-21 20:25:15 -04:00
void Events::pollEvents() {
checkForNextFrameCounter();
Common::Event event;
while (g_system->getEventManager()->pollEvent(event)) {
// Handle keypress
switch (event.type) {
case Common::EVENT_QUIT:
case Common::EVENT_RTL:
return;
case Common::EVENT_KEYDOWN:
// Check for debugger
if (event.kbd.keycode == Common::KEYCODE_d && (event.kbd.flags & Common::KBD_CTRL)) {
// Attach to the debugger
_vm->_debugger->attach();
_vm->_debugger->onFrame();
} else {
_pendingKeys.push(event.kbd);
}
return;
case Common::EVENT_KEYUP:
return;
case Common::EVENT_LBUTTONDOWN:
_mouseButtons |= 1;
return;
case Common::EVENT_RBUTTONDOWN:
_mouseButtons |= 2;
return;
case Common::EVENT_LBUTTONUP:
_mouseButtons &= ~1;
return;
case Common::EVENT_RBUTTONUP:
_mouseButtons &= ~2;
return;
default:
break;
}
}
}
/**
* Poll for events and introduce a small delay, to allow the system to
* yield to other running programs
*/
2015-03-21 20:25:15 -04:00
void Events::pollEventsAndWait() {
pollEvents();
g_system->delayMillis(10);
}
/**
* Check whether it's time to display the next screen frame
*/
2015-03-21 20:25:15 -04:00
bool Events::checkForNextFrameCounter() {
// Check for next game frame
uint32 milli = g_system->getMillis();
if ((milli - _priorFrameTime) >= GAME_FRAME_TIME) {
++_frameCounter;
_priorFrameTime = milli;
// Give time to the debugger
_vm->_debugger->onFrame();
// Display the frame
_vm->_screen->update();
return true;
}
return false;
}
/**
* Get the current mouse position
*/
Common::Point Events::mousePos() const {
return g_system->getEventManager()->getMousePos();
}
/**
* Get a pending keypress
*/
2015-05-07 19:33:44 +02:00
Common::KeyState Events::getKey() {
2015-05-18 18:34:10 -04:00
return _pendingKeys.pop();
}
/**
* Clear any current keypress or mouse click
*/
2015-03-21 20:25:15 -04:00
void Events::clearEvents() {
_pendingKeys.clear();
2015-04-29 18:02:08 -10:00
_mouseButtons = 0;
_pressed = _released = false;
_rightPressed = _rightReleased = false;
_oldButtons = _oldRightButton = false;
}
2015-03-27 08:36:04 -04:00
/**
* Clear any pending keyboard inputs
*/
void Events::clearKeyboard() {
_pendingKeys.clear();
}
/**
* Delay for a given number of game frames, where each frame is 1/60th of a second
*/
2015-03-21 20:25:15 -04:00
void Events::wait(int numFrames) {
uint32 totalMilli = numFrames * 1000 / GAME_FRAME_RATE;
delay(totalMilli);
}
2015-05-01 17:17:24 -10:00
/**
* Does a delay of the specified number of milliseconds
*/
2015-03-21 20:25:15 -04:00
bool Events::delay(uint32 time, bool interruptable) {
// Different handling for really short versus extended times
if (time < 10) {
// For really short periods, simply delay by the desired amount
pollEvents();
g_system->delayMillis(time);
bool result = !(interruptable && (kbHit() || _pressed));
clearEvents();
return result;
} else {
// For long periods go into a loop where we delay by 10ms at a time and then
// check for events. This ensures for longer delays that responsiveness is
// maintained
uint32 delayEnd = g_system->getMillis() + time;
while (!_vm->shouldQuit() && g_system->getMillis() < delayEnd) {
pollEventsAndWait();
if (interruptable && (kbHit() || _pressed)) {
clearEvents();
return false;
}
}
return true;
}
}
/**
* Sets the pressed and released button flags on the raw button state previously set in pollEvents calls.
* @remarks The events manager has separate variables for the raw immediate and old button state
* versus the current buttons states for the frame. This method is expected to be called only once
* per game frame
*/
void Events::setButtonState() {
_released = _rightReleased = false;
if (_mouseButtons & 1)
_pressed = _oldButtons = true;
if ((_mouseButtons & 1) == 0 && _oldButtons) {
_pressed = _oldButtons = false;
_released = true;
}
if (_mouseButtons & 2)
_rightPressed = _oldRightButton = true;
if ((_mouseButtons & 2) == 0 && _oldRightButton) {
_rightPressed = _oldRightButton = false;
_rightReleased = true;
}
}
/**
* Checks to see to see if a key or a mouse button is pressed.
*/
bool Events::checkInput() {
setButtonState();
return kbHit() || _pressed || _released || _rightPressed || _rightReleased;
}
2015-05-09 10:01:36 -04:00
} // End of namespace Sherlock