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

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 (276) 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 +696 -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 +136 -0
  45. package/lib/components/respArea/ExplicitConstructedResponse.js.map +1 -0
  46. package/lib/components/respArea/InlineDropdown.js +209 -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/package.json +24 -40
  97. package/src/__tests__/EditableHtml.test.jsx +554 -0
  98. package/src/__tests__/constants.test.js +19 -0
  99. package/src/__tests__/div-to-paragraph-conversion.test.jsx +125 -0
  100. package/src/__tests__/extensions.test.js +208 -0
  101. package/src/__tests__/index.test.jsx +154 -0
  102. package/src/__tests__/size-utils.test.js +64 -0
  103. package/src/__tests__/theme.test.js +17 -0
  104. package/src/components/CharacterPicker.jsx +207 -0
  105. package/src/components/EditableHtml.jsx +440 -0
  106. package/src/components/MenuBar.jsx +554 -0
  107. package/src/components/TiptapContainer.jsx +219 -0
  108. package/src/components/__tests__/AltDialog.test.jsx +147 -0
  109. package/src/components/__tests__/CharacterPicker.test.jsx +261 -0
  110. package/src/components/__tests__/CssIcon.test.jsx +46 -0
  111. package/src/components/__tests__/DragInTheBlank.test.jsx +255 -0
  112. package/src/components/__tests__/ExplicitConstructedResponse.test.jsx +204 -0
  113. package/src/components/__tests__/ImageToolbar.test.jsx +128 -0
  114. package/src/components/__tests__/InlineDropdown.test.jsx +388 -0
  115. package/src/components/__tests__/InsertImageHandler.test.js +161 -0
  116. package/src/components/__tests__/MediaDialog.test.jsx +293 -0
  117. package/src/components/__tests__/MediaToolbar.test.jsx +74 -0
  118. package/src/components/__tests__/MediaWrapper.test.jsx +81 -0
  119. package/src/components/__tests__/MenuBar.test.jsx +250 -0
  120. package/src/components/__tests__/RespArea.test.jsx +122 -0
  121. package/src/components/__tests__/TableIcons.test.jsx +149 -0
  122. package/src/components/__tests__/TextAlign.test.jsx +167 -0
  123. package/src/components/__tests__/TiptapContainer.test.jsx +138 -0
  124. package/src/components/__tests__/characterUtils.test.js +166 -0
  125. package/src/components/__tests__/choice.test.jsx +171 -0
  126. package/src/components/__tests__/custom-popper.test.jsx +82 -0
  127. package/src/components/__tests__/done-button.test.jsx +54 -0
  128. package/src/components/__tests__/toolbar-buttons.test.jsx +234 -0
  129. package/src/components/characters/characterUtils.js +447 -0
  130. package/src/components/characters/custom-popper.js +38 -0
  131. package/src/components/common/done-button.jsx +27 -0
  132. package/src/components/common/toolbar-buttons.jsx +122 -0
  133. package/src/components/icons/CssIcon.jsx +15 -0
  134. package/src/components/icons/RespArea.jsx +71 -0
  135. package/src/components/icons/TableIcons.jsx +52 -0
  136. package/src/components/icons/TextAlign.jsx +114 -0
  137. package/src/components/image/AltDialog.jsx +82 -0
  138. package/src/components/image/ImageToolbar.jsx +99 -0
  139. package/src/components/image/InsertImageHandler.js +107 -0
  140. package/src/components/media/MediaDialog.jsx +596 -0
  141. package/src/components/media/MediaToolbar.jsx +49 -0
  142. package/src/components/media/MediaWrapper.jsx +39 -0
  143. package/src/components/respArea/DragInTheBlank/DragInTheBlank.jsx +76 -0
  144. package/src/components/respArea/DragInTheBlank/choice.jsx +256 -0
  145. package/src/components/respArea/ExplicitConstructedResponse.jsx +135 -0
  146. package/src/components/respArea/InlineDropdown.jsx +220 -0
  147. package/src/components/respArea/MathTemplated.jsx +124 -0
  148. package/src/components/respArea/ToolbarIcon.jsx +66 -0
  149. package/src/components/respArea/__tests__/MathTemplated.test.jsx +210 -0
  150. package/src/components/respArea/inlineDropdownUtils.js +79 -0
  151. package/src/constants.js +5 -0
  152. package/src/extensions/__tests__/css.test.js +196 -0
  153. package/src/extensions/__tests__/custom-toolbar-wrapper.test.jsx +180 -0
  154. package/src/extensions/__tests__/divNode.test.js +87 -0
  155. package/src/extensions/__tests__/ensure-empty-root-div.test.js +57 -0
  156. package/src/extensions/__tests__/ensure-list-item-content-is-div.test.js +44 -0
  157. package/src/extensions/__tests__/extended-list-item.test.js +13 -0
  158. package/src/extensions/__tests__/extended-table-cell.test.js +22 -0
  159. package/src/extensions/__tests__/extended-table.test.js +183 -0
  160. package/src/extensions/__tests__/image-component.test.jsx +345 -0
  161. package/src/extensions/__tests__/image.test.js +237 -0
  162. package/src/extensions/__tests__/math.test.js +603 -0
  163. package/src/extensions/__tests__/media-node-view.test.jsx +298 -0
  164. package/src/extensions/__tests__/media.test.js +271 -0
  165. package/src/extensions/__tests__/responseArea.test.js +601 -0
  166. package/src/extensions/css.js +220 -0
  167. package/src/extensions/custom-toolbar-wrapper.jsx +78 -0
  168. package/src/extensions/div-node.js +86 -0
  169. package/src/extensions/ensure-empty-root-div.js +47 -0
  170. package/src/extensions/ensure-list-item-content-is-div.js +62 -0
  171. package/src/extensions/extended-list-item.js +10 -0
  172. package/src/extensions/extended-table-cell.js +19 -0
  173. package/src/extensions/extended-table.js +60 -0
  174. package/src/extensions/heading-paragraph.js +53 -0
  175. package/src/extensions/image-component.jsx +338 -0
  176. package/src/extensions/image.js +109 -0
  177. package/src/extensions/index.js +81 -0
  178. package/src/extensions/math.js +326 -0
  179. package/src/extensions/media.js +188 -0
  180. package/src/extensions/responseArea.js +401 -0
  181. package/src/index.jsx +5 -0
  182. package/src/styles/editorContainerStyles.js +145 -0
  183. package/src/theme.js +1 -0
  184. package/src/utils/__tests__/helper.test.js +126 -0
  185. package/src/utils/helper.js +69 -0
  186. package/src/utils/size.js +32 -0
  187. package/dist/components/CharacterPicker.d.ts +0 -31
  188. package/dist/components/CharacterPicker.js +0 -131
  189. package/dist/components/EditableHtml.d.ts +0 -11
  190. package/dist/components/EditableHtml.js +0 -291
  191. package/dist/components/MenuBar.d.ts +0 -11
  192. package/dist/components/MenuBar.js +0 -462
  193. package/dist/components/TiptapContainer.d.ts +0 -11
  194. package/dist/components/TiptapContainer.js +0 -154
  195. package/dist/components/characters/characterUtils.d.ts +0 -35
  196. package/dist/components/characters/characterUtils.js +0 -465
  197. package/dist/components/characters/custom-popper.d.ts +0 -14
  198. package/dist/components/characters/custom-popper.js +0 -32
  199. package/dist/components/common/done-button.d.ts +0 -30
  200. package/dist/components/common/done-button.js +0 -26
  201. package/dist/components/common/toolbar-buttons.d.ts +0 -38
  202. package/dist/components/common/toolbar-buttons.js +0 -91
  203. package/dist/components/icons/CssIcon.d.ts +0 -11
  204. package/dist/components/icons/CssIcon.js +0 -14
  205. package/dist/components/icons/RespArea.d.ts +0 -26
  206. package/dist/components/icons/RespArea.js +0 -42
  207. package/dist/components/icons/TableIcons.d.ts +0 -14
  208. package/dist/components/icons/TableIcons.js +0 -32
  209. package/dist/components/icons/TextAlign.d.ts +0 -18
  210. package/dist/components/icons/TextAlign.js +0 -134
  211. package/dist/components/image/AltDialog.d.ts +0 -22
  212. package/dist/components/image/AltDialog.js +0 -61
  213. package/dist/components/image/ImageToolbar.d.ts +0 -24
  214. package/dist/components/image/ImageToolbar.js +0 -80
  215. package/dist/components/image/InsertImageHandler.d.ts +0 -32
  216. package/dist/components/image/InsertImageHandler.js +0 -53
  217. package/dist/components/media/MediaDialog.d.ts +0 -43
  218. package/dist/components/media/MediaDialog.js +0 -389
  219. package/dist/components/media/MediaToolbar.d.ts +0 -19
  220. package/dist/components/media/MediaToolbar.js +0 -41
  221. package/dist/components/media/MediaWrapper.d.ts +0 -19
  222. package/dist/components/respArea/DragInTheBlank/DragInTheBlank.d.ts +0 -23
  223. package/dist/components/respArea/DragInTheBlank/DragInTheBlank.js +0 -58
  224. package/dist/components/respArea/DragInTheBlank/choice.d.ts +0 -56
  225. package/dist/components/respArea/DragInTheBlank/choice.js +0 -156
  226. package/dist/components/respArea/ExplicitConstructedResponse.d.ts +0 -20
  227. package/dist/components/respArea/ExplicitConstructedResponse.js +0 -83
  228. package/dist/components/respArea/InlineDropdown.d.ts +0 -18
  229. package/dist/components/respArea/InlineDropdown.js +0 -119
  230. package/dist/components/respArea/MathTemplated.d.ts +0 -19
  231. package/dist/components/respArea/MathTemplated.js +0 -97
  232. package/dist/components/respArea/ToolbarIcon.d.ts +0 -14
  233. package/dist/components/respArea/ToolbarIcon.js +0 -17
  234. package/dist/components/respArea/inlineDropdownUtils.d.ts +0 -15
  235. package/dist/components/respArea/inlineDropdownUtils.js +0 -15
  236. package/dist/constants.d.ts +0 -13
  237. package/dist/constants.js +0 -4
  238. package/dist/extensions/css.d.ts +0 -11
  239. package/dist/extensions/css.js +0 -115
  240. package/dist/extensions/custom-toolbar-wrapper.d.ts +0 -11
  241. package/dist/extensions/custom-toolbar-wrapper.js +0 -61
  242. package/dist/extensions/div-node.d.ts +0 -10
  243. package/dist/extensions/div-node.js +0 -42
  244. package/dist/extensions/ensure-empty-root-div.d.ts +0 -14
  245. package/dist/extensions/ensure-empty-root-div.js +0 -24
  246. package/dist/extensions/ensure-list-item-content-is-div.d.ts +0 -15
  247. package/dist/extensions/ensure-list-item-content-is-div.js +0 -31
  248. package/dist/extensions/extended-list-item.d.ts +0 -13
  249. package/dist/extensions/extended-list-item.js +0 -5
  250. package/dist/extensions/extended-table-cell.d.ts +0 -10
  251. package/dist/extensions/extended-table-cell.js +0 -6
  252. package/dist/extensions/extended-table.d.ts +0 -17
  253. package/dist/extensions/extended-table.js +0 -34
  254. package/dist/extensions/heading-paragraph.d.ts +0 -17
  255. package/dist/extensions/heading-paragraph.js +0 -30
  256. package/dist/extensions/image-component.d.ts +0 -22
  257. package/dist/extensions/image-component.js +0 -220
  258. package/dist/extensions/image.d.ts +0 -10
  259. package/dist/extensions/image.js +0 -68
  260. package/dist/extensions/index.d.ts +0 -16
  261. package/dist/extensions/index.js +0 -64
  262. package/dist/extensions/math.d.ts +0 -15
  263. package/dist/extensions/math.js +0 -158
  264. package/dist/extensions/media.d.ts +0 -19
  265. package/dist/extensions/media.js +0 -149
  266. package/dist/extensions/responseArea.d.ts +0 -27
  267. package/dist/extensions/responseArea.js +0 -259
  268. package/dist/index.d.ts +0 -13
  269. package/dist/index.js +0 -7
  270. package/dist/node_modules/.bun/clsx@2.1.1/node_modules/clsx/dist/clsx.js +0 -16
  271. package/dist/styles/editorContainerStyles.d.ts +0 -134
  272. package/dist/theme.d.ts +0 -9
  273. package/dist/utils/helper.d.ts +0 -9
  274. package/dist/utils/helper.js +0 -27
  275. package/dist/utils/size.d.ts +0 -9
  276. 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,135 @@
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
+
7
+ const ExplicitConstructedResponse = (props) => {
8
+ const { editor, node, getPos, options, selected } = props;
9
+ const { attrs: attributes } = node;
10
+ const { value } = attributes;
11
+ const { respAreaToolbar, error: errorFn } = options;
12
+ const pos = getPos();
13
+ const [showToolbar, setShowToolbar] = useState(false);
14
+ const EcrToolbar = respAreaToolbar([node, pos], editor, () => {});
15
+ const toolbarRef = useRef(null);
16
+
17
+ let error;
18
+
19
+ if (errorFn) {
20
+ const errorValue = errorFn();
21
+ const respIndex = parseInt(attributes.index, 10);
22
+
23
+ error = !!errorValue?.[respIndex]?.[0];
24
+ }
25
+
26
+ useEffect(() => {
27
+ const { selection } = editor.state;
28
+ const onlyThisNodeSelected = selection.from + node.nodeSize === selection.to;
29
+
30
+ if (selected) {
31
+ if (onlyThisNodeSelected) {
32
+ setShowToolbar(selected);
33
+ }
34
+ } else {
35
+ setShowToolbar(selected);
36
+ }
37
+ }, [editor, node, selected]);
38
+
39
+ useEffect(() => {
40
+ const handleClickOutside = (event) => {
41
+ const insideCharacterPicker =
42
+ event.target.closest('.insert-character-dialog') || event.target.closest('[data-toolbar-for]');
43
+
44
+ if (
45
+ !insideCharacterPicker &&
46
+ toolbarRef.current &&
47
+ !toolbarRef.current.contains(event.target) &&
48
+ !event.target.closest('[data-inline-node]')
49
+ ) {
50
+ setShowToolbar(false);
51
+ }
52
+ };
53
+
54
+ if (showToolbar) {
55
+ document.addEventListener('mousedown', handleClickOutside);
56
+ } else {
57
+ document.removeEventListener('mousedown', handleClickOutside);
58
+ }
59
+
60
+ return () => document.removeEventListener('mousedown', handleClickOutside);
61
+ }, [showToolbar]);
62
+
63
+ return (
64
+ <NodeViewWrapper
65
+ className="drag-in-the-blank"
66
+ data-selected={selected}
67
+ style={{
68
+ display: 'inline-flex',
69
+ minHeight: '55px',
70
+ position: 'relative',
71
+ cursor: 'pointer',
72
+ }}
73
+ >
74
+ <div
75
+ {...attributes}
76
+ style={{
77
+ display: 'inline-flex',
78
+ width: '100%',
79
+ minHeight: '46px',
80
+ height: '46px',
81
+ backgroundColor: '#FFF',
82
+ border: `1px solid ${error ? 'red' : '#C0C3CF'}`,
83
+ boxSizing: 'border-box',
84
+ borderRadius: '4px',
85
+ overflow: 'hidden',
86
+ padding: '12px 21px',
87
+ margin: '0 4px',
88
+ minWidth: '178px',
89
+ visibility: showToolbar ? 'hidden' : 'visible',
90
+ }}
91
+ onClick={() => setShowToolbar(true)}
92
+ dangerouslySetInnerHTML={{
93
+ __html: value || '<div>&nbsp;</div>',
94
+ }}
95
+ />
96
+ {showToolbar && (
97
+ <React.Fragment>
98
+ <div ref={toolbarRef} className="absolute z-50 bg-white shadow-lg rounded p-2" style={{ zIndex: 1 }}>
99
+ <EcrToolbar />
100
+ </div>
101
+ {editor._tiptapContainerEl &&
102
+ ReactDOM.createPortal(
103
+ <CustomToolbarWrapper
104
+ deletable
105
+ toolbarOpts={{ minWidth: 'auto' }}
106
+ autoWidth
107
+ style={{ top: -40, left: 0, right: 0 }}
108
+ onDelete={() => {
109
+ const { tr } = editor.state;
110
+ tr.delete(pos, pos + node.nodeSize);
111
+ // Prevent the debounced onBlur/onDone from firing into the
112
+ // now-deleted node's stale position
113
+ editor._toolbarOpened = false;
114
+ editor.view.dispatch(tr);
115
+ setShowToolbar(false);
116
+ editor.commands.focus();
117
+ }}
118
+ showDone={false}
119
+ />,
120
+ editor._tiptapContainerEl,
121
+ )}
122
+ </React.Fragment>
123
+ )}
124
+ </NodeViewWrapper>
125
+ );
126
+ };
127
+
128
+ ExplicitConstructedResponse.propTypes = {
129
+ attributes: PropTypes.object,
130
+ error: PropTypes.any,
131
+ value: PropTypes.string,
132
+ isFocused: PropTypes.bool,
133
+ };
134
+
135
+ export default ExplicitConstructedResponse;
@@ -0,0 +1,220 @@
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
+
9
+ const InlineDropdown = (props) => {
10
+ const { editor, node, getPos, options, selected } = props;
11
+ const { attrs: attributes } = node;
12
+ const { value, error } = attributes;
13
+ const html = value || '<div>&nbsp</div>';
14
+ const pos = getPos();
15
+ const toolbarRef = useRef(null);
16
+ const toolbarEditor = useRef(null);
17
+ const pendingCloseRequest = useRef(false);
18
+
19
+ const isHeld = () =>
20
+ editor._holdInlineDropdownToolbarIndex != null &&
21
+ String(editor._holdInlineDropdownToolbarIndex) === String(node.attrs.index);
22
+
23
+ const [showToolbar, setShowToolbar] = useState(false);
24
+ const [position, setPosition] = useState({ top: 0, left: 0 });
25
+
26
+ const closeToolbar = () => {
27
+ if (isHeld()) {
28
+ return;
29
+ }
30
+
31
+ setShowToolbar(false);
32
+ };
33
+
34
+ const InlineDropdownToolbar = options.respAreaToolbar([node, pos], editor, closeToolbar);
35
+
36
+ const reselectNode = () => {
37
+ const { tr } = editor.state;
38
+ const nodeAtPos = tr.doc.nodeAt(pos);
39
+
40
+ if (!nodeAtPos) {
41
+ return;
42
+ }
43
+
44
+ const { selection } = tr;
45
+
46
+ if (selection.from === pos && selection.to === pos + nodeAtPos.nodeSize) {
47
+ return;
48
+ }
49
+
50
+ tr.setSelection(NodeSelection.create(tr.doc, pos));
51
+ editor.view.dispatch(tr);
52
+ };
53
+
54
+ const requestClose = () => {
55
+ if (pendingCloseRequest.current) {
56
+ return;
57
+ }
58
+
59
+ if (options.onToolbarCloseRequest) {
60
+ pendingCloseRequest.current = true;
61
+
62
+ options.onToolbarCloseRequest(
63
+ [node, pos],
64
+ editor,
65
+ () => {
66
+ pendingCloseRequest.current = false;
67
+ delete editor._holdInlineDropdownToolbarIndex;
68
+ closeToolbar();
69
+ },
70
+ () => {
71
+ pendingCloseRequest.current = false;
72
+ delete editor._holdInlineDropdownToolbarIndex;
73
+ setShowToolbar(true);
74
+ setTimeout(reselectNode, 0);
75
+ },
76
+ );
77
+ } else {
78
+ closeToolbar();
79
+ }
80
+ };
81
+
82
+ useEffect(() => {
83
+ const { selection } = editor.state;
84
+ const onlyThisNodeSelected = selection.from + node.nodeSize === selection.to;
85
+
86
+ if (selected) {
87
+ if (onlyThisNodeSelected) {
88
+ setShowToolbar(true);
89
+ }
90
+ } else if (showToolbar) {
91
+ requestClose();
92
+ }
93
+ }, [editor, node, selected]);
94
+
95
+ useEffect(() => {
96
+ // Calculate position relative to selection
97
+ const bodyRect = document.body.getBoundingClientRect();
98
+ const { from } = editor.state.selection;
99
+ const start = editor.view.coordsAtPos(from);
100
+
101
+ setPosition({
102
+ top: start.top + Math.abs(bodyRect.top) + 40, // shift above
103
+ left: start.left,
104
+ });
105
+
106
+ const handleClickOutside = (event) => {
107
+ const insideSomeEditor = event.target.closest('[data-toolbar-for]');
108
+
109
+ if (
110
+ !event.target.closest('[data-inline-dropdown-toolbar]') &&
111
+ (!insideSomeEditor || insideSomeEditor.dataset.toolbarFor !== toolbarEditor.current?.instanceId) &&
112
+ !editor._toolbarOpened &&
113
+ toolbarRef.current &&
114
+ !toolbarRef.current.contains(event.target) &&
115
+ !event.target.closest('[data-inline-node]')
116
+ ) {
117
+ requestClose();
118
+ }
119
+ };
120
+
121
+ if (showToolbar) {
122
+ document.addEventListener('mousedown', handleClickOutside);
123
+ } else {
124
+ document.removeEventListener('mousedown', handleClickOutside);
125
+ }
126
+
127
+ return () => document.removeEventListener('mousedown', handleClickOutside);
128
+ }, [showToolbar]);
129
+
130
+ return (
131
+ <NodeViewWrapper
132
+ className="inline-dropdown"
133
+ data-selected={selected}
134
+ style={{
135
+ display: 'inline-flex',
136
+ height: '50px',
137
+ cursor: 'pointer',
138
+ }}
139
+ >
140
+ <div
141
+ style={{
142
+ display: 'inline-flex',
143
+ minWidth: '178px',
144
+ height: '36px',
145
+ background: '#FFF',
146
+ border: '1px solid #C0C3CF',
147
+ boxSizing: 'border-box',
148
+ borderRadius: '3px',
149
+ margin: '0 2px',
150
+ position: 'relative',
151
+ alignItems: 'center',
152
+ }}
153
+ onClick={() => setShowToolbar(true)}
154
+ >
155
+ <div
156
+ style={{
157
+ flex: 1,
158
+ overflow: 'hidden',
159
+ padding: '0 25px 0 8px',
160
+ whiteSpace: 'nowrap',
161
+ textOverflow: 'ellipsis',
162
+ }}
163
+ >
164
+ <span
165
+ style={{
166
+ display: 'inline-block',
167
+ verticalAlign: 'middle',
168
+ }}
169
+ dangerouslySetInnerHTML={{ __html: html }}
170
+ />
171
+ </div>
172
+ <Chevron direction="down" style={{ position: 'absolute', top: '5px', right: '5px' }} />
173
+ </div>
174
+ {showToolbar && (
175
+ <React.Fragment>
176
+ {ReactDOM.createPortal(
177
+ <div ref={toolbarRef} style={{ zIndex: 1 }}>
178
+ <InlineDropdownToolbar
179
+ editorCallback={(instance) => {
180
+ toolbarEditor.current = instance;
181
+ }}
182
+ />
183
+ </div>,
184
+ document.body,
185
+ )}
186
+
187
+ {editor._tiptapContainerEl &&
188
+ ReactDOM.createPortal(
189
+ <CustomToolbarWrapper
190
+ deletable
191
+ toolbarOpts={{ minWidth: 'auto' }}
192
+ autoWidth
193
+ style={{ top: -40, left: 0, right: 0 }}
194
+ onDelete={() => {
195
+ const { tr } = editor.state;
196
+ tr.delete(pos, pos + node.nodeSize);
197
+ // Prevent the debounced onBlur/onDone from firing into the
198
+ // now-deleted node's stale position
199
+ editor._toolbarOpened = false;
200
+ delete editor._holdInlineDropdownToolbarIndex;
201
+ editor.view.dispatch(tr);
202
+ setShowToolbar(false);
203
+ editor.commands.focus();
204
+ }}
205
+ showDone={false}
206
+ />,
207
+ editor._tiptapContainerEl,
208
+ )}
209
+ </React.Fragment>
210
+ )}
211
+ </NodeViewWrapper>
212
+ );
213
+ };
214
+
215
+ InlineDropdown.propTypes = {
216
+ attributes: PropTypes.object,
217
+ selectedItem: PropTypes.oneOfType([PropTypes.string, PropTypes.object]),
218
+ };
219
+
220
+ export default InlineDropdown;