@pie-lib/editable-html-tip-tap 2.1.2-next.31 → 2.1.2-next.36

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (280) hide show
  1. package/CHANGELOG.json +32 -0
  2. package/CHANGELOG.md +2532 -0
  3. package/LICENSE.md +5 -0
  4. package/lib/components/CharacterPicker.js +201 -0
  5. package/lib/components/CharacterPicker.js.map +1 -0
  6. package/lib/components/EditableHtml.js +376 -0
  7. package/lib/components/EditableHtml.js.map +1 -0
  8. package/lib/components/MenuBar.js +697 -0
  9. package/lib/components/MenuBar.js.map +1 -0
  10. package/lib/components/TiptapContainer.js +234 -0
  11. package/lib/components/TiptapContainer.js.map +1 -0
  12. package/lib/components/characters/characterUtils.js +378 -0
  13. package/lib/components/characters/characterUtils.js.map +1 -0
  14. package/lib/components/characters/custom-popper.js +44 -0
  15. package/lib/components/characters/custom-popper.js.map +1 -0
  16. package/lib/components/common/done-button.js +34 -0
  17. package/lib/components/common/done-button.js.map +1 -0
  18. package/lib/components/common/toolbar-buttons.js +144 -0
  19. package/lib/components/common/toolbar-buttons.js.map +1 -0
  20. package/lib/components/icons/CssIcon.js +25 -0
  21. package/lib/components/icons/CssIcon.js.map +1 -0
  22. package/lib/components/icons/RespArea.js +72 -0
  23. package/lib/components/icons/RespArea.js.map +1 -0
  24. package/lib/components/icons/TableIcons.js +53 -0
  25. package/lib/components/icons/TableIcons.js.map +1 -0
  26. package/lib/components/icons/TextAlign.js +157 -0
  27. package/lib/components/icons/TextAlign.js.map +1 -0
  28. package/lib/components/image/AltDialog.js +98 -0
  29. package/lib/components/image/AltDialog.js.map +1 -0
  30. package/lib/components/image/ImageToolbar.js +137 -0
  31. package/lib/components/image/ImageToolbar.js.map +1 -0
  32. package/lib/components/image/InsertImageHandler.js +135 -0
  33. package/lib/components/image/InsertImageHandler.js.map +1 -0
  34. package/lib/components/media/MediaDialog.js +594 -0
  35. package/lib/components/media/MediaDialog.js.map +1 -0
  36. package/lib/components/media/MediaToolbar.js +74 -0
  37. package/lib/components/media/MediaToolbar.js.map +1 -0
  38. package/lib/components/media/MediaWrapper.js +67 -0
  39. package/lib/components/media/MediaWrapper.js.map +1 -0
  40. package/lib/components/respArea/DragInTheBlank/DragInTheBlank.js +84 -0
  41. package/lib/components/respArea/DragInTheBlank/DragInTheBlank.js.map +1 -0
  42. package/lib/components/respArea/DragInTheBlank/choice.js +250 -0
  43. package/lib/components/respArea/DragInTheBlank/choice.js.map +1 -0
  44. package/lib/components/respArea/ExplicitConstructedResponse.js +137 -0
  45. package/lib/components/respArea/ExplicitConstructedResponse.js.map +1 -0
  46. package/lib/components/respArea/InlineDropdown.js +210 -0
  47. package/lib/components/respArea/InlineDropdown.js.map +1 -0
  48. package/lib/components/respArea/MathTemplated.js +130 -0
  49. package/lib/components/respArea/MathTemplated.js.map +1 -0
  50. package/lib/components/respArea/ToolbarIcon.js +81 -0
  51. package/lib/components/respArea/ToolbarIcon.js.map +1 -0
  52. package/lib/components/respArea/inlineDropdownUtils.js +67 -0
  53. package/lib/components/respArea/inlineDropdownUtils.js.map +1 -0
  54. package/lib/constants.js +11 -0
  55. package/lib/constants.js.map +1 -0
  56. package/lib/extensions/css.js +217 -0
  57. package/lib/extensions/css.js.map +1 -0
  58. package/lib/extensions/custom-toolbar-wrapper.js +92 -0
  59. package/lib/extensions/custom-toolbar-wrapper.js.map +1 -0
  60. package/lib/extensions/div-node.js +83 -0
  61. package/lib/extensions/div-node.js.map +1 -0
  62. package/lib/extensions/ensure-empty-root-div.js +48 -0
  63. package/lib/extensions/ensure-empty-root-div.js.map +1 -0
  64. package/lib/extensions/ensure-list-item-content-is-div.js +64 -0
  65. package/lib/extensions/ensure-list-item-content-is-div.js.map +1 -0
  66. package/lib/extensions/extended-list-item.js +15 -0
  67. package/lib/extensions/extended-list-item.js.map +1 -0
  68. package/lib/extensions/extended-table-cell.js +22 -0
  69. package/lib/extensions/extended-table-cell.js.map +1 -0
  70. package/lib/extensions/extended-table.js +75 -0
  71. package/lib/extensions/extended-table.js.map +1 -0
  72. package/lib/extensions/heading-paragraph.js +61 -0
  73. package/lib/extensions/heading-paragraph.js.map +1 -0
  74. package/lib/extensions/image-component.js +348 -0
  75. package/lib/extensions/image-component.js.map +1 -0
  76. package/lib/extensions/image.js +134 -0
  77. package/lib/extensions/image.js.map +1 -0
  78. package/lib/extensions/index.js +46 -0
  79. package/lib/extensions/index.js.map +1 -0
  80. package/lib/extensions/math.js +342 -0
  81. package/lib/extensions/math.js.map +1 -0
  82. package/lib/extensions/media.js +243 -0
  83. package/lib/extensions/media.js.map +1 -0
  84. package/lib/extensions/responseArea.js +446 -0
  85. package/lib/extensions/responseArea.js.map +1 -0
  86. package/lib/index.js +37 -0
  87. package/lib/index.js.map +1 -0
  88. package/lib/styles/editorContainerStyles.js +137 -0
  89. package/lib/styles/editorContainerStyles.js.map +1 -0
  90. package/lib/theme.js +8 -0
  91. package/lib/theme.js.map +1 -0
  92. package/lib/utils/helper.js +73 -0
  93. package/lib/utils/helper.js.map +1 -0
  94. package/lib/utils/size.js +26 -0
  95. package/lib/utils/size.js.map +1 -0
  96. package/lib/utils/toolbar.js +19 -0
  97. package/lib/utils/toolbar.js.map +1 -0
  98. package/package.json +24 -40
  99. package/src/__tests__/EditableHtml.test.jsx +554 -0
  100. package/src/__tests__/constants.test.js +19 -0
  101. package/src/__tests__/div-to-paragraph-conversion.test.jsx +125 -0
  102. package/src/__tests__/extensions.test.js +208 -0
  103. package/src/__tests__/index.test.jsx +154 -0
  104. package/src/__tests__/size-utils.test.js +64 -0
  105. package/src/__tests__/theme.test.js +17 -0
  106. package/src/components/CharacterPicker.jsx +207 -0
  107. package/src/components/EditableHtml.jsx +440 -0
  108. package/src/components/MenuBar.jsx +556 -0
  109. package/src/components/TiptapContainer.jsx +219 -0
  110. package/src/components/__tests__/AltDialog.test.jsx +147 -0
  111. package/src/components/__tests__/CharacterPicker.test.jsx +261 -0
  112. package/src/components/__tests__/CssIcon.test.jsx +46 -0
  113. package/src/components/__tests__/DragInTheBlank.test.jsx +255 -0
  114. package/src/components/__tests__/ExplicitConstructedResponse.test.jsx +209 -0
  115. package/src/components/__tests__/ImageToolbar.test.jsx +128 -0
  116. package/src/components/__tests__/InlineDropdown.test.jsx +393 -0
  117. package/src/components/__tests__/InsertImageHandler.test.js +161 -0
  118. package/src/components/__tests__/MediaDialog.test.jsx +293 -0
  119. package/src/components/__tests__/MediaToolbar.test.jsx +74 -0
  120. package/src/components/__tests__/MediaWrapper.test.jsx +81 -0
  121. package/src/components/__tests__/MenuBar.test.jsx +250 -0
  122. package/src/components/__tests__/RespArea.test.jsx +122 -0
  123. package/src/components/__tests__/TableIcons.test.jsx +149 -0
  124. package/src/components/__tests__/TextAlign.test.jsx +167 -0
  125. package/src/components/__tests__/TiptapContainer.test.jsx +138 -0
  126. package/src/components/__tests__/characterUtils.test.js +166 -0
  127. package/src/components/__tests__/choice.test.jsx +171 -0
  128. package/src/components/__tests__/custom-popper.test.jsx +82 -0
  129. package/src/components/__tests__/done-button.test.jsx +54 -0
  130. package/src/components/__tests__/toolbar-buttons.test.jsx +234 -0
  131. package/src/components/characters/characterUtils.js +447 -0
  132. package/src/components/characters/custom-popper.js +38 -0
  133. package/src/components/common/done-button.jsx +27 -0
  134. package/src/components/common/toolbar-buttons.jsx +122 -0
  135. package/src/components/icons/CssIcon.jsx +15 -0
  136. package/src/components/icons/RespArea.jsx +71 -0
  137. package/src/components/icons/TableIcons.jsx +52 -0
  138. package/src/components/icons/TextAlign.jsx +114 -0
  139. package/src/components/image/AltDialog.jsx +82 -0
  140. package/src/components/image/ImageToolbar.jsx +99 -0
  141. package/src/components/image/InsertImageHandler.js +107 -0
  142. package/src/components/media/MediaDialog.jsx +596 -0
  143. package/src/components/media/MediaToolbar.jsx +49 -0
  144. package/src/components/media/MediaWrapper.jsx +39 -0
  145. package/src/components/respArea/DragInTheBlank/DragInTheBlank.jsx +76 -0
  146. package/src/components/respArea/DragInTheBlank/choice.jsx +256 -0
  147. package/src/components/respArea/ExplicitConstructedResponse.jsx +136 -0
  148. package/src/components/respArea/InlineDropdown.jsx +221 -0
  149. package/src/components/respArea/MathTemplated.jsx +124 -0
  150. package/src/components/respArea/ToolbarIcon.jsx +66 -0
  151. package/src/components/respArea/__tests__/MathTemplated.test.jsx +210 -0
  152. package/src/components/respArea/inlineDropdownUtils.js +79 -0
  153. package/src/constants.js +5 -0
  154. package/src/extensions/__tests__/css.test.js +196 -0
  155. package/src/extensions/__tests__/custom-toolbar-wrapper.test.jsx +180 -0
  156. package/src/extensions/__tests__/divNode.test.js +87 -0
  157. package/src/extensions/__tests__/ensure-empty-root-div.test.js +57 -0
  158. package/src/extensions/__tests__/ensure-list-item-content-is-div.test.js +44 -0
  159. package/src/extensions/__tests__/extended-list-item.test.js +13 -0
  160. package/src/extensions/__tests__/extended-table-cell.test.js +22 -0
  161. package/src/extensions/__tests__/extended-table.test.js +183 -0
  162. package/src/extensions/__tests__/image-component.test.jsx +345 -0
  163. package/src/extensions/__tests__/image.test.js +237 -0
  164. package/src/extensions/__tests__/math.test.js +604 -0
  165. package/src/extensions/__tests__/media-node-view.test.jsx +298 -0
  166. package/src/extensions/__tests__/media.test.js +271 -0
  167. package/src/extensions/__tests__/responseArea.test.js +601 -0
  168. package/src/extensions/css.js +220 -0
  169. package/src/extensions/custom-toolbar-wrapper.jsx +78 -0
  170. package/src/extensions/div-node.js +86 -0
  171. package/src/extensions/ensure-empty-root-div.js +47 -0
  172. package/src/extensions/ensure-list-item-content-is-div.js +62 -0
  173. package/src/extensions/extended-list-item.js +10 -0
  174. package/src/extensions/extended-table-cell.js +19 -0
  175. package/src/extensions/extended-table.js +60 -0
  176. package/src/extensions/heading-paragraph.js +53 -0
  177. package/src/extensions/image-component.jsx +338 -0
  178. package/src/extensions/image.js +109 -0
  179. package/src/extensions/index.js +81 -0
  180. package/src/extensions/math.js +325 -0
  181. package/src/extensions/media.js +188 -0
  182. package/src/extensions/responseArea.js +401 -0
  183. package/src/index.jsx +5 -0
  184. package/src/styles/editorContainerStyles.js +145 -0
  185. package/src/theme.js +1 -0
  186. package/src/utils/__tests__/helper.test.js +126 -0
  187. package/src/utils/__tests__/toolbar.test.js +43 -0
  188. package/src/utils/helper.js +69 -0
  189. package/src/utils/size.js +32 -0
  190. package/src/utils/toolbar.js +15 -0
  191. package/dist/components/CharacterPicker.d.ts +0 -31
  192. package/dist/components/CharacterPicker.js +0 -131
  193. package/dist/components/EditableHtml.d.ts +0 -11
  194. package/dist/components/EditableHtml.js +0 -291
  195. package/dist/components/MenuBar.d.ts +0 -11
  196. package/dist/components/MenuBar.js +0 -462
  197. package/dist/components/TiptapContainer.d.ts +0 -11
  198. package/dist/components/TiptapContainer.js +0 -154
  199. package/dist/components/characters/characterUtils.d.ts +0 -35
  200. package/dist/components/characters/characterUtils.js +0 -465
  201. package/dist/components/characters/custom-popper.d.ts +0 -14
  202. package/dist/components/characters/custom-popper.js +0 -32
  203. package/dist/components/common/done-button.d.ts +0 -30
  204. package/dist/components/common/done-button.js +0 -26
  205. package/dist/components/common/toolbar-buttons.d.ts +0 -38
  206. package/dist/components/common/toolbar-buttons.js +0 -91
  207. package/dist/components/icons/CssIcon.d.ts +0 -11
  208. package/dist/components/icons/CssIcon.js +0 -14
  209. package/dist/components/icons/RespArea.d.ts +0 -26
  210. package/dist/components/icons/RespArea.js +0 -42
  211. package/dist/components/icons/TableIcons.d.ts +0 -14
  212. package/dist/components/icons/TableIcons.js +0 -32
  213. package/dist/components/icons/TextAlign.d.ts +0 -18
  214. package/dist/components/icons/TextAlign.js +0 -134
  215. package/dist/components/image/AltDialog.d.ts +0 -22
  216. package/dist/components/image/AltDialog.js +0 -61
  217. package/dist/components/image/ImageToolbar.d.ts +0 -24
  218. package/dist/components/image/ImageToolbar.js +0 -80
  219. package/dist/components/image/InsertImageHandler.d.ts +0 -32
  220. package/dist/components/image/InsertImageHandler.js +0 -53
  221. package/dist/components/media/MediaDialog.d.ts +0 -43
  222. package/dist/components/media/MediaDialog.js +0 -389
  223. package/dist/components/media/MediaToolbar.d.ts +0 -19
  224. package/dist/components/media/MediaToolbar.js +0 -41
  225. package/dist/components/media/MediaWrapper.d.ts +0 -19
  226. package/dist/components/respArea/DragInTheBlank/DragInTheBlank.d.ts +0 -23
  227. package/dist/components/respArea/DragInTheBlank/DragInTheBlank.js +0 -58
  228. package/dist/components/respArea/DragInTheBlank/choice.d.ts +0 -56
  229. package/dist/components/respArea/DragInTheBlank/choice.js +0 -156
  230. package/dist/components/respArea/ExplicitConstructedResponse.d.ts +0 -20
  231. package/dist/components/respArea/ExplicitConstructedResponse.js +0 -83
  232. package/dist/components/respArea/InlineDropdown.d.ts +0 -18
  233. package/dist/components/respArea/InlineDropdown.js +0 -119
  234. package/dist/components/respArea/MathTemplated.d.ts +0 -19
  235. package/dist/components/respArea/MathTemplated.js +0 -97
  236. package/dist/components/respArea/ToolbarIcon.d.ts +0 -14
  237. package/dist/components/respArea/ToolbarIcon.js +0 -17
  238. package/dist/components/respArea/inlineDropdownUtils.d.ts +0 -15
  239. package/dist/components/respArea/inlineDropdownUtils.js +0 -15
  240. package/dist/constants.d.ts +0 -13
  241. package/dist/constants.js +0 -4
  242. package/dist/extensions/css.d.ts +0 -11
  243. package/dist/extensions/css.js +0 -115
  244. package/dist/extensions/custom-toolbar-wrapper.d.ts +0 -11
  245. package/dist/extensions/custom-toolbar-wrapper.js +0 -61
  246. package/dist/extensions/div-node.d.ts +0 -10
  247. package/dist/extensions/div-node.js +0 -42
  248. package/dist/extensions/ensure-empty-root-div.d.ts +0 -14
  249. package/dist/extensions/ensure-empty-root-div.js +0 -24
  250. package/dist/extensions/ensure-list-item-content-is-div.d.ts +0 -15
  251. package/dist/extensions/ensure-list-item-content-is-div.js +0 -31
  252. package/dist/extensions/extended-list-item.d.ts +0 -13
  253. package/dist/extensions/extended-list-item.js +0 -5
  254. package/dist/extensions/extended-table-cell.d.ts +0 -10
  255. package/dist/extensions/extended-table-cell.js +0 -6
  256. package/dist/extensions/extended-table.d.ts +0 -17
  257. package/dist/extensions/extended-table.js +0 -34
  258. package/dist/extensions/heading-paragraph.d.ts +0 -17
  259. package/dist/extensions/heading-paragraph.js +0 -30
  260. package/dist/extensions/image-component.d.ts +0 -22
  261. package/dist/extensions/image-component.js +0 -220
  262. package/dist/extensions/image.d.ts +0 -10
  263. package/dist/extensions/image.js +0 -68
  264. package/dist/extensions/index.d.ts +0 -16
  265. package/dist/extensions/index.js +0 -64
  266. package/dist/extensions/math.d.ts +0 -15
  267. package/dist/extensions/math.js +0 -158
  268. package/dist/extensions/media.d.ts +0 -19
  269. package/dist/extensions/media.js +0 -149
  270. package/dist/extensions/responseArea.d.ts +0 -27
  271. package/dist/extensions/responseArea.js +0 -259
  272. package/dist/index.d.ts +0 -13
  273. package/dist/index.js +0 -7
  274. package/dist/node_modules/.bun/clsx@2.1.1/node_modules/clsx/dist/clsx.js +0 -16
  275. package/dist/styles/editorContainerStyles.d.ts +0 -134
  276. package/dist/theme.d.ts +0 -9
  277. package/dist/utils/helper.d.ts +0 -9
  278. package/dist/utils/helper.js +0 -27
  279. package/dist/utils/size.d.ts +0 -9
  280. package/dist/utils/size.js +0 -14
