mirror of
https://github.com/hedgedoc/hedgedoc.git
synced 2025-04-04 19:27:01 +00:00
Restructure the max-length-warning and the status bar (#1654)
Signed-off-by: Tilman Vatteroth <git@tilmanvatteroth.de>
This commit is contained in:
parent
7182f22e52
commit
4023acc9d6
10 changed files with 291 additions and 92 deletions
|
@ -5,14 +5,12 @@
|
|||
*/
|
||||
|
||||
import type { Editor, EditorChange } from 'codemirror'
|
||||
import React, { useCallback, useRef, useState } from 'react'
|
||||
import React, { useCallback, useState } from 'react'
|
||||
import { Controlled as ControlledCodeMirror } from 'react-codemirror2'
|
||||
import { MaxLengthWarningModal } from '../editor-modals/max-length-warning-modal'
|
||||
import type { ScrollProps } from '../synced-scroll/scroll-props'
|
||||
import { allHinters, findWordAtCursor } from './autocompletion'
|
||||
import './editor-pane.scss'
|
||||
import type { StatusBarInfo } from './status-bar/status-bar'
|
||||
import { createStatusInfo, defaultState, StatusBar } from './status-bar/status-bar'
|
||||
import { StatusBar } from './status-bar/status-bar'
|
||||
import { ToolBar } from './tool-bar/tool-bar'
|
||||
import { useApplicationState } from '../../../hooks/common/use-application-state'
|
||||
import './codemirror-imports'
|
||||
|
@ -23,6 +21,8 @@ import { useOnEditorPasteCallback } from './hooks/use-on-editor-paste-callback'
|
|||
import { useOnEditorFileDrop } from './hooks/use-on-editor-file-drop'
|
||||
import { useOnEditorScroll } from './hooks/use-on-editor-scroll'
|
||||
import { useApplyScrollState } from './hooks/use-apply-scroll-state'
|
||||
import { MaxLengthWarning } from './max-length-warning/max-length-warning'
|
||||
import { useCreateStatusBarInfo } from './hooks/use-create-status-bar-info'
|
||||
|
||||
const onChange = (editor: Editor) => {
|
||||
for (const hinter of allHinters) {
|
||||
|
@ -41,53 +41,34 @@ const onChange = (editor: Editor) => {
|
|||
|
||||
export const EditorPane: React.FC<ScrollProps> = ({ scrollState, onScroll, onMakeScrollSource }) => {
|
||||
const markdownContent = useNoteMarkdownContent()
|
||||
const maxLength = useApplicationState((state) => state.config.maxDocumentLength)
|
||||
const [showMaxLengthWarning, setShowMaxLengthWarning] = useState(false)
|
||||
const maxLengthWarningAlreadyShown = useRef(false)
|
||||
|
||||
const [editor, setEditor] = useState<Editor>()
|
||||
const [statusBarInfo, setStatusBarInfo] = useState<StatusBarInfo>(defaultState)
|
||||
const ligaturesEnabled = useApplicationState((state) => state.editorConfig.ligatures)
|
||||
|
||||
const onPaste = useOnEditorPasteCallback()
|
||||
const onEditorScroll = useOnEditorScroll(onScroll)
|
||||
useApplyScrollState(editor, scrollState)
|
||||
|
||||
const onBeforeChange = useCallback(
|
||||
(editor: Editor, data: EditorChange, value: string) => {
|
||||
if (value.length > maxLength && !maxLengthWarningAlreadyShown.current) {
|
||||
setShowMaxLengthWarning(true)
|
||||
maxLengthWarningAlreadyShown.current = true
|
||||
}
|
||||
if (value.length <= maxLength) {
|
||||
maxLengthWarningAlreadyShown.current = false
|
||||
}
|
||||
setNoteContent(value)
|
||||
},
|
||||
[maxLength]
|
||||
)
|
||||
const onBeforeChange = useCallback((editor: Editor, data: EditorChange, value: string) => {
|
||||
setNoteContent(value)
|
||||
}, [])
|
||||
|
||||
const [statusBarInfo, updateStatusBarInfo] = useCreateStatusBarInfo()
|
||||
|
||||
const onEditorDidMount = useCallback(
|
||||
(mountedEditor: Editor) => {
|
||||
setStatusBarInfo(createStatusInfo(mountedEditor, maxLength))
|
||||
updateStatusBarInfo(mountedEditor)
|
||||
setEditor(mountedEditor)
|
||||
},
|
||||
[maxLength]
|
||||
)
|
||||
|
||||
const onCursorActivity = useCallback(
|
||||
(editorWithActivity: Editor) => {
|
||||
setStatusBarInfo(createStatusInfo(editorWithActivity, maxLength))
|
||||
},
|
||||
[maxLength]
|
||||
[updateStatusBarInfo]
|
||||
)
|
||||
|
||||
const onDrop = useOnEditorFileDrop()
|
||||
const onMaxLengthHide = useCallback(() => setShowMaxLengthWarning(false), [])
|
||||
const codeMirrorOptions = useCodeMirrorOptions()
|
||||
|
||||
return (
|
||||
<div className={'d-flex flex-column h-100 position-relative'} onMouseEnter={onMakeScrollSource}>
|
||||
<MaxLengthWarningModal show={showMaxLengthWarning} onHide={onMaxLengthHide} maxLength={maxLength} />
|
||||
<MaxLengthWarning />
|
||||
<ToolBar editor={editor} />
|
||||
<ControlledCodeMirror
|
||||
className={`overflow-hidden w-100 flex-fill ${ligaturesEnabled ? '' : 'no-ligatures'}`}
|
||||
|
@ -96,12 +77,12 @@ export const EditorPane: React.FC<ScrollProps> = ({ scrollState, onScroll, onMak
|
|||
onChange={onChange}
|
||||
onPaste={onPaste}
|
||||
onDrop={onDrop}
|
||||
onCursorActivity={onCursorActivity}
|
||||
onCursorActivity={updateStatusBarInfo}
|
||||
editorDidMount={onEditorDidMount}
|
||||
onBeforeChange={onBeforeChange}
|
||||
onScroll={onEditorScroll}
|
||||
/>
|
||||
<StatusBar {...statusBarInfo} />
|
||||
<StatusBar statusBarInfo={statusBarInfo} />
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
|
|
@ -0,0 +1,38 @@
|
|||
/*
|
||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
||||
*
|
||||
* SPDX-License-Identifier: AGPL-3.0-only
|
||||
*/
|
||||
|
||||
import type { StatusBarInfo } from '../status-bar/status-bar'
|
||||
import { defaultState } from '../status-bar/status-bar'
|
||||
import type { Editor } from 'codemirror'
|
||||
import { useCallback, useState } from 'react'
|
||||
import { useApplicationState } from '../../../../hooks/common/use-application-state'
|
||||
|
||||
/**
|
||||
* Provides a {@link StatusBarInfo} object and a function that can update this object using a {@link CodeMirror code mirror instance}.
|
||||
*/
|
||||
export const useCreateStatusBarInfo = (): [
|
||||
statusBarInfo: StatusBarInfo,
|
||||
updateStatusBarInfo: (editor: Editor) => void
|
||||
] => {
|
||||
const maxDocumentLength = useApplicationState((state) => state.config.maxDocumentLength)
|
||||
const [statusBarInfo, setStatusBarInfo] = useState(defaultState)
|
||||
|
||||
const updateStatusBarInfo = useCallback(
|
||||
(editor: Editor): void => {
|
||||
setStatusBarInfo({
|
||||
position: editor.getCursor(),
|
||||
charactersInDocument: editor.getValue().length,
|
||||
remainingCharacters: maxDocumentLength - editor.getValue().length,
|
||||
linesInDocument: editor.lineCount(),
|
||||
selectedColumns: editor.getSelection().length,
|
||||
selectedLines: editor.getSelection().split('\n').length
|
||||
})
|
||||
},
|
||||
[maxDocumentLength]
|
||||
)
|
||||
|
||||
return [statusBarInfo, updateStatusBarInfo]
|
||||
}
|
|
@ -7,16 +7,20 @@
|
|||
import React from 'react'
|
||||
import { Button, Modal } from 'react-bootstrap'
|
||||
import { Trans, useTranslation } from 'react-i18next'
|
||||
import type { ModalVisibilityProps } from '../../common/modals/common-modal'
|
||||
import { CommonModal } from '../../common/modals/common-modal'
|
||||
import { cypressId } from '../../../utils/cypress-attribute'
|
||||
import type { ModalVisibilityProps } from '../../../common/modals/common-modal'
|
||||
import { CommonModal } from '../../../common/modals/common-modal'
|
||||
import { cypressId } from '../../../../utils/cypress-attribute'
|
||||
import { useApplicationState } from '../../../../hooks/common/use-application-state'
|
||||
|
||||
export interface MaxLengthWarningModalProps extends ModalVisibilityProps {
|
||||
maxLength: number
|
||||
}
|
||||
|
||||
export const MaxLengthWarningModal: React.FC<MaxLengthWarningModalProps> = ({ show, onHide, maxLength }) => {
|
||||
/**
|
||||
* Shows a modal that informs the user that the document is too long.
|
||||
*
|
||||
* @param show is {@code true} if the modal should be shown
|
||||
* @param onHide gets called if the modal was closed
|
||||
*/
|
||||
export const MaxLengthWarningModal: React.FC<ModalVisibilityProps> = ({ show, onHide }) => {
|
||||
useTranslation()
|
||||
const maxDocumentLength = useApplicationState((state) => state.config.maxDocumentLength)
|
||||
|
||||
return (
|
||||
<CommonModal
|
||||
|
@ -26,7 +30,7 @@ export const MaxLengthWarningModal: React.FC<MaxLengthWarningModalProps> = ({ sh
|
|||
title={'editor.error.limitReached.title'}
|
||||
showCloseButton={true}>
|
||||
<Modal.Body>
|
||||
<Trans i18nKey={'editor.error.limitReached.description'} values={{ maxLength }} />
|
||||
<Trans i18nKey={'editor.error.limitReached.description'} values={{ maxDocumentLength }} />
|
||||
<strong className='mt-2 d-block'>
|
||||
<Trans i18nKey={'editor.error.limitReached.advice'} />
|
||||
</strong>
|
|
@ -0,0 +1,33 @@
|
|||
/*
|
||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
||||
*
|
||||
* SPDX-License-Identifier: AGPL-3.0-only
|
||||
*/
|
||||
|
||||
import React, { useCallback, useEffect, useRef, useState } from 'react'
|
||||
import { MaxLengthWarningModal } from './max-length-warning-modal'
|
||||
import { useNoteMarkdownContent } from '../../../../hooks/common/use-note-markdown-content'
|
||||
import { useApplicationState } from '../../../../hooks/common/use-application-state'
|
||||
|
||||
/**
|
||||
* Watches the length of the document and shows a warning modal to the user if the document length exceeds the configured value.
|
||||
*/
|
||||
export const MaxLengthWarning: React.FC = () => {
|
||||
const [showMaxLengthWarningModal, setShowMaxLengthWarningModal] = useState(false)
|
||||
const maxLengthWarningAlreadyShown = useRef(false)
|
||||
const maxDocumentLength = useApplicationState((state) => state.config.maxDocumentLength)
|
||||
const hideWarning = useCallback(() => setShowMaxLengthWarningModal(false), [])
|
||||
const markdownContent = useNoteMarkdownContent()
|
||||
|
||||
useEffect(() => {
|
||||
if (markdownContent.length > maxDocumentLength && !maxLengthWarningAlreadyShown.current) {
|
||||
setShowMaxLengthWarningModal(true)
|
||||
maxLengthWarningAlreadyShown.current = true
|
||||
}
|
||||
if (markdownContent.length <= maxDocumentLength) {
|
||||
maxLengthWarningAlreadyShown.current = false
|
||||
}
|
||||
}, [markdownContent, maxDocumentLength])
|
||||
|
||||
return <MaxLengthWarningModal show={showMaxLengthWarningModal} onHide={hideWarning} />
|
||||
}
|
|
@ -0,0 +1,34 @@
|
|||
/*
|
||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
||||
*
|
||||
* SPDX-License-Identifier: AGPL-3.0-only
|
||||
*/
|
||||
|
||||
import React, { useMemo } from 'react'
|
||||
import { Trans } from 'react-i18next'
|
||||
import type { Position } from 'codemirror'
|
||||
|
||||
export interface CursorPositionInfoProps {
|
||||
cursorPosition: Position
|
||||
}
|
||||
|
||||
/**
|
||||
* Renders a translated text that shows the given cursor position.
|
||||
*
|
||||
* @param cursorPosition The cursor position that should be included
|
||||
*/
|
||||
export const CursorPositionInfo: React.FC<CursorPositionInfoProps> = ({ cursorPosition }) => {
|
||||
const translationOptions = useMemo(
|
||||
() => ({
|
||||
line: cursorPosition.line + 1,
|
||||
columns: cursorPosition.ch + 1
|
||||
}),
|
||||
[cursorPosition.ch, cursorPosition.line]
|
||||
)
|
||||
|
||||
return (
|
||||
<span>
|
||||
<Trans i18nKey={'editor.statusBar.cursor'} values={translationOptions} />
|
||||
</span>
|
||||
)
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
/*
|
||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
||||
*
|
||||
* SPDX-License-Identifier: AGPL-3.0-only
|
||||
*/
|
||||
|
||||
import React, { useMemo } from 'react'
|
||||
import { Trans, useTranslation } from 'react-i18next'
|
||||
|
||||
export interface LinesInDocumentInfoProps {
|
||||
numberOfLinesInDocument: number
|
||||
}
|
||||
|
||||
/**
|
||||
* Renders a translated text that shows the number of lines in the document.
|
||||
*
|
||||
* @param linesInDocument The number of lines in the document
|
||||
*/
|
||||
export const NumberOfLinesInDocumentInfo: React.FC<LinesInDocumentInfoProps> = ({ numberOfLinesInDocument }) => {
|
||||
useTranslation()
|
||||
|
||||
const translationOptions = useMemo(() => ({ lines: numberOfLinesInDocument }), [numberOfLinesInDocument])
|
||||
|
||||
return (
|
||||
<span>
|
||||
<Trans i18nKey={'editor.statusBar.lines'} values={translationOptions} />
|
||||
</span>
|
||||
)
|
||||
}
|
|
@ -0,0 +1,52 @@
|
|||
/*
|
||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
||||
*
|
||||
* SPDX-License-Identifier: AGPL-3.0-only
|
||||
*/
|
||||
|
||||
import React, { useMemo } from 'react'
|
||||
import { cypressId } from '../../../../utils/cypress-attribute'
|
||||
import { Trans, useTranslation } from 'react-i18next'
|
||||
|
||||
export interface LengthInfoProps {
|
||||
remainingCharacters: number
|
||||
charactersInDocument: number
|
||||
}
|
||||
|
||||
/**
|
||||
* Renders a translated text that shows the number of remaining characters.
|
||||
*
|
||||
* @param remainingCharacters The number of characters that are still available in this document
|
||||
* @param charactersInDocument The total number of characters in the document
|
||||
*/
|
||||
export const RemainingCharactersInfo: React.FC<LengthInfoProps> = ({ remainingCharacters, charactersInDocument }) => {
|
||||
const { t } = useTranslation()
|
||||
|
||||
const remainingCharactersClass = useMemo(() => {
|
||||
if (remainingCharacters <= 0) {
|
||||
return 'text-danger'
|
||||
} else if (remainingCharacters <= 100) {
|
||||
return 'text-warning'
|
||||
} else {
|
||||
return ''
|
||||
}
|
||||
}, [remainingCharacters])
|
||||
|
||||
const lengthTooltip = useMemo(() => {
|
||||
if (remainingCharacters === 0) {
|
||||
return t('editor.statusBar.lengthTooltip.maximumReached')
|
||||
} else if (remainingCharacters < 0) {
|
||||
return t('editor.statusBar.lengthTooltip.exceeded', { exceeded: -remainingCharacters })
|
||||
} else {
|
||||
return t('editor.statusBar.lengthTooltip.remaining', { remaining: remainingCharacters })
|
||||
}
|
||||
}, [remainingCharacters, t])
|
||||
|
||||
const translationOptions = useMemo(() => ({ length: charactersInDocument }), [charactersInDocument])
|
||||
|
||||
return (
|
||||
<span {...cypressId('remainingCharacters')} title={lengthTooltip} className={remainingCharactersClass}>
|
||||
<Trans i18nKey={'editor.statusBar.length'} values={translationOptions} />
|
||||
</span>
|
||||
)
|
||||
}
|
|
@ -0,0 +1,32 @@
|
|||
/*
|
||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
||||
*
|
||||
* SPDX-License-Identifier: AGPL-3.0-only
|
||||
*/
|
||||
|
||||
import React, { useMemo } from 'react'
|
||||
import { Trans, useTranslation } from 'react-i18next'
|
||||
|
||||
export interface SelectionInfoProps {
|
||||
count: number
|
||||
translationKey: 'column' | 'line'
|
||||
}
|
||||
|
||||
/**
|
||||
* Renders a translated text that shows the number of selected columns or lines.
|
||||
*
|
||||
* @param count The number that should be included in the text
|
||||
* @param translationKey Defines if the text for selected columns or lines should be used
|
||||
*/
|
||||
export const SelectionInfo: React.FC<SelectionInfoProps> = ({ count, translationKey }) => {
|
||||
useTranslation()
|
||||
|
||||
const countTranslationOptions = useMemo(() => ({ count: count }), [count])
|
||||
|
||||
console.log(count, translationKey)
|
||||
return (
|
||||
<span>
|
||||
<Trans i18nKey={`editor.statusBar.selection.${translationKey}`} values={countTranslationOptions} />
|
||||
</span>
|
||||
)
|
||||
}
|
|
@ -0,0 +1,14 @@
|
|||
/*
|
||||
* SPDX-FileCopyrightText: 2021 The HedgeDoc developers (see AUTHORS file)
|
||||
*
|
||||
* SPDX-License-Identifier: AGPL-3.0-only
|
||||
*/
|
||||
|
||||
import React, { Fragment } from 'react'
|
||||
|
||||
/**
|
||||
* Renders a dash without breaking spaces around.
|
||||
*/
|
||||
export const SeparatorDash: React.FC = () => {
|
||||
return <Fragment> – </Fragment>
|
||||
}
|
|
@ -4,12 +4,15 @@
|
|||
* SPDX-License-Identifier: AGPL-3.0-only
|
||||
*/
|
||||
|
||||
import type { Editor, Position } from 'codemirror'
|
||||
import React, { useMemo } from 'react'
|
||||
import { useTranslation } from 'react-i18next'
|
||||
import { ShowIf } from '../../../common/show-if/show-if'
|
||||
import type { Position } from 'codemirror'
|
||||
import React from 'react'
|
||||
import './status-bar.scss'
|
||||
import { cypressId } from '../../../../utils/cypress-attribute'
|
||||
import { RemainingCharactersInfo } from './remaining-characters-info'
|
||||
import { NumberOfLinesInDocumentInfo } from './number-of-lines-in-document-info'
|
||||
import { CursorPositionInfo } from './cursor-position-info'
|
||||
import { SelectionInfo } from './selection-info'
|
||||
import { ShowIf } from '../../../common/show-if/show-if'
|
||||
import { SeparatorDash } from './separator-dash'
|
||||
|
||||
export interface StatusBarInfo {
|
||||
position: Position
|
||||
|
@ -29,57 +32,36 @@ export const defaultState: StatusBarInfo = {
|
|||
remainingCharacters: 0
|
||||
}
|
||||
|
||||
export const createStatusInfo = (editor: Editor, maxDocumentLength: number): StatusBarInfo => ({
|
||||
position: editor.getCursor(),
|
||||
charactersInDocument: editor.getValue().length,
|
||||
remainingCharacters: maxDocumentLength - editor.getValue().length,
|
||||
linesInDocument: editor.lineCount(),
|
||||
selectedColumns: editor.getSelection().length,
|
||||
selectedLines: editor.getSelection().split('\n').length
|
||||
})
|
||||
|
||||
export const StatusBar: React.FC<StatusBarInfo> = ({
|
||||
position,
|
||||
selectedColumns,
|
||||
selectedLines,
|
||||
charactersInDocument,
|
||||
linesInDocument,
|
||||
remainingCharacters
|
||||
}) => {
|
||||
const { t } = useTranslation()
|
||||
|
||||
const getLengthTooltip = useMemo(() => {
|
||||
if (remainingCharacters === 0) {
|
||||
return t('editor.statusBar.lengthTooltip.maximumReached')
|
||||
}
|
||||
if (remainingCharacters < 0) {
|
||||
return t('editor.statusBar.lengthTooltip.exceeded', { exceeded: -remainingCharacters })
|
||||
}
|
||||
return t('editor.statusBar.lengthTooltip.remaining', { remaining: remainingCharacters })
|
||||
}, [remainingCharacters, t])
|
||||
export interface StatusBarProps {
|
||||
statusBarInfo: StatusBarInfo
|
||||
}
|
||||
|
||||
/**
|
||||
* Shows additional information about the document length and the current selection.
|
||||
*
|
||||
* @param statusBarInfo The information to show
|
||||
*/
|
||||
export const StatusBar: React.FC<StatusBarProps> = ({ statusBarInfo }) => {
|
||||
return (
|
||||
<div className='d-flex flex-row status-bar px-2'>
|
||||
<div>
|
||||
<span>{t('editor.statusBar.cursor', { line: position.line + 1, columns: position.ch + 1 })}</span>
|
||||
<ShowIf condition={selectedColumns !== 0 && selectedLines !== 0}>
|
||||
<ShowIf condition={selectedLines === 1}>
|
||||
<span> – {t('editor.statusBar.selection.column', { count: selectedColumns })}</span>
|
||||
</ShowIf>
|
||||
<ShowIf condition={selectedLines > 1}>
|
||||
<span> – {t('editor.statusBar.selection.line', { count: selectedLines })}</span>
|
||||
</ShowIf>
|
||||
<CursorPositionInfo cursorPosition={statusBarInfo.position} />
|
||||
<ShowIf condition={statusBarInfo.selectedLines === 1 && statusBarInfo.selectedColumns > 0}>
|
||||
<SeparatorDash />
|
||||
<SelectionInfo count={statusBarInfo.selectedColumns} translationKey={'column'} />
|
||||
</ShowIf>
|
||||
<ShowIf condition={statusBarInfo.selectedLines > 1}>
|
||||
<SeparatorDash />
|
||||
<SelectionInfo count={statusBarInfo.selectedLines} translationKey={'line'} />
|
||||
</ShowIf>
|
||||
</div>
|
||||
<div className='ml-auto'>
|
||||
<span>{t('editor.statusBar.lines', { lines: linesInDocument })}</span>
|
||||
–
|
||||
<span
|
||||
{...cypressId('remainingCharacters')}
|
||||
title={getLengthTooltip}
|
||||
className={remainingCharacters <= 0 ? 'text-danger' : remainingCharacters <= 100 ? 'text-warning' : ''}>
|
||||
{t('editor.statusBar.length', { length: charactersInDocument })}
|
||||
</span>
|
||||
<NumberOfLinesInDocumentInfo numberOfLinesInDocument={statusBarInfo.linesInDocument} />
|
||||
<SeparatorDash />
|
||||
<RemainingCharactersInfo
|
||||
remainingCharacters={statusBarInfo.remainingCharacters}
|
||||
charactersInDocument={statusBarInfo.charactersInDocument}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
|
|
Loading…
Reference in a new issue