mirror of
				https://github.com/zenorogue/hyperrogue.git
				synced 2025-10-25 02:47:40 +00:00 
			
		
		
		
	all Mobius bands implemented
This commit is contained in:
		
							
								
								
									
										143
									
								
								cell.cpp
									
									
									
									
									
								
							
							
						
						
									
										143
									
								
								cell.cpp
									
									
									
									
									
								
							| @@ -355,6 +355,105 @@ namespace torusconfig { | ||||
|     else | ||||
|       ginf[gTorus].quotientstyle |= qFULLTORUS; | ||||
|     } | ||||
|  | ||||
|   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); | ||||
|     } | ||||
|      | ||||
|   gp::loc sdxy() { return gp::loc(sdx, sdy); } | ||||
|    | ||||
|   int mobius_dir_basic() { | ||||
|     int dscalars[6]; | ||||
|     for(int a=0; a<SG6; a++) | ||||
|       dscalars[a] = dscalar(gp::eudir(a), sdxy()); | ||||
|     for(int a=0; a<SG6; a++) | ||||
|     for(int b=0; b<SG6; b++) | ||||
|       if(a != b && dscalars[a] == dscalars[b]) { | ||||
|         return (a + b) % SG6; | ||||
|       } | ||||
|     return -1; | ||||
|     } | ||||
|    | ||||
|   bool mobius_symmetric(bool square, int dx, int dy) { | ||||
|     dynamicval<eGeometry> g(geometry, square ? gEuclidSquare : gEuclid); | ||||
|     dynamicval<int> gx(sdx, dx); | ||||
|     dynamicval<int> gy(sdy, dy); | ||||
|     return mobius_dir_basic() != -1; | ||||
|     } | ||||
|  | ||||
|   void mobius_flip(int&x, int& y) { | ||||
|    | ||||
|     int d = mobius_dir_basic(); | ||||
|     int a, b; | ||||
|     if(d == 0) a = 1, b = SG6-1; | ||||
|     else a = 0, b = d; | ||||
|     auto p1 = gp::eudir(a); | ||||
|     auto p2 = gp::eudir(b); | ||||
|  | ||||
|     // x = sdx * s + px * t | ||||
|     // y = sdy * s + py * t | ||||
|     // py * x = py * sdx * s + px * py * t | ||||
|     // px * y = px * sdy * s + px + py * t | ||||
|     // py * x - px * y = py * sdx * s - px * sdy * s | ||||
|     // s = (py * x - px * y) / (py * sdx - px * sdy) | ||||
|      | ||||
|     int det = p1.second * sdx - p1.first * sdy; | ||||
|     int smul = p1.second * x - p1.first * y; | ||||
|     int tmul = sdx * y - sdy * x; | ||||
|      | ||||
|     x = (tmul * p2.first + smul * sdx) / det; | ||||
|     y = (tmul * p2.second + smul * sdy) / det; | ||||
|      | ||||
|     // println(hlog, make_pair(ox,oy), " [", d, "] ", make_pair(x,y), " p1 = ", p1, " p2 = ", p2, " det = ", det, " smul = ", smul, " tmul = ", tmul); | ||||
|     } | ||||
|  | ||||
|   int mobius_dir(cell *c) { | ||||
|     if(c->type == 8) return mobius_dir_basic() * 2; | ||||
|     else return mobius_dir_basic(); | ||||
|     } | ||||
|    | ||||
|   bool be_canonical(int& x, int& y) { | ||||
|     using namespace torusconfig; | ||||
|      | ||||
|     int periods = gdiv(dscalar(gp::loc(x,y), sdxy()), dscalar(sdxy(), sdxy())); | ||||
|      | ||||
|     y -= sdy * periods; | ||||
|     x -= sdx * periods;       | ||||
|      | ||||
|     bool b = false; | ||||
|  | ||||
|     if(nonorientable && (periods & 1)) { | ||||
|       mobius_flip(x, y); | ||||
|       b = true; | ||||
|       } | ||||
|      | ||||
|     return b; | ||||
|     } | ||||
|    | ||||
|   int cyldist(int id1, int id2) { | ||||
|    | ||||
|     int x1, y1, x2, y2; | ||||
|     tie(x1, y1) = vec_to_pair(id1); | ||||
|     tie(x2, y2) = vec_to_pair(id2); | ||||
|     be_canonical(x1, y1); | ||||
|     be_canonical(x2, y2); | ||||
|      | ||||
|     int dist = 1000000000; | ||||
|      | ||||
|     for(int a1=-1; a1<=1; a1++)  | ||||
|     for(int a2=-1; a2<=1; a2++) { | ||||
|       int ax1 = x1 + sdx * a1; | ||||
|       int ay1 = y1 + sdy * a1; | ||||
|       if(nonorientable && a1) mobius_flip(ax1, ay1); | ||||
|       int ax2 = x2 + sdx * a2; | ||||
|       int ay2 = y2 + sdy * a2; | ||||
|       if(nonorientable && a2) mobius_flip(ax2, ay2); | ||||
|       dist = min(dist, eudist(ax1 - ax2, ay1 - ay2)); | ||||
|        | ||||
|       } | ||||
|      | ||||
|     return dist; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| int euclid_getvec(int dx, int dy) { | ||||
| @@ -478,18 +577,8 @@ struct hrmap_euclidean : hrmap { | ||||
|     auto p = vec_to_pair(vec); | ||||
|     int x = p.first, y = p.second; | ||||
|     bool mobius = false; | ||||
|     if(euwrap) { | ||||
|       int zx = torusconfig::sdx; | ||||
|       int zy = torusconfig::sdy; | ||||
|        | ||||
|       int periods = gdiv(x * zx + y * zy, zx * zx + zy * zy); | ||||
|        | ||||
|       if(nonorientable) mobius = (periods&1) ? S6 : 0, periods &=~ 1; | ||||
|       y -= zy * periods; | ||||
|       x -= zx * periods;       | ||||
|        | ||||
|       if(mobius) x -= zx, y -= zy, y = -y; | ||||
|       } | ||||
|     if(euwrap)  | ||||
|       mobius = torusconfig::be_canonical(x, y); | ||||
|     euclideanSlab*& slab = euclidean[(y>>8)&(slabs-1)][(x>>8)&(slabs-1)]; | ||||
|     if(!slab) slab = new hrmap_euclidean::euclideanSlab; | ||||
|     return make_pair(&(slab->a[y&255][x&255]), mobius); | ||||
| @@ -510,7 +599,10 @@ struct hrmap_euclidean : hrmap { | ||||
| cellwalker vec_to_cellwalker(int vec) { | ||||
|   if(!fulltorus) { | ||||
|     auto p = euclideanAtCreate(vec); | ||||
|     return cellwalker(*p.first, 0, p.second); | ||||
|     if(p.second) | ||||
|       return cellwalker(*p.first, torusconfig::mobius_dir(*p.first), true); | ||||
|     else | ||||
|       return cellwalker(*p.first, 0, false); | ||||
|     } | ||||
|   else { | ||||
|     hrmap_torus *cur = torusmap(); | ||||
| @@ -528,7 +620,10 @@ int cellwalker_to_vec(cellwalker cw) { | ||||
|       if(ep.second != cw.mirrored) { | ||||
|         int x, y; | ||||
|         tie(x, y) = vec_to_pair(id); | ||||
|         return pair_to_vec(x + torusconfig::sdx, torusconfig::sdy - y); | ||||
|         x += torusconfig::sdx; | ||||
|         y += torusconfig::sdy; | ||||
|         torusconfig::mobius_flip(x, y); | ||||
|         return pair_to_vec(x, y); | ||||
|         } | ||||
|       } | ||||
|     return id; | ||||
| @@ -1050,15 +1145,21 @@ euc_pointer euclideanAtCreate(int vec) { | ||||
|   euc_pointer ep = euclideanAt(vec); | ||||
|   cell*& c = *ep.first; | ||||
|   if(!c) { | ||||
|     if(euwrap) { | ||||
|       int x, y; | ||||
|       tie(x, y) = vec_to_pair(vec); | ||||
|       torusconfig::be_canonical(x, y); | ||||
|       vec = pair_to_vec(x, y); | ||||
|       } | ||||
|     c = newCell(8, encodeId(vec)); | ||||
|     // euclideanAt(vec) = c; | ||||
|     build_euclidean_moves(c, vec, [ep, c,vec] (int delta, int d, int d2) {  | ||||
|       euc_pointer ep2 = euclideanAt(vec + delta); | ||||
|       cell* c2 = *ep2.first; | ||||
|       if(!c2) return; | ||||
|       if(ep.second) d = c->c.fix(-d); | ||||
|       if(ep2.second) d2 = c2->c.fix(-d2); | ||||
|       eumerge(c, d, c2, d2, ep.second != ep2.second); | ||||
|       // if(ep.second) d = c->c.fix(torusconfig::mobius_dir(c) - d); | ||||
|       if(ep2.second) d2 = c2->c.fix(torusconfig::mobius_dir(c2) - d2); | ||||
|       eumerge(c, d, c2, d2, ep2.second); | ||||
|       }); | ||||
|     } | ||||
|   return ep; | ||||
| @@ -1231,8 +1332,10 @@ int compdist(int dx[]) { | ||||
| int celldist(cell *c) { | ||||
|   if(fulltorus)  | ||||
|     return torusmap()->dists[decodeId(c->master)]; | ||||
|   if(euwrap) | ||||
|     return torusconfig::cyldist(decodeId(c->master), 0); | ||||
|   if(masterless) | ||||
|     return eudist(decodeId(c->master)); // fix cylinder | ||||
|     return eudist(decodeId(c->master)); | ||||
|   if(sphere || binarytiling) return celldistance(c, currentmap->gamestart()); | ||||
|   if(IRREGULAR) return irr::celldist(c, false); | ||||
|   if(archimedean || ctof(c)) return c->master->distance; | ||||
| @@ -1250,7 +1353,7 @@ int celldist(cell *c) { | ||||
|  | ||||
| int celldistAlt(cell *c) { | ||||
|   if(masterless) { | ||||
|     if(fulltorus) return celldist(c); // fix cylinder | ||||
|     if(fulltorus) return celldist(c); | ||||
|     int x, y; | ||||
|     tie(x,y) = vec_to_pair(decodeId(c->master)); | ||||
|     return euclidAlt(x, y); | ||||
| @@ -1624,6 +1727,8 @@ int celldistance(cell *c1, cell *c2) { | ||||
|       return eudist(decodeId(c1->master) - decodeId(c2->master)); // fix cylinder | ||||
|     else if(euwrap && torusconfig::torus_mode == 0)  | ||||
|       return torusmap()->dists[torusconfig::vec_to_id(decodeId(c1->master)-decodeId(c2->master))]; | ||||
|     else if(euwrap && !fulltorus) | ||||
|       return torusconfig::cyldist(decodeId(c1->master), decodeId(c2->master)); | ||||
|     } | ||||
|    | ||||
|   if(geometry == gFieldQuotient && !GOLDBERG) | ||||
|   | ||||
| @@ -170,21 +170,16 @@ void showTorusConfig() { | ||||
|         dialog::addInfo(XLAT("best if %1 is divisible by %2", "x", "2"), 0x808080), valid = 1; | ||||
|       if(!torus_bitrunc && valid == 1) | ||||
|         dialog::addInfo("incompatible with bitruncating", 0x808080), valid = 0;       | ||||
|       if(klein && abs(adx) != abs(ady) && adx != 0 && ady != 0) | ||||
|       if(klein && !torusconfig::mobius_symmetric(square, adx, ady)) | ||||
|         dialog::addInfo("Möbius band requires a symmetric period", 0x800000), valid = 0; | ||||
|  | ||||
|       if(klein && ady) | ||||
|         dialog::addInfo("not implemented", 0x800000), valid = 0; | ||||
|       } | ||||
|     else { | ||||
|       if(torusconfig::newsdy % 3) | ||||
|         dialog::addInfo(XLAT("best if %1 is divisible by %2", "y", "3"), 0x808080), valid = 1; | ||||
|       if(torusconfig::newsdx % 3) | ||||
|         dialog::addInfo(XLAT("best if %1 is divisible by %2", "x", "3"), 0x808080), valid = 1; | ||||
|       if(klein && adx != 0 && ady != 0 && adx != -ady) | ||||
|       if(klein && !torusconfig::mobius_symmetric(square, adx, ady)) | ||||
|         dialog::addInfo("Möbius band requires a symmetric period", 0x800000), valid = 0; | ||||
|       if(klein) | ||||
|         dialog::addInfo("not implemented", 0x800000), valid = 0; | ||||
|       } | ||||
|     } | ||||
|   else { | ||||
|   | ||||
| @@ -225,6 +225,7 @@ void virtualRebase(cell*& base, T& at, bool tohex, const U& check) { | ||||
|   if(euclid || sphere) { | ||||
|     again: | ||||
|     if(euwrap) for(int i=0; i<6; i++) { | ||||
|       // fix cylinder and square grid | ||||
|       auto newat = eumovedir(3+i) * at; | ||||
|       if(hdist0(check(newat)) < hdist0(check(at))) { | ||||
|         at = newat; | ||||
|   | ||||
| @@ -55,10 +55,6 @@ namespace hr { namespace gp { | ||||
|       } | ||||
|     } | ||||
|    | ||||
| #define SG6 (S3==3?6:4) | ||||
| #define SG3 (S3==3?3:2) | ||||
| #define SG2 (S3==3?2:1) | ||||
|  | ||||
|   int fixg6(int x) { return (x + MODFIXER) % SG6; } | ||||
|    | ||||
| #define WHD(x) // x | ||||
|   | ||||
							
								
								
									
										8
									
								
								hyper.h
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								hyper.h
									
									
									
									
									
								
							| @@ -134,6 +134,12 @@ void addMessage(string s, char spamtype = 0); | ||||
| #define MAX_S3 4 | ||||
| #define MAX_S84 240 | ||||
|  | ||||
| #define eurad crossf | ||||
|  | ||||
| #define SG6 (S3==3?6:4) | ||||
| #define SG3 (S3==3?3:2) | ||||
| #define SG2 (S3==3?2:1) | ||||
|  | ||||
| #define GOLDBERG (variation == eVariation::goldberg) | ||||
| #define IRREGULAR (variation == eVariation::irregular) | ||||
| #define PURE (variation == eVariation::pure) | ||||
| @@ -2860,7 +2866,7 @@ namespace princess { | ||||
| #define GRAIL_FOUND 0x4000 | ||||
| #define GRAIL_RADIUS_MASK 0x3FFF | ||||
|  | ||||
| int eudist(short sx, short sy); | ||||
| int eudist(int sx, int sy); | ||||
|  | ||||
| heptagon *createStep(heptagon *h, int d); | ||||
|  | ||||
|   | ||||
| @@ -580,8 +580,6 @@ void applymodel(hyperpoint H, hyperpoint& ret) { | ||||
| transmatrix sphereflip; // on the sphere, flip | ||||
| bool playerfound; // has player been found in the last drawing? | ||||
|  | ||||
| #define eurad crossf | ||||
|  | ||||
| double q3 = sqrt(double(3)); | ||||
|  | ||||
| bool outofmap(hyperpoint h) { | ||||
| @@ -935,7 +933,7 @@ void drawEuclidean() { | ||||
|       } | ||||
|  | ||||
|     if(do_draw(cw.at, Mat)) { | ||||
|       drawcell(cw.at, cw.mirrored ? Mat * Mirror : Mat, cw.spin, cw.mirrored); | ||||
|       drawcell(cw.at, cw.mirrored ? Mat * spin(-2*M_PI*cw.spin / cw.at->type) * Mirror : Mat, cw.spin, cw.mirrored); | ||||
|       for(int x=-1; x<=+1; x++) | ||||
|       for(int y=-1; y<=+1; y++) { | ||||
|         euspot p(dx+x, dy+y); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Zeno Rogue
					Zeno Rogue