move markdown related code to proper component

This commit is contained in:
keyan 2023-06-12 17:39:20 -05:00
parent 6a49f37c68
commit ef533d41a6
1 changed files with 38 additions and 29 deletions

View File

@ -3,7 +3,7 @@ import InputGroup from 'react-bootstrap/InputGroup'
import BootstrapForm from 'react-bootstrap/Form' import BootstrapForm from 'react-bootstrap/Form'
import Alert from 'react-bootstrap/Alert' import Alert from 'react-bootstrap/Alert'
import { Formik, Form as FormikForm, useFormikContext, useField, FieldArray } from 'formik' import { Formik, Form as FormikForm, useFormikContext, useField, FieldArray } from 'formik'
import React, { createContext, useContext, useEffect, useState } from 'react' import React, { createContext, useContext, useEffect, useRef, useState } from 'react'
import copy from 'clipboard-copy' import copy from 'clipboard-copy'
import Thumb from '../svgs/thumb-up-fill.svg' import Thumb from '../svgs/thumb-up-fill.svg'
import { Col, Dropdown as BootstrapDropdown, Nav } from 'react-bootstrap' import { Col, Dropdown as BootstrapDropdown, Nav } from 'react-bootstrap'
@ -76,14 +76,25 @@ export function InputSkeleton ({ label, hint }) {
) )
} }
export function MarkdownInput ({ label, topLevel, groupClassName, onChange, setHasImgLink, ...props }) { export function MarkdownInput ({ label, topLevel, groupClassName, onChange, setHasImgLink, onKeyDown, innerRef, ...props }) {
const [tab, setTab] = useState('write') const [tab, setTab] = useState('write')
const [, meta] = useField(props) const [, meta, helpers] = useField(props)
const formik = useFormikContext()
const [selectionRange, setSelectionRange] = useState({ start: 0, end: 0 })
innerRef = innerRef || useRef(null)
useEffect(() => { useEffect(() => {
!meta.value && setTab('write') !meta.value && setTab('write')
}, [meta.value]) }, [meta.value])
useEffect(() => {
if (selectionRange.start <= selectionRange.end && innerRef?.current) {
const { start, end } = selectionRange
const input = innerRef.current
input.setSelectionRange(start, end)
}
}, [innerRef?.current, selectionRange.start, selectionRange.end])
return ( return (
<FormGroup label={label} className={groupClassName}> <FormGroup label={label} className={groupClassName}>
<div className={`${styles.markdownInput} ${tab === 'write' ? styles.noTopLeftRadius : ''}`}> <div className={`${styles.markdownInput} ${tab === 'write' ? styles.noTopLeftRadius : ''}`}>
@ -109,6 +120,30 @@ export function MarkdownInput ({ label, topLevel, groupClassName, onChange, setH
setHasImgLink(mdHas(e.target.value, ['link', 'image'])) setHasImgLink(mdHas(e.target.value, ['link', 'image']))
} }
}} }}
innerRef={innerRef}
onKeyDown={(e) => {
const metaOrCtrl = e.metaKey || e.ctrlKey
if (metaOrCtrl) {
if (e.key === 'Enter') formik?.submitForm()
if (e.key === 'k') {
// some browsers use CTRL+K to focus search bar so we have to prevent that behavior
e.preventDefault()
insertMarkdownLinkFormatting(innerRef.current, helpers.setValue, setSelectionRange)
}
if (e.key === 'b') {
// some browsers use CTRL+B to open bookmarks so we have to prevent that behavior
e.preventDefault()
insertMarkdownBoldFormatting(innerRef.current, helpers.setValue, setSelectionRange)
}
if (e.key === 'i') {
// some browsers might use CTRL+I to do something else so prevent that behavior too
e.preventDefault()
insertMarkdownItalicFormatting(innerRef.current, helpers.setValue, setSelectionRange)
}
}
if (onKeyDown) onKeyDown(e)
}}
/> />
</div> </div>
<div className={tab !== 'preview' ? 'd-none' : 'form-group'}> <div className={tab !== 'preview' ? 'd-none' : 'form-group'}>
@ -163,7 +198,6 @@ function InputInner ({
const [field, meta, helpers] = noForm ? [{}, {}, {}] : useField(props) const [field, meta, helpers] = noForm ? [{}, {}, {}] : useField(props)
const formik = noForm ? null : useFormikContext() const formik = noForm ? null : useFormikContext()
const storageKeyPrefix = useContext(StorageKeyPrefixContext) const storageKeyPrefix = useContext(StorageKeyPrefixContext)
const [selectionRange, setSelectionRange] = useState({ start: 0, end: 0 })
const storageKey = storageKeyPrefix ? storageKeyPrefix + '-' + props.name : undefined const storageKey = storageKeyPrefix ? storageKeyPrefix + '-' + props.name : undefined
@ -183,16 +217,6 @@ function InputInner ({
} }
}, [overrideValue]) }, [overrideValue])
useEffect(() => {
if (selectionRange.start <= selectionRange.end && innerRef?.current) {
const { start, end } = selectionRange
const input = innerRef.current
if (props.type === 'textarea') {
input.setSelectionRange(start, end)
}
}
}, [selectionRange.start, selectionRange.end])
const invalid = (!formik || formik.submitCount > 0) && meta.touched && meta.error const invalid = (!formik || formik.submitCount > 0) && meta.touched && meta.error
return ( return (
@ -208,21 +232,6 @@ function InputInner ({
const metaOrCtrl = e.metaKey || e.ctrlKey const metaOrCtrl = e.metaKey || e.ctrlKey
if (metaOrCtrl) { if (metaOrCtrl) {
if (e.key === 'Enter') formik?.submitForm() if (e.key === 'Enter') formik?.submitForm()
if (e.key === 'k') {
// some browsers use CTRL+K to focus search bar so we have to prevent that behavior
e.preventDefault()
insertMarkdownLinkFormatting(innerRef.current, helpers.setValue, setSelectionRange)
}
if (e.key === 'b') {
// some browsers use CTRL+B to open bookmarks so we have to prevent that behavior
e.preventDefault()
insertMarkdownBoldFormatting(innerRef.current, helpers.setValue, setSelectionRange)
}
if (e.key === 'i') {
// some browsers might use CTRL+I to do something else so prevent that behavior too
e.preventDefault()
insertMarkdownItalicFormatting(innerRef.current, helpers.setValue, setSelectionRange)
}
} }
if (onKeyDown) onKeyDown(e) if (onKeyDown) onKeyDown(e)