mirror of
https://github.com/zenorogue/hyperrogue.git
synced 2025-01-04 14:30:35 +00:00
shmup:: parent-child relation is now based on ref counting
This commit is contained in:
parent
68ab6fedc6
commit
1612b767b3
82
shmup.cpp
82
shmup.cpp
@ -62,7 +62,6 @@ struct monster {
|
|||||||
bool inBoat;
|
bool inBoat;
|
||||||
bool no_targetting;
|
bool no_targetting;
|
||||||
monster *parent; // who shot this missile
|
monster *parent; // who shot this missile
|
||||||
eMonster parenttype; // type of the parent
|
|
||||||
int nextshot; // when will it be able to shot (players/flailers)
|
int nextshot; // when will it be able to shot (players/flailers)
|
||||||
int pid; // player ID
|
int pid; // player ID
|
||||||
int hitpoints; // hitpoints; or time elapsed in Asteroids
|
int hitpoints; // hitpoints; or time elapsed in Asteroids
|
||||||
@ -74,12 +73,16 @@ struct monster {
|
|||||||
bool isVirtual; // off the screen: gmatrix is unknown, and pat equals at
|
bool isVirtual; // off the screen: gmatrix is unknown, and pat equals at
|
||||||
hyperpoint inertia;// for frictionless lands
|
hyperpoint inertia;// for frictionless lands
|
||||||
|
|
||||||
|
int refs; // +1 for every reference (parent, lists of active monsters)
|
||||||
|
|
||||||
monster() {
|
monster() {
|
||||||
dead = false; inBoat = false; parent = NULL; nextshot = 0;
|
dead = false; inBoat = false; parent = NULL; nextshot = 0;
|
||||||
stunoff = 0; blowoff = 0; footphase = 0; no_targetting = false;
|
stunoff = 0; blowoff = 0; footphase = 0; no_targetting = false;
|
||||||
swordangle = 0; inertia = Hypc; ori = Id;
|
swordangle = 0; inertia = Hypc; ori = Id; refs = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
eMonster get_parenttype() { return parent ? parent->type : moNone; }
|
||||||
|
|
||||||
void store();
|
void store();
|
||||||
|
|
||||||
void findpat();
|
void findpat();
|
||||||
@ -88,6 +91,20 @@ struct monster {
|
|||||||
|
|
||||||
void rebasePat(const shiftmatrix& new_pat, cell *tgt);
|
void rebasePat(const shiftmatrix& new_pat, cell *tgt);
|
||||||
|
|
||||||
|
void remove_reference() {
|
||||||
|
refs--;
|
||||||
|
if(!refs) {
|
||||||
|
if(parent) parent->remove_reference();
|
||||||
|
delete this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void set_parent(monster *par) {
|
||||||
|
if(parent) parent->remove_reference();
|
||||||
|
parent = par;
|
||||||
|
parent->refs++;
|
||||||
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -337,10 +354,9 @@ void awakenMimics(monster *m, cell *c2) {
|
|||||||
m2->base = c;
|
m2->base = c;
|
||||||
|
|
||||||
if(isBullet(m)) {
|
if(isBullet(m)) {
|
||||||
m2->parenttype = m->parenttype;
|
|
||||||
m2->type = m->type;
|
m2->type = m->type;
|
||||||
m2->vel = m->vel;
|
m2->vel = m->vel;
|
||||||
m2->parent = m->parent;
|
m2->set_parent(m->parent);
|
||||||
m2->pid = m->pid;
|
m2->pid = m->pid;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -402,9 +418,8 @@ void shootBullet(monster *m) {
|
|||||||
if(WDIM == 3) bullet->at = bullet->at * cpush(2, 0.15 * SCALE);
|
if(WDIM == 3) bullet->at = bullet->at * cpush(2, 0.15 * SCALE);
|
||||||
if(prod) bullet->ori = m->ori;
|
if(prod) bullet->ori = m->ori;
|
||||||
bullet->type = moBullet;
|
bullet->type = moBullet;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
bullet->pid = m->pid;
|
bullet->pid = m->pid;
|
||||||
bullet->parenttype = m->type;
|
|
||||||
bullet->inertia = m->inertia;
|
bullet->inertia = m->inertia;
|
||||||
bullet->inertia[frontdir()] += bullet_velocity(m->type) * SCALE;
|
bullet->inertia[frontdir()] += bullet_velocity(m->type) * SCALE;
|
||||||
bullet->hitpoints = 0;
|
bullet->hitpoints = 0;
|
||||||
@ -422,9 +437,8 @@ void shootBullet(monster *m) {
|
|||||||
if(prod) bullet->ori = m->ori;
|
if(prod) bullet->ori = m->ori;
|
||||||
if(WDIM == 3) bullet->at = bullet->at * cpush(2, 0.15 * SCALE);
|
if(WDIM == 3) bullet->at = bullet->at * cpush(2, 0.15 * SCALE);
|
||||||
bullet->type = moBullet;
|
bullet->type = moBullet;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
bullet->pid = m->pid;
|
bullet->pid = m->pid;
|
||||||
bullet->parenttype = m->type;
|
|
||||||
bullet->hitpoints = 0;
|
bullet->hitpoints = 0;
|
||||||
bullet->inertia = cspin(0, WDIM-1, -M_PI/4 * i) * m->inertia;
|
bullet->inertia = cspin(0, WDIM-1, -M_PI/4 * i) * m->inertia;
|
||||||
bullet->inertia[frontdir()] += bullet_velocity(m->type) * SCALE;
|
bullet->inertia[frontdir()] += bullet_velocity(m->type) * SCALE;
|
||||||
@ -642,7 +656,7 @@ EX void activateArrow(cell *c) {
|
|||||||
traplist.emplace(ticks + 500, c);
|
traplist.emplace(ticks + 500, c);
|
||||||
}
|
}
|
||||||
|
|
||||||
monster arrowtrap_fakeparent;
|
monster arrowtrap_fakeparent, dragon_fakeparent;
|
||||||
|
|
||||||
void doTraps() {
|
void doTraps() {
|
||||||
while(true) {
|
while(true) {
|
||||||
@ -662,9 +676,9 @@ void doTraps() {
|
|||||||
bullet->base = tl[i];
|
bullet->base = tl[i];
|
||||||
bullet->at = rspintox(inverse_shift(tu, tC0(tv)));
|
bullet->at = rspintox(inverse_shift(tu, tC0(tv)));
|
||||||
bullet->type = moArrowTrap;
|
bullet->type = moArrowTrap;
|
||||||
bullet->parent = &arrowtrap_fakeparent;
|
bullet->set_parent(&arrowtrap_fakeparent);
|
||||||
|
arrowtrap_fakeparent.type = moArrowTrap;
|
||||||
bullet->pid = 0;
|
bullet->pid = 0;
|
||||||
bullet->parenttype = moArrowTrap;
|
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
}
|
}
|
||||||
catch(out_of_range&) {}
|
catch(out_of_range&) {}
|
||||||
@ -1487,9 +1501,8 @@ void shoot(eItem it, monster *m) {
|
|||||||
/* ori */
|
/* ori */
|
||||||
if(WDIM == 3) bullet->at = bullet->at * cpush(2, 0.15 * SCALE);
|
if(WDIM == 3) bullet->at = bullet->at * cpush(2, 0.15 * SCALE);
|
||||||
bullet->type = it == itOrbDragon ? moFireball : it == itOrbAir ? moAirball : moBullet;
|
bullet->type = it == itOrbDragon ? moFireball : it == itOrbAir ? moAirball : moBullet;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
bullet->pid = m->pid;
|
bullet->pid = m->pid;
|
||||||
bullet->parenttype = m->type;
|
|
||||||
items[it]--;
|
items[it]--;
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
}
|
}
|
||||||
@ -1596,9 +1609,7 @@ void spawn_asteroids(monster *bullet, monster *target) {
|
|||||||
child->at = target->at;
|
child->at = target->at;
|
||||||
child->ori = target->ori;
|
child->ori = target->ori;
|
||||||
child->type = target->type;
|
child->type = target->type;
|
||||||
child->parent = NULL;
|
|
||||||
child->pid = target->pid;
|
child->pid = target->pid;
|
||||||
child->parenttype = target->type;
|
|
||||||
child->inertia = target->inertia;
|
child->inertia = target->inertia;
|
||||||
child->inertia += bullet_inertia / 5;
|
child->inertia += bullet_inertia / 5;
|
||||||
child->hitpoints = target->hitpoints - 1;
|
child->hitpoints = target->hitpoints - 1;
|
||||||
@ -1668,7 +1679,8 @@ void moveBullet(monster *m, int delta) {
|
|||||||
|
|
||||||
if(m->type != moTongue && !(godragon || (c2==m->base && m->type == moArrowTrap) || passable(c2, m->base, P_BULLET | P_MIRRORWALL))) {
|
if(m->type != moTongue && !(godragon || (c2==m->base && m->type == moArrowTrap) || passable(c2, m->base, P_BULLET | P_MIRRORWALL))) {
|
||||||
m->dead = true;
|
m->dead = true;
|
||||||
if(m->type != moAirball) killMonster(c2, m->parent ? m->parent->type : moNone);
|
if(m->type != moAirball)
|
||||||
|
killMonster(c2, m->get_parenttype());
|
||||||
// cell *c = m->base;
|
// cell *c = m->base;
|
||||||
if(m->parent && isPlayer(m->parent)) {
|
if(m->parent && isPlayer(m->parent)) {
|
||||||
if(c2->wall == waBigTree) {
|
if(c2->wall == waBigTree) {
|
||||||
@ -1809,7 +1821,7 @@ void moveBullet(monster *m, int delta) {
|
|||||||
nat = spin_towards(nat, m->ori, tC0(m->parent->pat), bulletdir(), 1);
|
nat = spin_towards(nat, m->ori, tC0(m->parent->pat), bulletdir(), 1);
|
||||||
m->rebasePat(nat, m->base);
|
m->rebasePat(nat, m->base);
|
||||||
}
|
}
|
||||||
m->parent = m2;
|
m->set_parent(m2);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
m->dead = true;
|
m->dead = true;
|
||||||
@ -1844,7 +1856,7 @@ EX bool dragonbreath(cell *dragon) {
|
|||||||
bullet->base = dragon;
|
bullet->base = dragon;
|
||||||
bullet->at = spin_towards(Id, bullet->ori, inverse_shift(gmatrix[dragon], tC0(pc[randplayer]->pat)), bulletdir(), 1);
|
bullet->at = spin_towards(Id, bullet->ori, inverse_shift(gmatrix[dragon], tC0(pc[randplayer]->pat)), bulletdir(), 1);
|
||||||
bullet->type = moFireball;
|
bullet->type = moFireball;
|
||||||
bullet->parent = bullet;
|
bullet->set_parent(&dragon_fakeparent); dragon_fakeparent.type = moDragonHead;
|
||||||
bullet->pid = randplayer;
|
bullet->pid = randplayer;
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
return true;
|
return true;
|
||||||
@ -2214,8 +2226,7 @@ void moveMonster(monster *m, int delta) {
|
|||||||
bullet->at = m->at;
|
bullet->at = m->at;
|
||||||
bullet->ori = m->ori;
|
bullet->ori = m->ori;
|
||||||
bullet->type = moTongue;
|
bullet->type = moTongue;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
bullet->parenttype = m->type;
|
|
||||||
bullet->pid = whichPlayerOn(c2);
|
bullet->pid = whichPlayerOn(c2);
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
return;
|
return;
|
||||||
@ -2287,8 +2298,7 @@ void moveMonster(monster *m, int delta) {
|
|||||||
undead->base = c2;
|
undead->base = c2;
|
||||||
undead->at = Id;
|
undead->at = Id;
|
||||||
undead->type = moZombie;
|
undead->type = moZombie;
|
||||||
undead->parent = m;
|
undead->set_parent(m);
|
||||||
undead->parenttype = m->type;
|
|
||||||
undead->pid = 0;
|
undead->pid = 0;
|
||||||
undead->findpat();
|
undead->findpat();
|
||||||
additional.push_back(undead);
|
additional.push_back(undead);
|
||||||
@ -2297,8 +2307,7 @@ void moveMonster(monster *m, int delta) {
|
|||||||
undead->base = c3;
|
undead->base = c3;
|
||||||
undead->at = Id;
|
undead->at = Id;
|
||||||
undead->type = moGhost;
|
undead->type = moGhost;
|
||||||
undead->parent = m;
|
undead->set_parent(m);
|
||||||
undead->parenttype = m->type;
|
|
||||||
undead->findpat();
|
undead->findpat();
|
||||||
undead->pid = 0;
|
undead->pid = 0;
|
||||||
additional.push_back(undead);
|
additional.push_back(undead);
|
||||||
@ -2367,7 +2376,7 @@ void moveMonster(monster *m, int delta) {
|
|||||||
bullet->at = m->at;
|
bullet->at = m->at;
|
||||||
bullet->ori = m->ori;
|
bullet->ori = m->ori;
|
||||||
bullet->type = moFireball;
|
bullet->type = moFireball;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
bullet->pid = directi;
|
bullet->pid = directi;
|
||||||
if(m->type == moPyroCultist)
|
if(m->type == moPyroCultist)
|
||||||
@ -2381,8 +2390,7 @@ void moveMonster(monster *m, int delta) {
|
|||||||
bullet->at = m->at;
|
bullet->at = m->at;
|
||||||
bullet->ori = m->ori;
|
bullet->ori = m->ori;
|
||||||
bullet->type = moBullet;
|
bullet->type = moBullet;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
bullet->parenttype = moOutlaw;
|
|
||||||
bullet->pid = directi;
|
bullet->pid = directi;
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
m->nextshot = curtime + 1500;
|
m->nextshot = curtime + 1500;
|
||||||
@ -2394,7 +2402,7 @@ void moveMonster(monster *m, int delta) {
|
|||||||
bullet->at = m->at;
|
bullet->at = m->at;
|
||||||
bullet->ori = m->ori;
|
bullet->ori = m->ori;
|
||||||
bullet->type = moAirball;
|
bullet->type = moAirball;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
bullet->pid = i;
|
bullet->pid = i;
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
m->nextshot = curtime + 1500;
|
m->nextshot = curtime + 1500;
|
||||||
@ -2414,7 +2422,7 @@ void moveMonster(monster *m, int delta) {
|
|||||||
bullet->at = m->at;
|
bullet->at = m->at;
|
||||||
bullet->ori = m->ori;
|
bullet->ori = m->ori;
|
||||||
bullet->type = moFlailBullet;
|
bullet->type = moFlailBullet;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
bullet->vel = 1/400.0;
|
bullet->vel = 1/400.0;
|
||||||
bullet->pid = i;
|
bullet->pid = i;
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
@ -2428,7 +2436,7 @@ void moveMonster(monster *m, int delta) {
|
|||||||
bullet->at = m->at;
|
bullet->at = m->at;
|
||||||
bullet->ori = m->ori;
|
bullet->ori = m->ori;
|
||||||
bullet->type = moCrushball;
|
bullet->type = moCrushball;
|
||||||
bullet->parent = m;
|
bullet->set_parent(m);
|
||||||
bullet->pid = i;
|
bullet->pid = i;
|
||||||
additional.push_back(bullet);
|
additional.push_back(bullet);
|
||||||
break;
|
break;
|
||||||
@ -2730,11 +2738,9 @@ EX void turn(int delta) {
|
|||||||
// deactivate all monsters
|
// deactivate all monsters
|
||||||
for(monster *m: active)
|
for(monster *m: active)
|
||||||
if(m->dead && m->type != moPlayer) {
|
if(m->dead && m->type != moPlayer) {
|
||||||
for(monster *m2: active) if(m2->parent == m)
|
|
||||||
m2->parent = m->parent;
|
|
||||||
if(m == mousetarget) mousetarget = NULL;
|
if(m == mousetarget) mousetarget = NULL;
|
||||||
if(m == lmousetarget) lmousetarget = NULL;
|
if(m == lmousetarget) lmousetarget = NULL;
|
||||||
delete m;
|
m->remove_reference();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
m->store();
|
m->store();
|
||||||
@ -2802,7 +2808,7 @@ EX hookset<bool(const shiftmatrix&, cell*, shmup::monster*)> hooks_draw;
|
|||||||
EX void clearMonsters() {
|
EX void clearMonsters() {
|
||||||
for(mit it = monstersAt.begin(); it != monstersAt.end(); it++)
|
for(mit it = monstersAt.begin(); it != monstersAt.end(); it++)
|
||||||
delete(it->second);
|
delete(it->second);
|
||||||
for(monster *m: active) delete m;
|
for(monster *m: active) m->remove_reference();
|
||||||
mousetarget = NULL;
|
mousetarget = NULL;
|
||||||
lmousetarget = NULL;
|
lmousetarget = NULL;
|
||||||
monstersAt.clear();
|
monstersAt.clear();
|
||||||
@ -3005,12 +3011,12 @@ bool celldrawer::draw_shmup_monster() {
|
|||||||
case moBullet: {
|
case moBullet: {
|
||||||
color_t col;
|
color_t col;
|
||||||
cpid = m->pid;
|
cpid = m->pid;
|
||||||
if(m->parenttype == moPlayer)
|
if(m->get_parenttype() == moPlayer)
|
||||||
col = getcs().swordcolor;
|
col = getcs().swordcolor;
|
||||||
else if(m->parenttype == moMimic)
|
else if(m->get_parenttype() == moMimic)
|
||||||
col = (mirrorcolor(det(view.T) < 0) << 8) | 0xFF;
|
col = (mirrorcolor(det(view.T) < 0) << 8) | 0xFF;
|
||||||
else
|
else
|
||||||
col = (minf[m->parenttype].color << 8) | 0xFF;
|
col = (minf[m->get_parenttype()].color << 8) | 0xFF;
|
||||||
if(getcs().charid >= 4) {
|
if(getcs().charid >= 4) {
|
||||||
queuepoly(mmscale(view, 1.15), cgi.shPHead, col);
|
queuepoly(mmscale(view, 1.15), cgi.shPHead, col);
|
||||||
ShadowV(view, cgi.shPHead);
|
ShadowV(view, cgi.shPHead);
|
||||||
@ -3032,7 +3038,7 @@ bool celldrawer::draw_shmup_monster() {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case moTongue: {
|
case moTongue: {
|
||||||
queuepoly(mmscale(view, 1.15), cgi.shTongue, (minf[m->parenttype].color << 8) | 0xFF);
|
queuepoly(mmscale(view, 1.15), cgi.shTongue, (minf[m->get_parenttype()].color << 8) | 0xFF);
|
||||||
ShadowV(view, cgi.shTongue);
|
ShadowV(view, cgi.shTongue);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user