2019-08-10 11:43:24 +00:00
// Hyperbolic Rogue - Complex features part II
2019-01-02 21:01:00 +00:00
// Copyright (C) 2011-2019 Zeno Rogue, see 'hyper.cpp' for details
2018-10-25 00:43:14 +00:00
2019-08-10 11:43:24 +00:00
/** \file complex2.cpp
* \ brief Continuation of complex . cpp
*
* Includes : Brownian , Irradiated , Free Fall
*/
2019-09-05 07:15:40 +00:00
# include "hyper.h"
2018-10-25 00:43:14 +00:00
namespace hr {
2022-05-06 10:40:48 +00:00
# if CAP_COMPLEX2
2019-08-09 20:37:11 +00:00
EX namespace brownian {
# if HDR
const int level = 5 ;
# endif
2018-10-25 00:43:14 +00:00
2019-01-20 11:46:05 +00:00
map < cell * , vector < pair < cell * , int > > > futures ;
2018-10-25 00:43:14 +00:00
int centersteps = 0 ;
int totalsteps = 0 ;
void rise ( cell * c , int val ) {
if ( c - > wall = = waSea ) c - > wall = waNone ;
if ( c - > land = = laOcean | | c - > land = = laNone ) {
c - > land = laBrownian ;
c - > landparam = 0 ;
}
c - > bardir = NOBARRIERS ;
forCellCM ( c1 , c ) {
c1 - > bardir = NOBARRIERS ;
if ( c1 - > mpdist > BARLEV ) {
setdist ( c1 , BARLEV , c ) ;
}
if ( c1 - > land = = laOcean ) {
c1 - > land = laBrownian ;
c1 - > landparam = 0 ;
2019-01-03 16:19:23 +00:00
c1 - > wall = waSea ;
2018-10-25 00:43:14 +00:00
}
}
c - > landparam + = val ;
}
2019-01-20 11:46:05 +00:00
2023-08-23 17:44:37 +00:00
static constexpr int FAT = ( - 100 ) ; // less than 0
2018-10-25 00:43:14 +00:00
2019-01-20 11:46:05 +00:00
void recurse ( cell * c , int fatten_from ) {
2019-01-03 01:00:15 +00:00
int dl = getDistLimit ( ) ;
2018-10-25 00:43:14 +00:00
while ( true ) {
2019-01-20 11:46:05 +00:00
int cd = celldist ( c ) ;
bool fat = cd > fatten_from ;
2018-10-25 00:43:14 +00:00
totalsteps + + ;
2019-01-20 11:46:05 +00:00
if ( cd > = dl * ( fat ? 4 : ISMOBILE ? 2 : 3 ) + celldist ( cwt . at ) ) {
2018-10-25 00:43:14 +00:00
cell * c1 = c ;
while ( true ) {
cell * c2 = ts : : left_parent ( c1 , celldist ) ;
if ( ! c2 | | c2 - > mpdist < BARLEV ) break ;
setdist ( c2 , BARLEV , c1 ) ;
c1 = c2 ;
}
2019-01-20 11:46:05 +00:00
futures [ c1 ] . emplace_back ( c , fatten_from ) ;
2018-10-25 00:43:14 +00:00
return ;
}
2019-01-03 16:19:23 +00:00
if ( c - > mpdist < = 7 | | ! among ( c - > land , laNone , laOcean , laBrownian ) | | ( c - > land ! = laBrownian & & c - > bardir ! = NODIR ) ) {
centersteps + + ; return ;
}
cell * c2 = c - > cmove ( hrand ( c - > type ) ) ;
2019-01-20 11:46:05 +00:00
int cd2 = celldist ( c2 ) ;
2018-10-25 00:43:14 +00:00
// while(hrand(1000) < 1000 * chance) recurse(c);
2019-01-20 11:46:05 +00:00
if ( ! fat & & ( cd2 > fatten_from | | hrand ( 100000 ) = = 0 ) ) {
recurse ( c , FAT ) ;
fatten_from = FAT ;
2019-01-02 21:01:00 +00:00
}
2019-01-20 11:46:05 +00:00
else if ( fat ) recurse ( c , cd + dl * 6 ) ;
2018-10-25 00:43:14 +00:00
rise ( c , fat ? 256 : 1 ) ;
2019-01-03 16:19:23 +00:00
c = c2 ;
2018-10-25 00:43:14 +00:00
}
}
2019-09-05 10:00:55 +00:00
EX void dissolve_brownian ( cell * c , int x ) {
2018-10-25 00:43:14 +00:00
if ( c - > land = = laBrownian ) {
if ( among ( c - > wall , waNone , waStrandedBoat , waMineOpen , waFire ) ) {
if ( c - > landparam > = 4 * level ) c - > landparam = 4 * level - 1 ;
c - > landparam - = level * x ;
c - > wall = waNone ;
if ( c - > landparam < 0 ) c - > wall = waSea , c - > landparam = 0 ;
if ( c - > landparam = = 0 ) c - > landparam = 1 ;
}
}
}
2019-09-05 10:00:55 +00:00
EX void dissolve ( cell * c , int x ) {
2018-10-25 00:43:14 +00:00
destroyTrapsAround ( c ) ;
if ( c - > land = = laBrownian )
dissolve_brownian ( c , x ) ;
else if ( c - > wall = = waRed2 ) c - > wall = waRed1 ;
else if ( c - > wall = = waRed3 ) c - > wall = waRed2 ;
else if ( among ( c - > wall , waRed1 , waDeadfloor2 , waRubble , waBoat , waFire , waCIsland , waCIsland2 , waBigBush , waSmallBush ) ) c - > wall = waNone ;
else if ( c - > wall = = waStrandedBoat ) c - > wall = waNone ;
else if ( c - > wall = = waFrozenLake ) c - > wall = waLake ;
else if ( among ( c - > wall , waReptile , waGargoyleFloor ) | | cellUnstable ( c ) ) c - > wall = waChasm ;
else if ( among ( c - > wall , waNone , waDock , waBurningDock , waFloorA , waFloorB , waCavefloor , waDeadfloor , waMineMine , waMineUnknown , waMineOpen , waOpenGate , waClosePlate , waOpenPlate , waGargoyleBridge , waReptileBridge ) )
c - > wall = waSea ;
else if ( cellHalfvine ( c ) ) destroyHalfvine ( c , waNone , 4 ) ;
}
2019-08-09 20:37:11 +00:00
EX void init ( cell * c ) {
2019-01-24 13:49:50 +00:00
if ( ! hyperbolic ) return ;
2019-01-20 11:46:05 +00:00
recurse ( c , FAT ) ;
recurse ( c , FAT ) ;
2019-01-02 21:01:00 +00:00
}
2019-09-06 06:17:02 +00:00
EX void init_further ( cell * c ) {
2019-01-24 13:49:50 +00:00
if ( ! hyperbolic ) return ;
2019-01-20 11:46:05 +00:00
int dl = getDistLimit ( ) ;
2019-01-02 21:01:00 +00:00
dynamicval < bool > be ( generatingEquidistant , true ) ;
int gdir = - 1 ;
for ( int i = 0 ; i < c - > type ; i + + ) {
if ( c - > move ( i ) & & c - > move ( i ) - > mpdist < c - > mpdist ) gdir = i ;
}
if ( gdir < 0 ) return ;
cellwalker cw ( c , gdir ) ;
for ( int i = 0 ; i < 4 ; i + + ) {
cw + = revstep ;
setdist ( cw . at , BARLEV , cw . peek ( ) ) ;
buildEquidistant ( cw . at ) ;
println ( hlog , " from " , cw . peek ( ) , " to " , cw . at , " , land = " , dnameof ( cw . at - > land ) , " lp = " , cw . at - > landparam ) ;
}
if ( c - > land ! = laOcean | | ! no_barriers_in_radius ( cw . at , 2 ) ) return ;
println ( hlog , " brownian::init " , cw . at , " in distance " , celldistance ( cw . at , cwt . at ) ) ;
2019-01-20 11:46:05 +00:00
recurse ( cw . at , celldist ( c ) + dl * 3 ) ;
recurse ( cw . at , celldist ( c ) + dl * 3 ) ;
2019-01-02 21:01:00 +00:00
cell * c2 = c ;
while ( c2 - > mpdist > 7 ) {
forCellEx ( c3 , c2 ) if ( c3 - > mpdist < c2 - > mpdist ) { c2 = c3 ; goto next ; }
break ;
next : ;
}
if ( ! c2 - > monst & & c2 - > wall ! = waBoat ) c2 - > monst = moAcidBird ;
2018-10-25 00:43:14 +00:00
}
2019-01-03 16:19:23 +00:00
2019-09-05 10:00:55 +00:00
EX void apply_futures ( cell * c ) {
2018-10-25 00:43:14 +00:00
if ( futures . count ( c ) ) {
2022-07-05 17:23:05 +00:00
auto m = std : : move ( futures [ c ] ) ;
2019-01-03 16:19:23 +00:00
futures . erase ( c ) ;
2019-01-20 11:46:05 +00:00
for ( auto p : m )
2018-10-25 00:43:14 +00:00
recurse ( p . first , p . second ) ;
futures . erase ( c ) ;
printf ( " centersteps = %d futures = %d totalsteps = %d \n " , centersteps , isize ( futures ) , totalsteps ) ;
}
2019-01-03 16:19:23 +00:00
}
2019-08-09 20:37:11 +00:00
EX void build ( cell * c , int d ) {
2019-01-24 13:49:50 +00:00
if ( ! hyperbolic ) c - > wall = waNone , c - > landparam = 256 ;
2021-04-23 18:02:20 +00:00
2023-09-18 10:12:13 +00:00
if ( c - > landparam = = 0 & & ( ls : : single ( ) | | ls : : hv_structure ( ) ) ) c - > land = laOcean ;
2018-10-25 00:43:14 +00:00
ONEMPTY {
2019-01-02 21:01:00 +00:00
if ( hrand ( 10000 ) < min ( 250 , 100 + 2 * PT ( kills [ moAcidBird ] + kills [ moBrownBug ] , 50 ) ) * ( 25 + min ( items [ itBrownian ] , 100 ) ) / 25 & & c - > landparam > = 4 & & c - > landparam < 24 )
2018-10-25 00:43:14 +00:00
c - > item = itBrownian ;
2024-03-14 18:27:08 +00:00
if ( hrand_monster_in ( laBrownian , 8000 ) < 15 + items [ itBrownian ] )
2018-10-25 00:43:14 +00:00
c - > monst = moAcidBird ;
2024-03-14 18:27:08 +00:00
else if ( hrand_monster_in ( laBrownian , 8000 ) < 15 )
2018-10-25 00:43:14 +00:00
c - > monst = moAlbatross ;
2024-03-14 18:27:08 +00:00
else if ( hrand_monster_in ( laBrownian , 8000 ) < 15 + items [ itBrownian ] ) {
2018-10-25 00:43:14 +00:00
c - > monst = moBrownBug ;
c - > hitpoints = 3 ;
}
}
}
2018-12-25 11:49:54 +00:00
2019-09-05 10:00:55 +00:00
EX colortable colors = { 0x603000 , 0x804000 , 0xA05000 , 0xC09050 , 0xE0D0A0 } ;
2019-01-03 16:21:11 +00:00
2019-09-06 06:17:02 +00:00
EX color_t get_color ( int y ) {
2018-12-25 11:49:54 +00:00
return
2019-01-03 16:21:11 +00:00
y < level ? gradient ( colors [ 0 ] , colors [ 1 ] , 1 , y , level - 1 ) :
y < 2 * level ? colors [ 2 ] :
y < 3 * level ? colors [ 3 ] :
colors [ 4 ] ;
}
2019-09-05 10:00:55 +00:00
EX color_t & get_color_edit ( int y ) {
2019-01-03 16:21:11 +00:00
return
y < level / 2 ? colors [ 0 ] :
y < level ? colors [ 1 ] :
y < 2 * level ? colors [ 2 ] :
y < 3 * level ? colors [ 3 ] :
colors [ 4 ] ;
2018-12-25 11:49:54 +00:00
}
2018-10-25 00:43:14 +00:00
2019-01-11 01:23:00 +00:00
int hrc = addHook ( hooks_removecells , 0 , [ ] ( ) {
vector < cell * > to_remove ;
for ( auto p : futures ) if ( is_cell_removed ( p . first ) ) to_remove . push_back ( p . first ) ;
for ( auto r : to_remove ) futures . erase ( r ) ;
2020-04-11 18:40:12 +00:00
} ) + addHook ( hooks_clearmemory , 0 , [ ] ( ) { futures . clear ( ) ; } )
2019-05-30 14:12:38 +00:00
+ addHook ( hooks_gamedata , 0 , [ ] ( gamedata * gd ) { gd - > store ( futures ) ; } ) ;
2019-01-11 01:23:00 +00:00
2019-09-05 10:00:55 +00:00
EX }
2018-10-25 00:43:14 +00:00
2019-09-05 10:00:55 +00:00
EX namespace westwall {
2018-12-21 13:41:23 +00:00
2019-09-05 10:00:55 +00:00
EX void switchTreasure ( cell * c ) {
2018-12-21 13:41:23 +00:00
c - > item = itNone ;
if ( safety ) return ;
2019-01-24 13:49:25 +00:00
if ( hrand ( 5000 ) < PT ( 100 + 2 * ( kills [ moAirElemental ] + kills [ moWindCrow ] ) , 200 ) & & c - > landparam > = 5 + items [ itWest ] )
2018-12-21 13:41:23 +00:00
c - > item = itWest ;
else if ( hrand ( 5000 ) < 20 * PRIZEMUL )
placeLocalOrbs ( c ) ;
}
2019-09-06 06:17:02 +00:00
EX int coastvalEdge1 ( cell * c ) {
2018-12-22 21:39:16 +00:00
if ( c - > land = = laWestWall & & ! c - > landparam ) buildEquidistant ( c ) ;
return coastvalEdge ( c ) ;
}
2018-12-21 13:41:23 +00:00
void build ( vector < cell * > & whirlline , int d ) {
again :
cell * at = whirlline [ isize ( whirlline ) - 1 ] ;
cell * prev = whirlline [ isize ( whirlline ) - 2 ] ;
2021-08-08 17:13:09 +00:00
if ( looped ( whirlline ) ) return ;
2018-12-21 13:41:23 +00:00
for ( int i = 0 ; i < at - > type ; i + + )
2018-12-22 21:39:16 +00:00
if ( at - > move ( i ) & & coastvalEdge1 ( at - > move ( i ) ) = = d & & at - > move ( i ) ! = prev ) {
2018-12-21 13:41:23 +00:00
whirlline . push_back ( at - > move ( i ) ) ;
goto again ;
}
}
void moveAt ( cell * c , manual_celllister & cl ) {
if ( cl . listed ( c ) ) return ;
if ( c - > land ! = laWestWall ) return ;
vector < cell * > whirlline ;
int d = coastvalEdge ( c ) ;
whirlline . push_back ( c ) ;
2018-12-23 02:14:48 +00:00
whirlline . push_back ( gravity_state = = gsAnti ? ts : : right_of ( c , coastvalEdge1 ) : ts : : left_of ( c , coastvalEdge1 ) ) ;
2018-12-21 13:41:23 +00:00
build ( whirlline , d ) ;
reverse ( whirlline . begin ( ) , whirlline . end ( ) ) ;
build ( whirlline , d ) ;
int z = isize ( whirlline ) ;
for ( int i = 0 ; i < z ; i + + ) {
cl . add ( whirlline [ i ] ) ;
if ( whirlline [ i ] - > mpdist = = BARLEV )
switchTreasure ( whirlline [ i ] ) ;
}
for ( int i = 0 ; i < z - 1 ; i + + ) {
moveItem ( whirlline [ i ] , whirlline [ i + 1 ] , true ) ;
if ( whirlline [ i ] - > item )
2019-11-22 17:48:51 +00:00
animateMovement ( match ( whirlline [ i + 1 ] , whirlline [ i ] ) , LAYER_BOAT ) ;
2018-12-21 13:41:23 +00:00
}
for ( int i = 0 ; i < z ; i + + )
2021-08-08 17:04:58 +00:00
pickupMovedItems ( whirlline [ i ] , i < z - 1 ? whirlline [ i + 1 ] : whirlline [ 0 ] ) ;
2018-12-21 13:41:23 +00:00
}
2019-09-05 10:00:55 +00:00
EX void move ( ) {
2018-12-21 13:41:23 +00:00
manual_celllister cl ;
2018-12-23 02:14:48 +00:00
if ( gravity_state = = gsLevitation ) return ;
2018-12-21 13:41:23 +00:00
for ( cell * c : dcal ) moveAt ( c , cl ) ;
// Keys and Orbs of Yendor always move
using namespace yendor ;
for ( int i = 0 ; i < isize ( yi ) ; i + + ) {
moveAt ( yi [ i ] . path [ 0 ] , cl ) ;
2018-12-22 21:39:16 +00:00
// println(hlog, "coastval of actual key is ", coastvalEdge1(yi[i].actual_key()), " and item is ", dnameof(yi[i].actual_key()->item), "and mpdist is ", yi[i].actual_key()->mpdist);
moveAt ( yi [ i ] . actual_key ( ) , cl ) ;
if ( yi [ i ] . actualKey ) {
2018-12-23 02:14:48 +00:00
if ( gravity_state = = gsAnti ) yi [ i ] . age - - ;
else yi [ i ] . age + + ;
2018-12-22 21:39:16 +00:00
setdist ( yi [ i ] . actual_key ( ) , 8 , NULL ) ;
}
2018-12-21 13:41:23 +00:00
}
}
2019-09-05 10:00:55 +00:00
EX }
2018-12-21 13:41:23 +00:00
2019-12-08 18:17:28 +00:00
EX namespace variant {
2019-09-05 10:00:55 +00:00
# if HDR
2019-12-08 18:17:28 +00:00
struct feature {
2018-12-25 11:51:25 +00:00
color_t color_change ;
int rate_change ;
eMonster wanderer ;
void ( * build ) ( cell * ) ;
} ;
2019-12-08 18:17:28 +00:00
extern array < feature , 21 > features ;
2019-09-05 10:00:55 +00:00
# endif
2018-12-25 11:51:25 +00:00
# define VF [] (cell *c)
2024-03-14 18:27:08 +00:00
bool hrand_var ( int i ) { return hrand_monster_in ( laVariant , i ) < 25 + items [ itVarTreasure ] + yendor : : hardness ( ) ; }
2020-02-26 00:23:42 +00:00
2019-12-08 18:17:28 +00:00
array < feature , 21 > features { {
feature { ( color_t ) ( - 0x202020 ) , 5 , moNecromancer , VF {
2018-12-25 22:52:54 +00:00
if ( c - > wall = = waNone & & hrand ( 1500 ) < 20 ) c - > wall = waFreshGrave ;
2018-12-25 11:51:25 +00:00
if ( hrand ( 20000 ) < 10 + items [ itVarTreasure ] )
c - > monst = moNecromancer ;
} } ,
2020-02-26 00:23:42 +00:00
{ 0x000010 , 5 , moLancer , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 80000 ) ) c - > monst = moLancer ; } } ,
{ 0x100008 , 15 , moMonk , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 80000 ) ) c - > monst = moMonk ; } } ,
{ 0x080010 , 5 , moCrusher , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 80000 ) ) c - > monst = moCrusher ; } } ,
{ 0x181418 , 5 , moSkeleton , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 80000 ) ) c - > monst = moSkeleton , c - > hitpoints = 3 ; } } ,
{ 0x180000 , 5 , moPyroCultist , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 80000 ) ) c - > monst = moPyroCultist ; } } ,
{ 0x00000C , 2 , moFlailer , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 80000 ) ) c - > monst = moFlailer ; } } ,
{ 0x1C0700 , 1 , moHedge , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 80000 ) & & valence ( ) = = 3 ) c - > monst = moHedge ; } } ,
2019-01-02 15:19:22 +00:00
{ 0x000c00 , - 1 , moNone , VF { if ( hrand ( 1500 ) < 30 ) createArrowTrapAt ( c , laVariant ) ; } } ,
{ 0x001200 , - 1 , moNone , VF { if ( hrand ( 1500 ) < 50 & & c - > wall = = waNone ) c - > wall = waTrapdoor ; } } ,
{ 0x000c18 , - 1 , moNone , VF { if ( hrand ( 1500 ) < 30 ) build_pool ( c , true ) ; } } ,
{ 0x040A00 , - 1 , moNone , VF { if ( c - > wall = = waNone & & ! c - > monst & & ! c - > monst & & hrand ( 1500 ) < 10 ) c - > wall = waThumperOff ; } } ,
{ 0x080A00 , - 1 , moNone , VF { if ( hrand ( 1500 ) < 20 & & ! c - > monst & & ! c - > wall ) c - > wall = waFireTrap ; } } ,
2019-01-16 23:45:36 +00:00
{ 0x0C0A00 , 0 , moNone , VF {
bool inyendor = yendor : : on & & specialland = = laVariant & & celldist ( c ) < 7 ;
int chance = inyendor ? 800 : 100 ;
if ( c - > wall = = waNone & & ! c - > monst & & hrand ( 5000 ) < chance ) c - > wall = waExplosiveBarrel ;
} } ,
2019-01-02 15:19:22 +00:00
{ 0x060D04 , 0 , moNone , VF {
2018-12-25 11:51:25 +00:00
if ( c - > wall = = waNone & & ! c - > monst & & pseudohept ( c ) & & hrand ( 30000 ) < 25 + items [ itVarTreasure ] )
if ( buildIvy ( c , 0 , c - > type ) & & ! peace : : on ) c - > item = itVarTreasure ;
} } ,
2019-01-02 15:19:22 +00:00
{ 0x000A08 , 0 , moNone , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand ( 5000 ) < 100 ) c - > wall = waSmallTree ; } } ,
2021-07-10 04:30:13 +00:00
{ 0x100A10 , 1 , moRagingBull , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand ( 10000 ) < 10 + items [ itVarTreasure ] ) c - > monst = moSleepBull , c - > hitpoints = 3 ; } } ,
2019-01-02 15:19:22 +00:00
{ 0x00110C , 0 , moNone , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand ( 5000 ) < 100 ) c - > wall = waBigTree ; } } ,
{ 0x000A28 , 1 , moNone , VF { if ( hrand ( 500 ) < 10 ) build_pool ( c , false ) ; } } ,
2020-02-26 00:23:42 +00:00
{ 0x100A00 , 2 , moVariantWarrior , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 40000 ) ) c - > monst = moVariantWarrior ; } } ,
{ 0x100708 , 1 , moRatling , VF { if ( c - > wall = = waNone & & ! c - > monst & & hrand_var ( 50000 ) ) c - > monst = moRatling ; } }
2018-12-25 11:51:25 +00:00
} } ;
# undef VF
2019-12-08 18:17:28 +00:00
EX }
EX namespace camelot {
/** number of Grails collected, to show you as a knight */
EX int knighted = 0 ;
/** this value is used when using Orb of Safety in the Camelot in Pure Tactics Mode */
EX int anthraxBonus = 0 ;
2020-03-06 14:17:25 +00:00
vector < string > knight_names = {
2021-04-15 14:12:45 +00:00
" DivisionByZero " , " Phoenyx " , " tricosahedron " , " BillSYZ0317 " , " rjdimo " , " Person " , " Strange Yeah " , " paradoxica " , " godamnsteemsux " , " Allalinor " , " Spuddnik " ,
" QETC Crimson " , " aca " , " cannobeens " , " Dylgear " , " Patashu " , " SirLight_XXVII " , " Lord Ignus " , " Lotho " , " hotdogPi " , " vincent " , " pofoss " , " Zekava " , " Chimera245 " ,
" BreedPineapple " , " TaeK " , " Aliased " , " Vipul " , " jkvw3 " , " english5040 " , " gregwar69 " , " marvincast " , " Panacea108 " , " FEDPOL1 " , " Jenkar " , " inakilbss " , " *****-mail " ,
" Wheat Wizard " , " xiqnyl " , " zelda0x181e " , " ad-jones " , " mtomato " , " Fififionek " , " J Pystynen " , " krstefan11 " , " green orange " , " ZombieGirl1616 " , " z.chlebicki " ,
" 9427 " , " Lokki " , " gabbalis " , " The Horrendous Space Kablooie " , " zeno " , " ETsc2 " , " Fulgur14 " , " Roger " , " uqblf " , " Grep " , " Abigail " , " Cyberfox VII " , " ballom " , " CtrlAltDestroy " ,
" RushSecond " , " ozir " , " Vee Nought " , " archmageomega " , " Wroclaw " , " lunaduskjr " , " LucLucLuc " , " loper " , " RedsToad " , " Sharklilly " , " Dasypus Daimon " , " Mateusz " , " joshua " , " guysige " ,
" MrSprucetree " , " Atia " , " nerdyjoe " , " florrat " , " bderoo121 " , " Sprite Guard " , " fischpferd " , " Fluffiest Princess " , " no stop " , " Kieroshark " , " elgan65536 " , " T-Jski " ,
" martinus de monte " , " Inthar " , " mj.titze " , " thelast19digitsofpi " , " abacussssss " , " ann_dec " , " juhdanad " , " lllllllllwith10ls " , " rsstein " , " The Emerald Derp Leader " ,
" NattieGilgamesh " , " chocokels " , " oren " , " y " , " Nathanbananas " , " juliuskwan " , " wojtekor9 " , " CoderBro " , " RidiculousPyro64 " , " pringle " , " Spicy " , " Cheetahs " , " ShiningKatana " ,
" xy2 " , " starchewer " , " 2jjy " , " Nibaw " , " Goon in Chief " , " the.goosh222 " , " Toras " , " ooshoo " , " TravelDemon " , " The Big Extent " , " droplet739 " , " AlliterativeAnchovies " ,
" fones4jenke13 " , " ekisacik " , " j0eymcgrath " , " EatChangmyeong " , " Strichcoder " , " jwhirpl " , " craus " , " akei_arkay " , " lilypad " , " Ichigo Jam " , " supernewton " , " Neapolitan Sixth " ,
" Nemo " , " Westville " , " Huja " , " lapinozz " , " Just A Witch " , " Borador " , " Particles " , " dewhi100 " , " nat.mayer " , " ddtizm " , " Tism Jones " , " amagikuser " , " vkozulya " , " gassa " ,
" Factitious " , " wonderfullizardofoz " , " woofmao " , " CandorVelexion " , " Toricon " , " Vectis99 " , " RobotNerd277 " , " jerrypoons " , " MagmaMcFry " , " unczane " , " glass " , " Wegener " ,
" JeLomun " , " kip " , " Warmonger " , " Fooruman " , " Zyalin " , " Prezombie " , " ashley89 " , " bjobae " , " MFErtre " , " Roaringdragon2 " , " howilovepi " , " Yulgash " , " Sir Endipitous " , " Roshlev " ,
" BTernaryTau " , " HiGuy " , " coper " , " Tirear " , " qoala _ " , " Tyzone " , " Tiegon " , " Airin " , " Metroid26 " , " Sklorg " , " Fumblestealth " , " Toph " , " Tzaphqiel " , " jruderman " , " ray " ,
" Deathroll " , " Sinquetica " , " mootmoot " , " Noobinator " , " freeofme " , " Helyea " , " Snakebird Priestess " , " brisingre " , " Khashishi " , " Shiny " , " kabado " , " Berenthas " , " Misery " , " Altripp " , " Aldrenean " ,
2020-03-06 14:17:25 +00:00
} ;
map < cell * , int > knight_id ;
EX string knight_name ( cell * c ) {
if ( ! knight_id . count ( c ) ) knight_id [ c ] = rand ( ) % isize ( knight_names ) ;
return knight_names [ knight_id [ c ] ] ;
}
EX void move_knight ( cell * c1 , cell * c2 ) {
LATE ( move_knight ( c1 , c2 ) ; )
if ( knight_id . count ( c1 ) ) {
knight_id [ c2 ] = knight_id [ c1 ] ;
knight_id . erase ( c1 ) ;
}
}
2019-12-08 18:17:28 +00:00
EX void roundTableMessage ( cell * c2 ) {
if ( ! euclid & & ! cwt . at - > master - > alt ) return ;
if ( ! euclid & & ! c2 - > master - > alt ) return ;
int dd = celldistAltRelative ( c2 ) - celldistAltRelative ( cwt . at ) ;
bool tooeasy = ( roundTableRadius ( c2 ) < newRoundTableRadius ( ) ) ;
if ( dd > 0 ) {
if ( grailWasFound ( cwt . at ) ) {
addMessage ( XLAT ( " The Knights congratulate you on your success! " ) ) ;
knighted = roundTableRadius ( cwt . at ) ;
}
else if ( ! tooeasy )
addMessage ( XLAT ( " The Knights laugh at your failure! " ) ) ;
}
else {
if ( grailWasFound ( cwt . at ) )
addMessage ( XLAT ( " The Knights stare at you! " ) ) ;
else if ( tooeasy ) {
if ( ! tactic : : on )
addMessage ( XLAT ( " Come on, this is too easy... find a bigger castle! " ) ) ;
}
else
addMessage ( XLAT ( " The Knights wish you luck! " ) ) ;
}
}
EX void knightFlavorMessage ( cell * c2 ) {
if ( ! eubinary & & ! c2 - > master - > alt ) {
addMessage ( XLAT ( " \" I am lost... \" " ) ) ;
return ;
}
if ( tactic : : on ) {
addMessage ( XLAT ( " \" The Knights of the Horocyclic Table salute you! \" " ) ) ;
return ;
}
bool grailfound = grailWasFound ( c2 ) ;
int rad = roundTableRadius ( c2 ) ;
bool tooeasy = ( rad < newRoundTableRadius ( ) ) ;
static int msgid = 0 ;
2020-02-29 16:58:59 +00:00
changes . value_keep ( msgid ) ;
2019-12-08 18:17:28 +00:00
retry :
if ( msgid > = 32 ) msgid = 0 ;
if ( msgid = = 0 & & grailfound ) {
addMessage ( XLAT ( " \" I would like to congratulate you again! \" " ) ) ;
}
else if ( msgid = = 1 & & ! tooeasy ) {
addMessage ( XLAT ( " \" Find the Holy Grail to become one of us! \" " ) ) ;
}
else if ( msgid = = 2 & & ! tooeasy ) {
addMessage ( XLAT ( " \" The Holy Grail is in the center of the Round Table. \" " ) ) ;
}
# if CAP_CRYSTAL
else if ( msgid = = 3 & & cryst ) {
if ( crystal : : pure ( ) )
addMessage ( XLAT ( " \" Each piece of the Round Table is exactly %1 steps away from the Holy Grail. \" " , its ( roundTableRadius ( c2 ) ) ) ) ;
else
addMessage ( XLAT ( " \" According to Merlin, the Round Table is a perfect Euclidean sphere in %1 dimensions. \" " , its ( ginf [ gCrystal ] . sides / 2 ) ) ) ;
}
# endif
else if ( msgid = = 3 & & ! peace : : on & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" I enjoy watching the hyperbug battles. \" " ) ) ;
}
else if ( msgid = = 4 & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" Have you visited a temple in R'Lyeh? \" " ) ) ;
}
else if ( msgid = = 5 & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" Nice castle, eh? \" " ) ) ;
}
else if ( msgid = = 6 & & items [ itSpice ] < 10 & & ! peace : : on & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" The Red Rock Valley is dangerous, but beautiful. \" " ) ) ;
}
else if ( msgid = = 7 & & items [ itSpice ] < 10 & & ! peace : : on & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" Train in the Desert first! \" " ) ) ;
}
else if ( msgid = = 8 & & sizes_known ( ) & & ! tactic : : on ) {
string s = " " ;
if ( 0 ) ;
# if CAP_CRYSTAL
else if ( cryst )
s = crystal : : get_table_boundary ( ) ;
# endif
2021-07-10 04:33:25 +00:00
else if ( ! quotient & & rad )
2021-12-11 22:28:05 +00:00
s = get_expansion ( ) . get_descendants ( rad ) . get_str ( 100 ) ;
2019-12-08 18:17:28 +00:00
if ( s = = " " ) { msgid + + ; goto retry ; }
addMessage ( XLAT ( " \" Our Table seats %1 Knights! \" " , s ) ) ;
}
else if ( msgid = = 9 & & sizes_known ( ) & & ! tactic : : on ) {
string s = " " ;
if ( 0 ) ;
# if CAP_CRYSTAL
else if ( cryst )
s = crystal : : get_table_volume ( ) ;
# endif
2021-07-10 04:33:25 +00:00
else if ( ! quotient & & rad )
2021-12-11 22:28:05 +00:00
s = get_expansion ( ) . get_descendants ( rad - 1 , get_expansion ( ) . diskid ) . get_str ( 100 ) ;
2019-12-08 18:17:28 +00:00
if ( s = = " " ) { msgid + + ; goto retry ; }
addMessage ( XLAT ( " \" There are %1 floor tiles inside our Table! \" " , s ) ) ;
}
else if ( msgid = = 10 & & ! items [ itPirate ] & & ! items [ itWhirlpool ] & & ! peace : : on & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" Have you tried to take a boat and go into the Ocean? Try it! \" " ) ) ;
}
else if ( msgid = = 11 & & ! princess : : saved & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" When I visited the Palace, a mouse wanted me to go somewhere. \" " ) ) ;
}
else if ( msgid = = 12 & & ! princess : : saved & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" I wonder what was there... \" " ) ) ;
}
else if ( msgid = = 13 & & ! peace : : on & & in_full_game ( ) ) {
addMessage ( XLAT ( " \" Be careful in the Rose Garden! It is beautiful, but very dangerous! \" " ) ) ;
}
else if ( msgid = = 14 ) {
addMessage ( XLAT ( " \" There is no royal road to geometry. \" " ) ) ;
}
else if ( msgid = = 15 ) {
addMessage ( XLAT ( " \" There is no branch of mathematics, however abstract, " ) ) ;
addMessage ( XLAT ( " which may not some day be applied to phenomena of the real world. \" " ) ) ;
}
else if ( msgid = = 16 ) {
addMessage ( XLAT ( " \" It is not possession but the act of getting there, " ) ) ;
addMessage ( XLAT ( " which grants the greatest enjoyment. \" " ) ) ;
}
else if ( msgid = = 17 ) {
addMessage ( XLAT ( " \" We live in a beautiful and orderly world, " ) ) ;
addMessage ( XLAT ( " and not in a chaos without norms. \" " ) ) ;
}
else if ( msgid = = 25 ) {
addMessage ( XLAT ( " \" Thank you very much for talking, and have a great rest of your day! \" " ) ) ;
}
else {
msgid + + ; goto retry ;
}
msgid + + ;
}
EX }
EX namespace mine {
2020-08-02 00:29:29 +00:00
EX int victory_time ;
EX void count_status ( ) {
bool last = kills [ moBomberbird ] ;
kills [ moBomberbird ] = 0 ;
kills [ moTameBomberbird ] = 0 ;
for ( cell * c : currentmap - > allcells ( ) ) if ( c - > wall = = waMineUnknown ) kills [ moBomberbird ] + + ;
for ( cell * c : currentmap - > allcells ( ) ) if ( among ( c - > wall , waMineMine , waMineUnknown ) & & mine : : marked_mine ( c ) ) kills [ moTameBomberbird ] + + ;
if ( last & & ! kills [ moBomberbird ] ) {
mine : : victory_time = getgametime ( ) ;
2022-06-16 21:54:47 +00:00
showMissionScreen ( ) ;
2020-08-02 00:29:29 +00:00
}
}
EX bool in_minesweeper ( ) {
2022-05-21 11:08:42 +00:00
return closed_or_bounded & & specialland = = laMinefield ;
2020-08-02 00:29:29 +00:00
}
2019-12-08 18:17:28 +00:00
EX bool uncoverMines ( cell * c , int lev , int dist , bool just_checking ) {
bool b = false ;
if ( c - > wall = = waMineMine & & just_checking ) return true ;
if ( c - > wall = = waMineUnknown ) {
if ( just_checking )
return true ;
else {
2020-03-01 13:09:24 +00:00
changes . ccell ( c ) ;
2019-12-08 18:17:28 +00:00
c - > wall = waMineOpen ;
b = true ;
}
}
bool minesNearby = false ;
bool nominesNearby = false ;
bool mineopens = false ;
auto adj = adj_minefield_cells ( c ) ;
for ( cell * c2 : adj ) {
if ( c2 - > wall = = waMineMine ) minesNearby = true ;
if ( c2 - > wall = = waMineOpen ) mineopens = true ;
if ( c2 - > wall = = waMineUnknown & & ! c2 - > item ) nominesNearby = true ;
}
if ( lev & & ( nominesNearby | | mineopens ) & & ! minesNearby ) for ( cell * c2 : adj )
if ( c2 - > wall = = waMineUnknown | | c2 - > wall = = waMineOpen ) {
b | = uncoverMines ( c2 , lev - 1 , dist + 1 , just_checking ) ;
if ( b & & just_checking ) return true ;
}
if ( minesNearby & & ! nominesNearby & & dist = = 0 ) {
for ( cell * c2 : adj )
if ( c2 - > wall = = waMineMine & & c2 - > land = = laMinefield )
2020-03-01 13:09:24 +00:00
changes . ccell ( c2 ) ,
2019-12-08 18:17:28 +00:00
c2 - > landparam | = 1 ;
}
return b ;
}
EX bool mightBeMine ( cell * c ) {
return c - > wall = = waMineUnknown | | c - > wall = = waMineMine ;
}
2020-04-11 18:47:14 +00:00
EX hookset < bool ( cell * ) > hooks_mark ;
2019-12-08 18:17:28 +00:00
2022-05-21 15:09:20 +00:00
EX bool mark_always = true ;
2019-12-08 18:17:28 +00:00
EX void performMarkCommand ( cell * c ) {
if ( ! c ) return ;
if ( callhandlers ( false , hooks_mark , c ) ) return ;
2020-04-12 18:33:53 +00:00
if ( c - > land = = laCA & & c - > wall = = waNone ) {
c - > wall = ca : : wlive ;
ca : : list_adj ( c ) ;
}
else if ( c - > land = = laCA & & c - > wall = = ca : : wlive ) {
2019-12-08 18:17:28 +00:00
c - > wall = waNone ;
2020-04-12 18:33:53 +00:00
ca : : list_adj ( c ) ;
}
2019-12-08 18:17:28 +00:00
if ( c - > land ! = laMinefield ) return ;
if ( c - > item ) return ;
if ( ! mightBeMine ( c ) ) return ;
bool adj = false ;
2022-05-21 15:09:20 +00:00
if ( mark_always ) adj = true ;
2019-12-08 18:17:28 +00:00
forCellEx ( c2 , c ) if ( c2 - > wall = = waMineOpen ) adj = true ;
if ( adj ) c - > landparam ^ = 1 ;
}
EX bool marked_mine ( cell * c ) {
if ( ! mightBeMine ( c ) ) return false ;
if ( c - > item ) return false ;
if ( c - > land ! = laMinefield ) return true ;
return c - > landparam & 1 ;
}
EX bool marked_safe ( cell * c ) {
if ( ! mightBeMine ( c ) ) return false ;
if ( c - > item ) return true ;
if ( c - > land ! = laMinefield ) return false ;
return c - > landparam & 2 ;
}
EX bool safe ( ) {
return items [ itOrbAether ] ;
}
EX void uncover_full ( cell * c2 ) {
int mineradius =
2022-05-21 11:08:42 +00:00
closed_or_bounded ? 3 :
2019-12-08 18:17:28 +00:00
( items [ itBombEgg ] < 1 & & ! tactic : : on ) ? 0 :
items [ itBombEgg ] < 20 ? 1 :
items [ itBombEgg ] < 30 ? 2 :
3 ;
bool nomine = ! normal_gravity_at ( c2 ) ;
if ( ! nomine & & uncoverMines ( c2 , mineradius , 0 , true ) & & markOrb ( itOrbAether ) )
nomine = true ;
if ( ! nomine ) {
uncoverMines ( c2 , mineradius , 0 , false ) ;
mayExplodeMine ( c2 , moPlayer ) ;
}
}
EX void auto_teleport_charges ( ) {
2020-08-02 00:29:29 +00:00
if ( in_minesweeper ( ) ) {
mine : : count_status ( ) ;
2019-12-08 18:17:28 +00:00
items [ itOrbTeleport ] = isFire ( cwt . at - > wall ) ? 0 : 1 ;
2020-08-02 00:29:29 +00:00
}
2019-12-08 18:17:28 +00:00
}
EX }
EX namespace terracotta {
# if HDR
// predictable or not
static constexpr bool randterra = false ;
# endif
EX void check ( cell * c ) {
if ( c - > wall = = waTerraWarrior & & ! c - > monst & & ! racing : : on ) {
2021-11-07 09:11:16 +00:00
changes . ccell ( c ) ;
2019-12-08 18:17:28 +00:00
bool live = false ;
if ( randterra ) {
2021-08-04 00:15:53 +00:00
c - > wparam + + ;
if ( ( c - > wparam = = 3 & & hrand ( 3 ) = = 0 ) | |
( c - > wparam = = 4 & & hrand ( 2 ) = = 0 ) | |
c - > wparam = = 5 )
2019-12-08 18:17:28 +00:00
live = true ;
}
else {
2021-08-04 00:15:53 +00:00
c - > wparam - - ;
live = ! c - > wparam ;
2019-12-08 18:17:28 +00:00
}
if ( live )
c - > monst = moTerraWarrior ,
c - > hitpoints = 7 ,
c - > wall = waNone ;
}
}
EX void check_around ( cell * c ) {
forCellEx ( c2 , c )
check ( c2 ) ;
}
EX void check ( ) {
2021-03-06 10:46:13 +00:00
for ( cell * pc : player_positions ( ) )
forCellEx ( c , pc ) {
2019-12-08 18:17:28 +00:00
if ( shmup : : on ) {
forCellEx ( c2 , c )
check ( c2 ) ;
}
else
check ( c ) ;
}
}
EX }
EX namespace ambush {
EX void mark ( cell * c , manual_celllister & cl ) {
if ( ! cl . add ( c ) ) return ;
forCellEx ( c2 , c )
if ( c2 - > cpdist < c - > cpdist )
mark ( c2 , cl ) ;
}
EX int distance ;
EX bool ambushed ;
EX void check_state ( ) {
if ( havewhat & HF_HUNTER ) {
manual_celllister cl ;
for ( cell * c : dcal ) {
if ( c - > monst = = moHunterDog ) {
if ( c - > cpdist > distance )
distance = c - > cpdist ;
mark ( c , cl ) ;
}
if ( c - > monst = = moHunterGuard & & c - > cpdist < = 4 )
mark ( c , cl ) ;
}
if ( items [ itHunting ] > 5 & & items [ itHunting ] < = 22 ) {
int q = 0 ;
2021-03-06 10:46:13 +00:00
for ( cell * pc : player_positions ( ) )
forCellEx ( c2 , pc )
2019-12-08 18:17:28 +00:00
if ( cl . listed ( c2 ) )
q + + ;
if ( q = = 1 ) havewhat | = HF_FAILED_AMBUSH ;
if ( q = = 2 ) {
2021-03-06 10:46:13 +00:00
for ( cell * pc : player_positions ( ) )
forCellEx ( c2 , pc )
2019-12-08 18:17:28 +00:00
if ( cl . listed ( c2 ) )
2021-03-06 10:46:13 +00:00
forCellEx ( c3 , pc )
2019-12-08 18:17:28 +00:00
if ( c3 ! = c2 & & isNeighbor ( c2 , c3 ) )
if ( cl . listed ( c3 ) )
havewhat | = HF_FAILED_AMBUSH ;
}
if ( havewhat & HF_FAILED_AMBUSH & & ambushed ) {
addMessage ( XLAT ( " The Hunting Dogs give up. " ) ) ;
ambushed = false ;
}
}
}
}
EX int fixed_size ;
EX int size ( cell * c , eItem what ) {
bool restricted = false ;
for ( cell * c2 : dcal ) {
if ( c2 - > cpdist > 3 ) break ;
if ( c2 - > monst & & ! isFriendly ( c2 ) & & ! slowMover ( c2 ) & & ! isMultitile ( c2 ) ) restricted = true ;
}
int qty = items [ itHunting ] ;
if ( fixed_size )
return fixed_size ;
switch ( what ) {
case itCompass :
return 0 ;
case itHunting :
return min ( min ( qty , max ( 33 - qty , 6 ) ) , 15 ) ;
case itOrbSide3 :
return restricted ? 10 : 20 ;
case itOrbFreedom :
return restricted ? 10 : 60 ;
2020-03-05 17:48:11 +00:00
case itOrbImpact :
case itOrbPlague :
return 10 ;
2019-12-08 18:17:28 +00:00
case itOrbThorns :
case itOrb37 :
2020-03-05 17:48:11 +00:00
case itOrbChaos :
2019-12-08 18:17:28 +00:00
return 20 ;
case itOrbLava :
return 20 ;
case itOrbBeauty :
return 35 ;
case itOrbShell :
return 35 ;
case itOrbPsi :
// return 40; -> no benefits
return 20 ;
case itOrbDash :
case itOrbFrog :
return 40 ;
case itOrbAir :
case itOrbDragon :
return 50 ;
case itOrbStunning :
// return restricted ? 50 : 60; -> no benefits
return 30 ;
case itOrbBull :
case itOrbSpeed :
case itOrbShield :
return 60 ;
case itOrbInvis :
return 80 ;
case itOrbTeleport :
return 300 ;
case itGreenStone :
case itOrbSafety :
case itOrbYendor :
return 0 ;
case itKey :
return 16 ;
case itWarning :
return qty ;
default :
return restricted ? 6 : 10 ;
break ;
// Flash can survive about 70, but this gives no benefits
}
}
2020-03-07 03:37:24 +00:00
EX void ambush ( cell * c , int dogs ) {
LATE ( ambush ( c , dogs ) ; )
2019-12-08 18:17:28 +00:00
int maxdist = gamerange ( ) ;
celllister cl ( c , maxdist , 1000000 , NULL ) ;
cell * c0 = c ;
int d = 0 ;
int dogs0 = 0 ;
for ( cell * cx : cl . lst ) {
int dh = cl . getdist ( cx ) ;
if ( dh < = 2 & & cx - > monst = = moHunterGuard )
cx - > monst = moHunterDog , dogs0 + + ;
if ( dh > d ) c0 = cx , d = dh ;
}
if ( sphere ) {
for ( int i = cl . lst . size ( ) - 1 ; i > 0 & & dogs ; i - - )
if ( ! isPlayerOn ( cl . lst [ i ] ) & & ! cl . lst [ i ] - > monst )
cl . lst [ i ] - > monst = moHunterDog , dogs - - ;
}
vector < cell * > around ;
cell * clast = NULL ;
cell * ccur = c0 ;
2019-12-14 11:31:20 +00:00
int v = valence ( ) ;
2019-12-08 18:17:28 +00:00
if ( v > 4 ) {
for ( cell * c : cl . lst ) if ( cl . getdist ( c ) = = d ) around . push_back ( c ) ;
2021-07-12 03:54:25 +00:00
hrandom_shuffle ( around ) ;
2019-12-08 18:17:28 +00:00
}
else {
for ( int tries = 0 ; tries < 10000 ; tries + + ) {
cell * c2 = NULL ;
if ( v = = 3 ) {
forCellEx ( c1 , ccur )
if ( c1 ! = clast & & cl . listed ( c1 ) & & cl . getdist ( c1 ) = = d )
c2 = c1 ;
}
if ( v = = 4 ) {
for ( int i = 0 ; i < ccur - > type ; i + + ) {
cell * c1 = ( cellwalker ( ccur , i ) + wstep + 1 ) . peek ( ) ;
if ( ! c1 ) continue ;
if ( c1 ! = clast & & cl . listed ( c1 ) & & cl . getdist ( c1 ) = = d )
c2 = c1 ;
}
}
if ( ! c2 ) break ;
if ( c2 - > land = = laHunting & & c2 - > wall = = waNone & & c2 - > monst = = moNone )
around . push_back ( c2 ) ;
clast = ccur ; ccur = c2 ;
if ( c2 = = c0 ) break ;
}
}
int N = isize ( around ) ;
int gaps = dogs ;
2020-03-05 17:48:59 +00:00
int result = dogs0 ;
if ( N ) {
ambushed = true ;
int shift = hrand ( N ) ;
dogs = min ( dogs , N ) ;
gaps = min ( gaps , N ) ;
for ( int i = 0 ; i < dogs ; i + + ) {
int pos = ( shift + ( N * i ) / gaps ) % N ;
cell * nextdog = around [ pos ] ;
nextdog - > monst = moHunterDog ;
nextdog - > stuntime = 1 ;
drawFlash ( nextdog ) ;
}
result + = dogs ;
}
if ( result )
addMessage ( XLAT ( " You are ambushed! " ) ) ;
2019-12-08 18:17:28 +00:00
}
2021-05-23 13:57:39 +00:00
2023-10-28 21:44:05 +00:00
EX void guard_attack ( ) {
2023-10-29 06:57:20 +00:00
addMessage ( XLAT ( " %The1 alarms other dogs as it dies! " , moHunterDog ) ) ;
2023-10-28 21:44:05 +00:00
for ( cell * c : dcal ) if ( c - > monst = = moHunterGuard ) c - > monst = moHunterDog ;
ambush ( cwt . at , 7 ) ;
}
2021-05-23 13:57:39 +00:00
EX }
EX namespace dice {
2021-05-27 11:00:20 +00:00
struct die_structure {
vector < vector < int > > sides ;
vector < vector < int > > spins ;
vector < int > hardness ;
int faces ;
2021-05-29 13:46:35 +00:00
int facesides ;
2021-05-27 11:00:20 +00:00
int order ;
2021-07-28 10:24:30 +00:00
int highest_hardness ;
2021-05-27 11:00:20 +00:00
die_structure ( int ord , const vector < vector < int > > & v ) {
sides = v ;
spins = sides ;
faces = isize ( sides ) ;
2021-05-29 13:46:35 +00:00
facesides = isize ( sides [ 0 ] ) ;
2021-05-27 11:00:20 +00:00
order = ord ;
for ( int i = 0 ; i < faces ; i + + )
for ( int j = 0 ; j < isize ( sides [ i ] ) ; j + + ) {
int i1 = sides [ i ] [ j ] ;
spins [ i ] [ j ] = - 1 ;
for ( int k = 0 ; k < isize ( sides [ i1 ] ) ; k + + )
if ( sides [ i1 ] [ k ] = = i )
spins [ i ] [ j ] = k ;
if ( spins [ i ] [ j ] = = - 1 )
println ( hlog , " asymmetric " ) ;
}
hardness . resize ( faces , 99 ) ;
hardness . back ( ) = 0 ;
for ( int it = 0 ; it < faces ; it + + )
for ( int i = 0 ; i < faces ; i + + )
for ( int j : sides [ i ] )
hardness [ i ] = min ( hardness [ i ] , hardness [ j ] + 1 ) ;
2021-07-28 10:24:30 +00:00
highest_hardness = 0 ;
for ( int i = 0 ; i < faces ; i + + )
highest_hardness = max ( highest_hardness , hardness [ i ] ) ;
2021-05-27 11:00:20 +00:00
}
} ;
die_structure d20 ( 5 , {
2021-05-23 13:57:39 +00:00
{ 13 - 1 , 7 - 1 , 19 - 1 } , { 20 - 1 , 12 - 1 , 18 - 1 } , { 19 - 1 , 17 - 1 , 16 - 1 } , { 14 - 1 , 18 - 1 , 11 - 1 } , { 13 - 1 , 18 - 1 , 15 - 1 } ,
{ 14 - 1 , 9 - 1 , 16 - 1 } , { 1 - 1 , 15 - 1 , 17 - 1 } , { 20 - 1 , 16 - 1 , 10 - 1 } , { 19 - 1 , 6 - 1 , 11 - 1 } , { 8 - 1 , 17 - 1 , 12 - 1 } ,
{ 13 - 1 , 9 - 1 , 4 - 1 } , { 2 - 1 , 10 - 1 , 15 - 1 } , { 1 - 1 , 11 - 1 , 5 - 1 } , { 20 - 1 , 4 - 1 , 6 - 1 } , { 7 - 1 , 5 - 1 , 12 - 1 } ,
{ 8 - 1 , 6 - 1 , 3 - 1 } , { 7 - 1 , 10 - 1 , 3 - 1 } , { 2 - 1 , 5 - 1 , 4 - 1 } , { 1 - 1 , 3 - 1 , 9 - 1 } , { 8 - 1 , 2 - 1 , 14 - 1 }
2021-05-27 11:00:20 +00:00
} ) ;
die_structure d8 ( 4 , {
{ 1 , 3 , 5 } , { 0 , 4 , 2 } , { 3 , 1 , 7 } , { 2 , 6 , 0 } , { 5 , 7 , 1 } , { 4 , 0 , 6 } , { 7 , 5 , 3 } , { 6 , 2 , 4 }
} ) ;
2021-05-23 13:57:39 +00:00
2021-05-27 11:00:20 +00:00
die_structure d4 ( 3 , { { 3 , 2 , 1 } , { 3 , 0 , 2 } , { 1 , 0 , 3 } , { 0 , 1 , 2 } } ) ;
2021-05-29 13:46:35 +00:00
die_structure d6 ( 3 , { { 1 , 2 , 4 , 3 } , { 5 , 2 , 0 , 3 } , { 5 , 4 , 0 , 1 } , { 5 , 1 , 0 , 4 } , { 0 , 2 , 5 , 3 } , { 4 , 2 , 1 , 3 } } ) ;
die_structure d12 ( 3 , {
{ 3 , 5 , 4 , 9 , 1 } , { 0 , 9 , 6 , 7 , 3 } , { 11 , 10 , 5 , 3 , 7 } , { 0 , 1 , 7 , 2 , 5 } , { 0 , 5 , 10 , 8 , 9 } , { 0 , 3 , 2 , 10 , 4 } ,
{ 11 , 7 , 1 , 9 , 8 } , { 11 , 2 , 3 , 1 , 6 } , { 11 , 6 , 9 , 4 , 10 } , { 0 , 4 , 8 , 6 , 1 } , { 11 , 8 , 4 , 5 , 2 } , { 8 , 10 , 2 , 7 , 6 }
} ) ;
2021-05-31 18:43:50 +00:00
vector < die_structure * > die_list = { & d4 , & d6 , & d8 , & d12 , & d20 } ;
2021-05-29 13:46:35 +00:00
2021-05-27 11:00:20 +00:00
# if HDR
2021-06-16 11:13:13 +00:00
extern vector < struct die_structure * > die_list ;
2021-05-27 11:00:20 +00:00
struct die_data {
struct die_structure * which ;
2021-06-16 09:22:51 +00:00
int val ; /* the current face value */
int dir ; /* which direction is the first side (which->sides[val][0]) of the current face */
2021-06-16 09:12:42 +00:00
bool mirrored ;
2021-05-27 11:00:20 +00:00
int happy ( ) ;
} ;
# endif
2021-06-16 11:13:13 +00:00
EX int shape_faces ( die_structure * w ) { return w - > faces ; }
EX string die_name ( die_structure * w ) { return its ( w - > faces ) ; }
2021-05-27 11:00:20 +00:00
int die_data : : happy ( ) {
if ( val = = which - > faces - 1 ) return 1 ;
if ( val = = 0 ) return - 1 ;
return 0 ;
2021-05-23 13:57:39 +00:00
}
2021-05-31 18:43:50 +00:00
EX die_structure * get_by_id ( unsigned i ) { return die_list [ i % isize ( die_list ) ] ; }
EX int get_die_id ( die_structure * ds ) {
for ( int i = 0 ; i < isize ( die_list ) ; i + + )
if ( die_list [ i ] = = ds )
return i ;
return - 1 ;
}
2021-05-27 11:00:20 +00:00
EX map < cell * , die_data > data ;
EX void generate_specific ( cell * c , die_structure * ds , int min_hardness , int max_hardness ) {
auto & dd = data [ c ] ;
dd . which = ds ;
2021-05-29 13:46:35 +00:00
vector < int > dirs ;
for ( int i = 0 ; i < c - > type ; i + + ) createMov ( c , i ) ;
for ( int i = 0 ; i < c - > type ; i + + )
for ( int j = 0 ; j < c - > type ; j + + ) if ( can_roll ( ds - > facesides , i , movei ( c , j ) ) )
dirs . push_back ( i ) ;
2021-05-29 14:22:40 +00:00
if ( dirs . empty ( ) )
dd . dir = hrand ( c - > type ) ;
else
dd . dir = hrand_elt ( dirs ) ;
2021-05-27 11:00:20 +00:00
vector < int > sides ;
for ( int i = 0 ; i < ds - > faces ; i + + )
if ( ds - > hardness [ i ] > = min_hardness & & ds - > hardness [ i ] < = max_hardness )
sides . push_back ( i ) ;
2021-05-29 13:46:35 +00:00
dd . val = hrand_elt ( sides ) ;
}
EX int die_possible ( cell * c ) {
2021-05-29 14:22:40 +00:00
vector < int > res ;
for ( int i : { 3 , 4 , 5 } )
if ( ( c - > type % i ) = = 0 )
res . push_back ( i ) ;
if ( res . empty ( ) ) return 0 ;
return hrand_elt ( res ) ;
2021-05-29 13:46:35 +00:00
}
EX bool can_roll ( int sides , int cur , movei mi ) {
if ( mi . t - > type % sides ) return false ;
if ( ( cur - mi . d ) % ( mi . s - > type / sides ) ) return false ;
return true ;
}
EX bool can_roll ( movei mi ) {
auto & dd = data [ mi . s ] ;
auto & dw = dd . which ;
return can_roll ( dw - > facesides , dd . dir , mi ) ;
2021-05-23 13:57:39 +00:00
}
2021-05-31 18:43:50 +00:00
EX bool generate_random ( cell * c ) {
vector < die_structure * > ds ;
for ( die_structure * pds : { & d4 , & d6 , & d8 , & d12 , & d20 } )
if ( c - > type % pds - > facesides = = 0 )
ds . push_back ( pds ) ;
if ( ds . empty ( ) ) return false ;
generate_specific ( c , hrand_elt ( ds ) , 0 , 99 ) ;
return true ;
}
2021-05-23 13:57:39 +00:00
2021-05-27 11:00:20 +00:00
EX void generate_full ( cell * c , int hard ) {
2021-05-29 13:46:35 +00:00
int dp = die_possible ( c ) ;
if ( ! dp ) return ;
2021-05-31 09:38:51 +00:00
if ( safety ) return ;
2021-05-27 11:00:20 +00:00
int pct = hrand ( 100 ) ;
2021-05-27 13:54:38 +00:00
int pct2 = hrand ( 6000 ) ;
2021-05-29 13:46:35 +00:00
if ( dp = = 4 ) {
2021-05-29 14:22:40 +00:00
if ( pct < 20 ) {
c - > wall = ( pct < ( items [ itOrbLuck ] ? 9 : 11 ) ) ? waRichDie : waHappyDie ;
2021-05-29 13:46:35 +00:00
generate_specific ( c , & d6 , 1 , 2 ) ;
}
2021-07-20 12:50:02 +00:00
else if ( pct2 < 40 + hard ) {
2021-05-29 14:22:40 +00:00
c - > monst = moAnimatedDie ;
generate_specific ( c , & d6 , 0 , 99 ) ;
}
2021-05-29 13:46:35 +00:00
return ;
}
if ( dp = = 5 ) {
2021-05-29 14:22:40 +00:00
if ( pct < 20 ) {
c - > wall = ( pct < ( items [ itOrbLuck ] ? 9 : 11 ) ) ? waRichDie : waHappyDie ;
2021-05-29 13:46:35 +00:00
generate_specific ( c , & d12 , 2 , 3 ) ;
}
2021-07-20 12:50:02 +00:00
else if ( pct2 < 40 + hard ) {
2021-05-29 14:22:40 +00:00
c - > monst = moAnimatedDie ;
generate_specific ( c , & d12 , 0 , 99 ) ;
}
2021-05-29 13:46:35 +00:00
return ;
}
2021-05-27 13:54:38 +00:00
if ( pct < 3 ) {
2021-05-27 14:38:20 +00:00
c - > wall = waHappyDie ;
2021-05-29 18:06:32 +00:00
generate_specific ( c , & d4 , 0 , 0 ) ;
2021-05-27 11:00:20 +00:00
}
2021-05-27 13:54:38 +00:00
else if ( pct < 6 ) {
2021-05-27 14:38:20 +00:00
c - > wall = waHappyDie ;
2021-05-27 11:00:20 +00:00
generate_specific ( c , & d8 , 0 , 1 ) ;
}
2021-05-27 14:35:11 +00:00
else if ( pct < ( items [ itOrbLuck ] ? 8 : 9 ) ) {
2021-05-27 14:38:20 +00:00
c - > wall = waHappyDie ;
2021-05-27 11:00:20 +00:00
generate_specific ( c , & d20 , 0 , 1 ) ;
}
2021-05-27 13:54:38 +00:00
else if ( pct < 14 ) {
2021-05-27 11:00:20 +00:00
c - > wall = waRichDie ;
2021-05-27 14:35:11 +00:00
if ( items [ itOrbLuck ] )
generate_specific ( c , & d20 , 2 , 3 ) ;
else
generate_specific ( c , & d20 , 4 , 5 ) ;
2021-05-27 11:00:20 +00:00
}
2021-05-27 13:54:38 +00:00
else if ( pct < 15 ) {
2021-05-27 11:00:20 +00:00
c - > wall = waRichDie ;
2021-05-27 14:35:11 +00:00
if ( items [ itOrbLuck ] )
generate_specific ( c , & d8 , 1 , 2 ) ;
else
generate_specific ( c , & d8 , 2 , 3 ) ;
2021-05-27 11:00:20 +00:00
}
2021-05-27 14:35:11 +00:00
else if ( pct2 < ( items [ itOrbLuck ] ? 5 : 1 ) ) {
2021-05-27 11:00:20 +00:00
c - > monst = moAnimatedDie ;
generate_specific ( c , & d4 , 0 , 99 ) ;
}
else if ( pct2 < 40 ) {
c - > monst = moAnimatedDie ;
generate_specific ( c , & d8 , 0 , 99 ) ;
}
else if ( pct2 < 40 + hard ) {
c - > monst = moAnimatedDie ;
generate_specific ( c , & d20 , 0 , 99 ) ;
}
}
EX die_data roll_effect ( movei mi , die_data dd ) {
2021-05-23 13:57:39 +00:00
auto & cto = mi . t ;
auto & th = mi . s ;
int rdir = mi . dir_force ( ) ;
int t = th - > type ;
2021-05-27 11:00:20 +00:00
int val = dd . val ;
int dir = dd . dir ;
2021-05-23 13:57:39 +00:00
2021-05-27 11:00:20 +00:00
auto & dw = dd . which ;
2021-05-23 13:57:39 +00:00
2021-06-16 09:12:42 +00:00
int si = dw - > facesides ;
2021-05-27 11:00:20 +00:00
2021-05-29 14:23:21 +00:00
if ( t % si ) { println ( hlog , " error: bad roll " ) ; return dd ; }
2021-05-23 13:57:39 +00:00
2021-06-16 09:12:42 +00:00
int sideid = gmod ( ( rdir - dir ) * ( dd . mirrored ? - 1 : 1 ) * si / t , si ) ;
2021-05-23 13:57:39 +00:00
2021-05-27 11:00:20 +00:00
int val1 = dw - > sides [ val ] [ sideid ] ;
2021-05-23 13:57:39 +00:00
2021-05-27 11:00:20 +00:00
int si1 = isize ( dw - > sides [ val1 ] ) ;
2021-05-23 13:57:39 +00:00
2021-05-27 11:00:20 +00:00
int sideid1 = dw - > spins [ val ] [ sideid ] ;
2021-05-23 13:57:39 +00:00
int t1 = cto - > type ;
2021-05-29 14:23:21 +00:00
if ( t1 % si1 ) { println ( hlog , " error: bad roll target " ) ; return dd ; }
2021-05-23 13:57:39 +00:00
int rdir1 = mi . rev_dir_force ( ) ;
2021-06-16 09:12:42 +00:00
bool mirror1 = dd . mirrored ^ mi . mirror ( ) ;
int dir1 = rdir1 - ( mirror1 ? - 1 : 1 ) * sideid1 * t1 / si1 ;
2021-05-23 13:57:39 +00:00
dir1 = gmod ( dir1 , t1 ) ;
2021-06-16 09:12:42 +00:00
dd . mirrored = mirror1 ;
2021-05-27 11:00:20 +00:00
dd . val = val1 ;
dd . dir = dir1 ;
return dd ;
}
2021-05-30 11:10:00 +00:00
EX bool on ( cell * c ) {
2021-05-30 11:48:24 +00:00
return isDie ( c - > wall ) | | isDie ( c - > monst ) ;
2021-05-30 11:10:00 +00:00
}
2021-05-27 11:00:20 +00:00
2021-07-20 14:00:48 +00:00
EX string describe ( cell * c ) {
if ( ! data . count ( c ) ) return " BUG: die data missing " ;
else if ( ! data [ c ] . which ) return " BUG: die data default-initialized " ;
else return XLAT ( " d%1 rolled %2 " , its ( data [ c ] . which - > faces ) , its ( data [ c ] . val + 1 ) ) ;
}
2021-05-27 11:00:20 +00:00
EX void roll ( movei mi ) {
auto & cto = mi . t ;
auto & th = mi . s ;
changes . map_value ( data , cto ) ;
changes . map_value ( data , th ) ;
data [ cto ] = roll_effect ( mi , data [ th ] ) ;
data . erase ( th ) ;
2021-05-23 13:57:39 +00:00
}
2021-05-27 11:00:20 +00:00
EX void draw_die ( cell * c , const shiftmatrix & V , ld scale , color_t color ) {
if ( ! data . count ( c ) ) {
queuepoly ( V , cgi . shAsymmetric , 0xFF0000FF ) ;
return ;
}
2022-04-26 13:51:33 +00:00
/* priority used for dice */
const auto prio = PPR : : BIGSTATUE ;
2021-05-27 11:00:20 +00:00
eGeometry orig = geometry ;
bool fpp = GDIM = = 3 ;
auto & dd = data [ c ] ;
int val = dd . val ;
int dir = dd . dir ;
auto & dw = dd . which ;
2021-05-23 13:57:39 +00:00
2021-05-29 13:46:35 +00:00
int si = dw - > facesides ;
2021-05-27 11:00:20 +00:00
2023-09-18 10:07:29 +00:00
if ( inHighQual ) ;
else if ( c = = lmouseover_distant ) {
2021-05-27 11:00:20 +00:00
set < cell * > visited ;
struct celldata_t {
cell * c ;
die_data dd ;
shiftmatrix V ;
} ;
vector < celldata_t > data ;
auto visit = [ & ] ( cell * c , die_data dd , shiftmatrix V ) {
if ( visited . count ( c ) ) return ;
visited . insert ( c ) ;
data . emplace_back ( celldata_t { c , dd , V } ) ;
} ;
visit ( c , dd , V ) ;
for ( int i = 0 ; i < isize ( data ) ; i + + ) {
auto dat = data [ i ] ;
2021-07-28 10:24:30 +00:00
int wa = dw - > hardness [ dat . dd . val ] ;
int wb = dw - > highest_hardness ;
unsigned int red = ( ( wa * 0x00 ) + ( ( wb - wa ) * 0xff ) ) / wb ;
color_t col = 0xFF0000FF + ( red < < 16 ) ;
queuestr ( fpp ? dat . V * zpush ( cgi . FLOOR ) : dat . V , .5 , its ( dat . dd . val + 1 ) , col ) ;
2021-05-27 11:00:20 +00:00
if ( i < = 22 )
2021-05-29 13:46:35 +00:00
forCellIdEx ( c2 , id , dat . c ) if ( can_roll ( si , dat . dd . dir , movei ( dat . c , id ) ) & & ! visited . count ( c2 ) ) {
2021-05-27 11:00:20 +00:00
auto re = roll_effect ( movei ( dat . c , id ) , dat . dd ) ;
shiftmatrix V2 = dat . V * currentmap - > adj ( dat . c , id ) ;
gridline ( dat . V , C0 , V2 , C0 , 0xFF800080 , 0 ) ;
visit ( c2 , re , V2 ) ;
}
}
2021-05-23 13:57:39 +00:00
}
2021-06-16 09:23:49 +00:00
else if ( ! lmouseover_distant | | ! on ( lmouseover_distant ) ) if ( CAP_EXTFONT | | vid . usingGL = = false ) {
2021-05-31 18:44:10 +00:00
queuestr ( V , .5 , its ( val + 1 ) , 0xFFFFFFFF ) ;
auto & side = dw - > sides [ val ] ;
for ( int i = 0 ; i < si ; i + + ) {
int d = dir + c - > type * i / isize ( side ) ;
d = gmod ( d , c - > type ) ;
hyperpoint nxt = tC0 ( currentmap - > adj ( c , d ) ) ;
hyperpoint mid = normalize ( C0 * 1.3 + nxt * - .3 ) ;
queuestr ( V * rgpushxto0 ( mid ) , .25 , its ( side [ i ] + 1 ) , 0xFFFFFFFF ) ;
}
}
2021-05-23 13:57:39 +00:00
2023-02-04 19:03:32 +00:00
shiftmatrix V1 = V * ddspin ( c , dir , M_PI ) ;
2021-06-16 09:12:42 +00:00
if ( dd . mirrored ) V1 = V1 * MirrorY ;
2021-05-23 13:57:39 +00:00
2021-05-27 11:00:20 +00:00
// loop:
vector < bool > face_drawn ( dd . which - > faces , false ) ;
2021-05-23 13:57:39 +00:00
vector < pair < transmatrix , int > > facequeue ;
auto add_to_queue = [ & ] ( const transmatrix & T , int d ) {
if ( face_drawn [ d ] ) return ;
face_drawn [ d ] = true ;
facequeue . emplace_back ( T , d ) ;
} ;
2021-05-27 11:00:20 +00:00
transmatrix S = Id ;
2021-05-23 13:57:39 +00:00
ld outradius , inradius ;
if ( 1 ) {
dynamicval < eGeometry > g ( geometry , gSphere ) ;
2022-11-12 21:38:45 +00:00
ld alpha = TAU / dw - > order ;
ld beta = M_PI / dw - > facesides ;
2021-05-29 13:46:35 +00:00
inradius = edge_of_triangle_with_angles ( alpha , beta , beta ) ;
outradius = edge_of_triangle_with_angles ( beta , alpha , beta ) ;
2021-05-23 13:57:39 +00:00
}
2021-05-27 11:00:20 +00:00
hyperpoint shift = inverse_shift ( V1 , tC0 ( die_target ) ) ;
hyperpoint log_shift = inverse_exp ( shiftless ( shift ) ) * ( inradius / cgi . hexhexdist ) ;
if ( 1 ) {
dynamicval < eGeometry > g ( geometry , gSphere ) ;
hyperpoint de = direct_exp ( log_shift ) ;
S = rgpushxto0 ( de ) ;
2023-04-14 23:13:05 +00:00
# if MAXMDIM >= 4
2021-05-27 11:00:20 +00:00
if ( GDIM = = 3 ) {
2023-04-14 23:13:05 +00:00
for ( int i = 0 ; i < MAXMDIM ; i + + ) swap ( S [ i ] [ 2 ] , S [ i ] [ 3 ] ) ;
for ( int i = 0 ; i < MAXMDIM ; i + + ) swap ( S [ 2 ] [ i ] , S [ 3 ] [ i ] ) ;
2021-05-27 11:00:20 +00:00
}
2023-04-14 23:13:05 +00:00
# endif
for ( int i = 0 ; i < MAXMDIM ; i + + ) S [ i ] [ 1 ] * = - 1 ;
2021-05-27 11:00:20 +00:00
}
add_to_queue ( S , val ) ;
2021-05-29 14:22:40 +00:00
ld dieradius = cgi . scalefactor * scale / 2 ;
2021-05-27 11:00:20 +00:00
if ( dw - > faces = = 20 ) dieradius / = 1.3 ;
if ( dw - > faces = = 8 ) dieradius / = 1.15 ;
2021-05-29 14:22:40 +00:00
if ( dw - > faces = = 12 ) dieradius / = 1.15 ;
2021-05-23 13:57:39 +00:00
2021-05-31 14:11:49 +00:00
if ( hyperbolic ) dieradius / = 1.3 ;
2021-05-23 13:57:39 +00:00
ld base_to_base ;
2021-05-30 20:31:31 +00:00
bool osphere = sphere & & GDIM = = 2 ;
bool oeuclid = euclid & & GDIM = = 2 ;
eGeometry highdim =
( GDIM = = 3 ) ? geometry :
hyperbolic ? gSpace534 : gCubeTiling ;
2021-05-23 13:57:39 +00:00
if ( 1 ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2021-05-27 11:00:20 +00:00
hyperpoint h = cspin ( 2 , 0 , M_PI - outradius ) * zpush0 ( - dieradius ) ;
2021-05-30 20:31:31 +00:00
if ( osphere | | oeuclid )
base_to_base = - h [ 2 ] ;
else {
ld lim = sphere ? 1 : 5 ;
2021-06-03 13:11:24 +00:00
base_to_base = binsearch ( - lim , lim , [ h ] ( ld d ) {
2021-05-30 20:31:31 +00:00
return ( zpush ( d ) * h ) [ 2 ] > = sin_auto ( vid . depth ) ;
} ) ;
}
2021-05-23 13:57:39 +00:00
}
vector < pair < ld , int > > ordering ;
2021-05-27 11:00:20 +00:00
for ( int i = 0 ; i < dw - > faces ; i + + ) {
2021-05-23 13:57:39 +00:00
transmatrix T = facequeue [ i ] . first ;
int ws = facequeue [ i ] . second ;
for ( int d = 0 ; d < si ; d + + ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2022-11-12 21:38:45 +00:00
add_to_queue ( T * cspin ( 0 , 1 , TAU * d / si ) * cspin ( 2 , 0 , inradius ) * cspin ( 0 , 1 , M_PI - TAU * dw - > spins [ ws ] [ d ] / si ) , dw - > sides [ ws ] [ d ] ) ;
2021-05-23 13:57:39 +00:00
}
if ( 1 ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2021-05-23 13:57:39 +00:00
hyperpoint h = zpush ( base_to_base ) * T * zpush0 ( dieradius ) ;
2021-05-27 11:00:20 +00:00
ld z = fpp ? hdist0 ( h ) : asin_auto ( h [ 2 ] ) ;
2021-05-23 13:57:39 +00:00
ordering . emplace_back ( - z , i ) ;
}
}
sort ( ordering . begin ( ) , ordering . end ( ) ) ;
for ( auto o : ordering ) {
int i = o . second ;
transmatrix T = facequeue [ i ] . first ;
2021-05-29 13:46:35 +00:00
array < hyperpoint , 5 > face ;
2021-05-27 11:00:20 +00:00
2021-06-06 08:13:19 +00:00
hyperpoint dctr ;
if ( 1 ) {
dynamicval < eGeometry > g ( geometry , highdim ) ;
dctr = zpush ( base_to_base ) * C0 ;
}
2021-05-27 11:00:20 +00:00
auto sphere_to_space = [ & ] ( hyperpoint h ) {
if ( fpp ) return h ;
2021-05-30 20:31:31 +00:00
if ( osphere ) {
2023-04-14 23:13:05 +00:00
h [ 2 ] = 1 - h [ 2 ] ;
# if MAXMDIM > 3
h [ 3 ] = 0 ;
# endif
2021-05-30 20:31:31 +00:00
return h ;
}
if ( oeuclid ) { h [ 2 ] = 1 - h [ 2 ] ; return h ; }
2021-05-27 11:00:20 +00:00
ld z = asin_auto ( h [ 2 ] ) ;
2021-05-30 20:31:31 +00:00
h = zpush ( - z ) * h ;
2023-04-14 23:13:05 +00:00
# if MAXMDIM > 3
h [ 2 ] = h [ 3 ] ;
h [ 3 ] = 0 ;
# endif
2021-05-27 11:00:20 +00:00
dynamicval < eGeometry > g ( geometry , orig ) ;
2022-12-06 00:04:26 +00:00
return orthogonal_move ( h , z ) ;
2021-05-27 11:00:20 +00:00
} ;
2021-05-23 13:57:39 +00:00
for ( int d = 0 ; d < = si ; d + + ) {
2021-05-27 11:00:20 +00:00
hyperpoint h , hs ;
2021-05-23 13:57:39 +00:00
if ( 1 ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2022-11-12 21:38:45 +00:00
h = zpush ( base_to_base ) * T * cspin ( 0 , 1 , TAU * ( d + .5 ) / si ) * cspin ( 2 , 0 , outradius ) * zpush0 ( dieradius ) ;
2021-05-29 13:46:35 +00:00
if ( d < si ) face [ d ] = h ;
2021-05-27 11:00:20 +00:00
hs = sphere_to_space ( h ) ;
2021-05-23 13:57:39 +00:00
}
2021-05-27 11:00:20 +00:00
curvepoint ( hs ) ;
2021-05-23 13:57:39 +00:00
}
2021-05-27 11:00:20 +00:00
2021-05-29 13:46:35 +00:00
hyperpoint ctr , cx , cy ;
if ( dw - > facesides = = 3 ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2021-05-29 13:46:35 +00:00
ctr = ( face [ 0 ] + face [ 1 ] + face [ 2 ] ) / 3 ;
ctr = ctr * 1.01 - dctr * 0.01 ;
cx = face [ 2 ] - face [ 0 ] ;
cy = face [ 1 ] - ( face [ 0 ] + face [ 2 ] ) / 2 ;
}
if ( dw - > facesides = = 4 ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2021-05-29 13:46:35 +00:00
ctr = ( face [ 0 ] + face [ 1 ] + face [ 2 ] + face [ 3 ] ) / 4 ;
ctr = ctr * 1.01 - dctr * 0.01 ;
cx = face [ 1 ] - face [ 2 ] ;
cy = face [ 0 ] - face [ 1 ] ;
}
if ( dw - > facesides = = 5 ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2021-05-29 13:46:35 +00:00
ctr = ( face [ 0 ] + face [ 1 ] + face [ 2 ] + face [ 3 ] + face [ 4 ] ) / 5 ;
ctr = ctr * 1.01 - dctr * 0.01 ;
cx = ( face [ 2 ] - face [ 0 ] ) * .75 ;
cy = face [ 1 ] - ( face [ 3 ] + face [ 4 ] ) * .4 ;
}
2021-05-27 11:00:20 +00:00
2022-10-06 03:40:15 +00:00
queuecurve ( V1 , ( poly_outline = = OUTLINE_NONE ) ? 0xFFFFFFFF : poly_outline , color & 0xFFFFFF9F , prio ) ;
2021-05-27 11:00:20 +00:00
2021-05-29 10:16:32 +00:00
# if !CAP_EXTFONT
2021-06-16 09:23:49 +00:00
if ( ! vid . usingGL ) continue ;
2021-09-04 19:54:42 +00:00
# if CAP_GL
2021-05-27 11:00:20 +00:00
pointfunction pf = [ & ] ( ld x , ld y ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2021-05-29 13:46:35 +00:00
return sphere_to_space ( normalize ( ctr + cx * x + cy * y ) ) ;
2021-05-27 11:00:20 +00:00
} ;
2021-05-29 18:06:57 +00:00
if ( dw = = & d4 ) {
int q = facequeue [ i ] . second ;
for ( int j = 0 ; j < 3 ; j + + ) {
int j1 = ( j + 1 ) % 3 ;
int j2 = ( j + 2 ) % 3 ;
if ( 1 ) {
2021-05-30 20:31:31 +00:00
dynamicval < eGeometry > g ( geometry , highdim ) ;
2021-05-29 18:06:57 +00:00
ctr = ( face [ 0 ] + face [ 1 ] + face [ 2 ] + face [ j1 ] * 3 ) / 6 ;
ctr = ctr * 1.01 - dctr * 0.01 ;
cx = ( face [ j2 ] - face [ j ] ) / 2 ;
cy = face [ j1 ] - ( face [ j ] + face [ j2 ] ) / 4 ;
}
2022-04-26 13:51:33 +00:00
write_in_space ( V1 , max_glfont_size , - 1.2 , its ( 1 + dw - > sides [ q ] [ j ] ) , 0xFFFFFFFF , 0 , 8 , prio , pf ) ;
2021-05-29 18:06:57 +00:00
}
}
else {
int fid = dw - > faces - facequeue [ i ] . second ;
string s ;
if ( fid = = 6 ) s = " 6. " ;
else if ( fid = = 9 ) s = " 9. " ;
else s = its ( fid ) ;
2022-04-26 13:51:33 +00:00
write_in_space ( V1 , max_glfont_size , dw - > faces < 10 ? - 1.2 : - .75 , s , 0xFFFFFFFF , 0 , 8 , prio , pf ) ;
2021-05-29 18:06:57 +00:00
}
2021-05-29 10:16:32 +00:00
# endif
2021-09-04 19:54:42 +00:00
# endif
2021-05-23 13:57:39 +00:00
}
}
2021-06-16 09:22:41 +00:00
EX bool swap_forbidden ( cell * ca , cell * cb ) {
if ( ! on ( ca ) ) return false ;
return cb - > type % data [ ca ] . which - > facesides ;
}
EX void chaos_swap ( cellwalker wa , cellwalker wb ) {
swap_data ( data , wa . at , wb . at ) ;
if ( on ( wa . at ) ) {
auto & d = data [ wa . at ] ;
d . dir = chaos_mirror_dir ( d . dir , wb , wa ) ;
d . mirrored = ! d . mirrored ;
}
if ( on ( wb . at ) ) {
auto & d = data [ wb . at ] ;
d . dir = chaos_mirror_dir ( d . dir , wa , wb ) ;
d . mirrored = ! d . mirrored ;
}
}
2021-06-03 10:25:53 +00:00
int hook = addHook ( hooks_clearmemory , 0 , [ ] ( ) { data . clear ( ) ; } ) ;
2019-12-08 18:17:28 +00:00
EX }
2018-12-25 11:51:25 +00:00
2018-10-25 00:43:14 +00:00
# endif
2022-05-06 10:40:48 +00:00
# if !CAP_COMPLEX2
EX namespace dice {
EX bool on ( cell * c ) { return false ; }
EX bool swap_forbidden ( cell * a , cell * b ) { return false ; }
EX void chaos_swap ( cellwalker wa , cellwalker wb ) { }
EX }
2022-08-14 17:25:10 +00:00
EX namespace mine {
EX bool in_minesweeper ( ) { return false ; }
EX }
2022-05-06 10:40:48 +00:00
# endif
}