mirror of
https://github.com/overleaf/overleaf.git
synced 2024-12-11 23:42:17 -05:00
f409b96f00
GitOrigin-RevId: 8b6a2ad0beb802d917f3e998c378792c6156836a
144 lines
3.9 KiB
TypeScript
144 lines
3.9 KiB
TypeScript
import {
|
|
Decoration,
|
|
DecorationSet,
|
|
EditorView,
|
|
ViewPlugin,
|
|
ViewUpdate,
|
|
} from '@codemirror/view'
|
|
import { type Range, StateEffect, StateField } from '@codemirror/state'
|
|
import { sourceOnly } from './visual/visual'
|
|
|
|
const MAX_INDENT_FRACTION = 0.9
|
|
|
|
const setMaxIndentEffect = StateEffect.define<number>()
|
|
|
|
export const lineWrappingIndentation = (visual: boolean) => {
|
|
// this extension should only be active in the source editor
|
|
return sourceOnly(visual, [
|
|
// store the current maxIndent, based on the clientWidth
|
|
StateField.define<number>({
|
|
create() {
|
|
return 0
|
|
},
|
|
update(value, tr) {
|
|
for (const effect of tr.effects) {
|
|
if (effect.is(setMaxIndentEffect)) {
|
|
value = effect.value
|
|
}
|
|
}
|
|
|
|
return value
|
|
},
|
|
provide(field) {
|
|
return [
|
|
// calculate the max indent when the geometry changes
|
|
ViewPlugin.define(view => {
|
|
const measure = {
|
|
key: 'line-wrapping-indentation-max-indent',
|
|
read(view: EditorView) {
|
|
return (
|
|
(view.contentDOM.clientWidth / view.defaultCharacterWidth) *
|
|
MAX_INDENT_FRACTION
|
|
)
|
|
},
|
|
write(value: number, view: EditorView) {
|
|
if (view.state.field(field) !== value) {
|
|
window.setTimeout(() => {
|
|
view.dispatch({
|
|
effects: setMaxIndentEffect.of(value),
|
|
})
|
|
})
|
|
}
|
|
},
|
|
}
|
|
|
|
return {
|
|
update(update: ViewUpdate) {
|
|
if (update.geometryChanged) {
|
|
view.requestMeasure(measure)
|
|
}
|
|
},
|
|
}
|
|
}),
|
|
|
|
// rebuild the decorations when needed
|
|
ViewPlugin.define<{ decorations: DecorationSet }>(
|
|
view => {
|
|
let previousMaxIndent = 0
|
|
|
|
const value = {
|
|
decorations: buildDecorations(view, view.state.field(field)),
|
|
|
|
update(update: ViewUpdate) {
|
|
const maxIndent = view.state.field(field)
|
|
|
|
if (
|
|
maxIndent !== previousMaxIndent ||
|
|
update.geometryChanged ||
|
|
update.viewportChanged
|
|
) {
|
|
value.decorations = buildDecorations(view, maxIndent)
|
|
}
|
|
|
|
previousMaxIndent = maxIndent
|
|
},
|
|
}
|
|
|
|
return value
|
|
},
|
|
{
|
|
decorations: value => value.decorations,
|
|
}
|
|
),
|
|
]
|
|
},
|
|
}),
|
|
])
|
|
}
|
|
|
|
export const buildDecorations = (view: EditorView, maxIndent: number) => {
|
|
const { state } = view
|
|
const { doc, tabSize } = state
|
|
|
|
const decorations: Range<Decoration>[] = []
|
|
|
|
let from = 0
|
|
|
|
for (const line of doc.iterLines()) {
|
|
// const indent = line.match(/^(\s*)/)[1].length
|
|
const indent = calculateIndent(line, tabSize, maxIndent)
|
|
|
|
if (indent) {
|
|
decorations.push(lineIndentDecoration(indent).range(from))
|
|
}
|
|
|
|
from += line.length + 1
|
|
}
|
|
|
|
return Decoration.set(decorations)
|
|
}
|
|
|
|
const lineIndentDecoration = (indent: number) =>
|
|
Decoration.line({
|
|
attributes: {
|
|
// style: `text-indent: ${indent}ch hanging`, // "hanging" would be ideal, when browsers support it
|
|
style: `text-indent: -${indent}ch; padding-left: calc(${indent}ch + 6px)`, // add 6px to account for existing padding-left
|
|
},
|
|
})
|
|
|
|
// calculate the character width of whitespace at the start of a line
|
|
const calculateIndent = (line: string, tabSize: number, maxIndent: number) => {
|
|
let indent = 0
|
|
|
|
for (const char of line) {
|
|
if (char === ' ') {
|
|
indent++
|
|
} else if (char === '\t') {
|
|
indent += tabSize - (indent % tabSize)
|
|
} else {
|
|
break
|
|
}
|
|
}
|
|
|
|
return Math.min(indent, maxIndent)
|
|
}
|