text_utils.h 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. #ifndef TEXT_UTILS_H
  2. #define TEXT_UTILS_H
  3. #include <string.h>
  4. #include <stdlib.h>
  5. static char *my_strdup(const char *s)
  6. {
  7. if (!s) return NULL;
  8. size_t n = strlen(s) + 1;
  9. char *p = malloc(n);
  10. if (!p) return NULL;
  11. memcpy(p, s, n);
  12. return p;
  13. }
  14. static char *WrapText(Font font, const char *text, float fontSize,
  15. float spacing, float maxWidth)
  16. {
  17. if (!text || text[0] == '\0') return my_strdup("");
  18. char *result = my_strdup(text);
  19. if (!result) return NULL;
  20. int length = strlen(result);
  21. int lastSpaceIdx = -1;
  22. int currentLineStartIdx = 0;
  23. for (int i = 0; i < length; i++)
  24. {
  25. if (result[i] == ' ') {
  26. lastSpaceIdx = i;
  27. }
  28. else if (result[i] == '\n') {
  29. currentLineStartIdx = i + 1;
  30. lastSpaceIdx = -1;
  31. continue;
  32. }
  33. char savedChar = result[i + 1];
  34. result[i + 1] = '\0';
  35. Vector2 size = MeasureTextEx(font, &result[currentLineStartIdx],
  36. fontSize, spacing);
  37. result[i + 1] = savedChar;
  38. if (size.x > maxWidth)
  39. {
  40. if (lastSpaceIdx != -1 && lastSpaceIdx >= currentLineStartIdx)
  41. {
  42. result[lastSpaceIdx] = '\n';
  43. currentLineStartIdx = lastSpaceIdx + 1;
  44. lastSpaceIdx = -1;
  45. i = currentLineStartIdx - 1;
  46. }
  47. else
  48. {
  49. if (i > currentLineStartIdx)
  50. {
  51. result[i] = '\n';
  52. currentLineStartIdx = i;
  53. lastSpaceIdx = -1;
  54. }
  55. }
  56. }
  57. }
  58. return result;
  59. }
  60. static int GetUtf8ByteLength(const char *text, int count)
  61. {
  62. if (!text || count <= 0) return 0;
  63. int byte_len = 0;
  64. int glyph_count = 0;
  65. while (text[byte_len] != '\0' && glyph_count < count)
  66. {
  67. unsigned char c = (unsigned char)text[byte_len];
  68. int current_char_len = 1;
  69. if (c >= 0xF0) current_char_len = 4; // Emoji (unsupported)
  70. else if (c >= 0xE0) current_char_len = 3; // CJK
  71. else if (c >= 0xC0) current_char_len = 2; // Cyryllic
  72. else current_char_len = 1; // ASCII
  73. for (int i = 0; i < current_char_len; i++) {
  74. if (text[byte_len + i] == '\0') {
  75. return byte_len;
  76. }
  77. }
  78. // Skipping emojis...
  79. if (current_char_len == 4) {
  80. byte_len += 4;
  81. continue;
  82. }
  83. byte_len += current_char_len;
  84. glyph_count++;
  85. }
  86. return byte_len;
  87. }
  88. #endif