ref: c0cbe87b357a4464a7d10ab8f38a65c77ea3b629
dir: /appl/demo/camera/tkinterface.b/
# Sort out timing with taking photo & getting jpg/thumbnail - make sure it gets the right one when 2photos have been taken & sort out 'cannot communicate with camera' error implement tkinterface; include "tk.m"; tk: Tk; include "tkclient.m"; tkclient: Tkclient; include "sys.m"; sys : Sys; include "daytime.m"; daytime: Daytime; include "readdir.m"; readdir: Readdir; include "bufio.m"; bufio: Bufio; Iobuf: import bufio; include "selectfile.m"; selectfile: Selectfile; include "string.m"; str : String; include "draw.m"; draw: Draw; Context, Display, Point, Rect, Image, Screen, Font: import draw; include "grid/readjpg.m"; readjpg: Readjpg; display : ref draw->Display; context : ref draw->Context; camerapath := ""; savepath := ""; tmppath := "/tmp/"; usecache := 1; working := 0; processing := 0; coords: draw->Rect; DONE : con 1; KILLED : con 2; font: ref Draw->Font; tkfont := ""; tkfontb := ""; tkfontf := ""; ssize := 3; maxsize : Point; nilrect := Draw->Rect((0,0),(0,0)); runwithoutcam := 0; toplevels : list of (ref Tk->Toplevel, string, list of int, int) = nil; procimg : ref Draw->Image; loadimg: ref Draw->Image; tkinterface : module { init : fn (ctxt : ref Draw->Context, argv : list of string); }; init(ctxt : ref Draw->Context, argv : list of string) { display = ctxt.display; context = ctxt; sys = load Sys Sys->PATH; # sys->pctl(Sys->NEWPGRP, nil); # sys->pctl(Sys->FORKNS, nil); str = load String String->PATH; readdir = load Readdir Readdir->PATH; daytime = load Daytime Daytime->PATH; bufio = load Bufio Bufio->PATH; str = load String String->PATH; draw = load Draw Draw->PATH; tk = load Tk Tk->PATH; tkclient = load Tkclient Tkclient->PATH; tkclient->init(); selectfile = load Selectfile Selectfile->PATH; selectfile->init(); readjpg = load Readjpg Readjpg->PATH; readjpg->init(display); font = draw->Font.open(display, "/fonts/charon/plain.small.font"); runfrom := hd argv; p := isat2(runfrom,"/"); savepath = runfrom[:p+1]; argv = tl argv; while (argv != nil) { if (camerapath == "" && (hd argv)[0] == '/') camerapath = hd argv; if (hd argv == "nocache") usecache = 0; argv = tl argv; } if (camerapath == "") camerapath = "./"; if (camerapath != "" && camerapath[len camerapath - 1] != '/') camerapath[len camerapath] = '/'; r := display.image.r; # if (r.dx() < 800 || r.dy() < 600) ssize = 2; if (r.dx() < 400 || r.dy() < 300) ssize = 1; maxsize = (r.dx(), r.dy()); if (ssize == 1) { tkfont = "/fonts/charon/plain.tiny.font"; tkfontb = "/fonts/charon/bold.tiny.font"; tkfontf = "/fonts/pelm/unicode.8.font"; } else if (ssize == 2) { tkfont = "/fonts/charon/plain.small.font"; tkfontb = "/fonts/charon/bold.small.font"; tkfontf = "/fonts/pelm/unicode.8.font"; } else { tkfont = "/fonts/charon/plain.normal.font"; tkfontb = "/fonts/charon/bold.normal.font"; tkfontf = "/fonts/pelm/unicode.8.font"; } if ((sys->stat(tkfont)).t0 == -1) tkfont = ""; else tkfont = " -font " + tkfont; if ((sys->stat(tkfontb)).t0 == -1) tkfontb = ""; else tkfontb = " -font " + tkfontb; if ((sys->stat(tkfontf)).t0 == -1) tkfontf = ""; else tkfontf = " -font " + tkfontf; procimg = display.open("camproc.bit"); loadimg = display.open("camload.bit"); spawn tkstuff(); } # Tk stuff thumbscr := array[] of { "frame .f", "frame .fthumb -bg white", "frame .f.finfo", "frame .f.fsnap", "menubutton .f.fsnap.fsettings.mb2 -text {Selected\n(0 files)} -menu .m2 @", "menu .m2 @", ".m2 add command -text {Select All} -command {send butchan selectall 1}", ".m2 add command -text {Select None} -command {send butchan selectall 0}", ".m2 add command -text {Invert Selection} -command {send butchan invert}", ".m2 add command -text {Refresh Files} -command {send butchan refresh}", "menu .m @", "frame .f.fsnap.fsettings -borderwidth 1 -relief raised", "menubutton .f.fsnap.fsettings.mb -text {Settings} -menu .m &", "button .f.fsnap.fsettings.b -text {Information} -command {send butchan info} &", "grid .f.fsnap.fsettings.b -row 0 -column 0 -sticky ew", "grid .f.fsnap.fsettings.mb -row 1 -column 0 -sticky ew", "grid .f.fsnap.fsettings.mb2 -row 2 -column 0 -sticky ew", "frame .f.fsnap.fstore -borderwidth 1 -relief raised", "label .f.fsnap.fstore.l1 -text { Photos taken: } @", "label .f.fsnap.fstore.l2 -text { Remaining: } @", "label .f.fsnap.fstore.l3 -text { } @", "label .f.fsnap.fstore.l4 -text { } @", "grid .f.fsnap.fstore.l1 -row 0 -column 0 -sticky w", "grid .f.fsnap.fstore.l2 -row 1 -column 0 -sticky w", "grid .f.fsnap.fstore.l3 -row 0 -column 1 -sticky w", "grid .f.fsnap.fstore.l4 -row 1 -column 1 -sticky w", "frame .f.fsnap.ftime -borderwidth 1 -relief raised", "label .f.fsnap.ftime.l1 -text {Local: } @", "label .f.fsnap.ftime.l2 -text {Camera: } @", "label .f.fsnap.ftime.l3", "label .f.fsnap.ftime.l4", "checkbutton .f.fsnap.ftime.cb -text {Set camera to local time} -variable time &", "button .f.fsnap.ftime.b -text {refresh} -command {send butchan gettime} &", "grid .f.fsnap.ftime.l1 -row 0 -column 0 -sticky w", "grid .f.fsnap.ftime.l2 -row 1 -column 0 -sticky w", "grid .f.fsnap.ftime.l3 -row 0 -column 1 -sticky w", "grid .f.fsnap.ftime.l4 -row 1 -column 1 -sticky w", "grid .f.fsnap.ftime.cb -row 2 -column 0 -columnspan 2", "grid .f.fsnap.ftime.b -row 3 -column 0 -columnspan 2", "button .f.fsnap.b -text {Take Photo} -command {send butchan snap} &", "grid columnconfigure .f.fsnap 2 -minsize 150", "frame .f.fcom", "frame .f.f1 -background #0d0d0d1a", "canvas .f.f1.c1 -yscrollcommand {.f.f1.sb1 set} -height 255 -width 542 -bg white", ".f.f1.c1 create window 0 0 -window .fthumb -anchor nw", "scrollbar .f.f1.sb1 -command {.f.f1.c1 yview}", # "frame .f.f2", # "canvas .f.f2.c1 -width 556 -height 304", # ".f.f2.c1 create window 0 0 -window .f.fsnap -anchor nw", "grid .f.fsnap -column 0 -row 0", "grid .f.f1 -column 0 -row 1", "grid .f.f1.c1 -column 0 -row 0", "grid .f.f1.sb1 -column 1 -row 0 -sticky ns", # "grid .f.f2 -column 0 -row 0", # "grid .f.f2.c1 -column 0 -row 0 -sticky ew", "bind .Wm_t <ButtonPress-1> +{focus .}", "bind .Wm_t.title <ButtonPress-1> +{focus .}", }; lastpath := ""; Aitem: adt { pname,desc: string; dtype,factory: int; read, location: string; data: list of (string, int); }; LIST: con 0; MINMAX: con 1; OTHER: con 2; noabilities := 0; abilities : array of Aitem; getdesc(l : list of string): list of string { s := ""; while(hd l != "min" && hd l != "items" && tl l != nil) { s += hd l + " "; l = tl l; } while (s[len s - 1] == ' ' || s[len s - 1] == '\n') s = s[:len s -1]; l = s :: l; return l; } inflist : list of (string, string); ablmenu : array of string; getabilities() { inflist = nil; abilities = array[200] of Aitem; fd := bufio->open(camerapath+"abilities", bufio->OREAD); if (runwithoutcam) fd = bufio->open("/usr/danny/camera/abls", bufio->OREAD); i := 0; for (;;) { take := 0; s := fd.gets('\n'); if (s == "") break; (n, lst) := sys->tokenize(s," ,:\t\n"); abilities[i].data = nil; abilities[i].read = ""; if (lst != nil && len hd lst == 4) { abilities[i].pname = hd lst; lst = getdesc(tl lst); abilities[i].desc = hd lst; if (hd tl lst == "items") { abilities[i].dtype = LIST; abilities[i].factory = int hd tl tl tl tl lst; noitems := int hd tl tl lst; for (k := 0; k < noitems; k++) { s = fd.gets('\n'); (n2, lst2) := sys->tokenize(s,",:\t\n"); name := hd lst2; val := int hd tl lst2; if (k == 0) { if (abilities[i].pname == "ssiz") abilities[i].factory = val; else if (abilities[i].pname == "scpn") abilities[i].factory = val; } if (val == abilities[i].factory && noitems > 1) name += " *"; abilities[i].data = (name, val) :: abilities[i].data; } if (noitems < 2) { inflist = (abilities[i].desc, (hd abilities[i].data).t0) :: inflist; take = 1; } } else if (hd tl lst == "min") { abilities[i].dtype = MINMAX; abilities[i].factory = int hd tl tl tl tl tl tl lst; min := int hd tl tl lst; max := int hd tl tl tl tl lst; mul := 1; while (max > 200000) { min /= 10; max /= 10; mul *= 10; } abilities[i].data = ("min", min) :: abilities[i].data; abilities[i].data = ("max", max) :: abilities[i].data; abilities[i].data = ("mul", mul) :: abilities[i].data; } else { inflist = (abilities[i].desc,list2string(tl lst)) :: inflist; take = 1; } if (take || abilities[i].desc == "Time Format" || abilities[i].desc == "Date Format" || abilities[i].desc == "File Type" || contains(abilities[i].desc,"Video") || contains(abilities[i].desc,"Media") || contains(abilities[i].desc,"Sound") || contains(abilities[i].desc,"Volume") || contains(abilities[i].desc,"Slide") || contains(abilities[i].desc,"Timelapse") || contains(abilities[i].desc,"Burst") || contains(abilities[i].desc,"Power") || contains(abilities[i].desc,"Sleep")) i--; i++; } } noabilities = i; } isat(s: string, test: string): int { num := -1; if (len test > len s) return -1; for (i := 0; i < (1 + (len s) - (len test)); i++) { if (num == -1 && test == s[i:i+len test]) num = i; } return num; } isat2(s: string, test: string): int { num := -1; if (len test > len s) return -1; for (i := len s - len test; i >= 0; i--) { if (num == -1 && test == s[i:i+len test]) num = i; } return num; } nomatches(s: string): int { n := 0; for (i := 0; i < noabilities; i++) { test := abilities[i].desc; if (len s <= len test && test[:len s] == s) n++; } return n; } matches(s1,s2: string): int { if (len s1 < len s2) return 0; if (s1[:len s2] == s2) return 1; return 0; } biggestmatch(nm: int, s: string, l: int): string { bigmatch := s; match := s[:l]; for (;;) { if (bigmatch == match) break; if (nomatches(bigmatch) == nm) return bigmatch; p := isat2(bigmatch," "); if (p < len match) break; bigmatch = bigmatch[:p]; } return match; } getabllist(): array of string { los : list of string; los = nil; for (i := 0; i < noabilities; i++) { p := 0; p2 := 0; nm : int; for (;;) { nm = -1; tmpl := los; while (tmpl != nil) { if (matches(abilities[i].desc, hd tmpl)) nm = 0; tmpl = tl tmpl; } if (nm == 0) break; p += p2; tmp := abilities[i].desc[p:]; p2 = isat(tmp, " "); if (p2 == -1) p2 = len tmp; else p2++; nm = nomatches(abilities[i].desc[:p+p2]); if (nm <= 5) break; } if (nm > 0) { listitem := biggestmatch(nm, abilities[i].desc,p+p2); los = listitem :: los; } } ar := array[len los] of string; for (i = len ar - 1; i >= 0; i--) { ar[i] = hd los; los = tl los; } return ar; } buildabilitiesframes(top: ref Tk->Toplevel) { ablmenu = getabllist(); tkcmd(top, ".m add command -text {Refresh Main Screen} -command {send butchan refreshstate}"); tkcmd(top, ".m add command -text {Reset Camera} -command {send butchan reset}"); for (k := 0; k < len ablmenu; k++) { if (len ablmenu[k] > 4 && (ablmenu[k][:4] == "Zoom" || ablmenu[k][:5] == "Still")) buildabilitiesframe(top,k,"butchan"); else tkcmd(top, ".m add command -text {"+ablmenu[k]+ "} -command {send butchan abls "+string k+"}"); } tkcmd(top, "menu .mthumb "+tkfont); tkcmd(top, ".mthumb add command -label {Selection (88 files)}"); tkcmd(top, ".mthumb add separator"); for (k = nothumbs; k < len menu; k++) tkcmd(top, ".mthumb add command -text {"+menu[k].text+"} " + "-command {send butchan}"); } buildabilitiesframe(top: ref Tk->Toplevel,k: int, chanout: string) { nm := string nomatches(ablmenu[k]); count2 := 0; for (i := 0; i < noabilities; i++) { if (matches(abilities[i].desc,ablmenu[k])) { frame : string; case abilities[i].pname { "scpn" or "ssiz" or "zpos" => frame = ".f.fsnap.f"+abilities[i].pname; tkcmd(top, "frame "+frame+" -borderwidth 1 -relief raised"); * => frame = ".f"; if (count2 == 0) { tkcmd(top, "frame "+frame); tkcmd(top, "label "+frame+".l -text {"+ablmenu[k]+"}"+tkfontb); tkcmd(top, "grid "+frame+".l -row 0 -column 0 -columnspan "+nm); } frame = frame + ".f"+string count2; tkcmd(top, "frame "+frame+" -borderwidth 1 -relief raised"); tkcmd(top, "grid "+frame+" -row 1 -column "+string count2+ " -sticky nsew"); mul := getval(abilities[i].data,"mul"); s := abilities[i].desc[len ablmenu[k]:]; if (mul != 1 && abilities[i].dtype == MINMAX) s += " (x"+string mul+")"; tkcmd(top, "label "+frame+".l -text {"+s+"}"+tkfont); tkcmd(top, "grid "+frame+".l -row 0 -column 0 -sticky nw"); } if (abilities[i].dtype == MINMAX) { abilities[i].location = frame+".sc"; min := getval(abilities[i].data,"min"); max := getval(abilities[i].data,"max"); tkcmd(top, sys->sprint("scale %s.sc -to %d -from %d %s", frame,min,max,tkfont)); tkcmd(top, "bind "+frame+".sc <ButtonPress-3> {send " + chanout + " scaleval " + string i + " %X %Y}"); tkcmd(top, "grid "+frame+".sc -row 1 -column 0"); } else if (abilities[i].dtype == LIST) { tkcmd(top, "frame "+frame+".frb"); tkcmd(top, "grid "+frame+".frb -row 1 -column 0"); tmp := abilities[i].data; row := 0; while (tmp != nil) { (name, val) := hd tmp; s := sys->sprint("radiobutton %s.frb.rb%d -text {%s} -value %d -variable %s -height %d %s",frame,row,name,val,abilities[i].pname,24 - (3*(3-ssize)), tkfont); tkcmd(top,s); tkcmd(top, sys->sprint("grid %s.frb.rb%d -row %d -column 0 -sticky w", frame,row,row)); tmp = tl tmp; row++; } } tkcmd(top, "button "+frame+".bs -text {Set} -command "+ "{send "+chanout+" set "+string i+"}"+butheight+tkfont); tkcmd(top, "grid "+frame+".bs -row 2 -column 0 -sticky ew"); if (abilities[i].dtype == MINMAX) { tkcmd(top, "button "+frame+".bf -text {Default} -command "+ "{send "+chanout+" setdef "+string i+"}"+butheight+tkfont); tkcmd(top, "grid "+frame+".bf -row 3 -column 0 -sticky ew"); } count2++; } } } getvaluescr := array[] of { "frame .f -height 84 -width 114 -borderwidth 2 -relief raised", "label .f.l1 -text {Enter Value:} @", "entry .f.e1 -width 100 -bg white @", "button .f.b1 -text { ok } -command {send chanin ok} &", "button .f.b2 -text cancel -command {send chanin cancel} &", "grid .f.l1 -column 1 -row 0 -columnspan 2 -padx 0 -sticky w", "grid .f.e1 -column 1 -row 1 -columnspan 2 -padx 0 -pady 5", "grid .f.b1 -column 1 -row 2 -padx 0", "grid .f.b2 -column 2 -row 2 -padx 0", "grid columnconfigure .f 1 -minsize 20", "grid columnconfigure .f 2 -minsize 20", "grid columnconfigure .f 3 -minsize 5", "grid rowconfigure .f 0 -minsize 20", "grid rowconfigure .f 1 -minsize 20", "grid rowconfigure .f 2 -minsize 20", "grid columnconfigure .f 0 -minsize 5", "bind .f.e1 <Key> {send chanin key %s}", "focus .f.e1", "pack .f", "update", }; getvaluescreen(x,y: string): int { x = string ((int x) - 55); y = string ((int y) - 30); (top, nil) := tkclient->toplevel(context, "-x "+x+" -y "+y, nil, tkclient->Plain); chanin := chan of string; tk->namechan(top, chanin, "chanin"); for (tk1 := 0; tk1 < len getvaluescr; tk1++) tkcmd(top, getvaluescr[tk1]); tkclient->onscreen(top, "exact"); tkclient->startinput(top, "kbd"::"ptr"::nil); for(;;) { alt { s := <-top.ctxt.kbd => tk->keyboard(top, s); s := <-top.ctxt.ptr => tk->pointer(top, *s); inp := <- chanin => if (inp == "ok") return int tkcmd(top, ".f.e1 get"); else if (inp == "cancel") return -1; else if (inp[:3] == "key") { s := " "; s[0] = int inp[4:]; if (s[0] == '\n') return int tkcmd(top, ".f.e1 get"); if (s[0] >= '0' && s[0] <= '9') { tkcmd(top, ".f.e1 delete sel.first sel.last"); tkcmd(top, ".f.e1 insert insert {"+s+"}; update"); } } } } } infoscreen() { (top, titlebar) := tkclient->toplevel(context, "", "Information", Tkclient->Hide); tmp := inflist; tkcmd(top, "frame .f"); tkcmd(top, "label .f.l -text {Information}"); tkcmd(top, "grid .f.l -row 0 -column 0 -columnspan 2"); tkcmd(top, "frame .f.finfo -borderwidth 1 -relief raised"); tkcmd(top, "grid .f.finfo"); infrow := 0; while (tmp != nil) { infrow++; s := string infrow; (d1,d2) := hd tmp; tkcmd(top, "label .f.finfo.l"+s+"1 -text {"+d1+"}"); tkcmd(top, "label .f.finfo.l"+s+"2 -text {"+d2+"}"); tkcmd(top, "grid .f.finfo.l"+s+"1 -row "+s+" -column 0 -sticky w"); tkcmd(top, "grid .f.finfo.l"+s+"2 -row "+s+" -column 1 -sticky e"); tmp = tl tmp; } tkcmd(top, "pack .f; update"); tkclient->onscreen(top, nil); tkclient->startinput(top, "kbd"::"ptr"::nil); main: for(;;) { alt { s := <-top.ctxt.kbd => tk->keyboard(top, s); s := <-top.ctxt.ptr => tk->pointer(top, *s); title := <-top.ctxt.ctl or title = <-top.wreq or title = <- titlebar => if (title == "exit") break main; tkclient->wmctl(top, title); } } } settingsscreen(k: int, ctlchan: chan of int) { low := toplevels; for (;low != nil; low = tl low) { (tplvl, name, nil,nil) := hd low; if (name == ablmenu[k]) { tkcmd(tplvl, "raise .; focus .; update"); ctlchan <-= DONE; return; } } pid := sys->pctl(0, nil); (top, titlebar) := tkclient->toplevel(context, "", "Config", Tkclient->Appl); chanin := chan of string; tk->namechan(top,chanin, "chanin"); buildabilitiesframe(top,k, "chanin"); tkcmd(top,"bind .Wm_t <ButtonPress-1> +{focus .}"); tkcmd(top,"bind .Wm_t.title <ButtonPress-1> +{focus .}"); tkcmd(top, "pack .f; update"); err := 0; allread := 1; l : list of int = nil; for (i := 0; i < noabilities; i++) { if (matches(abilities[i].desc, ablmenu[k])) { l = i :: l; if (abilities[i].read != "") setmystate(top,i,abilities[i].read); else allread = 0; } } tkclient->onscreen(top, nil); tkclient->startinput(top, "kbd"::"ptr"::nil); if (!allread) { spawn workingscreen2(getcoords(top),pid, ctlchan,0); ltmp := l; for (;ltmp != nil; ltmp = tl ltmp) { if (abilities[hd ltmp].read == "" && getstate(top, hd ltmp) == -1) { err = 1; break; } } } if (!err) spawn settingsloop(top,chanin,titlebar,k,l); ctlchan <-= DONE; } settingsloop(top: ref Tk->Toplevel, chanin,titlebar: chan of string, k: int, abls: list of int) { tkcmd(top, "focus .Wm_t"); pid := sys->pctl(0,nil); addtoplevel(top,ablmenu[k], abls, pid); ctlchan := chan of int; main: for(;;) { alt { s := <-top.ctxt.kbd => tk->keyboard(top, s); s := <-top.ctxt.ptr => tk->pointer(top, *s); inp := <- chanin => tkcmd(top, "focus ."); (n, lst) := sys->tokenize(inp, " \t\n"); case hd lst { "scaleval" => i := int hd tl lst; val := getvaluescreen(hd tl tl lst, hd tl tl tl lst); if (val != -1) tkcmd(top, abilities[i].location+" set "+string val+";update"); "set" or "setdef" => if (working) dialog(" Camera is busy! ", 2,-1,getcoords(top)); else { spawn set(top, int hd tl lst, hd lst, ctlchan); <-ctlchan; working = 0; } } clearbuffer(chanin); title := <-top.ctxt.ctl or title = <-top.wreq or title = <- titlebar => if (title == "exit") break main; tkclient->wmctl(top, title); } } deltoplevel(top); } clearbuffer(c: chan of string) { tc := chan of int; spawn timer(tc); main: for (;;) alt { del := <-c => ; tick := <-tc => break main; } } timer(tick: chan of int) { sys->sleep(100); tick <- = 1; } getval(l: list of (string,int), s: string): int { while (l != nil) { (name,val) := hd l; if (name == s) return val; l = tl l; } return -2; } list2string(l : list of string): string { s := ""; while (l != nil) { s += " " + hd l; l = tl l; } if (s != "") return s[1:]; return s; } JPG: con 0; THUMB: con 1; Imgloaded: adt { name: string; imgtype: int; }; nofiles := 0; filelist := array[200] of string; thumbimg := array[200] of ref draw->Image; selected := array[200] of { * => 0 }; noselected := 0; fnew : list of int; imgloaded : list of Imgloaded; maxwidth, maxheight: int; nothumbs := 0; nocamera(): int { (n,dir) := sys->stat(camerapath+"ctl"); if (n != -1) return 0; return 1; } startuptkstuff(top: ref Tk->Toplevel, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(coords,pid, ctlchan,1); getabilities(); (dirs,n) := readdir->init(camerapath+"thumb", readdir->NAME); if (n == -1) nothumbs = 1; buildabilitiesframes(top); refreshfilelist(top,0); ctlchan <-= DONE; } tibuild := 0; butheight := ""; tkstuff() { if (!runwithoutcam && nocamera()) { dialog("Cannot find camera!",0,-1,nilrect); exit; } (win, titlebar) := tkclient->toplevel(context, "", "Camera", Tkclient->Appl); tkcmd(win, "frame .test"); if (tkcmd(win, ".test cget -bg") == "#ffffffff") tibuild = 1; tkcmd(win, "destroy .test"); butheight = " -height "+string (16 + (5*tibuild) - (3*(3-ssize))); butchan := chan of string; tk->namechan(win, butchan, "butchan"); for (tk1 := 0; tk1 < len thumbscr; tk1++) tkcmd(win, thumbscr[tk1]); coords = display.image.r; ctlchan := chan of int; imgloaded = nil; spawn startuptkstuff(win, ctlchan); e := <- ctlchan; if (e == KILLED) { dialog("Cancel during load!",0,-1,coords); exit; } working = 0; spawn mainscreen(win, 1, ctlchan); <- ctlchan; working = 0; processing = 0; tkcmd(win, "pack propagate . 0"); resizemain(win,1); tkcmd(win, "pack .f; update; focus ."); coords = getcoords(win); loadimg = nil; tkclient->onscreen(win, nil); tkclient->startinput(win, "kbd"::"ptr"::nil); main: for (;;) { alt { s := <-win.ctxt.kbd => tk->keyboard(win, s); s := <-win.ctxt.ptr => tk->pointer(win, *s); inp := <-butchan => tkcmd(win, "focus ."); (n, lst) := sys->tokenize(inp, "\t\n "); case hd lst { # Communicates internally "scaleval" => i := int hd tl lst; val := getvaluescreen(hd tl tl lst, hd tl tl tl lst); if (val != -1) tkcmd(win, abilities[i].location+" set "+string val); "info" => spawn infoscreen(); "unload" => i := int hd tl lst; for (k := 0; k < nofiles; k++) { if (i == k || (i == -1 && selected[k])) { delloaded(filelist[k],JPG); delloaded(filelist[k],THUMB); } } "invert" => nf := 0; for (i := 0; i < nofiles; i++) selected[i] = 1 - selected[i]; doselect(win); "selectall" => val := int hd tl lst; for (i := 0; i < nofiles; i++) selected[i] = val; doselect(win); "select" => i := int hd tl lst; selected[i] = 1 - selected[i]; doselect(win); "selectonly" => i := int hd tl lst; val := selected[i]; for (k := 0; k < nofiles; k++) selected[k] = 0; if (noselected - val == 0) selected[i] = 1 - val; else selected[i] = 1; doselect(win); "menu" => i := int hd tl lst; if (selected[i] && noselected > 1) i = -1; title := "Selection ("+string noselected+" files)"; if (i != -1) title = filelist[i]+".jpg"; si := string i; tkcmd(win, ".mthumb entryconfigure 0 -text {"+title+"}"); for (k := nothumbs; k < len menu; k++) tkcmd(win, ".mthumb entryconfigure "+string (2+k-nothumbs)+ " -command {send butchan "+ menu[k].com+" "+si+"}"); tkcmd(win, ".mthumb post "+hd tl tl lst+" "+hd tl tl tl lst); * => if (!processing) spawn dealwithcamera(win, lst); } tkcmd(win, "update"); clearbuffer(butchan); title := <-win.ctxt.ctl or title = <-win.wreq or title = <-titlebar => if (title == "exit") break main; err := tkclient->wmctl(win, title); if (err == nil && title == "!size") { (n, lst) := sys->tokenize(title, " "); if (hd tl lst == ".") resizemain(win,0); } coords = getcoords(win); } } for (; toplevels != nil; toplevels = tl toplevels) { (nil, nil, nil, pid) := hd toplevels; if (pid != -1) kill(pid); } while (imgloaded != nil) { (fname, ftype) := hd imgloaded; sys->remove(tmppath+fname+"."+string ftype+"~"); imgloaded = tl imgloaded; } tkcmd(win, "destroy ."); exit; } dealwithcamera(win: ref Tk->Toplevel, lst: list of string) { ctlchan := chan of int; processing = 1; case hd lst { "gettime" => spawn refreshtime(win, ctlchan); <- ctlchan; "show" => spawn loadthumb(win,int hd tl lst,ctlchan); <- ctlchan; "snap" => selected[nofiles+1] = 0; spawn takephoto(win, ctlchan); <- ctlchan; working = 0; if (fnew == nil) break; spawn waittilready(camerapath+"jpg/"+filelist[hd fnew]+".jpg", ctlchan); e := <- ctlchan; working = 0; if (e == DONE) { spawn loadnewthumb(win, ctlchan); <- ctlchan; working = 0; } "abls" => spawn settingsscreen(int hd tl lst, ctlchan); <- ctlchan; "set" or "setdef" => spawn set(win, int hd tl lst, hd lst, ctlchan); <- ctlchan; "del" => spawn delete(win, int hd tl lst, ctlchan); <- ctlchan; "view" => i := int hd tl lst; unnew(win, i); if (i == -1) multiview(); else vw(i); "refresh" => spawn refresh(win, ctlchan); <- ctlchan; "refreshstate" => spawn mainscreen(win, 0, ctlchan); <- ctlchan; "dnld" => i := int hd tl lst; unnew(win, i); if (i == -1) multidownload(); else dnld(i, ""); "reset" => if (dialog("reset camera to default settings?",1,-1,coords)) { spawn resetcam(win,1, ctlchan); <- ctlchan; } } processing = 0; working = 0; } refresh(top: ref Tk->Toplevel, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(coords,pid, ctlchan,0); refreshfilelist(top,1); ctlchan <-= DONE; } delete(top: ref Tk->Toplevel, i: int, ctlchan: chan of int) { pid := sys->pctl(0, nil); ok : int; s := ""; loi : list of int; loi = nil; if (i == -1) { for (k := 0; k < nofiles; k++) if (selected[k]) s+= filelist[k]+".jpg\n"; if (!dialog("Delete Selected files?\n\n"+s,1,-1,coords)) { ctlchan <-= DONE; return; } } else if (!dialog("Delete "+filelist[i]+".jpg?",1,i,coords)) { ctlchan <-= DONE; return; } spawn workingscreen2(coords,pid, ctlchan,0); s = ""; for (k := 0; k < nofiles; k++) { if ((i == -1 && selected[k]) || k == i) { s += filelist[k]+".jpg "; ok = sys->remove(camerapath+ "jpg/"+filelist[k]+".jpg"); if (ok == -1) s+="failed\n"; else { s+="ok\n"; loi = k :: loi; } } } if (loi == nil && i != -1) { dialog("cannot remove "+filelist[i]+".jpg?",0,i,coords); ctlchan <-= DONE; return; } while (loi != nil) { delloaded(filelist[hd loi],JPG); delloaded(filelist[hd loi],THUMB); delselect(hd loi); loi = tl loi; } refreshfilelist(top,0); getstore(top); if (i == -1) dialog("Files deleted:\n\n"+s,0,-1,coords); ctlchan <-= DONE; } delselect(n: int) { for (i := n; i < nofiles - 1; i++) selected[i] = selected[i+1]; selected[nofiles - 1] = 0; } doselect(top: ref Tk->Toplevel) { n := 0; for (i := 0; i < nofiles; i++) { col := "white"; if (selected[i]) { col = "blue"; n++; } tkcmd(top,".fthumb.p"+string i+" configure -bg "+col); } noselected = n; s := " files"; if (n == 1) s = " file"; tkcmd(top, ".f.fsnap.fsettings.mb2 configure -text {Selected\n("+string n+s+")}"); } takephoto(top: ref Tk->Toplevel, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(coords,pid, ctlchan,0); fd := sys->open(camerapath+"ctl",sys->OWRITE); if (fd != nil) { e := sys->fprint(fd, "snap"); if (e < 0) { dialog("Could not take photo",0,-1,coords); getstore(top); } else { getstore(top); n := nofiles; for (i := 0; i < 5; i++) { refreshfilelist(top,1); sys->sleep(1000); if (nofiles > n) break; } } } ctlchan <-= DONE; } unnew(top: ref Tk->Toplevel, i: int) { if (fnew == nil) return; tmp : list of int = nil; for (;fnew != nil; fnew = tl fnew) { if (i == -1 && selected[hd fnew]) i = hd fnew; if (hd fnew == i) tkcmd(top, ".fthumb.mb"+string hd fnew+" configure -fg black; update"); else tmp = hd fnew :: tmp; } fnew = tmp; } refreshtime(top: ref Tk->Toplevel, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(coords,pid, ctlchan,0); if (!samedate(top) && tkcmd(top, "variable time") == "1") settime(); gettime(top); ctlchan <-= DONE; } addtoplevel(top: ref Tk->Toplevel, name: string, abls: list of int, pid: int) { ltmp := toplevels; isin := 0; for (;ltmp != nil; ltmp = tl ltmp) { (tplvl, nil, nil, nil) := hd ltmp; if (tplvl == top) isin = 1; } if (!isin) toplevels = (top, name, abls, pid) :: toplevels; } deltoplevel(top: ref Tk->Toplevel) { ltmp : list of (ref Tk->Toplevel, string, list of int, int) = nil;; for (;toplevels != nil; toplevels = tl toplevels) { (tplvl, nm, loi, p) := hd toplevels; if (tplvl != top) ltmp = (tplvl, nm, loi, p) :: ltmp; } toplevels = ltmp; } resetcam(top: ref Tk->Toplevel, show: int, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(coords,pid, ctlchan,0); for (i := 0; i < noabilities; i++) setstate(i, string abilities[i].factory); if (show) { ltmp := toplevels; for (;ltmp != nil; ltmp = tl ltmp) { (tplvl, nm, loi, p) := hd ltmp; for (; loi != nil; loi = tl loi) setmystate(tplvl, hd loi, string abilities[hd loi].factory); } if (top != nil) getstore(top); } ctlchan <-= DONE; } set(top: ref Tk->Toplevel, i: int, s: string, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(getcoords(top),pid, ctlchan,0); val : string; if (s == "setdef") { val = string abilities[i].factory; setmystate(top,i,val); } else { if (abilities[i].dtype == MINMAX) { val = tkcmd(top, abilities[i].location+" get"); mul := getval(abilities[i].data, "mul"); val = string (int val * mul); } else { val = tkcmd(top, "variable "+abilities[i].pname); } } e := setstate(i,val); if (e == 2) getstore(top); else if (e == 0) dialog("cannot communicate with camera",0,-1,coords); ctlchan <-= DONE; } setstate(i: int, val: string): int { fd := sys->open(camerapath+"ctl",sys->OWRITE); if (fd != nil) { sys->fprint(fd, "%s %s",abilities[i].pname,val); abilities[i].read = val; if (abilities[i].pname == "ssiz" || abilities[i].pname == "scpn") return 2; return 1; } else return 0; } getfirst(s: string): string { (n, lst) := sys->tokenize(s," \n\t"); if (lst == nil) return ""; return hd lst; } getabl(pname: string): int { for (i := 0; i < noabilities; i++) if (abilities[i].pname == pname) return i; return -1; } getstate(top: ref Tk->Toplevel, i: int): int { fd := sys->open(camerapath+"state", sys->OWRITE); if (fd != nil) { sys->fprint(fd ,"%s", abilities[i].pname); sys->sleep(500); fdi := bufio->open(camerapath+"state",sys->OREAD); if (fdi != nil) { s := fdi.gets('\n'); if (s != nil) { (n,lst) := sys->tokenize(s,":\n"); val := hd tl lst; setmystate(top,i,val); } return 0; } } dialog("cannot communicate with camera",0,-1,coords); return -1; } setmystate(top: ref Tk->Toplevel, i: int, val: string) { abilities[i].read = val; if (abilities[i].dtype == LIST) tkcmd(top, "variable "+abilities[i].pname+" "+val); else if (abilities[i].dtype == MINMAX) { mul := getval(abilities[i].data, "mul"); tkcmd(top, abilities[i].location+" set "+string((int val)/mul)); } tkcmd(top, "update"); } max(a,b: int): int { if (a > b) return a; return b; } refreshfilelist(win: ref Tk->Toplevel, refresh: int): int { if (refresh) { fd := sys->open(camerapath+"ctl",sys->OWRITE); if (fd == nil) { dialog("cannot communicate with camera",0,-1,coords); return -1; } else sys->fprint(fd, "refresh"); } oldlist := filelist[:nofiles]; for (i := 0; i < nofiles; i++) { si := string i; tk->cmd(win, "grid forget .fthumb.mb"+si+" .fthumb.p"+si); tk->cmd(win, "destroy .fthumb.mb"+si+" .fthumb.p"+si+" .mthumb"+si); } (dirs,n) := readdir->init(camerapath+"jpg", readdir->NAME); if (n == -1) return -1; nofiles = n; row := 0; col := 0; nocols := -1; w1 := int tkcmd(win, ".f.f1.c1 cget -width"); w := 0; fnew = nil; for (i = 0; i < nofiles; i++) { filelist[i] = dirs[i].name; if (len filelist[i] > 3 && filelist[i][len filelist[i] - 4] == '.') filelist[i] = filelist[i][:len filelist[i]-4]; isnew := 1; for (k := 0; k < len oldlist; k++) { if (filelist[i] == oldlist[k]) { isnew = 0; break; } } si := string i; tkcmd(win, "menubutton .fthumb.mb"+si+" -bg white " + "-text {"+filelist[i]+".jpg} -menu .mthumb"+si+tkfontf); if (isnew && refresh) { fnew = i :: fnew; tkcmd(win, ".fthumb.mb"+si+" configure -fg red"); } thumbimg[i] = display.newimage(Rect((0,0),(90,90)),draw->RGB24,0,int 16rffcc00ff); e := tkcmd(win,"panel .fthumb.p"+si+" -borderwidth 2 -bg white"+ " -height 90 -width 90 -relief raised"); tk->putimage(win,".fthumb.p"+si, thumbimg[i],nil); tkcmd(win, "bind .fthumb.p"+si+" <Double-Button-1> {send butchan view "+si+"}"); tkcmd(win, "bind .fthumb.p"+si+" <ButtonPress-1> {send butchan selectonly "+si+"}"); tkcmd(win, "bind .fthumb.p"+si+" <ButtonPress-2> {send butchan select "+si+"}"); tkcmd(win, "bind .fthumb.p"+si+" <ButtonPress-3> {send butchan menu "+si+" %X %Y}"); thisw := int tkcmd(win, ".fthumb.mb"+si+" cget -width"); w += max(94, thisw); if ((nocols == -1 && w >= w1-(col*2)) || col == nocols) { nocols = col; col = 0; row+=2; w = thisw; } if (col == 0) tkcmd(win, "grid rowconfigure .fthumb "+string (row+1)+ " -minsize "+string (105 - 2*(3-ssize))); tkcmd(win, "grid .fthumb.mb"+si+" -row "+string row+" -column "+string col); tkcmd(win, "grid .fthumb.p"+si+" -row "+string (row+1)+" -column "+string col+" -sticky n"); tkcmd(win, "menu .mthumb"+si+tkfont); for (k = nothumbs; k < len menu; k++) tkcmd(win, ".mthumb"+si+" add command -text {"+menu[k].text+"} " + "-command {send butchan "+menu[k].com+" "+si+"}"); if (isloaded(filelist[i],THUMB) && usecache) loadthumbnail(win,i); col++; } if (row == 0) nocols = col; doselect(win); size := tkcmd(win, "grid size .fthumb"); csize := int size[:isat(size, " ")]; rsize := int size[isat(size, " ")+1:]; if (csize > nocols) tkcmd(win, "grid columndelete .fthumb "+string nocols+" "+string csize); if (rsize > row+1) tkcmd(win, "grid rowdelete .fthumb "+string (row+2)+" "+string rsize); height := string (2 + int tkcmd(win, ".fthumb cget -height")); width := tkcmd(win, ".f.f1.c1 cget -width"); colsize : int; if (nocols > 0) colsize = int width / nocols; else colsize = int width; for (i = 0; i < nocols; i++) tkcmd(win, "grid columnconfigure .fthumb "+string i+" -minsize "+string colsize); tkcmd(win, ".f.f1.c1 configure -scrollregion { 0 0 "+width+" "+height+"}"); tkcmd(win, "update"); return 0; } Mtype: adt { text, com: string; }; menu := array[] of { Mtype ("Show Thumbnail", "show"), Mtype ("Download", "dnld"), Mtype ("View", "view"), Mtype ("Delete", "del"), Mtype ("Clear Cache", "unload"), Mtype ("Refresh Files", "refresh"), }; tkcmd(top: ref Tk->Toplevel, cmd: string): string { if (cmd[len cmd - 1] == '$') cmd = cmd[:len cmd - 1] + tkfontb; else if (cmd[len cmd - 1] == '@') cmd = cmd[:len cmd - 1] + tkfont; if (cmd[len cmd - 1] == '&') cmd = cmd[:len cmd - 1] + butheight+tkfont; e := tk->cmd(top, cmd); if (e != "" && e[0] == '!') sys->print("tk error: '%s': %s\n",cmd,e); return e; } loadnewthumb(top: ref Tk->Toplevel, ctlchan: chan of int) { pid := sys->pctl(0,nil); spawn workingscreen2(coords,pid, ctlchan,0); getstore(top); for (tmp := fnew; tmp != nil; tmp = tl tmp) loadthumbnail(top,hd tmp); ctlchan <-= DONE; } loadthumb(top: ref Tk->Toplevel, i: int, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(coords,pid, ctlchan,0); if (i == -1) { for (k := 0; k < nofiles; k++) if (selected[k]) if (loadthumbnail(top, k) != 0) break; } else loadthumbnail(top, i); ctlchan <-= DONE; } loadthumbnail(top: ref Tk->Toplevel, i: int): int { fd : ref sys->FD; if (usecache && isloaded(filelist[i],THUMB)) fd = sys->open(tmppath+filelist[i]+"."+string THUMB+"~",sys->OREAD); else fd = sys->open(camerapath+"thumb/"+filelist[i]+".bit",sys->OREAD); if (fd == nil) { if (usecache && isloaded(filelist[i],THUMB)) { delloaded(filelist[i],THUMB); return loadthumbnail(top,i); } else dialog("cannot open "+filelist[i]+".bit",0,-1,coords); return -2; } image := display.readimage(fd); if (image == nil) { if (usecache && isloaded(filelist[i],THUMB)) { delloaded(filelist[i],THUMB); return loadthumbnail(top,i); } else dialog("Could not load thumbnail: "+filelist[i]+".jpg",0,-1,coords); return -1; } else { p := Point((90-image.r.max.x)/2,(90-image.r.max.y)/2); thumbimg[i].draw(image.r.addpt(p), image,nil,(0,0)); si := string i; tkcmd(top,".fthumb.p"+si+" dirty"); fd = nil; n := -1; if (usecache) { fd = sys->create(tmppath+filelist[i]+"."+string THUMB+"~",sys->OWRITE,8r666); n = display.writeimage(fd, image); } x := int tkcmd(top, ".fthumb.mb"+string i+" cget -actx"); y := int tkcmd(top, ".fthumb.mb"+string i+" cget -acty"); h := int tkcmd(top, ".fthumb.mb"+string i+" cget -height"); x1 := int tkcmd(top, ".fthumb cget -actx"); y1 := int tkcmd(top, ".fthumb cget -acty"); tkcmd(top, ".f.f1.c1 see "+string (x-x1)+" " +string (y-y1)+ " "+string (x-x1+90)+" " +string (y-y1+h+102)+"; update"); if (!usecache || n == 0) imgloaded = (filelist[i],THUMB) :: imgloaded; } return 0; } isloaded(name: string, ftype: int): int { tmp := imgloaded; while (tmp != nil) { ic := hd tmp; if (ic.name == name && ic.imgtype == ftype) return 1; tmp = tl tmp; } return 0; } delloaded(name: string, ftype: int) { tmp : list of Imgloaded; tmp = nil; while (imgloaded != nil) { ic := hd imgloaded; if (ic.name != name || ic.imgtype != ftype) tmp = ic :: tmp; else sys->remove(tmppath+ic.name+"."+string ic.imgtype+"~"); imgloaded = tl imgloaded; } imgloaded = tmp; } dialog(msg: string, diagtype, img: int, r: Rect): int { if (diagtype == 2) diagtype = 0; else working = 0; tmpimg : ref draw->Image; out := 0; title := "Dialog"; if (diagtype == 0) title = "Alert!"; (win, titlebar) := tkclient->toplevel(context, "" , title, Tkclient->Appl); diagchan := chan of string; tk->namechan(win, diagchan, "diagchan"); tkcmd(win, "frame .f"); tkcmd(win, "label .f.l -text {"+msg+"}"+tkfont); tkcmd(win, "button .f.bo -text { ok } -command {send diagchan ok} "+butheight+tkfont); tkcmd(win, "button .f.bc -text {cancel} -command {send diagchan cancel}"+butheight+tkfont); if (img >= 0 && isloaded(filelist[img], THUMB) && usecache) { fd := sys->open(tmppath+filelist[img]+"."+string THUMB+"~", sys->OREAD); if (fd != nil) { tmpimg = display.readimage(fd); tkcmd(win,"panel .f.p -height "+string tmpimg.r.max.y+ " -width "+string tmpimg.r.max.x+" -borderwidth 2 -relief raised"); tk->putimage(win,".f.p", tmpimg, nil); tkcmd(win, "grid .f.p -row 1 -column 0 -columnspan 2 -padx 5 -pady 5"); } } tkcmd(win, "grid .f.l -row 0 -column 0 -columnspan 2 -padx 10 -pady 5"); if (diagtype == 1) { tkcmd(win, "grid .f.bo -row 2 -column 0 -padx 5 -pady 5"); tkcmd(win, "grid .f.bc -row 2 -column 1 -padx 5 -pady 5"); } else tkcmd(win, "grid .f.bo -row 2 -column 0 -columnspan 2 -padx 5 -pady 5"); if (!r.eq(nilrect)) centrewin(win, r, 1); else tkcmd(win, "pack .f; focus .; update"); tkclient->onscreen(win, "exact"); tkclient->startinput(win, "kbd"::"ptr"::nil); main: for (;;) { alt { s := <-win.ctxt.kbd => tk->keyboard(win, s); s := <-win.ctxt.ptr => tk->pointer(win, *s); inp := <-diagchan => if (inp == "ok") { out = 1; break main; } if (inp == "cancel") break main; title = <-win.ctxt.ctl or title = <-win.wreq or title = <-titlebar => if (title == "exit") break main; else tkclient->wmctl(win, title); } } return out; } snapscr := array[] of { "label .f.fsnap.ltime -text {Date and Time} $", "label .f.fsnap.lstore -text {Memory Status} $", "label .f.fsnap.lzpos -text {Zoom} $", "label .f.fsnap.lssiz -text {Resolution} $", "label .f.fsnap.lscpn -text {Compression} $", "grid .f.fsnap.ltime -row 0 -column 0 -sticky sw", "grid .f.fsnap.lstore -row 0 -column 1 -sticky sw", "grid .f.fsnap.lscpn -row 2 -column 0 -sticky sw", "grid .f.fsnap.lssiz -row 2 -column 1 -sticky sw", "grid .f.fsnap.lzpos -row 2 -column 2 -sticky sw", "grid .f.fsnap.ftime -row 1 -column 0 -sticky nsew", "grid .f.fsnap.fstore -row 1 -column 1 -sticky nsew", "grid .f.fsnap.fsettings -row 1 -column 2 -sticky nsew", "grid .f.fsnap.fscpn -row 3 -column 0 -sticky nsew", "grid .f.fsnap.fssiz -row 3 -column 1 -sticky nsew", "grid .f.fsnap.fzpos -row 3 -column 2 -sticky nsew", "grid .f.fsnap.b -row 4 -column 0 -columnspan 3", "grid rowconfigure .f.fsnap 0 -minsize 30", "grid rowconfigure .f.fsnap 2 -minsize 30", "grid rowconfigure .f.fsnap 4 -minsize 30", "update", }; mainscreen(win: ref Tk->Toplevel, opt: int, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(coords, pid, ctlchan, opt); if (opt == 1) { for (tk1 := 0; tk1 < len snapscr; tk1++) tkcmd(win, snapscr[tk1]); gettime(win); if (samedate(win)) tkcmd(win, "variable time 1; update"); } getstore(win); lst := getabl("scpn") :: getabl("ssiz") :: getabl("zpos") :: nil; if (getstate(win, hd tl tl lst) == 0); if (getstate(win, hd tl lst) == 0); getstate(win, hd lst); if (opt == 1) { addtoplevel(win, "", lst, -1); height := tkcmd(win, ".f.fsnap cget -height"); width := tkcmd(win, ".f.fsnap cget -width"); # tkcmd(win, ".f.f2.c1 configure -scrollregion { 0 0 "+width+" "+height+"}"); # tkcmd(win, ".f.f2.c1 configure -height "+height+"}"); } ctlchan <-= DONE; } kill(pid: int) { pctl := sys->open("/prog/" + string pid + "/ctl", Sys->OWRITE); if (pctl != nil) sys->write(pctl, array of byte "kill", len "kill"); } gettime(win: ref Tk->Toplevel) { tkcmd(win,".f.fsnap.ftime.l3 configure -text {}"+tkfont); tkcmd(win,".f.fsnap.ftime.l4 configure -text {}"+tkfont); fdi := bufio->open(camerapath+"date",sys->OREAD); if (fdi != nil) { s := fdi.gets('\n'); if (s != nil) { if (s[len s - 1] == '\n') s = s[:len s - 1]; tm := daytime->local(daytime->now()); time := sys->sprint("%d/%d/%d %d:%d:%d", tm.mon+1, tm.mday, tm.year-100, tm.hour,tm.min,tm.sec); ltime = addzeros(time); ctime = addzeros(s[len "date is ":]); tk->cmd(win,".f.fsnap.ftime.l3 configure -text {"+ltime+"}"); tk->cmd(win,".f.fsnap.ftime.l4 configure -text {"+ctime+"}"); } } if (len ltime < 16) ltime = "??/??/?? ??:??:??"; if (len ctime < 16) ctime = "??/??/?? ??:??:??"; tkcmd(win, "update"); } addzeros(s: string): string { s[len s] = ' '; rs := ""; start := 0; isnum := 0; for (i := 0; i < len s; i++) { if (s[i] < '0' || s[i] > '9') { if (isnum && i - start < 2) rs[len rs] = '0'; rs += s[start:i+1]; start = i+1; isnum = 0; } else isnum = 1; } i = len rs - 1; while (i >= 0 && rs[i] == ' ') i--; return rs[:i+1]; } samedate(win: ref Tk->Toplevel): int { s1 := tkcmd(win, ".f.fsnap.ftime.l3 cget -text"); s2 := tkcmd(win, ".f.fsnap.ftime.l4 cget -text"); if (s1 == "" || s1 == "") return 0; if (s1[:len s1 - 3] == s2[:len s2 - 3]) return 1; return 0; } settime() { tm := daytime->local(daytime->now()); fd := sys->open(camerapath+"date", sys->OWRITE); if (fd != nil) { sys->fprint(fd, "%s", addzeros(sys->sprint("%d/%d/%d %d:%d:%d" ,tm.mon+1, tm.mday, tm.year-100, tm.hour,tm.min,tm.sec))); } } getstore(win: ref Tk->Toplevel) { fdi := bufio->open(camerapath+"storage",sys->OREAD); if (fdi != nil) { for(i := 0; i < 3; i++) { s := fdi.gets('\n'); if (s == nil) break; if (i > 0) { (n,lst) := sys->tokenize(s,"\t\n:"); val := string int hd tl lst; if (i == 2 && val == "0") tkcmd(win, ".f.fsnap.b configure -state disabled"); else tkcmd(win, ".f.fsnap.b configure -state normal"); tkcmd(win,".f.fsnap.fstore.l"+string (2+i)+" configure -text {"+val+" }"); } } tkcmd(win, "update"); } } contains(s: string, test: string): int { num :=0; if (len test > len s) return 0; for (i := 0; i < (1 + (len s) - (len test)); i++) { if (test == s[i:i+len test]) num++; } return num; } multidownload() { getpath := selectfile->filename(context, display.image, "Multiple download to directory...", nil, lastpath); if (getpath == "" || getpath[0] != '/' || getpath[len getpath - 1] != '/') return; s := ""; for (k := 0; k < nofiles; k++) { if (selected[k]) { e := dnld(k,getpath); if (e != 1) s += filelist[k]+".jpg "; if (e == 3) { s += "cancelled\n"; break; } else if (e == 0) s += "failed\n"; working = 0; } } if (s != "") s = ":\n\n"+s; dialog("Multiple download complete"+s,0,-1,coords); } downloading := ""; dnld(i: int, path: string): int { ctlchan := chan of int; ctlchans := chan of string; chanout := chan of string; spawn downloadscreen(coords, i, ctlchans, chanout); spawn download(i,path,ctlchan, ctlchans, chanout); pid := <-ctlchan; alt { s := <-ctlchans => chanout <-= "!done!"; if (s == "kill") { if (downloading != "") { (n,lst) := sys->tokenize(downloading, " \t\n"); for(;lst != nil; lst = tl lst) sys->remove(hd lst); } kill(pid); return 3; } else return dnld(i, "!"+s); e := <-ctlchan => chanout <-= "!show!"; chanout <-= "!done!"; return e; } return 0; } filelenrefresh(filename: string): int { fd := sys->open(camerapath+"ctl",sys->OWRITE); if (fd != nil) { sys->fprint(fd, "refresh"); (n, dir) := sys->stat(filename); if (n == -1) return -1; return int dir.length; } return -1; } testfilesize(filename: string): int { e := filelenrefresh(filename); if (e == 0) { e2 := dialog("Camera is still processing image\nwait until ready?",1,-1,coords); if (e2 == 0) return 0; ctlchan := chan of int; spawn waittilready(filename, ctlchan); e3 := <- ctlchan; working = 0; if (e3 == KILLED) return 0; return testfilesize(filename); } else return e; } waittilready(filename: string, ctlchan: chan of int) { pid := sys->pctl(0, nil); spawn workingscreen2(coords,pid,ctlchan,0); for (;;) { if (filelenrefresh(filename) != 0) break; sys->sleep(2000); } ctlchan <-= DONE; } download(i: int, path: string, ctlchan: chan of int, ctlchans, chanout: chan of string) { ctlchan <-= sys->pctl(0, nil); downloading = ""; savename : string; if (path == "") { savename = selectfile->filename(context, display.image, "Save "+filelist[i]+".jpg to directory...", "*.jpg" :: "*.jpeg" :: nil, lastpath); if (savename == "" || savename[0] != '/') { ctlchan <-= 0; return; } } else savename = path; # Used when retrying due to cache copy failing if (savename[0] == '!') { delloaded(filelist[i],JPG); savename = savename[1:]; path = ""; } confirm := 1; # Don't confirm overwrite if (savename[0] == '$') { confirm = 0; savename = savename[1:]; } if (savename[len savename - 1] == '/') savename += filelist[i]+".jpg"; if (!hasext(savename, ".jpg")) savename += ".jpg"; p := isat2(savename,"/"); lastpath = savename[:p+1]; filename := camerapath+"jpg/"+filelist[i]+".jpg"; filesize := testfilesize(filename); cached := 0; if (filesize > 0 && isloaded(filelist[i],JPG) && usecache) { cachefilename := tmppath+filelist[i]+"."+string JPG+"~"; if (testfilesize(cachefilename) == filesize) { cached = 1; filename = cachefilename; } else delloaded(filelist[i],JPG); } fd := sys->open(filename, sys->OREAD); if (filesize < 1 || fd == nil) { ctlchan <-= -1; return; } read := 0; cancel : int; buf : array of byte; fd2, fd3 : ref sys->FD = nil; cachename := tmppath+filelist[i]+"."+string JPG+"~"; if (confirm) (fd2, cancel) = create(savename, coords); else fd2 = sys->create(savename,sys->OWRITE, 8r666); if (fd2 == nil) { ctlchan <-= cancel; return; } if (usecache && !cached) fd3 = sys->create(cachename,sys->OWRITE,8r666); chanout <-= "!show!"; chanout <-= "l2 Downloading..."; chanout <-= "pc 0"; n : int; downloading = savename; if (fd3 != nil) downloading += " "+cachename; loop: for(;;) { rlen := 8192; if (read + rlen >= filesize) rlen = filesize - read; buf = array[rlen] of byte; n = sys->read(fd,buf,len buf); read += n; sout := "pc "+string ( (100*read)/filesize); chanout <-= sout; if (n < 1) break loop; written := 0; while (written < n) { n2 := sys->write(fd2,buf,n); if (n2 < 1) break loop; if (fd3 != nil) sys->write(fd3,buf,n); written += n2; } } chanout <-= "pc 100"; downloading = ""; fd = nil; fd2 = nil; if (read < filesize || n == -1) { if (cached) { ctlchans <-= savename; return; } sys->remove(savename); sys->remove(cachename); if (path == "") dialog(sys->sprint("Download Failed: %s.jpg\nread %d of %d bytes\n", filelist[i],read,filesize), 0, i,coords); ctlchan <-= 0; return; } # save it in cache if (usecache) imgloaded = (filelist[i],JPG) :: imgloaded; if (path == "") dialog(filelist[i]+".jpg downloaded",0,i,coords); ctlchan <-= 1; } downloadscr := array[] of { "frame .f -borderwidth 2 -relief raised", "label .f.l1 -text { } @", "label .f.l2 -text {Waiting...} @", "button .f.b -text {Cancel} -command {send ctlchans kill} &", "grid .f.l1 -row 0 -column 0 -columnspan 2 -pady 5", "grid .f.l2 -row 2 -column 1 -sticky w -padx 10", "grid .f.p -row 3 -column 1 -columnspan 1 -padx 10", "grid .f.b -row 4 -column 0 -pady 5 -columnspan 2", }; downloadscreen(r: Rect, i: int, ctlchans, chanin: chan of string) { working = 1; <- chanin; (top, nil) := tkclient->toplevel(context,"", nil, tkclient->Plain); progr := Rect((0,0),(100,15)); imgbg := display.newimage(progr,draw->CMAP8,1,draw->Black); black := display.newimage(progr,draw->CMAP8,1,draw->Black); white := display.newimage(progr,draw->CMAP8,1,draw->White); imgfg := display.newimage(progr,draw->CMAP8,1,draw->Blue); tkcmd(top, "panel .f.p -width 100 -height 15 -bg white -borderwidth 2 - relief raised"); tk->putimage(top, ".f.p",imgbg,nil); tk->namechan(top, ctlchans, "ctlchans"); for (tk1 := 0; tk1 < len downloadscr; tk1++) tkcmd(top, downloadscr[tk1]); tmpimg : ref Image = nil; if (i >= 0 && isloaded(filelist[i], THUMB) && usecache) tmpimg = display.open(tmppath+filelist[i]+"."+string THUMB+"~"); if (tmpimg == nil) tmpimg = procimg; if (tmpimg != nil) { w := tmpimg.r.dx(); h := tmpimg.r.dy(); tkcmd(top, "panel .f.p2 -width "+string w+" -height "+string h+ " -borderwidth 2 -relief raised"); tk->putimage(top, ".f.p2", tmpimg, nil); tkcmd(top, "grid .f.p2 -row 2 -column 0 -rowspan 2 -sticky e"); tkcmd(top, "grid columnconfigure .f 0 -minsize "+string (w + 14)); } tkcmd(top, ".f.l1 configure -text {"+filelist[i]+".jpg}"); centrewin(top,r,1); oldcoords := coords; tkclient->onscreen(top, "exact"); tkclient->startinput(top, "kbd"::"ptr"::nil); main: for (;;) { alt { s := <-top.ctxt.kbd => tk->keyboard(top, s); s := <-top.ctxt.ptr => tk->pointer(top, *s); text := <-chanin => if (!oldcoords.eq(coords)) { centrewin(top,coords,0); oldcoords = coords; } if (text == "!done!") break main; if (text[:2] == "pc") { val := int text[3:]; imgbg.draw(((0,0),(val,15)), imgfg,nil,(0,0)); if (val != 100) imgbg.draw(((val+1,0),(100,15)), black,nil,(0,0)); imgbg.text((42,1),white,(0,0),font, text[3:]+"%"); tkcmd(top,".f.p dirty; update"); } else if (text[:2] == "l2") tkcmd(top, ".f.l2 configure -text {"+text[3:]+"}; update"); } } working = 0; } centrewin(top: ref Tk->Toplevel, r: Rect, first: int) { s := ""; if (first) s = "pack .f;"; w := int tkcmd(top, ".f cget -width"); h := int tkcmd(top, ".f cget -height"); tmp := tk->cmd(top, ".Wm_t cget -height"); if (tmp != "" && tmp[0] != '!') { h += int tmp; s += "focus .;"; } px := r.min.x + ((r.max.x - r.min.x - w) / 2); py := r.min.y + ((r.max.y - r.min.y - h) / 2); tkcmd(top, ". configure -x "+string px+" -y "+string py); tkcmd(top, s+"raise .; update"); } workingscr2 := array[] of { "frame .f -borderwidth 2 -relief raised", "label .f.l3 -text { } -width 220 -height 2", "label .f.l -text {Please Wait} @", "label .f.l2 -text {|} -width 20 @", "button .f.b -text {Cancel} -command {send chanin kill} &", "grid .f.l -row 1 -column 0 -sticky e", "grid .f.l2 -row 1 -column 1 -sticky w", "grid .f.b -pady 5 -row 3 -column 0 -columnspan 2", "grid .f.l3 -row 4 -column 0 -columnspan 2", "grid rowconfigure .f 1 -minsize 80", }; workingscreen2(r : Rect, pid: int, ctlchan: chan of int, loading: int) { (top, nil) := tkclient->toplevel(context,"",nil, tkclient->Plain); chanin := chan of string; tk->namechan(top, chanin, "chanin"); for (tk1 := 0; tk1 < len workingscr2; tk1++) tkcmd(top, workingscr2[tk1]); if (loading) { # loadimg := display.open("camload.bit"); if (loadimg != nil) { w := loadimg.r.dx(); h := loadimg.r.dy(); tkcmd(top, "panel .f.p -width "+string w+" -height "+string h+ " -borderwidth 2 -relief raised"); tk->putimage(top, ".f.p", loadimg, nil); tkcmd(top, "grid .f.p -row 2 -column 0 -columnspan 2 -pady 5 -padx 20"); tkcmd(top, "grid forget .f.l .f.l2; grid rowconfigure .f 1 -minsize 20"); } } else { if (procimg != nil) { w := procimg.r.dx(); h := procimg.r.dy(); tkcmd(top, "panel .f.p -width "+string w+" -height "+string h+ " -borderwidth 2 -relief raised"); tk->putimage(top, ".f.p", procimg, nil); tkcmd(top, "grid .f.p -row 2 -column 0 -columnspan 2"); tkcmd(top, "grid rowconfigure .f 1 -minsize 30"); tkcmd(top, "grid rowconfigure .f 2 -minsize 50"); } } centrewin(top,r,1); spawn workingupdate(top,chanin); tkclient->onscreen(top, "exact"); tkclient->startinput(top, "kbd"::"ptr"::nil); main: for (;;) { alt { s := <-top.ctxt.kbd => tk->keyboard(top, s); s := <-top.ctxt.ptr => tk->pointer(top, *s); inp := <-chanin => if (inp == "done") break main; if (inp == "kill") { working = 0; if (pid != -1) kill(pid); ctlchan <-= KILLED; <-chanin; break main; } } } } workingupdate(top: ref Tk->Toplevel, chanout: chan of string) { show := array[] of { "/", "-", "\\\\", "|", }; if (working) { chanout <-= "done"; return; } working = 1; oldcoords := coords; hidden := 0; loop: for(;;) { for (i := 0; i < 4; i++) { sys->sleep(100); tkcmd(top, ".f.l2 configure -text {"+show[i]+"}; update"); if (!working) break loop; if (!oldcoords.eq(coords)) { centrewin(top, coords,0); oldcoords = coords; } } } chanout <-= "done"; } scrollx := 0; scrolly := 0; resizemain(top: ref Tk->Toplevel, init: int) { h, w: int; if (init) { growheight(top, 4000); h = int tkcmd(top, ".f.fsnap cget -height") + int tkcmd(top, ".Wm_t cget -height") + 2 * (124 - (5*(3-ssize))); if (h > display.image.r.dy()) h = display.image.r.dy(); w = display.image.r.dx(); } else { r := tk->rect(top, ".", 0); h = r.dy(); w = r.dx(); } ht := int tkcmd(top, ".Wm_t cget -height"); hf := int tkcmd(top, ".f cget -height"); wf := int tkcmd(top, ".f cget -width"); wsb := int tkcmd(top, ".f.f1.sb1 cget -width"); growwidth(top, w - 4); ws := int tkcmd(top, ".f.fsnap cget -width"); if (w > ws + 4) w = ws + 4; shrinkwidth(top,w - 4); ws = int tkcmd(top, ".f.fsnap cget -width"); if (w < ws || init) w = ws + 4; hmax := ((3*(h - ht))/5) - 4; growheight(top, hmax); shrinkheight(top, hmax); hs := int tkcmd(top, ".f.fsnap cget -height"); hmb := int tkcmd(top, ".f.fsnap.fsettings.mb cget -height"); if (h < ht+hs + 107 + hmb) h = ht+hs+107 + hmb; # hc2 = int tkcmd(top, ".f.fsnap cget -height"); wc2 := int tkcmd(top, ".f.fsnap cget -width"); hc1 := h - ht - hs - 4; wc1 := w-wsb-4; # wc1 = wc2 - wsb; tkcmd(top, ".f.f1.c1 configure -height "+string hc1+" -width "+string wc1); # tkcmd(top, ".f.f2.c1 configure -height "+string hc2+" -width "+string wc2); if (w < wc2 + 4) w = wc2 + 4; ws = int tkcmd(top, ".f.fsnap cget -width"); hs = int tkcmd(top, ".f.fsnap cget -height"); tkcmd(top, ". configure -height "+string h+" -width "+string w+"; update"); refreshfilelist(top, 0); } growwidth(top: ref Tk->Toplevel, wc2: int) { ws := int tkcmd(top, ".f.fsnap cget -width"); if (wc2 > ws && reducew[2]) { tkcmd(top, ".f.fsnap.ftime.l1 configure -text {Local:}"); tkcmd(top, ".f.fsnap.ftime.l2 configure -text {Camera:}"); tkcmd(top, ".f.fsnap.ftime.cb configure -text {Set to local time}"); reducew[2] = 0; } ws = int tkcmd(top, ".f.fsnap cget -width"); if (wc2 > ws && reducew[1]) { tkcmd(top, ".f.fsnap.ftime.l3 configure -text {"+ltime+"}"); tkcmd(top, ".f.fsnap.ftime.l4 configure -text {"+ctime+"}"); tkcmd(top, ".f.fsnap.ftime.cb configure -text {Set camera to local time}"); reducew[1] = 0; } ws = int tkcmd(top, ".f.fsnap cget -width"); if (wc2 > ws && reducew[0]) { tkcmd(top, ".f.fsnap.fstore.l1 configure -text { Photos taken:}"); reducew[0] = 0; } ws = int tkcmd(top, ".f.fsnap cget -width"); if (wc2 > ws) { wfs += wc2 - ws; if (wfs > 125-(20*(3-ssize))) wfs = 125-(20*(3-ssize)); tkcmd(top, "grid columnconfigure .f.fsnap 2 -minsize "+string wfs); } } growheight(top: ref Tk->Toplevel, hc2: int) { hs := int tkcmd(top, ".f.fsnap cget -height"); if (hc2 > hs) { tk->cmd(top, "grid .f.fsnap.fsettings.mb2 -row 2 -column 0 -sticky ew"); tk->cmd(top, "grid .f.fsnap.ftime.cb -row 2 -column 0 -columnspan 2"); tk->cmd(top, "grid .f.fsnap.ftime.b -row 3 -column 0 -columnspan 2"); } hs = int tkcmd(top, ".f.fsnap cget -height"); if (hc2 > hs) { hsc := int tkcmd(top, ".f.fsnap.fzpos.sc cget -height"); hsc += hc2 - hs; if (hsc > 88-(10*(3-ssize))) hsc = 88-(10*(3-ssize)); tkcmd(top, ".f.fsnap.fzpos.sc configure -height "+string hsc); } hs = int tkcmd(top, ".f.fsnap cget -height"); if (hc2 > hs) { hfs += hc2 - hs; if (hfs > 30 - (5*(3-ssize))) hfs = 30- (5*(3-ssize)); tkcmd(top, "grid rowconfigure .f.fsnap 0 -minsize "+string hfs); tkcmd(top, "grid rowconfigure .f.fsnap 2 -minsize "+string hfs); tkcmd(top, "grid rowconfigure .f.fsnap 4 -minsize "+string hfs); } } shrinkheight(top: ref Tk->Toplevel, hc2: int) { hs := int tkcmd(top, ".f.fsnap cget -height"); if (hc2 < hs) { hfs -= hs - hc2; if (hfs < 15) hfs = 15; tkcmd(top, "grid rowconfigure .f.fsnap 0 -minsize "+string hfs); tkcmd(top, "grid rowconfigure .f.fsnap 2 -minsize "+string hfs); tkcmd(top, "grid rowconfigure .f.fsnap 4 -minsize "+string hfs); } hs = int tkcmd(top, ".f.fsnap cget -height"); if (hc2 < hs) { hsc := int tkcmd(top, ".f.fsnap.fzpos.sc cget -height"); hsc -= hs - hc2; if (hsc < 55-(5*(3-ssize))) hsc = 55-(5*(3-ssize)); tkcmd(top, ".f.fsnap.fzpos.sc configure -height "+string hsc); } hs = int tkcmd(top, ".f.fsnap cget -height"); if (hc2 < hs) { tk->cmd(top, "grid forget .f.fsnap.fsettings.mb2"); tk->cmd(top, "grid forget .f.fsnap.ftime.cb"); tk->cmd(top, "grid forget .f.fsnap.ftime.b"); } } shrinkwidth(top: ref Tk->Toplevel, wc2: int) { ws := int tkcmd(top, ".f.fsnap cget -width"); wib := int tkcmd(top, ".f.fsnap.fsettings.b cget -width"); if (wc2 < ws) { diff := ws - wc2; wfs -= diff; if (wfs < wib) wfs = wib; tkcmd(top, "grid columnconfigure .f.fsnap 2 -minsize "+string wfs); } ws = int tkcmd(top, ".f.fsnap cget -width"); if (wc2 < ws) { tkcmd(top, ".f.fsnap.fstore.l1 configure -text { Taken:}"); reducew[0] = 1; } ws = int tkcmd(top, ".f.fsnap cget -width"); if (wc2 < ws) { tkcmd(top, ".f.fsnap.ftime.l3 configure -text {"+ltime[len ltime - 8:]+"}"); tkcmd(top, ".f.fsnap.ftime.l4 configure -text {"+ctime[len ctime - 8:]+"}"); tkcmd(top, ".f.fsnap.ftime.cb configure -text {Set to local time}"); reducew[1] = 1; } ws = int tkcmd(top, ".f.fsnap cget -width"); if (wc2 < ws) { tkcmd(top, ".f.fsnap.ftime.l1 configure -text {C:}"); tkcmd(top, ".f.fsnap.ftime.l2 configure -text {}"); tkcmd(top, ".f.fsnap.ftime.l3 configure -text {"+ctime[len ctime - 17:len ctime - 8]+"}"); tkcmd(top, ".f.fsnap.ftime.cb configure -text {Set local}"); reducew[2] = 1; } ws = int tkcmd(top, ".f.fsnap cget -width"); if (wc2 > ws) { wfs = 125-(20*(3-ssize)); tkcmd(top, "grid columnconfigure .f.fsnap 2 -minsize "+string wfs); } } ltime, ctime: string; wfs := 150; hfs := 30; reducew := array[10] of { * => 0 }; getcoords(top: ref Tk->Toplevel): Rect { h := int tkcmd(top, ". cget -height"); w := int tkcmd(top, ". cget -width"); x := int tkcmd(top, ". cget -actx"); y := int tkcmd(top, ". cget -acty"); r := Rect((x,y),(x+w,y+h)); return r; } viewscr := array[] of { "frame .f -bg", "canvas .f.c -yscrollcommand {.f.sy set} -xscrollcommand {.f.sx set} -height 300 -width 500", "scrollbar .f.sx -command {.f.c xview} -orient horizontal", "scrollbar .f.sy -command {.f.c yview}", "grid .f.c -row 0 -column 0", "grid .f.sy -row 0 -column 1 -sticky ns", "grid .f.sx -row 1 -column 0 -sticky ew", "bind .Wm_t <ButtonPress-1> +{focus .}", "bind .Wm_t.title <ButtonPress-1> +{focus .}", "pack propagate . 0", "menu .m @", ".m add command -text {Save As...}", ".m add separator", ".m add command -text {bit} -command {send butchan save bit}", ".m add command -text {jpeg} -command {send butchan save jpg}", }; resizeview(top: ref Tk->Toplevel, wp,hp: int) { w := int tkcmd(top, ". cget -width"); h := int tkcmd(top, ". cget -height"); hs := int tkcmd(top, ".f.sx cget -height"); ws := int tkcmd(top, ".f.sy cget -width"); ht := int tkcmd(top, ".Wm_t cget -height"); wc := w - ws - 4; hc := h - hs - ht - 6; wpc := wc - wp; hpc := hc - hp; if (wpc > 0) { wc -= wpc; w -= wpc; } if (hpc > 0) { hc -= hpc; h -= hpc; } tkcmd(top, ". configure -height "+string h+" -width "+string w); tkcmd(top, ".f.c configure -height "+string hc+" -width "+string wc); tkcmd(top, "update"); } multiview() { s := ""; for (k := 0; k < nofiles; k++) { if (selected[k]) { e := vw(k); if (e != 0) s += filelist[k]+".jpg "; if (e == 3) { s += "cancelled\n"; break; } else if (e == -1) s += "failed\n"; } } if (s != "") dialog("Multiple view complete:\n\n"+s,0,-1,coords); } vw(i: int): int { # raise window if it is already open low := toplevels; for(; low != nil; low = tl low) { (tplvl, name, nil, nil) := hd low; if (filelist[i]+".jpg" == name) { tkcmd(tplvl, "raise .; focus .; update"); return 0; } } ctlchan := chan of int; ctlchans := chan of string; chanout := chan of string; chanin := chan of string; spawn downloadscreen(coords, i, ctlchans, chanout); chanout <-= "!show!"; spawn view(i,ctlchan, chanin, chanout); pid := <-ctlchan; killed := 0; for (;;) alt { s := <-ctlchans => if (s == "kill") { chanin <-= "kill"; killed = 1; } e := <-ctlchan => chanout <-= "!done!"; if (killed) return 3; if (e == -1) dialog(sys->sprint("Cannot read file: %s.jpg\n%r",filelist[i]),0,i,coords); if (e == -2) return vw(i); else return e; } return 0; } view(i: int, ctlchan: chan of int, chanin, chanout: chan of string) { ctlchan <-= sys->pctl(0, nil); titlename := filelist[i]+".jpg"; filename := camerapath+"jpg/"+filelist[i]+".jpg"; filesize := testfilesize(filename); cached := 0; if (filesize > 0 && isloaded(filelist[i],JPG) && usecache) { cachefilename := tmppath+filelist[i]+"."+string JPG+"~"; if (testfilesize(cachefilename) == filesize) { cached = 1; filename = cachefilename; } else delloaded(filelist[i],JPG); } if (filesize < 1) { ctlchan <-= -1; return; } img: ref Image; cachepath := ""; if (!cached && usecache) cachepath = tmppath+filelist[i]+"."+string JPG+"~"; img = readjpg->jpg2img(filename, cachepath, chanin, chanout); if(img == nil) { if (cachepath != nil) sys->remove(cachepath); if (!cached) ctlchan <-= -1; else { delloaded(filelist[i], JPG); ctlchan <-= -2; } return; } else { chanout <-= "l2 Displaying"; if (cachepath != "") imgloaded = (filelist[i], JPG) :: imgloaded; (t, titlechan) := tkclient->toplevel(context, "", titlename, Tkclient->Appl); butchan := chan of string; tk->namechan(t, butchan, "butchan"); tkcmd(t, "focus .Wm_t; update"); for (tk1 := 0; tk1 < len viewscr; tk1++) tkcmd(t, viewscr[tk1]); w := img.r.dx(); h := img.r.dy(); tkcmd(t, "panel .p -width "+string w+" -height "+string h); tk->putimage(t, ".p",img,nil); tkcmd(t, "bind .p <ButtonPress-2> {send butchan move %X %Y}"); tkcmd(t, "bind .p <ButtonRelease-2> {send butchan release}"); tkcmd(t, "bind .p <ButtonPress-3> {send butchan menu %X %Y}"); tkcmd(t, ".f.c create window 0 0 -window .p -anchor nw"); tkcmd(t, ".f.c configure -scrollregion {0 0 "+string w+" "+string h+"}"); ctlchan <-= 0; addtoplevel(t,titlename,nil, sys->pctl(0,nil)); h1 := 300; w1 := 500; ht := int tkcmd(t, ".Wm_t cget -height"); if (h1 > display.image.r.dy() - ht) h1 = display.image.r.dy() - ht; if (w1 > display.image.r.dx()) w1 = display.image.r.dx(); tkcmd(t, ". configure -width "+string w1+" -height "+string h1); resizeview(t,w,h); tkcmd(t, "pack .f; update"); scrolling := 0; origin := Point (0,0); tkclient->onscreen(t, nil); tkclient->startinput(t, "kbd"::"ptr"::nil); loop: for(;;) alt{ s := <-t.ctxt.kbd => tk->keyboard(t, s); s := <-t.ctxt.ptr => tk->pointer(t, *s); inp := <- butchan => (n, lst) := sys->tokenize(inp, " \t\n"); case hd lst { "save" => ftype := "."+hd tl lst; savename := selectfile->filename(context, display.image, "Save "+filelist[i]+ftype+" to directory...", "*"+ftype :: nil, lastpath); if (savename != "" && savename[0] == '/') { lastpath = savename[:isat2(savename,"/")+1]; if (savename[len savename - 1] == '/') savename += filelist[i]+ftype; if (!hasext(savename, ftype)) savename += ftype; (fd, cancel) := create(savename, getcoords(t)); if (fd != nil) { n2 := -1; if (ftype == ".bit") n2 = display.writeimage(fd,img); if (ftype == ".jpg") n2 = 1 - dnld(i, "$"+savename); if (n2 == 0) { dialog(filelist[i]+ftype+" saved",0,i,getcoords(t)); break; } dialog("Could not save: "+filelist[i]+ftype,0,i,getcoords(t)); } if (!cancel) dialog("Could not save: "+filelist[i]+ftype,0,i,getcoords(t)); break; } "menu" => tkcmd(t, ".m post "+hd tl lst+" "+hd tl tl lst); "release" => scrolling = 0; "move" => newpoint := Point (int hd tl lst, int hd tl tl lst); if (scrolling) { diff := (origin.sub(newpoint)).mul(2); tkcmd(t, ".f.c xview scroll "+string diff.x+" units"); tkcmd(t, ".f.c yview scroll "+string diff.y+" units"); origin = newpoint; # clearbuffer(butchan); } else { origin = newpoint; scrolling = 1; } } s := <-t.ctxt.ctl or s = <-t.wreq or s = <-titlechan => if (s == "exit") break loop; e := tkclient->wmctl(t, s); if (e == nil && s[0] == '!') resizeview(t,w,h); } deltoplevel(t); } } create(filename: string, co: Rect): (ref sys->FD, int) { (n,dir) := sys->stat(filename); if (n != -1 && !dialog("overwrite "+filename+"?",1,-1,co)) return (nil,1); return (sys->create(filename,sys->OWRITE,8r666), 0); } hasext(name,ext: string): int { if (len name >= len ext && name[len name - len ext:] == ext) return 1; return 0; }