mirror of
https://github.com/overleaf/overleaf.git
synced 2024-11-07 20:31:06 -05:00
7f37ba737c
* Update Copybara options in preparation for open-sourcing the source editor * Move files * Update paths * Remove source-editor module and checks for its existence * Explicitly mention CM6 license in files that contain code adapted from CM6 GitOrigin-RevId: 89b7cc2b409db01ad103198ccbd1b126ab56349b
255 lines
7.1 KiB
TypeScript
255 lines
7.1 KiB
TypeScript
import {
|
|
searchKeymap,
|
|
search as searchExtension,
|
|
setSearchQuery,
|
|
getSearchQuery,
|
|
openSearchPanel,
|
|
SearchQuery,
|
|
searchPanelOpen,
|
|
} from '@codemirror/search'
|
|
import { EditorView, keymap, ViewPlugin } from '@codemirror/view'
|
|
import { Annotation, EditorState, TransactionSpec } from '@codemirror/state'
|
|
import { highlightSelectionMatches } from './highlight-selection-matches'
|
|
|
|
const restoreSearchQueryAnnotation = Annotation.define<boolean>()
|
|
|
|
const ignoredSearchKeybindings = new Set([
|
|
// This keybinding causes issues with entering @ on certain keyboard layouts
|
|
// https://github.com/overleaf/internal/issues/12119
|
|
'Alt-g',
|
|
])
|
|
|
|
const selectNextMatch = (query: SearchQuery, state: EditorState) => {
|
|
if (!query.valid) {
|
|
return false
|
|
}
|
|
|
|
let cursor = query.getCursor(state.doc, state.selection.main.from)
|
|
|
|
let result = cursor.next()
|
|
|
|
if (result.done) {
|
|
cursor = query.getCursor(state.doc)
|
|
result = cursor.next()
|
|
}
|
|
|
|
return result.done ? null : result.value
|
|
}
|
|
|
|
let searchQuery: SearchQuery | null
|
|
|
|
export const search = () => {
|
|
let open = false
|
|
|
|
return [
|
|
// keymap for search
|
|
keymap.of(
|
|
searchKeymap.filter(
|
|
item => !item.key || !ignoredSearchKeybindings.has(item.key)
|
|
)
|
|
),
|
|
|
|
// highlight text which matches the current selection
|
|
highlightSelectionMatches(),
|
|
|
|
// a wrapper round `search`, which creates a custom panel element and passes it to React by dispatching an event
|
|
searchExtension({
|
|
literal: true,
|
|
createPanel: () => {
|
|
const dom = document.createElement('div')
|
|
dom.className = 'ol-cm-search'
|
|
|
|
return {
|
|
dom,
|
|
mount() {
|
|
open = true
|
|
|
|
// focus the search input when the panel is already open
|
|
const searchInput =
|
|
dom.querySelector<HTMLInputElement>('[main-field]')
|
|
if (searchInput) {
|
|
searchInput.focus()
|
|
searchInput.select()
|
|
}
|
|
},
|
|
destroy() {
|
|
window.setTimeout(() => {
|
|
open = false // in a timeout, so the view plugin below can run its destroy method first
|
|
}, 0)
|
|
},
|
|
}
|
|
},
|
|
}),
|
|
|
|
// restore a stored search and re-open the search panel
|
|
ViewPlugin.define(view => {
|
|
if (searchQuery) {
|
|
const _searchQuery = searchQuery
|
|
window.setTimeout(() => {
|
|
openSearchPanel(view)
|
|
view.dispatch({
|
|
effects: setSearchQuery.of(_searchQuery),
|
|
annotations: restoreSearchQueryAnnotation.of(true),
|
|
})
|
|
}, 0)
|
|
}
|
|
|
|
return {
|
|
destroy() {
|
|
// persist the current search query if the panel is open
|
|
searchQuery = open ? getSearchQuery(view.state) : null
|
|
},
|
|
}
|
|
}),
|
|
|
|
// select a match while searching
|
|
EditorView.updateListener.of(update => {
|
|
for (const tr of update.transactions) {
|
|
// avoid changing the selection and viewport when switching between files
|
|
if (tr.annotation(restoreSearchQueryAnnotation)) {
|
|
continue
|
|
}
|
|
|
|
for (const effect of tr.effects) {
|
|
if (effect.is(setSearchQuery)) {
|
|
const query = effect.value
|
|
if (!query) return
|
|
|
|
// The rest of this messes up searching in Vim, which is handled by
|
|
// the Vim extension, so bail out here in Vim mode. Happily, the
|
|
// Vim extension sticks an extra property on the query value that
|
|
// can be checked
|
|
if ('forVim' in query) return
|
|
|
|
const next = selectNextMatch(query, tr.state)
|
|
|
|
if (next) {
|
|
// select a match if possible
|
|
const spec: TransactionSpec = {
|
|
selection: { anchor: next.from, head: next.to },
|
|
userEvent: 'select.search',
|
|
}
|
|
|
|
// scroll into view if not opening the panel
|
|
if (searchPanelOpen(tr.startState)) {
|
|
spec.effects = EditorView.scrollIntoView(next.from, {
|
|
y: 'center',
|
|
})
|
|
}
|
|
|
|
update.view.dispatch(spec)
|
|
} else {
|
|
// clear the selection if the query became invalid
|
|
const prevQuery = getSearchQuery(tr.startState)
|
|
|
|
if (prevQuery.valid) {
|
|
const { from } = tr.startState.selection.main
|
|
|
|
update.view.dispatch({
|
|
selection: { anchor: from },
|
|
})
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}),
|
|
|
|
// search form theme
|
|
EditorView.theme({
|
|
'.ol-cm-search-form': {
|
|
padding: '10px',
|
|
display: 'flex',
|
|
gap: '10px',
|
|
background: 'var(--ol-blue-gray-1)',
|
|
'--ol-cm-search-form-focus-shadow':
|
|
'inset 0 1px 1px rgb(0 0 0 / 8%), 0 0 8px rgb(102 175 233 / 60%)',
|
|
},
|
|
'.ol-cm-search-controls': {
|
|
display: 'grid',
|
|
gridTemplateColumns: 'auto auto',
|
|
gridTemplateRows: 'auto auto',
|
|
gap: '10px',
|
|
},
|
|
'.ol-cm-search-form-row': {
|
|
display: 'flex',
|
|
gap: '10px',
|
|
justifyContent: 'space-between',
|
|
},
|
|
'.ol-cm-search-form-group': {
|
|
display: 'flex',
|
|
gap: '10px',
|
|
alignItems: 'center',
|
|
},
|
|
'.ol-cm-search-input-group': {
|
|
border: '1px solid var(--input-border)',
|
|
borderRadius: '20px',
|
|
background: 'white',
|
|
width: '100%',
|
|
maxWidth: '25em',
|
|
'& input[type="text"]': {
|
|
background: 'none',
|
|
boxShadow: 'none',
|
|
},
|
|
'& input[type="text"]:focus': {
|
|
outline: 'none',
|
|
boxShadow: 'none',
|
|
},
|
|
'& .btn.btn': {
|
|
background: 'var(--ol-blue-gray-0)',
|
|
color: 'var(--ol-blue-gray-3)',
|
|
borderRadius: '50%',
|
|
height: '2em',
|
|
display: 'inline-flex',
|
|
alignItems: 'center',
|
|
justifyContent: 'center',
|
|
width: '2em',
|
|
marginRight: '3px',
|
|
'&.checked': {
|
|
color: '#fff',
|
|
backgroundColor: 'var(--ol-blue)',
|
|
},
|
|
'&:active': {
|
|
boxShadow: 'none',
|
|
},
|
|
},
|
|
'&:focus-within': {
|
|
borderColor: 'var(--input-border-focus)',
|
|
boxShadow: 'var(--ol-cm-search-form-focus-shadow)',
|
|
},
|
|
},
|
|
'.input-group .ol-cm-search-form-input': {
|
|
border: 'none',
|
|
},
|
|
'.ol-cm-search-input-button': {
|
|
background: '#fff',
|
|
color: 'inherit',
|
|
border: 'none',
|
|
},
|
|
'.ol-cm-search-input-button.focused': {
|
|
borderColor: 'var(--input-border-focus)',
|
|
boxShadow: 'var(--ol-cm-search-form-focus-shadow)',
|
|
},
|
|
'.ol-cm-search-form-button-group': {
|
|
flexShrink: 0,
|
|
},
|
|
'.ol-cm-search-form-position': {
|
|
flexShrink: 0,
|
|
color: 'var(--ol-blue-gray-4)',
|
|
},
|
|
'.ol-cm-search-hidden-inputs': {
|
|
position: 'absolute',
|
|
left: '-10000px',
|
|
},
|
|
'.ol-cm-search-form-close': {
|
|
flex: 1,
|
|
},
|
|
'.ol-cm-search-replace-input': {
|
|
order: 3,
|
|
},
|
|
'.ol-cm-search-replace-buttons': {
|
|
order: 4,
|
|
},
|
|
}),
|
|
]
|
|
}
|