1
0
mirror of https://github.com/zenorogue/hyperrogue.git synced 2024-11-27 14:37:16 +00:00
hyperrogue/geometry.cpp

1443 lines
44 KiB
C++
Raw Normal View History

// Hyperbolic Rogue -- basic geometry
// Copyright (C) 2011-2019 Zeno Rogue, see 'hyper.cpp' for details
2016-08-26 09:58:03 +00:00
/** \file geometry.cpp
* \brief Calculation of basic, and less basic, constants in each geometry
*/
2015-08-08 13:57:52 +00:00
#include "hyper.h"
namespace hr {
2019-08-09 23:56:00 +00:00
#if HDR
struct usershapelayer {
vector<hyperpoint> list;
bool sym;
int rots;
color_t color;
hyperpoint shift, spin;
ld zlevel;
int texture_offset;
PPR prio;
};
2019-08-10 08:57:14 +00:00
extern int usershape_changes;
static constexpr int USERLAYERS = 32;
2019-08-09 23:56:00 +00:00
struct usershape { usershapelayer d[USERLAYERS]; };
struct hpcshape {
int s, e;
PPR prio;
int flags;
hyperpoint intester;
struct basic_textureinfo *tinf;
int texture_offset;
int shs, she;
void clear() { s = e = shs = she = texture_offset = 0; prio = PPR::ZERO; tinf = NULL; flags = 0; }
2023-02-03 22:45:10 +00:00
hpcshape() { clear(); }
2019-08-09 23:56:00 +00:00
};
2019-08-10 08:57:14 +00:00
#define SIDE_SLEV 0
#define SIDE_WTS3 3
#define SIDE_WALL 4
#define SIDE_LAKE 5
#define SIDE_LTOB 6
#define SIDE_BTOI 7
#define SIDE_SKY 8
#define SIDE_HIGH 9
#define SIDE_HIGH2 10
2020-02-26 00:15:30 +00:00
#define SIDE_ASHA 11
#define SIDE_BSHA 12
#define SIDEPARS 13
2019-08-10 08:57:14 +00:00
/** GOLDBERG_BITS controls the size of tables for Goldberg. see gp::check_limits */
#ifndef GOLDBERG_BITS
#define GOLDBERG_BITS 5
#endif
static constexpr int GOLDBERG_LIMIT = (1<<GOLDBERG_BITS);
static constexpr int GOLDBERG_MASK = (GOLDBERG_LIMIT-1);
2020-05-25 00:27:04 +00:00
#ifndef BADMODEL
2019-08-10 08:57:14 +00:00
#define BADMODEL 0
2020-05-25 00:27:04 +00:00
#endif
2019-08-10 08:57:14 +00:00
2020-05-25 00:27:04 +00:00
#ifndef WINGS
static constexpr int WINGS = (BADMODEL ? 1 : 4);
2020-05-25 00:27:04 +00:00
#endif
2019-08-09 23:56:00 +00:00
typedef array<hpcshape, WINGS+1> hpcshape_animated;
extern vector<hpcshape> shPlainWall3D, shWireframe3D, shWall3D, shMiniWall3D;
struct floorshape {
bool is_plain;
int shapeid;
int id;
int pstrength; // pattern strength in 3D
int fstrength; // frame strength in 3D
PPR prio;
2020-01-18 15:03:32 +00:00
vector<hpcshape> b, shadow, side[SIDEPARS], levels[SIDEPARS], cone[2];
vector<vector<hpcshape>> gpside[SIDEPARS];
2019-08-09 23:56:00 +00:00
floorshape() { prio = PPR::FLOOR; pstrength = fstrength = 10; }
};
struct plain_floorshape : floorshape {
ld rad0, rad1;
void configure(ld r0, ld r1) { rad0 = r0; rad1 = r1; }
};
2019-08-10 08:57:14 +00:00
extern vector<ld> equal_weights;
2019-08-09 23:56:00 +00:00
// noftype: 0 (shapeid2 is heptagonal or just use shapeid1), 1 (shapeid2 is pure heptagonal), 2 (shapeid2 is Euclidean), 3 (shapeid2 is hexagonal)
struct escher_floorshape : floorshape {
int shapeid0, shapeid1, noftype, shapeid2;
ld scale;
};
struct basic_textureinfo {
int texture_id;
vector<glvertex> tvertices;
2020-09-11 09:08:27 +00:00
vector<glvertex> colors;
};
2021-03-31 08:30:03 +00:00
/** additional modules can add extra shapes etc. */
struct gi_extension {
2021-03-31 11:35:28 +00:00
virtual ~gi_extension() {}
2021-03-31 08:30:03 +00:00
};
2021-12-11 22:28:05 +00:00
struct expansion_analyzer;
/** both for 'heptagon' 3D cells and subdivided 3D cells */
2021-07-06 23:48:20 +00:00
struct subcellshape {
/** \brief raw coordinates of vertices of all faces */
2021-07-06 23:48:20 +00:00
vector<vector<hyperpoint>> faces;
/** \brief raw coordinates of all vertices in one vector */
2021-07-06 23:48:20 +00:00
vector<hyperpoint> vertices_only;
/** \brief cooked coordinates of vertices of all faces, computed from faces as: from_cellcenter * final_coords(v) */
vector<vector<hyperpoint>> faces_local;
/** \brief cooked coordinates of all vertices in one vector */
2021-07-06 23:48:20 +00:00
vector<hyperpoint> vertices_only_local;
/** \brief weights -- used to generate wall shapes in some geometries, empty otherwise */
vector<vector<double>> weights;
/** the center of every raw face */
2021-07-06 23:48:20 +00:00
vector<hyperpoint> face_centers;
2021-07-11 13:07:40 +00:00
vector<vector<char>> dirdist;
2021-07-06 23:48:20 +00:00
hyperpoint cellcenter;
transmatrix to_cellcenter;
transmatrix from_cellcenter;
/** \brief for adjacent directions a,b, next_dir[a][b] is the next direction adjacent to a, in (counter?)clockwise order from b */
vector<vector<char>> next_dir;
/** useful in product geometries */
vector<hyperpoint> walltester;
/** needed for twisted */
ld angle_of_zero;
/** compute all the properties based on `faces`, for the main heptagon cellshape */
void compute_hept();
/** compute all the properties based on `faces`, for subcells */
void compute_sub();
/** common part of compute_hept and compute_sub */
void compute_common();
2021-07-06 23:48:20 +00:00
};
2022-07-14 09:28:24 +00:00
enum class ePipeEnd {sharp, ball};
2023-01-26 23:27:10 +00:00
struct embedding_method;
/** basic geometry parameters */
2019-08-09 23:56:00 +00:00
struct geometry_information {
/** distance from heptagon center to another heptagon center */
ld tessf;
/** distance from heptagon center to adjacent cell center (either hcrossf or tessf) */
ld crossf;
/** distance from heptagon center to small heptagon vertex */
ld hexf;
/** distance from heptagon center to big heptagon vertex */
ld hcrossf;
/** distance between adjacent hexagon vertices */
ld hexhexdist;
/** distance between hexagon vertex and hexagon center */
ld hexvdist;
/** distance from heptagon center to heptagon vertex (either hexf or hcrossf) */
ld rhexf;
2019-08-09 23:56:00 +00:00
/** edge length */
ld edgelen;
2020-04-05 08:53:34 +00:00
/** basic parameters for 3D geometries */
map<int, int> close_distances;
int loop, face, schmid;
2020-04-05 08:53:34 +00:00
transmatrix spins[32], adjmoves[32];
2021-07-06 23:48:20 +00:00
unique_ptr<struct subcellshape> heptshape;
2021-07-06 23:48:20 +00:00
vector<struct subcellshape> subshapes;
2020-04-05 08:53:34 +00:00
ld adjcheck;
ld strafedist;
2020-05-27 23:50:00 +00:00
ld ultra_mirror_dist, ultra_material_part, ultra_mirror_part;
vector<transmatrix> ultra_mirrors;
2020-04-05 08:53:34 +00:00
int xp_order, r_order, rx_order;
transmatrix full_X, full_R, full_P;
/** for 2D geometries */
2020-01-18 15:03:32 +00:00
vector<transmatrix> heptmove, hexmove, invhexmove;
2019-08-09 23:56:00 +00:00
int base_distlimit;
2023-01-26 23:27:10 +00:00
unique_ptr<embedding_method> emb;
2023-01-05 23:24:45 +00:00
/** size of the Sword (from Orb of the Sword), used in the shmup mode */
2019-08-09 23:56:00 +00:00
ld sword_size;
/** scale factor for the graphics of most things*/
ld scalefactor;
ld orbsize, floorrad0, floorrad1, zhexf;
2019-08-09 23:56:00 +00:00
ld corner_bonus;
ld hexshift;
ld asteroid_size[8];
ld wormscale;
ld tentacle_length;
2024-06-29 10:41:39 +00:00
/** level in hybrid geometries */
2019-08-18 19:16:27 +00:00
ld plevel;
2019-08-26 12:16:55 +00:00
/** level for a z-step */
int single_step;
/** the number of levels in PSL */
int psl_steps;
2024-06-29 10:41:39 +00:00
/** level in twisted geometries -- rarely computed */
ld plevel_twisted;
/** for binary tilings */
transmatrix direct_tmatrix[14];
transmatrix inverse_tmatrix[14];
/** a bitmask for hr::bt::use_direct_for */
int use_direct;
2019-08-09 23:56:00 +00:00
/** various parameters related to the 3D view */
2019-08-09 23:56:00 +00:00
ld INFDEEP, BOTTOM, HELLSPIKE, LAKE, WALL, FLOOR, STUFF,
SLEV[4], FLATEYE,
LEG0, LEG1, LEG, LEG3, GROIN, GROIN1, GHOST,
BODY, BODY1, BODY2, BODY3,
NECK1, NECK, NECK3, HEAD, HEAD1, HEAD2, HEAD3,
2020-02-26 00:15:30 +00:00
ALEG0, ALEG, ABODY, AHEAD, BIRD, LOWSKY, SKY, HIGH, HIGH2,
2023-02-18 17:32:01 +00:00
HELL, STAR, SHALLOW;
2019-08-09 23:56:00 +00:00
ld human_height, slev;
ld eyelevel_familiar, eyelevel_human, eyelevel_dog;
#if CAP_SHAPES
hpcshape
shSemiFloorSide[SIDEPARS],
shBFloor[2],
shWave[8][2],
shCircleFloor,
shBarrel,
shWall[2], shMineMark[2], shBigMineMark[2], shFan,
shZebra[5],
shSwitchDisk,
shTower[11],
shEmeraldFloor[6],
shSemiFeatherFloor[2],
shSemiFloor[2], shSemiBFloor[2], shSemiFloorShadow,
shMercuryBridge[2],
shTriheptaSpecial[14],
shCross, shGiantStar[2], shLake, shMirror,
shHalfFloor[6], shHalfMirror[3],
shGem[2], shStar, shFlash, shDisk, shHalfDisk, shDiskT, shDiskS, shDiskM, shDiskSq, shEccentricDisk, shDiskSegment,
shHeptagon, shHeptagram,
2019-08-09 23:56:00 +00:00
shTinyBird, shTinyShark,
2022-08-29 02:06:01 +00:00
shEgg, shSmallEgg,
2022-08-27 19:43:22 +00:00
shRing, shSpikedRing, shTargetRing, shSawRing, shGearRing, shPeaceRing,
2022-09-17 22:00:54 +00:00
shHeptaRing, shSpearRing, shLoveRing, shFrogRing,
2022-08-27 19:43:22 +00:00
shPowerGearRing, shProtectiveRing, shTerraRing, shMoveRing,
shReserved4, shMoonDisk,
2022-08-28 17:25:42 +00:00
shDaisy, shSnowflake, shTriangle, shNecro, shStatue, shKey, shWindArrow,
2019-08-09 23:56:00 +00:00
shGun,
shFigurine, shTreat, shSmallTreat,
2019-08-09 23:56:00 +00:00
shElementalShard,
// shBranch,
2019-08-19 08:33:07 +00:00
shIBranch, shTentacle, shTentacleX, shILeaf[3],
2019-08-09 23:56:00 +00:00
shMovestar,
shWolf, shYeti, shDemon, shGDemon, shEagle, shGargoyleWings, shGargoyleBody,
shFoxTail1, shFoxTail2,
shDogBody, shDogHead, shDogFrontLeg, shDogRearLeg, shDogFrontPaw, shDogRearPaw,
shDogTorso,
shHawk,
shCatBody, shCatLegs, shCatHead, shFamiliarHead, shFamiliarEye,
shWolf1, shWolf2, shWolf3,
shRatEye1, shRatEye2, shRatEye3,
shDogStripes,
shPBody, shSmallPBody, shPSword, shSmallPSword, shPKnife,
2019-08-09 23:56:00 +00:00
shFerocityM, shFerocityF,
shHumanFoot, shHumanLeg, shHumanGroin, shHumanNeck, shSkeletalFoot, shYetiFoot,
shMagicSword, shSmallSword, shMagicShovel, shSeaTentacle, shKrakenHead, shKrakenEye, shKrakenEye2,
2019-08-09 23:56:00 +00:00
shArrow,
shPHead, shPFace, shGolemhead, shHood, shArmor,
shAztecHead, shAztecCap,
shSabre, shTurban1, shTurban2, shVikingHelmet, shRaiderHelmet, shRaiderArmor, shRaiderBody, shRaiderShirt,
shWestHat1, shWestHat2, shGunInHand,
shKnightArmor, shKnightCloak, shWightCloak,
shGhost, shEyes, shSlime, shJelly, shJoint, shWormHead, shSmallWormHead, shTentHead, shShark, shWormSegment, shSmallWormSegment, shWormTail, shSmallWormTail,
shSlimeEyes, shDragonEyes, shSmallDragonEyes, shWormEyes, shSmallWormEyes, shGhostEyes,
shMiniGhost, shSmallEyes, shMiniEyes,
shHedgehogBlade, shSmallHedgehogBlade, shHedgehogBladePlayer,
2019-08-09 23:56:00 +00:00
shWolfBody, shWolfHead, shWolfLegs, shWolfEyes,
shWolfFrontLeg, shWolfRearLeg, shWolfFrontPaw, shWolfRearPaw,
shFemaleBody, shFemaleHair, shFemaleDress, shWitchDress,
shWitchHair, shBeautyHair, shFlowerHair, shFlowerHand, shSuspenders, shTrophy,
shBugBody, shBugArmor, shBugLeg, shBugAntenna,
2022-08-29 04:09:24 +00:00
shPickAxe, shSmallPickAxe, shPike, shFlailBall, shSmallFlailBall, shFlailTrunk, shSmallFlailTrunk, shFlailChain, shHammerHead, shSmallHammerHead,
2019-08-09 23:56:00 +00:00
shBook, shBookCover, shGrail,
shBoatOuter, shBoatInner, shCompass1, shCompass2, shCompass3,
shKnife, shTongue, shFlailMissile, shTrapArrow,
2022-09-15 23:02:27 +00:00
shPirateHook, shSmallPirateHook, shPirateHood, shEyepatch, shPirateX,
2019-08-09 23:56:00 +00:00
// shScratch,
2023-02-18 17:32:01 +00:00
shHeptaMarker, shSnowball, shHugeDisk, shSkyboxSun, shSun, shNightStar, shEuclideanSky,
2019-08-09 23:56:00 +00:00
shSkeletonBody, shSkull, shSkullEyes, shFatBody, shWaterElemental,
shPalaceGate, shFishTail,
shMouse, shMouseLegs, shMouseEyes,
shPrincessDress, shPrinceDress,
shWizardCape1, shWizardCape2,
shBigCarpet1, shBigCarpet2, shBigCarpet3,
2022-08-26 19:34:04 +00:00
shGoatHead, shRose, shRoseItem, shSmallRose, shThorns,
2019-08-09 23:56:00 +00:00
shRatHead, shRatTail, shRatEyes, shRatCape1, shRatCape2,
shWizardHat1, shWizardHat2,
shTortoise[13][6],
shDragonLegs, shDragonTail, shDragonHead, shSmallDragonHead, shDragonSegment, shDragonNostril, shSmallDragonNostril,
2019-08-09 23:56:00 +00:00
shDragonWings,
shSolidBranch, shWeakBranch, shBead0, shBead1,
shBatWings, shBatBody, shBatMouth, shBatFang, shBatEye,
shParticle[16], shAsteroid[8],
shReptile[5][4],
shReptileBody, shReptileHead, shReptileFrontFoot, shReptileRearFoot,
shReptileFrontLeg, shReptileRearLeg, shReptileTail, shReptileEye,
shTrylobite, shTrylobiteHead, shTrylobiteBody,
shTrylobiteFrontLeg, shTrylobiteRearLeg, shTrylobiteFrontClaw, shTrylobiteRearClaw,
shBullBody, shBullHead, shBullHorn, shBullRearHoof, shBullFrontHoof,
2022-08-28 01:21:53 +00:00
shSmallBullHead, shSmallBullHorn,
shTinyBullHead, shTinyBullHorn, shTinyBullBody,
2019-08-09 23:56:00 +00:00
shButterflyBody, shButterflyWing, shGadflyBody, shGadflyWing, shGadflyEye,
shTerraArmor1, shTerraArmor2, shTerraArmor3, shTerraHead, shTerraFace,
shJiangShi, shJiangShiDress, shJiangShiCap1, shJiangShiCap2,
2020-02-26 00:19:49 +00:00
shPikeBody, shPikeEye,
2019-08-09 23:56:00 +00:00
shAsymmetric,
shPBodyOnly, shPBodyArm, shPBodyHand, shPHeadOnly,
2022-08-29 23:14:59 +00:00
shDodeca, shSmallerDodeca,
2022-09-16 19:13:20 +00:00
shLightningBolt, shHumanoid, shHalfHumanoid, shHourglass,
shShield, shSmallFan, shTreeIcon, shLeafIcon;
2019-08-09 23:56:00 +00:00
hpcshape shFrogRearFoot, shFrogFrontFoot, shFrogRearLeg, shFrogFrontLeg, shFrogRearLeg2, shFrogBody, shFrogEye, shFrogStripe, shFrogJumpFoot, shFrogJumpLeg, shSmallFrogRearFoot, shSmallFrogFrontFoot, shSmallFrogRearLeg, shSmallFrogFrontLeg, shSmallFrogRearLeg2, shSmallFrogBody;
2020-02-26 00:19:49 +00:00
2019-08-09 23:56:00 +00:00
hpcshape_animated
shAnimatedEagle, shAnimatedTinyEagle, shAnimatedGadfly, shAnimatedHawk, shAnimatedButterfly,
2021-05-09 00:25:05 +00:00
shAnimatedGargoyle, shAnimatedGargoyle2, shAnimatedBat, shAnimatedBat2;
2021-05-09 00:25:17 +00:00
hpcshape shTinyArrow;
2023-10-26 10:27:43 +00:00
hpcshape shCrossbow, shCrossbowBolt, shCrossbowstringLoaded, shCrossbowstringUnloaded, shCrossbowstringSemiloaded, shCrossbowIcon, shCrossbowstringIcon;
2021-05-09 00:25:05 +00:00
hpcshape shReserved[16];
int orb_inner_ring; //< for shDisk* shapes, the number of vertices in the inner ring
int res1, res2;
2019-08-09 23:56:00 +00:00
2020-01-25 23:48:56 +00:00
map<int, hpcshape> shPipe;
2019-08-09 23:56:00 +00:00
vector<hpcshape> shPlainWall3D, shWireframe3D, shWall3D, shMiniWall3D;
2019-08-21 05:33:24 +00:00
vector<hyperpoint> walltester;
2019-10-22 10:07:38 +00:00
vector<int> wallstart;
vector<ld> angle_of_zero; /* needed for twisted, especially Archimedean */
2019-10-22 10:07:38 +00:00
vector<transmatrix> raywall;
2019-08-09 23:56:00 +00:00
vector<struct plain_floorshape*> all_plain_floorshapes;
vector<struct escher_floorshape*> all_escher_floorshapes;
plain_floorshape
shFloor,
shMFloor, shMFloor2, shMFloor3, shMFloor4, shFullFloor,
shBigTriangle, shTriheptaFloor, shBigHepta;
escher_floorshape
shStarFloor, shCloudFloor, shCrossFloor, shChargedFloor,
shSStarFloor, shOverFloor, shTriFloor, shFeatherFloor,
shBarrowFloor, shNewFloor, shTrollFloor, shButterflyFloor,
shLavaFloor, shLavaSeabed, shSeabed, shCloudSeabed,
shCaveSeabed, shPalaceFloor, shDemonFloor, shCaveFloor,
shDesertFloor, shPowerFloor, shRoseFloor, shSwitchFloor,
shTurtleFloor, shRedRockFloor[3], shDragonFloor;
ld dlow_table[SIDEPARS], dhi_table[SIDEPARS], dfloor_table[SIDEPARS];
int prehpc;
2022-04-25 17:53:36 +00:00
/** list of points in all shapes */
2019-08-09 23:56:00 +00:00
vector<hyperpoint> hpc;
2022-04-25 17:53:36 +00:00
/** what shape are we currently creating */
hpcshape *last;
/** is the current shape already started? first = not yet */
2019-08-09 23:56:00 +00:00
bool first;
2022-04-25 17:53:36 +00:00
/** starting point of the current shape, can be ultraideal */
hyperpoint starting_point;
/** first ideal point of the current shape */
hyperpoint starting_ideal;
/** last added point of the current shape, can be ultraideal */
hyperpoint last_point;
/** last ideal point of the current shape */
hyperpoint last_ideal;
2019-08-09 23:56:00 +00:00
bool validsidepar[SIDEPARS];
vector<glvertex> ourshape;
#endif
hpcshape shFullCross[2];
int SD3, SD6, SD7, S12, S14, S21, S28, S42, S36, S84;
ld S_step;
2019-08-09 23:56:00 +00:00
vector<pair<int, cell*>> walloffsets;
2019-08-09 23:56:00 +00:00
vector<array<int, 3>> symmetriesAt;
struct cellrotation_t {
transmatrix M;
vector<int> mapping;
int inverse_id;
};
vector<cellrotation_t> cellrotations;
2019-10-12 09:22:13 +00:00
2019-08-09 23:56:00 +00:00
#ifndef SCALETUNER
static constexpr
#endif
double bscale7 = 1, brot7 = 0, bscale6 = 1, brot6 = 0;
vector<hpcshape*> allshapes;
transmatrix shadowmulmatrix;
map<usershapelayer*, hpcshape> ushr;
void prepare_basics();
void prepare_compute3();
void prepare_shapes();
void prepare_usershapes();
void hpcpush(hyperpoint h);
2022-04-25 17:53:36 +00:00
void hpc_connect_ideal(hyperpoint a, hyperpoint b);
2019-08-09 23:56:00 +00:00
void hpcsquare(hyperpoint h1, hyperpoint h2, hyperpoint h3, hyperpoint h4);
void chasmifyPoly(double fac, double fac2, int k);
void shift(hpcshape& sh, double dx, double dy, double dz);
void initPolyForGL();
void extra_vertices();
transmatrix ddi(int a, ld x);
void drawTentacle(hpcshape &h, ld rad, ld var, ld divby);
hyperpoint hpxyzsc(double x, double y, double z);
hyperpoint turtlevertex(int u, double x, double y, double z);
void bshape(hpcshape& sh, PPR prio);
void finishshape();
void bshape(hpcshape& sh, PPR prio, double shzoom, int shapeid, double bonus = 0, flagtype flags = 0);
void copyshape(hpcshape& sh, hpcshape& orig, PPR prio);
void zoomShape(hpcshape& old, hpcshape& newsh, double factor, PPR prio);
void pushShape(usershapelayer& ds);
void make_sidewalls();
void procedural_shapes();
2024-06-10 22:06:16 +00:00
void make_wall(int wo, int id, const vector<hyperpoint> vertices, vector<ld> weights = equal_weights);
void reserve_wall3d(int i);
void compute_cornerbonus();
2019-08-09 23:56:00 +00:00
void create_wall3d();
void configure_floorshapes();
void init_floorshapes();
void bshape2(hpcshape& sh, PPR prio, int shapeid, struct matrixlist& m);
2019-12-27 22:01:55 +00:00
void bshape_regular(floorshape &fsh, int id, int sides, ld shift, ld size, cell *model);
2019-08-09 23:56:00 +00:00
void generate_floorshapes_for(int id, cell *c, int siid, int sidir);
void generate_floorshapes();
void make_floor_textures_here();
void finish_apeirogon(hyperpoint center);
2019-08-09 23:56:00 +00:00
vector<hyperpoint> get_shape(hpcshape sh);
void add_cone(ld z0, const vector<hyperpoint>& vh, ld z1);
void add_prism_sync(ld z0, vector<hyperpoint> vh0, ld z1, vector<hyperpoint> vh1);
void add_prism(ld z0, vector<hyperpoint> vh0, ld z1, vector<hyperpoint> vh1);
void shift_last(ld z);
void shift_shape(hpcshape& sh, ld z);
void shift_shape_orthogonally(hpcshape& sh, ld z);
void add_texture(hpcshape& sh);
void make_ha_3d(hpcshape& sh, bool isarmor, ld scale);
void make_humanoid_3d(hpcshape& sh);
void addtri(array<hyperpoint, 3> hs, int kind);
void make_armor_3d(hpcshape& sh, int kind = 1);
void make_foot_3d(hpcshape& sh);
void make_head_only();
void make_head_3d(hpcshape& sh);
void make_paw_3d(hpcshape& sh, hpcshape& legsh);
void make_abody_3d(hpcshape& sh, ld tail);
void make_ahead_3d(hpcshape& sh);
void make_skeletal(hpcshape& sh, ld push = 0);
void make_revolution(hpcshape& sh, int mx = 180, ld push = 0);
void make_revolution_cut(hpcshape &sh, int each = 180, ld push = 0, ld width = 99);
void clone_shape(hpcshape& sh, hpcshape& target);
void animate_bird(hpcshape& orig, hpcshape_animated& animated, ld body);
void slimetriangle(hyperpoint a, hyperpoint b, hyperpoint c, ld rad, int lev);
void balltriangle(hyperpoint a, hyperpoint b, hyperpoint c, ld rad, int lev);
void make_ball(hpcshape& sh, ld rad, int lev);
void make_star(hpcshape& sh, ld rad);
void make_euclidean_sky();
void adjust_eye(hpcshape& eye, hpcshape head, ld shift_eye, ld shift_head, int q, ld zoom=1);
void shift_last_straight(ld z);
void queueball(const transmatrix& V, ld rad, color_t col, eItem what);
void make_shadow(hpcshape& sh);
void make_3d_models();
/* Goldberg parameters */
#if CAP_GP
struct gpdata_t {
vector<array<array<array<transmatrix, 6>, GOLDBERG_LIMIT>, GOLDBERG_LIMIT>> Tf;
2019-08-09 23:56:00 +00:00
transmatrix corners;
transmatrix corners_for_triangle;
2024-06-21 23:47:05 +00:00
transmatrix rotator;
2019-08-09 23:56:00 +00:00
ld alpha;
int area;
int pshid[3][8][GOLDBERG_LIMIT][GOLDBERG_LIMIT][8];
2020-07-12 18:43:58 +00:00
int nextid;
2019-08-09 23:56:00 +00:00
};
shared_ptr<gpdata_t> gpdata = nullptr;
2019-08-09 23:56:00 +00:00
#endif
2021-12-11 22:28:05 +00:00
shared_ptr<expansion_analyzer> expansion = nullptr;
2019-08-09 23:56:00 +00:00
int state = 0;
int usershape_state = 0;
/** contains the texture point coordinates for 3D models */
basic_textureinfo models_texture;
geometry_information() { last = NULL; use_count = 0; }
2019-08-09 23:56:00 +00:00
void require_basics() { if(state & 1) return; state |= 1; prepare_basics(); }
void require_shapes() { if(state & 2) return; state |= 2; prepare_shapes(); }
void require_usershapes() { if(usershape_state == usershape_changes) return; usershape_state = usershape_changes; prepare_usershapes(); }
int timestamp;
2020-01-25 23:48:56 +00:00
2023-06-10 09:05:57 +00:00
hpcshape& gen_pipe(hpcshape& pipe, ePipeEnd endtype, ld ratio, const hr::function<hyperpoint(ld,ld,ld)>& f);
hpcshape& get_pipe_iso(ld length, ld width, ePipeEnd endtype = ePipeEnd::sharp);
hpcshape& get_pipe_noniso(hyperpoint target, ld width, ePipeEnd endtype = ePipeEnd::sharp);
2021-03-31 08:30:03 +00:00
map<string, unique_ptr<gi_extension>> ext;
/** prevent from being destroyed */
int use_count;
2019-08-09 23:56:00 +00:00
};
#endif
EX subcellshape& get_hsh() {
if(!cgi.heptshape) cgi.heptshape = (unique_ptr<subcellshape>) (new subcellshape);
return *cgi.heptshape;
}
EX void add_wall(int i, const vector<hyperpoint>& h) {
auto& f = get_hsh().faces;
if(isize(f) <= i) f.resize(i+1);
f[i] = h;
}
/** values of hcrossf and hexf for the standard geometry. Since polygons are
* usually drawn in this geometry, the scale in other geometries is usually
* based on comparing these values to the values in the other geometry.
*/
2017-10-27 18:07:58 +00:00
2019-08-24 09:55:45 +00:00
#if HDR
static constexpr ld hcrossf7 = 0.620672, hexf7 = 0.378077, tessf7 = 1.090550, hexhexdist7 = 0.566256;
2019-08-24 09:55:45 +00:00
#endif
2015-08-08 13:57:52 +00:00
EX bool is_subcube_based(eVariation var) {
2021-07-13 13:12:03 +00:00
return among(var, eVariation::subcubes, eVariation::dual_subcubes, eVariation::bch, eVariation::bch_oct);
}
2021-07-08 22:07:35 +00:00
EX bool is_reg3_variation(eVariation var) {
return var == eVariation::coxeter;
}
EX bool special_fake() {
return fake::in() && (BITRUNCATED || (GOLDBERG && S3 == 4 && gp::param.first == 1 && gp::param.second == 1) || (UNRECTIFIED && gp::param.first == 1 && gp::param.second == 1));
}
void geometry_information::prepare_basics() {
2015-08-08 13:57:52 +00:00
DEBBI(DF_INIT | DF_POLY | DF_GEOM, ("prepare_basics"));
2017-10-27 18:07:58 +00:00
hexshift = 0;
2016-08-26 09:58:03 +00:00
ld ALPHA = TAU / S7;
2019-12-14 11:18:24 +00:00
2017-10-28 08:04:28 +00:00
ld fmin, fmax;
2020-05-31 01:30:14 +00:00
ld s3, beta;
heptshape = nullptr;
2017-10-28 08:04:28 +00:00
xp_order = 0;
2023-01-26 23:27:10 +00:00
emb = make_embed();
bool geuclid = euclid;
bool ghyperbolic = hyperbolic;
2022-12-08 18:38:06 +00:00
if(arcm::in() && !mproduct)
ginf[gArchimedean].cclass = gcHyperbolic;
2020-07-12 18:52:32 +00:00
dynamicval<eVariation> gv(variation, variation);
bool inv = INVERSE;
bool specfake = special_fake();
bool unrect = UNRECTIFIED;
2020-07-12 18:52:32 +00:00
if(INVERSE) {
variation = gp::variation_for(gp::param);
println(hlog, "bitruncated = ", BITRUNCATED);
}
2018-08-17 11:29:00 +00:00
2022-12-08 18:38:06 +00:00
if(mhybrid) {
2019-08-18 11:05:47 +00:00
auto t = this;
2022-12-08 18:38:06 +00:00
ld d = mproduct ? 1 : 2;
hybrid::in_underlying_geometry([&] {
t->rhexf = cgi.rhexf / d;
t->hexf = cgi.hexf / d;
t->crossf = cgi.crossf / d;
t->hcrossf = cgi.crossf / d;
t->tessf = cgi.tessf / d;
t->hexvdist = cgi.hexvdist / d;
t->hexhexdist = hdist(xpush0(cgi.hcrossf), xspinpush0(TAU/S7, cgi.hcrossf)) / d;
2019-08-18 11:05:47 +00:00
t->base_distlimit = cgi.base_distlimit-1;
});
goto hybrid_finish;
2019-08-18 11:05:47 +00:00
}
2019-08-24 09:55:45 +00:00
2022-12-08 18:38:06 +00:00
if(embedded_plane) geom3::light_flip(true);
if((sphere || hyperbolic) && WDIM == 3 && !bt::in()) {
rhexf = hexf = 0.378077;
crossf = hcrossf = 0.620672;
tessf = 1.090550;
hexhexdist = 0.566256;
goto finish;
}
2020-05-31 01:30:14 +00:00
s3 = S3;
2020-05-31 14:18:44 +00:00
if(fake::in() && !arcm::in()) s3 = fake::around;
2020-05-31 01:30:14 +00:00
beta = (S3 >= OINF && !fake::in()) ? 0 : TAU/s3;
2016-08-26 09:58:03 +00:00
2020-05-31 01:30:14 +00:00
tessf = euclid ? 1 : edge_of_triangle_with_angles(beta, M_PI/S7, M_PI/S7);
2015-08-08 13:57:52 +00:00
if(elliptic && S7 == 4 && !fake::in()) tessf = 90._deg;
2015-08-08 13:57:52 +00:00
hcrossf = euclid ? tessf / 2 / sin(M_PI/s3) : edge_of_triangle_with_angles(90._deg, M_PI/S7, beta/2);
2020-09-11 09:22:26 +00:00
if(S3 >= OINF) hcrossf = 10;
crossf = BITRUNCATED ? hcrossf : tessf;
2015-08-08 13:57:52 +00:00
fmin = 0, fmax = tessf;
for(int p=0; p<100; p++) {
ld f = (fmin+fmax) / 2;
2018-08-19 14:28:36 +00:00
hyperpoint H = xpush0(f);
hyperpoint H1 = spin(TAU/S7) * H;
2018-08-19 14:28:36 +00:00
hyperpoint H2 = xpush0(tessf-f);
2015-08-08 13:57:52 +00:00
ld v1 = intval(H, H1), v2 = intval(H, H2);
2020-05-31 01:30:14 +00:00
if(fake::in() && WDIM == 2) {
hexvdist = hdist(xpush0(f), xspinpush0(ALPHA/2, hcrossf));
v2 = hdist(
spin(90._deg/S3) * xpush0(hexvdist),
spin(-90._deg/S3) * xpush0(hexvdist)
2020-05-31 01:30:14 +00:00
);
v1 = hdist(
spin(M_PI/S7) * xpush0(f),
spin(-M_PI/S7) * xpush0(f)
);
}
2015-08-08 13:57:52 +00:00
if(v1 < v2) fmin = f; else fmax = f;
}
hexf = fmin;
rhexf = BITRUNCATED ? hexf : hcrossf;
edgelen = hdist(xpush0(rhexf), xspinpush0(TAU/S7, rhexf));
2017-10-29 16:12:40 +00:00
2019-11-27 00:01:20 +00:00
if(BITRUNCATED && !(S7&1))
2017-10-28 08:04:28 +00:00
hexshift = ALPHA/2 + ALPHA * ((S7-1)/2) + M_PI;
2019-11-27 00:01:20 +00:00
2017-10-28 08:04:28 +00:00
finish:
2017-10-28 23:57:34 +00:00
2018-08-19 14:28:36 +00:00
hexvdist = hdist(xpush0(hexf), xspinpush0(ALPHA/2, hcrossf));
2017-10-29 16:12:40 +00:00
2020-05-31 01:30:14 +00:00
hexhexdist = fake::in() ?
2 * hdist0(mid(xspinpush0(M_PI/S6, hexvdist), xspinpush0(-M_PI/S6, hexvdist)))
: hdist(xpush0(crossf), xspinpush0(TAU/S7, crossf));
if(specfake) {
vector<pair<ld, ld>> vals;
2024-06-21 16:34:37 +00:00
int s6 = BITRUNCATED ? S3*2 : S3;
vals.emplace_back(S7, unrect ? 0 : BITRUNCATED ? fake::around / 3 : fake::around / 2);
vals.emplace_back(s6, unrect ? fake::around : BITRUNCATED ? fake::around * 2 / 3 : fake::around / 2);
ld edgelength = euclid ? 1 : arcm::compute_edgelength(vals);
// circumradius and inradius, for S7 and S6 shapes
auto c7 = asin_auto(sin_auto(edgelength/2) / sin(M_PI / S7));
2024-06-21 16:34:37 +00:00
auto c6 = asin_auto(sin_auto(edgelength/2) / sin(M_PI / s6));
auto i7 = hdist0(mid(xpush0(c7), cspin(0, 1, TAU/S7) * xpush0(c7)));
auto i6 = hdist0(mid(xpush0(c6), cspin(0, 1, TAU/s6) * xpush0(c6)));
// note: tessf remains undefined
hcrossf = crossf = unrect ? i6+i6 : i7 + i6;
hexf = c7;
hexhexdist = i6 + i6;
hexvdist = c6;
2024-06-18 16:42:43 +00:00
rhexf = c7;
2024-06-29 10:41:39 +00:00
ld alpha6 = -atan2(xpush(c6) * cspin(0, 1, M_PI - TAU / s6) * xpush0(c6));
ld alpha7 = -atan2(xpush(c7) * cspin(0, 1, M_PI - TAU / S7) * xpush0(c7));
if(BITRUNCATED) plevel_twisted = (M_PI - 2 * alpha6 - alpha7) * fake::around * 2;
}
2020-05-31 01:30:14 +00:00
2019-05-12 23:57:40 +00:00
DEBB(DF_GEOM | DF_POLY,
(hr::format("S7=%d S6=%d hexf = " LDF" hcross = " LDF" tessf = " LDF" hexshift = " LDF " hexhex = " LDF " hexv = " LDF "\n", S7, S6, hexf, hcrossf, tessf, hexshift,
2019-05-12 23:57:40 +00:00
hexhexdist, hexvdist)));
2017-10-29 16:12:40 +00:00
base_distlimit = ginf[geometry].distlimit[!BITRUNCATED];
2018-08-28 17:05:57 +00:00
2023-01-03 19:50:48 +00:00
hybrid_finish:
2019-02-17 17:28:20 +00:00
#if CAP_GP
2020-07-12 18:52:32 +00:00
gp::compute_geometry(inv);
2019-02-17 17:28:20 +00:00
#endif
#if CAP_IRR
2018-07-16 18:05:23 +00:00
irr::compute_geometry();
2019-02-17 17:28:20 +00:00
#endif
#if CAP_ARCM
if(arcm::in()) {
2020-05-31 14:18:44 +00:00
auto& ac = arcm::current_or_fake();
if(fake::in_ext()) ac = arcm::current;
2020-05-31 14:18:44 +00:00
ac.compute_geometry();
crossf = hcrossf7 * ac.scale();
hexvdist = ac.scale() * .5;
rhexf = ac.scale() * .5;
edgelen = ac.edgelength;
2018-08-28 17:05:57 +00:00
}
2019-02-17 17:28:20 +00:00
#endif
#if CAP_BT
if(bt::in()) hexvdist = rhexf = 1, tessf = 1, scalefactor = 1, crossf = hcrossf7;
if(geometry == gHoroRec || kite::in() || sol || nil || nih) hexvdist = rhexf = .5, tessf = .5, scalefactor = .5, crossf = hcrossf7/2;
if(bt::in()) scalefactor *= min<ld>(vid.binary_width, 1), crossf *= min<ld>(vid.binary_width, 1);
#endif
2020-04-10 08:52:24 +00:00
#if MAXMDIM >= 4
2020-04-05 08:53:34 +00:00
if(reg3::in()) reg3::generate();
if(euc::in(3)) euc::generate();
#if CAP_SOLV
else if(sn::in()) sn::create_faces();
#endif
#if CAP_BT
else if(bt::in()) bt::create_faces();
#endif
else if(nil && !mtwisted) nilv::create_faces();
2020-04-10 08:52:24 +00:00
#endif
2018-08-28 17:05:57 +00:00
scalefactor = crossf / hcrossf7;
orbsize = crossf;
2020-05-31 01:30:14 +00:00
if(fake::in() && WDIM == 2) {
auto& u = *fake::underlying_cgip;
geometry = fake::underlying;
2020-06-03 13:11:20 +00:00
ld orig = xpush0(u.hcrossf)[0] / xpush0(u.hcrossf)[GDIM];
2020-05-31 01:30:14 +00:00
geometry = gFake;
2020-06-03 13:11:20 +00:00
ld our = xpush0(hcrossf)[0] / xpush0(hcrossf)[GDIM];
2020-05-31 01:30:14 +00:00
fake::scale = our / orig;
// if(debugflags & DF_GEOM)
}
if(fake::in() && WDIM == 3) {
2020-05-16 09:22:09 +00:00
auto& u = fake::underlying_cgip;
crossf = u->crossf * fake::scale;
scalefactor = u->scalefactor * fake::scale;
orbsize = u->orbsize * fake::scale;
hexf = u->hexf * fake::scale;
rhexf = u->rhexf * fake::scale;
hexvdist = u->hexvdist * fake::scale;
2020-05-31 01:30:14 +00:00
hcrossf = u->hcrossf * fake::scale;
2020-05-16 09:22:09 +00:00
}
2020-05-30 18:00:09 +00:00
if(arb::in()) {
2020-06-02 00:29:31 +00:00
auto csc = arb::current_or_slided().cscale;
2020-05-30 18:00:09 +00:00
scalefactor = csc;
hcrossf = crossf = orbsize = hcrossf7 * csc;
hexf = rhexf = hexvdist = csc * arb::current_or_slided().floor_scale;
base_distlimit = arb::current.range;
2020-05-30 18:00:09 +00:00
}
2021-07-06 23:48:20 +00:00
2022-02-17 20:00:10 +00:00
#if MAXMDIM >= 4
if(is_subcube_based(variation)) {
2021-07-06 23:48:20 +00:00
scalefactor /= reg3::subcube_count;
orbsize /= reg3::subcube_count;
}
2022-02-17 20:00:10 +00:00
#endif
2020-05-16 09:22:09 +00:00
2023-01-26 23:27:10 +00:00
if(meuclid && ghyperbolic) {
2022-12-08 18:38:06 +00:00
scalefactor *= exp(-vid.depth);
}
2023-01-26 23:27:10 +00:00
if(msphere && geuclid) scalefactor *= (1 + vid.depth);
if(msphere && ghyperbolic) scalefactor *= sinh(1 + vid.depth);
2022-12-08 18:38:06 +00:00
if(true) {
scalefactor *= vid.creature_scale;
orbsize *= vid.creature_scale;
}
2018-08-28 17:05:57 +00:00
zhexf = BITRUNCATED ? hexf : crossf* .55;
zhexf *= vid.creature_scale;
2019-05-15 16:43:40 +00:00
if(WDIM == 2 && GDIM == 3) zhexf *= 1.5, orbsize *= 1.2;
2018-08-28 17:05:57 +00:00
if(cgi.emb->is_euc_in_hyp()) {
zhexf *= exp(-vid.depth);
orbsize *= exp(-vid.depth);
}
2022-10-06 10:26:17 +00:00
floorrad0 = hexvdist* (GDIM == 3 ? 1 : 1 - 0.08 * global_boundary_ratio);
floorrad1 = rhexf * (GDIM == 3 ? 1 : 1 - 0.06 * global_boundary_ratio);
2018-08-28 17:05:57 +00:00
if(euc::in(2,4)) {
2018-08-28 17:05:57 +00:00
if(!BITRUNCATED)
floorrad0 = floorrad1 = rhexf * (GDIM == 3 ? 1 : .94);
2018-08-28 17:05:57 +00:00
else
floorrad0 = hexvdist * (GDIM == 3 ? 1 : .9),
floorrad1 = rhexf * (GDIM == 3 ? 1 : .8);
2018-08-28 17:05:57 +00:00
}
2019-08-18 19:16:27 +00:00
plevel = vid.plevel_factor * scalefactor;
2019-08-26 12:16:55 +00:00
single_step = 1;
2024-06-29 10:41:39 +00:00
auto fak = hybrid::underlying == gFake;
auto ug = fak ? fake::underlying : hybrid::underlying;
bool underlying_euclid = false;
if(mtwisted) { underlying_euclid = ginf[ug].cclass == gcEuclid; }
if(mtwisted && !underlying_euclid) {
2020-10-15 14:33:52 +00:00
#if CAP_ARCM
2019-08-27 16:59:26 +00:00
if(hybrid::underlying == gArchimedean)
arcm::current.get_step_values(psl_steps, single_step);
2020-10-15 14:33:52 +00:00
#else
if(0) ;
#endif
2019-08-26 13:39:37 +00:00
else {
single_step = S3 * S7 - 2 * S7 - 2 * S3;
psl_steps = 2 * S7;
if(BITRUNCATED) psl_steps *= S3;
2024-06-28 12:01:52 +00:00
if(GOLDBERG && S3 == 4 && gp::param == gp::loc{1,1}) psl_steps *= 2;
if(inv) psl_steps = 2 * S3;
if(single_step < 0) single_step = -single_step;
2019-08-26 13:39:37 +00:00
}
DEBB(DF_GEOM | DF_POLY, ("steps = ", psl_steps, " / ", single_step));
plevel = M_PI * single_step / psl_steps;
2024-06-28 12:01:52 +00:00
if(hybrid::underlying == gFake) {
auto s3 = fake::around;
ld fake_single_step = s3 * S7 - 2 * S7 - 2 * s3;
ld fake_psl_steps = 2 * S7;
if(BITRUNCATED) fake_psl_steps *= S3;
if(inv) fake_psl_steps = 2 * S3;
if(GOLDBERG && S3 == 4 && gp::param == gp::loc{1,1}) psl_steps *= 2;
if(fake_single_step < 0) fake_single_step = -fake_single_step;
plevel = M_PI * fake_single_step / fake_psl_steps;
2024-06-29 10:41:39 +00:00
/** fake Euclidean... */
if(abs(fake_single_step) < 1e-6)
plevel = 0.25 * s3 / tan(M_PI/s3);
2024-06-28 12:01:52 +00:00
}
}
2024-06-29 10:41:39 +00:00
if(mtwisted && underlying_euclid) {
single_step = 1;
if(ug == gArchimedean) plevel = arcm::current_or_fake().dual_tile_area();
2024-06-29 10:41:39 +00:00
if(ug == gEuclid && PURE) plevel = sqrt(3)/4.;
if(ug == gEuclidSquare && PURE) plevel = 1;
if(ug == gEuclidSquare && BITRUNCATED) plevel = 0.25;
if(ug == gEuclid && BITRUNCATED) plevel = sqrt(3)/12.;
if(ug == gEuclid && fak) plevel = 120._deg * fake::around - TAU;
if(ug == gEuclidSquare && fak && PURE) plevel = 90._deg * fake::around - TAU;
if(ug == gEuclidSquare && fak && BITRUNCATED) plevel = hybrid::underlying_cgip->plevel_twisted;
2019-08-26 12:16:55 +00:00
}
2019-08-18 19:16:27 +00:00
set_sibling_limit();
2022-12-08 18:38:06 +00:00
geom3::light_flip(false);
#if CAP_BT && MAXMDIM >= 4
if(bt::in()) bt::build_tmatrix();
#endif
prepare_compute3();
if(hyperbolic && &currfp != &fieldpattern::fp_invalid)
currfp.analyze();
2022-12-08 18:38:06 +00:00
heptmove.resize(S7);
hexmove.resize(S7);
invhexmove.resize(S7);
for(int d=0; d<S7; d++)
heptmove[d] = spin(-d * ALPHA) * lxpush(tessf) * spin(M_PI);
for(int d=0; d<S7; d++)
hexmove[d] = spin(hexshift-d * ALPHA) * lxpush(-crossf)* spin(M_PI);
for(int d=0; d<S7; d++) invhexmove[d] = iso_inverse(hexmove[d]);
gp::prepare_matrices(inv);
#if CAP_SOLV
if(asonov::in()) {
asonov::prepare();
asonov::prepare_walls();
}
#endif
2015-08-08 13:57:52 +00:00
}
2019-08-09 20:07:03 +00:00
EX purehookset hooks_swapdim;
EX namespace geom3 {
2017-03-23 10:53:57 +00:00
// Here we convert between the following parameters:
// abslev: level below the plane
// lev: level above the world (abslev = depth-lev)
// projection: projection parameter
// factor: zoom factor
2015-08-08 13:57:52 +00:00
2021-05-23 13:57:39 +00:00
EX ld abslev_to_projection(ld abslev) {
if(sphere || euclid) return vid.camera+abslev;
return tanh(abslev) / tanh(vid.camera);
2015-08-08 13:57:52 +00:00
}
2017-03-23 10:53:57 +00:00
ld projection_to_abslev(ld proj) {
if(sphere || euclid) return proj-vid.camera;
2017-03-23 10:53:57 +00:00
// tanh(abslev) / tanh(camera) = proj
return atanh(proj * tanh(vid.camera));
2015-08-08 13:57:52 +00:00
}
2017-03-23 10:53:57 +00:00
ld lev_to_projection(ld lev) {
return abslev_to_projection(vid.depth - lev);
2015-08-08 13:57:52 +00:00
}
2017-03-23 10:53:57 +00:00
ld projection_to_factor(ld proj) {
return lev_to_projection(0) / proj;
}
EX ld factor_to_projection(ld fac) {
2017-03-23 10:53:57 +00:00
return lev_to_projection(0) / fac;
}
EX ld lev_to_factor(ld lev) {
2022-12-08 18:38:06 +00:00
if(mproduct) return -lev;
2019-05-08 16:33:08 +00:00
if(WDIM == 3) return lev;
if(GDIM == 3) return vid.depth - lev;
2017-03-23 10:53:57 +00:00
return projection_to_factor(lev_to_projection(lev));
}
EX ld factor_to_lev(ld fac) {
2022-12-08 18:38:06 +00:00
if(mproduct) return -fac;
if(WDIM == 3) return fac;
if(GDIM == 3) return vid.depth - fac;
return vid.depth - projection_to_abslev(factor_to_projection(fac));
2017-03-23 10:53:57 +00:00
}
EX ld to_wh(ld val) {
return factor_to_lev(val / actual_wall_height());
}
2017-03-23 10:53:57 +00:00
EX void do_auto_eye() {
2019-06-01 17:58:07 +00:00
if(!vid.auto_eye) return;
auto& cs = getcs();
if(cs.charid < 4)
vid.eye = cgi.eyelevel_human;
else if(cs.charid < 8)
vid.eye = cgi.eyelevel_dog;
else if(cs.charid == 8)
vid.eye = cgi.eyelevel_familiar;
}
2017-03-23 10:53:57 +00:00
// how should we scale at level lev
EX ld scale_at_lev(ld lev) {
2017-03-23 10:53:57 +00:00
if(sphere || euclid) return 1;
return cosh(vid.depth - lev);
2017-03-23 10:53:57 +00:00
}
EX string invalid;
2023-02-10 22:39:49 +00:00
EX bool changing_embedded_settings;
2017-03-23 10:53:57 +00:00
EX ld actual_wall_height() {
2022-12-08 18:38:06 +00:00
if(mhybrid) return cgi.plevel;
2019-02-17 17:28:20 +00:00
#if CAP_GP
if(GOLDBERG && vid.gp_autoscale_heights)
return vid.wall_height * min<ld>(4 / hypot_d(2, gp::next), 1);
2019-02-17 17:28:20 +00:00
#endif
return vid.wall_height;
2018-04-23 10:34:14 +00:00
}
EX }
2018-04-23 10:34:14 +00:00
void geometry_information::prepare_compute3() {
using namespace geom3;
2019-05-12 23:57:40 +00:00
DEBBI(DF_INIT | DF_POLY | DF_GEOM, ("geom3::compute"));
// tanh(depth) / tanh(camera) == pconf.alpha
2017-03-23 10:53:57 +00:00
invalid = "";
2023-02-10 22:39:49 +00:00
if(GDIM == 3 || flipped || changing_embedded_settings);
else if(vid.tc_alpha < vid.tc_depth && vid.tc_alpha < vid.tc_camera)
pconf.alpha = tan_auto(vid.depth) / tan_auto(vid.camera);
else if(vid.tc_depth < vid.tc_alpha && vid.tc_depth < vid.tc_camera) {
ld v = pconf.alpha * tan_auto(vid.camera);
if(hyperbolic && (v<1e-6-12 || v>1-1e-12)) invalid = XLAT("cannot adjust depth"), vid.depth = vid.camera;
else vid.depth = atan_auto(v);
2017-03-23 10:53:57 +00:00
}
else {
ld v = tan_auto(vid.depth) / pconf.alpha;
if(hyperbolic && (v<1e-12-1 || v>1-1e-12)) invalid = XLAT("cannot adjust camera"), vid.camera = vid.depth;
else vid.camera = atan_auto(v);
2017-03-23 10:53:57 +00:00
}
if(fabs(pconf.alpha) < 1e-6) invalid = XLAT("does not work with perfect Klein");
2017-03-23 10:53:57 +00:00
if(invalid != "") {
INFDEEP = .7;
BOTTOM = .8;
HELLSPIKE = .85;
LAKE = .9;
2019-05-08 16:33:08 +00:00
FLOOR = 1;
2017-03-23 10:53:57 +00:00
WALL = 1.25;
SLEV[0] = 1;
SLEV[1] = 1.08;
SLEV[2] = 1.16;
SLEV[3] = 1.24;
FLATEYE = 1.03;
LEG1 = 1.025;
LEG = 1.05;
LEG3 = 1.075;
GROIN = 1.09;
GROIN1 = 1.105;
GHOST = 1.1;
BODY = 1.15;
BODY1 = 1.151;
BODY2 = 1.152;
BODY3 = 1.153;
2017-03-23 10:53:57 +00:00
NECK1 = 1.16;
NECK = 1.17;
NECK3 = 1.18;
2019-02-27 00:15:40 +00:00
HEAD = 1.188;
HEAD1= 1.189;
HEAD2= 1.190;
2019-04-20 23:00:15 +00:00
HEAD3= 1.191;
2017-03-23 10:53:57 +00:00
ABODY = 1.08;
AHEAD = 1.12;
BIRD = 1.20;
SHALLOW = .95;
STUFF = 1;
2023-02-18 17:32:01 +00:00
LOWSKY = SKY = HIGH = HIGH2 = STAR = 1;
2017-03-23 10:53:57 +00:00
}
else {
2018-04-23 10:34:14 +00:00
ld wh = actual_wall_height();
WALL = lev_to_factor(wh);
2019-05-08 16:33:08 +00:00
FLOOR = lev_to_factor(0);
2017-03-23 10:53:57 +00:00
human_height = vid.human_wall_ratio * wh;
if(WDIM == 3) human_height = scalefactor * vid.height_width / 2;
2022-12-08 18:38:06 +00:00
if(mhybrid) human_height = min(human_height, cgi.plevel * .9);
2019-05-08 16:33:08 +00:00
ld reduce = (WDIM == 3 ? human_height / 2 : 0);
LEG0 = lev_to_factor(human_height * .0 - reduce);
LEG1 = lev_to_factor(human_height * .1 - reduce);
LEG = lev_to_factor(human_height * .2 - reduce);
LEG3 = lev_to_factor(human_height * .3 - reduce);
GROIN = lev_to_factor(human_height * .4 - reduce);
GROIN1= lev_to_factor(human_height * .5 - reduce);
BODY = lev_to_factor(human_height * .6 - reduce);
BODY1 = lev_to_factor(human_height * .61 - reduce);
BODY2 = lev_to_factor(human_height * .62 - reduce);
BODY3 = lev_to_factor(human_height * .63 - reduce);
NECK1 = lev_to_factor(human_height * .7 - reduce);
NECK = lev_to_factor(human_height * .8 - reduce);
NECK3 = lev_to_factor(human_height * .9 - reduce);
2019-04-20 23:00:15 +00:00
HEAD = lev_to_factor(human_height * .97 - reduce);
HEAD1 = lev_to_factor(human_height * .98 - reduce);
HEAD2 = lev_to_factor(human_height * .99 - reduce);
HEAD3 = lev_to_factor(human_height - reduce);
2017-03-23 10:53:57 +00:00
2019-08-15 13:05:43 +00:00
reduce = (GDIM == 3 ? human_height * .3 : 0);
2022-12-08 20:20:17 +00:00
int sgn = vid.wall_height > 0 ? 1 : -1;
2023-01-26 23:27:10 +00:00
ld ees = cgi.emb->is_euc_in_noniso() ? geom3::euclid_embed_scale_mean() : 1;
2022-12-08 20:20:17 +00:00
2022-12-13 18:04:43 +00:00
STUFF = lev_to_factor(0) - sgn * max(orbsize * ees * 0.3, zhexf * ees * .6);
2019-05-08 16:33:08 +00:00
ABODY = lev_to_factor(human_height * .4 - reduce);
ALEG0 = lev_to_factor(human_height * .0 - reduce);
ALEG = lev_to_factor(human_height * .2 - reduce);
AHEAD = lev_to_factor(human_height * .6 - reduce);
BIRD = lev_to_factor(WDIM == 3 ? 0 : (vid.human_wall_ratio+1)/2 * wh * .8);
2019-05-12 18:06:24 +00:00
GHOST = lev_to_factor(WDIM == 3 ? 0 : human_height * .5);
2017-03-23 10:53:57 +00:00
FLATEYE = lev_to_factor(human_height * .15);
slev = vid.rock_wall_ratio * wh / 3;
2017-03-23 10:53:57 +00:00
for(int s=0; s<=3; s++)
SLEV[s] = lev_to_factor(vid.rock_wall_ratio * wh * s/3);
LAKE = lev_to_factor(wh * -vid.lake_top);
SHALLOW = lev_to_factor(wh * -vid.lake_shallow);
HELLSPIKE = lev_to_factor(wh * -(vid.lake_top+vid.lake_bottom)/2);
BOTTOM = lev_to_factor(wh * -vid.lake_bottom);
2023-02-18 17:32:01 +00:00
LOWSKY = lev_to_factor(vid.lowsky_height * wh);
HIGH = lev_to_factor(vid.wall_height2 * wh);
HIGH2 = lev_to_factor(vid.wall_height3 * wh);
SKY = vid.sky_height == use_the_default_value ? cgi.emb->height_limit(-sgn) : lev_to_factor(vid.sky_height * wh);
STAR = vid.star_height == use_the_default_value ? lerp(FLOOR, SKY, 0.95) : lev_to_factor(vid.star_height * wh);
HELL = -SKY;
if(embedded_plane)
INFDEEP = vid.infdeep_height == use_the_default_value ? cgi.emb->height_limit(sgn) : lev_to_factor(vid.infdeep_height * wh);
else
INFDEEP = (euclid || sphere) ? 0.01 : lev_to_projection(0) * tanh(vid.camera);
2022-12-08 18:38:06 +00:00
2023-01-24 14:17:09 +00:00
/* in spherical/cylindrical case, make sure that the high stuff does not go through the center */
2023-02-18 17:32:01 +00:00
if(vid.height_limits) {
auto hp = cgi.emb->height_limit(1);
auto hn = cgi.emb->height_limit(-1);
auto adjust = [&] (ld& val, ld& guide, ld lerpval) {
if(val > hp)
val = lerp(guide, hp, lerpval);
else if(val < hn)
val = lerp(guide, hn, lerpval);
};
adjust(HIGH, FLOOR, 0.8);
adjust(HIGH2, HIGH, 0.5);
adjust(SKY, FLOOR, 1);
adjust(STAR, FLOOR, 0.9);
adjust(LAKE, FLOOR, 0.8);
adjust(SHALLOW, LAKE, 0.9);
adjust(BOTTOM, SHALLOW, 0.5);
adjust(INFDEEP, FLOOR, 1);
2022-12-08 18:38:06 +00:00
}
2017-03-23 10:53:57 +00:00
}
}
2019-08-09 19:00:52 +00:00
EX namespace geom3 {
2022-12-08 18:38:06 +00:00
/** direction of swapping: +1 => from 2D to 3D; -1 => from 3D to 2D; 0 => make everything right */
EX int swap_direction;
EX void swapdim(int dir) {
swap_direction = dir;
decide_lpu();
swapmatrix_view(NLP, View);
swapmatrix_view(NLP, current_display->which_copy);
callhooks(hooks_swapdim);
for(auto m: allmaps) m->on_dim_change();
}
#if MAXMDIM >= 4
2023-01-26 23:27:10 +00:00
EX void switch_always3() {
2019-05-28 23:09:38 +00:00
if(dual::split(switch_always3)) return;
2020-10-15 14:33:52 +00:00
#if CAP_GL && CAP_RUG
if(rug::rugged) rug::close();
2020-07-03 12:42:33 +00:00
#endif
if(vid.always3) swapdim(-1);
vid.always3 = !vid.always3;
apply_always3();
check_cgi(); cgi.require_basics();
if(vid.always3) swapdim(+1);
}
2023-01-26 23:27:10 +00:00
#endif
2019-08-09 19:00:52 +00:00
EX void switch_tpp() {
2019-05-28 23:09:38 +00:00
if(dual::split(switch_fpp)) return;
2023-01-05 15:57:08 +00:00
if(rug::rugged) rug::close();
2023-08-14 16:08:28 +00:00
if(pmodel == mdDisk && !models::camera_straight) {
vid.yshift = 0;
2023-08-14 16:08:28 +00:00
pconf.cam() = Id;
pconf.xposition = 0;
pconf.yposition = 0;
pconf.scale = 1;
vid.fixed_facing = false;
}
else {
vid.yshift = -0.3;
2023-08-14 16:08:28 +00:00
pconf.cam() = cspin(1, 2, -45._deg);
pconf.scale = 18/16. * vid.xres / vid.yres / multi::players;
pconf.xposition = 0;
pconf.yposition = -0.9;
vid.fixed_facing = true;
vid.fixed_facing_dir = 90;
}
}
2023-02-18 11:55:33 +00:00
2019-08-09 19:00:52 +00:00
EX void switch_fpp() {
#if MAXMDIM >= 4
2020-10-15 14:33:52 +00:00
#if CAP_GL && CAP_RUG
if(rug::rugged) rug::close();
2020-07-03 12:42:33 +00:00
#endif
2019-05-28 23:09:38 +00:00
if(dual::split(switch_fpp)) return;
2023-02-18 11:55:33 +00:00
2023-05-30 16:40:14 +00:00
check_cgi();
cgi.require_basics();
2023-02-18 11:55:33 +00:00
if(!changing_embedded_settings)
View = inverse(models::rotmatrix()) * View;
if(!vid.always3) {
vid.always3 = true;
apply_always3();
2023-01-26 23:27:10 +00:00
auto emb = make_embed();
emb->auto_configure();
check_cgi();
2023-05-30 16:40:14 +00:00
cgi.require_basics();
swapdim(+1);
}
else {
swapdim(-1);
vid.always3 = false;
apply_always3();
2023-02-18 11:55:33 +00:00
if(!changing_embedded_settings) {
vid.wall_height = .3;
vid.human_wall_ratio = .7;
vid.camera = 1;
vid.depth = 1;
}
2023-02-02 18:07:02 +00:00
if(among(pmodel, mdPerspective, mdGeodesic)) pmodel = mdDisk;
swapdim(0);
}
2023-02-18 11:55:33 +00:00
if(!changing_embedded_settings)
View = models::rotmatrix() * View;
#endif
}
EX void apply_settings_full() {
if(cgip && vid.always3) {
2023-02-10 22:39:49 +00:00
changing_embedded_settings = true;
geom3::switch_fpp();
2023-04-11 15:10:04 +00:00
#if MAXMDIM >= 4
delete_sky();
2023-04-11 15:10:04 +00:00
#endif
// not sure why this is needed...
resetGL();
geom3::switch_fpp();
2023-02-10 22:39:49 +00:00
changing_embedded_settings = false;
}
}
EX void apply_settings_light() {
2023-04-11 15:14:19 +00:00
#if MAXMDIM >= 4
if(cgip && vid.always3) {
2023-02-10 22:39:49 +00:00
changing_embedded_settings = true;
geom3::switch_always3();
geom3::switch_always3();
2023-02-10 22:39:49 +00:00
changing_embedded_settings = false;
}
2023-04-11 15:10:04 +00:00
#endif
2023-04-11 15:14:19 +00:00
}
2019-08-09 19:00:52 +00:00
EX }
2015-08-08 13:57:52 +00:00
2019-08-09 20:37:11 +00:00
EX geometry_information *cgip;
EX map<string, geometry_information> cgis;
#if HDR
#define cgi (*cgip)
#endif
EX int last_texture_step;
2019-06-18 14:01:28 +00:00
2019-06-18 14:16:09 +00:00
int ntimestamp;
2021-03-21 10:33:25 +00:00
EX hookset<void(string&)> hooks_cgi_string;
EX string cgi_string() {
string s;
auto V = [&] (string a, string b) { s += a; s += ": "; s += b; s += "; "; };
V("GEO", its(int(geometry)));
V("VAR", its(int(variation)));
2022-04-24 20:09:54 +00:00
if(arb::in()) {
2020-06-02 00:29:31 +00:00
for(auto& sl: arb::current.sliders)
V("AS", fts(sl.current));
2022-04-24 20:09:54 +00:00
for(auto& sl: arb::current.intsliders)
V("AS", its(sl.current));
2020-06-02 00:29:31 +00:00
}
2020-05-15 09:46:26 +00:00
if(fake::in()) {
2020-05-31 01:30:14 +00:00
if(hyperbolic) V("H", fts(fake::around));
if(euclid) V("E", fts(fake::around));
if(sphere) V("S", fts(fake::around));
2020-05-15 09:46:26 +00:00
V("G", FPIU(cgi_string()));
2020-05-28 11:44:02 +00:00
return s;
2020-05-15 09:46:26 +00:00
}
2020-07-12 18:52:32 +00:00
if(GOLDBERG_INV) V("GP", its(gp::param.first) + "," + its(gp::param.second));
if(IRREGULAR) V("IRR", its(irr::irrid));
2022-02-17 20:00:10 +00:00
#if MAXMDIM >= 4
if(is_subcube_based(variation)) V("SC", its(reg3::subcube_count));
if(variation == eVariation::coxeter) V("COX", its(reg3::coxeter_param));
2022-02-17 20:00:10 +00:00
#endif
2020-10-15 14:33:52 +00:00
#if CAP_ARCM
if(arcm::in()) V("ARCM", arcm::current.symbol);
2020-10-15 14:33:52 +00:00
#endif
2019-12-27 21:59:02 +00:00
if(arb::in()) V("ARB", its(arb::current.order));
2022-05-01 09:18:39 +00:00
if(arb::in()) V("AP", its(arb::apeirogon_simplified_display));
2022-09-14 20:53:39 +00:00
if(arb::in()) V("F", its(arb::extended_football));
2022-10-06 10:26:17 +00:00
V("BR", fts(global_boundary_ratio));
if(cryst) V("CRYSTAL", its(ginf[gCrystal].sides) + its(ginf[gCrystal].vertex));
if(bt::in() || GDIM == 3) V("WQ", its(vid.texture_step));
2022-12-08 18:38:06 +00:00
if(mhybrid) {
2020-10-18 12:37:55 +00:00
V("U", PIU(cgi_string()));
}
2019-08-27 07:18:47 +00:00
2022-12-08 18:38:06 +00:00
if(mproduct) V("PL", fts(vid.plevel_factor));
2019-11-13 19:48:38 +00:00
2019-11-17 12:35:18 +00:00
if(geometry == gFieldQuotient) { V("S3=", its(S3)); V("S7=", its(S7)); }
2019-11-23 19:39:38 +00:00
if(nil) V("NIL", its(S7));
2019-11-17 12:35:18 +00:00
if(bt::in()) V("BT", fts(vid.binary_width));
2023-03-24 22:53:13 +00:00
if(hat::in()) V("H", fts(hat::hat_param));
2023-04-11 14:43:53 +00:00
if(hat::in() && hat::hat_param_imag) V("HI", fts(hat::hat_param_imag));
2021-03-21 11:50:51 +00:00
if(nil) V("NILW", fts(nilv::nilwidth));
if(GDIM == 2) {
V("CAMERA", fts(vid.camera));
}
if(WDIM == 2) {
V("WH", fts(vid.wall_height));
V("HW", fts(vid.human_wall_ratio));
V("RW", fts(vid.rock_wall_ratio));
V("DEPTH", fts(vid.depth));
V("ASH", ONOFF(vid.gp_autoscale_heights));
V("LT", fts(vid.lake_top));
V("LB", fts(vid.lake_bottom));
2023-05-23 18:22:14 +00:00
if(GDIM == 3 && vid.pseudohedral) {
V("PSH", fts(vid.pseudohedral));
V("PSD", fts(vid.depth_bonus));
}
2023-02-18 17:32:01 +00:00
V("LS", fts(vid.lake_shallow));
V("SSu", fts(vid.sun_size));
V("SSt", fts(vid.star_size));
V("WH2", fts(vid.wall_height2));
V("WH3", fts(vid.wall_height3));
V("WHL", fts(vid.lowsky_height));
if(vid.sky_height != use_the_default_value) V("SHe", fts(vid.sky_height));
if(vid.star_height != use_the_default_value) V("StH", fts(vid.star_height));
if(vid.infdeep_height != use_the_default_value) V("ID", fts(vid.infdeep_height));
}
V("3D", ONOFF(vid.always3));
2022-12-09 00:58:01 +00:00
if(embedded_plane) V("X:", its(geom3::ggclass()));
if(embedded_plane && meuclid) {
V("XS:", fts(geom3::euclid_embed_scale));
V("YS:", fts(geom3::euclid_embed_scale_y));
V("RS:", fts(geom3::euclid_embed_rotate));
}
2022-12-16 00:26:11 +00:00
if(vid.creature_scale != 1) V("CS", fts(vid.creature_scale));
if(WDIM == 3) V("HTW", fts(vid.height_width));
V("LQ", its(vid.linequality));
2021-03-21 10:33:25 +00:00
callhooks(hooks_cgi_string, s);
return s;
}
2023-04-14 23:27:35 +00:00
#if CAP_PORTALS
2022-02-17 20:00:10 +00:00
#define IFINTRA(x,y) x
#else
#define IFINTRA(x,y) y
#endif
EX void check_cgi() {
string s = cgi_string();
cgip = &cgis[s];
2019-06-18 14:16:09 +00:00
cgi.timestamp = ++ntimestamp;
2022-12-08 18:38:06 +00:00
if(mhybrid) hybrid::underlying_cgip->timestamp = ntimestamp;
if(fake::in() || (mhybrid && PIU(fake::in()))) fake::underlying_cgip->timestamp = ntimestamp;
2022-05-06 10:40:48 +00:00
#if CAP_ARCM
if(arcm::alt_cgip[0]) arcm::alt_cgip[0]->timestamp = ntimestamp;
if(arcm::alt_cgip[1]) arcm::alt_cgip[1]->timestamp = ntimestamp;
2022-05-06 10:40:48 +00:00
#endif
2019-06-18 14:01:28 +00:00
int limit = 3;
for(auto& t: cgis) if(t.second.use_count || t.second.timestamp == ntimestamp) limit++;
if(isize(cgis) > limit) {
2019-06-18 14:01:28 +00:00
vector<pair<int, string>> timestamps;
for(auto& t: cgis) if(!t.second.use_count) timestamps.emplace_back(-t.second.timestamp, t.first);
2019-06-18 14:01:28 +00:00
sort(timestamps.begin(), timestamps.end());
while(isize(timestamps) > limit && timestamps.back().first != -ntimestamp) {
DEBB(DF_GEOM, ("erasing geometry ", timestamps.back().second));
2019-06-18 14:01:28 +00:00
cgis.erase(timestamps.back().second);
timestamps.pop_back();
}
}
if(floor_textures && last_texture_step != vid.texture_step) {
println(hlog, "changed ", last_texture_step, " to ", vid.texture_step);
delete floor_textures;
floor_textures = NULL;
}
2020-07-03 12:48:36 +00:00
#if MAXMDIM >= 4 && CAP_GL
2019-08-15 13:05:43 +00:00
if(!floor_textures && GDIM == 3 && (cgi.state & 2))
2019-06-18 14:01:28 +00:00
make_floor_textures();
2019-07-12 21:17:00 +00:00
#endif
2019-06-18 14:01:28 +00:00
2015-08-08 13:57:52 +00:00
}
2019-06-18 14:01:28 +00:00
void clear_cgis() {
printf("clear_cgis\n");
for(auto& p: cgis) if(&p.second != &cgi) { cgis.erase(p.first); return; }
}
auto ah_clear_geo = addHook(hooks_clear_cache, 0, clear_cgis);
}