ref: 6d1aff1af30525e3dccc1ba6cb51704d0168f1d6
dir: /vmodeled/main.c/
#include <u.h> #include <libc.h> #include <bio.h> #include <thread.h> #include <draw.h> #include <mouse.h> #include <keyboard.h> #include <geometry.h> /* * Vector model - made out of lines and curves */ typedef struct VModel VModel; struct VModel { Point2 *pts; ulong npts; /* WIP * l(ine) → takes 2 points * c(urve) → takes 3 points */ char *strokefmt; }; typedef struct Object Object; struct Object { RFrame; VModel *mdl; }; RFrame worldrf; Object mainobj; double θ; double scale = 1; void resized(void); void* emalloc(ulong n) { void *p; p = malloc(n); if(p == nil) sysfatal("malloc: %r"); setmalloctag(p, getcallerpc(&n)); return p; } void* erealloc(void *p, ulong n) { void *np; np = realloc(p, n); if(np == nil){ if(n == 0) return nil; sysfatal("realloc: %r"); } if(p == nil) setmalloctag(np, getcallerpc(&p)); else setrealloctag(np, getcallerpc(&p)); return np; } Point toscreen(Point2 p) { p = invrframexform(p, worldrf); return Pt(p.x,p.y); } Point2 fromscreen(Point p) { return rframexform(Pt2(p.x,p.y,1), worldrf); } void rotateobj(double θ) { Matrix R = { cos(θ), -sin(θ), 0, sin(θ), cos(θ), 0, 0, 0, 1, }; mainobj.bx = xform(mainobj.bx, R); mainobj.by = xform(mainobj.by, R); } void zoomobj(double z) { mainobj.bx = mulpt2(mainobj.bx, z); mainobj.by = mulpt2(mainobj.by, z); } VModel * readvmodel(char *file) { ulong lineno; char *s, *args[2]; Biobuf *bin; VModel *mdl; bin = Bopen(file, OREAD); if(bin == nil) sysfatal("Bopen: %r"); mdl = emalloc(sizeof(VModel)); mdl->pts = nil; mdl->npts = 0; mdl->strokefmt = nil; lineno = 0; while(s = Brdline(bin, '\n')){ s[Blinelen(bin)-1] = 0; lineno++; switch(*s++){ case '#': continue; case 'v': if(tokenize(s, args, nelem(args)) != nelem(args)){ werrstr("syntax error: %s:%lud 'v' expects %d args", file, lineno, nelem(args)); free(mdl); Bterm(bin); return nil; } mdl->pts = erealloc(mdl->pts, ++mdl->npts*sizeof(Point2)); mdl->pts[mdl->npts-1].x = strtod(args[0], nil); mdl->pts[mdl->npts-1].y = strtod(args[1], nil); mdl->pts[mdl->npts-1].w = 1; break; case 'l': case 'c': mdl->strokefmt = strdup(s-1); break; } } Bterm(bin); return mdl; } int writevmodel(VModel *mdl, char *file) { USED(mdl); USED(file); return -1; } void drawvmodel(Image *dst, VModel *mdl, int t) { int i; char *s; Point pts[3]; Point2 *p; Matrix S = { 1/scale, 0, 0, 0, 1/scale, 0, 0, 0, 1, }, R = { cos(θ), -sin(θ), 0, sin(θ), cos(θ), 0, 0, 0, 1, }; if(t) mulm(S, R); p = mdl->pts; for(s = mdl->strokefmt; s != 0 && p-mdl->pts < mdl->npts; s++) switch(*s){ case 'l': if(t) line(dst, toscreen(invrframexform(xform(p[0], S), mainobj)), toscreen(invrframexform(xform(p[1], S), mainobj)), 0, 0, 0, display->white, ZP); else line(dst, toscreen(invrframexform(p[0], mainobj)), toscreen(invrframexform(p[1], mainobj)), 0, 0, 0, display->white, ZP); p += 2; break; case 'c': for(i = 0; i < nelem(pts); i++) if(t) pts[i] = toscreen(invrframexform(xform(p[i], S), mainobj)); else pts[i] = toscreen(invrframexform(p[i], mainobj)); bezspline(dst, pts, nelem(pts), 0, 0, 0, display->white, ZP); p += 3; break; } } void drawaxes(void) { line(screen, toscreen(Pt2(0,512,1)), toscreen(Pt2(0,-512,1)), 0, 0, 0, display->white, ZP); line(screen, toscreen(Pt2(512,0,1)), toscreen(Pt2(-512,0,1)), 0, 0, 0, display->white, ZP); } void drawinfo(void) { Point p; char buf[128]; p = Pt(10,3); snprint(buf, sizeof buf, "wbx %v", worldrf.bx); string(screen, addpt(screen->r.min, p), display->white, ZP, font, buf); p.y += font->height; snprint(buf, sizeof buf, "wby %v", worldrf.by); string(screen, addpt(screen->r.min, p), display->white, ZP, font, buf); p.y += font->height; snprint(buf, sizeof buf, "s %g", scale); string(screen, addpt(screen->r.min, p), display->white, ZP, font, buf); p.y += font->height; snprint(buf, sizeof buf, "θ %g", θ); string(screen, addpt(screen->r.min, p), display->white, ZP, font, buf); p.y += font->height; snprint(buf, sizeof buf, "obx %v", mainobj.bx); string(screen, addpt(screen->r.min, p), display->white, ZP, font, buf); p.y += font->height; snprint(buf, sizeof buf, "oby %v", mainobj.by); string(screen, addpt(screen->r.min, p), display->white, ZP, font, buf); } void redraw(int t) { lockdisplay(display); draw(screen, screen->r, display->black, nil, ZP); drawaxes(); drawvmodel(screen, mainobj.mdl, t); drawinfo(); flushimage(display, 1); unlockdisplay(display); } void lmb(Mousectl *mc, Keyboardctl *) { Point2 mpos; mpos = fromscreen(mc->xy); fprint(2, "wp %v\n", mpos); fprint(2, "op %v\n", rframexform(mpos, mainobj)); } void zoom(Mousectl *mc) { double z; Point oldxy, Δxy; oldxy = mc->xy; for(;;){ readmouse(mc); if(mc->buttons != 2) break; Δxy = subpt(mc->xy, oldxy); z = tanh((double)Δxy.y/100) + 1; scale = z; redraw(1); } zoomobj(scale); scale = 1; } void rmb(Mousectl *mc, Keyboardctl *) { Point2 p; double oldpθ, oldθ; p = rframexform(fromscreen(mc->xy), mainobj); oldpθ = atan2(p.y, p.x); oldθ = θ; for(;;){ readmouse(mc); if(mc->buttons != 4) break; p = rframexform(fromscreen(mc->xy), mainobj); θ = oldθ + atan2(p.y, p.x) - oldpθ; redraw(1); } rotateobj(θ); θ = 0; } void mouse(Mousectl *mc, Keyboardctl *kc) { if((mc->buttons&1) != 0) lmb(mc, kc); if((mc->buttons&2) != 0) zoom(mc); if((mc->buttons&4) != 0) rmb(mc, kc); } void key(Rune r) { switch(r){ case Kdel: case 'q': threadexitsall(nil); } } void usage(void) { fprint(2, "usage: %s\n", argv0); exits("usage"); } void threadmain(int argc, char *argv[]) { Mousectl *mc; Keyboardctl *kc; Rune r; GEOMfmtinstall(); ARGBEGIN{ default: usage(); }ARGEND; if(argc > 0) usage(); if(newwindow(nil) < 0) sysfatal("newwindow: %r"); if(initdraw(nil, nil, nil) < 0) sysfatal("initdraw: %r"); if((mc = initmouse(nil, screen)) == nil) sysfatal("initmouse: %r"); if((kc = initkeyboard(nil)) == nil) sysfatal("initkeyboard: %r"); worldrf.p = Pt2(screen->r.min.x+Dx(screen->r)/2,screen->r.max.y-Dy(screen->r)/2,1); worldrf.bx = Vec2(1, 0); worldrf.by = Vec2(0,-1); mainobj.bx = Vec2(1, 0); mainobj.by = Vec2(0, 1); mainobj.mdl = readvmodel("../assets/mdl/wedge.vmdl"); if(mainobj.mdl == nil) sysfatal("readvmodel: %r"); display->locking = 1; unlockdisplay(display); redraw(0); for(;;){ enum { MOUSE, RESIZE, KEYBOARD }; Alt a[] = { {mc->c, &mc->Mouse, CHANRCV}, {mc->resizec, nil, CHANRCV}, {kc->c, &r, CHANRCV}, {nil, nil, CHANEND} }; switch(alt(a)){ case MOUSE: mouse(mc, kc); break; case RESIZE: resized(); break; case KEYBOARD: key(r); break; } redraw(0); } } void resized(void) { lockdisplay(display); if(getwindow(display, Refnone) < 0) sysfatal("couldn't resize"); unlockdisplay(display); worldrf.p = Pt2(screen->r.min.x+Dx(screen->r)/2,screen->r.max.y-Dy(screen->r)/2,1); redraw(0); }