Splitting up the big evaluate() function

svn-id: r42171
This commit is contained in:
Sven Hesse 2009-07-06 11:19:37 +00:00
parent 87988c69ee
commit e182cc0b9f
2 changed files with 384 additions and 302 deletions

View File

@ -128,6 +128,16 @@ bool Hotspots::Hotspot::isActiveInput() const {
return true;
}
bool Hotspots::Hotspot::isInputLeave() const {
if (!isInput())
return false;
if (!(getType() & 1))
return true;
return false;
}
bool Hotspots::Hotspot::isFilled() const {
return getState() & kStateFilled;
}
@ -518,7 +528,7 @@ uint16 Hotspots::checkMouse(Type type, uint16 &id, uint16 &index) const {
// Check where the mouse was moved to
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
const Hotspot &spot = _hotspots[i];
if (spot.isDisabled())
// Only consider enabled hotspots
@ -548,7 +558,7 @@ uint16 Hotspots::checkMouse(Type type, uint16 &id, uint16 &index) const {
// Check if something was clicked
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
const Hotspot &spot = _hotspots[i];
if (spot.isDisabled())
// Only consider enabled hotspots
@ -646,7 +656,7 @@ uint16 Hotspots::check(uint8 handleMouse, int16 delay, uint16 &id, uint16 &index
// Update display
_vm->_draw->blitInvalidated();
_vm->_video->retrace();
_vm->_video->waitRetrace();
uint16 key = 0;
while (key == 0) {
@ -914,7 +924,8 @@ uint16 Hotspots::updateInput(uint16 xPos, uint16 yPos, uint16 width, uint16 heig
switch (key) {
case kKeyRight:
// If possible, move the cursor right
if ((pos > strlen(str)) || (pos > (editSize - 1)) || (editSize == 0)) {
if (((editSize != 0) && ((pos > strlen(str)) || (pos > (editSize - 1)))) ||
((editSize == 0) && (pos > strlen(str)))) {
pos++;
continue;
}
@ -1138,7 +1149,7 @@ uint16 Hotspots::handleInputs(int16 time, uint16 inputCount, uint16 &curInput,
}
void Hotspots::evaluateNew(uint16 i, uint16 *ids, InputDesc *inputs,
uint16 &validId, bool &hasInput, uint16 &inputCount) {
uint16 &inputId, bool &hasInput, uint16 &inputCount) {
ids[i] = 0;
@ -1300,7 +1311,7 @@ void Hotspots::evaluateNew(uint16 i, uint16 *ids, InputDesc *inputs,
break;
case 20:
validId = i;
inputId = i;
// Fall through to case 2
case kTypeClick:
key = _vm->_game->_script->readInt16();
@ -1334,13 +1345,62 @@ void Hotspots::evaluateNew(uint16 i, uint16 *ids, InputDesc *inputs,
flags, key, funcEnter, funcLeave, funcPos);
}
bool Hotspots::evaluateFind(uint16 key, int16 timeVal, const uint16 *ids,
uint16 hotspotIndex1, uint16 hotspotIndex2, uint16 endIndex,
int16 &duration, uint16 &id, uint16 &index, bool &finished) {
if (id != 0)
// We already found a hotspot, nothing to do
return true;
if (key != 0) {
// We've got a key
// Find the hotspot with that key associated
findKey(key, id, index);
if (id != 0)
// Found it
return true;
// Try it case insensitively
findKeyCaseInsensitive(key, id, index);
if (id != 0)
// Found it
return true;
return false;
}
if (duration != 0) {
// We've got a time duration
if (hotspotIndex1 != 0) {
finished =
leaveNthPlain(hotspotIndex1, endIndex, timeVal, ids, id, index, duration);
} else if (hotspotIndex2 != 0) {
findNthPlain(hotspotIndex2, endIndex, id, index);
} else {
findNthPlain(0, 0, id, index);
// Leave the current hotspot
if ((_currentKey != 0) && (_hotspots[_currentIndex].funcLeave != 0))
call(_hotspots[_currentIndex].funcLeave);
_currentKey = 0;
}
if (id != 0)
return true;
return false;
}
return false;
}
void Hotspots::evaluate() {
InputDesc inputs[20];
uint16 ids[kHotspotCount];
int16 counter;
int16 var_24;
int16 var_26;
int16 collStackPos;
// Push all local hotspots
push(0);
@ -1360,14 +1420,14 @@ void Hotspots::evaluate() {
// Parameters of this block
_vm->_game->_handleMouse = _vm->_game->_script->peekByte(0);
int16 duration = _vm->_game->_script->peekByte(1);
byte stackPos2 = _vm->_game->_script->peekByte(3);
byte descIndex = _vm->_game->_script->peekByte(4);
byte hotspotIndex1 = _vm->_game->_script->peekByte(3);
byte hotspotIndex2 = _vm->_game->_script->peekByte(4);
bool needRecalculation = _vm->_game->_script->peekByte(5) != 0;
// Seconds -> Milliseconds
duration *= 1000;
if ((stackPos2 != 0) || (descIndex != 0)) {
if ((hotspotIndex1 != 0) || (hotspotIndex2 != 0)) {
duration /= 100;
if (_vm->_game->_script->peekByte(1) == 100)
duration = 2;
@ -1377,14 +1437,12 @@ void Hotspots::evaluate() {
_vm->_game->_script->skip(6);
// Clear current ID
WRITE_VAR(16, 0);
setCurrentHotspot(0, 0);
byte var_41 = 0;
int16 var_46 = 0;
bool finishedDuration = false;
uint16 id = 0;
uint16 validId = 0xFFFF;
uint16 inputId = 0xFFFF;
uint16 index = 0;
bool hasInput = false;
@ -1392,7 +1450,7 @@ void Hotspots::evaluate() {
// Adding new hotspots
for (uint16 i = 0; i < count; i++)
evaluateNew(i, ids, inputs, validId, hasInput, inputCount);
evaluateNew(i, ids, inputs, inputId, hasInput, inputCount);
// Recalculate all hotspots if requested
if (needRecalculation)
@ -1401,7 +1459,7 @@ void Hotspots::evaluate() {
_vm->_game->_forceHandleMouse = 0;
_vm->_util->clearKeyBuf();
do {
while ((id == 0) && !_vm->_inter->_terminate && !_vm->shouldQuit()) {
uint16 key = 0;
if (hasInput) {
// Input
@ -1411,193 +1469,23 @@ void Hotspots::evaluate() {
key = handleInputs(duration, inputCount, curInput, inputs, id, index);
// Notify the script of the current input index
WRITE_VAR(55, curInput);
WRITE_VAR(17 + 38, curInput);
if (key == kKeyReturn) {
// Return pressed, invoke input leave
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
if (!spot.isFilledEnabled())
// Not filled or disabled
continue;
if ((spot.getType() & 1) != 0)
// Not an input with a leave function
continue;
if (spot.getType() <= kTypeClick)
// Not an input
continue;
id = spot.id;
validId = spot.id & 0x7FFF;
index = i;
break;
}
findFirstInputLeave(id, inputId, index);
break;
}
} else
// Normal move or click check
key = check(_vm->_game->_handleMouse, -duration, id, index);
// Handle special number keys
if (((key & 0xFF) >= ' ') && ((key & 0xFF) <= 0xFF) &&
((key >> 8) > 1) && ((key >> 8) < 12))
key = '0' + (((key >> 8) - 1) % 10) + (key & 0xFF00);
key = convertSpecialKey(key);
if (id == 0) {
// No hotspot area
// Try to find a fitting hotspot
Hotspots::evaluateFind(key, timeVal, ids, hotspotIndex1, hotspotIndex2, endIndex,
duration, id, index, finishedDuration);
if (key != 0) {
// But a key
// Find the hotspot with that key associated
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
if (!spot.isFilledEnabled())
// Not filled or disabled
continue;
// Key match Catch all
if ((spot.key == key) || (spot.key == 0x7FFF)) {
id = spot.id;
index = i;
break;
}
}
if (id == 0) {
// Didn't find such a hotspot
// Try it again, this time case insensitively
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
if (!spot.isFilledEnabled())
continue;
if ((spot.key & 0xFF00) != 0)
continue;
if (spot.key == 0)
continue;
if (toupper(key & 0xFF) == toupper(spot.key)) {
id = spot.id;
index = i;
break;
}
}
}
} else if (duration != 0) {
if (stackPos2 != 0) {
collStackPos = 0;
for (int i = endIndex; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
if (!spot.isFilledNew())
continue;
collStackPos++;
if (collStackPos != stackPos2)
// Isn't yet the one wanted
continue;
id = spot.id;
index = i;
_vm->_inter->storeMouse();
if (VAR(16) != 0)
// We already handle a hotspot
break;
// Notify the scripts that we now handle this hotspot
if (Hotspot::getState(id) == kStateFilled)
WRITE_VAR(16, ids[id & 0xFFF]);
else
WRITE_VAR(16, id & 0xFFF);
if (spot.funcLeave != 0) {
// It has a leave function
uint32 timeKey = _vm->_util->getTimeKey();
call(spot.funcLeave);
if (timeVal != 2) {
// Rest time we have left = time we had - time the leave took
duration = timeVal - (_vm->_util->getTimeKey() - timeKey);
// Remove the buffer time
if ((duration - var_46) < 3) {
var_46 -= (duration - 3);
duration = 3;
} else if (var_46 != 0) {
duration -= var_46;
var_46 = 0;
}
// Clamp
if (duration > timeVal)
duration = timeVal;
} else
duration = 2;
}
if (VAR(16) == 0)
// Didn't find anything
id = 0;
else
var_41 = 1;
break;
}
} else {
if (descIndex != 0) {
counter = 0;
// Enter the nth hotspot
for (int i = endIndex; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
if (spot.isFilledNew()) {
if (++counter == descIndex) {
id = spot.id;
index = i;
break;
}
}
}
} else {
// Enter the first hotspot
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
if (spot.isFilledNew()) {
id = spot.id;
index = i;
break;
}
}
// Leave the current hotspot
if ((_currentKey != 0) && (_hotspots[_currentIndex].funcLeave != 0))
call(_hotspots[_currentIndex].funcLeave);
_currentKey = 0;
}
}
}
}
if (var_41 != 0)
if (finishedDuration)
break;
if ((id == 0) || (_hotspots[index].funcLeave != 0))
@ -1606,114 +1494,30 @@ void Hotspots::evaluate() {
_vm->_inter->storeMouse();
// Notify the scripts of the currently handled hotspot
if (Hotspot::getState(id) == kStateFilled)
WRITE_VAR(16, ids[id & 0xFFF]);
else
WRITE_VAR(16, id & 0xFFF);
setCurrentHotspot(ids, id);
// Enter it
if (_hotspots[index].funcEnter != 0)
call(_hotspots[index].funcEnter);
WRITE_VAR(16, 0);
setCurrentHotspot(0, 0);
id = 0;
}
while ((id == 0) && !_vm->_inter->_terminate && !_vm->shouldQuit());
char tempStr[256];
if ((id & 0xFFF) == validId) {
collStackPos = 0;
var_24 = 0;
var_26 = 1;
for (int i = 0; i < kHotspotCount; i++) {
Hotspot &spot = _hotspots[i];
// Looking for all enabled inputs
if (spot.isEnd())
continue;
if (!spot.isFilledEnabled())
continue;
if (!spot.isInput())
continue;
// Clean up numerical floating values
if (spot.getType() >= kTypeInputFloatNoLeave) {
// Get the string
char *ptr;
strncpy0(tempStr, GET_VARO_STR(spot.key), 255);
// Remove spaces
while ((ptr = strchr(tempStr, ' ')))
_vm->_util->cutFromStr(tempStr, (ptr - tempStr), 1);
// Exchange decimal separator if needed
if (_vm->_global->_language == kLanguageBritish)
while ((ptr = strchr(tempStr, '.')))
*ptr = ',';
// Write it back
WRITE_VARO_STR(spot.key, tempStr);
}
if ((spot.getType() >= kTypeInput2NoLeave) && (spot.getType() <= kTypeInput3Leave)) {
const char *str = inputs[var_24].str;
strncpy0(tempStr, GET_VARO_STR(spot.key), 255);
if (spot.getType() < kTypeInput3NoLeave)
_vm->_util->cleanupStr(tempStr);
// Look if we find a match between the wanted and the typed string
int16 pos = 0;
do {
char spotStr[256];
strncpy0(spotStr, str, 255);
pos += strlen(str) + 1;
str += strlen(str) + 1;
if (spot.getType() < kTypeInput3NoLeave)
_vm->_util->cleanupStr(spotStr);
// Compare the entered string with the string we wanted
if (strcmp(tempStr, spotStr) == 0) {
WRITE_VAR(17, VAR(17) + 1);
WRITE_VAR(17 + var_26, 1);
break;
}
} while (inputs[var_24].length > pos);
collStackPos++;
} else
WRITE_VAR(17 + var_26, 2);
var_24++;
var_26++;
}
// Notify the scripts if we reached the requested hotspot
if (collStackPos != (int16) VAR(17))
WRITE_VAR(17, 0);
else
WRITE_VAR(17, 1);
}
if ((id & 0xFFF) == inputId)
matchInputStrings(inputs);
if (_vm->_game->_handleMouse == 1)
_vm->_draw->blitCursor();
if (!_vm->_inter->_terminate && (var_41 == 0)) {
if (!_vm->_inter->_terminate && (!finishedDuration)) {
_vm->_game->_script->seek(_hotspots[index].funcLeave);
_vm->_inter->storeMouse();
if (VAR(16) == 0) {
if (getCurrentHotspot() == 0) {
// No hotspot currently handled, now we'll handle the newly found one
if (Hotspot::getState(id) == kStateFilled)
WRITE_VAR(16, ids[id & 0xFFF]);
else
WRITE_VAR(16, id & 0xFFF);
setCurrentHotspot(ids, id);
}
} else
_vm->_game->_script->setFinished(true);
@ -1737,7 +1541,7 @@ int16 Hotspots::findCursor(uint16 x, uint16 y) const {
int16 cursor = 0;
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
const Hotspot &spot = _hotspots[i];
if ((spot.getWindow() != 0) || spot.isDisabled())
// Ignore disabled and non-main-windowed hotspots
@ -1767,7 +1571,7 @@ int16 Hotspots::findCursor(uint16 x, uint16 y) const {
uint16 Hotspots::inputToHotspot(uint16 input) const {
uint16 inputIndex = 0;
for (int i = 0; i < kHotspotCount; i++) {
Hotspot &spot = _hotspots[i];
const Hotspot &spot = _hotspots[i];
if (!spot.isActiveInput())
// Not an active input
@ -1789,7 +1593,7 @@ uint16 Hotspots::hotspotToInput(uint16 hotspot) const {
uint16 input = 0;
for (int i = 0; i < kHotspotCount; i++) {
Hotspot &spot = _hotspots[i];
const Hotspot &spot = _hotspots[i];
if (!spot.isActiveInput())
// Not an active input
@ -1808,7 +1612,7 @@ uint16 Hotspots::hotspotToInput(uint16 hotspot) const {
uint16 Hotspots::findClickedInput(uint16 index) const {
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
Hotspot &spot = _hotspots[i];
const Hotspot &spot = _hotspots[i];
if (spot.getWindow() != 0)
// Ignore other windows
@ -1837,6 +1641,254 @@ uint16 Hotspots::findClickedInput(uint16 index) const {
return index;
}
bool Hotspots::findFirstInputLeave(uint16 &id, uint16 &inputId, uint16 &index) const {
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
const Hotspot &spot = _hotspots[i];
if (!spot.isFilledEnabled())
// Not filled or disabled
continue;
if (!spot.isInputLeave())
// Not an input with a leave function
continue;
id = spot.id;
inputId = spot.id & 0x7FFF;
index = i;
return true;
}
return false;
}
bool Hotspots::findKey(uint16 key, uint16 &id, uint16 &index) const {
id = 0;
index = 0;
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
const Hotspot &spot = _hotspots[i];
if (!spot.isFilledEnabled())
// Not filled or disabled
continue;
// Key match Catch all
if ((spot.key == key) || (spot.key == 0x7FFF)) {
id = spot.id;
index = i;
return true;
}
}
return false;
}
bool Hotspots::findKeyCaseInsensitive(uint16 key, uint16 &id, uint16 &index) const {
id = 0;
index = 0;
for (int i = 0; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
const Hotspot &spot = _hotspots[i];
if (!spot.isFilledEnabled())
// Not filled or disabled, ignore
continue;
if ((spot.key & 0xFF00) != 0)
continue;
if (spot.key == 0)
// No associated key, ignore
continue;
// Compare
if (toupper(key & 0xFF) == toupper(spot.key)) {
id = spot.id;
index = i;
return true;
}
}
return false;
}
bool Hotspots::findNthPlain(uint16 n, uint16 startIndex, uint16 &id, uint16 &index) const {
id = 0;
index = 0;
uint16 counter = 0;
for (int i = startIndex; (i < kHotspotCount) && !_hotspots[i].isEnd(); i++) {
const Hotspot &spot = _hotspots[i];
if (!spot.isFilledNew())
// Not filled, ignore
continue;
if (++counter != n)
// Not yet the one we want
continue;
id = spot.id;
index = i;
return true;
}
return false;
}
bool Hotspots::leaveNthPlain(uint16 n, uint16 startIndex, int16 timeVal, const uint16 *ids,
uint16 &id, uint16 &index, int16 &duration) {
id = 0;
index = 0;
if (!findNthPlain(n, startIndex, id, index))
// Doesn't exist
return false;
_vm->_inter->storeMouse();
if (getCurrentHotspot() != 0)
// We already handle a hotspot
return false;
setCurrentHotspot(ids, id);
const Hotspot &spot = _hotspots[index];
if (spot.funcLeave != 0) {
// It has a leave function
uint32 startTime, callTime;
// Call the leave and time it
startTime = _vm->_util->getTimeKey();
call(spot.funcLeave);
callTime = _vm->_util->getTimeKey() - startTime;
// Remove the time it took from the time we have available
duration = CLIP<int>(timeVal - callTime, 2, timeVal);
}
if (getCurrentHotspot() == 0) {
id = 0;
index = 0;
}
return getCurrentHotspot() != 0;
}
void Hotspots::setCurrentHotspot(const uint16 *ids, uint16 id) const {
if (!ids) {
WRITE_VAR(16, 0);
return;
}
if (Hotspot::getState(id) == kStateFilled)
WRITE_VAR(16, ids[id & 0xFFF]);
else
WRITE_VAR(16, id & 0xFFF);
}
uint32 Hotspots::getCurrentHotspot() const {
return VAR(16);
}
void Hotspots::cleanFloatString(const Hotspot &spot) const {
char tempStr[256];
// Get the string
strncpy0(tempStr, GET_VARO_STR(spot.key), 255);
// Remove spaces
char *ptr;
while ((ptr = strchr(tempStr, ' ')))
_vm->_util->cutFromStr(tempStr, (ptr - tempStr), 1);
// Exchange decimal separator if needed
if (_vm->_global->_language == kLanguageBritish)
while ((ptr = strchr(tempStr, '.')))
*ptr = ',';
// Write it back
WRITE_VARO_STR(spot.key, tempStr);
}
void Hotspots::checkStringMatch(const Hotspot &spot, const InputDesc &input,
uint16 inputPos) const {
const char *str = input.str;
char tempStr[256];
char spotStr[256];
strncpy0(tempStr, GET_VARO_STR(spot.key), 255);
if (spot.getType() < kTypeInput3NoLeave)
_vm->_util->cleanupStr(tempStr);
uint16 pos = 0;
do {
strncpy0(spotStr, str, 255);
pos += strlen(str) + 1;
str += strlen(str) + 1;
if (spot.getType() < kTypeInput3NoLeave)
_vm->_util->cleanupStr(spotStr);
// Compare the entered string with the string we wanted
if (strcmp(tempStr, spotStr) == 0) {
WRITE_VAR(17, VAR(17) + 1);
WRITE_VAR(17 + inputPos, 1);
break;
}
} while (input.length > pos);
}
void Hotspots::matchInputStrings(const InputDesc *inputs) const {
uint16 strInputCount = 0;
uint16 inputIndex = 0;
uint16 inputPos = 1;
for (int i = 0; i < kHotspotCount; i++) {
const Hotspot &spot = _hotspots[i];
// Looking for all enabled inputs
if (spot.isEnd())
continue;
if (!spot.isFilledEnabled())
continue;
if (!spot.isInput())
continue;
if (spot.getType() >= kTypeInputFloatNoLeave)
cleanFloatString(spot);
if ((spot.getType() >= kTypeInput2NoLeave) && (spot.getType() <= kTypeInput3Leave)) {
// Look if we find a match between the wanted and the typed string
checkStringMatch(spot, inputs[inputIndex], inputPos);
strInputCount++;
} else
WRITE_VAR(17 + inputPos, 2);
inputIndex++;
inputPos++;
}
// Notify the scripts if we reached the requested hotspot
WRITE_VAR(17, (uint32) (strInputCount == ((uint16) VAR(17))));
}
uint16 Hotspots::convertSpecialKey(uint16 key) const {
if (((key & 0xFF) >= ' ') && ((key & 0xFF) <= 0xFF) &&
((key >> 8) > 1) && ((key >> 8) < 12))
key = '0' + (((key >> 8) - 1) % 10) + (key & 0xFF00);
return key;
}
void Hotspots::getTextCursorPos(const Video::FontDesc &font, const char *str,
uint32 pos, uint16 x, uint16 y, uint16 width, uint16 height,
uint16 &cursorX, uint16 &cursorY, uint16 &cursorWidth, uint16 &cursorHeight) const {

View File

@ -121,22 +121,23 @@ private:
void clear();
Type getType() const;
Type getType () const;
MouseButtons getButton() const;
uint8 getWindow() const;
uint8 getCursor() const;
uint8 getState() const;
uint8 getState () const;
/** Is this hotspot the block end marker? */
bool isEnd() const;
bool isInput() const;
bool isInput () const;
bool isActiveInput() const;
bool isInputLeave () const;
bool isFilled() const;
bool isFilled () const;
bool isFilledEnabled() const;
bool isFilledNew() const;
bool isDisabled() const;
bool isFilledNew () const;
bool isDisabled () const;
/** Are the specified coordinates in the hotspot? */
bool isIn(uint16 x, uint16 y) const;
@ -146,7 +147,7 @@ private:
static uint8 getState(uint16 id);
void disable();
void enable();
void enable ();
};
struct StackEntry {
@ -213,14 +214,43 @@ private:
/** Evaluate adding new hotspots script commands. */
void evaluateNew(uint16 i, uint16 *ids, InputDesc *inputs,
uint16 &validId, bool &hasInput, uint16 &inputCount);
uint16 &inputId, bool &hasInput, uint16 &inputCount);
/** Find the hotspot requested by script commands. */
bool evaluateFind(uint16 key, int16 timeVal, const uint16 *ids,
uint16 hotspotIndex1, uint16 hotspotIndex2, uint16 endIndex,
int16 &duration, uint16 &id, uint16 &index, bool &finished);
// Finding specific hotspots
/** Find the hotspot index that corresponds to the input index. */
uint16 inputToHotspot(uint16 input) const;
/** Find the input index that corresponds to the hotspot index. */
uint16 hotspotToInput(uint16 hotspot) const;
/** Find the input that was clicked on. */
uint16 findClickedInput(uint16 index) const;
/** Find the first input hotspot with a leave function. */
bool findFirstInputLeave(uint16 &id, uint16 &inputId, uint16 &index) const;
/** Find the hotspot with the matching key, case sensitively. */
bool findKey(uint16 key, uint16 &id, uint16 &index) const;
/** Find the hotspot with the matching key, case insensitively. */
bool findKeyCaseInsensitive(uint16 key, uint16 &id, uint16 &index) const;
/** Find the nth plain (without Type1 or Type2 state) hotspot. */
bool findNthPlain(uint16 n, uint16 startIndex, uint16 &id, uint16 &index) const;
/** Leave the nth plain (without Type1 or Type2 state) hotspot. */
bool leaveNthPlain(uint16 n, uint16 startIndex, int16 timeVal, const uint16 *ids,
uint16 &id, uint16 &index, int16 &duration);
// Hotspot ID variable access
void setCurrentHotspot(const uint16 *ids, uint16 id) const;
uint32 getCurrentHotspot() const;
// String input functions
void cleanFloatString(const Hotspot &spot) const;
void checkStringMatch(const Hotspot &spot, const InputDesc &input,
uint16 inputPos) const;
void matchInputStrings(const InputDesc *inputs) const;
uint16 convertSpecialKey(uint16 key) const;
/** Calculate the graphical cursor position. */
void getTextCursorPos(const Video::FontDesc &font, const char *str,