summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorXavier G <xavier.github@kindwolf.org>2016-05-13 03:04:08 +0200
committerXavier G <xavier.github@kindwolf.org>2016-05-13 03:04:08 +0200
commit719efc44a3c3c3db8f12b6ff40d2021837d9a902 (patch)
tree609b9a33b95790fb98137d4f30f37794f76dbb83 /src
parent35b3ccc6a407f83eba4f0c3787cc5c174bd3385c (diff)
downloadirssi-719efc44a3c3c3db8f12b6ff40d2021837d9a902.zip
Introduce string_chars_for_width().
Diffstat (limited to 'src')
-rw-r--r--src/core/utf8.c44
-rw-r--r--src/core/utf8.h6
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)