shithub: fnt

ref: f9bb63633ce47c39a98d929a98b8392c98b07e87
dir: /test_plan9.c/

View raw version
#include <u.h>
#include <libc.h>
#include <bio.h>
#include "otf.h"

typedef struct Image Image;

struct Image {
	int w;
	u8int *b;
};

static void
fit(u8int *b, int bw, int bh, Image *im, int h, int *x, int *y, int gap)
{
	int d;

	if((bw-*x) < im->w+gap){
		*y += h + gap;
		*x = gap;
	}
	if((bh-*y) < h+gap)
		return;
	for(d = 0; d < 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)
{
#define gap 2
	int total, mid, npix, bw, bh, x, y, i;
	u8int *b;

	total = 0;
	for(i = 0; i < n; i++)
		total += im[i].w;
	mid = total / n;
	npix = (mid+gap)*(h+gap)*n;
	bh = sqrt(npix);
	bw = npix/bh;
	bh *= 1.5;
	npix *= 1.5;
	npix += bw*gap;
	if((b = malloc(npix)) == nil)
		sysfatal("no memory");
	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));
	free(b);
}

static int
otfseek(void *aux, int off, int whence)
{
	return Bseek(aux, off, whence);
}

static int
otfread(void *aux, void *dst, int sz)
{
	return Bread(aux, dst, sz);
}

static void
usage(void)
{
	fprint(2, "usage: %s [-g GLYPH_ID] [-G] [-i N] [-s PX] 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");
	exits("usage");
}

void
main(int argc, char **argv)
{
	int i, gi, G, h;
	Otfile in, out;
	Otf *o;

	gi = -1;
	G = 0;
	h = 0;
	ARGBEGIN{
	case 'g':
		gi = strtol(EARGF(usage()), nil, 0);
		break;
	case 'G':
		G++;
		break;
	case 's':
		h = strtol(EARGF(usage()), nil, 0);
		break;
	default:
		usage();
	}ARGEND

	in.seek = otfseek;
	in.read = otfread;
	out.print = (void*)Bprint;

	out.aux = Bfdopen(1, OWRITE);
	for(i = 0; i < argc; i++){
		if(h <= 0)
			Bprint(out.aux, "%s\n", argv[i]);
		if((in.aux = Bopen(argv[i], OREAD)) == nil || (o = otfopen(&in)) == nil){
			fprint(2, "%r\n");
			continue;
		}
		if(G && gi < 0){
			int i, n = otfglyfnum(o);
			Image *im = h > 0 ? calloc(n, sizeof(*im)) : nil;
			for(i = 0; i < n; i++){
				Glyf *g = otfglyf(o, i);
				if(g == nil)
					continue;
				if(h > 0 && g->simple != nil && g->numberOfContours > 0){
					int w;
					u8int *b = otfdrawglyf(g, h, &w);
					if(b == nil)
						sysfatal("%r");
					im[i].w = w;
					im[i].b = b;
				}else if(h <= 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);
				for(i = 0; i < n; i++)
					free(im[i].b);
				free(im);
			}else{
				fprint(2, "\n");
			}
		}else if(gi < 0){
			otfprint(o, &out, indentΔ);
		}else{
			int n = otfglyfnum(o);
			if(gi >= n)
				sysfatal("glyph %d out of range, max %d\n", gi, n-1);
			Glyf *g = otfglyf(o, gi);
			if(g == nil){
				fprint(2, "%d: %r\n", gi);
			}else if(h > 0){
				if(g->component != nil)
					fprint(2, "%d: component\n", gi);
				else{
					int w;
					u8int *b = otfdrawglyf(g, h, &w);
					if(b == nil)
						sysfatal("%r");
					fprint(1, "%11s %11d %11d %11d %11d ", "k8", 0, 0, w, h);
					write(1, b, w*h);
					free(b);
				}
			}else{
				Bprint(out.aux, "\n%d:\n", gi);
				print_Glyf(&out, indentΔ, o, g);
			}
		}
		otfclose(o);
		Bterm(in.aux);
	}
	Bterm(out.aux);

	exits(nil);
}