@digabi/exam-engine-core 19.1.0-alpha.0 → 19.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (133) hide show
  1. package/dist/__tests__/tsconfig.tsbuildinfo +1 -1
  2. package/dist/components/grading/AnnotationPopup.d.ts +11 -0
  3. package/dist/components/grading/AnnotationPopup.d.ts.map +1 -0
  4. package/dist/components/grading/AnnotationPopup.js +20 -0
  5. package/dist/components/grading/AnnotationPopup.js.map +1 -0
  6. package/dist/components/grading/AnswerCharacterCounter.d.ts +6 -0
  7. package/dist/components/grading/AnswerCharacterCounter.d.ts.map +1 -0
  8. package/dist/components/grading/AnswerCharacterCounter.js +21 -0
  9. package/dist/components/grading/AnswerCharacterCounter.js.map +1 -0
  10. package/dist/components/grading/AnswerWithAnnotations.d.ts +11 -0
  11. package/dist/components/grading/AnswerWithAnnotations.d.ts.map +1 -0
  12. package/dist/components/grading/AnswerWithAnnotations.js +15 -0
  13. package/dist/components/grading/AnswerWithAnnotations.js.map +1 -0
  14. package/dist/components/grading/ChoiceAnswer.d.ts +6 -0
  15. package/dist/components/grading/ChoiceAnswer.d.ts.map +1 -0
  16. package/dist/components/grading/ChoiceAnswer.js +57 -0
  17. package/dist/components/grading/ChoiceAnswer.js.map +1 -0
  18. package/dist/components/grading/DropdownAnswer.d.ts +6 -0
  19. package/dist/components/grading/DropdownAnswer.d.ts.map +1 -0
  20. package/dist/components/grading/DropdownAnswer.js +43 -0
  21. package/dist/components/grading/DropdownAnswer.js.map +1 -0
  22. package/dist/components/grading/Grading.d.ts +4 -0
  23. package/dist/components/grading/Grading.d.ts.map +1 -0
  24. package/dist/components/grading/Grading.js +81 -0
  25. package/dist/components/grading/Grading.js.map +1 -0
  26. package/dist/components/grading/GradingAnswer.d.ts +22 -0
  27. package/dist/components/grading/GradingAnswer.d.ts.map +1 -0
  28. package/dist/components/grading/GradingAnswer.js +242 -0
  29. package/dist/components/grading/GradingAnswer.js.map +1 -0
  30. package/dist/components/grading/GradingAnswerAnnotationList.d.ts +9 -0
  31. package/dist/components/grading/GradingAnswerAnnotationList.d.ts.map +1 -0
  32. package/dist/components/grading/GradingAnswerAnnotationList.js +19 -0
  33. package/dist/components/grading/GradingAnswerAnnotationList.js.map +1 -0
  34. package/dist/components/grading/MultiLineAnswer.d.ts +14 -0
  35. package/dist/components/grading/MultiLineAnswer.d.ts.map +1 -0
  36. package/dist/components/grading/MultiLineAnswer.js +61 -0
  37. package/dist/components/grading/MultiLineAnswer.js.map +1 -0
  38. package/dist/components/grading/Question.d.ts +5 -0
  39. package/dist/components/grading/Question.d.ts.map +1 -0
  40. package/dist/components/grading/Question.js +23 -0
  41. package/dist/components/grading/Question.js.map +1 -0
  42. package/dist/components/grading/QuestionTitle.d.ts +6 -0
  43. package/dist/components/grading/QuestionTitle.d.ts.map +1 -0
  44. package/dist/components/grading/QuestionTitle.js +21 -0
  45. package/dist/components/grading/QuestionTitle.js.map +1 -0
  46. package/dist/components/grading/Results.d.ts +15 -0
  47. package/dist/components/grading/Results.d.ts.map +1 -0
  48. package/dist/components/grading/Results.js +77 -0
  49. package/dist/components/grading/Results.js.map +1 -0
  50. package/dist/components/grading/ScoredTextAnswer.d.ts +6 -0
  51. package/dist/components/grading/ScoredTextAnswer.d.ts.map +1 -0
  52. package/dist/components/grading/ScoredTextAnswer.js +22 -0
  53. package/dist/components/grading/ScoredTextAnswer.js.map +1 -0
  54. package/dist/components/grading/Section.d.ts +5 -0
  55. package/dist/components/grading/Section.d.ts.map +1 -0
  56. package/dist/components/grading/Section.js +18 -0
  57. package/dist/components/grading/Section.js.map +1 -0
  58. package/dist/components/grading/SingleLineAnswer.d.ts +12 -0
  59. package/dist/components/grading/SingleLineAnswer.d.ts.map +1 -0
  60. package/dist/components/grading/SingleLineAnswer.js +23 -0
  61. package/dist/components/grading/SingleLineAnswer.js.map +1 -0
  62. package/dist/components/grading/TextAnswer.d.ts +7 -0
  63. package/dist/components/grading/TextAnswer.d.ts.map +1 -0
  64. package/dist/components/grading/TextAnswer.js +53 -0
  65. package/dist/components/grading/TextAnswer.js.map +1 -0
  66. package/dist/components/grading/editAnnotations.d.ts +26 -0
  67. package/dist/components/grading/editAnnotations.d.ts.map +1 -0
  68. package/dist/components/grading/editAnnotations.js +177 -0
  69. package/dist/components/grading/editAnnotations.js.map +1 -0
  70. package/dist/components/grading/internal/AnnotationList.d.ts +5 -0
  71. package/dist/components/grading/internal/AnnotationList.d.ts.map +1 -0
  72. package/dist/components/grading/internal/AnnotationList.js +43 -0
  73. package/dist/components/grading/internal/AnnotationList.js.map +1 -0
  74. package/dist/components/grading/internal/QuestionAutoScore.d.ts +10 -0
  75. package/dist/components/grading/internal/QuestionAutoScore.d.ts.map +1 -0
  76. package/dist/components/grading/internal/QuestionAutoScore.js +17 -0
  77. package/dist/components/grading/internal/QuestionAutoScore.js.map +1 -0
  78. package/dist/components/grading/internal/QuestionManualScore.d.ts +12 -0
  79. package/dist/components/grading/internal/QuestionManualScore.d.ts.map +1 -0
  80. package/dist/components/grading/internal/QuestionManualScore.js +61 -0
  81. package/dist/components/grading/internal/QuestionManualScore.js.map +1 -0
  82. package/dist/components/grading/internal/QuestionScoresContainer.d.ts +10 -0
  83. package/dist/components/grading/internal/QuestionScoresContainer.d.ts.map +1 -0
  84. package/dist/components/grading/internal/QuestionScoresContainer.js +12 -0
  85. package/dist/components/grading/internal/QuestionScoresContainer.js.map +1 -0
  86. package/dist/components/grading/largeImageDetector.d.ts +3 -0
  87. package/dist/components/grading/largeImageDetector.d.ts.map +1 -0
  88. package/dist/components/grading/largeImageDetector.js +43 -0
  89. package/dist/components/grading/largeImageDetector.js.map +1 -0
  90. package/dist/components/grading/largeImageResizer.d.ts +2 -0
  91. package/dist/components/grading/largeImageResizer.d.ts.map +1 -0
  92. package/dist/components/grading/largeImageResizer.js +33 -0
  93. package/dist/components/grading/largeImageResizer.js.map +1 -0
  94. package/dist/components/results/Results2.d.ts +15 -0
  95. package/dist/components/results/Results2.d.ts.map +1 -0
  96. package/dist/components/results/Results2.js +79 -0
  97. package/dist/components/results/Results2.js.map +1 -0
  98. package/dist/components/results/internal/AnnotationList.d.ts +1 -1
  99. package/dist/components/results/internal/AnnotationList.d.ts.map +1 -1
  100. package/dist/components/results/internal/AnnotationList.js +3 -14
  101. package/dist/components/results/internal/AnnotationList.js.map +1 -1
  102. package/dist/components/shared/AnnotationListComponent.d.ts +1 -0
  103. package/dist/components/shared/AnnotationListComponent.d.ts.map +1 -0
  104. package/dist/components/shared/AnnotationListComponent.js +2 -0
  105. package/dist/components/shared/AnnotationListComponent.js.map +1 -0
  106. package/dist/components/shared/AnnotationLists.d.ts +13 -0
  107. package/dist/components/shared/AnnotationLists.d.ts.map +1 -0
  108. package/dist/components/shared/AnnotationLists.js +17 -0
  109. package/dist/components/shared/AnnotationLists.js.map +1 -0
  110. package/dist/editAnnotations.d.ts +2 -0
  111. package/dist/editAnnotations.d.ts.map +1 -0
  112. package/dist/editAnnotations.js +8 -0
  113. package/dist/editAnnotations.js.map +1 -0
  114. package/dist/editAnnotationsOld.d.ts +3 -0
  115. package/dist/editAnnotationsOld.d.ts.map +1 -0
  116. package/dist/editAnnotationsOld.js +340 -0
  117. package/dist/editAnnotationsOld.js.map +1 -0
  118. package/dist/i18n/fi-FI.d.ts +1 -0
  119. package/dist/i18n/fi-FI.d.ts.map +1 -1
  120. package/dist/i18n/fi-FI.js +1 -0
  121. package/dist/i18n/fi-FI.js.map +1 -1
  122. package/dist/i18n/index.d.ts +1 -0
  123. package/dist/i18n/index.d.ts.map +1 -1
  124. package/dist/i18n/sv-FI.d.ts.map +1 -1
  125. package/dist/i18n/sv-FI.js +1 -0
  126. package/dist/i18n/sv-FI.js.map +1 -1
  127. package/dist/main-bundle.js +1 -1
  128. package/dist/main.css +1 -1
  129. package/dist/renderAnnotations.d.ts +6 -2
  130. package/dist/renderAnnotations.d.ts.map +1 -1
  131. package/dist/renderAnnotations.js +61 -41
  132. package/dist/renderAnnotations.js.map +1 -1
  133. package/package.json +2 -2
