mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-03 16:09:32 +00:00
7632246264
This makes the bees level playable, removing the "lock-up". Collision detection between Little Red and the bees and butterflies doesn't work yet though, so they're just flying through her. Nevertheless, the game seems to be completable now.
618 lines
14 KiB
C++
618 lines
14 KiB
C++
/* 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/stream.h"
|
|
|
|
#include "graphics/palette.h"
|
|
|
|
#include "gob/gob.h"
|
|
#include "gob/util.h"
|
|
#include "gob/global.h"
|
|
#include "gob/dataio.h"
|
|
#include "gob/draw.h"
|
|
#include "gob/game.h"
|
|
#include "gob/video.h"
|
|
#include "gob/videoplayer.h"
|
|
#include "gob/sound/sound.h"
|
|
|
|
namespace Gob {
|
|
|
|
Util::Util(GobEngine *vm) : _vm(vm) {
|
|
_mouseButtons = kMouseButtonsNone,
|
|
_keyBufferHead = 0;
|
|
_keyBufferTail = 0;
|
|
_fastMode = 0;
|
|
_frameRate = 12;
|
|
_frameWaitTime = 0;
|
|
_startFrameTime = 0;
|
|
|
|
_keyState = 0;
|
|
}
|
|
|
|
uint32 Util::getTimeKey() {
|
|
return g_system->getMillis() * _vm->_global->_speedFactor;
|
|
}
|
|
|
|
int16 Util::getRandom(int16 max) {
|
|
if (max == 0)
|
|
return 0;
|
|
|
|
return _vm->_rnd.getRandomNumber(max - 1);
|
|
}
|
|
|
|
void Util::beep(int16 freq) {
|
|
if (_vm->_global->_soundFlags == 0)
|
|
return;
|
|
|
|
_vm->_sound->speakerOn(freq, 50);
|
|
}
|
|
|
|
void Util::notifyPaused(uint32 duration) {
|
|
_startFrameTime += duration;
|
|
}
|
|
|
|
void Util::delay(uint16 msecs) {
|
|
g_system->delayMillis(msecs / _vm->_global->_speedFactor);
|
|
}
|
|
|
|
void Util::longDelay(uint16 msecs) {
|
|
uint32 time = g_system->getMillis() * _vm->_global->_speedFactor + msecs;
|
|
do {
|
|
_vm->_video->waitRetrace();
|
|
processInput();
|
|
delay(15);
|
|
} while (!_vm->shouldQuit() &&
|
|
((g_system->getMillis() * _vm->_global->_speedFactor) < time));
|
|
}
|
|
|
|
void Util::initInput() {
|
|
_mouseButtons = kMouseButtonsNone;
|
|
_keyBufferHead = _keyBufferTail = 0;
|
|
}
|
|
|
|
void Util::processInput(bool scroll) {
|
|
Common::Event event;
|
|
Common::EventManager *eventMan = g_system->getEventManager();
|
|
int16 x = 0, y = 0;
|
|
bool hasMove = false;
|
|
|
|
_vm->_vidPlayer->updateLive();
|
|
|
|
while (eventMan->pollEvent(event)) {
|
|
switch (event.type) {
|
|
case Common::EVENT_MOUSEMOVE:
|
|
hasMove = true;
|
|
x = event.mouse.x;
|
|
y = event.mouse.y;
|
|
break;
|
|
case Common::EVENT_LBUTTONDOWN:
|
|
_mouseButtons = (MouseButtons) (((uint32) _mouseButtons) | ((uint32) kMouseButtonsLeft));
|
|
break;
|
|
case Common::EVENT_RBUTTONDOWN:
|
|
_mouseButtons = (MouseButtons) (((uint32) _mouseButtons) | ((uint32) kMouseButtonsRight));
|
|
break;
|
|
case Common::EVENT_LBUTTONUP:
|
|
_mouseButtons = (MouseButtons) (((uint32) _mouseButtons) & ~((uint32) kMouseButtonsLeft));
|
|
break;
|
|
case Common::EVENT_RBUTTONUP:
|
|
_mouseButtons = (MouseButtons) (((uint32) _mouseButtons) & ~((uint32) kMouseButtonsRight));
|
|
break;
|
|
case Common::EVENT_KEYDOWN:
|
|
keyDown(event);
|
|
|
|
if (event.kbd.hasFlags(Common::KBD_CTRL)) {
|
|
if (event.kbd.keycode == Common::KEYCODE_f)
|
|
_fastMode ^= 1;
|
|
else if (event.kbd.keycode == Common::KEYCODE_g)
|
|
_fastMode ^= 2;
|
|
else if (event.kbd.keycode == Common::KEYCODE_p)
|
|
_vm->pauseGame();
|
|
else if (event.kbd.keycode == Common::KEYCODE_d) {
|
|
_vm->getDebugger()->attach();
|
|
_vm->getDebugger()->onFrame();
|
|
}
|
|
break;
|
|
}
|
|
addKeyToBuffer(event.kbd);
|
|
break;
|
|
case Common::EVENT_KEYUP:
|
|
keyUp(event);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
_vm->_global->_speedFactor = MIN(_fastMode + 1, 3);
|
|
if (hasMove && scroll) {
|
|
x = CLIP(x, _vm->_global->_mouseMinX, _vm->_global->_mouseMaxX);
|
|
y = CLIP(y, _vm->_global->_mouseMinY, _vm->_global->_mouseMaxY);
|
|
|
|
x -= _vm->_video->_screenDeltaX;
|
|
y -= _vm->_video->_screenDeltaY;
|
|
|
|
_vm->_util->setMousePos(x, y);
|
|
_vm->_game->wantScroll(x, y);
|
|
|
|
// WORKAROUND:
|
|
// Force a check of the mouse in order to fix the sofa bug. This apply only for Gob3, and only
|
|
// in the impacted TOT file so that the second screen animation is not broken.
|
|
if ((_vm->getGameType() == kGameTypeGob3) && _vm->isCurrentTot("EMAP1008.TOT"))
|
|
_vm->_game->evaluateScroll();
|
|
}
|
|
}
|
|
|
|
void Util::clearKeyBuf() {
|
|
processInput();
|
|
_keyBufferHead = _keyBufferTail = 0;
|
|
}
|
|
|
|
bool Util::keyBufferEmpty() {
|
|
return (_keyBufferHead == _keyBufferTail);
|
|
}
|
|
|
|
void Util::addKeyToBuffer(const Common::KeyState &key) {
|
|
if ((_keyBufferHead + 1) % KEYBUFSIZE == _keyBufferTail) {
|
|
warning("key buffer overflow");
|
|
return;
|
|
}
|
|
|
|
_keyBuffer[_keyBufferHead] = key;
|
|
_keyBufferHead = (_keyBufferHead + 1) % KEYBUFSIZE;
|
|
}
|
|
|
|
bool Util::getKeyFromBuffer(Common::KeyState &key) {
|
|
if (_keyBufferHead == _keyBufferTail) return false;
|
|
|
|
key = _keyBuffer[_keyBufferTail];
|
|
_keyBufferTail = (_keyBufferTail + 1) % KEYBUFSIZE;
|
|
|
|
return true;
|
|
}
|
|
|
|
int16 Util::translateKey(const Common::KeyState &key) {
|
|
static struct keyS {
|
|
int16 from;
|
|
int16 to;
|
|
} keys[] = {
|
|
{Common::KEYCODE_INVALID, kKeyNone },
|
|
{Common::KEYCODE_BACKSPACE, kKeyBackspace},
|
|
{Common::KEYCODE_SPACE, kKeySpace },
|
|
{Common::KEYCODE_RETURN, kKeyReturn },
|
|
{Common::KEYCODE_ESCAPE, kKeyEscape },
|
|
{Common::KEYCODE_DELETE, kKeyDelete },
|
|
{Common::KEYCODE_UP, kKeyUp },
|
|
{Common::KEYCODE_DOWN, kKeyDown },
|
|
{Common::KEYCODE_RIGHT, kKeyRight },
|
|
{Common::KEYCODE_LEFT, kKeyLeft },
|
|
{Common::KEYCODE_F1, kKeyF1 },
|
|
{Common::KEYCODE_F2, kKeyF2 },
|
|
{Common::KEYCODE_F3, kKeyF3 },
|
|
{Common::KEYCODE_F4, kKeyF4 },
|
|
{Common::KEYCODE_F5, kKeyEscape },
|
|
{Common::KEYCODE_F6, kKeyF6 },
|
|
{Common::KEYCODE_F7, kKeyF7 },
|
|
{Common::KEYCODE_F8, kKeyF8 },
|
|
{Common::KEYCODE_F9, kKeyF9 },
|
|
{Common::KEYCODE_F10, kKeyF10 }
|
|
};
|
|
|
|
for (int i = 0; i < ARRAYSIZE(keys); i++)
|
|
if (key.keycode == keys[i].from)
|
|
return keys[i].to;
|
|
|
|
if ((key.keycode >= Common::KEYCODE_SPACE) &&
|
|
(key.keycode <= Common::KEYCODE_DELETE)) {
|
|
|
|
// Used as a user input in Gobliins 2 notepad, in the save dialog, ...
|
|
return key.ascii;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int16 Util::getKey() {
|
|
Common::KeyState key;
|
|
|
|
while (!getKeyFromBuffer(key)) {
|
|
processInput();
|
|
|
|
if (keyBufferEmpty())
|
|
g_system->delayMillis(10 / _vm->_global->_speedFactor);
|
|
}
|
|
return translateKey(key);
|
|
}
|
|
|
|
int16 Util::checkKey() {
|
|
Common::KeyState key;
|
|
|
|
getKeyFromBuffer(key);
|
|
|
|
return translateKey(key);
|
|
}
|
|
|
|
bool Util::checkKey(int16 &key) {
|
|
Common::KeyState keyS;
|
|
|
|
if (!getKeyFromBuffer(keyS))
|
|
return false;
|
|
|
|
key = translateKey(keyS);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Util::keyPressed() {
|
|
int16 key = checkKey();
|
|
if (key)
|
|
return true;
|
|
|
|
int16 x, y;
|
|
MouseButtons buttons;
|
|
|
|
getMouseState(&x, &y, &buttons);
|
|
return buttons != kMouseButtonsNone;
|
|
}
|
|
|
|
void Util::getMouseState(int16 *pX, int16 *pY, MouseButtons *pButtons) {
|
|
Common::Point mouse = g_system->getEventManager()->getMousePos();
|
|
*pX = mouse.x + _vm->_video->_scrollOffsetX - _vm->_video->_screenDeltaX;
|
|
*pY = mouse.y + _vm->_video->_scrollOffsetY - _vm->_video->_screenDeltaY;
|
|
|
|
if (pButtons != 0)
|
|
*pButtons = _mouseButtons;
|
|
}
|
|
|
|
void Util::setMousePos(int16 x, int16 y) {
|
|
x = CLIP<int>(x + _vm->_video->_screenDeltaX, 0, _vm->_width - 1);
|
|
y = CLIP<int>(y + _vm->_video->_screenDeltaY, 0, _vm->_height - 1);
|
|
g_system->warpMouse(x, y);
|
|
}
|
|
|
|
void Util::waitMouseUp() {
|
|
do {
|
|
processInput();
|
|
if (_mouseButtons != kMouseButtonsNone)
|
|
delay(10);
|
|
} while (_mouseButtons != kMouseButtonsNone);
|
|
}
|
|
|
|
void Util::waitMouseDown() {
|
|
int16 x;
|
|
int16 y;
|
|
MouseButtons buttons;
|
|
|
|
do {
|
|
processInput();
|
|
getMouseState(&x, &y, &buttons);
|
|
if (buttons == 0)
|
|
delay(10);
|
|
} while (buttons == 0);
|
|
}
|
|
|
|
void Util::waitMouseRelease(char drawMouse) {
|
|
MouseButtons buttons;
|
|
int16 mouseX;
|
|
int16 mouseY;
|
|
|
|
_vm->_game->checkKeys(&mouseX, &mouseY, &buttons, drawMouse);
|
|
while (buttons != 0) {
|
|
if (drawMouse != 0)
|
|
_vm->_draw->animateCursor(2);
|
|
delay(10);
|
|
_vm->_game->checkKeys(&mouseX, &mouseY, &buttons, drawMouse);
|
|
}
|
|
}
|
|
|
|
void Util::forceMouseUp(bool onlyWhenSynced) {
|
|
if (onlyWhenSynced && (_vm->_game->_mouseButtons != _mouseButtons))
|
|
return;
|
|
|
|
_vm->_game->_mouseButtons = kMouseButtonsNone;
|
|
_mouseButtons = kMouseButtonsNone;
|
|
}
|
|
|
|
void Util::clearPalette() {
|
|
int16 i;
|
|
byte colors[768];
|
|
|
|
_vm->validateVideoMode(_vm->_global->_videoMode);
|
|
|
|
if (_vm->_global->_setAllPalette) {
|
|
if (_vm->getPixelFormat().bytesPerPixel == 1) {
|
|
memset(colors, 0, sizeof(colors));
|
|
g_system->getPaletteManager()->setPalette(colors, 0, 256);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < 16; i++)
|
|
_vm->_video->setPalElem(i, 0, 0, 0, 0, _vm->_global->_videoMode);
|
|
}
|
|
|
|
int16 Util::getFrameRate() {
|
|
return _frameRate;
|
|
}
|
|
|
|
void Util::setFrameRate(int16 rate) {
|
|
if (rate == 0)
|
|
rate = 1;
|
|
|
|
_frameRate = rate;
|
|
_frameWaitTime = 1000 / rate;
|
|
_startFrameTime = getTimeKey();
|
|
}
|
|
|
|
void Util::notifyNewAnim() {
|
|
_startFrameTime = getTimeKey();
|
|
}
|
|
|
|
void Util::waitEndFrame() {
|
|
int32 time;
|
|
|
|
_vm->_video->waitRetrace();
|
|
|
|
time = getTimeKey() - _startFrameTime;
|
|
if ((time > 1000) || (time < 0)) {
|
|
_startFrameTime = getTimeKey();
|
|
return;
|
|
}
|
|
|
|
int32 toWait = _frameWaitTime - time;
|
|
|
|
if (toWait > 0)
|
|
delay(toWait);
|
|
|
|
_startFrameTime = getTimeKey();
|
|
}
|
|
|
|
void Util::setScrollOffset(int16 x, int16 y) {
|
|
processInput();
|
|
|
|
if (x >= 0)
|
|
_vm->_video->_scrollOffsetX = x;
|
|
else
|
|
_vm->_video->_scrollOffsetX = _vm->_draw->_scrollOffsetX;
|
|
|
|
if (y >= 0)
|
|
_vm->_video->_scrollOffsetY = y;
|
|
else
|
|
_vm->_video->_scrollOffsetY = _vm->_draw->_scrollOffsetY;
|
|
|
|
_vm->_video->waitRetrace();
|
|
}
|
|
|
|
void Util::insertStr(const char *str1, char *str2, int16 pos) {
|
|
int len1 = strlen(str1);
|
|
int len2 = strlen(str2);
|
|
int from = MIN((int) pos, len2);
|
|
|
|
for (int i = len2; i >= from; i--)
|
|
str2[len1 + i] = str2[i];
|
|
for (int i = 0; i < len1; i++)
|
|
str2[i + from] = str1[i];
|
|
}
|
|
|
|
void Util::cutFromStr(char *str, int16 from, int16 cutlen) {
|
|
int len = strlen(str);
|
|
|
|
if (from >= len)
|
|
return;
|
|
if ((from + cutlen) > len) {
|
|
str[from] = 0;
|
|
return;
|
|
}
|
|
|
|
int i = from;
|
|
do {
|
|
str[i] = str[i + cutlen];
|
|
i++;
|
|
} while (str[i] != 0);
|
|
}
|
|
|
|
void Util::replaceChar(char *str, char c1, char c2) {
|
|
while ((str = strchr(str, c1)))
|
|
*str = c2;
|
|
}
|
|
|
|
static const char trStr1[] =
|
|
" ' + - :0123456789: <=> abcdefghijklmnopqrstuvwxyz "
|
|
"abcdefghijklmnopqrstuvwxyz ";
|
|
static const char trStr2[] =
|
|
" ueaaaaceeeiii ooouu aioun"
|
|
" ";
|
|
static const char trStr3[] = " ";
|
|
|
|
void Util::cleanupStr(char *str) {
|
|
char *start, *end;
|
|
char buf[300];
|
|
|
|
strcpy(buf, trStr1);
|
|
strcat(buf, trStr2);
|
|
strcat(buf, trStr3);
|
|
|
|
// Translating "wrong" characters
|
|
for (size_t i = 0; i < strlen(str); i++)
|
|
str[i] = buf[MIN<int>(str[i] - 32, 32)];
|
|
|
|
// Trim spaces left
|
|
while (str[0] == ' ')
|
|
cutFromStr(str, 0, 1);
|
|
|
|
// Trim spaces right
|
|
while ((*str != '\0') && (str[strlen(str) - 1] == ' '))
|
|
cutFromStr(str, strlen(str) - 1, 1);
|
|
|
|
// Merge double spaces
|
|
start = strchr(str, ' ');
|
|
while (start) {
|
|
if (start[1] == ' ') {
|
|
cutFromStr(str, start - str, 1);
|
|
continue;
|
|
}
|
|
|
|
end = strchr(start + 1, ' ');
|
|
start = end ? end + 1 : 0;
|
|
}
|
|
}
|
|
|
|
void Util::listInsertFront(List *list, void *data) {
|
|
ListNode *node;
|
|
|
|
node = new ListNode;
|
|
if (list->pHead) {
|
|
node->pData = data;
|
|
node->pNext = list->pHead;
|
|
node->pPrev = 0;
|
|
list->pHead->pPrev = node;
|
|
list->pHead = node;
|
|
} else {
|
|
list->pHead = node;
|
|
list->pTail = node;
|
|
node->pData = data;
|
|
node->pNext = 0;
|
|
node->pPrev = 0;
|
|
}
|
|
}
|
|
|
|
void Util::listInsertBack(List *list, void *data) {
|
|
ListNode *node;
|
|
|
|
if (list->pHead != 0) {
|
|
if (list->pTail == 0) {
|
|
list->pTail = list->pHead;
|
|
warning("Util::listInsertBack(): Broken list");
|
|
}
|
|
|
|
node = new ListNode;
|
|
node->pData = data;
|
|
node->pPrev = list->pTail;
|
|
node->pNext = 0;
|
|
list->pTail->pNext = node;
|
|
list->pTail = node;
|
|
} else
|
|
listInsertFront(list, data);
|
|
}
|
|
|
|
void Util::listDropFront(List *list) {
|
|
if (list->pHead->pNext == 0) {
|
|
delete list->pHead;
|
|
list->pHead = 0;
|
|
list->pTail = 0;
|
|
} else {
|
|
list->pHead = list->pHead->pNext;
|
|
delete list->pHead->pPrev;
|
|
list->pHead->pPrev = 0;
|
|
}
|
|
}
|
|
|
|
void Util::deleteList(List *list) {
|
|
while (list->pHead)
|
|
listDropFront(list);
|
|
|
|
delete list;
|
|
}
|
|
|
|
char *Util::setExtension(char *str, const char *ext) {
|
|
assert(str && ext);
|
|
|
|
if (str[0] == '\0')
|
|
return str;
|
|
|
|
char *dot = strrchr(str, '.');
|
|
if (dot)
|
|
*dot = '\0';
|
|
|
|
strcat(str, ext);
|
|
return str;
|
|
}
|
|
|
|
Common::String Util::setExtension(const Common::String &str, const Common::String &ext) {
|
|
if (str.empty())
|
|
return str;
|
|
|
|
const char *dot = strrchr(str.c_str(), '.');
|
|
if (dot)
|
|
return Common::String(str.c_str(), dot - str.c_str()) + ext;
|
|
|
|
return str + ext;
|
|
}
|
|
|
|
Common::String Util::readString(Common::SeekableReadStream &stream, int n) {
|
|
Common::String str;
|
|
|
|
char c;
|
|
while (n-- > 0) {
|
|
if ((c = stream.readByte()) == '\0')
|
|
break;
|
|
|
|
str += c;
|
|
}
|
|
|
|
if (n > 0)
|
|
stream.skip(n);
|
|
|
|
return str;
|
|
}
|
|
|
|
/* NOT IMPLEMENTED */
|
|
void Util::checkJoystick() {
|
|
_vm->_global->_useJoystick = 0;
|
|
}
|
|
|
|
uint32 Util::getKeyState() const {
|
|
return _keyState;
|
|
}
|
|
|
|
void Util::keyDown(const Common::Event &event) {
|
|
if (event.kbd.keycode == Common::KEYCODE_UP)
|
|
_keyState |= 0x0001;
|
|
else if (event.kbd.keycode == Common::KEYCODE_DOWN)
|
|
_keyState |= 0x0002;
|
|
else if (event.kbd.keycode == Common::KEYCODE_RIGHT)
|
|
_keyState |= 0x0004;
|
|
else if (event.kbd.keycode == Common::KEYCODE_LEFT)
|
|
_keyState |= 0x0008;
|
|
else if (event.kbd.keycode == Common::KEYCODE_SPACE)
|
|
_keyState |= 0x0020;
|
|
else if (event.kbd.keycode == Common::KEYCODE_ESCAPE)
|
|
_keyState |= 0x0040;
|
|
}
|
|
|
|
void Util::keyUp(const Common::Event &event) {
|
|
if (event.kbd.keycode == Common::KEYCODE_UP)
|
|
_keyState &= ~0x0001;
|
|
else if (event.kbd.keycode == Common::KEYCODE_DOWN)
|
|
_keyState &= ~0x0002;
|
|
else if (event.kbd.keycode == Common::KEYCODE_RIGHT)
|
|
_keyState &= ~0x0004;
|
|
else if (event.kbd.keycode == Common::KEYCODE_LEFT)
|
|
_keyState &= ~0x0008;
|
|
else if (event.kbd.keycode == Common::KEYCODE_SPACE)
|
|
_keyState &= ~0x0020;
|
|
else if (event.kbd.keycode == Common::KEYCODE_ESCAPE)
|
|
_keyState &= ~0x0040;
|
|
}
|
|
|
|
} // End of namespace Gob
|