xref: /plan9/sys/src/cmd/aux/mnihongo/mnihongo.c (revision 7dd7cddf99dd7472612f1413b4da293630e6b1bc)
1219b2ee8SDavid du Colombier /*
2219b2ee8SDavid du Colombier output language from troff:
3219b2ee8SDavid du Colombier all numbers are character strings
4219b2ee8SDavid du Colombier 
5219b2ee8SDavid du Colombier sn	size in points
6219b2ee8SDavid du Colombier fn	font as number from 1-n
7219b2ee8SDavid du Colombier cx	ascii character x
8219b2ee8SDavid du Colombier Cxyz	funny char xyz. terminated by white space
9219b2ee8SDavid du Colombier Nn	absolute character number n on this font.  ditto
10219b2ee8SDavid du Colombier Hn	go to absolute horizontal position n
11219b2ee8SDavid du Colombier Vn	go to absolute vertical position n (down is positive)
12219b2ee8SDavid du Colombier hn	go n units horizontally (relative)
13219b2ee8SDavid du Colombier vn	ditto vertically
14219b2ee8SDavid du Colombier nnc	move right nn, then print c (exactly 2 digits!)
15219b2ee8SDavid du Colombier 		(this wart is an optimization that shrinks output file size
16219b2ee8SDavid du Colombier 		 about 35% and run-time about 15% while preserving ascii-ness)
17219b2ee8SDavid du Colombier Dt ...\n	draw operation 't':
18219b2ee8SDavid du Colombier 	Dl x y		line from here by x,y
19219b2ee8SDavid du Colombier 	Dc d		circle of diameter d with left side here
20219b2ee8SDavid du Colombier 	De x y		ellipse of axes x,y with left side here
21219b2ee8SDavid du Colombier 	Da dx dy dx dy	arc counter-clockwise, center at dx,dx, end at dx,dy
22219b2ee8SDavid du Colombier 	D~ x y x y ...	wiggly line by x,y then x,y ...
23219b2ee8SDavid du Colombier nb a	end of line (information only -- no action needed)
24219b2ee8SDavid du Colombier w	paddable word space -- no action needed
25219b2ee8SDavid du Colombier 	b = space before line, a = after
26219b2ee8SDavid du Colombier p	new page begins -- set v to 0
27219b2ee8SDavid du Colombier #...\n	comment
28219b2ee8SDavid du Colombier x ...\n	device control functions:
29219b2ee8SDavid du Colombier 	x i	init
30219b2ee8SDavid du Colombier 	x T s	name of device is s
31219b2ee8SDavid du Colombier 	x r n h v	resolution is n/inch
32219b2ee8SDavid du Colombier 		h = min horizontal motion, v = min vert
33219b2ee8SDavid du Colombier 	x p	pause (can restart)
34219b2ee8SDavid du Colombier 	x s	stop -- done for ever
35219b2ee8SDavid du Colombier 	x t	generate trailer
36219b2ee8SDavid du Colombier 	x f n s	font position n contains font s
37219b2ee8SDavid du Colombier 	x H n	set character height to n
38219b2ee8SDavid du Colombier 	x S n	set slant to N
39219b2ee8SDavid du Colombier 
40219b2ee8SDavid du Colombier 	Subcommands like "i" are often spelled out like "init".
41219b2ee8SDavid du Colombier */
42219b2ee8SDavid du Colombier 
43219b2ee8SDavid du Colombier #include <u.h>
44219b2ee8SDavid du Colombier #include <libc.h>
45*7dd7cddfSDavid du Colombier #include <draw.h>
46219b2ee8SDavid du Colombier #include <bio.h>
47219b2ee8SDavid du Colombier 
48219b2ee8SDavid du Colombier #define hmot(n)	hpos += n
49219b2ee8SDavid du Colombier #define hgoto(n)	hpos = n
50219b2ee8SDavid du Colombier #define vmot(n)	vgoto(vpos + n)
51219b2ee8SDavid du Colombier #define vgoto(n)	vpos = n
52219b2ee8SDavid du Colombier 
53219b2ee8SDavid du Colombier #define	putchar(x)	Bprint(&bout, "%C", x)
54219b2ee8SDavid du Colombier 
55219b2ee8SDavid du Colombier int	hpos;	/* horizontal position where we are supposed to be next (left = 0) */
56219b2ee8SDavid du Colombier int	vpos;	/* current vertical position (down positive) */
57219b2ee8SDavid du Colombier char	*fontfile	= "/lib/font/bit/pelm/unicode.9x24.font";
58219b2ee8SDavid du Colombier 
59219b2ee8SDavid du Colombier char	*pschar(char *, char *hex, int *wid, int *ht);
60219b2ee8SDavid du Colombier int	kanji(char *);
61219b2ee8SDavid du Colombier void	Bgetstr(Biobuf *bp, char *s);
62219b2ee8SDavid du Colombier void	Bgetline(Biobuf *bp, char *s);
63219b2ee8SDavid du Colombier void	Bgetint(Biobuf *bp, int *n);
64219b2ee8SDavid du Colombier 
65219b2ee8SDavid du Colombier Biobuf bin, bout;
66219b2ee8SDavid du Colombier 
67*7dd7cddfSDavid du Colombier void
main(void)68*7dd7cddfSDavid du Colombier main(void)
69219b2ee8SDavid du Colombier {
70219b2ee8SDavid du Colombier 	int c, n;
71219b2ee8SDavid du Colombier 	char str[100], *args[10];
72219b2ee8SDavid du Colombier 	int jfont, curfont;
73219b2ee8SDavid du Colombier 
74*7dd7cddfSDavid du Colombier 	if(initdraw(0, fontfile, 0) < 0){
75*7dd7cddfSDavid du Colombier 		fprint(2, "mnihongo: can't initialize display: %r\n");
76*7dd7cddfSDavid du Colombier 		exits("open");
77*7dd7cddfSDavid du Colombier 	}
78219b2ee8SDavid du Colombier 	Binit(&bin, 0, OREAD);
79219b2ee8SDavid du Colombier 	Binit(&bout, 1, OWRITE);
80219b2ee8SDavid du Colombier 
81219b2ee8SDavid du Colombier 	jfont = -1;
82219b2ee8SDavid du Colombier 	curfont = 1;
83219b2ee8SDavid du Colombier 	while ((c = Bgetc(&bin)) >= 0) {
84219b2ee8SDavid du Colombier 		switch (c) {
85219b2ee8SDavid du Colombier 		case '\n':	/* when input is text */
86219b2ee8SDavid du Colombier 		case ' ':
87219b2ee8SDavid du Colombier 		case '\0':		/* occasional noise creeps in */
88219b2ee8SDavid du Colombier 			putchar(c);
89219b2ee8SDavid du Colombier 			break;
90219b2ee8SDavid du Colombier 		case '0': case '1': case '2': case '3': case '4':
91219b2ee8SDavid du Colombier 		case '5': case '6': case '7': case '8': case '9':
92219b2ee8SDavid du Colombier 			/* two motion digits plus a character */
93219b2ee8SDavid du Colombier 			putchar(c);	/* digit 1 */
94219b2ee8SDavid du Colombier 			n = (c-'0')*10;
95219b2ee8SDavid du Colombier 			c = Bgetc(&bin);
96219b2ee8SDavid du Colombier 			putchar(c);	/* digit 2 */
97219b2ee8SDavid du Colombier 			n += c - '0';
98219b2ee8SDavid du Colombier 			hmot(n);
99219b2ee8SDavid du Colombier 			putchar(Bgetc(&bin));	/* char itself */
100219b2ee8SDavid du Colombier 			break;
101219b2ee8SDavid du Colombier 		case 'c':	/* single character */
102*7dd7cddfSDavid du Colombier 			c = Bgetrune(&bin);
103*7dd7cddfSDavid du Colombier 			if(c==' ')	/* why does this happen? it's troff - bwk */
104*7dd7cddfSDavid du Colombier 				break;
105*7dd7cddfSDavid du Colombier 			else if(jfont == curfont){
106*7dd7cddfSDavid du Colombier 				Bungetrune(&bin);
107219b2ee8SDavid du Colombier 				Bgetstr(&bin, str);
108219b2ee8SDavid du Colombier 				kanji(str);
109219b2ee8SDavid du Colombier 			}else{
110219b2ee8SDavid du Colombier 				putchar('c');
111*7dd7cddfSDavid du Colombier 				putchar(c);
112219b2ee8SDavid du Colombier 			}
113219b2ee8SDavid du Colombier 			break;
114219b2ee8SDavid du Colombier 		case 'C':
115219b2ee8SDavid du Colombier 			Bgetstr(&bin, str);
116219b2ee8SDavid du Colombier 			Bprint(&bout, "C%s", str);
117219b2ee8SDavid du Colombier 			break;
118219b2ee8SDavid du Colombier 		case 'f':
119219b2ee8SDavid du Colombier 			Bgetstr(&bin, str);
120219b2ee8SDavid du Colombier 			curfont = atoi(str);
121219b2ee8SDavid du Colombier 			if(curfont < 0 || curfont > 20)
122219b2ee8SDavid du Colombier 				curfont = 1;	/* sanity */
123219b2ee8SDavid du Colombier 			Bprint(&bout, "%c%s", c, str);
124219b2ee8SDavid du Colombier 			break;
125219b2ee8SDavid du Colombier 		case 'N':	/* absolute character number */
126219b2ee8SDavid du Colombier 		case 's':
127219b2ee8SDavid du Colombier 		case 'p':	/* new page */
128219b2ee8SDavid du Colombier 			Bgetint(&bin, &n);
129219b2ee8SDavid du Colombier 			Bprint(&bout, "%c%d", c, n);
130219b2ee8SDavid du Colombier 			break;
131219b2ee8SDavid du Colombier 		case 'H':	/* absolute horizontal motion */
132219b2ee8SDavid du Colombier 			Bgetint(&bin, &n);
133219b2ee8SDavid du Colombier 			Bprint(&bout, "%c%d", c, n);
134219b2ee8SDavid du Colombier 			hgoto(n);
135219b2ee8SDavid du Colombier 			break;
136219b2ee8SDavid du Colombier 		case 'h':	/* relative horizontal motion */
137219b2ee8SDavid du Colombier 			Bgetint(&bin, &n);
138219b2ee8SDavid du Colombier 			Bprint(&bout, "%c%d", c, n);
139219b2ee8SDavid du Colombier 			hmot(n);
140219b2ee8SDavid du Colombier 			break;
141219b2ee8SDavid du Colombier 		case 'V':
142219b2ee8SDavid du Colombier 			Bgetint(&bin, &n);
143219b2ee8SDavid du Colombier 			Bprint(&bout, "%c%d", c, n);
144219b2ee8SDavid du Colombier 			vgoto(n);
145219b2ee8SDavid du Colombier 			break;
146219b2ee8SDavid du Colombier 		case 'v':
147219b2ee8SDavid du Colombier 			Bgetint(&bin, &n);
148219b2ee8SDavid du Colombier 			Bprint(&bout, "%c%d", c, n);
149219b2ee8SDavid du Colombier 			vmot(n);
150219b2ee8SDavid du Colombier 			break;
151219b2ee8SDavid du Colombier 
152219b2ee8SDavid du Colombier 		case 'w':	/* word space */
153219b2ee8SDavid du Colombier 			putchar(c);
154219b2ee8SDavid du Colombier 			break;
155219b2ee8SDavid du Colombier 
156219b2ee8SDavid du Colombier 		case 'x':	/* device control */
157219b2ee8SDavid du Colombier 			Bgetline(&bin, str);
158219b2ee8SDavid du Colombier 			Bprint(&bout, "%c%s", c, str);
159219b2ee8SDavid du Colombier 			if(tokenize(str, args, 10)>2 && args[0][0]=='f' && ('0'<=args[1][0] && args[1][0]<='9')){
160219b2ee8SDavid du Colombier 				if(strncmp(args[2], "Jp", 2) == 0)
161219b2ee8SDavid du Colombier 					jfont = atoi(args[1]);
162219b2ee8SDavid du Colombier 				else if(atoi(args[1]) == jfont)
163219b2ee8SDavid du Colombier 					jfont = -1;
164219b2ee8SDavid du Colombier 			}
165219b2ee8SDavid du Colombier 			break;
166219b2ee8SDavid du Colombier 
167219b2ee8SDavid du Colombier 		case 'D':	/* draw function */
168219b2ee8SDavid du Colombier 		case 'n':	/* end of line */
169219b2ee8SDavid du Colombier 		case '#':	/* comment */
170219b2ee8SDavid du Colombier 			Bgetline(&bin, str);
171219b2ee8SDavid du Colombier 			Bprint(&bout, "%c%s", c, str);
172219b2ee8SDavid du Colombier 			break;
173219b2ee8SDavid du Colombier 		default:
174*7dd7cddfSDavid du Colombier 			fprint(2, "mnihongo: unknown input character %o %c\n", c, c);
175*7dd7cddfSDavid du Colombier 			exits("error");
176219b2ee8SDavid du Colombier 		}
177219b2ee8SDavid du Colombier 	}
178219b2ee8SDavid du Colombier }
179219b2ee8SDavid du Colombier 
kanji(char * s)180219b2ee8SDavid du Colombier int kanji(char *s)	/* very special pleading */
181219b2ee8SDavid du Colombier {			/* dump as kanji char if looks like one */
182219b2ee8SDavid du Colombier 	Rune r;
183219b2ee8SDavid du Colombier 	char hex[500];
184219b2ee8SDavid du Colombier 	int size = 10, ht, wid;
185219b2ee8SDavid du Colombier 
186219b2ee8SDavid du Colombier 	chartorune(&r, s);
187219b2ee8SDavid du Colombier 	pschar(s, hex, &wid, &ht);
188219b2ee8SDavid du Colombier 	Bprint(&bout, "x X PS save %d %d m\n", hpos, vpos);
189219b2ee8SDavid du Colombier 	Bprint(&bout, "x X PS currentpoint translate %d %d scale ptsize dup scale\n", size, size);
190219b2ee8SDavid du Colombier 	Bprint(&bout, "x X PS %d %d true [%d 0 0 -%d 0 %d]\n",
191219b2ee8SDavid du Colombier 		wid, ht, wid, wid, ht-2);	/* kludge; ought to use ->ascent */
192219b2ee8SDavid du Colombier 	Bprint(&bout, "x X PS {<%s>}\n", hex);
193219b2ee8SDavid du Colombier 	Bprint(&bout, "x X PS imagemask restore\n");
194219b2ee8SDavid du Colombier 	return 1;
195219b2ee8SDavid du Colombier }
196219b2ee8SDavid du Colombier 
pschar(char * s,char * hex,int * wid,int * ht)197219b2ee8SDavid du Colombier char *pschar(char *s, char *hex, int *wid, int *ht)
198219b2ee8SDavid du Colombier {
199219b2ee8SDavid du Colombier 	Point chpt, spt;
200*7dd7cddfSDavid du Colombier 	Image *b;
201219b2ee8SDavid du Colombier 	uchar rowdata[100];
202219b2ee8SDavid du Colombier 	char *hp = hex;
203219b2ee8SDavid du Colombier 	int y, i;
204219b2ee8SDavid du Colombier 
205*7dd7cddfSDavid du Colombier 	chpt = stringsize(font, s);		/* bounding box of char */
206219b2ee8SDavid du Colombier 	*wid = ((chpt.x+7) / 8) * 8;
207219b2ee8SDavid du Colombier 	*ht = chpt.y;
208*7dd7cddfSDavid du Colombier 	/* postscript is backwards to video, so draw white (ones) on black (zeros) */
209*7dd7cddfSDavid du Colombier 	b = allocimage(display, Rpt(ZP, chpt), GREY1, 0, DBlack);	/* place to put it */
210*7dd7cddfSDavid du Colombier 	spt = string(b, Pt(0,0), display->white, ZP, font, s);	/* put it there */
211219b2ee8SDavid du Colombier /* Bprint(&bout, "chpt %P, spt %P, wid,ht %d,%d\n", chpt, spt, *wid, *ht);
212219b2ee8SDavid du Colombier /* Bflush(&bout); */
213219b2ee8SDavid du Colombier 	for (y = 0; y < chpt.y; y++) {	/* read bits a row at a time */
214*7dd7cddfSDavid du Colombier 		memset(rowdata, 0, sizeof rowdata);
215*7dd7cddfSDavid du Colombier 		unloadimage(b, Rect(0, y, chpt.x, y+1), rowdata, sizeof rowdata);
216219b2ee8SDavid du Colombier 		for (i = 0; i < spt.x; i += 8) {	/* 8 == byte */
217219b2ee8SDavid du Colombier 			sprint(hp, "%2.2x", rowdata[i/8]);
218219b2ee8SDavid du Colombier 			hp += 2;
219219b2ee8SDavid du Colombier 		}
220219b2ee8SDavid du Colombier 	}
221219b2ee8SDavid du Colombier 	*hp = 0;
222*7dd7cddfSDavid du Colombier 	freeimage(b);
223219b2ee8SDavid du Colombier 	return hex;
224219b2ee8SDavid du Colombier }
225219b2ee8SDavid du Colombier 
226219b2ee8SDavid du Colombier 
Bgetstr(Biobuf * bp,char * s)227219b2ee8SDavid du Colombier void	Bgetstr(Biobuf *bp, char *s)	/* get a string */
228219b2ee8SDavid du Colombier {
229219b2ee8SDavid du Colombier 	int c;
230219b2ee8SDavid du Colombier 
231219b2ee8SDavid du Colombier 	while ((c = Bgetc(bp)) >= 0) {
232219b2ee8SDavid du Colombier 		if (c == ' ' || c == '\t' || c == '\n') {
233219b2ee8SDavid du Colombier 			Bungetc(bp);
234219b2ee8SDavid du Colombier 			break;
235219b2ee8SDavid du Colombier 		}
236219b2ee8SDavid du Colombier 		*s++ = c;
237219b2ee8SDavid du Colombier 	}
238219b2ee8SDavid du Colombier 	*s = 0;
239219b2ee8SDavid du Colombier }
240219b2ee8SDavid du Colombier 
Bgetline(Biobuf * bp,char * s)241219b2ee8SDavid du Colombier void	Bgetline(Biobuf *bp, char *s)	/* get a line, including newline */
242219b2ee8SDavid du Colombier {
243219b2ee8SDavid du Colombier 	int c;
244219b2ee8SDavid du Colombier 
245219b2ee8SDavid du Colombier 	while ((c = Bgetc(bp)) >= 0) {
246219b2ee8SDavid du Colombier 		*s++ = c;
247219b2ee8SDavid du Colombier 		if (c == '\n')
248219b2ee8SDavid du Colombier 			break;
249219b2ee8SDavid du Colombier 	}
250219b2ee8SDavid du Colombier 	*s = 0;
251219b2ee8SDavid du Colombier }
252219b2ee8SDavid du Colombier 
Bgetint(Biobuf * bp,int * n)253219b2ee8SDavid du Colombier void	Bgetint(Biobuf *bp, int *n)	/* get an integer */
254219b2ee8SDavid du Colombier {
255219b2ee8SDavid du Colombier 	double d;
256219b2ee8SDavid du Colombier 
257219b2ee8SDavid du Colombier 	Bgetd(bp, &d);
258219b2ee8SDavid du Colombier 	*n = d;
259219b2ee8SDavid du Colombier }
260