/* Missing. #if ISMOBILE==0 dialog::addItemHelp(16, XLAT("use Shift to decrease and Ctrl to fine tune ")); dialog::addItemHelp(17, XLAT("(e.g. Shift+Ctrl+Z)")); #endif if(xuni == 'i') { } */ const char* COLORBAR = "###"; namespace dialog { namespace zoom { int zoomf = 1, shiftx, shifty; bool zoomoff = false; }; #if CAP_MENUSCALING #if CAP_SDL void handleZooming(SDL_Event &ev) { using namespace zoom; if(zoomoff || (cmode != emOverview && cmode != emTactic)) { zoomf = 1; shiftx = shifty = 0; zoomoff = false; return; } if(ev.type == SDL_MOUSEBUTTONDOWN) { zoomf = 3; } if(zoomf == 3) { shiftx = -2*mousex; shifty = -2*mousey; } if(ev.type == SDL_MOUSEBUTTONUP && zoomf > 1) { zoomoff = true; } } #endif #else inline void handleZooming(SDL_Event &ev) {} #endif bool displayzoom(int x, int y, int b, int size, const string &s, int color, int align) { using namespace zoom; return displayfr(x * zoomf + shiftx, y * zoomf + shifty, b, size * zoomf, s, color, align); } vector<item> items; item& lastItem() { return items[items.size() - 1]; } void init() { items.clear(); } string keyname(int k) { if(k == 0) return ""; if(k == SDLK_ESCAPE) return "Esc"; if(k == SDLK_F5) return "F5"; if(k == SDLK_F10) return "F10"; if(k == SDLK_HOME) return "Home"; if(k == SDLK_BACKSPACE) return "Backspace"; if(k == SDLK_RETURN) return "Enter"; if(k == 32) return "space"; if(k >= 1 && k <= 26) { string s = "Ctrl+"; s += (k+64); return s; } if(k < 128) { string s; s += k; return s; } return "?"; } void addSlider(double d1, double d2, double d3, int key) { item it; it.type = diSlider; it.color = 0xC0C0C0; it.scale = 100; it.key = key; it.param = (d2-d1) / (d3-d1); items.push_back(it); } void addSelItem(string body, string value, int key) { item it; it.type = diItem; it.body = body; it.value = value; it.keycaption = keyname(key); it.key = key; it.color = 0xC0C0C0; it.colork = 0x808080; it.colorv = 0x80A040; it.colorc = 0xFFD500; it.colors = 0xFF8000; if(value == ONOFF(true)) it.colorv = 0x40FF40; if(value == ONOFF(false)) it.colorv = 0xC04040; it.scale = 100; items.push_back(it); } void addBoolItem(string body, bool value, int key) { addSelItem(body, ONOFF(value), key); } void addColorItem(string body, int value, int key) { item it; it.type = diItem; it.body = body; it.value = COLORBAR; it.keycaption = keyname(key); it.key = key; it.color = it.colorv = value >> 8; it.colors = it.color ^ 0x404040; it.colorc = it.color ^ 0x808080; it.colork = 0x808080; it.scale = 100; items.push_back(it); } void addHelp(string body) { item it; it.type = diHelp; it.body = body; it.scale = 100; if(size(body) >= 500) it.scale = 70; items.push_back(it); } void addInfo(string body, int color) { item it; it.type = diInfo; it.body = body; it.color = color; it.scale = 100; items.push_back(it); } void addItem(string body, int key) { item it; it.type = diItem; it.body = body; it.keycaption = keyname(key); it.key = key; it.color = 0xC0C0C0; it.colork = 0x808080; it.colors = 0xFFD500; it.colorc = 0xFF8000; it.scale = 100; items.push_back(it); } void addBigItem(string body, int key) { item it; it.type = diBigItem; it.body = body; it.keycaption = keyname(key); it.key = key; it.color = 0xC06000; it.colors = 0xFFD500; it.colorc = 0xFF8000; it.scale = 150; items.push_back(it); } int addBreak(int val) { item it; it.type = diBreak; it.scale = val; items.push_back(it); return items.size()-1; } void addTitle(string body, int color, int scale) { item it; it.type = diTitle; it.body = body; it.color = color; it.scale = scale; items.push_back(it); } void init(string title, int color, int scale, int brk) { init(); addTitle(title, color, scale); addBreak(brk); } int dcenter, dwidth; int dialogflags; int displayLong(string str, int siz, int y, bool measure) { int last = 0; int lastspace = 0; int xs, xo; if(sidescreen) xs = dwidth - vid.fsize*2, xo = vid.yres + vid.fsize; else xs = vid.xres * 618/1000, xo = vid.xres * 186/1000; for(int i=0; i<=size(str); i++) { int ls = 0; int prev = last; if(str[i] == ' ') lastspace = i; if(textwidth(siz, str.substr(last, i-last)) > xs) { if(lastspace == last) ls = i-1, last = i-1; else ls = lastspace, last = ls+1; } if(str[i] == 10 || i == size(str)) ls = i, last = i+1; if(ls) { if(!measure) displayfr(xo, y, 2, siz, str.substr(prev, ls-prev), 0xC0C0C0, 0); if(ls == prev) y += siz/2; else y += siz; lastspace = last; } } y += siz/2; return y; } int tothei, dialogwidth, dfsize, dfspace, leftwidth, rightwidth, innerwidth, itemx, keyx, valuex; string highlight_text; void measure() { tothei = 0; dialogwidth = 0; innerwidth = 0; int N = items.size(); for(int i=0; i<N; i++) { if(items[i].type == diHelp) tothei += displayLong(items[i].body, dfsize * items[i].scale / 100, 0, true); else { tothei += dfspace * items[i].scale / 100; if(items[i].type == diItem) innerwidth = max(innerwidth, textwidth(dfsize * items[i].scale / 100, items[i].body)); if(items[i].type == diTitle || items[i].type == diInfo || items[i].type == diBigItem) dialogwidth = max(dialogwidth, textwidth(dfsize * items[i].scale / 100, items[i].body) * 10/9); } } leftwidth = ISMOBILE ? 0 : textwidth(dfsize, "MMMMM") + dfsize/2; rightwidth = textwidth(dfsize, "MMMMMMMM") + dfsize/2; int fwidth = innerwidth + leftwidth + rightwidth; dialogwidth = max(dialogwidth, fwidth); itemx = dcenter - fwidth / 2 + leftwidth; keyx = dcenter - fwidth / 2 + leftwidth - dfsize/2; valuex = dcenter - fwidth / 2 + leftwidth + innerwidth + dfsize/2; } void display() { int N = items.size(); dfsize = vid.fsize; #if ISMOBILE || ISPANDORA dfsize *= 3; #endif dfspace = dfsize * 5/4; dcenter = vid.xres/2; dwidth = vid.xres; if(sidescreen) { dwidth = vid.xres - vid.yres; dcenter = vid.xres - dwidth / 2; } measure(); while(tothei > vid.yres - 5 * vid.fsize) { int adfsize = int(dfsize * sqrt((vid.yres - 5. * vid.fsize) / tothei)); if(adfsize < dfsize-1) dfsize = adfsize + 1; else dfsize--; dfspace = dfsize * 5/4; measure(); } while(dialogwidth > dwidth) { int adfsize = int(dfsize * sqrt(vid.xres * 1. / dialogwidth)); if(adfsize < dfsize-1) dfsize = adfsize + 1; else dfsize--; // keep dfspace measure(); } tothei = (vid.yres - tothei) / 2; for(int i=0; i<N; i++) { item& I = items[i]; if(I.type == diHelp) { tothei = displayLong(items[i].body, dfsize * items[i].scale / 100, tothei, false); continue; } int top = tothei; tothei += dfspace * I.scale / 100; int mid = (top + tothei) / 2; I.position = mid; if(I.type == diTitle || I.type == diInfo) { displayfr(dcenter, mid, 2, dfsize * I.scale/100, I.body, I.color, 8); } else if(I.type == diItem || I.type == diBigItem) { bool xthis = (mousey >= top && mousey < tothei); if(cmode && sm::DIALOG_STRICT_X) xthis = xthis && (mousex >= dcenter - dialogwidth/2 && mousex <= dcenter + dialogwidth/2); #if ISMOBILE if(xthis && mousepressed) I.color = I.colorc; #else if(xthis && mousemoved) { highlight_text = I.body; mousemoved = false; } if(highlight_text == I.body) { I.color = (xthis&&mousepressed&&actonrelease) ? I.colorc : I.colors; } #endif if(I.type == diBigItem) { displayfr(dcenter, mid, 2, dfsize * I.scale/100, I.body, I.color, 8); } else { if(!mousing) displayfr(keyx, mid, 2, dfsize * I.scale/100, I.keycaption, I.colork, 16); displayfr(itemx, mid, 2, dfsize * I.scale/100, I.body, I.color, 0); displayfr(valuex, mid, 2, dfsize * I.scale/100, I.value, I.colorv, 0); } if(xthis) getcstat = I.key; } else if(I.type == diSlider) { bool xthis = (mousey >= top && mousey < tothei); int sl, sr; if(sidescreen) sl = vid.yres + vid.fsize*2, sr = vid.xres - vid.fsize*2; else sl = vid.xres/4, sr = vid.xres*3/4; int sw = sr-sl; displayfr(sl, mid, 2, dfsize * I.scale/100, "(", I.color, 16); displayfr(sl + double(sw * I.param), mid, 2, dfsize * I.scale/100, "#", I.color, 8); displayfr(sr, mid, 2, dfsize * I.scale/100, ")", I.color, 0); if(xthis) getcstat = I.key, inslider = true; } } } bool isitem(item& it) { return it.type == diItem || it.type == diBigItem; } void handleNavigation(int &sym, int &uni) { #if !ISMOBILE if(uni == '\n' || uni == '\r' || sym == SDLK_KP5) for(int i=0; i<size(items); i++) if(isitem(items[i])) if(items[i].body == highlight_text) { uni = sym = items[i].key; return; } if(sym == SDLK_PAGEDOWN || sym == SDLK_KP3) { for(int i=0; i<size(items); i++) if(isitem(items[i])) highlight_text = items[i].body; } if(sym == SDLK_PAGEUP || sym == SDLK_KP9) { for(int i=0; i<size(items); i++) if(isitem(items[i])) { highlight_text = items[i].body; break; } } if(sym == SDLK_UP || sym == SDLK_KP8) { string last = ""; for(int i=0; i<size(items); i++) if(isitem(items[i])) last = items[i].body; uni = sym = 0; for(int i=0; i<size(items); i++) if(isitem(items[i])) { if(items[i].body == highlight_text) { highlight_text = last; return; } else last = items[i].body; } highlight_text = last; } if(sym == SDLK_DOWN || sym == SDLK_KP2) { int state = 0; for(int i=0; i<size(items); i++) if(isitem(items[i])) { if(state) { highlight_text = items[i].body; return; } else if(items[i].body == highlight_text) state = 1; } for(int i=0; i<size(items); i++) if(isitem(items[i])) highlight_text = items[i].body; uni = sym = 0; } #endif } unsigned int colorhistory[10] = { 0x202020FF, 0x800000FF, 0x008000FF, 0x000080FF, 0x404040FF, 0xC0C0C0FF, 0x804000FF, 0xC0C000FF, 0x408040FF, 0xFFD500FF }, lch; unsigned int *palette; int colorp = 0; unsigned int *colorPointer; bool handleKeyColor(int sym, int uni) { unsigned& color = *colorPointer; if(uni >= 'A' && uni <= 'D') { int x = (mousex - (dcenter-dwidth/4)) * 510 / dwidth; if(x < 0) x = 0; if(x > 255) x = 255; unsigned char* pts = (unsigned char*) &color; pts[uni - 'A'] = x; } else if(uni == ' ') { bool inHistory = false; for(int i=0; i<10; i++) if(colorhistory[i] == (unsigned) color) inHistory = true; if(!inHistory) { colorhistory[lch] = color; lch++; lch %= 10; } popScreen(); } else if(uni >= '0' && uni <= '9') { color = colorhistory[uni - '0']; } else if(palette && uni >= 'a' && uni < 'a'+(int) palette[0]) { color = palette[1 + uni - 'a']; } else if(sym == SDLK_DOWN || sym == SDLK_KP2) { colorp = (colorp-1) & 3; } else if(sym == SDLK_UP || sym == SDLK_KP8) { colorp = (colorp+1) & 3; } else if(sym == SDLK_LEFT || sym == SDLK_KP4) { unsigned char* pts = (unsigned char*) &color; pts[colorp] -= abs(shiftmul) < .6 ? 1 : 17; } else if(sym == SDLK_RIGHT || sym == SDLK_KP6) { unsigned char* pts = (unsigned char*) &color; pts[colorp] += abs(shiftmul) < .6 ? 1 : 17; } else if(doexiton(sym, uni)) { if(reaction) reaction(); popScreen(); } return false; } void drawColorDialog() { cmode = sm::NUMBER | dialogflags; if(cmode & sm::SIDE) gamescreen(0); dcenter = vid.xres/2; dwidth = vid.xres; if(sidescreen) { dwidth = vid.xres - vid.yres; dcenter = vid.xres - dwidth / 2; } int color = *colorPointer; int ash = 8; for(int j=0; j<10; j++) { int x = dcenter + vid.fsize * 2 * (j-5); int y = vid.yres / 2- 5 * vid.fsize; string s0 = ""; s0 += ('0'+j); vid.fsize *= 2; displayColorButton(x, y, s0, '0'+j, 0, 0, colorhistory[j] >> ash); vid.fsize /= 2; } if(palette) { int q = palette[0]; for(int i=0; i<q; i++) { int x = dcenter + vid.fsize * (2 * i-q); int y = vid.yres / 2- 7 * vid.fsize; string s0 = ""; s0 += ('a'+i); vid.fsize *= 2; displayColorButton(x, y, s0, 'a'+i, 0, 0, palette[i+1] >> ash); vid.fsize /= 2; } } for(int i=0; i<4; i++) { int y = vid.yres / 2 + (2-i) * vid.fsize * 2; int col = ((i==colorp) && !mousing) ? 0xFFD500 : forecolor; displayColorButton(dcenter - dwidth/4, y, "(", 0, 16, 0, col); string rgt = ") "; rgt += "ABGR" [i]; displayColorButton(dcenter + dwidth/4, y, rgt, 0, 0, 0, col); displayColorButton(dcenter - dwidth/4 + dwidth * ((color >> (8*i)) & 0xFF) / 510, y, "#", 0, 8, 0, col); if(mousey >= y - vid.fsize && mousey < y + vid.fsize) getcstat = 'A' + i, inslider = true; } displayColorButton(dcenter, vid.yres/2+vid.fsize * 6, XLAT("select this color") + " : " + itsh(color), ' ', 8, 0, color >> ash); keyhandler = handleKeyColor; } void openColorDialog(unsigned int& col, unsigned int *pal) { colorPointer = &col; palette = pal; dialogflags = 0; pushScreen(drawColorDialog); reaction = reaction_t(); } struct numberEditor { ld *editwhat; string s; ld vmin, vmax, step, dft; string title, help; ld (*scale) (ld); ld (*inverse_scale) (ld); int *intval; ld intbuf; bool positive; } ne; bool editingDetail() { return ne.editwhat == &geom3::highdetail || ne.editwhat == &geom3::middetail; } ld identity(ld x) { return x; } void scaleSinh() { ne.scale = ASINH; ne.inverse_scale = sinh; } void scaleLog() { ne.scale = log; ne.inverse_scale = exp; ne.positive = true; } int ldtoint(ld x) { if(x > 0) return int(x+.5); else return int(x-.5); } string disp(ld x) { if(ne.intval) return its(ldtoint(x)); else if(ne.vmax-ne.vmin < 1) return fts4(x); else return fts(x); } reaction_t reaction; void affect(char kind) { if(ne.intval) { if(kind == 's') sscanf(ne.s.c_str(), "%d", ne.intval), *ne.editwhat = *ne.intval; if(kind == 'v') *ne.intval = ldtoint(*ne.editwhat), ne.s = its(*ne.intval); } else { if(kind == 's') { ld x; sscanf(ne.s.c_str(), LDF, &x); if(ne.positive && x <= 0) return; *ne.editwhat = x; } if(kind == 'v') ne.s = fts(*ne.editwhat); } if(reaction) reaction(); #if CAP_AUDIO if(ne.intval == &musicvolume) { if(musicvolume < 0) *ne.editwhat = musicvolume = 0, affect('v'); else if(musicvolume > MIX_MAX_VOLUME) *ne.editwhat = musicvolume = MIX_MAX_VOLUME, affect('v'); #if CAP_SDLAUDIO else Mix_VolumeMusic(musicvolume); #endif #if ISANDROID settingsChanged = true; #endif } if(ne.intval == &effvolume) { if(effvolume < 0) *ne.editwhat = effvolume = 0, affect('v'); else if(effvolume > MIX_MAX_VOLUME) *ne.editwhat = effvolume = MIX_MAX_VOLUME, affect('v'); #if ISANDROID settingsChanged = true; #endif } #endif if(ne.intval == &vid.framelimit && vid.framelimit < 5) *ne.editwhat = vid.framelimit = 5, affect('v'); #if ISMOBILE==1 if(ne.intval == &fontscale && fontscale < 50) *ne.editwhat = fontscale = 50, affect('v'); #endif // if(ne.editwhat == &whatever) resetGeometry(); if(ne.intval == &sightrange && sightrange < 4) *ne.editwhat = sightrange = 4, affect('v'); int msr = allowIncreasedSight() ? 15 : 7; if(ne.intval == &sightrange && sightrange > msr) *ne.editwhat = sightrange = msr, affect('v'); if(ne.intval == &conformal::bandhalf && conformal::bandhalf < 5) *ne.editwhat = *ne.intval = 5, affect('v'); if(ne.intval == &conformal::bandsegment && conformal::bandsegment < 500) *ne.editwhat = *ne.intval = 500, affect('v'); if(ne.intval == &polygonal::coefid && polygonal::coefid < 0) *ne.editwhat = *ne.intval = 0, affect('v'); if(ne.intval == &polygonal::coefid && polygonal::coefid >= polygonal::MSI) *ne.editwhat = *ne.intval = polygonal::MSI-1, affect('v'); if(ne.intval == &polygonal::deg && polygonal::deg < 0) *ne.editwhat = *ne.intval = polygonal::MSI-1, affect('v'); if(ne.intval == &polygonal::deg && polygonal::deg >= polygonal::MSI) *ne.editwhat = *ne.intval = polygonal::MSI-1, affect('v'); if(ne.intval == &polygonal::SI) polygonal::solve(); if(ne.editwhat == &polygonal::STAR) polygonal::solve(); polygonal::solve(); if(ne.editwhat == &geom3::highdetail && geom3::highdetail > geom3::middetail) geom3::middetail = geom3::highdetail; if(ne.editwhat == &geom3::middetail && geom3::highdetail > geom3::middetail) geom3::highdetail = geom3::middetail; if(cmode & sm::A3) { buildpolys(); #if CAP_GL resetGL(); #endif } } void drawNumberDialog() { cmode = sm::NUMBER | dialogflags; gamescreen(0); init(ne.title); addInfo(ne.s); addSlider(ne.scale(ne.vmin), ne.scale(*ne.editwhat), ne.scale(ne.vmax), 500); addBreak(100); #if ISMOBILE==0 addHelp("You can scroll with arrow keys -- Ctrl to fine-tune"); addBreak(100); #endif addItem("return", ' '); addSelItem("default value", disp(ne.dft), SDLK_HOME); addBreak(100); if(cmode & sm::A3) ne.help = explain3D(ne.editwhat); if(ne.help != "") { addHelp(ne.help); // bool scal = !ISMOBILE && !ISPANDORA && size(ne.help) > 160; // if(scal) lastItem().scale = 30; } if(ne.editwhat == &vid.alpha) { addBreak(100); addSelItem(sphere ? "stereographic" : "Poincaré model", "1", 'p'); addSelItem(sphere ? "gnomonic" : "Klein model", "0", 'k'); addItem(sphere ? "towards orthographic" : "towards Gans model", 'o'); } if(ne.editwhat == &ne.intbuf && ne.intval == &sightrange && cheater) addBoolItem("overgenerate", overgenerate, 'o'); if(ne.editwhat == &vid.linewidth) addBoolItem("finer lines at the boundary", vid.antialias & AA_LINEWIDTH, 'o'); display(); keyhandler = [] (int sym, int uni) { handleNavigation(sym, uni); if((uni >= '0' && uni <= '9') || (uni == '.' && !ne.intval) || (uni == '-' && !ne.positive)) { ne.s += uni; affect('s'); } else if(uni == '\b' || uni == '\t') { ne.s = ne.s. substr(0, size(ne.s)-1); sscanf(ne.s.c_str(), LDF, ne.editwhat); affect('s'); } #if !ISMOBILE else if(sym == SDLK_RIGHT || sym == SDLK_KP6) { if(ne.intval && abs(shiftmul) < .6) (*ne.editwhat)++; else *ne.editwhat = ne.inverse_scale(ne.scale(*ne.editwhat) + shiftmul * ne.step); affect('v'); } else if(sym == SDLK_LEFT || sym == SDLK_KP4) { if(ne.intval && abs(shiftmul) < .6) (*ne.editwhat)--; else *ne.editwhat = ne.inverse_scale(ne.scale(*ne.editwhat) - shiftmul * ne.step); affect('v'); } #endif else if(sym == SDLK_HOME) { *ne.editwhat = ne.dft; affect('v'); } else if(uni == 500) { int sl, sr; if(sidescreen) sl = vid.yres + vid.fsize*2, sr = vid.xres - vid.fsize*2; else sl = vid.xres/4, sr = vid.xres*3/4; ld d = (mousex - sl + .0) / (sr-sl); *ne.editwhat = ne.inverse_scale(d * (ne.scale(ne.vmax) - ne.scale(ne.vmin)) + ne.scale(ne.vmin)); affect('v'); } else if(uni == 'o' && ne.editwhat == &ne.intbuf && ne.intval == &sightrange && cheater) overgenerate = !overgenerate; else if(uni == 'o' && ne.editwhat == &vid.linewidth) vid.antialias ^= AA_LINEWIDTH; else if(uni == 'p' && ne.editwhat == &vid.alpha) { *ne.editwhat = 1; vid.scale = 1; ne.s = "1"; } else if(uni == 'k' && ne.editwhat == &vid.alpha) { *ne.editwhat = 0; vid.scale = 1; ne.s = "0"; } else if((uni == 'i' || uni == 'I' || uni == 'o' || uni == 'O') && ne.editwhat == &vid.alpha) { double d = exp(shiftmul/10); vid.alpha *= d; vid.scale *= d; ne.s = fts(vid.alpha); } else if(doexiton(sym, uni)) popScreen(); }; } int nlpage = 1; int wheelshift = 0; int handlePage(int& nl, int& nlm, int perpage) { nlm = nl; int onl = nl; int ret = 0; if(nlpage) { nl = nlm = perpage; if(nlpage == 2) ret = nlm; int w = wheelshift; int realw = 0; while(w<0 && ret) { ret--; w++; realw--; } while(w>0 && ret+perpage < onl) { ret++; w--; realw++; } wheelshift = realw; if(ret+nl > onl) nl = onl-ret; } return ret; } void displayPageButtons(int i, bool pages) { int i0 = vid.yres - vid.fsize; int xr = vid.xres / 80; if(pages) if(displayfrZ(xr*8, i0, 1, vid.fsize, IFM("1 - ") + XLAT("page") + " 1", nlpage == 1 ? 0xD8D8C0 : 0xC0C0C0, 8)) getcstat = '1'; if(pages) if(displayfrZ(xr*24, i0, 1, vid.fsize, IFM("2 - ") + XLAT("page") + " 2", nlpage == 1 ? 0xD8D8C0 : 0xC0C0C0, 8)) getcstat = '2'; if(pages) if(displayfrZ(xr*40, i0, 1, vid.fsize, IFM("3 - ") + XLAT("all"), nlpage == 1 ? 0xD8D8C0 : 0xC0C0C0, 8)) getcstat = '3'; if(i&1) if(displayfrZ(xr*56, i0, 1, vid.fsize, IFM("0 - ") + XLAT("return"), 0xC0C0C0, 8)) getcstat = '0'; if(i&2) if(displayfrZ(xr*72, i0, 1, vid.fsize, IFM("F1 - ") + XLAT("help"), 0xC0C0C0, 8)) getcstat = SDLK_F1; } bool handlePageButtons(int uni) { if(uni == '1') nlpage = 1, wheelshift = 0; else if(uni == '2') nlpage = 2, wheelshift = 0; else if(uni == '3') nlpage = 0, wheelshift = 0; else if(uni == PSEUDOKEY_WHEELUP) wheelshift--; else if(uni == PSEUDOKEY_WHEELDOWN) wheelshift++; else return false; return true; } void editNumber(ld& x, ld vmin, ld vmax, ld step, ld dft, string title, string help) { ne.editwhat = &x; ne.s = fts(x); ne.vmin = vmin; ne.vmax = vmax; ne.step = step; ne.dft = dft; ne.title = title; ne.help = help; ne.scale = ne.inverse_scale = identity; ne.intval = NULL; ne.positive = false; dialogflags = (cmode & sm::A3); if(cmode & sm::SIDE) dialogflags |= sm::MAYDARK; cmode |= sm::NUMBER; pushScreen(drawNumberDialog); reaction = reaction_t(); } void editNumber(int& x, int vmin, int vmax, int step, int dft, string title, string help) { editNumber(ne.intbuf, vmin, vmax, step, dft, title, help); ne.intbuf = x; ne.intval = &x; ne.s = its(x); } void helpToEdit(int& x, int vmin, int vmax, int step, int dft) { popScreen(); string title = "help"; if(help[0] == '@') { int iv = help.find("\t"); int id = help.find("\n"); title = help.substr(iv+1, id-iv-1); help = help.substr(id+1); } editNumber(x, vmin, vmax, step, dft, title, help); } //-- choose file dialog-- bool filecmp(const pair<string,int> &f1, const pair<string,int> &f2) { if(f1.first == "../") return true; if(f2.first == "../") return false; if(f1.second != f2.second) return f1.second == CDIR; return f1.first < f2.first; } string filecaption, cfileext; string *cfileptr; bool editext = false; bool_reaction_t file_action; bool handleKeyFile(int sym, int uni); void drawFileDialog() { displayfr(vid.xres/2, 30 + vid.fsize, 2, vid.fsize, filecaption, forecolor, 8); string& cfile = *cfileptr; displayfr(vid.xres/2, 34 + vid.fsize * 2, 2, vid.fsize, cfile, editext ? 0xFF00FF : 0xFFFF00, 8); displayButton(vid.xres*1/5, 38+vid.fsize * 3, "F2 = save", 2000+SDLK_F2, 8); displayButton(vid.xres*2/5, 38+vid.fsize * 3, "F3 = load", 2000+SDLK_F3, 8); displayButton(vid.xres*3/5, 38+vid.fsize * 3, "F4 = extension", 2000+SDLK_F4, 8); displayButton(vid.xres*4/5, 38+vid.fsize * 3, "Enter = back", 2000+SDLK_RETURN, 8); v.clear(); DIR *d; struct dirent *dir; string where = "."; for(int i=0; i<size(cfile); i++) if(cfile[i] == '/' || cfile[i] == '\\') where = cfile.substr(0, i+1); d = opendir(where.c_str()); if (d) { while ((dir = readdir(d)) != NULL) { string s = dir->d_name; if(s != ".." && s[0] == '.') ; else if(size(s) > 4 && s.substr(size(s)-4) == cfileext) v.push_back(make_pair(s, CFILE)); else if(dir->d_type & DT_DIR) v.push_back(make_pair(s+"/", CDIR)); } closedir(d); } sort(v.begin(), v.end(), filecmp); int q = v.size(); int percolumn = (vid.yres-38) / (vid.fsize+5) - 4; int columns = 1 + (q-1) / percolumn; for(int i=0; i<q; i++) { int x = 16 + (vid.xres * (i/percolumn)) / columns; int y = 42 + vid.fsize * 4 + (vid.fsize+5) * (i % percolumn); displayColorButton(x, y, v[i].first, 1000 + i, 0, 0, v[i].second, 0xFFFF00); } keyhandler = handleKeyFile; } bool handleKeyFile(int uni, int sym) { string& s(*cfileptr); int i = size(s) - (editext?0:4); if(uni > 2000) sym = uni - 2000; if(sym == SDLK_ESCAPE) { popScreen(); } else if(sym == SDLK_RETURN || sym == SDLK_KP_ENTER) { // we pop and re-push, in case if action opens something popScreen(); if(!file_action()) pushScreen(drawFileDialog); } else if(sym == SDLK_F4) { editext = !editext; } else if(sym == SDLK_BACKSPACE && i) { s.erase(i-1, 1); } else if(uni >= 32 && uni < 127) { s.insert(i, s0 + char(uni)); } else if(uni >= 1000 && uni <= 1000+size(v)) { string where = "", what = s, whereparent = "../"; for(int i=0; i<size(s); i++) if(s[i] == '/') { if(i >= 2 && s.substr(i-2,3) == "../") whereparent = s.substr(0, i+1) + "../"; else whereparent = where; where = s.substr(0, i+1), what = s.substr(i+1); } int i = uni - 1000; if(v[i].first == "../") { s = whereparent + what; } else if(v[i].second == CDIR) s = where + v[i].first + what; else s = where + v[i].first; } return true; } void openFileDialog(string& filename, string fcap, string ext, bool_reaction_t action) { cfileptr = &filename; filecaption = fcap; cfileext = ext; file_action = action; pushScreen(dialog::drawFileDialog); } // infix/v/vpush string infix; bool hasInfix(const string &s) { if(infix == "") return true; string t = ""; for(int i=0; i<size(s); i++) { char c = s[i]; char tt = 0; if(c >= 'a' && c <= 'z') tt += c - 32; else if(c >= 'A' && c <= 'Z') tt += c; else if(c == '@') tt += c; if(tt) t += tt; } return t.find(infix) != string::npos; } bool editInfix(int uni) { if(uni >= 'A' && uni <= 'Z') infix += uni; else if(uni >= 'a' && uni <= 'z') infix += uni-32; else if(infix != "" && uni == 8) infix = infix.substr(0, size(infix)-1); else if(infix != "" && uni != 0) infix = ""; else return false; return true; } vector<pair<string, int> > v; void vpush(int i, const char *name) { string s = XLATN(name); if(!hasInfix(s)) return; dialog::v.push_back(make_pair(s, i)); } };