2017-05-26 05:24:38 +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 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.
|
|
|
|
*
|
|
|
|
*/
|
2017-06-05 19:10:47 +02:00
|
|
|
|
|
|
|
#include "sludge/allfiles.h"
|
2017-12-19 22:15:55 +01:00
|
|
|
#include "sludge/floor.h"
|
2017-07-20 00:41:13 +02:00
|
|
|
#include "sludge/graphics.h"
|
2017-12-19 22:15:55 +01:00
|
|
|
#include "sludge/loadsave.h"
|
|
|
|
#include "sludge/moreio.h"
|
|
|
|
#include "sludge/newfatal.h"
|
2017-06-05 19:10:47 +02:00
|
|
|
#include "sludge/objtypes.h"
|
|
|
|
#include "sludge/people.h"
|
2017-12-19 22:15:55 +01:00
|
|
|
#include "sludge/region.h"
|
2017-07-18 19:50:45 +02:00
|
|
|
#include "sludge/sludge.h"
|
2017-12-19 22:15:55 +01:00
|
|
|
#include "sludge/sludger.h"
|
2017-06-05 19:10:47 +02:00
|
|
|
#include "sludge/sound.h"
|
2017-12-19 21:24:31 +01:00
|
|
|
#include "sludge/speech.h"
|
2017-12-19 22:15:55 +01:00
|
|
|
#include "sludge/sprbanks.h"
|
|
|
|
#include "sludge/sprites.h"
|
|
|
|
#include "sludge/variable.h"
|
2017-07-13 18:08:30 +02:00
|
|
|
#include "sludge/version.h"
|
2017-12-19 22:15:55 +01:00
|
|
|
#include "sludge/zbuffer.h"
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
#define ANGLEFIX (180.0 / 3.14157)
|
|
|
|
#define ANI_STAND 0
|
|
|
|
#define ANI_WALK 1
|
|
|
|
#define ANI_TALK 2
|
|
|
|
|
2017-05-26 21:25:11 +02:00
|
|
|
namespace Sludge {
|
|
|
|
|
2017-07-20 10:39:24 +02:00
|
|
|
extern VariableStack *noStack;
|
2017-05-26 05:24:38 +02:00
|
|
|
extern int ssgVersion;
|
2017-07-20 10:39:24 +02:00
|
|
|
extern Floor *currentFloor;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2018-04-15 21:35:19 +02:00
|
|
|
PersonaAnimation::PersonaAnimation() {
|
|
|
|
theSprites = nullptr;
|
|
|
|
numFrames = 0;
|
|
|
|
frames = nullptr;
|
2018-04-15 21:10:02 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:35:19 +02:00
|
|
|
PersonaAnimation::~PersonaAnimation() {
|
|
|
|
if (numFrames) {
|
|
|
|
delete[] frames;
|
|
|
|
frames = nullptr;
|
|
|
|
}
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:35:19 +02:00
|
|
|
PersonaAnimation::PersonaAnimation(int num, VariableStack *&stacky) {
|
|
|
|
theSprites = nullptr;
|
|
|
|
numFrames = num;
|
|
|
|
frames = new AnimFrame[num];
|
2017-05-26 05:24:38 +02:00
|
|
|
int a = num, frameNum, howMany;
|
|
|
|
|
|
|
|
while (a) {
|
2017-05-29 08:02:59 +02:00
|
|
|
a--;
|
2018-04-15 21:35:19 +02:00
|
|
|
frames[a].noise = 0;
|
2017-05-27 20:16:54 +02:00
|
|
|
if (stacky->thisVar.varType == SVT_FILE) {
|
2018-04-15 21:35:19 +02:00
|
|
|
frames[a].noise = stacky->thisVar.varData.intValue;
|
2017-05-27 20:16:54 +02:00
|
|
|
} else if (stacky->thisVar.varType == SVT_FUNC) {
|
2018-04-15 21:35:19 +02:00
|
|
|
frames[a].noise = -stacky->thisVar.varData.intValue;
|
2017-05-27 20:16:54 +02:00
|
|
|
} else if (stacky->thisVar.varType == SVT_STACK) {
|
2017-06-05 11:49:19 +02:00
|
|
|
getValueType(frameNum, SVT_INT, stacky->thisVar.varData.theStack->first->thisVar);
|
|
|
|
getValueType(howMany, SVT_INT, stacky->thisVar.varData.theStack->first->next->thisVar);
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-27 20:16:54 +02:00
|
|
|
getValueType(frameNum, SVT_INT, stacky->thisVar);
|
2017-05-26 05:24:38 +02:00
|
|
|
howMany = 1;
|
|
|
|
}
|
|
|
|
trimStack(stacky);
|
2018-04-15 21:35:19 +02:00
|
|
|
frames[a].frameNum = frameNum;
|
|
|
|
frames[a].howMany = howMany;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:35:19 +02:00
|
|
|
PersonaAnimation::PersonaAnimation(PersonaAnimation *orig) {
|
2017-05-27 20:16:54 +02:00
|
|
|
int num = orig->numFrames;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// Copy the easy bits...
|
2018-04-15 21:35:19 +02:00
|
|
|
theSprites = orig->theSprites;
|
|
|
|
numFrames = num;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
if (num) {
|
2018-04-15 21:35:19 +02:00
|
|
|
// Argh! Frames! We need a whole NEW array of AnimFrame structures...
|
|
|
|
frames = new AnimFrame[num];
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-06-05 12:03:50 +02:00
|
|
|
for (int a = 0; a < num; a++) {
|
2018-04-15 21:35:19 +02:00
|
|
|
frames[a].frameNum = orig->frames[a].frameNum;
|
|
|
|
frames[a].howMany = orig->frames[a].howMany;
|
|
|
|
frames[a].noise = orig->frames[a].noise;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
} else {
|
2018-04-15 21:35:19 +02:00
|
|
|
frames = nullptr;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
2018-04-15 21:35:19 +02:00
|
|
|
}
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2018-04-15 21:35:19 +02:00
|
|
|
int PersonaAnimation::getTotalTime() {
|
|
|
|
int total = 0;
|
|
|
|
for (int a = 0; a < numFrames; a++) {
|
|
|
|
total += frames[a].howMany;
|
|
|
|
}
|
|
|
|
return total;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:35:19 +02:00
|
|
|
bool PersonaAnimation::save(Common::WriteStream *stream) {
|
|
|
|
stream->writeUint16BE(numFrames);
|
|
|
|
if (numFrames) {
|
|
|
|
stream->writeUint32LE(theSprites->ID);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2018-04-15 21:35:19 +02:00
|
|
|
for (int a = 0; a < numFrames; a++) {
|
|
|
|
stream->writeUint32LE(frames[a].frameNum);
|
|
|
|
stream->writeUint32LE(frames[a].howMany);
|
|
|
|
stream->writeUint32LE(frames[a].noise);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
2018-04-15 21:35:19 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PersonaAnimation::load(Common::SeekableReadStream *stream) {
|
|
|
|
numFrames = stream->readUint16BE();
|
|
|
|
|
|
|
|
if (numFrames) {
|
|
|
|
int a = stream->readUint32LE();
|
|
|
|
frames = new AnimFrame [numFrames];
|
|
|
|
if (!checkNew(frames))
|
|
|
|
return false;
|
|
|
|
theSprites = g_sludge->_gfxMan->loadBankForAnim(a);
|
|
|
|
|
|
|
|
for (a = 0; a < numFrames; a++) {
|
|
|
|
frames[a].frameNum = stream->readUint32LE();
|
|
|
|
frames[a].howMany = stream->readUint32LE();
|
|
|
|
if (ssgVersion >= VERSION(2, 0)) {
|
|
|
|
frames[a].noise = stream->readUint32LE();
|
|
|
|
} else {
|
|
|
|
frames[a].noise = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
theSprites = NULL;
|
|
|
|
frames = NULL;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:56:20 +02:00
|
|
|
bool Persona::save(Common::WriteStream *stream) {
|
|
|
|
int a;
|
|
|
|
stream->writeUint16BE(numDirections);
|
|
|
|
for (a = 0; a < numDirections * 3; a++) {
|
|
|
|
if (!animation[a]->save(stream))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Persona::load(Common::SeekableReadStream *stream) {
|
|
|
|
int a;
|
|
|
|
numDirections = stream->readUint16BE();
|
|
|
|
animation = new PersonaAnimation *[numDirections * 3];
|
|
|
|
if (!checkNew(animation))
|
|
|
|
return false;
|
|
|
|
for (a = 0; a < numDirections * 3; a++) {
|
2018-04-19 12:08:31 +02:00
|
|
|
animation[a] = new PersonaAnimation;
|
2018-04-15 21:56:20 +02:00
|
|
|
if (!checkNew(animation[a]))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!animation[a]->load(stream))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-15 22:09:37 +02:00
|
|
|
void OnScreenPerson::setFrames(int a) {
|
|
|
|
myAnim = myPersona->animation[(a * myPersona->numDirections) + direction];
|
|
|
|
}
|
|
|
|
|
|
|
|
void OnScreenPerson::makeTalker() {
|
|
|
|
setFrames(ANI_TALK);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OnScreenPerson::makeSilent() {
|
|
|
|
setFrames(ANI_STAND);
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:35:19 +02:00
|
|
|
PeopleManager::PeopleManager(SludgeEngine *vm) {
|
|
|
|
_vm = vm;
|
|
|
|
_allPeople = nullptr;
|
|
|
|
_scaleHorizon = 75;
|
|
|
|
_scaleDivide = 150;
|
|
|
|
_personRegion = new ScreenRegion;
|
|
|
|
}
|
|
|
|
|
|
|
|
PeopleManager::~PeopleManager() {
|
|
|
|
kill();
|
|
|
|
|
|
|
|
delete _personRegion;
|
|
|
|
_personRegion = nullptr;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::turnMeAngle(OnScreenPerson *thisPerson, int direc) {
|
2017-05-27 20:16:54 +02:00
|
|
|
int d = thisPerson->myPersona->numDirections;
|
|
|
|
thisPerson->angle = direc;
|
|
|
|
direc += (180 / d) + 180 + thisPerson->angleOffset;
|
2017-05-29 08:02:59 +02:00
|
|
|
while (direc >= 360)
|
|
|
|
direc -= 360;
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->direction = (direc * d) / 360;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::init() {
|
|
|
|
_personRegion->sX = 0;
|
|
|
|
_personRegion->sY = 0;
|
|
|
|
_personRegion->di = -1;
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::spinStep(OnScreenPerson *thisPerson) {
|
2017-05-27 20:16:54 +02:00
|
|
|
int diff = (thisPerson->angle + 360) - thisPerson->wantAngle;
|
2017-06-05 11:49:19 +02:00
|
|
|
int eachSlice = thisPerson->spinSpeed ? thisPerson->spinSpeed : (360 / thisPerson->myPersona->numDirections);
|
2017-05-26 05:24:38 +02:00
|
|
|
while (diff > 180) {
|
|
|
|
diff -= 360;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (diff >= eachSlice) {
|
2017-05-27 20:16:54 +02:00
|
|
|
turnMeAngle(thisPerson, thisPerson->angle - eachSlice);
|
2017-05-29 08:02:59 +02:00
|
|
|
} else if (diff <= -eachSlice) {
|
2017-05-27 20:16:54 +02:00
|
|
|
turnMeAngle(thisPerson, thisPerson->angle + eachSlice);
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-27 20:16:54 +02:00
|
|
|
turnMeAngle(thisPerson, thisPerson->wantAngle);
|
|
|
|
thisPerson->spinning = false;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::rethinkAngle(OnScreenPerson *thisPerson) {
|
2017-05-27 20:16:54 +02:00
|
|
|
int d = thisPerson->myPersona->numDirections;
|
|
|
|
int direc = thisPerson->angle + (180 / d) + 180 + thisPerson->angleOffset;
|
2017-05-29 08:02:59 +02:00
|
|
|
while (direc >= 360)
|
|
|
|
direc -= 360;
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->direction = (direc * d) / 360;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::turnPersonToFace(int thisNum, int direc) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *thisPerson = findPerson(thisNum);
|
2017-05-26 05:24:38 +02:00
|
|
|
if (thisPerson) {
|
2017-05-29 08:02:59 +02:00
|
|
|
if (thisPerson->continueAfterWalking)
|
|
|
|
abortFunction(thisPerson->continueAfterWalking);
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->continueAfterWalking = NULL;
|
|
|
|
thisPerson->walking = false;
|
|
|
|
thisPerson->spinning = false;
|
2017-05-26 05:24:38 +02:00
|
|
|
turnMeAngle(thisPerson, direc);
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_speechMan->isCurrentTalker(thisPerson) ?
|
|
|
|
thisPerson->makeTalker() : thisPerson->makeSilent();
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::setPersonExtra(int thisNum, int extra) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *thisPerson = findPerson(thisNum);
|
2017-05-26 05:24:38 +02:00
|
|
|
if (thisPerson) {
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->extra = extra;
|
2017-05-29 08:02:59 +02:00
|
|
|
if (extra & EXTRA_NOSCALE)
|
|
|
|
thisPerson->scale = 1;
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::setScale(int16 h, int16 d) {
|
|
|
|
_scaleHorizon = h;
|
|
|
|
_scaleDivide = d;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::moveAndScale(OnScreenPerson &me, float x, float y) {
|
2017-05-26 05:24:38 +02:00
|
|
|
me.x = x;
|
|
|
|
me.y = y;
|
2018-04-15 21:10:02 +02:00
|
|
|
if (!(me.extra & EXTRA_NOSCALE) && _scaleDivide)
|
|
|
|
me.scale = (me.y - _scaleHorizon) / _scaleDivide;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
OnScreenPerson *PeopleManager::findPerson(int v) {
|
|
|
|
OnScreenPerson *thisPerson = _allPeople;
|
2017-05-26 05:24:38 +02:00
|
|
|
while (thisPerson) {
|
2017-05-29 08:02:59 +02:00
|
|
|
if (v == thisPerson->thisType->objectNum)
|
|
|
|
break;
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson = thisPerson->next;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
return thisPerson;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::movePerson(int x, int y, int objNum) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(objNum);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (moveMe)
|
|
|
|
moveAndScale(*moveMe, x, y);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::setShown(bool h, int ob) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(ob);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (moveMe)
|
|
|
|
moveMe->show = h;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
enum drawModes {
|
|
|
|
drawModeNormal,
|
|
|
|
drawModeTransparent1,
|
|
|
|
drawModeTransparent2,
|
|
|
|
drawModeTransparent3,
|
|
|
|
drawModeDark1,
|
|
|
|
drawModeDark2,
|
|
|
|
drawModeDark3,
|
|
|
|
drawModeBlack,
|
|
|
|
drawModeShadow1,
|
|
|
|
drawModeShadow2,
|
|
|
|
drawModeShadow3,
|
|
|
|
drawModeFoggy1,
|
|
|
|
drawModeFoggy2,
|
|
|
|
drawModeFoggy3,
|
|
|
|
drawModeFoggy4,
|
|
|
|
drawModeGlow1,
|
|
|
|
drawModeGlow2,
|
|
|
|
drawModeGlow3,
|
|
|
|
drawModeGlow4,
|
|
|
|
drawModeInvisible,
|
|
|
|
numDrawModes
|
|
|
|
};
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::setMyDrawMode(OnScreenPerson *moveMe, int h) {
|
2017-05-26 05:24:38 +02:00
|
|
|
switch (h) {
|
2017-06-05 11:49:19 +02:00
|
|
|
case drawModeTransparent3:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 0;
|
|
|
|
moveMe->transparency = 64;
|
|
|
|
break;
|
|
|
|
case drawModeTransparent2:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 0;
|
|
|
|
moveMe->transparency = 128;
|
|
|
|
break;
|
|
|
|
case drawModeTransparent1:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 0;
|
|
|
|
moveMe->transparency = 192;
|
|
|
|
break;
|
|
|
|
case drawModeInvisible:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 0;
|
|
|
|
moveMe->transparency = 254;
|
|
|
|
break;
|
|
|
|
case drawModeDark1:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 192;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeDark2:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 128;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeDark3:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 64;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeBlack:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 255;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeShadow1:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 255;
|
|
|
|
moveMe->transparency = 64;
|
|
|
|
break;
|
|
|
|
case drawModeShadow2:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 255;
|
|
|
|
moveMe->transparency = 128;
|
|
|
|
break;
|
|
|
|
case drawModeShadow3:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 255;
|
|
|
|
moveMe->transparency = 192;
|
|
|
|
break;
|
|
|
|
case drawModeFoggy3:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 128;
|
|
|
|
moveMe->colourmix = 192;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeFoggy2:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 128;
|
|
|
|
moveMe->colourmix = 128;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeFoggy1:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 128;
|
|
|
|
moveMe->colourmix = 64;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeFoggy4:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 128;
|
|
|
|
moveMe->colourmix = 255;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeGlow3:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 255;
|
|
|
|
moveMe->colourmix = 192;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeGlow2:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 255;
|
|
|
|
moveMe->colourmix = 128;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeGlow1:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 255;
|
|
|
|
moveMe->colourmix = 64;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
case drawModeGlow4:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 255;
|
|
|
|
moveMe->colourmix = 255;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
moveMe->r = moveMe->g = moveMe->b = 0;
|
|
|
|
moveMe->colourmix = 0;
|
|
|
|
moveMe->transparency = 0;
|
|
|
|
break;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::setDrawMode(int h, int ob) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(ob);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!moveMe)
|
|
|
|
return;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
setMyDrawMode(moveMe, h);
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::setPersonTransparency(int ob, byte x) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(ob);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!moveMe)
|
|
|
|
return;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
if (x > 254)
|
|
|
|
x = 254;
|
2017-05-26 05:24:38 +02:00
|
|
|
moveMe->transparency = x;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::setPersonColourise(int ob, byte r, byte g, byte b, byte colourmix) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(ob);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!moveMe)
|
|
|
|
return;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
moveMe->r = r;
|
|
|
|
moveMe->g = g;
|
|
|
|
moveMe->b = b;
|
|
|
|
moveMe->colourmix = colourmix;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::shufflePeople() {
|
|
|
|
OnScreenPerson **thisReference = &_allPeople;
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *A, *B;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
if (!_allPeople)
|
2017-05-29 08:02:59 +02:00
|
|
|
return;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
while ((*thisReference)->next) {
|
|
|
|
float y1 = (*thisReference)->y;
|
|
|
|
if ((*thisReference)->extra & EXTRA_FRONT)
|
|
|
|
y1 += 1000;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
float y2 = (*thisReference)->next->y;
|
|
|
|
if ((*thisReference)->next->extra & EXTRA_FRONT)
|
|
|
|
y2 += 1000;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
if (y1 > y2) {
|
2017-05-29 08:02:59 +02:00
|
|
|
A = (*thisReference);
|
|
|
|
B = (*thisReference)->next;
|
2017-05-27 20:16:54 +02:00
|
|
|
A->next = B->next;
|
|
|
|
B->next = A;
|
2017-05-29 08:02:59 +02:00
|
|
|
(*thisReference) = B;
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-29 08:02:59 +02:00
|
|
|
thisReference = &((*thisReference)->next);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::drawPeople() {
|
2017-05-26 05:24:38 +02:00
|
|
|
shufflePeople();
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
OnScreenPerson *thisPerson = _allPeople;
|
|
|
|
PersonaAnimation *myAnim = NULL;
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_regionMan->resetOverRegion();
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
while (thisPerson) {
|
2017-05-27 20:16:54 +02:00
|
|
|
if (thisPerson->show) {
|
|
|
|
myAnim = thisPerson->myAnim;
|
|
|
|
if (myAnim != thisPerson->lastUsedAnim) {
|
|
|
|
thisPerson->lastUsedAnim = myAnim;
|
|
|
|
thisPerson->frameNum = 0;
|
|
|
|
thisPerson->frameTick = myAnim->frames[0].howMany;
|
|
|
|
if (myAnim->frames[thisPerson->frameNum].noise > 0) {
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_soundMan->startSound(myAnim->frames[thisPerson->frameNum].noise, false);
|
2017-05-29 08:02:59 +02:00
|
|
|
thisPerson->frameNum++;
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->frameNum %= thisPerson->myAnim->numFrames;
|
2017-06-05 11:49:19 +02:00
|
|
|
thisPerson->frameTick = thisPerson->myAnim->frames[thisPerson->frameNum].howMany;
|
2017-05-27 20:16:54 +02:00
|
|
|
} else if (myAnim->frames[thisPerson->frameNum].noise) {
|
2017-06-05 11:49:19 +02:00
|
|
|
startNewFunctionNum(-myAnim->frames[thisPerson->frameNum].noise, 0,
|
|
|
|
NULL, noStack);
|
2017-05-29 08:02:59 +02:00
|
|
|
thisPerson->frameNum++;
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->frameNum %= thisPerson->myAnim->numFrames;
|
2017-06-05 11:49:19 +02:00
|
|
|
thisPerson->frameTick = thisPerson->myAnim->frames[thisPerson->frameNum].howMany;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
2017-05-27 20:16:54 +02:00
|
|
|
int fNumSign = myAnim->frames[thisPerson->frameNum].frameNum;
|
2017-05-26 05:24:38 +02:00
|
|
|
int m = fNumSign < 0;
|
2017-07-11 00:14:47 +02:00
|
|
|
int fNum = ABS(fNumSign);
|
2017-05-27 20:16:54 +02:00
|
|
|
if (fNum >= myAnim->theSprites->bank.total) {
|
2017-05-26 05:24:38 +02:00
|
|
|
fNum = 0;
|
|
|
|
m = 2 - m;
|
|
|
|
}
|
|
|
|
if (m != 2) {
|
|
|
|
bool r = false;
|
2018-04-15 22:09:37 +02:00
|
|
|
r = _vm->_gfxMan->scaleSprite(myAnim->theSprites->bank.sprites[fNum], myAnim->theSprites->bank.myPalette, thisPerson, m);
|
2017-05-26 05:24:38 +02:00
|
|
|
if (r) {
|
2017-07-11 14:57:31 +02:00
|
|
|
if (!thisPerson->thisType->screenName.empty()) {
|
2018-04-15 21:10:02 +02:00
|
|
|
if (_personRegion->thisType != thisPerson->thisType)
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_regionMan->resetLastRegion();
|
2018-04-15 21:10:02 +02:00
|
|
|
_personRegion->thisType = thisPerson->thisType;
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_regionMan->setOverRegion(_personRegion);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!--thisPerson->frameTick) {
|
|
|
|
thisPerson->frameNum++;
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->frameNum %= thisPerson->myAnim->numFrames;
|
2017-06-05 11:49:19 +02:00
|
|
|
thisPerson->frameTick = thisPerson->myAnim->frames[thisPerson->frameNum].howMany;
|
2017-05-27 20:16:54 +02:00
|
|
|
if (thisPerson->show && myAnim && myAnim->frames) {
|
|
|
|
if (myAnim->frames[thisPerson->frameNum].noise > 0) {
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_soundMan->startSound(myAnim->frames[thisPerson->frameNum].noise, false);
|
2017-05-29 08:02:59 +02:00
|
|
|
thisPerson->frameNum++;
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->frameNum %= thisPerson->myAnim->numFrames;
|
2017-06-05 11:49:19 +02:00
|
|
|
thisPerson->frameTick = thisPerson->myAnim->frames[thisPerson->frameNum].howMany;
|
2017-05-27 20:16:54 +02:00
|
|
|
} else if (myAnim->frames[thisPerson->frameNum].noise) {
|
2017-06-05 11:49:19 +02:00
|
|
|
startNewFunctionNum(-myAnim->frames[thisPerson->frameNum].noise, 0,
|
|
|
|
NULL, noStack);
|
2017-05-29 08:02:59 +02:00
|
|
|
thisPerson->frameNum++;
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->frameNum %= thisPerson->myAnim->numFrames;
|
2017-06-05 11:49:19 +02:00
|
|
|
thisPerson->frameTick = thisPerson->myAnim->frames[thisPerson->frameNum].howMany;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson = thisPerson->next;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::handleClosestPoint(int &setX, int &setY, int &setPoly) {
|
2017-06-05 11:49:19 +02:00
|
|
|
int gotX = 320, gotY = 200, gotPoly = -1, i, j, xTest1, yTest1, xTest2, yTest2, closestX, closestY, oldJ, currentDistance = 0xFFFFF, thisDistance;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
for (i = 0; i < currentFloor->numPolygons; i++) {
|
2017-05-27 20:16:54 +02:00
|
|
|
oldJ = currentFloor->polygon[i].numVertices - 1;
|
2017-05-29 08:02:59 +02:00
|
|
|
for (j = 0; j < currentFloor->polygon[i].numVertices; j++) {
|
2017-06-05 11:49:19 +02:00
|
|
|
xTest1 = currentFloor->vertex[currentFloor->polygon[i].vertexID[j]].x;
|
|
|
|
yTest1 = currentFloor->vertex[currentFloor->polygon[i].vertexID[j]].y;
|
|
|
|
xTest2 = currentFloor->vertex[currentFloor->polygon[i].vertexID[oldJ]].x;
|
|
|
|
yTest2 = currentFloor->vertex[currentFloor->polygon[i].vertexID[oldJ]].y;
|
|
|
|
closestPointOnLine(closestX, closestY, xTest1, yTest1, xTest2, yTest2, setX, setY);
|
2017-05-26 05:24:38 +02:00
|
|
|
xTest1 = setX - closestX;
|
|
|
|
yTest1 = setY - closestY;
|
|
|
|
thisDistance = xTest1 * xTest1 + yTest1 * yTest1;
|
|
|
|
|
|
|
|
if (thisDistance < currentDistance) {
|
|
|
|
currentDistance = thisDistance;
|
|
|
|
gotX = closestX;
|
|
|
|
gotY = closestY;
|
|
|
|
gotPoly = i;
|
|
|
|
}
|
|
|
|
oldJ = j;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
if (gotPoly == -1)
|
|
|
|
return false;
|
2017-05-26 05:24:38 +02:00
|
|
|
setX = gotX;
|
|
|
|
setY = gotY;
|
|
|
|
setPoly = gotPoly;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::doBorderStuff(OnScreenPerson *moveMe) {
|
2017-05-27 20:16:54 +02:00
|
|
|
if (moveMe->inPoly == moveMe->walkToPoly) {
|
|
|
|
moveMe->inPoly = -1;
|
|
|
|
moveMe->thisStepX = moveMe->walkToX;
|
|
|
|
moveMe->thisStepY = moveMe->walkToY;
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
|
|
|
// The section in which we need to be next...
|
2017-05-27 20:16:54 +02:00
|
|
|
int newPoly = currentFloor->matrix[moveMe->inPoly][moveMe->walkToPoly];
|
2017-05-29 08:02:59 +02:00
|
|
|
if (newPoly == -1)
|
|
|
|
return false;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// Grab the index of the second matching corner...
|
|
|
|
int ID, ID2;
|
2017-06-05 11:49:19 +02:00
|
|
|
if (!getMatchingCorners(currentFloor->polygon[moveMe->inPoly], currentFloor->polygon[newPoly], ID, ID2))
|
2017-05-26 05:24:38 +02:00
|
|
|
return fatal("Not a valid floor plan!");
|
|
|
|
|
|
|
|
// Remember that we're walking to the new polygon...
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->inPoly = newPoly;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// Calculate the destination position on the coincidantal line...
|
2017-05-27 20:16:54 +02:00
|
|
|
int x1 = moveMe->x, y1 = moveMe->y;
|
|
|
|
int x2 = moveMe->walkToX, y2 = moveMe->walkToY;
|
|
|
|
int x3 = currentFloor->vertex[ID].x, y3 = currentFloor->vertex[ID].y;
|
|
|
|
int x4 = currentFloor->vertex[ID2].x, y4 = currentFloor->vertex[ID2].y;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
int xAB = x1 - x2;
|
|
|
|
int yAB = y1 - y2;
|
|
|
|
int xCD = x4 - x3;
|
|
|
|
int yCD = y4 - y3;
|
|
|
|
|
|
|
|
double m = (yAB * (x3 - x1) - xAB * (y3 - y1));
|
|
|
|
m /= ((xAB * yCD) - (yAB * xCD));
|
|
|
|
|
|
|
|
if (m > 0 && m < 1) {
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->thisStepX = x3 + m * xCD;
|
|
|
|
moveMe->thisStepY = y3 + m * yCD;
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
|
|
|
int dx13 = x1 - x3, dx14 = x1 - x4, dx23 = x2 - x3, dx24 = x2 - x4;
|
|
|
|
int dy13 = y1 - y3, dy14 = y1 - y4, dy23 = y2 - y3, dy24 = y2 - y4;
|
|
|
|
|
|
|
|
dx13 *= dx13;
|
|
|
|
dx14 *= dx14;
|
|
|
|
dx23 *= dx23;
|
|
|
|
dx24 *= dx24;
|
|
|
|
dy13 *= dy13;
|
|
|
|
dy14 *= dy14;
|
|
|
|
dy23 *= dy23;
|
|
|
|
dy24 *= dy24;
|
|
|
|
|
2017-06-05 11:49:19 +02:00
|
|
|
if (sqrt((double)dx13 + dy13) + sqrt((double)dx23 + dy23) < sqrt((double)dx14 + dy14) + sqrt((double)dx24 + dy24)) {
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->thisStepX = x3;
|
|
|
|
moveMe->thisStepY = y3;
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->thisStepX = x4;
|
|
|
|
moveMe->thisStepY = y4;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-27 20:16:54 +02:00
|
|
|
float yDiff = moveMe->thisStepY - moveMe->y;
|
|
|
|
float xDiff = moveMe->x - moveMe->thisStepX;
|
2017-05-26 05:24:38 +02:00
|
|
|
if (xDiff || yDiff) {
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->wantAngle = 180 + ANGLEFIX * atan2(xDiff, yDiff * 2);
|
|
|
|
moveMe->spinning = true;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 22:09:37 +02:00
|
|
|
moveMe->setFrames(ANI_WALK);
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::walkMe(OnScreenPerson *thisPerson, bool move) {
|
2017-05-26 05:24:38 +02:00
|
|
|
float xDiff, yDiff, maxDiff, s;
|
|
|
|
|
|
|
|
for (;;) {
|
2017-05-27 20:16:54 +02:00
|
|
|
xDiff = thisPerson->thisStepX - thisPerson->x;
|
|
|
|
yDiff = (thisPerson->thisStepY - thisPerson->y) * 2;
|
|
|
|
s = thisPerson->scale * thisPerson->walkSpeed;
|
2017-05-29 08:02:59 +02:00
|
|
|
if (s < 0.2)
|
|
|
|
s = 0.2;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-07-11 00:14:47 +02:00
|
|
|
maxDiff = (ABS(xDiff) >= ABS(yDiff)) ? ABS(xDiff) : ABS(yDiff);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-07-11 00:14:47 +02:00
|
|
|
if (ABS(maxDiff) > s) {
|
2017-05-27 20:16:54 +02:00
|
|
|
if (thisPerson->spinning) {
|
2017-05-26 05:24:38 +02:00
|
|
|
spinStep(thisPerson);
|
2018-04-15 22:09:37 +02:00
|
|
|
thisPerson->setFrames(ANI_WALK);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
s = maxDiff / s;
|
|
|
|
if (move)
|
2017-06-05 11:49:19 +02:00
|
|
|
moveAndScale(*thisPerson, thisPerson->x + xDiff / s, thisPerson->y + yDiff / (s * 2));
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-05-27 20:16:54 +02:00
|
|
|
if (thisPerson->inPoly == -1) {
|
|
|
|
if (thisPerson->directionWhenDoneWalking != -1) {
|
|
|
|
thisPerson->wantAngle = thisPerson->directionWhenDoneWalking;
|
|
|
|
thisPerson->spinning = true;
|
2017-05-26 05:24:38 +02:00
|
|
|
spinStep(thisPerson);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!doBorderStuff(thisPerson))
|
|
|
|
break;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson->walking = false;
|
2018-04-15 22:09:37 +02:00
|
|
|
thisPerson->setFrames(ANI_STAND);
|
2017-05-29 08:02:59 +02:00
|
|
|
moveAndScale(*thisPerson, thisPerson->walkToX, thisPerson->walkToY);
|
2017-05-26 05:24:38 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::makeWalkingPerson(int x, int y, int objNum, LoadedFunction *func, int di) {
|
2017-05-29 08:02:59 +02:00
|
|
|
if (x == 0 && y == 0)
|
|
|
|
return false;
|
|
|
|
if (currentFloor->numPolygons == 0)
|
|
|
|
return false;
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(objNum);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!moveMe)
|
|
|
|
return false;
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
if (moveMe->continueAfterWalking)
|
|
|
|
abortFunction(moveMe->continueAfterWalking);
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->continueAfterWalking = NULL;
|
|
|
|
moveMe->walking = true;
|
|
|
|
moveMe->directionWhenDoneWalking = di;
|
|
|
|
|
|
|
|
moveMe->walkToX = x;
|
|
|
|
moveMe->walkToY = y;
|
|
|
|
moveMe->walkToPoly = inFloor(x, y);
|
|
|
|
if (moveMe->walkToPoly == -1) {
|
2017-06-05 11:49:19 +02:00
|
|
|
if (!handleClosestPoint(moveMe->walkToX, moveMe->walkToY, moveMe->walkToPoly))
|
2017-05-29 08:02:59 +02:00
|
|
|
return false;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->inPoly = inFloor(moveMe->x, moveMe->y);
|
|
|
|
if (moveMe->inPoly == -1) {
|
|
|
|
int xxx = moveMe->x, yyy = moveMe->y;
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!handleClosestPoint(xxx, yyy, moveMe->inPoly))
|
|
|
|
return false;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
doBorderStuff(moveMe);
|
2017-05-27 20:16:54 +02:00
|
|
|
if (walkMe(moveMe, false) || moveMe->spinning) {
|
|
|
|
moveMe->continueAfterWalking = func;
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::stopPerson(int o) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(o);
|
2017-05-26 05:24:38 +02:00
|
|
|
if (moveMe)
|
2017-05-27 20:16:54 +02:00
|
|
|
if (moveMe->continueAfterWalking) {
|
|
|
|
abortFunction(moveMe->continueAfterWalking);
|
|
|
|
moveMe->continueAfterWalking = NULL;
|
|
|
|
moveMe->walking = false;
|
|
|
|
moveMe->spinning = false;
|
2018-04-15 22:09:37 +02:00
|
|
|
moveMe->setFrames(ANI_STAND);
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::forceWalkingPerson(int x, int y, int objNum, LoadedFunction *func, int di) {
|
2017-05-29 08:02:59 +02:00
|
|
|
if (x == 0 && y == 0)
|
|
|
|
return false;
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(objNum);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!moveMe)
|
|
|
|
return false;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
if (moveMe->continueAfterWalking)
|
|
|
|
abortFunction(moveMe->continueAfterWalking);
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->walking = true;
|
|
|
|
moveMe->continueAfterWalking = NULL;
|
|
|
|
moveMe->directionWhenDoneWalking = di;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->walkToX = x;
|
|
|
|
moveMe->walkToY = y;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// Let's pretend the start and end points are both in the same
|
|
|
|
// polygon (which one isn't important)
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->inPoly = 0;
|
|
|
|
moveMe->walkToPoly = 0;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
doBorderStuff(moveMe);
|
2017-05-27 20:16:54 +02:00
|
|
|
if (walkMe(moveMe) || moveMe->spinning) {
|
|
|
|
moveMe->continueAfterWalking = func;
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::jumpPerson(int x, int y, int objNum) {
|
2017-05-29 08:02:59 +02:00
|
|
|
if (x == 0 && y == 0)
|
|
|
|
return;
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(objNum);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!moveMe)
|
|
|
|
return;
|
|
|
|
if (moveMe->continueAfterWalking)
|
|
|
|
abortFunction(moveMe->continueAfterWalking);
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->continueAfterWalking = NULL;
|
|
|
|
moveMe->walking = false;
|
|
|
|
moveMe->spinning = false;
|
2017-05-29 08:02:59 +02:00
|
|
|
moveAndScale(*moveMe, x, y);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::floatCharacter(int f, int objNum) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(objNum);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!moveMe)
|
|
|
|
return false;
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->floaty = f;
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::setCharacterWalkSpeed(int f, int objNum) {
|
2017-05-29 08:02:59 +02:00
|
|
|
if (f <= 0)
|
|
|
|
return false;
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(objNum);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!moveMe)
|
|
|
|
return false;
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->walkSpeed = f;
|
2017-05-26 05:24:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::walkAllPeople() {
|
|
|
|
OnScreenPerson *thisPerson = _allPeople;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
while (thisPerson) {
|
2017-05-27 20:16:54 +02:00
|
|
|
if (thisPerson->walking) {
|
2017-05-26 05:24:38 +02:00
|
|
|
walkMe(thisPerson);
|
2017-05-27 20:16:54 +02:00
|
|
|
} else if (thisPerson->spinning) {
|
2017-05-26 05:24:38 +02:00
|
|
|
spinStep(thisPerson);
|
2018-04-15 22:09:37 +02:00
|
|
|
thisPerson->setFrames(ANI_STAND);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
2017-06-05 11:49:19 +02:00
|
|
|
if ((!thisPerson->walking) && (!thisPerson->spinning) && thisPerson->continueAfterWalking) {
|
2017-05-27 20:16:54 +02:00
|
|
|
restartFunction(thisPerson->continueAfterWalking);
|
|
|
|
thisPerson->continueAfterWalking = NULL;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
2017-05-27 20:16:54 +02:00
|
|
|
thisPerson = thisPerson->next;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::addPerson(int x, int y, int objNum, Persona *p) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *newPerson = new OnScreenPerson;
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!checkNew(newPerson))
|
|
|
|
return false;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// EASY STUFF
|
2018-04-15 22:09:37 +02:00
|
|
|
newPerson->thisType = _vm->_objMan->loadObjectType(objNum);
|
2017-05-27 20:16:54 +02:00
|
|
|
newPerson->scale = 1;
|
|
|
|
newPerson->extra = 0;
|
|
|
|
newPerson->continueAfterWalking = NULL;
|
2017-05-29 08:02:59 +02:00
|
|
|
moveAndScale(*newPerson, x, y);
|
2017-05-27 20:16:54 +02:00
|
|
|
newPerson->frameNum = 0;
|
|
|
|
newPerson->walkToX = x;
|
|
|
|
newPerson->walkToY = y;
|
|
|
|
newPerson->walking = false;
|
|
|
|
newPerson->spinning = false;
|
|
|
|
newPerson->show = true;
|
|
|
|
newPerson->direction = 0;
|
|
|
|
newPerson->angle = 180;
|
|
|
|
newPerson->wantAngle = 180;
|
|
|
|
newPerson->angleOffset = 0;
|
|
|
|
newPerson->floaty = 0;
|
|
|
|
newPerson->walkSpeed = newPerson->thisType->walkSpeed;
|
|
|
|
newPerson->myAnim = NULL;
|
|
|
|
newPerson->spinSpeed = newPerson->thisType->spinSpeed;
|
|
|
|
newPerson->r = 0;
|
|
|
|
newPerson->g = 0;
|
|
|
|
newPerson->b = 0;
|
|
|
|
newPerson->colourmix = 0;
|
|
|
|
newPerson->transparency = 0;
|
|
|
|
newPerson->myPersona = p;
|
2017-07-20 01:55:21 +02:00
|
|
|
newPerson->lastUsedAnim = 0;
|
2017-08-12 11:28:21 +02:00
|
|
|
newPerson->frameTick = 0;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2018-04-15 22:09:37 +02:00
|
|
|
newPerson->setFrames(ANI_STAND);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// HEIGHT (BASED ON 1st FRAME OF 1st ANIMATION... INC. SPECIAL CASES)
|
2017-05-27 20:16:54 +02:00
|
|
|
int fNumSigned = p->animation[0]->frames[0].frameNum;
|
2017-05-26 05:24:38 +02:00
|
|
|
int fNum = abs(fNumSigned);
|
2017-05-27 20:16:54 +02:00
|
|
|
if (fNum >= p->animation[0]->theSprites->bank.total) {
|
2017-05-26 05:24:38 +02:00
|
|
|
if (fNumSigned < 0) {
|
2017-05-27 20:16:54 +02:00
|
|
|
newPerson->height = 5;
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-06-05 11:49:19 +02:00
|
|
|
newPerson->height = p->animation[0]->theSprites->bank.sprites[0].yhot + 5;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
} else {
|
2017-06-05 11:49:19 +02:00
|
|
|
newPerson->height = p->animation[0]->theSprites->bank.sprites[fNum].yhot + 5;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOW ADD IT IN THE RIGHT PLACE
|
2018-04-15 21:10:02 +02:00
|
|
|
OnScreenPerson **changethat = &_allPeople;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
while (((*changethat) != NULL) && ((*changethat)->y < y))
|
|
|
|
changethat = &((*changethat)->next);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
newPerson->next = (*changethat);
|
|
|
|
(*changethat) = newPerson;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-06-05 11:49:19 +02:00
|
|
|
return (bool)(newPerson->thisType != NULL);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::animatePerson(int obj, PersonaAnimation *fram) { // Set a new SINGLE animation
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(obj);
|
2017-05-26 05:24:38 +02:00
|
|
|
if (moveMe) {
|
2017-05-29 08:02:59 +02:00
|
|
|
if (moveMe->continueAfterWalking)
|
|
|
|
abortFunction(moveMe->continueAfterWalking);
|
2017-05-27 20:16:54 +02:00
|
|
|
moveMe->continueAfterWalking = NULL;
|
|
|
|
moveMe->walking = false;
|
|
|
|
moveMe->spinning = false;
|
|
|
|
moveMe->myAnim = fram;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::animatePerson(int obj, Persona *per) { // Set a new costume
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *moveMe = findPerson(obj);
|
2017-05-26 05:24:38 +02:00
|
|
|
if (moveMe) {
|
2017-05-27 20:16:54 +02:00
|
|
|
// if (moveMe->continueAfterWalking) abortFunction (moveMe->continueAfterWalking);
|
|
|
|
// moveMe->continueAfterWalking = NULL;
|
|
|
|
// moveMe->walking = false;
|
|
|
|
moveMe->spinning = false;
|
|
|
|
moveMe->myPersona = per;
|
2017-05-26 05:24:38 +02:00
|
|
|
rethinkAngle(moveMe);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (moveMe->walking) {
|
2018-04-15 22:09:37 +02:00
|
|
|
moveMe->setFrames(ANI_WALK);
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2018-04-15 22:09:37 +02:00
|
|
|
moveMe->setFrames(ANI_STAND);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::kill() {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *killPeople;
|
2018-04-15 21:10:02 +02:00
|
|
|
while (_allPeople) {
|
|
|
|
if (_allPeople->continueAfterWalking)
|
|
|
|
abortFunction(_allPeople->continueAfterWalking);
|
|
|
|
_allPeople->continueAfterWalking = NULL;
|
|
|
|
killPeople = _allPeople;
|
|
|
|
_allPeople = _allPeople->next;
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_objMan->removeObjectType(killPeople->thisType);
|
2017-05-26 05:24:38 +02:00
|
|
|
delete killPeople;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::killMostPeople() {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *killPeople;
|
2018-04-15 21:10:02 +02:00
|
|
|
OnScreenPerson **lookyHere = &_allPeople;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
while (*lookyHere) {
|
|
|
|
if ((*lookyHere)->extra & EXTRA_NOREMOVE) {
|
|
|
|
lookyHere = &(*lookyHere)->next;
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-29 08:02:59 +02:00
|
|
|
killPeople = (*lookyHere);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// Change last pointer to NEXT in the list instead
|
2017-05-29 08:02:59 +02:00
|
|
|
(*lookyHere) = killPeople->next;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// Gone from the list... now free some memory
|
2017-05-29 08:02:59 +02:00
|
|
|
if (killPeople->continueAfterWalking)
|
|
|
|
abortFunction(killPeople->continueAfterWalking);
|
2017-05-27 20:16:54 +02:00
|
|
|
killPeople->continueAfterWalking = NULL;
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_objMan->removeObjectType(killPeople->thisType);
|
2017-05-26 05:24:38 +02:00
|
|
|
delete killPeople;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::removeOneCharacter(int i) {
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *p = findPerson(i);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
if (p) {
|
2018-04-15 22:09:37 +02:00
|
|
|
ScreenRegion *overRegion = _vm->_regionMan->getOverRegion();
|
2018-04-15 21:10:02 +02:00
|
|
|
if (overRegion == _personRegion && overRegion->thisType == p->thisType) {
|
2018-04-15 00:21:17 +02:00
|
|
|
overRegion = nullptr;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
if (p->continueAfterWalking)
|
|
|
|
abortFunction(p->continueAfterWalking);
|
2017-05-27 20:16:54 +02:00
|
|
|
p->continueAfterWalking = NULL;
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson **killPeople;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
for (killPeople = &_allPeople; *killPeople != p; killPeople = &((*killPeople)->next)) {
|
2017-05-26 05:24:38 +02:00
|
|
|
;
|
|
|
|
}
|
|
|
|
|
2017-05-29 08:02:59 +02:00
|
|
|
*killPeople = p->next;
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_objMan->removeObjectType(p->thisType);
|
2017-05-26 05:24:38 +02:00
|
|
|
delete p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::savePeople(Common::WriteStream *stream) {
|
|
|
|
OnScreenPerson *me = _allPeople;
|
2017-05-26 05:24:38 +02:00
|
|
|
int countPeople = 0, a;
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
stream->writeSint16LE(_scaleHorizon);
|
|
|
|
stream->writeSint16LE(_scaleDivide);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
while (me) {
|
2017-05-29 08:02:59 +02:00
|
|
|
countPeople++;
|
2017-05-27 20:16:54 +02:00
|
|
|
me = me->next;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->writeUint16BE(countPeople);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
me = _allPeople;
|
2017-06-05 12:03:50 +02:00
|
|
|
for (a = 0; a < countPeople; a++) {
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-07-11 00:02:46 +02:00
|
|
|
stream->writeFloatLE(me->x);
|
|
|
|
stream->writeFloatLE(me->y);
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2018-04-15 21:56:20 +02:00
|
|
|
me->myPersona->save(stream);
|
2018-04-15 21:35:19 +02:00
|
|
|
me->myAnim->save(stream);
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->writeByte(me->myAnim == me->lastUsedAnim);
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-07-11 00:02:46 +02:00
|
|
|
stream->writeFloatLE(me->scale);
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->writeUint16BE(me->extra);
|
|
|
|
stream->writeUint16BE(me->height);
|
|
|
|
stream->writeUint16BE(me->walkToX);
|
|
|
|
stream->writeUint16BE(me->walkToY);
|
|
|
|
stream->writeUint16BE(me->thisStepX);
|
|
|
|
stream->writeUint16BE(me->thisStepY);
|
|
|
|
stream->writeUint16BE(me->frameNum);
|
|
|
|
stream->writeUint16BE(me->frameTick);
|
|
|
|
stream->writeUint16BE(me->walkSpeed);
|
|
|
|
stream->writeUint16BE(me->spinSpeed);
|
2017-07-10 23:53:43 +02:00
|
|
|
stream->writeSint16LE(me->floaty);
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->writeByte(me->show);
|
|
|
|
stream->writeByte(me->walking);
|
|
|
|
stream->writeByte(me->spinning);
|
2017-05-27 20:16:54 +02:00
|
|
|
if (me->continueAfterWalking) {
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->writeByte(1);
|
2017-05-27 20:16:54 +02:00
|
|
|
saveFunction(me->continueAfterWalking, stream);
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->writeByte(0);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->writeUint16BE(me->direction);
|
|
|
|
stream->writeUint16BE(me->angle);
|
|
|
|
stream->writeUint16BE(me->angleOffset);
|
|
|
|
stream->writeUint16BE(me->wantAngle);
|
2017-07-10 23:53:43 +02:00
|
|
|
stream->writeSint16LE(me->directionWhenDoneWalking);
|
|
|
|
stream->writeSint16LE(me->inPoly);
|
|
|
|
stream->writeSint16LE(me->walkToPoly);
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->writeByte(me->r);
|
|
|
|
stream->writeByte(me->g);
|
|
|
|
stream->writeByte(me->b);
|
|
|
|
stream->writeByte(me->colourmix);
|
|
|
|
stream->writeByte(me->transparency);
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2018-04-15 22:09:37 +02:00
|
|
|
_vm->_objMan->saveObjectRef(me->thisType, stream);
|
2017-05-27 20:16:54 +02:00
|
|
|
|
|
|
|
me = me->next;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
bool PeopleManager::loadPeople(Common::SeekableReadStream *stream) {
|
|
|
|
kill();
|
|
|
|
|
|
|
|
OnScreenPerson **pointy = &_allPeople;
|
2017-07-20 10:39:24 +02:00
|
|
|
OnScreenPerson *me;
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
_scaleHorizon = stream->readSint16LE();
|
|
|
|
_scaleDivide = stream->readSint16LE();
|
2017-05-26 05:24:38 +02:00
|
|
|
|
2017-05-30 09:59:56 +02:00
|
|
|
int countPeople = stream->readUint16BE();
|
2017-05-26 05:24:38 +02:00
|
|
|
int a;
|
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
_allPeople = NULL;
|
2017-06-05 12:03:50 +02:00
|
|
|
for (a = 0; a < countPeople; a++) {
|
2017-07-20 10:39:24 +02:00
|
|
|
me = new OnScreenPerson;
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!checkNew(me))
|
|
|
|
return false;
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-07-20 10:39:24 +02:00
|
|
|
me->myPersona = new Persona;
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!checkNew(me->myPersona))
|
|
|
|
return false;
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2018-04-19 12:08:31 +02:00
|
|
|
me->myAnim = new PersonaAnimation;
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!checkNew(me->myAnim))
|
|
|
|
return false;
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-07-11 00:02:46 +02:00
|
|
|
me->x = stream->readFloatLE();
|
|
|
|
me->y = stream->readFloatLE();
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2018-04-15 21:56:20 +02:00
|
|
|
me->myPersona->load(stream);
|
2018-04-15 21:35:19 +02:00
|
|
|
me->myAnim->load(stream);
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-05-30 09:59:56 +02:00
|
|
|
me->lastUsedAnim = stream->readByte() ? me->myAnim : NULL;
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-07-11 00:02:46 +02:00
|
|
|
me->scale = stream->readFloatLE();
|
2017-05-27 20:16:54 +02:00
|
|
|
|
2017-05-30 09:59:56 +02:00
|
|
|
me->extra = stream->readUint16BE();
|
|
|
|
me->height = stream->readUint16BE();
|
|
|
|
me->walkToX = stream->readUint16BE();
|
|
|
|
me->walkToY = stream->readUint16BE();
|
|
|
|
me->thisStepX = stream->readUint16BE();
|
|
|
|
me->thisStepY = stream->readUint16BE();
|
|
|
|
me->frameNum = stream->readUint16BE();
|
|
|
|
me->frameTick = stream->readUint16BE();
|
|
|
|
me->walkSpeed = stream->readUint16BE();
|
|
|
|
me->spinSpeed = stream->readUint16BE();
|
2017-07-10 23:53:43 +02:00
|
|
|
me->floaty = stream->readSint16LE();
|
2017-05-30 09:59:56 +02:00
|
|
|
me->show = stream->readByte();
|
|
|
|
me->walking = stream->readByte();
|
|
|
|
me->spinning = stream->readByte();
|
|
|
|
if (stream->readByte()) {
|
2017-05-27 20:16:54 +02:00
|
|
|
me->continueAfterWalking = loadFunction(stream);
|
2017-05-29 08:02:59 +02:00
|
|
|
if (!me->continueAfterWalking)
|
|
|
|
return false;
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-27 20:16:54 +02:00
|
|
|
me->continueAfterWalking = NULL;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
2017-05-30 09:59:56 +02:00
|
|
|
me->direction = stream->readUint16BE();
|
|
|
|
me->angle = stream->readUint16BE();
|
2017-05-26 05:24:38 +02:00
|
|
|
if (ssgVersion >= VERSION(2, 0)) {
|
2017-05-30 09:59:56 +02:00
|
|
|
me->angleOffset = stream->readUint16BE();
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-27 20:16:54 +02:00
|
|
|
me->angleOffset = 0;
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
2017-05-30 09:59:56 +02:00
|
|
|
me->wantAngle = stream->readUint16BE();
|
2017-07-10 23:53:43 +02:00
|
|
|
me->directionWhenDoneWalking = stream->readSint16LE();
|
|
|
|
me->inPoly = stream->readSint16LE();
|
|
|
|
me->walkToPoly = stream->readSint16LE();
|
2017-05-26 05:24:38 +02:00
|
|
|
if (ssgVersion >= VERSION(2, 0)) {
|
2017-05-30 09:59:56 +02:00
|
|
|
me->r = stream->readByte();
|
|
|
|
me->g = stream->readByte();
|
|
|
|
me->b = stream->readByte();
|
|
|
|
me->colourmix = stream->readByte();
|
|
|
|
me->transparency = stream->readByte();
|
2017-05-26 05:24:38 +02:00
|
|
|
} else {
|
2017-05-30 09:59:56 +02:00
|
|
|
setMyDrawMode(me, stream->readUint16BE());
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
2018-04-15 22:09:37 +02:00
|
|
|
me->thisType = _vm->_objMan->loadObjectRef(stream);
|
2017-05-26 05:24:38 +02:00
|
|
|
|
|
|
|
// Anti-aliasing settings
|
|
|
|
if (ssgVersion >= VERSION(1, 6)) {
|
|
|
|
if (ssgVersion < VERSION(2, 0)) {
|
|
|
|
// aaLoad
|
2017-05-30 09:59:56 +02:00
|
|
|
stream->readByte();
|
2017-07-11 00:02:46 +02:00
|
|
|
stream->readFloatLE();
|
|
|
|
stream->readFloatLE();
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-27 20:16:54 +02:00
|
|
|
me->next = NULL;
|
2017-05-29 08:02:59 +02:00
|
|
|
*pointy = me;
|
|
|
|
pointy = &(me->next);
|
2017-05-26 05:24:38 +02:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2017-05-26 21:25:11 +02:00
|
|
|
|
2018-04-15 21:10:02 +02:00
|
|
|
void PeopleManager::freeze(FrozenStuffStruct *frozenStuff) {
|
|
|
|
frozenStuff->allPeople = _allPeople;
|
|
|
|
_allPeople = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PeopleManager::resotre(FrozenStuffStruct *frozenStuff) {
|
|
|
|
kill();
|
|
|
|
_allPeople = frozenStuff->allPeople;
|
|
|
|
}
|
|
|
|
|
2017-05-26 21:25:11 +02:00
|
|
|
} // End of namespace Sludge
|