mirror of
				https://github.com/zenorogue/hyperrogue.git
				synced 2025-11-04 07:43:02 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1184 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1184 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
// Hyperbolic Rogue -- Irregular (Voronoi) tilings
 | 
						|
// Copyright (C) 2011-2019 Zeno Rogue, see 'hyper.cpp' for details
 | 
						|
 | 
						|
/** \file irregular.cpp
 | 
						|
 *  \brief Irregular (Voronoi) tilings
 | 
						|
 */
 | 
						|
 | 
						|
#include "hyper.h"
 | 
						|
namespace hr { 
 | 
						|
 | 
						|
EX namespace irr {
 | 
						|
 | 
						|
EX int irrid;
 | 
						|
 | 
						|
#if CAP_IRR
 | 
						|
EX ld density = 2;
 | 
						|
EX ld quality = .2;
 | 
						|
EX int place_attempts = 10;
 | 
						|
EX int rearrange_max_attempts = 50;
 | 
						|
EX int rearrange_less = 10;
 | 
						|
 | 
						|
EX int cellcount;
 | 
						|
 | 
						|
#if HDR
 | 
						|
struct cellinfo {
 | 
						|
  cell *owner;
 | 
						|
  map<cell*, transmatrix> relmatrices;
 | 
						|
  vector<hyperpoint> jpoints;
 | 
						|
  hyperpoint p;
 | 
						|
  transmatrix pusher, rpusher;
 | 
						|
  vector<int> neid;
 | 
						|
  vector<int> spin;
 | 
						|
  vector<hyperpoint> vertices;
 | 
						|
  int localindex;
 | 
						|
  bool is_pseudohept;
 | 
						|
  int patterndir;
 | 
						|
  int generation;
 | 
						|
  };
 | 
						|
#endif
 | 
						|
 | 
						|
EX map<cell*, int> cellindex;
 | 
						|
 | 
						|
EX vector<cellinfo> cells;
 | 
						|
 | 
						|
EX map<heptagon*, vector<int> > cells_of_heptagon;
 | 
						|
 | 
						|
int runlevel;
 | 
						|
vector<ld> edgelens, distlens;
 | 
						|
 | 
						|
void make_cells_of_heptagon() {
 | 
						|
  cells_of_heptagon.clear();
 | 
						|
  for(int i=0; i<isize(cells); i++) {
 | 
						|
    auto &p1 = cells[i];
 | 
						|
    auto &vc = cells_of_heptagon[p1.owner->master];
 | 
						|
    p1.localindex = isize(vc);
 | 
						|
    vc.push_back(i);
 | 
						|
    }
 | 
						|
  }
 | 
						|
  
 | 
						|
string status[5];
 | 
						|
  
 | 
						|
EX hrmap *base;
 | 
						|
 | 
						|
EX euc::torus_config_full base_config;
 | 
						|
 | 
						|
EX bool gridmaking;
 | 
						|
 | 
						|
int rearrange_index;
 | 
						|
 | 
						|
bool cell_sorting;
 | 
						|
 | 
						|
EX int bitruncations_requested = 1;
 | 
						|
EX int bitruncations_performed = 0;
 | 
						|
 | 
						|
int black_adjacent, white_three;
 | 
						|
 | 
						|
void set_relmatrices(cellinfo& ci) {
 | 
						|
  auto& all = base->allcells();
 | 
						|
  ci.relmatrices.clear();
 | 
						|
  for(auto c0: all) ci.relmatrices[c0] = calc_relative_matrix(c0, ci.owner, ci.p);
 | 
						|
  }
 | 
						|
 | 
						|
void rebase(cellinfo& ci) {
 | 
						|
  cell *cx = ci.owner;
 | 
						|
  virtualRebase(ci.owner, ci.p);
 | 
						|
  if(ci.owner != cx) {
 | 
						|
    printf("rebased %p to %p\n", hr::voidp(cx), hr::voidp(ci.owner));
 | 
						|
    set_relmatrices(ci);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
void compute_jpoints() {
 | 
						|
  for(int i=0; i<isize(cells); i++) {
 | 
						|
    auto &ci = cells[i];
 | 
						|
 | 
						|
    ci.pusher = rgpushxto0(ci.p);
 | 
						|
    ci.rpusher = gpushxto0(ci.p);
 | 
						|
    
 | 
						|
    ci.jpoints.clear();
 | 
						|
 | 
						|
    for(int j=0; j<isize(cells); j++) {
 | 
						|
      auto &cj = cells[j];
 | 
						|
      ci.jpoints.push_back(ci.rpusher * ci.relmatrices[cj.owner] * cj.p);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
    
 | 
						|
void bitruncate() {
 | 
						|
  int cc = isize(cells);
 | 
						|
  map<pair<int, int>, int> bitruncated_id;
 | 
						|
  for(int i=0; i<cc; i++) {
 | 
						|
    int v = isize(cells[i].vertices);
 | 
						|
    for(int j=0; j<v; j++) {
 | 
						|
      int last = cells[i].neid[(j+v-1)%v];
 | 
						|
      int next = cells[i].neid[j];
 | 
						|
      if(!bitruncated_id.count(make_pair(i, last))) {
 | 
						|
        bitruncated_id[make_pair(i, last)] = 
 | 
						|
        bitruncated_id[make_pair(last, next)] = 
 | 
						|
        bitruncated_id[make_pair(next, i)] =                 
 | 
						|
          isize(cells);
 | 
						|
        cells.emplace_back();
 | 
						|
        cellinfo& s = cells.back();
 | 
						|
        s.patterndir = -1;
 | 
						|
        s.owner = cells[i].owner;
 | 
						|
        s.p = cells[i].pusher * cells[i].vertices[j];
 | 
						|
        s.neid.push_back(i);
 | 
						|
        s.neid.push_back(-1);
 | 
						|
        s.neid.push_back(last);
 | 
						|
        s.neid.push_back(-1);
 | 
						|
        s.neid.push_back(next);
 | 
						|
        s.neid.push_back(-1);
 | 
						|
        s.generation = bitruncations_performed + 1;
 | 
						|
        virtualRebase(s.owner, s.p);
 | 
						|
        set_relmatrices(s);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  for(int i=0; i<cc; i++) {
 | 
						|
    int v = isize(cells[i].vertices);
 | 
						|
    vector<int> newnei;
 | 
						|
    for(int j=0; j<v; j++) {
 | 
						|
      int last = cells[i].neid[(j+v-1)%v];
 | 
						|
      int next = cells[i].neid[j];
 | 
						|
      auto id = bitruncated_id[make_pair(i, last)];
 | 
						|
      newnei.push_back(id);
 | 
						|
      for(int k=0; k<6; k++) 
 | 
						|
        if(cells[id].neid[k] == i) {
 | 
						|
          cells[id].neid[(k+5)%6] = bitruncated_id[make_pair(i, next)];
 | 
						|
          }
 | 
						|
      }
 | 
						|
    cells[i].neid = std::move(newnei);
 | 
						|
    }
 | 
						|
  make_cells_of_heptagon();
 | 
						|
  compute_jpoints();
 | 
						|
  for(int i=0; i<isize(cells); i++) {
 | 
						|
    auto &ci = cells[i];
 | 
						|
    ci.vertices.clear();
 | 
						|
 | 
						|
    ci.pusher = rgpushxto0(ci.p);
 | 
						|
    ci.rpusher = gpushxto0(ci.p);
 | 
						|
 | 
						|
    int v = isize(ci.neid);
 | 
						|
    for(int j=0; j<v; j++) {
 | 
						|
      int last = ci.neid[(j+v-1)%v];
 | 
						|
      int next = ci.neid[j];
 | 
						|
      hyperpoint h1 = ci.rpusher * ci.relmatrices[cells[last].owner] * cells[last].p;
 | 
						|
      hyperpoint h2 = ci.rpusher * ci.relmatrices[cells[next].owner] * cells[next].p;
 | 
						|
      ci.vertices.push_back(mid3(C0, h1, h2));
 | 
						|
      }
 | 
						|
    }
 | 
						|
  bitruncations_performed++;
 | 
						|
  cell_sorting = false;
 | 
						|
  }
 | 
						|
 | 
						|
int rearrange(bool total, ld minedge) {
 | 
						|
  int tooshort = 0;
 | 
						|
  for(int i=0; i<isize(cells); i++) {
 | 
						|
    auto& p1 = cells[i];
 | 
						|
    hyperpoint h = Hypc;
 | 
						|
    for(auto v: p1.vertices) h = h + v;
 | 
						|
    
 | 
						|
    bool changed = total;
 | 
						|
 | 
						|
    for(int j=0; j<isize(p1.vertices); j++)
 | 
						|
      if(hdist(p1.vertices[j], p1.vertices[(j+1) % isize(p1.vertices)]) < minedge) {
 | 
						|
        tooshort++; changed = true;
 | 
						|
        h = h + p1.vertices[j] + p1.vertices[(j+1) % isize(p1.vertices)];
 | 
						|
        }
 | 
						|
    if(changed)
 | 
						|
      cells[i].p = p1.pusher * normalize(h);
 | 
						|
    }
 | 
						|
  return tooshort;
 | 
						|
  }
 | 
						|
 | 
						|
bool step(int delta) {
 | 
						|
 | 
						|
  if(!gridmaking) return false;
 | 
						|
  timetowait = 0;
 | 
						|
 | 
						|
  auto& all = base->allcells();
 | 
						|
 | 
						|
  auto t = SDL_GetTicks();
 | 
						|
  while(SDL_GetTicks() < t + 250)
 | 
						|
  switch(runlevel) {
 | 
						|
    case 0: {
 | 
						|
 | 
						|
     cells.clear();
 | 
						|
     cells_of_heptagon.clear();
 | 
						|
     cellindex.clear();
 | 
						|
     
 | 
						|
     if(0) if(cellcount <= isize(all) * 2) {
 | 
						|
       for(auto h: all) {
 | 
						|
         cells.emplace_back();
 | 
						|
         cellinfo& s = cells.back();
 | 
						|
         s.patterndir = -1;
 | 
						|
         s.owner = h, s.p = xspinpush0(hrand(1000), .01);
 | 
						|
         s.generation = 0;
 | 
						|
         set_relmatrices(s);
 | 
						|
         }
 | 
						|
       }
 | 
						|
      runlevel++;
 | 
						|
      break;
 | 
						|
      }
 | 
						|
     
 | 
						|
    case 1: {
 | 
						|
      while(isize(cells) < cellcount) {
 | 
						|
        if(SDL_GetTicks() > t + 250) { make_cells_of_heptagon(); status[0] = its(isize(cells)) + " cells"; return false; }
 | 
						|
        cells.emplace_back();
 | 
						|
        cellinfo& s = cells.back();
 | 
						|
        s.patterndir = -1;
 | 
						|
        ld bestval = 0;
 | 
						|
        for(int j=0; j<place_attempts; j++) {
 | 
						|
          int k = hrand(isize(all));
 | 
						|
          cell *c = all[k];
 | 
						|
          map<cell*, transmatrix> relmatrices;
 | 
						|
          hyperpoint h = randomPointIn(c->type);
 | 
						|
          for(auto c0: all) relmatrices[c0] = calc_relative_matrix(c0, c, h);
 | 
						|
          ld mindist = 1e6;
 | 
						|
          for(auto p: cells) {
 | 
						|
            if(!relmatrices.count(p.owner)) continue;
 | 
						|
            ld val = hdist(h, relmatrices[p.owner] * p.p);
 | 
						|
            if(val < mindist) mindist = val;
 | 
						|
            }
 | 
						|
          if(mindist > bestval) bestval = mindist, s.owner = c, s.p = h, s.relmatrices = std::move(relmatrices);
 | 
						|
          }
 | 
						|
        }
 | 
						|
      make_cells_of_heptagon();
 | 
						|
      cell_sorting = true; bitruncations_performed = 0;
 | 
						|
      runlevel++;
 | 
						|
      status[0] = "all " + its(isize(cells)) + " cells";
 | 
						|
      break;
 | 
						|
      }
 | 
						|
    
 | 
						|
    case 2: {
 | 
						|
 | 
						|
      if(cell_sorting)
 | 
						|
        sort(cells.begin(), cells.end(), [] (const cellinfo &s1, const cellinfo &s2) { return hdist0(s1.p) < hdist0(s2.p); });
 | 
						|
      make_cells_of_heptagon();
 | 
						|
 | 
						|
      edgelens.clear();
 | 
						|
      distlens.clear();
 | 
						|
  
 | 
						|
      int stats[16];
 | 
						|
      for(int k=0; k<16; k++) stats[k] = 0;
 | 
						|
      
 | 
						|
      compute_jpoints();
 | 
						|
      
 | 
						|
      for(int i=0; i<isize(cells); i++) {
 | 
						|
        auto &p1 = cells[i];
 | 
						|
    
 | 
						|
        p1.vertices.clear();
 | 
						|
        p1.neid.clear();
 | 
						|
    
 | 
						|
        int j = 0;
 | 
						|
        if(j == i) j = 1;
 | 
						|
    
 | 
						|
        for(int k=0; k<isize(cells); k++) if(k != i) {
 | 
						|
          if(hdist(p1.jpoints[k], C0) < hdist(p1.jpoints[j], C0))
 | 
						|
            j = k;
 | 
						|
          }
 | 
						|
          
 | 
						|
        hyperpoint t = mid(p1.jpoints[j], C0);
 | 
						|
              // p1.vertices.push_back(p1.pusher * t);
 | 
						|
        int j0 = j;
 | 
						|
        int oldj = j;
 | 
						|
        do {
 | 
						|
          int best_k = -1;
 | 
						|
          hyperpoint best_h;
 | 
						|
          for(int k=0; k<isize(cells); k++) if(k != i && k != j && k != oldj) {
 | 
						|
            hyperpoint h = circumscribe(C0, p1.jpoints[j], p1.jpoints[k]);
 | 
						|
            if(h[LDIM] < 0) continue;
 | 
						|
            if(!clockwise(t, h)) continue;
 | 
						|
            if(best_k == -1)
 | 
						|
              best_k = k, best_h = h;
 | 
						|
            else if(clockwise(h, best_h))
 | 
						|
              best_k = k, best_h = h;
 | 
						|
            }
 | 
						|
          p1.vertices.push_back(best_h);
 | 
						|
          p1.neid.push_back(best_k);
 | 
						|
          distlens.push_back(hdist0(best_h));
 | 
						|
          oldj = j, j = best_k, t = best_h;
 | 
						|
          if(j == -1) break;
 | 
						|
          if(isize(p1.vertices) == 15) break;
 | 
						|
          }
 | 
						|
        while(j != j0);
 | 
						|
        
 | 
						|
        for(int j=0; j<isize(p1.vertices); j++)
 | 
						|
          edgelens.push_back(hdist(p1.vertices[j], p1.vertices[(j+1) % isize(p1.vertices)]));
 | 
						|
    
 | 
						|
        stats[isize(p1.vertices)]++;
 | 
						|
        }
 | 
						|
    
 | 
						|
      for(int a=0; a<16; a++) printf("%3d ", stats[a]);
 | 
						|
      if(isize(edgelens)) {
 | 
						|
        printf("|");
 | 
						|
        printf("%4d ", isize(edgelens));
 | 
						|
        sort(edgelens.begin(), edgelens.end());
 | 
						|
        for(int a=0; a<=8; a++) printf("%6.3lf", double(edgelens[(a * isize(edgelens) - 1) / 8]));
 | 
						|
        printf(" | ");
 | 
						|
        sort(distlens.begin(), distlens.end());
 | 
						|
        for(int a=0; a<=8; a++) printf("%5.2lf", double(distlens[(a * isize(edgelens) - 1) / 8]));
 | 
						|
        }
 | 
						|
      printf("\n");
 | 
						|
      
 | 
						|
      runlevel++;
 | 
						|
      break;
 | 
						|
      }
 | 
						|
    
 | 
						|
    case 3: {
 | 
						|
    
 | 
						|
      int errors = 0, toobig = 0;
 | 
						|
  
 | 
						|
      for(int i=0; i<isize(cells); i++) {
 | 
						|
        int v = isize(cells[i].vertices);
 | 
						|
        if(v > 8 || v< 3) {
 | 
						|
          if(v < 3 || v >= 15)
 | 
						|
            errors++;
 | 
						|
          else toobig++;
 | 
						|
          cells[i] = cells.back();
 | 
						|
          i--; cells.pop_back();
 | 
						|
          }
 | 
						|
        }
 | 
						|
      
 | 
						|
      if(errors > 0) status[1] = XLAT("bad cells: %1", its(errors)); else status[1] = " ";
 | 
						|
      if(toobig > 0) status[2] = XLAT("too many edges: %1", its(toobig)); else status[2] = " ";
 | 
						|
      if(isize(cells) < cellcount*3/4) runlevel = 0;
 | 
						|
      else if(isize(cells) < cellcount) runlevel = 1;
 | 
						|
      else { rearrange_index = 0; runlevel++; }
 | 
						|
      break;
 | 
						|
      }
 | 
						|
    
 | 
						|
    case 4: {
 | 
						|
  
 | 
						|
      ld median = edgelens[isize(edgelens) / 2];
 | 
						|
      ld minedge = median * quality;
 | 
						|
      status[3] = XLAT("median edge: %1 minimum: %2", fts(median), fts(edgelens[0]));
 | 
						|
      if(!bitruncations_performed && edgelens[0] < minedge) {
 | 
						|
        if(rearrange_index >= rearrange_max_attempts) {
 | 
						|
          runlevel = 0; break;
 | 
						|
          }
 | 
						|
        int tooshort = rearrange(rearrange_index < rearrange_less, minedge);
 | 
						|
        
 | 
						|
        status[3] += XLAT(" (edges too short: %1)", its(tooshort));
 | 
						|
        runlevel = 2;
 | 
						|
        rearrange_index++;
 | 
						|
        break;
 | 
						|
        }
 | 
						|
      runlevel++;
 | 
						|
      break;
 | 
						|
      }
 | 
						|
      
 | 
						|
    case 5: {
 | 
						|
      if(bitruncations_performed < bitruncations_requested) 
 | 
						|
        bitruncate();
 | 
						|
      else
 | 
						|
        runlevel = 6;
 | 
						|
      break;
 | 
						|
      }
 | 
						|
    
 | 
						|
    case 6: {
 | 
						|
      
 | 
						|
      int notfound = 0;
 | 
						|
    
 | 
						|
      for(int i=0; i<isize(cells); i++) {
 | 
						|
        auto &p1 = cells[i];
 | 
						|
        int N = isize(p1.vertices);
 | 
						|
        p1.spin.resize(N);
 | 
						|
        for(int j=0; j<N; j++) {
 | 
						|
          auto i1 = p1.neid[j];
 | 
						|
          if(i1 < 0 || i1 >= isize(cells)) {
 | 
						|
            runlevel = 0;
 | 
						|
            return false;
 | 
						|
            }
 | 
						|
          bool found = false;          
 | 
						|
          for(int k=0; k < isize(cells[i1].vertices); k++)
 | 
						|
            if(cells[i1].neid[k] == i)
 | 
						|
              found = true, p1.spin[j] = k;
 | 
						|
          if(!found) notfound++;
 | 
						|
          }
 | 
						|
        }
 | 
						|
      
 | 
						|
      if(notfound) { status[4] = XLAT("cells badly paired: %1", its(notfound)); runlevel = 0; break; }
 | 
						|
      
 | 
						|
      int heptas = 0;
 | 
						|
      for(auto p: cells_of_heptagon) heptas++;
 | 
						|
      
 | 
						|
      if(heptas != isize(all)) {
 | 
						|
        status[4] = XLAT("cells not covered: %1", its(isize(all) - heptas));
 | 
						|
        printf("heptas = %d\n", heptas);
 | 
						|
        runlevel = 0; break;
 | 
						|
        }
 | 
						|
    
 | 
						|
      int faredge = 0;
 | 
						|
      for(int i=0; i<isize(cells); i++) {
 | 
						|
        auto &p1 = cells[i];
 | 
						|
        for(int j: p1.neid) {
 | 
						|
          auto &p2 = cells[j];
 | 
						|
          bool ok = p1.owner == p2.owner || isNeighbor(p1.owner, p2.owner);
 | 
						|
          if(!ok) faredge++;
 | 
						|
          }
 | 
						|
        }
 | 
						|
 | 
						|
      if(faredge) {
 | 
						|
        status[4] = XLAT("adjacent cells from nonadjacent heptagons: %1", its(faredge));
 | 
						|
        runlevel = 0; return false;
 | 
						|
        }
 | 
						|
      
 | 
						|
      /*
 | 
						|
      black_adjacent = 0;
 | 
						|
      white_three = 0;
 | 
						|
      for(int i=0; i<isize(cells); i++) {
 | 
						|
        if(!cells[i].by_bitruncation) {
 | 
						|
          for(int j: cells[i].neid) if(!cells[j].by_bitruncation) black_adjacent++;
 | 
						|
          }
 | 
						|
        else {
 | 
						|
          int v = isize(cells[i].neid);
 | 
						|
          for(int j=0; j<v; j++)
 | 
						|
            if(cells[cells[i].neid[j]].by_bitruncation)
 | 
						|
            if(cells[cells[i].neid[(j+1)%v]].by_bitruncation)
 | 
						|
              white_three++;
 | 
						|
          }
 | 
						|
        }
 | 
						|
      printf("black_adjacent = %d, white_three = %d\n", black_adjacent, white_three);
 | 
						|
      */
 | 
						|
 | 
						|
      status[4] = XLAT("OK");
 | 
						|
      runlevel = 10;
 | 
						|
      
 | 
						|
      for(auto& s: cells) s.is_pseudohept = false;
 | 
						|
      for(auto& s: cells) {
 | 
						|
        s.is_pseudohept = true;
 | 
						|
        for(int i: s.neid) if(cells[i].is_pseudohept) s.is_pseudohept = false;
 | 
						|
        }
 | 
						|
 | 
						|
      for(auto& s: cells) {
 | 
						|
        int d = -1;
 | 
						|
        ld dist = cgi.hcrossf / 2;
 | 
						|
        ld dists[8];
 | 
						|
        for(int i=0; i<S7; i++) {
 | 
						|
          dists[i] = hdist(s.p, xspinpush0(cgi.hexshift - i * TAU / S7, -cgi.hcrossf));
 | 
						|
          if(dists[i] < dist)
 | 
						|
            d = i, dist = dists[i];
 | 
						|
          }
 | 
						|
        if(d != -1 && dists[(d+1) % S7] > dists[(d+S7-1) % S7])
 | 
						|
          d = (d + S7 - 1) % S7;
 | 
						|
        s.patterndir = d;
 | 
						|
        }
 | 
						|
 | 
						|
      break;
 | 
						|
      }
 | 
						|
    
 | 
						|
    case 10:
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
  return false;
 | 
						|
  }
 | 
						|
 | 
						|
EX ld compute_scale() {
 | 
						|
  return sqrt(isize(cells_of_heptagon) * 1. / isize(cells));
 | 
						|
  }
 | 
						|
 | 
						|
EX void compute_geometry() {
 | 
						|
  if(IRREGULAR) {
 | 
						|
    ld scale = compute_scale();
 | 
						|
    cgi.crossf *= scale;
 | 
						|
    cgi.rhexf *= scale;
 | 
						|
    cgi.hexhexdist *= scale;
 | 
						|
    cgi.hexvdist *= scale;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
bool draw_cell_schematics(cell *c, const shiftmatrix& V) {
 | 
						|
  if(gridmaking) {
 | 
						|
    heptagon *h = c->master;
 | 
						|
    for(int i: cells_of_heptagon[h]) {
 | 
						|
      auto& p = cells[i];
 | 
						|
      if(p.owner == c) {
 | 
						|
        queuestr(V * rgpushxto0(p.p), .1, its(i), isize(p.vertices) > 8 ? 0xFF0000 : 0xFFFFFF);
 | 
						|
        int N = isize(p.vertices);
 | 
						|
        for(int j=0; j<N; j++)
 | 
						|
          gridline(V, p.pusher * p.vertices[j], p.pusher * p.vertices[(1+j)%N], 0xFFFFFFFF, 0);
 | 
						|
 | 
						|
        gridline(V, p.p, C0, 0xFF0000FF, 0);
 | 
						|
        if(p.patterndir != -1)
 | 
						|
          gridline(V, p.p, calc_relative_matrix(c->master->move(p.patterndir)->c7, c, p.p) * C0, 0x00FF00FF, 0);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  return false;
 | 
						|
  }
 | 
						|
 | 
						|
#if HDR
 | 
						|
struct heptinfo {
 | 
						|
  heptspin base;
 | 
						|
  vector<cell*> subcells;
 | 
						|
  vector<int> celldists[2];
 | 
						|
  };
 | 
						|
#endif
 | 
						|
 | 
						|
EX map<heptagon*, heptinfo> periodmap;
 | 
						|
 | 
						|
EX void link_to_base(heptagon *h, heptspin base) {
 | 
						|
  // printf("linking %p to %p/%d\n", hr::voidp(h), hr::voidp(base.at), base.spin);
 | 
						|
  auto &hi = periodmap[h];
 | 
						|
  hi.base = base;
 | 
						|
  for(int k: cells_of_heptagon[base.at]) {
 | 
						|
    cell *c = newCell(isize(cells[k].vertices), h);
 | 
						|
    hi.subcells.push_back(c);
 | 
						|
    cellindex[c] = k;
 | 
						|
    }
 | 
						|
  h->c7 = hi.subcells[0];
 | 
						|
  }
 | 
						|
 | 
						|
EX void clear_links(heptagon *h) {
 | 
						|
  auto& hi = periodmap[h];
 | 
						|
  for(cell *c: hi.subcells) {
 | 
						|
    for(int i=0; i<c->type; i++) if(c->move(i)) c->move(i)->move(c->c.spin(i)) = NULL;
 | 
						|
    cellindex.erase(c);
 | 
						|
    delete c;
 | 
						|
    }
 | 
						|
  h->c7 = NULL;
 | 
						|
  periodmap.erase(h);
 | 
						|
  }
 | 
						|
 | 
						|
EX void link_start(heptagon *h) {
 | 
						|
  link_to_base(h, heptspin(cells[0].owner->master, 0));
 | 
						|
  }
 | 
						|
 | 
						|
EX void link_next(heptagon *parent, int d) {
 | 
						|
  if(!periodmap.count(parent))
 | 
						|
    link_to_base(parent, heptspin(cells[0].owner->master, 0));
 | 
						|
  // printf("linking next: %p direction %d [s%d]\n", hr::voidp(parent), d, parent->c.spin(d));
 | 
						|
  auto *h = parent->move(d);
 | 
						|
  heptspin hs = periodmap[parent].base + d + wstep - parent->c.spin(d);
 | 
						|
  link_to_base(h, hs);
 | 
						|
  }
 | 
						|
 | 
						|
EX void may_link_next(heptagon *parent, int d) {
 | 
						|
  if(!periodmap.count(parent->move(d)))
 | 
						|
    link_next(parent, d);
 | 
						|
  }
 | 
						|
 | 
						|
 | 
						|
EX void link_cell(cell *c, int d) {
 | 
						|
  // printf("linking cell: %p direction %d\n", hr::voidp(c), d);
 | 
						|
  int ci = cellindex[c];
 | 
						|
  auto& sc = cells[ci];
 | 
						|
  int ci2 = sc.neid[d];
 | 
						|
  auto& sc2 = cells[ci2];
 | 
						|
  
 | 
						|
  heptagon *master2 = NULL;
 | 
						|
 | 
						|
  if(sc2.owner == sc.owner) {
 | 
						|
    master2 = c->master;
 | 
						|
    // printf("local\n");
 | 
						|
    }
 | 
						|
  else {
 | 
						|
    int dirs = 0;
 | 
						|
    int os = periodmap[c->master].base.spin;
 | 
						|
    for(int d=0; d<S7; d++) if(sc2.owner->master == sc.owner->master->modmove(os+d)) {
 | 
						|
      heptspin hss(c->master, d);
 | 
						|
      hss += wstep;
 | 
						|
      master2 = hss.at;
 | 
						|
      // printf("master2 is %p; base = %p; should be = %p\n", hr::voidp(master2), hr::voidp(periodmap[master2].base.at), hr::voidp(sc2.owner->master));
 | 
						|
      dirs++;
 | 
						|
      }
 | 
						|
    if(dirs != 1) { printf("dirs error\n"); exit(1); }
 | 
						|
    }
 | 
						|
  
 | 
						|
  cell *c2 = periodmap[master2].subcells[sc2.localindex];
 | 
						|
  c->c.connect(d, c2, sc.spin[d], false);
 | 
						|
  }
 | 
						|
 | 
						|
int hdist(heptagon *h1, heptagon *h2) {
 | 
						|
  if(h1 == h2) return 0;
 | 
						|
  for(int i=0; i<S7; i++) if(h1->move(i) == h2) return 1;
 | 
						|
  return 2;
 | 
						|
  }
 | 
						|
 | 
						|
// compute celldist or celldistalt for all the subcells of h.
 | 
						|
// We use the following algorithm:
 | 
						|
// - assume that everything is computed for all the adjacent heptagons of h which are closer to the origin
 | 
						|
// - consider h and its two neighbors which are in the same distance to the origin ('siblings')
 | 
						|
// - compute celldists for all the cells in these three heptagons, by bfs, based on the 'parent' heptagons adjacent to h
 | 
						|
// - record the computed distances for h, but not for its siblings
 | 
						|
 | 
						|
static constexpr int NODISTANCE = 2000000000;
 | 
						|
 | 
						|
map<heptagon*, heptagon*> last_on_horocycle;
 | 
						|
 | 
						|
void compute_horocycle(heptagon *);
 | 
						|
 | 
						|
void compute_distances(heptagon *h, bool alts) {
 | 
						|
  /* if(alts) printf("[%p] compute_distances %p\n", hr::voidp(h->alt->alt), hr::voidp(h));
 | 
						|
  printf("neighbors:"); for(int i=0; i<S7; i++) printf(" %p", createStep(h, i)); printf("\n"); */
 | 
						|
  
 | 
						|
  if(alts) {
 | 
						|
    if(!last_on_horocycle[h->alt->alt])
 | 
						|
      last_on_horocycle[h->alt->alt] = h;
 | 
						|
    
 | 
						|
    if(h->alt->alt->s != hsOrigin)
 | 
						|
    while(h->alt->distance <= last_on_horocycle[h->alt->alt]->alt->distance)
 | 
						|
      compute_horocycle(h->alt->alt);
 | 
						|
    }
 | 
						|
 | 
						|
  auto dm4 = [alts, h] (heptagon *h1) -> unsigned {
 | 
						|
    if(!alts) return h1->dm4;
 | 
						|
    if(alts && !h1->alt) return 100; // error
 | 
						|
    if(alts && h1->alt->alt != h->alt->alt) return 100; // error
 | 
						|
    return h1->alt->dm4;
 | 
						|
    };
 | 
						|
  unsigned cdm = dm4(h), pdm = (cdm-1)&3;
 | 
						|
  vector<heptagon*> hs;
 | 
						|
  hs.push_back(h);
 | 
						|
  for(int i=0; i<S7; i++) if(dm4(createStep(h, i)) == cdm)
 | 
						|
    hs.push_back(h->move(i));
 | 
						|
  
 | 
						|
  vector<vector<int>*> to_clear;
 | 
						|
 | 
						|
  for(auto hx: hs) {  
 | 
						|
    auto &hi = periodmap[hx];
 | 
						|
    int ct = isize(hi.subcells);
 | 
						|
    auto& cd = hi.celldists[alts];
 | 
						|
    if(cd.empty() && hx != h) to_clear.push_back(&cd);
 | 
						|
    cd.resize(ct, NODISTANCE);
 | 
						|
    if(h == hx && (alts ? h->alt->s == hsOrigin : h->s == hsOrigin))
 | 
						|
      cd[0] = 0;
 | 
						|
    }
 | 
						|
  while(true) {
 | 
						|
    bool changed = false;
 | 
						|
    for(auto hx: hs) {
 | 
						|
      auto& hi = periodmap[hx];
 | 
						|
      auto& cd = hi.celldists[alts];
 | 
						|
      for(int i=0; i<isize(hi.subcells); i++)
 | 
						|
        forCellCM(c2, hi.subcells[i]) 
 | 
						|
          if(among(dm4(c2->master), cdm, pdm) && hdist(h, c2->master) < 2) {
 | 
						|
            int d = irr::celldist(c2, alts) + 1;
 | 
						|
            if(d < cd[i]) cd[i] = d, changed = true;
 | 
						|
            }
 | 
						|
      }
 | 
						|
    if(!changed) break;
 | 
						|
    }
 | 
						|
 | 
						|
  /* for(auto hx: hs) {
 | 
						|
    auto& hi = periodmap[hx];
 | 
						|
    auto& cd = hi.celldists[alts];
 | 
						|
    // for(int i: cd) if(i == NODISTANCE) printf("distances not computed\n");
 | 
						|
    } */
 | 
						|
 | 
						|
  for(auto x: to_clear) x->clear();
 | 
						|
 | 
						|
  // for(int i: cd) printf(" %d", i); printf("\n");
 | 
						|
  }
 | 
						|
 | 
						|
void erase_alt(heptagon *alt) { 
 | 
						|
  last_on_horocycle.erase(alt);
 | 
						|
  }
 | 
						|
 | 
						|
void compute_horocycle(heptagon *alt) {
 | 
						|
  heptagon *master = last_on_horocycle[alt];
 | 
						|
  // printf("computing horocycle, master distance = %d [M=%p, A=%p]\n", master->alt->distance, hr::voidp(master), hr::voidp(alt));
 | 
						|
  
 | 
						|
  static constexpr int LOOKUP = 16;
 | 
						|
  set<heptagon*> hs[LOOKUP];
 | 
						|
  hs[0].insert(master);
 | 
						|
  set<heptagon*> region;
 | 
						|
  for(int i=0; i<LOOKUP-1; i++) {
 | 
						|
    for(auto h: hs[i]) {
 | 
						|
      currentmap->extend_altmap(h);
 | 
						|
      for(int j=0; j<S7; j++) {
 | 
						|
        if(h->move(j)->alt->alt != master->alt->alt) continue;
 | 
						|
        region.insert(h->move(j));
 | 
						|
        if(h->move(j)->alt->distance < h->alt->distance)
 | 
						|
          hs[i+1].insert(h->move(j));
 | 
						|
        }
 | 
						|
      }
 | 
						|
    if(hs[i+1].empty()) { printf("error: hs[%d] not found\n", i+1); exit(1); }
 | 
						|
    }
 | 
						|
  /* printf("[%p] compute_horocycle ");
 | 
						|
  for(int i=0; i<LOOKUP-1; i++) printf("%d -> ", isize(hs[i])); printf("%p\n", isize(hs[LOOKUP-1])); */
 | 
						|
  map<cell*, int> xdist; 
 | 
						|
  vector<cell*> xdqueue;
 | 
						|
  cell *orig = periodmap[*(hs[LOOKUP-1].begin())].subcells[0];
 | 
						|
  xdist[orig] = 0;
 | 
						|
  xdqueue.push_back(orig);
 | 
						|
  for(int i=0; i<isize(xdqueue); i++) {
 | 
						|
    forCellCM(c1, xdqueue[i])
 | 
						|
      if(!xdist.count(c1) && region.count(c1->master)) {
 | 
						|
        xdist[c1] = xdist[xdqueue[i]] + 1;
 | 
						|
        xdqueue.push_back(c1);
 | 
						|
        }
 | 
						|
    }
 | 
						|
  int delta = NODISTANCE;
 | 
						|
  for(int i=0; i<S7; i++) {
 | 
						|
    heptagon *h = master->move(i);
 | 
						|
    if(h->alt->alt != master->alt->alt) continue;
 | 
						|
    heptinfo& hi = periodmap[h];
 | 
						|
    if(!isize(hi.celldists[1])) continue;
 | 
						|
    for(int c=0; c<isize(hi.subcells); c++) {
 | 
						|
      if(hi.celldists[1][c] == NODISTANCE) continue;
 | 
						|
      int delta_candidate = hi.celldists[1][c] - xdist[hi.subcells[c]];
 | 
						|
      if(delta != NODISTANCE && delta_candidate != delta) {
 | 
						|
        printf("delta conflict: %d vs %d\n", delta, delta_candidate);
 | 
						|
        delta = max(delta, delta_candidate);
 | 
						|
        }
 | 
						|
      delta = delta_candidate;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  if(delta == NODISTANCE) {
 | 
						|
    delta = master->alt->distance - xdist[periodmap[master].subcells[0]];
 | 
						|
    // printf("delta not found, set to %d\n", delta);
 | 
						|
    }
 | 
						|
  // printf("using delta = %d\n", delta);
 | 
						|
 | 
						|
  for(int i=0; i<LOOKUP/2; i++) {
 | 
						|
    for(auto h: hs[i]) for(int j=-1; j<S7; j++) {
 | 
						|
      heptinfo& hi = periodmap[j == -1 ? h : h->move(j)];
 | 
						|
      hi.celldists[1].resize(isize(hi.subcells));
 | 
						|
      for(int c=0; c<isize(hi.subcells); c++)
 | 
						|
        hi.celldists[1][c] = delta + xdist[hi.subcells[c]];
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
  last_on_horocycle[alt] = *(hs[LOOKUP/2].begin());
 | 
						|
  }
 | 
						|
 | 
						|
EX int celldist(cell *c, bool alts) {
 | 
						|
  heptagon *master = c->master;
 | 
						|
  auto &hi = periodmap[master];
 | 
						|
  /* if(alts && master->alt->alt->s != hsOrigin && isize(hi.celldists[alts]) == 0) {
 | 
						|
    int doalts = 0;
 | 
						|
    for(int i=0; i<S7; i++) if(master->move(i)->alt == master->alt->move[0]) {
 | 
						|
      doalts = 1;
 | 
						|
      if(periodmap[master->move(i)].celldists[true].empty()) {
 | 
						|
        compute_horocycle(master);
 | 
						|
        doalts = 2;
 | 
						|
        }
 | 
						|
      }
 | 
						|
    if(doalts == 0) {
 | 
						|
      currentmap->extend_altmap(master);
 | 
						|
      for(int i=0; i<S7; i++) if(master->move(i)->alt == master->alt->move[0] && periodmap[master->move(i)].celldists[true].empty())
 | 
						|
        compute_horocycle(master);
 | 
						|
      }
 | 
						|
    } */
 | 
						|
  if(isize(hi.celldists[alts]) == 0) 
 | 
						|
    compute_distances(master, alts);
 | 
						|
  return hi.celldists[alts][cells[cellindex[c]].localindex];
 | 
						|
  }
 | 
						|
 | 
						|
eGeometry orig_geometry, base_geometry;
 | 
						|
 | 
						|
void start_game_on_created_map() {    
 | 
						|
  for(hrmap *& hm : allmaps) if(hm == base) hm = NULL;
 | 
						|
  stop_game();
 | 
						|
  geometry = orig_geometry;
 | 
						|
  variation = eVariation::irregular;
 | 
						|
  irrid++;
 | 
						|
  gridmaking = false;
 | 
						|
  start_game();
 | 
						|
  }
 | 
						|
 | 
						|
bool save_map(const string& fname) {
 | 
						|
  fhstream f(fname, "wt");
 | 
						|
  if(!f.f) return false;
 | 
						|
  auto& all = base->allcells();
 | 
						|
  int origcells = 0;
 | 
						|
  for(cellinfo& ci: cells) 
 | 
						|
    if(ci.generation == 0) 
 | 
						|
      origcells++;
 | 
						|
  println(f, spaced(int(geometry), isize(all), origcells));
 | 
						|
  
 | 
						|
  for(auto h: all) {
 | 
						|
    origcells = 0;
 | 
						|
    for(auto i: cells_of_heptagon[h->master]) 
 | 
						|
      if(cells[i].generation == 0)
 | 
						|
        origcells++;
 | 
						|
    println(f, origcells);
 | 
						|
    for(auto i: cells_of_heptagon[h->master]) if(cells[i].generation == 0) {
 | 
						|
      auto &ci = cells[i];
 | 
						|
      println(f, spaced(ci.p[0], ci.p[1], ci.p[LDIM]));
 | 
						|
      }
 | 
						|
    }
 | 
						|
  return true;
 | 
						|
  }
 | 
						|
 | 
						|
vector<ld> float_order;
 | 
						|
 | 
						|
EX void save_map_bin(hstream& f) {
 | 
						|
  if(!base) { f.write<short>(-1); return; }
 | 
						|
  auto& all = base->allcells();
 | 
						|
  int origcells = 0;
 | 
						|
  for(cellinfo& ci: cells)
 | 
						|
    if(ci.generation == 0)
 | 
						|
      origcells++;
 | 
						|
  f.write<short> (base_geometry);
 | 
						|
  f.write<short> (isize(all));
 | 
						|
  f.write<short> (origcells);
 | 
						|
  int foi = 0;
 | 
						|
 | 
						|
  auto check_float_order = [&] (ld x) {
 | 
						|
    if(foi >= isize(float_order)) {
 | 
						|
      float_order.push_back(x);
 | 
						|
      f.write<ld>(x);
 | 
						|
      }
 | 
						|
    else if(abs(float_order[foi] - x) > 1e-6) {
 | 
						|
      println(hlog, float_order[foi], " vs ", x, " : abs difference is ", abs(float_order[foi] - x));
 | 
						|
      float_order[foi] = x;
 | 
						|
      }
 | 
						|
    f.write<ld>(float_order[foi++]);
 | 
						|
    };
 | 
						|
 | 
						|
  for(auto h: all) {
 | 
						|
    origcells = 0;
 | 
						|
    for(auto i: cells_of_heptagon[h->master])
 | 
						|
      if(cells[i].generation == 0)
 | 
						|
        origcells++;
 | 
						|
    f.write<short> (origcells);
 | 
						|
    for(auto i: cells_of_heptagon[h->master]) if(cells[i].generation == 0) {
 | 
						|
      auto &ci = cells[i];
 | 
						|
      check_float_order(ci.p[0]);
 | 
						|
      check_float_order(ci.p[1]);
 | 
						|
      check_float_order(ci.p[LDIM]);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
bool load_map(const string &fname) {
 | 
						|
  fhstream f(fname, "rt");
 | 
						|
  if(!f.f) return false;
 | 
						|
  auto& all = base->allcells();
 | 
						|
  int g, sa;
 | 
						|
  scan(f, g, sa, cellcount);
 | 
						|
  if(sa != isize(all) || g != geometry) { printf("bad parameters\n"); addMessage(XLAT("bad format or bad map geometry")); return false; }
 | 
						|
  density = cellcount * 1. / isize(all);
 | 
						|
  
 | 
						|
  cells.clear();
 | 
						|
  
 | 
						|
  for(auto h: all) {
 | 
						|
    int q = 0;
 | 
						|
    scan(f, q);
 | 
						|
    if(q < 0 || q > cellcount) { runlevel = 0; return false; }
 | 
						|
    while(q--) {
 | 
						|
      cells.emplace_back();
 | 
						|
      cellinfo& s = cells.back();
 | 
						|
      s.patterndir = -1;
 | 
						|
      double a, b, c;
 | 
						|
      scan(f, a, b, c);
 | 
						|
      s.p = hpxyz(a, b, c);
 | 
						|
      for(auto c0: all) s.relmatrices[c0] = calc_relative_matrix(c0, h, s.p);
 | 
						|
      s.owner = h;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
  make_cells_of_heptagon();
 | 
						|
  runlevel = 2;
 | 
						|
  return true;
 | 
						|
  }
 | 
						|
 | 
						|
EX void load_map_bin(hstream& f) {
 | 
						|
  auto& all = base->allcells();
 | 
						|
  eGeometry g = (eGeometry) f.get<short>();
 | 
						|
  if(int(g) == -1) return;
 | 
						|
  int sa = f.get<short>();
 | 
						|
  cellcount = f.get<short>();
 | 
						|
 | 
						|
  if(g != geometry) throw hstream_exception("bad geometry");
 | 
						|
  if(sa != isize(all)) throw hstream_exception("bad size of all");
 | 
						|
  density = cellcount * 1. / isize(all);
 | 
						|
 | 
						|
  cells.clear();
 | 
						|
  float_order.clear();
 | 
						|
 | 
						|
  for(auto h: all) {
 | 
						|
    int q = f.get<short>();
 | 
						|
    if(q < 0 || q > cellcount) throw hstream_exception("incorrect quantity");
 | 
						|
    while(q--) {
 | 
						|
      cells.emplace_back();
 | 
						|
      cellinfo& s = cells.back();
 | 
						|
      s.patterndir = -1;
 | 
						|
      double a, b, c;
 | 
						|
      a = f.get<ld>();
 | 
						|
      b = f.get<ld>();
 | 
						|
      c = f.get<ld>();
 | 
						|
      float_order.push_back(a);
 | 
						|
      float_order.push_back(b);
 | 
						|
      float_order.push_back(c);
 | 
						|
      s.p = hpxyz(a, b, c);
 | 
						|
      s.p = normalize(s.p);
 | 
						|
      for(auto c0: all) s.relmatrices[c0] = calc_relative_matrix(c0, h, s.p);
 | 
						|
      s.owner = h;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
  make_cells_of_heptagon();
 | 
						|
  runlevel = 2;
 | 
						|
  }
 | 
						|
 | 
						|
EX void load_map_full(hstream& f) {
 | 
						|
  init();
 | 
						|
  try {
 | 
						|
    load_map_bin(f);
 | 
						|
    while(runlevel < 10) step(1000);
 | 
						|
    start_game_on_created_map();
 | 
						|
    }
 | 
						|
  catch(hr_exception& e) {
 | 
						|
    cancel_map_creation();
 | 
						|
    throw e;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
EX void cancel_map_creation() {
 | 
						|
  base = NULL;
 | 
						|
  runlevel = 0;
 | 
						|
  gridmaking = false;
 | 
						|
  stop_game();
 | 
						|
  geometry = orig_geometry;
 | 
						|
  }
 | 
						|
 | 
						|
string irrmapfile = "irregularmap.txt";
 | 
						|
 | 
						|
string irrhelp = 
 | 
						|
  "This option creates irregular grids to play the game on. "
 | 
						|
  "Currently rather slow algorithms are used, "
 | 
						|
  "so not recommended with too high density or "
 | 
						|
  "with too large periodic base geometry. "
 | 
						|
  "For technical reasons, the density cannot be too small.";
 | 
						|
 | 
						|
bool too_small_euclidean() {
 | 
						|
  for(cell *c: base->allcells())
 | 
						|
    forCellIdEx(c1, i1, c)
 | 
						|
    forCellIdEx(c2, i2, c)
 | 
						|
      if(i1 != i2 && c1 == c2) return true;
 | 
						|
  return false;
 | 
						|
  }
 | 
						|
 | 
						|
void show_gridmaker() {
 | 
						|
  cmode = sm::SIDE | sm::MAYDARK;
 | 
						|
  gamescreen();
 | 
						|
  dialog::init(XLAT("irregular grid"));
 | 
						|
  dialog::addSelItem(XLAT("density"), fts(density), 'd');
 | 
						|
  dialog::add_action([] {
 | 
						|
    dialog::editNumber(density, 1, 10, .1, 4, XLAT("density"), XLAT(irrhelp));
 | 
						|
    dialog::get_di().reaction = [] () {
 | 
						|
      int s = cellcount;
 | 
						|
      if(density < 1) density = 1;
 | 
						|
      cellcount = int(isize(currentmap->allcells()) * density + .5);
 | 
						|
      println(hlog, "density = ", fts(density), " cellcount = ", cellcount);
 | 
						|
      if(cellcount > s) runlevel = 1;
 | 
						|
      if(cellcount < s) runlevel = 0;
 | 
						|
      };
 | 
						|
    });
 | 
						|
  dialog::addSelItem(XLAT("min edge to median"), fts(quality), 'q');
 | 
						|
  dialog::add_action([] {
 | 
						|
    dialog::editNumber(quality, 0, 1, .05, .2, XLAT("quality"), XLAT(
 | 
						|
      "The smallest allowed ratio of edge length to median edge length. "
 | 
						|
      "Tilings with low values are easier to generate, but tend to be more ugly."
 | 
						|
      ));
 | 
						|
    dialog::get_di().reaction = [] () {
 | 
						|
      println(hlog, "quality = ", density);
 | 
						|
      if(runlevel > 4) runlevel = 4;
 | 
						|
      };
 | 
						|
    });
 | 
						|
  dialog::addBreak(100);
 | 
						|
  for(int i=0; i<5; i++)
 | 
						|
    dialog::addInfo(status[i]);
 | 
						|
  dialog::addBreak(100);
 | 
						|
  dialog::addSelItem(XLAT("activate"), runlevel == 10 ? XLAT("ready") : XLAT("wait..."), 'f');
 | 
						|
  if(runlevel == 10) dialog::add_action([] { popScreen(); start_game_on_created_map(); });
 | 
						|
  dialog::addItem(XLAT("cancel"), 'c');
 | 
						|
  dialog::add_action([] { cancel_map_creation(); popScreen(); start_game(); });
 | 
						|
  dialog::addItem(XLAT("save"), 's');
 | 
						|
  dialog::add_action([] () {
 | 
						|
    dialog::openFileDialog(irrmapfile, XLAT("irregular to save:"), ".txt", [] () {
 | 
						|
      if(save_map(irrmapfile)) {
 | 
						|
        addMessage(XLAT("Map saved to %1", irrmapfile));
 | 
						|
        return true;
 | 
						|
        }
 | 
						|
      else {
 | 
						|
        addMessage(XLAT("Failed to save map to %1", irrmapfile));
 | 
						|
        return false;
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
  dialog::addItem(XLAT("load"), 'l');
 | 
						|
  dialog::add_action([] () {
 | 
						|
    dialog::openFileDialog(irrmapfile, XLAT("irregular to load:"), ".txt", [] () {
 | 
						|
      if(load_map(irrmapfile)) {
 | 
						|
        addMessage(XLAT("Map loaded from %1", irrmapfile));
 | 
						|
        return true;
 | 
						|
        }
 | 
						|
      else {
 | 
						|
        addMessage(XLAT("Failed to load map from %1", irrmapfile));
 | 
						|
        return false;
 | 
						|
        }
 | 
						|
      });
 | 
						|
    });
 | 
						|
  dialog::addSelItem(XLAT("bitruncation count"), its(bitruncations_requested), 'b');
 | 
						|
  dialog::add_action([] () { 
 | 
						|
    dialog::editNumber(bitruncations_requested, 0, 5, 1, 1, XLAT("bitruncation const"),
 | 
						|
      XLAT("Bitruncation introduces some regularity, allowing more sophisticated floor tilings and textures."));
 | 
						|
    dialog::get_di().reaction = [] () {
 | 
						|
      if(bitruncations_requested > bitruncations_performed && runlevel > 5) runlevel = 5;
 | 
						|
      if(bitruncations_requested < bitruncations_performed) runlevel = 0;
 | 
						|
      };
 | 
						|
    });
 | 
						|
  if(too_small_euclidean())
 | 
						|
    dialog::addInfo(XLAT("too small period -- irregular tiling generation fails"));
 | 
						|
  dialog::addItem(XLAT("reset"), 'r');
 | 
						|
  dialog::add_action([] () { runlevel = 0; });
 | 
						|
  dialog::addHelp();
 | 
						|
  dialog::display();
 | 
						|
  keyhandler = [] (int sym, int uni) {
 | 
						|
    handlePanning(sym, uni);
 | 
						|
    if(uni == 'h' || sym == SDLK_F1) gotoHelp(XLAT(irrhelp));
 | 
						|
    dialog::handleNavigation(sym, uni);
 | 
						|
    // no exit
 | 
						|
    };
 | 
						|
  }
 | 
						|
 | 
						|
EX void init() {
 | 
						|
  stop_game();
 | 
						|
  orig_geometry = geometry;
 | 
						|
  switch(geometry) {
 | 
						|
    case gNormal:
 | 
						|
      geometry = gKleinQuartic;
 | 
						|
      break;
 | 
						|
    
 | 
						|
    case gOctagon:
 | 
						|
      geometry = gBolza2;
 | 
						|
      break;
 | 
						|
    
 | 
						|
    default: ;
 | 
						|
      break;
 | 
						|
    }
 | 
						|
 | 
						|
  base_geometry = geometry;
 | 
						|
  variation = eVariation::pure;
 | 
						|
  start_game();
 | 
						|
  if(base) delete base;
 | 
						|
  base = currentmap; 
 | 
						|
  base_config = euc::eu;
 | 
						|
  cellcount = int(isize(base->allcells()) * density + .5);
 | 
						|
  gridmaking = true;
 | 
						|
  drawthemap();
 | 
						|
  }
 | 
						|
 | 
						|
EX void visual_creator() {
 | 
						|
  init();
 | 
						|
  pushScreen(show_gridmaker);
 | 
						|
  runlevel = 0;
 | 
						|
  gridmaking = true;
 | 
						|
  }
 | 
						|
 | 
						|
EX void auto_creator() {
 | 
						|
  variation = eVariation::pure;
 | 
						|
  int cc = cellcount;
 | 
						|
  bitruncations_requested = bitruncations_performed;
 | 
						|
  visual_creator();
 | 
						|
  cellcount = cc; density = cc * 1. / isize(base->allcells());
 | 
						|
  printf("Creating the irregular map automatically...\n");
 | 
						|
  while(runlevel < 10) step(1000);
 | 
						|
  start_game_on_created_map();
 | 
						|
  }
 | 
						|
 | 
						|
#if CAP_COMMANDLINE  
 | 
						|
int readArgs() {
 | 
						|
  using namespace arg;
 | 
						|
           
 | 
						|
  if(0) ;
 | 
						|
  else if(argis("-irrvis")) {
 | 
						|
    PHASE(3);
 | 
						|
    restart_game();
 | 
						|
    visual_creator();
 | 
						|
    showstartmenu = false;
 | 
						|
    }
 | 
						|
  else if(argis("-irrdens")) {
 | 
						|
    PHASEFROM(2);
 | 
						|
    shift_arg_formula(density);
 | 
						|
    }
 | 
						|
  else if(argis("-irrb")) {
 | 
						|
    PHASEFROM(2);
 | 
						|
    shift(); bitruncations_requested = argi();
 | 
						|
    }
 | 
						|
  else if(argis("-irrq")) {
 | 
						|
    PHASEFROM(2);
 | 
						|
    shift_arg_formula(quality);
 | 
						|
    }
 | 
						|
  else if(argis("-irrload")) {
 | 
						|
    PHASE(3);
 | 
						|
    restart_game();
 | 
						|
    init();
 | 
						|
    showstartmenu = false;
 | 
						|
    shift();
 | 
						|
    load_map(args());
 | 
						|
    while(runlevel < 10) step(1000);
 | 
						|
    start_game_on_created_map();
 | 
						|
    }
 | 
						|
  else return 1;
 | 
						|
  return 0;
 | 
						|
  }
 | 
						|
#endif
 | 
						|
 | 
						|
EX unsigned char density_code() {
 | 
						|
  if(isize(cells) < 128) return isize(cells);
 | 
						|
  else {
 | 
						|
    int t = 127, a = isize(cells);
 | 
						|
    while(a > 127) a = a * 9/10, t++;
 | 
						|
    return t;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
EX bool pseudohept(cell* c) {
 | 
						|
  return cells[cellindex[c]].is_pseudohept;
 | 
						|
  }
 | 
						|
 | 
						|
EX bool ctof(cell* c) {
 | 
						|
  return cells[cellindex[c]].patterndir == -1;
 | 
						|
  }
 | 
						|
 | 
						|
EX bool supports(eGeometry g) {
 | 
						|
  if(g == gEuclid || g == gEuclidSquare) return ginf[g].flags & qCLOSED;
 | 
						|
  return among(g, gNormal, gKleinQuartic, gOctagon, gBolza2, gFieldQuotient, gSphere, gSmallSphere, gTinySphere);
 | 
						|
  }
 | 
						|
 | 
						|
EX array<heptagon*, 3> get_masters(cell *c) {
 | 
						|
  int d = cells[cellindex[c]].patterndir;
 | 
						|
  heptspin s = periodmap[c->master].base;
 | 
						|
  heptspin s0 = heptspin(c->master, 0) + (d - s.spin);
 | 
						|
  return make_array(s0.at, (s0 + wstep).at, (s0 + 1 + wstep).at);
 | 
						|
  }
 | 
						|
 | 
						|
EX void swap_vertices() {
 | 
						|
  for(auto& c: cells) {
 | 
						|
    swappoint(c.p);
 | 
						|
    swapmatrix(c.pusher);
 | 
						|
    swapmatrix(c.rpusher);
 | 
						|
    for(auto& jp: c.jpoints) swappoint(jp);
 | 
						|
    for(auto& rm: c.relmatrices) swapmatrix(rm.second);
 | 
						|
    for(auto& v: c.vertices) swappoint(v);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
auto hook = 
 | 
						|
#if CAP_COMMANDLINE
 | 
						|
  addHook(hooks_args, 100, readArgs) + 
 | 
						|
#endif
 | 
						|
#if MAXMDIM >= 4
 | 
						|
  addHook(hooks_swapdim, 100, swap_vertices) +
 | 
						|
#endif
 | 
						|
  addHook(hooks_drawcell, 100, draw_cell_schematics) +
 | 
						|
  addHook(shmup::hooks_turn, 100, step);
 | 
						|
 | 
						|
#endif
 | 
						|
  }}
 | 
						|
 | 
						|
/*
 | 
						|
  if(mouseover && !ctof(mouseover)) {
 | 
						|
    for(auto h: gp::get_masters(mouseover)) 
 | 
						|
      queueline(ggmatrix(h->c7)*C0, shmup::ggmatrix(mouseover)*C0, 0xFFFFFFFF);
 | 
						|
    }
 | 
						|
    
 | 
						|
*/
 |