mirror of
https://github.com/zenorogue/hyperrogue.git
synced 2025-01-25 00:16:59 +00:00
932 lines
30 KiB
C++
932 lines
30 KiB
C++
// Hyperbolic Rogue - environment
|
|
// Copyright (C) 2011-2019 Zeno Rogue, see 'hyper.cpp' for details
|
|
|
|
/** \file environment.cpp
|
|
* \brief game environment: routines related to the game that affect all the map. Monsters to move are detected here, but their moves are implemented in monstermove.cpp
|
|
*/
|
|
|
|
#include "hyper.h"
|
|
|
|
namespace hr {
|
|
|
|
#if HDR
|
|
#define HF_BUG Flag(0)
|
|
#define HF_EARTH Flag(1)
|
|
#define HF_BIRD Flag(2)
|
|
#define HF_LEADER Flag(3)
|
|
#define HF_HEX Flag(4)
|
|
#define HF_WHIRLPOOL Flag(5)
|
|
#define HF_WATER Flag(6)
|
|
#define HF_AIR Flag(7)
|
|
#define HF_MUTANT Flag(8)
|
|
#define HF_OUTLAW Flag(9)
|
|
#define HF_WHIRLWIND Flag(10)
|
|
#define HF_ROSE Flag(11)
|
|
#define HF_DRAGON Flag(12)
|
|
#define HF_KRAKEN Flag(13)
|
|
#define HF_SHARK Flag(14)
|
|
#define HF_BATS Flag(15)
|
|
#define HF_REPTILE Flag(16)
|
|
#define HF_EAGLES Flag(17)
|
|
#define HF_SLOW Flag(18)
|
|
#define HF_FAST Flag(19)
|
|
#define HF_WARP Flag(20)
|
|
#define HF_MOUSE Flag(21)
|
|
#define HF_RIVER Flag(22)
|
|
#define HF_MIRROR Flag(23)
|
|
#define HF_VOID Flag(24)
|
|
#define HF_HUNTER Flag(25)
|
|
#define HF_FAILED_AMBUSH Flag(26)
|
|
#define HF_MAGNET Flag(27)
|
|
#define HF_HEXD Flag(28)
|
|
#define HF_ALT Flag(29)
|
|
#define HF_MONK Flag(30)
|
|
#define HF_WESTWALL Flag(31)
|
|
#define HF_JUMP Flag(32)
|
|
#define HF_DICE Flag(33)
|
|
#endif
|
|
|
|
EX flagtype havewhat, hadwhat;
|
|
|
|
/** monsters of specific types to move */
|
|
EX vector<cell*> worms, ivies, ghosts, golems, hexsnakes;
|
|
|
|
/** temporary changes during bfs */
|
|
vector<pair<cell*, eMonster>> tempmonsters;
|
|
|
|
/** The position of the first cell in dcal in distance 7. New wandering monsters can be generated in dcal[first7..]. */
|
|
EX int first7;
|
|
|
|
/** the list of all nearby cells, according to cpdist */
|
|
EX vector<cell*> dcal;
|
|
|
|
/** the list of all nearby cells, according to current pathdist */
|
|
EX vector<cellwalker> pathq;
|
|
|
|
/** the number of big statues -- they increase monster generation */
|
|
EX int statuecount;
|
|
|
|
/** the number of slimes in Wetland -- they create ghosts */
|
|
EX int wetslime;
|
|
|
|
/** list of monsters to move (pathq restriced to monsters) */
|
|
EX vector<cell*> pathqm;
|
|
|
|
/** which hex snakes are there */
|
|
EX set<int> snaketypes;
|
|
|
|
EX int gamerange_bonus = 0;
|
|
EX int gamerange() { return getDistLimit() + gamerange_bonus; }
|
|
|
|
// pathdist begin
|
|
EX cell *pd_from;
|
|
EX int pd_range;
|
|
|
|
#if HDR
|
|
/** The pathdata is used to keep a list of visited cells. It is used as follows:
|
|
* 1) create pathdata object: pathdata pd(identifier)
|
|
* 2) use one of the following methods to mark cells as visited:
|
|
* 2a) onpath_with_dir or onpath_random_dir, to mark a cell together with its distance and the direction we came from (used by computePathdist to make pathfinding not sensitive to direction indexing)
|
|
* 2b) onpath, to mark a cell at its distance (used when ordering is irrelevant: compute_graphical_distance and in shmup)
|
|
* 2c) onpatk_mark, to just mark a cell (used in groupmove2)
|
|
* 3) All the visited cells are listed in pathq, and they have 'pathdist' set to their recorded distance (0 in case of onpath_mark).
|
|
* 4) When the pathdata object is deleted, all the pathdist values are cleared back to PINFD.
|
|
* The variable 'pathlock' ensures that we do not use two pathdata objects at once.
|
|
**/
|
|
|
|
struct pathdata {
|
|
void checklock();
|
|
~pathdata();
|
|
pathdata(eMonster m, bool include_allies IS(true));
|
|
pathdata(int i);
|
|
};
|
|
#endif
|
|
|
|
/** using pathdata, record a cell (together with direction) as visited */
|
|
EX void onpath_with_dir(cellwalker cw, int d) {
|
|
if(!pathlock) {
|
|
println(hlog, "onpath(", cw, ", ", d, ") without pathlock");
|
|
}
|
|
cw.at->pathdist = d;
|
|
pathq.push_back(cw);
|
|
}
|
|
|
|
/** using pathdata, record a cell as visited, with random direction */
|
|
EX void onpath_random_dir(cell *c, int d) {
|
|
onpath_with_dir(cellwalker(c, hrand(c->type), hrand(2)), d);
|
|
}
|
|
|
|
EX void onpath(cell *c, int d) {
|
|
onpath_with_dir(cellwalker(c, 0, 0), d);
|
|
}
|
|
|
|
EX void onpath_mark(cell *c) {
|
|
onpath_with_dir(cellwalker(c, 0, 0), 0);
|
|
}
|
|
|
|
EX void clear_pathdata() {
|
|
for(auto c: pathq) c.at->pathdist = PINFD;
|
|
pathq.clear();
|
|
pathqm.clear();
|
|
}
|
|
|
|
/** This ensures that we do not use two pathdata objects at once */
|
|
EX int pathlock = 0;
|
|
|
|
/** compute_graphical_distance determines the distance of every cell
|
|
* from the current FOV center. It uses the pathq structures but
|
|
* does not lock them */
|
|
|
|
EX void compute_graphical_distance() {
|
|
if(pathlock) { printf("path error: compute_graphical_distance\n"); }
|
|
cell *c1 = centerover ? centerover : pd_from ? pd_from : cwt.at;
|
|
int sr = get_sightrange_ambush();
|
|
if(pd_from == c1 && pd_range == sr) return;
|
|
clear_pathdata();
|
|
|
|
pathlock++;
|
|
pd_from = c1;
|
|
pd_range = sr;
|
|
onpath(c1, 0);
|
|
|
|
for(int qb=0; qb<isize(pathq); qb++) {
|
|
cell *c = pathq[qb].at;
|
|
if(c->pathdist == pd_range) break;
|
|
if(qb == 0) forCellCM(c1, c) ;
|
|
forCellEx(c1, c)
|
|
if(c1->pathdist == PINFD)
|
|
onpath(c1, c->pathdist + 1);
|
|
}
|
|
|
|
pathlock--;
|
|
}
|
|
|
|
const int max_radius = 16;
|
|
|
|
struct visit_set {
|
|
set<cell*> visited;
|
|
queue<cell*> q;
|
|
void visit(cell *c) {
|
|
if(visited.count(c)) return;
|
|
visited.insert(c);
|
|
q.push(c);
|
|
}
|
|
};
|
|
|
|
struct princess_ai {
|
|
array<visit_set, max_radius+1> info;
|
|
void visit_gate(cell *g) { info[0].visit(g); }
|
|
void run();
|
|
};
|
|
|
|
void princess_ai::run() {
|
|
int radius = toggle_radius(waOpenPlate);
|
|
if(pathq.empty()) return;
|
|
int d = pathq.back().at->pathdist;
|
|
if(d == PINFD - 1) return;
|
|
d++;
|
|
if(d < 5) d = 5; /* the Princess AI avoids plates when too close to the player */
|
|
hassert(radius <= max_radius);
|
|
|
|
for(int k=0; k<=radius; k++) while(!info[k].q.empty()) {
|
|
cell *c = info[k].q.front();
|
|
info[k].q.pop();
|
|
if(k < radius) forCellEx(c1, c) {
|
|
info[k+1].visit(c1);
|
|
if(k == 0 && c1->wall == waClosedGate)
|
|
info[0].visit(c1);
|
|
}
|
|
if(k == radius && c->wall == waOpenPlate && c->pathdist == PINFD)
|
|
onpath_random_dir(c, d);
|
|
}
|
|
}
|
|
|
|
EX void computePathdist(eMonster param, bool include_allies IS(true)) {
|
|
|
|
for(cell *c: targets)
|
|
if(include_allies || isPlayerOn(c))
|
|
onpath_random_dir(c, isPlayerOn(c) ? 0 : 1);
|
|
|
|
int qtarg = isize(targets);
|
|
|
|
int limit = gamerange();
|
|
|
|
int qb = 0;
|
|
|
|
bool princess = isPrincess(param);
|
|
princess_ai gd;
|
|
princess_retry:
|
|
|
|
for(; qb < isize(pathq); qb++) {
|
|
cellwalker cw = pathq[qb];
|
|
/* The opposite cell will be added to the queue first, which helps the AI. */
|
|
cw += cw.at->type/2;
|
|
cell*& c = cw.at;
|
|
if(c->monst && !isBug(c) && !(isFriendly(c) && !c->stuntime)) {
|
|
pathqm.push_back(c);
|
|
continue; // no paths going through monsters
|
|
}
|
|
if(isMounted(c) && !isPlayerOn(c)) {
|
|
// don't treat the Worm you are riding as passable
|
|
pathqm.push_back(c);
|
|
continue;
|
|
}
|
|
if(c->cpdist > limit && !(c->land == laTrollheim && turncount < c->landparam) && c->wall != waThumperOn) continue;
|
|
int d = c->pathdist;
|
|
if(d == PINFD - 1) continue;
|
|
for(int j=0; j<c->type; j++) {
|
|
cellwalker cw1 = cw + j;
|
|
// printf("i=%d cd=%d\n", i, c->move(i)->cpdist);
|
|
cell *c2 = cw1.peek();
|
|
|
|
flagtype f = P_MONSTER | P_REVDIR;
|
|
if(param == moTameBomberbird) f |= P_FLYING;
|
|
|
|
if(c2 && c2->pathdist == PINFD &&
|
|
passable(c2, (qb<qtarg) && !nonAdjacent(c,c2) && !thruVine(c,c2) ?NULL:c, f)) {
|
|
|
|
if(qb >= qtarg) {
|
|
if(param == moTortoise && nogoSlow(c, c2)) continue;
|
|
if(param == moIvyRoot && strictlyAgainstGravity(c, c2, false, MF_IVY)) continue;
|
|
if(param == moWorm && (cellUnstable(c) || cellEdgeUnstable(c) || prairie::no_worms(c))) continue;
|
|
if(!isFriendly(param) && items[itOrbLava] && c2->cpdist <= 5 && pseudohept(c) && makeflame(c2, 1, true))
|
|
continue;
|
|
}
|
|
|
|
onpath_with_dir(cw1 + wstep, d+1);
|
|
}
|
|
|
|
else if(c2 && c2->wall == waClosedGate && princess)
|
|
gd.visit_gate(c2);
|
|
}
|
|
}
|
|
|
|
if(princess) {
|
|
gd.run();
|
|
if(qb < isize(pathq)) goto princess_retry;
|
|
}
|
|
}
|
|
|
|
pathdata::~pathdata() {
|
|
pathlock--;
|
|
clear_pathdata();
|
|
}
|
|
|
|
void pathdata::checklock() {
|
|
if(pd_from) pd_from = NULL, clear_pathdata();
|
|
if(pathlock) printf("path error\n");
|
|
pathlock++;
|
|
}
|
|
|
|
pathdata::pathdata(int i) { checklock(); }
|
|
|
|
pathdata::pathdata(eMonster m, bool include_allies IS(true)) {
|
|
checklock();
|
|
if(isize(pathq))
|
|
println(hlog, "! we got tiles on pathq: ", isize(pathq));
|
|
|
|
computePathdist(m, include_allies);
|
|
}
|
|
|
|
// pathdist end
|
|
|
|
/** additional direction information for BFS algorithms.
|
|
* It remembers from where we have got to this location
|
|
* the opposite cell will be added to the queue first,
|
|
* which helps the AI. Used in bfs().
|
|
**/
|
|
EX vector<int> bfs_reachedfrom;
|
|
|
|
/** calculate cpdist, 'have' flags, and do general fixings */
|
|
EX void bfs() {
|
|
|
|
calcTidalPhase();
|
|
|
|
yendor::onpath();
|
|
|
|
int dcs = isize(dcal);
|
|
for(int i=0; i<dcs; i++) dcal[i]->cpdist = INFD;
|
|
worms.clear(); ivies.clear(); ghosts.clear(); golems.clear();
|
|
tempmonsters.clear(); targets.clear();
|
|
statuecount = 0;
|
|
wetslime = 0;
|
|
hexsnakes.clear();
|
|
|
|
hadwhat = havewhat;
|
|
havewhat = 0; jiangshi_on_screen = 0;
|
|
snaketypes.clear();
|
|
if(!(hadwhat & HF_WARP)) { avengers = 0; }
|
|
if(!(hadwhat & HF_MIRROR)) { mirrorspirits = 0; }
|
|
|
|
elec::havecharge = false;
|
|
elec::afterOrb = false;
|
|
elec::haveelec = false;
|
|
airmap.clear();
|
|
if(!(hadwhat & HF_ROSE)) rosemap.clear();
|
|
|
|
dcal.clear(); bfs_reachedfrom.clear();
|
|
|
|
recalcTide = false;
|
|
|
|
for(cell *c: player_positions()) {
|
|
if(c->cpdist == 0) continue;
|
|
c->cpdist = 0;
|
|
checkTide(c);
|
|
dcal.push_back(c);
|
|
bfs_reachedfrom.push_back(hrand(c->type));
|
|
if(!invismove) targets.push_back(c);
|
|
}
|
|
|
|
int distlimit = gamerange();
|
|
|
|
for(cell *c: player_positions()) {
|
|
if(items[itOrbDomination])
|
|
if(c->monst == moTentacle || c->monst == moTentaclewait || c->monst == moTentacleEscaping)
|
|
worms.push_back(c);
|
|
}
|
|
|
|
int qb = 0;
|
|
first7 = 0;
|
|
while(true) {
|
|
if(qb == isize(dcal)) break;
|
|
int i, fd = bfs_reachedfrom[qb] + dcal[qb]->type/2;
|
|
cell *c = dcal[qb++];
|
|
|
|
int d = c->cpdist;
|
|
|
|
if(WDIM == 2 && d == distlimit) { first7 = qb; break; }
|
|
|
|
for(int j=0; j<c->type; j++) if(i = (fd+j) % c->type, c->move(i)) {
|
|
// printf("i=%d cd=%d\n", i, c->move(i)->cpdist);
|
|
cell *c2 = c->move(i);
|
|
if(!c2) continue;
|
|
|
|
if(isWarpedType(c2->land)) havewhat |= HF_WARP;
|
|
if(c2->land == laMirror) havewhat |= HF_MIRROR;
|
|
|
|
if((c->wall == waBoat || c->wall == waSea) &&
|
|
(c2->wall == waSulphur || c2->wall == waSulphurC))
|
|
c2->wall = waSea;
|
|
|
|
if(c2 && signed(c2->cpdist) > d+1) {
|
|
if(WDIM == 3 && !gmatrix.count(c2)) {
|
|
if(!first7) first7 = qb;
|
|
continue;
|
|
}
|
|
c2->cpdist = d+1;
|
|
|
|
// remove treasures
|
|
if(!peace::on && c2->item && c2->cpdist == distlimit && itemclass(c2->item) == IC_TREASURE &&
|
|
c2->item != itBabyTortoise && WDIM != 3 &&
|
|
(items[c2->item] >= (ls::any_chaos()?10:20) + currentLocalTreasure || getGhostcount() >= 2)) {
|
|
c2->item = itNone;
|
|
if(c2->land == laMinefield) { c2->landparam &= ~3; }
|
|
}
|
|
|
|
if(c2->item == itBombEgg && c2->cpdist == distlimit && items[itBombEgg] >= c2->landparam) {
|
|
c2->item = itNone;
|
|
c2->landparam |= 2;
|
|
c2->landparam &= ~1;
|
|
if(!c2->monst) c2->monst = moBomberbird, c2->stuntime = 0;
|
|
}
|
|
|
|
if(c2->item == itBarrow && c2->cpdist == distlimit && c2->wall != waBarrowDig) {
|
|
c2->item = itNone;
|
|
}
|
|
|
|
if(c2->item == itLotus && c2->cpdist == distlimit && items[itLotus] >= getHauntedDepth(c2)) {
|
|
c2->item = itNone;
|
|
}
|
|
|
|
if(c2->item == itMutant2 && timerghost) {
|
|
bool rotten = true;
|
|
for(int i=0; i<c2->type; i++)
|
|
if(c2->move(i) && c2->move(i)->monst == moMutant)
|
|
rotten = false;
|
|
if(rotten) c2->item = itNone;
|
|
}
|
|
|
|
if(c2->item == itDragon && (shmup::on ? shmup::curtime-c2->landparam>300000 :
|
|
turncount-c2->landparam > 500))
|
|
c2->item = itNone;
|
|
|
|
if(c2->item == itTrollEgg && c2->cpdist == distlimit && !shmup::on && c2->landparam && turncount-c2->landparam > 650)
|
|
c2->item = itNone;
|
|
|
|
if(c2->item == itWest && c2->cpdist == distlimit && items[itWest] >= c2->landparam + 4)
|
|
c2->item = itNone;
|
|
|
|
if(c2->item == itMutant && c2->cpdist == distlimit && items[itMutant] >= c2->landparam) {
|
|
c2->item = itNone;
|
|
}
|
|
|
|
if(c2->item == itIvory && c2->cpdist == distlimit && items[itIvory] >= c2->landparam) {
|
|
c2->item = itNone;
|
|
}
|
|
|
|
if(c2->item == itAmethyst && c2->cpdist == distlimit && items[itAmethyst] >= -celldistAlt(c2)/5) {
|
|
c2->item = itNone;
|
|
}
|
|
|
|
if(!keepLightning) c2->ligon = 0;
|
|
dcal.push_back(c2);
|
|
bfs_reachedfrom.push_back(c->c.spin(i));
|
|
|
|
checkTide(c2);
|
|
|
|
if(c2->wall == waBigStatue && c2->land != laTemple)
|
|
statuecount++;
|
|
|
|
if(isAlch(c2->wall) && c2->land == laWet)
|
|
wetslime++;
|
|
|
|
if(cellHalfvine(c2) && isWarped(c2)) {
|
|
addMessage(XLAT("%The1 is destroyed!", c2->wall));
|
|
destroyHalfvine(c2);
|
|
}
|
|
|
|
if(c2->wall == waCharged) elec::havecharge = true;
|
|
if(isElectricLand(c2)) elec::haveelec = true;
|
|
|
|
if(c2->land == laWhirlpool) havewhat |= HF_WHIRLPOOL;
|
|
if(c2->land == laWhirlwind) havewhat |= HF_WHIRLWIND;
|
|
if(c2->land == laWestWall) havewhat |= HF_WESTWALL;
|
|
if(c2->land == laPrairie) havewhat |= HF_RIVER;
|
|
if(c2->land == laClearing) havewhat |= HF_MUTANT;
|
|
|
|
if(c2->wall == waRose) havewhat |= HF_ROSE;
|
|
|
|
if((hadwhat & HF_ROSE) && (rosemap[c2] & 3)) havewhat |= HF_ROSE;
|
|
|
|
if(c2->monst) {
|
|
if(isHaunted(c2->land) &&
|
|
c2->monst != moGhost && c2->monst != moZombie && c2->monst != moNecromancer)
|
|
fail_survivalist();
|
|
if(c2->monst == moHexSnake || c2->monst == moHexSnakeTail) {
|
|
havewhat |= HF_HEX;
|
|
if(c2->mondir != NODIR)
|
|
snaketypes.insert(snake_pair(c2));
|
|
if(c2->monst == moHexSnake) hexsnakes.push_back(c2);
|
|
else findWormIvy(c2);
|
|
}
|
|
else if(c2->monst == moKrakenT || c2->monst == moKrakenH) {
|
|
havewhat |= HF_KRAKEN;
|
|
}
|
|
else if(c2->monst == moDragonHead || c2->monst == moDragonTail) {
|
|
havewhat |= HF_DRAGON;
|
|
}
|
|
else if(c2->monst == moWitchSpeed)
|
|
havewhat |= HF_FAST;
|
|
else if(c2->monst == moMutant)
|
|
havewhat |= HF_MUTANT;
|
|
else if(c2->monst == moJiangshi)
|
|
jiangshi_on_screen++;
|
|
else if(c2->monst == moOutlaw)
|
|
havewhat |= HF_OUTLAW;
|
|
else if(isGhostMover(c2->monst))
|
|
ghosts.push_back(c2);
|
|
else if(isWorm(c2) || isIvy(c2)) findWormIvy(c2);
|
|
else if(isBug(c2)) {
|
|
havewhat |= HF_BUG;
|
|
targets.push_back(c2);
|
|
}
|
|
else if(isFriendly(c2)) {
|
|
if(c2->monst != moMouse && !markEmpathy(itOrbInvis) && !(isWatery(c2) && markEmpathy(itOrbFish)) &&
|
|
!c2->stuntime) targets.push_back(c2);
|
|
if(c2->monst == moGolem) golems.push_back(c2);
|
|
if(c2->monst == moFriendlyGhost) golems.push_back(c2);
|
|
if(c2->monst == moKnight) golems.push_back(c2);
|
|
if(c2->monst == moTameBomberbird) golems.push_back(c2);
|
|
if(c2->monst == moMouse) { golems.push_back(c2); havewhat |= HF_MOUSE; }
|
|
if(c2->monst == moPrincess || c2->monst == moPrincessArmed) golems.push_back(c2);
|
|
if(c2->monst == moIllusion) {
|
|
if(items[itOrbIllusion]) items[itOrbIllusion]--;
|
|
else c2->monst = moNone;
|
|
}
|
|
}
|
|
else if(c2->monst == moButterfly) {
|
|
addButterfly(c2);
|
|
}
|
|
else if(isAngryBird(c2->monst)) {
|
|
havewhat |= HF_BIRD;
|
|
if(c2->monst == moBat) havewhat |= HF_BATS | HF_EAGLES;
|
|
if(c2->monst == moEagle) havewhat |= HF_EAGLES;
|
|
}
|
|
else if(among(c2->monst, moFrog, moVaulter, moPhaser))
|
|
havewhat |= HF_JUMP;
|
|
else if(c2->monst == moReptile) havewhat |= HF_REPTILE;
|
|
else if(isLeader(c2->monst)) havewhat |= HF_LEADER;
|
|
else if(c2->monst == moEarthElemental) havewhat |= HF_EARTH;
|
|
else if(c2->monst == moWaterElemental) havewhat |= HF_WATER;
|
|
else if(c2->monst == moVoidBeast) havewhat |= HF_VOID;
|
|
else if(c2->monst == moHunterDog) havewhat |= HF_HUNTER;
|
|
else if(isMagneticPole(c2->monst)) havewhat |= HF_MAGNET;
|
|
else if(c2->monst == moAltDemon) havewhat |= HF_ALT;
|
|
else if(c2->monst == moHexDemon) havewhat |= HF_HEXD;
|
|
else if(among(c2->monst, moAnimatedDie, moAngryDie)) havewhat |= HF_DICE;
|
|
else if(c2->monst == moMonk) havewhat |= HF_MONK;
|
|
else if(c2->monst == moShark || c2->monst == moCShark || among(c2->monst, moRusalka, moPike)) havewhat |= HF_SHARK;
|
|
else if(c2->monst == moAirElemental)
|
|
havewhat |= HF_AIR, airmap.push_back(make_pair(c2,0));
|
|
}
|
|
// pheromones!
|
|
if(c2->land == laHive && c2->landparam >= 50 && c2->wall != waWaxWall)
|
|
havewhat |= HF_BUG;
|
|
if(c2->wall == waThumperOn)
|
|
targets.push_back(c2);
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
for(int i=first7; i<isize(dcal); i++)
|
|
forCellEx(c2, dcal[i])
|
|
if(c2->wall == waThumperOn) {
|
|
targets.push_back(c2);
|
|
}
|
|
|
|
while(recalcTide) {
|
|
recalcTide = false;
|
|
for(int i=0; i<isize(dcal); i++) checkTide(dcal[i]);
|
|
}
|
|
|
|
for(auto& t: tempmonsters) t.first->monst = t.second;
|
|
|
|
buildAirmap();
|
|
}
|
|
|
|
EX void moverefresh(bool turn IS(true)) {
|
|
int dcs = isize(dcal);
|
|
|
|
for(int i=0; i<dcs; i++) {
|
|
cell *c = dcal[i];
|
|
|
|
if(c->monst == moWolfMoved) c->monst = moWolf;
|
|
if(c->monst == moIvyNext) {
|
|
c->monst = moIvyHead; ivynext(c);
|
|
}
|
|
if(c->monst == moIvyDead)
|
|
removeIvy(c);
|
|
refreshFriend(c);
|
|
if(c->monst == moSlimeNextTurn) c->monst = moSlime;
|
|
if(c->monst == moLesser && !cellEdgeUnstable(c)) c->monst = moLesserM;
|
|
else if(c->monst == moLesserM) c->monst = moLesser;
|
|
if(c->monst == moGreater && !cellEdgeUnstable(c)) c->monst = moGreaterM;
|
|
else if(c->monst == moGreaterM) c->monst = moGreater;
|
|
|
|
if(c->monst == moPair && !c->stuntime) {
|
|
cell *c2 = c->move(c->mondir);
|
|
if(c2->monst != moPair) continue;
|
|
if(true) for(int i: {-1, 1}) {
|
|
cell *c3 = c->modmove(c->mondir + i);
|
|
if(among(c3->wall, waRuinWall, waColumn, waStone, waVinePlant, waPalace)) {
|
|
drawParticles(c3, winf[c3->wall].color, 30);
|
|
c3->wall = waNone;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(c->stuntime && !isMutantIvy(c)) {
|
|
if(turn) c->stuntime--;
|
|
int breathrange = sphere ? 2 : 3;
|
|
if(c->stuntime == 0 && c->monst == moDragonHead) {
|
|
// if moDragonHead is renamed to "Dragon Head", we might need to change this
|
|
eMonster subject = c->monst;
|
|
if(!c->hitpoints) c->hitpoints = 1;
|
|
else if(shmup::on && dragon::totalhp(c) > 2 && shmup::dragonbreath(c)) {
|
|
c->hitpoints = 0;
|
|
}
|
|
else if(dragon::totalhp(c) <= 2) ;
|
|
else if(isMounted(c)) {
|
|
if(dragon::target && celldistance(c, dragon::target) <= breathrange && makeflame(dragon::target, 5, true)) {
|
|
addMessage(XLAT("%The1 breathes fire!", subject));
|
|
makeflame(dragon::target, 5, false);
|
|
playSound(dragon::target, "fire");
|
|
c->hitpoints = 0;
|
|
}
|
|
}
|
|
else {
|
|
for(int i=0; i<isize(targets); i++) {
|
|
cell *t = targets[i];
|
|
if(celldistance(c, t) <= breathrange && makeflame(t, 5, true)) {
|
|
if(isPlayerOn(t)) addMessage(XLAT("%The1 breathes fire at you!", subject));
|
|
else if(t->monst)
|
|
addMessage(XLAT("%The1 breathes fire at %the2!", subject, t->monst));
|
|
else
|
|
addMessage(XLAT("%The1 breathes fire!", subject));
|
|
makeflame(t, 5, false);
|
|
playSound(t, "fire");
|
|
c->hitpoints = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// tortoises who have found their children no longer move
|
|
if(saved_tortoise_on(c))
|
|
c->stuntime = 2;
|
|
|
|
if(c->monst == moReptile) {
|
|
if(c->wall == waChasm || cellUnstable(c)) {
|
|
c->monst = moNone;
|
|
c->wall = waReptile;
|
|
c->wparam = reptilemax();
|
|
playSound(c, "click");
|
|
}
|
|
else if(isChasmy(c) || isWatery(c)) {
|
|
if(c->wall == waMercury) {
|
|
fallMonster(c, AF_FALL);
|
|
c->wall = waNone;
|
|
}
|
|
else {
|
|
c->wall = waReptileBridge;
|
|
c->wparam = reptilemax();
|
|
c->monst = moNone;
|
|
}
|
|
c->item = itNone;
|
|
playSound(c, "click");
|
|
}
|
|
}
|
|
|
|
if(c->wall == waChasm) {
|
|
if(c->land != laWhirlwind) c->item = itNone;
|
|
|
|
if(c->monst && !survivesChasm(c->monst) && c->monst != moReptile && normal_gravity_at(c)) {
|
|
if(c->monst != moRunDog && c->land == laMotion)
|
|
achievement_gain_once("FALLDEATH1");
|
|
addMessage(XLAT("%The1 falls!", c->monst));
|
|
fallMonster(c, AF_FALL);
|
|
}
|
|
}
|
|
|
|
else if(isReptile(c->wall) && turn) {
|
|
if(c->monst || isPlayerOn(c)) c->wparam = -1;
|
|
else if(c->cpdist <= 7) {
|
|
c->wparam--;
|
|
if(c->wparam == 0) {
|
|
if(c->wall == waReptile) c->wall = waChasm;
|
|
else placeWater(c, NULL);
|
|
c->monst = moReptile;
|
|
c->hitpoints = 3;
|
|
c->stuntime = 0;
|
|
vector<int> gooddirs;
|
|
// in the peace mode, a reptile will
|
|
// prefer to walk on the ground, rather than the chasm
|
|
for(int i=0; i<c->type; i++) {
|
|
int i0 = (i+3) % c->type;
|
|
int i1 = (i+c->type-3) % c->type;
|
|
if(c->move(i0) && passable(c->move(i0), c, 0))
|
|
if(c->move(i1) && passable(c->move(i1), c, 0))
|
|
gooddirs.push_back(i);
|
|
}
|
|
c->mondir = hrand_elt(gooddirs, c->mondir);
|
|
playSound(c, "click");
|
|
}
|
|
}
|
|
}
|
|
|
|
else if(isFire(c)) {
|
|
if(c->monst == moSalamander) c->stuntime = max<int>(c->stuntime, 1);
|
|
else if(c->monst == moVaulter && c->mondir == JUMP)
|
|
c->mondir = NODIR;
|
|
else if(c->monst && !survivesFire(c->monst) && !isWorm(c->monst)) {
|
|
addMessage(XLAT("%The1 burns!", c->monst));
|
|
if(isBull(c->monst)) {
|
|
addMessage(XLAT("Fire is extinguished!"));
|
|
c->wall = waNone;
|
|
}
|
|
fallMonster(c, AF_CRUSH);
|
|
}
|
|
if(c->item && itemBurns(c->item)) {
|
|
addMessage(XLAT("%The1 burns!", c->item));
|
|
c->item = itNone;
|
|
}
|
|
}
|
|
|
|
else if(isWatery(c)) {
|
|
if(c->monst == moLesser || c->monst == moLesserM || c->monst == moGreater || c->monst == moGreaterM)
|
|
c->monst = moGreaterShark;
|
|
if(c->monst && !survivesWater(c->monst) && normal_gravity_at(c)) {
|
|
playSound(c, "splash"+pick12());
|
|
if(isNonliving(c->monst))
|
|
addMessage(XLAT("%The1 sinks!", c->monst));
|
|
else
|
|
addMessage(XLAT("%The1 drowns!", c->monst));
|
|
if(isBull(c->monst)) {
|
|
addMessage(XLAT("%The1 is filled!", c->wall));
|
|
c->wall = waShallow;
|
|
}
|
|
fallMonster(c, AF_FALL);
|
|
}
|
|
}
|
|
else if(c->wall == waSulphur || c->wall == waSulphurC || c->wall == waMercury) {
|
|
if(c->monst && !survivesPoison(c->monst, c->wall) && normal_gravity_at(c)) {
|
|
playSound(c, "splash"+pick12());
|
|
if(isNonliving(c->monst))
|
|
addMessage(XLAT("%The1 sinks!", c->monst));
|
|
else
|
|
addMessage(XLAT("%The1 drowns!", c->monst));
|
|
if(isBull(c->monst)) {
|
|
addMessage(XLAT("%The1 is filled!", c->wall));
|
|
c->wall = waNone;
|
|
}
|
|
fallMonster(c, AF_FALL);
|
|
}
|
|
}
|
|
else if(c->wall == waMagma) {
|
|
if(c->monst == moSalamander) c->stuntime = max<int>(c->stuntime, 1);
|
|
else if(c->monst && !survivesPoison(c->monst, c->wall) && normal_gravity_at(c)) {
|
|
if(isNonliving(c->monst))
|
|
addMessage(XLAT("%The1 is destroyed by lava!", c->monst));
|
|
else
|
|
addMessage(XLAT("%The1 is killed by lava!", c->monst));
|
|
playSound(c, "steamhiss", 70);
|
|
fallMonster(c, AF_FALL);
|
|
}
|
|
}
|
|
else if(!isWateryOrBoat(c) && c->wall != waShallow) {
|
|
if(c->monst == moGreaterShark)
|
|
c->monst = moGreaterM;
|
|
else if(c->monst == moShark || c->monst == moCShark) {
|
|
addMessage(XLAT("%The1 suffocates!", c->monst));
|
|
fallMonster(c, AF_CRUSH);
|
|
}
|
|
else if(c->monst == moKrakenH) {
|
|
addMessage(XLAT("%The1 suffocates!", c->monst));
|
|
kraken::kill(c, moNone);
|
|
}
|
|
}
|
|
|
|
if(c->monst == moVineSpirit && !cellHalfvine(c) && c->wall != waVinePlant) {
|
|
addMessage(XLAT("%The1 is destroyed!", c->monst));
|
|
fallMonster(c, AF_CRUSH);
|
|
}
|
|
|
|
if(c->monst) mayExplodeMine(c, c->monst);
|
|
|
|
if(c->monst && c->wall == waClosedGate && !survivesWall(c->monst)) {
|
|
playSound(c, "hit-crush"+pick123());
|
|
addMessage(XLAT("%The1 is crushed!", c->monst));
|
|
fallMonster(c, AF_CRUSH);
|
|
}
|
|
|
|
if(c->monst && cellUnstable(c) && !ignoresPlates(c->monst) && !shmup::on)
|
|
doesFallSound(c);
|
|
}
|
|
}
|
|
|
|
// find worms and ivies
|
|
EX void settemp(cell *c) {
|
|
tempmonsters.emplace_back(c, (eMonster) c->monst);
|
|
c->monst = moNone;
|
|
}
|
|
|
|
EX void findWormIvy(cell *c) {
|
|
while(true) {
|
|
if(c->monst == moWorm || c->monst == moTentacle || c->monst == moWormwait || c->monst == moTentaclewait ||
|
|
c->monst == moTentacleEscaping) {
|
|
worms.push_back(c); settemp(c);
|
|
break;
|
|
}
|
|
else if(c->monst == moHexSnake) {
|
|
hexsnakes.push_back(c); settemp(c);
|
|
}
|
|
else if(c->monst == moWormtail || c->monst == moHexSnakeTail) {
|
|
bool bug = true;
|
|
for(int i=0; i<c->type; i++) {
|
|
cell* c2 = c->move(i);
|
|
if(c2 && isWorm(c2) && c2->mondir != NODIR && c2->move(c2->mondir) == c) {
|
|
settemp(c);
|
|
c = c2;
|
|
bug = false;
|
|
}
|
|
}
|
|
if(bug) break;
|
|
}
|
|
else if(c->monst == moIvyWait) {
|
|
cell* c2 = c->move(c->mondir);
|
|
settemp(c); c=c2;
|
|
}
|
|
else if(c->monst == moIvyHead) {
|
|
ivies.push_back(c); settemp(c);
|
|
break;
|
|
}
|
|
else if(c->monst == moIvyBranch || c->monst == moIvyRoot) {
|
|
bool bug = true;
|
|
for(int i=0; i<c->type; i++) {
|
|
cell* c2 = c->move(i);
|
|
if(c2 && (c2->monst == moIvyHead || c2->monst == moIvyBranch) && c2->move(c2->mondir) == c) {
|
|
settemp(c);
|
|
c = c2;
|
|
bug = false;
|
|
}
|
|
}
|
|
if(bug) break;
|
|
}
|
|
else break;
|
|
}
|
|
}
|
|
|
|
EX void monstersTurn() {
|
|
reset_spill();
|
|
checkSwitch();
|
|
mirror::breakAll();
|
|
DEBB(DF_TURN, ("bfs"));
|
|
bfs();
|
|
DEBB(DF_TURN, ("charge"));
|
|
if(elec::havecharge) elec::act();
|
|
DEBB(DF_TURN, ("mmo"));
|
|
int phase2 = (1 & items[itOrbSpeed]);
|
|
if(!phase2) movemonsters();
|
|
|
|
for(cell *pc: player_positions()) if(pc->item == itOrbSafety) {
|
|
collectItem(pc, pc, true);
|
|
return;
|
|
}
|
|
|
|
if(playerInPower() && (phase2 || !items[itOrbSpeed]) && (havewhat & HF_FAST))
|
|
moveNormals(moWitchSpeed);
|
|
|
|
if(phase2 && markOrb(itOrbEmpathy)) {
|
|
bfs();
|
|
movegolems(AF_FAST);
|
|
for(int i=0; i<isize(dcal); i++) {
|
|
if(dcal[i]->monst == moFriendlyGhost && dcal[i]->stuntime)
|
|
dcal[i]->stuntime--;
|
|
refreshFriend(dcal[i]);
|
|
}
|
|
}
|
|
DEBB(DF_TURN, ("rop"));
|
|
if(!dual::state) reduceOrbPowers();
|
|
int phase1 = (1 & items[itOrbSpeed]);
|
|
if(dual::state && items[itOrbSpeed]) phase1 = !phase1;
|
|
DEBB(DF_TURN, ("lc"));
|
|
if(!phase1) livecaves();
|
|
if(!phase1) ca::simulate();
|
|
if(!phase1) heat::processfires();
|
|
|
|
for(cell *c: crush_now) {
|
|
changes.ccell(c);
|
|
playSound(NULL, "closegate");
|
|
if(canAttack(c, moCrusher, c, c->monst, AF_GETPLAYER | AF_CRUSH)) {
|
|
attackMonster(c, AF_MSG | AF_GETPLAYER | AF_CRUSH, moCrusher);
|
|
}
|
|
moveEffect(movei(c, FALL), moDeadBird);
|
|
destroyBoats(c, NULL, true);
|
|
explodeBarrel(c);
|
|
}
|
|
|
|
changes.value_keep(crush_now);
|
|
changes.value_keep(crush_next);
|
|
crush_now = std::move(crush_next);
|
|
crush_next.clear();
|
|
|
|
DEBB(DF_TURN, ("heat"));
|
|
heat::processheat();
|
|
// if(elec::havecharge) elec::drawcharges();
|
|
|
|
orbbull::check();
|
|
|
|
#if CAP_COMPLEX2
|
|
if(!phase1) terracotta::check();
|
|
#endif
|
|
|
|
if(items[itOrbFreedom])
|
|
for(cell *pc: player_positions())
|
|
checkFreedom(pc);
|
|
|
|
DEBB(DF_TURN, ("check"));
|
|
checkmove();
|
|
if(canmove) elec::checklightningfast();
|
|
|
|
|
|
#if CAP_HISTORY
|
|
for(cell *pc: player_positions())
|
|
history::movehistory.push_back(pc);
|
|
#endif
|
|
}
|
|
|
|
/** check if whirlline is looped, if yes, remove the repeat; may not detect loops immediately */
|
|
EX bool looped(vector<cell*>& whirlline) {
|
|
if(isize(whirlline) == 1)
|
|
return false;
|
|
if(whirlline.back() == whirlline.front()) {
|
|
whirlline.pop_back();
|
|
return true;
|
|
}
|
|
int pos = isize(whirlline)/2;
|
|
if(isize(whirlline) > 2 && whirlline.back() == whirlline[pos]) {
|
|
while(pos && whirlline.back() == whirlline[pos])
|
|
whirlline.pop_back();
|
|
/* something weird must have happened... */
|
|
static bool once = true;
|
|
if(once) addMessage("warning: a looped line");
|
|
once = false;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
}
|