2015-03-15 18:42:24 -04:00
|
|
|
/* 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.
|
2015-05-09 18:04:13 +02:00
|
|
|
*
|
2015-03-15 18:42:24 -04:00
|
|
|
* 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.
|
2015-05-09 18:04:13 +02:00
|
|
|
*
|
2015-03-15 18:42:24 -04:00
|
|
|
* 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) {
|
2015-03-15 18:42:24 -04:00
|
|
|
_vm = vm;
|
2015-03-19 19:49:42 -04:00
|
|
|
_cursorImages = nullptr;
|
2015-03-21 18:18:12 -04:00
|
|
|
_cursorId = INVALID_CURSOR;
|
2015-03-15 18:42:24 -04:00
|
|
|
_frameCounter = 1;
|
|
|
|
_priorFrameTime = 0;
|
|
|
|
_mouseButtons = 0;
|
2015-03-23 20:34:34 -04:00
|
|
|
_pressed = _released = false;
|
|
|
|
_rightPressed = _rightReleased = false;
|
|
|
|
_oldButtons = _oldRightButton = false;
|
2015-03-15 18:42:24 -04:00
|
|
|
}
|
|
|
|
|
2015-03-21 20:25:15 -04:00
|
|
|
Events::~Events() {
|
2015-03-19 19:49:42 -04:00
|
|
|
delete _cursorImages;
|
2015-03-18 22:32:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load a set of cursors from the specified file
|
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
void Events::loadCursors(const Common::String &filename) {
|
2015-03-18 22:32:41 -04:00
|
|
|
hideCursor();
|
2015-03-19 19:49:42 -04:00
|
|
|
delete _cursorImages;
|
2015-03-18 22:32:41 -04:00
|
|
|
|
2015-03-19 19:49:42 -04:00
|
|
|
_cursorImages = new ImageFile(filename);
|
2015-04-22 05:06:13 -05:00
|
|
|
_cursorId = INVALID_CURSOR;
|
2015-03-15 18:42:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the cursor to show
|
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
void Events::setCursor(CursorId cursorId) {
|
2015-03-21 18:18:12 -04:00
|
|
|
if (cursorId == _cursorId)
|
|
|
|
return;
|
|
|
|
|
|
|
|
_cursorId = cursorId;
|
2015-03-18 22:32:41 -04:00
|
|
|
|
|
|
|
// Set the cursor data
|
2015-03-21 18:18:12 -04:00
|
|
|
Graphics::Surface &s = (*_cursorImages)[cursorId];
|
2015-03-15 18:42:24 -04:00
|
|
|
|
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();
|
2015-03-15 18:42:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Show the mouse cursor
|
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
void Events::showCursor() {
|
2015-03-15 18:42:24 -04:00
|
|
|
CursorMan.showMouse(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hide the mouse cursor
|
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
void Events::hideCursor() {
|
2015-03-15 18:42:24 -04:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2015-03-15 18:42:24 -04:00
|
|
|
/**
|
|
|
|
* Returns true if the mouse cursor is visible
|
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
bool Events::isCursorVisible() const {
|
2015-03-15 18:42:24 -04:00
|
|
|
return CursorMan.isVisible();
|
|
|
|
}
|
|
|
|
|
2015-04-10 23:35:26 -05:00
|
|
|
/**
|
|
|
|
* Move the mouse
|
|
|
|
*/
|
|
|
|
void Events::moveMouse(const Common::Point &pt) {
|
|
|
|
g_system->warpMouse(pt.x, pt.y);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-03-15 21:25:07 -04:00
|
|
|
/**
|
|
|
|
* Check for any pending events
|
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
void Events::pollEvents() {
|
2015-03-15 18:42:24 -04:00
|
|
|
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:
|
2015-03-23 20:34:34 -04:00
|
|
|
_mouseButtons |= 1;
|
|
|
|
return;
|
2015-03-15 18:42:24 -04:00
|
|
|
case Common::EVENT_RBUTTONDOWN:
|
2015-03-23 20:34:34 -04:00
|
|
|
_mouseButtons |= 2;
|
2015-03-15 18:42:24 -04:00
|
|
|
return;
|
|
|
|
case Common::EVENT_LBUTTONUP:
|
2015-03-23 20:34:34 -04:00
|
|
|
_mouseButtons &= ~1;
|
|
|
|
return;
|
2015-03-15 18:42:24 -04:00
|
|
|
case Common::EVENT_RBUTTONUP:
|
2015-03-23 20:34:34 -04:00
|
|
|
_mouseButtons &= ~2;
|
2015-03-15 18:42:24 -04:00
|
|
|
return;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-16 08:07:24 -04:00
|
|
|
/**
|
|
|
|
* 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() {
|
2015-03-16 08:07:24 -04:00
|
|
|
pollEvents();
|
|
|
|
g_system->delayMillis(10);
|
|
|
|
}
|
|
|
|
|
2015-03-15 21:25:07 -04:00
|
|
|
/**
|
|
|
|
* Check whether it's time to display the next screen frame
|
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
bool Events::checkForNextFrameCounter() {
|
2015-03-15 18:42:24 -04:00
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2015-05-18 18:43:29 -04:00
|
|
|
/**
|
|
|
|
* Get the current mouse position
|
|
|
|
*/
|
|
|
|
Common::Point Events::mousePos() const {
|
|
|
|
return g_system->getEventManager()->getMousePos();
|
|
|
|
}
|
|
|
|
|
2015-04-22 04:44:55 -05:00
|
|
|
/**
|
|
|
|
* 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();
|
2015-04-22 04:44:55 -05:00
|
|
|
}
|
|
|
|
|
2015-03-15 21:25:07 -04:00
|
|
|
/**
|
|
|
|
* Clear any current keypress or mouse click
|
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
void Events::clearEvents() {
|
2015-03-15 21:25:07 -04:00
|
|
|
_pendingKeys.clear();
|
2015-04-29 18:02:08 -10:00
|
|
|
_mouseButtons = 0;
|
2015-03-27 19:52:46 -04:00
|
|
|
_pressed = _released = false;
|
2015-03-23 20:34:34 -04:00
|
|
|
_rightPressed = _rightReleased = false;
|
|
|
|
_oldButtons = _oldRightButton = false;
|
2015-03-15 21:25:07 -04:00
|
|
|
}
|
|
|
|
|
2015-03-27 08:36:04 -04:00
|
|
|
/**
|
|
|
|
* Clear any pending keyboard inputs
|
|
|
|
*/
|
|
|
|
void Events::clearKeyboard() {
|
|
|
|
_pendingKeys.clear();
|
|
|
|
}
|
|
|
|
|
2015-03-15 21:25:07 -04:00
|
|
|
/**
|
2015-03-17 23:09:04 -04:00
|
|
|
* Delay for a given number of game frames, where each frame is 1/60th of a second
|
2015-03-15 21:25:07 -04:00
|
|
|
*/
|
2015-03-21 20:25:15 -04:00
|
|
|
void Events::wait(int numFrames) {
|
2015-03-17 23:09:04 -04:00
|
|
|
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) {
|
2015-03-18 19:02:17 -04:00
|
|
|
// 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);
|
2015-03-23 20:34:34 -04:00
|
|
|
bool result = !(interruptable && (kbHit() || _pressed));
|
2015-03-18 19:02:17 -04:00
|
|
|
|
|
|
|
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();
|
|
|
|
|
2015-03-23 20:34:34 -04:00
|
|
|
if (interruptable && (kbHit() || _pressed)) {
|
2015-03-18 19:02:17 -04:00
|
|
|
clearEvents();
|
|
|
|
return false;
|
|
|
|
}
|
2015-03-17 23:09:04 -04:00
|
|
|
}
|
|
|
|
|
2015-03-18 19:02:17 -04:00
|
|
|
return true;
|
|
|
|
}
|
2015-03-15 18:42:24 -04:00
|
|
|
}
|
|
|
|
|
2015-03-15 21:25:07 -04:00
|
|
|
/**
|
2015-05-09 10:32:45 -04:00
|
|
|
* 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
|
2015-03-15 21:25:07 -04:00
|
|
|
*/
|
2015-03-23 20:34:34 -04:00
|
|
|
void Events::setButtonState() {
|
|
|
|
_released = _rightReleased = false;
|
|
|
|
if (_mouseButtons & 1)
|
|
|
|
_pressed = _oldButtons = true;
|
|
|
|
|
|
|
|
if ((_mouseButtons & 1) == 0 && _oldButtons) {
|
|
|
|
_pressed = _oldButtons = false;
|
|
|
|
_released = true;
|
|
|
|
}
|
2015-03-15 18:42:24 -04:00
|
|
|
|
2015-03-23 20:34:34 -04:00
|
|
|
if (_mouseButtons & 2)
|
|
|
|
_rightPressed = _oldRightButton = true;
|
2015-03-15 18:42:24 -04:00
|
|
|
|
2015-03-23 20:34:34 -04:00
|
|
|
if ((_mouseButtons & 2) == 0 && _oldRightButton) {
|
|
|
|
_rightPressed = _oldRightButton = false;
|
|
|
|
_rightReleased = true;
|
2015-03-15 18:42:24 -04:00
|
|
|
}
|
2015-03-23 20:34:34 -04:00
|
|
|
}
|
2015-03-15 18:42:24 -04:00
|
|
|
|
2015-03-23 20:34:34 -04:00
|
|
|
/**
|
|
|
|
* 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-03-15 18:42:24 -04:00
|
|
|
}
|
|
|
|
|
2015-05-09 10:01:36 -04:00
|
|
|
} // End of namespace Sherlock
|