2013-07-01 22:19:09 +00: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.
|
|
|
|
|
|
|
|
* 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"
|
|
|
|
|
2013-07-11 05:26:32 +00:00
|
|
|
#include "common/algorithm.h"
|
|
|
|
#include "common/hashmap.h"
|
2013-08-06 00:09:28 +00:00
|
|
|
#include "common/debug.h"
|
2013-07-11 05:26:32 +00:00
|
|
|
|
2013-08-11 20:10:52 +00:00
|
|
|
#include "zvision/zvision.h"
|
2013-07-03 20:45:46 +00:00
|
|
|
#include "zvision/script_manager.h"
|
2013-08-14 15:45:39 +00:00
|
|
|
#include "zvision/render_manager.h"
|
|
|
|
#include "zvision/cursor_manager.h"
|
2013-07-11 05:26:32 +00:00
|
|
|
#include "zvision/actions.h"
|
|
|
|
#include "zvision/action_node.h"
|
|
|
|
#include "zvision/utility.h"
|
2013-07-01 22:19:09 +00:00
|
|
|
|
|
|
|
namespace ZVision {
|
|
|
|
|
2013-08-11 20:04:46 +00:00
|
|
|
ScriptManager::ScriptManager(ZVision *engine)
|
|
|
|
: _engine(engine),
|
|
|
|
_changeLocation(false) {
|
|
|
|
}
|
2013-07-11 05:26:32 +00:00
|
|
|
|
2013-08-21 01:50:14 +00:00
|
|
|
ScriptManager::~ScriptManager() {
|
|
|
|
for (Common::List<Puzzle *>::iterator iter = _activePuzzles.begin(); iter != _activePuzzles.end(); iter++) {
|
|
|
|
delete (*iter);
|
|
|
|
}
|
|
|
|
for (Common::List<Puzzle *>::iterator iter = _globalPuzzles.begin(); iter != _globalPuzzles.end(); iter++) {
|
|
|
|
delete (*iter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-01 22:19:09 +00:00
|
|
|
void ScriptManager::initialize() {
|
2013-08-10 22:23:47 +00:00
|
|
|
parseScrFile("universe.scr", true);
|
2013-08-14 15:29:16 +00:00
|
|
|
changeLocation('g', 'a', 'r', 'y', 0);
|
2013-07-01 22:19:09 +00:00
|
|
|
}
|
|
|
|
|
2013-08-10 22:28:41 +00:00
|
|
|
void ScriptManager::update(uint deltaTimeMillis) {
|
|
|
|
updateNodes(deltaTimeMillis);
|
|
|
|
checkPuzzleCriteria();
|
2013-08-10 22:31:57 +00:00
|
|
|
|
|
|
|
if (_changeLocation) {
|
|
|
|
changeLocationIntern();
|
|
|
|
_changeLocation = false;
|
|
|
|
}
|
2013-08-10 22:28:41 +00:00
|
|
|
}
|
|
|
|
|
2013-07-11 05:26:32 +00:00
|
|
|
void ScriptManager::createReferenceTable() {
|
2013-08-10 22:23:47 +00:00
|
|
|
// Iterate through each local Puzzle
|
2013-08-21 01:50:14 +00:00
|
|
|
for (Common::List<Puzzle *>::iterator activePuzzleIter = _activePuzzles.begin(); activePuzzleIter != _activePuzzles.end(); activePuzzleIter++) {
|
|
|
|
Puzzle *puzzlePtr = (*activePuzzleIter);
|
2013-07-11 05:26:32 +00:00
|
|
|
|
2013-08-18 20:38:56 +00:00
|
|
|
// Iterate through each CriteriaEntry and add a reference from the criteria key to the Puzzle
|
2013-08-21 01:50:14 +00:00
|
|
|
for (Common::List<Common::List<Puzzle::CriteriaEntry> >::iterator criteriaIter = (*activePuzzleIter)->criteriaList.begin(); criteriaIter != (*activePuzzleIter)->criteriaList.end(); criteriaIter++) {
|
2013-08-18 20:38:56 +00:00
|
|
|
for (Common::List<Puzzle::CriteriaEntry>::iterator entryIter = (*criteriaIter).begin(); entryIter != (*criteriaIter).end(); entryIter++) {
|
|
|
|
_referenceTable[entryIter->key].push_back(puzzlePtr);
|
|
|
|
|
|
|
|
// If the argument is a key, add a reference to it as well
|
|
|
|
if (entryIter->argumentIsAKey) {
|
|
|
|
_referenceTable[entryIter->argument].push_back(puzzlePtr);
|
|
|
|
}
|
|
|
|
}
|
2013-08-10 22:23:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate through each global Puzzle
|
2013-08-21 01:50:14 +00:00
|
|
|
for (Common::List<Puzzle *>::iterator globalPuzzleIter = _globalPuzzles.begin(); globalPuzzleIter != _globalPuzzles.end(); globalPuzzleIter++) {
|
|
|
|
Puzzle *puzzlePtr = (*globalPuzzleIter);
|
2013-08-10 22:23:47 +00:00
|
|
|
|
2013-08-18 20:38:56 +00:00
|
|
|
// Iterate through each CriteriaEntry and add a reference from the criteria key to the Puzzle
|
2013-08-21 01:50:14 +00:00
|
|
|
for (Common::List<Common::List<Puzzle::CriteriaEntry> >::iterator criteriaIter = (*globalPuzzleIter)->criteriaList.begin(); criteriaIter != (*globalPuzzleIter)->criteriaList.end(); criteriaIter++) {
|
2013-08-18 20:38:56 +00:00
|
|
|
for (Common::List<Puzzle::CriteriaEntry>::iterator entryIter = (*criteriaIter).begin(); entryIter != (*criteriaIter).end(); entryIter++) {
|
|
|
|
_referenceTable[entryIter->key].push_back(puzzlePtr);
|
2013-08-10 22:23:47 +00:00
|
|
|
|
2013-08-18 20:38:56 +00:00
|
|
|
// If the argument is a key, add a reference to it as well
|
|
|
|
if (entryIter->argumentIsAKey) {
|
|
|
|
_referenceTable[entryIter->argument].push_back(puzzlePtr);
|
|
|
|
}
|
|
|
|
}
|
2013-07-11 05:26:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove duplicate entries
|
2013-07-30 19:35:34 +00:00
|
|
|
for (Common::HashMap<uint32, Common::Array<Puzzle *> >::iterator referenceTableIter = _referenceTable.begin(); referenceTableIter != _referenceTable.end(); referenceTableIter++) {
|
2013-08-02 20:31:35 +00:00
|
|
|
removeDuplicateEntries(referenceTableIter->_value);
|
2013-07-11 05:26:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-30 19:25:31 +00:00
|
|
|
void ScriptManager::updateNodes(uint deltaTimeMillis) {
|
2013-07-11 05:26:32 +00:00
|
|
|
// If process() returns true, it means the node can be deleted
|
2013-08-24 04:53:19 +00:00
|
|
|
for (Common::List<ActionNode *>::iterator iter = _activeNodes.begin(); iter != _activeNodes.end();) {
|
|
|
|
if ((*iter)->process(deltaTimeMillis)) {
|
|
|
|
// Delete the node then remove the pointer
|
|
|
|
delete (*iter);
|
2013-07-11 05:26:32 +00:00
|
|
|
iter = _activeNodes.erase(iter);
|
|
|
|
} else {
|
|
|
|
iter++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptManager::checkPuzzleCriteria() {
|
|
|
|
while (!_puzzlesToCheck.empty()) {
|
|
|
|
Puzzle *puzzle = _puzzlesToCheck.pop();
|
2013-07-30 02:52:27 +00:00
|
|
|
|
2013-08-06 00:18:04 +00:00
|
|
|
// Check if the puzzle is already finished
|
|
|
|
// Also check that the puzzle isn't disabled
|
|
|
|
if (getStateValue(puzzle->key) == 1 &&
|
2013-08-18 01:38:06 +00:00
|
|
|
(puzzle->flags & Puzzle::DISABLED) == 0) {
|
2013-08-06 00:18:04 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2013-07-11 05:26:32 +00:00
|
|
|
// Check each Criteria
|
2013-08-18 20:38:56 +00:00
|
|
|
|
2013-07-30 02:52:27 +00:00
|
|
|
bool criteriaMet = false;
|
2013-08-18 20:38:56 +00:00
|
|
|
for (Common::List<Common::List<Puzzle::CriteriaEntry> >::iterator criteriaIter = puzzle->criteriaList.begin(); criteriaIter != puzzle->criteriaList.end(); criteriaIter++) {
|
|
|
|
criteriaMet = false;
|
|
|
|
|
|
|
|
for (Common::List<Puzzle::CriteriaEntry>::iterator entryIter = (*criteriaIter).begin(); entryIter != (*criteriaIter).end(); entryIter++) {
|
|
|
|
// Get the value to compare against
|
|
|
|
uint argumentValue;
|
|
|
|
if ((*entryIter).argumentIsAKey)
|
|
|
|
argumentValue = getStateValue(entryIter->argument);
|
|
|
|
else
|
|
|
|
argumentValue = entryIter->argument;
|
|
|
|
|
|
|
|
// Do the comparison
|
|
|
|
switch ((*entryIter).criteriaOperator) {
|
|
|
|
case Puzzle::EQUAL_TO:
|
|
|
|
criteriaMet = getStateValue(entryIter->key) == argumentValue;
|
|
|
|
break;
|
|
|
|
case Puzzle::NOT_EQUAL_TO:
|
|
|
|
criteriaMet = getStateValue(entryIter->key) != argumentValue;
|
|
|
|
break;
|
|
|
|
case Puzzle::GREATER_THAN:
|
|
|
|
criteriaMet = getStateValue(entryIter->key) > argumentValue;
|
|
|
|
break;
|
|
|
|
case Puzzle::LESS_THAN:
|
|
|
|
criteriaMet = getStateValue(entryIter->key) < argumentValue;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If one check returns false, don't keep checking
|
|
|
|
if (!criteriaMet) {
|
|
|
|
break;
|
|
|
|
}
|
2013-07-11 05:26:32 +00:00
|
|
|
}
|
|
|
|
|
2013-08-18 20:38:56 +00:00
|
|
|
// If any of the Criteria are *fully* met, then execute the results
|
|
|
|
if (criteriaMet) {
|
2013-07-30 02:52:27 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// criteriaList can be empty. Aka, the puzzle should be executed immediately
|
|
|
|
if (puzzle->criteriaList.empty() || criteriaMet) {
|
2013-08-06 00:09:28 +00:00
|
|
|
debug("Puzzle %u criteria passed. Executing its ResultActions", puzzle->key);
|
|
|
|
|
2013-08-10 22:36:57 +00:00
|
|
|
bool shouldContinue = true;
|
2013-08-19 00:51:27 +00:00
|
|
|
for (Common::List<ResultAction *>::iterator resultIter = puzzle->resultActions.begin(); resultIter != puzzle->resultActions.end(); resultIter++) {
|
2013-08-10 22:36:57 +00:00
|
|
|
shouldContinue = shouldContinue && (*resultIter)->execute(_engine);
|
2013-07-11 05:26:32 +00:00
|
|
|
}
|
2013-08-06 00:09:54 +00:00
|
|
|
|
|
|
|
// Set the puzzle as completed
|
|
|
|
setStateValue(puzzle->key, 1);
|
2013-08-10 22:36:57 +00:00
|
|
|
|
|
|
|
if (!shouldContinue) {
|
|
|
|
break;
|
|
|
|
}
|
2013-07-11 05:26:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-30 19:25:31 +00:00
|
|
|
uint ScriptManager::getStateValue(uint32 key) {
|
2013-08-05 05:00:21 +00:00
|
|
|
if (_globalState.contains(key))
|
|
|
|
return _globalState[key];
|
|
|
|
else
|
|
|
|
return 0;
|
2013-07-01 22:19:09 +00:00
|
|
|
}
|
|
|
|
|
2013-07-30 19:25:31 +00:00
|
|
|
void ScriptManager::setStateValue(uint32 key, uint value) {
|
2013-07-01 22:19:09 +00:00
|
|
|
_globalState[key] = value;
|
2013-08-03 19:59:00 +00:00
|
|
|
|
|
|
|
if (_referenceTable.contains(key)) {
|
|
|
|
for (Common::Array<Puzzle *>::iterator iter = _referenceTable[key].begin(); iter != _referenceTable[key].end(); iter++) {
|
|
|
|
_puzzlesToCheck.push((*iter));
|
|
|
|
}
|
|
|
|
}
|
2013-07-01 22:19:09 +00:00
|
|
|
}
|
|
|
|
|
2013-07-30 19:25:31 +00:00
|
|
|
void ScriptManager::addToStateValue(uint32 key, uint valueToAdd) {
|
2013-07-01 22:19:09 +00:00
|
|
|
_globalState[key] += valueToAdd;
|
|
|
|
}
|
|
|
|
|
2013-08-18 20:28:16 +00:00
|
|
|
bool ScriptManager::enableControl(uint32 key) {
|
|
|
|
if (!_activeControls.contains(key)) {
|
|
|
|
return false;
|
|
|
|
} else {
|
2013-08-24 04:59:28 +00:00
|
|
|
return _activeControls[key]->enable();
|
2013-08-18 20:28:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScriptManager::disableControl(uint32 key) {
|
|
|
|
if (!_activeControls.contains(key)) {
|
|
|
|
return false;
|
|
|
|
} else {
|
2013-08-24 04:53:19 +00:00
|
|
|
return _activeControls[key]->disable();
|
2013-08-18 20:28:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-24 04:53:19 +00:00
|
|
|
void ScriptManager::addActionNode(ActionNode *node) {
|
2013-07-11 05:26:32 +00:00
|
|
|
_activeNodes.push_back(node);
|
|
|
|
}
|
|
|
|
|
2013-08-16 22:27:51 +00:00
|
|
|
void ScriptManager::changeLocation(char world, char room, char node, char view, uint32 offset) {
|
2013-08-10 22:31:57 +00:00
|
|
|
_nextLocation.world = world;
|
|
|
|
_nextLocation.room = room;
|
|
|
|
_nextLocation.node = node;
|
|
|
|
_nextLocation.view = view;
|
2013-08-16 22:27:51 +00:00
|
|
|
_nextLocation.offset = offset;
|
2013-08-10 22:31:57 +00:00
|
|
|
|
|
|
|
_changeLocation = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScriptManager::changeLocationIntern() {
|
|
|
|
assert(_nextLocation.world != 0);
|
2013-08-16 22:27:51 +00:00
|
|
|
debug("Changing location to: %c %c %c %c %u", _nextLocation.world, _nextLocation.room, _nextLocation.node, _nextLocation.view, _nextLocation.offset);
|
2013-08-10 22:31:57 +00:00
|
|
|
|
2013-07-30 02:57:45 +00:00
|
|
|
// Clear all the containers
|
|
|
|
_referenceTable.clear();
|
|
|
|
_puzzlesToCheck.clear();
|
2013-08-21 01:55:00 +00:00
|
|
|
for (Common::List<Puzzle *>::iterator iter = _activePuzzles.begin(); iter != _activePuzzles.end(); iter++) {
|
|
|
|
delete (*iter);
|
|
|
|
}
|
2013-07-30 02:57:45 +00:00
|
|
|
_activePuzzles.clear();
|
2013-08-24 05:24:43 +00:00
|
|
|
for (Common::HashMap<uint32, Control *>::iterator iter = _activeControls.begin(); iter != _activeControls.end(); iter++) {
|
|
|
|
delete (*iter)._value;
|
|
|
|
}
|
2013-07-30 02:57:45 +00:00
|
|
|
_activeControls.clear();
|
2013-08-11 20:10:52 +00:00
|
|
|
_engine->clearAllMouseEvents();
|
2013-08-24 05:25:01 +00:00
|
|
|
// TODO: See if we need to clear _activeNodes as well. And if so, remember to delete the nodes before clearing the list
|
2013-07-30 02:57:45 +00:00
|
|
|
|
2013-08-14 15:45:39 +00:00
|
|
|
// Revert to the idle cursor
|
|
|
|
_engine->getCursorManager()->revertToIdle();
|
|
|
|
|
|
|
|
// Reset the background velocity
|
|
|
|
_engine->getRenderManager()->setBackgroundVelocity(0);
|
|
|
|
|
2013-07-30 02:57:45 +00:00
|
|
|
// Parse into puzzles and controls
|
2013-08-10 22:31:57 +00:00
|
|
|
Common::String fileName = Common::String::format("%c%c%c%c.scr", _nextLocation.world, _nextLocation.room, _nextLocation.node, _nextLocation.view);
|
2013-07-30 02:57:45 +00:00
|
|
|
parseScrFile(fileName);
|
|
|
|
|
2013-08-17 14:47:05 +00:00
|
|
|
// Change the background position
|
|
|
|
_engine->getRenderManager()->setBackgroundPosition(_nextLocation.offset);
|
|
|
|
|
2013-08-18 20:28:16 +00:00
|
|
|
// Enable all the controls
|
|
|
|
for (Common::HashMap<uint32, Control *>::iterator iter = _activeControls.begin(); iter != _activeControls.end(); iter++) {
|
2013-08-24 04:59:28 +00:00
|
|
|
(*iter)._value->enable();
|
2013-08-18 20:28:16 +00:00
|
|
|
}
|
|
|
|
|
2013-08-18 20:41:52 +00:00
|
|
|
// Add all the local puzzles to the queue to be checked
|
2013-08-21 01:50:14 +00:00
|
|
|
for (Common::List<Puzzle *>::iterator iter = _activePuzzles.begin(); iter != _activePuzzles.end(); iter++) {
|
2013-08-18 01:38:06 +00:00
|
|
|
// Reset any Puzzles that have the flag ONCE_PER_INST
|
2013-08-21 01:50:14 +00:00
|
|
|
if (((*iter)->flags & Puzzle::ONCE_PER_INST) == Puzzle::ONCE_PER_INST) {
|
|
|
|
setStateValue((*iter)->key, 0);
|
2013-08-18 01:38:06 +00:00
|
|
|
}
|
|
|
|
|
2013-08-21 01:50:14 +00:00
|
|
|
_puzzlesToCheck.push((*iter));
|
2013-07-30 02:57:45 +00:00
|
|
|
}
|
2013-08-10 22:31:57 +00:00
|
|
|
|
2013-08-18 20:41:52 +00:00
|
|
|
// Add all the global puzzles to the queue to be checked
|
2013-08-21 01:50:14 +00:00
|
|
|
for (Common::List<Puzzle *>::iterator iter = _globalPuzzles.begin(); iter != _globalPuzzles.end(); iter++) {
|
2013-08-18 01:38:06 +00:00
|
|
|
// Reset any Puzzles that have the flag ONCE_PER_INST
|
2013-08-21 01:50:14 +00:00
|
|
|
if (((*iter)->flags & Puzzle::ONCE_PER_INST) == Puzzle::ONCE_PER_INST) {
|
|
|
|
setStateValue((*iter)->key, 0);
|
2013-08-18 01:38:06 +00:00
|
|
|
}
|
|
|
|
|
2013-08-21 01:50:14 +00:00
|
|
|
_puzzlesToCheck.push((*iter));
|
2013-08-10 22:31:57 +00:00
|
|
|
}
|
2013-08-18 01:38:06 +00:00
|
|
|
|
|
|
|
// Create the puzzle reference table
|
|
|
|
createReferenceTable();
|
2013-07-30 02:57:45 +00:00
|
|
|
}
|
|
|
|
|
2013-07-01 22:19:09 +00:00
|
|
|
} // End of namespace ZVision
|