scummvm/engines/saga2/patrol.cpp

269 lines
8.1 KiB
C++
Raw Normal View History

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.
*
* 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 {
extern int16 worldCount; // Number of worlds
2021-05-17 20:47:39 +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
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
free(_route);
}
2021-05-17 20:47:39 +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 {
return *_route[index];
2021-06-01 21:31:22 +02:00
}
2021-05-17 20:47:39 +02:00
/* ===================================================================== *
PatrolRouteList member functions
* ===================================================================== */
PatrolRouteList::PatrolRouteList(Common::SeekableReadStream *stream) {
_numRoutes = stream->readSint16LE();
2021-05-17 20:47:39 +02:00
_routes = (PatrolRoute **)malloc(sizeof(PatrolRoute *) * _numRoutes);
2021-05-17 20:47:39 +02:00
for (int i = 0; i < _numRoutes; i++)
_routes[i] = new PatrolRoute(stream);
}
2021-05-17 20:47:39 +02:00
PatrolRouteList::~PatrolRouteList() {
for (int i = 0; i < _numRoutes; i++)
delete _routes[i];
2021-06-10 16:49:17 +02:00
free(_routes);
2021-05-17 20:47:39 +02:00
}
/* ===================================================================== *
PatrolRouteIterator member functions
* ===================================================================== */
PatrolRouteIterator::PatrolRouteIterator(uint8 map, int16 rte, uint8 type) :
_mapNum(map), _routeNo(rte), _flags(type & 0xF) {
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)
_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)
_vertexNo = route.vertices() - 1;
2021-05-17 20:47:39 +02:00
else
_vertexNo = 0;
2021-05-17 20:47:39 +02:00
}
}
PatrolRouteIterator::PatrolRouteIterator(uint8 map, int16 rte, uint8 type, int16 startingPoint) :
_mapNum(map), _routeNo(rte), _flags(type & 0xF) {
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
_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();
}
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() {
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
_vertexNo++;
if (_vertexNo >= route.vertices()) {
2022-10-29 13:44:19 +02:00
if (_flags & kPatrolRouteAlternate) {
// If alternating, initialize for iteration in the alternate
// direction
2022-10-29 13:44:19 +02:00
_flags |= kPatrolRouteInAlternate;
_vertexNo = MAX(route.vertices() - 2, 0);
2022-10-29 13:44:19 +02:00
} else if (_flags & kPatrolRouteRepeat)
// 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() {
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
_vertexNo--;
if (_vertexNo < 0) {
2022-10-29 13:44:19 +02:00
if (_flags & kPatrolRouteAlternate) {
// If alternating, initialize for iteration in the alternate
// direction
2022-10-29 13:44:19 +02:00
_flags |= kPatrolRouteInAlternate;
_vertexNo = MIN(1, route.vertices() - 1);
2022-10-29 13:44:19 +02:00
} else if (_flags & kPatrolRouteRepeat)
// 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() {
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
2021-05-17 20:47:39 +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)) {
// If repeating, initialize for iteration in the standard
// direction, and reset the waypoint index
2022-10-29 13:44:19 +02:00
_flags &= ~kPatrolRouteInAlternate;
_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() {
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
2021-05-17 20:47:39 +02:00
_vertexNo--;
2021-05-17 20:47:39 +02:00
2022-10-29 13:44:19 +02:00
if (_vertexNo < 0 && (_flags & kPatrolRouteRepeat)) {
// If repeating, initialize for iteration in the standard
// direction, and reset the waypoint index
2022-10-29 13:44:19 +02:00
_flags &= ~kPatrolRouteInAlternate;
_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 {
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
2021-05-17 20:47:39 +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++() {
const PatrolRoute &route = patrolRouteList[_mapNum]->getRoute(_routeNo);
2021-05-17 20:47:39 +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 {
_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() {
// Get patrol route resource context
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);
// Allocate the patrol route list array
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;
for (int16 i = 0; i < worldCount; i++) {
patrolRouteList[i] = nullptr;
2021-05-17 20:47:39 +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) {
Common::SeekableReadStream *stream = loadResourceToStream(patrolRouteRes, MKTAG('R', 'T', 'E', i), "patrol route data");
2021-05-17 20:47:39 +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-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);
// 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() {
if (!patrolRouteList)
return;
2021-05-17 20:47:39 +02:00
for (int16 i = 0; i < worldCount; i++)
delete patrolRouteList[i];
free(patrolRouteList);
2021-05-17 20:47:39 +02:00
patrolRouteList = nullptr;
2021-05-17 20:47:39 +02:00
}
} // end of namespace Saga2