1
0
mirror of https://github.com/zenorogue/hyperrogue.git synced 2025-01-15 11:45:48 +00:00

festive lights

This commit is contained in:
Zeno Rogue 2024-12-22 12:52:51 +01:00
parent 155053c9fb
commit 64f68e636a
6 changed files with 127 additions and 15 deletions

View File

@ -10,6 +10,7 @@ struct celldrawer {
color_t fcol; color_t fcol;
color_t wcol; color_t wcol;
color_t wcol_star;
color_t asciicol; color_t asciicol;
color_t aura_color; color_t aura_color;
int fd; int fd;
@ -632,6 +633,20 @@ void celldrawer::setcolors() {
if(c->wall == waGlass && !wmspatial) fcol = wcol; if(c->wall == waGlass && !wmspatial) fcol = wcol;
wcol_star = wcol;
if(festive) {
if(auto *at = at_or_null(old_shines, c)) {
fcol = darkenedby(fcol, fd);
wcol = darkenedby(wcol, fd);
fd = 0;
for(int p=0; p<3; p++) {
part(fcol, p) = min(255, part(fcol, p) + (*at)[p] / 4);
part(wcol, p) = min(255, part(wcol, p) + (*at)[p] / 8);
}
}
}
if(neon_mode == eNeon::illustration) { if(neon_mode == eNeon::illustration) {
fcol = highwall(c) ? w_monochromatize(fcol, 0) : w_monochromatize(fcol, 1); fcol = highwall(c) ? w_monochromatize(fcol, 0) : w_monochromatize(fcol, 1);
wcol = w_monochromatize(wcol, 0); wcol = w_monochromatize(wcol, 0);
@ -707,7 +722,7 @@ void celldrawer::draw_wall() {
if(GDIM == 3 && WDIM == 2) { if(GDIM == 3 && WDIM == 2) {
if(!qfi.fshape) qfi.fshape = &cgi.shFullFloor; if(!qfi.fshape) qfi.fshape = &cgi.shFullFloor;
if(conegraph(c)) { if(conegraph(c)) {
draw_shapevec(c, V, qfi.fshape->cone[0], darkena(wcol, 0, 0xFF), PPR::WALL); draw_shapevec(c, V, qfi.fshape->cone[0], darkena(wcol_star, 0, 0xFF), PPR::WALL);
draw_wallshadow(); draw_wallshadow();
return; return;
} }
@ -718,11 +733,11 @@ void celldrawer::draw_wall() {
queuepolyat(V * ddspin180(c, hdir), cgi.shPalaceGate, darkena(wcol, 0, 0xFF), wmspatial?PPR::WALL3A:PPR::WALL); queuepolyat(V * ddspin180(c, hdir), cgi.shPalaceGate, darkena(wcol, 0, 0xFF), wmspatial?PPR::WALL3A:PPR::WALL);
return; return;
} }
color_t wcol0 = wcol; color_t wcol0 = wcol_star;
color_t wcol2 = gradient(0, wcol0, 0, .8, 1); color_t wcol2 = gradient(0, wcol0, 0, .8, 1);
color_t wcol1 = wcol2; color_t wcol1 = wcol2;
if(geometry == gEuclidSquare) wcol1 = gradient(0, wcol0, 0, .9, 1); if(geometry == gEuclidSquare) wcol1 = gradient(0, wcol0, 0, .9, 1);
draw_shapevec(c, V, qfi.fshape->levels[SIDE_WALL], darkena(wcol, 0, 0xFF), PPR::WALL); draw_shapevec(c, V, qfi.fshape->levels[SIDE_WALL], darkena(wcol_star, 0, 0xFF), PPR::WALL);
forCellIdEx(c2, i, c) forCellIdEx(c2, i, c)
if(!highwall(c2) || conegraph(c2) || c2->wall == waClosedGate || fake::split()) if(!highwall(c2) || conegraph(c2) || c2->wall == waClosedGate || fake::split())
placeSidewall(c, i, SIDE_WALL, V, darkena((i&1)?wcol1:wcol2, fd, 255)); placeSidewall(c, i, SIDE_WALL, V, darkena((i&1)?wcol1:wcol2, fd, 255));
@ -733,7 +748,7 @@ void celldrawer::draw_wall() {
aura_color = wcol; aura_color = wcol;
color_t wcol0 = wcol; color_t wcol0 = wcol;
int starcol = wcol; int starcol = wcol_star;
if(c->wall == waWarpGate) starcol = 0; if(c->wall == waWarpGate) starcol = 0;
if(c->wall == waVinePlant) starcol = 0x60C000; if(c->wall == waVinePlant) starcol = 0x60C000;
@ -761,7 +776,7 @@ void celldrawer::draw_wall() {
for(int z=1; z<layers; z++) { for(int z=1; z<layers; z++) {
double zg = zgrad0(0, geom3::actual_wall_height(), z, layers); double zg = zgrad0(0, geom3::actual_wall_height(), z, layers);
draw_qfi(c, xyzscale(V, zg*(layers-z)/layers, zg), draw_qfi(c, xyzscale(V, zg*(layers-z)/layers, zg),
darkena(gradient(0, wcol, -layers, z, layers), 0, 0xFF), PPR::WALL3+z-layers+2); darkena(gradient(0, wcol_star, -layers, z, layers), 0, 0xFF), PPR::WALL3+z-layers+2);
} }
floorShadow(c, V, SHADOW_WALL); floorShadow(c, V, SHADOW_WALL);
} }

View File

@ -364,7 +364,9 @@ hpcshape
hpcshape shSpaceship, shMissile, shSpaceshipBase, shSpaceshipCockpit, shSpaceshipGun, shSpaceshipEngine; hpcshape shSpaceship, shMissile, shSpaceshipBase, shSpaceshipCockpit, shSpaceshipGun, shSpaceshipEngine;
hpcshape shReserved[10]; hpcshape shChristmasLight;
hpcshape shReserved[9];
int orb_inner_ring; //< for shDisk* shapes, the number of vertices in the inner ring int orb_inner_ring; //< for shDisk* shapes, the number of vertices in the inner ring
int res1, res2; int res1, res2;

View File

@ -2957,15 +2957,10 @@ EX bool drawMonster(const shiftmatrix& Vparam, int ct, cell *c, color_t col, col
if(mapeditor::drawUserShape(Vb, mapeditor::sgMonster, c->monst, (col << 8) + 0xFF, c)) if(mapeditor::drawUserShape(Vb, mapeditor::sgMonster, c->monst, (col << 8) + 0xFF, c))
return false; return false;
if(isIvy(c) || isMutantIvy(c) || c->monst == moFriendlyIvy) if(isIvy(c) || isMutantIvy(c) || c->monst == moFriendlyIvy) {
queuepoly(Vb, cgi.shIBranch, (col << 8) + 0xFF); queuepoly(Vb, cgi.shIBranch, (col << 8) + 0xFF);
/* else if(c->monst < moTentacle && wormstyle == 0) { christmas_lights(c, Vb);
ShadowV(Vb, cgi.shTentacleX, PPR::GIANTSHADOW); }
queuepoly(at_smart_lof(Vb, cgi.ABODY), cgi.shTentacleX, 0xFF);
queuepoly(at_smart_lof(Vb, cgi.ABODY), cgi.shTentacle, (col << 8) + 0xFF);
} */
// else if(c->monst < moTentacle) {
// }
else if(c->monst == moDragonHead || c->monst == moDragonTail) { else if(c->monst == moDragonHead || c->monst == moDragonTail) {
char part = dragon::bodypart(c, dragon::findhead(c)); char part = dragon::bodypart(c, dragon::findhead(c));
@ -3331,6 +3326,86 @@ EX bool drawMonster(const shiftmatrix& Vparam, int ct, cell *c, color_t col, col
return res; return res;
} }
EX color_t christmas_color(int d, bool simple) {
array<color_t, 5> cols = { 0x10101, 0x10100, 0x000001, 0x00100, 0x10000 };
color_t v = cols[gmod(d, 5)];
v *= 0x3F;
int mode;
if(false) {
mode = gmod(ticks / 20000, 2) ? 1 : 3;
}
else {
int t = gmod(ticks, 11000);
if(t > 10000) mode = 4;
else mode = 1 + gmod(ticks / 11000, 3);
}
switch(mode) {
case 0:
break;
case 1:
if((ticks/100 - d) % 16) v = 0;
break;
case 2: {
int z = gmod(d * 500 - ticks, 8000);
if(z > 1500) v = 0;
else if(z < 500) { v /= 0x3F; v *= 0x3F * z / 500; }
else if(z > 1000) { v /= 0x3F; v *= 0x3F * (1500 - z) / 500; }
break;
}
case 3:
if((-ticks / 100 - d) % 16) v = 0;
break;
case 4:
v /= 0x3F;
v *= int(0x20 + 0x1F * (1 - cos(ticks / 500. * TAU)) / 2);
break;
}
return v;
}
EX std::unordered_map<cell*, array<int, 3>> shines, old_shines;
EX bool festive, festive_date;
EX bool festive_option = true;
EX void christmas_lights(cell *c, const shiftmatrix& Vb) {
if(!festive) return;
int lev = 0;
if(isMutantIvy(c)) lev = (c->stuntime - 1) & 15;
else {
auto c1 = c;
while(lev < 100 && c1->cpdist <= 7 && (isIvy(c1->monst) || c1->monst == moFriendlyIvy) && !among(c1->monst, moIvyRoot) && c1->mondir != NODIR) {
c1 = c1->cmove(c1->mondir); lev++;
}
}
auto cn = c->cmove(c->mondir);
ld dist = hdist0(currentmap->adj(c, c->mondir) * C0);
for(int a: {0, 1, 2}) {
color_t col = christmas_color(3*lev-a, c->monst == moMutant);
int bri = max(part(col, 0), max(part(col, 1), part(col, 2)));
color_t fcol = 0x7E + (col << 9) + 0x1010101 * int(129 * bri / 0x3F);
queuepoly(Vb * xpush(dist * (a*2+1) / 6), cgi.shChristmasLight, fcol);
for(int p: {0,1,2}) {
int val = part(col, p);
if(!val) continue;
if(a == 0) {
shines[c][p] += 2 * val;
forCellEx(c1, c) shines[c1][p] += val;
}
if(a == 2) {
shines[cn][p] += 2 * val;
forCellEx(c1, cn) shines[c1][p] += val;
}
if(a == 1) {
shines[c][p] += val;
shines[cn][p] += val;
forCellEx(c1, c) shines[c1][p] += val / 2;
forCellEx(c1, cn) shines[c1][p] += val / 2;
}
}
}
}
#define AURA 180 #define AURA 180
array<array<int,4>,AURA+1> aurac; array<array<int,4>,AURA+1> aurac;
@ -5794,6 +5869,9 @@ EX void gamescreen() {
if(history::includeHistory) history::restore(); if(history::includeHistory) history::restore();
festive = festive_date && festive_option;
old_shines = std::move(shines); shines.clear();
anims::apply(); anims::apply();
#if CAP_RUG #if CAP_RUG
if(rug::rugged) { if(rug::rugged) {

View File

@ -846,6 +846,15 @@ EX bool showHalloween() {
return false; return false;
} }
EX bool showFestive() {
time_t t = time(NULL);
struct tm tm = *localtime(&t);
int month = tm.tm_mon + 1;
int day = tm.tm_mday;
if(month == 12 && day >= 24 && day <= 26) return true;
return false;
}
int daily_mode; int daily_mode;
void announce_random() { void announce_random() {

View File

@ -1358,6 +1358,13 @@ void geometry_information::prepare_shapes() {
bshape(shWormHead, PPR::ONTENTACLE, scalefactor * wormscale, 80); bshape(shWormHead, PPR::ONTENTACLE, scalefactor * wormscale, 80);
bshape(shSmallWormHead, PPR::ONTENTACLE, scalefactor * wormscale / 2, 80); bshape(shSmallWormHead, PPR::ONTENTACLE, scalefactor * wormscale / 2, 80);
bshape(shChristmasLight, PPR::ONTENTACLE_EYES);
hpcpush(hpxy(.05 * scalefactor, 0));
hpcpush(hpxy(0, .03 * scalefactor));
hpcpush(hpxy(-.05 * scalefactor, 0));
hpcpush(hpxy(0, -.03 * scalefactor));
hpcpush(hpxy(.05 * scalefactor, 0));
bshape(shWormSegment, PPR::TENTACLE1); bshape(shWormSegment, PPR::TENTACLE1);
auto TC0 = tile_center(); auto TC0 = tile_center();
RING(i) RING(i)

View File

@ -1808,6 +1808,7 @@ EX void initAll() {
callhooks(hooks_initialize); callhooks(hooks_initialize);
init_floorcolors(); init_floorcolors();
showstartmenu = true; showstartmenu = true;
festive_date = showFestive();
ca::init(); ca::init();
#if CAP_COMMANDLINE #if CAP_COMMANDLINE
arg::read(1); arg::read(1);