diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-08-13 07:19:28 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-08-13 07:19:28 -0700 |
commit | 4b308561021d4dedd7e83e5df30534d1fdde2216 (patch) | |
tree | 5dfe9d9f3463b3d819387618e1dc6005d831aec9 /stream.c | |
parent | 1684fc0bedfa9fa9a5a7a22d6fb44e660a203a1c (diff) | |
download | txr-4b308561021d4dedd7e83e5df30534d1fdde2216.tar.gz txr-4b308561021d4dedd7e83e5df30534d1fdde2216.tar.bz2 txr-4b308561021d4dedd7e83e5df30534d1fdde2216.zip |
Observe display width in format.
Take into account string display width in field trimming and padding
calculations, including situations where only half a character fits.
* stream.c (calc_fitlen): New function.
(vformat_str): Revised to use calc_fitlen. If calc_fitlen indicates
that there is no trimming or padding, then use put_string rather than
put_char.
* txr.1: Update description of format with regard to use of
dispaly width in field calculations.
Diffstat (limited to 'stream.c')
-rw-r--r-- | stream.c | 42 |
1 files changed, 36 insertions, 6 deletions
@@ -2081,21 +2081,51 @@ static void vformat_num(val stream, const char *str, vformat_align_post(stream, align, slack); } +static cnum calc_fitlen(const wchar_t *cstr, int precision, int width) +{ + cnum i, fitlen; + wchar_t ch; + + if (width == 0 && precision == 0) + return INT_PTR_MAX; + + for (i = 0, fitlen = 0; (ch = cstr[i]) != 0; i++) { + int chw = iswcntrl(ch) ? 0 : 1 + wide_display_char_p(ch); + + if (precision && fitlen + chw > precision) + break; + + if (width && fitlen + chw > width) + return INT_PTR_MAX; + + fitlen += chw; + } + + return fitlen; +} + static void vformat_str(val stream, val str, int width, enum align align, int precision) { const wchar_t *cstr = c_str(str); - int len = c_num(length_str(str)); - int truelen = (precision && precision < len) ? precision : len; - int slack = (truelen < width) ? width - truelen : 0; - int i; + cnum fitlen = calc_fitlen(cstr, precision, width); + cnum slack = (fitlen < width) ? width - fitlen : 0; + cnum i, w; + wchar_t wc; prot1(&str); vformat_align_pre(stream, align, slack); - for (i = 0; i < truelen; i++) - put_char(chr(cstr[i]), stream); + if (fitlen == INT_PTR_MAX) { + put_string(str, stream); + } else for (i = 0, w = 0; (wc = cstr[i]) != 0; i++) { + int cw = iswcntrl(wc) ? 0 : 1 + wide_display_char_p(wc); + if (w + cw > fitlen) + break; + put_char(chr(wc), stream); + w += cw; + } vformat_align_post(stream, align, slack); |