mirror of
				https://github.com/zenorogue/hyperrogue.git
				synced 2025-10-25 19:07:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			235 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			235 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "rogueviz.h"
 | |
| #include <iostream>
 | |
| #include <thread>
 | |
| 
 | |
| namespace rogueviz {
 | |
| 
 | |
| namespace rwalk {
 | |
| 
 | |
| struct line {
 | |
|   hyperpoint a;
 | |
|   hyperpoint b;
 | |
|   color_t col;
 | |
|   int timestamp;
 | |
|   };
 | |
| 
 | |
| struct rwalker {
 | |
|   cell *at;
 | |
|   transmatrix ori;
 | |
|   transmatrix T;
 | |
|   color_t col;
 | |
|   int simulated;
 | |
|   };
 | |
| 
 | |
| map<cell*, vector<line> > drawn;
 | |
| 
 | |
| vector<rwalker> walkers;
 | |
| 
 | |
| ld total_time;
 | |
| 
 | |
| bool draw_rwalk(cell *c, const shiftmatrix& V) {
 | |
| 
 | |
|   vid.linewidth *= 3;
 | |
|   for(auto p: drawn[c]) if(p.timestamp <= total_time)
 | |
|     queueline(V * p.a, V * p.b, p.col, 0);
 | |
|   vid.linewidth /= 3;
 | |
| 
 | |
|   return false;
 | |
|   }
 | |
| 
 | |
| bool in_video;
 | |
| 
 | |
| ld sim_speed = 5;
 | |
| 
 | |
| int branch_each = 50000;
 | |
| 
 | |
| ld step_size = 0.02;
 | |
| 
 | |
| bool advance_walkers(int delta) {
 | |
|   if(walkers.empty()) {
 | |
|     walkers.emplace_back(rwalker{currentmap->gamestart(), Id, Id, 0xFFFFFFFF, 0});
 | |
|     }
 | |
|   
 | |
|   if(in_video) {
 | |
|     ld t = history::phase / (isize(history::v) - 1);
 | |
|     total_time = walkers[0].simulated * t;
 | |
|     }
 | |
|   else {
 | |
|     total_time += delta * sim_speed;
 | |
|     }
 | |
| 
 | |
|   for(int i=0; i<isize(walkers); i++) {
 | |
|     if(heptdistance(walkers[i].at, centerover) > 7)
 | |
|       continue;
 | |
|     while(walkers[i].simulated < total_time) {
 | |
|       walkers[i].simulated++;
 | |
|       if(branch_each && rand() % branch_each == 0) {
 | |
|         walkers.push_back(walkers[i]);
 | |
|         walkers.back().col = hrandpos() | 0x808080FF;
 | |
|         walkers[i].col = hrandpos() | 0x808080FF;
 | |
|         }
 | |
|       auto& w = walkers[i];
 | |
|       hyperpoint h = tC0(w.T);
 | |
|       if(WDIM == 2) {
 | |
|         w.T = w.T * xspinpush(randd() * TAU, step_size);
 | |
|         }
 | |
|       else {
 | |
|         hyperpoint dir = random_spin() * xtangent(step_size);
 | |
|         apply_shift_object(w.T, w.ori, dir);        
 | |
|         }
 | |
|       fixmatrix(w.T);
 | |
|       hyperpoint h1 = tC0(w.T);
 | |
|       drawn[w.at].emplace_back(line{h, h1, w.col, w.simulated});
 | |
|       virtualRebase(w.at, w.T);
 | |
|       w.simulated++;
 | |
|       }
 | |
|     }
 | |
|   // centerover = walkers[0].at;
 | |
|   // View = inverse(walkers[0].T);
 | |
|   // setdist(centerover, 0, nullptr);
 | |
|   return false;
 | |
|   }
 | |
| 
 | |
| void enable();
 | |
| 
 | |
| int args() {
 | |
|   using namespace arg;
 | |
|            
 | |
|   if(0) ;
 | |
|   else if(argis("-rwalk")) {
 | |
|     enable();
 | |
|     }
 | |
|   else if(argis("-rwparam")) {
 | |
|     shift_arg_formula(sim_speed);
 | |
|     shift_arg_formula(step_size);
 | |
|     shift(); branch_each = argi();
 | |
|     }
 | |
| 
 | |
|   else return 1;
 | |
|   return 0;
 | |
|   }
 | |
| 
 | |
| void show() {
 | |
|   cmode = sm::SIDE | sm::MAYDARK;
 | |
|   gamescreen();
 | |
|   dialog::init(XLAT("random walk"), 0xFFFFFFFF, 150, 0);
 | |
| 
 | |
|   dialog::addSelItem("step size", fts(step_size), 'd');
 | |
|   dialog::add_action([]() {
 | |
|     dialog::editNumber(step_size, 1e-3, 10, 0.1, 1e-2, "step size", "");
 | |
|     dialog::scaleLog();
 | |
|     });
 | |
| 
 | |
|   dialog::addSelItem("steps per millisecond", fts(sim_speed), 'v');
 | |
|   dialog::add_action([]() {
 | |
|     dialog::editNumber(sim_speed, 0, 10, 0.1, 5, "steps per millisecond", "");
 | |
|     });
 | |
| 
 | |
|   dialog::addSelItem("steps per branch", its(branch_each), 'b');
 | |
|   dialog::add_action([]() {
 | |
|     dialog::editNumber(branch_each, 100, 1000000, 0.1, 50000, "steps per branch", "");
 | |
|     dialog::scaleLog();
 | |
|     });
 | |
| 
 | |
|   dialog::addBoolItem("create an animation", in_video, 'a');
 | |
|   dialog::add_action([]() {
 | |
|     in_video = !in_video;
 | |
|     if(!in_video) {
 | |
|       total_time = walkers[0].simulated;
 | |
|       history::on = false;
 | |
|       }
 | |
|     if(in_video) {
 | |
|       history::create(currentmap->gamestart(), walkers[0].at, walkers[0].T);
 | |
|       models::rotation = spin((rand() % 360) * degree);
 | |
|       }
 | |
|     });
 | |
| 
 | |
|   dialog::addBack();
 | |
|   dialog::display();    
 | |
|   }
 | |
| 
 | |
| void o_key(o_funcs& v) {
 | |
|   v.push_back(named_dialog("random walk", show));
 | |
|   }
 | |
| 
 | |
| string cap = "non-Euclidean random walk/";
 | |
| 
 | |
| void rw_slide(vector<tour::slide>& v, string title, string desc, reaction_t t) {
 | |
|   using namespace tour;
 | |
|   v.push_back(
 | |
|     tour::slide{cap + title, 18, LEGAL::NONE | QUICKGEO, desc, 
 | |
|    
 | |
|   [t] (presmode mode) {
 | |
|     setPlainCanvas(mode);
 | |
| 
 | |
|     if(mode == pmStart) {
 | |
|       tour::slide_backup(mapeditor::drawplayer, false);
 | |
|       stop_game();
 | |
|       t();
 | |
|       start_game();
 | |
|       enable();
 | |
|       }
 | |
| 
 | |
|     if(mode == pmKey) {
 | |
|       drawn.clear();
 | |
|       walkers.clear();
 | |
|       total_time = 0;
 | |
|       }
 | |
|     }}
 | |
|     );
 | |
|   }
 | |
| 
 | |
| void enable() {
 | |
|   rv_hook(hooks_drawcell, 100, draw_rwalk);
 | |
|   rv_hook(shmup::hooks_turn, 100, advance_walkers);
 | |
|   rv_hook(hooks_o_key, 80, o_key);
 | |
|   rv_hook(hooks_clearmemory, 40, [] () {
 | |
|     drawn.clear();
 | |
|     walkers.clear();
 | |
|     total_time = 0;
 | |
|     });
 | |
|   }
 | |
| 
 | |
| auto msc = 
 | |
|   addHook(hooks_args, 100, args)
 | |
| + addHook_rvslides(180, [] (string s, vector<tour::slide>& v) {
 | |
|   if(s != "mixed") return;
 | |
|   v.push_back(tour::slide{
 | |
|     cap+"random walk visualization", 10, tour::LEGAL::NONE | tour::QUICKSKIP,
 | |
|     "Here we see random walk in various geometries.\n"
 | |
|     "Press '5' to reset.\n"
 | |
|     ,
 | |
|     [] (tour::presmode mode) {
 | |
|       slide_url(mode, 'y', "YouTube link", "https://www.youtube.com/watch?v=sXNI_i6QZZY");
 | |
|       }
 | |
|     });
 | |
|   rw_slide(v, "Euclidean plane", "In Euclidean plane, the random walk always returns to the neighborhood of the starting point with probability 1.", [] {
 | |
|     set_geometry(gEuclid);
 | |
|     set_variation(eVariation::pure);
 | |
|     sim_speed = 5;
 | |
|     branch_each = 0;
 | |
|     step_size = 0.02;
 | |
|     });
 | |
|   rw_slide(v, "Euclidean 3-space", 
 | |
|     "However, in Euclidean 3-space, it does not return.", [] {
 | |
|     set_geometry(gCubeTiling);
 | |
|     set_variation(eVariation::pure);
 | |
|     sim_speed = 5;
 | |
|     branch_each = 0;
 | |
|     step_size = 0.02;
 | |
|     });
 | |
|   rw_slide(v, "Hyperbolic geometry", "In H2, it does not return, even if we branch from time to time.", [] {
 | |
|     set_geometry(gNormal);
 | |
|     set_variation(eVariation::bitruncated);
 | |
|     sim_speed = 5;
 | |
|     branch_each = 50000;
 | |
|     step_size = 0.02;
 | |
|     });
 | |
|   /* it works in other geometries too -- exercise left for the reader */
 | |
|   });
 | |
| 
 | |
| // {4,5} : 10 6 works
 | |
| 
 | |
| }
 | |
| 
 | |
| } | 
