#include "tests.h" #include "string_buffer.h" #include "wcwidth.h" #include #if defined(FT_HAVE_UTF8) #include "utf8.h" #endif size_t strchr_count(const char *str, char ch); size_t wstrchr_count(const wchar_t *str, wchar_t ch); #if defined(FT_HAVE_UTF8) size_t utf8chr_count(const void *str, utf8_int32_t ch); #endif const char *str_n_substring_beg(const char *str, char ch_separator, size_t n); const wchar_t *wstr_n_substring_beg(const wchar_t *str, wchar_t ch_separator, size_t n); #ifdef FT_HAVE_UTF8 const void *utf8_n_substring_beg(const void *str, utf8_int32_t ch_separator, size_t n); #endif f_status str_n_substring(const char *str, char ch_separator, size_t n, const char **begin, const char **end); void wstr_n_substring(const wchar_t *str, wchar_t ch_separator, size_t n, const wchar_t **begin, const wchar_t **end); #ifdef FT_HAVE_UTF8 void utf8_n_substring(const void *str, utf8_int32_t ch_separator, size_t n, const void **begin, const void **end); #endif void test_strchr_count(void); void test_str_n_substring(void); void test_buffer_text_visible_width(void); void test_buffer_text_visible_height(void); #if defined(FT_HAVE_WCHAR) void test_wchar_basics(void); #endif void test_print_n_strings(void); void test_string_buffer(void) { test_strchr_count(); test_str_n_substring(); test_buffer_text_visible_width(); test_buffer_text_visible_height(); #if defined(FT_HAVE_WCHAR) test_wchar_basics(); #endif test_print_n_strings(); } void test_strchr_count(void) { assert_true(strchr_count(NULL, '\n') == 0); assert_true(strchr_count("", '\n') == 0); assert_true(strchr_count("asbd", '\n') == 0); assert_true(strchr_count("asbd\n", '\n') == 1); assert_true(strchr_count("\nasbd", '\n') == 1); assert_true(strchr_count("a\nsbd", '\n') == 1); assert_true(strchr_count("\n12\n123", '\n') == 2); assert_true(strchr_count("\n12\n123\n", '\n') == 3); assert_true(strchr_count("\n\n\n", '\n') == 3); assert_true(strchr_count("\n123123\n123123\n\n\n123", '\n') == 5); assert_true(strchr_count("1a23123a123123aaa123", 'a') == 5); #ifdef FT_HAVE_WCHAR assert_true(wstrchr_count(NULL, L'\n') == 0); assert_true(wstrchr_count(L"", L'\n') == 0); assert_true(wstrchr_count(L"asbd", L'\n') == 0); assert_true(wstrchr_count(L"asbd\n", L'\n') == 1); assert_true(wstrchr_count(L"\nasbd", L'\n') == 1); assert_true(wstrchr_count(L"a\nsbd", L'\n') == 1); assert_true(wstrchr_count(L"\n12\n123", L'\n') == 2); assert_true(wstrchr_count(L"\n12\n123\n", L'\n') == 3); assert_true(wstrchr_count(L"\n\n\n", '\n') == 3); assert_true(wstrchr_count(L"\n123123\n123123\n\n\n123", L'\n') == 5); assert_true(wstrchr_count(L"1\xffffy23123\xffffy123123\xffffy\xffffy\xffffy123", L'\xffff') == 5); #endif /* FT_HAVE_WCHAR */ #ifdef FT_HAVE_UTF8 assert_true(utf8chr_count(NULL, '\n') == 0); assert_true(utf8chr_count("", '\n') == 0); assert_true(utf8chr_count("asbd", '\n') == 0); assert_true(utf8chr_count("Chinese L視野無限廣,窗外有藍天", '\n') == 0); assert_true(utf8chr_count("Hindi ऋषियों को सताने वाले दुष्ट राक्षसों के राजा रावण का सर्वनाश करने वाले विष्णुवतार भगवान श्रीराम, अयोध्या के महाराज दशरथ के बड़े सपुत्र थे।", '\n') == 0); assert_true(utf8chr_count("Portuguese Luís argüia à Júlia que «brações, fé, chá, óxido, pôr, zângão» eram palavras do português.", '\n') == 0); assert_true(utf8chr_count("Russian В чащах юга жил бы цитрус? Да, но фальшивый экземпляръ!", '\n') == 0); assert_true(utf8chr_count("Spanish La niña, viéndose atrapada en el áspero baúl índigo y sintiendo asfixia, lloró de vergüenza; mientras que la frustrada madre llamaba a su hija diciendo: ¿Dónde estás Waleska?", '\n') == 0); assert_true(utf8chr_count("asbd\n", '\n') == 1); assert_true(utf8chr_count("\nasbd", '\n') == 1); assert_true(utf8chr_count("a\nsbd", '\n') == 1); assert_true(utf8chr_count("Chinese L視野無限\n廣,窗外有藍天", '\n') == 1); assert_true(utf8chr_count("Hindi ऋषियों को सताने वा\nले दुष्ट राक्षसों के राजा रावण का सर्वनाश करने वाले विष्णुवतार भगवान श्रीराम, अयोध्या के महाराज दशरथ के बड़े सपुत्र थे।", '\n') == 1); assert_true(utf8chr_count("Portuguese Luís argüi\na à Júlia que «brações, fé, chá, óxido, pôr, zângão» eram palavras do português.", '\n') == 1); assert_true(utf8chr_count("Russian В чащах \nюга жил бы цитрус? Да, но фальшивый экземпляръ!", '\n') == 1); assert_true(utf8chr_count("Spanish La niña, vié\nndose atrapada en el áspero baúl índigo y sintiendo asfixia, lloró de vergüenza; mientras que la frustrada madre llamaba a su hija diciendo: ¿Dónde estás Waleska?", '\n') == 1); assert_true(utf8chr_count("\n12\n123", '\n') == 2); assert_true(utf8chr_count("\n12\n123\n", '\n') == 3); assert_true(utf8chr_count("\n\n\n", '\n') == 3); assert_true(utf8chr_count("\n123123\n123123\n\n\n123", '\n') == 5); assert_true(utf8chr_count("Chinese L視野無限\n廣,窗外有\n藍天", '\n') == 2); assert_true(utf8chr_count("Hindi ऋषियों को सताने वा\nले दुष्ट राक्षसों के राजा\n रावण का सर्वना\nश करने वाले विष्णुवतार भगवान श्रीराम, अयोध्या के महाराज दशरथ के बड़े सपुत्र थे।", '\n') == 3); assert_true(utf8chr_count("Portuguese Luís argüi\na à Júlia que «brações, fé, chá, óxido, \npôr, zângão» eram pal\navras do portu\nguês.", '\n') == 4); assert_true(utf8chr_count("Russian В чащах \nюга жил бы ц\nитрус? Да, но фальшивый экземпляръ!", '\n') == 2); assert_true(utf8chr_count("Spanish La niña, vié\nndose atrapada en el \n\n\náspero baúl índigo y \nsintiendo asfixia, lloró de vergüenza; mientras que la frustrada madre llamaba a su hija diciendo: ¿Dónde estás Waleska?", '\n') == 5); assert_true(utf8chr_count("1a23123a123123aaa123", 'a') == 5); #endif } void assert_str_equal_strong(const char *str1, const char *str2, const char *file, int line, const char *function) { if (!str1 && !str2) return; if ((str1 && !str2) || (!str1 && str2)) { fprintf(stderr, "%s:%d(%s):Abort! Not equals strings:\n", file, line, function); exit(EXIT_FAILURE); } else if (strcmp(str1, str2) != 0) { fprintf(stderr, "%s:%d(%s):Abort! Not equals strings:\n", file, line, function); exit(EXIT_FAILURE); } } #define ASSERT_STR_EQUAL_STRONG(str1, str2) assert_str_equal_strong(str1, str2, __FILE__,__LINE__, __func__) void test_str_n_substring(void) { const char *empty_str = ""; assert_true(str_n_substring_beg(empty_str, '\n', 0) == empty_str); assert_true(str_n_substring_beg(empty_str, '\n', 1) == NULL); assert_true(str_n_substring_beg(empty_str, '\n', 2) == NULL); #if defined(FT_HAVE_WCHAR) const wchar_t *empty_wstr = L""; assert_true(wstr_n_substring_beg(empty_wstr, L'\n', 0) == empty_wstr); assert_true(wstr_n_substring_beg(empty_wstr, L'\n', 1) == NULL); assert_true(wstr_n_substring_beg(empty_wstr, L'\n', 2) == NULL); #endif #ifdef FT_HAVE_UTF8 const char *utf8_empty_str = ""; assert_true(utf8_n_substring_beg(utf8_empty_str, '\n', 0) == utf8_empty_str); assert_true(utf8_n_substring_beg(utf8_empty_str, '\n', 1) == NULL); assert_true(utf8_n_substring_beg(utf8_empty_str, '\n', 2) == NULL); #endif const char *str = "123\n5678\n9"; assert_true(str_n_substring_beg(NULL, '\n', 0) == NULL); assert_true(str_n_substring_beg(str, '\n', 0) == str); assert_true(str_n_substring_beg(str, '1', 0) == str); assert_true(str_n_substring_beg(str, '\n', 1) == str + 4); assert_true(str_n_substring_beg(str, '\n', 2) == str + 9); assert_true(str_n_substring_beg(str, '\n', 3) == NULL); #if defined(FT_HAVE_WCHAR) const wchar_t *wstr = L"123\n5678\n9"; assert_true(wstr_n_substring_beg(NULL, L'\n', 0) == NULL); assert_true(wstr_n_substring_beg(wstr, L'\n', 0) == wstr); assert_true(wstr_n_substring_beg(wstr, L'1', 0) == wstr); assert_true(wstr_n_substring_beg(wstr, L'\n', 1) == wstr + 4); assert_true(wstr_n_substring_beg(wstr, L'\n', 2) == wstr + 9); assert_true(wstr_n_substring_beg(wstr, L'\n', 3) == NULL); #endif #ifdef FT_HAVE_UTF8 const char *utf8_str = "123\n5678\n9"; ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(NULL, '\n', 0), NULL); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str, '\n', 0), utf8_str); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str, '1', 0), utf8_str); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str, '\n', 1), "5678\n9"); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str, '\n', 2), "9"); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str, '\n', 3), NULL); #endif const char *str2 = "\n123\n56\n\n9\n"; assert_true(str_n_substring_beg(str2, '\n', 0) == str2); assert_true(str_n_substring_beg(str2, '\n', 1) == str2 + 1); assert_true(str_n_substring_beg(str2, '\n', 2) == str2 + 5); assert_true(str_n_substring_beg(str2, '\n', 3) == str2 + 8); assert_true(str_n_substring_beg(str2, '\n', 4) == str2 + 9); assert_true(str_n_substring_beg(str2, '\n', 5) == str2 + 11); assert_true(str_n_substring_beg(str2, '\n', 6) == NULL); #if defined(FT_HAVE_WCHAR) const wchar_t *wstr2 = L"\xff0fy23\xff0fy6\xff0f\xff0fy\xff0f"; assert_true(wstr_n_substring_beg(wstr2, L'\xff0f', 0) == wstr2); assert_true(wstr_n_substring_beg(wstr2, L'\xff0f', 1) == wstr2 + 1); assert_true(wstr_n_substring_beg(wstr2, L'\xff0f', 2) == wstr2 + 5); assert_true(wstr_n_substring_beg(wstr2, L'\xff0f', 3) == wstr2 + 8); assert_true(wstr_n_substring_beg(wstr2, L'\xff0f', 4) == wstr2 + 9); assert_true(wstr_n_substring_beg(wstr2, L'\xff0f', 5) == wstr2 + 11); assert_true(wstr_n_substring_beg(wstr2, L'\xff0f', 6) == NULL); #endif #ifdef FT_HAVE_UTF8 const char *utf8_str2 = "\n123\n56\n\n9\n"; ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str2, '\n', 0), utf8_str2); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str2, '\n', 1), "123\n56\n\n9\n"); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str2, '\n', 2), "56\n\n9\n"); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str2, '\n', 3), "\n9\n"); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str2, '\n', 4), "9\n"); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str2, '\n', 5), ""); ASSERT_STR_EQUAL_STRONG(utf8_n_substring_beg(utf8_str2, '\n', 6), NULL); #endif const char *beg = NULL; const char *end = NULL; str_n_substring(empty_str, '\n', 0, &beg, &end); assert_true(beg == empty_str && end == empty_str + strlen(empty_str)); str_n_substring(empty_str, '\n', 1, &beg, &end); assert_true(beg == NULL && end == NULL); str_n_substring(empty_str, '\n', 2, &beg, &end); assert_true(beg == NULL && end == NULL); #if defined(FT_HAVE_WCHAR) const wchar_t *wbeg = NULL; const wchar_t *wend = NULL; wstr_n_substring(empty_wstr, L'\n', 0, &wbeg, &wend); assert_true(wbeg == empty_wstr && wend == empty_wstr + wcslen(empty_wstr)); wstr_n_substring(empty_wstr, L'\n', 1, &wbeg, &wend); assert_true(wbeg == NULL && wend == NULL); wstr_n_substring(empty_wstr, L'\n', 2, &wbeg, &wend); assert_true(wbeg == NULL && wend == NULL); #endif #ifdef FT_HAVE_UTF8 const void *utf8_beg = NULL; const void *utf8_end = NULL; utf8_n_substring(utf8_empty_str, '\n', 0, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_empty_str && utf8_end == utf8_empty_str + strlen(utf8_empty_str)); utf8_n_substring(utf8_empty_str, '\n', 1, &utf8_beg, &utf8_end); assert_true(utf8_beg == NULL && utf8_end == NULL); utf8_n_substring(utf8_empty_str, '\n', 2, &utf8_beg, &utf8_end); assert_true(utf8_beg == NULL && utf8_end == NULL); #endif str_n_substring(NULL, '\n', 0, &beg, &end); assert_true(beg == NULL && end == NULL); str_n_substring(str, '\n', 0, &beg, &end); assert_true(beg == str && end == str + 3); str_n_substring(str, '2', 0, &beg, &end); assert_true(beg == str && end == str + 1); #if defined(FT_HAVE_WCHAR) wstr_n_substring(NULL, L'\n', 0, &wbeg, &wend); assert_true(wbeg == NULL && wend == NULL); wstr_n_substring(wstr, L'\n', 0, &wbeg, &wend); assert_true(wbeg == wstr && wend == wstr + 3); wstr_n_substring(wstr, L'2', 0, &wbeg, &wend); assert_true(wbeg == wstr && wend == wstr + 1); #endif #ifdef FT_HAVE_UTF8 utf8_n_substring(NULL, '\n', 0, &utf8_beg, &utf8_end); assert_true(utf8_beg == NULL && utf8_end == NULL); utf8_n_substring(utf8_str, '\n', 0, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str && utf8_end == utf8_str + 3); utf8_n_substring(utf8_str, '2', 0, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str && utf8_end == utf8_str + 1); #endif str_n_substring(str, '\n', 1, &beg, &end); assert_true(beg == str + 4 && end == str + 8); str_n_substring(str, '\n', 2, &beg, &end); assert_true(beg == str + 9 && end == str + strlen(str)); str_n_substring(str, '\n', 3, &beg, &end); assert_true(beg == NULL && end == NULL); #if defined(FT_HAVE_WCHAR) wstr_n_substring(wstr, L'\n', 1, &wbeg, &wend); assert_true(wbeg == wstr + 4 && wend == wstr + 8); wstr_n_substring(wstr, L'\n', 2, &wbeg, &wend); assert_true(wbeg == wstr + 9 && wend == wstr + wcslen(wstr)); wstr_n_substring(wstr, L'\n', 3, &wbeg, &wend); assert_true(wbeg == NULL && wend == NULL); #endif #ifdef FT_HAVE_UTF8 utf8_n_substring(utf8_str, '\n', 1, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str + 4 && utf8_end == utf8_str + 8); utf8_n_substring(utf8_str, '\n', 2, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str + 9 && utf8_end == utf8_str + strlen(utf8_str)); utf8_n_substring(utf8_str, '\n', 3, &utf8_beg, &utf8_end); assert_true(utf8_beg == NULL && end == NULL); #endif str_n_substring(str2, '\n', 0, &beg, &end); assert_true(beg == str2 && end == str2); str_n_substring(str2, '\n', 1, &beg, &end); assert_true(beg == str2 + 1 && end == str2 + 4); str_n_substring(str2, '\n', 2, &beg, &end); assert_true(beg == str2 + 5 && end == str2 + 7); str_n_substring(str2, '\n', 3, &beg, &end); assert_true(beg == str2 + 8 && end == str2 + 8); str_n_substring(str2, '\n', 4, &beg, &end); assert_true(beg == str2 + 9 && end == str2 + 10); str_n_substring(str2, '\n', 5, &beg, &end); assert_true(beg == str2 + 11 && end == str2 + 11); str_n_substring(str2, '\n', 6, &beg, &end); assert_true(beg == NULL && end == NULL); #if defined(FT_HAVE_WCHAR) wstr_n_substring(wstr2, L'\xff0f', 0, &wbeg, &wend); assert_true(wbeg == wstr2 && wend == wstr2); wstr_n_substring(wstr2, L'\xff0f', 1, &wbeg, &wend); assert_true(wbeg == wstr2 + 1 && wend == wstr2 + 4); wstr_n_substring(wstr2, L'\xff0f', 2, &wbeg, &wend); assert_true(wbeg == wstr2 + 5 && wend == wstr2 + 7); wstr_n_substring(wstr2, L'\xff0f', 3, &wbeg, &wend); assert_true(wbeg == wstr2 + 8 && wend == wstr2 + 8); wstr_n_substring(wstr2, L'\xff0f', 4, &wbeg, &wend); assert_true(wbeg == wstr2 + 9 && wend == wstr2 + 10); wstr_n_substring(wstr2, L'\xff0f', 5, &wbeg, &wend); assert_true(wbeg == wstr2 + 11 && wend == wstr2 + 11); wstr_n_substring(wstr2, L'\xff0f', 6, &wbeg, &wend); assert_true(wbeg == NULL && wend == NULL); #endif #if defined(FT_HAVE_WCHAR) utf8_n_substring(utf8_str2, '\n', 0, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str2 && utf8_end == utf8_str2); utf8_n_substring(utf8_str2, '\n', 1, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str2 + 1 && utf8_end == utf8_str2 + 4); utf8_n_substring(utf8_str2, '\n', 2, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str2 + 5 && utf8_end == utf8_str2 + 7); utf8_n_substring(utf8_str2, '\n', 3, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str2 + 8 && utf8_end == utf8_str2 + 8); utf8_n_substring(utf8_str2, '\n', 4, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str2 + 9 && utf8_end == utf8_str2 + 10); utf8_n_substring(utf8_str2, '\n', 5, &utf8_beg, &utf8_end); assert_true(utf8_beg == utf8_str2 + 11 && utf8_end == utf8_str2 + 11); utf8_n_substring(utf8_str2, '\n', 6, &utf8_beg, &utf8_end); assert_true(utf8_beg == NULL && utf8_end == NULL); #endif } #if defined(FT_HAVE_UTF8) /* Custom function to compute visible width of utf8 strings */ int u8strwid(const void *beg, const void *end, size_t *width) { const char *custom_str = "custom_string"; const size_t raw_len = (const char *)end - (const char *)beg; if (memcmp(beg, custom_str, MIN(strlen(custom_str), raw_len)) == 0) { *width = 25; return 0; } return 1; } #endif void test_buffer_text_visible_width(void) { f_string_buffer_t *buffer = create_string_buffer(200, CHAR_BUF); buffer->type = CHAR_BUF; char *old_value = buffer->str.cstr; buffer->str.cstr = (char *)""; assert_true(buffer_text_visible_width(buffer) == 0); buffer->str.cstr = (char *)"\n\n\n\n"; assert_true(buffer_text_visible_width(buffer) == 0); buffer->str.cstr = (char *)"12345"; assert_true(buffer_text_visible_width(buffer) == 5); buffer->str.cstr = (char *)"12345\n1234567"; assert_true(buffer_text_visible_width(buffer) == 7); buffer->str.cstr = (char *)"12345\n1234567\n"; assert_true(buffer_text_visible_width(buffer) == 7); buffer->str.cstr = (char *)"12345\n1234567\n123"; assert_true(buffer_text_visible_width(buffer) == 7); #if defined(FT_HAVE_WCHAR) buffer->type = W_CHAR_BUF; buffer->str.wstr = (wchar_t *)L""; assert_true(buffer_text_visible_width(buffer) == 0); buffer->str.wstr = (wchar_t *)L"\n\n\n\n"; assert_true(buffer_text_visible_width(buffer) == 0); buffer->str.wstr = (wchar_t *)L"12345"; assert_true(buffer_text_visible_width(buffer) == 5); buffer->str.wstr = (wchar_t *)L"12345\n1234567"; assert_true(buffer_text_visible_width(buffer) == 7); buffer->str.wstr = (wchar_t *)L"12345\n1234567\n"; assert_true(buffer_text_visible_width(buffer) == 7); buffer->str.wstr = (wchar_t *)L"12345\n1234567\n123"; assert_true(buffer_text_visible_width(buffer) == 7); #endif #if defined(FT_HAVE_UTF8) buffer->type = UTF8_BUF; buffer->str.u8str = (void *)""; assert_true(buffer_text_visible_width(buffer) == 0); buffer->str.u8str = (void *)"\n\n\n\n"; assert_true(buffer_text_visible_width(buffer) == 0); buffer->str.u8str = (void *)"12345"; assert_true(buffer_text_visible_width(buffer) == 5); buffer->str.u8str = (void *)"12345\n1234567"; assert_true(buffer_text_visible_width(buffer) == 7); buffer->str.u8str = (void *)"12345\n1234567\n"; assert_true(buffer_text_visible_width(buffer) == 7); buffer->str.u8str = (void *)"12345\n1234567\n123"; assert_true(buffer_text_visible_width(buffer) == 7); /* panagrams from http://clagnut.com/blog/2380/ */ /* 10 20 30 40 50 60 70 80 90 100 110 */ buffer->str.u8str = (void *)"Numbers 01234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890"; assert_true(buffer_text_visible_width(buffer) == 110); buffer->str.u8str = (void *)"Chinese 視野無限廣, 窗外有藍天"; assert_true(buffer_text_visible_width(buffer) == 32); buffer->str.u8str = (void *)"German Falsches Üben von Xylophonmusik quält jeden größeren Zwerg"; assert_true(buffer_text_visible_width(buffer) == 68); buffer->str.u8str = (void *)"Greek Ταχίστη αλώπηξ βαφής ψημένη γη, δρασκελίζει υπέρ νωθρού κυνός Takhístè"; assert_true(buffer_text_visible_width(buffer) == 80); buffer->str.u8str = (void *)"Irish D’ḟuascail Íosa Úrṁac na hÓiġe Beannaiṫe pór Éaḃa agus Áḋaiṁ"; assert_true(buffer_text_visible_width(buffer) == 70); buffer->str.u8str = (void *)"Japanese いろはにほへと ちりぬるを わかよたれそ つねならむ うゐ"; assert_true(buffer_text_visible_width(buffer) == 64); buffer->str.u8str = (void *)"Polish Pójdźże, kiń tę chmurność w głąb flaszy"; assert_true(buffer_text_visible_width(buffer) == 49); buffer->str.u8str = (void *)"Portuguese Luís argüia à Júlia que «brações, fé, chá, óxido, pôr, zângão» eram palavras do português"; assert_true(buffer_text_visible_width(buffer) == 100); buffer->str.u8str = (void *)"Russian Съешь же ещё этих мягких французских булок, да выпей чаю"; assert_true(buffer_text_visible_width(buffer) == 66); buffer->str.u8str = (void *)"Spanish Benjamín pidió una bebida de kiwi y fresa; Noé, sin vergüenza, la más exquisita champaña del menú"; assert_true(buffer_text_visible_width(buffer) == 107); buffer->str.u8str = (void *)"Turkish Vakfın çoğu bu huysuz genci plajda görmüştü"; assert_true(buffer_text_visible_width(buffer) == 53); /* 10 20 30 40 50 60 70 80 90 100 110 */ buffer->str.u8str = (void *)"Numbers 01234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890"; assert_true(buffer_text_visible_width(buffer) == 110); buffer->str.u8str = (void *)"Chinese 視野無限廣,\n 窗外有藍天"; assert_true(buffer_text_visible_width(buffer) == 21); buffer->str.u8str = (void *)"German Falsches Üben von Xy\nlophonmusik quält \njeden größeren Zwerg"; assert_true(buffer_text_visible_width(buffer) == 30); buffer->str.u8str = (void *)"Greek Ταχίστη αλώπηξ βαφής\n ψημένη γη, δρασκελίζει\n υπέρ νωθρού \nκυνός Takhístè"; assert_true(buffer_text_visible_width(buffer) == 30); buffer->str.u8str = (void *)"Irish D’ḟuascail Íosa Úrṁa\nc na hÓiġe Beannaiṫe\n pór Éaḃa agus Áḋaiṁ"; assert_true(buffer_text_visible_width(buffer) == 30); buffer->str.u8str = (void *)"Japanese いろはにほへと ちり\nぬるを わかよたれそ つねならむ うゐ"; assert_true(buffer_text_visible_width(buffer) == 35); buffer->str.u8str = (void *)"Polish Pójdźże, kiń tę chmu\nrność w głąb flaszy"; assert_true(buffer_text_visible_width(buffer) == 30); buffer->str.u8str = (void *)"Portuguese Luís argüia à Júlia\n que «brações, fé, chá,\n óxido, pôr, \nzângão» eram palavras\n do português"; assert_true(buffer_text_visible_width(buffer) == 30); buffer->str.u8str = (void *)"Russian Съешь же ещё этих мя\nгких французских булок,\n да выпей чаю"; assert_true(buffer_text_visible_width(buffer) == 30); buffer->str.u8str = (void *)"Spanish Benjamín pidió una b\nebida de kiwi y fresa;\n Noé, sin vergüenza,\n la más exquisita\n champaña del menú"; assert_true(buffer_text_visible_width(buffer) == 30); buffer->str.u8str = (void *)"Turkish Vakfın çoğu bu huysu\nz genci plajda gö\nrmüştü"; assert_true(buffer_text_visible_width(buffer) == 30); /* Test custom width function for utf8 strings */ ft_set_u8strwid_func(&u8strwid); buffer->str.u8str = (void *)"custom_string"; assert_true(buffer_text_visible_width(buffer) == 25); buffer->str.u8str = (void *)"123456789012345678901234\ncustom_string"; assert_true(buffer_text_visible_width(buffer) == 25); buffer->str.u8str = (void *)"12345678901234567890123456\ncustom_string"; assert_true(buffer_text_visible_width(buffer) == 26); buffer->str.u8str = (void *)"common_string"; assert_true(buffer_text_visible_width(buffer) == 13); ft_set_u8strwid_func(NULL); #endif buffer->type = CHAR_BUF; buffer->str.cstr = old_value; destroy_string_buffer(buffer); } void test_buffer_text_visible_height(void) { f_string_buffer_t *buffer = create_string_buffer(200, CHAR_BUF); buffer->type = CHAR_BUF; char *old_value = buffer->str.cstr; buffer->str.cstr = (char *)""; assert_true(buffer_text_visible_height(buffer) == 0); buffer->str.cstr = (char *)"\n"; assert_true(buffer_text_visible_height(buffer) == 2); buffer->str.cstr = (char *)"\n\n"; assert_true(buffer_text_visible_height(buffer) == 3); buffer->str.cstr = (char *)"\n\n\n\n"; assert_true(buffer_text_visible_height(buffer) == 5); buffer->str.cstr = (char *)"12345"; assert_true(buffer_text_visible_height(buffer) == 1); buffer->str.cstr = (char *)"\n12345"; assert_true(buffer_text_visible_height(buffer) == 2); buffer->str.cstr = (char *)"\n12345\n\n2"; assert_true(buffer_text_visible_height(buffer) == 4); #if defined(FT_HAVE_WCHAR) buffer->type = W_CHAR_BUF; buffer->str.wstr = (wchar_t *)L""; assert_true(buffer_text_visible_height(buffer) == 0); buffer->str.wstr = (wchar_t *)L"\n"; assert_true(buffer_text_visible_height(buffer) == 2); buffer->str.wstr = (wchar_t *)L"\n\n"; assert_true(buffer_text_visible_height(buffer) == 3); buffer->str.wstr = (wchar_t *)L"\n\n\n\n"; assert_true(buffer_text_visible_height(buffer) == 5); buffer->str.wstr = (wchar_t *)L"\xff0fy2345\xff0f"; assert_true(buffer_text_visible_height(buffer) == 1); buffer->str.wstr = (wchar_t *)L"\n12345"; assert_true(buffer_text_visible_height(buffer) == 2); buffer->str.wstr = (wchar_t *)L"\n12345\n\n2"; assert_true(buffer_text_visible_height(buffer) == 4); #endif #if defined(FT_HAVE_UTF8) buffer->type = UTF8_BUF; buffer->str.u8str = (void *)""; assert_true(buffer_text_visible_height(buffer) == 0); buffer->str.u8str = (void *)"\n"; assert_true(buffer_text_visible_height(buffer) == 2); buffer->str.u8str = (void *)"\n\n"; assert_true(buffer_text_visible_height(buffer) == 3); buffer->str.u8str = (void *)"\n\n\n\n"; assert_true(buffer_text_visible_height(buffer) == 5); buffer->str.u8str = (void *)"12345"; assert_true(buffer_text_visible_height(buffer) == 1); buffer->str.u8str = (void *)"\n12345"; assert_true(buffer_text_visible_height(buffer) == 2); buffer->str.u8str = (void *)"\n12345\n\n2"; assert_true(buffer_text_visible_height(buffer) == 4); #endif buffer->type = CHAR_BUF; buffer->str.cstr = old_value; destroy_string_buffer(buffer); } #if defined(FT_HAVE_WCHAR) void test_wchar_basics(void) { #if !defined(FT_MICROSOFT_COMPILER) assert_true(mk_wcswidth(L"", 0) == 0); assert_true(mk_wcswidth(L"1", 0) == 0); assert_true(mk_wcswidth(L"1", 1) == 1); assert_true(mk_wcswidth(L"λ", 1) == 1); // Greek assert_true(mk_wcswidth(L"ऋ", 1) == 1); // Hindi assert_true(mk_wcswidth(L"ź", 1) == 1); // Polish assert_true(mk_wcswidth(L"ü", 1) == 1); // Portuguese assert_true(mk_wcswidth(L"ц", 1) == 1); // Russian assert_true(mk_wcswidth(L"ñ", 1) == 1); // Spanish assert_true(mk_wcswidth(L"ğ", 1) == 1); // Turkish assert_true(mk_wcswidth(L"ФЦУЙъхЭЯЧьЮЪЁ", 13) == 13); assert_true(mk_wcswidth(L"ФЦУЙъхЭЯЧьЮЪЁ", 14) == 13); assert_true(mk_wcswidth(L"ФЦУЙъхЭЯЧьЮЪЁ", 10) == 10); /* panagrams from http://clagnut.com/blog/2380/ */ const wchar_t *str = NULL; /* 10 20 30 40 50 60 70 80 90 100 110 */ str = L"Numbers 01234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890"; assert_true(mk_wcswidth(str, 500) == 110); str = L"German Falsches Üben von Xylophonmusik quält jeden größeren Zwerg"; assert_true(mk_wcswidth(str, 500) == 68); str = L"Greek Ταχίστη αλώπηξ βαφής ψημένη γη, δρασκελίζει υπέρ νωθρού κυνός Takhístè"; assert_true(mk_wcswidth(str, 500) == 80); str = L"Irish D’ḟuascail Íosa Úrṁac na hÓiġe Beannaiṫe pór Éaḃa agus Áḋaiṁ"; assert_true(mk_wcswidth(str, 500) == 70); str = L"Polish Pójdźże, kiń tę chmurność w głąb flaszy"; assert_true(mk_wcswidth(str, 500) == 49); str = L"Portuguese Luís argüia à Júlia que «brações, fé, chá, óxido, pôr, zângão» eram palavras do português"; assert_true(mk_wcswidth(str, 500) == 100); str = L"Russian Съешь же ещё этих мягких французских булок, да выпей чаю"; assert_true(mk_wcswidth(str, 500) == 66); str = L"Spanish Benjamín pidió una bebida de kiwi y fresa; Noé, sin vergüenza, la más exquisita champaña del menú"; assert_true(mk_wcswidth(str, 500) == 107); str = L"Turkish Vakfın çoğu bu huysuz genci plajda görmüştü"; assert_true(mk_wcswidth(str, 500) == 53); #endif } #endif static void test_print_n_strings_(const char *str, size_t n) { int sz = n * strlen(str); { f_string_buffer_t *buffer = create_string_buffer(200, CHAR_BUF); const char *origin = (char *)buffer_get_data(buffer); f_conv_context_t cntx; cntx.u.buf = (char *)buffer_get_data(buffer); cntx.raw_avail = 200; cntx.b_type = CHAR_BUF; assert_true(print_n_strings(&cntx, n, str) == sz); assert_true(cntx.u.buf - origin == (ptrdiff_t)sz); destroy_string_buffer(buffer); } #ifdef FT_HAVE_WCHAR { f_string_buffer_t *buffer = create_string_buffer(200, W_CHAR_BUF); const char *origin = (char *)buffer_get_data(buffer); f_conv_context_t cntx; cntx.u.buf = (char *)buffer_get_data(buffer); cntx.raw_avail = 200; cntx.b_type = W_CHAR_BUF; assert_true(print_n_strings(&cntx, n, str) == /*sizeof(wchar_t) **/ sz); assert_true(cntx.u.buf - origin == (ptrdiff_t)sizeof(wchar_t) * sz); destroy_string_buffer(buffer); } #endif /* FT_HAVE_WCHAR */ #ifdef FT_HAVE_UTF8 { f_string_buffer_t *buffer = create_string_buffer(200, UTF8_BUF); const char *origin = (char *)buffer_get_data(buffer); f_conv_context_t cntx; cntx.u.buf = (char *)buffer_get_data(buffer); cntx.raw_avail = 200; cntx.b_type = UTF8_BUF; assert_true(print_n_strings(&cntx, n, str) == sz); assert_true(cntx.u.buf - origin == (ptrdiff_t)sz); destroy_string_buffer(buffer); } #endif /* FT_HAVE_UTF8 */ } void test_print_n_strings(void) { test_print_n_strings_("", 0); test_print_n_strings_("", 1); test_print_n_strings_("", 2); test_print_n_strings_(" ", 0); test_print_n_strings_(" ", 1); test_print_n_strings_(" ", 2); test_print_n_strings_("foo", 0); test_print_n_strings_("foo", 1); test_print_n_strings_("foo", 2); }