2019-08-10 11:43:24 +00:00
|
|
|
// Hyperbolic Rogue -- Euclidean geometry
|
|
|
|
// Copyright (C) 2011-2019 Zeno Rogue, see 'hyper.cpp' for details
|
|
|
|
|
|
|
|
/** \file euclid.cpp
|
|
|
|
* \brief Euclidean geometry, including 2D, 3D, and quotient spaces
|
|
|
|
*/
|
2019-02-24 21:12:32 +00:00
|
|
|
|
2019-09-05 07:15:40 +00:00
|
|
|
#include "hyper.h"
|
2019-02-24 21:12:32 +00:00
|
|
|
namespace hr {
|
|
|
|
|
2019-02-24 22:04:52 +00:00
|
|
|
// 3D Euclidean space
|
|
|
|
|
2019-02-27 22:30:26 +00:00
|
|
|
#if MAXMDIM == 4
|
2019-02-24 22:04:52 +00:00
|
|
|
|
2019-08-09 20:37:11 +00:00
|
|
|
EX namespace euclid3 {
|
2019-02-24 21:12:32 +00:00
|
|
|
|
2019-09-05 09:57:38 +00:00
|
|
|
#if HDR
|
2019-11-29 13:34:40 +00:00
|
|
|
struct coord : array<int, 3> {
|
|
|
|
coord() {}
|
|
|
|
coord(int x, int y, int z) { self[0] = x; self[1] = y; self[2] = z; }
|
|
|
|
coord& operator += (coord b) { for(int i: {0,1,2}) self[i] += b[i]; return self; }
|
|
|
|
coord& operator -= (coord b) { for(int i: {0,1,2}) self[i] -= b[i]; return self; }
|
|
|
|
coord operator + (coord b) const { coord a = self; return a += b; }
|
|
|
|
coord operator - (coord b) const { coord a = self; return a -= b; }
|
|
|
|
coord operator -() const { return coord(-self[0], -self[1], -self[2]); }
|
|
|
|
coord& operator +() { return self; }
|
|
|
|
const coord& operator +() const { return self; }
|
|
|
|
coord operator *(int x) const { return coord(x*self[0], x*self[1], x*self[2]); }
|
|
|
|
friend coord operator *(int x, const coord& y) { return coord(x*y[0], x*y[1], x*y[2]); }
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef array<coord, 3> intmatrix;
|
2019-11-27 21:01:02 +00:00
|
|
|
#endif
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
EX coord euzero = coord(0,0,0);
|
2019-03-01 03:18:19 +00:00
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
static const intmatrix main_axes = make_array<coord>(coord(1,0,0), coord(0,1,0), coord(0,0,1));
|
|
|
|
|
2019-09-05 09:57:38 +00:00
|
|
|
EX vector<coord> get_shifttable() {
|
2019-04-15 21:29:07 +00:00
|
|
|
static const coord D0 = main_axes[0];
|
|
|
|
static const coord D1 = main_axes[1];
|
|
|
|
static const coord D2 = main_axes[2];
|
2019-03-01 17:53:20 +00:00
|
|
|
vector<coord> shifttable;
|
|
|
|
switch(geometry) {
|
|
|
|
case gCubeTiling:
|
|
|
|
shifttable = { +D0, +D1, +D2 };
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gRhombic3:
|
|
|
|
shifttable = { D0+D1, D0+D2, D1+D2, D1-D2, D0-D2, D0-D1 };
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gBitrunc3:
|
|
|
|
shifttable = { 2*D0, 2*D1, 2*D2, D0+D1+D2, D0+D1-D2, D0-D1-D2, D0-D1+D2 };
|
|
|
|
break;
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
case gEuclid:
|
|
|
|
shifttable = { D0, D1, D1-D0, -D0, -D1, D0-D1 };
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gEuclidSquare:
|
|
|
|
shifttable = { D0, D1, -D0, -D1 };
|
|
|
|
break;
|
|
|
|
|
2019-03-01 17:53:20 +00:00
|
|
|
default:
|
|
|
|
printf("euclid3::get_shifttable() called in geometry that is not euclid3");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// reverse everything
|
|
|
|
int s = isize(shifttable);
|
|
|
|
for(int i=0; i<s; i++) shifttable.push_back(-shifttable[i]);
|
|
|
|
return shifttable;
|
|
|
|
}
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
EX coord canonicalize(coord x);
|
2019-11-02 19:27:44 +00:00
|
|
|
EX int twisted;
|
2019-11-27 21:01:02 +00:00
|
|
|
EX intmatrix T0;
|
2019-11-27 00:01:20 +00:00
|
|
|
EX gp::loc twisted_vec, ortho_vec;
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
struct hrmap_euclid3 : hrmap_standard {
|
2019-03-01 17:53:20 +00:00
|
|
|
vector<coord> shifttable;
|
|
|
|
vector<transmatrix> tmatrix;
|
2019-02-24 21:12:32 +00:00
|
|
|
map<coord, heptagon*> spacemap;
|
|
|
|
map<heptagon*, coord> ispacemap;
|
2019-04-11 22:17:50 +00:00
|
|
|
cell *camelot_center;
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
map<gp::loc, struct cdata> eucdata;
|
|
|
|
|
2019-04-15 21:29:07 +00:00
|
|
|
vector<cell*> toruscells;
|
|
|
|
vector<cell*>& allcells() override {
|
|
|
|
if(bounded) {
|
|
|
|
if(isize(toruscells) == 0) {
|
|
|
|
celllister cl(getOrigin()->c7, 1000, 1000000, NULL);
|
|
|
|
toruscells = cl.lst;
|
|
|
|
}
|
|
|
|
return toruscells;
|
|
|
|
}
|
|
|
|
return hrmap::allcells();
|
|
|
|
}
|
|
|
|
|
2019-02-24 22:04:52 +00:00
|
|
|
hrmap_euclid3() {
|
2019-03-01 17:53:20 +00:00
|
|
|
shifttable = get_shifttable();
|
|
|
|
tmatrix.resize(S7);
|
2019-11-27 00:01:20 +00:00
|
|
|
for(int i=0; i<S7; i++)
|
|
|
|
tmatrix[i] = eumove(shifttable[i]);
|
2019-04-11 22:17:50 +00:00
|
|
|
camelot_center = NULL;
|
2019-11-27 00:01:20 +00:00
|
|
|
build_torus3(geometry);
|
2019-02-24 21:12:32 +00:00
|
|
|
}
|
2019-03-06 15:36:10 +00:00
|
|
|
|
2019-05-10 02:02:37 +00:00
|
|
|
heptagon *getOrigin() override {
|
2019-11-29 13:34:40 +00:00
|
|
|
return get_at(euzero);
|
2019-02-24 21:12:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
heptagon *get_at(coord at) {
|
|
|
|
if(spacemap.count(at))
|
|
|
|
return spacemap[at];
|
|
|
|
else {
|
2019-03-01 17:53:20 +00:00
|
|
|
auto h = tailored_alloc<heptagon> (S7);
|
2019-11-27 00:03:57 +00:00
|
|
|
if(!IRREGULAR)
|
|
|
|
h->c7 = newCell(S7, h);
|
|
|
|
else
|
|
|
|
irr::link_to_base(h, ((hrmap_euclid3*)irr::base)->get_at(at));
|
2019-02-24 21:12:32 +00:00
|
|
|
h->distance = 0;
|
|
|
|
h->cdata = NULL;
|
2019-03-09 00:18:47 +00:00
|
|
|
h->alt = NULL;
|
2019-03-01 17:53:20 +00:00
|
|
|
if(S7 != 14)
|
2019-11-29 13:34:40 +00:00
|
|
|
h->zebraval = gmod(at[0] + at[1] * 2 + at[2] * 4, 5);
|
2019-03-01 17:53:20 +00:00
|
|
|
else
|
2019-11-29 13:34:40 +00:00
|
|
|
h->zebraval = at[0] & 1;
|
2019-02-24 21:12:32 +00:00
|
|
|
spacemap[at] = h;
|
|
|
|
ispacemap[h] = at;
|
2019-11-29 13:34:40 +00:00
|
|
|
|
2019-02-24 21:12:32 +00:00
|
|
|
return h;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
heptagon *build(heptagon *parent, int d, coord at) {
|
|
|
|
auto h = get_at(at);
|
2019-04-16 02:15:00 +00:00
|
|
|
int d1 = (d+S7/2)%S7;
|
2019-11-27 00:01:20 +00:00
|
|
|
bool mirr = false;
|
2019-04-16 02:15:00 +00:00
|
|
|
if(twisted) {
|
2019-11-27 00:01:20 +00:00
|
|
|
transmatrix I;
|
2019-11-29 13:34:40 +00:00
|
|
|
auto st = shifttable[d1];
|
2019-11-27 00:01:20 +00:00
|
|
|
twist(ispacemap[parent] + shifttable[d], st, I, mirr);
|
2019-11-29 13:34:40 +00:00
|
|
|
for(int i=0; i<S7; i++) if(shifttable[i] == st) d1 = i;
|
2019-04-16 02:15:00 +00:00
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
h->c.connect(d1, parent, d, mirr);
|
2019-02-24 21:12:32 +00:00
|
|
|
return h;
|
|
|
|
}
|
|
|
|
|
2019-05-10 02:02:37 +00:00
|
|
|
heptagon *create_step(heptagon *parent, int d) override {
|
2019-04-15 21:29:07 +00:00
|
|
|
return build(parent, d, canonicalize(ispacemap[parent] + shifttable[d]));
|
2019-03-08 21:38:44 +00:00
|
|
|
}
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
transmatrix adj(heptagon *h, int i) override {
|
2019-04-16 01:22:52 +00:00
|
|
|
if(!twisted) return tmatrix[i];
|
|
|
|
transmatrix res = tmatrix[i];
|
2019-11-27 00:01:20 +00:00
|
|
|
coord id = ispacemap[h];
|
2019-04-16 01:22:52 +00:00
|
|
|
id += shifttable[i];
|
2019-11-29 13:34:40 +00:00
|
|
|
auto dummy = euzero;
|
2019-11-27 00:01:20 +00:00
|
|
|
bool dm = false;
|
|
|
|
twist(id, dummy, res, dm);
|
|
|
|
|
2019-04-16 01:22:52 +00:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
void draw() {
|
2019-04-15 21:29:07 +00:00
|
|
|
dq::visited_by_matrix.clear();
|
2019-11-27 00:01:20 +00:00
|
|
|
dq::enqueue_by_matrix(centerover->master, cview() * master_relative(centerover, true));
|
2019-03-08 21:38:44 +00:00
|
|
|
|
|
|
|
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;
|
2019-11-27 00:01:20 +00:00
|
|
|
|
|
|
|
bool draw = drawcell_subs(c, V * spin(master_to_c7_angle()));
|
2019-10-12 09:21:00 +00:00
|
|
|
if(wallopt && isWall3(c) && isize(dq::drawqueue) > 1000) continue;
|
2019-03-08 21:38:44 +00:00
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
if(draw) for(int i=0; i<S7; i++)
|
|
|
|
dq::enqueue_by_matrix(h->move(i), V * adj(h, i));
|
2019-03-08 21:38:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-26 23:39:41 +00:00
|
|
|
transmatrix relative_matrix(heptagon *h2, heptagon *h1, const hyperpoint& hint) override {
|
2019-04-16 01:22:52 +00:00
|
|
|
if(twisted) {
|
2019-11-26 23:39:41 +00:00
|
|
|
if(h1 == h2) return Id;
|
|
|
|
for(int s=0; s<S7; s++) if(h2 == h1->move(s)) return adj(h1, s);
|
2019-04-16 01:22:52 +00:00
|
|
|
coord c1 = ispacemap[h1];
|
|
|
|
coord c2 = ispacemap[h2];
|
2019-11-28 21:15:39 +00:00
|
|
|
transmatrix T = eumove(c2 - c1);
|
|
|
|
|
|
|
|
transmatrix I = Id;
|
|
|
|
coord cs = c1;
|
|
|
|
for(int s=0; s<4; s++) {
|
|
|
|
for(int a=-1; a<=1; a++)
|
|
|
|
for(int b=-1; b<=1; b++) {
|
|
|
|
if(b && WDIM == 2) continue;
|
2019-11-29 13:34:40 +00:00
|
|
|
transmatrix T1 = I * eumove((c2 - cs) + a*T0[0] + b*T0[1]);
|
2019-04-16 01:22:52 +00:00
|
|
|
if(hdist0(tC0(T1)) < hdist0(tC0(T)))
|
|
|
|
T = T1;
|
|
|
|
}
|
2019-11-29 13:34:40 +00:00
|
|
|
auto co = T0[WDIM-1];
|
2019-11-28 21:15:39 +00:00
|
|
|
cs += co;
|
|
|
|
I = I * eumove(co);
|
2019-11-29 13:34:40 +00:00
|
|
|
auto dummy = euzero;
|
2019-11-28 21:15:39 +00:00
|
|
|
bool dm = false;
|
|
|
|
cs = twist(cs, dummy, I, dm);
|
2019-04-16 01:22:52 +00:00
|
|
|
}
|
|
|
|
return T;
|
|
|
|
}
|
2019-04-15 21:29:07 +00:00
|
|
|
auto d = ispacemap[h2] - ispacemap[h1];
|
|
|
|
d = canonicalize(d);
|
2019-11-27 00:01:20 +00:00
|
|
|
return eumove(d);
|
2019-02-24 21:12:32 +00:00
|
|
|
}
|
2019-03-08 21:38:44 +00:00
|
|
|
|
2019-04-08 14:16:16 +00:00
|
|
|
vector<hyperpoint> get_vertices(cell* c) override {
|
|
|
|
vector<hyperpoint> res;
|
|
|
|
if(S7 < 14)
|
|
|
|
for(ld a: {-.5,.5}) for(ld b: {-.5,.5}) for(ld c: {-.5, .5}) res.push_back(hpxy3(a,b,c));
|
|
|
|
if(S7 == 12) {
|
|
|
|
res.push_back(hpxy3(1,0,0));
|
|
|
|
res.push_back(hpxy3(-1,0,0));
|
|
|
|
res.push_back(hpxy3(0,1,0));
|
|
|
|
res.push_back(hpxy3(0,-1,0));
|
|
|
|
res.push_back(hpxy3(0,0,1));
|
|
|
|
res.push_back(hpxy3(0,0,-1));
|
|
|
|
}
|
|
|
|
if(S7 == 14) {
|
|
|
|
for(ld a: {-1.,-.5,0.,.5,1.})
|
|
|
|
for(ld b: {-1.,-.5,0.,.5,1.})
|
|
|
|
for(ld c: {-1.,-.5,0.,.5,1.})
|
|
|
|
if(a == 0 || b == 0 || c == 0)
|
|
|
|
if(a == .5 || a == -.5 || b == .5 || b == -.5 || c == .5 || c == -.5)
|
|
|
|
if(a == 1 || a == -1 || b == 1 || b == -1 || c == 1 || c == -1)
|
|
|
|
res.push_back(hpxy3(a,b,c));
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
2019-02-24 21:12:32 +00:00
|
|
|
};
|
|
|
|
|
2019-02-24 22:04:52 +00:00
|
|
|
hrmap_euclid3* cubemap() {
|
|
|
|
return ((hrmap_euclid3*) currentmap);
|
2019-02-24 21:12:32 +00:00
|
|
|
}
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
hrmap_euclid3* eucmap() { return cubemap(); }
|
|
|
|
|
2019-09-05 09:57:38 +00:00
|
|
|
EX vector<coord>& get_current_shifttable() { return cubemap()->shifttable; }
|
|
|
|
EX map<coord, heptagon*>& get_spacemap() { return cubemap()->spacemap; }
|
|
|
|
EX map<heptagon*, coord>& get_ispacemap() { return cubemap()->ispacemap; }
|
|
|
|
EX cell *& get_camelot_center() { return cubemap()->camelot_center; }
|
|
|
|
|
2019-08-09 20:37:11 +00:00
|
|
|
EX hrmap* new_map() {
|
2019-02-24 22:04:52 +00:00
|
|
|
return new hrmap_euclid3;
|
2019-02-24 21:12:32 +00:00
|
|
|
}
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
EX transmatrix move_matrix(heptagon *h, int i) {
|
|
|
|
return cubemap()->adj(h, i);
|
2019-04-15 21:29:07 +00:00
|
|
|
}
|
|
|
|
|
2019-09-05 09:57:38 +00:00
|
|
|
EX bool pseudohept(cell *c) {
|
2019-02-27 13:40:56 +00:00
|
|
|
coord co = cubemap()->ispacemap[c->master];
|
2019-03-02 00:19:31 +00:00
|
|
|
if(S7 == 12) {
|
2019-11-29 13:34:40 +00:00
|
|
|
for(int i=0; i<3; i++) if((co[i] & 1)) return false;
|
2019-03-02 00:19:31 +00:00
|
|
|
}
|
|
|
|
else {
|
2019-11-29 13:34:40 +00:00
|
|
|
for(int i=0; i<3; i++) if(!(co[i] & 1)) return false;
|
2019-03-02 00:19:31 +00:00
|
|
|
}
|
2019-02-27 13:40:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-09-05 09:57:38 +00:00
|
|
|
EX int dist_alt(cell *c) {
|
2019-04-11 22:17:50 +00:00
|
|
|
if(specialland == laCamelot) return dist_relative(c) + roundTableRadius(c);
|
2019-11-29 13:34:40 +00:00
|
|
|
auto v = cubemap()->ispacemap[c->master];
|
2019-03-02 00:19:31 +00:00
|
|
|
if(S7 == 6) return v[2];
|
|
|
|
else if(S7 == 12) return (v[0] + v[1] + v[2]) / 2;
|
|
|
|
else return v[2]/2;
|
2019-02-27 14:50:26 +00:00
|
|
|
}
|
|
|
|
|
2019-09-05 09:57:38 +00:00
|
|
|
EX bool get_emerald(cell *c) {
|
2019-11-29 13:34:40 +00:00
|
|
|
auto v = cubemap()->ispacemap[c->master];
|
2019-03-02 00:19:31 +00:00
|
|
|
int s0 = 0, s1 = 0;
|
|
|
|
for(int i=0; i<3; i++) {
|
|
|
|
v[i] = gmod(v[i], 6);
|
|
|
|
int d = min(v[i], 6-v[i]);;
|
|
|
|
s0 += min(v[i], 6-v[i]);
|
|
|
|
s1 += 3-d;
|
|
|
|
}
|
|
|
|
if(s0 == s1) println(hlog, "equality");
|
|
|
|
return s0 > s1;
|
2019-02-24 21:12:32 +00:00
|
|
|
}
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
bool cellvalid(coord v) {
|
2019-04-15 21:29:07 +00:00
|
|
|
if(S7 == 6) return true;
|
|
|
|
if(S7 == 12) return (v[0] + v[1] + v[2]) % 2 == 0;
|
|
|
|
if(S7 == 14) return v[0] % 2 == v[1] % 2 && v[0] % 2 == v[2] % 2;
|
|
|
|
return false;
|
|
|
|
}
|
2019-02-24 21:12:32 +00:00
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
EX int celldistance(coord v) {
|
2019-03-01 17:53:20 +00:00
|
|
|
if(S7 == 6)
|
2019-03-02 00:19:31 +00:00
|
|
|
return abs(v[0]) + abs(v[1]) + abs(v[2]);
|
2019-03-01 17:53:20 +00:00
|
|
|
else {
|
2019-03-02 00:19:31 +00:00
|
|
|
for(int i=0; i<3; i++) v[i] = abs(v[i]);
|
|
|
|
sort(v.begin(), v.end());
|
2019-03-01 17:53:20 +00:00
|
|
|
int dist = 0;
|
|
|
|
if(S7 == 12) {
|
2019-03-02 00:19:31 +00:00
|
|
|
int d = v[1] - v[0]; v[1] -= d; v[2] -= d;
|
2019-03-01 17:53:20 +00:00
|
|
|
dist += d;
|
2019-04-11 22:18:18 +00:00
|
|
|
int m = min((v[2] - v[0]), v[0]);
|
|
|
|
dist += 2 * m;
|
|
|
|
v[0] -= m; v[1] -= m; v[2] -= m * 2;
|
2019-03-02 00:19:31 +00:00
|
|
|
if(v[0])
|
|
|
|
dist += (v[0] + v[1] + v[2]) / 2;
|
2019-03-01 17:53:20 +00:00
|
|
|
else
|
2019-03-02 00:19:31 +00:00
|
|
|
dist += v[2];
|
2019-03-01 17:53:20 +00:00
|
|
|
}
|
|
|
|
else {
|
2019-03-02 00:19:31 +00:00
|
|
|
dist = v[0] + (v[1] - v[0]) / 2 + (v[2] - v[0]) / 2;
|
2019-03-01 17:53:20 +00:00
|
|
|
}
|
|
|
|
return dist;
|
|
|
|
}
|
2019-02-24 21:12:32 +00:00
|
|
|
}
|
|
|
|
|
2019-09-05 09:57:38 +00:00
|
|
|
EX int celldistance(cell *c1, cell *c2) {
|
2019-04-15 21:29:07 +00:00
|
|
|
auto cm = cubemap();
|
2019-11-02 21:17:57 +00:00
|
|
|
return celldistance(canonicalize(cm->ispacemap[c1->master] - cm->ispacemap[c2->master]));
|
2019-04-15 21:29:07 +00:00
|
|
|
}
|
|
|
|
|
2019-09-05 09:57:38 +00:00
|
|
|
EX void set_land(cell *c) {
|
2019-04-11 22:17:50 +00:00
|
|
|
setland(c, specialland);
|
|
|
|
auto m = cubemap();
|
2019-11-29 13:34:40 +00:00
|
|
|
auto co = m->ispacemap[c->master];
|
2019-04-11 22:17:50 +00:00
|
|
|
|
|
|
|
int dv = 1;
|
|
|
|
if(geometry != gCubeTiling) dv = 2;
|
|
|
|
|
|
|
|
int hash = 0;
|
|
|
|
for(int a=0; a<3; a++) hash = 1317 * hash + co[a] / 4;
|
|
|
|
|
|
|
|
set_euland3(c, co[0]*120, co[1]*120, (co[1]+co[2]) / dv, hash);
|
|
|
|
}
|
|
|
|
|
2019-08-09 20:37:11 +00:00
|
|
|
EX int dist_relative(cell *c) {
|
2019-04-11 22:17:50 +00:00
|
|
|
auto m = cubemap();
|
|
|
|
auto& cc = m->camelot_center;
|
|
|
|
int r = roundTableRadius(NULL);
|
|
|
|
cell *start = m->gamestart();
|
|
|
|
if(!cc) {
|
|
|
|
cc = start;
|
|
|
|
while(euclid3::celldistance(cc, start) < r + 5)
|
|
|
|
cc = cc->cmove(hrand(cc->type));
|
|
|
|
}
|
|
|
|
|
|
|
|
return euclid3::celldistance(cc, c) - r;
|
|
|
|
}
|
|
|
|
|
2019-04-15 21:29:07 +00:00
|
|
|
/* quotient spaces */
|
|
|
|
|
|
|
|
int determinant(const intmatrix T) {
|
|
|
|
int det = 0;
|
|
|
|
for(int i=0; i<3; i++)
|
|
|
|
det += T[0][i] * T[1][(i+1)%3] * T[2][(i+2)%3];
|
|
|
|
for(int i=0; i<3; i++)
|
|
|
|
det -= T[0][i] * T[1][(i+2)%3] * T[2][(i+1)%3];
|
|
|
|
return det;
|
|
|
|
}
|
|
|
|
|
|
|
|
intmatrix scaled_inverse(const intmatrix T) {
|
|
|
|
intmatrix T2;
|
|
|
|
for(int i=0; i<3; i++)
|
|
|
|
for(int j=0; j<3; j++)
|
|
|
|
T2[j][i] = (T[(i+1)%3][(j+1)%3] * T[(i+2)%3][(j+2)%3] - T[(i+1)%3][(j+2)%3] * T[(i+2)%3][(j+1)%3]);
|
|
|
|
return T2;
|
|
|
|
}
|
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
intmatrix user_axes;
|
|
|
|
intmatrix optimal_axes;
|
|
|
|
intmatrix regular_axes;
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-11-02 19:27:44 +00:00
|
|
|
intmatrix T, T2, T_edit;
|
2019-11-27 17:07:39 +00:00
|
|
|
EX int det;
|
2019-11-27 00:01:20 +00:00
|
|
|
int infinite_dims;
|
2019-11-02 19:27:44 +00:00
|
|
|
int twisted0, twisted_edit;
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-11-02 19:27:44 +00:00
|
|
|
EX void set_torus3(int x, int y, int z) {
|
|
|
|
for(int i=0; i<3; i++) for(int j=0; j<3; j++) T0[i][j] = 0;
|
|
|
|
tie(T0[0][0], T0[1][1], T0[2][2]) = make_tuple(x, y, z);
|
2019-11-02 21:17:57 +00:00
|
|
|
twisted = 0;
|
2019-11-02 19:27:44 +00:00
|
|
|
}
|
2019-11-02 21:17:57 +00:00
|
|
|
|
|
|
|
EX void clear_torus3() {
|
|
|
|
set_torus3(0, 0, 0);
|
|
|
|
}
|
2019-11-02 19:27:44 +00:00
|
|
|
|
2019-04-15 21:29:07 +00:00
|
|
|
unordered_map<coord, int> canonical_hash;
|
|
|
|
vector<coord> canonical_seq;
|
|
|
|
int canonical_index;
|
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
coord compute_cat(coord coo) {
|
|
|
|
coord cat = euzero;
|
2019-04-15 21:29:07 +00:00
|
|
|
for(int i=0; i<3; i++) {
|
|
|
|
int val = T2[0][i] * coo[0] + T2[1][i] * coo[1] + T2[2][i] * coo[2];
|
2019-11-27 00:01:20 +00:00
|
|
|
if(i < WDIM - infinite_dims) val = gmod(val, det);
|
2019-04-15 21:29:07 +00:00
|
|
|
cat += val * main_axes[i];
|
|
|
|
}
|
|
|
|
return cat;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
void add_canonical(coord val) {
|
|
|
|
auto cat = compute_cat(val);
|
|
|
|
if(canonical_hash.count(cat)) return;
|
|
|
|
canonical_hash[cat] = isize(canonical_seq);
|
|
|
|
canonical_seq.push_back(val);
|
|
|
|
}
|
|
|
|
|
2019-11-29 13:15:49 +00:00
|
|
|
EX bool valid_third_turn(const intmatrix& m) {
|
|
|
|
if(T0[0][2] != -T0[0][0]-T0[0][1]) return false;
|
|
|
|
if(T0[1][0] != T0[0][1]) return false;
|
|
|
|
if(T0[1][1] != T0[0][2]) return false;
|
|
|
|
if(T0[1][2] != T0[0][0]) return false;
|
|
|
|
if(T0[2][0] != T0[2][1]) return false;
|
|
|
|
if(T0[2][0] != T0[2][2]) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
EX void build_torus3(eGeometry g) {
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
int dim = ginf[g].g.gameplay_dimension;
|
2019-11-29 13:34:40 +00:00
|
|
|
|
|
|
|
user_axes = T0;
|
|
|
|
if(dim == 2) user_axes[2] = euzero;
|
2019-04-15 21:29:07 +00:00
|
|
|
|
|
|
|
optimal_axes = user_axes;
|
|
|
|
|
|
|
|
again:
|
2019-11-29 13:34:40 +00:00
|
|
|
for(int i=0; i<dim; i++) if(optimal_axes[i] < euzero) optimal_axes[i] = -optimal_axes[i];
|
2019-04-15 21:29:07 +00:00
|
|
|
if(optimal_axes[0] < optimal_axes[1]) swap(optimal_axes[0], optimal_axes[1]);
|
2019-11-27 00:01:20 +00:00
|
|
|
if(optimal_axes[1] < optimal_axes[dim]) swap(optimal_axes[1], optimal_axes[dim]);
|
2019-04-15 21:29:07 +00:00
|
|
|
if(optimal_axes[0] < optimal_axes[1]) swap(optimal_axes[0], optimal_axes[1]);
|
|
|
|
for(int i=0; i<3; i++) {
|
|
|
|
int i1 = (i+1) % 3;
|
|
|
|
int i2 = (i+2) % 3;
|
|
|
|
for(int a=-10; a<=10; a++)
|
|
|
|
for(int b=-10; b<=10; b++) {
|
|
|
|
coord cand = optimal_axes[i] + optimal_axes[i1] * a + optimal_axes[i2] * b;
|
|
|
|
if(celldistance(cand) < celldistance(optimal_axes[i])) {
|
|
|
|
optimal_axes[i] = cand;
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
regular_axes = optimal_axes;
|
2019-11-27 00:01:20 +00:00
|
|
|
infinite_dims = dim;
|
2019-11-29 13:34:40 +00:00
|
|
|
for(int i=0; i<dim; i++) if(optimal_axes[i] != euzero) infinite_dims--;
|
2019-11-27 00:01:20 +00:00
|
|
|
|
2019-04-15 21:29:07 +00:00
|
|
|
int attempt = 0;
|
|
|
|
next_attempt:
|
2019-11-27 00:01:20 +00:00
|
|
|
for(int i=dim-infinite_dims; i<3; i++)
|
2019-04-15 21:29:07 +00:00
|
|
|
regular_axes[i] = main_axes[(attempt+i)%3];
|
2019-11-27 00:01:20 +00:00
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
T = regular_axes;
|
2019-04-15 21:29:07 +00:00
|
|
|
det = determinant(T);
|
|
|
|
if(det == 0) {
|
|
|
|
attempt++;
|
|
|
|
if(attempt == 3) {
|
|
|
|
println(hlog, "weird singular!\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
goto next_attempt;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(det < 0) det = -det;
|
|
|
|
|
|
|
|
T2 = scaled_inverse(T);
|
|
|
|
canonical_hash.clear();
|
|
|
|
canonical_seq.clear();
|
|
|
|
canonical_index = 0;
|
2019-11-29 13:34:40 +00:00
|
|
|
add_canonical(euzero);
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-04-16 01:22:52 +00:00
|
|
|
twisted = twisted0;
|
2019-11-27 00:01:20 +00:00
|
|
|
if(dim == 3) {
|
2019-11-29 13:15:49 +00:00
|
|
|
if(valid_third_turn(T0)) {
|
|
|
|
twisted &= 16;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
twisted &= 7;
|
|
|
|
if(g != gCubeTiling && ((T0[0][0]+T0[2][2]) & 1)) twisted &=~ 1;
|
|
|
|
if(g != gCubeTiling && ((T0[1][1]+T0[2][2]) & 1)) twisted &=~ 2;
|
|
|
|
for(int i=0; i<3; i++) for(int j=0; j<3; j++)
|
|
|
|
if(i != j && T0[i][j]) twisted = 0;
|
|
|
|
if(T0[2][2] == 0) twisted = 0;
|
|
|
|
if(T0[0][0] != T0[1][1]) twisted &= 3;
|
|
|
|
}
|
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
else {
|
|
|
|
twisted &= 8;
|
|
|
|
twisted_vec = as_gp(T0[1]);
|
|
|
|
ortho_vec = as_gp(T0[0]);
|
|
|
|
if(twisted_vec == gp::loc{0,0}) twisted = 0;
|
|
|
|
if(chiral(twisted_vec)) twisted = 0;
|
|
|
|
if(dscalar(twisted_vec, ortho_vec))
|
|
|
|
twisted = 0;
|
2019-04-15 21:29:07 +00:00
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
|
|
|
|
set_flag(ginf[g].flags, qANYQ, infinite_dims < dim);
|
|
|
|
set_flag(ginf[g].flags, qBOUNDED, infinite_dims == 0);
|
2019-11-27 23:14:57 +00:00
|
|
|
set_flag(ginf[g].flags, qSMALL, infinite_dims == 0 && det <= 4096);
|
2019-11-27 00:01:20 +00:00
|
|
|
bool nonori = false;
|
|
|
|
if(twisted&1) nonori = !nonori;
|
|
|
|
if(twisted&2) nonori = !nonori;
|
|
|
|
if(twisted&4) nonori = !nonori;
|
2019-11-28 20:53:56 +00:00
|
|
|
if(twisted&8) nonori = !nonori;
|
2019-11-27 00:01:20 +00:00
|
|
|
set_flag(ginf[g].flags, qNONORIENTABLE, nonori);
|
|
|
|
}
|
|
|
|
|
|
|
|
void build_torus3() {
|
|
|
|
for(eGeometry g: { gEuclid, gEuclidSquare, gCubeTiling, gRhombic3, gBitrunc3})
|
|
|
|
build_torus3(g);
|
2019-04-15 21:29:07 +00:00
|
|
|
}
|
|
|
|
|
2019-04-16 01:22:52 +00:00
|
|
|
void swap01(transmatrix& M) {
|
|
|
|
for(int i=0; i<4; i++) swap(M[i][0], M[i][1]);
|
|
|
|
}
|
2019-11-27 21:01:36 +00:00
|
|
|
|
|
|
|
gp::loc ort1() { return (S3 == 3 ? gp::loc(1, -2) : gp::loc(0, 1)); }
|
2019-11-29 13:15:49 +00:00
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
int diagonal_cross(const coord& a, const coord& b) {
|
2019-11-29 13:15:49 +00:00
|
|
|
return a[0]*b[1] + a[1]*b[2] + a[2]*b[0]
|
|
|
|
- b[0]*a[1] - b[1]*a[2] - b[2]*a[0];
|
|
|
|
};
|
2019-11-27 21:01:36 +00:00
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
EX coord twist(coord x, coord& d, transmatrix& M, bool& mirr) {
|
2019-11-29 13:15:49 +00:00
|
|
|
if(!twisted) return x;
|
|
|
|
if(twisted & 16) {
|
|
|
|
int period = T0[2][2];
|
|
|
|
transmatrix RotYZX = Zero;
|
|
|
|
RotYZX[0][1] = 1;
|
|
|
|
RotYZX[1][2] = 1;
|
|
|
|
RotYZX[2][0] = 1;
|
|
|
|
RotYZX[3][3] = 1;
|
2019-11-29 13:34:40 +00:00
|
|
|
auto& coo = x;
|
2019-11-29 13:15:49 +00:00
|
|
|
while(true) {
|
|
|
|
auto coosum = coo[0] + coo[1] + coo[2];
|
|
|
|
if(coosum >= 3 * period) {
|
|
|
|
coo[0] -= period, coo[1] -= period, coo[2] -= period;
|
|
|
|
tie(d[0], d[1], d[2]) = make_tuple(d[1], d[2], d[0]);
|
|
|
|
M = M * RotYZX;
|
|
|
|
}
|
|
|
|
else if(coosum < 0) {
|
|
|
|
coo[0] += period, coo[1] += period, coo[2] += period;
|
|
|
|
tie(d[0], d[1], d[2]) = make_tuple(d[2], d[0], d[1]);
|
|
|
|
M = M * RotYZX * RotYZX;
|
|
|
|
}
|
|
|
|
else break;
|
|
|
|
}
|
2019-11-29 13:34:40 +00:00
|
|
|
if(T0[0] != euzero) {
|
2019-11-29 13:15:49 +00:00
|
|
|
while(diagonal_cross(coo, T0[1]) < 0) for(int i=0; i<3; i++) coo[i] -= T0[0][i];
|
|
|
|
while(diagonal_cross(coo, T0[1]) > 0) for(int i=0; i<3; i++) coo[i] += T0[0][i];
|
|
|
|
while(diagonal_cross(coo, T0[0]) > 0) for(int i=0; i<3; i++) coo[i] -= T0[1][i];
|
|
|
|
while(diagonal_cross(coo, T0[0]) < 0) for(int i=0; i<3; i++) coo[i] += T0[1][i];
|
|
|
|
}
|
2019-11-29 13:34:40 +00:00
|
|
|
return coo;
|
2019-11-29 13:15:49 +00:00
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
if(WDIM == 3) {
|
2019-11-29 13:34:40 +00:00
|
|
|
auto& coo = x;
|
2019-11-27 00:01:20 +00:00
|
|
|
while(coo[2] >= T0[2][2]) {
|
|
|
|
coo[2] -= T0[2][2];
|
|
|
|
if(twisted & 1) coo[0] *= -1, d[0] *= -1, M = M * MirrorX;
|
|
|
|
if(twisted & 2) coo[1] *= -1, d[1] *= -1, M = M * MirrorY;
|
|
|
|
if(twisted & 4) swap(coo[0], coo[1]), swap01(M), swap(d[0], d[1]);
|
|
|
|
}
|
|
|
|
while(coo[2] < 0) {
|
|
|
|
coo[2] += T0[2][2];
|
|
|
|
if(twisted & 4) swap(coo[0], coo[1]), swap(d[0], d[1]), swap01(M);
|
|
|
|
if(twisted & 1) coo[0] *= -1, d[0] *= -1, M = M * MirrorX;
|
|
|
|
if(twisted & 2) coo[1] *= -1, d[1] *= -1, M = M * MirrorY;
|
|
|
|
}
|
|
|
|
for(int i: {0,1})
|
|
|
|
if(T0[i][i]) coo[i] = gmod(coo[i], T0[i][i]);
|
2019-11-29 13:34:40 +00:00
|
|
|
return coo;
|
2019-04-16 01:22:52 +00:00
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
else {
|
2019-11-29 13:34:40 +00:00
|
|
|
gp::loc coo = as_gp(x);
|
2019-11-27 21:01:36 +00:00
|
|
|
gp::loc ort = ort1() * twisted_vec;
|
2019-11-27 00:01:20 +00:00
|
|
|
int dsc = dscalar(twisted_vec, twisted_vec);
|
|
|
|
gp::loc d0 (d[0], d[1]);
|
|
|
|
hyperpoint h = eumove(as_coord(twisted_vec)) * C0;
|
|
|
|
while(true) {
|
|
|
|
int dsx = dscalar(coo, twisted_vec);
|
|
|
|
if(dsx >= dsc) coo = coo - twisted_vec;
|
|
|
|
else if (dsx < 0) coo = coo + twisted_vec;
|
|
|
|
else break;
|
|
|
|
M = M * spintox(h) * MirrorY * rspintox(h);
|
|
|
|
auto s = ort * dscalar(d0, ort) * 2;
|
|
|
|
auto v = dscalar(ort, ort);
|
|
|
|
s.first /= v;
|
|
|
|
s.second /= v;
|
|
|
|
d0 = d0 - s;
|
|
|
|
s = ort * dscalar(coo, ort) * 2;
|
|
|
|
s.first /= v;
|
|
|
|
s.second /= v;
|
|
|
|
coo = coo - s;
|
|
|
|
mirr = !mirr;
|
|
|
|
}
|
|
|
|
if(ortho_vec != gp::loc{0,0}) {
|
|
|
|
int osc = dscalar(ortho_vec, ortho_vec);
|
|
|
|
while(true) {
|
|
|
|
int dsx = dscalar(coo, ortho_vec);
|
|
|
|
if(dsx >= osc) coo = coo - ortho_vec;
|
|
|
|
else if(dsx < 0) coo = coo + ortho_vec;
|
|
|
|
else break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
d[0] = d0.first; d[1] = d0.second;
|
2019-11-29 13:15:49 +00:00
|
|
|
return as_coord(coo);
|
2019-04-16 01:22:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-15 21:29:07 +00:00
|
|
|
coord canonicalize(coord x) {
|
2019-04-16 01:22:52 +00:00
|
|
|
if(twisted) {
|
|
|
|
transmatrix M = Id;
|
2019-11-29 13:34:40 +00:00
|
|
|
auto dummy = euzero;
|
2019-11-27 00:01:20 +00:00
|
|
|
bool dm = false;
|
|
|
|
return twist(x, dummy, M, dm);
|
2019-04-16 01:22:52 +00:00
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
if(infinite_dims == WDIM) return x;
|
|
|
|
if(infinite_dims == WDIM-1) {
|
2019-04-15 21:29:07 +00:00
|
|
|
while(celldistance(x + optimal_axes[0]) <= celldistance(x)) x += optimal_axes[0];
|
|
|
|
while(celldistance(x - optimal_axes[0]) < celldistance(x)) x -= optimal_axes[0];
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
auto cat = compute_cat(x);
|
|
|
|
auto& st = cubemap()->shifttable;
|
|
|
|
while(!canonical_hash.count(cat)) {
|
|
|
|
if(canonical_index == isize(canonical_seq)) throw hr_exception();
|
|
|
|
auto v = canonical_seq[canonical_index++];
|
|
|
|
for(auto s: st) add_canonical(v + s);
|
|
|
|
}
|
|
|
|
return canonical_seq[canonical_hash[cat]];
|
|
|
|
}
|
|
|
|
|
2019-09-06 06:17:02 +00:00
|
|
|
EX void prepare_torus3() {
|
2019-04-15 21:29:07 +00:00
|
|
|
T_edit = T0;
|
2019-04-16 01:22:52 +00:00
|
|
|
twisted_edit = twisted0;
|
2019-04-15 21:29:07 +00:00
|
|
|
}
|
2019-11-27 00:04:15 +00:00
|
|
|
|
|
|
|
EX void show_fundamental() {
|
|
|
|
initquickqueue();
|
|
|
|
transmatrix M = ggmatrix(cwt.at);
|
|
|
|
hyperpoint h0 = M*C0;
|
2019-11-29 13:34:40 +00:00
|
|
|
hyperpoint ha = M*(eumove(T_edit[0]) * C0 - C0) / 2;
|
|
|
|
hyperpoint hb = M*(eumove(T_edit[1]) * C0 - C0) / 2;
|
2019-11-27 00:04:15 +00:00
|
|
|
if(WDIM == 3) {
|
2019-11-29 13:34:40 +00:00
|
|
|
hyperpoint hc = M*(eumove(T_edit[2]) * C0 - C0) / 2;
|
2019-11-27 00:04:15 +00:00
|
|
|
for(int d:{-1,1}) for(int e:{-1,1}) {
|
|
|
|
queueline(h0+d*ha+e*hb-hc, h0+d*ha+e*hb+hc, 0xFFFFFFFF);
|
|
|
|
queueline(h0+d*hb+e*hc-ha, h0+d*hb+e*hc+ha, 0xFFFFFFFF);
|
|
|
|
queueline(h0+d*hc+e*ha-hb, h0+d*hc+e*ha+hb, 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
queueline(h0+ha+hb, h0+ha-hb, 0xFFFFFFFF);
|
|
|
|
queueline(h0-ha+hb, h0-ha-hb, 0xFFFFFFFF);
|
|
|
|
queueline(h0+ha+hb, h0-ha+hb, 0xFFFFFFFF);
|
|
|
|
queueline(h0+ha-hb, h0-ha-hb, 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
quickqueue();
|
|
|
|
}
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-11-27 21:01:36 +00:00
|
|
|
using torus_config = pair<euclid3::intmatrix, int>;
|
|
|
|
|
|
|
|
euclid3::intmatrix on_periods(gp::loc a, gp::loc b) {
|
|
|
|
euclid3::intmatrix res;
|
|
|
|
for(int i=0; i<3; i++) for(int j=0; j<3; j++) res[i][j] = 0;
|
|
|
|
res[0][0] = a.first;
|
|
|
|
res[0][1] = a.second;
|
|
|
|
res[1][0] = b.first;
|
|
|
|
res[1][1] = b.second;
|
|
|
|
res[2][2] = 1;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
torus_config single_row_torus(int qty, int dy) {
|
2019-11-27 23:15:14 +00:00
|
|
|
return { on_periods({dy, -1}, {qty, 0}), false };
|
2019-11-27 21:01:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
torus_config regular_torus(gp::loc p) {
|
|
|
|
return { on_periods(p, gp::loc(0,1) * p), false };
|
|
|
|
}
|
|
|
|
|
|
|
|
torus_config rectangular_torus(int x, int y, bool klein) {
|
|
|
|
if(S3 == 3) y /= 2;
|
|
|
|
return { on_periods(euclid3::ort1() * gp::loc(y,0), gp::loc(x,0)), klein?8:0 };
|
|
|
|
}
|
|
|
|
|
|
|
|
void torus_config_option(string name, char key, torus_config tc) {
|
|
|
|
dialog::addBoolItem(name, make_pair(T_edit, twisted_edit) == tc && PURE, key);
|
|
|
|
dialog::add_action([tc] {
|
|
|
|
tie(euclid3::T0, euclid3::twisted0) = tc;
|
|
|
|
tie(T_edit, twisted_edit) = tc;
|
|
|
|
set_variation(eVariation::pure);
|
|
|
|
start_game();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-08-09 20:37:11 +00:00
|
|
|
EX void show_torus3() {
|
2019-11-27 00:01:20 +00:00
|
|
|
int dim = WDIM;
|
|
|
|
cmode = sm::SIDE | sm::MAYDARK | sm::TORUSCONFIG;
|
2019-04-15 21:29:07 +00:00
|
|
|
gamescreen(1);
|
2019-11-27 00:01:20 +00:00
|
|
|
dialog::init(XLAT("Euclidean quotient spaces"));
|
|
|
|
|
|
|
|
for(int y=0; y<dim+1; y++)
|
2019-04-15 21:29:07 +00:00
|
|
|
dialog::addBreak(100);
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
dialog::addInfo(XLAT("columns specify periods"));
|
|
|
|
dialog::addInfo(XLAT("(vectors you need to take to get back to start)"));
|
|
|
|
|
2019-04-16 01:22:52 +00:00
|
|
|
dialog::addBreak(50);
|
|
|
|
|
2019-11-27 00:04:15 +00:00
|
|
|
show_fundamental();
|
2019-11-27 00:01:20 +00:00
|
|
|
if(dim == 3) {
|
|
|
|
bool nondiag = false;
|
|
|
|
for(int i=0; i<dim; i++)
|
|
|
|
for(int j=0; j<dim; j++)
|
|
|
|
if(T_edit[i][j] && i != j) nondiag = true;
|
|
|
|
|
|
|
|
if(nondiag) {
|
|
|
|
dialog::addInfo(XLAT("twisting implemented only for diagonal matrices"));
|
|
|
|
dialog::addBreak(200);
|
|
|
|
}
|
|
|
|
else if(T_edit[dim-1][dim-1] == 0) {
|
|
|
|
dialog::addInfo(XLAT("nothing to twist"));
|
|
|
|
dialog::addInfo(XLAT("change the bottom left corner"));
|
|
|
|
dialog::addBreak(100);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
auto g = geometry;
|
|
|
|
if(g == gCubeTiling || (T_edit[0][0]+T_edit[2][2]) % 2 == 0)
|
|
|
|
dialog::addBoolItem(XLAT("flip X coordinate"), twisted_edit & 1, 'x');
|
|
|
|
else
|
|
|
|
dialog::addBoolItem(XLAT("flipping X impossible"), twisted_edit & 1, 'x');
|
|
|
|
dialog::add_action([] { twisted_edit ^= 1; });
|
|
|
|
|
|
|
|
if(g == gCubeTiling || (T_edit[1][1]+T_edit[2][2]) % 2 == 0)
|
|
|
|
dialog::addBoolItem(XLAT("flip Y coordinate"), twisted_edit & 2, 'y');
|
|
|
|
else
|
|
|
|
dialog::addBoolItem(XLAT("flipping Y impossible"), twisted_edit & 2, 'y');
|
|
|
|
dialog::add_action([] { twisted_edit ^= 2; });
|
|
|
|
|
|
|
|
if(T_edit[0][0] == T_edit[1][1])
|
|
|
|
dialog::addBoolItem(XLAT("swap X and Y"), twisted_edit & 4, 'z');
|
|
|
|
else
|
|
|
|
dialog::addBoolItem(XLAT("swapping impossible"), twisted_edit & 4, 'z');
|
|
|
|
dialog::add_action([] { twisted_edit ^= 4; });
|
|
|
|
}
|
2019-04-16 01:22:52 +00:00
|
|
|
}
|
|
|
|
else {
|
2019-11-27 00:01:20 +00:00
|
|
|
if(T_edit[1][0] == 0 && T_edit[1][1] == 0)
|
|
|
|
dialog::addInfo(XLAT("change the second column for Möbius bands and Klein bottles"));
|
|
|
|
else if(chiral(as_gp(T_edit[1])))
|
|
|
|
dialog::addInfo(XLAT("second period is chiral -- cannot be mirrored"));
|
|
|
|
else if(dscalar(as_gp(T_edit[1]), as_gp(T_edit[0])))
|
|
|
|
dialog::addInfo(XLAT("periods must be orthogonal for mirroring"));
|
|
|
|
else {
|
|
|
|
dialog::addBoolItem(XLAT("mirror flip in the second period"), twisted_edit & 8, 'x');
|
|
|
|
dialog::add_action([] { twisted_edit ^= 8; });
|
|
|
|
}
|
2019-11-27 21:01:36 +00:00
|
|
|
|
|
|
|
dialog::addBreak(50);
|
|
|
|
torus_config_option(XLAT("single-cell torus"), 'A', regular_torus({1,0}));
|
|
|
|
torus_config_option(XLAT("large regular torus"), 'B', regular_torus({12, 0}));
|
|
|
|
torus_config_option(XLAT("Klein bottle"), 'C', rectangular_torus(12, 6, true));
|
|
|
|
torus_config_option(XLAT("cylinder"), 'D', rectangular_torus(6, 0, false));
|
|
|
|
torus_config_option(XLAT("Möbius band"), 'E', rectangular_torus(6, 0, true));
|
|
|
|
if(S3 == 3) torus_config_option(XLAT("seven-colorable torus"), 'F', regular_torus({1,2}));
|
|
|
|
if(S3 == 3) torus_config_option(XLAT("HyperRogue classic torus"), 'G', single_row_torus(381, -22));
|
2019-04-16 01:22:52 +00:00
|
|
|
}
|
2019-11-28 19:37:43 +00:00
|
|
|
|
|
|
|
dialog::addBreak(50);
|
|
|
|
dialog::addBoolItem(XLAT("standard rotation"), eqmatrix(models::euclidean_spin, Id), 's');
|
|
|
|
dialog::add_action([] { rotate_view(models::euclidean_spin); });
|
|
|
|
|
2019-11-28 19:37:56 +00:00
|
|
|
#if CAP_RUG
|
|
|
|
if(GDIM == 2) {
|
|
|
|
dialog::addBoolItem(XLAT("hypersian rug mode"), (rug::rugged), 'u');
|
|
|
|
dialog::add_action(rug::select);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-04-16 01:22:52 +00:00
|
|
|
dialog::addBreak(50);
|
|
|
|
|
2019-04-15 21:29:07 +00:00
|
|
|
char xch = 'p';
|
|
|
|
for(eGeometry g: {gCubeTiling, gRhombic3, gBitrunc3}) {
|
2019-11-27 00:01:20 +00:00
|
|
|
if(dim == 2) g = geometry;
|
2019-04-15 21:29:07 +00:00
|
|
|
dialog::addItem(XLAT(ginf[g].menu_displayed_name), xch++);
|
|
|
|
dialog::add_action([g] {
|
|
|
|
stop_game();
|
|
|
|
set_geometry(g);
|
|
|
|
T0 = T_edit;
|
2019-04-16 01:22:52 +00:00
|
|
|
twisted0 = twisted_edit;
|
2019-04-15 21:29:07 +00:00
|
|
|
start_game();
|
|
|
|
});
|
2019-11-27 00:01:20 +00:00
|
|
|
if(dim == 2) break;
|
2019-04-15 21:29:07 +00:00
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
|
2019-04-15 21:29:07 +00:00
|
|
|
dialog::addBreak(50);
|
|
|
|
dialog::addBack();
|
|
|
|
dialog::display();
|
|
|
|
|
|
|
|
int i = -1;
|
|
|
|
for(auto& v: dialog::items) if(v.type == dialog::diBreak) {
|
2019-11-27 00:01:20 +00:00
|
|
|
if(i >= 0 && i < dim) {
|
|
|
|
for(int j=0; j < dim; j++) {
|
2019-04-15 21:29:07 +00:00
|
|
|
char ch = 'a' + i * 3 + j;
|
2019-11-27 00:01:20 +00:00
|
|
|
if(displayfr(dialog::dcenter + dialog::dfspace * 4 * (j-(dim-1.)/2), v.position, 2, dialog::dfsize, its(T_edit[j][i]), 0xFFFFFF, 8))
|
2019-04-15 21:29:07 +00:00
|
|
|
getcstat = ch;
|
|
|
|
dialog::add_key_action(ch, [=] {
|
2019-04-17 23:04:22 +00:00
|
|
|
dialog::editNumber(T_edit[j][i], -10, +10, 1, 0, "", XLAT(
|
|
|
|
"This matrix lets you play on the quotient spaces of three-dimensional. "
|
|
|
|
"Euclidean space. Every column specifies a translation vector which "
|
|
|
|
"takes you back to the starting point. For example, if you put "
|
|
|
|
"set 2, 6, 0 on the diagonal, you get back to the starting point "
|
|
|
|
"if you move 2 steps in the X direction, 6 steps in the Y direction "
|
|
|
|
"(the quotient space is infinite in the Z direction).\n\n"
|
|
|
|
"You can also introduce twists for diagonal matrices: after going "
|
|
|
|
"the given number of steps in the Z direction, the space is also "
|
|
|
|
"mirrored or rotated. (More general 'twisted' spaces are currently "
|
|
|
|
"not implemented.)"
|
|
|
|
)
|
2019-04-15 21:29:07 +00:00
|
|
|
);
|
2019-11-27 00:04:15 +00:00
|
|
|
dialog::extra_options = show_fundamental;
|
2019-04-15 21:29:07 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-28 07:54:10 +00:00
|
|
|
#if CAP_COMMANDLINE
|
2019-04-15 21:29:07 +00:00
|
|
|
int euArgs() {
|
|
|
|
using namespace arg;
|
|
|
|
|
|
|
|
if(0) ;
|
|
|
|
else if(argis("-t3")) {
|
|
|
|
PHASEFROM(2);
|
|
|
|
stop_game();
|
|
|
|
for(int i=0; i<3; i++)
|
|
|
|
for(int j=0; j<3; j++) {
|
|
|
|
shift(); T0[i][j] = argi();
|
|
|
|
}
|
|
|
|
build_torus3();
|
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
else if(argis("-t2")) {
|
|
|
|
PHASEFROM(2);
|
|
|
|
stop_game();
|
|
|
|
for(int i=0; i<2; i++)
|
|
|
|
for(int j=0; j<2; j++) {
|
|
|
|
shift(); T0[i][j] = argi();
|
|
|
|
}
|
|
|
|
shift(); twisted0 = argi();
|
|
|
|
build_torus3();
|
|
|
|
}
|
2019-11-29 13:15:49 +00:00
|
|
|
else if(argis("-twistthird")) {
|
|
|
|
PHASEFROM(2);
|
|
|
|
stop_game();
|
|
|
|
shift(); T0[0][0] = argi();
|
|
|
|
shift(); T0[0][1] = argi();
|
|
|
|
shift(); T0[2][0] = argi();
|
|
|
|
T0[0][2] = -T0[0][0]-T0[0][1];
|
|
|
|
T0[1][0] = T0[0][1];
|
|
|
|
T0[1][1] = T0[0][2];
|
|
|
|
T0[1][2] = T0[0][0];
|
|
|
|
T0[2][1] = T0[2][2] = argi();
|
|
|
|
twisted0 = 16;
|
|
|
|
build_torus3();
|
|
|
|
}
|
2019-04-16 01:22:52 +00:00
|
|
|
else if(argis("-twist3")) {
|
|
|
|
PHASEFROM(2);
|
|
|
|
stop_game();
|
|
|
|
for(int i=0; i<3; i++)
|
|
|
|
for(int j=0; j<3; j++) T0[i][j] = 0;
|
|
|
|
|
|
|
|
for(int i=0; i<3; i++) {
|
|
|
|
shift(); T0[i][i] = argi();
|
|
|
|
}
|
|
|
|
shift(); twisted0 = argi();
|
|
|
|
build_torus3();
|
|
|
|
}
|
2019-04-16 02:14:36 +00:00
|
|
|
else if(argis("-twisttest")) {
|
|
|
|
start_game();
|
|
|
|
celllister cl(cwt.at, 10000, 10000, NULL);
|
|
|
|
for(cell *c: cl.lst) {
|
2019-11-27 00:01:20 +00:00
|
|
|
heptagon *h = c->master;
|
2019-04-16 02:14:36 +00:00
|
|
|
for(int i=0; i<S7; i++)
|
|
|
|
for(int j=0; j<S7; j++)
|
|
|
|
for(int k=0; k<S7; k++)
|
|
|
|
for(int l=0; l<S7; l++)
|
2019-11-27 00:01:20 +00:00
|
|
|
if(h->move(i) && c->move(k) && h->move(i)->move(j) == h->move(k)->move(l) && h->move(i)->move(j)) {
|
|
|
|
transmatrix T1 = move_matrix(h, i) * move_matrix(h->move(i), j);
|
|
|
|
transmatrix T2 = move_matrix(h, k) * move_matrix(h->move(k), l);
|
2019-04-16 02:14:36 +00:00
|
|
|
if(!eqmatrix(T1, T2)) {
|
2019-11-29 13:34:40 +00:00
|
|
|
println(hlog, c, " @ ", cubemap()->ispacemap[c->master], " : ", i, "/", j, "/", k, "/", l, " :: ", T1, " vs ", T2);
|
2019-04-16 02:14:36 +00:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-04-15 21:29:07 +00:00
|
|
|
|
|
|
|
else return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto euhook = addHook(hooks_args, 100, euArgs);
|
2019-06-28 07:54:10 +00:00
|
|
|
#endif
|
2019-08-09 20:37:11 +00:00
|
|
|
EX }
|
2019-04-15 21:29:07 +00:00
|
|
|
|
2019-02-24 21:12:32 +00:00
|
|
|
#endif
|
2019-03-08 21:38:44 +00:00
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
EX int dscalar(gp::loc e1, gp::loc e2) {
|
|
|
|
return 2 * (e1.first * e2.first + e1.second*e2.second) + (S3 == 3 ? e1.first*e2.second + e2.first * e1.second : 0);
|
2019-03-08 21:38:44 +00:00
|
|
|
}
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
EX int dsquare(gp::loc e) { return dscalar(e, e)/2; }
|
2019-03-08 21:38:44 +00:00
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
EX int dcross(gp::loc e1, gp::loc e2) {
|
|
|
|
return e1.first * e2.second - e1.second*e2.first;
|
|
|
|
}
|
2019-03-08 21:38:44 +00:00
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
EX gp::loc euc2_coordinates(cell *c) {
|
2019-11-29 13:34:40 +00:00
|
|
|
auto ans = euclid3::eucmap()->ispacemap[c->master];
|
2019-11-27 20:21:38 +00:00
|
|
|
if(BITRUNCATED)
|
|
|
|
return as_gp(ans) * gp::loc(1,1) + (c == c->master->c7 ? gp::loc(0,0) : gp::eudir((c->c.spin(0)+4)%6));
|
|
|
|
if(GOLDBERG) {
|
|
|
|
auto li = gp::get_local_info(c);
|
|
|
|
gp::loc shift(0,0);
|
|
|
|
if(li.first_dir >= 0) shift = gp::eudir(li.last_dir) * li.relative;
|
|
|
|
return as_gp(ans) * gp::param + shift;
|
|
|
|
}
|
|
|
|
return as_gp(ans);
|
2019-03-08 21:38:44 +00:00
|
|
|
}
|
|
|
|
|
2019-11-27 20:21:38 +00:00
|
|
|
/** this is slow, but we use it only for small p's */
|
2019-11-27 00:01:20 +00:00
|
|
|
EX cell* at_euc2_coordinates(gp::loc p) {
|
2019-11-27 20:21:38 +00:00
|
|
|
cellwalker cw(currentmap->gamestart());
|
|
|
|
while(p.first--) cw += revstep;
|
|
|
|
cw ++;
|
|
|
|
while(p.second--) cw += revstep;
|
|
|
|
return cw.at;
|
2019-11-27 00:01:20 +00:00
|
|
|
}
|
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
EX euclid3::coord as_coord(gp::loc p) { return euclid3::coord(p.first, p.second, 0); }
|
2019-11-27 00:01:20 +00:00
|
|
|
|
2019-11-27 21:01:48 +00:00
|
|
|
EX gp::loc sdxy() { return as_gp(euclid3::T[1]) * gp::univ_param(); }
|
2019-11-27 00:01:20 +00:00
|
|
|
|
|
|
|
EX pair<bool, string> coord_display(const transmatrix& V, cell *c) {
|
|
|
|
if(c != c->master->c7) return {false, ""};
|
2019-11-29 13:34:40 +00:00
|
|
|
hyperpoint hx = eumove(euclid3::main_axes[0]) * C0;
|
|
|
|
hyperpoint hy = eumove(euclid3::main_axes[1]) * C0;
|
2019-11-27 00:01:20 +00:00
|
|
|
hyperpoint hz = WDIM == 2 ? C0 : eumove(euclid3::main_axes[2]) * C0;
|
|
|
|
hyperpoint h = kz(inverse(build_matrix(hx, hy, hz, C03)) * inverse(ggmatrix(cwt.at->master->c7)) * V * C0);
|
|
|
|
|
|
|
|
if(WDIM == 3)
|
|
|
|
return {true, fts(h[0]) + "," + fts(h[1]) + "," + fts(h[2]) };
|
|
|
|
else
|
|
|
|
return {true, fts(h[0]) + "," + fts(h[1]) };
|
2019-09-05 09:57:38 +00:00
|
|
|
}
|
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
EX gp::loc as_gp(const euclid3::coord& v) { return gp::loc(v[0], v[1]); }
|
2019-11-27 00:01:20 +00:00
|
|
|
|
|
|
|
EX map<gp::loc, cdata>& get_cdata() { return euclid3::eucmap()->eucdata; }
|
|
|
|
|
2019-11-29 13:34:40 +00:00
|
|
|
EX transmatrix eumove(euclid3::coord co) {
|
2019-11-27 00:01:20 +00:00
|
|
|
constexpr double q3 = sqrt(double(3));
|
|
|
|
if(WDIM == 3) {
|
|
|
|
return eupush3(co[0], co[1], co[2]);
|
2019-09-05 09:57:38 +00:00
|
|
|
}
|
2019-11-27 00:01:20 +00:00
|
|
|
transmatrix Mat = Id;
|
|
|
|
if(a4) {
|
|
|
|
Mat[0][LDIM] += co[0] * cgi.tessf;
|
|
|
|
Mat[1][LDIM] += co[1] * cgi.tessf;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
Mat[0][LDIM] += (co[0] + co[1] * .5) * cgi.tessf;
|
|
|
|
Mat[1][LDIM] += co[1] * q3 /2 * cgi.tessf;
|
|
|
|
}
|
|
|
|
return Mat;
|
2019-09-05 09:57:38 +00:00
|
|
|
}
|
|
|
|
|
2019-11-27 00:01:20 +00:00
|
|
|
EX bool chiral(gp::loc g) {
|
|
|
|
int x = g.first;
|
|
|
|
int y = g.second;
|
|
|
|
if(x == 0) return false;
|
|
|
|
if(y == 0) return false;
|
|
|
|
if(x+y == 0) return false;
|
|
|
|
if(x==y) return false;
|
|
|
|
if(S3 == 3 && y == -2*x) return false;
|
|
|
|
if(S3 == 3 && x == -2*y) return false;
|
|
|
|
return true;
|
|
|
|
}
|
2019-09-05 09:57:38 +00:00
|
|
|
|
2019-11-27 20:08:09 +00:00
|
|
|
EX void twist_once(gp::loc coo) {
|
|
|
|
coo = coo - euclid3::twisted_vec * gp::univ_param();
|
|
|
|
if(euclid3::twisted&8) {
|
2019-11-27 21:01:36 +00:00
|
|
|
gp::loc ort = euclid3::ort1() * euclid3::twisted_vec * gp::univ_param();
|
2019-11-27 20:08:09 +00:00
|
|
|
auto s = ort * dscalar(coo, ort) * 2;
|
|
|
|
auto v = dscalar(ort, ort);
|
|
|
|
s.first /= v;
|
|
|
|
s.second /= v;
|
|
|
|
coo = coo - s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
EX int eudist(int sx, int sy, bool reduce IS(true)) {
|
|
|
|
int z0 = abs(sx);
|
|
|
|
int z1 = abs(sy);
|
|
|
|
if(a4 && BITRUNCATED)
|
|
|
|
return (z0 == z1 && z0 > 0 && reduce) ? z0+1: max(z0, z1);
|
|
|
|
if(a4) return z0 + z1;
|
|
|
|
int z2 = abs(sx+sy);
|
|
|
|
return max(max(z0,z1), z2);
|
|
|
|
}
|
|
|
|
|
|
|
|
EX int eudist(gp::loc a, gp::loc b) {
|
|
|
|
return eudist(a.first-b.first, a.second-b.second, (a.first ^ a.second)&1);
|
|
|
|
}
|
|
|
|
|
|
|
|
EX int cyldist(gp::loc a, gp::loc b) {
|
2019-11-29 13:34:40 +00:00
|
|
|
a = as_gp(euclid3::canonicalize(as_coord(a)));
|
|
|
|
b = as_gp(euclid3::canonicalize(as_coord(b)));
|
2019-11-27 20:08:09 +00:00
|
|
|
|
|
|
|
if(!quotient) return eudist(a, b);
|
|
|
|
|
|
|
|
int best = 0;
|
|
|
|
for(int sa=0; sa<16; sa++) {
|
|
|
|
auto _a = a, _b = b;
|
|
|
|
if(sa&1) twist_once(_a);
|
|
|
|
if(sa&2) twist_once(_b);
|
|
|
|
if(sa&4) _a = _a + euclid3::ortho_vec * gp::univ_param();
|
|
|
|
if(sa&8) _b = _b + euclid3::ortho_vec * gp::univ_param();
|
|
|
|
int val = eudist(_a, _b);
|
|
|
|
if(sa == 0 || val < best) best = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
return best;
|
|
|
|
}
|
|
|
|
|
2019-02-24 21:12:32 +00:00
|
|
|
}
|