mirror of
https://github.com/hedgedoc/hedgedoc.git
synced 2025-05-16 08:04:45 -04:00
Move toolbar functionality from redux to codemirror dispatch (#2083)
Signed-off-by: Tilman Vatteroth <git@tilmanvatteroth.de>
This commit is contained in:
parent
a8bd22aef3
commit
e93607c96e
99 changed files with 1730 additions and 1721 deletions
|
@ -28,6 +28,7 @@ describe('File upload', () => {
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
it('via button', () => {
|
it('via button', () => {
|
||||||
|
cy.getByCypressId('editor-pane').should('have.attr', 'data-cypress-editor-ready', 'true')
|
||||||
cy.getByCypressId('editor-toolbar-upload-image-button').should('be.visible')
|
cy.getByCypressId('editor-toolbar-upload-image-button').should('be.visible')
|
||||||
cy.getByCypressId('editor-toolbar-upload-image-input').selectFile(
|
cy.getByCypressId('editor-toolbar-upload-image-input').selectFile(
|
||||||
{
|
{
|
||||||
|
@ -37,15 +38,16 @@ describe('File upload', () => {
|
||||||
},
|
},
|
||||||
{ force: true }
|
{ force: true }
|
||||||
)
|
)
|
||||||
cy.get('.cm-line').contains(``)
|
cy.get('.cm-line').contains(``)
|
||||||
})
|
})
|
||||||
|
|
||||||
it('via paste', () => {
|
it('via paste', () => {
|
||||||
|
cy.getByCypressId('editor-pane').should('have.attr', 'data-cypress-editor-ready', 'true')
|
||||||
cy.fixture('demo.png').then((image: string) => {
|
cy.fixture('demo.png').then((image: string) => {
|
||||||
const pasteEvent = {
|
const pasteEvent = {
|
||||||
clipboardData: {
|
clipboardData: {
|
||||||
files: [Cypress.Blob.base64StringToBlob(image, 'image/png')],
|
files: [Cypress.Blob.base64StringToBlob(image, 'image/png')],
|
||||||
getData: (_: string) => ''
|
getData: () => ''
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
cy.get('.cm-content').trigger('paste', pasteEvent)
|
cy.get('.cm-content').trigger('paste', pasteEvent)
|
||||||
|
@ -54,6 +56,7 @@ describe('File upload', () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
it('via drag and drop', () => {
|
it('via drag and drop', () => {
|
||||||
|
cy.getByCypressId('editor-pane').should('have.attr', 'data-cypress-editor-ready', 'true')
|
||||||
cy.get('.cm-content').selectFile(
|
cy.get('.cm-content').selectFile(
|
||||||
{
|
{
|
||||||
contents: '@demoImage',
|
contents: '@demoImage',
|
||||||
|
@ -62,11 +65,12 @@ describe('File upload', () => {
|
||||||
},
|
},
|
||||||
{ action: 'drag-drop', force: true }
|
{ action: 'drag-drop', force: true }
|
||||||
)
|
)
|
||||||
cy.get('.cm-line').contains(``)
|
cy.get('.cm-line').contains(``)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
it('fails', () => {
|
it('fails', () => {
|
||||||
|
cy.getByCypressId('editor-pane').should('have.attr', 'data-cypress-editor-ready', 'true')
|
||||||
cy.intercept(
|
cy.intercept(
|
||||||
{
|
{
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
|
@ -89,12 +93,16 @@ describe('File upload', () => {
|
||||||
})
|
})
|
||||||
|
|
||||||
it('lets text paste still work', () => {
|
it('lets text paste still work', () => {
|
||||||
|
cy.getByCypressId('editor-pane').should('have.attr', 'data-cypress-editor-ready', 'true')
|
||||||
const testText = 'a long test text'
|
const testText = 'a long test text'
|
||||||
const pasteEvent = {
|
|
||||||
|
const pasteEvent: Event = Object.assign(new Event('paste', { bubbles: true, cancelable: true }), {
|
||||||
clipboardData: {
|
clipboardData: {
|
||||||
getData: (type = 'text') => testText
|
files: [],
|
||||||
}
|
getData: () => testText
|
||||||
}
|
}
|
||||||
|
})
|
||||||
|
|
||||||
cy.get('.cm-content').trigger('paste', pasteEvent)
|
cy.get('.cm-content').trigger('paste', pasteEvent)
|
||||||
cy.get('.cm-line').contains(`${testText}`)
|
cy.get('.cm-line').contains(`${testText}`)
|
||||||
})
|
})
|
||||||
|
|
|
@ -0,0 +1,57 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
import type { PropsWithChildren } from 'react'
|
||||||
|
import React, { createContext, useContext, useState } from 'react'
|
||||||
|
import Optional from 'optional-js'
|
||||||
|
import type { EditorView } from '@codemirror/view'
|
||||||
|
import type { ContentEdits } from '../editor-pane/tool-bar/formatters/types/changes'
|
||||||
|
import type { CursorSelection } from '../editor-pane/tool-bar/formatters/types/cursor-selection'
|
||||||
|
|
||||||
|
export type CodeMirrorReference = EditorView | undefined
|
||||||
|
type SetCodeMirrorReference = (value: CodeMirrorReference) => void
|
||||||
|
|
||||||
|
export type ContentFormatter = (parameters: {
|
||||||
|
currentSelection: CursorSelection
|
||||||
|
markdownContent: string
|
||||||
|
}) => [ContentEdits, CursorSelection | undefined]
|
||||||
|
|
||||||
|
type ChangeEditorContentContext = [CodeMirrorReference, SetCodeMirrorReference]
|
||||||
|
|
||||||
|
const changeEditorContentContext = createContext<ChangeEditorContentContext | undefined>(undefined)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extracts the code mirror reference from the parent context
|
||||||
|
*/
|
||||||
|
export const useCodeMirrorReference = (): CodeMirrorReference => {
|
||||||
|
const contextContent = Optional.ofNullable(useContext(changeEditorContentContext)).orElseThrow(
|
||||||
|
() => new Error('No change content received. Did you forget to use the provider component')
|
||||||
|
)
|
||||||
|
return contextContent[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extracts the code mirror reference from the parent context
|
||||||
|
*/
|
||||||
|
export const useSetCodeMirrorReference = (): SetCodeMirrorReference => {
|
||||||
|
const contextContent = Optional.ofNullable(useContext(changeEditorContentContext)).orElseThrow(
|
||||||
|
() => new Error('No change content received. Did you forget to use the provider component')
|
||||||
|
)
|
||||||
|
return contextContent[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a context for the child components that contains a ref to the current code mirror instance and a callback that posts changes to this codemirror.
|
||||||
|
*/
|
||||||
|
export const ChangeEditorContentContextProvider: React.FC<PropsWithChildren<unknown>> = ({ children }) => {
|
||||||
|
const [codeMirrorRef, setCodeMirrorRef] = useState<CodeMirrorReference>(undefined)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<changeEditorContentContext.Provider value={[codeMirrorRef, setCodeMirrorRef]}>
|
||||||
|
{children}
|
||||||
|
</changeEditorContentContext.Provider>
|
||||||
|
)
|
||||||
|
}
|
10
src/components/editor-page/change-content-context/code-mirror-selection.d.ts
vendored
Normal file
10
src/components/editor-page/change-content-context/code-mirror-selection.d.ts
vendored
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
export interface CodeMirrorSelection {
|
||||||
|
anchor: number
|
||||||
|
head?: number
|
||||||
|
}
|
|
@ -0,0 +1,50 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { useMemo } from 'react'
|
||||||
|
import Optional from 'optional-js'
|
||||||
|
import type { CodeMirrorSelection } from './code-mirror-selection'
|
||||||
|
import type { ContentFormatter } from './change-content-context'
|
||||||
|
import { useCodeMirrorReference } from './change-content-context'
|
||||||
|
import type { CursorSelection } from '../editor-pane/tool-bar/formatters/types/cursor-selection'
|
||||||
|
import type { EditorView } from '@codemirror/view'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Changes the content of the given CodeMirror view using the given formatter function.
|
||||||
|
*
|
||||||
|
* @param view The CodeMirror view whose content should be changed
|
||||||
|
* @param formatter A function that generates changes that get dispatched to CodeMirror
|
||||||
|
*/
|
||||||
|
export const changeEditorContent = (view: EditorView, formatter: ContentFormatter): void => {
|
||||||
|
const [changes, selection] = formatter({
|
||||||
|
currentSelection: {
|
||||||
|
from: view.state.selection.main.from,
|
||||||
|
to: view.state.selection.main.to
|
||||||
|
},
|
||||||
|
markdownContent: view.state.doc.toString()
|
||||||
|
})
|
||||||
|
|
||||||
|
view.dispatch({ changes: changes, selection: convertSelectionToCodeMirrorSelection(selection) })
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a {@link ContentFormatter formatter function} that is linked to the current CodeMirror-View
|
||||||
|
* @see changeEditorContent
|
||||||
|
*/
|
||||||
|
export const useChangeEditorContentCallback = () => {
|
||||||
|
const codeMirrorRef = useCodeMirrorReference()
|
||||||
|
return useMemo(() => {
|
||||||
|
if (codeMirrorRef) {
|
||||||
|
return (callback: ContentFormatter) => changeEditorContent(codeMirrorRef, callback)
|
||||||
|
}
|
||||||
|
}, [codeMirrorRef])
|
||||||
|
}
|
||||||
|
|
||||||
|
const convertSelectionToCodeMirrorSelection = (selection: CursorSelection | undefined) => {
|
||||||
|
return Optional.ofNullable(selection)
|
||||||
|
.map<CodeMirrorSelection | undefined>((selection) => ({ anchor: selection.from, head: selection.to }))
|
||||||
|
.orElse(undefined)
|
||||||
|
}
|
|
@ -9,8 +9,15 @@ import type { RenderIframeProps } from '../renderer-pane/render-iframe'
|
||||||
import { RenderIframe } from '../renderer-pane/render-iframe'
|
import { RenderIframe } from '../renderer-pane/render-iframe'
|
||||||
import { useSendFrontmatterInfoFromReduxToRenderer } from '../renderer-pane/hooks/use-send-frontmatter-info-from-redux-to-renderer'
|
import { useSendFrontmatterInfoFromReduxToRenderer } from '../renderer-pane/hooks/use-send-frontmatter-info-from-redux-to-renderer'
|
||||||
import { useTrimmedNoteMarkdownContentWithoutFrontmatter } from '../../../hooks/common/use-trimmed-note-markdown-content-without-frontmatter'
|
import { useTrimmedNoteMarkdownContentWithoutFrontmatter } from '../../../hooks/common/use-trimmed-note-markdown-content-without-frontmatter'
|
||||||
|
import { NoteType } from '../../../redux/note-details/types/note-details'
|
||||||
|
import { useApplicationState } from '../../../hooks/common/use-application-state'
|
||||||
|
import { RendererType } from '../../render-page/window-post-message-communicator/rendering-message'
|
||||||
|
import { useSetCheckboxInEditor } from './hooks/use-set-checkbox-in-editor'
|
||||||
|
|
||||||
export type EditorDocumentRendererProps = Omit<RenderIframeProps, 'markdownContentLines'>
|
export type EditorDocumentRendererProps = Omit<
|
||||||
|
RenderIframeProps,
|
||||||
|
'markdownContentLines' | 'rendererType' | 'onTaskCheckedChange'
|
||||||
|
>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Renders the markdown content from the global application state with the iframe renderer.
|
* Renders the markdown content from the global application state with the iframe renderer.
|
||||||
|
@ -20,6 +27,15 @@ export type EditorDocumentRendererProps = Omit<RenderIframeProps, 'markdownConte
|
||||||
export const EditorDocumentRenderer: React.FC<EditorDocumentRendererProps> = (props) => {
|
export const EditorDocumentRenderer: React.FC<EditorDocumentRendererProps> = (props) => {
|
||||||
useSendFrontmatterInfoFromReduxToRenderer()
|
useSendFrontmatterInfoFromReduxToRenderer()
|
||||||
const trimmedContentLines = useTrimmedNoteMarkdownContentWithoutFrontmatter()
|
const trimmedContentLines = useTrimmedNoteMarkdownContentWithoutFrontmatter()
|
||||||
|
const noteType: NoteType = useApplicationState((state) => state.noteDetails.frontmatter.type)
|
||||||
|
const setCheckboxInEditor = useSetCheckboxInEditor()
|
||||||
|
|
||||||
return <RenderIframe {...props} markdownContentLines={trimmedContentLines} />
|
return (
|
||||||
|
<RenderIframe
|
||||||
|
{...props}
|
||||||
|
onTaskCheckedChange={setCheckboxInEditor}
|
||||||
|
rendererType={noteType === NoteType.SLIDE ? RendererType.SLIDESHOW : RendererType.DOCUMENT}
|
||||||
|
markdownContentLines={trimmedContentLines}
|
||||||
|
/>
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,72 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { useChangeEditorContentCallback } from '../../change-content-context/use-change-editor-content-callback'
|
||||||
|
import { useCallback } from 'react'
|
||||||
|
import type { ContentEdits } from '../../editor-pane/tool-bar/formatters/types/changes'
|
||||||
|
import Optional from 'optional-js'
|
||||||
|
|
||||||
|
const TASK_REGEX = /(\s*(?:[-*+]|\d+[.)]) )(\[[ xX]?])/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a callback that changes the state of a checkbox in a given line in the current codemirror instance.
|
||||||
|
*/
|
||||||
|
export const useSetCheckboxInEditor = () => {
|
||||||
|
const changeEditorContent = useChangeEditorContentCallback()
|
||||||
|
|
||||||
|
return useCallback(
|
||||||
|
(changedLineIndex: number, checkboxChecked: boolean): void => {
|
||||||
|
changeEditorContent?.(({ markdownContent }) => {
|
||||||
|
const lines = markdownContent.split('\n')
|
||||||
|
const lineStartIndex = findStartIndexOfLine(lines, changedLineIndex)
|
||||||
|
const edits = Optional.ofNullable(TASK_REGEX.exec(lines[changedLineIndex]))
|
||||||
|
.map<ContentEdits>(([, beforeCheckbox, oldCheckbox]) => {
|
||||||
|
const checkboxStartIndex = lineStartIndex + beforeCheckbox.length
|
||||||
|
return createCheckboxContentEdit(checkboxStartIndex, oldCheckbox, checkboxChecked)
|
||||||
|
})
|
||||||
|
.orElse([])
|
||||||
|
return [edits, undefined]
|
||||||
|
})
|
||||||
|
},
|
||||||
|
[changeEditorContent]
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Finds the start position of the wanted line index if the given lines would be concat with new-line-characters.
|
||||||
|
*
|
||||||
|
* @param lines The lines to search through
|
||||||
|
* @param wantedLineIndex The index of the line whose start position should be found
|
||||||
|
* @return the found start position
|
||||||
|
*/
|
||||||
|
const findStartIndexOfLine = (lines: string[], wantedLineIndex: number): number => {
|
||||||
|
return lines
|
||||||
|
.map((value) => value.length)
|
||||||
|
.filter((value, index) => index < wantedLineIndex)
|
||||||
|
.reduce((state, lineLength) => state + lineLength + 1, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a {@link ContentEdits content edit} for the change of a checkbox at a given position.
|
||||||
|
*
|
||||||
|
* @param checkboxStartIndex The start index of the checkbox
|
||||||
|
* @param oldCheckbox The old checkbox that should be replaced
|
||||||
|
* @param newCheckboxState The new status of the checkbox
|
||||||
|
* @return the created {@link ContentEdits edit}
|
||||||
|
*/
|
||||||
|
const createCheckboxContentEdit = (
|
||||||
|
checkboxStartIndex: number,
|
||||||
|
oldCheckbox: string,
|
||||||
|
newCheckboxState: boolean
|
||||||
|
): ContentEdits => {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
from: checkboxStartIndex,
|
||||||
|
to: checkboxStartIndex + oldCheckbox.length,
|
||||||
|
insert: `[${newCheckboxState ? 'x' : ' '}]`
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
|
@ -4,10 +4,10 @@
|
||||||
* SPDX-License-Identifier: AGPL-3.0-only
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import React, { Fragment, useCallback, useEffect, useMemo, useRef, useState } from 'react'
|
import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react'
|
||||||
import { useTranslation } from 'react-i18next'
|
import { useTranslation } from 'react-i18next'
|
||||||
import { useApplyDarkMode } from '../../hooks/common/use-apply-dark-mode'
|
import { useApplyDarkMode } from '../../hooks/common/use-apply-dark-mode'
|
||||||
import { setCheckboxInMarkdownContent, updateNoteTitleByFirstHeading } from '../../redux/note-details/methods'
|
import { updateNoteTitleByFirstHeading } from '../../redux/note-details/methods'
|
||||||
import { MotdModal } from '../common/motd-modal/motd-modal'
|
import { MotdModal } from '../common/motd-modal/motd-modal'
|
||||||
import { AppBar, AppBarMode } from './app-bar/app-bar'
|
import { AppBar, AppBarMode } from './app-bar/app-bar'
|
||||||
import { EditorMode } from './app-bar/editor-view-mode'
|
import { EditorMode } from './app-bar/editor-view-mode'
|
||||||
|
@ -15,17 +15,16 @@ import { useViewModeShortcuts } from './hooks/useViewModeShortcuts'
|
||||||
import { Sidebar } from './sidebar/sidebar'
|
import { Sidebar } from './sidebar/sidebar'
|
||||||
import { Splitter } from './splitter/splitter'
|
import { Splitter } from './splitter/splitter'
|
||||||
import type { DualScrollState, ScrollState } from './synced-scroll/scroll-props'
|
import type { DualScrollState, ScrollState } from './synced-scroll/scroll-props'
|
||||||
import { RendererType } from '../render-page/window-post-message-communicator/rendering-message'
|
|
||||||
import { useEditorModeFromUrl } from './hooks/useEditorModeFromUrl'
|
import { useEditorModeFromUrl } from './hooks/useEditorModeFromUrl'
|
||||||
import { UiNotifications } from '../notifications/ui-notifications'
|
import { UiNotifications } from '../notifications/ui-notifications'
|
||||||
import { useUpdateLocalHistoryEntry } from './hooks/useUpdateLocalHistoryEntry'
|
import { useUpdateLocalHistoryEntry } from './hooks/useUpdateLocalHistoryEntry'
|
||||||
import { useApplicationState } from '../../hooks/common/use-application-state'
|
import { useApplicationState } from '../../hooks/common/use-application-state'
|
||||||
import { EditorDocumentRenderer } from './editor-document-renderer/editor-document-renderer'
|
import { EditorDocumentRenderer } from './editor-document-renderer/editor-document-renderer'
|
||||||
import { Logger } from '../../utils/logger'
|
import { Logger } from '../../utils/logger'
|
||||||
import { NoteType } from '../../redux/note-details/types/note-details'
|
|
||||||
import { NoteAndAppTitleHead } from '../layout/note-and-app-title-head'
|
import { NoteAndAppTitleHead } from '../layout/note-and-app-title-head'
|
||||||
import equal from 'fast-deep-equal'
|
import equal from 'fast-deep-equal'
|
||||||
import { EditorPane } from './editor-pane/editor-pane'
|
import { EditorPane } from './editor-pane/editor-pane'
|
||||||
|
import { ChangeEditorContentContextProvider } from './change-content-context/change-content-context'
|
||||||
|
|
||||||
export enum ScrollSource {
|
export enum ScrollSource {
|
||||||
EDITOR = 'editor',
|
EDITOR = 'editor',
|
||||||
|
@ -112,7 +111,6 @@ export const EditorPageContent: React.FC = () => {
|
||||||
),
|
),
|
||||||
[onEditorScroll, scrollState.editorScrollState, setEditorToScrollSource]
|
[onEditorScroll, scrollState.editorScrollState, setEditorToScrollSource]
|
||||||
)
|
)
|
||||||
const noteType: NoteType = useApplicationState((state) => state.noteDetails.frontmatter.type)
|
|
||||||
|
|
||||||
const rightPane = useMemo(
|
const rightPane = useMemo(
|
||||||
() => (
|
() => (
|
||||||
|
@ -120,17 +118,15 @@ export const EditorPageContent: React.FC = () => {
|
||||||
frameClasses={'h-100 w-100'}
|
frameClasses={'h-100 w-100'}
|
||||||
onMakeScrollSource={setRendererToScrollSource}
|
onMakeScrollSource={setRendererToScrollSource}
|
||||||
onFirstHeadingChange={updateNoteTitleByFirstHeading}
|
onFirstHeadingChange={updateNoteTitleByFirstHeading}
|
||||||
onTaskCheckedChange={setCheckboxInMarkdownContent}
|
|
||||||
onScroll={onMarkdownRendererScroll}
|
onScroll={onMarkdownRendererScroll}
|
||||||
scrollState={scrollState.rendererScrollState}
|
scrollState={scrollState.rendererScrollState}
|
||||||
rendererType={noteType === NoteType.SLIDE ? RendererType.SLIDESHOW : RendererType.DOCUMENT}
|
|
||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
[noteType, onMarkdownRendererScroll, scrollState.rendererScrollState, setRendererToScrollSource]
|
[onMarkdownRendererScroll, scrollState.rendererScrollState, setRendererToScrollSource]
|
||||||
)
|
)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Fragment>
|
<ChangeEditorContentContextProvider>
|
||||||
<NoteAndAppTitleHead />
|
<NoteAndAppTitleHead />
|
||||||
<UiNotifications />
|
<UiNotifications />
|
||||||
<MotdModal />
|
<MotdModal />
|
||||||
|
@ -147,6 +143,6 @@ export const EditorPageContent: React.FC = () => {
|
||||||
<Sidebar />
|
<Sidebar />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Fragment>
|
</ChangeEditorContentContextProvider>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
* SPDX-License-Identifier: AGPL-3.0-only
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import React, { useCallback, useMemo, useRef } from 'react'
|
import React, { useCallback, useMemo } from 'react'
|
||||||
import type { ScrollProps } from '../synced-scroll/scroll-props'
|
import type { ScrollProps } from '../synced-scroll/scroll-props'
|
||||||
import { StatusBar } from './status-bar/status-bar'
|
import { StatusBar } from './status-bar/status-bar'
|
||||||
import { ToolBar } from './tool-bar/tool-bar'
|
import { ToolBar } from './tool-bar/tool-bar'
|
||||||
|
@ -12,54 +12,50 @@ import { useApplicationState } from '../../../hooks/common/use-application-state
|
||||||
import { setNoteContent } from '../../../redux/note-details/methods'
|
import { setNoteContent } from '../../../redux/note-details/methods'
|
||||||
import { useNoteMarkdownContent } from '../../../hooks/common/use-note-markdown-content'
|
import { useNoteMarkdownContent } from '../../../hooks/common/use-note-markdown-content'
|
||||||
import { MaxLengthWarning } from './max-length-warning/max-length-warning'
|
import { MaxLengthWarning } from './max-length-warning/max-length-warning'
|
||||||
import { useOnImageUploadFromRenderer } from './hooks/use-on-image-upload-from-renderer'
|
|
||||||
import type { ReactCodeMirrorRef } from '@uiw/react-codemirror'
|
|
||||||
import ReactCodeMirror from '@uiw/react-codemirror'
|
import ReactCodeMirror from '@uiw/react-codemirror'
|
||||||
import { useCursorActivityCallback } from './hooks/use-cursor-activity-callback'
|
|
||||||
import { useApplyScrollState } from './hooks/use-apply-scroll-state'
|
import { useApplyScrollState } from './hooks/use-apply-scroll-state'
|
||||||
import styles from './extended-codemirror/codemirror.module.scss'
|
import styles from './extended-codemirror/codemirror.module.scss'
|
||||||
import { oneDark } from '@codemirror/theme-one-dark'
|
import { oneDark } from '@codemirror/theme-one-dark'
|
||||||
import { useTranslation } from 'react-i18next'
|
import { useTranslation } from 'react-i18next'
|
||||||
import { Logger } from '../../../utils/logger'
|
|
||||||
import { useCodeMirrorScrollWatchExtension } from './hooks/code-mirror-extensions/use-code-mirror-scroll-watch-extension'
|
import { useCodeMirrorScrollWatchExtension } from './hooks/code-mirror-extensions/use-code-mirror-scroll-watch-extension'
|
||||||
import { useCodeMirrorPasteExtension } from './hooks/code-mirror-extensions/use-code-mirror-paste-extension'
|
import { useCodeMirrorFileInsertExtension } from './hooks/code-mirror-extensions/use-code-mirror-file-insert-extension'
|
||||||
import { useCodeMirrorFileDropExtension } from './hooks/code-mirror-extensions/use-code-mirror-file-drop-extension'
|
|
||||||
import { markdown, markdownLanguage } from '@codemirror/lang-markdown'
|
import { markdown, markdownLanguage } from '@codemirror/lang-markdown'
|
||||||
import { EditorView } from '@codemirror/view'
|
import { EditorView } from '@codemirror/view'
|
||||||
import { autocompletion } from '@codemirror/autocomplete'
|
import { autocompletion } from '@codemirror/autocomplete'
|
||||||
import { useCodeMirrorFocusReference } from './hooks/use-code-mirror-focus-reference'
|
import { cypressAttribute, cypressId } from '../../../utils/cypress-attribute'
|
||||||
import { useOffScreenScrollProtection } from './hooks/use-off-screen-scroll-protection'
|
|
||||||
import { cypressId } from '../../../utils/cypress-attribute'
|
|
||||||
import { findLanguageByCodeBlockName } from '../../markdown-renderer/markdown-extension/code-block-markdown-extension/find-language-by-code-block-name'
|
import { findLanguageByCodeBlockName } from '../../markdown-renderer/markdown-extension/code-block-markdown-extension/find-language-by-code-block-name'
|
||||||
import { languages } from '@codemirror/language-data'
|
import { languages } from '@codemirror/language-data'
|
||||||
|
import { useCursorActivityCallback } from './hooks/use-cursor-activity-callback'
|
||||||
const logger = new Logger('EditorPane')
|
import { useCodeMirrorReference, useSetCodeMirrorReference } from '../change-content-context/change-content-context'
|
||||||
|
import { useCodeMirrorTablePasteExtension } from './hooks/table-paste/use-code-mirror-table-paste-extension'
|
||||||
|
import { useOnImageUploadFromRenderer } from './hooks/image-upload-from-renderer/use-on-image-upload-from-renderer'
|
||||||
|
|
||||||
export const EditorPane: React.FC<ScrollProps> = ({ scrollState, onScroll, onMakeScrollSource }) => {
|
export const EditorPane: React.FC<ScrollProps> = ({ scrollState, onScroll, onMakeScrollSource }) => {
|
||||||
const markdownContent = useNoteMarkdownContent()
|
const markdownContent = useNoteMarkdownContent()
|
||||||
|
|
||||||
const ligaturesEnabled = useApplicationState((state) => state.editorConfig.ligatures)
|
const ligaturesEnabled = useApplicationState((state) => state.editorConfig.ligatures)
|
||||||
const codeMirrorRef = useRef<ReactCodeMirrorRef | null>(null)
|
|
||||||
|
|
||||||
useApplyScrollState(codeMirrorRef, scrollState)
|
useApplyScrollState(scrollState)
|
||||||
|
|
||||||
const editorScrollExtension = useCodeMirrorScrollWatchExtension(onScroll)
|
const editorScrollExtension = useCodeMirrorScrollWatchExtension(onScroll)
|
||||||
const editorPasteExtension = useCodeMirrorPasteExtension()
|
const tablePasteExtensions = useCodeMirrorTablePasteExtension()
|
||||||
const dropExtension = useCodeMirrorFileDropExtension()
|
const fileInsertExtension = useCodeMirrorFileInsertExtension()
|
||||||
const [focusExtension, editorFocused] = useCodeMirrorFocusReference()
|
const cursorActivityExtension = useCursorActivityCallback()
|
||||||
const saveOffFocusScrollStateExtensions = useOffScreenScrollProtection()
|
|
||||||
const cursorActivityExtension = useCursorActivityCallback(editorFocused)
|
|
||||||
|
|
||||||
const onBeforeChange = useCallback(
|
const onBeforeChange = useCallback((value: string): void => {
|
||||||
(value: string): void => {
|
|
||||||
if (!editorFocused.current) {
|
|
||||||
logger.debug("Don't post content change because editor isn't focused")
|
|
||||||
} else {
|
|
||||||
setNoteContent(value)
|
setNoteContent(value)
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const codeMirrorRef = useCodeMirrorReference()
|
||||||
|
const setCodeMirrorReference = useSetCodeMirrorReference()
|
||||||
|
|
||||||
|
const updateViewContext = useMemo(() => {
|
||||||
|
return EditorView.updateListener.of((update) => {
|
||||||
|
if (codeMirrorRef !== update.view) {
|
||||||
|
setCodeMirrorReference(update.view)
|
||||||
}
|
}
|
||||||
},
|
})
|
||||||
[editorFocused]
|
}, [codeMirrorRef, setCodeMirrorReference])
|
||||||
)
|
|
||||||
|
|
||||||
const extensions = useMemo(
|
const extensions = useMemo(
|
||||||
() => [
|
() => [
|
||||||
|
@ -67,23 +63,15 @@ export const EditorPane: React.FC<ScrollProps> = ({ scrollState, onScroll, onMak
|
||||||
base: markdownLanguage,
|
base: markdownLanguage,
|
||||||
codeLanguages: (input) => findLanguageByCodeBlockName(languages, input)
|
codeLanguages: (input) => findLanguageByCodeBlockName(languages, input)
|
||||||
}),
|
}),
|
||||||
...saveOffFocusScrollStateExtensions,
|
|
||||||
focusExtension,
|
|
||||||
EditorView.lineWrapping,
|
EditorView.lineWrapping,
|
||||||
editorScrollExtension,
|
editorScrollExtension,
|
||||||
editorPasteExtension,
|
tablePasteExtensions,
|
||||||
dropExtension,
|
fileInsertExtension,
|
||||||
autocompletion(),
|
autocompletion(),
|
||||||
cursorActivityExtension
|
|
||||||
],
|
|
||||||
[
|
|
||||||
cursorActivityExtension,
|
cursorActivityExtension,
|
||||||
dropExtension,
|
updateViewContext
|
||||||
editorPasteExtension,
|
],
|
||||||
editorScrollExtension,
|
[cursorActivityExtension, fileInsertExtension, tablePasteExtensions, editorScrollExtension, updateViewContext]
|
||||||
focusExtension,
|
|
||||||
saveOffFocusScrollStateExtensions
|
|
||||||
]
|
|
||||||
)
|
)
|
||||||
|
|
||||||
useOnImageUploadFromRenderer()
|
useOnImageUploadFromRenderer()
|
||||||
|
@ -100,7 +88,8 @@ export const EditorPane: React.FC<ScrollProps> = ({ scrollState, onScroll, onMak
|
||||||
className={`d-flex flex-column h-100 position-relative`}
|
className={`d-flex flex-column h-100 position-relative`}
|
||||||
onTouchStart={onMakeScrollSource}
|
onTouchStart={onMakeScrollSource}
|
||||||
onMouseEnter={onMakeScrollSource}
|
onMouseEnter={onMakeScrollSource}
|
||||||
{...cypressId('editor-pane')}>
|
{...cypressId('editor-pane')}
|
||||||
|
{...cypressAttribute('editor-ready', String(codeMirrorRef !== undefined))}>
|
||||||
<MaxLengthWarning />
|
<MaxLengthWarning />
|
||||||
<ToolBar />
|
<ToolBar />
|
||||||
<ReactCodeMirror
|
<ReactCodeMirror
|
||||||
|
@ -115,7 +104,6 @@ export const EditorPane: React.FC<ScrollProps> = ({ scrollState, onScroll, onMak
|
||||||
theme={oneDark}
|
theme={oneDark}
|
||||||
value={markdownContent}
|
value={markdownContent}
|
||||||
onChange={onBeforeChange}
|
onChange={onBeforeChange}
|
||||||
ref={codeMirrorRef}
|
|
||||||
/>
|
/>
|
||||||
<StatusBar />
|
<StatusBar />
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,42 +0,0 @@
|
||||||
/*
|
|
||||||
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
|
||||||
*
|
|
||||||
* SPDX-License-Identifier: AGPL-3.0-only
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { useCallback, useMemo } from 'react'
|
|
||||||
import { handleUpload } from '../../upload-handler'
|
|
||||||
import { EditorView } from '@codemirror/view'
|
|
||||||
import type { Extension } from '@codemirror/state'
|
|
||||||
import Optional from 'optional-js'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a callback that is used to process file drops on the code mirror editor
|
|
||||||
*
|
|
||||||
* @return the code mirror callback
|
|
||||||
*/
|
|
||||||
export const useCodeMirrorFileDropExtension = (): Extension => {
|
|
||||||
const onDrop = useCallback((event: DragEvent, view: EditorView): void => {
|
|
||||||
if (!event.pageX || !event.pageY) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
Optional.ofNullable(event.dataTransfer?.files)
|
|
||||||
.filter((files) => files.length > 0)
|
|
||||||
.ifPresent((files) => {
|
|
||||||
event.preventDefault()
|
|
||||||
const newCursor = view.posAtCoords({ y: event.pageY, x: event.pageX })
|
|
||||||
if (newCursor === null) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
handleUpload(files[0], { from: newCursor })
|
|
||||||
})
|
|
||||||
}, [])
|
|
||||||
|
|
||||||
return useMemo(
|
|
||||||
() =>
|
|
||||||
EditorView.domEventHandlers({
|
|
||||||
drop: onDrop
|
|
||||||
}),
|
|
||||||
[onDrop]
|
|
||||||
)
|
|
||||||
}
|
|
|
@ -0,0 +1,51 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { useMemo } from 'react'
|
||||||
|
import { EditorView } from '@codemirror/view'
|
||||||
|
import type { Extension } from '@codemirror/state'
|
||||||
|
import { handleUpload } from '../use-handle-upload'
|
||||||
|
import Optional from 'optional-js'
|
||||||
|
import type { CursorSelection } from '../../tool-bar/formatters/types/cursor-selection'
|
||||||
|
|
||||||
|
const calculateCursorPositionInEditor = (view: EditorView, event: MouseEvent): number => {
|
||||||
|
return Optional.ofNullable(event.pageX)
|
||||||
|
.flatMap((posX) => {
|
||||||
|
return Optional.ofNullable(event.pageY).map((posY) => {
|
||||||
|
return view.posAtCoords({ x: posX, y: posY })
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.orElse(view.state.selection.main.head)
|
||||||
|
}
|
||||||
|
|
||||||
|
const processFileList = (view: EditorView, fileList?: FileList, cursorSelection?: CursorSelection): boolean => {
|
||||||
|
return Optional.ofNullable(fileList)
|
||||||
|
.filter((files) => files.length > 0)
|
||||||
|
.map((files) => {
|
||||||
|
handleUpload(view, files[0], cursorSelection)
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
.orElse(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a callback that is used to process file drops and pastes on the code mirror editor
|
||||||
|
*
|
||||||
|
* @return the code mirror callback
|
||||||
|
*/
|
||||||
|
export const useCodeMirrorFileInsertExtension = (): Extension => {
|
||||||
|
return useMemo(() => {
|
||||||
|
return EditorView.domEventHandlers({
|
||||||
|
drop: (event, view) => {
|
||||||
|
processFileList(view, event.dataTransfer?.files, { from: calculateCursorPositionInEditor(view, event) }) &&
|
||||||
|
event.preventDefault()
|
||||||
|
},
|
||||||
|
paste: (event, view) => {
|
||||||
|
processFileList(view, event.clipboardData?.files) && event.preventDefault()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, [])
|
||||||
|
}
|
|
@ -1,34 +0,0 @@
|
||||||
/*
|
|
||||||
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
|
||||||
*
|
|
||||||
* SPDX-License-Identifier: AGPL-3.0-only
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { useMemo } from 'react'
|
|
||||||
import { handleFilePaste, handleTablePaste } from '../../tool-bar/utils/pasteHandlers'
|
|
||||||
import { EditorView } from '@codemirror/view'
|
|
||||||
import type { Extension } from '@codemirror/state'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a {@link Extension code mirror extension} that handles the table or file paste action.
|
|
||||||
*
|
|
||||||
* @return the created {@link Extension code mirror extension}
|
|
||||||
*/
|
|
||||||
export const useCodeMirrorPasteExtension = (): Extension => {
|
|
||||||
return useMemo(
|
|
||||||
() =>
|
|
||||||
EditorView.domEventHandlers({
|
|
||||||
paste: (event: ClipboardEvent) => {
|
|
||||||
const clipboardData = event.clipboardData
|
|
||||||
if (!clipboardData) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if (handleTablePaste(clipboardData) || handleFilePaste(clipboardData)) {
|
|
||||||
event.preventDefault()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
[]
|
|
||||||
)
|
|
||||||
}
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
*
|
*
|
||||||
* SPDX-License-Identifier: AGPL-3.0-only
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
*/
|
*/
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
*
|
*
|
||||||
* SPDX-License-Identifier: AGPL-3.0-only
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
*/
|
*/
|
|
@ -1,19 +1,19 @@
|
||||||
/*
|
/*
|
||||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
*
|
*
|
||||||
* SPDX-License-Identifier: AGPL-3.0-only
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { useEditorReceiveHandler } from '../../../render-page/window-post-message-communicator/hooks/use-editor-receive-handler'
|
import { useEditorReceiveHandler } from '../../../../render-page/window-post-message-communicator/hooks/use-editor-receive-handler'
|
||||||
import type { ImageUploadMessage } from '../../../render-page/window-post-message-communicator/rendering-message'
|
import type { ImageUploadMessage } from '../../../../render-page/window-post-message-communicator/rendering-message'
|
||||||
import { CommunicationMessageType } from '../../../render-page/window-post-message-communicator/rendering-message'
|
import { CommunicationMessageType } from '../../../../render-page/window-post-message-communicator/rendering-message'
|
||||||
import { useCallback } from 'react'
|
import { useCallback } from 'react'
|
||||||
import { getGlobalState } from '../../../../redux'
|
import { getGlobalState } from '../../../../../redux'
|
||||||
import { handleUpload } from '../upload-handler'
|
import { Logger } from '../../../../../utils/logger'
|
||||||
import { Logger } from '../../../../utils/logger'
|
import { findRegexMatchInText } from './find-regex-match-in-text'
|
||||||
import { findRegexMatchInText } from '../find-regex-match-in-text'
|
|
||||||
import Optional from 'optional-js'
|
import Optional from 'optional-js'
|
||||||
import type { CursorSelection } from '../../../../redux/editor/types'
|
import { useHandleUpload } from '../use-handle-upload'
|
||||||
|
import type { CursorSelection } from '../../tool-bar/formatters/types/cursor-selection'
|
||||||
|
|
||||||
const log = new Logger('useOnImageUpload')
|
const log = new Logger('useOnImageUpload')
|
||||||
const imageWithPlaceholderLinkRegex = /!\[([^\]]*)]\(https:\/\/([^)]*)\)/g
|
const imageWithPlaceholderLinkRegex = /!\[([^\]]*)]\(https:\/\/([^)]*)\)/g
|
||||||
|
@ -22,9 +22,12 @@ const imageWithPlaceholderLinkRegex = /!\[([^\]]*)]\(https:\/\/([^)]*)\)/g
|
||||||
* Receives {@link CommunicationMessageType.IMAGE_UPLOAD image upload events} via iframe communication and processes the attached uploads.
|
* Receives {@link CommunicationMessageType.IMAGE_UPLOAD image upload events} via iframe communication and processes the attached uploads.
|
||||||
*/
|
*/
|
||||||
export const useOnImageUploadFromRenderer = (): void => {
|
export const useOnImageUploadFromRenderer = (): void => {
|
||||||
|
const handleUpload = useHandleUpload()
|
||||||
|
|
||||||
useEditorReceiveHandler(
|
useEditorReceiveHandler(
|
||||||
CommunicationMessageType.IMAGE_UPLOAD,
|
CommunicationMessageType.IMAGE_UPLOAD,
|
||||||
useCallback((values: ImageUploadMessage) => {
|
useCallback(
|
||||||
|
(values: ImageUploadMessage) => {
|
||||||
const { dataUri, fileName, lineIndex, placeholderIndexInLine } = values
|
const { dataUri, fileName, lineIndex, placeholderIndexInLine } = values
|
||||||
if (!dataUri.startsWith('')
|
||||||
|
})
|
||||||
|
})
|
40
src/utils/read-file.ts
Normal file
40
src/utils/read-file.ts
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: 2022 The HedgeDoc developers (see AUTHORS file)
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
export enum FileContentFormat {
|
||||||
|
TEXT,
|
||||||
|
DATA_URL
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the given {@link File}.
|
||||||
|
*
|
||||||
|
* @param file The file to read
|
||||||
|
* @param fileReaderMode Defines as what the file content should be formatted.
|
||||||
|
* @throws Error if an invalid read mode was given or if the file couldn't be read.
|
||||||
|
* @return the file content
|
||||||
|
*/
|
||||||
|
export const readFile = async (file: Blob, fileReaderMode: FileContentFormat): Promise<string> => {
|
||||||
|
return new Promise<string>((resolve, reject) => {
|
||||||
|
const fileReader = new FileReader()
|
||||||
|
fileReader.addEventListener('load', () => {
|
||||||
|
resolve(fileReader.result as string)
|
||||||
|
})
|
||||||
|
fileReader.addEventListener('error', (error) => {
|
||||||
|
reject(error)
|
||||||
|
})
|
||||||
|
switch (fileReaderMode) {
|
||||||
|
case FileContentFormat.DATA_URL:
|
||||||
|
fileReader.readAsDataURL(file)
|
||||||
|
break
|
||||||
|
case FileContentFormat.TEXT:
|
||||||
|
fileReader.readAsText(file)
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
throw new Error('Unknown file reader mode')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue