diff --git a/src/components/FileAttachmentPanel/FileAttachmentPanel.stories.js b/src/components/FileAttachmentPanel/FileAttachmentPanel.stories.js
new file mode 100644
index 0000000000..ef08c4250a
--- /dev/null
+++ b/src/components/FileAttachmentPanel/FileAttachmentPanel.stories.js
@@ -0,0 +1,74 @@
+import React from 'react';
+import FileAttachmentPanelComponent from './FileAttachmentPanel';
+import { Provider } from 'react-redux';
+import { configureStore } from '@reduxjs/toolkit';
+import Panel from 'components/Panel';
+
+export default {
+ title: 'ModularComponents/FileAttachmentPanel',
+ component: FileAttachmentPanelComponent
+};
+
+const initialState = {
+ viewer: {
+ openElements: {
+ panel: true,
+ },
+ disabledElements: {},
+ customElementOverrides: {},
+ tab: {},
+ panelWidths: { panel: 300 },
+ modularHeaders: {},
+ }
+};
+
+export function FileAttachmentPanelLeftEmpty() {
+ return (
+
initialState })}>
+
+
+
+
+ );
+}
+
+export function FileAttachmentPanelRightEmpty() {
+ return (
+
initialState })}>
+
+
+
+
+ );
+}
+
+const filesMock = {
+ embeddedFiles: [
+ { filename: '1.png' },
+ { filename: '15pages.pdf' },
+ ],
+ fileAttachmentAnnotations: [],
+};
+filesMock.fileAttachmentAnnotations[1] = [{ PageNumber: 1, filename: '2.png' }];
+filesMock.fileAttachmentAnnotations[5] = [{ PageNumber: 5, filename: 'signature.png' }];
+filesMock.fileAttachmentAnnotations[8] = [{ PageNumber: 8, filename: 'q.jpeg' }];
+
+export function FileAttachmentPanelLeftWithFiles() {
+ return (
+
initialState })}>
+
+
+
+
+ );
+}
+
+export function FileAttachmentPanelRightWithFiles() {
+ return (
+
initialState })}>
+
+
+
+
+ );
+}
diff --git a/src/components/FilePicker/FilePicker.js b/src/components/FilePicker/FilePicker.js
new file mode 100644
index 0000000000..1dad49a155
--- /dev/null
+++ b/src/components/FilePicker/FilePicker.js
@@ -0,0 +1,125 @@
+import React, { useState, useRef } from 'react';
+import { useTranslation } from 'react-i18next';
+import Icon from 'components/Icon';
+import classNames from 'classnames';
+import { isMobile } from 'helpers/device';
+
+import './FilePicker.scss';
+
+const FilePicker = ({
+ onChange = () => { },
+ onDrop = () => { },
+ shouldShowIcon = false,
+ acceptFormats,
+ allowMultiple = false,
+ errorMessage = ''
+}) => {
+ const [t] = useTranslation();
+ const [isDragging, setIsDragging] = useState(false);
+ const fileInputRef = useRef(null);
+
+ const onClick = () => {
+ fileInputRef?.current?.click();
+ };
+
+ const onKeyDown = (event) => {
+ if (event.key === 'Enter') {
+ onClick();
+ }
+ };
+
+ const handleChange = (e) => {
+ const files = e.target.files;
+ files.length > 0 && onChange(Array.from(files));
+ };
+
+ const handleDragEnter = (e) => {
+ e.preventDefault();
+ setIsDragging(true);
+ };
+
+ const handleDragOver = (e) => {
+ e.preventDefault();
+ };
+
+ const handleDragLeave = (e) => {
+ e.preventDefault();
+
+ if (!e.target.parentNode.contains(e.relatedTarget)) {
+ setIsDragging(false);
+ }
+ };
+
+ const handleDragExit = (e) => {
+ e.preventDefault();
+ setIsDragging(false);
+ };
+
+ const handleFileDrop = async (e) => {
+ e.preventDefault();
+ setIsDragging(false);
+ const { files } = e.dataTransfer;
+ files.length > 0 && onDrop(Array.from(files));
+ };
+
+ const renderPrompt = () => {
+ if (isMobile()) {
+ return (
+
+ {t('filePicker.selectFile')}
+
+ );
+ }
+ return (
+ <>
+
+ {t('filePicker.dragAndDrop')}
+
+
+ {t('filePicker.or')}
+
+ >
+ );
+ };
+
+ return (
+
+
+
+ {shouldShowIcon &&
}
+ {renderPrompt()}
+
{t('action.browse')}
+ {
+ handleChange(event);
+ event.target.value = null;
+ }}
+ />
+
+
+ {errorMessage && (
+
{errorMessage}
+ )}
+
+
+ );
+};
+
+export default FilePicker;
diff --git a/src/components/FilePicker/FilePicker.scss b/src/components/FilePicker/FilePicker.scss
new file mode 100644
index 0000000000..c0bb05c3c7
--- /dev/null
+++ b/src/components/FilePicker/FilePicker.scss
@@ -0,0 +1,74 @@
+@import '../../constants/styles';
+@import '../../constants/modal';
+
+.file-picker-component {
+ width: 100%;
+ height: 100%;
+ position: relative;
+ border-radius: 4px;
+
+ .file-picker-container {
+ position: relative;
+ border: 1px dashed var(--modal-stroke-and-border);
+ display: flex;
+ flex-direction: column;
+ justify-content: center;
+ align-items: center;
+ width: 100%;
+ height: 100%;
+
+ &.dragging {
+ background: var(--file-picker-drop-background);
+ border: 1px dashed var(--file-picker-drop-border);
+ }
+
+ .file-picker-body {
+ display: flex;
+ flex-direction: column;
+ justify-content: center;
+ align-items: center;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+
+ .label-separator {
+ margin: 10px;
+ }
+
+ color: var(--faded-text);
+
+ .modal-btn-file {
+ border-radius: 4px;
+ border: 1px solid var(--primary-button);
+ color: var(--primary-button);
+ padding-top: 2px;
+ padding-right: 20px;
+ padding-bottom: 4px;
+ padding-left: 20px;
+ cursor: pointer;
+ }
+
+ .Icon {
+ width: fit-content;
+ height: fit-content;
+ margin-bottom: 15px;
+
+ svg {
+ height: 45px;
+ }
+ }
+
+ .file-picker-separator {
+ margin: 10px;
+ }
+ }
+
+ .file-picker-error {
+ position: absolute;
+ color: red;
+ bottom: 0px;
+ right: 0px;
+ margin: 0px 5px 5px 0px;
+ }
+ }
+}
diff --git a/src/components/FilePicker/index.js b/src/components/FilePicker/index.js
new file mode 100644
index 0000000000..e6a188ce94
--- /dev/null
+++ b/src/components/FilePicker/index.js
@@ -0,0 +1,3 @@
+import FilePicker from './FilePicker';
+
+export default FilePicker;
diff --git a/src/components/FilterAnnotModal/FilterAnnotModal.spec.js b/src/components/FilterAnnotModal/FilterAnnotModal.spec.js
new file mode 100644
index 0000000000..718fde7f65
--- /dev/null
+++ b/src/components/FilterAnnotModal/FilterAnnotModal.spec.js
@@ -0,0 +1,33 @@
+import React from 'react';
+import { render } from '@testing-library/react';
+import { UserPanel, ColorPanel, TypePanel, DocumentFilterActive } from './FilterAnnotModal.stories';
+
+const noop = () => { };
+
+jest.mock('core', () => ({
+ getDisplayAuthor: () => 'Test Author',
+ addEventListener: noop,
+ removeEventListener: noop,
+ getAnnotationsList: () => [],
+ getDocumentViewers: () => [{
+ getAnnotationManager: () => ({
+ getAnnotationsList: () => []
+ })
+ }],
+}));
+
+const UserPanelFilterAnnotModalStory = withI18n(UserPanel);
+const ColorPanelFilterAnnotModalStory = withI18n(ColorPanel);
+const TypePanelFilterAnnotModalStory = withI18n(TypePanel);
+const DocumentFilterActiveStory = withI18n(DocumentFilterActive);
+
+describe('FilterAnnotModal', () => {
+ it('Stories should not throw any errors', () => {
+ expect(() => {
+ render(
);
+ render(
);
+ render(
);
+ render(
);
+ }).not.toThrow();
+ });
+});
\ No newline at end of file
diff --git a/src/components/FontSizeDropdown/FontSizeDropdown.scss b/src/components/FontSizeDropdown/FontSizeDropdown.scss
index 96c8159abe..e42ad93c0c 100644
--- a/src/components/FontSizeDropdown/FontSizeDropdown.scss
+++ b/src/components/FontSizeDropdown/FontSizeDropdown.scss
@@ -21,11 +21,15 @@
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
-
&.error {
border-color: var(--error-border-color);
}
}
+
+ .disabledText {
+ color: var(--disabled-text);
+ }
+
// To remove arrows from input
/* Chrome, Safari, Edge, Opera */
input::-webkit-outer-spin-button,
diff --git a/src/components/FontSizeDropdown/pdfEditHelper.js b/src/components/FontSizeDropdown/pdfEditHelper.js
new file mode 100644
index 0000000000..585a982193
--- /dev/null
+++ b/src/components/FontSizeDropdown/pdfEditHelper.js
@@ -0,0 +1,111 @@
+const SELECTION_BACKGROUND = '#50A5F5FE';
+let range;
+let docViewer;
+let inputElement;
+
+/**
+ * @ignore
+ * Helper function to keep the highlight of the selected text in the text edit box before the elemnt focus is changed.
+ */
+export function keepTextEditSelectionOnInputFocus(core) {
+ inputElement = document.activeElement;
+ docViewer = core.getDocumentViewer();
+ // When the input is still in focus but we changed page, we need to un-focus the input.
+ docViewer.addEventListener('pageNumberUpdated', handlePageChange, { once: true });
+ // When we click anywhere other than the input field itself, it should unfocus.
+ document.addEventListener('mousedown', handleClick);
+
+ const currentRange = window.getSelection().getRangeAt(0);
+ const isFocusOutsideTextBox = currentRange.startContainer.nodeName === 'DIV';
+ // Component re-renders when we focus into the input field because it is a dropdown/input combo.
+ // In that case the focus is already shifted out from text edit box and this function is executed again
+ // due to re-render. The selection will no longer include the text nodes we initially selected
+ // in the text edit box.
+ if (isFocusOutsideTextBox) {
+ return;
+ }
+
+ // When we have nothing selected, simply return
+ const isEmptySelection = currentRange.startContainer === currentRange.endContainer && currentRange.startOffset === currentRange.endOffset;
+ if (isEmptySelection) {
+ return;
+ }
+
+ // When the color / font style of the selected text was changed at least once before changing
+ // the font size with input field, the selection we get from `getSelection` would have been modified
+ // by worker API. In this case, we need to reinitialize the range so that the range object returns to its initial state.
+ const isRangeModifiedByWorkerAPI = currentRange.startContainer.nodeName === 'SPAN';
+ if (isRangeModifiedByWorkerAPI) {
+ range = reinitializeRange(currentRange);
+ } else {
+ range = currentRange;
+ }
+
+ toggleSelectionHighlight(range.startContainer, range.endContainer);
+}
+
+/**
+ * @ignore
+ * Helper function to restore the text edit box selection
+ */
+export function restoreSelection() {
+ if (!range) {
+ return;
+ }
+ const selection = window.getSelection();
+ selection.removeAllRanges();
+ selection.addRange(range);
+ range = null;
+ docViewer.removeEventListener('pageNumberUpdated', handlePageChange);
+ document.removeEventListener('mousedown', handleClick);
+}
+
+/**
+ * @ignore
+ * Toggles the background color of the selected elements recursively.
+ * @param {Text} startTextNode
+ * @param {Text} endTextNode
+ */
+function toggleSelectionHighlight(startTextNode, endTextNode) {
+ const startElement = startTextNode.parentElement;
+ const selectionEnd = !startElement.nextElementSibling && startElement.parentElement.nextElementSibling === endTextNode;
+ if (selectionEnd) {
+ return;
+ }
+ startElement.style.background = SELECTION_BACKGROUND;
+ const highlighNextCharacterSameLine = startElement.nextElementSibling?.tagName === 'SPAN' && startTextNode !== endTextNode;
+ const highlighNextCharacterNextLine = startElement.nextElementSibling === null && startElement.parentElement.nextElementSibling;
+ if (highlighNextCharacterSameLine) {
+ // When the next character in selection is in the same line of current character, we simply pass that in
+ toggleSelectionHighlight(startElement.nextElementSibling.firstChild, endTextNode);
+ } else if (highlighNextCharacterNextLine) {
+ // when the next character in selection is in the next line, we need to go into the next line and continue our recursion
+ toggleSelectionHighlight(startElement.parentElement.nextElementSibling.firstElementChild.firstChild, endTextNode);
+ }
+}
+
+function reinitializeRange(workerAPIEditedRange) {
+ const selection = window.getSelection();
+ selection.removeAllRanges();
+ const startNode = workerAPIEditedRange.startContainer.firstChild;
+ const endNode = workerAPIEditedRange.endContainer.previousSibling.firstChild;
+ workerAPIEditedRange.setStart(startNode, 0);
+ workerAPIEditedRange.setEnd(endNode, 1);
+ return workerAPIEditedRange;
+}
+
+function handlePageChange() {
+ if (document.activeElement?.tagName === 'INPUT') {
+ document.activeElement.blur();
+ }
+ docViewer.removeEventListener('pageNumberUpdated', handlePageChange);
+ document.removeEventListener('mousedown', handleClick);
+}
+
+function handleClick(e) {
+ const isClickingFontSizeInput = e.target === inputElement;
+ if (!isClickingFontSizeInput) {
+ document.activeElement.blur();
+ document.removeEventListener('mousedown', handleClick);
+ }
+}
\ No newline at end of file
diff --git a/src/components/FormFieldEditPopup/FormFieldEditPopupIndicator/FormFieldEditPopupIndicator.js b/src/components/FormFieldEditPopup/FormFieldEditPopupIndicator/FormFieldEditPopupIndicator.js
new file mode 100644
index 0000000000..453917ec9c
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldEditPopupIndicator/FormFieldEditPopupIndicator.js
@@ -0,0 +1,37 @@
+import React from 'react';
+import { Choice, Input } from '@pdftron/webviewer-react-toolkit';
+import { useTranslation } from 'react-i18next';
+
+
+const FormFieldEditPopupIndicator = ({ indicator, indicatorPlaceholder }) => {
+ const { t } = useTranslation();
+ const onIndicatorChange = (showIndicator) => {
+ if (indicator.value.length < 1 && showIndicator) {
+ indicator.onChange(indicatorPlaceholder);
+ }
+ indicator.toggleIndicator(showIndicator);
+ };
+
+ return (
+
+
{t('formField.formFieldPopup.fieldIndicator')}
+
onIndicatorChange(event.target.checked)}
+ label={t(indicator.label)}
+ />
+
+ indicator.onChange(event.target.value)}
+ value={indicator.value}
+ fillWidth="false"
+ placeholder={indicatorPlaceholder}
+ disabled={!indicator.isChecked}
+ />
+
+
+ );
+};
+
+export default FormFieldEditPopupIndicator;
\ No newline at end of file
diff --git a/src/components/FormFieldEditPopup/FormFieldEditPopupIndicator/index.js b/src/components/FormFieldEditPopup/FormFieldEditPopupIndicator/index.js
new file mode 100644
index 0000000000..da4b611b59
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldEditPopupIndicator/index.js
@@ -0,0 +1,3 @@
+import FormFieldEditPopupIndicator from './FormFieldEditPopupIndicator';
+
+export default FormFieldEditPopupIndicator;
\ No newline at end of file
diff --git a/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/FormFieldEditSignaturePopup.stories.js b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/FormFieldEditSignaturePopup.stories.js
new file mode 100644
index 0000000000..b09af8cdab
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/FormFieldEditSignaturePopup.stories.js
@@ -0,0 +1,75 @@
+import React from 'react';
+import FormFieldEditSignaturePopup from './FormFieldEditSignaturePopup';
+import { configureStore } from '@reduxjs/toolkit';
+
+
+import { Provider } from 'react-redux';
+
+
+export default {
+ title: 'Components/FormFieldEditPopup',
+ component: FormFieldEditSignaturePopup,
+};
+
+const initialState = {
+ viewer: {
+ disabledElements: {},
+ customElementOverrides: {},
+ }
+};
+
+const store = configureStore({ reducer: () => initialState });
+
+const annotation = {
+ Width: 100,
+ Height: 100,
+};
+
+export function SignatureFieldPopup() {
+ const signatureFields = [
+ {
+ label: 'formField.formFieldPopup.fieldName',
+ onChange: () => { },
+ value: 'SignatureField1',
+ required: true,
+ type: 'text',
+ },
+ ];
+
+ const signatureFlags = [
+ {
+ label: 'formField.formFieldPopup.readOnly',
+ onChange: () => { },
+ isChecked: true,
+ },
+ {
+ label: 'formField.formFieldPopup.required',
+ onChange: () => { },
+ isChecked: true,
+ }
+ ];
+
+ const indicator = {
+ label: 'formField.formFieldPopup.documentFieldIndicator',
+ toggleIndicator: () => { },
+ isChecked: true,
+ onChange: () => { },
+ value: 'This is an indicator'
+ };
+
+ const props = {
+ fields: signatureFields,
+ flags: signatureFlags,
+ annotation,
+ isValid: true,
+ getSignatureOptionHandler: () => 'initialsSignature',
+ indicator,
+ };
+ return (
+
+
+
+
+
+ );
+}
diff --git a/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/SignatureOptionsDropdown.js b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/SignatureOptionsDropdown.js
new file mode 100644
index 0000000000..dd643496b8
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/SignatureOptionsDropdown.js
@@ -0,0 +1,82 @@
+import React, { useState } from 'react';
+import Select from 'react-select';
+import { useTranslation } from 'react-i18next';
+import DataElementWrapper from 'components/DataElementWrapper';
+import SignatureModes from 'constants/signatureModes';
+import ReactSelectCustomArrowIndicator from 'components/ReactSelectCustomArrowIndicator';
+import ReactSelectWebComponentProvider from 'src/components/ReactSelectWebComponentProvider';
+
+import './SignatureOptionsDropdown.scss';
+
+const getStyles = () => ({
+ control: (provided) => ({
+ ...provided,
+ minHeight: '28px',
+ backgroundColor: 'var(--component-background)',
+ borderColor: 'hsl(0, 0%, 80%)',
+ boxShadow: null,
+ '&:hover': null,
+ }),
+ valueContainer: (provided) => ({
+ ...provided,
+ padding: '2px',
+ }),
+ singleValue: (provided) => ({
+ ...provided,
+ color: 'var(--text-color)',
+ }),
+ menu: (provided) => ({
+ ...provided,
+ backgroundColor: 'var(--component-background)',
+ }),
+ option: (provided) => ({
+ ...provided,
+ backgroundColor: 'var(--component-background)',
+ color: 'var(--text-color)',
+ '&:hover': {
+ backgroundColor: 'var(--popup-button-hover)',
+ }
+ }),
+ indicatorsContainer: (provided) => ({
+ ...provided,
+ paddingRight: '6px',
+ height: '26px',
+ }),
+});
+
+const SignatureOptionsDropdown = (props) => {
+ const { onChangeHandler, initialOption } = props;
+ const { t } = useTranslation();
+ const styles = getStyles();
+ const signatureOptions = [
+ { value: SignatureModes.FULL_SIGNATURE, label: t('formField.types.signature') },
+ { value: SignatureModes.INITIALS, label: t('option.type.initials') },
+ ];
+
+ const init = signatureOptions.find((option) => option.value === initialOption);
+ const [value, setValue] = useState(init);
+
+ const onChange = (option) => {
+ setValue(option);
+ onChangeHandler(option);
+ };
+
+ return (
+
+
+
+
+
+
+ );
+};
+
+export default SignatureOptionsDropdown;
\ No newline at end of file
diff --git a/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/SignatureOptionsDropdown.scss b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/SignatureOptionsDropdown.scss
new file mode 100644
index 0000000000..07d5aab5ef
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/SignatureOptionsDropdown.scss
@@ -0,0 +1,17 @@
+.signature-options-container {
+ padding: 5px 0px 5px 0px;
+ display: grid;
+ grid-template-columns: 0.5fr 1fr;
+ grid-template-areas: "label dropdown";
+ align-items: center;
+
+ label {
+ grid-area: label;
+ }
+}
+
+.arrow {
+ width: 12px;
+ height: 16px;
+ margin-top: 2px;
+}
\ No newline at end of file
diff --git a/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/index.js b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/index.js
new file mode 100644
index 0000000000..6d7db923f7
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/SignatureOptionsDropdown/index.js
@@ -0,0 +1,3 @@
+import SignatureOptionsDropdown from './SignatureOptionsDropdown';
+
+export default SignatureOptionsDropdown;
\ No newline at end of file
diff --git a/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/index.js b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/index.js
new file mode 100644
index 0000000000..7ef2728376
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldEditSignaturePopup/index.js
@@ -0,0 +1,3 @@
+import FormFieldEditSignaturePopup from './FormFieldEditSignaturePopup';
+
+export default FormFieldEditSignaturePopup;
\ No newline at end of file
diff --git a/src/components/FormFieldEditPopup/FormFieldPopupDimensionsInput/FormFieldPopupDimensionsInput.js b/src/components/FormFieldEditPopup/FormFieldPopupDimensionsInput/FormFieldPopupDimensionsInput.js
new file mode 100644
index 0000000000..a98434ef19
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldPopupDimensionsInput/FormFieldPopupDimensionsInput.js
@@ -0,0 +1,32 @@
+import React from 'react';
+import { useTranslation } from 'react-i18next';
+
+const FormFieldPopupDimensionsInput = ({ width, height, onWidthChange, onHeightChange }) => {
+ const { t } = useTranslation();
+
+ return (
+
+
{t('formField.formFieldPopup.size')}:
+
+ onWidthChange(e.target.value)}
+ /> {t('formField.formFieldPopup.width')}
+
+
+ onHeightChange(e.target.value)}
+ /> {t('formField.formFieldPopup.height')}
+
+
+ );
+};
+
+export default FormFieldPopupDimensionsInput;
\ No newline at end of file
diff --git a/src/components/FormFieldEditPopup/FormFieldPopupDimensionsInput/index.js b/src/components/FormFieldEditPopup/FormFieldPopupDimensionsInput/index.js
new file mode 100644
index 0000000000..928474c06d
--- /dev/null
+++ b/src/components/FormFieldEditPopup/FormFieldPopupDimensionsInput/index.js
@@ -0,0 +1,3 @@
+import FormFieldPopupDimensionsInput from './FormFieldPopupDimensionsInput';
+
+export default FormFieldPopupDimensionsInput;
\ No newline at end of file
diff --git a/src/components/FormFieldIndicator/FormFieldIndicator.scss b/src/components/FormFieldIndicator/FormFieldIndicator.scss
new file mode 100644
index 0000000000..64ef8fad85
--- /dev/null
+++ b/src/components/FormFieldIndicator/FormFieldIndicator.scss
@@ -0,0 +1,60 @@
+@import '../../constants/modal';
+@import '../../constants/popup';
+
+@media print {
+ #form-field-indicator-wrapper {
+ opacity: 0;
+ }
+}
+
+#form-field-indicator-wrapper {
+ position: relative;
+ z-index: min($modal-z-index, $popup-z-index) - 10;
+}
+
+.formFieldIndicator {
+ background-color: var(--color-blue-gray-5);
+ color: var(--color-gray-1);
+ width: 98px;
+ min-height: 32px;
+ font-size: 13px;
+ position: fixed;
+ padding: 4px 0px;
+ display: flex;
+ justify-content: center;
+ align-items: center;
+ font-family: 'Lato', sans-serif;
+
+ .formFieldIndicator-text {
+ margin: 0;
+ position: absolute;
+ padding: 0 8px;
+ }
+}
+
+.formFieldIndicator::before {
+ content: '';
+ display: block;
+ position: absolute;
+ right: -20px;
+ bottom: 0;
+ width: 0;
+ height: 0;
+ border-left: 20px solid var(--color-blue-gray-5);
+ border-top: 20px solid transparent;
+ border-bottom: 20px solid transparent;
+}
+
+.formFieldIndicator.rightSidePage::before {
+ content: '';
+ display: block;
+ position: absolute;
+ left: -20px;
+ bottom: 0;
+ width: 0;
+ height: 0;
+ border-right: 20px solid var(--color-blue-gray-5);
+ border-left: 0;
+ border-top: 20px solid transparent;
+ border-bottom: 20px solid transparent;
+}
\ No newline at end of file
diff --git a/src/components/FormFieldIndicator/FormFieldIndicatorContainer.js b/src/components/FormFieldIndicator/FormFieldIndicatorContainer.js
new file mode 100644
index 0000000000..4aa3d6634e
--- /dev/null
+++ b/src/components/FormFieldIndicator/FormFieldIndicatorContainer.js
@@ -0,0 +1,113 @@
+import React, { useEffect, useState } from 'react';
+import { useSelector } from 'react-redux';
+import debounce from 'lodash/debounce';
+import useOnFormFieldsChanged from '../../hooks/useOnFormFieldsChanged';
+import core from 'core';
+import selectors from 'selectors';
+import FormFieldIndicator from './FormFieldIndicator';
+import './FormFieldIndicator.scss';
+import DataElements from 'src/constants/dataElement';
+import getRootNode from 'helpers/getRootNode';
+import { createPortal } from 'react-dom';
+
+const FormFieldIndicatorContainer = () => {
+ const [
+ isOpen,
+ isDisabled,
+ documentContainerWidth,
+ documentContainerHeight,
+ leftPanelWidth,
+ notePanelWidth,
+ ] = useSelector((state) => [
+ selectors.isElementOpen(state, DataElements['FORM_FIELD_INDICATOR_CONTAINER']),
+ selectors.isElementDisabled(state, DataElements['FORM_FIELD_INDICATOR_CONTAINER']),
+ selectors.getDocumentContainerWidth(state),
+ selectors.getDocumentContainerHeight(state),
+ selectors.getLeftPanelWidth(state),
+ selectors.getDocumentContentContainerWidthStyle(state),
+ selectors.getNotesPanelWidth(state),
+ ]);
+ const formFieldAnnotationsList = useOnFormFieldsChanged();
+ const [indicators, setIndicators] = useState([]);
+
+ const getIndicators = () => {
+ if (!core.getDocument()) {
+ return [];
+ }
+ return formFieldAnnotationsList
+ .filter((fieldAnnotation) => {
+ return fieldAnnotation.getCustomData('trn-form-field-show-indicator') === 'true';
+ }).map((fieldAnnotation) => {
+ return createFormFieldIndicator(fieldAnnotation);
+ });
+ };
+
+ const resetIndicators = () => {
+ setIndicators([]);
+ };
+
+ useEffect(() => {
+ core.addEventListener('documentUnloaded', resetIndicators);
+ return () => {
+ core.removeEventListener('documentUnloaded', resetIndicators);
+ };
+ }, []);
+
+
+ useEffect(() => {
+ setIndicators(getIndicators());
+
+ const onScroll = debounce(() => {
+ if (isOpen && !isDisabled) {
+ setIndicators(getIndicators());
+ }
+ }, 0);
+
+ const scrollViewElement = core.getScrollViewElement();
+
+ scrollViewElement?.addEventListener('scroll', onScroll);
+ return () => {
+ scrollViewElement?.removeEventListener('scroll', onScroll);
+ };
+ }, [
+ formFieldAnnotationsList,
+ isOpen,
+ isDisabled,
+ documentContainerWidth,
+ documentContainerHeight,
+ leftPanelWidth,
+ notePanelWidth,
+ ]);
+
+ const createFormFieldIndicator = (annotation) => {
+ const { scrollLeft, scrollTop } = core.getScrollViewElement();
+ const payload = {
+ displayMode: core.getDocumentViewer().getDisplayModeManager().getDisplayMode(),
+ viewerBoundingRect: core.getViewerElement().getBoundingClientRect(),
+ appBoundingRect: getRootNode().getElementById('app').getBoundingClientRect(),
+ scrollLeft: scrollLeft,
+ scrollTop: scrollTop,
+ };
+ return (
);
+ };
+
+ if (isOpen && !isDisabled) {
+ return (<>
+ {
+ createPortal(
, (window.isApryseWebViewerWebComponent)
+ ? getRootNode().getElementById('app') : document.body)
+ }
+ >);
+ }
+
+ return null;
+};
+
+export default FormFieldIndicatorContainer;
diff --git a/src/components/FormFieldIndicator/index.js b/src/components/FormFieldIndicator/index.js
new file mode 100644
index 0000000000..33db0c2d60
--- /dev/null
+++ b/src/components/FormFieldIndicator/index.js
@@ -0,0 +1,3 @@
+import FormFieldIndicator from './FormFieldIndicatorContainer';
+
+export default FormFieldIndicator;
diff --git a/src/components/InlineCommentingPopup/InlineCommentingPopup.scss b/src/components/InlineCommentingPopup/InlineCommentingPopup.scss
new file mode 100644
index 0000000000..cdfb46b88f
--- /dev/null
+++ b/src/components/InlineCommentingPopup/InlineCommentingPopup.scss
@@ -0,0 +1,171 @@
+@import '../../constants/styles.scss';
+@import '../../constants/popup';
+
+.InlineCommentingPopup {
+ @extend %popup;
+ border-radius: 4px;
+ box-shadow: 0 0 3px 0 var(--document-box-shadow);
+ background: var(--component-background);
+ align-items: flex-start;
+
+ @include mobile {
+ position: fixed;
+ left: 0;
+ bottom: 0;
+ z-index: 100;
+ flex-direction: column;
+ justify-content: flex-end;
+ width: 100%;
+ background: var(--modal-negative-space);
+ }
+
+ @include tablet-and-desktop {
+ overflow: auto;
+ max-height: calc(100% - 100px);
+ }
+
+ .inline-comment-container {
+ display: flex;
+ flex-direction: column;
+
+ @include mobile {
+ flex-basis: auto;
+ width: 100%;
+ max-height: 40%;
+ background: var(--component-background);
+ box-shadow: 0 0 3px 0 var(--document-box-shadow);
+ border-radius: 4px 4px 0px 0px;
+ }
+
+ @include tablet-and-desktop {
+ max-width: 260px;
+ }
+
+ &.expanded {
+ @include mobile {
+ flex-grow: 1;
+ max-height: 90%;
+ }
+ }
+ }
+
+ .Note {
+ border-radius: 0;
+ background: none;
+ margin: 0;
+ cursor: default;
+
+ @include mobile {
+ flex-grow: 1;
+ display: flex;
+ flex-direction: column;
+ overflow: auto;
+ box-shadow: 0 0 3px 0 var(--document-box-shadow);
+ }
+
+ @include tablet-and-desktop {
+ box-shadow: none;
+ }
+
+ &>div:not(:nth-last-child(2)) {
+ @include mobile {
+ flex-grow: 0;
+ }
+ }
+
+ &>div:nth-last-child(2) {
+ @include mobile {
+ flex-grow: 1;
+ }
+ }
+
+ &>.NoteContent:only-child {
+ @include mobile {
+ flex-grow: 1;
+ }
+
+ .edit-content {
+ @include mobile {
+ flex-grow: 0;
+ }
+ }
+ }
+ }
+
+ .NoteHeader {
+ @include mobile {
+ flex-grow: 0;
+ }
+ }
+
+ .NoteContent .edit-content {
+ margin-top: 16px;
+ }
+
+ .note-popup-options:not(.options-reply) {
+ top: 33px;
+ }
+
+ .quill,
+ .ql-container,
+ .ql-editor {
+ @include mobile {
+ font-size: 16px;
+ }
+ }
+
+ .inline-comment-header {
+ flex-grow: 0;
+ flex-shrink: 0;
+ display: flex;
+ flex-direction: row;
+ align-items: center;
+ }
+
+ .inline-comment-header-title {
+ flex-grow: 1;
+ font-size: 16px;
+ }
+
+ .Button {
+
+ &.add-attachment,
+ &.reply-button {
+ margin: 0;
+
+ .Icon {
+ width: 22px;
+ height: 22px;
+ }
+ }
+
+ &.add-attachment {
+ width: 24px;
+ height: 24px;
+
+ @include mobile {
+ width: 24px;
+ height: 24px;
+ }
+ }
+
+ &.reply-button {
+ width: 28px;
+ height: 28px;
+
+ @include mobile {
+ width: 28px;
+ height: 28px;
+ }
+ }
+ }
+}
+
+// fix for storybook
+.sb-show-main {
+ .InlineCommentingPopup {
+ .quill.comment-textarea {
+ padding: 0;
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/InlineCommentingPopup/InlineCommentingPopup.spec.js b/src/components/InlineCommentingPopup/InlineCommentingPopup.spec.js
new file mode 100644
index 0000000000..d02323dba1
--- /dev/null
+++ b/src/components/InlineCommentingPopup/InlineCommentingPopup.spec.js
@@ -0,0 +1,44 @@
+import React from 'react';
+import { render } from '@testing-library/react';
+import * as reactRedux from 'react-redux';
+import { Basic, Mobile, initialState } from './InlineCommentingPopup.stories';
+
+const TestInlineCommentPopup = withProviders(Basic);
+const TestInlineCommentPopupMobile = withProviders(Mobile);
+
+jest.mock('core', () => ({
+ getGroupAnnotations: () => [],
+ getDisplayAuthor: () => '',
+ canModify: () => true,
+ canModifyContents: () => true,
+ addEventListener: () => { },
+ removeEventListener: () => { },
+}));
+
+describe('InlineCommentPopup Component', () => {
+ beforeEach(() => {
+ const useDispatchMock = jest.spyOn(reactRedux, 'useDispatch');
+ useDispatchMock.mockImplementation(() => { });
+
+ const useSelectorMock = jest.spyOn(reactRedux, 'useSelector');
+ useSelectorMock.mockImplementation((selector) => selector(initialState));
+ });
+
+ afterEach(() => {
+ jest.restoreAllMocks();
+ });
+
+ it('Should not throw any errors when rendering storybook component', () => {
+ expect(() => {
+ render(
);
+ }).not.toThrow();
+ });
+
+ it('Should show header for mobile correctly', () => {
+ const { container } = render(
+
+ );
+ const mobileHeader = container.querySelector('.inline-comment-header');
+ expect(mobileHeader).not.toBeNull();
+ });
+});
\ No newline at end of file
diff --git a/src/components/InlineCommentingPopup/InlineCommentingPopup.stories.js b/src/components/InlineCommentingPopup/InlineCommentingPopup.stories.js
new file mode 100644
index 0000000000..a8d6ed08dd
--- /dev/null
+++ b/src/components/InlineCommentingPopup/InlineCommentingPopup.stories.js
@@ -0,0 +1,83 @@
+import React from 'react';
+import InlineCommentingPopup from './InlineCommentingPopup';
+import { configureStore } from '@reduxjs/toolkit';
+import { Provider } from 'react-redux';
+
+const noop = () => { };
+
+export default {
+ title: 'Components/InlineCommentPopup',
+ component: InlineCommentingPopup,
+ includeStories: ['Basic', 'Mobile'],
+};
+
+export const initialState = {
+ viewer: {
+ disabledElements: {},
+ customElementOverrides: {},
+ openElements: { inlineCommentPopup: true },
+ customPanels: [],
+ unreadAnnotationIdSet: new Set(),
+ colorMap: [{ colorMapKey: () => '#F1A099' }],
+ },
+};
+
+export const context = {
+ searchInput: '',
+ resize: noop,
+ isSelected: true,
+ setCurAnnotId: noop,
+ onTopNoteContentClicked: noop,
+ pendingEditTextMap: {},
+ pendingReplyMap: {},
+ pendingAttachmentMap: {}
+};
+
+
+const mockAnnotation = {
+ Author: 'Mikel Landa',
+ isFormFieldPlaceholder: () => false,
+ getReplies: () => [],
+ getStatus: () => '',
+ isReply: () => false,
+ getAssociatedNumber: () => 1,
+ getContents: noop,
+ getCustomData: () => '',
+ getAttachments: noop,
+ getRichTextStyle: noop,
+};
+
+export const basicProps = {
+ isOpen: true,
+ isNotesPanelOpen: false,
+ commentingAnnotation: mockAnnotation,
+ position: { top: 0, left: 0 },
+ contextValue: context,
+};
+
+export const Basic = () => {
+ return (
+
initialState })}>
+
+
+ );
+};
+
+export const mobileProps = {
+ ...basicProps,
+ isMobile: true,
+};
+
+export const Mobile = () => {
+ return (
+
initialState })}>
+
+
+ );
+};
+
+Mobile.parameters = {
+ viewport: {
+ defaultViewport: 'mobile1',
+ },
+};
\ No newline at end of file
diff --git a/src/components/InlineCommentingPopup/index.js b/src/components/InlineCommentingPopup/index.js
new file mode 100644
index 0000000000..6af61d6503
--- /dev/null
+++ b/src/components/InlineCommentingPopup/index.js
@@ -0,0 +1,3 @@
+import InlineCommentingPopupContainer from './InlineCommentingPopupContainer';
+
+export default InlineCommentingPopupContainer;
\ No newline at end of file
diff --git a/src/components/InsertPageModal/InsertBlankPagePanel/InsertBlankPagePanel.scss b/src/components/InsertPageModal/InsertBlankPagePanel/InsertBlankPagePanel.scss
new file mode 100644
index 0000000000..ebf9c666ea
--- /dev/null
+++ b/src/components/InsertPageModal/InsertBlankPagePanel/InsertBlankPagePanel.scss
@@ -0,0 +1,202 @@
+@import '../../../constants/styles';
+
+.insert-blank-page-panel {
+ width: 100%;
+
+ .dimension-input-container {
+ min-width: 100%;
+ margin: 0;
+ height: 32px;
+ }
+
+ .subheader {
+ font-size: 13px;
+ font-weight: 700;
+ padding-top: 8px;
+ padding-bottom: 8px;
+ }
+
+ .panel-container {
+
+ .section {
+ display: flex;
+ padding-top: 8px;
+ padding-bottom: 8px;
+ gap: 16px;
+
+ .input-container {
+ display: flex;
+ flex-direction: column;
+
+ p {
+ margin: 0;
+ padding-bottom: 8px;
+ font-size: 13px;
+ }
+
+ .page-number-input {
+ width: 100%;
+ height: 32px;
+ margin: 0;
+ }
+
+ .customSelector {
+ margin-left: 0;
+ height: 28px;
+
+ .customSelector__selectedItem {
+ width: 100%;
+ border-radius: 4px;
+ }
+
+ .Icon {
+ width: 13px;
+ height: 13px;
+ }
+
+ ul {
+ width: 100%;
+
+ @include mobile {
+ top: auto;
+ bottom: calc(100% + 4px);
+ }
+ }
+
+ & li:first-child {
+ color: var(--faded-text);
+ font-size: 13px;
+
+ @include mobile {
+ display: none;
+ }
+ }
+
+ li .optionSelected {
+ color: var(--text-color);
+ background: var(--popup-button-active);
+ }
+ }
+
+ select {
+ height: 28px;
+ width: 100%;
+ }
+
+ .Dropdown {
+ height: 32px;
+ min-width: 150px;
+ width: 100% !important;
+
+ .arrow {
+ flex: 0 1 auto;
+ }
+
+ .picked-option .picked-option-text {
+ width: 150px;
+ text-align: left;
+ }
+ }
+
+ .Dropdown__items {
+ top: -52px;
+ z-index: 80;
+ width: 100%;
+ }
+
+ .input-sub-text {
+ margin-top: 8px;
+ padding-bottom: 0;
+ color: var(--faded-text);
+ }
+
+ .page-number-error {
+ margin-top: 8px;
+ font-size: 13px;
+ color: var(--color-message-error);
+ }
+ }
+
+ @include mobile {
+ .ui__choice__label, input, button {
+ font-size: 13px;
+ }
+ }
+ }
+
+ .section > * {
+ flex: 1;
+ }
+ }
+}
+.incrementNumberInput {
+ border: 1px solid var(--border);
+ border-radius: 4px;
+ display: flex;
+ height: 32px;
+
+ input[type=number]::-webkit-inner-spin-button,
+ input[type=number]::-webkit-outer-spin-button {
+ -webkit-appearance: none;
+ margin: 0;
+ }
+
+ input[type=number] {
+ -moz-appearance:textfield;
+ }
+
+ .ui__input {
+ border: 0;
+ height: 100%;
+
+ .ui__input__input {
+ width: 100%;
+ height: 100%;
+ padding: 6px;
+ line-height: normal;
+
+ }
+ }
+
+ .ui__input--message-default.ui__input--focused {
+ outline: none;
+ box-shadow: none;
+ }
+
+ .increment-buttons {
+ @include mobile {
+ display: none;
+ }
+
+ display: flex;
+ flex-direction: column;
+ gap: 2px;
+ justify-content: center;
+ padding: 2px;
+
+ .increment-arrow-button {
+ border: 0;
+ border-radius: 2px;
+ height: 10px;
+ width: 20px;
+ display: flex;
+ justify-content: center;
+ align-items: center;
+ line-height: 10px;
+ padding: 0;
+
+ .Icon {
+ height: 10px;
+ width: 10px;
+ }
+
+ &:active {
+ box-shadow: 0 0 1px 0 var(--document-box-shadow);
+ }
+ }
+ }
+
+ &:focus-within {
+ border: 1px solid var(--focus-border);
+ }
+}
diff --git a/src/components/InsertPageModal/InsertBlankPagePanel/index.js b/src/components/InsertPageModal/InsertBlankPagePanel/index.js
new file mode 100644
index 0000000000..35b87e586d
--- /dev/null
+++ b/src/components/InsertPageModal/InsertBlankPagePanel/index.js
@@ -0,0 +1,3 @@
+import InsertBlankPagePanel from './InsertBlankPagePanel';
+
+export default InsertBlankPagePanel;
\ No newline at end of file
diff --git a/src/components/InsertPageModal/InsertUploadedPagePanel/InsertUploadedPagePanel.scss b/src/components/InsertPageModal/InsertUploadedPagePanel/InsertUploadedPagePanel.scss
index 8ba7533d5f..8691b1083e 100644
--- a/src/components/InsertPageModal/InsertUploadedPagePanel/InsertUploadedPagePanel.scss
+++ b/src/components/InsertPageModal/InsertUploadedPagePanel/InsertUploadedPagePanel.scss
@@ -17,7 +17,7 @@
width: 100%;
font-size: 16px;
line-height: 24px;
- color: var(--gray-9);
+ color: var(--gray-8);
font-weight: 700;
box-shadow: inset 0px -1px 0px var(--divider);
padding: 20px 16px 20px 16px;
diff --git a/src/components/LazyLoadWrapper/index.js b/src/components/LazyLoadWrapper/index.js
new file mode 100644
index 0000000000..c0f8af0a5e
--- /dev/null
+++ b/src/components/LazyLoadWrapper/index.js
@@ -0,0 +1,5 @@
+import LazyLoadWrapper from './LazyLoadWrapper';
+
+export { default as LazyLoadComponents } from './LazyLoadComponents';
+
+export default LazyLoadWrapper;
diff --git a/src/components/LeftPanelOverlay/ThumbnailMoreOptionsPopupSmall.js b/src/components/LeftPanelOverlay/ThumbnailMoreOptionsPopupSmall.js
new file mode 100644
index 0000000000..5fe573e7f5
--- /dev/null
+++ b/src/components/LeftPanelOverlay/ThumbnailMoreOptionsPopupSmall.js
@@ -0,0 +1,29 @@
+import React from 'react';
+import { useSelector } from 'react-redux';
+import selectors from 'selectors';
+import FlyoutMenu from 'components/FlyoutMenu/FlyoutMenu';
+import PageAdditionalControls from 'components/PageManipulationOverlay/PageAdditionalControls';
+import PageManipulationControls from '../PageManipulationOverlay/PageManipulationControls';
+import DataElements from 'constants/dataElement';
+
+const ThumbnailMoreOptionsPopupSmall = () => {
+ const selectedPageIndexes = useSelector((state) => selectors.getSelectedThumbnailPageIndexes(state));
+
+ return (
+
+ i + 1)}
+ warn
+ />
+ i + 1)}
+ warn
+ />
+
+ );
+};
+
+export default ThumbnailMoreOptionsPopupSmall;
diff --git a/src/components/LinkAnnotationPopup/LinkAnnotationPopup.js b/src/components/LinkAnnotationPopup/LinkAnnotationPopup.js
new file mode 100644
index 0000000000..e5639f5d9f
--- /dev/null
+++ b/src/components/LinkAnnotationPopup/LinkAnnotationPopup.js
@@ -0,0 +1,71 @@
+import ActionButton from 'components/ActionButton';
+import classNames from 'classnames';
+import DataElements from 'constants/dataElement';
+import PropTypes from 'prop-types';
+import React from 'react';
+
+import './LinkAnnotationPopup.scss';
+
+const propTypes = {
+ handleUnLink: PropTypes.func,
+ isAnnotation: PropTypes.bool,
+ isMobileDevice: PropTypes.bool,
+ linkText: PropTypes.string,
+ handleOnMouseEnter: PropTypes.func,
+ handleOnMouseLeave: PropTypes.func,
+ handleMouseMove: PropTypes.func,
+};
+
+const LinkAnnotationPopup = ({
+ handleUnLink,
+ isAnnotation,
+ isMobileDevice,
+ linkText,
+ handleOnMouseEnter,
+ handleOnMouseLeave,
+ handleMouseMove
+}) => {
+ const renderContents = () => (
+
+ {linkText && (
+ <>
+
+ {linkText}
+
+
+ >
+ )}
+
+
+ );
+
+ if (isMobileDevice || !isAnnotation) {
+ return null;
+ }
+
+ return (
+
+ {renderContents()}
+
+ );
+};
+
+LinkAnnotationPopup.propTypes = propTypes;
+
+export default LinkAnnotationPopup;
diff --git a/src/components/LinkAnnotationPopup/LinkAnnotationPopup.scss b/src/components/LinkAnnotationPopup/LinkAnnotationPopup.scss
new file mode 100644
index 0000000000..c09a744bbe
--- /dev/null
+++ b/src/components/LinkAnnotationPopup/LinkAnnotationPopup.scss
@@ -0,0 +1,52 @@
+@import '../../constants/styles.scss';
+@import '../../constants/popup';
+
+.LinkAnnotationPopupContainer {
+ @extend %popup;
+ border-radius: 4px;
+ box-shadow: 0 0 3px 0 var(--document-box-shadow);
+ background: var(--component-background);
+}
+
+.LinkAnnotationPopup {
+ &.is-horizontal {
+ .contents {
+ display: flex;
+ justify-content: center;
+ align-items: center;
+ gap: 8px;
+
+ .link-annot-input {
+ margin: 8px 0 8px 8px;
+ color: #485056;
+ font-style: normal;
+ font-weight: 400;
+ line-height: normal;
+ /* top right bottom left */
+ border: none;
+ width: fit-content;
+ max-width: 240px;
+ cursor: pointer;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ display: -webkit-box;
+ -webkit-line-clamp: 3;
+ line-clamp: 3;
+ -webkit-box-orient: vertical;
+ box-sizing: border-box;
+ word-break: break-all;
+ }
+
+ .divider {
+ width: 1px;
+ height: 20px;
+ background: var(--divider);
+ flex-shrink: 0;
+ }
+
+ .main-menu-button {
+ margin: 4px 8px 4px 0;
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/LinkAnnotationPopup/LinkAnnotationPopup.stories.js b/src/components/LinkAnnotationPopup/LinkAnnotationPopup.stories.js
new file mode 100644
index 0000000000..8065cd5f77
--- /dev/null
+++ b/src/components/LinkAnnotationPopup/LinkAnnotationPopup.stories.js
@@ -0,0 +1,40 @@
+import React from 'react';
+import LinkAnnotationPopup from './LinkAnnotationPopup';
+import { configureStore } from '@reduxjs/toolkit';
+import { Provider } from 'react-redux';
+
+const noop = () => { };
+
+export default {
+ title: 'Components/LinkAnnotationPopup',
+ component: LinkAnnotationPopup,
+};
+
+const initialState = {
+ viewer: {
+ disabledElements: {},
+ customElementOverrides: {},
+ annotationPopup: [
+ ],
+ activeDocumentViewerKey: 1,
+ },
+};
+
+export const Basic = () => {
+ const props = {
+ linkText: 'https://www.Apryse.com',
+ handleUnLink: noop,
+ handleOnMouseEnter: noop,
+ handleOnMouseLeave: noop,
+ handleMouseMove: noop,
+ isAnnotation: true,
+ isMobileDevice: false,
+ };
+ return (
+
initialState })}>
+
+
+
+
+ );
+};
diff --git a/src/components/LinkAnnotationPopup/index.js b/src/components/LinkAnnotationPopup/index.js
new file mode 100644
index 0000000000..a60b394985
--- /dev/null
+++ b/src/components/LinkAnnotationPopup/index.js
@@ -0,0 +1,3 @@
+import LinkAnnotationPopupContainer from './LinkAnnotationPopupContainer';
+
+export default LinkAnnotationPopupContainer;
\ No newline at end of file
diff --git a/src/components/LinkModal/LinkModal.spec.js b/src/components/LinkModal/LinkModal.spec.js
new file mode 100644
index 0000000000..5ebc24e6a3
--- /dev/null
+++ b/src/components/LinkModal/LinkModal.spec.js
@@ -0,0 +1,16 @@
+import React from 'react';
+import { render, fireEvent } from '@testing-library/react';
+import { NoURLInput } from './LinkModal.stories';
+import core from 'core';
+
+core.addEventListener = jest.fn();
+
+describe('LinkModal', () => {
+ describe('Component', () => {
+ it('Story should not throw any errors', () => {
+ expect(() => {
+ render(
);
+ }).not.toThrow();
+ });
+ });
+});
diff --git a/src/components/LinkModal/LinkModal.stories.js b/src/components/LinkModal/LinkModal.stories.js
new file mode 100644
index 0000000000..61a3896bac
--- /dev/null
+++ b/src/components/LinkModal/LinkModal.stories.js
@@ -0,0 +1,54 @@
+import React, { useState } from 'react';
+import { configureStore } from '@reduxjs/toolkit';
+import { Provider as ReduxProvider } from 'react-redux';
+import LinkModal from './LinkModal';
+import core from 'core';
+
+export default {
+ title: 'Components/LinkModal',
+ component: LinkModal,
+};
+
+const initialState = {
+ viewer: {
+ disabledElements: {},
+ openElements: {
+ 'linkModal': true,
+ },
+ currentPage: 1,
+ selectedTab: 'notesPanel',
+ tab: {
+ linkModal: 'URLPanelButton'
+ },
+ customElementOverrides: {},
+ pageLabels: []
+ },
+ document: {
+ totalPages: 1
+ }
+};
+const store = configureStore({
+ reducer: () => initialState
+});
+
+export function NoURLInput() {
+ core.getDocumentViewer = () => ({
+ getAnnotationManager: () => ({
+ isAnnotationSelected: () => true
+ }),
+ getSelectedText: () => 'selected text'
+ });
+
+ core.getSelectedAnnotations = () => ({});
+
+ return (
+
+
+
+
+
+ );
+}
\ No newline at end of file
diff --git a/src/components/LogoBar/LogoBar.js b/src/components/LogoBar/LogoBar.js
new file mode 100644
index 0000000000..f28018de26
--- /dev/null
+++ b/src/components/LogoBar/LogoBar.js
@@ -0,0 +1,42 @@
+import './LogoBar.scss';
+import React from 'react';
+import selectors from 'selectors';
+import DataElements from 'constants/dataElement';
+import { useSelector } from 'react-redux';
+import packageConfig from '../../../package.json';
+import Button from '../Button';
+import { isMobileSize } from 'src/helpers/getDeviceSize';
+
+const LogoBar = () => {
+ const [
+ isDisabled,
+ ] = useSelector((state) => [
+ selectors.isElementDisabled(state, DataElements.LOGO_BAR),
+ ]);
+
+ const logoText = isMobileSize() ? 'Apryse' : 'Powered by Apryse';
+ const versionText = isMobileSize() ? packageConfig.version : `Version ${packageConfig.version}`;
+ const apryseURL = 'https://apryse.com/products/webviewer';
+ const apryseRedirect = () => {
+ window.top.location.href = apryseURL;
+ };
+
+ return isDisabled ? null : (
+
+ );
+};
+
+export default LogoBar;
diff --git a/src/components/LogoBar/LogoBar.scss b/src/components/LogoBar/LogoBar.scss
new file mode 100644
index 0000000000..e203b2202c
--- /dev/null
+++ b/src/components/LogoBar/LogoBar.scss
@@ -0,0 +1,38 @@
+@import '../../constants/styles';
+
+.LogoBar {
+ display: flex;
+ flex-direction: row;
+ justify-content: space-between;
+ align-items: center;
+ padding: 4px 16px;
+ width: 100%;
+ height: $logo-bar-height;
+ background: var(--gray-0);
+ color: #747C84;
+ font-size: 12px;
+ line-height: 16px;
+
+ .logo-container {
+ display: flex;
+ flex-direction: row;
+ align-items: center;
+ }
+
+ .logo-button {
+ width: 100%;
+ gap: 4px;
+ color: #747C84;
+ font-size: 12px !important;
+
+ .Icon {
+ width: 14px;
+ height: 14px;
+ }
+ }
+
+ .version {
+ text-decoration: none;
+ color: #747C84;
+ }
+}
\ No newline at end of file
diff --git a/src/components/LogoBar/LogoBar.stories.js b/src/components/LogoBar/LogoBar.stories.js
new file mode 100644
index 0000000000..e90c79b544
--- /dev/null
+++ b/src/components/LogoBar/LogoBar.stories.js
@@ -0,0 +1,29 @@
+import React from 'react';
+import { configureStore } from '@reduxjs/toolkit';
+import LogoBar from './LogoBar';
+
+import { Provider } from 'react-redux';
+
+export default {
+ title: 'Components/LogoBar',
+ component: LogoBar,
+};
+
+const initialState = {
+ viewer: {
+ disabledElements: {},
+ customElementOverrides: {},
+ },
+};
+
+const store = configureStore({
+ reducer: () => initialState
+});
+
+export const LogoBarComponent = () => (
+
+
+
+
+
+);
\ No newline at end of file
diff --git a/src/components/LogoBar/index.js b/src/components/LogoBar/index.js
new file mode 100644
index 0000000000..561567d51e
--- /dev/null
+++ b/src/components/LogoBar/index.js
@@ -0,0 +1,3 @@
+import LogoBar from './LogoBar';
+
+export default LogoBar;
diff --git a/src/components/ModularComponents/BottomHeader/BottomHeader.scss b/src/components/ModularComponents/BottomHeader/BottomHeader.scss
new file mode 100644
index 0000000000..ac3c7ba51c
--- /dev/null
+++ b/src/components/ModularComponents/BottomHeader/BottomHeader.scss
@@ -0,0 +1,23 @@
+.bottom-headers-wrapper {
+ position: absolute;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ pointer-events: none;
+}
+
+.BottomHeader {
+ display: inline-block;
+ width: 100%;
+ pointer-events: auto;
+}
+
+.BottomHeader.closed {
+ display: none;
+}
+
+.BottomHeader.stroke {
+ border-top-width: 1px;
+ border-top-style: solid;
+ border-top-color: var(--gray-5);
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/BottomHeader/index.js b/src/components/ModularComponents/BottomHeader/index.js
new file mode 100644
index 0000000000..e4d6606372
--- /dev/null
+++ b/src/components/ModularComponents/BottomHeader/index.js
@@ -0,0 +1,3 @@
+import BottomHeader from './BottomHeaderContainer';
+
+export default BottomHeader;
\ No newline at end of file
diff --git a/src/components/ModularComponents/CustomButton/CustomButton.js b/src/components/ModularComponents/CustomButton/CustomButton.js
new file mode 100644
index 0000000000..029a7407c6
--- /dev/null
+++ b/src/components/ModularComponents/CustomButton/CustomButton.js
@@ -0,0 +1,47 @@
+import React from 'react';
+import '../../Button/Button.scss';
+import './CustomButton.scss';
+import classNames from 'classnames';
+import PropTypes from 'prop-types';
+import Button from 'components/Button';
+import { PLACEMENT } from 'constants/customizationVariables';
+
+const CustomButton = (props) => {
+ const { title, dataElement, label, img, onClick, disabled, className, preset, headerPlacement, ariaLabel } = props;
+ let forceTooltipPosition;
+ if ([PLACEMENT.LEFT, PLACEMENT.RIGHT].includes(headerPlacement)) {
+ forceTooltipPosition = headerPlacement === PLACEMENT.LEFT ? PLACEMENT.RIGHT : PLACEMENT.LEFT;
+ } else if ([PLACEMENT.TOP, PLACEMENT.BOTTOM].includes(headerPlacement)) {
+ forceTooltipPosition = headerPlacement === PLACEMENT.TOP ? PLACEMENT.BOTTOM : PLACEMENT.TOP;
+ }
+ return (
+
+ );
+};
+
+CustomButton.propTypes = {
+ dataElement: PropTypes.string,
+ title: PropTypes.string,
+ label: PropTypes.string,
+ img: PropTypes.string,
+ onClick: PropTypes.func,
+ disabled: PropTypes.bool,
+};
+
+export default CustomButton;
diff --git a/src/components/ModularComponents/CustomButton/CustomButton.scss b/src/components/ModularComponents/CustomButton/CustomButton.scss
new file mode 100644
index 0000000000..3a5f9f65b6
--- /dev/null
+++ b/src/components/ModularComponents/CustomButton/CustomButton.scss
@@ -0,0 +1,46 @@
+@import '../../../constants/styles';
+
+.CustomButton {
+ padding: 5px;
+ width: fit-content;
+
+ &:hover {
+ background-color: var(--view-header-button-hover);
+ }
+}
+
+.confirm-button {
+ background-color: var(--primary-button);
+ border: 1px solid var(--primary-button);
+ color: var(--primary-button-text);
+ padding: 7px 14px;
+ width: -webkit-fit-content;
+ width: -moz-fit-content;
+ width: fit-content;
+ border-radius: 5px;
+ height: 30px;
+ cursor: pointer;
+
+ &:hover {
+ background: var(--primary-button-hover) !important;
+ border: 1px solid var(--primary-button-hover) !important;
+ border-radius: 5px !important;
+ }
+}
+
+.cancel-button {
+ color: var(--secondary-button-text);
+ background-color: transparent;
+ padding: 7px 14px;
+ width: -webkit-fit-content;
+ width: -moz-fit-content;
+ width: fit-content;
+ border-radius: 5px;
+ height: 30px;
+ cursor: pointer;
+
+ &:hover {
+ color: var(--secondary-button-hover);
+ background: transparent;
+ }
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/CustomButton/CustomButton.stories.js b/src/components/ModularComponents/CustomButton/CustomButton.stories.js
new file mode 100644
index 0000000000..a099cfae2a
--- /dev/null
+++ b/src/components/ModularComponents/CustomButton/CustomButton.stories.js
@@ -0,0 +1,56 @@
+import React from 'react';
+import CustomButton from './CustomButton';
+import initialState from 'src/redux/initialState';
+import { Provider } from 'react-redux';
+import { configureStore } from '@reduxjs/toolkit';
+
+export default {
+ title: 'Components/CustomButton',
+ component: CustomButton,
+};
+
+const store = configureStore({
+ reducer: () => initialState
+});
+
+const BasicComponent = (props) => {
+ return (
+
+
+
+ );
+};
+
+export const DefaultButton = BasicComponent.bind({});
+DefaultButton.args = {
+ dataElement: 'button-data-element',
+ title: 'Button title',
+ disabled: false,
+ label: 'Click',
+ img: 'icon-save',
+ onClick: () => {
+ alert('Clicked!');
+ }
+};
+
+export const ConfirmButton = BasicComponent.bind({});
+ConfirmButton.args = {
+ dataElement: 'button-data-element',
+ title: 'Apply Fields',
+ label: 'Apply Fields',
+ preset: 'confirm',
+ onClick: () => {
+ alert('Apply Fields button clicked!');
+ }
+};
+
+export const CancelButton = BasicComponent.bind({});
+CancelButton.args = {
+ dataElement: 'button-data-element',
+ title: 'Cancel',
+ label: 'Cancel',
+ preset: 'cancel',
+ onClick: () => {
+ alert('Cancel button clicked!');
+ }
+};
\ No newline at end of file
diff --git a/src/components/ModularComponents/CustomButton/index.js b/src/components/ModularComponents/CustomButton/index.js
new file mode 100644
index 0000000000..e5f8cd2a41
--- /dev/null
+++ b/src/components/ModularComponents/CustomButton/index.js
@@ -0,0 +1,3 @@
+import CustomButton from './CustomButton';
+
+export default CustomButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/Divider/Divider.js b/src/components/ModularComponents/Divider/Divider.js
new file mode 100644
index 0000000000..c7b9aa1e30
--- /dev/null
+++ b/src/components/ModularComponents/Divider/Divider.js
@@ -0,0 +1,13 @@
+import React from 'react';
+import classNames from 'classnames';
+import './Divider.scss';
+
+const Divider = ({ headerDirection }) => {
+ const className = classNames('Divider', `${headerDirection || 'column'}`);
+
+ return (
+
+ );
+};
+
+export default Divider;
\ No newline at end of file
diff --git a/src/components/ModularComponents/Divider/Divider.scss b/src/components/ModularComponents/Divider/Divider.scss
new file mode 100644
index 0000000000..03f39f4228
--- /dev/null
+++ b/src/components/ModularComponents/Divider/Divider.scss
@@ -0,0 +1,15 @@
+.Divider {
+ background: var(--gray-5);
+
+ &.row {
+ width: 1px;
+ height: auto;
+ margin: 4px 0px;
+ }
+
+ &.column {
+ width: auto;
+ height: 1px;
+ margin: 0px 4px;
+ }
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/Divider/index.js b/src/components/ModularComponents/Divider/index.js
new file mode 100644
index 0000000000..7909d0ef6b
--- /dev/null
+++ b/src/components/ModularComponents/Divider/index.js
@@ -0,0 +1,3 @@
+import Divider from './Divider';
+
+export default Divider;
\ No newline at end of file
diff --git a/src/components/ModularComponents/FlexDropdown/FlexDropdown.js b/src/components/ModularComponents/FlexDropdown/FlexDropdown.js
new file mode 100644
index 0000000000..62a510a650
--- /dev/null
+++ b/src/components/ModularComponents/FlexDropdown/FlexDropdown.js
@@ -0,0 +1,12 @@
+import React from 'react';
+import Dropdown from 'components/Dropdown';
+
+import './FlexDropdown.scss';
+
+const FlexDropdown = (props) => {
+ return (
+
+ );
+};
+
+export default FlexDropdown;
\ No newline at end of file
diff --git a/src/components/ModularComponents/FlexDropdown/FlexDropdown.scss b/src/components/ModularComponents/FlexDropdown/FlexDropdown.scss
new file mode 100644
index 0000000000..94b98f1f51
--- /dev/null
+++ b/src/components/ModularComponents/FlexDropdown/FlexDropdown.scss
@@ -0,0 +1,51 @@
+@import '../../../constants/styles';
+
+.FlexDropdown {
+ &__wrapper {
+ position: relative;
+ }
+
+ .Dropdown__items {
+ z-index: $headers-z-index;
+ }
+
+ .Dropdown__item-object {
+ display: flex;
+ align-items: center;
+ gap: 4px;
+ }
+
+ &.column {
+ height: auto;
+ padding: 0;
+ .picked-option {
+ flex-direction: column;
+ gap: 4px;
+ padding: 4px;
+ }
+ .Dropdown__items {
+ z-index: $headers-z-index;
+ width: 100%;
+ padding: 0;
+ gap: 12px;
+ }
+ .Dropdown__item {
+ min-height: 28px;
+ height: 100%;
+ padding: 4px;
+ }
+ .Dropdown__item-object {
+ display: flex;
+ flex-direction: column;
+ max-width: 100%;
+ flex: 1;
+ }
+ }
+
+ .Dropdown__item-text {
+ white-space: nowrap;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ max-width: 100%;
+ }
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/FlexDropdown/index.js b/src/components/ModularComponents/FlexDropdown/index.js
new file mode 100644
index 0000000000..4d12ec3a2d
--- /dev/null
+++ b/src/components/ModularComponents/FlexDropdown/index.js
@@ -0,0 +1,3 @@
+import FlexDropdown from './FlexDropdown';
+
+export default FlexDropdown;
\ No newline at end of file
diff --git a/src/components/ModularComponents/FloatingHeader/FloatingHeader.scss b/src/components/ModularComponents/FloatingHeader/FloatingHeader.scss
new file mode 100644
index 0000000000..c556e61fed
--- /dev/null
+++ b/src/components/ModularComponents/FloatingHeader/FloatingHeader.scss
@@ -0,0 +1,117 @@
+@import '../../../constants/styles.scss';
+
+.FloatingHeaderContainer {
+ transition: all .3s ease-in-out;
+ position: absolute;
+ width: 100%;
+ box-sizing: border-box;
+ display: flex;
+ flex-direction: row;
+ align-items: baseline;
+ min-height: $top-bar-height;
+ padding: 24px;
+ z-index: $headers-z-index;
+ pointer-events: none;
+
+ &.bottom {
+ position: relative;
+ }
+
+ &.left {
+ width: $left-header-width;
+ padding: 24px 0px 24px 24px;
+ }
+
+ &.right {
+ width: $right-header-width;
+ align-items: end;
+ }
+
+ &.vertical {
+ flex-direction: column;
+ }
+
+ .FloatSection {
+ display: flex;
+ flex: 1;
+
+ &.vertical {
+ flex-direction: column;
+ }
+
+ // Maybe we'll need these maybe we wont
+ // &.start__top,
+ // &.start__bottom {
+ // // align-self: flex-start;
+ // }
+
+ // &.start__left,
+ // &.start__right {
+ // // align-self: flex-start;
+ // }
+
+ &.center__top,
+ &.center__bottom {
+ justify-content: center;
+ }
+
+ &.center__left,
+ &.center__right {
+ justify-content: center;
+ }
+
+ &.end__top,
+ &.end__bottom {
+ justify-content: flex-end;
+ }
+
+ &.end__left,
+ &.end__right {
+ justify-content: flex-end;
+ }
+ }
+}
+
+.FloatingHeader {
+ border-radius: 4px;
+ pointer-events: auto;
+ transition: opacity .2s ease;
+
+ // customizable styles
+ padding: 8px 12px;
+ background: var(--gray-0);
+
+ &.opacity-full {
+ opacity: 1;
+ }
+
+ &.opacity-low {
+ opacity: 0.5;
+
+ &.isVisible {
+ opacity: 1;
+ }
+ }
+
+ &.opacity-none {
+ opacity: 0;
+
+ &.isVisible {
+ opacity: 1;
+ }
+ }
+
+ &.opacity-mode-dynamic:hover {
+ opacity: 1;
+ }
+}
+
+.FloatingHeader.stroke {
+ border-width: 1px;
+ border-style: solid;
+ border-color: var(--gray-5);
+}
+
+.FloatingHeader.VerticalHeader {
+ padding: 12px 8px;
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/FloatingHeader/FloatingHeaderContainer.js b/src/components/ModularComponents/FloatingHeader/FloatingHeaderContainer.js
new file mode 100644
index 0000000000..9e95a0ae35
--- /dev/null
+++ b/src/components/ModularComponents/FloatingHeader/FloatingHeaderContainer.js
@@ -0,0 +1,120 @@
+import React, { useMemo } from 'react';
+import useFloatingHeaderSelectors from 'hooks/useFloatingHeaderSelectors';
+import FloatingHeader from './FloatingHeader';
+import './FloatingHeader.scss';
+import classNames from 'classnames';
+import { PLACEMENT, POSITION, DEFAULT_GAP } from 'src/constants/customizationVariables';
+
+const FloatSection = ({ position, isVertical, children, gap = DEFAULT_GAP }) => {
+ const className = classNames('FloatSection', position, { 'vertical': isVertical });
+ return (
+
+ {children}
+
+ );
+};
+
+const FloatingHeaderContainer = React.forwardRef((props, ref) => {
+ const { floatingHeaders, placement } = props;
+ const isHorizontalHeader = [PLACEMENT.TOP, PLACEMENT.BOTTOM].includes(placement);
+ const selectors = useFloatingHeaderSelectors();
+
+ const style = useMemo(() => computeFloatContainerStyle({
+ ...selectors,
+ isHorizontalHeader,
+ placement,
+ }), [selectors, isHorizontalHeader, placement]);
+
+ const renderHeaders = (headers, positionPrefix) => (
+
+ {headers.map((header) => )}
+
+ );
+
+ return (
+
+ {renderHeaders(floatingHeaders.filter((h) => h.position === POSITION.START), POSITION.START)}
+ {renderHeaders(floatingHeaders.filter((h) => h.position === POSITION.CENTER), POSITION.CENTER)}
+ {renderHeaders(floatingHeaders.filter((h) => h.position === POSITION.END), POSITION.END)}
+
+ );
+});
+
+FloatingHeaderContainer.displayName = 'FloatingHeaderContainer';
+
+function computeFloatContainerStyle(params) {
+ const {
+ isLeftPanelOpen,
+ leftPanelWidth,
+ isRightPanelOpen,
+ rightPanelWidth,
+ leftHeaderWidth,
+ rightHeaderWidth,
+ isHorizontalHeader,
+ topFloatingContainerHeight,
+ bottomFloatingContainerHeight,
+ topStartFloatingHeaders,
+ bottomStartFloatingHeaders,
+ topHeadersHeight,
+ bottomHeadersHeight,
+ bottomEndFloatingHeaders,
+ topEndFloatingHeaders,
+ placement
+ } = params;
+
+ const styles = {};
+ const verticalHeaderWidth = rightHeaderWidth + leftHeaderWidth;
+ let panelsWidth = 0;
+ let leftOffset = leftHeaderWidth;
+
+ if (isLeftPanelOpen) {
+ panelsWidth += leftPanelWidth;
+ leftOffset += leftPanelWidth;
+ }
+ if (isRightPanelOpen) {
+ panelsWidth += rightPanelWidth;
+ }
+
+ if (leftOffset !== 0) {
+ styles.transform = `translate(${leftOffset}px, 0px)`;
+ }
+ if (placement === PLACEMENT.RIGHT) {
+ styles.transform = 'translate(-48px, 0px)';
+ }
+ if (isHorizontalHeader && (panelsWidth || verticalHeaderWidth)) {
+ styles.width = `calc(100% - ${panelsWidth + verticalHeaderWidth}px)`;
+ }
+ if (!isHorizontalHeader) {
+ // if it is the left float header, and there are no top start floating headers, then we can take the full height
+ // otherwise the height must accotun for the floating header container
+ let topFloatingHeaderOffset = 0;
+ let bottomFloatingHeaderOffset = 0;
+
+ if (placement === PLACEMENT.LEFT) {
+ topFloatingHeaderOffset = topStartFloatingHeaders.length === 0 ? 0 : topFloatingContainerHeight;
+ bottomFloatingHeaderOffset = bottomStartFloatingHeaders.length === 0 ? 0 : bottomFloatingContainerHeight;
+ }
+
+ if (placement === PLACEMENT.RIGHT) {
+ topFloatingHeaderOffset = topEndFloatingHeaders.length === 0 ? 0 : topFloatingContainerHeight;
+ bottomFloatingHeaderOffset = bottomEndFloatingHeaders.length === 0 ? 0 : bottomFloatingContainerHeight;
+ }
+
+ styles.height = `calc(100% - ${topHeadersHeight + bottomHeadersHeight + topFloatingHeaderOffset + bottomFloatingHeaderOffset}px)`;
+ if (topFloatingHeaderOffset) {
+ styles.marginTop = `${topFloatingContainerHeight}px`;
+ styles.paddingTop = '0px';
+ }
+ if (bottomFloatingHeaderOffset) {
+ styles.paddingBottom = '0px';
+ }
+ }
+
+ return styles;
+}
+
+export default FloatingHeaderContainer;
\ No newline at end of file
diff --git a/src/components/ModularComponents/FloatingHeader/Stories/HeadersInApp.stories.js b/src/components/ModularComponents/FloatingHeader/Stories/HeadersInApp.stories.js
new file mode 100644
index 0000000000..b86672fa3c
--- /dev/null
+++ b/src/components/ModularComponents/FloatingHeader/Stories/HeadersInApp.stories.js
@@ -0,0 +1,233 @@
+import React from 'react';
+import { Provider } from 'react-redux';
+import { configureStore } from '@reduxjs/toolkit';
+import App from 'components/App';
+import initialState from 'src/redux/initialState';
+import rootReducer from 'reducers/rootReducer';
+import {
+ defaultLeftHeader,
+ defaultTopHeader,
+ floatStartHeader,
+ secondFloatStartHeader,
+ floatCenterHeader,
+ floatEndHeader,
+ secondFloatStartLeftHeader,
+ floatStartLeftHeader,
+ floatCenterLeftHeader,
+ floatEndLeftHeader,
+ defaultRightHeader,
+ secondFloatStartRightHeader,
+ floatStartRightHeader,
+ floatCenterRightHeader,
+ floatEndRightHeader,
+ defaultBottomHeader,
+ floatStartBottomHeader,
+ secondFloatStartBottomHeader,
+ floatCenterBottomHeader,
+ floatEndBottomHeader,
+ floatStarTopHeaderStatic,
+ floatCenterTopHeaderDynamic,
+ floatEndTopHeaderNone,
+ mockModularComponents,
+} from '../../Helpers/mockHeaders';
+
+export default {
+ title: 'ModularComponents/FloatingHeader/App',
+ component: App,
+};
+
+
+const noop = () => { };
+
+const MockApp = ({ initialState }) => {
+ return (
+
getDefaultMiddleware({ serializableCheck: false })
+ })}>
+
+
+ );
+};
+
+const Template = (args) => {
+ const stateWithHeaders = {
+ ...initialState,
+ viewer: {
+ ...initialState.viewer,
+ modularHeaders: args.headers,
+ modularComponents: mockModularComponents,
+ openElements: {},
+ },
+ featureFlags: {
+ customizableUI: true,
+ },
+ };
+ return
;
+};
+
+function createTemplate(headers) {
+ const template = Template.bind({});
+ template.args = { headers };
+ template.parameters = {
+ layout: 'fullscreen',
+ chromatic: { disableSnapshot: true }
+ };
+ return template;
+}
+
+export const TopAndLeftHeaders = createTemplate({
+ defaultLeftHeader,
+ secondFloatStartLeftHeader,
+ floatStartLeftHeader,
+ floatCenterLeftHeader,
+ floatEndLeftHeader,
+ defaultTopHeader,
+ floatStartHeader,
+ secondFloatStartHeader,
+ floatCenterHeader,
+ floatEndHeader,
+});
+
+export const TopCenterAndLeftHeaders = createTemplate({
+ defaultLeftHeader,
+ secondFloatStartLeftHeader,
+ floatStartLeftHeader,
+ floatCenterLeftHeader,
+ floatEndLeftHeader,
+ defaultTopHeader,
+ floatCenterHeader,
+ floatEndHeader,
+});
+
+export const TopAndRightHeaders = createTemplate({
+ defaultRightHeader,
+ secondFloatStartRightHeader,
+ floatStartRightHeader,
+ floatCenterRightHeader,
+ floatEndRightHeader,
+ defaultTopHeader,
+ floatStartHeader,
+ secondFloatStartHeader,
+ floatCenterHeader,
+ floatEndHeader,
+});
+
+export const TopCenterAndRightHeaders = createTemplate({
+ defaultRightHeader,
+ secondFloatStartRightHeader,
+ floatStartRightHeader,
+ floatCenterRightHeader,
+ floatEndRightHeader,
+ defaultTopHeader,
+ floatCenterHeader,
+});
+
+export const BottomAndLeftHeaders = createTemplate({
+ defaultLeftHeader,
+ secondFloatStartLeftHeader,
+ floatStartLeftHeader,
+ floatCenterLeftHeader,
+ floatEndLeftHeader,
+ defaultBottomHeader,
+ floatStartBottomHeader,
+ secondFloatStartBottomHeader,
+ floatCenterBottomHeader,
+ floatEndBottomHeader,
+});
+
+export const BottomCenterAndLeftHeaders = createTemplate({
+ defaultLeftHeader,
+ secondFloatStartLeftHeader,
+ floatStartLeftHeader,
+ floatCenterLeftHeader,
+ floatEndLeftHeader,
+ defaultBottomHeader,
+ floatCenterBottomHeader,
+ floatEndBottomHeader,
+});
+
+export const BottomAndRightHeaders = createTemplate({
+ defaultRightHeader,
+ secondFloatStartRightHeader,
+ floatStartRightHeader,
+ floatCenterRightHeader,
+ floatEndRightHeader,
+ defaultBottomHeader,
+ floatStartBottomHeader,
+ secondFloatStartBottomHeader,
+ floatCenterBottomHeader,
+ floatEndBottomHeader,
+});
+
+export const BottomCenterAndRightHeaders = createTemplate({
+ defaultRightHeader,
+ secondFloatStartRightHeader,
+ floatStartRightHeader,
+ floatCenterRightHeader,
+ floatEndRightHeader,
+ defaultBottomHeader,
+ floatCenterBottomHeader,
+});
+
+export const TopAndBottomHeaders = createTemplate({
+ defaultBottomHeader,
+ floatStartBottomHeader,
+ secondFloatStartBottomHeader,
+ floatCenterBottomHeader,
+ floatEndBottomHeader,
+ defaultTopHeader,
+ floatStartHeader,
+ secondFloatStartHeader,
+ floatCenterHeader,
+ floatEndHeader,
+});
+
+export const FloatingOnAllSides = createTemplate({
+ floatStartHeader,
+ secondFloatStartHeader,
+ floatCenterHeader,
+ floatEndHeader,
+ floatStartLeftHeader,
+ secondFloatStartLeftHeader,
+ floatCenterLeftHeader,
+ floatEndLeftHeader,
+ floatStartRightHeader,
+ secondFloatStartRightHeader,
+ floatCenterRightHeader,
+ floatEndRightHeader,
+ floatStartBottomHeader,
+ secondFloatStartBottomHeader,
+ floatCenterBottomHeader,
+ floatEndBottomHeader,
+});
+
+export const FloatiesWithOpacityLevels = createTemplate({
+ floatStarTopHeaderStatic,
+ floatCenterTopHeaderDynamic,
+ floatEndTopHeaderNone,
+});
+
+export const AllHeaders = createTemplate({
+ defaultTopHeader,
+ floatStartHeader,
+ secondFloatStartHeader,
+ floatCenterHeader,
+ floatEndHeader,
+ defaultLeftHeader,
+ floatStartLeftHeader,
+ secondFloatStartLeftHeader,
+ floatCenterLeftHeader,
+ floatEndLeftHeader,
+ defaultRightHeader,
+ floatStartRightHeader,
+ secondFloatStartRightHeader,
+ floatCenterRightHeader,
+ floatEndRightHeader,
+ defaultBottomHeader,
+ floatStartBottomHeader,
+ secondFloatStartBottomHeader,
+ floatCenterBottomHeader,
+ floatEndBottomHeader,
+});
\ No newline at end of file
diff --git a/src/components/ModularComponents/FloatingHeader/index.js b/src/components/ModularComponents/FloatingHeader/index.js
new file mode 100644
index 0000000000..6f3f435969
--- /dev/null
+++ b/src/components/ModularComponents/FloatingHeader/index.js
@@ -0,0 +1,3 @@
+import FloatingHeaderContainer from './FloatingHeaderContainer';
+
+export default FloatingHeaderContainer;
\ No newline at end of file
diff --git a/src/components/ModularComponents/GenericOutlinesPanel/GenericOutlinesPanel.stories.js b/src/components/ModularComponents/GenericOutlinesPanel/GenericOutlinesPanel.stories.js
index 974a922073..7328dccd98 100644
--- a/src/components/ModularComponents/GenericOutlinesPanel/GenericOutlinesPanel.stories.js
+++ b/src/components/ModularComponents/GenericOutlinesPanel/GenericOutlinesPanel.stories.js
@@ -30,7 +30,7 @@ export const Editable = () => {
},
panelWidths: { 'outlines-panel': DEFAULT_NOTES_PANEL_WIDTH },
isInDesktopOnlyMode: true,
- modularHeaders: []
+ modularHeaders: {}
},
document: {
outlines: getDefaultOutlines(),
@@ -63,7 +63,7 @@ export const NonEditable = () => {
},
panelWidths: { 'outlines-panel': DEFAULT_NOTES_PANEL_WIDTH },
isInDesktopOnlyMode: true,
- modularHeaders: []
+ modularHeaders: {}
},
document: {
outlines: getDefaultOutlines(),
@@ -98,7 +98,7 @@ export const Expanded = () => {
},
panelWidths: { 'outlines-panel': DEFAULT_NOTES_PANEL_WIDTH },
isInDesktopOnlyMode: true,
- modularHeaders: []
+ modularHeaders: {}
},
document: {
outlines: getDefaultOutlines(),
@@ -132,7 +132,7 @@ export const NoOutlines = () => {
},
panelWidths: { 'outlines-panel': DEFAULT_NOTES_PANEL_WIDTH },
isInDesktopOnlyMode: true,
- modularHeaders: []
+ modularHeaders: {}
},
document: {
outlines: [],
diff --git a/src/components/ModularComponents/GroupedItems/GroupedItems.scss b/src/components/ModularComponents/GroupedItems/GroupedItems.scss
new file mode 100644
index 0000000000..e68163fef2
--- /dev/null
+++ b/src/components/ModularComponents/GroupedItems/GroupedItems.scss
@@ -0,0 +1,3 @@
+.GroupedItems {
+ display: flex;
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/GroupedItems/index.js b/src/components/ModularComponents/GroupedItems/index.js
new file mode 100644
index 0000000000..bf4e4bb82f
--- /dev/null
+++ b/src/components/ModularComponents/GroupedItems/index.js
@@ -0,0 +1,3 @@
+import GroupedItems from './GroupedItems';
+
+export default GroupedItems;
\ No newline at end of file
diff --git a/src/components/ModularComponents/Helpers/menuItems.js b/src/components/ModularComponents/Helpers/menuItems.js
new file mode 100644
index 0000000000..7120298b93
--- /dev/null
+++ b/src/components/ModularComponents/Helpers/menuItems.js
@@ -0,0 +1,129 @@
+import React from 'react';
+import DataElements from 'constants/dataElement';
+import { PRESET_BUTTON_TYPES } from 'constants/customizationVariables';
+import ActionButton from 'components/ActionButton';
+
+export const menuItems = {
+ [PRESET_BUTTON_TYPES.UNDO]: {
+ dataElement: 'undoButton',
+ presetDataElement: DataElements.UNDO_PRESET_BUTTON,
+ icon: 'icon-operation-undo',
+ label: 'action.undo',
+ title: 'action.undo',
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.REDO]: {
+ dataElement: 'redoButton',
+ presetDataElement: DataElements.REDO_PRESET_BUTTON,
+ icon: 'icon-operation-redo',
+ label: 'action.redo',
+ title: 'action.redo',
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.FORM_FIELD_EDIT]: {
+ dataElement: 'formFieldEditButton',
+ presetDataElement: DataElements.FORM_FIELD_EDIT_PRESET_BUTTON,
+ icon: 'ic-fill-and-sign',
+ label: 'action.formFieldEditMode',
+ title: 'action.formFieldEditMode',
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.CONTENT_EDIT]: {
+ dataElement: 'contentEditButton',
+ presetDataElement: DataElements.CONTENT_EDIT_PRESET_BUTTON,
+ icon: 'icon-content-edit',
+ label: 'action.contentEditMode',
+ title: 'action.contentEditMode',
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.NEW_DOCUMENT]: {
+ dataElement: DataElements.NEW_DOCUMENT_BUTTON,
+ presetDataElement: DataElements.NEW_DOCUMENT_PRESET_BUTTON,
+ icon: 'icon-plus-sign',
+ label: 'action.newDocument',
+ title: 'action.newDocument',
+ isActive: false,
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.FILE_PICKER]: {
+ dataElement: DataElements.FILE_PICKER_BUTTON,
+ presetDataElement: DataElements.FILE_PICKER_PRESET_BUTTON,
+ icon: 'icon-header-file-picker-line',
+ label: 'action.openFile',
+ title: 'action.openFile',
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.DOWNLOAD]: {
+ dataElement: DataElements.DOWNLOAD_BUTTON,
+ presetDataElement: DataElements.DOWNLOAD_PRESET_BUTTON,
+ icon: 'icon-download',
+ label: 'action.download',
+ title: 'action.download',
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.SAVE_AS]: {
+ dataElement: DataElements.SAVE_AS_BUTTON,
+ presetDataElement: DataElements.SAVE_AS_PRESET_BUTTON,
+ icon: 'icon-save',
+ label: 'saveModal.saveAs',
+ title: 'saveModal.saveAs',
+ isActive: false,
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.PRINT]: {
+ dataElement: DataElements.PRINT_BUTTON,
+ presetDataElement: DataElements.PRINT_PRESET_BUTTON,
+ icon: 'icon-header-print-line',
+ label: 'action.print',
+ title: 'action.print',
+ isActive: false,
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.CREATE_PORTFOLIO]: {
+ dataElement: DataElements.CREATE_PORTFOLIO_BUTTON,
+ presetDataElement: DataElements.CREATE_PORTFOLIO_PRESET_BUTTON,
+ icon: 'icon-pdf-portfolio',
+ label: 'portfolio.createPDFPortfolio',
+ title: 'portfolio.createPDFPortfolio',
+ isActive: false,
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.SETTINGS]: {
+ dataElement: DataElements.SETTINGS_BUTTON,
+ presetDataElement: DataElements.SETTINGS_PRESET_BUTTON,
+ icon: 'icon-header-settings-line',
+ label: 'option.settings.settings',
+ title: 'option.settings.settings',
+ isActive: false,
+ hidden: false,
+ },
+ [PRESET_BUTTON_TYPES.FULLSCREEN]: {
+ dataElement: DataElements.FULLSCREEN_BUTTON,
+ presetDataElement: DataElements.FULLSCREEN_PRESET_BUTTON,
+ icon: 'icon-header-full-screen',
+ label: 'action.enterFullscreen',
+ title: 'action.enterFullscreen',
+ hidden: false,
+ },
+};
+
+export const getPresetButtonDOM = (buttonType, isDisabled, onClick, isFullScreen) => {
+ const { dataElement, presetDataElement } = menuItems[buttonType];
+ let { icon, title } = menuItems[buttonType];
+
+ if (buttonType === PRESET_BUTTON_TYPES.FULLSCREEN) {
+ icon = isFullScreen ? 'icon-header-full-screen-exit' : 'icon-header-full-screen';
+ title = isFullScreen ? 'action.exitFullscreen' : 'action.enterFullscreen';
+ }
+
+ return (
+
+ );
+};
\ No newline at end of file
diff --git a/src/components/ModularComponents/Helpers/mockHeaders.js b/src/components/ModularComponents/Helpers/mockHeaders.js
new file mode 100644
index 0000000000..4ff660e6e1
--- /dev/null
+++ b/src/components/ModularComponents/Helpers/mockHeaders.js
@@ -0,0 +1,301 @@
+/* eslint-disable no-alert */
+const baseButton = {
+ dataElement: 'button',
+ onClick: () => alert('Added'),
+ disabled: false,
+ title: 'Button 1',
+ label: 'Add',
+ type: 'customButton'
+};
+
+const divider = {
+ type: 'divider',
+ dataElement: 'divider-1',
+};
+
+const leftPanelToggle = {
+ dataElement: 'leftPanelToggle',
+ toggleElement: 'leftPanel',
+ disabled: false,
+ title: 'Left Panel',
+ img: 'icon-header-sidebar-line',
+ type: 'toggleButton',
+};
+
+const notesPanelToggle = {
+ dataElement: 'notesPanelToggle',
+ toggleElement: 'notesPanel',
+ disabled: false,
+ title: 'Notes Panel',
+ img: 'icon-header-chat-line',
+ type: 'toggleButton',
+};
+
+// Handy mock buttons
+const button1 = { ...baseButton, dataElement: 'button1', label: 'Button 1' };
+const button2 = { ...baseButton, dataElement: 'button2', label: 'Button 2' };
+const button3 = { ...baseButton, dataElement: 'button3', label: 'Button 3' };
+const button4 = { ...baseButton, dataElement: 'button4', label: 'Button 4' };
+const button5 = { ...baseButton, dataElement: 'button5', label: 'Button 5' };
+const button6 = { ...baseButton, dataElement: 'button6', label: 'Button 6' };
+const button7 = { ...baseButton, dataElement: 'button7', label: 'Button 7' };
+const button8 = { ...baseButton, dataElement: 'button8', label: 'Button 8' };
+const button9 = { ...baseButton, dataElement: 'button9', label: 'Button 9' };
+
+// These are our headers
+const defaultTopHeader = {
+ dataElement: 'defaultHeader',
+ placement: 'top',
+ gap: 20,
+ items: ['button1', 'button2', 'divider', 'button3'],
+};
+
+const floatStartHeader = {
+ dataElement: 'floatStartHeader',
+ placement: 'top',
+ float: true,
+ position: 'start',
+ items: ['button1', 'button2'],
+ gap: 20
+};
+
+const secondFloatStartHeader = {
+ dataElement: 'floatStartHeader-2',
+ placement: 'top',
+ float: true,
+ position: 'start',
+ items: ['button3', 'button4'],
+ gap: 20
+};
+
+const floatCenterHeader = {
+ dataElement: 'floatCenterHeader',
+ placement: 'top',
+ float: true,
+ position: 'center',
+ items: ['button5', 'divider', 'button6'],
+ gap: 20
+};
+
+const floatEndHeader = {
+ dataElement: 'floatEndHeader',
+ placement: 'top',
+ float: true,
+ position: 'end',
+ items: ['button7', 'divider', 'button8', 'button9'],
+ gap: 20
+};
+
+const defaultLeftHeader = {
+ dataElement: 'defaultHeader',
+ placement: 'left',
+ gap: 20,
+ items: ['button1', 'button2', 'divider', 'button3'],
+};
+
+const floatStartLeftHeader = {
+ dataElement: 'floatStartLeftHeader',
+ placement: 'left',
+ float: true,
+ position: 'start',
+ items: ['button3', 'button4', 'leftPanelToggle'],
+ gap: 20
+};
+
+const secondFloatStartLeftHeader = {
+ dataElement: 'secondFloatLeftBottomHeader',
+ placement: 'left',
+ float: true,
+ position: 'start',
+ items: ['button5', 'button6'],
+ gap: 20
+};
+
+const floatCenterLeftHeader = {
+ dataElement: 'floatCenterLeftHeader',
+ placement: 'left',
+ float: true,
+ position: 'center',
+ items: ['button1', 'button2'],
+ gap: 20
+};
+
+const floatEndLeftHeader = {
+ dataElement: 'floatEndLeftHeader',
+ placement: 'left',
+ float: true,
+ position: 'end',
+ items: ['button7', 'button8', 'divider', 'button9'],
+ gap: 20
+};
+
+const defaultRightHeader = {
+ dataElement: 'defaultHeader',
+ placement: 'right',
+ gap: 20,
+ items: ['button1', 'button2', 'divider', 'button3'],
+};
+
+const floatStartRightHeader = {
+ dataElement: 'floatStartRightHeader',
+ placement: 'right',
+ float: true,
+ position: 'start',
+ items: ['button3', 'button4', 'notesPanelToggle'],
+ gap: 20
+};
+
+const secondFloatStartRightHeader = {
+ dataElement: 'secondFloatRightBottomHeader',
+ placement: 'right',
+ float: true,
+ position: 'start',
+ items: ['button5', 'button6'],
+ gap: 20
+};
+
+const floatCenterRightHeader = {
+ dataElement: 'floatCenterRightHeader',
+ placement: 'right',
+ float: true,
+ position: 'center',
+ items: ['button1', 'button2'],
+ gap: 20
+};
+
+const floatEndRightHeader = {
+ dataElement: 'floatEndRightHeader',
+ placement: 'right',
+ float: true,
+ position: 'end',
+ items: ['button7', 'button8', 'divider', 'button9'],
+ gap: 20
+};
+
+const defaultBottomHeader = {
+ dataElement: 'defaultBottomHeader',
+ placement: 'bottom',
+ gap: 20,
+ items: ['button1', 'button2', 'divider', 'button3'],
+ getDimensionTotal: () => {
+ return 32;
+ }
+};
+
+const floatStartBottomHeader = {
+ dataElement: 'floatStartBottomHeader',
+ placement: 'bottom',
+ float: true,
+ position: 'start',
+ items: ['button3', 'button4'],
+ gap: 20
+};
+
+const secondFloatStartBottomHeader = {
+ dataElement: 'secondFloatStartBottomHeader',
+ placement: 'bottom',
+ float: true,
+ position: 'start',
+ items: ['button5', 'button6'],
+ gap: 20
+};
+
+const floatCenterBottomHeader = {
+ dataElement: 'floatCenterBottomHeader',
+ placement: 'bottom',
+ float: true,
+ position: 'center',
+ items: ['button1', 'button2'],
+ gap: 20
+};
+
+const floatEndBottomHeader = {
+ dataElement: 'floatEndBottomHeader',
+ placement: 'bottom',
+ float: true,
+ position: 'end',
+ items: ['button7', 'button8', 'divider', 'button9'],
+ gap: 20
+};
+
+const floatStarTopHeaderStatic = {
+ dataElement: 'floatStarTopHeaderStatic',
+ placement: 'top',
+ float: true,
+ position: 'start',
+ items: ['button1', 'button2'],
+ opacityMode: 'static',
+ opacity: 'full',
+};
+
+const floatCenterTopHeaderDynamic = {
+ dataElement: 'floatStarTopHeaderDynamic',
+ placement: 'top',
+ float: true,
+ position: 'center',
+ items: ['button1', 'button2'],
+ opacityMode: 'dynamic',
+ opacity: 'low',
+};
+
+const floatEndTopHeaderNone = {
+ dataElement: 'floatStarTopHeaderNone',
+ placement: 'top',
+ float: true,
+ position: 'end',
+ items: ['button1', 'button2'],
+ opacityMode: 'dynamic',
+ opacity: 'none',
+};
+
+const mockModularComponents = {
+ 'button1': button1,
+ 'button2': button2,
+ 'button3': button3,
+ 'button4': button4,
+ 'button5': button5,
+ 'button6': button6,
+ 'button7': button7,
+ 'button8': button8,
+ 'button9': button9,
+ 'divider': divider,
+ 'leftPanelToggle': leftPanelToggle,
+ 'notesPanelToggle': notesPanelToggle,
+};
+
+export {
+ button1,
+ button2,
+ button3,
+ button4,
+ button5,
+ button6,
+ button7,
+ button8,
+ button9,
+ defaultTopHeader,
+ floatStartHeader,
+ secondFloatStartHeader,
+ floatCenterHeader,
+ floatEndHeader,
+ defaultLeftHeader,
+ floatStartLeftHeader,
+ secondFloatStartLeftHeader,
+ floatCenterLeftHeader,
+ floatEndLeftHeader,
+ defaultRightHeader,
+ floatStartRightHeader,
+ secondFloatStartRightHeader,
+ floatCenterRightHeader,
+ floatEndRightHeader,
+ defaultBottomHeader,
+ floatStartBottomHeader,
+ secondFloatStartBottomHeader,
+ floatCenterBottomHeader,
+ floatEndBottomHeader,
+ floatStarTopHeaderStatic,
+ floatCenterTopHeaderDynamic,
+ floatEndTopHeaderNone,
+ // Modular stuff
+ mockModularComponents,
+};
\ No newline at end of file
diff --git a/src/components/ModularComponents/Helpers/validation-helper.js b/src/components/ModularComponents/Helpers/validation-helper.js
new file mode 100644
index 0000000000..44c9e533f9
--- /dev/null
+++ b/src/components/ModularComponents/Helpers/validation-helper.js
@@ -0,0 +1,26 @@
+import { JUSTIFY_CONTENT } from 'constants/customizationVariables';
+
+export const isJustifyContentValid = (justifyContent) => {
+ const validJustifications = Object.values(JUSTIFY_CONTENT);
+ if (!validJustifications.includes(justifyContent)) {
+ console.warn(`${justifyContent} is not a valid value for justifyContent. Please use one of the following: ${validJustifications}`);
+ return false;
+ }
+ return true;
+};
+
+export const isGapValid = (gap) => {
+ if (isNaN(gap) || gap < 0) {
+ console.warn(`${gap} is not a valid value for gap. Please use a number, which represents the gap between items in pixels.`);
+ return false;
+ }
+ return true;
+};
+
+export const isGrowValid = (grow) => {
+ if (isNaN(grow) || grow < 0) {
+ console.warn(`${grow} is not a valid value for grow. Please use a number, which represents the flex-grow property of item.`);
+ return false;
+ }
+ return true;
+};
\ No newline at end of file
diff --git a/src/components/ModularComponents/InnerItem/index.js b/src/components/ModularComponents/InnerItem/index.js
new file mode 100644
index 0000000000..9549d3edc3
--- /dev/null
+++ b/src/components/ModularComponents/InnerItem/index.js
@@ -0,0 +1,3 @@
+import InnerItem from './InnerItem';
+
+export default InnerItem;
\ No newline at end of file
diff --git a/src/components/ModularComponents/LeftHeader/LeftHeader.scss b/src/components/ModularComponents/LeftHeader/LeftHeader.scss
new file mode 100644
index 0000000000..313ddc7cfe
--- /dev/null
+++ b/src/components/ModularComponents/LeftHeader/LeftHeader.scss
@@ -0,0 +1,20 @@
+@import '../../../constants/styles.scss';
+
+.LeftHeader {
+ height: 100%;
+ width: $left-header-width;
+ min-width: fit-content;
+ padding: 12px 8px;
+}
+
+.LeftHeader.closed {
+ position: fixed;
+ left: 0;
+ display: none;
+}
+
+.LeftHeader.stroke {
+ border-right-width: 1px;
+ border-right-style: solid;
+ border-right-color: var(--gray-5);
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/LeftHeader/index.js b/src/components/ModularComponents/LeftHeader/index.js
new file mode 100644
index 0000000000..aa454c14b8
--- /dev/null
+++ b/src/components/ModularComponents/LeftHeader/index.js
@@ -0,0 +1,3 @@
+import LeftHeader from './LeftHeaderContainer';
+
+export default LeftHeader;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PageControls/PageControls.js b/src/components/ModularComponents/PageControls/PageControls.js
new file mode 100644
index 0000000000..c2f0f1a6dd
--- /dev/null
+++ b/src/components/ModularComponents/PageControls/PageControls.js
@@ -0,0 +1,119 @@
+import React from 'react';
+import PropTypes from 'prop-types';
+import classNames from 'classnames';
+import CustomButton from '../CustomButton';
+import ToggleElementButton from '../ToggleElementButton';
+import { isMobileSize } from 'helpers/getDeviceSize';
+import { DIRECTION } from 'constants/customizationVariables';
+import { useTranslation } from 'react-i18next';
+import './PageControls.scss';
+
+function PageControls(props) {
+ const {
+ size,
+ dataElement,
+ onFlyoutToggle,
+ leftChevron,
+ rightChevron,
+ currentPage,
+ totalPages,
+ elementRef,
+ headerDirection,
+ onBlur,
+ onFocus,
+ onClick,
+ onChange,
+ onSubmit,
+ isFocused,
+ input,
+ inputRef,
+ allowPageNavigation,
+ } = props;
+
+ const { t } = useTranslation();
+ const isMobile = isMobileSize();
+ let inputWidth = 0;
+ if (input) {
+ inputWidth = 26 + input.length * (isMobile ? 10 : 7);
+ }
+
+ const style = { width: inputWidth };
+ if (headerDirection === DIRECTION.COLUMN) {
+ style.minHeight = 32;
+ }
+
+ return (
+
+ {size === 0 && <>
+
+
+
+
{totalPages}
+
+ >}
+ {size === 1 &&
+
+ }
+
+
+
+
+ );
+}
+
+PageControls.propTypes = {
+ size: PropTypes.number,
+ dataElement: PropTypes.string.isRequired,
+ onFlyoutToggle: PropTypes.func,
+ leftChevron: PropTypes.object,
+ rightChevron: PropTypes.object,
+ currentPage: PropTypes.number,
+ totalPages: PropTypes.number,
+ elementRef: PropTypes.any,
+ headerDirection: PropTypes.string,
+ onBlur: PropTypes.func,
+ onFocus: PropTypes.func,
+ onClick: PropTypes.func,
+ onChange: PropTypes.func,
+ onSubmit: PropTypes.func,
+ isFocused: PropTypes.bool,
+ input: PropTypes.string,
+ inputRef: PropTypes.any,
+ allowPageNavigation: PropTypes.bool,
+};
+
+export default PageControls;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PageControls/PageControls.scss b/src/components/ModularComponents/PageControls/PageControls.scss
new file mode 100644
index 0000000000..8da2251226
--- /dev/null
+++ b/src/components/ModularComponents/PageControls/PageControls.scss
@@ -0,0 +1,74 @@
+@import '../../../constants/styles';
+@import '../../../constants/overlay';
+
+.PageControlsWrapper {
+ display: flex;
+ gap: 8px;
+ flex-direction: row;
+ justify-content: flex-start;
+ flex-grow: 0;
+
+ .total-page {
+ margin: auto;
+ }
+
+ .paddingTop {
+ padding-top: 4px;
+ }
+
+ .paddingLeft {
+ padding-left: 4px;
+ }
+
+ form {
+ height: 100%;
+ }
+
+ form input {
+ height: 100%;
+ text-align: center;
+ font-family: Lato, sans-serif;
+ font-weight: 700;
+ background: var(--gray-0);
+ cursor: pointer;
+ outline: none;
+ border-radius: 3px;
+ min-width: 32px;
+ }
+}
+
+.pageNavFlyoutMenu {
+ .page-nav-display .flyout-item-label>* {
+ margin: auto;
+ }
+
+ form input {
+ font-size: 13px !important;
+ width: 17px;
+ font-family: Lato, sans-serif;
+ cursor: pointer;
+ outline: none;
+ margin: 0px 6px;
+ padding: 0;
+ text-align: center;
+ background: transparent;
+
+ min-width: 26px;
+ min-height: 22px;
+ }
+
+ form input:not(:focus) {
+ font-weight: 700;
+ border: none;
+ border-bottom: 1px solid var(--icon-color);
+ border-radius: 0;
+ }
+
+ form input:focus {
+ height: 100%;
+ background: var(--gray-0);
+ border-radius: 3px;
+ border: 1px solid var(--border);
+ min-height: 26px;
+ }
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/PageControls/PageControls.spec.js b/src/components/ModularComponents/PageControls/PageControls.spec.js
new file mode 100644
index 0000000000..63ac8dac5c
--- /dev/null
+++ b/src/components/ModularComponents/PageControls/PageControls.spec.js
@@ -0,0 +1,67 @@
+import React from 'react';
+import { render, screen, fireEvent } from '@testing-library/react';
+import userEvent from '@testing-library/user-event';
+import PageControls from './PageControls';
+import core from 'core';
+
+const PageControlWithRedux = withProviders(PageControls);
+
+const props = {
+ dataElement: 'page-controls-container',
+ size: 0,
+ leftChevron: {
+ dataElement: 'leftChevronBtn',
+ title: 'action.pagePrev',
+ label: null,
+ img: 'icon-chevron-up',
+ type: 'customButton',
+ disabled: false,
+ ariaLabel: 'action.pagePrev',
+ onClick: jest.fn(),
+ },
+ rightChevron: {
+ dataElement: 'rightChevronBtn',
+ title: 'action.pageNext',
+ label: null,
+ img: 'icon-chevron-right',
+ type: 'customButton',
+ disabled: false,
+ ariaLabel: 'action.pageNext',
+ onClick: jest.fn(),
+ },
+ input: '7',
+ totalPages: 11,
+ onChange: jest.fn(),
+};
+
+describe('Page Controls Container component', () => {
+ beforeEach(() => {
+ const documentViewer = core.setDocumentViewer(1, new window.Core.DocumentViewer());
+ documentViewer.doc = new window.Core.Document('dummy', 'pdf');
+ });
+
+ it('Should be able to find input and check input value', () => {
+ render(
);
+ const input = screen.getByRole('textbox');
+ expect(input.value).toEqual(props.input);
+ });
+
+ it('Should be able to type into input of Page Controls', () => {
+ render(
);
+ const input = screen.getByRole('textbox');
+ userEvent.type(input, '8');
+ expect(input.value).toEqual(props.input);
+ });
+
+ it('Should call onClick on left/right button on Page Controls component', () => {
+ render(
);
+ const leftBtn = screen.getByRole('button', { name: 'action.pagePrev' });
+ const rightBtn = screen.getByRole('button', { name: 'action.pageNext' });
+ expect(leftBtn).toBeInTheDocument();
+ expect(rightBtn).toBeInTheDocument();
+ fireEvent.click(leftBtn);
+ fireEvent.click(rightBtn);
+ expect(props.leftChevron.onClick).toHaveBeenCalledTimes(1);
+ expect(props.rightChevron.onClick).toHaveBeenCalledTimes(1);
+ });
+});
diff --git a/src/components/ModularComponents/PageControls/PageControlsFlyout.js b/src/components/ModularComponents/PageControls/PageControlsFlyout.js
new file mode 100644
index 0000000000..406e6ef3d1
--- /dev/null
+++ b/src/components/ModularComponents/PageControls/PageControlsFlyout.js
@@ -0,0 +1,57 @@
+import React, { useState } from 'react';
+import PropTypes from 'prop-types';
+
+function PageControlsFlyout(props) {
+ const {
+ onSubmit,
+ onChange,
+ input,
+ totalPages,
+ inputWidth,
+ } = props;
+ const [isFocused, setIsFocused] = useState(false);
+
+ const onBlur = () => {
+ setIsFocused(false);
+ props.onBlur();
+ };
+
+ const onFocus = () => {
+ setIsFocused(true);
+ props.onFocus();
+ };
+
+ const style = {};
+ if (isFocused) {
+ style.width = inputWidth;
+ } else {
+ style.width = inputWidth - 10;
+ }
+
+ return (
+
+
{'Pages: '}
+
+
{` of ${totalPages}`}
+
+ );
+}
+
+PageControlsFlyout.propTypes = {
+ onSubmit: PropTypes.func,
+ onChange: PropTypes.func,
+ onFocus: PropTypes.func,
+ onBlur: PropTypes.func,
+ input: PropTypes.string,
+ totalPages: PropTypes.number,
+ inputWidth: PropTypes.number,
+};
+
+export default PageControlsFlyout;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PageControls/index.js b/src/components/ModularComponents/PageControls/index.js
new file mode 100644
index 0000000000..df6986a39e
--- /dev/null
+++ b/src/components/ModularComponents/PageControls/index.js
@@ -0,0 +1,3 @@
+import PageControls from './PageControlsContainer';
+
+export default PageControls;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/PresetButton.js b/src/components/ModularComponents/PresetButton/PresetButton.js
new file mode 100644
index 0000000000..d0ca16cbd8
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/PresetButton.js
@@ -0,0 +1,56 @@
+import React from 'react';
+import PropTypes from 'prop-types';
+import { PRESET_BUTTON_TYPES } from 'constants/customizationVariables';
+import './PresetButton.scss';
+import NewDocumentButton from './buttons/NewDocument';
+import FilePickerButton from './buttons/FilePicker';
+import UndoButton from './buttons/Undo';
+import RedoButton from './buttons/Redo';
+import DownloadButton from './buttons/Download';
+import FullScreenButton from './buttons/FullScreen';
+import SaveAsButton from './buttons/SaveAs';
+import PrintButton from './buttons/Print';
+import NewPortfolioButton from './buttons/NewPortfolio';
+import SettingsButton from './buttons/Settings';
+import FormFieldEditButton from './buttons/FormFieldEdit';
+import ContentEditButton from './buttons/ContentEdit';
+
+const PresetButton = (props) => {
+ const { buttonType } = props;
+
+ switch (buttonType) {
+ case PRESET_BUTTON_TYPES.UNDO:
+ return
;
+ case PRESET_BUTTON_TYPES.REDO:
+ return
;
+ case PRESET_BUTTON_TYPES.NEW_DOCUMENT:
+ return
;
+ case PRESET_BUTTON_TYPES.FILE_PICKER:
+ return
;
+ case PRESET_BUTTON_TYPES.DOWNLOAD:
+ return
;
+ case PRESET_BUTTON_TYPES.FULLSCREEN:
+ return
;
+ case PRESET_BUTTON_TYPES.SAVE_AS:
+ return
;
+ case PRESET_BUTTON_TYPES.PRINT:
+ return
;
+ case PRESET_BUTTON_TYPES.CREATE_PORTFOLIO:
+ return
;
+ case PRESET_BUTTON_TYPES.SETTINGS:
+ return
;
+ case PRESET_BUTTON_TYPES.FORM_FIELD_EDIT:
+ return
;
+ case PRESET_BUTTON_TYPES.CONTENT_EDIT:
+ return
;
+ default:
+ console.warn(`${buttonType} is not a valid item type.`);
+ return null;
+ }
+};
+
+PresetButton.propTypes = {
+ buttonType: PropTypes.string.isRequired
+};
+
+export default PresetButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/PresetButton.scss b/src/components/ModularComponents/PresetButton/PresetButton.scss
new file mode 100644
index 0000000000..b0df6a3610
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/PresetButton.scss
@@ -0,0 +1,19 @@
+@import '../../../../src/constants/styles';
+
+.PresetButton {
+ &:hover {
+ background: var(--tools-button-hover);
+ }
+
+ &.formFieldEditButton, &.contentEditButton {
+ &.active {
+ background: var(--tools-button-active);
+ color: var(--view-header-icon-active-fill);
+ cursor: default;
+
+ .Icon {
+ color: var(--view-header-icon-active-fill)
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/PresetButton.stories.js b/src/components/ModularComponents/PresetButton/PresetButton.stories.js
new file mode 100644
index 0000000000..44494bd676
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/PresetButton.stories.js
@@ -0,0 +1,74 @@
+import React from 'react';
+import { Provider } from 'react-redux';
+import { configureStore } from '@reduxjs/toolkit';
+import initialState from 'src/redux/initialState';
+import PresetButton from './PresetButton';
+import { PRESET_BUTTON_TYPES } from 'src/constants/customizationVariables';
+
+export default {
+ title: 'ModularComponents/PresetButton',
+ component: PresetButton,
+};
+
+initialState.viewer.activeDocumentViewerKey = 1;
+const store = configureStore({ reducer: () => initialState });
+
+const prepareButtonStory = (buttonType) => {
+ const props = {
+ buttonType: buttonType,
+ };
+
+ return (
+
+
+
+ );
+};
+
+export function UndoButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.UNDO);
+}
+
+export function RedoButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.REDO);
+}
+
+export function NewDocumentButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.NEW_DOCUMENT);
+}
+
+export function FilePickerButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.FILE_PICKER);
+}
+
+export function DownloadButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.DOWNLOAD);
+}
+
+export function FullscreenButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.FULLSCREEN);
+}
+
+export function SaveAsButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.SAVE_AS);
+}
+
+export function PrintButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.PRINT);
+}
+
+export function CreatePortfolioButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.CREATE_PORTFOLIO);
+}
+
+export function SettingsButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.SETTINGS);
+}
+
+export function FormFieldEditButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.FORM_FIELD_EDIT);
+}
+
+export function ContentEditButton() {
+ return prepareButtonStory(PRESET_BUTTON_TYPES.CONTENT_EDIT);
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/buttons/FullScreen.js b/src/components/ModularComponents/PresetButton/buttons/FullScreen.js
new file mode 100644
index 0000000000..031f514502
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/buttons/FullScreen.js
@@ -0,0 +1,43 @@
+import { useSelector } from 'react-redux';
+import PropTypes from 'prop-types';
+import selectors from 'selectors';
+import { useTranslation } from 'react-i18next';
+import toggleFullscreen from 'helpers/toggleFullscreen';
+import { innerItemToFlyoutItem } from 'src/helpers/itemToFlyoutHelper';
+import { getPresetButtonDOM } from '../../Helpers/menuItems';
+import { PRESET_BUTTON_TYPES } from 'src/constants/customizationVariables';
+
+/**
+ * A button that toggles fullscreen mode.
+ * @name fullscreenButton
+ * @memberof UI.Components.PresetButton
+ */
+const FullScreenButton = (props) => {
+ const { isFlyoutItem, iconDOMElement } = props;
+ const { t } = useTranslation();
+ const [
+ isFullScreen,
+ ] = useSelector(
+ (state) => [
+ selectors.isFullScreen(state),
+ ],
+ );
+
+ return (
+ isFlyoutItem ?
+ innerItemToFlyoutItem({
+ isDisabled: false,
+ icon: iconDOMElement,
+ label: isFullScreen ? t('action.exitFullscreen') : t('action.enterFullscreen'),
+ }, toggleFullscreen)
+ :
+ getPresetButtonDOM(PRESET_BUTTON_TYPES.FULLSCREEN, false, toggleFullscreen, isFullScreen)
+ );
+};
+
+FullScreenButton.propTypes = {
+ isFlyoutItem: PropTypes.bool,
+ iconDOMElement: PropTypes.object,
+};
+
+export default FullScreenButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/buttons/Print.js b/src/components/ModularComponents/PresetButton/buttons/Print.js
new file mode 100644
index 0000000000..17530ff243
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/buttons/Print.js
@@ -0,0 +1,62 @@
+import { useDispatch, shallowEqual, useSelector } from 'react-redux';
+import PropTypes from 'prop-types';
+import { getPresetButtonDOM, menuItems } from '../../Helpers/menuItems';
+import { print } from 'helpers/print';
+import selectors from 'selectors';
+import core from 'core';
+import { innerItemToFlyoutItem } from 'helpers/itemToFlyoutHelper';
+import { useTranslation } from 'react-i18next';
+import { PRESET_BUTTON_TYPES } from 'constants/customizationVariables';
+
+/**
+ * A button that prints the document.
+ * @name printButton
+ * @memberof UI.Components.PresetButton
+ */
+const PrintButton = (props) => {
+ const { isFlyoutItem, iconDOMElement } = props;
+ const { label } = menuItems.printButton;
+ const dispatch = useDispatch();
+ const { t } = useTranslation();
+
+ const [
+ isEmbedPrintSupported,
+ sortStrategy,
+ colorMap,
+ timezone,
+ ] = useSelector(
+ (state) => [
+ selectors.isEmbedPrintSupported(state),
+ selectors.getSortStrategy(state),
+ selectors.getColorMap(state),
+ selectors.getTimezone(state),
+ ],
+ shallowEqual,
+ );
+
+ const handlePrintButtonClick = () => {
+ print(dispatch, isEmbedPrintSupported, sortStrategy, colorMap, { isGrayscale: core.getDocumentViewer().isGrayscaleModeEnabled(), timezone });
+ };
+
+ return (
+ isFlyoutItem ?
+ innerItemToFlyoutItem({
+ isDisabled: false,
+ icon: iconDOMElement,
+ label: t(label),
+ }, handlePrintButtonClick)
+ :
+ getPresetButtonDOM(
+ PRESET_BUTTON_TYPES.PRINT,
+ false,
+ handlePrintButtonClick
+ )
+ );
+};
+
+PrintButton.propTypes = {
+ isFlyoutItem: PropTypes.bool,
+ iconDOMElement: PropTypes.object,
+};
+
+export default PrintButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/buttons/Redo.js b/src/components/ModularComponents/PresetButton/buttons/Redo.js
new file mode 100644
index 0000000000..5c4b2b2103
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/buttons/Redo.js
@@ -0,0 +1,61 @@
+import React from 'react';
+import { useSelector } from 'react-redux';
+import { useTranslation } from 'react-i18next';
+import selectors from 'selectors';
+import PropTypes from 'prop-types';
+import ActionButton from 'components/ActionButton';
+import { menuItems } from '../../Helpers/menuItems';
+import core from 'core';
+
+/**
+ * A button that performs the redo action.
+ * @name redoButton
+ * @memberof UI.Components.PresetButton
+ */
+const RedoButton = (props) => {
+ const { isFlyoutItem, iconDOMElement } = props;
+ const { label, presetDataElement, icon, title } = menuItems.redoButton;
+ const activeDocumentViewerKey = useSelector((state) => selectors.getActiveDocumentViewerKey(state));
+ const { t } = useTranslation();
+
+ const handleClick = () => {
+ core.redo(activeDocumentViewerKey);
+ };
+
+ const onKeyDown = (e) => {
+ if (e.key === 'Enter') {
+ handleClick();
+ }
+ };
+
+ return (
+ isFlyoutItem ?
+ (
+
+
+ {iconDOMElement}
+ {label &&
{t(label)}
}
+
+
+ )
+ : (
+
!state.viewer.canRedo[state.viewer.activeDocumentViewerKey]}
+ />
+ )
+ );
+};
+
+RedoButton.propTypes = {
+ isFlyoutItem: PropTypes.bool,
+ iconDOMElement: PropTypes.object,
+};
+
+export default RedoButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/buttons/Settings.js b/src/components/ModularComponents/PresetButton/buttons/Settings.js
new file mode 100644
index 0000000000..005196fcb2
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/buttons/Settings.js
@@ -0,0 +1,42 @@
+import { useDispatch } from 'react-redux';
+import actions from 'actions';
+import PropTypes from 'prop-types';
+import { getPresetButtonDOM, menuItems } from '../../Helpers/menuItems';
+import DataElements from 'constants/dataElement';
+import { PRESET_BUTTON_TYPES } from 'constants/customizationVariables';
+import { innerItemToFlyoutItem } from 'helpers/itemToFlyoutHelper';
+import { useTranslation } from 'react-i18next';
+
+/**
+ * A button that opens the settings modal.
+ * @name settingsButton
+ * @memberof UI.Components.PresetButton
+ */
+const SettingsButton = (props) => {
+ const { isFlyoutItem, iconDOMElement } = props;
+ const { label } = menuItems.settingsButton;
+ const dispatch = useDispatch();
+ const { t } = useTranslation();
+
+ const handleSettingsButtonClick = () => {
+ dispatch(actions.openElement(DataElements.SETTINGS_MODAL));
+ };
+
+ return (
+ isFlyoutItem ?
+ innerItemToFlyoutItem({
+ isDisabled: false,
+ icon: iconDOMElement,
+ label: t(label),
+ }, handleSettingsButtonClick)
+ :
+ getPresetButtonDOM(PRESET_BUTTON_TYPES.SETTINGS, false, handleSettingsButtonClick)
+ );
+};
+
+SettingsButton.propTypes = {
+ isFlyoutItem: PropTypes.bool,
+ iconDOMElement: PropTypes.object,
+};
+
+export default SettingsButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/buttons/Undo.js b/src/components/ModularComponents/PresetButton/buttons/Undo.js
new file mode 100644
index 0000000000..07ca5b8200
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/buttons/Undo.js
@@ -0,0 +1,61 @@
+import React from 'react';
+import { useSelector } from 'react-redux';
+import { useTranslation } from 'react-i18next';
+import selectors from 'selectors';
+import PropTypes from 'prop-types';
+import ActionButton from 'components/ActionButton';
+import { menuItems } from '../../Helpers/menuItems';
+import core from 'core';
+
+/**
+ * A button that performs the undo action.
+ * @name undoButton
+ * @memberof UI.Components.PresetButton
+ */
+const UndoButton = (props) => {
+ const { isFlyoutItem, iconDOMElement } = props;
+ const { label, presetDataElement, icon, title } = menuItems.undoButton;
+ const activeDocumentViewerKey = useSelector((state) => selectors.getActiveDocumentViewerKey(state));
+ const { t } = useTranslation();
+
+ const handleClick = () => {
+ core.undo(activeDocumentViewerKey);
+ };
+
+ const onKeyDown = (e) => {
+ if (e.key === 'Enter') {
+ handleClick();
+ }
+ };
+
+ return (
+ isFlyoutItem ?
+ (
+
+
+ {iconDOMElement}
+ {label &&
{t(label)}
}
+
+
+ )
+ : (
+ !state.viewer.canUndo[state.viewer.activeDocumentViewerKey]}
+ />
+ )
+ );
+};
+
+UndoButton.propTypes = {
+ isFlyoutItem: PropTypes.bool,
+ iconDOMElement: PropTypes.object,
+};
+
+export default UndoButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/PresetButton/index.js b/src/components/ModularComponents/PresetButton/index.js
new file mode 100644
index 0000000000..204e7d4a6d
--- /dev/null
+++ b/src/components/ModularComponents/PresetButton/index.js
@@ -0,0 +1,3 @@
+import PresetButton from './PresetButton';
+
+export default PresetButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/RibbonGroup/index.js b/src/components/ModularComponents/RibbonGroup/index.js
new file mode 100644
index 0000000000..7301b1f53d
--- /dev/null
+++ b/src/components/ModularComponents/RibbonGroup/index.js
@@ -0,0 +1,3 @@
+import RibbonGroup from './RibbonGroup';
+
+export default RibbonGroup;
\ No newline at end of file
diff --git a/src/components/ModularComponents/RibbonItem/RibbonItem.scss b/src/components/ModularComponents/RibbonItem/RibbonItem.scss
new file mode 100644
index 0000000000..98e531dd7c
--- /dev/null
+++ b/src/components/ModularComponents/RibbonItem/RibbonItem.scss
@@ -0,0 +1,57 @@
+.RibbonItem {
+ padding: 0;
+ border: none;
+ background-color: transparent;
+ cursor: pointer;
+ color: var(--faded-text);
+ white-space: nowrap;
+ height: auto;
+
+ .Button {
+ width: 100%;
+ padding: 8px;
+ column-gap: 8px;
+ row-gap: 4px;
+ color: var(--faded-text);
+ &:hover {
+ background-color: var(--blue-1);
+ }
+ &.active {
+ color: var(--ribbon-active-color);
+ background-color: transparent;
+ .Icon {
+ color: var(--ribbon-active-color);
+ }
+ }
+ }
+
+ &:not(.vertical) .Button {
+ &.active {
+ border-bottom-left-radius: 0;
+ border-bottom-right-radius: 0;
+ border-bottom: 2px solid var(--ribbon-active-color);
+ }
+ }
+
+ &.vertical {
+ white-space: wrap;
+ .Button {
+ flex-direction: column;
+ height: 100%;
+ &.active {
+ border-bottom-left-radius: 0;
+ border-top-left-radius: 0;
+ border-left: 2px solid var(--ribbon-active-color);
+ }
+ }
+ }
+
+ &.vertical:not(.left) .Button {
+ &.active {
+ border-bottom-right-radius: 0;
+ border-top-right-radius: 0;
+ border-left: none;
+ border-right: 2px solid var(--ribbon-active-color);
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/RibbonItem/index.js b/src/components/ModularComponents/RibbonItem/index.js
new file mode 100644
index 0000000000..a44f180ef1
--- /dev/null
+++ b/src/components/ModularComponents/RibbonItem/index.js
@@ -0,0 +1,3 @@
+import RibbonItem from './RibbonItem';
+
+export default RibbonItem;
\ No newline at end of file
diff --git a/src/components/ModularComponents/RibbonOverflowFlyout/RibbonOverflowFlyout.js b/src/components/ModularComponents/RibbonOverflowFlyout/RibbonOverflowFlyout.js
new file mode 100644
index 0000000000..c21fce7d1f
--- /dev/null
+++ b/src/components/ModularComponents/RibbonOverflowFlyout/RibbonOverflowFlyout.js
@@ -0,0 +1,22 @@
+import { useEffect } from 'react';
+import { useDispatch } from 'react-redux';
+import actions from 'actions';
+import './RibbonOverflowFlyout.scss';
+
+const RibbonOverflowFlyout = (props) => {
+ const dispatch = useDispatch();
+ const { items } = props;
+
+ useEffect(() => {
+ const RibbonOverflowFlyout = {
+ dataElement: 'RibbonOverflowFlyout',
+ className: 'RibbonOverflowFlyout',
+ items: items || [],
+ };
+ dispatch(actions.addFlyout(RibbonOverflowFlyout));
+ }, []);
+
+ return null;
+};
+
+export default RibbonOverflowFlyout;
diff --git a/src/components/ModularComponents/RibbonOverflowFlyout/RibbonOverflowFlyout.scss b/src/components/ModularComponents/RibbonOverflowFlyout/RibbonOverflowFlyout.scss
new file mode 100644
index 0000000000..58f7019b2a
--- /dev/null
+++ b/src/components/ModularComponents/RibbonOverflowFlyout/RibbonOverflowFlyout.scss
@@ -0,0 +1,11 @@
+.Flyout .RibbonOverflowFlyout {
+ .RibbonItem {
+ width: 100%;
+ .Button {
+ justify-content: left;
+ &:hover {
+ background-color: transparent;
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/RibbonOverflowFlyout/index.js b/src/components/ModularComponents/RibbonOverflowFlyout/index.js
new file mode 100644
index 0000000000..cade412da3
--- /dev/null
+++ b/src/components/ModularComponents/RibbonOverflowFlyout/index.js
@@ -0,0 +1,3 @@
+import RibbonOverflowFlyout from './RibbonOverflowFlyout';
+
+export default RibbonOverflowFlyout;
\ No newline at end of file
diff --git a/src/components/ModularComponents/RightHeader/RightHeader.scss b/src/components/ModularComponents/RightHeader/RightHeader.scss
new file mode 100644
index 0000000000..ff7d1e55df
--- /dev/null
+++ b/src/components/ModularComponents/RightHeader/RightHeader.scss
@@ -0,0 +1,19 @@
+@import '../../../constants/styles';
+
+.RightHeader {
+ height: 100%;
+ width: $right-header-width;
+ min-width: fit-content;
+ z-index: 63;
+ padding: 12px 8px;
+}
+
+.RightHeader.closed {
+ display: none;
+}
+
+.RightHeader.stroke {
+ border-left-width: 1px;
+ border-left-style: solid;
+ border-left-color: var(--gray-5);
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/RightHeader/index.js b/src/components/ModularComponents/RightHeader/index.js
new file mode 100644
index 0000000000..48ec4457e1
--- /dev/null
+++ b/src/components/ModularComponents/RightHeader/index.js
@@ -0,0 +1,3 @@
+import RightHeader from './RightHeaderContainer';
+
+export default RightHeader;
\ No newline at end of file
diff --git a/src/components/ModularComponents/StatefulButton/StatefulButton.js b/src/components/ModularComponents/StatefulButton/StatefulButton.js
new file mode 100644
index 0000000000..0333abe22f
--- /dev/null
+++ b/src/components/ModularComponents/StatefulButton/StatefulButton.js
@@ -0,0 +1,91 @@
+import React, { useEffect } from 'react';
+import '../../Button/Button.scss';
+import './StatefulButton.scss';
+import classNames from 'classnames';
+import PropTypes from 'prop-types';
+import Button from 'components/Button';
+
+const StatefulButton = (props) => {
+ const { dataElement, disabled, mount, unmount, states } = props;
+ const [, updateState] = React.useState();
+ const forceUpdate = React.useCallback(() => updateState({}), []);
+
+ const [activeState, setActiveState] = React.useState(props.initialState);
+
+ useEffect(() => {
+ if (mount) {
+ mount(update);
+ }
+ return function() {
+ if (unmount) {
+ unmount();
+ }
+ };
+ });
+
+ const update = (newState) => {
+ if (newState) {
+ setActiveState(newState);
+ } else {
+ forceUpdate();
+ }
+ };
+
+ const onClick = () => {
+ const { dispatch } = props;
+
+ states[activeState].onClick(
+ update,
+ states[activeState],
+ dispatch,
+ );
+ };
+
+ const { title, img, getContent, isActive } = states[activeState];
+ const content = getContent ? getContent(states[activeState]) : '';
+ const className = [
+ 'StatefulButton',
+ states[activeState].className ? states[activeState].className : '',
+ ].join(' ').trim();
+
+ return (
+
+ );
+};
+
+StatefulButton.propTypes = {
+ initialState: PropTypes.string.isRequired,
+ mount: PropTypes.func.isRequired,
+ unmount: PropTypes.func,
+ states: PropTypes.shape({
+ activeState: PropTypes.shape({
+ img: PropTypes.string,
+ label: PropTypes.string,
+ onClick: PropTypes.func.isRequired,
+ title: PropTypes.string.isRequired,
+ getContent: PropTypes.func.isRequired,
+ }),
+ AnotherState: PropTypes.shape({
+ img: PropTypes.string,
+ label: PropTypes.string,
+ onClick: PropTypes.func.isRequired,
+ title: PropTypes.string.isRequired,
+ getContent: PropTypes.func.isRequired,
+ }),
+ }),
+};
+
+export default React.memo(StatefulButton);
diff --git a/src/components/ModularComponents/StatefulButton/StatefulButton.scss b/src/components/ModularComponents/StatefulButton/StatefulButton.scss
new file mode 100644
index 0000000000..1700e53217
--- /dev/null
+++ b/src/components/ModularComponents/StatefulButton/StatefulButton.scss
@@ -0,0 +1,7 @@
+@import '../../../constants/styles';
+
+.StatefulButton {
+ padding: 5px;
+ width: fit-content;
+ background-color: var(--gray-4);
+}
\ No newline at end of file
diff --git a/src/components/ModularComponents/StatefulButton/StatefulButton.stories.js b/src/components/ModularComponents/StatefulButton/StatefulButton.stories.js
new file mode 100644
index 0000000000..b628d65961
--- /dev/null
+++ b/src/components/ModularComponents/StatefulButton/StatefulButton.stories.js
@@ -0,0 +1,75 @@
+import React from 'react';
+import { createStore } from 'redux';
+import { Provider } from 'react-redux';
+import StatefulButtonComponent from './StatefulButton';
+
+const initialState = {
+ viewer: {
+ disabledElements: {},
+ customElementOverrides: {},
+ openElements: [],
+ }
+};
+function rootReducer(state = initialState) {
+ return state;
+}
+
+const store = createStore(rootReducer);
+
+const BasicComponent = (props) => {
+ return (
+
+
+
+ );
+};
+
+export default {
+ title: 'ModularComponents/StatefulButton',
+ component: StatefulButtonComponent,
+};
+
+
+export const StatefulButtonCounter = BasicComponent.bind({});
+StatefulButtonCounter.args = {
+ type: 'statefulButton',
+ dataElement: 'countButton',
+ initialState: 'Count',
+ states: {
+ Count: {
+ number: 3,
+ getContent: (activeState) => {
+ return activeState.number;
+ },
+ onClick: (update, activeState) => {
+ activeState.number += 1;
+ update();
+ }
+ }
+ },
+ mount: () => {},
+};
+
+export const StatefulButtonStates = BasicComponent.bind({});
+StatefulButtonStates.args = {
+ type: 'statefulButton',
+ dataElement: 'singlePageBtn',
+ initialState: 'SinglePage',
+ states: {
+ SinglePage: {
+ img: 'icon-header-page-manipulation-page-layout-single-page-line',
+ onClick: (update) => {
+ update('DoublePage');
+ },
+ title: 'Single Page',
+ },
+ DoublePage: {
+ img: 'icon-header-page-manipulation-page-layout-double-page-line',
+ onClick: (update) => {
+ update('SinglePage');
+ },
+ title: 'Single Page',
+ },
+ },
+ mount: () => {},
+};
diff --git a/src/components/ModularComponents/StatefulButton/index.js b/src/components/ModularComponents/StatefulButton/index.js
new file mode 100644
index 0000000000..a340a837d2
--- /dev/null
+++ b/src/components/ModularComponents/StatefulButton/index.js
@@ -0,0 +1,3 @@
+import StatefulButton from './StatefulButton';
+
+export default StatefulButton;
\ No newline at end of file
diff --git a/src/components/ModularComponents/TabPanel/index.js b/src/components/ModularComponents/TabPanel/index.js
new file mode 100644
index 0000000000..b5620b9e57
--- /dev/null
+++ b/src/components/ModularComponents/TabPanel/index.js
@@ -0,0 +1,3 @@
+import TabPanel from './TabPanel';
+
+export default TabPanel;
\ No newline at end of file
diff --git a/src/components/ModularComponents/TopHeader/index.js b/src/components/ModularComponents/TopHeader/index.js
new file mode 100644
index 0000000000..b3e5f5206d
--- /dev/null
+++ b/src/components/ModularComponents/TopHeader/index.js
@@ -0,0 +1,3 @@
+import TopHeader from './TopHeaderContainer';
+
+export default TopHeader;
\ No newline at end of file
diff --git a/src/components/ModularComponents/ViewControls/ViewControlsFlyout.js b/src/components/ModularComponents/ViewControls/ViewControlsFlyout.js
new file mode 100644
index 0000000000..97252b4007
--- /dev/null
+++ b/src/components/ModularComponents/ViewControls/ViewControlsFlyout.js
@@ -0,0 +1,227 @@
+import displayModeObjects from 'constants/displayModeObjects';
+import core from 'core';
+import { useLayoutEffect } from 'react';
+import { useSelector, useStore, useDispatch } from 'react-redux';
+import selectors from 'selectors';
+import { enterReaderMode, exitReaderMode } from 'helpers/readerMode';
+import actions from 'actions';
+import toggleFullscreen from 'helpers/toggleFullscreen';
+import { isIE11, isIOS, isIOSFullScreenSupported } from 'helpers/device';
+import DataElements from 'constants/dataElement';
+
+const ViewControlsFlyout = () => {
+ const store = useStore();
+ const dispatch = useDispatch();
+ const [
+ totalPages,
+ displayMode,
+ isDisabled,
+ isReaderMode,
+ isMultiViewerMode,
+ isFullScreen,
+ activeDocumentViewerKey,
+ isMultiTab,
+ isMultiViewerModeAvailable,
+ currentFlyout
+ ] = useSelector((state) => [
+ selectors.getTotalPages(state),
+ selectors.getDisplayMode(state),
+ selectors.isElementDisabled(state, 'viewControlsFlyout'),
+ selectors.isReaderMode(state),
+ selectors.isMultiViewerMode(state),
+ selectors.isFullScreen(state),
+ selectors.getActiveDocumentViewerKey(state),
+ selectors.getIsMultiTab(state),
+ selectors.getIsMultiViewerModeAvailable(state),
+ selectors.getFlyout(state, 'viewControlsFlyout')
+ ]);
+
+ const totalPageThreshold = 1000;
+ let isPageTransitionEnabled = totalPages < totalPageThreshold;
+
+ useLayoutEffect(() => {
+ const viewControlsFlyout = {
+ dataElement: 'viewControlsFlyout',
+ className: 'ViewControlsFlyout',
+ items: getViewControlsFlyoutItems()
+ };
+
+ if (!currentFlyout) {
+ dispatch(actions.addFlyout(viewControlsFlyout));
+ } else {
+ dispatch(actions.updateFlyout(viewControlsFlyout.dataElement, viewControlsFlyout));
+ }
+ }, [isFullScreen, isMultiViewerModeAvailable, isMultiViewerMode, displayMode]);
+
+ if (isDisabled) {
+ return;
+ }
+
+ const documentViewer = core.getDocumentViewer();
+ const displayModeManager = documentViewer?.getDisplayModeManager();
+ if (displayModeManager?.isVirtualDisplayEnabled()) {
+ isPageTransitionEnabled = true;
+ }
+
+ const showReaderButton = core.isFullPDFEnabled() && core.getDocument()?.getType() === 'pdf';
+ const showCompareButton = !isIE11 && !isMultiTab && isMultiViewerModeAvailable;
+ const toggleCompareMode = () => {
+ store.dispatch(actions.setIsMultiViewerMode(!isMultiViewerMode));
+ };
+
+ const handleClick = (pageTransition, layout) => {
+ const setDisplayMode = () => {
+ const displayModeObject = displayModeObjects.find(
+ (obj) => obj.pageTransition === pageTransition && obj.layout === layout,
+ );
+ core.setDisplayMode(displayModeObject.displayMode);
+ };
+
+ if (isReaderMode) {
+ exitReaderMode(store);
+ setTimeout(() => {
+ setDisplayMode();
+ });
+ } else {
+ setDisplayMode();
+ }
+ };
+
+ const handleReaderModeClick = () => {
+ if (isReaderMode) {
+ return;
+ }
+ enterReaderMode(store);
+ };
+
+ let pageTransition;
+ let layout;
+
+ const displayModeObject = displayModeObjects.find((obj) => obj.displayMode === displayMode);
+ if (displayModeObject) {
+ pageTransition = displayModeObject.pageTransition;
+ layout = displayModeObject.layout;
+ }
+
+ const getViewControlsFlyoutItems = () => {
+ let viewControlsFlyoutItems = [];
+
+ const continuousPageTransitionButton = {
+ icon: 'icon-header-page-manipulation-page-transition-continuous-page-line',
+ label: 'option.pageTransition.continuous',
+ title: 'option.pageTransition.continuous',
+ onClick: () => handleClick('continuous', layout),
+ dataElement: 'continuousPageTransitionButton',
+ isActive: pageTransition === 'continuous' && !isReaderMode
+ };
+ const defaultPageTransitionButton = {
+ icon: 'icon-header-page-manipulation-page-transition-page-by-page-line',
+ label: 'option.pageTransition.default',
+ title: 'option.pageTransition.default',
+ onClick: () => handleClick('default', layout),
+ dataElement: 'defaultPageTransitionButton',
+ isActive: pageTransition === 'default' && !isReaderMode
+ };
+ const readerPageTransitionButton = {
+ icon: 'icon-header-page-manipulation-page-transition-reader',
+ label: 'option.pageTransition.reader',
+ title: 'option.pageTransition.reader',
+ onClick: () => handleReaderModeClick(),
+ dataElement: 'readerPageTransitionButton',
+ isActive: isReaderMode
+ };
+ const rotateClockwiseButton = {
+ icon: 'icon-header-page-manipulation-page-rotation-clockwise-line',
+ label: 'action.rotateClockwise',
+ title: 'action.rotateClockwise',
+ onClick: () => core.rotateClockwise(activeDocumentViewerKey),
+ dataElement: 'rotateClockwiseButton'
+ };
+ const rotateCounterClockwiseButton = {
+ icon: 'icon-header-page-manipulation-page-rotation-clockwise-line',
+ label: 'action.rotateCounterClockwise',
+ title: 'action.rotateCounterClockwise',
+ onClick: () => core.rotateCounterClockwise(activeDocumentViewerKey),
+ dataElement: 'rotateCounterClockwiseButton'
+ };
+ const singleLayoutButton = {
+ icon: 'icon-header-page-manipulation-page-layout-single-page-line',
+ label: 'option.layout.single',
+ title: 'option.layout.single',
+ onClick: () => handleClick(pageTransition, 'single'),
+ dataElement: 'singleLayoutButton',
+ isActive: layout === 'single'
+ };
+ const doubleLayoutButton = {
+ icon: 'icon-header-page-manipulation-page-layout-double-page-line',
+ label: 'option.layout.double',
+ title: 'option.layout.double',
+ onClick: () => handleClick(pageTransition, 'double'),
+ dataElement: 'doubleLayoutButton',
+ isActive: layout === 'double'
+ };
+ const coverLayoutButton = {
+ icon: 'icon-header-page-manipulation-page-layout-cover-line',
+ label: 'option.layout.cover',
+ title: 'option.layout.cover',
+ onClick: () => handleClick(pageTransition, 'cover'),
+ dataElement: 'coverLayoutButton',
+ isActive: layout === 'cover'
+ };
+
+ const divider = 'divider';
+
+ if (isPageTransitionEnabled) {
+ viewControlsFlyoutItems.push('option.displayMode.pageTransition');
+ viewControlsFlyoutItems = [...viewControlsFlyoutItems, continuousPageTransitionButton, defaultPageTransitionButton];
+
+ if (showReaderButton) {
+ viewControlsFlyoutItems.push(readerPageTransitionButton);
+ }
+ if (!isReaderMode) {
+ viewControlsFlyoutItems.push(divider);
+ }
+ }
+ if (!isReaderMode) {
+ viewControlsFlyoutItems = [...viewControlsFlyoutItems,
+ 'action.rotate',
+ rotateClockwiseButton,
+ rotateCounterClockwiseButton,
+ divider,
+ 'option.displayMode.layout',
+ singleLayoutButton,
+ doubleLayoutButton,
+ coverLayoutButton
+ ];
+ }
+ if (showCompareButton) {
+ const toggleCompareModeButton = {
+ icon: 'icon-header-compare',
+ label: 'action.comparePages',
+ title: 'action.comparePages',
+ onClick: toggleCompareMode,
+ dataElement: 'toggleCompareModeButton',
+ isActive: isMultiViewerMode
+ };
+ viewControlsFlyoutItems.push(toggleCompareModeButton);
+ }
+
+ if (!isIOS || isIOSFullScreenSupported) {
+ const fullScreenButton = {
+ icon: isFullScreen ? 'icon-header-full-screen-exit' : 'icon-header-full-screen',
+ label: isFullScreen ? 'action.exitFullscreen' : 'action.enterFullscreen',
+ title: isFullScreen ? 'action.exitFullscreen' : 'action.enterFullscreen',
+ onClick: toggleFullscreen,
+ dataElement: DataElements.FULLSCREEN_BUTTON
+ };
+ viewControlsFlyoutItems.push(divider);
+ viewControlsFlyoutItems.push(fullScreenButton);
+ }
+
+ return viewControlsFlyoutItems;
+ };
+
+ return null;
+};
+
+export default ViewControlsFlyout;
diff --git a/src/components/ModularComponents/ViewControls/ViewControlsToggleButton.js b/src/components/ModularComponents/ViewControls/ViewControlsToggleButton.js
new file mode 100644
index 0000000000..3142668655
--- /dev/null
+++ b/src/components/ModularComponents/ViewControls/ViewControlsToggleButton.js
@@ -0,0 +1,19 @@
+import React from 'react';
+import ToggleElementButton from '../ToggleElementButton';
+
+const ViewControlsToggleButton = () => {
+ return (
+
+
+
+ );
+};
+
+export default ViewControlsToggleButton;
diff --git a/src/components/ModularComponents/ViewControls/index.js b/src/components/ModularComponents/ViewControls/index.js
new file mode 100644
index 0000000000..59927601c3
--- /dev/null
+++ b/src/components/ModularComponents/ViewControls/index.js
@@ -0,0 +1,3 @@
+import ViewControls from './ViewControlsToggleButton';
+
+export default ViewControls;
\ No newline at end of file
diff --git a/src/components/ModularComponents/ZoomControls/ZoomControls.spec.js b/src/components/ModularComponents/ZoomControls/ZoomControls.spec.js
new file mode 100644
index 0000000000..db6e5590ca
--- /dev/null
+++ b/src/components/ModularComponents/ZoomControls/ZoomControls.spec.js
@@ -0,0 +1,58 @@
+import React from 'react';
+import { render, screen, fireEvent } from '@testing-library/react';
+import userEvent from '@testing-library/user-event';
+import ZoomControls from './ZoomControls';
+import core from 'core';
+
+const ZoomControlWithRedux = withProviders(ZoomControls);
+
+const props = {
+ setZoomHandler: jest.fn(),
+ zoomValue: '100',
+ zoomTo: jest.fn(),
+ onZoomInClicked: jest.fn(),
+ onZoomOutClicked: jest.fn(),
+ isZoomFlyoutMenuActive: false,
+ isActive: true,
+ dataElement: 'zoom-container',
+ size: 0,
+};
+
+describe('Zoom Container component', () => {
+ beforeEach(() => {
+ const documentViewer = core.setDocumentViewer(1, new window.Core.DocumentViewer());
+ documentViewer.doc = new window.Core.Document('dummy', 'pdf');
+ });
+
+ it('it renders the zoomvalue correctly', () => {
+ render();
+ const input = screen.getByRole('textbox');
+ expect(input.value).toEqual(props.zoomValue);
+ });
+
+ it('it ignores invalid values that you input', () => {
+ render();
+ const input = screen.getByRole('textbox');
+ userEvent.type(input, 'zoom');
+ expect(input.value).toEqual(props.zoomValue);
+ });
+
+ it('Should execute zoomIn/zoomOut when zoom in/out button is clicked', async () => {
+ render();
+ const zoomInButton = screen.getByRole('button', { name: 'Zoom in' });
+ const zoomOutButton = screen.getByRole('button', { name: 'Zoom out' });
+ expect(zoomInButton).toBeInTheDocument();
+ fireEvent.click(zoomInButton);
+ expect(props.onZoomInClicked).toHaveBeenCalledTimes(1);
+ fireEvent.click(zoomOutButton);
+ expect(props.onZoomOutClicked).toHaveBeenCalledTimes(1);
+ });
+
+ it('it renders the zoomvalue correctly', () => {
+ render();
+ const input = screen.getByRole('textbox');
+ userEvent.type(input, '66');
+ fireEvent.keyDown(input, { key: 'Enter', code: 'Enter' });
+ expect(props.zoomTo).toHaveBeenCalledTimes(1);
+ });
+});
diff --git a/src/components/ModularComponents/ZoomControls/index.js b/src/components/ModularComponents/ZoomControls/index.js
new file mode 100644
index 0000000000..96f7b1a7b6
--- /dev/null
+++ b/src/components/ModularComponents/ZoomControls/index.js
@@ -0,0 +1,3 @@
+import ZoomControls from './ZoomControlsContainer';
+
+export default ZoomControls;
diff --git a/src/components/ModularHeaderItems/index.js b/src/components/ModularHeaderItems/index.js
new file mode 100644
index 0000000000..178f651f00
--- /dev/null
+++ b/src/components/ModularHeaderItems/index.js
@@ -0,0 +1,3 @@
+import ModularHeaderItems from './ModularHeaderItems';
+
+export default ModularHeaderItems;
\ No newline at end of file
diff --git a/src/components/MoreOptionsContextMenuPopup/MoreOptionsContextMenuPopup.js b/src/components/MoreOptionsContextMenuPopup/MoreOptionsContextMenuPopup.js
new file mode 100644
index 0000000000..262be70a3d
--- /dev/null
+++ b/src/components/MoreOptionsContextMenuPopup/MoreOptionsContextMenuPopup.js
@@ -0,0 +1,143 @@
+import React, { useState, useRef, useEffect, useCallback } from 'react';
+import { useTranslation } from 'react-i18next';
+import { createPortal } from 'react-dom';
+import PropTypes from 'prop-types';
+
+import DataElementWrapper from 'components/DataElementWrapper';
+
+import useOnClickOutside from 'hooks/useOnClickOutside';
+import getOverlayPositionBasedOn from 'helpers/getOverlayPositionBasedOn';
+
+import './MoreOptionsContextMenuPopup.scss';
+import Button from '../Button';
+import getRootNode from 'helpers/getRootNode';
+
+const propTypes = {
+ type: PropTypes.oneOf(['bookmark', 'outline', 'portfolio']).isRequired,
+ anchorButton: PropTypes.string.isRequired,
+ shouldDisplayDeleteButton: PropTypes.bool,
+ onClosePopup: PropTypes.func.isRequired,
+ onRenameClick: PropTypes.func,
+ onSetDestinationClick: PropTypes.func,
+ onDownloadClick: PropTypes.func,
+ onDeleteClick: PropTypes.func,
+ onOpenClick: PropTypes.func,
+};
+
+const MoreOptionsContextMenuPopup = ({
+ type,
+ anchorButton,
+ shouldDisplayDeleteButton,
+ onClosePopup,
+ onRenameClick,
+ onSetDestinationClick,
+ onDownloadClick,
+ onDeleteClick,
+ onOpenClick,
+}) => {
+ const [t] = useTranslation();
+ const containerRef = useRef(null);
+ const [position, setPosition] = useState({ left: -100, right: 'auto', top: 'auto' });
+
+ const Portal = ({ children, position }) => {
+ const mount = getRootNode().querySelector('#outline-edit-popup-portal');
+ mount.style.position = 'absolute';
+ mount.style.top = position.top === 'auto' ? position.top : `${position.top}px`;
+ mount.style.left = position.left === 'auto' ? position.left : `${position.left}px`;
+ mount.style.right = position.right === 'auto' ? position.right : `${position.right}px`;
+ mount.style.zIndex = 999;
+
+ return createPortal(children, mount);
+ };
+
+ useEffect(() => {
+ const position = getOverlayPositionBasedOn(anchorButton, containerRef);
+ setPosition(position);
+ }, [anchorButton]);
+
+ const onClickOutside = useCallback((e) => {
+ if (!containerRef?.current.contains(e.target)) {
+ onClosePopup();
+ }
+ });
+
+ useOnClickOutside(containerRef, onClickOutside);
+
+ return (
+
+
+ {type === 'portfolio' && onOpenClick &&
+
+
+ );
+};
+
+MoreOptionsContextMenuPopup.propTypes = propTypes;
+
+export default MoreOptionsContextMenuPopup;
diff --git a/src/components/MoreOptionsContextMenuPopup/MoreOptionsContextMenuPopup.scss b/src/components/MoreOptionsContextMenuPopup/MoreOptionsContextMenuPopup.scss
new file mode 100644
index 0000000000..ec949b4f3a
--- /dev/null
+++ b/src/components/MoreOptionsContextMenuPopup/MoreOptionsContextMenuPopup.scss
@@ -0,0 +1,29 @@
+.more-options-context-menu-popup {
+ padding-top: var(--padding-small);
+ padding-bottom: var(--padding-small);
+
+ background-color: var(--component-background);
+ box-shadow: 0px 0px 3px var(--document-box-shadow);
+ border-radius: 4px;
+
+ .option-button {
+ justify-content: flex-start;
+ width: 100%;
+ padding: var(--padding-small) var(--padding-medium);
+ border-radius: 0;
+
+ &:not(:first-child) {
+ margin-top: var(--padding-small);
+ }
+
+ &:hover {
+ background-color: var(--tools-header-background);
+ }
+
+ .Icon {
+ width: 20px;
+ height: auto;
+ margin-right: 10px;
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/MoreOptionsContextMenuPopup/index.js b/src/components/MoreOptionsContextMenuPopup/index.js
new file mode 100644
index 0000000000..1351cedcfe
--- /dev/null
+++ b/src/components/MoreOptionsContextMenuPopup/index.js
@@ -0,0 +1,3 @@
+import MoreOptionsContextMenuPopup from './MoreOptionsContextMenuPopup';
+
+export default MoreOptionsContextMenuPopup;
diff --git a/src/components/MultiTabEmptyPage/MultiTabEmptyPage.scss b/src/components/MultiTabEmptyPage/MultiTabEmptyPage.scss
new file mode 100644
index 0000000000..fc77503b32
--- /dev/null
+++ b/src/components/MultiTabEmptyPage/MultiTabEmptyPage.scss
@@ -0,0 +1,13 @@
+.MultiTabEmptyPage {
+ width: 100%;
+
+ &.closed {
+ display: none;
+ }
+
+ .empty-page-body,
+ .image-signature {
+ height: 100%;
+ width: 100%;
+ }
+}
\ No newline at end of file
diff --git a/src/components/MultiTabEmptyPage/index.js b/src/components/MultiTabEmptyPage/index.js
new file mode 100644
index 0000000000..50b46342da
--- /dev/null
+++ b/src/components/MultiTabEmptyPage/index.js
@@ -0,0 +1,3 @@
+import MultiTabEmptyPage from './MultiTabEmptyPage';
+
+export default MultiTabEmptyPage;
\ No newline at end of file
diff --git a/src/components/MultiViewer/ComparePanel/ChangeListItem/ChangeListItem.js b/src/components/MultiViewer/ComparePanel/ChangeListItem/ChangeListItem.js
new file mode 100644
index 0000000000..d63a1781c7
--- /dev/null
+++ b/src/components/MultiViewer/ComparePanel/ChangeListItem/ChangeListItem.js
@@ -0,0 +1,74 @@
+import React, { useCallback } from 'react';
+import './ChangeListItem.scss';
+import PropTypes from 'prop-types';
+import { useTranslation } from 'react-i18next';
+import Icon from 'components/Icon';
+import core from 'core';
+import selectors from 'selectors/index';
+import { useSelector, useDispatch } from 'react-redux';
+import actions from 'actions';
+import { setIsScrolledByClickingChangeItem } from 'helpers/multiViewerHelper';
+
+const propTypes = {
+ oldText: PropTypes.string,
+ newText: PropTypes.string,
+ oldCount: PropTypes.number,
+ newCount: PropTypes.number,
+ type: PropTypes.string,
+ old: PropTypes.object,
+ new: PropTypes.object,
+};
+
+const ChangeListItem = (props) => {
+ const { t } = useTranslation();
+ const dispatch = useDispatch();
+ const [syncViewer] = useSelector((state) => [
+ selectors.getSyncViewer(state),
+ ]);
+
+ const onClickItem = useCallback(() => {
+ setIsScrolledByClickingChangeItem(true);
+ if (props.old && props.new && props.old.getPageNumber() !== props.new.getPageNumber()) {
+ dispatch(actions.setSyncViewer(null));
+ }
+ for (const annotation of [props.old, props.new]) {
+ const viewerNumber = annotation === props.old ? 1 : 2;
+ if (!annotation) {
+ continue;
+ }
+ core.getDocumentViewer(viewerNumber).getAnnotationManager().deselectAllAnnotations();
+ core.jumpToAnnotation(annotation, viewerNumber);
+ core.getDocumentViewer(viewerNumber).getAnnotationManager().selectAnnotation(annotation);
+ }
+ }, [syncViewer]);
+
+ return (
+
+
+
+
+
+
{t('multiViewer.comparePanel.change')}
+
{props.type}
+ {props.old &&
+
+
{t('multiViewer.comparePanel.old')}
+
-{props.oldCount}
+
+
{props.oldText}
+
}
+ {props.new &&
+
+
{t('multiViewer.comparePanel.new')}
+
+{props.newCount}
+
+
{props.newText}
+
}
+
+
+ );
+};
+
+ChangeListItem.propTypes = propTypes;
+
+export default ChangeListItem;
diff --git a/src/components/MultiViewer/ComparePanel/ChangeListItem/ChangeListItem.scss b/src/components/MultiViewer/ComparePanel/ChangeListItem/ChangeListItem.scss
new file mode 100644
index 0000000000..928e98a843
--- /dev/null
+++ b/src/components/MultiViewer/ComparePanel/ChangeListItem/ChangeListItem.scss
@@ -0,0 +1,73 @@
+@import "../../../../constants/styles";
+
+.ChangeListItem {
+ display: flex;
+ flex-direction: row;
+ border-radius: 4px;
+ align-self: stretch;
+ filter: drop-shadow(0px 0px 4px var(--gray-7));
+ background-color: var(--component-background);
+ font-family: var(--font-family);
+
+ .icon-change{
+ padding: 14px 0 14px 14px;
+
+ height: 100%;
+ .Icon {
+ width: 24px;
+ height: 24px;
+ }
+ }
+
+ .container-right {
+ padding: 12px;
+ width: calc(100% - 14px - 24px); // Subtract icon padding + icon width
+ display: flex;
+ flex-direction: column;
+ overflow-wrap: break-word;
+
+ .title {
+ font-size: var(--font-size-medium);
+ }
+
+ .type {
+ font-size: var(--font-size-small);
+ color: var(--gray-7);
+ text-transform: capitalize;
+ }
+
+ .value-container {
+ padding: 12px;
+ margin-top: 10px;
+ width: 100%;
+ display: flex;
+ flex-direction: column;
+ border-radius: 4px;
+ align-self: stretch;
+ color: var(--gray-8);
+ font-size: var(--font-size-medium);
+ font-weight: var(--font-weight-medium);
+
+ &.old {
+ .value-title {
+ color: #CF0101;
+ }
+ background-color: rgba(255, 73, 73, 0.1);
+ }
+ &.new {
+ .value-title{
+ color: #006D41;
+ }
+ background-color: rgba(21, 205, 131, 0.1);
+ }
+
+ .value-title {
+ display: flex;
+ justify-content: space-between;
+ font-size: var(--font-size-small);
+ font-weight: var(--font-weight-bold);
+ }
+ }
+ }
+
+}
diff --git a/src/components/MultiViewer/ComparePanel/ChangeListItem/index.js b/src/components/MultiViewer/ComparePanel/ChangeListItem/index.js
new file mode 100644
index 0000000000..522d18d3f0
--- /dev/null
+++ b/src/components/MultiViewer/ComparePanel/ChangeListItem/index.js
@@ -0,0 +1,3 @@
+import ChangeListItem from './ChangeListItem';
+
+export default ChangeListItem;
\ No newline at end of file
diff --git a/src/components/MultiViewer/ComparePanel/ComparePanel.js b/src/components/MultiViewer/ComparePanel/ComparePanel.js
new file mode 100644
index 0000000000..02abebad6e
--- /dev/null
+++ b/src/components/MultiViewer/ComparePanel/ComparePanel.js
@@ -0,0 +1,305 @@
+import React, { useEffect, useRef, useState } from 'react';
+import './ComparePanel.scss';
+import DataElementWrapper from 'components/DataElementWrapper';
+import classNames from 'classnames';
+import { useSelector, useDispatch } from 'react-redux';
+import { useTranslation } from 'react-i18next';
+import selectors from 'selectors';
+import { isMobileSize } from 'helpers/getDeviceSize';
+import Events from 'constants/events';
+import ChangeListItem from 'components/MultiViewer/ComparePanel/ChangeListItem';
+import core from 'core';
+import throttle from 'lodash/throttle';
+import { SYNC_MODES } from 'constants/multiViewerContants';
+import multiViewerHelper from 'helpers/multiViewerHelper';
+import fireEvent from 'helpers/fireEvent';
+import actions from 'actions';
+import DataElements from 'src/constants/dataElement';
+import { panelMinWidth } from 'constants/panel';
+
+const specialChar = /([!@#$%^&*()+=\[\]\\';,./{}|":<>?~_-])/gm;
+
+const ComparePanel = ({
+ dataElement = 'comparePanel',
+ // For storybook tests
+ initialChangeListData = {},
+ initialTotalChanges = 0,
+}) => {
+ const dispatch = useDispatch();
+ const { t } = useTranslation();
+ const isMobile = isMobileSize();
+ const [
+ isOpen,
+ currentWidth,
+ isInDesktopOnlyMode,
+ isComparisonOverlayEnabled,
+ multiViewerSyncScrollMode,
+ ] = useSelector((state) => [
+ selectors.isElementOpen(state, dataElement),
+ selectors.getComparePanelWidth(state),
+ selectors.isInDesktopOnlyMode(state),
+ selectors.getIsComparisonOverlayEnabled(state),
+ selectors.getMultiViewerSyncScrollMode(state),
+ ]);
+ const isCustom = dataElement !== 'comparePanel';
+ const panelWidth = !isCustom ? currentWidth : panelMinWidth - 32;
+ const [searchValue, setSearchValue] = React.useState('');
+ const style = !isInDesktopOnlyMode && isMobile ? {} : { width: `${panelWidth}px`, minWidth: `${panelWidth}px` };
+ const changeListData = useRef(initialChangeListData);
+ const [totalChanges, setTotalChanges] = useState(initialTotalChanges);
+ const [filteredListData, setFilteredListData] = useState(initialChangeListData);
+ const filterfuncRef = useRef(
+ throttle(
+ (searchValue) => {
+ if (!changeListData.current) {
+ return [];
+ }
+ if (!searchValue) {
+ setFilteredListData(changeListData.current);
+ return;
+ }
+ searchValue = searchValue.replace(specialChar, '\\$&');
+ const keys = Object.keys(changeListData.current);
+ const newData = {};
+ for (const key of keys) {
+ for (const item of changeListData.current[key]) {
+ if (
+ item.oldText?.toLowerCase().match(searchValue.toLowerCase()) ||
+ item.newText?.toLowerCase().match(searchValue.toLowerCase())
+ ) {
+ if (!newData[key]) {
+ newData[key] = [];
+ }
+ newData[key].push(item);
+ }
+ }
+ }
+ setFilteredListData(newData);
+ },
+ 100,
+ { leading: true },
+ ),
+ );
+
+ useEffect(() => {
+ const updatePanelItems = (doc1Annotations, doc2Annotations, diffCount) => {
+ dispatch(actions.setIsCompareStarted(true));
+ dispatch(actions.enableElement('comparePanelToggleButton'));
+ dispatch(actions.openElement(DataElements.LOADING_MODAL));
+ const annotMap = {};
+ if (multiViewerSyncScrollMode === SYNC_MODES.SKIP_UNMATCHED) {
+ const doc1AnnotationsClone = [...doc1Annotations];
+ const doc2AnnotationsClone = [...doc2Annotations];
+ const annotationMatches1 = [];
+ const annotationMatches2 = [];
+ while (doc1AnnotationsClone.length || doc2AnnotationsClone.length) {
+ const annotation = doc1AnnotationsClone[0] || doc2AnnotationsClone[0];
+ const ids = [annotation.getCustomData('TextDiffID')];
+ const pageNumbers1 = [];
+ const pageNumbers2 = [];
+ const side1Annotations = [];
+ const side2Annotations = [];
+ const checkIfAnnotionMatches1 = (annotation) => ids.includes(annotation.getCustomData('TextDiffID')) || pageNumbers1.includes(annotation.PageNumber);
+ const checkIfAnnotionMatches2 = (annotation) => ids.includes(annotation.getCustomData('TextDiffID')) || pageNumbers2.includes(annotation.PageNumber);
+ let added = false;
+ do {
+ added = false;
+ for (const annotation of doc1AnnotationsClone) {
+ if (checkIfAnnotionMatches1(annotation)) {
+ const annotId = annotation.getCustomData('TextDiffID');
+ const annotPageNumber = annotation.PageNumber;
+ !ids.includes(annotId) && ids.push(annotId);
+ !pageNumbers1.includes(annotPageNumber) && pageNumbers1.push(annotation.PageNumber);
+ doc1AnnotationsClone.splice(doc1AnnotationsClone.indexOf(annotation), 1);
+ side1Annotations.push(annotation);
+ added = true;
+ }
+ }
+ for (const annotation of doc2AnnotationsClone) {
+ if (checkIfAnnotionMatches2(annotation)) {
+ const annotId = annotation.getCustomData('TextDiffID');
+ const annotPageNumber = annotation.PageNumber;
+ !ids.includes(annotId) && ids.push(annotId);
+ !pageNumbers2.includes(annotPageNumber) && pageNumbers2.push(annotation.PageNumber);
+ doc2AnnotationsClone.splice(doc2AnnotationsClone.indexOf(annotation), 1);
+ side2Annotations.push(annotation);
+ added = true;
+ }
+ }
+ } while (added);
+ if (side1Annotations.length && side2Annotations.length) {
+ annotationMatches1.push(side1Annotations);
+ annotationMatches2.push(side2Annotations);
+ }
+ }
+
+ const matchedPages = { 1: {}, 2: {} };
+ for (const i in annotationMatches1) {
+ const doc1Annotations = annotationMatches1[i];
+ const doc2Annotations = annotationMatches2[i];
+ const doc1Pages = Array.from(new Set(doc1Annotations.map((annotation) => annotation.PageNumber)));
+ const doc2Pages = Array.from(new Set(doc2Annotations.map((annotation) => annotation.PageNumber)));
+ for (const pageNumber of doc1Pages) {
+ matchedPages[1][pageNumber] = {
+ otherSidePages: doc2Pages,
+ thisSidePages: doc1Pages,
+ };
+ }
+ for (const pageNumber of doc2Pages) {
+ matchedPages[2][pageNumber] = {
+ otherSidePages: doc1Pages,
+ thisSidePages: doc2Pages,
+ };
+ }
+ }
+ multiViewerHelper.matchedPages = matchedPages;
+ }
+ const matchedIds = [];
+ for (const index in doc1Annotations) {
+ const annotation = doc1Annotations[index];
+ const type = annotation.getCustomData('TextDiffType');
+ const id = annotation.getCustomData('TextDiffID');
+ const otherAnnotations = doc2Annotations.filter((annotation) => annotation?.getCustomData('TextDiffID') === id);
+ if (!otherAnnotations.length) {
+ continue;
+ }
+ if (!matchedIds.includes(id)) {
+ matchedIds.push(id);
+ }
+ if (!annotMap[annotation.PageNumber]) {
+ annotMap[annotation.PageNumber] = [];
+ }
+ annotMap[annotation.PageNumber].push({
+ new: otherAnnotations[0],
+ newText: otherAnnotations[0]?.Author,
+ newCount: otherAnnotations[0]?.Author?.length,
+ old: annotation,
+ oldText: annotation?.Author,
+ oldCount: annotation?.Author?.length,
+ type: `${t('multiViewer.comparePanel.textContent')} - ${t(`multiViewer.comparePanel.${type}`)}`,
+ });
+ if (otherAnnotations.length > 1) {
+ for (const i in otherAnnotations) {
+ if (i === '0') {
+ continue;
+ }
+ annotMap[annotation.PageNumber].push({
+ new: otherAnnotations[i],
+ newText: otherAnnotations[i]?.Author,
+ newCount: otherAnnotations[i]?.Author?.length,
+ old: annotation,
+ oldText: annotation?.Author,
+ oldCount: annotation?.Author?.length,
+ type: `${t('multiViewer.comparePanel.textContent')} - ${t(`multiViewer.comparePanel.${type}`)}`,
+ });
+ }
+ }
+ }
+ const unmatchedAnnotations1 = doc1Annotations.filter((annotation) => !matchedIds.includes(annotation.getCustomData('TextDiffID')));
+ const unmatchedAnnotations2 = doc2Annotations.filter((annotation) => !matchedIds.includes(annotation.getCustomData('TextDiffID')));
+ for (const annotation of unmatchedAnnotations1) {
+ if (!annotMap[annotation.PageNumber]) {
+ annotMap[annotation.PageNumber] = [];
+ }
+ annotMap[annotation.PageNumber].push({
+ old: annotation,
+ oldText: annotation?.Author,
+ oldCount: annotation?.Author?.length,
+ type: `${t('multiViewer.comparePanel.textContent')} - ${t(`multiViewer.comparePanel.${annotation.getCustomData('TextDiffType')}`)}`,
+ });
+ }
+ for (const annotation of unmatchedAnnotations2) {
+ if (!annotMap[annotation.PageNumber]) {
+ annotMap[annotation.PageNumber] = [];
+ }
+ annotMap[annotation.PageNumber].push({
+ new: annotation,
+ newText: annotation?.Author,
+ newCount: annotation?.Author?.length,
+ type: `${t('multiViewer.comparePanel.textContent')} - ${t(`multiViewer.comparePanel.${annotation.getCustomData('TextDiffType')}`)}`,
+ });
+ }
+ for (const pageNumber of Object.keys(annotMap)) {
+ annotMap[pageNumber] = annotMap[pageNumber].sort((a, b) => {
+ if (a?.new?.PageNumber && b?.new?.PageNumber && a?.new?.PageNumber !== b?.new?.PageNumber) {
+ return a?.new?.PageNumber - b?.new?.PageNumber;
+ }
+ const aY = a?.new?.Y || a?.old?.Y;
+ const bY = b?.new?.Y || b?.old?.Y;
+ if (aY === bY) {
+ return (a?.new?.X || a?.old?.X) - (b?.new?.X || b?.old?.X);
+ }
+ return aY - bY;
+ });
+ }
+ dispatch(actions.closeElement(DataElements.LOADING_MODAL));
+ fireEvent(Events.COMPARE_ANNOTATIONS_LOADED, { annotMap, diffCount });
+ if (!isComparisonOverlayEnabled) {
+ core.hideAnnotations(core.getSemanticDiffAnnotations(1), 1);
+ core.hideAnnotations(core.getSemanticDiffAnnotations(2), 2);
+ }
+ setTotalChanges(diffCount);
+ changeListData.current = annotMap;
+ setFilteredListData(annotMap);
+ };
+ const resetPanelItems = () => {
+ setTotalChanges(0);
+ changeListData.current = {};
+ };
+ core.addEventListener(Events.COMPARE_ANNOTATIONS_LOADED, updatePanelItems, undefined, 1);
+ core.addEventListener('documentUnloaded', resetPanelItems, undefined, 1);
+ core.addEventListener('documentUnloaded', resetPanelItems, undefined, 2);
+ return () => {
+ core.removeEventListener(Events.COMPARE_ANNOTATIONS_LOADED, updatePanelItems, 1);
+ core.removeEventListener('documentUnloaded', resetPanelItems, 1);
+ core.removeEventListener('documentUnloaded', resetPanelItems, 2);
+ };
+ }, []);
+
+ const renderPageItem = (pageNum) => {
+ const changeListItems = filteredListData[pageNum];
+ return
+ {t('multiViewer.comparePanel.page')} {pageNum}
+ {changeListItems.map(((props) => ))}
+ ;
+ };
+
+ const onSearchInputChange = (e) => {
+ const newValue = e.target.value;
+ setSearchValue(newValue);
+ filterfuncRef.current(newValue);
+ };
+
+ const shouldRenderItems = !!(totalChanges && filteredListData && Object.keys(filteredListData).length);
+ return (
+
+
+
+
+
+
+ {t('multiViewer.comparePanel.changesList')} ({totalChanges})
+
+
+ {shouldRenderItems && Object.keys(filteredListData).map((key) => renderPageItem(key))}
+
+
+
+ );
+};
+
+export default ComparePanel;
diff --git a/src/components/MultiViewer/ComparePanel/ComparePanel.scss b/src/components/MultiViewer/ComparePanel/ComparePanel.scss
new file mode 100644
index 0000000000..09ad6fc44f
--- /dev/null
+++ b/src/components/MultiViewer/ComparePanel/ComparePanel.scss
@@ -0,0 +1,62 @@
+@import "../../../constants/styles";
+
+$padding-main: 0px;
+$padding-inner: 16px;
+
+.ComparePanel {
+ padding: $padding-main;
+ height: 100%;
+
+ .input-container {
+ margin: $padding-inner;
+ display: flex;
+ position: relative;
+ box-sizing: border-box;
+ border: 1px solid var(--border);
+ border-radius: 4px;
+ height: 28px;
+ align-items: center;
+ justify-content: flex-end;
+ color: var(--text-color);
+ padding: 6px 2px 6px 6px;
+ background: var(--gray-0);
+
+ input {
+ width: 100%;
+ padding: 6px 26px 6px 6px;
+ height: 20px;
+ border: none;
+ background: transparent;
+ }
+ }
+
+ .changeListContainer{
+ height: calc(100% - 28px);
+ .changeListTitle {
+ padding: 8px $padding-inner 2px;
+ font-size: var(--font-size-medium);
+ font-family: var(--font-family);
+ font-weight: var(--font-weight-bold);
+ color: var(--gray-8);
+ span {
+ color: var(--gray-7)
+ }
+ }
+
+ .changeList {
+ padding: $padding-inner;
+ height: calc(100% - 43px);
+ overflow-y: auto;
+ display: flex;
+ flex-direction: column;
+ gap: 8px;
+
+ .page-number {
+ margin-top: 10px;
+ font-size: var(--font-size-default);
+ font-family: var(--font-family);
+ color: var(--gray-7);
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/MultiViewer/ComparePanel/ComparePanel.spec.js b/src/components/MultiViewer/ComparePanel/ComparePanel.spec.js
new file mode 100644
index 0000000000..666561faea
--- /dev/null
+++ b/src/components/MultiViewer/ComparePanel/ComparePanel.spec.js
@@ -0,0 +1,60 @@
+import { createStore } from 'redux';
+import { Provider } from 'react-redux';
+import React from 'react';
+import { render } from '@testing-library/react';
+import ComparePanel from './ComparePanel';
+import core from 'core';
+
+// mock initial state.
+// UI Buttons are redux connected, and they need a state or the
+// tests will error out
+const initialState = {
+ viewer: {
+ syncViewer: null,
+ openElements: {
+ comparePanel: true
+ },
+ panelWidths: {
+ comparePanel: 330,
+ },
+ isInDesktopOnlyMode: false,
+ disabledElements: {},
+ }
+};
+
+function rootReducer(state = initialState, action) { // eslint-disable-line no-unused-vars
+ return state;
+}
+
+const store = createStore(rootReducer);
+const ComparePanelWithRedux = (props) => (
+
+
+
+);
+
+const noop = () => {
+};
+
+jest.mock('core', () => ({
+ addEventListener: noop,
+ removeEventListener: noop,
+ jumpToAnnotation: noop,
+ getDocumentViewer: () => ({
+ getAnnotationManager: () => ({
+ deselectAllAnnotations: noop,
+ selectAnnotation: noop,
+ })
+ })
+}));
+describe('ComparePanel component', () => {
+ it('Should render correctly', () => {
+ const { container } = render();
+ const searchBar = container.querySelector('.input-container > input');
+ const changeList = container.querySelector('.changeList');
+ const changeCountTitle = container.querySelector('.changeListTitle');
+ expect(searchBar).toBeInTheDocument();
+ expect(changeList).toBeInTheDocument();
+ expect(changeCountTitle).toBeInTheDocument();
+ });
+});
diff --git a/src/components/MultiViewer/ComparePanel/ComparePanel.stories.js b/src/components/MultiViewer/ComparePanel/ComparePanel.stories.js
new file mode 100644
index 0000000000..705f734724
--- /dev/null
+++ b/src/components/MultiViewer/ComparePanel/ComparePanel.stories.js
@@ -0,0 +1,118 @@
+import React from 'react';
+import { Provider } from 'react-redux';
+import { configureStore } from '@reduxjs/toolkit';
+import ComparePanel from './ComparePanel';
+import rootReducer from 'reducers/rootReducer';
+
+export default {
+ title: 'ModularComponents/ComparePanel',
+ component: ComparePanel,
+};
+
+const store = configureStore({
+ reducer: rootReducer,
+});
+
+export const Empty = () => {
+ return (
+
+
+
+ );
+};
+
+const mockItems = {
+ '1': [
+ {
+ 'new': {
+ 'Id': '24ba68bd-4a45-4c52-b80f-b543ac5a0336'
+ },
+ 'newText': 'Important Factors ',
+ 'newCount': 18,
+ 'old': {
+ 'Id': '5e45eea5-d63d-4328-811f-410ee769fe30'
+ },
+ 'oldText': '',
+ 'oldCount': 0,
+ 'type': 'Text Content - insert'
+ },
+ {
+ 'new': {
+ 'Id': '99c5eacc-325c-4b40-be0c-3eb71d20430e'
+ },
+ 'newText': ' you',
+ 'newCount': 4,
+ 'old': {
+ 'Id': 'fa896655-c4e2-4133-8cda-63d15d6038d3'
+ },
+ 'oldText': '',
+ 'oldCount': 0,
+ 'type': 'Text Content - insert'
+ }
+ ],
+ '2': [
+ {
+ 'new': {
+ 'Id': 'fe9079d4-e139-44c1-afbc-48de7483efae'
+ },
+ 'newText': 'into',
+ 'newCount': 4,
+ 'old': {
+ 'Id': '9608d6a2-d690-47d2-aef0-350d7ba90602'
+ },
+ 'oldText': 'intoadsf',
+ 'oldCount': 8,
+ 'type': 'Text Content - edit'
+ },
+ {
+ 'new': {
+ 'Id': '4629e9e5-126e-4925-aa63-c69a103d3ace'
+ },
+ 'newText': 'This',
+ 'newCount': 4,
+ 'old': {
+ 'Id': '341415d1-d4f7-4ac9-ad04-3d4ce60cdd34'
+ },
+ 'oldText': 'Thasdfis',
+ 'oldCount': 8,
+ 'type': 'Text Content - edit'
+ }
+ ],
+ '3': [
+ {
+ 'new': {
+ 'Id': '93875dd2-8eaf-416d-a01b-b9cb807ad9a3'
+ },
+ 'newText': 'of thing',
+ 'newCount': 9,
+ 'old': {
+ 'Id': 'f0e5be8a-11c0-4b25-9fc4-833a9175875e'
+ },
+ 'oldText': '3',
+ 'oldCount': 1,
+ 'type': 'Text Content - edit'
+ },
+ {
+ 'new': {
+ 'Id': '790fc686-4a83-4654-b74f-4f911614fe11'
+ },
+ 'newText': ' initially on the main platforms preferred by your users. But later if you wish to expand, the library does ',
+ 'newCount': 108,
+ 'old': {
+ 'Id': 'c2ecc98b-3ef5-4935-838e-fc9dfc626537'
+ },
+ 'oldText': 'y does ',
+ 'oldCount': 7,
+ 'type': 'Text Content - edit'
+ }
+ ],
+};
+const mockItemCount = 6;
+
+export const Populated = () => {
+ return (
+
+
+
+ );
+};
diff --git a/src/components/MultiViewer/ComparePanel/index.js b/src/components/MultiViewer/ComparePanel/index.js
new file mode 100644
index 0000000000..67d9ee7f39
--- /dev/null
+++ b/src/components/MultiViewer/ComparePanel/index.js
@@ -0,0 +1,3 @@
+import ComparePanel from './ComparePanel';
+
+export default ComparePanel;
\ No newline at end of file
diff --git a/src/components/MultiViewer/CompareZoomOverlay/CompareZoomOverlay.js b/src/components/MultiViewer/CompareZoomOverlay/CompareZoomOverlay.js
new file mode 100644
index 0000000000..00386de815
--- /dev/null
+++ b/src/components/MultiViewer/CompareZoomOverlay/CompareZoomOverlay.js
@@ -0,0 +1,77 @@
+import React from 'react';
+import FlyoutMenu from 'components/FlyoutMenu/FlyoutMenu';
+import ZoomOverlay from 'components/ZoomOverlay/ZoomOverlay';
+import { useSelector, useDispatch } from 'react-redux';
+import { fitToWidth, fitToPage, zoomTo } from 'helpers/zoom';
+import selectors from 'selectors';
+import actions from 'actions';
+import { useTranslation } from 'react-i18next';
+import PropTypes from 'prop-types';
+
+const propTypes = {
+ zoom1: PropTypes.number,
+ zoom2: PropTypes.number,
+};
+
+const CompareZoomOverlay = ({
+ zoom1,
+ zoom2,
+}) => {
+ const { t } = useTranslation();
+ const dispatch = useDispatch();
+
+ const onClickMarqueeZoom = (documentViewerKey) => () => {
+ dispatch(actions.closeElements([`zoomOverlay${documentViewerKey}`]));
+ };
+ const onClickZoomLevelOption = (documentViewerKey) => (zoomLevel) => {
+ zoomTo(zoomLevel, true, documentViewerKey);
+ dispatch(actions.closeElements([`zoomOverlay${documentViewerKey}`]));
+ };
+ const onClickMarqueeZoom1 = onClickMarqueeZoom(1);
+ const onClickMarqueeZoom2 = onClickMarqueeZoom(2);
+ const onClickZoomLevelOption1 = onClickZoomLevelOption(1);
+ const onClickZoomLevelOption2 = onClickZoomLevelOption(2);
+ const onFitToWidth1 = () => fitToWidth(1);
+ const onFitToWidth2 = () => fitToWidth(2);
+ const onFitToPage1 = () => fitToPage(1);
+ const onFitToPage2 = () => fitToPage(2);
+
+ return (
+ <>
+
+ selectors.isElementDisabled(state, 'marqueeToolButton'))}
+ fitToWidth={onFitToWidth1}
+ fitToPage={onFitToPage1}
+ onClickZoomLevelOption={onClickZoomLevelOption1}
+ onClickMarqueeZoom={onClickMarqueeZoom1}
+ />
+
+
+ selectors.isElementDisabled(state, 'marqueeToolButton'))}
+ fitToWidth={onFitToWidth2}
+ fitToPage={onFitToPage2}
+ onClickZoomLevelOption={onClickZoomLevelOption2}
+ onClickMarqueeZoom={onClickMarqueeZoom2}
+ />
+
+ >
+ );
+};
+
+CompareZoomOverlay.propTypes = propTypes;
+
+export default CompareZoomOverlay;
diff --git a/src/components/MultiViewer/CompareZoomOverlay/index.js b/src/components/MultiViewer/CompareZoomOverlay/index.js
new file mode 100644
index 0000000000..134e7c8a5f
--- /dev/null
+++ b/src/components/MultiViewer/CompareZoomOverlay/index.js
@@ -0,0 +1,3 @@
+import CompareZoomOverlay from './CompareZoomOverlay';
+
+export default CompareZoomOverlay;
\ No newline at end of file
diff --git a/src/components/MultiViewer/ComparisonButton/ComparisonButton.js b/src/components/MultiViewer/ComparisonButton/ComparisonButton.js
new file mode 100644
index 0000000000..a86d7a4789
--- /dev/null
+++ b/src/components/MultiViewer/ComparisonButton/ComparisonButton.js
@@ -0,0 +1,103 @@
+import React, { useCallback, useState, useEffect } from 'react';
+import './ComparisonButton.scss';
+import { useSelector, useDispatch } from 'react-redux';
+import selectors from 'selectors';
+import actions from 'actions';
+import Choice from 'components/Choice';
+import DataElements from 'src/constants/dataElement';
+import core from 'core';
+import { useTranslation } from 'react-i18next';
+
+const ComparisonButton = () => {
+ const { t } = useTranslation();
+ const dispatch = useDispatch();
+ const [disabled, setDisabled] = useState(true);
+ const [isCompareStarted, isComparisonOverlayEnabled] = useSelector((state) => [
+ selectors.isCompareStarted(state),
+ selectors.getIsComparisonOverlayEnabled(state),
+ ]);
+ const semanticDiffAnnotations = core.getSemanticDiffAnnotations();
+
+ useEffect(() => {
+ const checkDisabled = () => {
+ const documentsLoaded = core.getDocument(1) && core.getDocument(2);
+ const document1IsValidType = core.getDocument(1)?.getType() === 'pdf' ||
+ (core.getDocument(1)?.getType() === 'webviewerServer' && !core.getDocument(1)?.isWebViewerServerDocument());
+ const document2IsValidType = core.getDocument(2)?.getType() === 'pdf' ||
+ (core.getDocument(2)?.getType() === 'webviewerServer' && !core.getDocument(2)?.isWebViewerServerDocument());
+ if (documentsLoaded && document1IsValidType && document2IsValidType) {
+ setDisabled(false);
+ } else {
+ setDisabled(true);
+ }
+ };
+ const unLoaded = () => {
+ setDisabled(true);
+ dispatch(actions.setIsCompareStarted(false));
+ dispatch(actions.disableElement('comparePanelToggleButton'));
+ dispatch(actions.closeElement('comparePanel'));
+ };
+ checkDisabled();
+ core.addEventListener('documentLoaded', checkDisabled, undefined, 1);
+ core.addEventListener('documentLoaded', checkDisabled, undefined, 2);
+ core.addEventListener('documentUnloaded', unLoaded, undefined, 1);
+ core.addEventListener('documentUnloaded', unLoaded, undefined, 2);
+
+ if (!semanticDiffAnnotations.length && isComparisonOverlayEnabled) {
+ dispatch(actions.setIsComparisonOverlayEnabled(false));
+ }
+
+ if (semanticDiffAnnotations.length && !isComparisonOverlayEnabled) {
+ dispatch(actions.setIsComparisonOverlayEnabled(true));
+ }
+
+ return () => {
+ core.removeEventListener('documentLoaded', checkDisabled, undefined, 1);
+ core.removeEventListener('documentLoaded', checkDisabled, undefined, 2);
+ core.removeEventListener('documentUnloaded', unLoaded, undefined, 1);
+ core.removeEventListener('documentUnloaded', unLoaded, undefined, 2);
+ };
+ }, [semanticDiffAnnotations]);
+
+ const startComparison = useCallback(() => {
+ const [documentViewer, documentViewer2] = core.getDocumentViewers();
+ const shouldDiff = documentViewer?.getDocument() && documentViewer2?.getDocument();
+ if (shouldDiff) {
+ dispatch(actions.setIsCompareStarted(true));
+ dispatch(actions.enableElement('comparePanelToggleButton'));
+ dispatch(actions.openElement(DataElements.LOADING_MODAL));
+ documentViewer.startSemanticDiff(documentViewer2).catch((error) => {
+ console.error(error);
+ dispatch(actions.closeElement(DataElements.LOADING_MODAL));
+ });
+ }
+ }, []);
+
+ const toggleComparisonOverlay = async () => {
+ const enable = !isComparisonOverlayEnabled;
+ const [documentViewerOne, documentViewerTwo] = core.getDocumentViewers();
+ if (enable) {
+ await documentViewerOne.startSemanticDiff(documentViewerTwo);
+ } else {
+ await documentViewerOne.stopSemanticDiff();
+ }
+ dispatch(actions.setIsComparisonOverlayEnabled(enable));
+ };
+
+ return (
+
+ {!isCompareStarted ?
+ {t('action.startComparison')}
+ :
+
+ }
+
+ );
+};
+
+export default ComparisonButton;
diff --git a/src/components/MultiViewer/ComparisonButton/ComparisonButton.scss b/src/components/MultiViewer/ComparisonButton/ComparisonButton.scss
new file mode 100644
index 0000000000..123009c8a4
--- /dev/null
+++ b/src/components/MultiViewer/ComparisonButton/ComparisonButton.scss
@@ -0,0 +1,36 @@
+@import '../../../constants/styles';
+
+.ComparisonButton {
+ display: flex;
+ align-items: center;
+ font-size: var(--font-size-medium);
+
+ button {
+ @include button-reset;
+ background: var(--primary-button);
+ border-radius: 4px;
+ display: flex;
+ align-items: center;
+ padding: 8px 16px;
+ justify-content: center;
+ position: relative;
+ color: var(--primary-button-text);
+ cursor: pointer;
+
+ @include mobile {
+ font-size: 13px;
+ }
+
+ &:hover {
+ background: var(--primary-button-hover);
+ &:disabled {
+ background: var(--primary-button);
+ }
+ }
+
+ &:disabled {
+ opacity: 0.8;
+ cursor: not-allowed;
+ }
+ }
+}
diff --git a/src/components/MultiViewer/ComparisonButton/index.js b/src/components/MultiViewer/ComparisonButton/index.js
new file mode 100644
index 0000000000..8dd89c8e24
--- /dev/null
+++ b/src/components/MultiViewer/ComparisonButton/index.js
@@ -0,0 +1,3 @@
+import ComparisonButton from './ComparisonButton';
+
+export default ComparisonButton;
\ No newline at end of file
diff --git a/src/components/MultiViewer/DocumentContainer/DocumentContainer.scss b/src/components/MultiViewer/DocumentContainer/DocumentContainer.scss
new file mode 100644
index 0000000000..4b25ab223b
--- /dev/null
+++ b/src/components/MultiViewer/DocumentContainer/DocumentContainer.scss
@@ -0,0 +1,58 @@
+@import '../../../constants/styles';
+
+.MultiViewer {
+ .DocumentContainer {
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ width: calc(100% - 4px);
+ height: calc(100% - 32px);
+ overflow: overlay;
+ user-select: none;
+
+ @include ie11 {
+ margin-left: 0 !important;
+ width: 100% !important
+ }
+
+ .document {
+ overflow-x: visible;
+ overflow-y: visible;
+ margin: auto; // vertical centering when content is smaller than document container
+ // can't use 'justify-content: center;' due to losing access to content when overflowing
+ // see: https://stackoverflow.com/questions/33454533/cant-scroll-to-top-of-flex-item-that-is-overflowing-container
+ outline: none;
+ -webkit-tap-highlight-color: transparent;
+
+ &.hidden {
+ display: none;
+ }
+
+ .pageSection {
+ &[id*=pageSectionb] {
+ box-shadow: none;
+ }
+
+ .pageContainer {
+ background-color: $document-bg-color;
+ position: relative;
+ box-shadow: $md-shadow1;
+
+ span.link {
+ cursor: pointer;
+ }
+ }
+ }
+
+ textarea.freetext {
+ position: absolute;
+ z-index: 20;
+ border: 0;
+ padding: 0;
+ box-sizing: border-box;
+ resize: none;
+ outline: 1px solid transparent;
+ }
+ }
+ }
+}
diff --git a/src/components/MultiViewer/DocumentContainer/index.js b/src/components/MultiViewer/DocumentContainer/index.js
new file mode 100644
index 0000000000..57c02d0d56
--- /dev/null
+++ b/src/components/MultiViewer/DocumentContainer/index.js
@@ -0,0 +1,3 @@
+import DocumentContainer from './DocumentContainer';
+
+export default DocumentContainer;
\ No newline at end of file
diff --git a/src/components/MultiViewer/DocumentHeader/DocumentHeader.js b/src/components/MultiViewer/DocumentHeader/DocumentHeader.js
new file mode 100644
index 0000000000..736328d4b1
--- /dev/null
+++ b/src/components/MultiViewer/DocumentHeader/DocumentHeader.js
@@ -0,0 +1,72 @@
+import React, { useEffect, useState } from 'react';
+import ToggleZoomOverlay from 'components/ToggleZoomOverlay';
+import PropTypes from 'prop-types';
+import Button from 'components/Button';
+import classNames from 'classnames';
+import core from 'core';
+import { useTranslation } from 'react-i18next';
+import downloadPdf from 'helpers/downloadPdf';
+import { useDispatch, useSelector } from 'react-redux';
+import selectors from 'selectors';
+import DataElements from 'constants/dataElement';
+import actions from 'actions';
+
+import './DocumentHeader.scss';
+
+const propTypes = {
+ documentViewerKey: PropTypes.number.isRequired,
+ docLoaded: PropTypes.bool.isRequired,
+ isSyncing: PropTypes.bool.isRequired,
+};
+
+// Todo Compare: Make stories for this component
+const DocumentHeader = ({
+ documentViewerKey,
+ docLoaded,
+ isSyncing,
+}) => {
+ const { t } = useTranslation();
+ const dispatch = useDispatch();
+ const [filename, setFileName] = useState('Untitled');
+ const [saveButtonDisabled] = useSelector((state) => [selectors.isElementDisabled(state, DataElements.MULTI_VIEWER_SAVE_DOCUMENT_BUTTON)]);
+
+ useEffect(() => {
+ const stopSyncing = () => dispatch(actions.setSyncViewer(null));
+ const onLoaded = () => setFileName(core.getDocument(documentViewerKey)?.getFilename());
+ const unLoaded = () => setFileName('Untitled');
+ core.addEventListener('documentLoaded', onLoaded, undefined, documentViewerKey);
+ core.addEventListener('documentUnloaded', unLoaded, undefined, documentViewerKey);
+ core.addEventListener('displayModeUpdated', stopSyncing, undefined, documentViewerKey);
+ setFileName(core.getDocument(1)?.getFilename() || 'Untitled');
+ return () => {
+ core.removeEventListener('documentLoaded', onLoaded, documentViewerKey);
+ core.removeEventListener('documentUnloaded', unLoaded, documentViewerKey);
+ core.removeEventListener('displayModeUpdated', stopSyncing, documentViewerKey);
+ };
+ }, [documentViewerKey]);
+
+ const closeDocument = () => core.closeDocument(documentViewerKey);
+ const onClickSync = () => dispatch(actions.setSyncViewer(isSyncing ? null : documentViewerKey));
+ const onSaveDocument = () => downloadPdf(dispatch, undefined, documentViewerKey);
+
+ return (
+
+ );
+};
+
+DocumentHeader.propTypes = propTypes;
+
+export default DocumentHeader;
diff --git a/src/components/MultiViewer/DocumentHeader/DocumentHeader.scss b/src/components/MultiViewer/DocumentHeader/DocumentHeader.scss
new file mode 100644
index 0000000000..cdc8ccc387
--- /dev/null
+++ b/src/components/MultiViewer/DocumentHeader/DocumentHeader.scss
@@ -0,0 +1,48 @@
+@import '../../../constants/styles';
+
+.DocumentHeader {
+ width: 100%;
+ height: fit-content;
+ display: flex;
+ flex-direction: row;
+ align-items: center;
+ justify-content: space-between;
+ padding: 4px 0;
+
+ &.hidden {
+ visibility: hidden;
+ }
+
+ .Button {
+ width: 28px;
+ height: 28px;
+
+ &:hover {
+ background-color: var(--popup-button-hover);
+ }
+
+ &.active{
+ background-color: var(--view-header-button-active);
+ .Icon {
+ color: var(--view-header-icon-active-fill);
+ }
+ }
+ }
+
+ .zoom-overlay {
+ //align-self: flex-start;
+ }
+
+ .file-name {
+ font-family: var(--font-family);
+ font-size: var(--font-size-medium);
+ white-space: nowrap;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ }
+
+ .control-buttons {
+ display: flex;
+ gap: 8px;
+ }
+}
diff --git a/src/components/MultiViewer/DocumentHeader/index.js b/src/components/MultiViewer/DocumentHeader/index.js
new file mode 100644
index 0000000000..8429f904c7
--- /dev/null
+++ b/src/components/MultiViewer/DocumentHeader/index.js
@@ -0,0 +1,3 @@
+import DocumentHeader from './DocumentHeader';
+
+export default DocumentHeader;
\ No newline at end of file
diff --git a/src/components/MultiViewer/DropArea/DropArea.js b/src/components/MultiViewer/DropArea/DropArea.js
new file mode 100644
index 0000000000..e56b4c6fa0
--- /dev/null
+++ b/src/components/MultiViewer/DropArea/DropArea.js
@@ -0,0 +1,63 @@
+import React, { useRef } from 'react';
+import selectors from 'selectors';
+import './DropArea.scss';
+import { useTranslation } from 'react-i18next';
+import Icon from 'components/Icon';
+import PropTypes from 'prop-types';
+import loadDocument from 'helpers/loadDocument';
+import { useDispatch, useSelector } from 'react-redux';
+import getHashParameters from 'helpers/getHashParameters';
+
+const propTypes = {
+ documentViewerKey: PropTypes.number.isRequired,
+};
+
+// Todo Compare: Make stories for this component
+const DropArea = ({ documentViewerKey }) => {
+ const { t } = useTranslation();
+ const dispatch = useDispatch();
+ const fileInput = useRef();
+ const [
+ customMultiViewerAcceptedFileFormats,
+ ] = useSelector((state) => [
+ selectors.getCustomMultiViewerAcceptedFileFormats(state),
+ ]);
+
+ const browseFiles = () => fileInput.current.click();
+
+ const onDrop = (e) => {
+ e.preventDefault();
+ const { files } = e.dataTransfer;
+ if (files.length) {
+ loadDocument(dispatch, files[0], {}, documentViewerKey);
+ }
+ };
+ const loadFile = (e) => {
+ e.preventDefault();
+ const { files } = e.target;
+ if (files.length) {
+ loadDocument(dispatch, files[0], {}, documentViewerKey);
+ }
+ };
+
+ const wvServer = !!getHashParameters('webviewerServerURL', null);
+ const acceptFormats = wvServer ? window.Core.SupportedFileFormats.SERVER : window.Core.SupportedFileFormats.CLIENT;
+
+ return (
+ e.preventDefault()} onDrop={onDrop}>
+
+
{t('multiViewer.dragAndDrop')}
+
{t('multiViewer.or')}
+
{t('multiViewer.browse')}
+
`.${format}`,
+ )).join(', ')}
+ />
+
+ );
+};
+
+DropArea.propTypes = propTypes;
+
+export default DropArea;
diff --git a/src/components/MultiViewer/DropArea/DropArea.scss b/src/components/MultiViewer/DropArea/DropArea.scss
new file mode 100644
index 0000000000..8353ce4412
--- /dev/null
+++ b/src/components/MultiViewer/DropArea/DropArea.scss
@@ -0,0 +1,44 @@
+@import '../../../constants/styles';
+
+.DropArea {
+ gap: 10px;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ flex-direction: column;
+ z-index: 10;
+ top: 0;
+ bottom: 0;
+ left: 0;
+ right: 0;
+ width: 100%;
+ height: 100%;
+ background-color: var(--document-background-color);
+
+ border: 4px dashed #E1E1E3;
+ box-sizing: border-box;
+
+ .Icon {
+ color: var(--gray-7);
+ height: 65px;
+ width: 65px;
+ }
+ .hidden {
+ display: none;
+ }
+
+ button {
+ width: 105px;
+ height: 32px;
+ border-radius: 4px;
+ inset: 1px;
+ color: var(--blue-5);
+ background-color: var(--document-background-color);
+ border: 1px solid var(--blue-5);
+ pointer-events: auto;
+
+ &:hover {
+ background-color: var(--tools-button-hover);
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/MultiViewer/DropArea/index.js b/src/components/MultiViewer/DropArea/index.js
new file mode 100644
index 0000000000..c19e0dc81c
--- /dev/null
+++ b/src/components/MultiViewer/DropArea/index.js
@@ -0,0 +1,3 @@
+import DropArea from './DropArea';
+
+export default DropArea;
\ No newline at end of file
diff --git a/src/components/MultiViewer/MultiViewerWrapper/MultiViewerWrapper.js b/src/components/MultiViewer/MultiViewerWrapper/MultiViewerWrapper.js
new file mode 100644
index 0000000000..b431a3491a
--- /dev/null
+++ b/src/components/MultiViewer/MultiViewerWrapper/MultiViewerWrapper.js
@@ -0,0 +1,11 @@
+import React from 'react';
+import './MultiViewerWrapper.scss';
+import { useSelector } from 'react-redux';
+import selectors from 'selectors';
+
+const MultiViewerWrapper = ({ children }) => {
+ const isMultiViewerReady = useSelector((state) => selectors.isMultiViewerReady(state));
+ return isMultiViewerReady ? <>{children}> : null;
+};
+
+export default MultiViewerWrapper;
\ No newline at end of file
diff --git a/src/components/MultiViewer/MultiViewerWrapper/MultiViewerWrapper.scss b/src/components/MultiViewer/MultiViewerWrapper/MultiViewerWrapper.scss
new file mode 100644
index 0000000000..7820f5b31d
--- /dev/null
+++ b/src/components/MultiViewer/MultiViewerWrapper/MultiViewerWrapper.scss
@@ -0,0 +1,3 @@
+.MultiViewerWrapper {
+
+}
\ No newline at end of file
diff --git a/src/components/MultiViewer/MultiViewerWrapper/index.js b/src/components/MultiViewer/MultiViewerWrapper/index.js
new file mode 100644
index 0000000000..c5717b7f47
--- /dev/null
+++ b/src/components/MultiViewer/MultiViewerWrapper/index.js
@@ -0,0 +1,3 @@
+import MultiViewerWrapper from 'src/components/MultiViewer/MultiViewerWrapper/MultiViewerWrapper';
+
+export default MultiViewerWrapper;
\ No newline at end of file
diff --git a/src/components/MultiViewer/index.js b/src/components/MultiViewer/index.js
new file mode 100644
index 0000000000..27038c49a7
--- /dev/null
+++ b/src/components/MultiViewer/index.js
@@ -0,0 +1,3 @@
+import MultiViewer from 'src/components/MultiViewer/MultiViewer';
+
+export default MultiViewer;
\ No newline at end of file
diff --git a/src/components/NoteTextarea/NoteTextarea.stories.js b/src/components/NoteTextarea/NoteTextarea.stories.js
new file mode 100644
index 0000000000..93de470a49
--- /dev/null
+++ b/src/components/NoteTextarea/NoteTextarea.stories.js
@@ -0,0 +1,77 @@
+import React, { useRef } from 'react';
+import NoteContext from '../Note/Context';
+import NoteTextarea from './NoteTextarea';
+import { createStore } from 'redux';
+import { Provider } from 'react-redux';
+
+export default {
+ title: 'Components/NotesPanel/NoteTextarea',
+ component: NoteTextarea,
+};
+
+function handleStateChange(newValue) {
+ // eslint-disable-next-line no-console
+}
+
+const context = {
+ pendingEditTextMap: {},
+ pendingReplyMap: {},
+};
+
+const initialState = {
+ viewer: {
+ disabledElements: {},
+ openElements: { audioPlaybackPopup: true },
+ customElementOverrides: {},
+ userData: [
+ {
+ value: 'John Doe',
+ id: 'johndoe@gmail.com',
+ email: 'johndoe@gmail.com',
+ },
+ {
+ value: 'Jane Doe',
+ id: 'janedoe@gmail.com',
+ email: 'janedoe@gmail.com'
+ },
+ {
+ value: 'Jane Doe',
+ id: 'janedoejanedoejanedoejanedoe@gmail.com',
+ email: 'janedoejanedoejanedoejanedoe@gmail.com'
+ },
+ ]
+ }
+};
+
+function rootReducer(state = initialState, action) {
+ return state;
+}
+
+const store = createStore(rootReducer);
+
+const props = {
+ value: 'test',
+ onChange: handleStateChange,
+ onSubmit: () => console.log('onSubmit'),
+ onBlur: () => console.log('onBlur'),
+ onFocus: () => console.log('onFocus')
+};
+
+export const Basic = () => {
+ const textareaRef = useRef(null);
+
+ return (
+
+
+ {
+ textareaRef.current = el;
+ }
+ }
+ />
+
+
+ );
+};
\ No newline at end of file
diff --git a/src/components/NotesPanel/ReplyAttachmentPicker.js b/src/components/NotesPanel/ReplyAttachmentPicker.js
new file mode 100644
index 0000000000..ee3bf62f12
--- /dev/null
+++ b/src/components/NotesPanel/ReplyAttachmentPicker.js
@@ -0,0 +1,38 @@
+import React from 'react';
+import { useSelector } from 'react-redux';
+import selectors from 'selectors';
+
+const ReplyAttachmentPicker = ({ annotationId, addAttachments }) => {
+ const replyAttachmentHandler = useSelector((state) => selectors.getReplyAttachmentHandler(state));
+
+ const onChange = async (e) => {
+ const file = e.target.files[0];
+ if (file) {
+ let attachment = file;
+ if (replyAttachmentHandler) {
+ const url = await replyAttachmentHandler(file);
+ attachment = {
+ url,
+ name: file.name,
+ size: file.size,
+ type: file.type
+ };
+ }
+ addAttachments(annotationId, [attachment]);
+ }
+ };
+
+ return (
+ {
+ e.target.value = '';
+ }}
+ />
+ );
+};
+
+export default ReplyAttachmentPicker;
diff --git a/src/components/NotesPanelHeader/index.js b/src/components/NotesPanelHeader/index.js
new file mode 100644
index 0000000000..8a772a9429
--- /dev/null
+++ b/src/components/NotesPanelHeader/index.js
@@ -0,0 +1,3 @@
+import NotesPanelHeader from './NotesPanelHeader';
+
+export default NotesPanelHeader;
\ No newline at end of file
diff --git a/src/components/OfficeEditorCreateTablePopup/OfficeEditorCreateTablePopup.scss b/src/components/OfficeEditorCreateTablePopup/OfficeEditorCreateTablePopup.scss
new file mode 100644
index 0000000000..8c2c4b6a13
--- /dev/null
+++ b/src/components/OfficeEditorCreateTablePopup/OfficeEditorCreateTablePopup.scss
@@ -0,0 +1,26 @@
+.office-editor-create-table {
+ margin: 8px;
+ width: 120px;
+ height: 150px;
+ cursor: pointer;
+
+ table, td {
+ border: 1px solid var(--gray-8);
+ border-collapse: collapse;
+ }
+
+ td {
+ width: 12px;
+ height: 12px;
+
+ &.selected-cell {
+ background-color: var(--oe-table-dropdown-highlight);
+ }
+ }
+
+ .create-table-rows-columns {
+ display: block;
+ text-align: center;
+ margin: 10px auto;
+ }
+}
diff --git a/src/components/OfficeEditorCreateTablePopup/OfficeEditorCreateTablePopup.stories.js b/src/components/OfficeEditorCreateTablePopup/OfficeEditorCreateTablePopup.stories.js
new file mode 100644
index 0000000000..ab70942ea3
--- /dev/null
+++ b/src/components/OfficeEditorCreateTablePopup/OfficeEditorCreateTablePopup.stories.js
@@ -0,0 +1,13 @@
+import React from 'react';
+import OfficeEditorCreateTablePopup from './OfficeEditorCreateTablePopup';
+
+export default {
+ title: 'Components/OfficeEditorCreateTablePopup',
+ component: OfficeEditorCreateTablePopup
+};
+
+export function Basic() {
+ return (
+
+ );
+}
diff --git a/src/components/OfficeEditorCreateTablePopup/index.js b/src/components/OfficeEditorCreateTablePopup/index.js
new file mode 100644
index 0000000000..3bd366ecdf
--- /dev/null
+++ b/src/components/OfficeEditorCreateTablePopup/index.js
@@ -0,0 +1,3 @@
+import OfficeEditorCreateTablePopup from './OfficeEditorCreateTablePopup';
+
+export default OfficeEditorCreateTablePopup;
diff --git a/src/components/OfficeEditorImageFilePickerHandler/OfficeEditorImageFilePickerHandler.js b/src/components/OfficeEditorImageFilePickerHandler/OfficeEditorImageFilePickerHandler.js
new file mode 100644
index 0000000000..2204df4ccd
--- /dev/null
+++ b/src/components/OfficeEditorImageFilePickerHandler/OfficeEditorImageFilePickerHandler.js
@@ -0,0 +1,59 @@
+import React from 'react';
+import actions from 'actions';
+import { useDispatch } from 'react-redux';
+import getRootNode from 'helpers/getRootNode';
+import core from 'core';
+import DataElements from 'constants/dataElement';
+
+import '../FilePickerHandler/FilePickerHandler.scss';
+
+// TODO: Can we accept any other image formats?
+const ACCEPTED_FORMATS = ['jpg', 'jpeg', 'png', 'bmp'].map(
+ (format) => `.${format}`,
+).join(', ');
+
+const toBase64 = (file) => new Promise((resolve, reject) => {
+ const reader = new FileReader();
+ reader.readAsDataURL(file);
+ reader.onload = () => resolve(reader.result);
+ reader.onerror = reject;
+});
+
+const FilePickerHandler = () => {
+ const dispatch = useDispatch();
+
+ const openDocument = async (e) => {
+ const file = e.target.files[0];
+ if (file) {
+ try {
+ dispatch(actions.openElement(DataElements.LOADING_MODAL));
+ const base64 = await toBase64(file);
+ await core.getOfficeEditor().insertImageAtCursor(base64);
+ dispatch(actions.closeElement(DataElements.LOADING_MODAL));
+ } catch (error) {
+ dispatch(actions.closeElement(DataElements.LOADING_MODAL));
+ dispatch(actions.showWarningMessage({
+ title: 'Error',
+ message: error.message,
+ }));
+ }
+ const picker = getRootNode().querySelector('#office-editor-file-picker');
+ if (picker) {
+ picker.value = '';
+ }
+ }
+ };
+
+ return (
+
+
+
+ );
+};
+
+export default FilePickerHandler;
diff --git a/src/components/OfficeEditorImageFilePickerHandler/index.js b/src/components/OfficeEditorImageFilePickerHandler/index.js
new file mode 100644
index 0000000000..9babab9332
--- /dev/null
+++ b/src/components/OfficeEditorImageFilePickerHandler/index.js
@@ -0,0 +1,3 @@
+import OfficeEditorImageFilePickerHandler from './OfficeEditorImageFilePickerHandler';
+
+export default OfficeEditorImageFilePickerHandler;
diff --git a/src/components/OutlineContent/OutlineContent.js b/src/components/OutlineContent/OutlineContent.js
new file mode 100644
index 0000000000..de85a11ef3
--- /dev/null
+++ b/src/components/OutlineContent/OutlineContent.js
@@ -0,0 +1,267 @@
+import React, { useContext, useEffect, useRef, useState } from 'react';
+import { useTranslation } from 'react-i18next';
+import PropTypes from 'prop-types';
+import core from 'core';
+
+import Button from '../Button';
+import MoreOptionsContextMenuPopup from '../MoreOptionsContextMenuPopup';
+import OutlineContext from '../Outline/Context';
+import './OutlineContent.scss';
+
+const propTypes = {
+ text: PropTypes.string.isRequired,
+ outlinePath: PropTypes.string,
+ isAdding: PropTypes.bool,
+ isOutlineRenaming: PropTypes.bool,
+ setOutlineRenaming: PropTypes.func,
+ isOutlineChangingDest: PropTypes.bool,
+ setOutlineChangingDest: PropTypes.func,
+ setIsHovered: PropTypes.func,
+ onCancel: PropTypes.func,
+ textColor: PropTypes.string,
+};
+
+const OutlineContent = ({
+ text,
+ outlinePath,
+ isAdding,
+ isOutlineRenaming,
+ setOutlineRenaming,
+ isOutlineChangingDest,
+ setOutlineChangingDest,
+ setIsHovered,
+ onCancel,
+ textColor,
+}) => {
+ const {
+ currentDestPage,
+ currentDestText,
+ editingOutlines,
+ setEditingOutlines,
+ isMultiSelectMode,
+ isOutlineEditable,
+ addNewOutline,
+ renameOutline,
+ updateOutlineDest,
+ updateOutlines,
+ removeOutlines,
+ } = useContext(OutlineContext);
+
+ const [t] = useTranslation();
+ const TOOL_NAME = 'OutlineDestinationCreateTool';
+
+ const [isDefault, setIsDefault] = useState(false);
+ const [outlineText, setOutlineText] = useState(text);
+ const [isContextMenuOpen, setContextMenuOpen] = useState(false);
+ const inputRef = useRef();
+
+ const handleKeyDown = (e) => {
+ if (e.key === 'Enter') {
+ e.stopPropagation();
+ if (isAdding) {
+ onAddOutline();
+ }
+ if (isOutlineRenaming && !isRenameButtonDisabled()) {
+ onRenameOutline();
+ }
+ }
+ if (e.key === 'Escape') {
+ onCancelOutline();
+ }
+ };
+
+ const onAddOutline = () => {
+ addNewOutline(outlineText.trim() === '' ? '' : outlineText);
+ };
+
+ const onRenameOutline = () => {
+ setOutlineRenaming(false);
+ renameOutline(outlinePath, outlineText);
+ };
+
+ const onCancelOutline = () => {
+ updateOutlines();
+ if (isOutlineRenaming) {
+ setOutlineRenaming(false);
+ setOutlineText(text);
+ }
+ if (isOutlineChangingDest) {
+ setOutlineChangingDest(false);
+ }
+ if (isAdding) {
+ onCancel();
+ }
+ };
+
+ const isRenameButtonDisabled = () => {
+ return !outlineText || text === outlineText;
+ };
+
+ useEffect(() => {
+ if (outlineText !== text) {
+ setOutlineText(text);
+ }
+ }, [text]);
+
+ useEffect(() => {
+ if (isAdding || isOutlineRenaming) {
+ inputRef.current.focus();
+ inputRef.current.select();
+ }
+
+ setIsDefault(!isAdding && !isOutlineRenaming && !isOutlineChangingDest);
+ }, [isOutlineRenaming, isOutlineChangingDest]);
+
+ useEffect(() => {
+ const editingOutlinesClone = { ...editingOutlines };
+ const isOutlineEditing = isOutlineRenaming || isOutlineChangingDest;
+ if (isOutlineEditing) {
+ editingOutlinesClone[outlinePath] = (isOutlineEditing);
+ } else {
+ delete editingOutlinesClone[outlinePath];
+ }
+ setEditingOutlines({ ...editingOutlinesClone });
+ }, [isOutlineRenaming, isOutlineChangingDest]);
+
+ useEffect(() => {
+ if (!isAdding) {
+ setIsHovered(isContextMenuOpen);
+ }
+ }, [isContextMenuOpen]);
+
+ const textStyle = {
+ color: textColor || 'auto'
+ };
+
+ return (
+
+ {isAdding &&
+
+ {t('component.newOutlineTitle')}
+
+ }
+
+ {isDefault &&
+ <>
+
{
+ if (isOutlineEditable) {
+ setOutlineRenaming(true);
+ }
+ }}
+ style={textStyle}
+ >
+ {text}
+
+
+ {isOutlineEditable &&
+
{
+ e.stopPropagation();
+ setContextMenuOpen(true);
+ }}
+ />
+ }
+ {isContextMenuOpen &&
+ setContextMenuOpen(false)}
+ onRenameClick={() => {
+ setContextMenuOpen(false);
+ setOutlineRenaming(true);
+ }}
+ onSetDestinationClick={() => {
+ setContextMenuOpen(false);
+ setOutlineChangingDest(true);
+ core.setToolMode(TOOL_NAME);
+ }}
+ onDeleteClick={() => {
+ setContextMenuOpen(false);
+ removeOutlines([outlinePath]);
+ }}
+ />
+ }
+ >
+ }
+
+ {isOutlineChangingDest &&
+
+ {text}
+
+ }
+
+ {(isAdding || isOutlineRenaming) &&
+ setOutlineText(e.target.value)}
+ />
+ }
+
+ {(isAdding || isOutlineChangingDest) &&
+
+ {t('component.destination')}: {t('component.bookmarkPage')} {currentDestPage},
+ “{currentDestText}”
+
+ }
+
+ {(isAdding || isOutlineRenaming || isOutlineChangingDest) &&
+
+
+ {isAdding &&
+
+ }
+ {isOutlineRenaming &&
+
+ }
+ {isOutlineChangingDest &&
+ {
+ setOutlineChangingDest(false);
+ updateOutlineDest(outlinePath);
+ }}
+ />
+ }
+
+ }
+
+ );
+};
+
+OutlineContent.propTypes = propTypes;
+
+export default OutlineContent;
diff --git a/src/components/OutlineContent/OutlineContent.scss b/src/components/OutlineContent/OutlineContent.scss
new file mode 100644
index 0000000000..88577c5f59
--- /dev/null
+++ b/src/components/OutlineContent/OutlineContent.scss
@@ -0,0 +1,13 @@
+.outline-text,
+.outline-destination {
+ white-space: nowrap;
+ overflow: hidden;
+ text-overflow: ellipsis;
+}
+
+.outline-destination {
+ flex-basis: 100%;
+ font-size: 10px;
+ color: var(--faded-text);
+ margin-top: var(--padding-small);
+}
\ No newline at end of file
diff --git a/src/components/OutlineContent/OutlineContent.spec.js b/src/components/OutlineContent/OutlineContent.spec.js
new file mode 100644
index 0000000000..b6093a0fbb
--- /dev/null
+++ b/src/components/OutlineContent/OutlineContent.spec.js
@@ -0,0 +1,35 @@
+import React from 'react';
+import { render, fireEvent } from '@testing-library/react';
+import { Basic, Renaming, ColoredOutline } from './OutlineContent.stories';
+
+const BasicOutline = withProviders(Basic);
+const RenamingOutline = withProviders(Renaming);
+
+describe('Outline', () => {
+ it('Story should not throw any errors', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('Save button in renaming outline is disabled if text is empty or text is the same as current name', () => {
+ const { container } = render();
+
+ const saveButton = container.querySelector('.bookmark-outline-save-button');
+ expect(saveButton.disabled).toBe(true);
+
+ const textInput = container.querySelector('.bookmark-outline-input');
+ fireEvent.change(textInput, { target: { value: 'new outline' } });
+ expect(saveButton.disabled).toBe(false);
+
+ fireEvent.change(textInput, { target: { value: '' } });
+ expect(saveButton.disabled).toBe(true);
+ });
+
+ it('should set font color if textColor is passed to OutlineContent', () => {
+ const { container } = render();
+
+ const outline = container.querySelector('.bookmark-outline-text');
+ expect(outline.style.color).toBe('rgb(255, 0, 0)');
+ });
+});
diff --git a/src/components/OutlineContent/OutlineContent.stories.js b/src/components/OutlineContent/OutlineContent.stories.js
new file mode 100644
index 0000000000..e92329339b
--- /dev/null
+++ b/src/components/OutlineContent/OutlineContent.stories.js
@@ -0,0 +1,190 @@
+import React from 'react';
+import { legacy_createStore as createStore } from 'redux';
+import { Provider as ReduxProvider } from 'react-redux';
+import OutlineContent from './OutlineContent';
+import OutlineContext from '../Outline/Context';
+import '../LeftPanel/LeftPanel.scss';
+
+const NOOP = () => { };
+
+export default {
+ title: 'Components/OutlineContent',
+ component: OutlineContent,
+};
+
+const reducer = () => {
+ return {
+ viewer: {
+ disabledElements: {},
+ customElementOverrides: {},
+ isOutlineEditingEnabled: true,
+ },
+ document: {
+ outlines: {},
+ },
+ };
+};
+
+export const Basic = () => {
+ return (
+
+
+
+ );
+};
+
+export const Adding = () => {
+ return (
+
+
+
+ );
+};
+
+export const Renaming = () => {
+ return (
+
+
+
+ );
+};
+
+export const ChangingDestination = () => {
+ return (
+
+
+
+ );
+};
+
+export const ColoredOutline = () => {
+ return (
+
+
+
+ );
+};
\ No newline at end of file
diff --git a/src/components/OutlineContent/index.js b/src/components/OutlineContent/index.js
new file mode 100644
index 0000000000..a1e905f50e
--- /dev/null
+++ b/src/components/OutlineContent/index.js
@@ -0,0 +1,3 @@
+import OutlineContent from './OutlineContent';
+
+export default OutlineContent;
diff --git a/src/components/OutlinesPanel/OutlinesDragLayer.js b/src/components/OutlinesPanel/OutlinesDragLayer.js
new file mode 100644
index 0000000000..6eff6e115e
--- /dev/null
+++ b/src/components/OutlinesPanel/OutlinesDragLayer.js
@@ -0,0 +1,77 @@
+import React from 'react';
+import { useDragLayer } from 'react-dnd';
+import { ItemTypes } from 'constants/dnd';
+
+const layerStyles = {
+ position: 'fixed',
+ pointerEvents: 'none',
+ zIndex: 99999,
+ left: 0,
+ top: 0,
+ width: '100%',
+ height: '100%'
+};
+
+const getItemStyles = (initialOffset, currentOffset) => {
+ if (!initialOffset || !currentOffset) {
+ return {
+ display: 'none'
+ };
+ }
+ const { x, y } = currentOffset;
+ const transform = `translate(calc(${x}px - 50%), calc(${y}px - 100%))`;
+ return {
+ transform,
+ WebkitTransform: transform,
+ };
+};
+
+export const OutlinesDragLayer = () => {
+ const {
+ itemType,
+ item,
+ isDragging,
+ initialOffset,
+ currentOffset
+ } = useDragLayer((dragLayerState) => ({
+ itemType: dragLayerState.getItemType(),
+ item: dragLayerState.getItem(),
+ isDragging: dragLayerState.isDragging(),
+ initialOffset: dragLayerState.getInitialSourceClientOffset(),
+ currentOffset: dragLayerState.getClientOffset(),
+ }));
+
+ const renderDragItem = () => {
+ if (!item) {
+ return null;
+ }
+
+ const { dragOutline } = item;
+
+ switch (itemType) {
+ case ItemTypes.OUTLINE:
+ return (
+ <>
+ {dragOutline.getName()}
+ >
+ );
+ default:
+ return null;
+ }
+ };
+
+ if (!isDragging) {
+ return null;
+ }
+
+ return (
+
+
+ {renderDragItem()}
+
+
+ );
+};
diff --git a/src/components/Panel/Panel.stories.js b/src/components/Panel/Panel.stories.js
index df6c061d8c..ac207f171e 100644
--- a/src/components/Panel/Panel.stories.js
+++ b/src/components/Panel/Panel.stories.js
@@ -27,7 +27,7 @@ const initialState = {
panelWidths: { panel: DEFAULT_NOTES_PANEL_WIDTH },
sortStrategy: 'position',
isInDesktopOnlyMode: true,
- modularHeaders: []
+ modularHeaders: {}
}
};
diff --git a/src/components/PortfolioItem/PortfolioItem.scss b/src/components/PortfolioItem/PortfolioItem.scss
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/src/components/PortfolioItem/index.js b/src/components/PortfolioItem/index.js
new file mode 100644
index 0000000000..8711af14dc
--- /dev/null
+++ b/src/components/PortfolioItem/index.js
@@ -0,0 +1,3 @@
+import PortfolioItem from './PortfolioItem';
+
+export default PortfolioItem;
diff --git a/src/components/PortfolioItemContent/PortfolioItemContent.scss b/src/components/PortfolioItemContent/PortfolioItemContent.scss
index 65fbbe5495..3a2afe8993 100644
--- a/src/components/PortfolioItemContent/PortfolioItemContent.scss
+++ b/src/components/PortfolioItemContent/PortfolioItemContent.scss
@@ -1,5 +1,7 @@
.PortfolioPanel {
.bookmark-outline-single-container {
+ margin-top: 6px;
+ margin-bottom: 6px;
.bookmark-outline-label-row {
align-items: center;
diff --git a/src/components/PortfolioItemContent/index.js b/src/components/PortfolioItemContent/index.js
new file mode 100644
index 0000000000..0765d3b00b
--- /dev/null
+++ b/src/components/PortfolioItemContent/index.js
@@ -0,0 +1,3 @@
+import PortfolioItemContent from './PortfolioItemContent';
+
+export default PortfolioItemContent;
diff --git a/src/components/PortfolioPanel/PortfolioContext.js b/src/components/PortfolioPanel/PortfolioContext.js
new file mode 100644
index 0000000000..484ecdd0b5
--- /dev/null
+++ b/src/components/PortfolioPanel/PortfolioContext.js
@@ -0,0 +1,5 @@
+import React from 'react';
+
+const PortfolioContext = React.createContext();
+
+export default PortfolioContext;
\ No newline at end of file
diff --git a/src/components/PortfolioPanel/PortfolioDragLayer.js b/src/components/PortfolioPanel/PortfolioDragLayer.js
new file mode 100644
index 0000000000..be3ff191b0
--- /dev/null
+++ b/src/components/PortfolioPanel/PortfolioDragLayer.js
@@ -0,0 +1,76 @@
+import React from 'react';
+import { useDragLayer } from 'react-dnd';
+import { ItemTypes } from 'constants/dnd';
+
+const layerStyles = {
+ position: 'fixed',
+ pointerEvents: 'none',
+ zIndex: 99999,
+ left: 0,
+ top: 0,
+ width: '100%',
+ height: '100%'
+};
+
+const getItemStyles = (initialOffset, currentOffset) => {
+ if (!initialOffset || !currentOffset) {
+ return {
+ display: 'none'
+ };
+ }
+ const { x, y } = currentOffset;
+ const transform = `translate(calc(${x}px - 50%), calc(${y}px - 100%))`;
+ return {
+ transform,
+ WebkitTransform: transform,
+ };
+};
+
+export const PortfolioDragLayer = () => {
+ const {
+ itemType,
+ item,
+ isDragging,
+ initialOffset,
+ currentOffset
+ } = useDragLayer((dragLayerState) => ({
+ itemType: dragLayerState.getItemType(),
+ item: dragLayerState.getItem(),
+ isDragging: dragLayerState.isDragging(),
+ initialOffset: dragLayerState.getInitialSourceClientOffset(),
+ currentOffset: dragLayerState.getClientOffset(),
+ }));
+
+ const renderDragItemPreview = () => {
+ if (!item) {
+ return null;
+ }
+
+ const { dragPortfolioItem } = item;
+
+ if (itemType === ItemTypes.PORTFOLIO) {
+ return (
+ <>
+ {dragPortfolioItem.name}
+ >
+ );
+ }
+
+ return null;
+ };
+
+ if (!isDragging) {
+ return null;
+ }
+
+ return (
+
+
+ {renderDragItemPreview()}
+
+
+ );
+};
diff --git a/src/components/PortfolioPanel/PortfolioPanel.scss b/src/components/PortfolioPanel/PortfolioPanel.scss
new file mode 100644
index 0000000000..a4a0d22370
--- /dev/null
+++ b/src/components/PortfolioPanel/PortfolioPanel.scss
@@ -0,0 +1,12 @@
+@import '../../constants/styles';
+@import '../../constants/panel';
+
+.PortfolioPanel {
+ .portfolio-panel-control {
+ display: flex;
+ }
+
+ .bookmark-outline-row {
+ padding-top: 6px;
+ }
+}
\ No newline at end of file
diff --git a/src/components/PortfolioPanel/index.js b/src/components/PortfolioPanel/index.js
new file mode 100644
index 0000000000..33783b3238
--- /dev/null
+++ b/src/components/PortfolioPanel/index.js
@@ -0,0 +1,3 @@
+import PortfolioPanel from './PortfolioPanel';
+
+export default PortfolioPanel;
diff --git a/src/components/ReactSelectCustomArrowIndicator/ReactSelectCustomArrowIndicator.js b/src/components/ReactSelectCustomArrowIndicator/ReactSelectCustomArrowIndicator.js
new file mode 100644
index 0000000000..db31b603bb
--- /dev/null
+++ b/src/components/ReactSelectCustomArrowIndicator/ReactSelectCustomArrowIndicator.js
@@ -0,0 +1,15 @@
+import React from 'react';
+import { components } from 'react-select';
+import Icon from 'components/Icon';
+
+const ReactSelectCustomArrowIndicator = (props) => {
+ const { selectProps } = props;
+ const { menuIsOpen } = selectProps;
+ return (
+
+
+
+ );
+};
+
+export default ReactSelectCustomArrowIndicator;
\ No newline at end of file
diff --git a/src/components/ReactSelectCustomArrowIndicator/index.js b/src/components/ReactSelectCustomArrowIndicator/index.js
new file mode 100644
index 0000000000..1a2e698e7f
--- /dev/null
+++ b/src/components/ReactSelectCustomArrowIndicator/index.js
@@ -0,0 +1,3 @@
+import ReactSelectCustomArrowIndicator from './ReactSelectCustomArrowIndicator';
+
+export default ReactSelectCustomArrowIndicator;
\ No newline at end of file
diff --git a/src/components/ReactSelectWebComponentProvider/index.js b/src/components/ReactSelectWebComponentProvider/index.js
new file mode 100644
index 0000000000..3a1f0e7dc1
--- /dev/null
+++ b/src/components/ReactSelectWebComponentProvider/index.js
@@ -0,0 +1,3 @@
+import ReactSelectWebComponentProvider from './ReactSelectWebComponentProvider';
+
+export default ReactSelectWebComponentProvider;
\ No newline at end of file
diff --git a/src/components/RedactionPanel/RedactionPanel.stories.js b/src/components/RedactionPanel/RedactionPanel.stories.js
index b1f2574537..b5af2f69fb 100644
--- a/src/components/RedactionPanel/RedactionPanel.stories.js
+++ b/src/components/RedactionPanel/RedactionPanel.stories.js
@@ -6,13 +6,20 @@ import RedactionPanelContainerWithProvider from './RedactionPanelContainer';
import RightPanel from 'components/RightPanel';
import { RedactionPanelContext } from './RedactionPanelContext';
import { defaultRedactionTypes, redactionTypeMap } from 'constants/redactionTypes';
+import Panel from 'components/Panel';
const noop = () => { };
export default {
title: 'Components/RedactionPanel',
component: RedactionPanel,
- includeStories: ['EmptyList', 'PanelWithRedactionItems', 'RedactionPanelWithSearch']
+ includeStories: [
+ 'EmptyList', 'PanelWithRedactionItems', 'RedactionPanelWithSearch',
+ 'RedactionLeftGenericPanel',
+ 'RedactionRightGenericPanel',
+ 'RightPanelWithRedactionItems',
+ 'LeftPanelWithRedactionItems',
+ ]
};
export const RedactionContextMock = ({ children, mockContext }) => {
@@ -50,12 +57,14 @@ const initialState = {
openElements: {
header: true,
redactionPanel: true,
+ panel: true,
},
currentLanguage: 'en',
panelWidths: {
redactionPanel: 330,
+ panel: 300,
},
- modularHeaders: [],
+ modularHeaders: {},
modularHeadersHeight: {
topHeaders: 40,
bottomHeaders: 40
@@ -200,7 +209,65 @@ export function PanelWithRedactionItems() {
export function RedactionPanelWithSearch() {
return (
-
+
);
}
+
+
+export function RedactionLeftGenericPanel() {
+ return (
+
+
+
+
+
+
+
+
+
+ );
+}
+
+export function RedactionRightGenericPanel() {
+ return (
+
+
+
+
+
+
+
+
+
+ );
+}
+
+
+export function RightPanelWithRedactionItems() {
+ return (
+
+
+
+
+
+
+
+
+
+ );
+}
+
+export function LeftPanelWithRedactionItems() {
+ return (
+
+
+
+
+
+
+
+
+
+ );
+}
\ No newline at end of file
diff --git a/src/components/RedactionSearchResultGroup/RedactionSearchResultGroup.scss b/src/components/RedactionSearchResultGroup/RedactionSearchResultGroup.scss
index 7c35f384a6..aa3e037d00 100644
--- a/src/components/RedactionSearchResultGroup/RedactionSearchResultGroup.scss
+++ b/src/components/RedactionSearchResultGroup/RedactionSearchResultGroup.scss
@@ -1,7 +1,7 @@
.redaction-search-results-page-number {
display: flex;
align-items: center;
- margin-left: 11px;
+ margin-left: 11px !important;
label {
font-size: 13px;
diff --git a/src/components/ReplyAttachmentList/ReplyAttachmentList.scss b/src/components/ReplyAttachmentList/ReplyAttachmentList.scss
new file mode 100644
index 0000000000..910cd5b46b
--- /dev/null
+++ b/src/components/ReplyAttachmentList/ReplyAttachmentList.scss
@@ -0,0 +1,84 @@
+.reply-attachment-list {
+ display: flex;
+ flex-direction: column;
+ cursor: default;
+
+ .reply-attachment {
+ display: flex;
+ flex-direction: column;
+ background-color: var(--gray-1);
+ border-radius: 4px;
+ cursor: pointer;
+ overflow: hidden;
+
+ &:not(:last-child) {
+ margin-bottom: 8px;
+ }
+
+ .reply-attachment-preview {
+ width: 100%;
+ max-height: 80px;
+ display: flex;
+ justify-content: center;
+
+ &.dirty {
+ position: relative;
+ margin-bottom: 15px;
+ }
+
+ img {
+ max-width: 100%;
+ max-height: 100%;
+ object-fit: contain;
+ }
+
+ .reply-attachment-preview-message {
+ font-size: 11px;
+ color: var(--error-text-color);
+ position: absolute;
+ bottom: -15px;
+ left: 10px;
+ }
+ }
+
+ .reply-attachment-info {
+ display: flex;
+ align-items: center;
+ height: 40px;
+ padding: 8px;
+
+ .reply-attachment-icon {
+ height: 24px;
+ min-height: 24px;
+ width: 24px;
+ min-width: 24px;
+ }
+
+ .reply-file-name {
+ height: 16px;
+ width: 100%;
+ white-space: nowrap;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ margin-left: 8px;
+ margin-right: 8px;
+ }
+
+ .attachment-button {
+ height: 24px;
+ min-height: 24px;
+ width: 24px;
+ min-width: 24px;
+
+ &:hover {
+ background-color: var(--blue-1);
+ }
+
+ .Icon {
+ height: 16px;
+ width: 16px;
+ }
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/components/ReplyAttachmentList/ReplyAttachmentList.stories.js b/src/components/ReplyAttachmentList/ReplyAttachmentList.stories.js
new file mode 100644
index 0000000000..f994ca3e33
--- /dev/null
+++ b/src/components/ReplyAttachmentList/ReplyAttachmentList.stories.js
@@ -0,0 +1,95 @@
+import React from 'react';
+import ReplyAttachmentList from './ReplyAttachmentList';
+import { createStore } from 'redux';
+import { Provider } from 'react-redux';
+
+export default {
+ title: 'Components/ReplyAttachmentList',
+ component: ReplyAttachmentList
+};
+
+const initialState = {
+ viewer: {
+ disabledElements: {},
+ customElementOverrides: {},
+ replyAttachmentPreviewEnabled: true,
+ }
+};
+function rootReducer(state = initialState, action) {
+ return state;
+}
+const store = createStore(rootReducer);
+
+const files = [
+ {
+ name: 'file_1.pdf'
+ },
+ {
+ name: 'file_2.doc'
+ },
+ {
+ name: 'file_3_extra_long_file_name.cad'
+ }
+];
+
+// State 1
+export function DisplayMode() {
+ const props = {
+ files,
+ isEditing: false
+ };
+
+ return (
+
+
+
+
+
+ );
+}
+
+// State 2
+export function EditMode() {
+ const props = {
+ files,
+ isEditing: true
+ };
+
+ return (
+
+
+
+
+
+ );
+}
+
+const SVG_MIME_TYPE = 'image/svg+xml';
+const svgString = `
+
+
+
+`;
+const svgBlob = new Blob([svgString], { type: SVG_MIME_TYPE });
+const svgFile = new File([svgBlob], 'redirect.svg', { type: SVG_MIME_TYPE });
+
+// State 3
+export function UnsafeSVGAttachment() {
+ const props = {
+ files: [...files, svgFile],
+ isEditing: false
+ };
+
+ return (
+
+
+
+
+
+ );
+}
diff --git a/src/components/ReplyAttachmentList/index.js b/src/components/ReplyAttachmentList/index.js
new file mode 100644
index 0000000000..4487c18bf9
--- /dev/null
+++ b/src/components/ReplyAttachmentList/index.js
@@ -0,0 +1,3 @@
+import ReplyAttachmentList from './ReplyAttachmentList';
+
+export default ReplyAttachmentList;
\ No newline at end of file
diff --git a/src/components/RichTextStyleEditor/RichTextStyleEditor.stories.js b/src/components/RichTextStyleEditor/RichTextStyleEditor.stories.js
new file mode 100644
index 0000000000..6c095e8388
--- /dev/null
+++ b/src/components/RichTextStyleEditor/RichTextStyleEditor.stories.js
@@ -0,0 +1,117 @@
+import React from 'react';
+import { configureStore } from '@reduxjs/toolkit';
+import initialState from 'src/redux/initialState';
+import { Provider } from 'react-redux';
+import RichTextStyleEditor from './RichTextStyleEditor';
+import Panel from '../Panel';
+import '../StylePicker/StylePicker.scss';
+import { initialTextColors } from 'helpers/initialColorStates';
+
+export default {
+ title: 'Components/RichTextStyleEditor',
+ component: RichTextStyleEditor,
+};
+
+// Mock some state to show the style popups
+const state = {
+ ...initialState,
+ viewer: {
+ openElements: {
+ watermarkPanel: true,
+ stylePopup: true,
+ stylePanel: true,
+ stylePopupTextStyleContainer: true,
+ stylePopupColorsContainer: true,
+ stylePopupLabelTextContainer: true,
+ panel: true,
+ header: true,
+ },
+ selectedScale: undefined,
+ colorMap: {
+ textField: {
+ currentStyleTab: 'StrokeColor',
+ iconColor: 'StrokeColor',
+ }
+ },
+ fonts: ['Helvetica', 'Times New Roman', 'Arimo'],
+ isSnapModeEnabled: false,
+ customElementOverrides: {},
+ panelWidths: { panel: 264 },
+ colors: [
+ '#fdac0f', '#fa9933', '#f34747', '#21905b', '#c531a4',
+ '#e5631a', '#3e5ece', '#dc9814', '#c27727', '#b11c1c',
+ '#13558c', '#76287b', '#347842', '#318f29', '#ffffff',
+ '#cdcdcd', '#9c9c9c', '#696969', '#272727', '#000000'
+ ],
+ textColors: initialTextColors,
+ toolColorOverrides: {},
+ disabledElements: {
+ logoBar: { disabled: true },
+ },
+ sortStrategy: 'position',
+ isInDesktopOnlyMode: true,
+ modularHeaders: {}
+ }
+};
+
+const noop = () => {};
+
+const store = configureStore({
+ reducer: () => state
+});
+
+const BasicComponent = (props) => {
+ return (
+
+
+
+
+
+
+
+ );
+};
+
+export const Basic = BasicComponent.bind({
+ annotation: '',
+ editor: {},
+ style: {},
+ isFreeTextAutoSize: false,
+ onFreeTextSizeToggle: () => {},
+ onPropertyChange: () => {},
+ onRichTextStyleChange: () => {},
+});
+Basic.args = {
+ currentStyleTab: 'StrokeColor',
+ isInFormBuilderAndNotFreeText: true,
+ style: {
+ 'FillColor': new window.Core.Annotations.Color(212, 211, 211),
+ 'StrokeColor': new window.Core.Annotations.Color(0, 0, 0),
+ 'TextColor': new window.Core.Annotations.Color(0, 0, 0),
+ 'Opacity': null,
+ 'StrokeThickness': 1,
+ 'FontSize': '12pt',
+ 'Style': 'solid'
+ },
+ colorMapKey: 'textField',
+ colorPalette: 'StrokeColor',
+ disableSeparator: true,
+ hideSnapModeCheckbox: true,
+ isFreeText: false,
+ isEllipse: false,
+ isTextStyleContainerActive: true,
+ isLabelTextContainerActive: true,
+ properties: {
+ 'StrokeStyle': 'solid',
+ },
+ isRedaction: false,
+ fonts: ['Helvetica', 'Times New Roman', 'Arimo'],
+ isSnapModeEnabled: false,
+ onSliderChange: noop,
+ onStyleChange: noop,
+ closeElement: noop,
+ openElement: noop,
+ onPropertyChange: noop,
+ onRichTextStyleChange: noop,
+ onLineStyleChange: noop,
+};
diff --git a/src/components/RichTextStyleEditor/index.js b/src/components/RichTextStyleEditor/index.js
new file mode 100644
index 0000000000..54d5259661
--- /dev/null
+++ b/src/components/RichTextStyleEditor/index.js
@@ -0,0 +1,3 @@
+import RichTextStyleEditor from './RichTextStyleEditor';
+
+export default RichTextStyleEditor;
\ No newline at end of file
diff --git a/src/components/RightPanel/RightPanel.js b/src/components/RightPanel/RightPanel.js
index bafca271cf..5b965d5b61 100644
--- a/src/components/RightPanel/RightPanel.js
+++ b/src/components/RightPanel/RightPanel.js
@@ -50,7 +50,10 @@ const RightPanel = ({ children, dataElement, onResize }) => {
const legacyToolsHeaderOpen = isToolsHeaderOpen && currentToolbarGroup !== 'toolbarGroup-View';
const legacyAllHeadersHidden = !isHeaderOpen && !legacyToolsHeaderOpen;
const { customizableUI } = featureFlags;
- const style = {};
+ const style = {
+ // prevent panel from appearing until scss is loaded
+ display: 'none',
+ };
// Calculating its height according to the existing horizontal modular headers
if (customizableUI) {
const horizontalHeadersHeight = topHeadersHeight + bottomHeadersHeight;
diff --git a/src/components/ScaleModal/ScaleCustom.js b/src/components/ScaleModal/ScaleCustom.js
new file mode 100644
index 0000000000..ab1198e133
--- /dev/null
+++ b/src/components/ScaleModal/ScaleCustom.js
@@ -0,0 +1,318 @@
+import React, { useState, useEffect, useRef } from 'react';
+import PropTypes from 'prop-types';
+import { useSelector, shallowEqual } from 'react-redux';
+import { useTranslation } from 'react-i18next';
+import selectors from 'selectors';
+import Tooltip from '../Tooltip';
+import Dropdown from '../Dropdown';
+import {
+ ifFractionalPrecision,
+ hintValues,
+ convertUnit,
+ fractionalUnits,
+ floatRegex,
+ inFractionalRegex,
+ ftInFractionalRegex,
+ ftInDecimalRegex,
+ parseFtInDecimal,
+ parseInFractional,
+ parseFtInFractional
+} from 'constants/measurementScale';
+import classNames from 'classnames';
+
+const Scale = window.Core.Scale;
+
+const ScaleCustomProps = {
+ scale: PropTypes.array,
+ onScaleChange: PropTypes.func,
+ precision: PropTypes.number
+};
+
+function ScaleCustom({ scale, onScaleChange, precision }) {
+ const [measurementUnits] = useSelector((state) => [selectors.getMeasurementUnits(state)], shallowEqual);
+ const [pageValueDisplay, setPageValueDisplay] = useState('');
+ const [worldValueDisplay, setWorldValueDisplay] = useState('');
+ const [isFractionalPrecision, setIsFractionalPrecision] = useState(false);
+ const [pageWarningMessage, setPageWarningMessage] = useState('');
+ const [worldWarningMessage, setWorldWarningMessage] = useState('');
+ const [scaleValueBlurFlag, setScaleValueBlurFlag] = useState(false);
+
+ const pageValueInput = useRef(null);
+ const worldValueInput = useRef(null);
+
+ const [t] = useTranslation();
+
+ const filterFractionalUnits = (units) => units.filter((unit) => fractionalUnits.includes(unit));
+ const unitFromOptions = isFractionalPrecision ? filterFractionalUnits(measurementUnits.from) : measurementUnits.from;
+ const unitToOptions = isFractionalPrecision ? filterFractionalUnits(measurementUnits.to) : measurementUnits.to;
+
+ // If our scale has a unit that is not in the current 'from' measurement units, change it
+ // to the first unit in the list.
+ useEffect(() => {
+ if (!unitFromOptions.includes(scale[0][1])) {
+ onScaleUnitChange(unitFromOptions[0], true);
+ }
+ }, [scale[0][1]]);
+
+ // If our scale has a unit that is not in the current 'to' measurement units, change it
+ // to the first unit in the list. We want to wait until the 'from' unit is valid before
+ // setting the 'to' unit. Otherwise, we will reset the 'from' unit.
+ useEffect(() => {
+ if (unitFromOptions.includes(scale[0][1]) && !unitToOptions.includes(scale[1][1])) {
+ onScaleUnitChange(unitToOptions[0], false);
+ }
+ }, [scale[0][1], scale[1][1]]);
+
+ useEffect(() => {
+ const formatDecimal = (value) => {
+ return value?.toFixed((1 / precision).toString().length - 1);
+ };
+
+ if (scale[0][0] && pageValueInput?.current !== document.activeElement) {
+ if (!isFractionalPrecision) {
+ setPageValueDisplay(formatDecimal(scale[0][0]) || '');
+ } else {
+ setPageValueDisplay(Scale.getFormattedValue(scale[0][0], scale[0][1], precision, false, true) || '');
+ }
+ }
+ if (scale[1][0] && !(worldValueInput && worldValueInput.current === document.activeElement)) {
+ if (!isFractionalPrecision && scale[1][1] !== 'ft-in') {
+ setWorldValueDisplay(formatDecimal(scale[1][0]) || '');
+ } else {
+ setWorldValueDisplay(Scale.getFormattedValue(scale[1][0], scale[1][1], precision, false, true) || '');
+ }
+ }
+ }, [scale, precision, worldValueInput, pageValueInput, isFractionalPrecision, scaleValueBlurFlag]);
+
+ useEffect(() => {
+ setIsFractionalPrecision(ifFractionalPrecision(precision));
+ }, [precision]);
+
+ useEffect(() => {
+ if (isFractionalPrecision) {
+ setPageWarningMessage(hintValues[scale[0][1]]);
+ setWorldWarningMessage(hintValues[scale[1][1]]);
+ } else if (scale[1][1] === 'ft-in') {
+ setPageWarningMessage('');
+ setWorldWarningMessage(hintValues['ft-in decimal']);
+ } else {
+ setPageWarningMessage('');
+ setWorldWarningMessage('');
+ }
+ }, [scale, isFractionalPrecision]);
+
+ // Re-validate invalid world value input when world unit changes
+ useEffect(() => {
+ !isWorldValueValid && onInputValueChange(worldValueInput.current.value, false);
+ }, [scale[1][1]]);
+
+ // Re-validate invalid scale value input when isFractionalPrecision value changes
+ useEffect(() => {
+ if (!isPageValueValid && !isWorldValueValid) {
+ let pageScale = {
+ value: scale[0][0],
+ unit: scale[0][1]
+ };
+ onInputValueChange(pageValueInput.current.value, true, (newScale) => {
+ pageScale = newScale.pageScale;
+ });
+ let worldScale = {
+ value: scale[1][0],
+ unit: scale[1][1]
+ };
+ onInputValueChange(worldValueInput.current.value, false, (newScale) => {
+ worldScale = newScale.worldScale;
+ });
+
+ _onScaleChange(new Scale({ pageScale, worldScale }));
+ } else {
+ !isPageValueValid && onInputValueChange(pageValueInput.current.value, true);
+ !isWorldValueValid && onInputValueChange(worldValueInput.current.value, false);
+ }
+ }, [isFractionalPrecision]);
+
+ const isPageValueValid = !!scale[0][0];
+ const isWorldValueValid = !!scale[1][0];
+
+ const pageValueClass = classNames('scale-input', {
+ 'invalid-value': !isPageValueValid
+ });
+ const worldValueClass = classNames('scale-input', {
+ 'invalid-value': !isWorldValueValid
+ });
+
+ // If scale value is smaller than the current precision, replace it with precision value to prevent 0 value.
+ const _onScaleChange = (newScale) => {
+ const getPrecision = (unit) => (unit === 'ft-in' ? precision / 12 : precision);
+
+ if (newScale.pageScale.value && newScale.pageScale.value < precision) {
+ newScale.pageScale.value = getPrecision(newScale.pageScale.unit);
+ }
+ if (newScale.worldScale.value && newScale.worldScale.value < precision) {
+ newScale.worldScale.value = getPrecision(newScale.worldScale.unit);
+ }
+ onScaleChange(newScale);
+ };
+
+ const onInputValueChange = (value, isPageValue, getNewScale) => {
+ const updateScaleValue = (scaleValue) => {
+ if ((isPageValue && scaleValue !== scale[0][0]) || (!isPageValue && scaleValue !== scale[1][0])) {
+ const newScale = new Scale({
+ pageScale: { value: isPageValue ? scaleValue : scale[0][0], unit: scale[0][1] },
+ worldScale: { value: !isPageValue ? scaleValue : scale[1][0], unit: scale[1][1] }
+ });
+ if (getNewScale) {
+ getNewScale(newScale);
+ } else {
+ _onScaleChange(newScale);
+ }
+ }
+ };
+
+ if (isPageValue) {
+ setPageValueDisplay(value);
+ } else {
+ setWorldValueDisplay(value);
+ }
+ const inputValue = value.trim();
+ if (!isFractionalPrecision) {
+ if (!isPageValue && scale[1][1] === 'ft-in') {
+ if (ftInDecimalRegex.test(inputValue)) {
+ const result = parseFtInDecimal(inputValue);
+ if (result > 0) {
+ updateScaleValue(result);
+ return;
+ }
+ }
+ } else if (floatRegex.test(inputValue)) {
+ const scaleValue = parseFloat(inputValue) || 0;
+ updateScaleValue(scaleValue);
+ return;
+ }
+ } else {
+ const scaleUnit = isPageValue ? scale[0][1] : scale[1][1];
+ if (scaleUnit === 'in') {
+ if (inFractionalRegex.test(inputValue)) {
+ const result = parseInFractional(inputValue);
+ if (result > 0) {
+ updateScaleValue(result);
+ return;
+ }
+ }
+ } else if (scaleUnit === 'ft-in') {
+ if (ftInFractionalRegex.test(inputValue)) {
+ const result = parseFtInFractional(inputValue);
+ if (result > 0) {
+ updateScaleValue(result);
+ return;
+ }
+ }
+ }
+ }
+ updateScaleValue(undefined);
+ };
+
+ const onScaleUnitChange = (newUnit, isPageUnit) => {
+ let newPageScale;
+ if (isPageUnit && newUnit !== scale[0][1]) {
+ newPageScale = {
+ value: scale[0][0] ? convertUnit(scale[0][0], scale[0][1], newUnit) : scale[0][0],
+ unit: newUnit
+ };
+ } else {
+ newPageScale = { value: scale[0][0], unit: scale[0][1] };
+ }
+ let newWorldScale;
+ if (!isPageUnit && newUnit !== scale[1][1]) {
+ newWorldScale = {
+ value: scale[1][0] ? convertUnit(scale[1][0], scale[1][1], newUnit) : scale[1][0],
+ unit: newUnit
+ };
+ } else {
+ newWorldScale = { value: scale[1][0], unit: scale[1][1] };
+ }
+
+ _onScaleChange(new Scale({ pageScale: newPageScale, worldScale: newWorldScale }));
+ };
+
+ const getInputPlaceholder = (isPageValue) => {
+ const unit = isPageValue ? scale[0][1] : scale[1][1];
+ return isFractionalPrecision ? hintValues[unit] : (unit === 'ft-in' ? hintValues['ft-in decimal'] : '');
+ };
+
+ const onInputBlur = () => {
+ setScaleValueBlurFlag((flag) => !flag);
+ };
+
+ return (
+
+
+
+
+
onInputValueChange(e.target.value, true)}
+ placeholder={getInputPlaceholder(true)}
+ ref={pageValueInput}
+ onBlur={onInputBlur}
+ />
+
+
+ onScaleUnitChange(value, true)}
+ currentSelectionKey={scale[0][1]}
+ />
+
+
+
+ {' = '}
+
+
onInputValueChange(e.target.value, false)}
+ placeholder={getInputPlaceholder(false)}
+ ref={worldValueInput}
+ onBlur={onInputBlur}
+ />
+
+
+ onScaleUnitChange(value, false)}
+ currentSelectionKey={scale[1][1]}
+ />
+
+
+
+
+
+
+ {!isPageValueValid && (
+
+ {`${t('option.measurement.scaleModal.incorrectSyntax')} ${pageWarningMessage}`}
+
+ )}
+ {!isWorldValueValid && (
+
+ {`${t('option.measurement.scaleModal.incorrectSyntax')} ${worldWarningMessage}`}
+
+ )}
+
+
+ );
+}
+
+ScaleCustom.propTypes = ScaleCustomProps;
+
+export default ScaleCustom;
diff --git a/src/components/ScaleModal/index.js b/src/components/ScaleModal/index.js
new file mode 100644
index 0000000000..73f40ff610
--- /dev/null
+++ b/src/components/ScaleModal/index.js
@@ -0,0 +1,3 @@
+import ScaleModal from './ScaleModal';
+
+export default ScaleModal;
\ No newline at end of file
diff --git a/src/components/ScaleOverlay/MeasurementDetail.spec.js b/src/components/ScaleOverlay/MeasurementDetail.spec.js
new file mode 100644
index 0000000000..1aaee4fa38
--- /dev/null
+++ b/src/components/ScaleOverlay/MeasurementDetail.spec.js
@@ -0,0 +1,62 @@
+import React from 'react';
+import { render, screen, fireEvent } from '@testing-library/react';
+import { EllipseScaleOverlay } from './MeasurementDetail.stories';
+
+const noop = () => { };
+
+jest.mock('core', () => ({
+ addEventListener: noop,
+ removeEventListener: noop,
+ jumpToAnnotation: noop,
+ getDocumentViewer: () => ({
+ getAnnotationManager: () => ({
+ deselectAllAnnotations: noop,
+ selectAnnotation: noop,
+ })
+ }),
+ getAnnotationManager: () => ({
+ selectAnnotation: noop,
+ redrawAnnotation: noop,
+ trigger: noop
+ }),
+ getTool: () => ({
+ finish: noop
+ })
+}));
+
+
+describe('MeasurementDetail', () => {
+ it('renders the EllipseScaleOverlay storybook component', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('increases the area when the radius increases', () => {
+ render();
+
+ const radiusInput = screen.getByDisplayValue('1.74');
+ const areaTextElement = screen.getByText('10.00 sq in');
+
+ const beforeText = areaTextElement.textContent;
+ fireEvent.change(radiusInput, { target: { value: '2' } });
+ const afterText = areaTextElement.textContent;
+ const beforeValue = parseFloat(beforeText.substring(0, beforeText.indexOf(' ')));
+ const afterValue = parseFloat(afterText.substring(0, afterText.indexOf(' ')));
+
+ expect(beforeValue).toBeLessThan(afterValue);
+ });
+
+ it('sets area to zero when radius is zero', () => {
+ render();
+
+ const radiusInput = screen.getByDisplayValue('1.74');
+ const areaTextElement = screen.getByText('10.00 sq in');
+
+ fireEvent.change(radiusInput, { target: { value: '0' } });
+ const afterText = areaTextElement.textContent;
+ const afterValue = parseFloat(afterText.substring(0, afterText.indexOf(' ')));
+
+ expect(afterValue).toEqual(0);
+ });
+});
\ No newline at end of file
diff --git a/src/components/ScaleOverlay/ScaleHeader.js b/src/components/ScaleOverlay/ScaleHeader.js
new file mode 100644
index 0000000000..aaffcb4eed
--- /dev/null
+++ b/src/components/ScaleOverlay/ScaleHeader.js
@@ -0,0 +1,36 @@
+import React from 'react';
+import Icon from 'components/Icon';
+import ScaleSelector from './ScaleSelector';
+import PropTypes from 'prop-types';
+import { useTranslation } from 'react-i18next';
+
+const propTypes = {
+ scales: PropTypes.arrayOf(PropTypes.object).isRequired,
+ selectedScales: PropTypes.arrayOf(PropTypes.string).isRequired,
+ onScaleSelected: PropTypes.func.isRequired,
+ onAddingNewScale: PropTypes.func.isRequired,
+};
+
+const ScaleHeader = ({ scales, selectedScales, onScaleSelected, onAddingNewScale }) => {
+ const [t] = useTranslation();
+
+ return (
+
+
+
{t('option.measurementOption.scale')}
+ {scales.length ? (
+
+ ) : (
+
{t('option.measurement.scaleOverlay.addNewScale')}
+ )}
+
+ );
+};
+
+ScaleHeader.propTypes = propTypes;
+export default ScaleHeader;
diff --git a/src/components/SearchPanel/SearchPanel.js b/src/components/SearchPanel/SearchPanel.js
index a9f73d2d66..94c7b0a094 100644
--- a/src/components/SearchPanel/SearchPanel.js
+++ b/src/components/SearchPanel/SearchPanel.js
@@ -20,7 +20,8 @@ const propTypes = {
setActiveResult: PropTypes.func,
isInDesktopOnlyMode: PropTypes.bool,
isProcessingSearchResults: PropTypes.bool,
- activeDocumentViewerKey: PropTypes.number
+ activeDocumentViewerKey: PropTypes.number,
+ isCustomPanel: PropTypes.bool,
};
function noop() { }
@@ -36,7 +37,9 @@ function SearchPanel(props) {
isMobile = false,
isInDesktopOnlyMode,
isProcessingSearchResults,
- activeDocumentViewerKey
+ activeDocumentViewerKey,
+ dataElement = 'searchPanel',
+ isCustomPanel = false,
} = props;
const { t } = useTranslation();
@@ -76,12 +79,15 @@ function SearchPanel(props) {
}, []);
const className = getClassName('Panel SearchPanel', { isOpen });
- const style = !isInDesktopOnlyMode && isMobile ? {} : { width: `${currentWidth}px`, minWidth: `${currentWidth}px` };
+ let style = {};
+ if (!isCustomPanel && (isInDesktopOnlyMode || !isMobile)) {
+ style = { width: `${currentWidth}px`, minWidth: `${currentWidth}px` };
+ }
return (
{!isInDesktopOnlyMode && isMobile &&
diff --git a/src/components/SearchPanel/SearchPanel.stories.js b/src/components/SearchPanel/SearchPanel.stories.js
new file mode 100644
index 0000000000..c5c9dcece5
--- /dev/null
+++ b/src/components/SearchPanel/SearchPanel.stories.js
@@ -0,0 +1,46 @@
+import React from 'react';
+import SearchPanel from './SearchPanelContainer';
+import { Provider } from 'react-redux';
+import { configureStore } from '@reduxjs/toolkit';
+import Panel from 'components/Panel';
+
+export default {
+ title: 'ModularComponents/SearchPanel',
+ component: SearchPanel
+};
+
+const initialState = {
+ viewer: {
+ openElements: {
+ panel: true,
+ },
+ disabledElements: {},
+ customElementOverrides: {},
+ tab: {},
+ panelWidths: { panel: 300 },
+ modularHeaders: {},
+ },
+ search: {},
+};
+
+const store = configureStore({ reducer: () => initialState });
+
+export function SearchPanelLeft() {
+ return (
+
+
+
+
+
+ );
+}
+
+export function SearchPanelRight() {
+ return (
+ initialState })}>
+
+
+
+
+ );
+}
diff --git a/src/components/Selector/Selector.js b/src/components/Selector/Selector.js
new file mode 100644
index 0000000000..323102e77d
--- /dev/null
+++ b/src/components/Selector/Selector.js
@@ -0,0 +1,51 @@
+import classNames from 'classnames';
+import PropTypes from 'prop-types';
+import React from 'react';
+import Icon from 'components/Icon';
+import './Selector.scss';
+
+const propTypes = {
+ className: PropTypes.string,
+ items: PropTypes.arrayOf(PropTypes.string).isRequired,
+ selectedItem: PropTypes.string,
+ onItemSelected: PropTypes.func.isRequired,
+ placeHolder: PropTypes.string,
+ selectedItemStyle: PropTypes.object,
+};
+
+const Selector = ({ className, items = [], selectedItem = '', onItemSelected, placeHolder, selectedItemStyle }) => {
+ return (
+
+
+ {!selectedItem && placeHolder ? placeHolder : selectedItem}
+
+
+
+
+ );
+};
+
+Selector.propTypes = propTypes;
+
+export default Selector;
diff --git a/src/components/Selector/Selector.scss b/src/components/Selector/Selector.scss
new file mode 100644
index 0000000000..157cc5fa68
--- /dev/null
+++ b/src/components/Selector/Selector.scss
@@ -0,0 +1,93 @@
+.customSelector {
+ position: relative;
+ .customSelector__selectedItem {
+ height: 2rem;
+ width: 8.5rem;
+ position: relative;
+ background-color: transparent;
+ border: solid 1px;
+ border-color: var(--border);
+ padding: 0 4px 0 8px;
+ color: var(--text-color);
+ font-family: Lato;
+ font-style: normal;
+ font-weight: normal;
+ font-size: 0.8rem;
+ text-align: left;
+ border-radius: 0.3rem;
+ display: flex;
+ flex-direction: row;
+ justify-content: space-between;
+ align-items: center;
+ }
+
+ & ul {
+ margin: 0;
+ list-style-type: none;
+ position: absolute;
+ width: 10rem;
+ left: 0;
+ top: 0;
+ text-align: left;
+ letter-spacing: 0px;
+ display: none;
+ border-radius: 4px;
+ pointer-events: all;
+ z-index: 1000;
+ background-color: var(--component-background);
+ box-shadow: 0 0 3px 0 var(--document-box-shadow);
+ padding-left: 0px;
+ }
+
+ & li {
+ display: block;
+ height: 2rem;
+ position: relative;
+ font-family: Lato;
+ font-style: normal;
+ font-weight: normal;
+ font-size: 0.8rem;
+ padding-left: 0.5rem;
+ :hover {
+ cursor: pointer;
+ }
+ }
+
+ & li:first-child {
+ display: flex;
+ flex-direction: row;
+ justify-content: space-between;
+ align-items: center;
+ padding: 0 4px 0 8px;
+ }
+
+ & li:not(:first-child) .options:hover {
+ background-color: var(--blue-1);
+ }
+
+ & li .optionSelected {
+ color: var(--blue-5);
+ }
+
+ & li .options {
+ border: none;
+ background-color: transparent;
+ padding-right: 0.65rem;
+ padding-left: 0.5rem;
+ width: calc(100% + 0.5rem);
+ text-align: left;
+ display: flex;
+ align-items: center;
+ justify-content: space-between;
+ height: 100%;
+ margin-left: -0.5rem;
+ z-index: 1;
+ }
+
+ .customSelector__selectedItem:focus + ul {
+ display: block;
+ pointer-events: all;
+ transform: translateY(0px);
+ }
+
+}
diff --git a/src/components/Selector/Selector.spec.js b/src/components/Selector/Selector.spec.js
new file mode 100644
index 0000000000..9f4f11044b
--- /dev/null
+++ b/src/components/Selector/Selector.spec.js
@@ -0,0 +1,24 @@
+import React from 'react';
+import { render } from '@testing-library/react';
+import { Basic as BasicStory, Placeholder as PlaceholderStory } from './Selector.stories';
+
+const BasicSelectStory = withI18n(BasicStory);
+
+describe('Selector', () => {
+ it('Basic story should not throw any errors', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+});
+
+const PlaceHolderSelectStory = withI18n(PlaceholderStory);
+
+describe('Placeholder version', () => {
+ it('Should have placeholder text as default item', () => {
+ const { container } = render();
+
+ const selectedItem = container.querySelector('.customSelector__selectedItem');
+ expect(selectedItem).toHaveTextContent('PLACEHOLDER');
+ });
+});
\ No newline at end of file
diff --git a/src/components/Selector/Selector.stories.js b/src/components/Selector/Selector.stories.js
new file mode 100644
index 0000000000..4d72f16456
--- /dev/null
+++ b/src/components/Selector/Selector.stories.js
@@ -0,0 +1,47 @@
+import React, { useState } from 'react';
+import Selector from './Selector';
+
+export default {
+ title: 'Components/Selector',
+ component: Selector,
+};
+
+export function Basic() {
+ const items = ['ITEM 1', 'ITEM 2'];
+ const [selectedItem, setSelectedItem] = useState(items[0]);
+ const placeHolder = 'PLACEHOLDER';
+ const onItemSelected = (item) => {
+ setSelectedItem(item);
+ };
+
+ return (
+
+
+
+ );
+}
+
+export function Placeholder() {
+ const items = ['ITEM 1', 'ITEM 2'];
+ const [selectedItem, setSelectedItem] = useState();
+ const placeHolder = 'PLACEHOLDER';
+ const onItemSelected = (item) => {
+ setSelectedItem(item);
+ };
+
+ return (
+
+
+
+ );
+}
diff --git a/src/components/Selector/index.js b/src/components/Selector/index.js
new file mode 100644
index 0000000000..837acf7044
--- /dev/null
+++ b/src/components/Selector/index.js
@@ -0,0 +1,3 @@
+import Selector from './Selector';
+
+export default Selector;
\ No newline at end of file
diff --git a/src/components/SettingsModal/AdvancedTab.scss b/src/components/SettingsModal/AdvancedTab.scss
new file mode 100644
index 0000000000..51ec77df99
--- /dev/null
+++ b/src/components/SettingsModal/AdvancedTab.scss
@@ -0,0 +1,22 @@
+.setting-item {
+ border: 1px var(--border) solid;
+ padding: 16px;
+ display: flex;
+ align-items: flex-start;
+ justify-content: space-between;
+
+ &:not(:last-child) {
+ border-bottom: 0px
+ }
+
+ .setting-item-info {
+ display: flex;
+ flex-direction: column;
+ margin-right: 18px;
+
+ .setting-item-label {
+ font-weight: 700;
+ margin-bottom: 10px;
+ }
+ }
+}
diff --git a/src/components/SettingsModal/GeneralTab.js b/src/components/SettingsModal/GeneralTab.js
new file mode 100644
index 0000000000..a69b139c49
--- /dev/null
+++ b/src/components/SettingsModal/GeneralTab.js
@@ -0,0 +1,109 @@
+import React from 'react';
+import { useSelector, useDispatch, useStore } from 'react-redux';
+import selectors from 'selectors';
+import actions from 'actions';
+import { useTranslation } from 'react-i18next';
+import { isIE } from 'helpers/device';
+import Languages from 'constants/languages';
+import Theme from 'constants/theme';
+import DataElements from 'constants/dataElement';
+import setLanguage from '../../apis/setLanguage';
+import Dropdown from 'components/Dropdown';
+import Icon from 'components/Icon';
+import Choice from 'components/Choice';
+import DataElementWrapper from 'components/DataElementWrapper';
+import { SearchWrapper } from './SearchWrapper';
+
+import './GeneralTab.scss';
+
+const GeneralTab = () => {
+ const [
+ currentLanguage,
+ activeTheme
+ ] = useSelector((state) => [
+ selectors.getCurrentLanguage(state),
+ selectors.getActiveTheme(state)
+ ]);
+ const [t] = useTranslation();
+ const dispatch = useDispatch();
+ const store = useStore();
+
+ const changeLanguage = (value) => {
+ if (value !== currentLanguage) {
+ setLanguage(store)(value);
+ }
+ };
+
+ const isLightMode = activeTheme === Theme.LIGHT;
+
+ const setTheme = (theme) => {
+ dispatch(actions.setActiveTheme(theme));
+ };
+
+ return (
+ <>
+
+
+ {t('option.settings.language')}
+ item[0]}
+ getDisplayValue={(item) => item[1]}
+ onClickItem={changeLanguage}
+ maxHeight={200}
+ width={336}
+ getCustomItemStyle={() => ({ textAlign: 'left', width: '326px' })}
+ className="language-dropdown"
+ />
+
+
+
+ {!isIE && (
+
+ {t('option.settings.theme')}
+
+
+
+
+ setTheme(Theme.LIGHT)}
+ label={t('option.settings.lightMode')}
+ name="theme_choice"
+ />
+
+
+
+
+
+ setTheme(Theme.DARK)}
+ label={t('option.settings.darkMode')}
+ name="theme_choice"
+ />
+
+
+
+
+ )}
+
+ >
+ );
+};
+
+export default GeneralTab;
diff --git a/src/components/SettingsModal/GeneralTab.scss b/src/components/SettingsModal/GeneralTab.scss
new file mode 100644
index 0000000000..9d85ad9fe0
--- /dev/null
+++ b/src/components/SettingsModal/GeneralTab.scss
@@ -0,0 +1,61 @@
+.language-dropdown {
+ .Dropdown__items {
+ left: 0;
+ width: 336px;
+ }
+}
+
+.theme-options {
+ width: 336px;
+ height: 160px;
+ display: flex;
+ justify-content: space-between;
+
+ .theme-option {
+ width: 160px;
+ height: 160px;
+ display: flex;
+ flex-direction: column;
+
+ .Icon {
+ width: 160px;
+ height: 120px;
+
+ &.light-mode-icon {
+ color: white;
+ }
+
+ &.dark-mode-icon {
+ color: black;
+ }
+
+ svg {
+ border: 1px solid;
+ border-color: var(--border);
+ border-top-left-radius: 4px;
+ border-top-right-radius: 4px;
+ }
+ }
+
+ .theme-choice {
+ height: 100%;
+ border: 1px solid;
+ border-color: var(--border);
+ border-top: 0px;
+ border-bottom-left-radius: 4px;
+ border-bottom-right-radius: 4px;
+ display: flex;
+ padding-left: 12px;
+ }
+
+ &.active-theme {
+ .Icon svg {
+ border-color: var(--focus-border);
+ }
+
+ .theme-choice {
+ border-color: var(--focus-border);
+ }
+ }
+ }
+}
diff --git a/src/components/SettingsModal/SearchWrapper.js b/src/components/SettingsModal/SearchWrapper.js
new file mode 100644
index 0000000000..a50fb0179d
--- /dev/null
+++ b/src/components/SettingsModal/SearchWrapper.js
@@ -0,0 +1,13 @@
+import React, { createContext, useContext } from 'react';
+
+export const SearchContext = createContext();
+
+export const SearchWrapper = ({ children, keywords = [] }) => {
+ const searchTerm = useContext(SearchContext).trim();
+
+ return (!searchTerm || keywords.some((keyword) => keyword.toLowerCase().includes(searchTerm.toLowerCase()))) ? (
+ <>
+ {children}
+ >
+ ) : null;
+};
diff --git a/src/components/SettingsModal/index.js b/src/components/SettingsModal/index.js
new file mode 100644
index 0000000000..09a83d8fe7
--- /dev/null
+++ b/src/components/SettingsModal/index.js
@@ -0,0 +1,3 @@
+import SettingsModal from './SettingsModal';
+
+export default SettingsModal;
\ No newline at end of file
diff --git a/src/components/SignatureListPanel/index.js b/src/components/SignatureListPanel/index.js
new file mode 100644
index 0000000000..bdc6bf76e5
--- /dev/null
+++ b/src/components/SignatureListPanel/index.js
@@ -0,0 +1,3 @@
+import SignatureListPanel from './SignatureListPanel';
+
+export default SignatureListPanel;
\ No newline at end of file
diff --git a/src/components/SignatureModal/SavedSignatures/index.js b/src/components/SignatureModal/SavedSignatures/index.js
new file mode 100644
index 0000000000..91ec23ef95
--- /dev/null
+++ b/src/components/SignatureModal/SavedSignatures/index.js
@@ -0,0 +1,3 @@
+import SavedSignatures from './SavedSignatures';
+
+export default SavedSignatures;
\ No newline at end of file
diff --git a/src/components/SignatureModal/SignatureModal.js b/src/components/SignatureModal/SignatureModal.js
index fb12f41490..1978ca1388 100644
--- a/src/components/SignatureModal/SignatureModal.js
+++ b/src/components/SignatureModal/SignatureModal.js
@@ -82,6 +82,7 @@ const SignatureModal = () => {
for (const signatureTool of signatureToolArray) {
signatureTool.clearLocation();
signatureTool.setSignature(null);
+ signatureTool.setInitials(null);
}
dispatch(actions.closeElement(DataElements.SIGNATURE_MODAL));
};
diff --git a/src/components/SignatureStylePopup/SignatureStylePopup.stories.js b/src/components/SignatureStylePopup/SignatureStylePopup.stories.js
index be4bcc81af..617444ca02 100644
--- a/src/components/SignatureStylePopup/SignatureStylePopup.stories.js
+++ b/src/components/SignatureStylePopup/SignatureStylePopup.stories.js
@@ -4,30 +4,13 @@ import SelectedSignatureRow from './SelectedSignatureRow';
import { configureStore } from '@reduxjs/toolkit';
import { Provider } from 'react-redux';
import SignatureModes from 'constants/signatureModes';
+import { mockSavedSignatures, mockSavedInitials } from './mockedSignatures';
export default {
title: 'Components/SavedSignaturesOverlay',
component: SignatureStylePopup,
};
-const mockSavedSignatures = [
- {
- imgSrc: ''
- },
- {
- imgSrc: ''
- }
-];
-
-const mockSavedInitials = [
- {
- imgSrc: ''
- },
- {
- imgSrc: ''
- }
-];
-
const initialState = {
viewer: {
isInitialsModeEnabled: true,
@@ -76,14 +59,14 @@ const savedInitialsStore = configureStore({
export const SavedInitialsTab = () => (
-
+
);
export const SelectedSignature = () => (
-
+
);
@@ -100,6 +83,6 @@ const initialsModeStore = configureStore({
export const SelectedInitials = () => (
-
+
);
diff --git a/src/components/SignatureStylePopup/mockedSignatures.js b/src/components/SignatureStylePopup/mockedSignatures.js
new file mode 100644
index 0000000000..b943b99509
--- /dev/null
+++ b/src/components/SignatureStylePopup/mockedSignatures.js
@@ -0,0 +1,17 @@
+export const mockSavedSignatures = [
+ {
+ imgSrc: ''
+ },
+ {
+ imgSrc: ''
+ }
+];
+
+export const mockSavedInitials = [
+ {
+ imgSrc: ''
+ },
+ {
+ imgSrc: ''
+ }
+];
diff --git a/src/components/SignatureValidationModal/SignatureValidationModal.scss b/src/components/SignatureValidationModal/SignatureValidationModal.scss
index 566593902b..5a6190250a 100644
--- a/src/components/SignatureValidationModal/SignatureValidationModal.scss
+++ b/src/components/SignatureValidationModal/SignatureValidationModal.scss
@@ -75,7 +75,7 @@
div.body > div.section {
margin: 16px 16px;
padding-bottom: 16px;
- border-bottom: 1px solid var(--gray-4);
+ border-bottom: 1px solid var(--gray-5);
}
div.body > div.section:last-child {
diff --git a/src/components/SnippingToolPopup/SnippingToolPopup.js b/src/components/SnippingToolPopup/SnippingToolPopup.js
new file mode 100644
index 0000000000..1cc68936d1
--- /dev/null
+++ b/src/components/SnippingToolPopup/SnippingToolPopup.js
@@ -0,0 +1,155 @@
+import React from 'react';
+import classNames from 'classnames';
+import Icon from 'components/Icon';
+import { useTranslation } from 'react-i18next';
+import { Choice } from '@pdftron/webviewer-react-toolkit';
+import Dropdown from 'components/Dropdown';
+import actions from 'actions';
+import { useDispatch } from 'react-redux';
+import DataElements from 'constants/dataElement';
+
+import './SnippingToolPopup.scss';
+
+const SnippingToolPopup = ({
+ snippingMode,
+ onSnippingModeChange,
+ closeSnippingPopup,
+ applySnipping,
+ isSnipping,
+ isInDesktopOnlyMode,
+ isMobile,
+ shouldShowApplySnippingWarning,
+}) => {
+ const { t } = useTranslation();
+
+ const snippingNames = {
+ 'CLIPBOARD': t('snippingPopUp.clipboard'),
+ 'DOWNLOAD': t('snippingPopUp.download'),
+ 'CROP_AND_REMOVE': t('snippingPopUp.cropAndRemove'),
+ };
+
+ const className = classNames({
+ Popup: true,
+ SnippingToolPopup: true,
+ mobile: isMobile,
+ });
+
+ const handleButtonPressed = (button) => {
+ switch (button) {
+ case 'apply':
+ shouldShowApplySnippingWarning && snippingMode === 'CROP_AND_REMOVE' ? openSnippingConfirmationWarning() : applySnipping();
+ break;
+ case 'cancel':
+ isSnipping ? openSnippingCancellationWarning() : closeSnippingPopup();
+ break;
+ }
+ };
+
+ const dispatch = useDispatch();
+
+ const openSnippingConfirmationWarning = () => {
+ const title = t('snippingPopUp.snippingModal.applyTitle');
+ const message = t('snippingPopUp.snippingModal.applyMessage');
+ const confirmationWarning = {
+ message,
+ title,
+ onConfirm: () => {
+ applySnipping();
+ },
+ };
+ dispatch(actions.showWarningMessage(confirmationWarning));
+ };
+
+ const openSnippingCancellationWarning = () => {
+ const title = t('snippingPopUp.snippingModal.cancelTitle');
+ const message = t('snippingPopUp.snippingModal.cancelMessage');
+ const cancellationWarning = {
+ message,
+ title,
+ onConfirm: () => {
+ closeSnippingPopup();
+ },
+ };
+ dispatch(actions.showWarningMessage(cancellationWarning));
+ };
+
+ if (isMobile && !isInDesktopOnlyMode) {
+ return (
+
+
+
+
+ onSnippingModeChange(Object.keys(snippingNames).find((key) => snippingNames[key] === e))}
+ currentSelectionKey={snippingNames[snippingMode]}
+ />
+
+
handleButtonPressed('apply')}
+ disabled={!isSnipping}
+ >
+ {t('action.apply')}
+
+
+
handleButtonPressed('cancel')}
+ >
+
+
+
+
+ );
+ }
+ return (
+
+
+ {t('snippingPopUp.title')}
+ onSnippingModeChange('CLIPBOARD')}
+ checked={snippingMode === 'CLIPBOARD'}
+ radio
+ />
+ onSnippingModeChange('DOWNLOAD')}
+ checked={snippingMode === 'DOWNLOAD'}
+ radio
+ />
+ onSnippingModeChange('CROP_AND_REMOVE')}
+ checked={snippingMode === 'CROP_AND_REMOVE'}
+ radio
+ />
+
+
+
+ handleButtonPressed('cancel')}>
+ {t('action.cancel')}
+
+ handleButtonPressed('apply')}
+ disabled={!isSnipping}
+ >
+ {t('action.apply')}
+
+
+
+ );
+};
+
+export default SnippingToolPopup;
diff --git a/src/components/SnippingToolPopup/SnippingToolPopup.scss b/src/components/SnippingToolPopup/SnippingToolPopup.scss
new file mode 100644
index 0000000000..7e9acbfaaa
--- /dev/null
+++ b/src/components/SnippingToolPopup/SnippingToolPopup.scss
@@ -0,0 +1,204 @@
+@import '../../constants/styles';
+@import '../../constants/modal';
+@import '../../constants/popup.scss';
+
+.SnippingToolPopup {
+ width: 250px;
+
+ .snipping-section {
+ padding: 16px;
+ display: flex;
+ flex-direction: column;
+
+ .ui__choice {
+ margin: 0;
+ }
+
+ .ui__choice:not(:last-of-type) {
+ padding-bottom: 12px;
+ }
+ }
+
+ .menu-title {
+ padding-bottom: 16px;
+ font-weight: bold;
+ }
+
+ .crop-inactive {
+ color: var(--gray-6);
+ }
+
+ .Icon {
+ height: 18px;
+ width: 18px;
+ }
+
+ .divider {
+ border-top: 1px solid var(--divider);
+ width: 100%;
+ }
+
+ .buttons {
+ padding: 12px;
+ text-align: right;
+ font-size: 13px;
+ display: flex;
+ justify-content: space-between;
+ }
+
+ .save-button {
+ color: var(--primary-button-text);
+ padding: 6px 16px;
+ background: var(--primary-button);
+ border-radius: 4px;
+ border: 0;
+ height: 32px;
+ cursor: pointer;
+
+ &:disabled {
+ opacity: 0.5;
+ cursor: auto;
+ }
+ }
+
+ .cancel-button {
+ cursor: pointer;
+ background: none;
+ border: 0;
+ color: var(--secondary-button-text);
+ padding: 6px 16px;
+ margin-right: 4px;
+ height: 32px;
+ &:hover {
+ color: var(--secondary-button-hover);
+ }
+ &:focus {
+ outline: none;
+ }
+ &:disabled {
+ opacity: 0.5;
+ cursor: auto;
+ color: var(--secondary-button-text);
+ }
+ }
+}
+
+.custom-select {
+ flex-grow: 2;
+ max-width: 100%;
+ margin: 4px;
+
+ .customSelector {
+ margin-left: 0;
+ height: 28px;
+ width: 100% !important;
+
+ .customSelector__selectedItem {
+ width: 100%;
+ }
+
+ ul {
+ width: 100%;
+ }
+
+ .customSelector__arrow {
+ height: 18px;
+ width: 18px;
+ }
+ }
+
+ select {
+ height: 28px;
+ width: 100%;
+ }
+}
+
+.SnippingPopupContainer {
+ @extend %popup;
+ border-radius: 4px;
+ box-shadow: 0 0 3px 0 var(--document-box-shadow);
+ background: var(--component-background);
+ top: 0;
+
+ @include mobile {
+ width: 100%;
+ position: fixed;
+ bottom: 0 !important;
+ border-radius: 0;
+ justify-content: start;
+ top: auto;
+
+ .snipping-mobile-section {
+ padding-top: 8px;
+ padding-bottom: 8px;
+ padding-left: 12px;
+ padding-right: 12px;
+ display: flex;
+ align-items: center;
+ justify-content: space-between;
+ width: 100%;
+ }
+
+ .SnippingToolPopup {
+ width: 100%;
+ }
+
+ .snipping-mobile-container {
+ display: flex;
+ align-items: center;
+
+ .customSelector {
+ width: 100%;
+ }
+
+ .Dropdown {
+ height: 32px;
+ min-width: 150px;
+ width: 100% !important;
+
+ .arrow {
+ flex: 0 1 auto;
+ }
+
+ .picked-option .picked-option-text {
+ width: 150px;
+ text-align: left;
+ }
+ }
+
+ .Dropdown__items {
+ top: -52px;
+ z-index: 80;
+ width: 100%;
+ }
+
+ .wrapper {
+ z-index: 79;
+ }
+
+ .save-button {
+ margin-left: 6px;
+ min-width: 75px;
+ }
+ }
+ .cancel-button {
+ padding: 0;
+
+ .Icon {
+ width: 24px;
+ height: 24px;
+ }
+ }
+
+ .snipping-selector {
+ width: 100%;
+ display: flex;
+ }
+
+ @media (max-width: 430px) {
+ .snipping-selector {
+ display: block;
+ }
+ }
+ }
+}
diff --git a/src/components/SnippingToolPopup/SnippingToolPopup.spec.js b/src/components/SnippingToolPopup/SnippingToolPopup.spec.js
new file mode 100644
index 0000000000..01edeabec2
--- /dev/null
+++ b/src/components/SnippingToolPopup/SnippingToolPopup.spec.js
@@ -0,0 +1,17 @@
+import React from 'react';
+import { render } from '@testing-library/react';
+import { Basic } from './SnippingToolPopup.stories';
+
+const BasicSnippingToolPopupStory = withI18n(Basic);
+
+jest.mock('core');
+
+describe('SnippingToolPopup', () => {
+ describe('Component', () => {
+ it('Story should not throw any errors', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+ });
+});
diff --git a/src/components/SnippingToolPopup/SnippingToolPopup.stories.js b/src/components/SnippingToolPopup/SnippingToolPopup.stories.js
new file mode 100644
index 0000000000..0cf959cec2
--- /dev/null
+++ b/src/components/SnippingToolPopup/SnippingToolPopup.stories.js
@@ -0,0 +1,42 @@
+import React from 'react';
+import SnippingToolPopup from './SnippingToolPopup';
+import { createStore } from 'redux';
+import { Provider } from 'react-redux';
+
+export default {
+ title: 'Components/SnippingToolPopup',
+ component: SnippingToolPopup,
+};
+
+const initialState = {
+ viewer: {
+ disabledElements: {},
+ customElementOverrides: {},
+ },
+};
+
+function rootReducer(state = initialState, action) {
+ return state;
+}
+
+const store = createStore(rootReducer);
+
+const noop = () => {};
+
+const popupProps = {
+ closeSnippingPopup: noop,
+ applySnipping: noop,
+ isSnipping: true,
+ isInDesktopOnlyMode: false,
+ isMobile: false,
+};
+
+export function Basic() {
+ return (
+
+
+
+
+
+ );
+}
diff --git a/src/components/SnippingToolPopup/SnippingToolPopupContainer.js b/src/components/SnippingToolPopup/SnippingToolPopupContainer.js
new file mode 100644
index 0000000000..56295559a0
--- /dev/null
+++ b/src/components/SnippingToolPopup/SnippingToolPopupContainer.js
@@ -0,0 +1,200 @@
+import React, { useEffect, useState, useRef, useCallback } from 'react';
+import { useSelector, useDispatch } from 'react-redux';
+import actions from 'actions';
+import selectors from 'selectors';
+import core from 'core';
+import SnippingToolPopup from './SnippingToolPopup';
+import './SnippingToolPopup.scss';
+import Draggable from 'react-draggable';
+import useOnSnippingAnnotationChangedOrSelected from '../../hooks/useOnSnippingAnnotationChangedOrSelected';
+import { isMobileSize } from 'helpers/getDeviceSize';
+import getRootNode from 'helpers/getRootNode';
+import DataElements from 'constants/dataElement';
+
+function SnippingToolPopupContainer() {
+ const snippingToolName = window.Core.Tools.ToolNames['SNIPPING'];
+ const snippingCreateTool = core.getTool(snippingToolName);
+ const [
+ isOpen,
+ isInDesktopOnlyMode,
+ shouldShowApplySnippingWarning,
+ ] = useSelector((state) => [
+ selectors.getActiveToolName(state) === snippingToolName && selectors.isElementOpen(state, DataElements.SNIPPING_TOOL_POPUP),
+ selectors.isInDesktopOnlyMode(state),
+ selectors.shouldShowApplySnippingWarning(state),
+ ]);
+ const dispatch = useDispatch();
+ const [isSnipping, setIsSnipping] = useState(snippingCreateTool.getIsSnipping());
+
+ const elementsToClose = ['leftPanel', 'searchPanel', 'notesPanel', 'redactionPanel', 'textEditingPanel'];
+
+ const openSnippingPopup = () => {
+ dispatch(actions.openElement(DataElements.SNIPPING_TOOL_POPUP));
+ // eslint-disable-next-line no-undef
+ dispatch(actions.closeElements(elementsToClose));
+ setIsSnipping(snippingCreateTool.getIsSnipping());
+ };
+
+ useEffect(() => {
+ const handleSnippingCancellation = () => setIsSnipping(false);
+
+ const handleToolModeChange = (newTool, oldTool) => {
+ if (newTool instanceof Core.Tools.SnippingCreateTool) { // eslint-disable-line no-undef
+ newTool.addEventListener(window.Core.Tools.SnippingCreateTool.Events['SNIPPING_CANCELLED'], handleSnippingCancellation);
+ openSnippingPopup();
+ } else if (oldTool instanceof Core.Tools.SnippingCreateTool) { // eslint-disable-line no-undef
+ newTool.removeEventListener(window.Core.Tools.SnippingCreateTool.Events['SNIPPING_CANCELLED'], handleSnippingCancellation);
+ setIsSnipping(false);
+ snippingCreateTool.reset();
+ reenableHeader();
+ }
+ };
+
+ core.addEventListener('toolModeUpdated', handleToolModeChange);
+
+ return () => {
+ core.removeEventListener('toolModeUpdated', handleToolModeChange);
+ };
+ });
+
+ const disableHeader = () => {
+ const header = getRootNode().querySelector('[data-element=header]');
+ if (header) {
+ header.style.pointerEvents = 'none';
+ header.style.opacity = '0.5';
+ }
+
+ const toolsHeader = getRootNode().querySelector('[data-element=toolsHeader]');
+ if (toolsHeader) {
+ toolsHeader.style.pointerEvents = 'none';
+ toolsHeader.style.opacity = '0.5';
+ }
+ };
+
+ const reenableHeader = () => {
+ const header = getRootNode().querySelector('[data-element=header]');
+ if (header) {
+ header.style.pointerEvents = '';
+ header.style.opacity = '1';
+ }
+
+ const toolsHeader = getRootNode().querySelector('[data-element=toolsHeader]');
+ if (toolsHeader) {
+ toolsHeader.style.pointerEvents = '';
+ toolsHeader.style.opacity = '1';
+ }
+ };
+
+ const snippingAnnotation = useOnSnippingAnnotationChangedOrSelected(openSnippingPopup);
+
+ // re-enable other tools and panels while not snipping
+ useEffect(() => {
+ if (!isSnipping) {
+ reenableHeader();
+ } else {
+ disableHeader();
+ }
+ }, [isSnipping]);
+
+ const [snippingMode, setSnippingMode] = useState(null);
+
+ useEffect(() => {
+ snippingCreateTool.setSnippingMode('CLIPBOARD');
+ setSnippingMode('CLIPBOARD');
+ }, []);
+
+ const onSnippingModeChange = (option) => {
+ snippingCreateTool.setSnippingMode(option);
+ setSnippingMode(option);
+ };
+
+ const snippingPopupRef = useRef();
+ const DEFAULT_POPUP_WIDTH = 250;
+ const DEFAULT_POPUP_HEIGHT = 200;
+ const documentContainerElement = core.getScrollViewElement();
+ const popupWidth = snippingPopupRef.current?.getBoundingClientRect().width || DEFAULT_POPUP_WIDTH;
+ const popupHeight = snippingPopupRef.current?.getBoundingClientRect().height || DEFAULT_POPUP_HEIGHT;
+ const documentViewer = core.getDocumentViewer(1);
+ // eslint-disable-next-line no-undef
+ const xOffset = documentViewer.getViewerElement()?.getBoundingClientRect().right || 0;
+
+ const getSnippingPopupOffset = () => {
+ const offset = {
+ x: xOffset + 35,
+ y: documentContainerElement?.offsetTop + 10,
+ };
+ if (snippingAnnotation && snippingPopupRef?.current) {
+ offset.x = Math.min(offset.x, documentContainerElement.offsetWidth - popupWidth);
+ }
+ return offset;
+ };
+
+ const getSnippingPopupBounds = () => {
+ const bounds = {
+ top: 0,
+ bottom: documentContainerElement.offsetHeight - popupHeight,
+ left: 0 - getSnippingPopupOffset()['x'],
+ right: documentContainerElement.offsetWidth - getSnippingPopupOffset()['x'] - popupWidth,
+ };
+ return bounds;
+ };
+
+ const closeAndReset = () => {
+ snippingCreateTool.reset();
+ dispatch(actions.closeElement(DataElements.SNIPPING_TOOL_POPUP));
+ reenableHeader();
+ core.setToolMode(window.Core.Tools.ToolNames.SNIPPING);
+ };
+
+ const closeSnippingPopup = useCallback(() => {
+ closeAndReset();
+ }, []);
+
+ // disable/enable the 'apply' button when snipping
+ useEffect(() => {
+ setIsSnipping(snippingCreateTool.getIsSnipping());
+ }, [snippingAnnotation]);
+
+ const applySnipping = async () => {
+ await snippingCreateTool.applySnipping();
+ snippingCreateTool.reset();
+ reenableHeader();
+ };
+
+ const props = {
+ snippingMode,
+ onSnippingModeChange,
+ closeSnippingPopup,
+ applySnipping,
+ isSnipping,
+ isInDesktopOnlyMode,
+ shouldShowApplySnippingWarning,
+ };
+
+ const isMobile = isMobileSize();
+
+ if (isOpen && core.getDocument()) {
+ if (isMobile && !isInDesktopOnlyMode) {
+ // disable draggable on mobile devices
+ return (
+
+
+
+ );
+ }
+ return (
+
+
+
+
+
+ );
+ }
+ return null;
+}
+
+export default SnippingToolPopupContainer;
diff --git a/src/components/SnippingToolPopup/index.js b/src/components/SnippingToolPopup/index.js
new file mode 100644
index 0000000000..c95f3125e3
--- /dev/null
+++ b/src/components/SnippingToolPopup/index.js
@@ -0,0 +1,3 @@
+import SnippingToolPopup from './SnippingToolPopupContainer';
+
+export default SnippingToolPopup;
\ No newline at end of file
diff --git a/src/components/StylePicker/ColorPicker/index.js b/src/components/StylePicker/ColorPicker/index.js
new file mode 100644
index 0000000000..60951c677e
--- /dev/null
+++ b/src/components/StylePicker/ColorPicker/index.js
@@ -0,0 +1,3 @@
+import ColorPicker from './ColorPicker';
+
+export default ColorPicker;
\ No newline at end of file
diff --git a/src/components/StylePicker/index.js b/src/components/StylePicker/index.js
new file mode 100644
index 0000000000..8114839de3
--- /dev/null
+++ b/src/components/StylePicker/index.js
@@ -0,0 +1,3 @@
+import StylePicker from './StylePicker';
+
+export default StylePicker;
\ No newline at end of file
diff --git a/src/components/StylePopup/StylePopup.stories.js b/src/components/StylePopup/StylePopup.stories.js
new file mode 100644
index 0000000000..ce01b333a5
--- /dev/null
+++ b/src/components/StylePopup/StylePopup.stories.js
@@ -0,0 +1,213 @@
+import React from 'react';
+import { configureStore } from '@reduxjs/toolkit';
+import initialState from 'src/redux/initialState';
+import { Provider } from 'react-redux';
+import StylePopup from './StylePopup';
+import core from 'core';
+
+export default {
+ title: 'Components/StylePopup',
+ component: StylePopup,
+};
+
+// Mock some state to show the style popups
+const state = {
+ ...initialState,
+ viewer: {
+ openElements: {
+ watermarkPanel: true,
+ stylePopup: true,
+ stylePopupTextStyleContainer: true,
+ stylePopupColorsContainer: true,
+ stylePopupLabelTextContainer: true
+ },
+ disabledElements: {},
+ selectedScale: undefined,
+ colorMap: {
+ textField: {
+ currentStyleTab: 'StrokeColor',
+ iconColor: 'StrokeColor',
+ }
+ },
+ fonts: ['Helvetica', 'Times New Roman', 'Arimo'],
+ isSnapModeEnabled: false,
+ customElementOverrides: {}
+ }
+};
+
+const noop = () => {};
+
+const store = configureStore({
+ reducer: () => state
+});
+
+const BasicComponent = (props) => {
+ core.getFormFieldCreationManager = () => ({
+ isInFormFieldCreationMode: () => true,
+ });
+
+ return (
+
+
+
+ );
+};
+
+export const StylePopupInFormBuilder = BasicComponent.bind({});
+StylePopupInFormBuilder.args = {
+ currentStyleTab: 'StrokeColor',
+ isInFormBuilderAndNotFreeText: true,
+ style: {
+ 'FillColor': new window.Core.Annotations.Color(212, 211, 211),
+ 'StrokeColor': new window.Core.Annotations.Color(0, 0, 0),
+ 'TextColor': new window.Core.Annotations.Color(0, 0, 0),
+ 'Opacity': null,
+ 'StrokeThickness': 1,
+ 'FontSize': '12pt',
+ 'Style': 'solid'
+ },
+ colorMapKey: 'textField',
+ colorPalette: 'StrokeColor',
+ disableSeparator: true,
+ hideSnapModeCheckbox: true,
+ isFreeText: false,
+ isEllipse: false,
+ isTextStyleContainerActive: true,
+ isLabelTextContainerActive: true,
+ properties: {
+ 'StrokeStyle': 'solid',
+ },
+ isRedaction: false,
+ fonts: ['Helvetica', 'Times New Roman', 'Arimo'],
+ isSnapModeEnabled: false,
+ onSliderChange: noop,
+ onStyleChange: noop,
+ closeElement: noop,
+ openElement: noop,
+ onPropertyChange: noop,
+ onRichTextStyleChange: noop,
+ onLineStyleChange: noop,
+};
+
+export const StylePopupForRedactionToolInHeaderItem = () => {
+ const props = {
+ currentStyleTab: 'TextColor',
+ isInFormBuilderAndNotFreeText: false,
+ style: {
+ 'FillColor': new window.Core.Annotations.Color(212, 211, 211),
+ 'StrokeColor': new window.Core.Annotations.Color(0, 0, 0),
+ 'TextColor': new window.Core.Annotations.Color(0, 0, 0),
+ 'Opacity': null,
+ 'StrokeThickness': 1,
+ 'FontSize': '12pt',
+ 'Style': 'solid'
+ },
+ colorMapKey: 'textField',
+ colorPalette: 'TextColor',
+ disableSeparator: true,
+ hideSnapModeCheckbox: true,
+ isFreeText: false,
+ isEllipse: false,
+ isTextStyleContainerActive: true,
+ isLabelTextContainerActive: true,
+ properties: {
+ 'StrokeStyle': 'solid',
+ },
+ isRedaction: true,
+ fonts: ['Helvetica', 'Times New Roman', 'Arimo'],
+ isSnapModeEnabled: false,
+ onSliderChange: noop,
+ onStyleChange: noop,
+ closeElement: noop,
+ openElement: noop,
+ onPropertyChange: noop,
+ onRichTextStyleChange: noop,
+ onLineStyleChange: noop,
+ };
+
+ const stateForTextTab = {
+ ...state,
+ viewer: {
+ ...state.viewer,
+ colorMap: {
+ textField: {
+ currentStyleTab: 'TextColor',
+ iconColor: 'TextColor',
+ }
+ },
+ }
+ };
+
+ const store = configureStore({
+ reducer: () => stateForTextTab
+ });
+
+ return (
+
+
+
+
+
+ );
+};
+
+export const StylePopupForFreeTextToolInHeaderItem = () => {
+ const props = {
+ currentStyleTab: 'TextColor',
+ isInFormBuilderAndNotFreeText: false,
+ style: {
+ 'FillColor': new window.Core.Annotations.Color(212, 211, 211),
+ 'StrokeColor': new window.Core.Annotations.Color(0, 0, 0),
+ 'TextColor': new window.Core.Annotations.Color(0, 0, 0),
+ 'Opacity': 1,
+ 'StrokeThickness': 1,
+ 'FontSize': '12pt',
+ 'Style': 'solid'
+ },
+ colorMapKey: 'textField',
+ colorPalette: 'TextColor',
+ disableSeparator: true,
+ hideSnapModeCheckbox: true,
+ isFreeText: true,
+ isEllipse: false,
+ isTextStyleContainerActive: true,
+ isLabelTextContainerActive: false,
+ properties: {
+ 'StrokeStyle': 'solid',
+ },
+ fonts: ['Helvetica', 'Times New Roman', 'Arimo'],
+ isSnapModeEnabled: false,
+ onSliderChange: noop,
+ onStyleChange: noop,
+ closeElement: noop,
+ openElement: noop,
+ onPropertyChange: noop,
+ onRichTextStyleChange: noop,
+ onLineStyleChange: noop,
+ };
+
+ const stateForTextTab = {
+ ...state,
+ viewer: {
+ ...state.viewer,
+ colorMap: {
+ textField: {
+ currentStyleTab: 'TextColor',
+ iconColor: 'TextColor',
+ }
+ },
+ }
+ };
+
+ const store = configureStore({
+ reducer: () => stateForTextTab
+ });
+
+ return (
+
+
+
+
+
+ );
+};
diff --git a/src/components/TextEditingPanel/TextEditingPanel.scss b/src/components/TextEditingPanel/TextEditingPanel.scss
new file mode 100644
index 0000000000..ae3ed0c28f
--- /dev/null
+++ b/src/components/TextEditingPanel/TextEditingPanel.scss
@@ -0,0 +1,152 @@
+@import '../../constants/styles';
+@import '../../constants/panel';
+
+.TextEditingPanel {
+ padding: 16px 16px 0px 16px;
+ display: flex;
+ flex-direction: column;
+
+ .text-editing-panel-text-style-picker {
+ margin-top: 16px;
+ }
+
+ .text-editing-panel-section {
+ .text-editing-panel-heading {
+ font-size: var(--font-size-default);
+ font-weight: 700;
+ }
+
+ .text-editing-panel-menu-items {
+ margin-top: 16px;
+ margin-bottom: 16px;
+
+ .text-editing-panel-menu-items-buttons {
+ display: flex;
+ gap: 8px;
+ }
+
+ .text-editing-panel-menu-items-buttons.undo-redo {
+ gap: 12px;
+ }
+
+ // align font family dropdown items right
+ .Dropdown__items {
+ right: auto;
+ }
+
+ .FontSizeDropdown .Dropdown__items {
+ right: 0;
+ }
+
+ .Dropdown__items .Dropdown__item {
+ font-size: var(--font-size-default);
+ font-family: var(--font-family);
+ }
+
+ .Dropdown .picked-option .picked-option-text {
+ font-family: var(--font-family);
+ }
+
+ .inactive {
+ opacity: 0.5;
+ pointer-Events: none;
+ }
+ }
+
+ .top-panel {
+ margin-top: 0px;
+ }
+
+ .icon-grid .row {
+ padding-top: 12px;
+ }
+
+ .link-section {
+ margin-top: 12px;
+ }
+
+ .text-editing-row {
+ display: flex;
+ flex-direction: row;
+ justify-content: flex-start;
+ gap: 8px;
+
+ width: 100%;
+ margin-top: 8px;
+ margin-bottom: 8px;
+ .Button .Icon {
+ width: 32px;
+ height: 32px;
+ }
+
+ .color-picker-container {
+ width: 100%;
+ }
+
+ .ColorPalette {
+ display: flex;
+ flex-wrap: wrap;
+ justify-content: left;
+ align-content: flex-start;
+ gap: 8px;
+ }
+ }
+
+ .custom-colors-pallete {
+ .cell-container {
+ flex: none;
+ }
+ }
+
+ .custom-colors-section {
+ margin-top: 12px;
+ }
+
+ .addToCustomButton {
+ width: 24px;
+ height: 24px;
+ padding-top: 8px;
+ align-self: center;
+ }
+
+ .text-editing-panel-color-palette {
+ display: flex;
+ margin-bottom: 16px;
+ }
+
+ .opacity-slider {
+ @include ie11 {
+ align-items: stretch;
+ }
+ }
+ }
+
+ @include mobile {
+ width: 100%;
+ min-width: 100%;
+ padding-top: 0px;
+
+ .icon-grid .text-horizontal-alignment {
+ float: none;
+ }
+
+ .close-container {
+ display: flex;
+ align-items: center;
+ justify-content: flex-end;
+ height: 64px;
+
+ width: 100%;
+ padding-right: 12px;
+
+ .close-icon-container {
+ cursor: pointer;
+ .close-icon {
+ width: 24px;
+ height: 24px;
+ }
+ }
+ }
+ }
+}
+
diff --git a/src/components/TextEditingPanel/TextEditingPanel.spec.js b/src/components/TextEditingPanel/TextEditingPanel.spec.js
new file mode 100644
index 0000000000..5d6bf69fba
--- /dev/null
+++ b/src/components/TextEditingPanel/TextEditingPanel.spec.js
@@ -0,0 +1,121 @@
+import React from 'react';
+import { render, screen } from '@testing-library/react';
+import { TextEditingUndoRedo as TextEditUndoRedoStory } from './TextEditingPanel.stories';
+import TextEditingPanel from './TextEditingPanel';
+import core from 'core';
+
+const TestTextEditingPanel = withProviders(TextEditingPanel);
+
+const noop = () => { };
+
+const mockProps = {
+ handlePropertyChange: noop,
+ handleTextFormatChange: noop,
+ handleColorChange: noop,
+ format: {
+ bold: false,
+ italic: false,
+ underline: false,
+ },
+ undoRedoProperties: undefined,
+};
+
+core.getContentEditManager = () => ({
+ isInContentEditMode: () => true,
+});
+
+describe('TextEditingPanel', () => {
+ it('Undo/redo story should render without errors', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('should render without undo/redo buttons', () => {
+ render();
+ const undoButton = screen.queryByRole('Undo', { name: 'Undo' });
+ const redoButton = screen.queryByRole('Redo', { name: 'Redo' });
+ expect(undoButton).not.toBeInTheDocument();
+ expect(redoButton).not.toBeInTheDocument();
+ });
+
+ it('should render with undo/redo buttons disabled', () => {
+ mockProps.undoRedoProperties = {
+ canUndo: false,
+ canRedo: false
+ };
+
+ render();
+ const undoButton = screen.getByRole('button', { name: 'Undo' });
+ const redoButton = screen.getByRole('button', { name: 'Redo' });
+ expect(undoButton.disabled).toBe(true);
+ expect(redoButton.disabled).toBe(true);
+ });
+
+ it('should render with undo/redo buttons enabled', () => {
+ mockProps.undoRedoProperties = {
+ canUndo: true,
+ canRedo: true
+ };
+
+ render();
+ const undoButton = screen.getByRole('button', { name: 'Undo' });
+ const redoButton = screen.getByRole('button', { name: 'Redo' });
+ expect(undoButton.disabled).toBe(false);
+ expect(redoButton.disabled).toBe(false);
+ });
+
+ it('should render with undo enabled but redo disabled', () => {
+ mockProps.undoRedoProperties = {
+ canUndo: true,
+ canRedo: false
+ };
+
+ render();
+ const undoButton = screen.getByRole('button', { name: 'Undo' });
+ const redoButton = screen.getByRole('button', { name: 'Redo' });
+ expect(undoButton.disabled).toBe(false);
+ expect(redoButton.disabled).toBe(true);
+ });
+
+ it('should render with undo disabled but redo enabled', () => {
+ mockProps.undoRedoProperties = {
+ canUndo: false,
+ canRedo: true
+ };
+
+ render();
+ const undoButton = screen.getByRole('button', { name: 'Undo' });
+ const redoButton = screen.getByRole('button', { name: 'Redo' });
+ expect(undoButton.disabled).toBe(true);
+ expect(redoButton.disabled).toBe(false);
+ });
+
+ it('should fire undo handler when button is clicked', () => {
+ mockProps.undoRedoProperties = {
+ canUndo: true,
+ handleUndo: jest.fn()
+ };
+
+ render();
+ const undoButton = screen.getByRole('button', { name: 'Undo' });
+ expect(undoButton.disabled).toBe(false);
+
+ undoButton.click();
+ expect(mockProps.undoRedoProperties.handleUndo).toHaveBeenCalled();
+ });
+
+ it('should fire redo handler when button is clicked', () => {
+ mockProps.undoRedoProperties = {
+ canRedo: true,
+ handleRedo: jest.fn()
+ };
+
+ render();
+ const redoButton = screen.getByRole('button', { name: 'Redo' });
+ expect(redoButton.disabled).toBe(false);
+
+ redoButton.click();
+ expect(mockProps.undoRedoProperties.handleRedo).toHaveBeenCalled();
+ });
+});
\ No newline at end of file
diff --git a/src/components/TextEditingPanel/index.js b/src/components/TextEditingPanel/index.js
new file mode 100644
index 0000000000..0547944cee
--- /dev/null
+++ b/src/components/TextEditingPanel/index.js
@@ -0,0 +1,3 @@
+import TextEditingPanel from './TextEditingPanelContainer';
+
+export default TextEditingPanel;
diff --git a/src/components/TextStylePicker/TextStylePicker.spec.js b/src/components/TextStylePicker/TextStylePicker.spec.js
new file mode 100644
index 0000000000..2af910b86e
--- /dev/null
+++ b/src/components/TextStylePicker/TextStylePicker.spec.js
@@ -0,0 +1,58 @@
+import React from 'react';
+import { render, screen, fireEvent } from '@testing-library/react';
+import TextStylePicker from './TextStylePicker';
+import { DEBOUNCE_TIME } from '../FontSizeDropdown/FontSizeDropdown';
+import { Provider } from 'react-redux';
+import { configureStore } from '@reduxjs/toolkit';
+import userEvent from '@testing-library/user-event';
+
+jest.mock('core', () => ({
+ getContentEditManager: () => ({
+ isInContentEditMode: () => false,
+ }),
+}));
+
+// mock initial state.
+// UI Buttons are redux connected, and they need a state or the
+const initialState = {
+ viewer: {
+ openElements: {
+ },
+ disabledElements: {},
+ customElementOverrides: {},
+ }
+};
+
+const store = configureStore({
+ reducer: () => initialState
+});
+
+const TextStylePickerWithRedux = (props) => (
+
+
+
+);
+
+
+const noop = () => { };
+
+describe.only('TextStylePicker Component', () => {
+ it('should render without errors', () => {
+ const props = {
+ onPropertyChange: noop
+ };
+ render();
+ });
+
+ it('should render a warning if you enter an invalid font size', async () => {
+ const props = {
+ onPropertyChange: noop
+ };
+ render();
+ const fontSizeInput = screen.getByRole('textbox');
+ userEvent.type(fontSizeInput, '12345');
+ // Assert that a warning exists
+ await new Promise((r) => setTimeout(r, DEBOUNCE_TIME + 5));
+ expect(screen.getByText('Font size must be in the following range: 1 - 512')).toBeInTheDocument();
+ });
+});
diff --git a/src/components/TextStylePicker/TextStylePicker.stories.js b/src/components/TextStylePicker/TextStylePicker.stories.js
new file mode 100644
index 0000000000..b47ce1abeb
--- /dev/null
+++ b/src/components/TextStylePicker/TextStylePicker.stories.js
@@ -0,0 +1,82 @@
+import React from 'react';
+import { configureStore } from '@reduxjs/toolkit';
+import initialState from 'src/redux/initialState';
+import i18n from 'i18next';
+import { I18nextProvider } from 'react-i18next';
+import { Provider } from 'react-redux';
+import TextStylePicker from './TextStylePicker';
+import core from 'core';
+
+export default {
+ title: 'Components/TextStylePicker',
+ component: TextStylePicker,
+};
+
+const noop = () => {};
+
+const state = {
+ ...initialState,
+ viewer: {
+ currentLanguage: 'ja',
+ disabledElements: {},
+ customElementOverrides: {}
+ }
+};
+const store = configureStore({
+ reducer: () => state
+});
+
+const BasicComponent = (props) => {
+ return (
+
+
+
+
+
+
+
+ );
+};
+
+const DisabledFontSelectorComponent = (props) => {
+ return (
+
+
+
+
+
+
+
+ );
+};
+
+export const TextStylePickerSection = BasicComponent.bind({});
+TextStylePickerSection.args = {
+ properties: {
+ FontSize: '128'
+ },
+ isRedaction: false,
+ onPropertyChange: noop
+};
+
+export const TextStylePickerFreeTextDisabled = DisabledFontSelectorComponent.bind({});
+TextStylePickerFreeTextDisabled.args = {
+ properties: {
+ FontSize: '128'
+ },
+ isFreeText: true,
+ isFreeTextAutoSize: true,
+ isRedaction: false,
+ onPropertyChange: noop
+};
+
+export const TextStylePickerFreeTextEnabled = DisabledFontSelectorComponent.bind({});
+TextStylePickerFreeTextEnabled.args = {
+ properties: {
+ FontSize: '128'
+ },
+ isFreeText: true,
+ isFreeTextAutoSize: false,
+ isRedaction: false,
+ onPropertyChange: noop
+};
\ No newline at end of file
diff --git a/src/components/Wv3dPropertiesPanel/GeneralValuesSection/GeneralValuesSection.js b/src/components/Wv3dPropertiesPanel/GeneralValuesSection/GeneralValuesSection.js
new file mode 100644
index 0000000000..b262ba0922
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/GeneralValuesSection/GeneralValuesSection.js
@@ -0,0 +1,16 @@
+import React from 'react';
+import PropertyKeyValuePair from '../PropertyKeyValuePair/PropertyKeyValuePair';
+
+const GeneralValuesSection = (props) => {
+ const { entities } = props;
+
+ const elements = [];
+
+ for (const entity in entities) {
+ elements.push();
+ }
+
+ return {elements}
;
+};
+
+export default GeneralValuesSection;
diff --git a/src/components/Wv3dPropertiesPanel/GeneralValuesSection/GeneralValuesSection.stories.js b/src/components/Wv3dPropertiesPanel/GeneralValuesSection/GeneralValuesSection.stories.js
index ca0f98d39b..23ca96ae95 100644
--- a/src/components/Wv3dPropertiesPanel/GeneralValuesSection/GeneralValuesSection.stories.js
+++ b/src/components/Wv3dPropertiesPanel/GeneralValuesSection/GeneralValuesSection.stories.js
@@ -24,7 +24,7 @@ const initialState = {
panelWidths: {
wv3dPropertiesPanel: 330,
},
- modularHeaders: [],
+ modularHeaders: {},
modularHeadersHeight: {
topHeaders: 40,
bottomHeaders: 40
diff --git a/src/components/Wv3dPropertiesPanel/Group/Group.js b/src/components/Wv3dPropertiesPanel/Group/Group.js
new file mode 100644
index 0000000000..c3e4b9d825
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/Group/Group.js
@@ -0,0 +1,37 @@
+import React, { useState, useMemo } from 'react';
+import Icon from 'components/Icon';
+import PropertyKeyValuePair from '../PropertyKeyValuePair/PropertyKeyValuePair';
+import './Group.scss';
+
+const Group = (props) => {
+ const { name, data, open } = props;
+
+ const [isActive, setIsActive] = useState(open);
+ const downArrow = 'ic_chevron_down_black_24px';
+ const rightArrow = 'ic_chevron_right_black_24px';
+
+ const onClick = () => {
+ setIsActive(!isActive);
+ };
+
+ const elements = useMemo(() => {
+ return Object.entries(data).map((entity) => (
+
+ ));
+ }, [data]);
+
+ return (
+
+ );
+};
+
+export default React.memo(Group);
diff --git a/src/components/Wv3dPropertiesPanel/Group/Group.scss b/src/components/Wv3dPropertiesPanel/Group/Group.scss
new file mode 100644
index 0000000000..d9c1a61bf7
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/Group/Group.scss
@@ -0,0 +1,15 @@
+.group-title {
+ display: flex;
+ align-items: center;
+ margin-bottom: 5px;
+ user-select: all !important;
+}
+
+.dropdown.active {
+ visibility: visible;
+}
+
+.dropdown.inactive {
+ visibility: hidden;
+ display: none;
+}
diff --git a/src/components/Wv3dPropertiesPanel/Group/Group.stories.js b/src/components/Wv3dPropertiesPanel/Group/Group.stories.js
index 948ac7dfd7..20b590332c 100644
--- a/src/components/Wv3dPropertiesPanel/Group/Group.stories.js
+++ b/src/components/Wv3dPropertiesPanel/Group/Group.stories.js
@@ -24,7 +24,7 @@ const initialState = {
panelWidths: {
wv3dPropertiesPanel: 330,
},
- modularHeaders: [],
+ modularHeaders: {},
modularHeadersHeight: {
topHeaders: 40,
bottomHeaders: 40
diff --git a/src/components/Wv3dPropertiesPanel/GroupsContainer/GroupsContainer.js b/src/components/Wv3dPropertiesPanel/GroupsContainer/GroupsContainer.js
new file mode 100644
index 0000000000..521d13ec1a
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/GroupsContainer/GroupsContainer.js
@@ -0,0 +1,39 @@
+import React from 'react';
+import Group from '../Group/Group';
+
+function addOrderedGroups(orderedGroup, groups) {
+ const orderArray = [];
+ for (const group in orderedGroup) {
+ const groupName = orderedGroup[group];
+
+ if (groupName in groups) {
+ orderArray.push();
+ }
+ }
+
+ return orderArray;
+}
+
+const GroupsContainer = (props) => {
+ const { groups, groupOrder } = props;
+
+ let combinedGroups = [];
+
+ if (groupOrder && groupOrder.length > 0) {
+ combinedGroups = addOrderedGroups(groupOrder, groups);
+
+ for (const group in groups) {
+ if (!groupOrder.includes(group)) {
+ combinedGroups.push();
+ }
+ }
+ } else {
+ for (const group in groups) {
+ combinedGroups.push();
+ }
+ }
+
+ return {combinedGroups}
;
+};
+
+export default GroupsContainer;
diff --git a/src/components/Wv3dPropertiesPanel/GroupsContainer/GroupsContainer.stories.js b/src/components/Wv3dPropertiesPanel/GroupsContainer/GroupsContainer.stories.js
index 0fd8b3db72..cf6b73fa5f 100644
--- a/src/components/Wv3dPropertiesPanel/GroupsContainer/GroupsContainer.stories.js
+++ b/src/components/Wv3dPropertiesPanel/GroupsContainer/GroupsContainer.stories.js
@@ -24,7 +24,7 @@ const initialState = {
panelWidths: {
wv3dPropertiesPanel: 330,
},
- modularHeaders: [],
+ modularHeaders: {},
modularHeadersHeight: {
topHeaders: 40,
bottomHeaders: 40
diff --git a/src/components/Wv3dPropertiesPanel/HeaderTitle/HeaderTitle.js b/src/components/Wv3dPropertiesPanel/HeaderTitle/HeaderTitle.js
new file mode 100644
index 0000000000..8b664b6215
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/HeaderTitle/HeaderTitle.js
@@ -0,0 +1,18 @@
+import React from 'react';
+import { useTranslation } from 'react-i18next';
+import './HeaderTitle.scss';
+
+const HeaderTitle = (attributes) => {
+ const { title } = attributes;
+ const { t } = useTranslation();
+
+ return (
+
+ {t('wv3dPropertiesPanel.propertiesHeader')}
+
+ {title}
+
+ );
+};
+
+export default HeaderTitle;
diff --git a/src/components/Wv3dPropertiesPanel/HeaderTitle/HeaderTitle.scss b/src/components/Wv3dPropertiesPanel/HeaderTitle/HeaderTitle.scss
new file mode 100644
index 0000000000..9a7c4c74f2
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/HeaderTitle/HeaderTitle.scss
@@ -0,0 +1,7 @@
+.header-value {
+ color: var(--gray-7);
+}
+
+.header-title {
+ font-size: 16px;
+}
diff --git a/src/components/Wv3dPropertiesPanel/PropertiesElement/PropertiesElement.js b/src/components/Wv3dPropertiesPanel/PropertiesElement/PropertiesElement.js
new file mode 100644
index 0000000000..cb2f466444
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/PropertiesElement/PropertiesElement.js
@@ -0,0 +1,91 @@
+import React from 'react';
+import { useTranslation } from 'react-i18next';
+
+import GeneralValuesSection from '../GeneralValuesSection/GeneralValuesSection';
+import GroupsContainer from '../GroupsContainer/GroupsContainer';
+import Group from '../Group/Group';
+import HeaderTitle from '../HeaderTitle/HeaderTitle';
+
+function createDataSet(dataMap, propertySet, removeEmptyRows) {
+ const combinedMap = {};
+
+ if (removeEmptyRows) {
+ for (const item in dataMap) {
+ const dataPoint = propertySet[dataMap[item]];
+ if (dataPoint !== undefined && dataPoint !== '') {
+ combinedMap[item] = dataPoint;
+ }
+ }
+ } else {
+ for (const item in dataMap) {
+ combinedMap[item] = propertySet[dataMap[item]];
+ }
+ }
+
+ return combinedMap;
+}
+
+function checkForEmptyKeys(data) {
+ for (const key in data) {
+ const value = data[key];
+ if (value !== undefined && value !== '') {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+function generateGroupDataSet(dataMap, propertySet, removeEmptyRows, removeEmptyGroups) {
+ const combinedGroupMap = {};
+
+ if (removeEmptyGroups) {
+ for (const group in dataMap) {
+ const dataset = createDataSet(dataMap[group], propertySet, removeEmptyRows);
+ if (Object.keys(dataset).length > 0) {
+ if (!checkForEmptyKeys(dataset)) {
+ combinedGroupMap[group] = dataset;
+ }
+ }
+ }
+ } else {
+ for (const group in dataMap) {
+ combinedGroupMap[group] = createDataSet(dataMap[group], propertySet, removeEmptyRows);
+ }
+ }
+
+ return combinedGroupMap;
+}
+
+const PropertiesElement = (props) => {
+ const { element, schema } = props;
+
+ const {
+ headerName,
+ defaultValues,
+ groups,
+ groupOrder,
+ removeEmptyRows,
+ removeEmptyGroups,
+ createRawValueGroup,
+ } = schema;
+
+ const { t } = useTranslation();
+
+ const defaultItems = createDataSet(defaultValues, element, removeEmptyRows);
+ const groupsItems = generateGroupDataSet(groups, element, removeEmptyRows, removeEmptyGroups);
+ const name = element[headerName];
+
+ return (
+
+
+
+
+ {createRawValueGroup ? (
+
+ ) : null}
+
+ );
+};
+
+export default PropertiesElement;
diff --git a/src/components/Wv3dPropertiesPanel/PropertiesElement/PropertiesElement.stories.js b/src/components/Wv3dPropertiesPanel/PropertiesElement/PropertiesElement.stories.js
index df09df1d00..bd4e731d8a 100644
--- a/src/components/Wv3dPropertiesPanel/PropertiesElement/PropertiesElement.stories.js
+++ b/src/components/Wv3dPropertiesPanel/PropertiesElement/PropertiesElement.stories.js
@@ -106,7 +106,7 @@ const initialState = {
panelWidths: {
wv3dPropertiesPanel: 330,
},
- modularHeaders: [],
+ modularHeaders: {},
modularHeadersHeight: {
topHeaders: 40,
bottomHeaders: 40
diff --git a/src/components/Wv3dPropertiesPanel/PropertyKeyValuePair/PropertyKeyValuePair.js b/src/components/Wv3dPropertiesPanel/PropertyKeyValuePair/PropertyKeyValuePair.js
new file mode 100644
index 0000000000..c585d70bf4
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/PropertyKeyValuePair/PropertyKeyValuePair.js
@@ -0,0 +1,15 @@
+import React from 'react';
+import './PropertyKeyValuePair.scss';
+
+const PropertyKeyValuePair = (props) => {
+ const { name, value } = props;
+
+ return (
+
+ {name}
+ {value}
+
+ );
+};
+
+export default PropertyKeyValuePair;
diff --git a/src/components/Wv3dPropertiesPanel/PropertyKeyValuePair/PropertyKeyValuePair.scss b/src/components/Wv3dPropertiesPanel/PropertyKeyValuePair/PropertyKeyValuePair.scss
new file mode 100644
index 0000000000..2780e41b17
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/PropertyKeyValuePair/PropertyKeyValuePair.scss
@@ -0,0 +1,25 @@
+.property-pair {
+ margin-left: 24px;
+ display: flex;
+ flex-direction: row;
+ flex-wrap: nowrap;
+ justify-content: flex-start;
+ align-items: center;
+ padding-bottom: 10px;
+ user-select: all !important;
+}
+
+.property-key {
+ flex-basis: 140px;
+ white-space: nowrap;
+ overflow: hidden;
+ text-overflow: ellipsis;
+}
+
+.property-value {
+ white-space: nowrap;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ flex-basis: 300px;
+ padding-left: 20px;
+}
diff --git a/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.js b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.js
new file mode 100644
index 0000000000..b4bf9da7a2
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.js
@@ -0,0 +1,53 @@
+import React from 'react';
+import { useTranslation } from 'react-i18next';
+import Icon from 'components/Icon';
+import DataElementWrapper from '../DataElementWrapper';
+import { v4 as uuidv4 } from 'uuid';
+import './Wv3dPropertiesPanel.scss';
+
+import PropertiesElement from './PropertiesElement/PropertiesElement';
+
+const Wv3dPropertiesPanel = (props) => {
+ const { currentWidth, isInDesktopOnlyMode, isMobile = false, closeWv3dPropertiesPanel, schema, modelData } = props;
+
+ const { t } = useTranslation();
+ const style = !isInDesktopOnlyMode && isMobile ? {} : { width: `${currentWidth}px`, minWidth: `${currentWidth}px` };
+
+ const renderMobileCloseButton = () => {
+ return (
+
+ );
+ };
+
+ let propertiesCollection = modelData.map((element) => (
+
+ ));
+
+ const emptyPanelPlaceholder = () => {
+ return (
+
+
+
+
+
{t('wv3dPropertiesPanel.emptyPanelMessage')}
+
+ );
+ };
+
+ if (modelData.length < 1) {
+ propertiesCollection = emptyPanelPlaceholder();
+ }
+
+ return (
+
+ {!isInDesktopOnlyMode && isMobile && renderMobileCloseButton()}
+ {propertiesCollection}
+
+ );
+};
+
+export default Wv3dPropertiesPanel;
diff --git a/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.scss b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.scss
new file mode 100644
index 0000000000..5488f9b3b1
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.scss
@@ -0,0 +1,64 @@
+@import '../../constants/styles';
+@import '../../constants/panel';
+
+.wv3d-properties-panel {
+ padding: 16px 0px 0px 16px;
+ display: flex;
+ flex-direction: column;
+ overflow-y: auto;
+ user-select: all !important;
+
+ .no-selections {
+ flex-direction: column;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+
+ .empty-icon {
+ width: 100px;
+ height: 100px;
+
+ svg {
+ width: 100px;
+ height: 100px;
+ }
+
+ * {
+ fill: var(--gray-5);
+ color: var(--gray-5);
+ }
+ }
+
+ .empty-text {
+ margin-top: 4px;
+ padding-left: 4px;
+ padding-right: 4px;
+ width: 68%;
+ text-align: center;
+ }
+ }
+
+ @include mobile {
+ width: 100%;
+ min-width: 100%;
+ padding-top: 0px;
+
+ .close-container {
+ display: flex;
+ align-items: center;
+ justify-content: flex-end;
+ height: 64px;
+
+ width: 100%;
+ padding-right: 12px;
+
+ .close-icon-container {
+ cursor: pointer;
+ .close-icon {
+ width: 24px;
+ height: 24px;
+ }
+ }
+ }
+ }
+}
diff --git a/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.spec.js b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.spec.js
new file mode 100644
index 0000000000..337fa72f85
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.spec.js
@@ -0,0 +1,322 @@
+import React from 'react';
+import { render } from '@testing-library/react';
+import Wv3dPropertiesPanel from './Wv3dPropertiesPanel';
+
+import {
+ DefaultStandard,
+ GroupOrderSpecified,
+ MultiplePropertiesElements,
+ EmptyPanel,
+ RemoveEmptyRows,
+ RemoveEmptyGroups,
+ RemoveRawValues,
+} from './Wv3dPropertiesPanel.stories';
+
+const sampleData = [
+ {
+ ConnectedFrom: '2274843',
+ ConnectedTo: '2274847',
+ ContainedInStructure: '2258156',
+ Declares: '',
+ Decomposes: '',
+ Description: 'ÿ',
+ ExtendToStructure: 'T',
+ FillsVoids: '',
+ GlobalId: '3_YR89Qiz6UgyQsBcI$FJz',
+ GrossFootprintArea: '317.638889',
+ GrossSideArea: '4879.727431',
+ GrossVolume: '8132.879051',
+ HasAssignments: '',
+ HasAssociations: '2260414',
+ HasContext: '',
+ HasCoverings: '',
+ HasOpenings: '',
+ HasProjections: '',
+ Height: '25.604167',
+ InterferesElements: '',
+ IsConnectionRealization: '',
+ IsDeclaredBy: '',
+ IsDefinedBy: '29092,29099',
+ IsExternal: 'T',
+ IsInterferedByElements: '',
+ IsNestedBy: '',
+ IsTypedBy: '2266845',
+ Length: '190.583333',
+ LoadBearing: 'F',
+ Name: 'Basic Wall:Reinforced Concrete - 1\'-8":117463',
+ Nests: '',
+ ObjectPlacement: '29040',
+ ObjectType: 'Basic Wall:Reinforced Concrete - 1\'-8":118691',
+ OwnerHistory: '42',
+ PredefinedType: 'NOTDEFINED',
+ ProvidesBoundaries: '',
+ Reference: 'Basic Wall:Reinforced Concrete - 1\'-8"',
+ ReferencedBy: '',
+ ReferencedInStructures: '',
+ Representation: '29077',
+ Tag: '117463',
+ Width: '1.666667',
+ handle: '29081',
+ },
+];
+
+const sampleSchema = {
+ headerName: 'Name',
+ defaultValues: {},
+ groups: {},
+ groupOrder: [],
+ removeEmptyRows: false,
+ removeEmptyGroups: false,
+ createRawValueGroup: false,
+};
+
+describe('Wv3dPropertiesPanel', () => {
+ it('The Header is populated with a title', () => {
+ render(
+ ,
+ );
+
+ const name = sampleSchema['headerName'];
+ const expectedHeader = sampleData[0][name];
+ const res = document.body.getElementsByClassName('header-value');
+
+ expect(res[0].innerHTML).toContain(expectedHeader);
+ });
+
+ it('Default Values are generated', () => {
+ sampleSchema.defaultValues = {
+ 'GrossVolume': 'GrossVolume',
+ 'OwnerHistory': 'OwnerHistory',
+ };
+
+ const { getByText } = render(
+ ,
+ );
+
+ expect(getByText('GrossVolume')).toBeInTheDocument();
+ expect(getByText(sampleData[0]['GrossVolume'])).toBeInTheDocument();
+
+ expect(getByText('OwnerHistory')).toBeInTheDocument();
+ expect(getByText(sampleData[0]['OwnerHistory'])).toBeInTheDocument();
+ });
+
+ it('Groups are created succesfully', async () => {
+ sampleSchema.defaultValues = {};
+ sampleSchema.groups = {
+ TestGroup: {
+ 'GrossVolume': 'GrossVolume',
+ 'OwnerHistory': 'OwnerHistory',
+ },
+ };
+
+ const { getByText } = render(
+ ,
+ );
+
+ expect(getByText('GrossVolume')).toBeInTheDocument();
+ expect(getByText(sampleData[0]['GrossVolume'])).toBeInTheDocument();
+
+ expect(getByText('OwnerHistory')).toBeInTheDocument();
+ expect(getByText(sampleData[0]['OwnerHistory'])).toBeInTheDocument();
+
+ expect(getByText('TestGroup')).toBeInTheDocument();
+ });
+
+ it('Empty Values are removed when removeEmptyRows is true', () => {
+ sampleSchema.defaultValues = { 'GrossVolume': 'GrossVolume', 'OwnerHistory': 'OwnerHistory', 'EmptyTest': '' };
+ sampleSchema.removeEmptyRows = true;
+ sampleSchema.groups = {};
+
+ const { getByText, queryByText } = render(
+ ,
+ );
+
+ expect(getByText('GrossVolume')).toBeInTheDocument();
+ expect(getByText(sampleData[0]['GrossVolume'])).toBeInTheDocument();
+
+ expect(getByText('OwnerHistory')).toBeInTheDocument();
+ expect(getByText(sampleData[0]['OwnerHistory'])).toBeInTheDocument();
+
+ expect(queryByText('EmptyTest')).toBe(null);
+ });
+
+ it('Empty Groups are removed when removeEmptyGroups is true', () => {
+ sampleSchema.defaultValues = {};
+ sampleSchema.groups = {
+ TestGroup: {
+ 'GrossVolume': 'GrossVolume',
+ 'OwnerHistory': 'OwnerHistory',
+ 'EmptyTest': '',
+ },
+ TestGroup2: {
+ 'EmptyTest': '',
+ 'EmptyTest2': '',
+ 'EmptyTest3': '',
+ },
+ };
+ sampleSchema.removeEmptyRows = true;
+ sampleSchema.removeEmptyGroups = true;
+
+ const { getByText, queryByText } = render(
+ ,
+ );
+
+ expect(getByText('TestGroup')).toBeInTheDocument();
+ expect(queryByText('TestGroup2')).toBe(null);
+ });
+
+ it('Groups are ordered correctly', () => {
+ sampleSchema.defaultValues = {};
+ sampleSchema.groups = {
+ TestGroup: {
+ 'GrossVolume': 'GrossVolume',
+ 'OwnerHistory': 'OwnerHistory',
+ 'EmptyTest': '',
+ },
+ TestGroup2: {
+ 'EmptyTest': '',
+ 'EmptyTest2': '',
+ 'EmptyTest3': '',
+ },
+ TestGroup3: {
+ 'EmptyTest': '',
+ 'EmptyTest2': '',
+ 'EmptyTest3': '',
+ },
+ };
+
+ sampleSchema.groupOrder = ['TestGroup2', 'TestGroup3'];
+ sampleSchema.removeEmptyRows = false;
+ sampleSchema.removeEmptyGroups = false;
+
+ const { asFragment } = render(
+ ,
+ );
+
+ const fragment = asFragment();
+ const groupContainer = fragment.querySelector('[data-element="groupsContainer"]');
+
+ expect(groupContainer.children.length).toBe(3);
+ expect(groupContainer.children[0]).toHaveTextContent('TestGroup2');
+ expect(groupContainer.children[1]).toHaveTextContent('TestGroup3');
+ expect(groupContainer.children[2]).toHaveTextContent('TestGroup');
+ });
+
+ it('Raw Value Group was created successfully', () => {
+ sampleSchema.createRawValueGroup = true;
+
+ const { getByText } = render(
+ ,
+ );
+
+ expect(getByText('All')).toBeInTheDocument();
+ });
+
+ it('Raw Value Group should not be in the document', () => {
+ sampleSchema.createRawValueGroup = false;
+
+ const { queryByText } = render(
+ ,
+ );
+
+ expect(queryByText('All')).toBe(null);
+ });
+
+ it('renders the storybook component with defaults correctly', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('renders the storybook component with ordered groups correctly', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('renders the storybook component with empty rows removed correctly', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('renders the storybook component with empty groups removed correctly', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('renders the storybook component with the raw values section removed correctly', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('renders the storybook component with multiple elements correctly', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+
+ it('renders the storybook component with zero elements correctly', () => {
+ expect(() => {
+ render();
+ }).not.toThrow();
+ });
+});
diff --git a/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.stories.js b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.stories.js
index 4484ba570e..3a897e67e3 100644
--- a/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.stories.js
+++ b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanel.stories.js
@@ -201,7 +201,7 @@ const initialState = {
panelWidths: {
wv3dPropertiesPanel: 330,
},
- modularHeaders: [],
+ modularHeaders: {},
modularHeadersHeight: {
topHeaders: 40,
bottomHeaders: 40
diff --git a/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanelContainer.js b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanelContainer.js
new file mode 100644
index 0000000000..af740bdf12
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/Wv3dPropertiesPanelContainer.js
@@ -0,0 +1,57 @@
+import React, { useEffect, useState } from 'react';
+import Wv3dPropertiesPanel from './Wv3dPropertiesPanel';
+import { useSelector, shallowEqual, useDispatch } from 'react-redux';
+import selectors from 'selectors';
+import actions from 'actions';
+
+import { isMobileSize } from 'helpers/getDeviceSize';
+
+const Wv3dPropertiesPanelContainer = () => {
+ const [isOpen, isDisabled, wv3dPropertiesPanelWidth, isInDesktopOnlyMode, modelData, schema] = useSelector(
+ (state) => [
+ selectors.isElementOpen(state, 'wv3dPropertiesPanel'),
+ selectors.isElementDisabled(state, 'wv3dPropertiesPanel'),
+ selectors.getWv3dPropertiesPanelWidth(state),
+ selectors.isInDesktopOnlyMode(state),
+ selectors.getWv3dPropertiesPanelModelData(state),
+ selectors.getWv3dPropertiesPanelSchema(state),
+ ],
+ shallowEqual,
+ );
+
+ const isMobile = isMobileSize();
+
+ const dispatch = useDispatch();
+
+ const closeWv3dPropertiesPanel = () => {
+ dispatch(actions.closeElement('wv3dPropertiesPanel'));
+ };
+
+ const [renderNull, setRenderNull] = useState(false);
+
+ useEffect(() => {
+ const timeout = setTimeout(() => {
+ setRenderNull(!isOpen);
+ }, 500);
+ return () => {
+ clearTimeout(timeout);
+ };
+ }, [isOpen]);
+
+ if (isDisabled || (!isOpen && renderNull)) {
+ return null;
+ }
+
+ return (
+
+ );
+};
+
+export default Wv3dPropertiesPanelContainer;
diff --git a/src/components/Wv3dPropertiesPanel/index.js b/src/components/Wv3dPropertiesPanel/index.js
new file mode 100644
index 0000000000..c0e4c60388
--- /dev/null
+++ b/src/components/Wv3dPropertiesPanel/index.js
@@ -0,0 +1,3 @@
+import Wv3dPropertiesPanel from './Wv3dPropertiesPanelContainer';
+
+export default Wv3dPropertiesPanel;
diff --git a/src/constants/defaultToolsWithInlineCommentOnAnnotationSelected.js b/src/constants/defaultToolsWithInlineCommentOnAnnotationSelected.js
new file mode 100644
index 0000000000..516ca89263
--- /dev/null
+++ b/src/constants/defaultToolsWithInlineCommentOnAnnotationSelected.js
@@ -0,0 +1,10 @@
+// default annotations that have inline comment enabled on select
+export default [
+ window.Core.Annotations.TextUnderlineAnnotation,
+ window.Core.Annotations.TextHighlightAnnotation,
+ window.Core.Annotations.FreeTextAnnotation,
+ window.Core.Annotations.CaretAnnotation,
+ window.Core.Annotations.StickyAnnotation,
+ window.Core.Annotations.TextSquigglyAnnotation,
+ window.Core.Annotations.TextStrikeoutAnnotation,
+];
\ No newline at end of file
diff --git a/src/constants/featureFlags.js b/src/constants/featureFlags.js
new file mode 100644
index 0000000000..d49647551d
--- /dev/null
+++ b/src/constants/featureFlags.js
@@ -0,0 +1,11 @@
+/**
+ * Contains string enums for WebViewer feature flags
+ * @name UI.FeatureFlags
+ * @property {string} CUSTOMIZABLE_UI Feature flag for the new customizable UI
+ * @ignore
+ */
+const FEATURE_FLAGS = {
+ CUSTOMIZABLE_UI: 'customizableUI',
+};
+
+export default FEATURE_FLAGS;
\ No newline at end of file
diff --git a/src/constants/highContrastDark.scss b/src/constants/highContrastDark.scss
index 9ff92de340..297aa3aadd 100644
--- a/src/constants/highContrastDark.scss
+++ b/src/constants/highContrastDark.scss
@@ -100,4 +100,6 @@
--outline-color: var(--blue-7);
--outline-hover: var(--blue-3);
+
+ --oe-table-dropdown-highlight: var(--blue-5);
}
diff --git a/src/constants/highContrastLight.scss b/src/constants/highContrastLight.scss
index be1b459121..5bcf4c3b76 100644
--- a/src/constants/highContrastLight.scss
+++ b/src/constants/highContrastLight.scss
@@ -103,4 +103,6 @@
--outline-hover: var(--blue-1);
--preset-background: var(--gray-1);
+
+ --oe-table-dropdown-highlight: var(--blue-4);
}
diff --git a/src/constants/languages.js b/src/constants/languages.js
new file mode 100644
index 0000000000..e4681ca970
--- /dev/null
+++ b/src/constants/languages.js
@@ -0,0 +1,72 @@
+// The values in this array should match the language codes of the json files inside the i18n folder
+const Languages = [
+ ['en', 'English'],
+ ['el', 'Ελληνικά'],
+ ['de', 'Deutsch'],
+ ['es', 'Español'],
+ ['fr', 'Français'],
+ ['hu', 'Magyar'],
+ ['it', 'Italiano'],
+ ['ja', '日本語'],
+ ['ko', '한국어'],
+ ['nl', 'Nederlands'],
+ ['pt_br', 'Português'],
+ ['pl', 'Polski'],
+ ['uk', 'українська'],
+ ['ru', 'Pусский'],
+ ['ro', 'Romanian'],
+ ['sv', 'Svenska'],
+ ['tr', 'Türk'],
+ ['th', 'ไทย'],
+ ['vi', 'Tiếng Việt'],
+ ['ms', 'Melayu'],
+ ['hi', 'हिन्दी'],
+ ['bn', 'বাংলা'],
+ ['zh_cn', '简体中文'],
+ ['zh_tw', '繁體中文'],
+ ['cs', 'česky, čeština'],
+ ['id', 'Bahasa Indonesia']
+];
+
+/**
+ * Contains string enums for the default languages found in WebViewer.
+ * @name UI.Languages
+ * @property {string} EN English (en)
+ * @property {string} CS česky, čeština (cs)
+ * @property {string} EL Ελληνικά (el)
+ * @property {string} DE Deutsch (de)
+ * @property {string} ES Español (es)
+ * @property {string} FR Français (fr)
+ * @property {string} HU Magyar (hu)
+ * @property {string} IT Italiano (it)
+ * @property {string} JA 日本語 (ja)
+ * @property {string} KO 한국어 (ko)
+ * @property {string} NL Nederlands (nl)
+ * @property {string} PT_BR Português (pt_br)
+ * @property {string} PL Polski (pl)
+ * @property {string} UK українська (uk)
+ * @property {string} RU Pусский (ru)
+ * @property {string} RO Romanian (ro)
+ * @property {string} SV Svenska (sv)
+ * @property {string} TR Türk (tr)
+ * @property {string} TH ไทย (th)
+ * @property {string} VI Tiếng Việt (vi)
+ * @property {string} ID Bahasa Indonesia (id)
+ * @property {string} MS Melayu (ms)
+ * @property {string} HI हिन्दी (hi)
+ * @property {string} BN বাংলা (bn)
+ * @property {string} ZH_CN 简体中文 (zh_cn)
+ * @property {string} ZH_TW 繁體中文 (zh_tw)
+ * @example
+ WebViewer(...).then(function(instance) {
+ instance.UI.setLanguage(instance.UI.Languages.FR);
+ });
+ */
+
+export const languageEnum = Languages.reduce((acc, pair) => {
+ const code = pair[0];
+ acc[code.toUpperCase()] = code;
+ return acc;
+}, {});
+
+export default Languages;
diff --git a/src/constants/measurementScale.js b/src/constants/measurementScale.js
new file mode 100644
index 0000000000..bef86c3b5c
--- /dev/null
+++ b/src/constants/measurementScale.js
@@ -0,0 +1,150 @@
+const Scale = window.Core.Scale;
+
+export const PresetMeasurementSystems = {
+ METRIC: 'metric',
+ IMPERIAL: 'imperial'
+};
+
+const metricPreset = [
+ ['1:10', new Scale([[1, 'mm'], [10, 'mm']])],
+ ['1:20', new Scale([[1, 'mm'], [20, 'mm']])],
+ ['1:50', new Scale([[1, 'mm'], [50, 'mm']])],
+ ['1:100', new Scale([[1, 'mm'], [100, 'mm']])],
+ ['1:200', new Scale([[1, 'mm'], [200, 'mm']])],
+ ['1:500', new Scale([[1, 'mm'], [500, 'mm']])],
+ ['1:1000', new Scale([[1, 'mm'], [1000, 'mm']])]
+];
+const imperialPreset = [
+ ['1/16"=1\'-0"', new Scale([[1 / 16, 'in'], [1, 'ft-in']])],
+ ['3/32"=1\'-0"', new Scale([[3 / 32, 'in'], [1, 'ft-in']])],
+ ['1/8"=1\'-0"', new Scale([[1 / 8, 'in'], [1, 'ft-in']])],
+ ['3/16"=1\'-0"', new Scale([[3 / 16, 'in'], [1, 'ft-in']])],
+ ['1/4"=1\'-0"', new Scale([[1 / 4, 'in'], [1, 'ft-in']])],
+ ['3/8"=1\'-0"', new Scale([[3 / 8, 'in'], [1, 'ft-in']])],
+ ['1/2"=1\'-0"', new Scale([[1 / 2, 'in'], [1, 'ft-in']])],
+ ['3/4"=1\'-0"', new Scale([[3 / 4, 'in'], [1, 'ft-in']])],
+ ['1"=1\'-0"', new Scale([[1, 'in'], [1, 'ft-in']])]
+];
+
+export const getMeasurementScalePreset = () => ({
+ [PresetMeasurementSystems.METRIC]: metricPreset,
+ [PresetMeasurementSystems.IMPERIAL]: imperialPreset
+});
+
+const decimalPrecisions = [
+ ['0.1', 0.1],
+ ['0.01', 0.01],
+ ['0.001', 0.001],
+ ['0.0001', 0.0001]
+];
+const fractionalPrecisions = [
+ ['1/8', 0.125],
+ ['1/16', 0.0625],
+ ['1/32', 0.03125],
+ ['1/64', 0.015625]
+];
+export const PrecisionType = {
+ DECIMAL: 'decimal',
+ FRACTIONAL: 'fractional'
+};
+export const precisionOptions = {
+ [PrecisionType.DECIMAL]: decimalPrecisions,
+ [PrecisionType.FRACTIONAL]: fractionalPrecisions
+};
+
+export const precisionFractions = {
+ 0.125: '1/8',
+ 0.0625: '1/16',
+ 0.03125: '1/32',
+ 0.015625: '1/64'
+};
+
+export const numberRegex = /^\d*(\.\d*)?$/;
+export const fractionRegex = /^\d*(\s\d\/\d*)$/;
+export const pureFractionRegex = /^(\d\/\d*)*$/;
+export const floatRegex = /^(\d+)?(\.)?(\d+)?$/;
+export const inFractionalRegex = /^((\d+) )?((\d+)\/)?(\d+)"$/;
+export const ftInFractionalRegex = /^((\d+)'-)?((\d+) )?((\d+)\/)?(\d+)"$/;
+export const ftInDecimalRegex = /^((\d+)ft-)?(((\d+).)?(\d+))in$/;
+
+export const parseFtInDecimal = (valueStr) => {
+ const matches = valueStr.match(ftInDecimalRegex);
+ let sum = 0;
+ sum += matches[2] ? Number(matches[2]) : 0;
+ if (matches[3] && Number(matches[3])) {
+ sum += (Number(matches[3]) / 12);
+ }
+ return sum;
+};
+export const parseInFractional = (valueStr) => {
+ const matches = valueStr.match(inFractionalRegex);
+ let sum = 0;
+ sum += matches[2] ? Number(matches[2]) : 0;
+ if (matches[5] && Number(matches[5])) {
+ if (matches[4] && Number(matches[4])) {
+ sum += (Number(matches[4]) / Number(matches[5]));
+ } else {
+ sum += Number(matches[5]);
+ }
+ }
+ return sum;
+};
+export const parseFtInFractional = (valueStr) => {
+ const matches = valueStr.match(ftInFractionalRegex);
+ let sum = 0;
+ sum += matches[2] ? Number(matches[2]) : 0;
+ sum += matches[4] ? Number(matches[4]) / 12 : 0;
+ if (matches[7] && Number(matches[7])) {
+ if (matches[6] && Number(matches[6])) {
+ sum += (Number(matches[6]) / Number(matches[7])) / 12;
+ } else {
+ sum += Number(matches[7]) / 12;
+ }
+ }
+ return sum;
+};
+
+export const fractionalUnits = ['in', 'ft-in'];
+export const metricUnits = ['mm', 'cm', 'm', 'km'];
+
+export const ifFractionalPrecision = (precision) => fractionalPrecisions.map((item) => item[0]).includes(precision) || fractionalPrecisions.map((item) => item[1]).includes(precision);
+
+export const hintValues = {
+ 'in': 'eg. 1 1/2"',
+ 'ft-in': 'eg. 1\'-1 1/2"',
+ 'ft-in decimal': 'eg. 1ft-10.5in'
+};
+
+// the base unit is cm
+const unitConversion = {
+ 'mm': 0.1,
+ 'cm': 1,
+ 'm': 100,
+ 'km': 100000,
+ 'mi': 160394,
+ 'yd': 91.44,
+ 'ft': 30.48,
+ 'in': 2.54,
+ 'ft\'': 30.48,
+ 'in"': 2.54,
+ 'pt': 0.0352778,
+ 'ft-in': 30.48
+};
+
+export const convertUnit = (value, unit, newUnit) => {
+ return value * unitConversion[unit] / unitConversion[newUnit];
+};
+
+export const scalePresetPrecision = {
+ [imperialPreset[0][0]]: fractionalPrecisions[1],
+ [imperialPreset[1][0]]: fractionalPrecisions[2],
+ [imperialPreset[2][0]]: fractionalPrecisions[0],
+ [imperialPreset[3][0]]: fractionalPrecisions[1],
+ [imperialPreset[4][0]]: fractionalPrecisions[0],
+ [imperialPreset[5][0]]: fractionalPrecisions[0],
+ [imperialPreset[6][0]]: fractionalPrecisions[0],
+ [imperialPreset[7][0]]: fractionalPrecisions[0],
+ [imperialPreset[8][0]]: fractionalPrecisions[0]
+};
+
+export const initialScale = new Scale({ pageScale: { value: 1, unit: 'in' }, worldScale: { value: 1, unit: 'in' } });
diff --git a/src/constants/measurementTypes.js b/src/constants/measurementTypes.js
new file mode 100644
index 0000000000..2ef30b46ff
--- /dev/null
+++ b/src/constants/measurementTypes.js
@@ -0,0 +1,9 @@
+export const measurementTypeTranslationMap = {
+ distanceMeasurement: 'option.measurementOverlay.distanceMeasurement',
+ perimeterMeasurement: 'option.measurementOverlay.perimeterMeasurement',
+ areaMeasurement: 'option.measurementOverlay.areaMeasurement',
+ rectangularAreaMeasurement: 'option.measurementOverlay.areaMeasurement',
+ cloudyRectangularAreaMeasurement: 'option.measurementOverlay.areaMeasurement',
+ ellipseMeasurement: 'option.measurementOverlay.areaMeasurement',
+ arcMeasurement: 'option.measurementOverlay.arcMeasurement',
+};
\ No newline at end of file
diff --git a/src/constants/multiViewerContants.js b/src/constants/multiViewerContants.js
new file mode 100644
index 0000000000..c0c1865cb2
--- /dev/null
+++ b/src/constants/multiViewerContants.js
@@ -0,0 +1,7 @@
+export const DISABLED_TOOL_GROUPS = ['toolbarGroup-Edit', 'toolbarGroup-Forms', 'toolbarGroup-EditText'];
+export const DISABLED_TOOLS_KEYWORDS = ['Content', 'AddParagraphTool', 'FormField', 'Crop'];
+
+export const SYNC_MODES = {
+ 'SYNC': 'SYNC',
+ 'SKIP_UNMATCHED': 'SKIP_UNMATCHED',
+};
diff --git a/src/constants/officeEditorFonts.js b/src/constants/officeEditorFonts.js
new file mode 100644
index 0000000000..d53ebdfe48
--- /dev/null
+++ b/src/constants/officeEditorFonts.js
@@ -0,0 +1,170 @@
+export const availableFontFaces = [
+ 'Arial',
+ 'Arial Black',
+ 'Arial Narrow',
+ 'Arial Rounded MT Bold',
+ 'Baskerville Old Face',
+ 'Bookman Old Style',
+ 'Bookshelf Symbol 7',
+ 'Brush Script MT',
+ 'Calibri',
+ 'Calibri Light',
+ 'Cambria',
+ 'Cambria Math',
+ 'Century',
+ 'Century Schoolbook',
+ 'Comic Sans MS',
+ 'Consolas',
+ 'Cooper Black',
+ 'Copperplate Gothic Light',
+ 'Courier',
+ 'Courier New',
+ 'Garamond',
+ 'Georgia',
+ 'Gill Sans MT',
+ 'Gill Sans MT Condensed',
+ 'Helvetica',
+ 'Lucida Console',
+ 'MS Outlook',
+ 'Malgun Gothic',
+ 'Meiryo',
+ 'monospace',
+ 'Myriad Pro',
+ 'sans-serif',
+ 'serif',
+ 'SimSun',
+ 'Symbol',
+ 'Tahoma',
+ 'Tahoma Bold',
+ 'Times New Roman',
+ 'Trebuchet MS',
+ 'Verdana',
+];
+
+export const cssFontValues = {
+ 'Arial': {
+ fontFamily: 'Arial, Helvetica, sans-serif',
+ },
+ 'Arial Black': {
+ fontFamily: '"Arial Black", Gadget, sans-serif',
+ },
+ 'Arial Italic': {
+ fontFamily: 'Arial, Helvetica, sans-serif',
+ fontStyle: 'italic',
+ },
+ 'Arial Narrow': {
+ fontFamily: '"Arial Narrow", sans-serif',
+ },
+ 'Arial Rounded MT Bold': {
+ fontFamily: '"Arial Rounded MT Bold", sans-serif',
+ },
+ 'Baskerville Old Face': {
+ fontFamily: '"Baskerville Old Face", "Book Antiqua", Palatino, serif',
+ },
+ 'Bookman Old Style': {
+ fontFamily: '"Bookman Old Style", serif',
+ },
+ 'Bookshelf Symbol 7': {
+ fontFamily: '"Bookshelf Symbol 7", sans-serif',
+ },
+ 'Brush Script MT': {
+ fontFamily: '"Brush Script MT", cursive',
+ },
+ 'Calibri': {
+ fontFamily: 'Calibri, Candara, Segoe, "Segoe UI", Optima, Arial, sans-serif',
+ },
+ 'Calibri Light': {
+ fontFamily: 'Calibri, Candara, Segoe, "Segoe UI", Optima, Arial, sans-serif',
+ },
+ 'Cambria': {
+ fontFamily: 'Cambria, Georgia, serif',
+ },
+ 'Cambria Math': {
+ fontFamily: '"Cambria Math", serif',
+ },
+ 'Century': {
+ fontFamily: 'Century, sans-serif',
+ },
+ 'Century Schoolbook': {
+ fontFamily: '"Century Schoolbook", serif',
+ },
+ 'Comic Sans MS': {
+ fontFamily: '"Comic Sans MS", cursive, sans-serif',
+ },
+ 'Consolas': {
+ fontFamily: 'Consolas, monaco, monospace',
+ },
+ 'Consolas Italic': {
+ fontFamily: 'Consolas, monaco, monospace',
+ fontStyle: 'italic',
+ },
+ 'Cooper Black': {
+ fontFamily: '"Cooper Black", sans-serif',
+ },
+ 'Copperplate Gothic Light': {
+ fontFamily: '"Copperplate Gothic Light", sans-serif',
+ },
+ 'Courier': {
+ fontFamily: 'Courier, monospace',
+ },
+ 'Courier New': {
+ fontFamily: '"Courier New", Courier, monospace',
+ },
+ 'Garamond': {
+ fontFamily: 'Garamond, serif',
+ },
+ 'Georgia': {
+ fontFamily: 'Georgia, serif',
+ },
+ 'Gill Sans MT': {
+ fontFamily: '"Gill Sans MT", sans-serif',
+ },
+ 'Gill Sans MT Condensed': {
+ fontFamily: '"Gill Sans MT Condensed", sans-serif',
+ },
+ 'Helvetica': {
+ fontFamily: 'Helvetica, Arial, sans-serif',
+ },
+ 'Lucida Console': {
+ fontFamily: '"Lucida Console", Monaco, monospace',
+ },
+ 'MS Outlook': {
+ fontFamily: '"MS Outlook", sans-serif',
+ },
+ 'Malgun Gothic': {
+ fontFamily: '"Malgun Gothic", sans-serif',
+ },
+ 'Meiryo': {
+ fontFamily: 'Meiryo, sans-serif',
+ },
+ 'monospace': {
+ fontFamily: 'monospace',
+ },
+ 'Myriad Pro': {
+ fontFamily: '"Myriad Pro", Myriad, sans-serif',
+ },
+ 'sans-serif': {
+ fontFamily: 'sans-serif',
+ },
+ 'serif': {
+ fontFamily: 'serif',
+ },
+ 'Symbol': {
+ fontFamily: 'Symbol, sans-serif',
+ },
+ 'SimSun': {
+ fontFamily: '"SimSun", sans-serif',
+ },
+ 'Tahoma': {
+ fontFamily: 'Tahoma, Geneva, sans-serif',
+ },
+ 'Times New Roman': {
+ fontFamily: '"Times New Roman", Times, serif',
+ },
+ 'Trebuchet MS': {
+ fontFamily: '"Trebuchet MS", Helvetica, sans-serif',
+ },
+ 'Verdana': {
+ fontFamily: 'Verdana, Geneva, sans-serif',
+ },
+};
diff --git a/src/constants/pageNumberPlaceholder.js b/src/constants/pageNumberPlaceholder.js
new file mode 100644
index 0000000000..c597b8f901
--- /dev/null
+++ b/src/constants/pageNumberPlaceholder.js
@@ -0,0 +1 @@
+export default '1, 3, 5-10';
\ No newline at end of file
diff --git a/src/constants/presetNewPageDimensions.js b/src/constants/presetNewPageDimensions.js
new file mode 100644
index 0000000000..5535946921
--- /dev/null
+++ b/src/constants/presetNewPageDimensions.js
@@ -0,0 +1,14 @@
+export default {
+ 'Letter': {
+ 'height': 11,
+ 'width': 8.5,
+ },
+ 'Half letter': {
+ 'height': 5.5,
+ 'width': 8.5,
+ },
+ 'Junior legal': {
+ 'height': 5,
+ 'width': 8,
+ },
+};
\ No newline at end of file
diff --git a/src/constants/signatureModes.js b/src/constants/signatureModes.js
new file mode 100644
index 0000000000..a28b91c554
--- /dev/null
+++ b/src/constants/signatureModes.js
@@ -0,0 +1,6 @@
+const SignatureModes = {
+ FULL_SIGNATURE: window.Core.Tools.SignatureCreateTool.SignatureTypes.FULL_SIGNATURE,
+ INITIALS: window.Core.Tools.SignatureCreateTool.SignatureTypes.INITIALS,
+};
+
+export default SignatureModes;
\ No newline at end of file
diff --git a/src/constants/webFonts.js b/src/constants/webFonts.js
new file mode 100644
index 0000000000..09788bd9a5
--- /dev/null
+++ b/src/constants/webFonts.js
@@ -0,0 +1,14 @@
+// web fonts from https://www.pdftron.com/webfonts/v2/fonts.json
+// that support bold, italic, and bold-italic
+// and can be used in content editing
+export default [
+ 'Arimo',
+ 'Caladea',
+ 'Carlito',
+ 'Cousine',
+ 'Liberation Serif',
+ 'Open Sans',
+ 'Roboto',
+ 'Roboto Mono',
+ 'Tinos',
+];
diff --git a/src/core/createAndApplyScale.js b/src/core/createAndApplyScale.js
new file mode 100644
index 0000000000..800947737b
--- /dev/null
+++ b/src/core/createAndApplyScale.js
@@ -0,0 +1,8 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.MeasurementManager.html#createScale__anchor
+ */
+export default (scale, applyTo, documentViewerKey = 1) => {
+ core.getDocumentViewer(documentViewerKey).getMeasurementManager().createAndApplyScale({ scale, applyTo });
+};
diff --git a/src/core/deleteScale.js b/src/core/deleteScale.js
new file mode 100644
index 0000000000..df4f9cc2d3
--- /dev/null
+++ b/src/core/deleteScale.js
@@ -0,0 +1,8 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.MeasurementManager.html#deleteScale__anchor
+ */
+export default (scale, documentViewerKey = 1) => {
+ core.getDocumentViewer(documentViewerKey).getMeasurementManager().deleteScale(scale);
+};
diff --git a/src/core/deselectAnnotations.js b/src/core/deselectAnnotations.js
new file mode 100644
index 0000000000..7bf77bbf79
--- /dev/null
+++ b/src/core/deselectAnnotations.js
@@ -0,0 +1,10 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.AnnotationManager.html#deselectAnnotations__anchor
+ * @fires annotationSelected on AnnotationManager
+ * @see https://docs.apryse.com/api/web/Core.AnnotationManager.html#event:annotationSelected__anchor
+ */
+export default (annotations, documentViewerKey = 1) => {
+ core.getDocumentViewer(documentViewerKey).getAnnotationManager().deselectAnnotations(annotations);
+};
diff --git a/src/core/documentViewers.js b/src/core/documentViewers.js
new file mode 100644
index 0000000000..10ee9e51a4
--- /dev/null
+++ b/src/core/documentViewers.js
@@ -0,0 +1,18 @@
+const documentViewerMap = new Map();
+
+export const setDocumentViewer = (number, documentViewer) => {
+ documentViewerMap.set(number, documentViewer);
+ return documentViewer;
+};
+
+export const deleteDocumentViewer = (number) => {
+ documentViewerMap.delete(number);
+};
+
+export const getDocumentViewer = (number = 1) => {
+ return documentViewerMap.get(number);
+};
+
+export const getDocumentViewers = () => {
+ return Array.from(documentViewerMap.values());
+};
\ No newline at end of file
diff --git a/src/core/enableAnnotationNumbering.js b/src/core/enableAnnotationNumbering.js
new file mode 100644
index 0000000000..e05e8dc68c
--- /dev/null
+++ b/src/core/enableAnnotationNumbering.js
@@ -0,0 +1,8 @@
+import getAnnotationManager from './getAnnotationManager';
+
+/**
+ * https://docs.apryse.com/api/web/Core.AnnotationManager.html#enableAnnotationNumbering__anchor
+ */
+export default (documentViewerKey = 1) => {
+ getAnnotationManager(documentViewerKey).enableAnnotationNumbering();
+};
\ No newline at end of file
diff --git a/src/core/getAllowedFileExtensions.js b/src/core/getAllowedFileExtensions.js
new file mode 100644
index 0000000000..f87656ffeb
--- /dev/null
+++ b/src/core/getAllowedFileExtensions.js
@@ -0,0 +1,8 @@
+/**
+ * https://docs.apryse.com/api/web/Core.html#.getAllowedFileExtensions__anchor
+ */
+export default () => {
+ return window.Core.getAllowedFileExtensions().length > 0 ?
+ window.Core.getAllowedFileExtensions().map((format) => `.${format}`,).join(', ') :
+ window.Core.SupportedFileFormats.CLIENT.map((format) => `.${format}`,).join(', ');
+};
diff --git a/src/core/getContentEditManager.js b/src/core/getContentEditManager.js
new file mode 100644
index 0000000000..9a7e2438a0
--- /dev/null
+++ b/src/core/getContentEditManager.js
@@ -0,0 +1,3 @@
+import core from 'core';
+
+export default (pageNumber, documentViewerKey = 1) => core.getDocumentViewer(documentViewerKey).getContentEditManager();
diff --git a/src/core/getOfficeEditor.js b/src/core/getOfficeEditor.js
new file mode 100644
index 0000000000..1d53fcdc17
--- /dev/null
+++ b/src/core/getOfficeEditor.js
@@ -0,0 +1,3 @@
+import core from 'core';
+
+export default (documentViewerKey = 1) => core.getDocumentViewer(documentViewerKey).getDocument().getOfficeEditor();
diff --git a/src/core/getResultCode.js b/src/core/getResultCode.js
new file mode 100644
index 0000000000..8a6dd62c4a
--- /dev/null
+++ b/src/core/getResultCode.js
@@ -0,0 +1 @@
+export default () => window.Core.Search.ResultCode;
diff --git a/src/core/getScalePrecision.js b/src/core/getScalePrecision.js
new file mode 100644
index 0000000000..713cf553f7
--- /dev/null
+++ b/src/core/getScalePrecision.js
@@ -0,0 +1,8 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.MeasurementManager.html#getScalePrecision__anchor
+ */
+export default (scale, documentViewerKey = 1) => {
+ return core.getDocumentViewer(documentViewerKey).getMeasurementManager().getScalePrecision(scale);
+};
diff --git a/src/core/getScales.js b/src/core/getScales.js
new file mode 100644
index 0000000000..61fe650045
--- /dev/null
+++ b/src/core/getScales.js
@@ -0,0 +1,8 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.MeasurementManager.html#getScales__anchor
+ */
+export default (documentViewerKey = 1) => {
+ return core.getDocumentViewer(documentViewerKey).getMeasurementManager().getScales();
+};
diff --git a/src/core/getSemanticDiffAnnotations.js b/src/core/getSemanticDiffAnnotations.js
new file mode 100644
index 0000000000..6a3e498b78
--- /dev/null
+++ b/src/core/getSemanticDiffAnnotations.js
@@ -0,0 +1,6 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.AnnotationManager.html#getSemanticDiffAnnotations__anchor
+ */
+export default (documentViewerKey = 1) => core.getDocumentViewer(documentViewerKey).getAnnotationManager().getSemanticDiffAnnotations();
\ No newline at end of file
diff --git a/src/core/getToolsFromAllDocumentViewers.js b/src/core/getToolsFromAllDocumentViewers.js
new file mode 100644
index 0000000000..8087d59535
--- /dev/null
+++ b/src/core/getToolsFromAllDocumentViewers.js
@@ -0,0 +1,6 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.DocumentViewer.html#getTool__anchor
+ */
+export default (toolName) => core.getDocumentViewers().map((documentViewer) => documentViewer.getTool(toolName));
diff --git a/src/core/isSearchResultEqual.js b/src/core/isSearchResultEqual.js
new file mode 100644
index 0000000000..e3b5c1e2d5
--- /dev/null
+++ b/src/core/isSearchResultEqual.js
@@ -0,0 +1 @@
+export default (resultA, resultB) => window.Core.Search.isSearchResultEqual(resultA, resultB);
diff --git a/src/core/loadBlankOfficeEditorDocument.js b/src/core/loadBlankOfficeEditorDocument.js
new file mode 100644
index 0000000000..2ad4f2ad33
--- /dev/null
+++ b/src/core/loadBlankOfficeEditorDocument.js
@@ -0,0 +1,6 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.DocumentViewer.html#loadBlankOfficeEditorDocument
+ */
+export default (options, documentViewerKey = 1) => core.getDocumentViewer(documentViewerKey).loadBlankOfficeEditorDocument(options);
diff --git a/src/core/replaceScales.js b/src/core/replaceScales.js
new file mode 100644
index 0000000000..83422b43bd
--- /dev/null
+++ b/src/core/replaceScales.js
@@ -0,0 +1,8 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.MeasurementManager.html#replaceScales__anchor
+ */
+export default (originalScales, scale, documentViewerKey = 1) => {
+ core.getDocumentViewer(documentViewerKey).getMeasurementManager().replaceScales(originalScales, scale);
+};
diff --git a/src/core/setBookmarkIconShortcutVisibility.js b/src/core/setBookmarkIconShortcutVisibility.js
new file mode 100644
index 0000000000..6aa7f2c17c
--- /dev/null
+++ b/src/core/setBookmarkIconShortcutVisibility.js
@@ -0,0 +1,9 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.DocumentViewer.html#setBookmarkIconShortcutVisibility__anchor
+ * @see https://docs.apryse.com/api/web/Core.DocumentViewer.html#setBookmarkIconShortcutVisibility__anchor
+ */
+export default (isEnabled, documentViewerKey = 1) => {
+ core.getDocumentViewer(documentViewerKey).setBookmarkIconShortcutVisibility(isEnabled);
+};
diff --git a/src/core/setBookmarkShortcutToggleOffFunction.js b/src/core/setBookmarkShortcutToggleOffFunction.js
new file mode 100644
index 0000000000..257d402db9
--- /dev/null
+++ b/src/core/setBookmarkShortcutToggleOffFunction.js
@@ -0,0 +1,9 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.DocumentViewer.html#setBookmarkShortcutToggleOffFunction__anchor
+ * @see https://docs.apryse.com/api/web/Core.DocumentViewer.html#setBookmarkShortcutToggleOffFunction__anchor
+ */
+export default (callback, documentViewerKey = 1) => {
+ core.getDocumentViewer(documentViewerKey).setBookmarkShortcutToggleOffFunction(callback);
+};
diff --git a/src/core/setBookmarkShortcutToggleOnFunction.js b/src/core/setBookmarkShortcutToggleOnFunction.js
new file mode 100644
index 0000000000..173af37655
--- /dev/null
+++ b/src/core/setBookmarkShortcutToggleOnFunction.js
@@ -0,0 +1,9 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.DocumentViewer.html#setBookmarkShortcutToggleOnFunction__anchor
+ * @see https://docs.apryse.com/api/web/Core.DocumentViewer.html#setBookmarkShortcutToggleOnFunction__anchor
+ */
+export default (callback, documentViewerKey = 1) => {
+ core.getDocumentViewer(documentViewerKey).setBookmarkShortcutToggleOnFunction(callback);
+};
diff --git a/src/core/setUserBookmarks.js b/src/core/setUserBookmarks.js
new file mode 100644
index 0000000000..24dd049e21
--- /dev/null
+++ b/src/core/setUserBookmarks.js
@@ -0,0 +1,9 @@
+import core from 'core';
+
+/**
+ * https://docs.apryse.com/api/web/Core.DocumentViewer.html#setUserBookmarks__anchor
+ * @see https://docs.apryse.com/api/web/Core.DocumentViewer.html#setUserBookmarks__anchor
+ */
+export default (bookmarks, documentViewerKey = 1) => {
+ core.getDocumentViewer(documentViewerKey).setUserBookmarks(bookmarks);
+};
diff --git a/src/event-listeners/onCaretAnnotationAdded.js b/src/event-listeners/onCaretAnnotationAdded.js
new file mode 100644
index 0000000000..836f7204d7
--- /dev/null
+++ b/src/event-listeners/onCaretAnnotationAdded.js
@@ -0,0 +1,29 @@
+import core from 'core';
+import actions from 'actions';
+import selectors from 'selectors';
+import DataElements from 'src/constants/dataElement';
+
+export default ({ dispatch, getState }) => (annotation) => {
+ const state = getState();
+ const isNotesPanelDisabled = selectors.isElementDisabled(state, DataElements.NOTES_PANEL);
+ const isNotesPanelOpen = selectors.isElementOpen(state, DataElements.NOTES_PANEL);
+ const isInlineCommentDisabled = selectors.isElementDisabled(state, DataElements.INLINE_COMMENT_POPUP);
+
+ if (isNotesPanelDisabled) {
+ return;
+ }
+
+ dispatch(actions.closeElement('searchPanel'));
+ dispatch(actions.closeElement(DataElements.REDACTION_PANEL));
+ if (!isInlineCommentDisabled || isNotesPanelOpen) {
+ core.selectAnnotation(annotation);
+ dispatch(actions.triggerNoteEditing());
+ } else {
+ dispatch(actions.openElement(DataElements.NOTES_PANEL));
+ // wait for the notes panel to be fully opened before focusing
+ setTimeout(() => {
+ core.selectAnnotation(annotation);
+ dispatch(actions.triggerNoteEditing());
+ }, 400);
+ }
+};
diff --git a/src/event-listeners/onContentEditModeEnded.js b/src/event-listeners/onContentEditModeEnded.js
index 99814b480c..213007167a 100644
--- a/src/event-listeners/onContentEditModeEnded.js
+++ b/src/event-listeners/onContentEditModeEnded.js
@@ -1,5 +1,13 @@
import actions from 'actions';
+import selectors from 'selectors';
+import core from 'core';
-export default (dispatch) => () => {
+export default (dispatch, store) => () => {
dispatch(actions.enableElements(['thumbnailControl', 'documentControl']));
+ const featureFlags = selectors.getFeatureFlags(store.getState());
+ const { customizableUI } = featureFlags;
+
+ if (customizableUI) {
+ core.setToolMode('AnnotationEdit');
+ }
};
diff --git a/src/event-listeners/onFormFieldCreationModeStarted.js b/src/event-listeners/onFormFieldCreationModeStarted.js
index 929d3dc8e3..19545b7f5a 100644
--- a/src/event-listeners/onFormFieldCreationModeStarted.js
+++ b/src/event-listeners/onFormFieldCreationModeStarted.js
@@ -15,7 +15,6 @@ const formBuilderDefaultDisabledKeys = {
PRINT: 'print',
BOOKMARK: 'bookmark',
SWITCH_PAN: 'switchPan',
- SELECT: 'select',
PAN: 'pan',
ARROW: 'arrow',
CALLOUT: 'callout',
diff --git a/src/event-listeners/onImageContentAdded.js b/src/event-listeners/onImageContentAdded.js
new file mode 100644
index 0000000000..02cc392960
--- /dev/null
+++ b/src/event-listeners/onImageContentAdded.js
@@ -0,0 +1,9 @@
+import core from 'core';
+import actions from 'actions';
+import defaultTool from 'constants/defaultTool';
+
+export default (dispatch) => (annotation) => {
+ core.setToolMode(defaultTool);
+ dispatch(actions.setActiveToolGroup(''));
+ core.selectAnnotation(annotation);
+};
\ No newline at end of file
diff --git a/src/event-listeners/onInitialDeleted.js b/src/event-listeners/onInitialDeleted.js
new file mode 100644
index 0000000000..b6fe9a6c44
--- /dev/null
+++ b/src/event-listeners/onInitialDeleted.js
@@ -0,0 +1,10 @@
+import core from 'core';
+import actions from 'actions';
+import getSignatureDataToStore from 'helpers/getSignatureDataToStore';
+
+export default (dispatch) => async () => {
+ const signatureTool = core.getTool('AnnotationCreateSignature');
+ const savedInitials = signatureTool.getSavedInitials();
+ const newSavedInitials = await getSignatureDataToStore(savedInitials);
+ dispatch(actions.setSavedInitials(newSavedInitials));
+};
diff --git a/src/event-listeners/onInitialSaved.js b/src/event-listeners/onInitialSaved.js
new file mode 100644
index 0000000000..a1296c41a3
--- /dev/null
+++ b/src/event-listeners/onInitialSaved.js
@@ -0,0 +1,22 @@
+import core from 'core';
+import selectors from 'selectors';
+import actions from 'actions';
+import getSignatureDataToStore from 'helpers/getSignatureDataToStore';
+
+export default (dispatch, store) => async () => {
+ const signatureTool = core.getTool('AnnotationCreateSignature');
+ let savedInitials = signatureTool.getSavedInitials();
+ const maxSignaturesCount = selectors.getMaxSignaturesCount(store.getState());
+ const numberOfInitialsToRemove = savedInitials.length - maxSignaturesCount;
+
+ if (numberOfInitialsToRemove > 0) {
+ // to keep the UI sync with the signatures saved in the tool
+ for (let i = 0; i < numberOfInitialsToRemove; i++) {
+ signatureTool.deleteSavedInitials(0);
+ }
+ }
+
+ savedInitials = signatureTool.getSavedInitials();
+ const initialsToStore = await getSignatureDataToStore(savedInitials);
+ dispatch(actions.setSavedInitials(initialsToStore));
+};
\ No newline at end of file
diff --git a/src/event-listeners/onLayersUpdated.js b/src/event-listeners/onLayersUpdated.js
index da98b2f6d2..edc3c4d549 100644
--- a/src/event-listeners/onLayersUpdated.js
+++ b/src/event-listeners/onLayersUpdated.js
@@ -1,10 +1,10 @@
import actions from 'actions';
-import equal from 'fast-deep-equal';
+import _isEqual from 'lodash/isEqual';
import setUIPropertiesForLayers from 'helpers/setUIPropertiesForLayers';
export default (newOCGLayers, currentOCGLayers, dispatch) => {
- const isEqual = equal(newOCGLayers, currentOCGLayers);
- if (!isEqual) {
+ const layersEqual = _isEqual(newOCGLayers, currentOCGLayers);
+ if (!layersEqual) {
const layersToSet = setUIPropertiesForLayers(newOCGLayers);
dispatch(actions.setLayers(layersToSet));
}
diff --git a/src/event-listeners/onSignatureSaved.js b/src/event-listeners/onSignatureSaved.js
index cbe46d890f..20275a6026 100644
--- a/src/event-listeners/onSignatureSaved.js
+++ b/src/event-listeners/onSignatureSaved.js
@@ -18,5 +18,7 @@ export default (dispatch, store) => async () => {
savedSignatures = signatureTool.getSavedSignatures();
const signaturesToStore = await getSignatureDataToStore(savedSignatures);
+ // get the last element of the array (LIFO) and set it as active so it can be shown in the new signature list panel
+ dispatch(actions.setSelectedDisplayedSignatureIndex(signaturesToStore.length - 1));
dispatch(actions.setSavedSignatures(signaturesToStore));
};
\ No newline at end of file
diff --git a/src/helpers/checkFeaturesToEnable.js b/src/helpers/checkFeaturesToEnable.js
new file mode 100644
index 0000000000..277981c571
--- /dev/null
+++ b/src/helpers/checkFeaturesToEnable.js
@@ -0,0 +1,38 @@
+import { mapKeyToToolNames, annotationMapKeys } from 'constants/map';
+import Feature from 'constants/feature';
+import { getInstanceNode } from './getRootNode';
+import DataElements from 'constants/dataElement';
+
+const checkFeaturesToEnable = (componentsMap) => {
+ const keys = Object.keys(componentsMap);
+ const measurementTools = [
+ mapKeyToToolNames(annotationMapKeys.DISTANCE_MEASUREMENT),
+ mapKeyToToolNames(annotationMapKeys.PERIMETER_MEASUREMENT),
+ mapKeyToToolNames(annotationMapKeys.ARC_MEASUREMENT),
+ mapKeyToToolNames(annotationMapKeys.RECTANGULAR_AREA_MEASUREMENT),
+ mapKeyToToolNames(annotationMapKeys.CLOUDY_RECTANGULAR_AREA_MEASUREMENT),
+ mapKeyToToolNames(annotationMapKeys.AREA_MEASUREMENT),
+ mapKeyToToolNames(annotationMapKeys.ELLIPSE_MEASUREMENT),
+ mapKeyToToolNames(annotationMapKeys.COUNT_MEASUREMENT),
+ ].flat();
+ const contentEditTools = mapKeyToToolNames(annotationMapKeys.CONTENT_EDIT_TOOL);
+ const redactTools = mapKeyToToolNames(annotationMapKeys.REDACTION);
+ const instance = getInstanceNode().instance;
+
+ for (let index = 0, len = keys.length; index < len; index++) {
+ const element = componentsMap[keys[index]];
+ if (element.dataElement === DataElements.FILE_PICKER_BUTTON) {
+ instance.UI.enableFilePicker();
+ } else if (element.dataElement === DataElements.CREATE_PORTFOLIO) {
+ instance.UI.enableFeatures(Feature.Portfolio);
+ } else if (redactTools.indexOf(element.toolName) > -1) {
+ instance.UI.enableRedaction();
+ } else if (measurementTools.indexOf(element.toolName) > -1) {
+ instance.UI.enableMeasurement();
+ } else if (contentEditTools.indexOf(element.toolName) > -1) {
+ instance.UI.enableFeatures(Feature.ContentEdit);
+ }
+ }
+};
+
+export default checkFeaturesToEnable;
\ No newline at end of file
diff --git a/src/helpers/checkFeaturesToEnable.spec.js b/src/helpers/checkFeaturesToEnable.spec.js
new file mode 100644
index 0000000000..31e58825e7
--- /dev/null
+++ b/src/helpers/checkFeaturesToEnable.spec.js
@@ -0,0 +1,48 @@
+import checkFeaturesToEnable from './checkFeaturesToEnable';
+import { getInstanceNode } from './getRootNode';
+
+jest.mock('./getRootNode', () => {
+ const original = jest.requireActual('./getRootNode'); // Step 2.
+
+ return {
+ ...original,
+ getInstanceNode: jest.fn(() => {})
+ };
+});
+
+describe('checkFeaturesToEnable', () => {
+ it('Should call checkFeaturesToEnable functions', () => {
+ const mockMethod = jest.fn();
+ const mockMethod2 = jest.fn();
+ const mockMethod3 = jest.fn();
+ const mockMethod4 = jest.fn();
+ getInstanceNode.mockImplementation(() => {
+ return {
+ instance: {
+ UI: {
+ enableFeatures: mockMethod,
+ enableFilePicker: mockMethod2,
+ enableRedaction: mockMethod3,
+ enableMeasurement: mockMethod4,
+ }
+ }
+ };
+ });
+ const instance = getInstanceNode().instance;
+ checkFeaturesToEnable({
+ arcMeasurementToolButton: {
+ toolName: 'AnnotationCreateArcMeasurement',
+ dataElement: 'arcMeasurementToolButton',
+ },
+ contentEditButton: {
+ toolName: 'ContentEditTool',
+ dataElement: 'contentEditButton',
+ }
+ });
+
+ expect(instance.UI.enableFeatures).toHaveBeenCalledTimes(1);
+ expect(instance.UI.enableFilePicker).not.toBeCalled();
+ expect(instance.UI.enableMeasurement).toHaveBeenCalledTimes(1);
+ expect(instance.UI.enableRedaction).not.toBeCalled();
+ });
+});
\ No newline at end of file
diff --git a/src/helpers/clickTracker.js b/src/helpers/clickTracker.js
new file mode 100644
index 0000000000..ef215d72db
--- /dev/null
+++ b/src/helpers/clickTracker.js
@@ -0,0 +1,11 @@
+export const ClickedItemTypes = {
+ BUTTON: 'button',
+};
+
+let clickMiddleWare;
+
+export const setClickMiddleWare = (middleware) => {
+ clickMiddleWare = middleware;
+};
+
+export const getClickMiddleWare = () => clickMiddleWare;
diff --git a/src/helpers/getAngleInRadians.js b/src/helpers/getAngleInRadians.js
new file mode 100644
index 0000000000..1a20add5b5
--- /dev/null
+++ b/src/helpers/getAngleInRadians.js
@@ -0,0 +1,21 @@
+export default (pt1, pt2, pt3) => {
+ let angle;
+
+ if (pt1 && pt2) {
+ if (pt3) {
+ // calculate the angle using Law of cosines
+ const AB = Math.sqrt(Math.pow(pt2.x - pt1.x, 2) + Math.pow(pt2.y - pt1.y, 2));
+ const BC = Math.sqrt(Math.pow(pt2.x - pt3.x, 2) + Math.pow(pt2.y - pt3.y, 2));
+ const AC = Math.sqrt(Math.pow(pt3.x - pt1.x, 2) + Math.pow(pt3.y - pt1.y, 2));
+ angle = Math.acos((BC * BC + AB * AB - AC * AC) / (2 * BC * AB));
+ } else {
+ // if there are only two points returns the angle in the plane (in radians) between the positive x-axis and the ray from (0,0) to the point (x,y)
+ angle = Math.atan2(pt2.y - pt1.y, pt2.x - pt1.x);
+ // keep the angle range between 0 and Math.PI / 2
+ angle = Math.abs(angle);
+ angle = angle > Math.PI / 2 ? Math.PI - angle : angle;
+ }
+ }
+
+ return angle;
+};
diff --git a/src/helpers/getDeviceSize.js b/src/helpers/getDeviceSize.js
new file mode 100644
index 0000000000..9e1fbc502b
--- /dev/null
+++ b/src/helpers/getDeviceSize.js
@@ -0,0 +1,55 @@
+import getRootNode from './getRootNode';
+import useMedia from 'hooks/useMedia';
+import { MOBILE_SIZE, TABLET_SIZE } from 'constants/deviceSizes';
+
+export const isMobileSize = () => {
+ if (window.isApryseWebViewerWebComponent) {
+ return getRootNode()?.host.clientWidth <= MOBILE_SIZE;
+ }
+ return useMedia(
+ // Media queries
+ [`(max-width: ${MOBILE_SIZE}px)`],
+ [true],
+ // Default value
+ false,
+ );
+};
+
+export const isTabletSize = () => {
+ if (window.isApryseWebViewerWebComponent) {
+ return getRootNode()?.host.clientWidth > MOBILE_SIZE && getRootNode()?.host.clientWidth <= TABLET_SIZE;
+ }
+ return useMedia(
+ // Media queries
+ [`(min-width: ${MOBILE_SIZE + 1}px) and (max-width: ${TABLET_SIZE}px)`],
+ [true],
+ // Default value
+ false,
+ );
+};
+
+export const isTabletAndMobileSize = () => {
+ if (window.isApryseWebViewerWebComponent) {
+ return getRootNode()?.host.clientWidth <= TABLET_SIZE;
+ }
+ return useMedia(
+ // Media queries
+ [`(max-width: ${TABLET_SIZE}px)`],
+ [true],
+ // Default value
+ false,
+ );
+};
+
+export const isDesktopSize = () => {
+ if (window.isApryseWebViewerWebComponent) {
+ return getRootNode()?.host.clientWidth > TABLET_SIZE;
+ }
+ return useMedia(
+ // Media queries
+ [`(min-width: ${TABLET_SIZE + 1}px)`],
+ [true],
+ // Default value
+ false,
+ );
+};
\ No newline at end of file
diff --git a/src/helpers/getElements.js b/src/helpers/getElements.js
index c42cc6ea55..5d9f558b6d 100644
--- a/src/helpers/getElements.js
+++ b/src/helpers/getElements.js
@@ -12,8 +12,4 @@ export function getAllOpenedModals() {
export function getDatePicker() {
return document.querySelector('[data-element="datePickerContainer"]');
-}
-
-export function getAllPanels(location) {
- return document.querySelectorAll(`.flx-Panel.${location}`);
-}
+}
\ No newline at end of file
diff --git a/src/helpers/getNumberOfDecimalPlaces.js b/src/helpers/getNumberOfDecimalPlaces.js
new file mode 100644
index 0000000000..7469c00e83
--- /dev/null
+++ b/src/helpers/getNumberOfDecimalPlaces.js
@@ -0,0 +1 @@
+export default (precision) => (precision === 1 ? 0 : precision?.toString().split('.')[1].length);
diff --git a/src/helpers/getRootNode.js b/src/helpers/getRootNode.js
index ae37eb3f8e..7b900be470 100644
--- a/src/helpers/getRootNode.js
+++ b/src/helpers/getRootNode.js
@@ -1,5 +1,21 @@
let rootNode;
+function findNestedWebComponents(tagName, root = document) {
+ const elements = [];
+
+ // Check direct children
+ root.querySelectorAll(tagName).forEach((el) => elements.push(el));
+
+ // Check shadow DOMs
+ root.querySelectorAll('*').forEach((el) => {
+ if (el.shadowRoot) {
+ elements.push(...findNestedWebComponents(tagName, el.shadowRoot));
+ }
+ });
+
+ return elements;
+}
+
const getRootNode = () => {
if (!window.isApryseWebViewerWebComponent) {
return document;
@@ -7,7 +23,13 @@ const getRootNode = () => {
if (rootNode) {
return rootNode;
}
- const elementList = document.getElementsByTagName('apryse-webviewer');
+
+ let elementList;
+ elementList = document.getElementsByTagName('apryse-webviewer');
+ if (elementList.length === 0) {
+ elementList = findNestedWebComponents('apryse-webviewer');
+ }
+
if (elementList?.length) {
for (const element of elementList) {
const foundNode = element.shadowRoot;
diff --git a/src/helpers/handleFreeTextAutoSizeToggle.js b/src/helpers/handleFreeTextAutoSizeToggle.js
new file mode 100644
index 0000000000..9fd6420a32
--- /dev/null
+++ b/src/helpers/handleFreeTextAutoSizeToggle.js
@@ -0,0 +1,22 @@
+import core from 'core';
+
+/**
+ * @ignore
+ * handler for auto size font toggle
+ * @param {FreeTextAnnotation} annotation annotation to toggle auto size font
+ * @param {function} setAutoSizeFont function to set auto size font
+ * @param {boolean} isAutoSizeFont current auto size font value
+ */
+export default (annotation, setAutoSizeFont, isAutoSizeFont) => {
+ const freeTextAnnot = annotation;
+ const calculatedFontSize = freeTextAnnot.getCalculatedFontSize();
+ if (isAutoSizeFont) {
+ freeTextAnnot.FontSize = calculatedFontSize;
+ } else {
+ freeTextAnnot.switchToAutoFontSize();
+ }
+
+ setAutoSizeFont(!isAutoSizeFont);
+ core.getAnnotationManager().redrawAnnotation(freeTextAnnot);
+};
+
diff --git a/src/helpers/initialColorStates.js b/src/helpers/initialColorStates.js
new file mode 100644
index 0000000000..fb754c8a4c
--- /dev/null
+++ b/src/helpers/initialColorStates.js
@@ -0,0 +1,50 @@
+const initialColors = [
+ '#e44234',
+ '#ff8d00',
+ '#ffcd45',
+ '#5cc96e',
+ '#25d2d1',
+ '#597ce2',
+ '#c544ce',
+ '#7d2e25',
+ '#a84f1d',
+ '#e99e38',
+ '#347842',
+ '#167e7d',
+ '#354a87',
+ '#76287b',
+ '#ffffff',
+ '#cdcdcd',
+ '#9c9c9c',
+ '#696969',
+ '#272727',
+ '#000000'
+];
+
+const initialTextColors = [
+ '#000000',
+ '#272727',
+ '#696969',
+ '#9c9c9c',
+ '#cdcdcd',
+ '#ffffff',
+ '#7d2e25',
+ '#a84f1d',
+ '#e99e38',
+ '#347842',
+ '#167e7d',
+ '#354a87',
+ '#76287b',
+ '#e44234',
+ '#ff8d00',
+ '#ffcd45',
+ '#5cc96e',
+ '#25d2d1',
+ '#597ce2',
+ '#c544ce'
+];
+
+export {
+ initialColors,
+ initialTextColors,
+};
\ No newline at end of file
diff --git a/src/helpers/multiViewerHelper.js b/src/helpers/multiViewerHelper.js
new file mode 100644
index 0000000000..34503633be
--- /dev/null
+++ b/src/helpers/multiViewerHelper.js
@@ -0,0 +1,14 @@
+const multiViewerHelper = {
+ matchedPages: null,
+ isScrolledByClickingChangeItem: false,
+};
+
+export const setIsScrolledByClickingChangeItem = (value) => {
+ multiViewerHelper.isScrolledByClickingChangeItem = value;
+};
+
+export const getIsScrolledByClickingChangeItem = () => {
+ return multiViewerHelper.isScrolledByClickingChangeItem;
+};
+
+export default multiViewerHelper;
diff --git a/src/helpers/officeEditor.js b/src/helpers/officeEditor.js
new file mode 100644
index 0000000000..b94a760de3
--- /dev/null
+++ b/src/helpers/officeEditor.js
@@ -0,0 +1,6 @@
+import core from 'core';
+import { workerTypes } from 'constants/types';
+
+export function isOfficeEditorMode() {
+ return core.getDocument()?.getType() === workerTypes.OFFICE_EDITOR;
+}
diff --git a/src/helpers/openOfficeEditorFilePicker.js b/src/helpers/openOfficeEditorFilePicker.js
new file mode 100644
index 0000000000..9d3e3e8c1b
--- /dev/null
+++ b/src/helpers/openOfficeEditorFilePicker.js
@@ -0,0 +1,5 @@
+import getRootNode from 'helpers/getRootNode';
+
+export default () => {
+ getRootNode().querySelector('#office-editor-file-picker')?.click();
+};
diff --git a/src/helpers/sanitizeSVG.js b/src/helpers/sanitizeSVG.js
new file mode 100644
index 0000000000..06208e4d7a
--- /dev/null
+++ b/src/helpers/sanitizeSVG.js
@@ -0,0 +1,43 @@
+import DOMPurify from 'dompurify';
+
+const SVG_MIME_TYPE = 'image/svg+xml';
+
+const hasFileSize = (file) => {
+ return file.size !== undefined;
+};
+
+// Taken from https://github.com/mattkrick/sanitize-svg/blob/master/src/sanitizeSVG.ts#L31
+const readAsText = (svg) => {
+ return new Promise((resolve) => {
+ if (!hasFileSize(svg)) {
+ resolve(svg.toString('utf-8'));
+ } else {
+ const fileReader = new FileReader();
+ fileReader.onload = () => resolve(fileReader.result);
+ fileReader.readAsText(svg);
+ }
+ });
+};
+
+export const isSVG = (file) => {
+ return file.type === SVG_MIME_TYPE;
+};
+
+export const sanitizeSVG = async (file) => {
+ const svgText = await readAsText(file);
+ if (!svgText) {
+ return { svg: file };
+ }
+
+ const forbiddenTags = [];
+ DOMPurify.addHook('uponSanitizeElement', (_, hookEvent) => {
+ const { tagName, allowedTags } = hookEvent;
+ if (!allowedTags[tagName]) {
+ forbiddenTags.push(tagName);
+ }
+ });
+
+ const clean = DOMPurify.sanitize(svgText);
+ const svg = new Blob([clean], { type: SVG_MIME_TYPE });
+ return { svg, isDirty: forbiddenTags.length > 0 };
+};
\ No newline at end of file
diff --git a/src/helpers/setEnableAnnotationNumbering.js b/src/helpers/setEnableAnnotationNumbering.js
new file mode 100644
index 0000000000..1233663334
--- /dev/null
+++ b/src/helpers/setEnableAnnotationNumbering.js
@@ -0,0 +1,7 @@
+import core from 'core';
+
+export default (state) => {
+ if (state.viewer.isAnnotationNumberingEnabled) {
+ core.enableAnnotationNumbering();
+ }
+};
\ No newline at end of file
diff --git a/src/helpers/useWindowsDimensions.js b/src/helpers/useWindowsDimensions.js
new file mode 100644
index 0000000000..89f0f7bd84
--- /dev/null
+++ b/src/helpers/useWindowsDimensions.js
@@ -0,0 +1,24 @@
+import { useState, useEffect } from 'react';
+
+function getWindowDimensions() {
+ const { innerWidth: width, innerHeight: height } = window;
+ return {
+ width,
+ height
+ };
+}
+
+export default function useWindowDimensions() {
+ const [windowDimensions, setWindowDimensions] = useState(getWindowDimensions());
+
+ useEffect(() => {
+ function handleResize() {
+ setWindowDimensions(getWindowDimensions());
+ }
+
+ window.addEventListener('resize', handleResize);
+ return () => window.removeEventListener('resize', handleResize);
+ }, []);
+
+ return windowDimensions;
+}
diff --git a/src/hooks/useFloatingHeaderSelectors/index.js b/src/hooks/useFloatingHeaderSelectors/index.js
new file mode 100644
index 0000000000..d197bf916d
--- /dev/null
+++ b/src/hooks/useFloatingHeaderSelectors/index.js
@@ -0,0 +1,3 @@
+import useFloatingHeaderSelectors from './useFloatingHeaderSelectors';
+
+export default useFloatingHeaderSelectors;
\ No newline at end of file
diff --git a/src/hooks/useFloatingHeaderSelectors/useFloatingHeaderSelectors.spec.js b/src/hooks/useFloatingHeaderSelectors/useFloatingHeaderSelectors.spec.js
new file mode 100644
index 0000000000..43f557219c
--- /dev/null
+++ b/src/hooks/useFloatingHeaderSelectors/useFloatingHeaderSelectors.spec.js
@@ -0,0 +1,102 @@
+import { renderHook } from '@testing-library/react-hooks';
+import { useSelector, Provider } from 'react-redux';
+import useFloatingHeaderSelectors from './useFloatingHeaderSelectors';
+import rootReducer from 'reducers/rootReducer';
+import initialState from 'src/redux/initialState';
+import { configureStore } from '@reduxjs/toolkit';
+import React from 'react';
+import { RESIZE_BAR_WIDTH } from 'src/constants/panel';
+// Mocking useSelector
+jest.mock('react-redux', () => ({
+ ...jest.requireActual('react-redux'),
+ useSelector: jest.fn(),
+}));
+
+const floatEndBottomHeader = {
+ dataElement: 'floatEndBottomHeader',
+ placement: 'bottom',
+ float: true,
+ position: 'end',
+ items: [],
+};
+
+const floatStartBottomHeader = {
+ dataElement: 'floatStartBottomHeader',
+ placement: 'bottom',
+ float: true,
+ position: 'start',
+ items: [],
+};
+
+const floatStartTopHeader = {
+ dataElement: 'topStartFloatingHeader',
+ placement: 'top',
+ float: true,
+ position: 'start',
+ items: [],
+};
+
+const floatEndTopHeader = {
+ dataElement: 'topEndFloatingHeader',
+ placement: 'top',
+ float: true,
+ position: 'end',
+ items: [],
+};
+
+describe('useFloatingHeaderSelectors hook', () => {
+ it('should return the correct values from the state', () => {
+ const mockState = {
+ ...initialState,
+ viewer: {
+ ...initialState.viewer,
+ openElements: {
+ ...initialState.viewer.openElements,
+ leftPanel: false,
+ notesPanel: false,
+ redactionPanel: true,
+ },
+ modularHeadersWidth: {
+ rightHeader: 23,
+ leftHeader: 48,
+ },
+ floatingContainersDimensions: {
+ topFloatingContainerHeight: 36,
+ bottomFloatingContainerHeight: 28,
+ },
+ modularHeaders: [
+ floatEndBottomHeader,
+ floatStartTopHeader,
+ floatEndTopHeader,
+ floatStartBottomHeader,
+ ],
+ }
+ };
+
+ const store = configureStore({
+ reducer: rootReducer,
+ preloadedState: mockState,
+ });
+
+ useSelector.mockImplementation((callback) => callback(mockState));
+
+ const { result } = renderHook(() => useFloatingHeaderSelectors(), {
+ wrapper: ({ children }) => {children},
+ });
+
+ expect(result.current.isLeftPanelOpen).toBeFalsy();
+ expect(result.current.isRightPanelOpen).toBe(true);
+ expect(result.current.leftPanelWidth).toBe(mockState.viewer.panelWidths.leftPanel + RESIZE_BAR_WIDTH);
+ expect(result.current.rightPanelWidth).toBe(mockState.viewer.panelWidths.redactionPanel);
+ expect(result.current.leftHeaderWidth).toBe(0);
+ expect(result.current.rightHeaderWidth).toBe(0);
+ expect(result.current.topHeadersHeight).toBe(0);
+ expect(result.current.bottomHeadersHeight).toBe(0);
+ expect(result.current.topFloatingContainerHeight).toBe(mockState.viewer.floatingContainersDimensions.topFloatingContainerHeight);
+ expect(result.current.bottomFloatingContainerHeight).toBe(mockState.viewer.floatingContainersDimensions.bottomFloatingContainerHeight);
+ expect(result.current.topStartFloatingHeaders).toEqual([floatStartTopHeader]);
+ expect(result.current.bottomStartFloatingHeaders).toEqual([floatStartBottomHeader]);
+ expect(result.current.bottomEndFloatingHeaders).toEqual([floatEndBottomHeader]);
+ expect(result.current.topEndFloatingHeaders).toEqual([floatEndTopHeader]);
+ });
+});
diff --git a/src/hooks/useOnAnnotationContentOverlayOpen/index.js b/src/hooks/useOnAnnotationContentOverlayOpen/index.js
new file mode 100644
index 0000000000..8d414fb75f
--- /dev/null
+++ b/src/hooks/useOnAnnotationContentOverlayOpen/index.js
@@ -0,0 +1,3 @@
+import useOnAnnotationContentOverlayOpen from './useOnAnnotationContentOverlayOpen';
+
+export default useOnAnnotationContentOverlayOpen;
\ No newline at end of file
diff --git a/src/hooks/useOnAnnotationContentOverlayOpen/useOnAnnotationContentOverlayOpen.js b/src/hooks/useOnAnnotationContentOverlayOpen/useOnAnnotationContentOverlayOpen.js
new file mode 100644
index 0000000000..4accc4344b
--- /dev/null
+++ b/src/hooks/useOnAnnotationContentOverlayOpen/useOnAnnotationContentOverlayOpen.js
@@ -0,0 +1,49 @@
+import { useEffect, useState } from 'react';
+import { useDispatch, useSelector } from 'react-redux';
+import actions from 'actions';
+import selectors from 'selectors';
+import core from 'core';
+import DataElements from 'constants/dataElement';
+
+export default function useOnAnnotationContentOverlayOpen() {
+ // Clients have the option to customize how the tooltip is rendered by passing a handler
+ const customHandler = useSelector((state) => selectors.getAnnotationContentOverlayHandler(state));
+
+ const dispatch = useDispatch();
+ const [annotation, setAnnotation] = useState(null);
+ const [clientXY, setClientXY] = useState({ clientX: 0, clientY: 0 });
+ const isUsingCustomHandler = customHandler !== null;
+
+ useEffect(() => {
+ const viewElement = core.getViewerElement();
+
+ const onMouseHover = (e) => {
+ if (e.buttons !== 0) {
+ return;
+ }
+ let annotation = core.getAnnotationManager().getAnnotationByMouseEvent(e);
+
+ if (annotation && viewElement.contains(e.target)) {
+ // if hovered annot is grouped, pick the "primary" annot to match Adobe's behavior
+ const groupedAnnots = core.getAnnotationManager().getGroupAnnotations(annotation);
+ const ungroupedAnnots = groupedAnnots.filter((annot) => !annot.isGrouped());
+ annotation = ungroupedAnnots.length > 0 ? ungroupedAnnots[0] : annotation;
+
+ const isFreeTextAnnotation = annotation instanceof window.Core.Annotations.FreeTextAnnotation;
+ if (isUsingCustomHandler || !isFreeTextAnnotation) {
+ setClientXY({ clientX: e.clientX, clientY: e.clientY });
+ setAnnotation(annotation);
+ dispatch(actions.openElement(DataElements.ANNOTATION_CONTENT_OVERLAY));
+ }
+ } else {
+ setAnnotation(null);
+ dispatch(actions.closeElement(DataElements.ANNOTATION_CONTENT_OVERLAY));
+ }
+ };
+
+ core.addEventListener('mouseMove', onMouseHover);
+ return () => core.removeEventListener('mouseMove', onMouseHover);
+ }, [annotation, isUsingCustomHandler]);
+
+ return { annotation, clientXY };
+}
\ No newline at end of file
diff --git a/src/hooks/useOnAnnotationCreateSignatureToolMode/index.js b/src/hooks/useOnAnnotationCreateSignatureToolMode/index.js
new file mode 100644
index 0000000000..2264e9c3af
--- /dev/null
+++ b/src/hooks/useOnAnnotationCreateSignatureToolMode/index.js
@@ -0,0 +1,3 @@
+import useOnAnnotationCreateSignatureToolMode from './useOnAnnotationCreateSignatureToolMode';
+
+export default useOnAnnotationCreateSignatureToolMode;
\ No newline at end of file
diff --git a/src/hooks/useOnAnnotationPopupOpen/index.js b/src/hooks/useOnAnnotationPopupOpen/index.js
new file mode 100644
index 0000000000..62fb3ba75c
--- /dev/null
+++ b/src/hooks/useOnAnnotationPopupOpen/index.js
@@ -0,0 +1,3 @@
+import useOnAnnotationPopupOpen from './useOnAnnotationPopupOpen';
+
+export default useOnAnnotationPopupOpen;
\ No newline at end of file
diff --git a/src/hooks/useOnCountMeasurementAnnotationSelected/useOnCountMeasurementAnnotationSelected.js b/src/hooks/useOnCountMeasurementAnnotationSelected/useOnCountMeasurementAnnotationSelected.js
index 1223246222..80e6de8fab 100644
--- a/src/hooks/useOnCountMeasurementAnnotationSelected/useOnCountMeasurementAnnotationSelected.js
+++ b/src/hooks/useOnCountMeasurementAnnotationSelected/useOnCountMeasurementAnnotationSelected.js
@@ -59,10 +59,7 @@ export default function useOnCountMeasurementAnnotationSelected() {
) {
setAnnotation(annotations[0]);
dispatch(actions.openElement(DataElements.MEASUREMENT_OVERLAY));
- } else if (
- action === 'deselected' &&
- !core.isAnnotationSelected(annotation)
- ) {
+ } else if (action === 'deselected' && !core.isAnnotationSelected(annotation)) {
dispatch(actions.closeElement(DataElements.MEASUREMENT_OVERLAY));
}
};
diff --git a/src/hooks/useOnCropAnnotationAdded/index.js b/src/hooks/useOnCropAnnotationAdded/index.js
new file mode 100644
index 0000000000..66ebd9d402
--- /dev/null
+++ b/src/hooks/useOnCropAnnotationAdded/index.js
@@ -0,0 +1,3 @@
+import useOnCropAnnotationAdded from './useOnCropAnnotationAdded';
+
+export default useOnCropAnnotationAdded;
\ No newline at end of file
diff --git a/src/hooks/useOnCropAnnotationAdded/useOnCropAnnotationAdded.js b/src/hooks/useOnCropAnnotationAdded/useOnCropAnnotationAdded.js
new file mode 100644
index 0000000000..ceb8edb931
--- /dev/null
+++ b/src/hooks/useOnCropAnnotationAdded/useOnCropAnnotationAdded.js
@@ -0,0 +1,24 @@
+import { useEffect, useState } from 'react';
+import core from 'core';
+
+export default function useOnCropAnnotationAdded(openDocumentCropPopup) {
+ const [cropAnnotation, setCropAnnotation] = useState(null);
+
+ useEffect(() => {
+ const onAnnotationChanged = (annotations, action) => {
+ const annotation = annotations[0];
+ if (action === 'add' && annotation.Subject === 'Rectangle' && annotation.ToolName === 'CropPage') {
+ setCropAnnotation(annotation);
+ openDocumentCropPopup();
+ }
+ };
+
+ core.addEventListener('annotationChanged', onAnnotationChanged);
+
+ return () => {
+ core.removeEventListener('annotationChanged', onAnnotationChanged);
+ };
+ }, []);
+
+ return cropAnnotation;
+}
\ No newline at end of file
diff --git a/src/hooks/useOnCropAnnotationChangedOrSelected/index.js b/src/hooks/useOnCropAnnotationChangedOrSelected/index.js
new file mode 100644
index 0000000000..91ececbb66
--- /dev/null
+++ b/src/hooks/useOnCropAnnotationChangedOrSelected/index.js
@@ -0,0 +1,3 @@
+import useOnCropAnnotationChangedOrSelected from './useOnCropAnnotationChangedOrSelected';
+
+export default useOnCropAnnotationChangedOrSelected;
\ No newline at end of file
diff --git a/src/hooks/useOnCropAnnotationChangedOrSelected/useOnCropAnnotationChangedOrSelected.js b/src/hooks/useOnCropAnnotationChangedOrSelected/useOnCropAnnotationChangedOrSelected.js
new file mode 100644
index 0000000000..387599b8df
--- /dev/null
+++ b/src/hooks/useOnCropAnnotationChangedOrSelected/useOnCropAnnotationChangedOrSelected.js
@@ -0,0 +1,37 @@
+import { useEffect, useState } from 'react';
+import core from 'core';
+
+export default function useOnCropAnnotationChangedOrSelected(openDocumentCropPopup) {
+ const [cropAnnotation, setCropAnnotation] = useState(null);
+
+ useEffect(() => {
+ const onAnnotationChanged = (annotations, action) => {
+ const annotation = annotations[0];
+ if (action === 'add' && annotation.ToolName === window.Core.Tools.ToolNames['CROP'] && annotation instanceof window.Core.Annotations.RectangleAnnotation) {
+ setCropAnnotation(annotation);
+ openDocumentCropPopup();
+ }
+ if (action === 'delete' && annotation.ToolName === window.Core.Tools.ToolNames['CROP'] && annotation instanceof window.Core.Annotations.RectangleAnnotation) {
+ setCropAnnotation(null);
+ }
+ };
+
+ const onAnnotationSelected = (annotations, action) => {
+ const annotation = annotations[0];
+ if (action === 'selected' && annotation.ToolName === window.Core.Tools.ToolNames['CROP'] && annotation instanceof window.Core.Annotations.RectangleAnnotation) {
+ setCropAnnotation(annotation);
+ openDocumentCropPopup();
+ }
+ };
+
+ core.addEventListener('annotationChanged', onAnnotationChanged);
+ core.addEventListener('annotationSelected', onAnnotationSelected);
+
+ return () => {
+ core.removeEventListener('annotationChanged', onAnnotationChanged);
+ core.removeEventListener('annotationSelected', onAnnotationSelected);
+ };
+ }, []);
+
+ return cropAnnotation;
+}
diff --git a/src/hooks/useOnFormFieldsChanged/index.js b/src/hooks/useOnFormFieldsChanged/index.js
new file mode 100644
index 0000000000..db652dca01
--- /dev/null
+++ b/src/hooks/useOnFormFieldsChanged/index.js
@@ -0,0 +1,3 @@
+import useOnFormFieldsChanged from './useOnFormFieldsChanged';
+
+export default useOnFormFieldsChanged;
\ No newline at end of file
diff --git a/src/hooks/useOnFormFieldsChanged/useOnFormFieldsChanged.js b/src/hooks/useOnFormFieldsChanged/useOnFormFieldsChanged.js
new file mode 100644
index 0000000000..08a24daf40
--- /dev/null
+++ b/src/hooks/useOnFormFieldsChanged/useOnFormFieldsChanged.js
@@ -0,0 +1,64 @@
+import { useState, useEffect } from 'react';
+import core from 'core';
+
+export default function useOnFormFieldsChanged() {
+ const [formFieldAnnotationsList, setFormFieldAnnotationsList] = useState([]);
+
+ useEffect(() => {
+ const setFormFieldIndicators = () => {
+ let formFieldIndicators = [];
+ const annotations = core.getAnnotationsList();
+ const formFieldCreationManager = core.getFormFieldCreationManager();
+ if (formFieldCreationManager.isInFormFieldCreationMode()) {
+ const formFieldPlaceholders = annotations.filter((annotation) => annotation.isFormFieldPlaceholder());
+ formFieldIndicators = [
+ ...formFieldPlaceholders
+ .reduce(
+ (fieldNameMap, field) => {
+ if (!fieldNameMap.has(field.getCustomData(formFieldCreationManager.getFieldLabels().FIELD_NAME))) {
+ fieldNameMap.set(field.getCustomData(formFieldCreationManager.getFieldLabels().FIELD_NAME), field);
+ }
+ return fieldNameMap;
+ },
+ new Map()
+ ).values(),
+ ];
+ } else {
+ const widgets = annotations.filter(
+ (annotation) => formFieldCreationManager.getShowIndicator(annotation)
+ );
+ formFieldIndicators = [
+ ...widgets
+ .reduce(
+ (fieldNameMap, field) => {
+ if (!fieldNameMap.has(field['fieldName'])) {
+ fieldNameMap.set(field['fieldName'], field);
+ }
+ return fieldNameMap;
+ },
+ new Map()
+ ).values(),
+ ];
+ }
+ setFormFieldAnnotationsList(formFieldIndicators);
+ };
+
+ const onDocumentLoaded = () => {
+ setFormFieldAnnotationsList([]);
+ };
+
+ core.addEventListener('documentLoaded', onDocumentLoaded);
+ core.addEventListener('zoomUpdated', setFormFieldIndicators);
+ core.addEventListener('annotationChanged', setFormFieldIndicators);
+ core.addEventListener('pageNumberUpdated', setFormFieldIndicators);
+
+ return () => {
+ core.removeEventListener('documentLoaded', onDocumentLoaded);
+ core.removeEventListener('zoomUpdated', setFormFieldIndicators);
+ core.removeEventListener('annotationChanged', setFormFieldIndicators);
+ core.removeEventListener('pageNumberUpdated', setFormFieldIndicators);
+ };
+ });
+
+ return formFieldAnnotationsList;
+}
diff --git a/src/hooks/useOnInlineCommentPopupOpen/index.js b/src/hooks/useOnInlineCommentPopupOpen/index.js
new file mode 100644
index 0000000000..664fdf586b
--- /dev/null
+++ b/src/hooks/useOnInlineCommentPopupOpen/index.js
@@ -0,0 +1,3 @@
+import useOnInlineCommentPopupOpen from './useOnInlineCommentPopupOpen';
+
+export default useOnInlineCommentPopupOpen;
\ No newline at end of file
diff --git a/src/hooks/useOnInlineCommentPopupOpen/useOnInlineCommentPopupOpen.js b/src/hooks/useOnInlineCommentPopupOpen/useOnInlineCommentPopupOpen.js
new file mode 100644
index 0000000000..ca8a8b5282
--- /dev/null
+++ b/src/hooks/useOnInlineCommentPopupOpen/useOnInlineCommentPopupOpen.js
@@ -0,0 +1,125 @@
+import { useEffect, useState } from 'react';
+import { useDispatch, useSelector, shallowEqual } from 'react-redux';
+import actions from 'actions';
+import selectors from 'selectors';
+import core from 'core';
+import DataElements from 'constants/dataElement';
+
+export default function useOnInlineCommentPopupOpen() {
+ const [
+ isNotesPanelOpen,
+ notesInLeftPanel,
+ leftPanelOpen,
+ activeLeftPanel,
+ inlineCommentFilter,
+ activeDocumentViewerKey,
+ ] = useSelector(
+ (state) => [
+ selectors.isElementOpen(state, DataElements.NOTES_PANEL),
+ selectors.getNotesInLeftPanel(state),
+ selectors.isElementOpen(state, DataElements.LEFT_PANEL),
+ selectors.getActiveLeftPanel(state),
+ selectors.getInlineCommentFilter(state),
+ selectors.getActiveDocumentViewerKey(state),
+ ],
+ shallowEqual,
+ );
+ const dispatch = useDispatch();
+
+ const [annotation, setAnnotation] = useState(null);
+ const [isFreeTextAnnotationAdded, setFreeTextAnnotationAdded] = useState(false);
+ const { ToolNames } = window.Core.Tools;
+
+ const isNotesPanelOpenOrActive = isNotesPanelOpen || (notesInLeftPanel && leftPanelOpen && activeLeftPanel === 'notesPanel');
+
+ const closeAndReset = () => {
+ dispatch(actions.closeElement(DataElements.INLINE_COMMENT_POPUP));
+ setAnnotation(null);
+ setFreeTextAnnotationAdded(false);
+ };
+
+ const isFreeTextAnnotation = (annot) => {
+ return annot instanceof window.Core.Annotations.FreeTextAnnotation;
+ };
+
+ useEffect(() => {
+ const onAnnotationDoubleClicked = (annot) => {
+ if (isFreeTextAnnotation(annot)) {
+ closeAndReset();
+ }
+ };
+
+ core.addEventListener('annotationDoubleClicked', onAnnotationDoubleClicked, null, activeDocumentViewerKey);
+ return () => core.removeEventListener('annotationDoubleClicked', onAnnotationDoubleClicked, null, activeDocumentViewerKey);
+ }, [activeDocumentViewerKey]);
+
+ useEffect(() => {
+ const onAnnotationSelected = (annotations, action) => {
+ const selectedAnnotationTool = annotations[0].ToolName;
+ const shouldSetCommentingAnnotation =
+ action === 'selected'
+ && annotations.length
+ && !isFreeTextAnnotationAdded
+ && selectedAnnotationTool !== ToolNames.CROP;
+ if (shouldSetCommentingAnnotation) {
+ setAnnotation(annotations[0]);
+ }
+
+ if (action === 'deselected' && annotations.length) {
+ setFreeTextAnnotationAdded(false);
+ if (annotations.some((annot) => annot === annotation)) {
+ closeAndReset();
+ }
+ }
+ };
+
+ core.addEventListener('annotationSelected', onAnnotationSelected, null, activeDocumentViewerKey);
+ return () => {
+ core.removeEventListener('annotationSelected', onAnnotationSelected, null, activeDocumentViewerKey);
+ };
+ }, [annotation, isFreeTextAnnotationAdded, activeDocumentViewerKey]);
+
+ useEffect(() => {
+ setFreeTextAnnotationAdded(false);
+ const onMouseLeftUp = (e) => {
+ // WILL BE TRIGGERED ON MOBILE: happens before annotationSelected
+ // clicking on the selected annotation is considered clicking outside of this component
+ // so this component will close due to useOnClickOutside
+ // this handler is used to make sure that if we click on the selected annotation, this component will show up again
+ const annotUnderMouse = core.getAnnotationByMouseEvent(e, activeDocumentViewerKey);
+
+ if (annotation) {
+ if (!annotUnderMouse) {
+ closeAndReset();
+ }
+
+ if (core.isAnnotationSelected(annotUnderMouse) && annotUnderMouse !== annotation) {
+ setAnnotation(annotUnderMouse);
+ }
+ }
+ };
+
+ const onAnnotationChanged = (annotations, action) => {
+ setFreeTextAnnotationAdded(action === 'add' && isFreeTextAnnotation(annotations[0]));
+ const isCommentingAnnotationSelected = core.isAnnotationSelected(annotation);
+ if (annotation && !isCommentingAnnotationSelected) {
+ closeAndReset();
+ }
+ };
+
+ core.addEventListener('mouseLeftUp', onMouseLeftUp, null, activeDocumentViewerKey);
+ core.addEventListener('annotationChanged', onAnnotationChanged, null, activeDocumentViewerKey);
+ return () => {
+ core.removeEventListener('mouseLeftUp', onMouseLeftUp, null, activeDocumentViewerKey);
+ core.removeEventListener('annotationChanged', onAnnotationChanged, null, activeDocumentViewerKey);
+ };
+ }, [annotation, activeDocumentViewerKey]);
+
+ useEffect(() => {
+ if (!isNotesPanelOpenOrActive && annotation && inlineCommentFilter(annotation)) {
+ dispatch(actions.openElement(DataElements.INLINE_COMMENT_POPUP));
+ }
+ }, [annotation, inlineCommentFilter]);
+
+ return { annotation, closeAndReset };
+}
\ No newline at end of file
diff --git a/src/hooks/useOnLinkAnnotationPopupOpen/index.js b/src/hooks/useOnLinkAnnotationPopupOpen/index.js
new file mode 100644
index 0000000000..3dcf1661e8
--- /dev/null
+++ b/src/hooks/useOnLinkAnnotationPopupOpen/index.js
@@ -0,0 +1,3 @@
+import useOnLinkAnnotationPopupOpen from './useOnLinkAnnotationPopupOpen';
+
+export default useOnLinkAnnotationPopupOpen;
\ No newline at end of file
diff --git a/src/hooks/useOnLinkAnnotationPopupOpen/useOnLinkAnnotationPopupOpen.spec.js b/src/hooks/useOnLinkAnnotationPopupOpen/useOnLinkAnnotationPopupOpen.spec.js
new file mode 100644
index 0000000000..7b8237f80e
--- /dev/null
+++ b/src/hooks/useOnLinkAnnotationPopupOpen/useOnLinkAnnotationPopupOpen.spec.js
@@ -0,0 +1,47 @@
+import React from 'react';
+import { renderHook } from '@testing-library/react-hooks';
+import useOnLinkAnnotationPopupOpen from './useOnLinkAnnotationPopupOpen';
+import core from 'core';
+
+jest.mock('core');
+
+const MockComponent = ({ children }) => ({children}
);
+const wrapper = withProviders(MockComponent);
+
+const scrollContainer = {
+ scrollLeft: 100,
+ scrollTop: 100,
+ addEventListener: jest.fn(),
+ removeEventListener: jest.fn(),
+};
+
+describe('useOnLinkAnnotationPopupOpen hook', () => {
+ beforeAll(() => {
+ core.addEventListener = jest.fn();
+ core.getScrollViewElement = jest.fn();
+ core.getScrollViewElement.mockReturnValue(scrollContainer);
+ core.getAnnotationManager = jest.fn().mockReturnValue({
+ getContentEditHistoryManager: jest.fn().mockReturnValue({
+ getAnnotationsByMouseEvent: jest.fn(),
+ }),
+ });
+ });
+
+ it('adds event listeners on mouse move', () => {
+ const { result } = renderHook(() => {
+ return useOnLinkAnnotationPopupOpen();
+ }, { wrapper });
+
+ expect(result.error).toBeUndefined();
+ expect(core.addEventListener).toBeCalledWith('mouseMove', expect.any(Function));
+ });
+
+ it('removes event listeners to mouse move when component is unmounted', () => {
+ const { result, unmount } = renderHook(() => useOnLinkAnnotationPopupOpen(), { wrapper });
+
+ expect(result.error).toBeUndefined();
+ unmount();
+
+ expect(core.removeEventListener).toBeCalledWith('mouseMove', expect.any(Function));
+ });
+});
diff --git a/src/hooks/useOnMeasurementToolOrAnnotationSelected/index.js b/src/hooks/useOnMeasurementToolOrAnnotationSelected/index.js
new file mode 100644
index 0000000000..ec3414a8f9
--- /dev/null
+++ b/src/hooks/useOnMeasurementToolOrAnnotationSelected/index.js
@@ -0,0 +1,3 @@
+import useOnMeasurementToolOrAnnotationSelected from './useOnMeasurementToolOrAnnotationSelected';
+
+export default useOnMeasurementToolOrAnnotationSelected;
\ No newline at end of file
diff --git a/src/hooks/useOnRightClick/index.js b/src/hooks/useOnRightClick/index.js
new file mode 100644
index 0000000000..ce45aa251c
--- /dev/null
+++ b/src/hooks/useOnRightClick/index.js
@@ -0,0 +1,3 @@
+import useOnRightClick from './useOnRightClick';
+
+export default useOnRightClick;
\ No newline at end of file
diff --git a/src/hooks/useOnRightClick/useOnRightClick.js b/src/hooks/useOnRightClick/useOnRightClick.js
new file mode 100644
index 0000000000..f3329aac40
--- /dev/null
+++ b/src/hooks/useOnRightClick/useOnRightClick.js
@@ -0,0 +1,53 @@
+import { useEffect } from 'react';
+import core from 'core';
+import getRootNode from 'helpers/getRootNode';
+import { shallowEqual, useSelector } from 'react-redux';
+import selectors from 'selectors';
+
+export default (handler) => {
+ const [
+ activeDocumentViewerKey,
+ isMultiViewerMode,
+ ] = useSelector(
+ (state) => [
+ selectors.getActiveDocumentViewerKey(state),
+ selectors.isMultiViewerMode(state),
+ ],
+ shallowEqual,
+ );
+
+ useEffect(() => {
+ const listener = (e) => {
+ const { tagName } = e.target;
+ const clickedOnInput = tagName === 'INPUT';
+ const clickedOnTextarea = tagName === 'TEXTAREA';
+ const clickedOnFreeTextarea = !!((
+ e.target.className === 'ql-editor'
+ || e.target.parentNode.className === 'ql-editor'
+ || e.target.parentNode.parentNode.className === 'ql-editor'
+ ));
+
+ const documentContainer =
+ isMultiViewerMode
+ ? getRootNode().querySelector(`#DocumentContainer${activeDocumentViewerKey}`)
+ : getRootNode().querySelector('.DocumentContainer');
+ const clickedOnDocumentContainer = documentContainer.contains(e.target);
+
+ if (
+ clickedOnDocumentContainer &&
+ // when clicking on these two elements we want to display the default context menu so that users can use auto-correction, look up dictionary, etc...
+ !(clickedOnInput || clickedOnTextarea || clickedOnFreeTextarea)
+ ) {
+ e.preventDefault();
+ handler(e);
+ }
+ };
+
+ getRootNode().addEventListener('contextmenu', listener);
+ core.addEventListener('longTap', listener, null, activeDocumentViewerKey);
+ return () => {
+ getRootNode().removeEventListener('contextmenu', listener);
+ core.removeEventListener('longTap', listener, null, activeDocumentViewerKey);
+ };
+ }, [handler, activeDocumentViewerKey, isMultiViewerMode]);
+};
diff --git a/src/hooks/useOnRightClick/useOnRightClick.spec.js b/src/hooks/useOnRightClick/useOnRightClick.spec.js
new file mode 100644
index 0000000000..d1597e94ec
--- /dev/null
+++ b/src/hooks/useOnRightClick/useOnRightClick.spec.js
@@ -0,0 +1,32 @@
+import React from 'react';
+import { renderHook } from '@testing-library/react-hooks';
+import useOnRightClick from './useOnRightClick';
+import core from 'core';
+
+jest.mock('core');
+
+const MockComponent = ({ children }) => ({children}
);
+const wrapper = withProviders(MockComponent);
+
+describe('useOnRightClick hook', () => {
+ it('adds event listeners to longTap on mobile', () => {
+ core.addEventListener = jest.fn();
+
+ const { result } = renderHook(() => useOnRightClick(), { wrapper });
+
+ expect(result.error).toBeUndefined();
+
+ expect(core.addEventListener).toBeCalledWith('longTap', expect.any(Function), null, 1);
+ });
+
+ it('removes event listeners to longTap when component is unmounted', () => {
+ core.removeEventListener = jest.fn();
+
+ const { result, unmount } = renderHook(() => useOnRightClick(), { wrapper });
+
+ expect(result.error).toBeUndefined();
+ unmount();
+
+ expect(core.removeEventListener).toBeCalledWith('longTap', expect.any(Function), null, 1);
+ });
+});
\ No newline at end of file
diff --git a/src/hooks/useOnRightClickAnnotation/index.js b/src/hooks/useOnRightClickAnnotation/index.js
new file mode 100644
index 0000000000..98132ed95f
--- /dev/null
+++ b/src/hooks/useOnRightClickAnnotation/index.js
@@ -0,0 +1,3 @@
+import useOnRightClickAnnotation from './useOnRightClickAnnotation';
+
+export default useOnRightClickAnnotation;
\ No newline at end of file
diff --git a/src/hooks/useOnRightClickAnnotation/useOnRightClickAnnotation.js b/src/hooks/useOnRightClickAnnotation/useOnRightClickAnnotation.js
new file mode 100644
index 0000000000..13b143ca35
--- /dev/null
+++ b/src/hooks/useOnRightClickAnnotation/useOnRightClickAnnotation.js
@@ -0,0 +1,20 @@
+import { useState, useCallback } from 'react';
+import core from 'core';
+import useOnRightClick from 'hooks/useOnRightClick';
+
+export default () => {
+ const [rightClickedAnnotation, setRightClickedAnnotation] = useState(null);
+
+ useOnRightClick(
+ useCallback((e) => {
+ const annotUnderMouse = core.getAnnotationByMouseEvent(e);
+ if (annotUnderMouse && annotUnderMouse.ToolName !== window.Core.Tools.ToolNames.CROP) {
+ if (annotUnderMouse !== rightClickedAnnotation) {
+ setRightClickedAnnotation(annotUnderMouse);
+ }
+ }
+ }, [rightClickedAnnotation])
+ );
+
+ return { rightClickedAnnotation, setRightClickedAnnotation };
+};
diff --git a/src/hooks/useOnSnippingAnnotationChangedOrSelected/index.js b/src/hooks/useOnSnippingAnnotationChangedOrSelected/index.js
new file mode 100644
index 0000000000..30a2f600cb
--- /dev/null
+++ b/src/hooks/useOnSnippingAnnotationChangedOrSelected/index.js
@@ -0,0 +1,3 @@
+import useOnSnippingAnnotationChangedOrSelected from './useOnSnippingAnnotationChangedOrSelected';
+
+export default useOnSnippingAnnotationChangedOrSelected;
\ No newline at end of file
diff --git a/src/hooks/useOnSnippingAnnotationChangedOrSelected/useOnSnippingAnnotationChangedOrSelected.js b/src/hooks/useOnSnippingAnnotationChangedOrSelected/useOnSnippingAnnotationChangedOrSelected.js
new file mode 100644
index 0000000000..cca8767535
--- /dev/null
+++ b/src/hooks/useOnSnippingAnnotationChangedOrSelected/useOnSnippingAnnotationChangedOrSelected.js
@@ -0,0 +1,37 @@
+import { useEffect, useState } from 'react';
+import core from 'core';
+
+export default function useOnCropAnnotationChangedOrSelected(openSnippingPopup) {
+ const [snippingAnnotation, setSnippingAnnotation] = useState(null);
+
+ useEffect(() => {
+ const onAnnotationChanged = (annotations, action) => {
+ const annotation = annotations[0];
+ if (action === 'add' && annotation.ToolName === window.Core.Tools.ToolNames['SNIPPING'] && annotation instanceof window.Core.Annotations.RectangleAnnotation) {
+ setSnippingAnnotation(annotation);
+ openSnippingPopup();
+ }
+ if (action === 'delete' && annotation.ToolName === window.Core.Tools.ToolNames['SNIPPING'] && annotation instanceof window.Core.Annotations.RectangleAnnotation) {
+ setSnippingAnnotation(null);
+ }
+ };
+
+ const onAnnotationSelected = (annotations, action) => {
+ const annotation = annotations[0];
+ if (action === 'selected' && annotation.ToolName === window.Core.Tools.ToolNames['SNIPPING'] && annotation instanceof window.Core.Annotations.RectangleAnnotation) {
+ setSnippingAnnotation(annotation);
+ openSnippingPopup();
+ }
+ };
+
+ core.addEventListener('annotationChanged', onAnnotationChanged);
+ core.addEventListener('annotationSelected', onAnnotationSelected);
+
+ return () => {
+ core.removeEventListener('annotationChanged', onAnnotationChanged);
+ core.removeEventListener('annotationSelected', onAnnotationSelected);
+ };
+ }, []);
+
+ return snippingAnnotation;
+}
diff --git a/src/hooks/useResizeObserver/index.js b/src/hooks/useResizeObserver/index.js
new file mode 100644
index 0000000000..4fa6b2d3ef
--- /dev/null
+++ b/src/hooks/useResizeObserver/index.js
@@ -0,0 +1,3 @@
+import useResizeObserver from './useResizeObserver';
+
+export default useResizeObserver;
\ No newline at end of file
diff --git a/src/hooks/useResizeObserver/useResizeObserver.js b/src/hooks/useResizeObserver/useResizeObserver.js
new file mode 100644
index 0000000000..ba446181da
--- /dev/null
+++ b/src/hooks/useResizeObserver/useResizeObserver.js
@@ -0,0 +1,34 @@
+import { useEffect, useRef, useState } from 'react';
+
+const useResizeObserver = () => {
+ const [dimensions, setDimensions] = useState({ width: null, height: null });
+ const elementRef = useRef(null);
+
+ useEffect(() => {
+ const node = elementRef.current;
+
+ if (node) {
+ const observer = new ResizeObserver((entries) => {
+ for (const entry of entries) {
+ const observedWidth = entry.borderBoxSize[0].inlineSize;
+ const observedHeight = entry.borderBoxSize[0].blockSize;
+ setDimensions({
+ width: observedWidth,
+ height: observedHeight
+ });
+ }
+ });
+
+ observer.observe(node);
+
+ // Cleanup: stop observing when the component unmounts
+ return () => {
+ observer.unobserve(node);
+ };
+ }
+ }, [elementRef.current]);
+
+ return [elementRef, dimensions];
+};
+
+export default useResizeObserver;
diff --git a/src/hooks/useResizeObserver/useResizeObserver.spec.js b/src/hooks/useResizeObserver/useResizeObserver.spec.js
new file mode 100644
index 0000000000..5f52f85886
--- /dev/null
+++ b/src/hooks/useResizeObserver/useResizeObserver.spec.js
@@ -0,0 +1,32 @@
+import { act } from '@testing-library/react-hooks';
+import { render } from '@testing-library/react';
+import React from 'react';
+import useResizeObserver from './useResizeObserver';
+
+// Mocking ResizeObserver as it's not available in Jest's JSDOM
+global.ResizeObserver = class ResizeObserver {
+ constructor(callback) {
+ this.callback = callback;
+ }
+ observe() {
+ this.callback([{ borderBoxSize: [{ inlineSize: 100, blockSize: 200 }] }]);
+ }
+ unobserve() { }
+};
+
+describe('useResizeObserver', () => {
+ it('returns correct dimensions after observing', () => {
+ let dimensions;
+ function DummyComponent() {
+ const [ref, size] = useResizeObserver();
+ dimensions = size;
+ return ;
+ }
+
+ act(() => {
+ render();
+ });
+
+ expect(dimensions).toEqual({ width: 100, height: 200 });
+ });
+});
\ No newline at end of file
diff --git a/src/redux/reducers/digitalSignatureValidationReducer.js b/src/redux/reducers/digitalSignatureValidationReducer.js
new file mode 100644
index 0000000000..08369bc8d7
--- /dev/null
+++ b/src/redux/reducers/digitalSignatureValidationReducer.js
@@ -0,0 +1,46 @@
+export default (initialState) => (state = initialState, action) => {
+ const { type, payload } = action;
+
+ switch (type) {
+ case 'SET_VALIDATION_MODAL_WIDGET_NAME':
+ return {
+ ...state,
+ validationModalWidgetName: payload.validationModalWidgetName,
+ };
+ case 'SET_VERIFICATION_RESULT':
+ return { ...state, verificationResult: payload.result };
+ case 'ADD_TRUSTED_CERTIFICATES':
+ /**
+ * To mimic the behavior of the Core implementation, where certificates
+ * can only be added but not removed, only allow this action to append
+ * to the existing array
+ */
+ return {
+ ...state,
+ certificates: [...state.certificates, ...payload.certificates],
+ };
+ case 'ADD_TRUST_LIST':
+ /**
+ * The Core implementation only allows a single Trust List to be passed
+ * as a parameter, but in order to allow flexibility of future potential
+ * requirements where a developer may want to add multiple Trust Lists,
+ * we are storing an Array of Trust Lists
+ */
+ return {
+ ...state,
+ trustLists: [...state.trustLists, payload.trustList],
+ };
+ case 'SET_IS_REVOCATION_CHECKING_ENABLED':
+ return {
+ ...state,
+ isRevocationCheckingEnabled: payload.isRevocationCheckingEnabled,
+ };
+ case 'SET_REVOCATION_PROXY_PREFIX':
+ return {
+ ...state,
+ revocationProxyPrefix: payload.revocationProxyPrefix,
+ };
+ default:
+ return state;
+ }
+};
diff --git a/src/redux/reducers/featureFlagsReducer.js b/src/redux/reducers/featureFlagsReducer.js
new file mode 100644
index 0000000000..af3ad53f08
--- /dev/null
+++ b/src/redux/reducers/featureFlagsReducer.js
@@ -0,0 +1,12 @@
+export default (initialState) => (state = initialState, action) => {
+ const { type, payload } = action;
+
+ switch (type) {
+ case 'ENABLE_FEATURE_FLAG':
+ return { ...state, [payload.featureFlag]: true };
+ case 'DISABLE_FEATURE_FLAG':
+ return { ...state, [payload.featureFlag]: false };
+ default:
+ return state;
+ }
+};
\ No newline at end of file
diff --git a/src/redux/reducers/wv3dPropertiesPanelReducer.js b/src/redux/reducers/wv3dPropertiesPanelReducer.js
new file mode 100644
index 0000000000..2ae898d19d
--- /dev/null
+++ b/src/redux/reducers/wv3dPropertiesPanelReducer.js
@@ -0,0 +1,26 @@
+export default (initialState) => (state = initialState, action) => {
+ const { type, payload } = action;
+
+ switch (type) {
+ case 'SET_WV3D_PROPERTIES_PANEL_MODEL_DATA': {
+ const { modelData } = payload;
+
+ return {
+ ...state,
+ modelData,
+ };
+ }
+
+ case 'SET_WV3D_PROPERTIES_PANEL_SCHEMA': {
+ const { schema } = payload;
+
+ return {
+ ...state,
+ schema,
+ };
+ }
+
+ default:
+ return state;
+ }
+};
diff --git a/webpack.config.dev.js b/webpack.config.dev.js
index aca52ad8b1..faff432e65 100644
--- a/webpack.config.dev.js
+++ b/webpack.config.dev.js
@@ -6,7 +6,7 @@ module.exports = {
mode: 'development',
devtool: 'cheap-module-eval-source-map',
entry: [
- 'webpack-hot-middleware/client?name=ui&path=/__webpack_hmr',
+ 'webpack-hot-middleware/client?name=ui&path=/__webpack_hmr&noInfo=true',
path.resolve(__dirname, 'src'),
],
output: {
@@ -57,29 +57,53 @@ module.exports = {
loader: 'style-loader',
options: {
insert: function (styleTag) {
- const webComponents = document.getElementsByTagName('apryse-webviewer');
- if (webComponents.length > 0) {
- const clonedStyleTags = [];
- for (let i = 0; i < webComponents.length; i++) {
- const webComponent = webComponents[i];
- if (i === 0) {
- webComponent.shadowRoot.appendChild(styleTag);
- styleTag.onload = function () {
- if (clonedStyleTags.length > 0) {
- clonedStyleTags.forEach((styleNode) => {
- // eslint-disable-next-line no-unsanitized/property
- styleNode.innerHTML = styleTag.innerHTML;
- });
- }
- };
- } else {
- const styleNode = styleTag.cloneNode(true);
- webComponent.shadowRoot.appendChild(styleNode);
- clonedStyleTags.push(styleNode);
+ function findNestedWebComponents(tagName, root = document) {
+ const elements = [];
+
+ // Check direct children
+ root.querySelectorAll(tagName).forEach(el => elements.push(el));
+
+ // Check shadow DOMs
+ root.querySelectorAll('*').forEach(el => {
+ if (el.shadowRoot) {
+ elements.push(...findNestedWebComponents(tagName, el.shadowRoot));
}
- }
- } else {
+ });
+
+ return elements;
+ }
+ // If its the iframe we just append to the document head
+ if (!window.isApryseWebViewerWebComponent) {
document.head.appendChild(styleTag);
+ return;
+ }
+
+ let webComponents;
+ // First we see if the webcomponent is at the document level
+ webComponents = document.getElementsByTagName('apryse-webviewer');
+ // If not, we check have to check if it is nested in another webcomponent
+ if (!webComponents.length) {
+ webComponents = findNestedWebComponents('apryse-webviewer');
+ }
+ // Now we append the style tag to each webcomponent
+ const clonedStyleTags = [];
+ for (let i = 0; i < webComponents.length; i++) {
+ const webComponent = webComponents[i];
+ if (i === 0) {
+ webComponent.shadowRoot.appendChild(styleTag);
+ styleTag.onload = function () {
+ if (clonedStyleTags.length > 0) {
+ clonedStyleTags.forEach((styleNode) => {
+ // eslint-disable-next-line no-unsanitized/property
+ styleNode.innerHTML = styleTag.innerHTML;
+ });
+ }
+ };
+ } else {
+ const styleNode = styleTag.cloneNode(true);
+ webComponent.shadowRoot.appendChild(styleNode);
+ clonedStyleTags.push(styleNode);
+ }
}
},
},
diff --git a/webpack.config.prod.js b/webpack.config.prod.js
index 8c112a59c9..96784a33e6 100644
--- a/webpack.config.prod.js
+++ b/webpack.config.prod.js
@@ -1,7 +1,7 @@
const path = require('path');
const CopyWebpackPlugin = require('copy-webpack-plugin');
// const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin;
-const MiniCssExtractPlugin = require('mini-css-extract-plugin');
+// const MiniCssExtractPlugin = require('mini-css-extract-plugin');
module.exports = {
mode: 'production',
@@ -32,10 +32,10 @@ module.exports = {
to: '../build/configorigin.txt',
},
]),
- new MiniCssExtractPlugin({
- filename: 'style.css',
- chunkFilename: 'chunks/[name].chunk.css'
- }),
+ // new MiniCssExtractPlugin({
+ // filename: 'style.css',
+ // chunkFilename: 'chunks/[name].chunk.css'
+ // }),
// new BundleAnalyzerPlugin()
],
module: {
@@ -45,12 +45,16 @@ module.exports = {
use: {
loader: 'babel-loader',
options: {
+ ignore: [
+ /\/core-js/,
+ ],
+ sourceType: "unambiguous",
presets: [
'@babel/preset-react',
[
'@babel/preset-env',
{
- useBuiltIns: 'entry',
+ useBuiltIns: 'usage',
corejs: 3,
},
],
@@ -66,14 +70,67 @@ module.exports = {
},
},
include: [path.resolve(__dirname, 'src'), path.resolve(__dirname, 'node_modules')],
- exclude: function(modulePath) {
+ exclude: function (modulePath) {
return /node_modules/.test(modulePath) && !/node_modules.+react-dnd/.test(modulePath);
}
},
{
test: /\.scss$/,
use: [
- MiniCssExtractPlugin.loader,
+ {
+ loader: 'style-loader',
+ options: {
+ insert: function (styleTag) {
+ function findNestedWebComponents(tagName, root = document) {
+ const elements = [];
+
+ // Check direct children
+ root.querySelectorAll(tagName).forEach(el => elements.push(el));
+
+ // Check shadow DOMs
+ root.querySelectorAll('*').forEach(el => {
+ if (el.shadowRoot) {
+ elements.push(...findNestedWebComponents(tagName, el.shadowRoot));
+ }
+ });
+
+ return elements;
+ }
+ if (!window.isApryseWebViewerWebComponent) {
+ document.head.appendChild(styleTag);
+ return;
+ }
+
+ let webComponents;
+ // First we see if the webcomponent is at the document level
+ webComponents = document.getElementsByTagName('apryse-webviewer');
+ // If not, we check have to check if it is nested in another webcomponent
+ if (!webComponents.length) {
+ webComponents = findNestedWebComponents('apryse-webviewer');
+ }
+ // Now we append the style tag to each webcomponent
+ const clonedStyleTags = [];
+ for (let i = 0; i < webComponents.length; i++) {
+ const webComponent = webComponents[i];
+ if (i === 0) {
+ webComponent.shadowRoot.appendChild(styleTag);
+ styleTag.onload = function () {
+ if (clonedStyleTags.length > 0) {
+ clonedStyleTags.forEach((styleNode) => {
+ // eslint-disable-next-line no-unsanitized/property
+ styleNode.innerHTML = styleTag.innerHTML;
+ });
+ }
+ };
+ } else {
+ const styleNode = styleTag.cloneNode(true);
+ webComponent.shadowRoot.appendChild(styleNode);
+ clonedStyleTags.push(styleNode);
+ }
+ }
+ },
+ },
+ },
'css-loader',
{
loader: 'postcss-loader',