From 5c82e439a785c5b2652d3a8bedc83d6830689f24 Mon Sep 17 00:00:00 2001 From: Kenichi Kobayashi Date: Thu, 16 Dec 2021 03:03:20 +0900 Subject: [PATCH] Desktop: Fixes #5708: Scroll positions are preserved (#5826) Features: - Scroll position is preserved when the editor layout changes. - Scroll position is remembered when a note selection changes. Modifications: - The current Sync Scroll feature (in v2.6.2) is modified to use line-percent-based scroll positions. - Scroll position translation functions, Viewer-to-Editor and Editor-to-Viewer, are separated into V2L / L2E and E2L / L2V respectively. - The scrollmap is moved from gui/utils/SyncScrollMap.ts to note-viewer/scrollmap.js. - IPC Protocol about the scrollmap becomes not necessary and is removed. - Ignores non-user scroll events to avoid sync with incorrect scroll positions. - When CodeMirror is not ready, setEditorPercentScroll() is waited. - Fixes the bug: An incorrect scroll position is sometimes recorded. - Since scroll positions become line-percent-based, the following incompatibilities of scroll positions are fixed: - Between Editor and Viewer. - Between Viewer Layout and Split Layout of Viewer - Between Editor Layout and Split Layout of Editor --- .eslintignore | 3 - .gitignore | 3 - .../NoteBody/CodeMirror/CodeMirror.tsx | 28 +-- .../NoteEditor/NoteBody/CodeMirror/Editor.tsx | 5 + .../CodeMirror/utils/useScrollHandler.ts | 178 ++++++++++------- .../app-desktop/gui/NoteEditor/NoteEditor.tsx | 5 +- packages/app-desktop/gui/NoteTextViewer.tsx | 12 -- .../app-desktop/gui/note-viewer/index.html | 184 ++++++++++++------ .../app-desktop/gui/note-viewer/scrollmap.js | 113 +++++++++++ .../app-desktop/gui/utils/SyncScrollMap.ts | 92 --------- 10 files changed, 371 insertions(+), 252 deletions(-) create mode 100644 packages/app-desktop/gui/note-viewer/scrollmap.js delete mode 100644 packages/app-desktop/gui/utils/SyncScrollMap.ts diff --git a/.eslintignore b/.eslintignore index 60d282904..a2eb738b3 100644 --- a/.eslintignore +++ b/.eslintignore @@ -705,9 +705,6 @@ packages/app-desktop/gui/style/StyledTextInput.js.map packages/app-desktop/gui/utils/NoteListUtils.d.ts packages/app-desktop/gui/utils/NoteListUtils.js packages/app-desktop/gui/utils/NoteListUtils.js.map -packages/app-desktop/gui/utils/SyncScrollMap.d.ts -packages/app-desktop/gui/utils/SyncScrollMap.js -packages/app-desktop/gui/utils/SyncScrollMap.js.map packages/app-desktop/gui/utils/convertToScreenCoordinates.d.ts packages/app-desktop/gui/utils/convertToScreenCoordinates.js packages/app-desktop/gui/utils/convertToScreenCoordinates.js.map diff --git a/.gitignore b/.gitignore index 388107a1d..fe324bfc8 100644 --- a/.gitignore +++ b/.gitignore @@ -688,9 +688,6 @@ packages/app-desktop/gui/style/StyledTextInput.js.map packages/app-desktop/gui/utils/NoteListUtils.d.ts packages/app-desktop/gui/utils/NoteListUtils.js packages/app-desktop/gui/utils/NoteListUtils.js.map -packages/app-desktop/gui/utils/SyncScrollMap.d.ts -packages/app-desktop/gui/utils/SyncScrollMap.js -packages/app-desktop/gui/utils/SyncScrollMap.js.map packages/app-desktop/gui/utils/convertToScreenCoordinates.d.ts packages/app-desktop/gui/utils/convertToScreenCoordinates.js packages/app-desktop/gui/utils/convertToScreenCoordinates.js.map diff --git a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/CodeMirror.tsx b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/CodeMirror.tsx index 830ad837d..43e6efe76 100644 --- a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/CodeMirror.tsx +++ b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/CodeMirror.tsx @@ -7,7 +7,7 @@ import { commandAttachFileToBody, handlePasteEvent } from '../../utils/resourceH import { ScrollOptions, ScrollOptionTypes } from '../../utils/types'; import { CommandValue } from '../../utils/types'; import { usePrevious, cursorPositionToTextOffset } from './utils'; -import useScrollHandler, { translateScrollPercentToEditor, translateScrollPercentToViewer } from './utils/useScrollHandler'; +import useScrollHandler from './utils/useScrollHandler'; import useElementSize from '@joplin/lib/hooks/useElementSize'; import Toolbar from './Toolbar'; import styles_ from './styles'; @@ -65,7 +65,8 @@ function CodeMirror(props: NoteBodyEditorProps, ref: any) { usePluginServiceRegistration(ref); - const { resetScroll, editor_scroll, setEditorPercentScroll, setViewerPercentScroll } = useScrollHandler(editorRef, webviewRef, props.onScroll); + const { resetScroll, editor_scroll, setEditorPercentScroll, setViewerPercentScroll, editor_resize, + } = useScrollHandler(editorRef, webviewRef, props.onScroll); const codeMirror_change = useCallback((newBody: string) => { props_onChangeRef.current({ changeId: null, content: newBody }); @@ -115,10 +116,9 @@ function CodeMirror(props: NoteBodyEditorProps, ref: any) { if (!webviewRef.current) return; webviewRef.current.wrappedInstance.send('scrollToHash', options.value as string); } else if (options.type === ScrollOptionTypes.Percent) { - const editorPercent = options.value as number; - setEditorPercentScroll(editorPercent); - const viewerPercent = translateScrollPercentToViewer(editorRef, webviewRef, editorPercent); - setViewerPercentScroll(viewerPercent); + const percent = options.value as number; + setEditorPercentScroll(percent); + setViewerPercentScroll(percent); } else { throw new Error(`Unsupported scroll options: ${options.type}`); } @@ -581,17 +581,8 @@ function CodeMirror(props: NoteBodyEditorProps, ref: any) { editorRef.current.updateBody(newBody); } } else if (msg === 'percentScroll') { - const viewerPercent = arg0; - const editorPercent = translateScrollPercentToEditor(editorRef, webviewRef, viewerPercent); - setEditorPercentScroll(editorPercent); - } else if (msg === 'syncViewerScrollWithEditor') { - const force = !!arg0; - webviewRef.current?.wrappedInstance?.refreshSyncScrollMap(force); - const editorPercent = Math.max(0, Math.min(1, editorRef.current?.getScrollPercent())); - if (!isNaN(editorPercent)) { - const viewerPercent = translateScrollPercentToViewer(editorRef, webviewRef, editorPercent); - setViewerPercentScroll(viewerPercent); - } + const percent = arg0; + setEditorPercentScroll(percent); } else { props.onMessage(event); } @@ -644,6 +635,7 @@ function CodeMirror(props: NoteBodyEditorProps, ref: any) { const options: any = { pluginAssets: renderedBody.pluginAssets, downloadResources: Setting.value('sync.resourceDownloadMode'), + markupLineCount: editorRef.current?.lineCount() || 0, }; // It seems when there's an error immediately when the component is @@ -652,7 +644,6 @@ function CodeMirror(props: NoteBodyEditorProps, ref: any) { // Since we can't do much about it we just print an error. if (webviewRef.current && webviewRef.current.wrappedInstance) { webviewRef.current.wrappedInstance.send('setHtml', renderedBody.html, options); - webviewRef.current.wrappedInstance.refreshSyncScrollMap(true); } else { console.error('Trying to set HTML on an undefined webview ref'); } @@ -829,6 +820,7 @@ function CodeMirror(props: NoteBodyEditorProps, ref: any) { onScroll={editor_scroll} onEditorPaste={onEditorPaste} isSafeMode={props.isSafeMode} + onResize={editor_resize} /> ); diff --git a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/Editor.tsx b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/Editor.tsx index e423a46a5..70f92d83e 100644 --- a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/Editor.tsx +++ b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/Editor.tsx @@ -93,6 +93,7 @@ export interface EditorProps { onScroll: any; onEditorPaste: any; isSafeMode: boolean; + onResize: any; } function Editor(props: EditorProps, ref: any) { @@ -189,6 +190,8 @@ function Editor(props: EditorProps, ref: any) { cm.on('paste', editor_paste); cm.on('drop', editor_drop); cm.on('dragover', editor_drag); + cm.on('refresh', props.onResize); + cm.on('update', props.onResize); // It's possible for searchMarkers to be available before the editor // In these cases we set the markers asap so the user can see them as @@ -202,6 +205,8 @@ function Editor(props: EditorProps, ref: any) { cm.off('paste', editor_paste); cm.off('drop', editor_drop); cm.off('dragover', editor_drag); + cm.off('refresh', props.onResize); + cm.off('update', props.onResize); editorParent.current.removeChild(cm.getWrapperElement()); setEditor(null); }; diff --git a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/utils/useScrollHandler.ts b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/utils/useScrollHandler.ts index 228f57087..a4efe3e66 100644 --- a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/utils/useScrollHandler.ts +++ b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/utils/useScrollHandler.ts @@ -1,10 +1,34 @@ import { useCallback, useRef } from 'react'; import shim from '@joplin/lib/shim'; -import { SyncScrollMap } from '../../../../utils/SyncScrollMap'; export default function useScrollHandler(editorRef: any, webviewRef: any, onScroll: Function) { - const ignoreNextEditorScrollEvent_ = useRef(false); const scrollTimeoutId_ = useRef(null); + const scrollPercent_ = useRef(0); + const ignoreNextEditorScrollTime_ = useRef(Date.now()); + const ignoreNextEditorScrollEventCount_ = useRef(0); + const delayedSetEditorPercentScrollTimeoutID_ = useRef(null); + + // Ignores one next scroll event for a short time. + const ignoreNextEditorScrollEvent = () => { + const now = Date.now(); + if (now >= ignoreNextEditorScrollTime_.current) ignoreNextEditorScrollEventCount_.current = 0; + if (ignoreNextEditorScrollEventCount_.current < 10) { // for safety + ignoreNextEditorScrollTime_.current = now + 200; + ignoreNextEditorScrollEventCount_.current += 1; + } + }; + + // Tests the next scroll event should be ignored and then decrements the count. + const isNextEditorScrollEventIgnored = () => { + if (ignoreNextEditorScrollEventCount_.current) { + if (Date.now() < ignoreNextEditorScrollTime_.current) { + ignoreNextEditorScrollEventCount_.current -= 1; + return true; + } + ignoreNextEditorScrollEventCount_.current = 0; + } + return false; + }; const scheduleOnScroll = useCallback((event: any) => { if (scrollTimeoutId_.current) { @@ -18,108 +42,128 @@ export default function useScrollHandler(editorRef: any, webviewRef: any, onScro }, 10); }, [onScroll]); - const setEditorPercentScroll = useCallback((p: number) => { - ignoreNextEditorScrollEvent_.current = true; + const setEditorPercentScrollInternal = (percent: number) => { + scrollPercent_.current = percent; + let retry = 0; + const fn = () => { + if (delayedSetEditorPercentScrollTimeoutID_.current) { + shim.clearInterval(delayedSetEditorPercentScrollTimeoutID_.current); + delayedSetEditorPercentScrollTimeoutID_.current = null; + } + const cm = editorRef.current; + if (isCodeMirrorReady(cm)) { + // calculates editor's GUI-dependent pixel-based raw percent + const newEditorPercent = translateScrollPercentL2E(cm, scrollPercent_.current); + const oldEditorPercent = cm.getScrollPercent(); + if (!(Math.abs(newEditorPercent - oldEditorPercent) < 1e-8)) { + ignoreNextEditorScrollEvent(); + cm.setScrollPercent(newEditorPercent); + } + } else { + retry += 1; + if (retry <= 10) { + delayedSetEditorPercentScrollTimeoutID_.current = shim.setTimeout(fn, 50); + } + } + }; + fn(); + }; + const restoreEditorPercentScroll = () => { + if (isCodeMirrorReady(editorRef.current)) { + setEditorPercentScrollInternal(scrollPercent_.current); + } + }; + + const setEditorPercentScroll = useCallback((percent: number) => { + setEditorPercentScrollInternal(percent); if (editorRef.current) { - editorRef.current.setScrollPercent(p); - - scheduleOnScroll({ percent: p }); + scheduleOnScroll({ percent }); } }, [scheduleOnScroll]); - const setViewerPercentScroll = useCallback((p: number) => { + const setViewerPercentScroll = useCallback((percent: number) => { if (webviewRef.current) { - webviewRef.current.wrappedInstance.send('setPercentScroll', p); - scheduleOnScroll({ percent: p }); + webviewRef.current.wrappedInstance.send('setPercentScroll', percent); + scheduleOnScroll({ percent }); } }, [scheduleOnScroll]); const editor_scroll = useCallback(() => { - if (ignoreNextEditorScrollEvent_.current) { - ignoreNextEditorScrollEvent_.current = false; - return; - } + if (isNextEditorScrollEventIgnored()) return; - if (editorRef.current) { - const editorPercent = Math.max(0, Math.min(1, editorRef.current.getScrollPercent())); + const cm = editorRef.current; + if (isCodeMirrorReady(cm)) { + const editorPercent = Math.max(0, Math.min(1, cm.getScrollPercent())); if (!isNaN(editorPercent)) { // when switching to another note, the percent can sometimes be NaN // this is coming from `gui/NoteEditor/NoteBody/CodeMirror/utils/useScrollUtils.ts` // when CodeMirror returns scroll info with heigth == clientHeigth // https://github.com/laurent22/joplin/issues/4797 - const viewerPercent = translateScrollPercentToViewer(editorRef, webviewRef, editorPercent); - setViewerPercentScroll(viewerPercent); + + // calculates GUI-independent line-based percent + const percent = translateScrollPercentE2L(cm, editorPercent); + scrollPercent_.current = percent; + setViewerPercentScroll(percent); } } }, [setViewerPercentScroll]); const resetScroll = useCallback(() => { + scrollPercent_.current = 0; if (editorRef.current) { editorRef.current.setScrollPercent(0); } }, []); - return { resetScroll, setEditorPercentScroll, setViewerPercentScroll, editor_scroll }; + const editor_resize = useCallback((cm) => { + if (cm) { + restoreEditorPercentScroll(); + } + }, []); + + return { + resetScroll, setEditorPercentScroll, setViewerPercentScroll, editor_scroll, editor_resize, + }; } -const translateScrollPercent_ = (editorRef: any, webviewRef: any, percent: number, editorToViewer: boolean) => { +const translateLE_ = (codeMirror: any, percent: number, l2e: boolean) => { // If the input is out of (0,1) or not number, it is not translated. if (!(0 < percent && percent < 1)) return percent; - const map: SyncScrollMap = webviewRef.current?.wrappedInstance.getSyncScrollMap(); - const cm = editorRef.current; - if (!map || map.line.length <= 2 || !cm) return percent; // No translation - const lineCount = cm.lineCount(); - if (map.line[map.line.length - 2] >= lineCount) { - // Discarded a obsolete map and use no translation. - webviewRef.current.wrappedInstance.refreshSyncScrollMap(false); - return percent; - } - const info = cm.getScrollInfo(); - const height = Math.max(1, info.height - info.clientHeight); - let values = map.percent, target = percent; - if (editorToViewer) { - const top = percent * height; - const line = cm.lineAtHeight(top, 'local'); - values = map.line; - target = line; - } - // Binary search (rightmost): finds where map[r-1][field] <= target < map[r][field] - let l = 1, r = values.length - 1; - while (l < r) { - const m = Math.floor(l + (r - l) / 2); - if (target < values[m]) r = m; else l = m + 1; - } - const lineU = map.line[r - 1]; - const lineL = Math.min(lineCount, map.line[r]); - const ePercentU = r == 1 ? 0 : Math.min(1, cm.heightAtLine(lineU, 'local') / height); - const ePercentL = Math.min(1, cm.heightAtLine(lineL, 'local') / height); - const vPercentU = map.percent[r - 1]; - const vPercentL = ePercentL == 1 ? 1 : map.percent[r]; - let result; - if (editorToViewer) { - const linInterp = (percent - ePercentU) / (ePercentL - ePercentU); - result = vPercentU + (vPercentL - vPercentU) * linInterp; - } else { - const linInterp = (percent - vPercentU) / (vPercentL - vPercentU); + if (!codeMirror) return percent; // No translation + const info = codeMirror.getScrollInfo(); + const height = info.height - info.clientHeight; + if (height <= 1) return percent; // No translation for non-displayed CodeMirror. + const lineCount = codeMirror.lineCount(); + let lineU = l2e ? Math.floor(percent * lineCount) : codeMirror.lineAtHeight(percent * height, 'local'); + lineU = Math.max(0, Math.min(lineCount - 1, lineU)); + const ePercentU = codeMirror.heightAtLine(lineU, 'local') / height; + const ePercentL = codeMirror.heightAtLine(lineU + 1, 'local') / height; + let linInterp, result; + if (l2e) { + linInterp = percent * lineCount - lineU; result = ePercentU + (ePercentL - ePercentU) * linInterp; + } else { + linInterp = Math.max(0, Math.min(1, (percent - ePercentU) / (ePercentL - ePercentU))) || 0; + result = (lineU + linInterp) / lineCount; } return Math.max(0, Math.min(1, result)); }; -// translateScrollPercentToEditor() and translateScrollPercentToViewer() are -// the translation functions between Editor's scroll percent and Viewer's scroll +// translateScrollPercentL2E() and translateScrollPercentE2L() are +// the translation functions between Editor's scroll percent and line-based scroll // percent. They are used for synchronous scrolling between Editor and Viewer. -// They use a SyncScrollMap provided by Viewer for its translation. // To see the detail of synchronous scrolling, refer the following design document. -// https://github.com/laurent22/joplin/pull/5512#issuecomment-931277022 - -export const translateScrollPercentToEditor = (editorRef: any, webviewRef: any, viewerPercent: number) => { - const editorPercent = translateScrollPercent_(editorRef, webviewRef, viewerPercent, false); - return editorPercent; +// Replace me! https://github.com/laurent22/joplin/pull/5512#issuecomment-931277022 +const translateScrollPercentL2E = (cm: any, lPercent: number) => { + return translateLE_(cm, lPercent, true); }; -export const translateScrollPercentToViewer = (editorRef: any, webviewRef: any, editorPercent: number) => { - const viewerPercent = translateScrollPercent_(editorRef, webviewRef, editorPercent, true); - return viewerPercent; +const translateScrollPercentE2L = (cm: any, ePercent: number) => { + return translateLE_(cm, ePercent, false); }; + +function isCodeMirrorReady(cm: any) { + const info = cm?.getScrollInfo(); + return info && info.height - info.clientHeight > 0; +} diff --git a/packages/app-desktop/gui/NoteEditor/NoteEditor.tsx b/packages/app-desktop/gui/NoteEditor/NoteEditor.tsx index cea6e30e6..01cc4d9ac 100644 --- a/packages/app-desktop/gui/NoteEditor/NoteEditor.tsx +++ b/packages/app-desktop/gui/NoteEditor/NoteEditor.tsx @@ -344,7 +344,10 @@ function NoteEditor(props: NoteEditorProps) { const onScroll = useCallback((event: any) => { props.dispatch({ type: 'EDITOR_SCROLL_PERCENT_SET', - noteId: formNote.id, + // In callbacks of setTimeout()/setInterval(), props/state cannot be used + // to refer the current value, since they would be one or more generations old. + // For the purpose, useRef value should be used. + noteId: formNoteRef.current.id, percent: event.percent, }); }, [props.dispatch, formNote]); diff --git a/packages/app-desktop/gui/NoteTextViewer.tsx b/packages/app-desktop/gui/NoteTextViewer.tsx index 9f1212e10..d211fa724 100644 --- a/packages/app-desktop/gui/NoteTextViewer.tsx +++ b/packages/app-desktop/gui/NoteTextViewer.tsx @@ -2,7 +2,6 @@ import PostMessageService, { MessageResponse, ResponderComponentType } from '@jo import * as React from 'react'; const { connect } = require('react-redux'); import { reg } from '@joplin/lib/registry'; -import { SyncScrollMap, SyncScrollMapper } from './utils/SyncScrollMap'; interface Props { onDomReady: Function; @@ -168,17 +167,6 @@ class NoteTextViewerComponent extends React.Component { } } - private syncScrollMapper_ = new SyncScrollMapper; - - refreshSyncScrollMap(forced: boolean) { - return this.syncScrollMapper_.refresh(forced); - } - - getSyncScrollMap(): SyncScrollMap { - const doc = this.webviewRef_.current?.contentWindow?.document; - return this.syncScrollMapper_.get(doc); - } - // ---------------------------------------------------------------- // Wrap WebView functions (END) // ---------------------------------------------------------------- diff --git a/packages/app-desktop/gui/note-viewer/index.html b/packages/app-desktop/gui/note-viewer/index.html index b3da31481..eb4638bdf 100644 --- a/packages/app-desktop/gui/note-viewer/index.html +++ b/packages/app-desktop/gui/note-viewer/index.html @@ -42,6 +42,7 @@
+