unistr/u8-mbtouc \
unistr/u8-strlen \
unistr/u8-strncat \
+ uniwidth/u8-strwidth \
unitypes \
unlocked-io \
vasprintf-posix \
data_out_pool (const union value *input, const char *encoding,
const struct fmt_spec *format, struct pool *pool)
{
+ const struct fmt_number_style *style = settings_get_style (format->type);
char *output;
char *t ;
assert (fmt_check_output (format));
- output = xmalloc (format->w + 1);
+ output = xmalloc (format->w + style->extra_bytes + 1);
converters[format->type] (input, format, output);
the negative suffix, plus (if negative) the negative
prefix. */
width = rounder_width (r, decimals, &integer_digits, &add_neg_prefix);
- width += strlen (style->neg_suffix.s);
+ width += style->neg_suffix.width;
if (add_neg_prefix)
- width += strlen (style->neg_prefix.s);
+ width += style->neg_prefix.width;
if (width > format->w)
continue;
if (add_neg_prefix)
p = stpcpy (p, style->neg_suffix.s);
else
- p = mempset (p, ' ', strlen (style->neg_suffix.s));
- assert (p == output + format->w);
+ p = mempset (p, ' ', style->neg_suffix.width);
+
+ assert (p >= output + format->w);
+ assert (p <= output + format->w + style->extra_bytes);
+ *p = '\0';
return true;
}
char buf[64], *p;
/* Allocate minimum required space. */
- width = 6 + strlen (style->neg_suffix.s);
+ width = 6 + style->neg_suffix.width;
if (number < 0)
- width += strlen (style->neg_prefix.s);
+ width += style->neg_prefix.width;
if (width > format->w)
return false;
if (number < 0)
p = stpcpy (p, style->neg_suffix.s);
else
- p = mempset (p, ' ', strlen (style->neg_suffix.s));
+ p = mempset (p, ' ', style->neg_suffix.width);
- assert (p == buf + format->w);
- memcpy (output, buf, format->w);
- output[format->w] = '\0';
+ assert (p >= output + format->w);
+ assert (p <= output + format->w + style->extra_bytes);
+ *p = '\0';
return true;
}
#include <ctype.h>
#include <stdlib.h>
+#include <uniwidth.h>
#include "data/identifier.h"
#include "data/settings.h"
/* Sets the number style for TYPE to have the given DECIMAL and GROUPING
characters, negative prefix NEG_PREFIX, prefix PREFIX, suffix SUFFIX, and
- negative suffix NEG_SUFFIX. */
+ negative suffix NEG_SUFFIX. All of the strings are UTF-8 encoded. */
void
fmt_settings_set_style (struct fmt_settings *settings, enum fmt_type type,
char decimal, char grouping,
const char *suffix, const char *neg_suffix)
{
struct fmt_number_style *style = &settings->styles[type];
+ int total_bytes, total_width;
assert (grouping == '.' || grouping == ',' || grouping == 0);
assert (decimal == '.' || decimal == ',');
fmt_affix_set (&style->neg_suffix, neg_suffix);
style->decimal = decimal;
style->grouping = grouping;
+
+ total_bytes = (strlen (neg_prefix) + strlen (prefix)
+ + strlen (suffix) + strlen (neg_suffix));
+ total_width = (style->neg_prefix.width + style->prefix.width
+ + style->suffix.width + style->neg_suffix.width);
+ style->extra_bytes = MAX (0, total_bytes - total_width);
}
/* Sets the decimal point character for the settings in S to DECIMAL.
return map[bytes - 1];
}
\f
-/* Sets AFFIX's string value to S. */
+/* Sets AFFIX's string value to S, a UTF-8 encoded string. */
static void
fmt_affix_set (struct fmt_affix *affix, const char *s)
{
affix->s = s[0] == '\0' ? CONST_CAST (char *, "") : xstrdup (s);
+ affix->width = u8_strwidth (CHAR_CAST (const uint8_t *, s), "UTF-8");
}
/* Frees data in AFFIX. */
fmt_affix_set (&new->neg_suffix, old->neg_suffix.s);
new->decimal = old->decimal;
new->grouping = old->grouping;
+ new->extra_bytes = old->extra_bytes;
}
/* Destroys a struct fmt_number_style. */
}
}
-/* Returns the total width of the standard prefix and suffix for
- STYLE. */
+/* Returns the total width of the standard prefix and suffix for STYLE, in
+ display columns (e.g. as returned by u8_strwidth()). */
int
fmt_affix_width (const struct fmt_number_style *style)
{
- return strlen (style->prefix.s) + strlen (style->suffix.s);
+ return style->prefix.width + style->suffix.width;
}
-/* Returns the total width of the negative prefix and suffix for
- STYLE. */
+/* Returns the total width of the negative prefix and suffix for STYLE, in
+ display columns (e.g. as returned by u8_strwidth()). */
int
fmt_neg_affix_width (const struct fmt_number_style *style)
{
- return strlen (style->neg_prefix.s) + strlen (style->neg_suffix.s);
+ return style->neg_prefix.width + style->neg_suffix.width;
}
/* Returns the struct fmt_desc for the given format TYPE. */
/* A prefix or suffix for a numeric output format. */
struct fmt_affix
{
- char *s; /* String contents of affix. */
+ char *s; /* String contents of affix, in UTF-8. */
+ int width; /* Display width in columns (see wcwidth()). */
};
/* A numeric output style. */
struct fmt_affix neg_suffix; /* Negative suffix. */
char decimal; /* Decimal point: '.' or ','. */
char grouping; /* Grouping character: ',', '.', or 0. */
+
+ /* A fmt_affix may require more bytes than its display width; for example,
+ U+00A5 (¥) is 3 bytes in UTF-8 but occupies only one display column.
+ This member is the sum of the number of bytes required by all of the
+ fmt_affix members in this struct, minus their display widths. Thus, it
+ can be used to size memory allocations: for example, the formatted
+ result of CCA20.5 requires no more than (20 + extra_bytes) bytes in
+ UTF-8. */
+ int extra_bytes;
};
int fmt_affix_width (const struct fmt_number_style *);