#ifndef lint static char sccsid[] = "@(#)n7.c 2.4 (CWI) 86/08/15"; #endif lint #include "tdef.h" #ifdef NROFF #include "tw.h" #endif #ifdef NROFF #define GETCH gettch #endif #ifndef NROFF #define GETCH getch #endif /* * troff7.c * * text */ #include #include #include "ext.h" int brflg; tbreak() { register pad, k; register tchar *i, j; register int resol = 0; trap = 0; if (nb) { /* if( dip == d && numtab[NL].val == -1) newline(1); else */ return; } if (dip == d && numtab[NL].val == -1) { newline(1); return; } if (!nc) { setnel(); if (!wch) return; if (pendw) getword(1); movword(); } else if (pendw && !brflg) { getword(1); movword(); } *linep = dip->nls = 0; #ifdef NROFF if (dip == d) horiz(po); #endif if (lnmod) donum(); lastl = ne; if (brflg != 1) { totout = 0; } else if (ad) { if ((lastl = ll - un) < ne) lastl = ne; } if (admod && ad && (brflg != 2)) { lastl = ne; adsp = adrem = 0; if (admod == 1) un += quant(nel / 2, HOR); else if (admod == 2) un += nel; } totout++; brflg = 0; if (lastl + un > dip->maxl) dip->maxl = lastl + un; horiz(un); #ifdef NROFF if (adrem % t.Adj) resol = t.Hor; else resol = t.Adj; #else resol = HOR; #endif for (i = line; nc > 0; ) { if ((cbits(j = *i++)) == ' ') { pad = 0; do { pad += width(j); nc--; } while ((cbits(j = *i++)) == ' '); i--; pad += adsp; --nwd; if (adrem) { if (adrem < 0) { pad -= resol; adrem += resol; } else if ((totout & 01) || adrem / resol >= nwd) { pad += resol; adrem -= resol; } } pchar((tchar) WORDSP); horiz(pad); } else { pchar(j); nc--; } } if (ic) { if ((k = ll - un - lastl + ics) > 0) horiz(k); pchar(ic); } if (icf) icf++; else ic = 0; ne = nwd = 0; un = in; setnel(); newline(0); if (dip != d) { if (dip->dnl > dip->hnl) dip->hnl = dip->dnl; } else { if (numtab[NL].val > dip->hnl) dip->hnl = numtab[NL].val; } for (k = ls - 1; k > 0 && !trap; k--) newline(0); spread = 0; } donum() { register i, nw; extern pchar(); nrbits = nmbits; nw = width('1' | nrbits); if (nn) { nn--; goto d1; } if (numtab[LN].val % ndf) { numtab[LN].val++; d1: un += nw * (3 + nms + ni); return; } i = 0; if (numtab[LN].val < 100) i++; if (numtab[LN].val < 10) i++; horiz(nw * (ni + i)); nform = 0; fnumb(numtab[LN].val, pchar); un += nw * nms; numtab[LN].val++; } text() { register tchar i; static int spcnt; nflush++; numtab[HP].val = 0; if ((dip == d) && (numtab[NL].val == -1)) { newline(1); return; } setnel(); if (ce || !fi) { nofill(); return; } if (pendw) goto t4; if (pendt) if (spcnt) goto t2; else goto t3; pendt++; if (spcnt) goto t2; while ((cbits(i = GETCH())) == ' ') { spcnt++; numtab[HP].val += sps; widthp = sps; } if (nlflg) { t1: nflush = pendt = ch = spcnt = 0; callsp(); return; } ch = i; if (spcnt) { t2: tbreak(); if (nc || wch) goto rtn; un += spcnt * sps; spcnt = 0; setnel(); if (trap) goto rtn; if (nlflg) goto t1; } t3: if (spread) goto t5; if (pendw || !wch) t4: if (getword(0)) goto t6; if (!movword()) goto t3; t5: if (nlflg) pendt = 0; adsp = adrem = 0; if (ad) { if (nwd == 1) adsp = nel; else adsp = nel / (nwd - 1); adsp = (adsp / HOR) * HOR; adrem = nel - adsp*(nwd-1); } brflg = 1; tbreak(); spread = 0; if (!trap) goto t3; if (!nlflg) goto rtn; t6: pendt = 0; ckul(); rtn: nflush = 0; } nofill() { register j; register tchar i; if (!pendnf) { over = 0; tbreak(); if (trap) goto rtn; if (nlflg) { ch = nflush = 0; callsp(); return; } adsp = adrem = 0; nwd = 10000; } while ((j = (cbits(i = GETCH()))) != '\n') { if (j == ohc) continue; if (j == CONT) { pendnf++; nflush = 0; flushi(); ckul(); return; } j = width(i); widthp = j; numtab[HP].val += j; storeline(i, j); } if (ce) { ce--; if ((i = quant(nel / 2, HOR)) > 0) un += i; } if (!nc) storeline((tchar)FILLER, 0); brflg = 2; tbreak(); ckul(); rtn: pendnf = nflush = 0; } callsp() { register i; if (flss) i = flss; else i = lss; flss = 0; casesp(i); } ckul() { if (ul && (--ul == 0)) { cu = 0; font = sfont; mchbits(); } if (it && (--it == 0) && itmac) control(itmac, 0); } storeline(c, w) register tchar c; { if (linep >= line + lnsize - 1) { if (!over) { flusho(); errprint("Line overflow."); over++; c = LEFTHAND; w = -1; goto s1; } return; } s1: if (w == -1) w = width(c); ne += w; nel -= w; *linep++ = c; nc++; } newline(a) int a; { register i, j, nlss; int opn; if (a) goto nl1; if (dip != d) { j = lss; pchar1((tchar)FLSS); if (flss) lss = flss; i = lss + dip->blss; dip->dnl += i; pchar1((tchar)i); pchar1((tchar)'\n'); lss = j; dip->blss = flss = 0; if (dip->alss) { pchar1((tchar)FLSS); pchar1((tchar)dip->alss); pchar1((tchar)'\n'); dip->dnl += dip->alss; dip->alss = 0; } if (dip->ditrap && !dip->ditf && dip->dnl >= dip->ditrap && dip->dimac) if (control(dip->dimac, 0)) { trap++; dip->ditf++; } return; } j = lss; if (flss) lss = flss; nlss = dip->alss + dip->blss + lss; numtab[NL].val += nlss; #ifndef NROFF if (ascii) { dip->alss = dip->blss = 0; } #endif pchar1((tchar)'\n'); flss = 0; lss = j; if (numtab[NL].val < pl) goto nl2; nl1: ejf = dip->hnl = numtab[NL].val = 0; ejl = frame; if (donef) { if ((!nc && !wch) || ndone) done1(0); ndone++; donef = 0; if (frame == stk) nflush++; } opn = numtab[PN].val; numtab[PN].val++; if (npnflg) { numtab[PN].val = npn; npn = npnflg = 0; } nlpn: if (numtab[PN].val == pfrom) { print++; pfrom = -1; } else if (opn == pto) { print = 0; opn = -1; chkpn(); goto nlpn; } if (print) newpage(numtab[PN].val); /* supposedly in a clean state so can pause */ #ifdef NROFF if (print && *t.eject) { oputs(t.eject); flusho(); } #endif if (stop && print) { dpn++; if (dpn >= stop) { dpn = 0; dostop(); } } nl2: trap = 0; if (numtab[NL].val == 0) { if ((j = findn(0)) != NTRAP) trap = control(mlist[j], 0); } else if ((i = findt(numtab[NL].val - nlss)) <= nlss) { if ((j = findn1(numtab[NL].val - nlss + i)) == NTRAP) { flusho(); errprint("Trap botch."); done2(-5); } trap = control(mlist[j], 0); } } findn1(a) int a; { register i, j; for (i = 0; i < NTRAP; i++) { if (mlist[i]) { if ((j = nlist[i]) < 0) j += pl; if (j == a) break; } } return(i); } chkpn() { pto = *(pnp++); pfrom = pto>=0 ? pto : -pto; if (pto == -32767) { flusho(); done1(0); } if (pto < 0) { pto = -pto; print++; pfrom = 0; } } findt(a) int a; { register i, j, k; k = 32767; if (dip != d) { if (dip->dimac && (i = dip->ditrap - a) > 0) k = i; return(k); } for (i = 0; i < NTRAP; i++) { if (mlist[i]) { if ((j = nlist[i]) < 0) j += pl; if ((j -= a) <= 0) continue; if (j < k) k = j; } } i = pl - a; if (k > i) k = i; return(k); } findt1() { register i; if (dip != d) i = dip->dnl; else i = numtab[NL].val; return(findt(i)); } eject(a) struct s *a; { register savlss; if (dip != d) return; ejf++; if (a) ejl = a; else ejl = frame; if (trap) return; e1: savlss = lss; lss = findt(numtab[NL].val); newline(0); lss = savlss; if (numtab[NL].val && !trap) goto e1; } movword() { register w; register tchar i, *wp; int savwch, hys; over = 0; wp = wordp; if (!nwd) { while (cbits(i = *wp++) == ' ') { wch--; wne -= sps; } wp--; } if (wne > nel && !hyoff && hyf && (!nwd || nel > 3 * sps) && (!(hyf & 02) || (findt1() > lss))) hyphen(wp); savwch = wch; hyp = hyptr; nhyp = 0; while (*hyp && *hyp <= wp) hyp++; while (wch) { if (hyoff != 1 && *hyp == wp) { hyp++; if (!wdstart || (wp > wdstart + 1 && wp < wdend && (!(hyf & 04) || wp < wdend - 1) && /* 04 => last 2 */ (!(hyf & 010) || wp > wdstart + 2))) { /* 010 => 1st 2 */ nhyp++; storeline((tchar)IMP, 0); } } i = *wp++; w = width(i); wne -= w; wch--; storeline(i, w); } if (nel >= 0) { nwd++; return(0); /* line didn't fill up */ } #ifndef NROFF xbits((tchar)HYPHEN, 1); #endif hys = width((tchar)HYPHEN); m1: if (!nhyp) { if (!nwd) goto m3; if (wch == savwch) goto m4; } if (*--linep != IMP) goto m5; if (!(--nhyp)) if (!nwd) goto m2; if (nel < hys) { nc--; goto m1; } m2: if ((i = cbits(*(linep - 1))) != '-' && i != EMDASH) { *linep = (*(linep - 1) & SFMASK) | HYPHEN; w = width(*linep); nel -= w; ne += w; linep++; } m3: nwd++; m4: wordp = wp; return(1); /* line filled up */ m5: nc--; w = width(*linep); ne -= w; nel += w; wne += w; wch++; wp--; goto m1; } horiz(i) int i; { vflag = 0; if (i) pchar(makem(i)); } setnel() { if (!nc) { linep = line; if (un1 >= 0) { un = un1; un1 = -1; } nel = ll - un; ne = adsp = adrem = 0; } } getword(x) int x; { register int j, k; register tchar i, *wp; int noword; noword = 0; if (x) if (pendw) { *pendw = 0; goto rtn; } if (wordp = pendw) goto g1; hyp = hyptr; wordp = word; over = wne = wch = 0; hyoff = 0; while (1) { /* picks up 1st char of word */ j = cbits(i = GETCH()); if (j == '\n') { wne = wch = 0; noword = 1; goto rtn; } if (j == ohc) { hyoff = 1; /* 1 => don't hyphenate */ continue; } if (j == ' ') { numtab[HP].val += sps; widthp = sps; storeword(i, sps); continue; } break; } storeword(' ' | chbits, sps); if (spflg) { storeword(' ' | chbits, sps); spflg = 0; } g0: if (j == CONT) { pendw = wordp; nflush = 0; flushi(); return(1); } if (hyoff != 1) { if (j == ohc) { hyoff = 2; *hyp++ = wordp; if (hyp > (hyptr + NHYP - 1)) hyp = hyptr + NHYP - 1; goto g1; } if (j == '-' || j == EMDASH) if (wordp > word + 1) { hyoff = 2; *hyp++ = wordp + 1; if (hyp > (hyptr + NHYP - 1)) hyp = hyptr + NHYP - 1; } } j = width(i); numtab[HP].val += j; storeword(i, j); g1: j = cbits(i = GETCH()); if (j != ' ') { static char *sentchar = ".?!"; /* sentence terminators */ if (j != '\n') goto g0; wp = wordp-1; /* handle extra space at end of sentence */ while (wp >= word) { j = cbits(*wp--); if (j=='"' || j=='\'' || j==')' || j==']' || j=='*' || j==DAGGER) continue; for (k = 0; sentchar[k]; k++) if (j == sentchar[k]) { spflg++; break; } break; } } *wordp = 0; numtab[HP].val += sps; rtn: for (wp = word; *wp; wp++) { j = cbits(*wp); if (j == ' ') continue; if (!isdigit(j) && j != '-') break; } if (*wp == 0) /* all numbers, so don't hyphenate */ hyoff = 1; wdstart = 0; wordp = word; pendw = 0; *hyp++ = 0; setnel(); return(noword); } storeword(c, w) register tchar c; register int w; { if (wordp >= &word[WDSIZE - 3]) { if (!over) { flusho(); errprint("Word overflow."); over++; c = LEFTHAND; w = -1; goto s1; } return; } s1: if (w == -1) w = width(c); widthp = w; wne += w; *wordp++ = c; wch++; } #ifdef NROFF tchar gettch() { extern int c_isalnum; tchar i; int j; i = getch(); j = cbits(i); if (ismot(i) || fbits(i) != ulfont) return(i); if (cu) { if (trtab[j] == ' ') { setcbits(i, '_'); setfbits(i, FT); /* default */ } return(i); } /* should test here for characters that ought to be underlined */ /* in the old nroff, that was the 200 bit on the width! */ /* for now, just do letters, digits and certain special chars */ if (j <= 127) { if (!isalnum(j)) setfbits(i, FT); } else { if (j < c_isalnum) setfbits(i, FT); } return(i); } #endif