// Hyperbolic Rogue -- screenshots in SVG and PNG formats // Copyright (C) 2011-2018 Zeno Rogue, see 'hyper.cpp' for details namespace hr { #if ISMOBILE==1 namespace svg { bool in = false; } #endif #if ISMOBILE==0 // svg renderer namespace svg { FILE *f; bool in = false; ld cta(color_t col) { // col >>= 24; col &= 0xFF; return col / 255.0; } bool invisible(color_t col) { return (col & 0xFF) == 0; } ld gamma = .5; void fixgamma(unsigned int& color) { unsigned char *c = (unsigned char*) (&color); for(int i=1; i<4; i++) c[i] = 255 * pow(float(c[i] / 255.0), float(gamma)); } int svgsize; int divby = 10; const char* coord(int val) { static char buf[10][20]; static int id; id++; id %= 10; if(divby == 1) { sprintf(buf[id], "%d", val); return buf[id]; } else if(divby <= 10) { sprintf(buf[id], "%.1f", val*1./divby); return buf[id]; } else { sprintf(buf[id], "%.2f", val*1./divby); return buf[id]; } } char* stylestr(unsigned int fill, unsigned int stroke, ld width=1) { fixgamma(fill); fixgamma(stroke); static char buf[600]; // printf("fill = %08X stroke = %08x\n", fill, stroke); if(stroke == 0xFF00FF && false) { stroke = 0x000000FF; if(fill == 0x332a22ff) fill = 0x000000FF; else if(fill == 0x686868FF) fill = 0x000000FF; else if(fill == 0xd0d0d0FF) fill = 0x000000FF; else fill = 0xFFFFFFFF; } sprintf(buf, "style=\"stroke:#%06x;stroke-opacity:%.3" PLDF ";stroke-width:%" PLDF "px;fill:#%06x;fill-opacity:%.3" PLDF "\"", (stroke>>8) & 0xFFFFFF, cta(stroke), width/divby, (fill>>8) & 0xFFFFFF, cta(fill) ); return buf; } void circle(int x, int y, int size, color_t col, color_t fillcol) { if(!invisible(col) && !invisible(fillcol)) { if(vid.stretch == 1) fprintf(f, "\n", coord(x), coord(y), coord(size), stylestr(fillcol, col)); else fprintf(f, "\n", coord(x), coord(y), coord(size), coord(size*vid.stretch), stylestr(fillcol, col)); } } string link; void startstring() { if(link != "") fprintf(f, "", link.c_str()); } void stopstring() { if(link != "") fprintf(f, ""); } string font = "Times"; void text(int x, int y, int size, const string& str, bool frame, color_t col, int align) { double dfc = (x - vid.xcenter) * (x - vid.xcenter) + (y - vid.ycenter) * (y - vid.ycenter); dfc /= vid.radius; dfc /= vid.radius; // 0 = center, 1 = edge dfc = 1 - dfc; col = 0xFF + (col << 8); bool uselatex = font == "latex"; if(!invisible(col)) { startstring(); string str2 = ""; for(int i=0; i<(int) str.size(); i++) if(str[i] == '&') str2 += "&"; else if(str[i] == '<') str2 += "<"; else if(str[i] == '>') str2 += ">"; else if(uselatex && str[i] == '#') str2 += "\\#"; else str2 += str[i]; if(uselatex) str2 = string("\\myfont{")+coord(size)+"}{" + str2 + "}"; fprintf(f, "%s", stylestr(col, frame ? 0x0000000FF : 0, (1<", stylestr(col, outline, (hyperbolic ? vid.radius : vid.scrsize) * linewidth/256)); stopstring(); fprintf(f, "\n"); } void render(const char *fname, const function& what) { dynamicval v(vid, vid); dynamicval v2(in, true); // dynamicval v5(ringcolor, 0x808080FF); vid.usingGL = false; vid.xres = vid.yres = svgsize ? svgsize : min(1 << (get_sightrange()+7), 16384); calcparam(); dynamicval v6(inHighQual, true); darken = 0; time_t timer; timer = time(NULL); char buf[128]; strftime(buf, 128, "svgshot-%y%m%d-%H%M%S.svg", localtime(&timer)); if(!fname) fname = buf; f = fopen(fname, "wt"); fprintf(f, "\n", coord(vid.xres), coord(vid.yres)); what(); fprintf(f, "\n"); fclose(f); addMessage(XLAT("Saved the SVG shot to %1 (sightrange %2)", fname, its(get_sightrange()))); } #if CAP_COMMANDLINE int read_args() { using namespace arg; if(argis("-svgsize")) { shift(); sscanf(argcs(), "%d/%d", &svg::svgsize, &svg::divby); } else if(argis("-svgfont")) { shift(); svg::font = args(); // note: use '-svgfont latex' to produce text output as: \myfont{size}{text} // (this is helpful with Inkscape's PDF+TeX output feature; define \myfont yourself) } else if(argis("-svggamma")) { shift(); svg::gamma = argf(); } else if(argis("-svgshot")) { PHASE(3); shift(); start_game(); printf("saving SVG screenshot to %s\n", argcs()); svg::render(argcs()); } else return 1; return 0; } auto ah = addHook(hooks_args, 0, read_args); #endif } #endif #if CAP_SDL int pngres = 2000; int pngformat = 0; #if CAP_PNG void IMAGESAVE(SDL_Surface *s, const char *fname) { SDL_Surface *s2 = SDL_PNGFormatAlpha(s); SDL_SavePNG(s2, fname); SDL_FreeSurface(s2); } #endif hookset *hooks_hqshot; color_t fillmodel; void saveHighQualityShot(const char *fname, const char *caption, int fade) { resetbuffer rb; // int maxrange = getDistLimit() * 3/2; // dynamicval v3(sightrange, (cheater && sightrange < maxrange) ? maxrange : sightrange); if(cheater) doOvergenerate(); time_t timer; timer = time(NULL); dynamicval v(vid, vid); dynamicval v2(inHighQual, true); dynamicval v6(auraNOGL, fname ? true : false); vid.xres = vid.yres = pngres; if(pngformat == 1) vid.xres = vid.yres * 4/3; if(pngformat == 2) vid.xres = vid.yres * 16/9; if(pngformat == 3) { vid.xres = vid.yres * 22/16; while(vid.xres & 15) vid.xres++; } // if(vid.pmodel == 0) vid.scale = 0.99; calcparam(); renderbuffer glbuf(vid.xres, vid.yres, vid.usingGL); glbuf.enable(); stereo::set_viewport(0); // printf("format = %d, %d x %d\n", pngformat, vid.xres, vid.yres); darken = 0; int numi = (fname?1:2); for(int i=0; ic7, 7 - getDistLimit() - genrange_bonus, NULL); } playermoved = false; } struct animatable_parameter { ld& value; dialog::scaler sc; ld values[2]; bool need_reset; animatable_parameter(ld& val, bool r = false) : value(val), sc(dialog::identity), need_reset(r) {} animatable_parameter(ld& val, dialog::scaler s) : value(val), sc(s), need_reset(false) {} }; vector animatable_parameters = { animatable_parameter(vid.scale, dialog::asinhic), animatable_parameter(vid.alpha, dialog::asinhic), animatable_parameter(vid.linewidth), animatable_parameter(vid.xposition), animatable_parameter(vid.yposition), animatable_parameter(vid.ballangle), animatable_parameter(vid.yshift), animatable_parameter(polygonal::STAR), animatable_parameter(stereo::ipd), animatable_parameter(stereo::lr_eyewidth), animatable_parameter(stereo::anaglyph_eyewidth), animatable_parameter(stereo::fov), animatable_parameter(vid.euclid_to_sphere), animatable_parameter(vid.twopoint_param), animatable_parameter(vid.stretch), animatable_parameter(vid.binary_width, true), animatable_parameter(geom3::depth, false), animatable_parameter(geom3::camera, true), animatable_parameter(geom3::wall_height, true), animatable_parameter(vid.ballproj), animatable_parameter(surface::dini_b), animatable_parameter(surface::hyper_b), animatable_parameter(conformal::halfplane_scale), animatable_parameter(conformal::model_transition), animatable_parameter(conformal::top_z, dialog::logarithmic), }; ld anim_param = 0; int paramstate = 0; bool needs_highqual; bool joukowsky_anim; void reflect_view() { if(centerover.at) { transmatrix T = Id; cell *mbase = centerover.at; cell *c = centerover.at; if(shmup::reflect(c, mbase, T)) View = inverse(T) * View; } } void apply() { int t = ticks - lastticks; lastticks = ticks; switch(ma) { case maTranslation: if(conformal::on) { conformal::phase = (isize(conformal::v) - 1) * ticks * 1. / period; conformal::movetophase(); } else if(centerover.at) { reflect_view(); if((hyperbolic && !quotient && (centerover.at->land != cwt.at->land || memory_saving_mode) && among(centerover.at->land, laHaunted, laIvoryTower, laDungeon, laEndorian) && centerover.at->landparam >= 10 ) ) { if(memory_saving_mode) { activateSafety(laIce); return; } else { fullcenter(); View = spin(rand() % 1000) * View; } } View = spin(movement_angle * M_PI / 180) * ypush(shift_angle * M_PI / 180) * xpush(cycle_length * t / period) * ypush(-shift_angle * M_PI / 180) * spin(-movement_angle * M_PI / 180) * View; moved(); } break; case maRotation: View = spin(2 * M_PI * t / period) * View; break; case maParabolic: reflect_view(); View = spin(movement_angle * M_PI / 180) * ypush(shift_angle * M_PI / 180) * binary::parabolic(parabolic_length * t / period) * ypush(-shift_angle * M_PI / 180) * spin(-movement_angle * M_PI / 180) * View; moved(); break; case maCircle: { if(masterless) centerover = rotation_center_c; else viewctr = rotation_center_h; ld alpha = circle_spins * 2 * M_PI * ticks / period; View = spin(-cos_auto(circle_radius)*alpha) * xpush(circle_radius) * spin(alpha) * rotation_center_View; moved(); break; } default: break; } if(env_ocean) { turncount += env_ocean * ticks * tidalsize / period; calcTidalPhase(); for(auto& p: gmatrix) if(p.first->land == laOcean) checkTide(p.first); turncount -= ticks * tidalsize / period; } if(env_volcano) { bak_turncount = turncount; turncount += env_volcano * ticks * 64 / period; for(auto& p: gmatrix) if(p.first->land == laVolcano) checkTide(p.first); } if(rug::rugged) { if(rug_rotation1) { rug::apply_rotation(rotmatrix(rug_angle * M_PI / 180, 1, 2)); rug::apply_rotation(rotmatrix(rug_rotation1 * 2 * M_PI * t / period, 0, 2)); rug::apply_rotation(rotmatrix(-rug_angle * M_PI / 180, 1, 2)); } if(rug_rotation2) { rug::apply_rotation(rug::currentrot * rotmatrix(rug_rotation2 * 2 * M_PI * t / period, 0, 1) * inverse(rug::currentrot)); } } vid.skiprope += skiprope_rotation * t * 2 * M_PI / period; if(ballangle_rotation) { if(conformal::model_has_orientation()) conformal::model_orientation += ballangle_rotation * 360 * t / period; else vid.ballangle += ballangle_rotation * 360 * t / period; } if(paramstate == 2 && anim_param) { ld phase = (1 + sin(anim_param * 2 * M_PI * ticks / period)) / 2; for(auto& ap: animatable_parameters) if(ap.values[0] != ap.values[1]) { ap.value = ap.sc.inverse(ap.sc.direct(ap.values[0]) * phase + ap.sc.direct(ap.values[1]) * (1-phase)); if(ap.need_reset) { if(!inHighQual) needs_highqual = true; else need_reset_geometry = true; } if(&ap.value == &surface::hyper_b) run_shape(surface::dsHyperlike); if(&ap.value == &surface::dini_b) { if(!inHighQual) needs_highqual = true; else surface::run_shape(surface::dsDini); } } if(need_reset_geometry) resetGeometry(), need_reset_geometry = false; calcparam(); } if(joukowsky_anim) { ld t = ticks / period; t = t - floor(t); if(pmodel == mdBand) { conformal::model_transition = t * 4 - 1; } else { conformal::model_transition = t / 1.1; vid.scale = (1 - conformal::model_transition) / 2.; } calcparam(); } } void rollback() { if(env_volcano) { turncount = bak_turncount; } } #if CAP_FILES string animfile = "animation-%04d.png"; bool record_animation() { for(int i=0; i 1) { dialog::addSelItem(XLAT("game animation period"), fts(animation_period)+ " ms", 'G'); dialog::add_action([] () { dialog::editNumber(animation_period, 0, 10000, 1000, 1000, XLAT("game animation period"), XLAT("Least common multiple of the animation periods of all the game objects on screen, such as rotating items.") ); dialog::reaction = [] () { animation_factor = 2 * M_PI * animation_lcm / animation_period; }; dialog::extra_options = [] () { dialog::addItem("default", 'D'); dialog::add_action([] () { animation_factor = 1; popScreen(); }); }; }); } dialog::addBoolItem(XLAT("no movement animation"), ma == maNone, '0'); dialog::add_action([] () { ma = maNone; }); dialog::addBoolItem(XLAT("translation"), ma == maTranslation, '1'); dialog::add_action([] () { ma = maTranslation; }); dialog::addBoolItem(XLAT("rotation"), ma == maRotation, '2'); dialog::add_action([] () { ma = maRotation; }); if(hyperbolic) { dialog::addBoolItem(XLAT("parabolic"), ma == maParabolic, '3'); dialog::add_action([] () { ma = maParabolic; }); } if(among(pmodel, mdJoukowsky, mdJoukowskyInverted)) { dialog::addBoolItem(XLAT("joukowsky_anim"), joukowsky_anim, 'j'); dialog::add_action([] () { joukowsky_anim = !joukowsky_anim; }); } if(among(pmodel, mdJoukowsky, mdJoukowskyInverted)) { animator(XLAT("Möbius transformations"), skiprope_rotation, 'S'); } dialog::addBoolItem(XLAT("circle"), ma == maCircle, '4'); dialog::add_action([] () { ma = maCircle; rotation_center_h = viewctr; rotation_center_c = centerover; rotation_center_View = View; }); switch(ma) { case maCircle: { animator(XLAT("circle spins"), circle_spins, 's'); dialog::addSelItem(XLAT("circle radius"), fts(circle_radius), 'c'); dialog::add_action([] () { dialog::editNumber(circle_radius, 0, 10, 0.1, acosh(1.), XLAT("circle radius"), ""); dialog::extra_options = [] () { if(hyperbolic) { // area = 2pi (cosh(r)-1) dialog::addSelItem(XLAT("double spin"), fts(acosh(2.)), 'a'); dialog::add_action([] () { circle_radius = acosh(2.); }); dialog::addSelItem(XLAT("triple spin"), fts(acosh(3.)), 'b'); dialog::add_action([] () { circle_radius = acosh(3.); }); } if(sphere) { dialog::addSelItem(XLAT("double spin"), fts(acos(1/2.)), 'a'); dialog::add_action([] () { circle_radius = acos(1/2.); }); dialog::addSelItem(XLAT("triple spin"), fts(acos(1/3.)), 'b'); dialog::add_action([] () { circle_radius = acos(1/3.); }); } }; }); dialog::addColorItem(XLAT("draw the circle"), circle_display_color, 'd'); dialog::add_action([] () { dialog::openColorDialog(circle_display_color, NULL); }); dialog::addBreak(100); } case maTranslation: case maParabolic: { if(ma == maTranslation && conformal::on) dialog::addBreak(300); else if(ma == maTranslation) { dialog::addSelItem(XLAT("cycle length"), fts(cycle_length), 'c'); dialog::add_action([] () { dialog::editNumber(cycle_length, 0, 10, 0.1, 2*M_PI, "shift", ""); dialog::extra_options = [] () { dialog::addSelItem(XLAT("full circle"), fts(2 * M_PI), 'a'); dialog::add_action([] () { cycle_length = 2 * M_PI; }); dialog::addSelItem(XLAT("Zebra period"), fts(2.898149445355172), 'b'); dialog::add_action([] () { cycle_length = 2.898149445355172; }); dialog::addSelItem(XLAT("Bolza period"), fts(2 * 1.528571), 'c'); dialog::add_action([] () { cycle_length = 2 * 1.528571; }); }; }); } else { dialog::addSelItem(XLAT("cells to go"), fts(parabolic_length), 'c'); dialog::add_action([] () { dialog::editNumber(parabolic_length, 0, 10, 1, 1, "cells to go", ""); }); } dialog::addSelItem(XLAT("shift"), fts(shift_angle) + "°", 's'); dialog::add_action([] () { dialog::editNumber(shift_angle, 0, 90, 15, 0, XLAT("shift"), ""); }); dialog::addSelItem(XLAT("movement angle"), fts(movement_angle) + "°", 'm'); dialog::add_action([] () { dialog::editNumber(movement_angle, 0, 360, 15, 0, XLAT("movement angle"), ""); }); break; } default: { dialog::addBreak(300); } } animator(XLATN("Ocean"), env_ocean, 'o'); animator(XLATN("Volcanic Wasteland"), env_volcano, 'v'); #if CAP_RUG if(rug::rugged) { animator(XLAT("screen-relative rotation"), rug_rotation1, 'r'); if(rug_rotation1) { dialog::addSelItem(XLAT("angle"), fts(rug_angle) + "°", 'a'); dialog::add_action([] () { dialog::editNumber(rug_angle, 0, 360, 15, 0, "Rug angle", ""); }); } else dialog::addBreak(100); animator(XLAT("model-relative rotation"), rug_rotation2, 'r'); animator(XLAT("automatic move speed"), rug::ruggo, 'M'); dialog::add_action([] () { dialog::editNumber(rug::ruggo, 0, 10, 1, 1, XLAT("automatic move speed"), XLAT("Move automatically without pressing any keys.")); if(among(rug::gwhere, gSphere, gElliptic)) dialog::extra_options = [] () { dialog::addItem(XLAT("synchronize"), 'S'); dialog::add_action([] () { rug::ruggo = 2 * M_PI * 1000 / period; popScreen(); }); }; }); } #endif if(conformal::model_has_orientation()) animator(XLAT("model rotation"), ballangle_rotation, 'r'); else if(among(pmodel, mdHyperboloid, mdHemisphere, mdBall)) animator(XLAT("3D rotation"), ballangle_rotation, 'r'); animator(XLAT("animate parameter change"), anim_param, 'P'); dialog::addSelItem(XLAT("choose parameters to animate"), its(paramstate), 'C'); dialog::add_action(next_paramstate); dialog::addBoolItem(XLAT("history mode"), (conformal::on || conformal::includeHistory), 'h'); dialog::add_action([] () { pushScreen(conformal::history_menu); }); #if CAP_FILES if(needs_highqual) dialog::addInfo(XLAT("some parameters will only change in recorded animation")); else dialog::addBreak(100); dialog::addSelItem(XLAT("frames to record"), its(noframes), 'n'); dialog::add_action([] () { dialog::editNumber(noframes, 0, 300, 30, 5, XLAT("frames to record"), ""); }); dialog::addSelItem(XLAT("record to a file"), animfile, 'R'); dialog::add_action([] () { dialog::openFileDialog(animfile, XLAT("record to a file"), ".png", record_animation); }); #endif dialog::addBack(); dialog::display(); } int readArgs() { using namespace arg; if(0) ; else if(argis("-animmenu")) { PHASE(3); showstartmenu = false; pushScreen(show); } else if(argis("-animparam")) { PHASE(2); next_paramstate(); if(paramstate >= 2) anim_param = 1; } else if(argis("-animperiod")) { PHASE(2); shift(); period = argf(); } else if(argis("-animrecord")) { PHASE(3); shift(); noframes = argi(); shift(); animfile = args(); record_animation(); } else if(argis("-animcircle")) { PHASE(3); start_game(); ma = maCircle; rotation_center_h = viewctr; rotation_center_c = cwt.at; rotation_center_View = View; shift(); circle_spins = argf(); shift(); circle_radius = argf(); shift(); circle_display_color = arghex(); } else if(argis("-animmove")) { ma = maTranslation; shift(); cycle_length = argf(); shift(); shift_angle = argf(); shift(); movement_angle = argf(); } else if(argis("-animpar")) { ma = maParabolic; shift(); parabolic_length = argf(); shift(); shift_angle = argf(); shift(); movement_angle = argf(); } else if(argis("-animrot")) { ma = maRotation; } else if(argis("-animrug")) { shift(); rug_rotation1 = argf(); shift(); rug_angle = argf(); shift(); rug_rotation2 = argf(); } else if(argis("-animenv")) { shift(); env_ocean = argf(); shift(); env_volcano = argf(); } else if(argis("-animball")) { shift(); ballangle_rotation = argf(); } else if(argis("-animj")) { shift(); joukowsky_anim = true; } else if(argis("-animrec")) { PHASE(3); shift(); noframes = argi(); shift(); animfile = args(); record_animation(); } else return 1; return 0; } auto animhook = addHook(hooks_args, 100, readArgs) + addHook(hooks_frame, 100, display_animation) + 0; bool any_animation() { if(conformal::on) return true; if(ma) return true; if(ballangle_rotation || rug_rotation1 || rug_rotation2) return true; if(paramstate == 2) return true; return false; } bool any_on() { return any_animation() || conformal::includeHistory; } bool center_music() { return among(ma, maParabolic, maTranslation); } } #endif }