Skip to content

Commit

Permalink
fix(compass-crud): persist modified document value COMPASS-8373 (#6404)
Browse files Browse the repository at this point in the history
* persist json value in document

* use ref and remove useEffect dep

* ref current assignment
  • Loading branch information
mabaasit authored Oct 29, 2024
1 parent 67cb7b7 commit daaa6e3
Show file tree
Hide file tree
Showing 3 changed files with 70 additions and 1 deletion.
19 changes: 18 additions & 1 deletion packages/compass-crud/src/components/json-editor.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -76,9 +76,26 @@ const JSONEditor: React.FunctionComponent<JSONEditorProps> = ({
const [expanded, setExpanded] = useState<boolean>(doc.expanded);
const [editing, setEditing] = useState<boolean>(doc.editing);
const [deleting, setDeleting] = useState<boolean>(doc.markedForDeletion);
const [value, setValue] = useState<string>(() => doc.toEJSON());
const [value, setValue] = useState<string>(
() => doc.modifiedEJSONString ?? doc.toEJSON()
);
const [initialValue] = useState<string>(() => doc.toEJSON());
const [containsErrors, setContainsErrors] = useState<boolean>(false);
const setModifiedEJSONStringRef = useRef<(value: string | null) => void>(
doc.setModifiedEJSONString.bind(doc)
);
setModifiedEJSONStringRef.current = doc.setModifiedEJSONString.bind(doc);

useEffect(() => {
const setModifiedEJSONString = setModifiedEJSONStringRef.current;
return () => {
// When this component is used in virtualized list, the editor is
// unmounted on scroll and if the user is editing the document, the
// editor value is lost. This is a way to keep track of the editor
// value when the it's unmounted and is restored on next mount.
setModifiedEJSONString(editing ? value : null);
};
}, [value, editing]);

const handleCopy = useCallback(() => {
copyToClipboard?.(doc);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,10 @@ import {
import { type VirtualListRef } from '@mongodb-js/compass-components';

import VirtualizedDocumentJsonView from './virtualized-document-json-view';
import {
getCodemirrorEditorValue,
setCodemirrorEditorValue,
} from '@mongodb-js/compass-editor';

const createBigDocument = (variable: number) =>
new HadronDocument({
Expand Down Expand Up @@ -168,4 +172,44 @@ describe('VirtualizedDocumentJsonView', function () {
expect(() => within(documentElement).getByText('Cancel')).to.throw;
expect(() => within(documentElement).getByText('Replace')).to.throw;
});

it('preserves the edit state of document when a document goes out of visible viewport when scrolling', async function () {
const bigDocuments = Array.from({ length: 20 }, (_, idx) =>
createBigDocument(idx)
);
const listRef: VirtualListRef = React.createRef();
render(
<VirtualizedDocumentJsonView
namespace="x.y"
docs={bigDocuments}
isEditable={true}
__TEST_OVERSCAN_COUNT={0}
__TEST_LIST_HEIGHT={178}
__TEST_LIST_REF={listRef}
/>
);

let [firstDocumentElement] = screen.getAllByTestId('editable-json');
// Trigger the edit state (we only set the edited value if the document is being edited)
userEvent.click(within(firstDocumentElement).getByLabelText('Edit'));

let cmEditor = firstDocumentElement.querySelector(
'[data-codemirror="true"]'
);
await setCodemirrorEditorValue(cmEditor, '{value: "edited"}');

// Scroll down and then scroll back up
act(() => {
listRef.current?.scrollToItem(15);
});
act(() => {
listRef.current?.scrollToItem(0);
});

[firstDocumentElement] = screen.getAllByTestId('editable-json');
cmEditor = firstDocumentElement.querySelector('[data-codemirror="true"]');

const value = getCodemirrorEditorValue(cmEditor);
expect(value).to.equal('{value: "edited"}');
});
});
8 changes: 8 additions & 0 deletions packages/hadron-document/src/document.ts
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,9 @@ export class Document extends EventEmitter {
maxVisibleElementsCount = DEFAULT_VISIBLE_ELEMENTS;
editing = false;
markedForDeletion = false;
// This is used to store the changed EJSON string when the document is modified
// via the JSONEditor.
modifiedEJSONString: string | null = null;

/**
* Send cancel event.
Expand Down Expand Up @@ -456,6 +459,7 @@ export class Document extends EventEmitter {
finishEditing() {
if (this.editing) {
this.editing = false;
this.setModifiedEJSONString(null);
this.emit(DocumentEvents.EditingFinished);
}
}
Expand Down Expand Up @@ -503,6 +507,10 @@ export class Document extends EventEmitter {
onRemoveError(error: Error) {
this.emit('remove-error', error.message);
}

setModifiedEJSONString(ejson: string | null) {
this.modifiedEJSONString = ejson;
}
}

export default Document;

0 comments on commit daaa6e3

Please sign in to comment.