mirror of
				https://github.com/zenorogue/hyperrogue.git
				synced 2025-10-26 03:17:39 +00:00 
			
		
		
		
	rogueviz:: added objmodels.cpp and some demos using it
This commit is contained in:
		
							
								
								
									
										293
									
								
								rogueviz/ascending-descending.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										293
									
								
								rogueviz/ascending-descending.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,293 @@ | |||||||
|  | #include "rogueviz.h" | ||||||
|  |  | ||||||
|  | namespace hr { | ||||||
|  |  | ||||||
|  | namespace ply { | ||||||
|  |  | ||||||
|  | bool in, animated; | ||||||
|  |  | ||||||
|  | using namespace rogueviz::objmodels; | ||||||
|  |  | ||||||
|  | hyperpoint A = point31(9.7, -38.1, 10.116); | ||||||
|  | hyperpoint B = point31(17.3, -38.1, 10.95); | ||||||
|  | hyperpoint C = point31(17.3, -46.1, 11.75); | ||||||
|  | hyperpoint D = point31(12.95, -46.08, 12.28); | ||||||
|  | hyperpoint E = point31(12.86, -44.53, 12.4); | ||||||
|  |  | ||||||
|  | ld radius = (12.86 - 11.11) / 2; | ||||||
|  |  | ||||||
|  | transmatrix nilform; | ||||||
|  |  | ||||||
|  | map<hyperpoint, pair<int, hyperpoint> > cache; | ||||||
|  |  | ||||||
|  | bool debugnil = false; | ||||||
|  | bool usecache = true; | ||||||
|  |  | ||||||
|  | pair<int, hyperpoint> nilize(hyperpoint h) { | ||||||
|  |    | ||||||
|  |   hyperpoint hc = h; | ||||||
|  |   for(int i=0; i<4; i++) hc[i] = floor(h[i] * 1000 + .5); | ||||||
|  |   if(usecache && cache.count(hc)) return cache[hc]; | ||||||
|  |    | ||||||
|  |   hyperpoint at = A; | ||||||
|  |    | ||||||
|  |   hyperpoint result = C0; | ||||||
|  |    | ||||||
|  |   auto move_coord_full = [&] (hyperpoint tgt) { | ||||||
|  |     for(int i=0; i<100; i++) { | ||||||
|  |       result = nisot::translate(result) * nilform * ((tgt - at) / 100 + C0); | ||||||
|  |       } | ||||||
|  |     at = tgt; | ||||||
|  |     }; | ||||||
|  |    | ||||||
|  |   auto move_coord = [&] (int c, hyperpoint upto) { | ||||||
|  |     ld part = (h[c] - at[c]) / (upto[c] - at[c]); | ||||||
|  |     if(part > 1) part = 1; | ||||||
|  |     move_coord_full(at + part * (upto-at)); | ||||||
|  |     }; | ||||||
|  |    | ||||||
|  |   int cat = 0; | ||||||
|  |    | ||||||
|  |   if(h[1] > -39) { | ||||||
|  |     move_coord(0, B); | ||||||
|  |     cat = 1; | ||||||
|  |     goto finish; | ||||||
|  |     } | ||||||
|  |   move_coord_full(B); | ||||||
|  |   if(h[0] > 17.2) { | ||||||
|  |     move_coord(1, C); | ||||||
|  |     cat = 2; | ||||||
|  |     goto finish;     | ||||||
|  |     } | ||||||
|  |   move_coord_full(C); | ||||||
|  |   if(h[1] < -45.5) { | ||||||
|  |     move_coord(0, D); | ||||||
|  |     cat = 3; | ||||||
|  |     goto finish; | ||||||
|  |     } | ||||||
|  |   cat = 4; | ||||||
|  |   move_coord_full(D); | ||||||
|  |   move_coord(1, E); | ||||||
|  |   finish: | ||||||
|  |    | ||||||
|  |   move_coord_full(h); | ||||||
|  |   return cache[hc] = {cat, result}; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | ld vperiod; | ||||||
|  |  | ||||||
|  | pair<hyperpoint, hyperpoint> trace_path(ld v) { | ||||||
|  |  | ||||||
|  |   ld vorig = v; | ||||||
|  |    | ||||||
|  |   hyperpoint lctr = A; | ||||||
|  |   ld angle = 0; | ||||||
|  |    | ||||||
|  |   ld arclen = radius * M_PI/2; | ||||||
|  |    | ||||||
|  |   auto change_angle = [&] (ld x) { | ||||||
|  |     if(v == 0) return; | ||||||
|  |     else if(v >= arclen) v -= arclen, angle += 1; | ||||||
|  |     else angle += v / arclen, v = 0; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |   auto shift_to = [&] (hyperpoint h, bool last = false) { | ||||||
|  |     ld seglen = hypot_d(3, h - lctr); | ||||||
|  |     if(v == 0) return; | ||||||
|  |     else if(v >= seglen && !last) v -= seglen, lctr = h; | ||||||
|  |     else lctr = lerp(lctr, h, v / seglen), v = 0; | ||||||
|  |     }; | ||||||
|  |    | ||||||
|  |   change_angle(1); | ||||||
|  |   shift_to(B); | ||||||
|  |   change_angle(2); | ||||||
|  |   shift_to(C); | ||||||
|  |   change_angle(3); | ||||||
|  |   shift_to(D); | ||||||
|  |   change_angle(4); | ||||||
|  |   shift_to(E, true); | ||||||
|  |    | ||||||
|  |   angle *= M_PI/2; | ||||||
|  |    | ||||||
|  |   if(v > 0) vperiod = vorig - v; | ||||||
|  |    | ||||||
|  |   return { lctr + point3(radius * -cos(angle), radius * sin(angle), 0), point3(-sin(angle), -cos(angle), 0) };   | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | transmatrix scaleby(ld x) { | ||||||
|  |   transmatrix res = Id; | ||||||
|  |   for(int i=0; i<LDIM; i++) | ||||||
|  |     res[i][i] = x; | ||||||
|  |   return res; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | vector<hyperpoint> route(1000), forwards(1000), overroute(1000); | ||||||
|  |  | ||||||
|  | hyperpoint interpolate_at(vector<hyperpoint>& v, ld x) { | ||||||
|  |   while(x > 1000) x -= 1000; | ||||||
|  |   return lerp(v[x], v[(int(x)+1) % 1000], frac(x)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | ld advance = 2; | ||||||
|  | ld over = 1.5; | ||||||
|  | ld over2 = 1; | ||||||
|  |  | ||||||
|  | void make_routes() { | ||||||
|  |   for(int i=0; i<1000; i++) { | ||||||
|  |     ld v = vperiod * i / 1000; | ||||||
|  |     route[i] = nilize(trace_path(v).first + point3(0,0,over)).second; | ||||||
|  |     ld vb = v + advance; | ||||||
|  |     if(vb > vperiod) vb -= vperiod; | ||||||
|  |     forwards[i] = nilize(trace_path(vb).first).second; | ||||||
|  |     overroute[i] = nilize(trace_path(v).first + point3(0,0,over+over2)).second; | ||||||
|  |     } | ||||||
|  |    | ||||||
|  |   auto smoothen = [&] (vector<hyperpoint>& v) { | ||||||
|  |     for(int it=0; it<100; it++) { | ||||||
|  |       vector<hyperpoint> smoother; | ||||||
|  |       for(int i=0; i<1000; i++) { | ||||||
|  |         hyperpoint& a = v[i]; | ||||||
|  |         hyperpoint& b = v[(i+1) % 1000]; | ||||||
|  |         hyperpoint& c = v[(i+2) % 1000]; | ||||||
|  |         smoother.push_back((a + b + c) / 3); | ||||||
|  |         } | ||||||
|  |       v = smoother; | ||||||
|  |       } | ||||||
|  |     }; | ||||||
|  |    | ||||||
|  |   smoothen(route); | ||||||
|  |   smoothen(forwards); | ||||||
|  |   smoothen(overroute); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | bool prepared; | ||||||
|  |  | ||||||
|  | void prepare_nilform() { | ||||||
|  |   if(prepared) return; | ||||||
|  |   prepared = true; | ||||||
|  |   hyperpoint axis = (E - A); | ||||||
|  |   transmatrix rotator = Id; | ||||||
|  |   rotator[2] = axis; | ||||||
|  |   println(hlog, " = ", (rotator[2] | rotator[2])); | ||||||
|  |   rotator[2] /= sqrt(rotator[2] | rotator[2]); | ||||||
|  |   rotator[1] -= (rotator[1] | rotator[2]) * rotator[2]; | ||||||
|  |   rotator[1] /= sqrt(rotator[1] | rotator[1]); | ||||||
|  |   rotator[0] -= (rotator[0] | rotator[2]) * rotator[2]; | ||||||
|  |   rotator[0] -= (rotator[0] | rotator[1]) * rotator[1]; | ||||||
|  |   rotator[0] /= sqrt(rotator[0] | rotator[0]); | ||||||
|  |   println(hlog, "rotator = ", kz(rotator)); | ||||||
|  |   rotator = inverse(transpose(rotator)); | ||||||
|  |   println(hlog, "rotator = ", kz(rotator));     | ||||||
|  |    | ||||||
|  |   ld minscale = 0.5; // positive | ||||||
|  |   ld maxscale = 1.5; // negative | ||||||
|  |   ld scale; | ||||||
|  |    | ||||||
|  |   for(int it=0; it<100; it++) {   | ||||||
|  |     scale = (minscale + maxscale) / 2; | ||||||
|  |     nilform = rotator * scaleby(scale); | ||||||
|  |     cache.clear(); | ||||||
|  |     hyperpoint nE = nilize(E).second; | ||||||
|  |     if(nE[2] < 0) maxscale = scale; | ||||||
|  |     else minscale = scale; | ||||||
|  |     } | ||||||
|  |    | ||||||
|  |   println(hlog, "scale = ", scale); | ||||||
|  |   println(hlog, nilize(E).second);   | ||||||
|  |    | ||||||
|  |   vperiod = radius * 2 * M_PI + hypot_d(3, B-A) + hypot_d(3, C-B) + hypot_d(3, D-C) + hypot_d(3, E-D); | ||||||
|  |   println(hlog, "vperiod = ", vperiod); | ||||||
|  |    | ||||||
|  |   make_routes(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | model staircase("rogueviz/nil/", "aco.obj", nilize); | ||||||
|  |  | ||||||
|  | bool draw_ply() { | ||||||
|  |  | ||||||
|  |   if(!in) return false; | ||||||
|  |    | ||||||
|  |   if(nil) prepare_nilform(); | ||||||
|  |  | ||||||
|  |   staircase.render(ggmatrix(currentmap->gamestart())); | ||||||
|  |    | ||||||
|  |   return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | void show() { | ||||||
|  |   cmode = sm::SIDE | sm::MAYDARK; | ||||||
|  |   gamescreen(0); | ||||||
|  |   dialog::init(XLAT("Ascending & Descending"), 0xFFFFFFFF, 150, 0); | ||||||
|  |  | ||||||
|  |   dialog::addSelItem("advance", fts(advance), 'a'); | ||||||
|  |   dialog::add_action([]() { | ||||||
|  |     dialog::editNumber(advance, 0, 100, 1, 1, "advance", ""); | ||||||
|  |     dialog::reaction = make_routes; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |   dialog::addSelItem("over", fts(over), 'o'); | ||||||
|  |   dialog::add_action([]() { | ||||||
|  |     dialog::editNumber(over, 0, 100, 1, 1, "over", ""); | ||||||
|  |     dialog::reaction = make_routes; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |   dialog::addSelItem("over2", fts(over2), 'p'); | ||||||
|  |   dialog::add_action([]() { | ||||||
|  |     dialog::editNumber(over2, 0, 100, 1, 1, "over2", ""); | ||||||
|  |     dialog::reaction = make_routes; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |   dialog::addBoolItem_action("animated", animated, 'a'); | ||||||
|  |  | ||||||
|  |   dialog::addBack(); | ||||||
|  |   dialog::display();     | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | void o_key(o_funcs& v) { | ||||||
|  |   if(in) v.push_back(named_dialog("Ascending & Descending", show)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | auto plyhook = addHook(hooks_frame, 100, draw_ply) | ||||||
|  |   | ||||||
|  | + addHook(anims::hooks_anim, 100, [] {  | ||||||
|  |   if(!in || !animated) return; | ||||||
|  |   usecache = false; | ||||||
|  |   ld t = ticks * 1. / anims::period; | ||||||
|  |   t = t - floor(t); | ||||||
|  |   t *= 1000; | ||||||
|  |    | ||||||
|  |   centerover = currentmap->gamestart(); | ||||||
|  |   set_view( | ||||||
|  |     interpolate_at(route, t), | ||||||
|  |     interpolate_at(forwards, t), | ||||||
|  |     interpolate_at(overroute, t) | ||||||
|  |     ); | ||||||
|  |    | ||||||
|  |   anims::moved(); | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  | + addHook(hooks_o_key, 80, o_key) | ||||||
|  |  | ||||||
|  | #if CAP_COMMANDLINE | ||||||
|  | + addHook(hooks_args, 100, [] { | ||||||
|  |   using namespace arg; | ||||||
|  |             | ||||||
|  |   if(0) ; | ||||||
|  |   else if(argis("-asd")) { | ||||||
|  |     in = true; | ||||||
|  |     } | ||||||
|  |   else if(argis("-asd-prec")) { | ||||||
|  |     shift(); prec = argi(); | ||||||
|  |     } | ||||||
|  |   else if(argis("-asd-anim")) { | ||||||
|  |     in = true; | ||||||
|  |     animated = true; | ||||||
|  |     } | ||||||
|  |   else return 1; | ||||||
|  |   return 0; | ||||||
|  |   }) | ||||||
|  | #endif | ||||||
|  |   ;  | ||||||
|  |  | ||||||
|  | } | ||||||
|  | } | ||||||
							
								
								
									
										189
									
								
								rogueviz/backhead.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										189
									
								
								rogueviz/backhead.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,189 @@ | |||||||
|  | #include "rogueviz.h" | ||||||
|  |  | ||||||
|  | /*  | ||||||
|  |  | ||||||
|  | used for https://youtu.be/KxjnibOkuLs | ||||||
|  |  | ||||||
|  | you need the models from https://renderpeople.com/free-3d-people/ (and convert them to the Wavefront OBJ format) | ||||||
|  |  | ||||||
|  | -geo 120c -canvas 1 -noplayer camspd=.1 | ||||||
|  |  | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | namespace hr { | ||||||
|  |  | ||||||
|  | namespace backhead { | ||||||
|  |  | ||||||
|  | int view_model_id = 0; | ||||||
|  | int camera_model = 1; | ||||||
|  |  | ||||||
|  | using namespace rogueviz::objmodels; | ||||||
|  |  | ||||||
|  | map<int, model> models; | ||||||
|  |  | ||||||
|  | tf_result person_tf(hyperpoint h) { return {0, direct_exp(h*5)}; } | ||||||
|  |  | ||||||
|  | ld smoothen(ld x) { return x*x*(3-2*x); } | ||||||
|  |  | ||||||
|  | ld cm = 0.08 / 160; | ||||||
|  |  | ||||||
|  | bool use_camera; | ||||||
|  |  | ||||||
|  | transmatrix eyematrix(int id) { | ||||||
|  |   transmatrix S = Id; | ||||||
|  |    | ||||||
|  |   int i = 0; | ||||||
|  |   if(id == 1) for(ld val: {-0.997647,0.00073308,-0.0685486,7.63928e-05,0.0147183,-0.971198,-0.224516,0.0784166,-0.0669523,-0.224937,0.972022,0.00970247,-0.000429702,0.0785862,0.00820562,0.996873}) | ||||||
|  |     S[0][i++] = val; | ||||||
|  |   if(id == 0) for(ld val: {-0.976869,0.119464,-0.17719,-0.00766071,-0.0534002,-0.935386,-0.339797,0.0820953,-0.207062,-0.322171,0.923596,0.0173391,0.000492418,0.0835894,0.0105615,0.996444}) | ||||||
|  |     S[0][i++] = val; | ||||||
|  |    | ||||||
|  |   return S; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | ld back = 0; | ||||||
|  |  | ||||||
|  | bool do_anim = false; | ||||||
|  |  | ||||||
|  | int tf; | ||||||
|  |  | ||||||
|  | vector<string> captions = { | ||||||
|  |   "as seen by another person", | ||||||
|  |   "changing the view...", | ||||||
|  |   "L/R", | ||||||
|  |   "downward", | ||||||
|  |   "upwards", | ||||||
|  |   "back" | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  | EX bool ourStats() { | ||||||
|  |  | ||||||
|  |   /* | ||||||
|  |   if(tf < 0 || tf >= 5) { println(hlog, "tf=", tf); tf = gmod(tf, 5); } | ||||||
|  |    | ||||||
|  |   println(hlog, "displaying tf = ", tf); | ||||||
|  |  | ||||||
|  |   displayfr(10, 10 + 7 * vid.fsize, 2, vid.fsize * 7, captions[tf], 0xFFFFFF, 0); | ||||||
|  |  | ||||||
|  |   nohelp = true; | ||||||
|  |   nomenukey = true; | ||||||
|  |   clearMessages(); | ||||||
|  |  | ||||||
|  |   glflush(); | ||||||
|  |  | ||||||
|  |   println(hlog, "done?"); | ||||||
|  |   */ | ||||||
|  |    | ||||||
|  |   return true; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | void face_animation() { | ||||||
|  |  | ||||||
|  |   if(!do_anim) return; | ||||||
|  |  | ||||||
|  |   ld t = ticks / anims::period; | ||||||
|  |    | ||||||
|  |   t = t - floor(t); | ||||||
|  |    | ||||||
|  |   t *= 2; | ||||||
|  |   view_model_id = (t >= 1 ? 0 : 1); | ||||||
|  |   camera_model = 1 - view_model_id; | ||||||
|  |   if(t>=1) t -= 1; | ||||||
|  |    | ||||||
|  |   t *= 4.5; | ||||||
|  |    | ||||||
|  |   if(t < 1) { | ||||||
|  |     if(t < .7) t /= .7; | ||||||
|  |     else t = (t-.7) / .3 + 1; | ||||||
|  |     } | ||||||
|  |   else | ||||||
|  |     t++; | ||||||
|  |  | ||||||
|  |   // t *= 5; | ||||||
|  |    | ||||||
|  |   tf = t; | ||||||
|  |   t -= tf; | ||||||
|  |    | ||||||
|  |   View = Id; | ||||||
|  |    | ||||||
|  |   ld up = view_model_id == 0 ? 82 : 90; | ||||||
|  |   ld down = view_model_id == 1 ? 65 : 70; | ||||||
|  |    | ||||||
|  |   up -= 15; down -= 15; | ||||||
|  |  | ||||||
|  |   tie(up, down) = make_pair(-down, -up); | ||||||
|  |    | ||||||
|  |   if(tf == 0) { | ||||||
|  |     View = cspin(0, 2, 360 * degree * smoothen(t)) * View; | ||||||
|  |     View = zpush(-0.1) * View; | ||||||
|  |     View = cspin(0, 2, M_PI) * View; | ||||||
|  |     back = 0; | ||||||
|  |     } | ||||||
|  |   else if(tf == 1) { | ||||||
|  |     View = zpush(-0.1 * (1-smoothen(t))) * View; | ||||||
|  |     if(t > .9) | ||||||
|  |     back = -0.1 * smoothen(10*t-9); | ||||||
|  |     View = cspin(0, 2, M_PI) * View; | ||||||
|  |     } | ||||||
|  |   else if(tf == 2) { | ||||||
|  |     View = cspin(0, 2, 75*degree*sin(2*M_PI*smoothen(t))) * View; | ||||||
|  |     } | ||||||
|  |   else if(tf == 3) { | ||||||
|  |     View = cspin(1, 2, up*degree*smoothen(t)) * View; | ||||||
|  |     } | ||||||
|  |   else if(tf == 4) { | ||||||
|  |     View = cspin(1, 2, degree*(up-(up+down)*smoothen(t))) * View; | ||||||
|  |     } | ||||||
|  |   else if(tf == 5) { | ||||||
|  |     View = cspin(1, 2, degree*-down*(1-smoothen(t*2))) * View; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |   use_camera = tf <= 1; | ||||||
|  |  | ||||||
|  |   sightranges[geometry] = use_camera ? 30 : 100; | ||||||
|  |   if(tf == 5) sightranges[geometry] = 100 - 70 * smoothen(t*2); | ||||||
|  |    | ||||||
|  |   anims::moved(); | ||||||
|  |  | ||||||
|  |   hide_hud = false; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  | bool draw_ply() { | ||||||
|  |  | ||||||
|  |   if(models.empty()) { | ||||||
|  |     models[0] = model("rogueviz/models/", "dennis.obj", person_tf); | ||||||
|  |     models[1] = model("rogueviz/models/", "mei.obj", person_tf); | ||||||
|  |     } | ||||||
|  |    | ||||||
|  |   shiftmatrix Zero = ggmatrix(currentmap->gamestart()); | ||||||
|  |    | ||||||
|  |   Zero = Zero * eyematrix(view_model_id); | ||||||
|  |   println(hlog, Zero); | ||||||
|  |    | ||||||
|  |   models[view_model_id].render(Zero); | ||||||
|  |  | ||||||
|  |   if(use_camera) models[camera_model].render(shiftless(eyematrix(camera_model) * zpush(back))); | ||||||
|  |    | ||||||
|  |   return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | auto plyhook = addHook(hooks_frame, 100, draw_ply) | ||||||
|  |   + addHook(anims::hooks_anim, 100, face_animation) | ||||||
|  | + addHook(hooks_args, 100, [] { | ||||||
|  |   using namespace arg; | ||||||
|  |             | ||||||
|  |   if(0) ; | ||||||
|  |   else if(argis("-head-swap")) { | ||||||
|  |     swap(view_model_id, camera_model); | ||||||
|  |     } | ||||||
|  |   else if(argis("-head-anim")) { | ||||||
|  |     do_anim = !do_anim; | ||||||
|  |     } | ||||||
|  |   else return 1; | ||||||
|  |   return 0; | ||||||
|  |   }) | ||||||
|  |   + addHook(hooks_prestats, 100, ourStats) | ||||||
|  |     ; | ||||||
|  |  | ||||||
|  | } | ||||||
|  | } | ||||||
							
								
								
									
										94
									
								
								rogueviz/hypcity.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								rogueviz/hypcity.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,94 @@ | |||||||
|  | #include "rogueviz.h" | ||||||
|  |  | ||||||
|  | // 'hyperbolic city' demo | ||||||
|  | // download the model from https://sketchfab.com/3d-models/night-city-p2-82637933a7cb4fafadb0e2a79415c438 as rogueviz/models/emilejohansson_p2.obj | ||||||
|  |  | ||||||
|  | // see the results posted here: | ||||||
|  |  | ||||||
|  | // https://twitter.com/ZenoRogue/status/1375750351391981570 | ||||||
|  | // -noplayer -geo 4x5 -gp 1 1 -unrectified -switch-fpp -canvas 303030 camera=0 depth=0 -sr 3 -PM 0 -alpha 1 -zoom .95 | ||||||
|  |  | ||||||
|  | // https://twitter.com/ZenoRogue/status/1375748835046215682 | ||||||
|  | // -noplayer -geo nil -canvas 303030 -back 44e4 -sight3 3 | ||||||
|  |  | ||||||
|  | // https://twitter.com/ZenoRogue/status/1375754422752575488 | ||||||
|  | // add -PM 0 -alpha 1 | ||||||
|  |  | ||||||
|  | namespace hr { | ||||||
|  |  | ||||||
|  | using namespace rogueviz::objmodels; | ||||||
|  |  | ||||||
|  | model city("rogueviz/models/", "emilejohansson_p2.obj"); | ||||||
|  |  | ||||||
|  | hyperpoint low, high; | ||||||
|  |  | ||||||
|  | void prepare_tf() { | ||||||
|  |   if(!city.models.empty()) return; | ||||||
|  |    | ||||||
|  |   prec = 40; | ||||||
|  |  | ||||||
|  |   for(int i=0; i<4; i++) low[i] = 100, high[i] = -100; | ||||||
|  |  | ||||||
|  |   cgi.require_basics(); | ||||||
|  |   hyperpoint corner = get_corner_position(cwt.at, 0); | ||||||
|  |    | ||||||
|  |   ld t = abs(corner[0] / corner[3]); | ||||||
|  |  | ||||||
|  |   city.tf = [=] (hyperpoint h) -> pair<int, hyperpoint> { | ||||||
|  |     swap(h[1], h[2]); | ||||||
|  |     h[2] = -h[2]; | ||||||
|  |     h[2] += 0.063; | ||||||
|  |     h[0] -= 0.063; | ||||||
|  |     h[1] += 0.063; | ||||||
|  |     h *= 6; | ||||||
|  |     // h[0] -= .5; | ||||||
|  |     // h[1] += .5; | ||||||
|  |  | ||||||
|  |     for(int i=0; i<4; i++) | ||||||
|  |       low[i] = min(low[i], h[i]), | ||||||
|  |       high[i] = max(high[i], h[i]); | ||||||
|  |        | ||||||
|  |     if(hyperbolic) { | ||||||
|  |      | ||||||
|  |       hyperpoint hx; | ||||||
|  |       hx[0] = h[0] * t * 2; | ||||||
|  |       hx[1] = h[1] * t * 2; | ||||||
|  |       hx[2] = 0; | ||||||
|  |       hx[3] = 1; | ||||||
|  |       hx = spin(45 * degree) * hx; | ||||||
|  |       normalize(hx); | ||||||
|  |       hx = zshift(hx, h[2]*(t*7)); | ||||||
|  |  | ||||||
|  |       return {0, hx};  | ||||||
|  |       } | ||||||
|  |      | ||||||
|  |     if(nil) { | ||||||
|  |       swap(h[1], h[2]); | ||||||
|  |       h *= 0.5 / 0.378; | ||||||
|  |       h[3] = 1; | ||||||
|  |       return {0, h}; | ||||||
|  |       } | ||||||
|  |      | ||||||
|  |     return {0, h}; | ||||||
|  |     }; | ||||||
|  |   println(hlog, "low = ", low); | ||||||
|  |   println(hlog, "high = ", high); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | bool draw_city_at(cell *c, const shiftmatrix& V) { | ||||||
|  |   prepare_tf(); | ||||||
|  |    | ||||||
|  |   if(nil) { | ||||||
|  |     auto co = nilv::get_coord(c->master); | ||||||
|  |     if(co[1]) return false; | ||||||
|  |     } | ||||||
|  |    | ||||||
|  |   if(c == cwt.at || true)  | ||||||
|  |     city.render(V); | ||||||
|  |  | ||||||
|  |   return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | auto hypcity_ah = addHook(hooks_drawcell, 100, draw_city_at); | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										243
									
								
								rogueviz/objmodels.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										243
									
								
								rogueviz/objmodels.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,243 @@ | |||||||
|  | /* a library for loading 3D models */ | ||||||
|  |  | ||||||
|  | #include "rogueviz.h" | ||||||
|  |  | ||||||
|  | namespace rogueviz { | ||||||
|  |  | ||||||
|  | namespace objmodels { | ||||||
|  |  | ||||||
|  | bool scan(fhstream& hs, char& c) { return fscanf(hs.f, "%c", &c) == 1; } | ||||||
|  |  | ||||||
|  | char peek(fhstream& fs) { | ||||||
|  |   char g = fgetc(fs.f); | ||||||
|  |   ungetc(g, fs.f); | ||||||
|  |   return g; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | void model::load_obj(model_type& objects) { | ||||||
|  |   fhstream fs(path+fname, "rt"); | ||||||
|  |  | ||||||
|  |   if(!fs.f)  | ||||||
|  |     throw hr_exception("failed to open model file: " + path + fname); | ||||||
|  |  | ||||||
|  |   vector<hyperpoint> vertices; | ||||||
|  |   vector<hyperpoint> normals; | ||||||
|  |   vector<hyperpoint> tvertices; | ||||||
|  |  | ||||||
|  |   while(!feof(fs.f)) { | ||||||
|  |     string s; | ||||||
|  |     scan(fs, s); | ||||||
|  |     if(s == "#") { scanline(fs); } | ||||||
|  |     else if(s == "mtllib") { | ||||||
|  |       string mtllib; | ||||||
|  |       scan(fs, mtllib); | ||||||
|  |       fhstream fsm(path+mtllib, "rt"); | ||||||
|  |       if(!fsm.f)  | ||||||
|  |         throw hr_exception("failed to open mtllib: " + mtllib); | ||||||
|  |       color_t nextcol = 0xFFFFFFFF; | ||||||
|  |       string mtlname, texname = ""; | ||||||
|  |       auto emit_material = [&] { | ||||||
|  |         if(texname != "") { | ||||||
|  |           texture::texture_data tdata; | ||||||
|  |           materials[mtlname] = tdata; | ||||||
|  |           auto& mat = materials[mtlname]; | ||||||
|  |           mat.twidth = mat.theight = 0; mat.stretched = true; | ||||||
|  |           println(hlog, "texname: ", texname); | ||||||
|  |           mat.readtexture(path+texname); | ||||||
|  |           mat.loadTextureGL(); | ||||||
|  |           println(hlog, "texture ID: ", mat.textureid); | ||||||
|  |           } | ||||||
|  |         colors[mtlname] = nextcol; | ||||||
|  |         println(hlog, "color of ", mtlname, " is ", nextcol); | ||||||
|  |         }; | ||||||
|  |       while(!feof(fsm.f)) { | ||||||
|  |         string s; | ||||||
|  |         scan(fsm, s); | ||||||
|  |         if(s == "#") { scanline(fsm); } | ||||||
|  |         if(s == "Kd") { | ||||||
|  |           ld a, b, c; | ||||||
|  |           scan(fsm, a, b, c); | ||||||
|  |           part(nextcol, 1) = a * 319.99; | ||||||
|  |           part(nextcol, 2) = b * 319.99; | ||||||
|  |           part(nextcol, 3) = c * 319.99; | ||||||
|  |           } | ||||||
|  |         if(s == "newmtl") { | ||||||
|  |           emit_material(); | ||||||
|  |           nextcol = 0xFFFFFFFF; | ||||||
|  |           texname = ""; | ||||||
|  |           mtlname = scanline(fsm); | ||||||
|  |           } | ||||||
|  |         if(s == "map_Kd") { | ||||||
|  |           scan(fsm, texname); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       emit_material(); | ||||||
|  |       } | ||||||
|  |     else if(s == "o" || s == "g") { | ||||||
|  |       next_object: | ||||||
|  |       object *co = nullptr; | ||||||
|  |       bool textured = false; | ||||||
|  |       string oname = scanline(fs); | ||||||
|  |       println(hlog, "reading object: ", oname); | ||||||
|  |       while(true) { | ||||||
|  |         if(feof(fs.f)) { | ||||||
|  |           if(co) cgi.finishshape(); | ||||||
|  |           if(co) println(hlog, "vertices = ", co->sh.e-co->sh.s, " tvertices = ", isize(co->tv.tvertices)); | ||||||
|  |           break; | ||||||
|  |           } | ||||||
|  |         scan(fs, s); | ||||||
|  |         if(s == "#") { | ||||||
|  |           scanline(fs); | ||||||
|  |           } | ||||||
|  |         else if(s == "o" || s == "g") { | ||||||
|  |           if(co) cgi.finishshape(); | ||||||
|  |           if(co) println(hlog, "vertices = ", co->sh.e-co->sh.s, " tvertices = ", isize(co->tv.tvertices)); | ||||||
|  |           goto next_object; | ||||||
|  |           } | ||||||
|  |         else if(s == "v") {         | ||||||
|  |           hyperpoint h = C0; | ||||||
|  |           scan(fs, h[0], h[1], h[2]); // assume all | ||||||
|  |           h[0] /= 100; | ||||||
|  |           h[1] /= 100; | ||||||
|  |           h[2] /= 100; | ||||||
|  |           vertices.push_back(h); | ||||||
|  |           } | ||||||
|  |         else if(s == "vt") { | ||||||
|  |           ld u, v; | ||||||
|  |           scan(fs, u, v); | ||||||
|  |           tvertices.push_back(point3(u, 1-v, 0)); | ||||||
|  |           } | ||||||
|  |         else if(s == "vn") { | ||||||
|  |           hyperpoint hn = C0; | ||||||
|  |           scan(fs, hn[0], hn[1], hn[2]); | ||||||
|  |           normals.push_back(hn); | ||||||
|  |           } | ||||||
|  |         else if(s == "s") { | ||||||
|  |           scan(fs, s); | ||||||
|  |           } | ||||||
|  |         else if(s == "usemtl") { | ||||||
|  |           if(co) cgi.finishshape(); | ||||||
|  |           if(co) println(hlog, "vertices = ", co->sh.e-co->sh.s, " tvertices = ", isize(co->tv.tvertices)); | ||||||
|  |           string mtlname = scanline(fs); | ||||||
|  |           co = nullptr; | ||||||
|  |           if(mtlname.find("Layer_Layer0") != string::npos) continue; | ||||||
|  |           objects.push_back(make_shared<object>()); | ||||||
|  |           co = &*objects.back(); | ||||||
|  |           cgi.bshape(co->sh, PPR::WALL); | ||||||
|  |           cgi.last->flags |= POLY_TRIANGLES; | ||||||
|  |           cgi.last->texture_offset = 0; | ||||||
|  |           if(materials.count(mtlname)) { | ||||||
|  |             textured = true; | ||||||
|  |             cgi.last->tinf = &co->tv; | ||||||
|  |             co->tv.texture_id = materials[mtlname].textureid; | ||||||
|  |             println(hlog, "using texture_id : ", co->tv.texture_id); | ||||||
|  |             co->color = 0xFFFFFFFF; | ||||||
|  |             } | ||||||
|  |           else { | ||||||
|  |             textured = false; | ||||||
|  |             cgi.last->tinf = &co->tv; | ||||||
|  |             co->tv.texture_id = floor_textures->renderedTexture; | ||||||
|  |             if(colors.count(mtlname)) | ||||||
|  |               co->color = colors[mtlname]; | ||||||
|  |             else | ||||||
|  |               co->color = 0xFFFFFFFF; | ||||||
|  |             } | ||||||
|  |           println(hlog, "set textured to ", textured); | ||||||
|  |           } | ||||||
|  |         else if(s == "f") { | ||||||
|  |           struct vertexinfo { int f, t, n; }; | ||||||
|  |           array<vertexinfo, 3> vis; | ||||||
|  |           vector<hyperpoint> hys; | ||||||
|  |           vector<hyperpoint> tot; | ||||||
|  |           char bar; | ||||||
|  |           for(int i=0; i<3; i++) { | ||||||
|  |             vis[i].f = vis[i].t = vis[i].n = 1; | ||||||
|  |             scan(fs, vis[i].f); | ||||||
|  |             if(peek(fs) == '/') { | ||||||
|  |               scan(fs, bar); | ||||||
|  |               if(peek(fs) != '/') scan(fs, vis[i].t); | ||||||
|  |               } | ||||||
|  |             if(peek(fs) == '/') { | ||||||
|  |               scan(fs, bar); | ||||||
|  |               scan(fs, vis[i].n); | ||||||
|  |               } | ||||||
|  |  | ||||||
|  |             vis[i].f--; vis[i].t--; vis[i].n--; | ||||||
|  |             if(vis[i].f < 0 || vis[i].f >= isize(vertices))  | ||||||
|  |               throw hr_exception("illegal ID"); | ||||||
|  |             hys.push_back(vertices[vis[i].f]); | ||||||
|  |             tot.push_back(textured ? tvertices[vis[i].t] : point3(0,0,0)); | ||||||
|  |             } | ||||||
|  |           if(!co) continue; | ||||||
|  |            | ||||||
|  |           hyperpoint norm = (hys[1] - hys[0]) ^ (hys[2] - hys[0]); | ||||||
|  |           norm /= hypot_d(3, norm); | ||||||
|  |           ld y = .5 + (.2 * norm[0] + .16 * norm[1] + .14 * norm[2]); | ||||||
|  |           glvertex shade = glhr::makevertex(0, y, 0); | ||||||
|  |           glvertex shadecol = glhr::makevertex(y, y, y); | ||||||
|  |            | ||||||
|  |           auto n0 = tf(hys[0]); | ||||||
|  |           auto n1 = tf(hys[1]); | ||||||
|  |           auto n2 = tf(hys[2]); | ||||||
|  |           auto mi = min(n0.first, min(n1.first, n2.first)); | ||||||
|  |           auto ma = max(n0.first, max(n1.first, n2.first)); | ||||||
|  |           if(ma - mi > 1) continue; | ||||||
|  |            | ||||||
|  |           int parts = sd(hys); | ||||||
|  |           auto tri = [&] (int a, int b) { | ||||||
|  |             cgi.hpcpush(tf(hys[0] + (hys[1] - hys[0]) * a / parts + (hys[2] - hys[0]) * b / parts).second); | ||||||
|  |             // cgi.hpcpush(tf(tot[0] + (tot[1] - tot[0]) * a / parts + (tot[2] - tot[0]) * b / parts).second); | ||||||
|  |             if(textured) { | ||||||
|  |               co->tv.tvertices.push_back(glhr::pointtogl(tot[0] + (tot[1] - tot[0]) * a / parts + (tot[2] - tot[0]) * b / parts)); | ||||||
|  |               co->tv.colors.push_back(shadecol); | ||||||
|  |               } | ||||||
|  |             else { | ||||||
|  |               co->tv.tvertices.push_back(shade); | ||||||
|  |               } | ||||||
|  |             }; | ||||||
|  |            | ||||||
|  |           for(int a=0; a<parts; a++) | ||||||
|  |           for(int b=0; b<parts-a; b++) { | ||||||
|  |             tri(a, b); | ||||||
|  |             tri(a+1, b); | ||||||
|  |             tri(a, b+1); | ||||||
|  |             if(a+b < parts-1) { | ||||||
|  |               tri(a, b+1); | ||||||
|  |               tri(a+1, b); | ||||||
|  |               tri(a+1, b+1); | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         else if(s == "l") { | ||||||
|  |           int a, b; | ||||||
|  |           scan(fs, a, b); | ||||||
|  |           /* ignore */ | ||||||
|  |           } | ||||||
|  |         else if(s == "") { } | ||||||
|  |         else  | ||||||
|  |           throw hr_exception("unknown command: " + s); | ||||||
|  |         }       | ||||||
|  |        | ||||||
|  |       } | ||||||
|  |     else  | ||||||
|  |       throw("unknown command: " + s); | ||||||
|  |     } | ||||||
|  |    | ||||||
|  |   println(hlog, "reading finished"); | ||||||
|  |  | ||||||
|  |   cgi.extra_vertices();   | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | void model::render(const shiftmatrix& V) { | ||||||
|  |  | ||||||
|  |   auto& objs = models[cgi_string()]; | ||||||
|  |    | ||||||
|  |   if(objs.empty()) load_obj(objs); | ||||||
|  |  | ||||||
|  |   for(auto& obj: objs) { | ||||||
|  |     queuepoly(V, obj->sh, obj->color);   | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | } | ||||||
|  | } | ||||||
| @@ -181,6 +181,55 @@ function<void(presmode)> roguevizslide_action(char c, const T& t, const U& act) | |||||||
|   colorpair perturb(colorpair cp); |   colorpair perturb(colorpair cp); | ||||||
|   void queuedisk(const shiftmatrix& V, const colorpair& cp, bool legend, const string* info, int i); |   void queuedisk(const shiftmatrix& V, const colorpair& cp, bool legend, const string* info, int i); | ||||||
|  |  | ||||||
|  | /* 3D models */ | ||||||
|  |  | ||||||
|  | namespace objmodels { | ||||||
|  |  | ||||||
|  |   using tf_result = pair<int, hyperpoint>; | ||||||
|  |  | ||||||
|  |   using transformer = std::function<tf_result(hyperpoint)>; | ||||||
|  |   using subdivider = std::function<int(vector<hyperpoint>&)>; | ||||||
|  |    | ||||||
|  |   inline int prec = 10; | ||||||
|  |    | ||||||
|  |   struct object { | ||||||
|  |     hpcshape sh; | ||||||
|  |     basic_textureinfo tv; | ||||||
|  |     color_t color; | ||||||
|  |     }; | ||||||
|  |    | ||||||
|  |   using model_type = vector<shared_ptr<object>>;   | ||||||
|  |    | ||||||
|  |   struct model { | ||||||
|  |    | ||||||
|  |     string path, fname; | ||||||
|  |     transformer tf; | ||||||
|  |     subdivider sd; | ||||||
|  |    | ||||||
|  |     model(string path = "", string fn = "",  | ||||||
|  |       transformer tf = [] (hyperpoint h) {  | ||||||
|  |         return tf_result{0, direct_exp(h)}; | ||||||
|  |         }, | ||||||
|  |       subdivider sd = [] (vector<hyperpoint>& hys) {  | ||||||
|  |         if(euclid) return 1; | ||||||
|  |         ld maxlen = prec * max(hypot_d(3, hys[1] - hys[0]), max(hypot_d(3, hys[2] - hys[0]), hypot_d(3, hys[2] - hys[1]))); | ||||||
|  |         return int(ceil(maxlen)); | ||||||
|  |         } | ||||||
|  |       ) : path(path), fname(fn), tf(tf), sd(sd) {} | ||||||
|  |    | ||||||
|  |     map<string, texture::texture_data> materials; | ||||||
|  |     map<string, color_t> colors; | ||||||
|  |      | ||||||
|  |     map<string, model_type> models; | ||||||
|  |  | ||||||
|  |     /* private */ | ||||||
|  |     void load_obj(model_type& objects); | ||||||
|  |      | ||||||
|  |     void render(const shiftmatrix& V); | ||||||
|  |     }; | ||||||
|  |    | ||||||
|  |   } | ||||||
|  |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Zeno Rogue
					Zeno Rogue