mirror of
https://github.com/zenorogue/hyperrogue.git
synced 2025-04-03 09:17:02 +00:00
more movei. Also animated Friendly Ivy growth
This commit is contained in:
parent
2172ca6fd7
commit
6d44105fef
94
game.cpp
94
game.cpp
@ -3588,10 +3588,6 @@ EX void activateArrowTrap(cell *c) {
|
||||
* or when a dead bird falls (then m == moDeadBird)
|
||||
*/
|
||||
|
||||
EX void moveEffect(cell *ct, cell *cf, eMonster m, int direction_hint) {
|
||||
moveEffect(movei(cf, ct, direction_hint), m);
|
||||
}
|
||||
|
||||
EX void moveEffect(const movei& mi, eMonster m) {
|
||||
|
||||
auto& cf = mi.s;
|
||||
@ -3905,7 +3901,7 @@ EX void moveMonster(const movei& mi) {
|
||||
|
||||
if(m == moEarthElemental) {
|
||||
if(!passable(ct, cf, 0)) earthFloor(ct);
|
||||
earthMove(cf, neighborId(cf, ct));
|
||||
earthMove(mi);
|
||||
}
|
||||
|
||||
if(m == moWaterElemental) {
|
||||
@ -4804,19 +4800,20 @@ EX void moveWorm(cell *c) {
|
||||
return;
|
||||
}
|
||||
|
||||
cell* goal = c->move(dir);
|
||||
movei mi(c, dir);
|
||||
auto& goal = mi.t;
|
||||
|
||||
if(isPlayerOn(goal) || goal->monst)
|
||||
attackMonster(goal, AF_EAT | AF_MSG | AF_GETPLAYER, c->monst);
|
||||
|
||||
for(int j=0; j<c->type; j++) if(c->move(j) == goal) {
|
||||
if(1) {
|
||||
goal->monst = eMonster(moWormwait + id);
|
||||
moveEffect(goal, NULL, eMonster(moWormwait + id), NOHINT);
|
||||
moveEffect(mi, eMonster(moWormwait + id));
|
||||
|
||||
animateMovement(c, goal, LAYER_BIG, dir);
|
||||
animateMovement(mi, LAYER_BIG);
|
||||
c->monst = eMonster(moWormtail + id);
|
||||
goal->mondir = c->c.spin(j);
|
||||
goal->monmirror = c->monmirror ^ c->c.mirror(j);
|
||||
goal->mondir = mi.rev_dir_or(NODIR);
|
||||
goal->monmirror = c->monmirror ^ c->c.mirror(dir);
|
||||
|
||||
mountmove(goal, goal->mondir, true, c);
|
||||
|
||||
@ -4841,9 +4838,10 @@ EX void moveWorm(cell *c) {
|
||||
drawParticles(c2, (linf[c2->land].color & 0xF0F0F0), 16);
|
||||
return;
|
||||
}
|
||||
c3 = c2, c2 = c3->move(c2->mondir);
|
||||
movei mim(c2, c2->mondir);
|
||||
c3 = c2, c2 = mim.t;
|
||||
mountmove(c3, c3->mondir, true, c2);
|
||||
animateMovement(c2, c3, LAYER_BIG, revhint(c2, c2->mondir));
|
||||
animateMovement(mim.rev(), LAYER_BIG);
|
||||
}
|
||||
}
|
||||
|
||||
@ -4912,45 +4910,47 @@ EX void moveivy() {
|
||||
if(c->monst != moIvyHead) continue;
|
||||
ivynext(c);
|
||||
|
||||
cell *mto = NULL;
|
||||
int pd = c->pathdist;
|
||||
int sp = 0;
|
||||
|
||||
movei mi(nullptr, nullptr, NODIR);
|
||||
|
||||
while(c->monst != moIvyRoot) {
|
||||
if(!isIvy(c->monst)) {
|
||||
raiseBuggyGeneration(c, "that's not an Ivy!");
|
||||
}
|
||||
if(c->mondir == NODIR)
|
||||
raiseBuggyGeneration(c, "wrong mondir!");
|
||||
for(int j=0; j<c->type; j++) {
|
||||
if(c->move(j) && canAttack(c, c->monst, c->move(j), c->move(j)->monst, AF_ONLY_FRIEND | AF_GETPLAYER)) {
|
||||
if(isPlayerOn(c->move(j)))
|
||||
killThePlayerAt(c->monst, c->move(j), 0);
|
||||
|
||||
forCellIdEx(c2, j, c) {
|
||||
if(canAttack(c, c->monst, c2, c2->monst, AF_ONLY_FRIEND | AF_GETPLAYER)) {
|
||||
if(isPlayerOn(c2))
|
||||
killThePlayerAt(c->monst, c2, 0);
|
||||
else {
|
||||
if(attackJustStuns(c->move(j), 0, c->monst))
|
||||
addMessage(XLAT("The ivy attacks %the1!", c->move(j)->monst));
|
||||
else if(isNonliving(c->move(j)->monst))
|
||||
addMessage(XLAT("The ivy destroys %the1!", c->move(j)->monst));
|
||||
if(attackJustStuns(c2, 0, c->monst))
|
||||
addMessage(XLAT("The ivy attacks %the1!", c2->monst));
|
||||
else if(isNonliving(c2->monst))
|
||||
addMessage(XLAT("The ivy destroys %the1!", c2->monst));
|
||||
else
|
||||
addMessage(XLAT("The ivy kills %the1!", c->move(j)->monst));
|
||||
attackMonster(c->move(j), AF_NORMAL, c->monst);
|
||||
addMessage(XLAT("The ivy kills %the1!", c2->monst));
|
||||
attackMonster(c2, AF_NORMAL, c->monst);
|
||||
}
|
||||
continue;
|
||||
}
|
||||
if(c->move(j) && signed(c->move(j)->pathdist) < pd && passable(c->move(j), c, 0)
|
||||
&& !strictlyAgainstGravity(c->move(j), c, false, MF_IVY))
|
||||
mto = c->move(j), pd = mto->pathdist, sp = c->c.spin(j);
|
||||
if(c2 && c2->pathdist < pd && passable(c2, c, 0) && !strictlyAgainstGravity(c2, c, false, MF_IVY))
|
||||
mi = movei(c, j), pd = c2->pathdist;
|
||||
}
|
||||
c = c->move(c->mondir);
|
||||
}
|
||||
|
||||
auto& mto = mi.t;
|
||||
|
||||
if(mto && mto->cpdist) {
|
||||
animateMovement(mto->move(sp), mto, LAYER_BIG, mto->c.spin(sp));
|
||||
mto->monst = moIvyWait, mto->mondir = sp;
|
||||
mto->monmirror = c->monmirror ^ c->c.mirror(sp);
|
||||
moveEffect(mto, NULL, moIvyWait, NOHINT);
|
||||
animateMovement(mi, LAYER_BIG);
|
||||
mto->monst = moIvyWait, mto->mondir = mi.rev_dir_or(NODIR);
|
||||
mto->monmirror = mi.s->monmirror ^ mi.mirror();
|
||||
moveEffect(mi, moIvyWait);
|
||||
// if this is the only branch, we want to move the head immediately to mto instead
|
||||
if(mto->move(mto->mondir)->monst == moIvyHead) {
|
||||
if(mi.s->monst == moIvyHead) {
|
||||
mto->monst = moIvyHead; co->monst = moIvyBranch;
|
||||
}
|
||||
}
|
||||
@ -4962,11 +4962,13 @@ EX void moveivy() {
|
||||
}
|
||||
}
|
||||
|
||||
EX bool earthMove(cell *from, int dir) {
|
||||
EX bool earthMove(const movei& mi) {
|
||||
auto& from = mi.s;
|
||||
bool b = false;
|
||||
cell *c2 = from->move(dir);
|
||||
int d = from->c.spin(dir);
|
||||
cell *c2 = mi.t;
|
||||
b |= earthWall(from);
|
||||
if(!mi.proper()) return b;
|
||||
int d = mi.rev_dir_or(0);
|
||||
if(c2) for(int u=2; u<=c2->type-2; u++) {
|
||||
cell *c3 = c2->modmove(d + u);
|
||||
if(c3) b |= earthFloor(c3);
|
||||
@ -5143,15 +5145,17 @@ EX void groupmove(eMonster movtype, flagtype mf) {
|
||||
|
||||
vector<cell*> hexdfs;
|
||||
|
||||
// note: move from 'c' to 'from'!
|
||||
EX void moveHexSnake(cell *from, cell *c, int d, bool mounted) {
|
||||
EX void moveHexSnake(const movei& mi, bool mounted) {
|
||||
// note: move from 'c' to 'from'!
|
||||
auto& from = mi.t;
|
||||
auto& c = mi.s;
|
||||
if(from->wall == waBoat) from->wall = waSea;
|
||||
moveEffect(from, c, c->monst, revhint(from, d));
|
||||
from->monst = c->monst; from->mondir = d; from->hitpoints = c->hitpoints;
|
||||
moveEffect(mi, c->monst);
|
||||
from->monst = c->monst; from->mondir = mi.rev_dir_or(NODIR); from->hitpoints = c->hitpoints;
|
||||
c->monst = moHexSnakeTail;
|
||||
preventbarriers(from);
|
||||
|
||||
animateMovement(c, from, LAYER_BIG, revhint(from, d));
|
||||
animateMovement(mi, LAYER_BIG);
|
||||
mountmove(from, from->mondir, true, c);
|
||||
|
||||
cell *c2 = c, *c3=c2;
|
||||
@ -5159,7 +5163,7 @@ EX void moveHexSnake(cell *from, cell *c, int d, bool mounted) {
|
||||
if(a == ROCKSNAKELENGTH) { c2->monst = moNone, c3->mondir = NODIR; break; }
|
||||
if(c2->mondir == NODIR) break;
|
||||
mountmove(c2, c2->mondir, true, c2->move(c2->mondir));
|
||||
animateMovement(c2->move(c2->mondir), c2, LAYER_BIG, revhint(c2, c2->mondir));
|
||||
animateMovement(movei(c2, c2->mondir).rev(), LAYER_BIG);
|
||||
c3 = c2, c2 = c3->move(c2->mondir);
|
||||
}
|
||||
else break;
|
||||
@ -5217,7 +5221,7 @@ EX void hexvisit(cell *c, cell *from, int d, bool mounted, int colorpair) {
|
||||
if(from->cpdist == 0 || from->monst) return;
|
||||
|
||||
snakeAttack(c, mounted);
|
||||
moveHexSnake(from, c, d, mounted);
|
||||
moveHexSnake(movei(from, d).rev(), mounted);
|
||||
}
|
||||
|
||||
onpath(c, 0);
|
||||
@ -7856,7 +7860,7 @@ EX void monstersTurn() {
|
||||
if(canAttack(c, moCrusher, c, c->monst, AF_GETPLAYER | AF_CRUSH)) {
|
||||
attackMonster(c, AF_MSG | AF_GETPLAYER | AF_CRUSH, moCrusher);
|
||||
}
|
||||
moveEffect(c, c, moDeadBird, -1);
|
||||
moveEffect(movei(c, FALL), moDeadBird);
|
||||
destroyBoats(c, NULL, true);
|
||||
explodeBarrel(c);
|
||||
}
|
||||
@ -8596,7 +8600,7 @@ EX bool movepcto(int d, int subdir IS(1), bool checkonly IS(false)) {
|
||||
|
||||
if(items[itOrbDigging]) {
|
||||
invismove = false;
|
||||
if(earthMove(cwt.at, d)) markOrb(itOrbDigging);
|
||||
if(earthMove(mi)) markOrb(itOrbDigging);
|
||||
}
|
||||
|
||||
movecost(cwt.at, c2, 1);
|
||||
|
@ -463,6 +463,7 @@ struct movei {
|
||||
movei rev() const { return movei(t, s, rev_dir_or(d)); }
|
||||
int dir_or(int x) const { return proper() ? d : x; }
|
||||
int rev_dir_or(int x) const { return proper() ? s->c.spin(d) : x; }
|
||||
bool mirror() { return s->c.mirror(d); }
|
||||
};
|
||||
#endif
|
||||
|
||||
|
36
orbs.cpp
36
orbs.cpp
@ -50,21 +50,25 @@ EX void drainOrb(eItem it, int target IS(0)) {
|
||||
}
|
||||
|
||||
EX void empathyMove(cell *c, cell *cto, int dir) {
|
||||
empathyMove(movei(c, cto, dir)); // to erase
|
||||
}
|
||||
|
||||
EX void empathyMove(const movei& mi) {
|
||||
if(!items[itOrbEmpathy]) return;
|
||||
|
||||
if(items[itOrbFire]) {
|
||||
invismove = false;
|
||||
if(makeflame(c, 10, false)) markEmpathy(itOrbFire);
|
||||
if(makeflame(mi.s, 10, false)) markEmpathy(itOrbFire);
|
||||
}
|
||||
|
||||
if(items[itOrbDigging]) {
|
||||
if(dir != STRONGWIND && earthMove(c, dir))
|
||||
if(mi.proper() && earthMove(mi))
|
||||
markEmpathy(itOrbDigging), invismove = false;
|
||||
}
|
||||
|
||||
if(items[itOrbWinter] && isIcyLand(c) && c->wall == waNone) {
|
||||
if(items[itOrbWinter] && isIcyLand(mi.s) && mi.s->wall == waNone) {
|
||||
invismove = false;
|
||||
c->wall = waIcewall;
|
||||
mi.s->wall = waIcewall;
|
||||
markEmpathy(itOrbWinter);
|
||||
}
|
||||
}
|
||||
@ -636,14 +640,16 @@ EX void jumpTo(cell *dest, eItem byWhat, int bonuskill IS(0), eMonster dashmon I
|
||||
monstersTurn();
|
||||
}
|
||||
|
||||
void growIvyTo(cell *dest, cell *src, int direction_hint) {
|
||||
void growIvyTo(const movei& mi) {
|
||||
auto& dest = mi.t;
|
||||
if(dest->monst)
|
||||
attackMonster(dest, AF_NORMAL | AF_MSG, moFriendlyIvy);
|
||||
else {
|
||||
dest->monst = moFriendlyIvy;
|
||||
dest->mondir = neighborId(dest, src);
|
||||
moveEffect(dest, src, moFriendlyIvy, direction_hint);
|
||||
empathyMove(src, dest, neighborId(src, dest));
|
||||
dest->mondir = mi.rev_dir_or(NODIR);
|
||||
animateMovement(mi, LAYER_BIG);
|
||||
moveEffect(mi, moFriendlyIvy);
|
||||
empathyMove(mi);
|
||||
}
|
||||
createNoise(1);
|
||||
monstersTurn();
|
||||
@ -817,7 +823,7 @@ void summonAt(cell *dest) {
|
||||
if(dest->monst == moTortoise)
|
||||
tortoise::emap[dest] = getBits(dest), dest->hitpoints = 3;
|
||||
addMessage(XLAT("You summon %the1!", dest->monst));
|
||||
moveEffect(dest, dest, dest->monst, -1);
|
||||
moveEffect(movei(dest, FALL), dest->monst);
|
||||
if(dest->wall == waClosePlate || dest->wall == waOpenPlate)
|
||||
toggleGates(dest, dest->wall);
|
||||
|
||||
@ -886,7 +892,7 @@ void gun_attack(cell *dest) {
|
||||
|
||||
EX void checkStunKill(cell *dest) {
|
||||
if(isBird(dest->monst)) {
|
||||
moveEffect(dest, dest, moDeadBird, NOHINT);
|
||||
moveEffect(movei(dest, FALL), moDeadBird);
|
||||
doesFall(dest);
|
||||
if(isWatery(dest) || dest->wall == waChasm || isFire(dest)) {
|
||||
addMessage(XLAT("%The1 falls!", dest->monst));
|
||||
@ -1094,7 +1100,6 @@ EX eItem targetRangedOrb(cell *c, orbAction a) {
|
||||
|
||||
// nature
|
||||
if(items[itOrbNature] && numplayers() == 1 && c->monst != moFriendlyIvy) {
|
||||
cell *sides[MAX_EDGE];
|
||||
int dirs[MAX_EDGE];
|
||||
int qsides = 0;
|
||||
forCellIdCM(cf, d, c)
|
||||
@ -1109,13 +1114,12 @@ EX eItem targetRangedOrb(cell *c, orbAction a) {
|
||||
if(strictlyAgainstGravity(c, cf, false, MF_IVY)) continue;
|
||||
if(monstersnear(cwt.at, NULL, moPlayer, c, cwt.at)) continue;
|
||||
}
|
||||
dirs[qsides] = d;
|
||||
sides[qsides++] = cf;
|
||||
dirs[qsides++] = d;
|
||||
}
|
||||
|
||||
|
||||
if(qsides > 0) {
|
||||
int di = hrand(qsides);
|
||||
if(!isCheck(a)) growIvyTo(c, sides[di], revhint(c, dirs[di]));
|
||||
int di = dirs[hrand(qsides)];
|
||||
if(!isCheck(a)) growIvyTo(movei(c, di).rev());
|
||||
return itOrbNature;
|
||||
}
|
||||
}
|
||||
|
11
shmup.cpp
11
shmup.cpp
@ -1062,7 +1062,8 @@ void movePlayer(monster *m, int delta) {
|
||||
}
|
||||
}
|
||||
else if(isPushable(c2->wall) && !nonAdjacent(c2, m->base)) {
|
||||
int sd = neighborId(m->base, c2);
|
||||
int sd1 = neighborId(m->base, c2);
|
||||
int sd = m->base->c.spin(sd1);
|
||||
int subdir = 1;
|
||||
double bestd = 9999;
|
||||
for(int di=-1; di<2; di+=2) {
|
||||
@ -1073,7 +1074,7 @@ void movePlayer(monster *m, int delta) {
|
||||
if(d<bestd) bestd=d, subdir = di;
|
||||
}
|
||||
pushmonsters();
|
||||
auto mip = determinePush(cellwalker(c2, sd), subdir, [m, c2] (cell *c) { return canPushThumperOn(c, c2, m->base); });
|
||||
auto mip = determinePush(cellwalker(c2, sd1), subdir, [m, c2] (cell *c) { return canPushThumperOn(c, c2, m->base); });
|
||||
visibleFor(300);
|
||||
if(!mip.proper()) go = false;
|
||||
else pushThumper(mip);
|
||||
@ -1141,8 +1142,7 @@ void movePlayer(monster *m, int delta) {
|
||||
|
||||
if(items[itOrbDigging]) {
|
||||
visibleFor(400);
|
||||
int d = neighborId(m->base, c2);
|
||||
if(d >= 0 && earthMove(m->base, d)) markOrb(itOrbDigging);
|
||||
if(earthMove(match(m->base, c2))) markOrb(itOrbDigging);
|
||||
}
|
||||
|
||||
cwt.at = c2; afterplayermoved();
|
||||
@ -2220,8 +2220,7 @@ void moveMonster(monster *m, int delta) {
|
||||
if(c2 != m->base && m->type == moFireElemental) makeflame(m->base, 20, false);
|
||||
if(c2 != m->base && m->type == moWaterElemental) placeWater(c2, m->base);
|
||||
if(c2 != m->base && m->type == moEarthElemental) {
|
||||
int d = neighborId(m->base, c2);
|
||||
if(d >= 0) earthMove(m->base, d);
|
||||
earthMove(match(m->base, c2));
|
||||
}
|
||||
|
||||
if(m->type == moReptile && c2 != m->base) {
|
||||
|
Loading…
x
Reference in New Issue
Block a user