#include <u.h> #include <libc.h> #include <draw.h> #include <event.h> #include <bio.h> #include "proof.h" char fname[NFONT][20]; /* font names */ char lastload[NFONT][20]; /* last file name prefix loaded for this font */ Font *fonttab[NFONT][NSIZE]; /* pointers to fonts */ int fmap[NFONT]; /* what map to use with this font */ static void loadfont(int, int); static void fontlookup(int, char *); static void buildxheight(Biobuf*); static void buildmap(Biobuf*); static void buildtroff(char *); static void addmap(int, char *, int); static char *map(Rune*, int); static void scanstr(char *, char *, char **); int specfont; /* somehow, number of special font */ #define NMAP 5 #define QUICK 2048 /* char values less than this are quick to look up */ #define eq(s,t) strcmp((char *) s, (char *) t) == 0 int curmap = -1; /* what map are we working on */ typedef struct Link Link; struct Link /* link names together */ { uchar *name; int val; Link *next; }; typedef struct Map Map; struct Map /* holds a mapping from uchar name to index */ { double xheight; Rune quick[QUICK]; /* low values get special treatment */ Link *slow; /* other stuff goes into a link list */ }; Map charmap[5]; typedef struct Fontmap Fontmap; struct Fontmap /* mapping from troff name to filename */ { char *troffname; char *prefix; int map; /* which charmap to use for this font */ char *fallback; /* font to look in if can't find char here */ }; Fontmap fontmap[100]; int pos2fontmap[NFONT]; /* indexed by troff font position, gives Fontmap */ int nfontmap = 0; /* how many are there */ void dochar(Rune r[]) { char *s, *fb; Font *f; Point p; int fontno, fm, i; char buf[10]; fontno = curfont; if((s = map(r, curfont)) == 0){ /* not on current font */ if ((s = map(r, specfont)) != 0) /* on special font */ fontno = specfont; else{ /* look for fallback */ fm = pos2fontmap[curfont]; fb = fontmap[fm].fallback; if(fb){ /* see if fallback is mounted */ for(i = 0; i < NFONT; i++){ if(eq(fb, fontmap[pos2fontmap[i]].troffname)){ s = map(r, i); if(s){ fontno = i; goto found; } } } } /* no such char; use name itself on defont */ /* this is not a general solution */ p.x = hpos/DIV + xyoffset.x + offset.x; p.y = vpos/DIV + xyoffset.y + offset.y; p.y -= font->ascent; sprint(buf, "%S", r); string(screen, p, display->black, ZP, font, buf); return; } } found: p.x = hpos/DIV + xyoffset.x + offset.x; p.y = vpos/DIV + xyoffset.y + offset.y; while ((f = fonttab[fontno][cursize]) == 0) loadfont(fontno, cursize); p.y -= f->ascent; dprint(2, "putting %S at %d,%d font %d, size %d\n", r, p.x, p.y, fontno, cursize); string(screen, p, display->black, ZP, f, s); } static int drawlog2[] = { 0, 0, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 5 }; static void loadfont(int n, int s) { char file[100]; int i, fd, t, deep; static char *try[3] = {"", "times/R.", "pelm/"}; Subfont *f; Font *ff; try[0] = fname[n]; dprint(2, "loadfont %d %d\n", n, s); for (t = 0; t < 3; t++){ i = s * mag * charmap[fmap[n]].xheight/0.72; /* a pixel is 0.72 points */ if (i < MINSIZE) i = MINSIZE; dprint(2, "size %d, i %d, mag %g\n", s, i, mag); for(; i >= MINSIZE; i--){ /* if .font file exists, take that */ sprint(file, "%s/%s%d.font", libfont, try[t], i); ff = openfont(display, file); if(ff != 0){ fonttab[n][s] = ff; dprint(2, "using %s for font %d %d\n", file, n, s); return; } /* else look for a subfont file */ for (deep = drawlog2[screen->depth]; deep >= 0; deep--){ sprint(file, "%s/%s%d.%d", libfont, try[t], i, deep); dprint(2, "trying %s for %d\n", file, i); if ((fd = open(file, 0)) >= 0){ f = readsubfont(display, file, fd, 0); if (f == 0) { fprint(2, "can't rdsubfontfile %s: %r\n", file); exits("rdsubfont"); } close(fd); ff = mkfont(f, 0); if(ff == 0){ fprint(2, "can't mkfont %s: %r\n", file); exits("rdsubfont"); } fonttab[n][s] = ff; dprint(2, "using %s for font %d %d\n", file, n, s); return; } } } } fprint(2, "can't find font %s.%d or substitute, quitting\n", fname[n], s); exits("no font"); } void loadfontname(int n, char *s) { int i; Font *f, *g = 0; if (strcmp(s, fname[n]) == 0) return; if(fname[n] && fname[n][0]){ if(lastload[n] && strcmp(lastload[n], fname[n]) == 0) return; strcpy(lastload[n], fname[n]); } fontlookup(n, s); for (i = 0; i < NSIZE; i++) if (f = fonttab[n][i]){ if (f != g) { freefont(f); g = f; } fonttab[n][i] = 0; } } void allfree(void) { int i; for (i=0; i<NFONT; i++) loadfontname(i, "??"); } void readmapfile(char *file) { Biobuf *fp; char *p, cmd[100]; if ((fp=Bopen(file, OREAD)) == 0){ fprint(2, "proof: can't open map file %s\n", file); exits("urk"); } while((p=Brdline(fp, '\n')) != 0) { p[Blinelen(fp)-1] = 0; scanstr(p, cmd, 0); if(p[0]=='\0' || eq(cmd, "#")) /* skip comments, empty */ continue; else if(eq(cmd, "xheight")) buildxheight(fp); else if(eq(cmd, "map")) buildmap(fp); else if(eq(cmd, "special")) buildtroff(p); else if(eq(cmd, "troff")) buildtroff(p); else fprint(2, "weird map line %s\n", p); } Bterm(fp); } static void buildxheight(Biobuf *fp) /* map goes from char name to value to print via *string() */ { char *line; line = Brdline(fp, '\n'); if(line == 0){ fprint(2, "proof: bad map file\n"); exits("map"); } charmap[curmap].xheight = atof(line); } static void buildmap(Biobuf *fp) /* map goes from char name to value to print via *string() */ { uchar *p, *line, ch[100]; int val; Rune r; curmap++; if(curmap >= NMAP){ fprint(2, "proof: out of char maps; recompile\n"); exits("charmap"); } while ((line = Brdline(fp, '\n'))!= 0){ if (line[0] == '\n') return; line[Blinelen(fp)-1] = 0; scanstr((char *) line, (char *) ch, (char **)(void*)&p); if (ch[0] == '\0') { fprint(2, "bad map file line '%s'\n", (char*)line); continue; } val = strtol((char *) p, 0, 10); dprint(2, "buildmap %s (%x %x) %s %d\n", (char*)ch, ch[0], ch[1], (char*)p, val); chartorune(&r, (char*)ch); if(utflen((char*)ch)==1 && r<QUICK) charmap[curmap].quick[r] = val; else addmap(curmap, strdup((char *) ch), val); /* put somewhere else */ } } static void addmap(int n, char *s, int val) /* stick a new link on */ { Link *p = (Link *) malloc(sizeof(Link)); Link *prev = charmap[n].slow; if(p == 0) exits("out of memory in addmap"); p->name = (uchar *) s; p->val = val; p->next = prev; charmap[n].slow = p; } static void buildtroff(char *buf) /* map troff names into bitmap filenames */ { /* e.g., R -> times/R., I -> times/I., etc. */ char *p, cmd[100], name[200], prefix[400], fallback[100]; scanstr(buf, cmd, &p); scanstr(p, name, &p); scanstr(p, prefix, &p); while(*p!=0 && isspace(*p)) p++; if(*p != 0){ scanstr(p, fallback, &p); fontmap[nfontmap].fallback = strdup(fallback); }else fontmap[nfontmap].fallback = 0; fontmap[nfontmap].troffname = strdup(name); fontmap[nfontmap].prefix = strdup(prefix); fontmap[nfontmap].map = curmap; dprint(2, "troff name %s is bitmap %s map %d in slot %d fallback %s\n", name, prefix, curmap, nfontmap, fontmap[nfontmap].fallback? fontmap[nfontmap].fallback : "<null>"); nfontmap++; } static void fontlookup(int n, char *s) /* map troff name of s into position n */ { int i; for(i = 0; i < nfontmap; i++) if (eq(s, fontmap[i].troffname)) { strcpy(fname[n], fontmap[i].prefix); fmap[n] = fontmap[i].map; pos2fontmap[n] = i; if (eq(s, "S")) specfont = n; dprint(2, "font %d %s is %s\n", n, s, fname[n]); return; } /* god help us if this font isn't there */ } static char * map(Rune rp[], int font) /* figure out mapping for char in this font */ { static char s[100]; char c[10]; Link *p; Rune r; if(rp[1]==0 && rp[0]<QUICK) /* fast lookup */ r = charmap[fmap[font]].quick[rp[0]]; else { /* high-valued or compound character name */ sprint(c, "%S", rp); r = 0; for (p = charmap[fmap[font]].slow; p; p = p->next) if(eq(c, p->name)){ r = p->val; break; } } if(r == 0){ /* not there */ dprint(2, "didn't find %S font# %d\n", rp, font); return 0; } dprint(2, "map %S to %s font# %d\n", rp, s, font); s[runetochar(s, &r)] = 0; return s; } static void scanstr(char *s, char *ans, char **ep) { for (; isspace((uchar) *s); s++) ; for (; *s!=0 && !isspace((uchar) *s); ) *ans++ = *s++; *ans = 0; if (ep) *ep = s; }