mirror of
https://github.com/zenorogue/hyperrogue.git
synced 2025-01-26 00:47:00 +00:00
269 lines
7.6 KiB
C++
269 lines
7.6 KiB
C++
// Hyperbolic Rogue -- Blizzard
|
|
// Copyright (C) 2011-2019 Zeno Rogue, see 'hyper.cpp' for details
|
|
|
|
/** \file blizzard.cpp
|
|
* \brief special graphical effects, such as the Blizzard and arrow traps
|
|
*/
|
|
|
|
#include "hyper.h"
|
|
namespace hr {
|
|
|
|
struct snowball {
|
|
transmatrix T;
|
|
transmatrix global;
|
|
snowball *prev;
|
|
snowball *next;
|
|
double phase;
|
|
snowball(int t) { T = rgpushxto0(randomPointIn(t)); phase = randd(); }
|
|
};
|
|
|
|
#if HDR
|
|
struct blizzardcell;
|
|
#endif
|
|
|
|
struct blizzardcell {
|
|
cell *c;
|
|
int frame;
|
|
int tmp;
|
|
transmatrix *gm;
|
|
char wmap;
|
|
int inward, outward, ward;
|
|
int qty[MAX_EDGE];
|
|
vector<snowball*> inorder, outorder;
|
|
int inid, outid;
|
|
~blizzardcell() { for(auto i: inorder) delete i; }
|
|
};
|
|
|
|
EX map<cell*, blizzardcell> blizzardcells;
|
|
|
|
EX void set_blizzard_frame(cell *c, int frameid) {
|
|
blizzardcells[c].frame = frameid;
|
|
}
|
|
|
|
EX vector<blizzardcell*> bcells;
|
|
|
|
int blizzard_N;
|
|
|
|
EX blizzardcell* getbcell(cell *c) {
|
|
int i = c->listindex;
|
|
if(i<0 || i >= blizzard_N) return NULL;
|
|
if(bcells[i]->c != c) return NULL;
|
|
return bcells[i];
|
|
}
|
|
|
|
EX void drawBlizzards() {
|
|
#if CAP_SHAPES && CAP_FIELD
|
|
poly_outline = OUTLINE_NONE;
|
|
auto it = blizzardcells.begin();
|
|
bcells.clear();
|
|
while(it != blizzardcells.end())
|
|
if(it->second.frame != frameid || !gmatrix.count(it->first))
|
|
it = blizzardcells.erase(it);
|
|
else {
|
|
it->second.c = it->first;
|
|
bcells.push_back(&it->second);
|
|
it++;
|
|
}
|
|
blizzard_N = isize(bcells);
|
|
for(int i=0; i<blizzard_N; i++) {
|
|
auto& bc = *bcells[i];
|
|
bc.tmp = bc.c->listindex,
|
|
bc.c->listindex = i;
|
|
bc.gm = &gmatrix[bc.c];
|
|
bc.wmap = windmap::at(bc.c);
|
|
}
|
|
|
|
for(int i=0; i<blizzard_N; i++) {
|
|
auto& bc = *bcells[i];
|
|
cell *c = bc.c;
|
|
bc.inward = bc.outward = 0;
|
|
for(int i=0; i<c->type; i++) {
|
|
int& qty = bc.qty[i];
|
|
qty = 0;
|
|
cell *c2 = c->move(i);
|
|
if(!c2) continue;
|
|
auto bc2 = getbcell(c2);
|
|
if(!bc2) continue;
|
|
int z = (bc2->wmap - bc.wmap) & 255;
|
|
if(z >= windmap::NOWINDBELOW && z < windmap::NOWINDFROM)
|
|
bc.outward += qty = z / 8;
|
|
z = (-z) & 255;
|
|
if(z >= windmap::NOWINDBELOW && z < windmap::NOWINDFROM)
|
|
bc.inward += z / 8, qty = -z/8;
|
|
}
|
|
bc.ward = max(bc.inward, bc.outward);
|
|
while(isize(bc.inorder) < bc.ward) {
|
|
auto sb = new snowball(c->type);
|
|
bc.inorder.push_back(sb);
|
|
bc.outorder.push_back(sb);
|
|
}
|
|
for(auto& sb: bc.inorder) sb->prev = sb->next = NULL;
|
|
bc.inid = 0;
|
|
}
|
|
|
|
double at = fractick(40);
|
|
|
|
for(int i=0; i<blizzard_N; i++) {
|
|
auto& bc = *bcells[i];
|
|
for(auto sb: bc.inorder)
|
|
sb->global = (*bc.gm) * sb->T;
|
|
}
|
|
|
|
for(int i=0; i<blizzard_N; i++) {
|
|
auto& bc = *bcells[i];
|
|
cell *c = bc.c;
|
|
|
|
bc.outid = 0;
|
|
|
|
for(int d=0; d<c->type; d++) for(int k=0; k<bc.qty[d]; k++) {
|
|
auto& bc2 = *getbcell(c->move(d));
|
|
auto& sball = *bc.outorder[bc.outid++];
|
|
auto& sball2 = *bc2.inorder[bc2.inid++];
|
|
sball.next = &sball2;
|
|
sball2.prev = &sball;
|
|
|
|
hyperpoint t = inverse(sball.global) * tC0(sball2.global);
|
|
double at0 = at + sball.phase;
|
|
if(at0>1) at0 -= 1;
|
|
transmatrix tpartial = sball.global * rspintox(t) * xpush(hdist0(t) * at0);
|
|
|
|
if(wmascii || wmblack)
|
|
queuechr(tpartial, .2, '.', 0xFFFFFF);
|
|
else
|
|
queuepoly(tpartial, cgi.shSnowball, 0xFFFFFF80);
|
|
}
|
|
}
|
|
|
|
for(int ii=0; ii<blizzard_N; ii++) {
|
|
auto& bc = *bcells[ii];
|
|
|
|
/* if(isNeighbor(bc.c, mouseover)) {
|
|
if(againstWind(mouseover, bc.c))
|
|
queuepoly(*bc.gm, cgi.shHeptaMarker, 0x00C00040);
|
|
if(againstWind(bc.c, mouseover))
|
|
queuepoly(*bc.gm, cgi.shHeptaMarker, 0xC0000040);
|
|
} */
|
|
|
|
forCellIdEx(c2, i, bc.c) if(bc.c == mouseover || c2 == mouseover) {
|
|
color_t col = 0x00C00080;
|
|
if(c2 == mouseover)
|
|
col ^= 0xC0C00000;
|
|
if(isPlayerOn(c2))
|
|
col ^= 0x00000040;
|
|
if(isPlayerOn(bc.c))
|
|
col ^= 0x00000040;
|
|
if(againstWind(bc.c, c2))
|
|
queuepoly(*bc.gm * ddspin(bc.c, i) * xpush(cellgfxdist(bc.c, i)/2), cgi.shWindArrow, col);
|
|
}
|
|
|
|
int B = isize(bc.outorder);
|
|
if(B<2) continue;
|
|
int i = rand() % B;
|
|
int j = rand() % (B-1);
|
|
if(i==j) j++;
|
|
|
|
if(1) {
|
|
auto& sb1 = *bc.outorder[i];
|
|
auto& sb2 = *bc.outorder[j];
|
|
|
|
double swapcost = 0;
|
|
if(sb1.next) swapcost -= hdist(tC0(sb1.global), tC0(sb1.next->global));
|
|
if(sb2.next) swapcost -= hdist(tC0(sb2.global), tC0(sb2.next->global));
|
|
if(sb1.next) swapcost += hdist(tC0(sb2.global), tC0(sb1.next->global));
|
|
if(sb2.next) swapcost += hdist(tC0(sb1.global), tC0(sb2.next->global));
|
|
if(swapcost < 0) {
|
|
swap(bc.outorder[i], bc.outorder[j]);
|
|
swap(sb1.next, sb2.next);
|
|
if(sb1.next) sb1.next->prev = &sb1;
|
|
if(sb2.next) sb2.next->prev = &sb2;
|
|
}
|
|
}
|
|
|
|
if(1) {
|
|
auto& sb1 = *bc.inorder[i];
|
|
auto& sb2 = *bc.inorder[j];
|
|
|
|
double swapcost = 0;
|
|
if(sb1.prev) swapcost -= hdist(tC0(sb1.global), tC0(sb1.prev->global));
|
|
if(sb2.prev) swapcost -= hdist(tC0(sb2.global), tC0(sb2.prev->global));
|
|
if(sb1.prev) swapcost += hdist(tC0(sb2.global), tC0(sb1.prev->global));
|
|
if(sb2.prev) swapcost += hdist(tC0(sb1.global), tC0(sb2.prev->global));
|
|
if(swapcost < 0) {
|
|
swap(bc.inorder[i], bc.inorder[j]);
|
|
swap(sb1.prev, sb2.prev);
|
|
if(sb1.prev) sb1.prev->next = &sb1;
|
|
if(sb2.prev) sb2.prev->next = &sb2;
|
|
}
|
|
}
|
|
|
|
auto& sbp = *bc.inorder[i];
|
|
if(sbp.next && sbp.prev) {
|
|
double p1 = sbp.next->phase;
|
|
double p2 = sbp.prev->phase;
|
|
double d = p2-p1;
|
|
if(d<=.5) d+=1;
|
|
if(d>=.5) d-=1;
|
|
sbp.phase = p1 + d/2;
|
|
if(sbp.phase >= 1) sbp.phase -= 1;
|
|
if(sbp.phase < 0) sbp.phase += 1;
|
|
}
|
|
}
|
|
|
|
for(auto bc: bcells)
|
|
bc->c->listindex = bc->tmp;
|
|
#endif
|
|
}
|
|
|
|
EX vector<cell*> arrowtraps;
|
|
|
|
EX void drawArrowTraps() {
|
|
for(cell *c: arrowtraps) {
|
|
auto r = traplimits(c);
|
|
|
|
try {
|
|
transmatrix& t0 = gmatrix.at(r[0]);
|
|
transmatrix& t1 = gmatrix.at(r[4]);
|
|
ignore(t0);
|
|
ignore(t1);
|
|
|
|
#if CAP_QUEUE
|
|
queueline(tC0(t0), tC0(t1), 0xFF0000FF, 4 + vid.linequality, PPR::ITEM);
|
|
#endif
|
|
#if CAP_SHAPES
|
|
if((c->wparam & 7) == 3 && !shmup::on) {
|
|
// queueline(t0 * randomPointIn(r[0]->type), t1 * randomPointIn(r[1]->type), 0xFFFFFFFF, 4, PPR::ITEM);
|
|
int tt = int(fractick(64) * 401);
|
|
|
|
for(int u=0; u<2; u++) {
|
|
transmatrix& tu = u ? t0 : t1;
|
|
transmatrix& tv = u ? t1 : t0;
|
|
hyperpoint trel = inverse(tu) * tC0(tv);
|
|
transmatrix tpartial = tu * rspintox(trel) * xpush(hdist0(trel) * tt / 401.0);
|
|
tpartial = tpartial * ypush(.05);
|
|
if(GDIM == 3) tpartial = tpartial * cspin(1, 2, M_PI/2);
|
|
queuepoly(tpartial, cgi.shTrapArrow, 0xFFFFFFFF);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
catch(out_of_range&) {}
|
|
}
|
|
}
|
|
|
|
auto ccm_blizzard = addHook(clearmemory, 0, [] () {
|
|
arrowtraps.clear();
|
|
blizzardcells.clear();
|
|
bcells.clear();
|
|
}) +
|
|
+ addHook(hooks_gamedata, 0, [] (gamedata* gd) {
|
|
gd->store(arrowtraps);
|
|
gd->store(blizzardcells);
|
|
gd->store(bcells);
|
|
gd->store(blizzard_N);
|
|
})
|
|
+ addHook(hooks_removecells, 0, [] () {
|
|
eliminate_if(arrowtraps, is_cell_removed);
|
|
});
|
|
|
|
}
|