2021-05-17 20:47:39 +02: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.
|
|
|
|
*
|
2021-12-26 18:47:58 +01:00
|
|
|
* 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 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
2021-05-17 20:47:39 +02: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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* aint32 with this program; if not, write to the Free Software
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Based on the original sources
|
|
|
|
* Faery Tale II -- The Halls of the Dead
|
|
|
|
* (c) 1993-1996 The Wyrmkeep Entertainment Co.
|
|
|
|
*/
|
|
|
|
|
2021-06-23 14:41:01 +02:00
|
|
|
#include "saga2/saga2.h"
|
2021-05-17 20:47:39 +02:00
|
|
|
#include "saga2/fta.h"
|
|
|
|
#include "saga2/patrol.h"
|
2021-06-01 21:31:22 +02:00
|
|
|
#include "saga2/cmisc.h"
|
|
|
|
#include "saga2/tcoords.h"
|
2021-06-02 00:16:29 +02:00
|
|
|
#include "saga2/hresmgr.h"
|
2021-05-17 20:47:39 +02:00
|
|
|
|
|
|
|
namespace Saga2 {
|
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
extern int16 worldCount; // Number of worlds
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
PatrolRouteList **patrolRouteList = nullptr; // Global patrol route array
|
|
|
|
|
|
|
|
PatrolRoute::PatrolRoute(Common::SeekableReadStream *stream) {
|
|
|
|
_wayPoints = stream->readSint16LE();
|
|
|
|
|
|
|
|
_route = (TilePoint **)malloc(sizeof(TilePoint *) * _wayPoints);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
for (int i = 0; i < _wayPoints; i++)
|
|
|
|
_route[i] = new TilePoint(stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
PatrolRoute::~PatrolRoute() {
|
|
|
|
for (int i = 0; i < _wayPoints; i++)
|
|
|
|
delete _route[i];
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
free(_route);
|
|
|
|
}
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
// Returns a const reference to a specified way point
|
2021-06-01 21:31:22 +02:00
|
|
|
const TilePoint &PatrolRoute::operator [](int16 index) const {
|
2021-06-11 19:32:51 +02:00
|
|
|
return *_route[index];
|
2021-06-01 21:31:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-05-17 20:47:39 +02:00
|
|
|
/* ===================================================================== *
|
|
|
|
PatrolRouteList member functions
|
|
|
|
* ===================================================================== */
|
2021-06-11 19:32:51 +02:00
|
|
|
PatrolRouteList::PatrolRouteList(Common::SeekableReadStream *stream) {
|
|
|
|
_numRoutes = stream->readSint16LE();
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
_routes = (PatrolRoute **)malloc(sizeof(PatrolRoute *) * _numRoutes);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
for (int i = 0; i < _numRoutes; i++)
|
|
|
|
_routes[i] = new PatrolRoute(stream);
|
|
|
|
}
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
PatrolRouteList::~PatrolRouteList() {
|
|
|
|
for (int i = 0; i < _numRoutes; i++)
|
|
|
|
delete _routes[i];
|
2021-06-10 16:49:17 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
free(_routes);
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* ===================================================================== *
|
|
|
|
PatrolRouteIterator member functions
|
|
|
|
* ===================================================================== */
|
|
|
|
|
|
|
|
PatrolRouteIterator::PatrolRouteIterator(uint8 map, int16 rte, uint8 type) :
|
2021-06-11 15:47:18 +02:00
|
|
|
_mapNum(map), _routeNo(rte), _flags(type & 0xF) {
|
2021-06-11 19:32:51 +02:00
|
|
|
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2022-10-29 13:44:19 +02:00
|
|
|
if (_flags & kPatrolRouteRandom)
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo = g_vm->_rnd->getRandomNumber(route.vertices() - 1);
|
2021-05-17 20:47:39 +02:00
|
|
|
else {
|
2022-10-29 13:44:19 +02:00
|
|
|
if (_flags & kPatrolRouteReverse)
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo = route.vertices() - 1;
|
2021-05-17 20:47:39 +02:00
|
|
|
else
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo = 0;
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
PatrolRouteIterator::PatrolRouteIterator(uint8 map, int16 rte, uint8 type, int16 startingPoint) :
|
|
|
|
_mapNum(map), _routeNo(rte), _flags(type & 0xF) {
|
2021-06-11 19:32:51 +02:00
|
|
|
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
|
2021-06-11 15:47:18 +02:00
|
|
|
|
|
|
|
_vertexNo = clamp(0, startingPoint, route.vertices() - 1);
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
|
2021-07-11 03:18:07 +09:00
|
|
|
void PatrolRouteIterator::read(Common::InSaveFile *in) {
|
|
|
|
_routeNo = in->readSint16LE();
|
|
|
|
_vertexNo = in->readSint16LE();
|
|
|
|
_mapNum = in->readByte();
|
|
|
|
_flags = in->readByte();
|
|
|
|
}
|
|
|
|
|
2021-07-14 14:19:21 +09:00
|
|
|
void PatrolRouteIterator::write(Common::MemoryWriteStreamDynamic *out) const {
|
2021-07-11 03:18:07 +09:00
|
|
|
out->writeSint16LE(_routeNo);
|
|
|
|
out->writeSint16LE(_vertexNo);
|
|
|
|
out->writeByte(_mapNum);
|
|
|
|
out->writeByte(_flags);
|
|
|
|
}
|
|
|
|
|
2021-05-17 20:47:39 +02:00
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// Increment the waypoint index
|
2021-09-11 12:13:35 +03:00
|
|
|
void PatrolRouteIterator::increment() {
|
2021-06-11 19:32:51 +02:00
|
|
|
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
|
2021-06-11 15:47:18 +02:00
|
|
|
|
|
|
|
_vertexNo++;
|
|
|
|
|
|
|
|
if (_vertexNo >= route.vertices()) {
|
2022-10-29 13:44:19 +02:00
|
|
|
if (_flags & kPatrolRouteAlternate) {
|
2021-06-11 15:47:18 +02:00
|
|
|
// If alternating, initialize for iteration in the alternate
|
|
|
|
// direction
|
2022-10-29 13:44:19 +02:00
|
|
|
_flags |= kPatrolRouteInAlternate;
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo = MAX(route.vertices() - 2, 0);
|
2022-10-29 13:44:19 +02:00
|
|
|
} else if (_flags & kPatrolRouteRepeat)
|
2021-06-11 15:47:18 +02:00
|
|
|
// If repeating, reset the waypoint index
|
|
|
|
_vertexNo = 0;
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// Decrement the waypoint index
|
2021-09-11 12:13:35 +03:00
|
|
|
void PatrolRouteIterator::decrement() {
|
2021-06-11 19:32:51 +02:00
|
|
|
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
|
2021-06-11 15:47:18 +02:00
|
|
|
|
|
|
|
_vertexNo--;
|
|
|
|
|
|
|
|
if (_vertexNo < 0) {
|
2022-10-29 13:44:19 +02:00
|
|
|
if (_flags & kPatrolRouteAlternate) {
|
2021-06-11 15:47:18 +02:00
|
|
|
// If alternating, initialize for iteration in the alternate
|
|
|
|
// direction
|
2022-10-29 13:44:19 +02:00
|
|
|
_flags |= kPatrolRouteInAlternate;
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo = MIN(1, route.vertices() - 1);
|
2022-10-29 13:44:19 +02:00
|
|
|
} else if (_flags & kPatrolRouteRepeat)
|
2021-06-11 15:47:18 +02:00
|
|
|
// If repeating, reset the waypoint index
|
|
|
|
_vertexNo = route.vertices() - 1;
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// Increment the waypoint index in the alternate direction
|
2021-09-11 12:13:35 +03:00
|
|
|
void PatrolRouteIterator::altIncrement() {
|
2021-06-11 19:32:51 +02:00
|
|
|
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo++;
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2022-10-29 13:44:19 +02:00
|
|
|
if (_vertexNo >= route.vertices() && (_flags & kPatrolRouteRepeat)) {
|
2021-06-11 15:47:18 +02:00
|
|
|
// If repeating, initialize for iteration in the standard
|
|
|
|
// direction, and reset the waypoint index
|
2022-10-29 13:44:19 +02:00
|
|
|
_flags &= ~kPatrolRouteInAlternate;
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo = MAX(route.vertices() - 2, 0);
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// Decrement the waypoint index in the alternate direction
|
2021-09-11 12:13:35 +03:00
|
|
|
void PatrolRouteIterator::altDecrement() {
|
2021-06-11 19:32:51 +02:00
|
|
|
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo--;
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2022-10-29 13:44:19 +02:00
|
|
|
if (_vertexNo < 0 && (_flags & kPatrolRouteRepeat)) {
|
2021-06-11 15:47:18 +02:00
|
|
|
// If repeating, initialize for iteration in the standard
|
|
|
|
// direction, and reset the waypoint index
|
2022-10-29 13:44:19 +02:00
|
|
|
_flags &= ~kPatrolRouteInAlternate;
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo = MIN(1, route.vertices() - 1);
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// Return the coordinates of the current waypoint
|
2021-09-11 12:13:35 +03:00
|
|
|
const TilePoint PatrolRouteIterator::operator*() const {
|
2021-06-11 19:32:51 +02:00
|
|
|
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
return _vertexNo >= 0 && _vertexNo < route.vertices() ? route[_vertexNo] : Nowhere;
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
|
2021-09-11 12:13:35 +03:00
|
|
|
const PatrolRouteIterator &PatrolRouteIterator::operator++() {
|
2021-06-11 19:32:51 +02:00
|
|
|
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-24 00:41:19 +02:00
|
|
|
if (_vertexNo >= 0 && _vertexNo < route.vertices()) {
|
2022-10-29 13:44:19 +02:00
|
|
|
if (!(_flags & kPatrolRouteRandom)) {
|
|
|
|
if (!(_flags & kPatrolRouteInAlternate)) {
|
|
|
|
if (!(_flags & kPatrolRouteReverse))
|
2021-05-17 20:47:39 +02:00
|
|
|
increment();
|
|
|
|
else
|
|
|
|
decrement();
|
|
|
|
} else {
|
2022-10-29 13:44:19 +02:00
|
|
|
if (!(_flags & kPatrolRouteReverse))
|
2021-05-17 20:47:39 +02:00
|
|
|
altDecrement();
|
|
|
|
else
|
|
|
|
altIncrement();
|
|
|
|
}
|
|
|
|
} else {
|
2021-06-11 15:47:18 +02:00
|
|
|
_vertexNo = g_vm->_rnd->getRandomNumber(route.vertices() - 1);
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
// Load the patrol routes from the resource file
|
2021-09-11 12:13:35 +03:00
|
|
|
void initPatrolRoutes() {
|
2021-06-11 15:47:18 +02:00
|
|
|
// Get patrol route resource context
|
2021-06-11 19:32:51 +02:00
|
|
|
hResContext *patrolRouteRes = auxResFile->newContext(MKTAG('P', 'T', 'R', 'L'), "patrol route resource");
|
2021-06-03 00:48:12 +09:00
|
|
|
if (patrolRouteRes == nullptr || !patrolRouteRes->_valid)
|
2021-05-17 20:47:39 +02:00
|
|
|
error("Error accessing patrol route resource group.");
|
|
|
|
|
2021-06-13 16:10:30 +02:00
|
|
|
debugC(1, kDebugLoading, "Loading Patrol Routes for %d worlds", worldCount);
|
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
// Allocate the patrol route list array
|
2021-06-11 19:32:51 +02:00
|
|
|
patrolRouteList = (PatrolRouteList **)malloc(sizeof(PatrolRouteList *) * worldCount);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-03 00:48:12 +09:00
|
|
|
if (patrolRouteList == nullptr)
|
2021-05-17 20:47:39 +02:00
|
|
|
error("Unable to allocate the patrol route list");
|
|
|
|
|
2021-06-13 16:10:30 +02:00
|
|
|
int count = 0;
|
2021-06-11 19:32:51 +02:00
|
|
|
for (int16 i = 0; i < worldCount; i++) {
|
|
|
|
patrolRouteList[i] = nullptr;
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
// Load this worlds's patrol routes
|
2021-06-08 16:46:55 +02:00
|
|
|
if (patrolRouteRes->size(MKTAG('R', 'T', 'E', i)) > 0) {
|
2021-06-11 19:32:51 +02:00
|
|
|
Common::SeekableReadStream *stream = loadResourceToStream(patrolRouteRes, MKTAG('R', 'T', 'E', i), "patrol route data");
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
patrolRouteList[i] = new PatrolRouteList(stream);
|
2021-06-13 16:10:30 +02:00
|
|
|
count++;
|
2021-06-11 22:37:05 +02:00
|
|
|
|
|
|
|
delete stream;
|
2021-06-11 19:32:51 +02:00
|
|
|
}
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
|
2021-06-13 16:10:30 +02:00
|
|
|
debugC(1, kDebugLoading, "Loading Patrol Routes, loaded %d entries", count);
|
|
|
|
|
2021-06-11 15:47:18 +02:00
|
|
|
// Dispose of the patrol route resource context
|
2021-05-17 20:47:39 +02:00
|
|
|
auxResFile->disposeContext(patrolRouteRes);
|
|
|
|
}
|
|
|
|
|
2021-09-11 12:13:35 +03:00
|
|
|
void cleanupPatrolRoutes() {
|
2021-06-11 19:32:51 +02:00
|
|
|
if (!patrolRouteList)
|
|
|
|
return;
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
for (int16 i = 0; i < worldCount; i++)
|
|
|
|
delete patrolRouteList[i];
|
|
|
|
|
|
|
|
free(patrolRouteList);
|
2021-05-17 20:47:39 +02:00
|
|
|
|
2021-06-11 19:32:51 +02:00
|
|
|
patrolRouteList = nullptr;
|
2021-05-17 20:47:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
} // end of namespace Saga2
|