@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,76 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import { NodeViewWrapper } from '@tiptap/react';
4
+ import DragDropTile from './choice';
5
+ import omit from 'lodash-es/omit';
6
+
7
+ export const onValueChange = (editor, node, pos, choice) => {
8
+ const { tr } = editor.state;
9
+
10
+ // Merge old and new attributes
11
+ tr.setNodeMarkup(pos, undefined, {
12
+ ...node.attrs,
13
+ ...choice.value,
14
+ });
15
+ tr.isDone = true;
16
+ editor.view.dispatch(tr);
17
+ };
18
+
19
+ export const onRemoveResponse = (editor, node, choice) => {
20
+ const { tr } = editor.state;
21
+
22
+ // Merge old and new attributes
23
+ tr.setNodeMarkup(choice.pos, undefined, omit(node.attrs, ['value', 'id']));
24
+ tr.isDone = true;
25
+ editor.view.dispatch(tr);
26
+ };
27
+
28
+ const DragDrop = (props) => {
29
+ const { editor, node, getPos, options, selected } = props;
30
+ const { attrs: attributes } = node;
31
+ const { inTable } = attributes;
32
+ const pos = getPos();
33
+
34
+ // console.log({nodeProps.children})
35
+ return (
36
+ <NodeViewWrapper
37
+ className="drag-in-the-blank"
38
+ data-selected={selected}
39
+ style={{ display: 'inline', whiteSpace: 'normal' }}
40
+ >
41
+ <span
42
+ {...attributes}
43
+ style={{
44
+ display: 'inline-flex',
45
+ minHeight: '50px',
46
+ minWidth: '178px',
47
+ position: 'relative',
48
+ margin: inTable ? '10px' : '0 10px',
49
+ cursor: 'pointer',
50
+ }}
51
+ >
52
+ <DragDropTile
53
+ n={attributes}
54
+ dragKey={attributes.id}
55
+ targetId="0"
56
+ pos={pos}
57
+ value={attributes}
58
+ duplicates={options.duplicates}
59
+ selected={selected}
60
+ onChange={(choice) => onValueChange(editor, node, pos, choice)}
61
+ removeResponse={(choice) => onRemoveResponse(editor, node, choice)}
62
+ ></DragDropTile>
63
+ </span>
64
+ </NodeViewWrapper>
65
+ );
66
+ };
67
+
68
+ DragDrop.propTypes = {
69
+ attributes: PropTypes.object,
70
+ data: PropTypes.object,
71
+ n: PropTypes.object,
72
+ nodeProps: PropTypes.object,
73
+ opts: PropTypes.object,
74
+ };
75
+
76
+ export default DragDrop;
@@ -0,0 +1,256 @@
1
+ import React, { useEffect, useRef, useState } from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import { useDraggable, useDroppable } from '@dnd-kit/core';
4
+ import { color } from '@pie-lib/render-ui';
5
+ import { renderMath } from '@pie-lib/math-rendering';
6
+ import { styled } from '@mui/material/styles';
7
+ import classnames from 'classnames';
8
+
9
+ import { GripIcon } from '../../icons/RespArea';
10
+
11
+ const StyledContent = styled('span')(({ theme }) => ({
12
+ border: `solid 0px ${theme.palette.primary.main}`,
13
+ '& mjx-frac': {
14
+ fontSize: '120% !important',
15
+ },
16
+ }));
17
+
18
+ export function BlankContent({ n, children, isDragging, isOver, dragItem, value, selected }) {
19
+ const [hoveredElementSize, setHoveredElementSize] = useState(null);
20
+ const elementRef = useRef(null);
21
+
22
+ const handleClick = (event) => {
23
+ if (!elementRef.current) return;
24
+
25
+ if (elementRef.current.contains(event.target)) {
26
+ elementRef.current.classList.add('selected');
27
+ } else {
28
+ elementRef.current.classList.remove('selected');
29
+ }
30
+ };
31
+
32
+ useEffect(() => {
33
+ document.addEventListener('click', handleClick);
34
+ return () => {
35
+ document.removeEventListener('click', handleClick);
36
+ };
37
+ }, []);
38
+
39
+ useEffect(() => {
40
+ if (elementRef.current && typeof renderMath === 'function') {
41
+ renderMath(elementRef.current);
42
+ }
43
+ }, [value?.value, isOver, dragItem?.value?.value]);
44
+
45
+ useEffect(() => {
46
+ if (isOver && elementRef.current && !hoveredElementSize) {
47
+ const node = elementRef.current;
48
+ setHoveredElementSize({ width: node.offsetWidth, height: node.offsetHeight });
49
+ } else if (!isOver && hoveredElementSize) {
50
+ setHoveredElementSize(null);
51
+ }
52
+ }, [isOver, hoveredElementSize]);
53
+
54
+ const label = dragItem && isOver ? dragItem.value.value : value.value || '\u00A0';
55
+ const finalLabel = isDragging ? '\u00A0' : label;
56
+ const hasGrip = finalLabel !== '\u00A0';
57
+ const isPreview = dragItem && isOver;
58
+
59
+ const borderStyle = selected
60
+ ? `2px solid ${color.primaryDark()}`
61
+ : isPreview
62
+ ? `1px solid ${color.defaults.BORDER_DARK}`
63
+ : `1px solid ${color.defaults.BORDER_LIGHT}`;
64
+
65
+ return (
66
+ <div
67
+ ref={elementRef}
68
+ className={selected ? 'selected' : undefined}
69
+ style={{
70
+ display: 'inline-flex',
71
+ minWidth: '178px',
72
+ minHeight: '36px',
73
+ background: isPreview ? `${color.defaults.BORDER_LIGHT}` : `${color.defaults.WHITE}`,
74
+ border: borderStyle,
75
+ boxSizing: 'border-box',
76
+ borderRadius: '3px',
77
+ overflow: 'hidden',
78
+ position: 'relative',
79
+ padding: '8px 8px 8px 35px',
80
+ width: hoveredElementSize ? hoveredElementSize.width : undefined,
81
+ height: hoveredElementSize ? hoveredElementSize.height : undefined,
82
+ touchAction: 'none',
83
+ }}
84
+ data-key={n.index}
85
+ contentEditable={false}
86
+ >
87
+ {hasGrip && (
88
+ <GripIcon
89
+ style={{
90
+ position: 'absolute',
91
+ top: '6px',
92
+ left: '15px',
93
+ color: '#9B9B9B',
94
+ }}
95
+ contentEditable={false}
96
+ />
97
+ )}
98
+ <span
99
+ dangerouslySetInnerHTML={{
100
+ __html: finalLabel,
101
+ }}
102
+ />
103
+ {children}
104
+ </div>
105
+ );
106
+ }
107
+
108
+ BlankContent.propTypes = {
109
+ n: PropTypes.object,
110
+ children: PropTypes.node,
111
+ isDragging: PropTypes.bool,
112
+ isOver: PropTypes.bool,
113
+ dragItem: PropTypes.object,
114
+ value: PropTypes.object,
115
+ selected: PropTypes.bool,
116
+ };
117
+
118
+ function DragDropChoice({
119
+ value,
120
+ disabled,
121
+ instanceId,
122
+ children,
123
+ n,
124
+ onChange,
125
+ removeResponse,
126
+ duplicates,
127
+ pos,
128
+ selected,
129
+ }) {
130
+ const {
131
+ attributes: dragAttributes,
132
+ listeners: dragListeners,
133
+ setNodeRef: setDragNodeRef,
134
+ isDragging,
135
+ } = useDraggable({
136
+ id: `drag-${n.index}`,
137
+ disabled: disabled || !value?.value,
138
+ data: {
139
+ id: `drag-${n.index}`,
140
+ value,
141
+ instanceId,
142
+ n,
143
+ pos,
144
+ opts: { duplicates },
145
+ type: 'drag-in-the-blank-placed-choice',
146
+ fromChoice: !value,
147
+ onRemove: (draggedData) => removeResponse(draggedData),
148
+ onDrop: (draggedData, dropData) => {
149
+ // check if we're dropping into a blank
150
+ const isValidBlank = dropData?.type === 'drag-in-the-blank-drop-choice';
151
+
152
+ if (!isValidBlank) return;
153
+
154
+ // place into blank
155
+ onChange(draggedData);
156
+
157
+ if (!duplicates && draggedData.fromChoice) {
158
+ removeResponse(draggedData);
159
+ }
160
+ },
161
+ },
162
+ });
163
+
164
+ const {
165
+ setNodeRef: setDropNodeRef,
166
+ isOver,
167
+ active: dragItem,
168
+ } = useDroppable({
169
+ id: `drop-${n.index}`,
170
+ data: {
171
+ type: 'drag-in-the-blank-drop-choice',
172
+ accepts: ['drag-in-the-blank-choice', 'drag-in-the-blank-placed-choice'],
173
+ instanceId: instanceId,
174
+ value: value,
175
+ id: `drop-${n.index}`,
176
+ pos,
177
+ n,
178
+ opts: { duplicates },
179
+ onDrop: (draggedData, dropData) => {
180
+ // check if we're dropping into a blank
181
+ const isValidBlank = dropData?.type === 'drag-in-the-blank-drop-choice';
182
+
183
+ if (!isValidBlank) return;
184
+
185
+ // if the dragged and dropped data are the same, do nothing
186
+ if (draggedData.value.id === dropData.value.id) return;
187
+
188
+ if (draggedData.type === 'drag-in-the-blank-choice') {
189
+ // place into blank
190
+ onChange(draggedData);
191
+
192
+ if (!duplicates && draggedData.fromChoice) {
193
+ removeResponse(draggedData);
194
+ }
195
+ return;
196
+ }
197
+
198
+ // moving placed choice between blanks
199
+ if (draggedData.type === 'drag-in-the-blank-placed-choice') {
200
+ // clear target blank
201
+ removeResponse(dropData);
202
+
203
+ // set new blank value
204
+ onChange(draggedData);
205
+
206
+ // clear original blank - slight delay to ensure state updates correctly
207
+ setTimeout(() => removeResponse(draggedData), 10);
208
+ }
209
+ },
210
+ },
211
+ });
212
+
213
+ const setNodeRef = (node) => {
214
+ setDragNodeRef(node);
215
+ setDropNodeRef(node);
216
+ };
217
+
218
+ const dragContent = (
219
+ <BlankContent
220
+ n={n}
221
+ isDragging={isDragging}
222
+ isOver={isOver}
223
+ dragItem={dragItem?.data?.current}
224
+ value={value}
225
+ selected={selected}
226
+ >
227
+ {children}
228
+ </BlankContent>
229
+ );
230
+
231
+ const dragEl = !value ? (
232
+ <span ref={setDropNodeRef}>{dragContent}</span>
233
+ ) : (
234
+ <span ref={setNodeRef} {...dragAttributes} {...dragListeners}>
235
+ {dragContent}
236
+ </span>
237
+ );
238
+
239
+ const content = <StyledContent className={classnames(isOver && 'over')}>{dragEl}</StyledContent>;
240
+
241
+ return content;
242
+ }
243
+
244
+ DragDropChoice.propTypes = {
245
+ value: PropTypes.object,
246
+ disabled: PropTypes.bool,
247
+ instanceId: PropTypes.string,
248
+ children: PropTypes.node,
249
+ n: PropTypes.object.isRequired,
250
+ onChange: PropTypes.func.isRequired,
251
+ removeResponse: PropTypes.func.isRequired,
252
+ duplicates: PropTypes.bool,
253
+ selected: PropTypes.bool,
254
+ };
255
+
256
+ export default DragDropChoice;
@@ -0,0 +1,136 @@
1
+ import React, { useEffect, useRef, useState } from 'react';
2
+ import { NodeViewWrapper } from '@tiptap/react';
3
+ import ReactDOM from 'react-dom';
4
+ import PropTypes from 'prop-types';
5
+ import CustomToolbarWrapper from '../../extensions/custom-toolbar-wrapper';
6
+ import { setToolbarOpened } from '../../utils/toolbar';
7
+
8
+ const ExplicitConstructedResponse = (props) => {
9
+ const { editor, node, getPos, options, selected } = props;
10
+ const { attrs: attributes } = node;
11
+ const { value } = attributes;
12
+ const { respAreaToolbar, error: errorFn } = options;
13
+ const pos = getPos();
14
+ const [showToolbar, setShowToolbar] = useState(false);
15
+ const EcrToolbar = respAreaToolbar([node, pos], editor, () => {});
16
+ const toolbarRef = useRef(null);
17
+
18
+ let error;
19
+
20
+ if (errorFn) {
21
+ const errorValue = errorFn();
22
+ const respIndex = parseInt(attributes.index, 10);
23
+
24
+ error = !!errorValue?.[respIndex]?.[0];
25
+ }
26
+
27
+ useEffect(() => {
28
+ const { selection } = editor.state;
29
+ const onlyThisNodeSelected = selection.from + node.nodeSize === selection.to;
30
+
31
+ if (selected) {
32
+ if (onlyThisNodeSelected) {
33
+ setShowToolbar(selected);
34
+ }
35
+ } else {
36
+ setShowToolbar(selected);
37
+ }
38
+ }, [editor, node, selected]);
39
+
40
+ useEffect(() => {
41
+ const handleClickOutside = (event) => {
42
+ const insideCharacterPicker =
43
+ event.target.closest('.insert-character-dialog') || event.target.closest('[data-toolbar-for]');
44
+
45
+ if (
46
+ !insideCharacterPicker &&
47
+ toolbarRef.current &&
48
+ !toolbarRef.current.contains(event.target) &&
49
+ !event.target.closest('[data-inline-node]')
50
+ ) {
51
+ setShowToolbar(false);
52
+ }
53
+ };
54
+
55
+ if (showToolbar) {
56
+ document.addEventListener('mousedown', handleClickOutside);
57
+ } else {
58
+ document.removeEventListener('mousedown', handleClickOutside);
59
+ }
60
+
61
+ return () => document.removeEventListener('mousedown', handleClickOutside);
62
+ }, [showToolbar]);
63
+
64
+ return (
65
+ <NodeViewWrapper
66
+ className="drag-in-the-blank"
67
+ data-selected={selected}
68
+ style={{
69
+ display: 'inline-flex',
70
+ minHeight: '55px',
71
+ position: 'relative',
72
+ cursor: 'pointer',
73
+ }}
74
+ >
75
+ <div
76
+ {...attributes}
77
+ style={{
78
+ display: 'inline-flex',
79
+ width: '100%',
80
+ minHeight: '46px',
81
+ height: '46px',
82
+ backgroundColor: '#FFF',
83
+ border: `1px solid ${error ? 'red' : '#C0C3CF'}`,
84
+ boxSizing: 'border-box',
85
+ borderRadius: '4px',
86
+ overflow: 'hidden',
87
+ padding: '12px 21px',
88
+ margin: '0 4px',
89
+ minWidth: '178px',
90
+ visibility: showToolbar ? 'hidden' : 'visible',
91
+ }}
92
+ onClick={() => setShowToolbar(true)}
93
+ dangerouslySetInnerHTML={{
94
+ __html: value || '<div>&nbsp;</div>',
95
+ }}
96
+ />
97
+ {showToolbar && (
98
+ <React.Fragment>
99
+ <div ref={toolbarRef} className="absolute z-50 bg-white shadow-lg rounded p-2" style={{ zIndex: 1 }}>
100
+ <EcrToolbar />
101
+ </div>
102
+ {editor._tiptapContainerEl &&
103
+ ReactDOM.createPortal(
104
+ <CustomToolbarWrapper
105
+ deletable
106
+ toolbarOpts={{ minWidth: 'auto' }}
107
+ autoWidth
108
+ style={{ top: -40, left: 0, right: 0 }}
109
+ onDelete={() => {
110
+ const { tr } = editor.state;
111
+ tr.delete(pos, pos + node.nodeSize);
112
+ // Prevent the debounced onBlur/onDone from firing into the
113
+ // now-deleted node's stale position
114
+ setToolbarOpened(editor, false);
115
+ editor.view.dispatch(tr);
116
+ setShowToolbar(false);
117
+ editor.commands.focus();
118
+ }}
119
+ showDone={false}
120
+ />,
121
+ editor._tiptapContainerEl,
122
+ )}
123
+ </React.Fragment>
124
+ )}
125
+ </NodeViewWrapper>
126
+ );
127
+ };
128
+
129
+ ExplicitConstructedResponse.propTypes = {
130
+ attributes: PropTypes.object,
131
+ error: PropTypes.any,
132
+ value: PropTypes.string,
133
+ isFocused: PropTypes.bool,
134
+ };
135
+
136
+ export default ExplicitConstructedResponse;
@@ -0,0 +1,221 @@
1
+ import React, { useEffect, useRef, useState } from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import { NodeViewWrapper } from '@tiptap/react';
4
+ import { NodeSelection } from 'prosemirror-state';
5
+ import { Chevron } from '../icons/RespArea';
6
+ import ReactDOM from 'react-dom';
7
+ import CustomToolbarWrapper from '../../extensions/custom-toolbar-wrapper';
8
+ import { setToolbarOpened } from '../../utils/toolbar';
9
+
10
+ const InlineDropdown = (props) => {
11
+ const { editor, node, getPos, options, selected } = props;
12
+ const { attrs: attributes } = node;
13
+ const { value, error } = attributes;
14
+ const html = value || '<div>&nbsp</div>';
15
+ const pos = getPos();
16
+ const toolbarRef = useRef(null);
17
+ const toolbarEditor = useRef(null);
18
+ const pendingCloseRequest = useRef(false);
19
+
20
+ const isHeld = () =>
21
+ editor._holdInlineDropdownToolbarIndex != null &&
22
+ String(editor._holdInlineDropdownToolbarIndex) === String(node.attrs.index);
23
+
24
+ const [showToolbar, setShowToolbar] = useState(false);
25
+ const [position, setPosition] = useState({ top: 0, left: 0 });
26
+
27
+ const closeToolbar = () => {
28
+ if (isHeld()) {
29
+ return;
30
+ }
31
+
32
+ setShowToolbar(false);
33
+ };
34
+
35
+ const InlineDropdownToolbar = options.respAreaToolbar([node, pos], editor, closeToolbar);
36
+
37
+ const reselectNode = () => {
38
+ const { tr } = editor.state;
39
+ const nodeAtPos = tr.doc.nodeAt(pos);
40
+
41
+ if (!nodeAtPos) {
42
+ return;
43
+ }
44
+
45
+ const { selection } = tr;
46
+
47
+ if (selection.from === pos && selection.to === pos + nodeAtPos.nodeSize) {
48
+ return;
49
+ }
50
+
51
+ tr.setSelection(NodeSelection.create(tr.doc, pos));
52
+ editor.view.dispatch(tr);
53
+ };
54
+
55
+ const requestClose = () => {
56
+ if (pendingCloseRequest.current) {
57
+ return;
58
+ }
59
+
60
+ if (options.onToolbarCloseRequest) {
61
+ pendingCloseRequest.current = true;
62
+
63
+ options.onToolbarCloseRequest(
64
+ [node, pos],
65
+ editor,
66
+ () => {
67
+ pendingCloseRequest.current = false;
68
+ delete editor._holdInlineDropdownToolbarIndex;
69
+ closeToolbar();
70
+ },
71
+ () => {
72
+ pendingCloseRequest.current = false;
73
+ delete editor._holdInlineDropdownToolbarIndex;
74
+ setShowToolbar(true);
75
+ setTimeout(reselectNode, 0);
76
+ },
77
+ );
78
+ } else {
79
+ closeToolbar();
80
+ }
81
+ };
82
+
83
+ useEffect(() => {
84
+ const { selection } = editor.state;
85
+ const onlyThisNodeSelected = selection.from + node.nodeSize === selection.to;
86
+
87
+ if (selected) {
88
+ if (onlyThisNodeSelected) {
89
+ setShowToolbar(true);
90
+ }
91
+ } else if (showToolbar) {
92
+ requestClose();
93
+ }
94
+ }, [editor, node, selected]);
95
+
96
+ useEffect(() => {
97
+ // Calculate position relative to selection
98
+ const bodyRect = document.body.getBoundingClientRect();
99
+ const { from } = editor.state.selection;
100
+ const start = editor.view.coordsAtPos(from);
101
+
102
+ setPosition({
103
+ top: start.top + Math.abs(bodyRect.top) + 40, // shift above
104
+ left: start.left,
105
+ });
106
+
107
+ const handleClickOutside = (event) => {
108
+ const insideSomeEditor = event.target.closest('[data-toolbar-for]');
109
+
110
+ if (
111
+ !event.target.closest('[data-inline-dropdown-toolbar]') &&
112
+ (!insideSomeEditor || insideSomeEditor.dataset.toolbarFor !== toolbarEditor.current?.instanceId) &&
113
+ !editor._toolbarOpened &&
114
+ toolbarRef.current &&
115
+ !toolbarRef.current.contains(event.target) &&
116
+ !event.target.closest('[data-inline-node]')
117
+ ) {
118
+ requestClose();
119
+ }
120
+ };
121
+
122
+ if (showToolbar) {
123
+ document.addEventListener('mousedown', handleClickOutside);
124
+ } else {
125
+ document.removeEventListener('mousedown', handleClickOutside);
126
+ }
127
+
128
+ return () => document.removeEventListener('mousedown', handleClickOutside);
129
+ }, [showToolbar]);
130
+
131
+ return (
132
+ <NodeViewWrapper
133
+ className="inline-dropdown"
134
+ data-selected={selected}
135
+ style={{
136
+ display: 'inline-flex',
137
+ height: '50px',
138
+ cursor: 'pointer',
139
+ }}
140
+ >
141
+ <div
142
+ style={{
143
+ display: 'inline-flex',
144
+ minWidth: '178px',
145
+ height: '36px',
146
+ background: '#FFF',
147
+ border: '1px solid #C0C3CF',
148
+ boxSizing: 'border-box',
149
+ borderRadius: '3px',
150
+ margin: '0 2px',
151
+ position: 'relative',
152
+ alignItems: 'center',
153
+ }}
154
+ onClick={() => setShowToolbar(true)}
155
+ >
156
+ <div
157
+ style={{
158
+ flex: 1,
159
+ overflow: 'hidden',
160
+ padding: '0 25px 0 8px',
161
+ whiteSpace: 'nowrap',
162
+ textOverflow: 'ellipsis',
163
+ }}
164
+ >
165
+ <span
166
+ style={{
167
+ display: 'inline-block',
168
+ verticalAlign: 'middle',
169
+ }}
170
+ dangerouslySetInnerHTML={{ __html: html }}
171
+ />
172
+ </div>
173
+ <Chevron direction="down" style={{ position: 'absolute', top: '5px', right: '5px' }} />
174
+ </div>
175
+ {showToolbar && (
176
+ <React.Fragment>
177
+ {ReactDOM.createPortal(
178
+ <div ref={toolbarRef} style={{ zIndex: 1 }}>
179
+ <InlineDropdownToolbar
180
+ editorCallback={(instance) => {
181
+ toolbarEditor.current = instance;
182
+ }}
183
+ />
184
+ </div>,
185
+ document.body,
186
+ )}
187
+
188
+ {editor._tiptapContainerEl &&
189
+ ReactDOM.createPortal(
190
+ <CustomToolbarWrapper
191
+ deletable
192
+ toolbarOpts={{ minWidth: 'auto' }}
193
+ autoWidth
194
+ style={{ top: -40, left: 0, right: 0 }}
195
+ onDelete={() => {
196
+ const { tr } = editor.state;
197
+ tr.delete(pos, pos + node.nodeSize);
198
+ // Prevent the debounced onBlur/onDone from firing into the
199
+ // now-deleted node's stale position
200
+ setToolbarOpened(editor, false);
201
+ delete editor._holdInlineDropdownToolbarIndex;
202
+ editor.view.dispatch(tr);
203
+ setShowToolbar(false);
204
+ editor.commands.focus();
205
+ }}
206
+ showDone={false}
207
+ />,
208
+ editor._tiptapContainerEl,
209
+ )}
210
+ </React.Fragment>
211
+ )}
212
+ </NodeViewWrapper>
213
+ );
214
+ };
215
+
216
+ InlineDropdown.propTypes = {
217
+ attributes: PropTypes.object,
218
+ selectedItem: PropTypes.oneOfType([PropTypes.string, PropTypes.object]),
219
+ };
220
+
221
+ export default InlineDropdown;