shithub: musw

Download patch

ref: 6d1aff1af30525e3dccc1ba6cb51704d0168f1d6
parent: 9e5da826ebcec801b11eefca036bfa408492d4c6
author: rodri <rgl@antares-labs.eu>
date: Thu Feb 2 16:45:22 EST 2023

vmodeled: fixed object scaling and rotation without affecting coords. created initial object structure.

--- a/vmodeled/main.c
+++ b/vmodeled/main.c
@@ -22,8 +22,15 @@
 	char *strokefmt;
 };
 
+typedef struct Object Object;
+struct Object
+{
+	RFrame;
+	VModel *mdl;
+};
+
 RFrame worldrf;
-VModel *model;
+Object mainobj;
 double θ;
 double scale = 1;
 
@@ -72,6 +79,25 @@
 	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)
 {
@@ -121,8 +147,16 @@
 	return mdl;
 }
 
+int
+writevmodel(VModel *mdl, char *file)
+{
+	USED(mdl);
+	USED(file);
+	return -1;
+}
+
 void
-drawvmodel(Image *dst, VModel *mdl)
+drawvmodel(Image *dst, VModel *mdl, int t)
 {
 	int i;
 	char *s;
@@ -129,26 +163,33 @@
 	Point pts[3];
 	Point2 *p;
 	Matrix S = {
-		scale, 0, 1,
-		0, scale, 1,
+		1/scale, 0, 0,
+		0, 1/scale, 0,
 		0, 0, 1,
 	}, R = {
-		cos(θ), -sin(θ), 1,
-		sin(θ),  cos(θ), 1,
+		cos(θ), -sin(θ), 0,
+		sin(θ),  cos(θ), 0,
 		0, 0, 1,
 	};
 
-	mulm(S, R);
+	if(t)
+		mulm(S, R);
 	p = mdl->pts;
 	for(s = mdl->strokefmt; s != 0 && p-mdl->pts < mdl->npts; s++)
 		switch(*s){
 		case 'l':
-			line(dst, toscreen(xform(p[0], S)), toscreen(xform(p[1], S)), 0, 0, 0, display->white, ZP);
+			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++)
-				pts[i] = toscreen(xform(p[i], S));
+				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;
@@ -163,53 +204,60 @@
 }
 
 void
-redraw(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, model);
+	drawvmodel(screen, mainobj.mdl, t);
+	drawinfo();
 	flushimage(display, 1);
 	unlockdisplay(display);
 }
 
 void
-rmb(Mousectl *mc, Keyboardctl *)
-{
-	Point2 p;
-	double oldmθ, oldθ;
-
-	p = fromscreen(mc->xy);
-	oldmθ = atan2(p.y, p.x);
-	oldθ = θ;
-
-	for(;;){
-		readmouse(mc);
-		if(mc->buttons != 4)
-			break;
-		p = fromscreen(mc->xy);
-		θ = oldθ + (atan2(p.y, p.x) - oldmθ);
-		fprint(2, "θ %g\n", θ);
-		redraw();
-	}
-}
-
-void
 lmb(Mousectl *mc, Keyboardctl *)
 {
 	Point2 mpos;
 
 	mpos = fromscreen(mc->xy);
-	fprint(2, "mpos %v\n", mpos);
+	fprint(2, "wp %v\n", mpos);
+	fprint(2, "op %v\n", rframexform(mpos, mainobj));
 }
 
 void
 zoom(Mousectl *mc)
 {
-	double oldscale, z;
+	double z;
 	Point oldxy, Δxy;
 
-	oldscale = scale;
 	oldxy = mc->xy;
 
 	for(;;){
@@ -218,12 +266,36 @@
 			break;
 		Δxy = subpt(mc->xy, oldxy);
 		z = tanh((double)Δxy.y/100) + 1;
-		scale = z*oldscale;
-		redraw();
+		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)
@@ -277,14 +349,16 @@
 	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);
 
-	model = readvmodel("../assets/mdl/wedge.vmdl");
-	if(model == nil)
+	mainobj.mdl = readvmodel("../assets/mdl/wedge.vmdl");
+	if(mainobj.mdl == nil)
 		sysfatal("readvmodel: %r");
 
 	display->locking = 1;
 	unlockdisplay(display);
-	redraw();
+	redraw(0);
 
 	for(;;){
 		enum { MOUSE, RESIZE, KEYBOARD };
@@ -307,7 +381,7 @@
 			break;
 		}
 
-		redraw();
+		redraw(0);
 	}
 }
 
@@ -319,5 +393,5 @@
 		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();
+	redraw(0);
 }
--- a/vmodeled/mkfile
+++ b/vmodeled/mkfile
@@ -6,3 +6,6 @@
 	main.$O\
 
 </sys/src/cmd/mkone
+
+syms:V:
+	$CC -a main.c > syms