1
0

cwchar.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. /*
  2. * Copyright (c) 2006-2022, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2014-01-01 mattn implement wcwidth
  9. */
  10. #include "posix/wchar.h"
  11. #include <string.h>
  12. struct interval
  13. {
  14. long first;
  15. long last;
  16. };
  17. static int bisearch(wchar_t ucs, const struct interval *table, int max)
  18. {
  19. int min = 0;
  20. int mid;
  21. if (ucs < table[0].first || ucs > table[max].last)
  22. {
  23. return 0;
  24. }
  25. while (max >= min)
  26. {
  27. mid = (min + max) / 2;
  28. if (ucs > table[mid].last)
  29. {
  30. min = mid + 1;
  31. }
  32. else if (ucs < table[mid].first)
  33. {
  34. max = mid - 1;
  35. }
  36. else
  37. {
  38. return 1;
  39. }
  40. }
  41. return 0;
  42. }
  43. int wcwidth(wchar_t ucs)
  44. {
  45. /* sorted list of non-overlapping intervals of non-spacing characters */
  46. static const struct interval combining[] = {
  47. { 0x0300, 0x034E }, { 0x0360, 0x0362 }, { 0x0483, 0x0486 },
  48. { 0x0488, 0x0489 }, { 0x0591, 0x05A1 }, { 0x05A3, 0x05B9 },
  49. { 0x05BB, 0x05BD }, { 0x05BF, 0x05BF }, { 0x05C1, 0x05C2 },
  50. { 0x05C4, 0x05C4 }, { 0x064B, 0x0655 }, { 0x0670, 0x0670 },
  51. { 0x06D6, 0x06E4 }, { 0x06E7, 0x06E8 }, { 0x06EA, 0x06ED },
  52. { 0x070F, 0x070F }, { 0x0711, 0x0711 }, { 0x0730, 0x074A },
  53. { 0x07A6, 0x07B0 }, { 0x0901, 0x0902 }, { 0x093C, 0x093C },
  54. { 0x0941, 0x0948 }, { 0x094D, 0x094D }, { 0x0951, 0x0954 },
  55. { 0x0962, 0x0963 }, { 0x0981, 0x0981 }, { 0x09BC, 0x09BC },
  56. { 0x09C1, 0x09C4 }, { 0x09CD, 0x09CD }, { 0x09E2, 0x09E3 },
  57. { 0x0A02, 0x0A02 }, { 0x0A3C, 0x0A3C }, { 0x0A41, 0x0A42 },
  58. { 0x0A47, 0x0A48 }, { 0x0A4B, 0x0A4D }, { 0x0A70, 0x0A71 },
  59. { 0x0A81, 0x0A82 }, { 0x0ABC, 0x0ABC }, { 0x0AC1, 0x0AC5 },
  60. { 0x0AC7, 0x0AC8 }, { 0x0ACD, 0x0ACD }, { 0x0B01, 0x0B01 },
  61. { 0x0B3C, 0x0B3C }, { 0x0B3F, 0x0B3F }, { 0x0B41, 0x0B43 },
  62. { 0x0B4D, 0x0B4D }, { 0x0B56, 0x0B56 }, { 0x0B82, 0x0B82 },
  63. { 0x0BC0, 0x0BC0 }, { 0x0BCD, 0x0BCD }, { 0x0C3E, 0x0C40 },
  64. { 0x0C46, 0x0C48 }, { 0x0C4A, 0x0C4D }, { 0x0C55, 0x0C56 },
  65. { 0x0CBF, 0x0CBF }, { 0x0CC6, 0x0CC6 }, { 0x0CCC, 0x0CCD },
  66. { 0x0D41, 0x0D43 }, { 0x0D4D, 0x0D4D }, { 0x0DCA, 0x0DCA },
  67. { 0x0DD2, 0x0DD4 }, { 0x0DD6, 0x0DD6 }, { 0x0E31, 0x0E31 },
  68. { 0x0E34, 0x0E3A }, { 0x0E47, 0x0E4E }, { 0x0EB1, 0x0EB1 },
  69. { 0x0EB4, 0x0EB9 }, { 0x0EBB, 0x0EBC }, { 0x0EC8, 0x0ECD },
  70. { 0x0F18, 0x0F19 }, { 0x0F35, 0x0F35 }, { 0x0F37, 0x0F37 },
  71. { 0x0F39, 0x0F39 }, { 0x0F71, 0x0F7E }, { 0x0F80, 0x0F84 },
  72. { 0x0F86, 0x0F87 }, { 0x0F90, 0x0F97 }, { 0x0F99, 0x0FBC },
  73. { 0x0FC6, 0x0FC6 }, { 0x102D, 0x1030 }, { 0x1032, 0x1032 },
  74. { 0x1036, 0x1037 }, { 0x1039, 0x1039 }, { 0x1058, 0x1059 },
  75. { 0x1160, 0x11FF }, { 0x17B7, 0x17BD }, { 0x17C6, 0x17C6 },
  76. { 0x17C9, 0x17D3 }, { 0x180B, 0x180E }, { 0x18A9, 0x18A9 },
  77. { 0x200B, 0x200F }, { 0x202A, 0x202E }, { 0x206A, 0x206F },
  78. { 0x20D0, 0x20E3 }, { 0x302A, 0x302F }, { 0x3099, 0x309A },
  79. { 0xFB1E, 0xFB1E }, { 0xFE20, 0xFE23 }, { 0xFEFF, 0xFEFF },
  80. { 0xFFF9, 0xFFFB }
  81. };
  82. /* test for 8-bit control characters */
  83. if (ucs == 0)
  84. {
  85. return 0;
  86. }
  87. if (ucs < 32 || (ucs >= 0x7f && ucs < 0xa0))
  88. {
  89. return -1;
  90. }
  91. /* binary search in table of non-spacing characters */
  92. if (bisearch(ucs, combining, sizeof(combining) / sizeof(struct interval) - 1))
  93. {
  94. return 0;
  95. }
  96. return 1 +
  97. (ucs >= 0x1100 &&
  98. (ucs <= 0x115f || /* Hangul Jamo init. consonants */
  99. (ucs >= 0x2e80 && ucs <= 0xa4cf && (ucs & ~0x0011) != 0x300a &&
  100. ucs != 0x303f) || /* CJK ... Yi */
  101. (ucs >= 0xac00 && ucs <= 0xd7a3) || /* Hangul Syllables */
  102. (ucs >= 0xf900 && ucs <= 0xfaff) || /* CJK Compatibility Ideographs */
  103. (ucs >= 0xfe30 && ucs <= 0xfe6f) || /* CJK Compatibility Forms */
  104. (ucs >= 0xff00 && ucs <= 0xff5f) || /* Fullwidth Forms */
  105. (ucs >= 0xffe0 && ucs <= 0xffe6) // ||
  106. //#ifndef _WIN32
  107. // (ucs >= 0x20000 && ucs <= 0x2ffff)
  108. //#else
  109. // 0
  110. //#endif
  111. ));
  112. }
  113. int wcswidth(const wchar_t *pwcs, size_t n)
  114. {
  115. int w, width = 0;
  116. for (;*pwcs && n-- > 0; pwcs++)
  117. {
  118. if ((w = wcwidth(*pwcs)) < 0)
  119. {
  120. return -1;
  121. }
  122. else
  123. {
  124. width += w;
  125. }
  126. }
  127. return width;
  128. }