mirror of https://github.com/sgoudham/carbon.git
parent
da5edfede1
commit
4c639a2872
@ -0,0 +1,215 @@
|
||||
// CodeMirror, copyright (c) by Marijn Haverbeke and others
|
||||
// Distributed under an MIT license: https://codemirror.net/LICENSE
|
||||
import CodeMirror from 'codemirror'
|
||||
|
||||
let defaults = {
|
||||
pairs: '()[]{}\'\'""',
|
||||
closeBefore: ')]}\'":;>',
|
||||
triples: '',
|
||||
explode: '[]{}',
|
||||
}
|
||||
|
||||
let Pos = CodeMirror.Pos
|
||||
|
||||
CodeMirror.defineOption('autoCloseBrackets', false, function (cm, val, old) {
|
||||
if (old && old != CodeMirror.Init) {
|
||||
cm.removeKeyMap(keyMap)
|
||||
cm.state.closeBrackets = null
|
||||
}
|
||||
if (val) {
|
||||
ensureBound(getOption(val, 'pairs'))
|
||||
cm.state.closeBrackets = val
|
||||
cm.addKeyMap(keyMap)
|
||||
}
|
||||
})
|
||||
|
||||
function getOption(conf, name) {
|
||||
if (name == 'pairs' && typeof conf == 'string') return conf
|
||||
if (typeof conf == 'object' && conf[name] != null) return conf[name]
|
||||
return defaults[name]
|
||||
}
|
||||
|
||||
let keyMap = { Backspace: handleBackspace, Enter: handleEnter }
|
||||
function ensureBound(chars) {
|
||||
for (let i = 0; i < chars.length; i++) {
|
||||
let ch = chars.charAt(i),
|
||||
key = "'" + ch + "'"
|
||||
if (!keyMap[key]) keyMap[key] = handler(ch)
|
||||
}
|
||||
}
|
||||
ensureBound(defaults.pairs + '`')
|
||||
|
||||
function handler(ch) {
|
||||
return function (cm) {
|
||||
return handleChar(cm, ch)
|
||||
}
|
||||
}
|
||||
|
||||
function getConfig(cm) {
|
||||
let deflt = cm.state.closeBrackets
|
||||
if (!deflt || deflt.override) return deflt
|
||||
let mode = cm.getModeAt(cm.getCursor())
|
||||
return mode.closeBrackets || deflt
|
||||
}
|
||||
|
||||
function handleBackspace(cm) {
|
||||
let conf = getConfig(cm)
|
||||
if (!conf || cm.getOption('disableInput')) return CodeMirror.Pass
|
||||
|
||||
let pairs = getOption(conf, 'pairs')
|
||||
let ranges = cm.listSelections()
|
||||
for (let i = 0; i < ranges.length; i++) {
|
||||
if (!ranges[i].empty()) return CodeMirror.Pass
|
||||
let around = charsAround(cm, ranges[i].head)
|
||||
if (!around || pairs.indexOf(around) % 2 != 0) return CodeMirror.Pass
|
||||
}
|
||||
for (let i = ranges.length - 1; i >= 0; i--) {
|
||||
let cur = ranges[i].head
|
||||
cm.replaceRange('', Pos(cur.line, cur.ch - 1), Pos(cur.line, cur.ch + 1), '+delete')
|
||||
}
|
||||
}
|
||||
|
||||
function handleEnter(cm) {
|
||||
let conf = getConfig(cm)
|
||||
let explode = conf && getOption(conf, 'explode')
|
||||
if (!explode || cm.getOption('disableInput')) return CodeMirror.Pass
|
||||
|
||||
let ranges = cm.listSelections()
|
||||
for (let i = 0; i < ranges.length; i++) {
|
||||
if (!ranges[i].empty()) return CodeMirror.Pass
|
||||
let around = charsAround(cm, ranges[i].head)
|
||||
if (!around || explode.indexOf(around) % 2 != 0) return CodeMirror.Pass
|
||||
}
|
||||
cm.operation(function () {
|
||||
let linesep = cm.lineSeparator() || '\n'
|
||||
cm.replaceSelection(linesep + linesep, null)
|
||||
moveSel(cm, -1)
|
||||
ranges = cm.listSelections()
|
||||
for (let i = 0; i < ranges.length; i++) {
|
||||
let line = ranges[i].head.line
|
||||
cm.indentLine(line, null, true)
|
||||
cm.indentLine(line + 1, null, true)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
function moveSel(cm, dir) {
|
||||
let newRanges = [],
|
||||
ranges = cm.listSelections(),
|
||||
primary = 0
|
||||
for (let i = 0; i < ranges.length; i++) {
|
||||
let range = ranges[i]
|
||||
if (range.head == cm.getCursor()) primary = i
|
||||
let pos =
|
||||
range.head.ch || dir > 0
|
||||
? { line: range.head.line, ch: range.head.ch + dir }
|
||||
: { line: range.head.line - 1 }
|
||||
newRanges.push({ anchor: pos, head: pos })
|
||||
}
|
||||
cm.setSelections(newRanges, primary)
|
||||
}
|
||||
|
||||
function contractSelection(sel) {
|
||||
let inverted = CodeMirror.cmpPos(sel.anchor, sel.head) > 0
|
||||
return {
|
||||
anchor: new Pos(sel.anchor.line, sel.anchor.ch + (inverted ? -1 : 1)),
|
||||
head: new Pos(sel.head.line, sel.head.ch + (inverted ? 1 : -1)),
|
||||
}
|
||||
}
|
||||
|
||||
function handleChar(cm, ch) {
|
||||
let conf = getConfig(cm)
|
||||
if (!conf || cm.getOption('disableInput')) return CodeMirror.Pass
|
||||
|
||||
let pairs = getOption(conf, 'pairs')
|
||||
let pos = pairs.indexOf(ch)
|
||||
if (pos == -1) return CodeMirror.Pass
|
||||
|
||||
let closeBefore = getOption(conf, 'closeBefore')
|
||||
|
||||
let triples = getOption(conf, 'triples')
|
||||
|
||||
let identical = pairs.charAt(pos + 1) == ch
|
||||
let ranges = cm.listSelections()
|
||||
let opening = pos % 2 == 0
|
||||
|
||||
let type
|
||||
for (let i = 0; i < ranges.length; i++) {
|
||||
let range = ranges[i],
|
||||
cur = range.head,
|
||||
curType
|
||||
let next = cm.getRange(cur, Pos(cur.line, cur.ch + 1))
|
||||
if (opening && !range.empty()) {
|
||||
curType = 'surround'
|
||||
} else if ((identical || !opening) && next == ch) {
|
||||
if (identical && stringStartsAfter(cm, cur)) curType = 'both'
|
||||
else if (
|
||||
triples.indexOf(ch) >= 0 &&
|
||||
cm.getRange(cur, Pos(cur.line, cur.ch + 3)) == ch + ch + ch
|
||||
)
|
||||
curType = 'skipThree'
|
||||
else curType = 'skip'
|
||||
} else if (
|
||||
identical &&
|
||||
cur.ch > 1 &&
|
||||
triples.indexOf(ch) >= 0 &&
|
||||
cm.getRange(Pos(cur.line, cur.ch - 2), cur) == ch + ch
|
||||
) {
|
||||
if (cur.ch > 2 && /\bstring/.test(cm.getTokenTypeAt(Pos(cur.line, cur.ch - 2))))
|
||||
return CodeMirror.Pass
|
||||
curType = 'addFour'
|
||||
} else if (identical) {
|
||||
let prev = cur.ch == 0 ? ' ' : cm.getRange(Pos(cur.line, cur.ch - 1), cur)
|
||||
if (!CodeMirror.isWordChar(next) && prev != ch && !CodeMirror.isWordChar(prev))
|
||||
curType = 'both'
|
||||
else return CodeMirror.Pass
|
||||
} else if (
|
||||
opening &&
|
||||
(next.length === 0 || /\s/.test(next) || closeBefore.indexOf(next) > -1)
|
||||
) {
|
||||
curType = 'both'
|
||||
} else {
|
||||
return CodeMirror.Pass
|
||||
}
|
||||
if (!type) type = curType
|
||||
else if (type != curType) return CodeMirror.Pass
|
||||
}
|
||||
|
||||
let left = pos % 2 ? pairs.charAt(pos - 1) : ch
|
||||
let right = pos % 2 ? ch : pairs.charAt(pos + 1)
|
||||
cm.operation(function () {
|
||||
if (type == 'skip') {
|
||||
moveSel(cm, 1)
|
||||
} else if (type == 'skipThree') {
|
||||
moveSel(cm, 3)
|
||||
} else if (type == 'surround') {
|
||||
let sels = cm.getSelections()
|
||||
for (let i = 0; i < sels.length; i++) sels[i] = left + sels[i] + right
|
||||
cm.replaceSelections(sels, 'around')
|
||||
sels = cm.listSelections().slice()
|
||||
for (let i = 0; i < sels.length; i++) sels[i] = contractSelection(sels[i])
|
||||
cm.setSelections(sels)
|
||||
} else if (type == 'both') {
|
||||
cm.replaceSelection(left + right, null)
|
||||
cm.triggerElectric(left + right)
|
||||
moveSel(cm, -1)
|
||||
} else if (type == 'addFour') {
|
||||
cm.replaceSelection(left + left + left + left, 'before')
|
||||
moveSel(cm, 1)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
function charsAround(cm, pos) {
|
||||
let str = cm.getRange(Pos(pos.line, pos.ch - 1), Pos(pos.line, pos.ch + 1))
|
||||
return str.length == 2 ? str : null
|
||||
}
|
||||
|
||||
function stringStartsAfter(cm, pos) {
|
||||
let token = cm.getTokenAt(Pos(pos.line, pos.ch + 1))
|
||||
return (
|
||||
/\bstring/.test(token.type) &&
|
||||
token.start == pos.ch &&
|
||||
(pos.ch == 0 || !/\bstring/.test(cm.getTokenTypeAt(pos)))
|
||||
)
|
||||
}
|
Loading…
Reference in New Issue