@@ -0,0 +1,11 @@
1
+ import React from 'react';
2
+ export declare const AnnotationPopup: React.FunctionComponent<{
3
+ setMessage: (message: string) => void;
4
+ message: string;
5
+ popupVisible: boolean;
6
+ position: {
7
+ left: number;
8
+ top: number;
9
+ };
10
+ }>;
11
+ //# sourceMappingURL=AnnotationPopup.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AnnotationPopup.d.ts","sourceRoot":"","sources":["../../../src/components/grading/AnnotationPopup.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAsC,MAAM,OAAO,CAAA;AAE1D,eAAO,MAAM,eAAe,EAAE,KAAK,CAAC,iBAAiB,CAAC;IACpD,UAAU,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAA;IACrC,OAAO,EAAE,MAAM,CAAA;IACf,YAAY,EAAE,OAAO,CAAA;IACrB,QAAQ,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,GAAG,EAAE,MAAM,CAAA;KAAE,CAAA;CACxC,CAuCA,CAAA"}
@@ -0,0 +1,20 @@
1
+ import React, { useEffect, useRef, useState } from 'react';
2
+ export const AnnotationPopup = ({ setMessage, message, popupVisible, position }) => {
3
+ const popupRef = useRef(null);
4
+ const [field, setField] = useState('');
5
+ useEffect(() => {
6
+ setField(message);
7
+ }, [message]);
8
+ return popupVisible ? (React.createElement("div", { style: {
9
+ display: popupVisible ? 'block' : 'none',
10
+ position: 'absolute',
11
+ left: position.left,
12
+ top: position.top,
13
+ }, className: "popup", ref: popupRef },
14
+ React.createElement("div", { className: "popup add-annotation-popup" },
15
+ React.createElement("form", { onSubmit: (e) => e.preventDefault() },
16
+ React.createElement("input", { onChange: (e) => setField(e.target.value), value: field, className: "add-annotation-text", type: "text", autoFocus: true }),
17
+ React.createElement("i", { className: "fa fa-comment" }),
18
+ React.createElement("button", { type: "submit", "data-i18n": "arpa.annotate", onClick: () => setMessage(field) }, "Merkitse"))))) : (React.createElement(React.Fragment, null));
19
+ };
20
+ //# sourceMappingURL=AnnotationPopup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AnnotationPopup.js","sourceRoot":"","sources":["../../../src/components/grading/AnnotationPopup.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EAAE,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAA;AAE1D,MAAM,CAAC,MAAM,eAAe,GAKvB,CAAC,EAAE,UAAU,EAAE,OAAO,EAAE,YAAY,EAAE,QAAQ,EAAE,EAAE,EAAE;IACvD,MAAM,QAAQ,GAAG,MAAM,CAAiB,IAAI,CAAC,CAAA;IAE7C,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAS,EAAE,CAAC,CAAA;IAE9C,SAAS,CAAC,GAAG,EAAE;QACb,QAAQ,CAAC,OAAO,CAAC,CAAA;IACnB,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,CAAA;IAEb,OAAO,YAAY,CAAC,CAAC,CAAC,CACpB,6BACE,KAAK,EAAE;YACL,OAAO,EAAE,YAAY,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM;YACxC,QAAQ,EAAE,UAAU;YACpB,IAAI,EAAE,QAAQ,CAAC,IAAI;YACnB,GAAG,EAAE,QAAQ,CAAC,GAAG;SAClB,EACD,SAAS,EAAC,OAAO,EACjB,GAAG,EAAE,QAAQ;QAEb,6BAAK,SAAS,EAAC,4BAA4B;YACzC,8BAAM,QAAQ,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,cAAc,EAAE;gBACvC,+BACE,QAAQ,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EACzC,KAAK,EAAE,KAAK,EACZ,SAAS,EAAC,qBAAqB,EAC/B,IAAI,EAAC,MAAM,EACX,SAAS,EAAE,IAAI,GACf;gBACF,2BAAG,SAAS,EAAC,eAAe,GAAK;gBACjC,gCAAQ,IAAI,EAAC,QAAQ,eAAW,eAAe,EAAC,OAAO,EAAE,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,CAAC,eAEvE,CACJ,CACH,CACF,CACP,CAAC,CAAC,CAAC,CACF,yCAAK,CACN,CAAA;AACH,CAAC,CAAA"}
@@ -0,0 +1,6 @@
1
+ /// <reference types="react" />
2
+ export declare function AnswerCharacterCounter({ characterCount, maxLength, }: {
3
+ characterCount: number;
4
+ maxLength: number | undefined;
5
+ }): JSX.Element;
6
+ //# sourceMappingURL=AnswerCharacterCounter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AnswerCharacterCounter.d.ts","sourceRoot":"","sources":["../../../src/components/grading/AnswerCharacterCounter.tsx"],"names":[],"mappings":";AAGA,wBAAgB,sBAAsB,CAAC,EACrC,cAAc,EACd,SAAS,GACV,EAAE;IACD,cAAc,EAAE,MAAM,CAAA;IACtB,SAAS,EAAE,MAAM,GAAG,SAAS,CAAA;CAC9B,eAqBA"}
@@ -0,0 +1,21 @@
1
+ import React from 'react';
2
+ import { useExamTranslation } from '../../i18n';
3
+ export function AnswerCharacterCounter({ characterCount, maxLength, }) {
4
+ const { t } = useExamTranslation();
5
+ const percentage = countSurplusPercentage(characterCount, maxLength);
6
+ return (React.createElement("div", { className: "e-font-size-xs e-grading-answer-length" },
7
+ t('answer-length', {
8
+ count: characterCount,
9
+ }),
10
+ percentage > 0 ? (React.createElement("span", { className: "e-grading-answer-max-length-surplus" }, t('max-length-surplus', {
11
+ maxLength,
12
+ percentage,
13
+ }))) : ('')));
14
+ }
15
+ function countSurplusPercentage(characters, maxCharacters) {
16
+ if (!maxCharacters || characters <= maxCharacters) {
17
+ return 0;
18
+ }
19
+ return Math.floor((100 * characters) / maxCharacters - 100);
20
+ }
21
+ //# sourceMappingURL=AnswerCharacterCounter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AnswerCharacterCounter.js","sourceRoot":"","sources":["../../../src/components/grading/AnswerCharacterCounter.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAA;AACzB,OAAO,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAA;AAE/C,MAAM,UAAU,sBAAsB,CAAC,EACrC,cAAc,EACd,SAAS,GAIV;IACC,MAAM,EAAE,CAAC,EAAE,GAAG,kBAAkB,EAAE,CAAA;IAClC,MAAM,UAAU,GAAG,sBAAsB,CAAC,cAAc,EAAE,SAAS,CAAC,CAAA;IAEpE,OAAO,CACL,6BAAK,SAAS,EAAC,wCAAwC;QACpD,CAAC,CAAC,eAAe,EAAE;YAClB,KAAK,EAAE,cAAc;SACtB,CAAC;QACD,UAAU,GAAG,CAAC,CAAC,CAAC,CAAC,CAChB,8BAAM,SAAS,EAAC,qCAAqC,IAClD,CAAC,CAAC,oBAAoB,EAAE;YACvB,SAAS;YACT,UAAU;SACX,CAAC,CACG,CACR,CAAC,CAAC,CAAC,CACF,EAAE,CACH,CACG,CACP,CAAA;AACH,CAAC;AAED,SAAS,sBAAsB,CAAC,UAAkB,EAAE,aAAwC;IAC1F,IAAI,CAAC,aAAa,IAAI,UAAU,IAAI,aAAa,EAAE;QACjD,OAAO,CAAC,CAAA;KACT;IACD,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,UAAU,CAAC,GAAG,aAAa,GAAG,GAAG,CAAC,CAAA;AAC7D,CAAC"}
@@ -0,0 +1,11 @@
1
+ import React from 'react';
2
+ import { Annotation } from '../..';
3
+ export declare const AnswerWithAnnotations: React.FunctionComponent<{
4
+ type: 'richText' | 'text';
5
+ value?: string;
6
+ annotations: {
7
+ pregrading: Annotation[];
8
+ censoring: Annotation[];
9
+ };
10
+ }>;
11
+ //# sourceMappingURL=AnswerWithAnnotations.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AnswerWithAnnotations.d.ts","sourceRoot":"","sources":["../../../src/components/grading/AnswerWithAnnotations.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAkC,MAAM,OAAO,CAAA;AACtD,OAAO,EAAE,UAAU,EAAE,MAAM,OAAO,CAAA;AAIlC,eAAO,MAAM,qBAAqB,EAAE,KAAK,CAAC,iBAAiB,CAAC;IAC1D,IAAI,EAAE,UAAU,GAAG,MAAM,CAAA;IACzB,KAAK,CAAC,EAAE,MAAM,CAAA;IACd,WAAW,EAAE;QAAE,UAAU,EAAE,UAAU,EAAE,CAAC;QAAC,SAAS,EAAE,UAAU,EAAE,CAAA;KAAE,CAAA;CACnE,CAmBA,CAAA"}
@@ -0,0 +1,15 @@
1
+ import React, { useLayoutEffect, useRef } from 'react';
2
+ import { renderAnnotations } from '../../renderAnnotations';
3
+ export const AnswerWithAnnotations = ({ type, annotations, value }) => {
4
+ const answerRef = useRef(null);
5
+ useLayoutEffect(() => {
6
+ console.log('LAYOUT', annotations.censoring.length);
7
+ if (answerRef.current) {
8
+ answerRef.current.innerHTML = value || '';
9
+ renderAnnotations(answerRef.current, annotations.pregrading, annotations.censoring);
10
+ }
11
+ });
12
+ return type === 'richText' ? (React.createElement("div", { ref: answerRef })) : (React.createElement("span", { className: "text-answer text-answer--single-line" },
13
+ React.createElement("span", { className: "e-inline-block", ref: answerRef }, value)));
14
+ };
15
+ //# sourceMappingURL=AnswerWithAnnotations.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AnswerWithAnnotations.js","sourceRoot":"","sources":["../../../src/components/grading/AnswerWithAnnotations.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EAAE,eAAe,EAAE,MAAM,EAAE,MAAM,OAAO,CAAA;AAEtD,OAAO,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAA;AAG3D,MAAM,CAAC,MAAM,qBAAqB,GAI7B,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,KAAK,EAAE,EAAE,EAAE;IACpC,MAAM,SAAS,GAAG,MAAM,CAAiB,IAAI,CAAC,CAAA;IAE9C,eAAe,CAAC,GAAG,EAAE;QACnB,OAAO,CAAC,GAAG,CAAC,QAAQ,EAAE,WAAW,CAAC,SAAS,CAAC,MAAM,CAAC,CAAA;QACnD,IAAI,SAAS,CAAC,OAAO,EAAE;YACrB,SAAS,CAAC,OAAO,CAAC,SAAS,GAAG,KAAK,IAAI,EAAE,CAAA;YACzC,iBAAiB,CAAC,SAAS,CAAC,OAAO,EAAE,WAAW,CAAC,UAAU,EAAE,WAAW,CAAC,SAAS,CAAC,CAAA;SACpF;IACH,CAAC,CAAC,CAAA;IACF,OAAO,IAAI,KAAK,UAAU,CAAC,CAAC,CAAC,CAC3B,6BAAK,GAAG,EAAE,SAAS,GAAI,CACxB,CAAC,CAAC,CAAC,CACF,8BAAM,SAAS,EAAC,sCAAsC;QACpD,8BAAM,SAAS,EAAC,gBAAgB,EAAC,GAAG,EAAE,SAAS,IAC5C,KAAK,CACD,CACF,CACR,CAAA;AACH,CAAC,CAAA"}
@@ -0,0 +1,6 @@
1
+ import React from 'react';
2
+ import { ExamComponentProps } from '../../createRenderChildNodes';
3
+ declare function ChoiceAnswer({ element, renderChildNodes }: ExamComponentProps): JSX.Element;
4
+ declare const _default: React.MemoExoticComponent<typeof ChoiceAnswer>;
5
+ export default _default;
6
+ //# sourceMappingURL=ChoiceAnswer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ChoiceAnswer.d.ts","sourceRoot":"","sources":["../../../src/components/grading/ChoiceAnswer.tsx"],"names":[],"mappings":"AACA,OAAO,KAAqB,MAAM,OAAO,CAAA;AAEzC,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAA;AAuEjE,iBAAS,YAAY,CAAC,EAAE,OAAO,EAAE,gBAAgB,EAAE,EAAE,kBAAkB,eAyCtE;;AAED,wBAAuC"}
@@ -0,0 +1,57 @@
1
+ import classNames from 'classnames';
2
+ import React, { useContext } from 'react';
3
+ import { getNumericAttribute, mapChildElements, query } from '../../dom-utils';
4
+ import { useExamTranslation } from '../../i18n';
5
+ import { ScreenReaderOnly } from '../ScreenReaderOnly';
6
+ import { findMultiChoiceFromGradingStructure, ResultsContext } from '../context/ResultsContext';
7
+ import ResultsExamQuestionAutoScore from './internal/QuestionAutoScore';
8
+ function ChoiceAnswerOption({ renderChildNodes, selected, element, questionId, direction, isCorrect, }) {
9
+ const { t } = useExamTranslation();
10
+ const className = element.getAttribute('class');
11
+ const optionId = element.getAttribute('option-id');
12
+ const content = (React.createElement("div", { className: classNames('e-choice-answer-option e-column', className, {
13
+ 'e-choice-answer-option--selected': selected,
14
+ }) }, renderChildNodes(element)));
15
+ const Content = ({ narrow = false }) => (React.createElement(React.Fragment, null,
16
+ React.createElement("input", { type: "radio", className: classNames('e-radio-button', { 'e-column e-column--narrow': narrow }), name: String(questionId), value: optionId, checked: selected, readOnly: true }),
17
+ content,
18
+ isCorrect && React.createElement(ScreenReaderOnly, null, t('screen-reader.correct-answer'))));
19
+ return direction === 'vertical' ? (React.createElement("div", { className: "e-mrg-b-1" },
20
+ React.createElement("label", { className: classNames('e-columns', {
21
+ 'e-correct-answer-left': isCorrect,
22
+ 'e-columns--inline': query(element, ['image', 'video']) == null /* Force full width for options containing responsive media */,
23
+ }) },
24
+ React.createElement(Content, { narrow: true })))) : (React.createElement("div", { className: "e-column e-mrg-b-1" },
25
+ React.createElement("label", { className: classNames('e-block e-text-center', { 'e-correct-answer-bottom': isCorrect }) },
26
+ React.createElement(Content, null))));
27
+ }
28
+ function ChoiceAnswer({ element, renderChildNodes }) {
29
+ const { answersByQuestionId, gradingStructure } = useContext(ResultsContext);
30
+ const questionId = getNumericAttribute(element, 'question-id');
31
+ const answer = answersByQuestionId[questionId];
32
+ const direction = element.getAttribute('direction') || 'vertical';
33
+ const className = element.getAttribute('class');
34
+ const choice = findMultiChoiceFromGradingStructure(gradingStructure, questionId);
35
+ const scoreValue = answer && choice.options.find((option) => option.id === Number(answer.value)).score;
36
+ const maxScore = getNumericAttribute(element, 'max-score');
37
+ return (React.createElement(React.Fragment, null,
38
+ scoreValue !== undefined && React.createElement(ResultsExamQuestionAutoScore, { score: scoreValue, maxScore: maxScore }),
39
+ React.createElement("div", { className: classNames('e-choice-answer', className, {
40
+ 'e-columns': direction === 'horizontal',
41
+ }) }, mapChildElements(element, (childElement) => {
42
+ const optionId = getNumericAttribute(childElement, 'option-id');
43
+ const selected = answer != null && Number(answer.value) === optionId;
44
+ const grading = choice.options.find((option) => option.id === optionId);
45
+ return (React.createElement(ChoiceAnswerOption, { ...{
46
+ element: childElement,
47
+ renderChildNodes,
48
+ questionId,
49
+ key: optionId,
50
+ direction,
51
+ selected,
52
+ isCorrect: grading.correct,
53
+ } }));
54
+ }))));
55
+ }
56
+ export default React.memo(ChoiceAnswer);
57
+ //# sourceMappingURL=ChoiceAnswer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ChoiceAnswer.js","sourceRoot":"","sources":["../../../src/components/grading/ChoiceAnswer.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ import React from 'react';
2
+ import { ExamComponentProps } from '../../createRenderChildNodes';
3
+ declare function DropdownAnswer({ element, renderChildNodes }: ExamComponentProps): JSX.Element | null;
4
+ declare const _default: React.MemoExoticComponent<typeof DropdownAnswer>;
5
+ export default _default;
6
+ //# sourceMappingURL=DropdownAnswer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DropdownAnswer.d.ts","sourceRoot":"","sources":["../../../src/components/grading/DropdownAnswer.tsx"],"names":[],"mappings":"AACA,OAAO,KAAqB,MAAM,OAAO,CAAA;AAEzC,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAA;AASjE,iBAAS,cAAc,CAAC,EAAE,OAAO,EAAE,gBAAgB,EAAE,EAAE,kBAAkB,sBA2DxE;;AAED,wBAAyC"}
@@ -0,0 +1,43 @@
1
+ import classNames from 'classnames';
2
+ import React, { useContext } from 'react';
3
+ import { findChildElement, getNumericAttribute } from '../../dom-utils';
4
+ import { useExamTranslation } from '../../i18n';
5
+ import { shortDisplayNumber } from '../../shortDisplayNumber';
6
+ import { QuestionContext } from '../context/QuestionContext';
7
+ import { ScreenReaderOnly } from '../ScreenReaderOnly';
8
+ import { findMultiChoiceFromGradingStructure, ResultsContext } from '../context/ResultsContext';
9
+ import ResultsExamQuestionAutoScore from './internal/QuestionAutoScore';
10
+ function DropdownAnswer({ element, renderChildNodes }) {
11
+ const { t } = useExamTranslation();
12
+ const { answersByQuestionId, gradingStructure } = useContext(ResultsContext);
13
+ const { answers } = useContext(QuestionContext);
14
+ const questionId = getNumericAttribute(element, 'question-id');
15
+ const answer = answersByQuestionId[questionId];
16
+ const selectedOption = findChildElement(element, (childElement) => (answer === null || answer === void 0 ? void 0 : answer.value) === childElement.getAttribute('option-id'));
17
+ const choice = findMultiChoiceFromGradingStructure(gradingStructure, questionId);
18
+ if (selectedOption) {
19
+ const correctIds = choice.options.filter((o) => o.correct).map((o) => o.id);
20
+ const correctOptions = Array.from(element.children).filter((childElement) => correctIds.includes(getNumericAttribute(childElement, 'option-id')));
21
+ const isAnswerCorrect = correctIds.includes(getNumericAttribute(selectedOption, 'option-id'));
22
+ const displayNumber = shortDisplayNumber(element.getAttribute('display-number'));
23
+ const scoreValue = answer && choice.options.find((option) => option.id === Number(answer.value)).score;
24
+ const maxScore = getNumericAttribute(element, 'max-score');
25
+ return (React.createElement(React.Fragment, null,
26
+ answers.length > 1 && React.createElement("sup", null, displayNumber),
27
+ React.createElement("span", { className: classNames('e-dropdown-answer__answered', {
28
+ 'e-dropdown-answer__answered--correct': isAnswerCorrect,
29
+ 'e-dropdown-answer__answered--wrong': !isAnswerCorrect,
30
+ }) },
31
+ React.createElement(ScreenReaderOnly, null, t('screen-reader.answer-begin')),
32
+ renderChildNodes(selectedOption),
33
+ React.createElement(ScreenReaderOnly, null, t('screen-reader.answer-end')),
34
+ isAnswerCorrect && React.createElement(ScreenReaderOnly, null, t('screen-reader.correct-answer'))),
35
+ !isAnswerCorrect && (React.createElement("span", { className: "e-dropdown-answer__correct", "aria-hidden": true }, correctOptions.map((correctOption, i) => (React.createElement(React.Fragment, { key: i },
36
+ renderChildNodes(correctOption),
37
+ i < correctOptions.length - 1 && ', '))))),
38
+ scoreValue != null && (React.createElement(ResultsExamQuestionAutoScore, { score: scoreValue, maxScore: maxScore, displayNumber: displayNumber }))));
39
+ }
40
+ return null;
41
+ }
42
+ export default React.memo(DropdownAnswer);
43
+ //# sourceMappingURL=DropdownAnswer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DropdownAnswer.js","sourceRoot":"","sources":["../../../src/components/grading/DropdownAnswer.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import React from 'react';
2
+ declare const _default: React.MemoExoticComponent<React.ComponentType<any>>;
3
+ export default _default;
4
+ //# sourceMappingURL=Grading.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Grading.d.ts","sourceRoot":"","sources":["../../../src/components/grading/Grading.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAkD,MAAM,OAAO,CAAA;;AAmHtE,wBAA6E"}
@@ -0,0 +1,81 @@
1
+ import React, { useContext, useEffect, useRef, useState } from 'react';
2
+ import { I18nextProvider } from 'react-i18next';
3
+ import { changeLanguage, initI18n } from '../../i18n';
4
+ import { useCached } from '../../useCached';
5
+ import { withCommonExamContext } from '../context/CommonExamContext';
6
+ import { ResultsContext, withResultsContext } from '../context/ResultsContext';
7
+ import { GradingAnswer } from './GradingAnswer';
8
+ function Grading() {
9
+ const { answersByQuestionId } = useContext(ResultsContext);
10
+ const i18n = useCached(() => initI18n('FI-fi'));
11
+ useEffect(changeLanguage(i18n, 'FI-fi'));
12
+ const answerIds = Object.keys(answersByQuestionId).map(Number);
13
+ if (answerIds.length === 0) {
14
+ return React.createElement("div", null, "No answers");
15
+ }
16
+ const [answerId, setAnswerId] = useState(answerIds[0]);
17
+ const annotationsStorage = useRef(Object.fromEntries(answerIds.map((id) => [
18
+ id,
19
+ {
20
+ pregrading: [
21
+ {
22
+ startIndex: 2,
23
+ length: 5,
24
+ message: '+1',
25
+ },
26
+ {
27
+ x: 0.07434944237918215,
28
+ y: 0.8599562363238512,
29
+ type: 'rect',
30
+ width: 0.16604708798017348,
31
+ height: 0.03063457330415753,
32
+ message: '+2',
33
+ attachmentIndex: 0,
34
+ },
35
+ {
36
+ x1: 0.4035532994923858,
37
+ x2: 0.5621827411167513,
38
+ y1: 0.957983193277311,
39
+ y2: 0.957983193277311,
40
+ type: 'line',
41
+ message: 'trk, +1',
42
+ attachmentIndex: 0,
43
+ },
44
+ ],
45
+ censoring: [],
46
+ },
47
+ ])));
48
+ const [annotations, setAnnotations] = useState({
49
+ pregrading: [],
50
+ censoring: [],
51
+ });
52
+ useEffect(() => {
53
+ setAnnotations(annotationsStorage.current[answerId]);
54
+ });
55
+ const { type: answerType, value } = answersByQuestionId[answerId];
56
+ if (answerType === 'choice') {
57
+ return React.createElement("div", null, "choice answer");
58
+ }
59
+ function selectQuestion(e, id) {
60
+ e.preventDefault();
61
+ setAnswerId(id);
62
+ setAnnotations(annotationsStorage.current[id]);
63
+ }
64
+ function saveAnnotations(annotations) {
65
+ annotationsStorage.current[answerId] = annotations;
66
+ setAnnotations({ ...annotationsStorage.current[answerId] });
67
+ }
68
+ return (React.createElement(I18nextProvider, { i18n: i18n },
69
+ React.createElement("main", { className: "e-exam" },
70
+ React.createElement("div", { className: "grading-navi" }, answerIds.map((id) => id === answerId ? (React.createElement("span", { key: id, className: "grading-navi-item" }, answersByQuestionId[id].displayNumber)) : (React.createElement("a", { className: "grading-navi-item", href: "", onClick: (e) => selectQuestion(e, id), key: id }, answersByQuestionId[id].displayNumber)))),
71
+ React.createElement(GradingAnswer, { ...{
72
+ isReadOnly: false,
73
+ answerType,
74
+ gradingRole: 'censoring',
75
+ value,
76
+ annotations,
77
+ saveAnnotations,
78
+ } }))));
79
+ }
80
+ export default React.memo(withResultsContext(withCommonExamContext(Grading)));
81
+ //# sourceMappingURL=Grading.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Grading.js","sourceRoot":"","sources":["../../../src/components/grading/Grading.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,22 @@
1
+ /// <reference types="react" />
2
+ import { Annotation } from '../..';
3
+ type Annotations = {
4
+ pregrading: Annotation[];
5
+ censoring: Annotation[];
6
+ };
7
+ type GradingRole = 'pregrading' | 'censoring';
8
+ export declare function GradingAnswer({ answer: { type, characterCount, value }, language, isReadOnly, gradingRole, annotations, saveAnnotations, maxLength, }: {
9
+ answer: {
10
+ type: 'richText' | 'text';
11
+ characterCount: number;
12
+ value: string;
13
+ };
14
+ language: string;
15
+ isReadOnly: boolean;
16
+ gradingRole: GradingRole;
17
+ maxLength?: number;
18
+ annotations: Annotations;
19
+ saveAnnotations: (annotations: Annotations) => void;
20
+ }): JSX.Element;
21
+ export {};
22
+ //# sourceMappingURL=GradingAnswer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GradingAnswer.d.ts","sourceRoot":"","sources":["../../../src/components/grading/GradingAnswer.tsx"],"names":[],"mappings":";AACA,OAAO,EAAE,UAAU,EAAE,MAAM,OAAO,CAAA;AAyBlC,KAAK,WAAW,GAAG;IAAE,UAAU,EAAE,UAAU,EAAE,CAAC;IAAC,SAAS,EAAE,UAAU,EAAE,CAAA;CAAE,CAAA;AAExE,KAAK,WAAW,GAAG,YAAY,GAAG,WAAW,CAAA;AAC7C,wBAAgB,aAAa,CAAC,EAC5B,MAAM,EAAE,EAAE,IAAI,EAAE,cAAc,EAAE,KAAK,EAAE,EACvC,QAAQ,EACR,UAAU,EACV,WAAW,EACX,WAAW,EACX,eAAe,EACf,SAAS,GACV,EAAE;IACD,MAAM,EAAE;QAAE,IAAI,EAAE,UAAU,GAAG,MAAM,CAAC;QAAC,cAAc,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAA;IAC5E,QAAQ,EAAE,MAAM,CAAA;IAChB,UAAU,EAAE,OAAO,CAAA;IACnB,WAAW,EAAE,WAAW,CAAA;IACxB,SAAS,CAAC,EAAE,MAAM,CAAA;IAClB,WAAW,EAAE,WAAW,CAAA;IACxB,eAAe,EAAE,CAAC,WAAW,EAAE,WAAW,KAAK,IAAI,CAAA;CACpD,eAkSA"}
@@ -0,0 +1,242 @@
1
+ import React, { useEffect, useLayoutEffect, useRef } from 'react';
2
+ import { annotationFromMousePosition, getOverlappingMessages, hasTextSelectedInAnswerText, imageAnnotationMouseDownInfo, mergeAnnotation, selectionHasNothingToUnderline, showAndPositionElement, textAnnotationFromRange, } from './editAnnotations';
3
+ import { renderAnnotations, renderImageAnnotationByImage, updateImageAnnotationMarkSize, wrapAllImages, } from '../../renderAnnotations';
4
+ import GradingAnswerAnnotationList from './GradingAnswerAnnotationList';
5
+ import { changeLanguage, initI18n } from '../../i18n';
6
+ import { updateLargeImageWarnings } from './largeImageDetector';
7
+ import { I18nextProvider } from 'react-i18next';
8
+ import { useCached } from '../../useCached';
9
+ import { AnswerCharacterCounter } from './AnswerCharacterCounter';
10
+ export function GradingAnswer({ answer: { type, characterCount, value }, language, isReadOnly, gradingRole, annotations, saveAnnotations, maxLength, }) {
11
+ const answerRef = useRef(null);
12
+ const popupRef = useRef(null);
13
+ const messageRef = useRef(null);
14
+ const tooltipRef = useRef(null);
15
+ let newAnnotationObject;
16
+ let savedAnnotations;
17
+ const imgAnnotationState = { start: undefined, element: undefined, img: undefined };
18
+ let isEditAnnotationPopupVisible = false;
19
+ let annotationDataForTooltip;
20
+ let annotationPositionForPopup;
21
+ let hideTooltipTimeout;
22
+ let windowResizeTimeout;
23
+ function onAnnotationOrListClick(e) {
24
+ var _a;
25
+ const element = e.target;
26
+ if (element instanceof HTMLElement) {
27
+ if (element.tagName === 'MARK') {
28
+ showTooltip(element);
29
+ }
30
+ else if (element.tagName === 'LI') {
31
+ const index = (_a = element.dataset.listNumber) === null || _a === void 0 ? void 0 : _a.replace(')', '');
32
+ const mark = document.querySelector(`.e-annotation[data-index="${index}"]`);
33
+ showTooltip(mark);
34
+ mark.scrollIntoView({ block: 'center', behavior: 'smooth' });
35
+ }
36
+ else if (annotationDataForTooltip) {
37
+ hideTooltip();
38
+ }
39
+ }
40
+ }
41
+ function editExistingAnnotation(e) {
42
+ if (e.target.closest('.editable')) {
43
+ toggle(tooltipRef.current, false);
44
+ showExistingAnnotationPopup(annotationPositionForPopup);
45
+ }
46
+ }
47
+ useLayoutEffect(() => {
48
+ if (answerRef.current) {
49
+ savedAnnotations = annotations;
50
+ toggle(tooltipRef.current, false);
51
+ toggle(popupRef.current, false);
52
+ renderAnswerWithAnnotations(savedAnnotations);
53
+ }
54
+ window.onresize = () => {
55
+ clearTimeout(windowResizeTimeout);
56
+ windowResizeTimeout = setTimeout(() => updateLargeImageWarnings(answerRef.current), 1000);
57
+ };
58
+ });
59
+ const i18n = useCached(() => initI18n(language));
60
+ useEffect(changeLanguage(i18n, language));
61
+ return (React.createElement(I18nextProvider, { i18n: i18n },
62
+ React.createElement("div", { onClick: (e) => onAnnotationOrListClick(e), className: "e-grading-answer-wrapper" },
63
+ React.createElement("div", { className: "e-grading-answer e-line-height-l e-mrg-b-1", ref: answerRef, onMouseDown: (e) => onAnswerMouseDown(e), onMouseOver: (e) => onMouseOverAnnotation(e.target) }),
64
+ React.createElement(AnswerCharacterCounter, { characterCount: characterCount, maxLength: maxLength }),
65
+ React.createElement(GradingAnswerAnnotationList, { censoring: annotations.censoring, pregrading: annotations.pregrading, singleGrading: false }),
66
+ React.createElement("form", { style: { display: 'none' }, ref: popupRef, className: "e-grading-answer-popup e-grading-answer-add-annotation", onSubmit: (e) => onSubmitAnnotation(e) },
67
+ React.createElement("input", { name: "message", className: "e-grading-answer-add-annotation-text", type: "text", ref: messageRef }),
68
+ React.createElement("button", { className: "e-grading-answer-add-annotation-button", type: "submit", "data-i18n": "arpa.annotate" }, "Merkitse"),
69
+ React.createElement("button", { className: "e-grading-answer-close-popup", onClick: (e) => {
70
+ e.preventDefault();
71
+ hideAnnotationPopupAndRefresh();
72
+ } }, "\u00D7")),
73
+ React.createElement("div", { style: { display: 'none' }, ref: tooltipRef, className: "e-grading-answer-tooltip e-grading-answer-popup", onMouseOver: onMouseOverTooltip, onMouseOut: hideTooltip },
74
+ React.createElement("span", { onClick: (e) => editExistingAnnotation(e), className: "e-grading-answer-tooltip-label" }, "tooltip text"),
75
+ React.createElement("button", { onClick: (e) => removeAnnotation(e), className: "e-grading-answer-tooltip-remove" }, "\u00D7")))));
76
+ function removeAnnotation(e) {
77
+ e.preventDefault();
78
+ hideTooltip();
79
+ savedAnnotations[annotationDataForTooltip.role].splice(annotationDataForTooltip.index, 1);
80
+ annotationDataForTooltip = undefined;
81
+ saveAnnotations(savedAnnotations);
82
+ }
83
+ function hideTooltip() {
84
+ hideTooltipTimeout = setTimeout(() => {
85
+ toggle(tooltipRef.current, false);
86
+ }, 400);
87
+ }
88
+ function onMouseOverTooltip() {
89
+ clearTimeout(hideTooltipTimeout);
90
+ }
91
+ function onMouseOutFromTooltip(e) {
92
+ const target = e.target;
93
+ if (target.tagName === 'MARK') {
94
+ clearTimeout(hideTooltipTimeout);
95
+ hideTooltip();
96
+ }
97
+ answerRef.current.removeEventListener('mouseout', onMouseOutFromTooltip);
98
+ }
99
+ function showTooltip(target) {
100
+ clearTimeout(hideTooltipTimeout);
101
+ const tooltip = tooltipRef.current;
102
+ const { type, listIndex, message } = target.dataset;
103
+ tooltip.classList.toggle('editable', !isReadOnly && type === gradingRole);
104
+ annotationPositionForPopup = target.getBoundingClientRect();
105
+ showAndPositionElement(annotationPositionForPopup, answerRef.current, tooltip);
106
+ tooltip.querySelector('.e-grading-answer-tooltip-label').textContent = message || '–';
107
+ annotationDataForTooltip = { index: Number(listIndex), role: type, message: message || '' };
108
+ answerRef.current.addEventListener('mouseout', onMouseOutFromTooltip);
109
+ }
110
+ function onMouseOverAnnotation(target) {
111
+ if (target.tagName === 'MARK' &&
112
+ !isEditAnnotationPopupVisible &&
113
+ (!hasTextSelectedInAnswerText() || isReadOnly) &&
114
+ !imgAnnotationState.element) {
115
+ showTooltip(target);
116
+ }
117
+ }
118
+ function showAnnotationPopup(rect, message) {
119
+ annotationDataForTooltip = undefined;
120
+ setupAnnotationPopup(rect, message);
121
+ }
122
+ function showExistingAnnotationPopup(rect) {
123
+ setupAnnotationPopup(rect, annotationDataForTooltip.message);
124
+ }
125
+ function setupAnnotationPopup(rect, message) {
126
+ // Could be active from previous popup
127
+ window.removeEventListener('keydown', onKeyUpInAnnotationPopup);
128
+ showAndPositionElement(rect, answerRef.current, popupRef.current);
129
+ const inputElement = messageRef.current;
130
+ inputElement.value = message;
131
+ inputElement.focus();
132
+ isEditAnnotationPopupVisible = true;
133
+ window.addEventListener('keydown', onKeyUpInAnnotationPopup);
134
+ }
135
+ function hideAnnotationPopupAndRefresh() {
136
+ newAnnotationObject = undefined;
137
+ imgAnnotationState.element = undefined;
138
+ imgAnnotationState.img = undefined;
139
+ isEditAnnotationPopupVisible = false;
140
+ window.removeEventListener('keydown', onKeyUpInAnnotationPopup);
141
+ toggle(popupRef.current, false);
142
+ renderAnswerWithAnnotations(savedAnnotations);
143
+ }
144
+ function onAnswerMouseDown(e) {
145
+ var _a;
146
+ if (isReadOnly) {
147
+ return;
148
+ }
149
+ window.addEventListener('mouseup', onWindowMouseUpAfterAnswerMouseDown);
150
+ // Do annotations only with left mouse buttons
151
+ if (e.button !== 0) {
152
+ return;
153
+ }
154
+ const target = e.target;
155
+ const img = ((_a = target.closest('.e-annotation-wrapper')) === null || _a === void 0 ? void 0 : _a.querySelector('img')) || undefined;
156
+ if (!img) {
157
+ return;
158
+ }
159
+ img.addEventListener('dragstart', preventDefaults); // Prevent dragging images when marking image annotations
160
+ imgAnnotationState.start = imageAnnotationMouseDownInfo(e, img);
161
+ imgAnnotationState.img = img;
162
+ window.addEventListener('mousemove', onMouseMoveForImageAnnotation);
163
+ }
164
+ function onWindowMouseUpAfterAnswerMouseDown() {
165
+ var _a, _b;
166
+ (_a = imgAnnotationState.img) === null || _a === void 0 ? void 0 : _a.removeEventListener('dragstart', preventDefaults);
167
+ window.removeEventListener('mousemove', onMouseMoveForImageAnnotation);
168
+ window.removeEventListener('mouseup', onWindowMouseUpAfterAnswerMouseDown);
169
+ // Image annotation is being created since shape exists
170
+ if (imgAnnotationState.element) {
171
+ showAnnotationPopup((_b = imgAnnotationState.element) === null || _b === void 0 ? void 0 : _b.getBoundingClientRect(), '');
172
+ return;
173
+ }
174
+ // Text annotation
175
+ const selection = window.getSelection();
176
+ if (selection && answerRef.current !== null && hasTextSelectedInAnswerText()) {
177
+ const range = selection.getRangeAt(0);
178
+ if (selectionHasNothingToUnderline(range)) {
179
+ return;
180
+ }
181
+ const position = textAnnotationFromRange(answerRef.current, range);
182
+ const message = getOverlappingMessages(savedAnnotations[gradingRole], position.startIndex, position.length);
183
+ newAnnotationObject = { ...position, type: 'text', message };
184
+ showAnnotationPopup(range.getBoundingClientRect(), message);
185
+ const newAnnotations = { ...savedAnnotations };
186
+ newAnnotations[gradingRole] = mergeAnnotation(answerRef.current, newAnnotationObject, savedAnnotations[gradingRole]);
187
+ renderAnswerWithAnnotations(newAnnotations);
188
+ }
189
+ }
190
+ function onMouseMoveForImageAnnotation(e) {
191
+ preventDefaults(e);
192
+ newAnnotationObject = annotationFromMousePosition(e, imgAnnotationState.start);
193
+ // Create shape on first mouse move and resize after that
194
+ if (imgAnnotationState.element) {
195
+ updateImageAnnotationMarkSize(imgAnnotationState.element, newAnnotationObject);
196
+ }
197
+ else {
198
+ imgAnnotationState.element = renderImageAnnotationByImage(imgAnnotationState.img, '', newAnnotationObject, gradingRole, 999);
199
+ }
200
+ }
201
+ function renderAnswerWithAnnotations(annotations) {
202
+ const container = answerRef.current;
203
+ if (type === 'richText') {
204
+ container.innerHTML = value;
205
+ wrapAllImages(container);
206
+ updateLargeImageWarnings(container);
207
+ }
208
+ else {
209
+ container.textContent = value;
210
+ }
211
+ renderAnnotations(container, annotations.pregrading, annotations.censoring, false);
212
+ }
213
+ function onSubmitAnnotation(e) {
214
+ e.preventDefault();
215
+ const message = messageRef.current.value;
216
+ if (annotationDataForTooltip) {
217
+ // Editing existing annotation message by clicking tooltip
218
+ savedAnnotations[annotationDataForTooltip.role][annotationDataForTooltip.index].message = message;
219
+ }
220
+ else {
221
+ // Saving new annotation
222
+ savedAnnotations[gradingRole] = mergeAnnotation(answerRef.current, { ...newAnnotationObject, message }, savedAnnotations[gradingRole] || []);
223
+ }
224
+ toggle(popupRef.current, false);
225
+ saveAnnotations(savedAnnotations);
226
+ }
227
+ function onKeyUpInAnnotationPopup(e) {
228
+ if (e.key === 'Escape' && isEditAnnotationPopupVisible) {
229
+ hideAnnotationPopupAndRefresh();
230
+ }
231
+ }
232
+ }
233
+ function preventDefaults(e) {
234
+ e.preventDefault();
235
+ e.stopPropagation();
236
+ }
237
+ function toggle(element, isVisible) {
238
+ if (element instanceof HTMLElement) {
239
+ element.style.display = isVisible ? 'initial' : 'none';
240
+ }
241
+ }
242
+ //# sourceMappingURL=GradingAnswer.js.map