implement Coffee; include "sys.m"; sys: Sys; include "draw.m"; draw: Draw; Context, Display, Point, Rect, Image, Screen: import draw; include "tk.m"; tk: Tk; Toplevel: import tk; include "wmlib.m"; wmlib: Wmlib; Coffee: module { init: fn(ctxt: ref Context, argv: list of string); }; screen: ref Screen; display: ref Display; t: ref Toplevel; NC: con 6; task_cfg := array[] of { "frame .f -height 306 -width 406", "frame .b", "button .b.Stop -text Stop -command {send cmd stop}", "scale .b.Rate -from 1 -to 10 -orient horizontal"+ " -showvalue 0 -command {send cmd rate}", "scale .b.Jitter -from 0 -to 5 -orient horizontal"+ " -showvalue 0 -command {send cmd jitter}", "scale .b.Skip -from 0 -to 25 -orient horizontal"+ " -showvalue 0 -command {send cmd skip}", ".b.Rate set 3", ".b.Jitter set 2", ".b.Skip set 5", "pack .b.Stop .b.Rate .b.Jitter .b.Skip -side left", "pack .b -anchor w", "pack .f -side bottom -fill both -expand 1", "pack propagate . 0", }; init(ctxt: ref Context, nil: list of string) { sys = load Sys Sys->PATH; draw = load Draw Draw->PATH; tk = load Tk Tk->PATH; wmlib = load Wmlib Wmlib->PATH; display = ctxt.display; screen = ctxt.screen; wmlib->init(); menubut: chan of string; (t, menubut) = wmlib->titlebar(screen, "", "Infernal Coffee", 0); cmd := chan of string; tk->namechan(t, cmd, "cmd"); wmlib->tkcmds(t, task_cfg); tk->cmd(t, "bind . {send cmd resize}"); tk->cmd(t, "update"); r := frameposn(t); ctl := chan of (string, int, int); spawn animate(r, ctl); for(;;) alt { menu := <-menubut => if(menu == "exit") { ctl <-= ("exit", 0, 0); return; } wmlib->titlectl(t, menu); press := <-cmd => (n, word) := sys->tokenize(press, " "); case hd word { "stop" or "go" => ctl <-= (hd word, 0, 0); "rate" or "jitter" or "skip" => ctl <-= (hd word, int hd tl word, 0); "resize" => r = frameposn(t); ctl <-= (hd word, r.min.x, r.min.y); } } } frameposn(t: ref Toplevel): Rect { r: Rect; r.min.x = int tk->cmd(t, ".f cget -actx") + int tk->cmd(t, ".f cget -bd"); r.min.y = int tk->cmd(t, ".f cget -acty") + int tk->cmd(t, ".f cget -bd"); r.max.x = r.min.x + int tk->cmd(t, ".f cget -actwidth"); r.max.y = r.min.y + int tk->cmd(t, ".f cget -actheight"); return r; } animate(r: Rect, ctl: chan of (string, int, int)) { stopped := 0; fill := display.open("/icons/bigdelight.bit"); c := array[NC] of ref Image; m := array[NC] of ref Image; for(i:=0; iBlack); org := buffer.r.min; hell := t.image; rate := 3; jitter := 2; skip := 5; i = 0; for(k:=0; ; k++){ sys->sleep(1); if(k%25 > 25-skip) i -= rate; else i += rate; buffer.draw(buffer.clipr, fill, nil, fill.r.min); center := buffer.r.min.add(buffer.r.max).div(2); for(j:=0; jsleep(0); } hell.draw((org, (org.x+buffer.r.dx(), org.y+buffer.r.dy())), buffer, nil, buffer.r.min); sys->sleep(5); alt{ (cmd, i0, i1) := <-ctl => Pause: for(;;){ case cmd{ "exit" => return; "go" => if(stopped){ tk->cmd(t, ".b.Stop configure -text Stop -command {send cmd stop}"); tk->cmd(t, "update"); stopped = 0; } break Pause; "stop" => if(!stopped){ tk->cmd(t, ".b.Stop configure -text { Go } -command {send cmd go}"); tk->cmd(t, "update"); stopped = 1; } "rate" => rate = i0; if(stopped == 0) break Pause; "jitter" => jitter = i0; if(stopped == 0) break Pause; "skip" => skip = i0; if(stopped == 0) break Pause; "resize" => org = (i0+3, i1+3); hell = t.image; hell.draw((org, (org.x+buffer.r.dx(), org.y+buffer.r.dy())), buffer, nil, buffer.r.min); if(stopped == 0) break Pause; } (cmd, i0, i1) = <-ctl; } * => ; } } } sintab := array[] of { 0000, 0017, 0035, 0052, 0070, 0087, 0105, 0122, 0139, 0156, 0174, 0191, 0208, 0225, 0242, 0259, 0276, 0292, 0309, 0326, 0342, 0358, 0375, 0391, 0407, 0423, 0438, 0454, 0469, 0485, 0500, 0515, 0530, 0545, 0559, 0574, 0588, 0602, 0616, 0629, 0643, 0656, 0669, 0682, 0695, 0707, 0719, 0731, 0743, 0755, 0766, 0777, 0788, 0799, 0809, 0819, 0829, 0839, 0848, 0857, 0866, 0875, 0883, 0891, 0899, 0906, 0914, 0921, 0927, 0934, 0940, 0946, 0951, 0956, 0961, 0966, 0970, 0974, 0978, 0982, 0985, 0988, 0990, 0993, 0995, 0996, 0998, 0999, 0999, 1000, 1000, }; sincos(a: int): (int, int) { a %= 360; if(a < 0) a += 360; if(a <= 90) return (sintab[a], sintab[90-a]); if(a <= 180) return (sintab[180-a], -sintab[a-90]); if(a <= 270) return (-sintab[a-180], -sintab[270-a]); return (-sintab[360-a], sintab[a-270]); }