textElement.test.ts 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. import { BOUND_TEXT_PADDING, FONT_FAMILY } from "../constants";
  2. import { API } from "../tests/helpers/api";
  3. import {
  4. computeContainerDimensionForBoundText,
  5. getContainerCoords,
  6. getMaxContainerWidth,
  7. getMaxContainerHeight,
  8. wrapText,
  9. detectLineHeight,
  10. getLineHeightInPx,
  11. getDefaultLineHeight,
  12. } from "./textElement";
  13. import { FontString } from "./types";
  14. describe("Test wrapText", () => {
  15. const font = "20px Cascadia, width: Segoe UI Emoji" as FontString;
  16. it("shouldn't add new lines for trailing spaces", () => {
  17. const text = "Hello whats up ";
  18. const maxWidth = 200 - BOUND_TEXT_PADDING * 2;
  19. const res = wrapText(text, font, maxWidth);
  20. expect(res).toBe(text);
  21. });
  22. it("should work with emojis", () => {
  23. const text = "😀";
  24. const maxWidth = 1;
  25. const res = wrapText(text, font, maxWidth);
  26. expect(res).toBe("😀");
  27. });
  28. it("should show the text correctly when max width reached", () => {
  29. const text = "Hello😀";
  30. const maxWidth = 10;
  31. const res = wrapText(text, font, maxWidth);
  32. expect(res).toBe("H\ne\nl\nl\no\n😀");
  33. });
  34. describe("When text doesn't contain new lines", () => {
  35. const text = "Hello whats up";
  36. [
  37. {
  38. desc: "break all words when width of each word is less than container width",
  39. width: 80,
  40. res: `Hello \nwhats \nup`,
  41. },
  42. {
  43. desc: "break all characters when width of each character is less than container width",
  44. width: 25,
  45. res: `H
  46. e
  47. l
  48. l
  49. o
  50. w
  51. h
  52. a
  53. t
  54. s
  55. u
  56. p`,
  57. },
  58. {
  59. desc: "break words as per the width",
  60. width: 140,
  61. res: `Hello whats \nup`,
  62. },
  63. {
  64. desc: "fit the container",
  65. width: 250,
  66. res: "Hello whats up",
  67. },
  68. {
  69. desc: "should push the word if its equal to max width",
  70. width: 60,
  71. res: `Hello
  72. whats
  73. up`,
  74. },
  75. ].forEach((data) => {
  76. it(`should ${data.desc}`, () => {
  77. const res = wrapText(text, font, data.width - BOUND_TEXT_PADDING * 2);
  78. expect(res).toEqual(data.res);
  79. });
  80. });
  81. });
  82. describe("When text contain new lines", () => {
  83. const text = `Hello
  84. whats up`;
  85. [
  86. {
  87. desc: "break all words when width of each word is less than container width",
  88. width: 80,
  89. res: `Hello\nwhats \nup`,
  90. },
  91. {
  92. desc: "break all characters when width of each character is less than container width",
  93. width: 25,
  94. res: `H
  95. e
  96. l
  97. l
  98. o
  99. w
  100. h
  101. a
  102. t
  103. s
  104. u
  105. p`,
  106. },
  107. {
  108. desc: "break words as per the width",
  109. width: 150,
  110. res: `Hello
  111. whats up`,
  112. },
  113. {
  114. desc: "fit the container",
  115. width: 250,
  116. res: `Hello
  117. whats up`,
  118. },
  119. ].forEach((data) => {
  120. it(`should respect new lines and ${data.desc}`, () => {
  121. const res = wrapText(text, font, data.width - BOUND_TEXT_PADDING * 2);
  122. expect(res).toEqual(data.res);
  123. });
  124. });
  125. });
  126. describe("When text is long", () => {
  127. const text = `hellolongtextthisiswhatsupwithyouIamtypingggggandtypinggg break it now`;
  128. [
  129. {
  130. desc: "fit characters of long string as per container width",
  131. width: 170,
  132. res: `hellolongtextth\nisiswhatsupwith\nyouIamtypingggg\ngandtypinggg \nbreak it now`,
  133. },
  134. {
  135. desc: "fit characters of long string as per container width and break words as per the width",
  136. width: 130,
  137. res: `hellolongte
  138. xtthisiswha
  139. tsupwithyou
  140. Iamtypinggg
  141. ggandtyping
  142. gg break it
  143. now`,
  144. },
  145. {
  146. desc: "fit the long text when container width is greater than text length and move the rest to next line",
  147. width: 600,
  148. res: `hellolongtextthisiswhatsupwithyouIamtypingggggandtypinggg \nbreak it now`,
  149. },
  150. ].forEach((data) => {
  151. it(`should ${data.desc}`, () => {
  152. const res = wrapText(text, font, data.width - BOUND_TEXT_PADDING * 2);
  153. expect(res).toEqual(data.res);
  154. });
  155. });
  156. });
  157. it("should wrap the text correctly when word length is exactly equal to max width", () => {
  158. const text = "Hello Excalidraw";
  159. // Length of "Excalidraw" is 100 and exacty equal to max width
  160. const res = wrapText(text, font, 100);
  161. expect(res).toEqual(`Hello \nExcalidraw`);
  162. });
  163. it("should return the text as is if max width is invalid", () => {
  164. const text = "Hello Excalidraw";
  165. expect(wrapText(text, font, NaN)).toEqual(text);
  166. expect(wrapText(text, font, -1)).toEqual(text);
  167. expect(wrapText(text, font, Infinity)).toEqual(text);
  168. });
  169. });
  170. describe("Test measureText", () => {
  171. describe("Test getContainerCoords", () => {
  172. const params = { width: 200, height: 100, x: 10, y: 20 };
  173. it("should compute coords correctly when ellipse", () => {
  174. const element = API.createElement({
  175. type: "ellipse",
  176. ...params,
  177. });
  178. expect(getContainerCoords(element)).toEqual({
  179. x: 44.2893218813452455,
  180. y: 39.64466094067262,
  181. });
  182. });
  183. it("should compute coords correctly when rectangle", () => {
  184. const element = API.createElement({
  185. type: "rectangle",
  186. ...params,
  187. });
  188. expect(getContainerCoords(element)).toEqual({
  189. x: 15,
  190. y: 25,
  191. });
  192. });
  193. it("should compute coords correctly when diamond", () => {
  194. const element = API.createElement({
  195. type: "diamond",
  196. ...params,
  197. });
  198. expect(getContainerCoords(element)).toEqual({
  199. x: 65,
  200. y: 50,
  201. });
  202. });
  203. });
  204. describe("Test computeContainerDimensionForBoundText", () => {
  205. const params = {
  206. width: 178,
  207. height: 194,
  208. };
  209. it("should compute container height correctly for rectangle", () => {
  210. const element = API.createElement({
  211. type: "rectangle",
  212. ...params,
  213. });
  214. expect(computeContainerDimensionForBoundText(150, element.type)).toEqual(
  215. 160,
  216. );
  217. });
  218. it("should compute container height correctly for ellipse", () => {
  219. const element = API.createElement({
  220. type: "ellipse",
  221. ...params,
  222. });
  223. expect(computeContainerDimensionForBoundText(150, element.type)).toEqual(
  224. 226,
  225. );
  226. });
  227. it("should compute container height correctly for diamond", () => {
  228. const element = API.createElement({
  229. type: "diamond",
  230. ...params,
  231. });
  232. expect(computeContainerDimensionForBoundText(150, element.type)).toEqual(
  233. 320,
  234. );
  235. });
  236. });
  237. describe("Test getMaxContainerWidth", () => {
  238. const params = {
  239. width: 178,
  240. height: 194,
  241. };
  242. it("should return max width when container is rectangle", () => {
  243. const container = API.createElement({ type: "rectangle", ...params });
  244. expect(getMaxContainerWidth(container)).toBe(168);
  245. });
  246. it("should return max width when container is ellipse", () => {
  247. const container = API.createElement({ type: "ellipse", ...params });
  248. expect(getMaxContainerWidth(container)).toBe(116);
  249. });
  250. it("should return max width when container is diamond", () => {
  251. const container = API.createElement({ type: "diamond", ...params });
  252. expect(getMaxContainerWidth(container)).toBe(79);
  253. });
  254. });
  255. describe("Test getMaxContainerHeight", () => {
  256. const params = {
  257. width: 178,
  258. height: 194,
  259. };
  260. it("should return max height when container is rectangle", () => {
  261. const container = API.createElement({ type: "rectangle", ...params });
  262. expect(getMaxContainerHeight(container)).toBe(184);
  263. });
  264. it("should return max height when container is ellipse", () => {
  265. const container = API.createElement({ type: "ellipse", ...params });
  266. expect(getMaxContainerHeight(container)).toBe(127);
  267. });
  268. it("should return max height when container is diamond", () => {
  269. const container = API.createElement({ type: "diamond", ...params });
  270. expect(getMaxContainerHeight(container)).toBe(87);
  271. });
  272. });
  273. });
  274. const textElement = API.createElement({
  275. type: "text",
  276. text: "Excalidraw is a\nvirtual \nopensource \nwhiteboard for \nsketching \nhand-drawn like\ndiagrams",
  277. fontSize: 20,
  278. fontFamily: 1,
  279. height: 175,
  280. });
  281. describe("Test detectLineHeight", () => {
  282. it("should return correct line height", () => {
  283. expect(detectLineHeight(textElement)).toBe(1.25);
  284. });
  285. });
  286. describe("Test getLineHeightInPx", () => {
  287. it("should return correct line height", () => {
  288. expect(
  289. getLineHeightInPx(textElement.fontSize, textElement.lineHeight),
  290. ).toBe(25);
  291. });
  292. });
  293. describe("Test getDefaultLineHeight", () => {
  294. it("should return line height using default font family when not passed", () => {
  295. //@ts-ignore
  296. expect(getDefaultLineHeight()).toBe(1.25);
  297. });
  298. it("should return line height using default font family for unknown font", () => {
  299. const UNKNOWN_FONT = 5;
  300. expect(getDefaultLineHeight(UNKNOWN_FONT)).toBe(1.25);
  301. });
  302. it("should return correct line height", () => {
  303. expect(getDefaultLineHeight(FONT_FAMILY.Cascadia)).toBe(1.2);
  304. });
  305. });