diff options
author | Xavier G <xavier.github@kindwolf.org> | 2016-05-13 03:04:08 +0200 |
---|---|---|
committer | Xavier G <xavier.github@kindwolf.org> | 2016-05-13 03:04:08 +0200 |
commit | 719efc44a3c3c3db8f12b6ff40d2021837d9a902 (patch) | |
tree | 609b9a33b95790fb98137d4f30f37794f76dbb83 /src | |
parent | 35b3ccc6a407f83eba4f0c3787cc5c174bd3385c (diff) | |
download | irssi-719efc44a3c3c3db8f12b6ff40d2021837d9a902.zip |
Introduce string_chars_for_width().
Diffstat (limited to 'src')
-rw-r--r-- | src/core/utf8.c | 44 | ||||
-rw-r--r-- | src/core/utf8.h | 6 |
2 files changed, 50 insertions, 0 deletions
diff --git a/src/core/utf8.c b/src/core/utf8.c index 7c75c374..d1f58599 100644 --- a/src/core/utf8.c +++ b/src/core/utf8.c @@ -89,3 +89,47 @@ int string_width(const char *str, int policy) } return len; } + +int string_chars_for_width(const char *str, int policy, unsigned int n, unsigned int *bytes) +{ + const char *c, *previous_c; + int str_width, char_width, char_count; + + g_return_val_if_fail(str != NULL, -1); + + /* Handle the dummy case where n is 0: */ + if (!n) { + if (bytes) { + *bytes = 0; + } + return 0; + } + + if (policy == -1) { + policy = string_policy(str); + } + + /* Iterate over characters until we reach n: */ + char_count = 0; + str_width = 0; + c = str; + while (*c != '\0') { + previous_c = c; + char_width = string_advance(&c, policy); + if (str_width + char_width > n) { + /* We stepped beyond n, get one step back and stop there: */ + c = previous_c; + break; + } + ++ char_count; + str_width += char_width; + } + /* At this point, we know that char_count characters reach str_width + * columns, which is less than or equal to n. */ + + /* Optionally provide the equivalent amount of bytes: */ + if (bytes) { + *bytes = c - str; + } + return char_count; +} diff --git a/src/core/utf8.h b/src/core/utf8.h index 6f02c5a0..411b2048 100644 --- a/src/core/utf8.h +++ b/src/core/utf8.h @@ -42,6 +42,12 @@ int string_length(const char *str, int policy); */ int string_width(const char *str, int policy); +/* Return the amount of characters from str it takes to reach n columns, or -1 if + * str is NULL. Optionally return the equivalent amount of bytes. + * If policy is -1, this function will call string_policy(). + */ +int string_chars_for_width(const char *str, int policy, unsigned int n, unsigned int *bytes); + #define unichar_isprint(c) (((c) & ~0x80) >= 32) #define is_utf8_leading(c) (((c) & 0xc0) != 0x80) |