mirror of
https://github.com/zenorogue/hyperrogue.git
synced 2024-12-24 17:10:36 +00:00
crystal {3,4,4}
This commit is contained in:
parent
352f9ed8a9
commit
d2b3df57c6
@ -593,6 +593,7 @@ vector<geometryinfo> ginf = {
|
|||||||
{"{3,oo}", "none", "{3,∞} (infinite triangles)", "oox3", 3, 100, qIDEAL, giHyperb2, 0x49400, {{7, 7}}, eVariation::pure},
|
{"{3,oo}", "none", "{3,∞} (infinite triangles)", "oox3", 3, 100, qIDEAL, giHyperb2, 0x49400, {{7, 7}}, eVariation::pure},
|
||||||
{"{3,3,6}","none", "{3,3,6} hyperbolic honeycomb", "336", 4, 6, qIDEAL | qEXPERIMENTAL, giHyperb3, 0x49600, {{7, 2}}, eVariation::pure},
|
{"{3,3,6}","none", "{3,3,6} hyperbolic honeycomb", "336", 4, 6, qIDEAL | qEXPERIMENTAL, giHyperb3, 0x49600, {{7, 2}}, eVariation::pure},
|
||||||
{"{3,4,4}","none", "{3,4,4} hyperbolic honeycomb", "344", 8, 4, qIDEAL | qEXPERIMENTAL, giHyperb3, 0x50000, {{7, 2}}, eVariation::pure},
|
{"{3,4,4}","none", "{3,4,4} hyperbolic honeycomb", "344", 8, 4, qIDEAL | qEXPERIMENTAL, giHyperb3, 0x50000, {{7, 2}}, eVariation::pure},
|
||||||
|
{"{3,4,4}","Crystal", "4D crystal in H3", "Cryst3" , 8, 4, qIDEAL | qANYQ | qCRYSTAL, giHyperb3, 0x52000, {{7, 3}}, eVariation::pure},
|
||||||
};
|
};
|
||||||
// bits: 9, 10, 15, 16, (reserved for later) 17, 18
|
// bits: 9, 10, 15, 16, (reserved for later) 17, 18
|
||||||
|
|
||||||
|
@ -215,7 +215,7 @@ enum eGeometry {
|
|||||||
gField435, gField534,
|
gField435, gField534,
|
||||||
gBinary4, gSol,
|
gBinary4, gSol,
|
||||||
gKiteDart2, gKiteDart3, gNil, gProduct, gRotSpace,
|
gKiteDart2, gKiteDart3, gNil, gProduct, gRotSpace,
|
||||||
gTernary, gNIH, gSolN, gInfOrder, gSpace336, gSpace344,
|
gTernary, gNIH, gSolN, gInfOrder, gSpace336, gSpace344, gCrystal344,
|
||||||
gGUARD};
|
gGUARD};
|
||||||
|
|
||||||
enum eGeometryClass { gcHyperbolic, gcEuclid, gcSphere, gcSolNIH, gcNil, gcProduct, gcSL2 };
|
enum eGeometryClass { gcHyperbolic, gcEuclid, gcSphere, gcSolNIH, gcNil, gcProduct, gcSL2 };
|
||||||
|
128
crystal.cpp
128
crystal.cpp
@ -13,7 +13,11 @@ EX namespace crystal {
|
|||||||
|
|
||||||
#if HDR
|
#if HDR
|
||||||
static const int MAXDIM = 7;
|
static const int MAXDIM = 7;
|
||||||
typedef array<int, MAXDIM> coord;
|
|
||||||
|
struct coord : public array<int, MAXDIM> {
|
||||||
|
coord operator + (coord b) { for(int i=0; i<MAXDIM; i++) b[i] += self[i]; return b; }
|
||||||
|
};
|
||||||
|
|
||||||
static const coord c0 = {};
|
static const coord c0 = {};
|
||||||
|
|
||||||
typedef array<ld, MAXDIM> ldcoord;
|
typedef array<ld, MAXDIM> ldcoord;
|
||||||
@ -356,6 +360,25 @@ int fiftyrule(coord c) {
|
|||||||
|
|
||||||
bool is_bi(crystal_structure& cs, coord co);
|
bool is_bi(crystal_structure& cs, coord co);
|
||||||
|
|
||||||
|
#if MAXMDIM >= 4
|
||||||
|
using shifttable = array<coord, 8>;
|
||||||
|
|
||||||
|
shifttable get_canonical(coord co) {
|
||||||
|
shifttable res;
|
||||||
|
for(int i=0; i<4; i++) {
|
||||||
|
res[i] = c0;
|
||||||
|
res[i][i] = 2;
|
||||||
|
res[i+4] = c0;
|
||||||
|
res[i+4][i] = -2;
|
||||||
|
}
|
||||||
|
for(int a=0; a<4; a++) if(co[a] & 2) swap(res[a], res[a+4]);
|
||||||
|
int bts = 0;
|
||||||
|
for(int a=0; a<4; a++) if(co[a] & 2) bts++;
|
||||||
|
if(bts & 1) swap(res[2], res[3]), swap(res[6], res[7]);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
struct hrmap_crystal : hrmap_standard {
|
struct hrmap_crystal : hrmap_standard {
|
||||||
heptagon *getOrigin() { return get_heptagon_at(c0, S7); }
|
heptagon *getOrigin() { return get_heptagon_at(c0, S7); }
|
||||||
|
|
||||||
@ -380,8 +403,12 @@ struct hrmap_crystal : hrmap_standard {
|
|||||||
return a;
|
return a;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool crystal3() { return WDIM == 3; }
|
||||||
|
|
||||||
hrmap_crystal() {
|
hrmap_crystal() {
|
||||||
cs.build();
|
if(crystal3()) cs.dim = 4;
|
||||||
|
else cs.build();
|
||||||
|
|
||||||
camelot_center = NULL;
|
camelot_center = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -447,6 +474,18 @@ struct hrmap_crystal : hrmap_standard {
|
|||||||
}
|
}
|
||||||
auto co = hcoords[h];
|
auto co = hcoords[h];
|
||||||
|
|
||||||
|
if(crystal3()) {
|
||||||
|
auto st = get_canonical(co);
|
||||||
|
auto co1 = co + st[d];
|
||||||
|
auto h1 = get_heptagon_at(co1, S7);
|
||||||
|
auto st1 = get_canonical(co1);
|
||||||
|
|
||||||
|
for(int d1=0; d1<8; d1++) if(st1[d1] == st[d])
|
||||||
|
h->c.connect(d, h1, d1 ^ 4, false);
|
||||||
|
|
||||||
|
return h1;
|
||||||
|
}
|
||||||
|
|
||||||
if(is_bi(cs, co)) {
|
if(is_bi(cs, co)) {
|
||||||
heptspin hs(h, d);
|
heptspin hs(h, d);
|
||||||
(hs + 1 + wstep + 1).cpeek();
|
(hs + 1 + wstep + 1).cpeek();
|
||||||
@ -477,6 +516,82 @@ struct hrmap_crystal : hrmap_standard {
|
|||||||
return h->move(d);
|
return h->move(d);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if MAXMDIM >= 4
|
||||||
|
map<int, transmatrix> adjs;
|
||||||
|
|
||||||
|
transmatrix adj(heptagon *h, int d) {
|
||||||
|
auto co = hcoords[h];
|
||||||
|
int id = 0;
|
||||||
|
for(int a=0; a<4; a++) id = (2*id) + ((co[a]>>1) & 1);
|
||||||
|
id = 8*id + d;
|
||||||
|
if(adjs.count(id)) return adjs[id];
|
||||||
|
transmatrix T = reg3::adjmoves[d];
|
||||||
|
reg3::generate_cellrotations();
|
||||||
|
auto st = get_canonical(co);
|
||||||
|
auto co1 = co + st[d];
|
||||||
|
auto st1 = get_canonical(co1);
|
||||||
|
int qty = 0;
|
||||||
|
transmatrix res;
|
||||||
|
for(auto& cr: cgi.cellrotations) {
|
||||||
|
transmatrix U = T * cr.first;
|
||||||
|
for(int s=0; s<8; s++)
|
||||||
|
if(reg3::dirs_adjacent[d][s])
|
||||||
|
for(int t=0; t<8; t++)
|
||||||
|
if(st1[t] == st[s]) {
|
||||||
|
if(hdist(U * tC0(reg3::adjmoves[t]), tC0(reg3::adjmoves[s])) > reg3::strafedist + .1)
|
||||||
|
goto wrong;
|
||||||
|
}
|
||||||
|
res = U;
|
||||||
|
qty++;
|
||||||
|
wrong: ;
|
||||||
|
}
|
||||||
|
adjs[id] = res;
|
||||||
|
if(qty == 1) return res;
|
||||||
|
println(hlog, "qty = ", qty);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void draw() override {
|
||||||
|
if(!crystal3()) { hrmap_standard::draw(); return; }
|
||||||
|
sphereflip = Id;
|
||||||
|
|
||||||
|
// for(int i=0; i<S6; i++) queuepoly(ggmatrix(cwt.at), shWall3D[i], 0xFF0000FF);
|
||||||
|
|
||||||
|
dq::visited_by_matrix.clear();
|
||||||
|
dq::enqueue_by_matrix(viewctr.at, cview());
|
||||||
|
|
||||||
|
while(!dq::drawqueue.empty()) {
|
||||||
|
auto& p = dq::drawqueue.front();
|
||||||
|
heptagon *h = get<0>(p);
|
||||||
|
transmatrix V = get<1>(p);
|
||||||
|
dynamicval<ld> b(band_shift, get<2>(p));
|
||||||
|
bandfixer bf(V);
|
||||||
|
dq::drawqueue.pop();
|
||||||
|
|
||||||
|
cell *c = h->c7;
|
||||||
|
if(!do_draw(c, V)) continue;
|
||||||
|
drawcell(c, V, 0, false);
|
||||||
|
|
||||||
|
if(wallopt && isWall3(c) && isize(dq::drawqueue) > 1000) continue;
|
||||||
|
|
||||||
|
for(int d=0; d<S7; d++) {
|
||||||
|
dq::enqueue_by_matrix(h->move(d), V * adj(h, d));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual transmatrix relative_matrix(cell *h2, cell *h1, const hyperpoint& hint) override {
|
||||||
|
if(!crystal3()) return hrmap_standard::relative_matrix(h2, h1, hint);
|
||||||
|
if(gmatrix0.count(h2) && gmatrix0.count(h1))
|
||||||
|
return inverse(gmatrix0[h1]) * gmatrix0[h2];
|
||||||
|
return xpush(999);
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual transmatrix relative_matrix(heptagon *h2, heptagon *h1) override {
|
||||||
|
if(!crystal3()) return hrmap::relative_matrix(h2, h1);
|
||||||
|
return relative_matrix(h2->c7, h1->c7, C0);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
hrmap_crystal *crystal_map() {
|
hrmap_crystal *crystal_map() {
|
||||||
@ -487,6 +602,8 @@ EX heptagon *get_heptagon_at(coord c) { return crystal_map()->get_heptagon_at(c,
|
|||||||
EX coord get_coord(heptagon *h) { return crystal_map()->hcoords[h]; }
|
EX coord get_coord(heptagon *h) { return crystal_map()->hcoords[h]; }
|
||||||
EX ldcoord get_ldcoord(cell *c) { return crystal_map()->get_coord(c); }
|
EX ldcoord get_ldcoord(cell *c) { return crystal_map()->get_coord(c); }
|
||||||
|
|
||||||
|
EX transmatrix get_adj(heptagon *h, int d) { return crystal_map()->adj(h, d); }
|
||||||
|
|
||||||
bool is_bi(crystal_structure& cs, coord co) {
|
bool is_bi(crystal_structure& cs, coord co) {
|
||||||
for(int i=0; i<cs.dim; i++) if(co[i] & HALFSTEP) return true;
|
for(int i=0; i<cs.dim; i++) if(co[i] & HALFSTEP) return true;
|
||||||
return false;
|
return false;
|
||||||
@ -1149,17 +1266,18 @@ EX void show() {
|
|||||||
dialog::addBreak(50);
|
dialog::addBreak(50);
|
||||||
dialog::addBoolItem_action(XLAT("view coordinates in the cheat mode"), view_coordinates, 'v');
|
dialog::addBoolItem_action(XLAT("view coordinates in the cheat mode"), view_coordinates, 'v');
|
||||||
dialog::addSelItem(XLAT("compass probability"), fts(compass_probability), 'p');
|
dialog::addSelItem(XLAT("compass probability"), fts(compass_probability), 'p');
|
||||||
|
if(geometry == gCrystal344) dialog::lastItem().value += " (N/A)";
|
||||||
dialog::add_action([]() {
|
dialog::add_action([]() {
|
||||||
dialog::editNumber(compass_probability, 0, 1, 0.1, 1, XLAT("compass probability"), compass_help());
|
dialog::editNumber(compass_probability, 0, 1, 0.1, 1, XLAT("compass probability"), compass_help());
|
||||||
dialog::bound_low(0);
|
dialog::bound_low(0);
|
||||||
});
|
});
|
||||||
if(cryst) {
|
if(cryst && WDIM == 2) {
|
||||||
dialog::addBoolItem(XLAT("3D display"), rug::rugged, 'r');
|
dialog::addBoolItem(XLAT("3D display"), rug::rugged, 'r');
|
||||||
dialog::add_action_push(rug::show);
|
dialog::add_action_push(rug::show);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
dialog::addBreak(100);
|
dialog::addBreak(100);
|
||||||
if(rug::rugged && cryst && ginf[gCrystal].sides == 8) {
|
if(rug::rugged && cryst && ginf[gCrystal].sides == 8 && WDIM == 2) {
|
||||||
dialog::addBoolItem(XLAT("render a cut"), draw_cut, 'x');
|
dialog::addBoolItem(XLAT("render a cut"), draw_cut, 'x');
|
||||||
dialog::add_action([]() {
|
dialog::add_action([]() {
|
||||||
draw_cut = true;
|
draw_cut = true;
|
||||||
@ -1461,7 +1579,7 @@ void transform_euclid_to_crystal () {
|
|||||||
|
|
||||||
EX void add_crystal_transform(char c) {
|
EX void add_crystal_transform(char c) {
|
||||||
if(shmup::on) return;
|
if(shmup::on) return;
|
||||||
if(cryst && ginf[gCrystal].sides == 6) {
|
if(cryst && ginf[gCrystal].sides == 6 && geometry != gCrystal344) {
|
||||||
dialog::addItem("convert Crystal to 3D", c);
|
dialog::addItem("convert Crystal to 3D", c);
|
||||||
dialog::add_action(transform_crystal_to_euclid);
|
dialog::add_action(transform_crystal_to_euclid);
|
||||||
}
|
}
|
||||||
|
2
reg3.cpp
2
reg3.cpp
@ -892,6 +892,8 @@ EX bool pseudohept(cell *c) {
|
|||||||
return c->master->distance & 1;
|
return c->master->distance & 1;
|
||||||
if(geometry == gField534)
|
if(geometry == gField534)
|
||||||
return hr::celldistance(c, currentmap->gamestart()) & 1;
|
return hr::celldistance(c, currentmap->gamestart()) & 1;
|
||||||
|
if(geometry == gCrystal344)
|
||||||
|
return false;
|
||||||
if(hyperbolic) {
|
if(hyperbolic) {
|
||||||
heptagon *h = m->reg_gmatrix[c->master].first;
|
heptagon *h = m->reg_gmatrix[c->master].first;
|
||||||
return (h->zebraval == 1) && (h->distance & 1);
|
return (h->zebraval == 1) && (h->distance & 1);
|
||||||
|
Loading…
Reference in New Issue
Block a user