summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--usr.bin/mandoc/term.c57
1 files changed, 33 insertions, 24 deletions
diff --git a/usr.bin/mandoc/term.c b/usr.bin/mandoc/term.c
index 31de5a40886..e84d09a2c8d 100644
--- a/usr.bin/mandoc/term.c
+++ b/usr.bin/mandoc/term.c
@@ -1,4 +1,4 @@
-/* $Id: term.c,v 1.21 2010/03/02 00:38:59 schwarze Exp $ */
+/* $Id: term.c,v 1.22 2010/03/05 20:46:48 schwarze Exp $ */
/*
* Copyright (c) 2008, 2009 Kristaps Dzonsons <kristaps@kth.se>
*
@@ -128,9 +128,10 @@ term_flushln(struct termp *p)
int i; /* current input position in p->buf */
size_t vis; /* current visual position on output */
size_t vbl; /* number of blanks to prepend to output */
- size_t vsz; /* visual characters to write to output */
+ size_t vend; /* end of word visual position on output */
size_t bp; /* visual right border position */
int j; /* temporary loop index */
+ int jhy; /* last hyphen before line overflow */
size_t maxvis, mmax;
static int overstep = 0;
@@ -157,8 +158,6 @@ term_flushln(struct termp *p)
* breaking the line.
*/
- vis = 0;
-
/*
* If in the standard case (left-justified), then begin with our
* indentation, otherwise (columns, etc.) just start spitting
@@ -170,7 +169,18 @@ term_flushln(struct termp *p)
for (j = 0; j < (int)p->offset; j++)
putchar(' ');
- for (i = 0; i < (int)p->col; i++) {
+ vis = vend = i = 0;
+ while (i < (int)p->col) {
+
+ /*
+ * Choose the number of blanks to prepend: no blank at the
+ * beginning of a line, one between words -- but do not
+ * actually write them yet.
+ */
+ vbl = (size_t)(ASCII_EOS == p->buf[i] ? 2 :
+ (0 == vis ? 0 : 1));
+ vis += vbl;
+
/*
* Count up visible word characters. Control sequences
* (starting with the CSI) aren't counted. A space
@@ -179,45 +189,41 @@ term_flushln(struct termp *p)
*/
/* LINTED */
- for (j = i, vsz = 0; j < (int)p->col; j++) {
+ for (j = i, jhy = 0, vend = vis; j < (int)p->col; j++) {
if (j && ' ' == p->buf[j])
break;
else if (8 == p->buf[j])
- vsz--;
- else if (ASCII_EOS != p->buf[j])
- vsz++;
+ vend--;
+ else if (ASCII_EOS != p->buf[j]) {
+ if (vend > vis && vend < bp &&
+ '-' == p->buf[j])
+ jhy = j;
+ vend++;
+ }
}
/*
* Skip empty words. This happens due to the ASCII_EOS
* after the end of the final sentence of a paragraph.
*/
- if (vsz == 0 && j == (int)p->col)
+ if (vend == vis && j == (int)p->col)
break;
/*
- * Choose the number of blanks to prepend: no blank at the
- * beginning of a line, one between words -- but do not
- * actually write them yet.
- */
- vbl = (size_t)(ASCII_EOS == p->buf[i] ? 2 :
- (0 == vis ? 0 : 1));
-
- /*
* Find out whether we would exceed the right margin.
* If so, break to the next line. (TODO: hyphenate)
* Otherwise, write the chosen number of blanks now.
*/
- if (vis && vis + vbl + vsz > bp) {
+ if (vend > bp && 0 == jhy) {
+ vend -= vis;
putchar('\n');
if (TERMP_NOBREAK & p->flags) {
for (j = 0; j < (int)p->rmargin; j++)
putchar(' ');
- vis = p->rmargin - p->offset;
+ vend += p->rmargin - p->offset;
} else {
for (j = 0; j < (int)p->offset; j++)
putchar(' ');
- vis = 0;
}
/* Remove the overstep width. */
bp += (int)/* LINTED */
@@ -226,21 +232,24 @@ term_flushln(struct termp *p)
} else {
for (j = 0; j < (int)vbl; j++)
putchar(' ');
- vis += vbl;
}
/*
* Finally, write out the word.
*/
for ( ; i < (int)p->col; i++) {
- if (' ' == p->buf[i])
+ if (vend > bp && i > jhy)
break;
+ if (' ' == p->buf[i]) {
+ i++;
+ break;
+ }
if (ASCII_NBRSP == p->buf[i])
putchar(' ');
else if (ASCII_EOS != p->buf[i])
putchar(p->buf[i]);
}
- vis += vsz;
+ vis = vend;
}
p->col = 0;