@@ -0,0 +1,604 @@
1
+ import React from 'react';
2
+ import { render, waitFor, fireEvent } from '@testing-library/react';
3
+ import { EnsureTextAfterMathPlugin, MathNode, MathNodeView, ZeroWidthSpaceHandlingPlugin } from '../math';
4
+
5
+ jest.mock('@tiptap/react', () => ({
6
+ NodeViewWrapper: ({ children, ...props }) => (
7
+ <div data-testid="node-view-wrapper" {...props}>
8
+ {children}
9
+ </div>
10
+ ),
11
+ ReactNodeViewRenderer: jest.fn((component) => component),
12
+ }));
13
+
14
+ const mockCreatePortal = jest.fn((node) => node);
15
+ jest.mock('react-dom', () => ({
16
+ ...jest.requireActual('react-dom'),
17
+ createPortal: (...args) => mockCreatePortal(...args),
18
+ }));
19
+
20
+ jest.mock('@pie-lib/math-toolbar', () => {
21
+ const React = require('react');
22
+ return {
23
+ MathPreview: ({ latex }) => <div data-testid="math-preview">{latex}</div>,
24
+ MathToolbar: ({ latex, onChange, onDone }) => {
25
+ const [localLatex, setLocalLatex] = React.useState(latex);
26
+ return (
27
+ <div data-testid="math-toolbar">
28
+ <input
29
+ data-testid="math-input"
30
+ value={localLatex}
31
+ onChange={(e) => {
32
+ setLocalLatex(e.target.value);
33
+ onChange(e.target.value);
34
+ }}
35
+ />
36
+ <button data-testid="done-button" onClick={() => onDone(localLatex)}>
37
+ Done
38
+ </button>
39
+ </div>
40
+ );
41
+ },
42
+ };
43
+ });
44
+
45
+ jest.mock('@pie-lib/math-rendering', () => ({
46
+ wrapMath: (latex, wrapper) => latex,
47
+ }));
48
+
49
+ jest.mock('@tiptap/core', () => ({
50
+ Node: {
51
+ create: jest.fn((config) => config),
52
+ },
53
+ }));
54
+
55
+ jest.mock('prosemirror-state', () => ({
56
+ Plugin: jest.fn(function (config) {
57
+ return config;
58
+ }),
59
+ PluginKey: jest.fn(function (key) {
60
+ this.key = key;
61
+ }),
62
+ TextSelection: {
63
+ create: jest.fn((doc, pos) => ({ type: 'text', pos })),
64
+ },
65
+ NodeSelection: {
66
+ create: jest.fn((doc, pos) => ({ type: 'node', pos })),
67
+ },
68
+ }));
69
+
70
+ describe('MathNode', () => {
71
+ describe('configuration', () => {
72
+ it('has correct name', () => {
73
+ expect(MathNode.name).toBe('math');
74
+ });
75
+
76
+ it('is inline', () => {
77
+ expect(MathNode.inline).toBe(true);
78
+ });
79
+
80
+ it('is in inline group', () => {
81
+ expect(MathNode.group).toBe('inline');
82
+ });
83
+
84
+ it('is atomic', () => {
85
+ expect(MathNode.atom).toBe(true);
86
+ });
87
+ });
88
+
89
+ describe('addAttributes', () => {
90
+ it('returns required attributes', () => {
91
+ const attributes = MathNode.addAttributes();
92
+
93
+ expect(attributes).toHaveProperty('latex');
94
+ expect(attributes).toHaveProperty('wrapper');
95
+ expect(attributes).toHaveProperty('html');
96
+
97
+ expect(attributes.latex).toEqual({ default: '' });
98
+ expect(attributes.wrapper).toEqual({ default: null });
99
+ expect(attributes.html).toEqual({ default: null });
100
+ });
101
+ });
102
+
103
+ describe('parseHTML', () => {
104
+ it('returns parsing rules for latex', () => {
105
+ const rules = MathNode.parseHTML();
106
+
107
+ expect(Array.isArray(rules)).toBe(true);
108
+ expect(rules).toHaveLength(2);
109
+ expect(rules[0]).toHaveProperty('tag', 'span[data-latex]');
110
+ });
111
+
112
+ it('returns parsing rules for mathml', () => {
113
+ const rules = MathNode.parseHTML();
114
+ expect(rules[1]).toHaveProperty('tag', 'span[data-type="mathml"]');
115
+ });
116
+ });
117
+
118
+ describe('renderHTML', () => {
119
+ it('renders mathml when html attribute is present', () => {
120
+ const result = MathNode.renderHTML({
121
+ HTMLAttributes: {
122
+ html: '<math><mi>x</mi></math>',
123
+ },
124
+ });
125
+
126
+ expect(result[0]).toBe('span');
127
+ expect(result[1]).toHaveProperty('data-type', 'mathml');
128
+ });
129
+
130
+ it('renders latex when html attribute is not present', () => {
131
+ const result = MathNode.renderHTML({
132
+ HTMLAttributes: {
133
+ latex: 'x^2',
134
+ },
135
+ });
136
+
137
+ expect(result[0]).toBe('span');
138
+ expect(result[1]).toHaveProperty('data-latex', '');
139
+ expect(result[1]).toHaveProperty('data-raw', 'x^2');
140
+ });
141
+ });
142
+
143
+ describe('addCommands', () => {
144
+ it('returns insertMath command', () => {
145
+ const commands = MathNode.addCommands();
146
+
147
+ expect(commands).toHaveProperty('insertMath');
148
+ expect(typeof commands.insertMath).toBe('function');
149
+ });
150
+ });
151
+
152
+ describe('addNodeView', () => {
153
+ it('returns ReactNodeViewRenderer result', () => {
154
+ const result = MathNode.addNodeView();
155
+
156
+ expect(result).toBeDefined();
157
+ });
158
+ });
159
+
160
+ describe('addProseMirrorPlugins', () => {
161
+ it('registers ensure-text-after-math and zero-width-space plugins', () => {
162
+ const plugins = MathNode.addProseMirrorPlugins();
163
+
164
+ expect(plugins).toHaveLength(2);
165
+ expect(plugins[0].appendTransaction).toBeDefined();
166
+ expect(plugins[1].props.handleKeyDown).toBeDefined();
167
+ });
168
+ });
169
+ });
170
+
171
+ describe('EnsureTextAfterMathPlugin', () => {
172
+ it('inserts a zero-width space after a math node when no text follows', () => {
173
+ const plugin = EnsureTextAfterMathPlugin('math');
174
+ const textNode = { type: { name: 'text' } };
175
+ const mathNode = { type: { name: 'math' }, nodeSize: 3 };
176
+ const tr = { insert: jest.fn() };
177
+
178
+ const newState = {
179
+ schema: { text: jest.fn((value) => ({ type: textNode.type, text: value })) },
180
+ tr,
181
+ doc: {
182
+ descendants: (cb) => cb(mathNode, 5),
183
+ nodeAt: jest.fn(() => null),
184
+ },
185
+ };
186
+
187
+ const result = plugin.appendTransaction([{ docChanged: true }], {}, newState);
188
+
189
+ expect(tr.insert).toHaveBeenCalledWith(8, expect.anything());
190
+ expect(result).toBe(tr);
191
+ });
192
+
193
+ it('does not insert when text already follows the math node', () => {
194
+ const plugin = EnsureTextAfterMathPlugin('math');
195
+ const tr = { insert: jest.fn() };
196
+ const mathNode = { type: { name: 'math' }, nodeSize: 3 };
197
+
198
+ const newState = {
199
+ schema: { text: jest.fn() },
200
+ tr,
201
+ doc: {
202
+ descendants: (cb) => cb(mathNode, 5),
203
+ nodeAt: jest.fn(() => ({ type: { name: 'text' } })),
204
+ },
205
+ };
206
+
207
+ const result = plugin.appendTransaction([{ docChanged: true }], {}, newState);
208
+
209
+ expect(tr.insert).not.toHaveBeenCalled();
210
+ expect(result).toBeNull();
211
+ });
212
+
213
+ it('returns null when the document did not change', () => {
214
+ const plugin = EnsureTextAfterMathPlugin('math');
215
+
216
+ const result = plugin.appendTransaction([{ docChanged: false }], {}, {});
217
+ expect(result).toBeNull();
218
+ });
219
+ });
220
+
221
+ describe('ZeroWidthSpaceHandlingPlugin', () => {
222
+ const createDefaultDoc = () => ({
223
+ textBetween: jest.fn(() => '\u200b'),
224
+ resolve: jest.fn(() => ({
225
+ nodeAfter: null,
226
+ nodeBefore: null,
227
+ })),
228
+ });
229
+
230
+ const createView = ({ state: stateOverrides = {} } = {}) => {
231
+ const dispatch = jest.fn();
232
+ const tr = {
233
+ delete: jest.fn().mockReturnThis(),
234
+ setSelection: jest.fn().mockReturnThis(),
235
+ };
236
+
237
+ return {
238
+ state: {
239
+ selection: { from: 2, empty: true },
240
+ doc: createDefaultDoc(),
241
+ tr,
242
+ ...stateOverrides,
243
+ doc: { ...createDefaultDoc(), ...stateOverrides.doc },
244
+ },
245
+ dispatch,
246
+ };
247
+ };
248
+
249
+ it('deletes math and zero-width space on Backspace', () => {
250
+ const view = createView();
251
+ const event = { key: 'Backspace' };
252
+ const handled = ZeroWidthSpaceHandlingPlugin.props.handleKeyDown(view, event);
253
+
254
+ expect(handled).toBe(true);
255
+ expect(view.state.tr.delete).toHaveBeenCalledWith(0, 2);
256
+ expect(view.dispatch).toHaveBeenCalledWith(view.state.tr);
257
+ });
258
+
259
+ it('selects the math node on ArrowLeft before a zero-width space', () => {
260
+ const mathNode = { nodeSize: 3 };
261
+ const view = createView({
262
+ state: {
263
+ doc: {
264
+ resolve: jest
265
+ .fn()
266
+ .mockReturnValueOnce({ nodeAfter: mathNode, nodeBefore: null })
267
+ .mockReturnValueOnce({ pos: 4 }),
268
+ },
269
+ },
270
+ });
271
+ const { NodeSelection } = require('prosemirror-state');
272
+
273
+ const handled = ZeroWidthSpaceHandlingPlugin.props.handleKeyDown(view, { key: 'ArrowLeft' });
274
+
275
+ expect(handled).toBe(true);
276
+ expect(NodeSelection.create).toHaveBeenCalledWith(view.state.doc, 4);
277
+ expect(view.dispatch).toHaveBeenCalled();
278
+ });
279
+
280
+ it('moves the text cursor before the zero-width space when no inline node precedes it', () => {
281
+ const view = createView();
282
+ const { TextSelection } = require('prosemirror-state');
283
+
284
+ const handled = ZeroWidthSpaceHandlingPlugin.props.handleKeyDown(view, { key: 'ArrowLeft' });
285
+
286
+ expect(handled).toBe(true);
287
+ expect(TextSelection.create).toHaveBeenCalledWith(view.state.doc, 0);
288
+ expect(view.dispatch).toHaveBeenCalled();
289
+ });
290
+
291
+ it('returns false for unrelated keys', () => {
292
+ const view = createView({
293
+ state: {
294
+ doc: {
295
+ textBetween: jest.fn(() => 'a'),
296
+ resolve: jest.fn(),
297
+ },
298
+ },
299
+ });
300
+
301
+ const handled = ZeroWidthSpaceHandlingPlugin.props.handleKeyDown(view, { key: 'Enter' });
302
+ expect(handled).toBe(false);
303
+ });
304
+ });
305
+
306
+ describe('MathNodeView', () => {
307
+ const createMockEditor = () => ({
308
+ state: {
309
+ selection: {
310
+ from: 0,
311
+ to: 1,
312
+ },
313
+ tr: {
314
+ setSelection: jest.fn().mockReturnThis(),
315
+ setMeta: jest.fn().mockReturnThis(),
316
+ },
317
+ doc: {},
318
+ },
319
+ view: {
320
+ coordsAtPos: jest.fn(() => ({ top: 100, left: 50 })),
321
+ dispatch: jest.fn(),
322
+ },
323
+ commands: {
324
+ focus: jest.fn(),
325
+ },
326
+ instanceId: 'editor-123',
327
+ _toolbarOpened: false,
328
+ });
329
+
330
+ const mockNode = {
331
+ attrs: {
332
+ latex: 'x^2',
333
+ },
334
+ };
335
+
336
+ let defaultProps;
337
+
338
+ beforeAll(() => {
339
+ Object.defineProperty(document.body, 'getBoundingClientRect', {
340
+ value: jest.fn(() => ({ top: 0, left: 0 })),
341
+ configurable: true,
342
+ });
343
+ });
344
+
345
+ beforeEach(() => {
346
+ jest.clearAllMocks();
347
+ mockCreatePortal.mockImplementation((node) => node);
348
+ defaultProps = {
349
+ node: mockNode,
350
+ updateAttributes: jest.fn(),
351
+ editor: createMockEditor(),
352
+ selected: false,
353
+ options: {},
354
+ };
355
+ });
356
+
357
+ it('renders without crashing', () => {
358
+ const { container } = render(<MathNodeView {...defaultProps} />);
359
+ expect(container).toBeInTheDocument();
360
+ });
361
+
362
+ it('renders NodeViewWrapper', () => {
363
+ const { getByTestId } = render(<MathNodeView {...defaultProps} />);
364
+ expect(getByTestId('node-view-wrapper')).toBeInTheDocument();
365
+ });
366
+
367
+ it('displays math preview', () => {
368
+ const { getByTestId } = render(<MathNodeView {...defaultProps} />);
369
+ expect(getByTestId('math-preview')).toBeInTheDocument();
370
+ });
371
+
372
+ it('shows toolbar when selected', async () => {
373
+ const { getByTestId } = render(<MathNodeView {...defaultProps} selected={true} />);
374
+ await waitFor(() => {
375
+ expect(getByTestId('math-toolbar')).toBeInTheDocument();
376
+ });
377
+ });
378
+
379
+ it('does not show toolbar when not selected', () => {
380
+ const { queryByTestId } = render(<MathNodeView {...defaultProps} selected={false} />);
381
+ expect(queryByTestId('math-toolbar')).not.toBeInTheDocument();
382
+ });
383
+
384
+ it('adds data-toolbar-for attribute with editor instanceId', async () => {
385
+ const { container } = render(<MathNodeView {...defaultProps} selected={true} />);
386
+ await waitFor(() => {
387
+ const toolbar = container.querySelector('[data-toolbar-for]');
388
+ expect(toolbar).toHaveAttribute('data-toolbar-for', 'editor-123');
389
+ });
390
+ });
391
+
392
+ describe('toolbar positioning', () => {
393
+ it('uses a fixed top offset and horizontal position from coordsAtPos', async () => {
394
+ const { container } = render(<MathNodeView {...defaultProps} selected={true} />);
395
+ await waitFor(() => {
396
+ const toolbar = container.querySelector('[data-toolbar-for]');
397
+ expect(toolbar).toBeInTheDocument();
398
+ expect(toolbar.style.top).toBe('40px');
399
+ expect(toolbar.style.left).toBe('50px');
400
+ });
401
+ });
402
+
403
+ it('keeps the fixed top offset when the editor container is scrolled', async () => {
404
+ const containerEl = document.createElement('div');
405
+ containerEl.getBoundingClientRect = jest.fn(() => ({ top: -200, left: 0, width: 600, height: 400 }));
406
+
407
+ const editor = {
408
+ ...defaultProps.editor,
409
+ _tiptapContainerEl: containerEl,
410
+ };
411
+
412
+ const { container } = render(<MathNodeView {...defaultProps} editor={editor} selected={true} />);
413
+ await waitFor(() => {
414
+ const toolbar = container.querySelector('[data-toolbar-for]');
415
+ expect(toolbar).toBeInTheDocument();
416
+ expect(toolbar.style.top).toBe('40px');
417
+ expect(toolbar.style.left).toBe('50px');
418
+ });
419
+ });
420
+
421
+ it('applies absolute positioning style to toolbar', async () => {
422
+ const { container } = render(<MathNodeView {...defaultProps} selected={true} />);
423
+ await waitFor(() => {
424
+ const toolbar = container.querySelector('[data-toolbar-for]');
425
+ expect(toolbar).toBeInTheDocument();
426
+ expect(toolbar.style.position).toBe('absolute');
427
+ });
428
+ });
429
+
430
+ it('updates horizontal position from coordsAtPos when selection changes', async () => {
431
+ const editor = {
432
+ ...defaultProps.editor,
433
+ view: {
434
+ ...defaultProps.editor.view,
435
+ coordsAtPos: jest.fn(() => ({ top: 200, left: 150 })),
436
+ dispatch: jest.fn(),
437
+ },
438
+ };
439
+
440
+ const { container } = render(<MathNodeView {...defaultProps} editor={editor} selected={true} />);
441
+ await waitFor(() => {
442
+ const toolbar = container.querySelector('[data-toolbar-for]');
443
+ expect(toolbar).toBeInTheDocument();
444
+ expect(toolbar.style.top).toBe('40px');
445
+ expect(toolbar.style.left).toBe('150px');
446
+ });
447
+ });
448
+
449
+ it('portals toolbar into _tiptapContainerEl when available', async () => {
450
+ const containerEl = document.createElement('div');
451
+ containerEl.getBoundingClientRect = jest.fn(() => ({ top: 0, left: 0, width: 600, height: 400 }));
452
+
453
+ const editor = {
454
+ ...defaultProps.editor,
455
+ _tiptapContainerEl: containerEl,
456
+ };
457
+
458
+ mockCreatePortal.mockClear();
459
+ render(<MathNodeView {...defaultProps} editor={editor} selected={true} />);
460
+ await waitFor(() => {
461
+ expect(mockCreatePortal).toHaveBeenCalled();
462
+ const lastCall = mockCreatePortal.mock.calls[mockCreatePortal.mock.calls.length - 1];
463
+ expect(lastCall[1]).toBe(containerEl);
464
+ });
465
+ });
466
+
467
+ it('portals toolbar into document.body when _tiptapContainerEl is not set', async () => {
468
+ const editor = {
469
+ ...defaultProps.editor,
470
+ _tiptapContainerEl: undefined,
471
+ };
472
+
473
+ mockCreatePortal.mockClear();
474
+ render(<MathNodeView {...defaultProps} editor={editor} selected={true} />);
475
+ await waitFor(() => {
476
+ expect(mockCreatePortal).toHaveBeenCalled();
477
+ const lastCall = mockCreatePortal.mock.calls[mockCreatePortal.mock.calls.length - 1];
478
+ expect(lastCall[1]).toBe(document.body);
479
+ });
480
+ });
481
+ });
482
+
483
+ it('calls updateAttributes when latex changes', async () => {
484
+ const { getByTestId } = render(<MathNodeView {...defaultProps} selected={true} />);
485
+ await waitFor(() => {
486
+ const input = getByTestId('math-input');
487
+ fireEvent.change(input, { target: { value: 'y^2' } });
488
+ });
489
+ expect(defaultProps.updateAttributes).toHaveBeenCalledWith({ latex: 'y^2' });
490
+ });
491
+
492
+ it('closes toolbar and updates attributes when done', async () => {
493
+ const updateAttributes = jest.fn();
494
+ const { getByTestId } = render(
495
+ <MathNodeView {...defaultProps} updateAttributes={updateAttributes} selected={true} />,
496
+ );
497
+
498
+ await waitFor(() => {
499
+ expect(getByTestId('done-button')).toBeInTheDocument();
500
+ });
501
+
502
+ const doneButton = getByTestId('done-button');
503
+ fireEvent.click(doneButton);
504
+
505
+ await waitFor(() => {
506
+ expect(updateAttributes).toHaveBeenCalledWith({ latex: 'x^2' });
507
+ });
508
+ });
509
+
510
+ it('sets editor._toolbarOpened when toolbar is shown', async () => {
511
+ const { getByTestId } = render(<MathNodeView {...defaultProps} selected={true} />);
512
+ await waitFor(() => {
513
+ expect(getByTestId('math-toolbar')).toBeInTheDocument();
514
+ expect(defaultProps.editor._toolbarOpened).toBe(true);
515
+ });
516
+ });
517
+
518
+ it('unsets editor._toolbarOpened when toolbar is closed', async () => {
519
+ const { getByTestId } = render(<MathNodeView {...defaultProps} selected={true} />);
520
+
521
+ await waitFor(() => {
522
+ expect(getByTestId('done-button')).toBeInTheDocument();
523
+ });
524
+
525
+ const doneButton = getByTestId('done-button');
526
+ fireEvent.click(doneButton);
527
+
528
+ await waitFor(() => {
529
+ expect(defaultProps.editor._toolbarOpened).toBe(false);
530
+ });
531
+ });
532
+
533
+ it('closes toolbar on outside click and runs handleDone', async () => {
534
+ const updateAttributes = jest.fn();
535
+ const editor = createMockEditor();
536
+ const { TextSelection } = require('prosemirror-state');
537
+ const { queryByTestId } = render(
538
+ <MathNodeView {...defaultProps} editor={editor} updateAttributes={updateAttributes} selected={true} />,
539
+ );
540
+
541
+ await waitFor(() => {
542
+ expect(queryByTestId('math-toolbar')).toBeInTheDocument();
543
+ });
544
+
545
+ fireEvent.click(document.body);
546
+
547
+ await waitFor(() => {
548
+ expect(queryByTestId('math-toolbar')).not.toBeInTheDocument();
549
+ expect(updateAttributes).toHaveBeenCalledWith({ latex: 'x^2' });
550
+ expect(TextSelection.create).toHaveBeenCalledWith(editor.state.doc, 1);
551
+ expect(editor.state.tr.setSelection).toHaveBeenCalled();
552
+ expect(editor.view.dispatch).toHaveBeenCalledWith(editor.state.tr);
553
+ expect(editor.commands.focus).toHaveBeenCalled();
554
+ expect(editor._toolbarOpened).toBe(false);
555
+ });
556
+ });
557
+
558
+ it('does not close toolbar when clicking the math node preview', async () => {
559
+ const { getByTestId, queryByTestId } = render(<MathNodeView {...defaultProps} selected={true} />);
560
+
561
+ await waitFor(() => {
562
+ expect(queryByTestId('math-toolbar')).toBeInTheDocument();
563
+ });
564
+
565
+ fireEvent.click(getByTestId('math-preview'));
566
+
567
+ await waitFor(() => {
568
+ expect(queryByTestId('math-toolbar')).toBeInTheDocument();
569
+ });
570
+ });
571
+
572
+ it('does not close toolbar when clicking equation editor dropdown', async () => {
573
+ const { queryByTestId } = render(<MathNodeView {...defaultProps} selected={true} />);
574
+
575
+ await waitFor(() => {
576
+ expect(queryByTestId('math-toolbar')).toBeInTheDocument();
577
+ });
578
+
579
+ // Simulate MUI Select's portal dropdown container.
580
+ const dropdown = document.createElement('div');
581
+ dropdown.id = 'equation-editor-select-listbox';
582
+ document.body.appendChild(dropdown);
583
+
584
+ fireEvent.click(dropdown);
585
+
586
+ await waitFor(() => {
587
+ expect(queryByTestId('math-toolbar')).toBeInTheDocument();
588
+ });
589
+
590
+ document.body.removeChild(dropdown);
591
+ });
592
+
593
+ it('renders with empty latex', () => {
594
+ const nodeWithEmptyLatex = { attrs: { latex: '' } };
595
+ const { getByTestId } = render(<MathNodeView {...defaultProps} node={nodeWithEmptyLatex} />);
596
+ expect(getByTestId('math-preview')).toBeInTheDocument();
597
+ });
598
+
599
+ it('has correct styling on NodeViewWrapper', () => {
600
+ const { getByTestId } = render(<MathNodeView {...defaultProps} />);
601
+ const wrapper = getByTestId('node-view-wrapper');
602
+ expect(wrapper).toHaveStyle({ display: 'inline-flex', cursor: 'pointer' });
603
+ });
604
+ });