mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-05-23 17:19:37 +02:00
93 lines
2.9 KiB
TypeScript
93 lines
2.9 KiB
TypeScript
import React, { useCallback, useEffect, useState } from 'react'
|
|
import ReferencePickerModal from './reference-picker-modal'
|
|
import { useEditorViewContext } from '@/features/ide-react/context/editor-view-context'
|
|
|
|
export default function ReferencePickerController() {
|
|
const { view } = useEditorViewContext()
|
|
|
|
const [open, setOpen] = useState(false)
|
|
const [from, setFrom] = useState<number | null>(null)
|
|
const [to, setTo] = useState<number | null>(null)
|
|
const [braceFrom, setBraceFrom] = useState<number | null>(null)
|
|
const [braceTo, setBraceTo] = useState<number | null>(null)
|
|
const [initialKeys, setInitialKeys] = useState<string[]>([])
|
|
|
|
const onClose = useCallback(() => setOpen(false), [])
|
|
|
|
const onApply = useCallback(
|
|
(selectedKeys: string[]) => {
|
|
if (!view || from == null || to == null) return
|
|
|
|
const doc = view.state.doc
|
|
const bFrom = braceFrom ?? from
|
|
const bTo = braceTo ?? to
|
|
|
|
let insertCore = selectedKeys.join(', ')
|
|
|
|
// Expand over spaces/tabs and commas around selection
|
|
let start = from
|
|
let end = to
|
|
while (start > bFrom && /[ \t,]/.test(doc.sliceString(start - 1, start))) start--
|
|
while (end < bTo && /[ \t,]/.test(doc.sliceString(end, end + 1))) end++
|
|
|
|
let prefix = ''
|
|
let suffix = ''
|
|
if (start > bFrom && doc.sliceString(start - 1, start) !== '\n') prefix = ', '
|
|
if (end < bTo && insertCore) suffix = ', '
|
|
|
|
const insert = prefix + insertCore + suffix
|
|
|
|
// Compute cursor position after inserted tokens
|
|
const cursorPos = start + prefix.length + insertCore.length + (end === bTo ? 0 : 1)
|
|
|
|
view.dispatch({
|
|
changes: { from: start, to: end, insert },
|
|
selection: { anchor: cursorPos },
|
|
})
|
|
view.focus()
|
|
},
|
|
[view, from, to, braceFrom, braceTo]
|
|
)
|
|
|
|
useEffect(() => {
|
|
const handler = (evt: CustomEvent) => {
|
|
const detail = evt.detail || {}
|
|
|
|
if (detail.insertFrom == null) {
|
|
if (!view) return
|
|
const pos = view.state.selection.main.head
|
|
const insertText = '\\cite{}'
|
|
view.dispatch({
|
|
changes: { from: pos, to: pos, insert: insertText }
|
|
})
|
|
const newFrom = pos + insertText.indexOf('{') + 1
|
|
setFrom(newFrom)
|
|
setTo(newFrom)
|
|
setBraceFrom(newFrom)
|
|
setBraceTo(newFrom)
|
|
setInitialKeys([])
|
|
} else {
|
|
setFrom(detail.insertFrom)
|
|
setTo(detail.insertTo)
|
|
setBraceFrom(detail.braceFrom ?? detail.insertFrom)
|
|
setBraceTo(detail.braceTo ?? detail.insertTo)
|
|
setInitialKeys(detail.selectedTokens || [])
|
|
}
|
|
|
|
setOpen(true)
|
|
}
|
|
|
|
window.addEventListener('reference:openPicker', handler as EventListener)
|
|
return () => window.removeEventListener('reference:openPicker', handler as EventListener)
|
|
}, [view])
|
|
|
|
return (
|
|
<ReferencePickerModal
|
|
show={open}
|
|
onClose={onClose}
|
|
onApply={onApply}
|
|
initialKeys={initialKeys}
|
|
/>
|
|
)
|
|
}
|