shithub: fnt

Download patch

ref: def6bc8aa2a70c6dd9f7974ca12992efd32b2ebd
parent: c623ba3f4452ad1fcb2f89bdd716b4a8831685ca
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Sun Jul 14 12:58:18 EDT 2024

raster: use ppem for scaling

--- a/otf.c.in
+++ b/otf.c.in
@@ -539,6 +539,10 @@
 		werrstr("no memory");
 		goto err;
 	}
+	if(o->td.head == nil){
+		werrstr("no head table");
+		goto err;
+	}
 	if(o->td.loca == nil){
 		werrstr("no loca table");
 		goto err;
@@ -577,6 +581,8 @@
 	if(read_Glyf(o, g) < 0){
 		free(g);
 		g = nil;
+	}else{
+		g->unitsPerEm = o->td.head->unitsPerEm;
 	}
 	otfpoprange(o);
 	otfpoprange(o);
--- a/otf.h.in
+++ b/otf.h.in
@@ -80,4 +80,4 @@
 typedef struct Glyf Glyf;
 Glyf *otfglyf(Otf *o, int index);
 int otfglyfnum(Otf *o);
-u8int *otfdrawglyf(Glyf *g, int h, int *wo);
+u8int *otfdrawglyf(Glyf *g, int ppem, int *wo, int *ho);
--- a/otf.rkt
+++ b/otf.rkt
@@ -209,7 +209,7 @@
  {int16 xMax}
  {int16 yMax}
  #:extra
- (list (cons 'field (list (~a "ComponentGlyph *component;") (~a "SimpleGlyph *simple;")))
+ (list (cons 'field (list (~a "ComponentGlyph *component;") (~a "SimpleGlyph *simple;") (~a "u16int unitsPerEm;")))
        (cons 'read
              (list (~a "if(v->numberOfContours < 0 && read_ComponentGlyph(o, &v->component, 0) < 0)")
                    (~a "	goto err;")
--- a/plan9/otf.c
+++ b/plan9/otf.c
@@ -553,6 +553,10 @@
 		werrstr("no memory");
 		goto err;
 	}
+	if(o->td.head == nil){
+		werrstr("no head table");
+		goto err;
+	}
 	if(o->td.loca == nil){
 		werrstr("no loca table");
 		goto err;
@@ -591,6 +595,8 @@
 	if(read_Glyf(o, g) < 0){
 		free(g);
 		g = nil;
+	}else{
+		g->unitsPerEm = o->td.head->unitsPerEm;
 	}
 	otfpoprange(o);
 	otfpoprange(o);
--- a/plan9/otf.h
+++ b/plan9/otf.h
@@ -80,7 +80,7 @@
 typedef struct Glyf Glyf;
 Glyf *otfglyf(Otf *o, int index);
 int otfglyfnum(Otf *o);
-u8int *otfdrawglyf(Glyf *g, int h, int *wo);
+u8int *otfdrawglyf(Glyf *g, int ppem, int *wo, int *ho);
 
 typedef struct SubHeader SubHeader;
 typedef struct MapGroup MapGroup;
@@ -451,6 +451,7 @@
 	s16int yMax;
 	ComponentGlyph *component;
 	SimpleGlyph *simple;
+	u16int unitsPerEm;
 };
 
 int read_Glyf(Otf *o, Glyf *v);
--- a/plan9/test.c
+++ b/plan9/test.c
@@ -6,38 +6,43 @@
 typedef struct Image Image;
 
 struct Image {
+	int h;
 	int w;
 	u8int *b;
 };
 
 static void
-fit(u8int *b, int bw, int bh, Image *im, int h, int *x, int *y, int gap)
+fit(u8int *b, int bw, int bh, Image *im, int maxh, int *x, int *y, int gap)
 {
 	int d;
 
 	if((bw-*x) < im->w+gap){
-		*y += h + gap;
+		*y += maxh + gap;
 		*x = gap;
 	}
-	if((bh-*y) < h+gap)
+	if((bh-*y) < maxh+gap)
 		return;
-	for(d = 0; d < h; d++)
+	for(d = 0; d < im->h; d++)
 		memcpy(b + (*y + d)*bw + *x, im->b + d*im->w, im->w);
 	*x += im->w + gap;
 }
 
 static void
-dumpmap(Biobuf *out, Image *im, int n, int h)
+dumpmap(Biobuf *out, Image *im, int n)
 {
-#define gap 2
-	int total, mid, npix, bw, bh, x, y, i;
+#define gap 1
+	int total, mid, npix, bw, bh, x, y, i, maxh;
 	u8int *b;
 
 	total = 0;
-	for(i = 0; i < n; i++)
+	maxh = 0;
+	for(i = 0; i < n; i++){
+		if(maxh < im[i].h)
+			maxh = im[i].h;
 		total += im[i].w;
+	}
 	mid = total / n;
-	npix = (mid+gap)*(h+gap)*n;
+	npix = (mid+gap)*(maxh+gap)*n;
 	bh = sqrt(npix);
 	bw = npix/bh;
 	bh *= 1.5;
@@ -48,9 +53,9 @@
 	memset(b, 0xff, npix);
 	x = y = gap;
 	for(i = 0; i < n; i++)
-		fit(b, bw, bh, im+i, h, &x, &y, gap);
-	Bprint(out, "%11s %11d %11d %11d %11d ", "k8", 0, 0, bw, y+h+gap);
-	Bwrite(out, b, bw*(y+h+gap));
+		fit(b, bw, bh, im+i, maxh, &x, &y, gap);
+	Bprint(out, "%11s %11d %11d %11d %11d ", "k8", 0, 0, bw, y+maxh+gap);
+	Bwrite(out, b, bw*(y+maxh+gap));
 	free(b);
 }
 
@@ -69,10 +74,10 @@
 static void
 usage(void)
 {
-	fprint(2, "usage: %s [-g GLYPH_ID] [-G] [-i N] [-s PX] font.otf ...\n", argv0);
+	fprint(2, "usage: %s [-g GLYPH_ID] [-G] [-i N] [-p PPM] font.otf ...\n", argv0);
 	fprint(2, " -g: specifies a single glyph id\n");
 	fprint(2, " -G: print out glyph ids, only ones that can be drawn atm (no compound yet)\n");
-	fprint(2, " -s: draw (of size PX) and write the image to stdout\n");
+	fprint(2, " -p: draw (of size in pixels per em) and write the image to stdout\n");
 	exits("usage");
 }
 
@@ -79,13 +84,13 @@
 void
 main(int argc, char **argv)
 {
-	int i, gi, G, h;
+	int i, gi, G, ppem;
 	Otfile in, out;
 	Otf *o;
 
 	gi = -1;
 	G = 0;
-	h = 0;
+	ppem = 0;
 	ARGBEGIN{
 	case 'g':
 		gi = strtol(EARGF(usage()), nil, 0);
@@ -93,8 +98,8 @@
 	case 'G':
 		G++;
 		break;
-	case 's':
-		h = strtol(EARGF(usage()), nil, 0);
+	case 'p':
+		ppem = strtol(EARGF(usage()), nil, 0);
 		break;
 	default:
 		usage();
@@ -110,30 +115,27 @@
 			fprint(2, "%r\n");
 			continue;
 		}
-		if(h <= 0)
+		if(ppem <= 0)
 			Bprint(out.aux, "%s\n", argv[i]);
 		if(G && gi < 0){
 			int i, n = otfglyfnum(o);
-			Image *im = h > 0 ? calloc(n, sizeof(*im)) : nil;
+			Image *im = ppem > 0 ? calloc(n, sizeof(*im)) : nil;
 			for(i = 0; i < n; i++){
 				Glyf *g = otfglyf(o, i);
 				if(g == nil)
 					sysfatal("%r");
-				if(h > 0 && g->simple != nil && g->numberOfContours > 0){
-					int w;
-					u8int *b = otfdrawglyf(g, h, &w);
-					if(b == nil)
+				if(ppem > 0 && g->simple != nil && g->numberOfContours > 0){
+					im[i].b = otfdrawglyf(g, ppem, &im[i].w, &im[i].h);
+					if(im[i].b == nil)
 						sysfatal("%r");
-					im[i].w = w;
-					im[i].b = b;
-				}else if(h <= 0){
+				}else if(ppem <= 0){
 					Bprint(out.aux, "%d (%s):\n", i, g->simple ? "simple" : "component");
 					print_Glyf(&out, indentΔ, o, g);
 				}
 				free(g);
 			}
-			if(h > 0){
-				dumpmap(out.aux, im, n, h);
+			if(ppem > 0){
+				dumpmap(out.aux, im, n);
 				for(i = 0; i < n; i++)
 					free(im[i].b);
 				free(im);
@@ -149,12 +151,12 @@
 			Glyf *g = otfglyf(o, gi);
 			if(g == nil){
 				fprint(2, "%d: %r\n", gi);
-			}else if(h > 0){
+			}else if(ppem > 0){
 				if(g->component != nil)
 					fprint(2, "%d: component\n", gi);
 				else{
-					int w;
-					u8int *b = otfdrawglyf(g, h, &w);
+					int w, h;
+					u8int *b = otfdrawglyf(g, ppem, &w, &h);
 					if(b == nil)
 						sysfatal("%r");
 					fprint(1, "%11s %11d %11d %11d %11d ", "k8", 0, 0, w, h);
--- a/rast.c
+++ b/rast.c
@@ -57,6 +57,8 @@
  */
 #define QBZR_PIX_SCALE 4.0
 
+#define GAP 0
+
 #define ε 1.0e-10
 
 static int
@@ -450,21 +452,17 @@
 }
 
 u8int *
-otfdrawglyf(Glyf *g, int h, int *wo)
+otfdrawglyf(Glyf *g, int ppem, int *wo, int *ho)
 {
-	int w, i, r, k, npts, npx, ns;
+	int w, h, i, r, k, npts, npx, ns;
 	Point *p₀, *p, *prev;
 	Spt *pts₀, *pts, *e;
-	double scale;
 	SegQ *s₀, *s;
 	Sval *fp;
 	u8int *b;
 
-	/* FIXME those epsilons and ceil is a rather dumb hack for now */
-	scale = (double)h / (g->yMax - g->yMin) + ε;
-	w = *wo = ceil((g->xMax - g->xMin) * scale + ε);
+#define scale (double)ppem/g->unitsPerEm
 
-	npx = w * h;
 	ns = 0;
 	s₀ = nil;
 
@@ -507,16 +505,19 @@
 		s = s₀ + ns;
 		pts = e;
 		for(e = pts₀; e <= pts-3; e += 2, s++, ns++){
-			s->v0[0] = (e[0].x - g->xMin) * scale;
-			s->v0[1] = (e[0].y - g->yMin) * scale;
-			s->v1[0] = (e[1].x - g->xMin) * scale;
-			s->v1[1] = (e[1].y - g->yMin) * scale;
-			s->v2[0] = (e[2].x - g->xMin) * scale;
-			s->v2[1] = (e[2].y - g->yMin) * scale;
+			s->v0[0] = (e[0].x - g->xMin) * scale + GAP;
+			s->v0[1] = (e[0].y - g->yMin) * scale + GAP;
+			s->v1[0] = (e[1].x - g->xMin) * scale + GAP;
+			s->v1[1] = (e[1].y - g->yMin) * scale + GAP;
+			s->v2[0] = (e[2].x - g->xMin) * scale + GAP;
+			s->v2[1] = (e[2].y - g->yMin) * scale + GAP;
 		}
 		free(pts₀);
 	}
 
+	w = ceil((g->xMax - g->xMin)*scale + 2*GAP);
+	h = ceil((g->yMax - g->yMin)*scale + 2*GAP);
+	npx = w*h;
 	fp = calloc(1, npx*sizeof(*fp));
 	r = qbzr(s₀, ns, w, h, fp);
 	free(s₀);
@@ -526,8 +527,11 @@
 		b = (u8int*)fp;
 		for(i = 0; i < npx; i++)
 			b[i] = 255 - (fp[i] <= 0 ? 0 : (fp[i] >= QBZR_PIX_SCALE ? 255 : fp[i]/QBZR_PIX_SCALE*255));
-		if((b = realloc(b, npx)) != nil)
+		if((b = realloc(b, npx)) != nil){
+			*wo = w;
+			*ho = h;
 			fp = nil;
+		}
 	}
 	free(fp);
 	return b;
--- a/unix/test.c
+++ b/unix/test.c
@@ -9,37 +9,42 @@
 
 struct Image {
 	int w;
+	int h;
 	u8int *b;
 };
 
 static void
-fit(u8int *b, int bw, int bh, Image *im, int h, int *x, int *y, int gap)
+fit(u8int *b, int bw, int bh, Image *im, int maxh, int *x, int *y, int gap)
 {
 	int d;
 
 	if((bw-*x) < im->w+gap){
-		*y += h + gap;
+		*y += maxh + gap;
 		*x = gap;
 	}
-	if((bh-*y) < h+gap)
+	if((bh-*y) < maxh+gap)
 		return;
-	for(d = 0; d < h; d++)
+	for(d = 0; d < im->h; d++)
 		memcpy(b + (*y + d)*bw + *x, im->b + d*im->w, im->w);
 	*x += im->w + gap;
 }
 
 static void
-dumpmap(FILE *out, Image *im, int n, int h)
+dumpmap(FILE *out, Image *im, int n)
 {
-#define gap 2
-	int total, mid, npix, bw, bh, x, y, i;
+#define gap 1
+	int total, mid, npix, bw, bh, x, y, i, maxh;
 	u8int *b;
 
 	total = 0;
-	for(i = 0; i < n; i++)
+	maxh = 0;
+	for(i = 0; i < n; i++){
+		if(maxh < im[i].h)
+			maxh = im[i].h;
 		total += im[i].w;
+	}
 	mid = total / n;
-	npix = (mid+gap)*(h+gap)*n;
+	npix = (mid+gap)*(maxh+gap)*n;
 	bh = sqrt(npix);
 	bw = npix/bh;
 	bh *= 1.5;
@@ -51,9 +56,9 @@
 	memset(b, 0xff, npix);
 	x = y = gap;
 	for(i = 0; i < n; i++)
-		fit(b, bw, bh, im+i, h, &x, &y, gap);
-	fprintf(out, "%11s %11d %11d %11d %11d ", "k8", 0, 0, bw, y+h+gap);
-	fwrite(b, 1, bw*(y+h+gap), out);
+		fit(b, bw, bh, im+i, maxh, &x, &y, gap);
+	fprintf(out, "%11s %11d %11d %11d %11d ", "k8", 0, 0, bw, y+maxh+gap);
+	fwrite(b, 1, bw*(y+maxh+gap), out);
 	free(b);
 }
 
@@ -99,7 +104,7 @@
 	fprintf(stderr, "usage: %s [-g GLYPH_ID] [-G] [-i N] [-s PX] font.otf ...\n", argv0);
 	fprintf(stderr, " -g: specifies a single glyph id\n");
 	fprintf(stderr, " -G: print out glyph ids, only ones that can be drawn atm (no compound yet)\n");
-	fprintf(stderr, " -s: draw (of size PX) and write the image to stdout\n");
+	fprintf(stderr, " -p: draw (of size in pixels per em) and write the image to stdout\n");
 	exit(1);
 }
 
@@ -106,13 +111,13 @@
 int
 main(int argc, char **argv)
 {
-	int i, gi, G, h;
+	int i, gi, G, ppem;
 	Otfile in, out;
 	Otf *o;
 
 	gi = -1;
 	G = 0;
-	h = 0;
+	ppem = 0;
 	ARGBEGIN{
 	case 'g':
 		gi = strtol(EARGF(usage()), NULL, 0);
@@ -120,8 +125,8 @@
 	case 'G':
 		G++;
 		break;
-	case 's':
-		h = strtol(EARGF(usage()), NULL, 0);
+	case 'p':
+		ppem = strtol(EARGF(usage()), NULL, 0);
 		break;
 	default:
 		usage();
@@ -137,30 +142,27 @@
 			err(1, "%s", argv[i]);
 		if((o = otfopen(&in)) == NULL)
 			errx(1, "%s: %s", argv[i], otferrstr());
-		if(h <= 0)
+		if(ppem <= 0)
 			fprintf(out.aux, "%s\n", argv[i]);
 		if(G && gi < 0){
 			int i, n = otfglyfnum(o);
-			Image *im = h > 0 ? calloc(n, sizeof(*im)) : NULL;
+			Image *im = ppem > 0 ? calloc(n, sizeof(*im)) : NULL;
 			for(i = 0; i < n; i++){
 				Glyf *g = otfglyf(o, i);
 				if(g == NULL)
 					errx(1, "glyph %d: %s", i, otferrstr());
-				if(h > 0 && g->simple != NULL && g->numberOfContours > 0){
-					int w;
-					u8int *b = otfdrawglyf(g, h, &w);
-					if(b == NULL)
+				if(ppem > 0 && g->simple != NULL && g->numberOfContours > 0){
+					im[i].b = otfdrawglyf(g, ppem, &im[i].w, &im[i].h);
+					if(im[i].b == NULL)
 						errx(1, "glyph %d: %s", i, otferrstr());
-					im[i].w = w;
-					im[i].b = b;
-				}else if(h <= 0){
+				}else if(ppem <= 0){
 					fprintf(out.aux, "%d (%s):\n", i, g->simple ? "simple" : "component");
 					print_Glyf(&out, indentΔ, o, g);
 				}
 				free(g);
 			}
-			if(h > 0){
-				dumpmap(out.aux, im, n, h);
+			if(ppem > 0){
+				dumpmap(out.aux, im, n);
 				for(i = 0; i < n; i++)
 					free(im[i].b);
 				free(im);
@@ -174,17 +176,17 @@
 			Glyf *g = otfglyf(o, gi);
 			if(g == NULL){
 				errx(1, "glyph %d: %s", gi, otferrstr());
-			}else if(h > 0){
+			}else if(ppem > 0){
 				if(g->component != NULL)
 					errx(1, "glyph %d: component", gi);
 				else{
-					int w;
-					u8int *b = otfdrawglyf(g, h, &w);
-					if(b == NULL)
+					Image im;
+					im.b = otfdrawglyf(g, ppem, &im.w, &im.h);
+					if(im.b == NULL)
 						errx(1, "glyph %d: %s", gi, otferrstr());
-					fprintf(out.aux, "%11s %11d %11d %11d %11d ", "k8", 0, 0, w, h);
-					fwrite(b, 1, w*h, out.aux);
-					free(b);
+					fprintf(out.aux, "%11s %11d %11d %11d %11d ", "k8", 0, 0, im.w, im.h);
+					fwrite(im.b, 1, im.w*im.h, out.aux);
+					free(im.b);
 				}
 			}else{
 				fprintf(out.aux, "\n%d:\n", gi);