2019-05-24 03:40:16 +00:00
|
|
|
/* global CodeMirror, $, editor, Cookies */
|
2020-02-04 06:36:19 +00:00
|
|
|
import { options, Alignment, FormatType } from '@susisu/mte-kernel'
|
|
|
|
import debounce from 'lodash/debounce'
|
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
import * as utils from './utils'
|
2017-03-28 09:16:32 +00:00
|
|
|
import config from './config'
|
2017-04-09 12:05:48 +00:00
|
|
|
import statusBarTemplate from './statusbar.html'
|
2018-06-19 14:03:32 +00:00
|
|
|
import toolBarTemplate from './toolbar.html'
|
2019-08-16 01:37:09 +00:00
|
|
|
import './markdown-lint'
|
2020-01-31 09:51:14 +00:00
|
|
|
import CodeMirrorSpellChecker, { supportLanguages, supportLanguageCodes } from './spellcheck'
|
2019-10-14 05:20:44 +00:00
|
|
|
import { initTableEditor } from './table-editor'
|
2020-01-31 07:41:37 +00:00
|
|
|
import { availableThemes } from './constants'
|
2017-03-07 14:35:54 +00:00
|
|
|
|
|
|
|
/* config section */
|
2017-03-09 07:39:42 +00:00
|
|
|
const isMac = CodeMirror.keyMap.default === CodeMirror.keyMap.macDefault
|
|
|
|
const defaultEditorMode = 'gfm'
|
|
|
|
const viewportMargin = 20
|
2017-03-07 14:35:54 +00:00
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
const jumpToAddressBarKeymapName = isMac ? 'Cmd-L' : 'Ctrl-L'
|
2017-03-07 14:35:54 +00:00
|
|
|
|
|
|
|
export default class Editor {
|
2017-03-09 07:39:42 +00:00
|
|
|
constructor () {
|
|
|
|
this.editor = null
|
2017-03-13 14:00:20 +00:00
|
|
|
this.jumpToAddressBarKeymapValue = null
|
2017-03-09 07:39:42 +00:00
|
|
|
this.defaultExtraKeys = {
|
|
|
|
F10: function (cm) {
|
|
|
|
cm.setOption('fullScreen', !cm.getOption('fullScreen'))
|
|
|
|
},
|
|
|
|
Esc: function (cm) {
|
2018-02-24 20:03:12 +00:00
|
|
|
if (cm.getOption('fullScreen') && !(cm.getOption('keyMap').substr(0, 3) === 'vim')) {
|
2017-03-09 07:39:42 +00:00
|
|
|
cm.setOption('fullScreen', false)
|
2018-02-24 20:03:12 +00:00
|
|
|
} else {
|
|
|
|
return CodeMirror.Pass
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
'Cmd-S': function () {
|
|
|
|
return false
|
|
|
|
},
|
|
|
|
'Ctrl-S': function () {
|
|
|
|
return false
|
|
|
|
},
|
|
|
|
Enter: 'newlineAndIndentContinueMarkdownList',
|
|
|
|
Tab: function (cm) {
|
|
|
|
var tab = '\t'
|
|
|
|
|
|
|
|
// contruct x length spaces
|
|
|
|
var spaces = Array(parseInt(cm.getOption('indentUnit')) + 1).join(' ')
|
|
|
|
|
|
|
|
// auto indent whole line when in list or blockquote
|
|
|
|
var cursor = cm.getCursor()
|
|
|
|
var line = cm.getLine(cursor.line)
|
|
|
|
|
|
|
|
// this regex match the following patterns
|
|
|
|
// 1. blockquote starts with "> " or ">>"
|
|
|
|
// 2. unorder list starts with *+-
|
|
|
|
// 3. order list starts with "1." or "1)"
|
|
|
|
var regex = /^(\s*)(>[> ]*|[*+-]\s|(\d+)([.)]))/
|
|
|
|
|
|
|
|
var match
|
|
|
|
var multiple = cm.getSelection().split('\n').length > 1 ||
|
|
|
|
cm.getSelections().length > 1
|
|
|
|
|
|
|
|
if (multiple) {
|
|
|
|
cm.execCommand('defaultTab')
|
|
|
|
} else if ((match = regex.exec(line)) !== null) {
|
|
|
|
var ch = match[1].length
|
|
|
|
var pos = {
|
|
|
|
line: cursor.line,
|
|
|
|
ch: ch
|
|
|
|
}
|
|
|
|
if (cm.getOption('indentWithTabs')) {
|
|
|
|
cm.replaceRange(tab, pos, pos, '+input')
|
|
|
|
} else {
|
|
|
|
cm.replaceRange(spaces, pos, pos, '+input')
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (cm.getOption('indentWithTabs')) {
|
|
|
|
cm.execCommand('defaultTab')
|
|
|
|
} else {
|
|
|
|
cm.replaceSelection(spaces)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'Cmd-Left': 'goLineLeftSmart',
|
|
|
|
'Cmd-Right': 'goLineRight',
|
2019-08-04 15:20:44 +00:00
|
|
|
Home: 'goLineLeftSmart',
|
|
|
|
End: 'goLineRight',
|
2017-03-09 07:39:42 +00:00
|
|
|
'Ctrl-C': function (cm) {
|
|
|
|
if (!isMac && cm.getOption('keyMap').substr(0, 3) === 'vim') {
|
|
|
|
document.execCommand('copy')
|
|
|
|
} else {
|
|
|
|
return CodeMirror.Pass
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'Ctrl-*': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '*')
|
|
|
|
},
|
|
|
|
'Shift-Ctrl-8': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '*')
|
|
|
|
},
|
|
|
|
'Ctrl-_': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '_')
|
|
|
|
},
|
|
|
|
'Shift-Ctrl--': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '_')
|
|
|
|
},
|
|
|
|
'Ctrl-~': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '~')
|
|
|
|
},
|
|
|
|
'Shift-Ctrl-`': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '~')
|
|
|
|
},
|
|
|
|
'Ctrl-^': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '^')
|
|
|
|
},
|
|
|
|
'Shift-Ctrl-6': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '^')
|
|
|
|
},
|
|
|
|
'Ctrl-+': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '+')
|
|
|
|
},
|
|
|
|
'Shift-Ctrl-=': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '+')
|
|
|
|
},
|
|
|
|
'Ctrl-=': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, '=')
|
|
|
|
},
|
|
|
|
'Shift-Ctrl-Backspace': cm => {
|
|
|
|
utils.wrapTextWith(this.editor, cm, 'Backspace')
|
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-28 03:18:36 +00:00
|
|
|
this.eventListeners = {}
|
2017-04-09 13:14:23 +00:00
|
|
|
this.config = config
|
2020-02-27 09:35:17 +00:00
|
|
|
|
|
|
|
// define modes from mode mime
|
2020-02-27 10:20:05 +00:00
|
|
|
const ignoreOverlay = {
|
2020-02-27 09:35:17 +00:00
|
|
|
token: function (stream, state) {
|
|
|
|
stream.next()
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
CodeMirror.defineMode('vega', function (config, modeConfig) {
|
|
|
|
return CodeMirror.overlayMode(CodeMirror.getMode(config, 'application/ld+json'), ignoreOverlay)
|
|
|
|
})
|
2017-03-28 03:18:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
on (event, cb) {
|
|
|
|
if (!this.eventListeners[event]) {
|
|
|
|
this.eventListeners[event] = [cb]
|
|
|
|
} else {
|
|
|
|
this.eventListeners[event].push(cb)
|
|
|
|
}
|
|
|
|
|
|
|
|
this.editor.on(event, (...args) => {
|
2017-03-28 09:11:20 +00:00
|
|
|
this.eventListeners[event].forEach(cb => cb.bind(this)(...args))
|
2017-03-28 03:18:36 +00:00
|
|
|
})
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
|
2018-06-19 14:03:32 +00:00
|
|
|
addToolBar () {
|
|
|
|
this.toolBar = $(toolBarTemplate)
|
|
|
|
this.toolbarPanel = this.editor.addPanel(this.toolBar[0], {
|
2018-06-23 18:55:32 +00:00
|
|
|
position: 'top'
|
2018-06-19 14:03:32 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
var makeBold = $('#makeBold')
|
|
|
|
var makeItalic = $('#makeItalic')
|
|
|
|
var makeStrike = $('#makeStrike')
|
|
|
|
var makeHeader = $('#makeHeader')
|
|
|
|
var makeCode = $('#makeCode')
|
|
|
|
var makeQuote = $('#makeQuote')
|
|
|
|
var makeGenericList = $('#makeGenericList')
|
|
|
|
var makeOrderedList = $('#makeOrderedList')
|
|
|
|
var makeCheckList = $('#makeCheckList')
|
|
|
|
var makeLink = $('#makeLink')
|
|
|
|
var makeImage = $('#makeImage')
|
|
|
|
var makeTable = $('#makeTable')
|
|
|
|
var makeLine = $('#makeLine')
|
|
|
|
var makeComment = $('#makeComment')
|
|
|
|
|
2019-10-14 05:22:36 +00:00
|
|
|
var insertRow = $('#insertRow')
|
|
|
|
var deleteRow = $('#deleteRow')
|
|
|
|
var moveRowUp = $('#moveRowUp')
|
|
|
|
var moveRowDown = $('#moveRowDown')
|
|
|
|
var insertColumn = $('#insertColumn')
|
|
|
|
var deleteColumn = $('#deleteColumn')
|
|
|
|
var moveColumnLeft = $('#moveColumnLeft')
|
|
|
|
var moveColumnRight = $('#moveColumnRight')
|
|
|
|
var alignLeft = $('#alignLeft')
|
|
|
|
var alignCenter = $('#alignCenter')
|
|
|
|
var alignRight = $('#alignRight')
|
|
|
|
var alignNone = $('#alignNone')
|
|
|
|
|
2018-06-19 14:03:32 +00:00
|
|
|
makeBold.click(() => {
|
2018-06-23 18:55:32 +00:00
|
|
|
utils.wrapTextWith(this.editor, this.editor, '**')
|
|
|
|
this.editor.focus()
|
2018-06-19 14:03:32 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
makeItalic.click(() => {
|
2018-06-23 18:55:32 +00:00
|
|
|
utils.wrapTextWith(this.editor, this.editor, '*')
|
|
|
|
this.editor.focus()
|
2018-06-19 14:03:32 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
makeStrike.click(() => {
|
2018-06-23 18:55:32 +00:00
|
|
|
utils.wrapTextWith(this.editor, this.editor, '~~')
|
|
|
|
this.editor.focus()
|
2018-06-19 14:03:32 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
makeHeader.click(() => {
|
|
|
|
utils.insertHeader(this.editor)
|
|
|
|
})
|
|
|
|
|
|
|
|
makeCode.click(() => {
|
|
|
|
utils.wrapTextWith(this.editor, this.editor, '```')
|
2018-06-23 18:55:32 +00:00
|
|
|
this.editor.focus()
|
2018-06-19 14:03:32 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
makeQuote.click(() => {
|
|
|
|
utils.insertOnStartOfLines(this.editor, '> ')
|
|
|
|
})
|
|
|
|
|
|
|
|
makeGenericList.click(() => {
|
|
|
|
utils.insertOnStartOfLines(this.editor, '* ')
|
|
|
|
})
|
|
|
|
|
|
|
|
makeOrderedList.click(() => {
|
|
|
|
utils.insertOnStartOfLines(this.editor, '1. ')
|
|
|
|
})
|
|
|
|
|
|
|
|
makeCheckList.click(() => {
|
|
|
|
utils.insertOnStartOfLines(this.editor, '- [ ] ')
|
|
|
|
})
|
|
|
|
|
|
|
|
makeLink.click(() => {
|
2018-06-23 18:55:32 +00:00
|
|
|
utils.insertLink(this.editor, false)
|
2018-06-19 14:03:32 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
makeImage.click(() => {
|
2018-06-23 18:55:32 +00:00
|
|
|
utils.insertLink(this.editor, true)
|
2018-06-19 14:03:32 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
makeTable.click(() => {
|
|
|
|
utils.insertText(this.editor, '\n\n| Column 1 | Column 2 | Column 3 |\n| -------- | -------- | -------- |\n| Text | Text | Text |\n')
|
|
|
|
})
|
|
|
|
|
|
|
|
makeLine.click(() => {
|
|
|
|
utils.insertText(this.editor, '\n----\n')
|
|
|
|
})
|
|
|
|
|
|
|
|
makeComment.click(() => {
|
2018-06-23 18:55:32 +00:00
|
|
|
utils.insertText(this.editor, '> []')
|
2018-06-19 14:03:32 +00:00
|
|
|
})
|
2019-10-14 05:22:36 +00:00
|
|
|
|
|
|
|
// table tools UI
|
|
|
|
const opts = options({
|
|
|
|
smartCursor: true,
|
|
|
|
formatType: FormatType.NORMAL
|
|
|
|
})
|
|
|
|
|
|
|
|
insertRow.click(() => {
|
|
|
|
this.tableEditor.insertRow(opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
deleteRow.click(() => {
|
|
|
|
this.tableEditor.deleteRow(opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
moveRowUp.click(() => {
|
|
|
|
this.tableEditor.moveRow(-1, opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
moveRowDown.click(() => {
|
|
|
|
this.tableEditor.moveRow(1, opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
insertColumn.click(() => {
|
|
|
|
this.tableEditor.insertColumn(opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
deleteColumn.click(() => {
|
|
|
|
this.tableEditor.deleteColumn(opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
moveColumnLeft.click(() => {
|
|
|
|
this.tableEditor.moveColumn(-1, opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
moveColumnRight.click(() => {
|
|
|
|
this.tableEditor.moveColumn(1, opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
alignLeft.click(() => {
|
|
|
|
this.tableEditor.alignColumn(Alignment.LEFT, opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
alignCenter.click(() => {
|
|
|
|
this.tableEditor.alignColumn(Alignment.CENTER, opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
alignRight.click(() => {
|
|
|
|
this.tableEditor.alignColumn(Alignment.RIGHT, opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
|
|
|
|
|
|
|
alignNone.click(() => {
|
|
|
|
this.tableEditor.alignColumn(Alignment.NONE, opts)
|
|
|
|
this.editor.focus()
|
|
|
|
})
|
2018-06-19 14:03:32 +00:00
|
|
|
}
|
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
addStatusBar () {
|
2017-04-09 12:05:48 +00:00
|
|
|
this.statusBar = $(statusBarTemplate)
|
|
|
|
this.statusCursor = this.statusBar.find('.status-cursor > .status-line-column')
|
|
|
|
this.statusSelection = this.statusBar.find('.status-cursor > .status-selection')
|
|
|
|
this.statusFile = this.statusBar.find('.status-file')
|
|
|
|
this.statusIndicators = this.statusBar.find('.status-indicators')
|
|
|
|
this.statusIndent = this.statusBar.find('.status-indent')
|
|
|
|
this.statusKeymap = this.statusBar.find('.status-keymap')
|
|
|
|
this.statusLength = this.statusBar.find('.status-length')
|
|
|
|
this.statusTheme = this.statusBar.find('.status-theme')
|
|
|
|
this.statusSpellcheck = this.statusBar.find('.status-spellcheck')
|
2019-08-16 16:17:22 +00:00
|
|
|
this.statusLinter = this.statusBar.find('.status-linter')
|
2017-04-09 12:05:48 +00:00
|
|
|
this.statusPreferences = this.statusBar.find('.status-preferences')
|
|
|
|
this.statusPanel = this.editor.addPanel(this.statusBar[0], {
|
|
|
|
position: 'bottom'
|
|
|
|
})
|
2017-03-09 07:39:42 +00:00
|
|
|
|
2017-04-09 12:05:48 +00:00
|
|
|
this.setIndent()
|
|
|
|
this.setKeymap()
|
|
|
|
this.setTheme()
|
|
|
|
this.setSpellcheck()
|
2019-08-16 16:17:22 +00:00
|
|
|
this.setLinter()
|
2017-04-09 12:05:48 +00:00
|
|
|
this.setPreferences()
|
2020-02-04 06:36:19 +00:00
|
|
|
|
|
|
|
this.handleStatusBarResize()
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
|
|
|
|
2017-03-28 09:16:32 +00:00
|
|
|
updateStatusBar () {
|
|
|
|
if (!this.statusBar) return
|
|
|
|
|
|
|
|
var cursor = this.editor.getCursor()
|
2019-12-01 15:22:12 +00:00
|
|
|
var cursorText = 'Line ' + (cursor.line + 1) + ', Column ' + (cursor.ch + 1)
|
2017-03-28 09:16:32 +00:00
|
|
|
this.statusCursor.text(cursorText)
|
|
|
|
var fileText = ' — ' + editor.lineCount() + ' Lines'
|
|
|
|
this.statusFile.text(fileText)
|
|
|
|
var docLength = editor.getValue().length
|
|
|
|
this.statusLength.text('Length ' + docLength)
|
|
|
|
if (docLength > (config.docmaxlength * 0.95)) {
|
|
|
|
this.statusLength.css('color', 'red')
|
2017-10-30 00:25:44 +00:00
|
|
|
this.statusLength.attr('title', 'You have almost reached the limit for this document.')
|
2017-03-28 09:16:32 +00:00
|
|
|
} else if (docLength > (config.docmaxlength * 0.8)) {
|
|
|
|
this.statusLength.css('color', 'orange')
|
2017-10-30 00:25:44 +00:00
|
|
|
this.statusLength.attr('title', 'This document is nearly full, consider splitting it or creating a new one.')
|
2017-03-28 09:16:32 +00:00
|
|
|
} else {
|
|
|
|
this.statusLength.css('color', 'white')
|
2017-10-30 00:25:44 +00:00
|
|
|
this.statusLength.attr('title', 'You can write up to ' + config.docmaxlength + ' characters in this document.')
|
2017-03-28 09:16:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-04 06:36:19 +00:00
|
|
|
handleStatusBarResize () {
|
|
|
|
const onResize = debounce(() => {
|
|
|
|
if (!this.statusBar) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
const maxHeight = window.innerHeight - this.statusBar.height() - 50 /* navbar height */ - 10 /* spacing */
|
|
|
|
this.statusBar.find('.status-theme ul.dropdown-menu').css('max-height', `${maxHeight}px`)
|
|
|
|
}, 300)
|
|
|
|
|
|
|
|
$(window).resize(onResize)
|
|
|
|
|
|
|
|
onResize()
|
|
|
|
}
|
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
setIndent () {
|
|
|
|
var cookieIndentType = Cookies.get('indent_type')
|
|
|
|
var cookieTabSize = parseInt(Cookies.get('tab_size'))
|
|
|
|
var cookieSpaceUnits = parseInt(Cookies.get('space_units'))
|
|
|
|
if (cookieIndentType) {
|
|
|
|
if (cookieIndentType === 'tab') {
|
|
|
|
this.editor.setOption('indentWithTabs', true)
|
|
|
|
if (cookieTabSize) {
|
|
|
|
this.editor.setOption('indentUnit', cookieTabSize)
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
} else if (cookieIndentType === 'space') {
|
|
|
|
this.editor.setOption('indentWithTabs', false)
|
|
|
|
if (cookieSpaceUnits) {
|
|
|
|
this.editor.setOption('indentUnit', cookieSpaceUnits)
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
if (cookieTabSize) {
|
|
|
|
this.editor.setOption('tabSize', cookieTabSize)
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
var type = this.statusIndicators.find('.indent-type')
|
|
|
|
var widthLabel = this.statusIndicators.find('.indent-width-label')
|
|
|
|
var widthInput = this.statusIndicators.find('.indent-width-input')
|
|
|
|
|
|
|
|
const setType = () => {
|
|
|
|
if (this.editor.getOption('indentWithTabs')) {
|
|
|
|
Cookies.set('indent_type', 'tab', {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
type.text('Tab Size:')
|
|
|
|
} else {
|
|
|
|
Cookies.set('indent_type', 'space', {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
type.text('Spaces:')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
setType()
|
|
|
|
|
|
|
|
const setUnit = () => {
|
|
|
|
var unit = this.editor.getOption('indentUnit')
|
|
|
|
if (this.editor.getOption('indentWithTabs')) {
|
|
|
|
Cookies.set('tab_size', unit, {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
Cookies.set('space_units', unit, {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
}
|
|
|
|
widthLabel.text(unit)
|
|
|
|
}
|
|
|
|
setUnit()
|
|
|
|
|
|
|
|
type.click(() => {
|
|
|
|
if (this.editor.getOption('indentWithTabs')) {
|
|
|
|
this.editor.setOption('indentWithTabs', false)
|
|
|
|
cookieSpaceUnits = parseInt(Cookies.get('space_units'))
|
|
|
|
if (cookieSpaceUnits) {
|
|
|
|
this.editor.setOption('indentUnit', cookieSpaceUnits)
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
} else {
|
|
|
|
this.editor.setOption('indentWithTabs', true)
|
|
|
|
cookieTabSize = parseInt(Cookies.get('tab_size'))
|
|
|
|
if (cookieTabSize) {
|
|
|
|
this.editor.setOption('indentUnit', cookieTabSize)
|
|
|
|
this.editor.setOption('tabSize', cookieTabSize)
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
|
|
|
setType()
|
|
|
|
setUnit()
|
|
|
|
})
|
|
|
|
widthLabel.click(() => {
|
|
|
|
if (widthLabel.is(':visible')) {
|
|
|
|
widthLabel.addClass('hidden')
|
|
|
|
widthInput.removeClass('hidden')
|
|
|
|
widthInput.val(this.editor.getOption('indentUnit'))
|
|
|
|
widthInput.select()
|
|
|
|
} else {
|
|
|
|
widthLabel.removeClass('hidden')
|
|
|
|
widthInput.addClass('hidden')
|
|
|
|
}
|
|
|
|
})
|
|
|
|
widthInput.on('change', () => {
|
|
|
|
var val = parseInt(widthInput.val())
|
|
|
|
if (!val) val = this.editor.getOption('indentUnit')
|
|
|
|
if (val < 1) val = 1
|
|
|
|
else if (val > 10) val = 10
|
|
|
|
|
|
|
|
if (this.editor.getOption('indentWithTabs')) {
|
|
|
|
this.editor.setOption('tabSize', val)
|
|
|
|
}
|
|
|
|
this.editor.setOption('indentUnit', val)
|
|
|
|
setUnit()
|
|
|
|
})
|
|
|
|
widthInput.on('blur', function () {
|
|
|
|
widthLabel.removeClass('hidden')
|
|
|
|
widthInput.addClass('hidden')
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
setKeymap () {
|
|
|
|
var cookieKeymap = Cookies.get('keymap')
|
|
|
|
if (cookieKeymap) {
|
|
|
|
this.editor.setOption('keyMap', cookieKeymap)
|
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
var label = this.statusIndicators.find('.ui-keymap-label')
|
|
|
|
var sublime = this.statusIndicators.find('.ui-keymap-sublime')
|
|
|
|
var emacs = this.statusIndicators.find('.ui-keymap-emacs')
|
|
|
|
var vim = this.statusIndicators.find('.ui-keymap-vim')
|
|
|
|
|
|
|
|
const setKeymapLabel = () => {
|
|
|
|
var keymap = this.editor.getOption('keyMap')
|
|
|
|
Cookies.set('keymap', keymap, {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
label.text(keymap)
|
|
|
|
this.restoreOverrideEditorKeymap()
|
|
|
|
this.setOverrideBrowserKeymap()
|
|
|
|
}
|
|
|
|
setKeymapLabel()
|
|
|
|
|
|
|
|
sublime.click(() => {
|
|
|
|
this.editor.setOption('keyMap', 'sublime')
|
|
|
|
setKeymapLabel()
|
|
|
|
})
|
|
|
|
emacs.click(() => {
|
|
|
|
this.editor.setOption('keyMap', 'emacs')
|
|
|
|
setKeymapLabel()
|
|
|
|
})
|
|
|
|
vim.click(() => {
|
|
|
|
this.editor.setOption('keyMap', 'vim')
|
|
|
|
setKeymapLabel()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
setTheme () {
|
2020-01-31 07:41:37 +00:00
|
|
|
this.statusIndicators.find('.status-theme ul.dropdown-menu').append(availableThemes.map(theme => {
|
|
|
|
return $(`<li value="${theme.value}"><a>${theme.name}</a></li>`)
|
|
|
|
}))
|
|
|
|
|
|
|
|
const activateThemeListItem = (theme) => {
|
|
|
|
this.statusIndicators.find('.status-theme li').removeClass('active')
|
|
|
|
this.statusIndicators.find(`.status-theme li[value="${theme}"]`).addClass('active')
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
|
2019-11-04 01:11:16 +00:00
|
|
|
const setTheme = theme => {
|
2017-03-09 07:39:42 +00:00
|
|
|
this.editor.setOption('theme', theme)
|
|
|
|
Cookies.set('theme', theme, {
|
|
|
|
expires: 365
|
|
|
|
})
|
2020-01-31 07:41:37 +00:00
|
|
|
this.statusIndicators.find('.status-theme li').removeClass('active')
|
|
|
|
this.statusIndicators.find(`.status-theme li[value="${theme}"]`).addClass('active')
|
|
|
|
}
|
|
|
|
|
|
|
|
const cookieTheme = Cookies.get('theme')
|
|
|
|
if (cookieTheme && availableThemes.find(theme => cookieTheme === theme.value)) {
|
|
|
|
setTheme(cookieTheme)
|
|
|
|
activateThemeListItem(cookieTheme)
|
|
|
|
} else {
|
|
|
|
activateThemeListItem(this.editor.getOption('theme'))
|
2019-11-04 01:11:16 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
|
2019-11-04 01:11:16 +00:00
|
|
|
this.statusIndicators.find('.status-theme li').click(function () {
|
2020-01-31 07:41:37 +00:00
|
|
|
const theme = $(this).attr('value')
|
|
|
|
setTheme(theme)
|
|
|
|
activateThemeListItem(theme)
|
2017-03-09 07:39:42 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-11-26 12:29:21 +00:00
|
|
|
setSpellcheckLang (lang) {
|
|
|
|
if (lang === 'disabled') {
|
|
|
|
this.statusIndicators.find('.spellcheck-lang').text('')
|
2020-02-05 10:49:27 +00:00
|
|
|
this.activateSpellcheckListItem(false)
|
2019-11-26 12:29:21 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-01-31 09:51:14 +00:00
|
|
|
if (!supportLanguageCodes.includes(lang)) {
|
2019-11-26 12:29:21 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
const langName = this.statusIndicators.find(`.status-spellcheck li[value="${lang}"]`).text()
|
|
|
|
this.statusIndicators.find('.spellcheck-lang').text(langName)
|
|
|
|
|
|
|
|
this.spellchecker.setDictLang(lang)
|
2020-02-05 10:49:27 +00:00
|
|
|
this.activateSpellcheckListItem(lang)
|
2019-11-26 12:29:21 +00:00
|
|
|
}
|
|
|
|
|
2020-01-31 09:51:14 +00:00
|
|
|
getExistingSpellcheckLang () {
|
|
|
|
const cookieSpellcheck = Cookies.get('spellcheck')
|
|
|
|
|
|
|
|
if (cookieSpellcheck) {
|
|
|
|
return cookieSpellcheck === 'false' ? undefined : cookieSpellcheck
|
|
|
|
} else {
|
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-05 10:49:27 +00:00
|
|
|
activateSpellcheckListItem (lang) {
|
|
|
|
this.statusIndicators.find('.status-spellcheck li').removeClass('active')
|
|
|
|
|
|
|
|
if (lang) {
|
|
|
|
this.statusIndicators.find(`.status-spellcheck li[value="${lang}"]`).addClass('active')
|
|
|
|
} else {
|
|
|
|
this.statusIndicators.find(`.status-spellcheck li[value="disabled"]`).addClass('active')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
setSpellcheck () {
|
2020-01-31 09:51:14 +00:00
|
|
|
this.statusSpellcheck.find('ul.dropdown-menu').append(supportLanguages.map(lang => {
|
|
|
|
return $(`<li value="${lang.value}"><a>${lang.name}</a></li>`)
|
|
|
|
}))
|
|
|
|
|
2019-11-26 12:29:21 +00:00
|
|
|
const cookieSpellcheck = Cookies.get('spellcheck')
|
2017-03-09 07:39:42 +00:00
|
|
|
if (cookieSpellcheck) {
|
2019-11-26 12:29:21 +00:00
|
|
|
let mode = null
|
|
|
|
let lang = 'en_US'
|
2020-02-05 10:49:27 +00:00
|
|
|
|
2019-11-26 12:29:21 +00:00
|
|
|
if (cookieSpellcheck === 'false' || !cookieSpellcheck) {
|
2017-03-09 07:39:42 +00:00
|
|
|
mode = defaultEditorMode
|
2020-02-05 10:49:27 +00:00
|
|
|
this.activateSpellcheckListItem(false)
|
2019-11-26 12:29:21 +00:00
|
|
|
} else {
|
|
|
|
mode = 'spell-checker'
|
2020-01-31 09:51:14 +00:00
|
|
|
if (supportLanguageCodes.includes(cookieSpellcheck)) {
|
2019-11-26 12:29:21 +00:00
|
|
|
lang = cookieSpellcheck
|
|
|
|
}
|
|
|
|
this.setSpellcheckLang(lang)
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
2020-02-05 10:49:27 +00:00
|
|
|
|
|
|
|
this.editor.setOption('mode', mode)
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
|
2019-11-26 12:29:21 +00:00
|
|
|
const spellcheckToggle = this.statusSpellcheck.find('.ui-spellcheck-toggle')
|
2017-03-07 14:35:54 +00:00
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
const checkSpellcheck = () => {
|
|
|
|
var mode = this.editor.getOption('mode')
|
|
|
|
if (mode === defaultEditorMode) {
|
|
|
|
spellcheckToggle.removeClass('active')
|
|
|
|
} else {
|
|
|
|
spellcheckToggle.addClass('active')
|
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
|
|
|
|
2019-11-26 12:29:21 +00:00
|
|
|
const self = this
|
|
|
|
this.statusIndicators.find(`.status-spellcheck li`).click(function () {
|
|
|
|
const lang = $(this).attr('value')
|
|
|
|
|
|
|
|
if (lang === 'disabled') {
|
|
|
|
spellcheckToggle.removeClass('active')
|
|
|
|
|
|
|
|
Cookies.set('spellcheck', false, {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
|
|
|
|
self.editor.setOption('mode', defaultEditorMode)
|
2017-03-09 07:39:42 +00:00
|
|
|
} else {
|
2019-11-26 12:29:21 +00:00
|
|
|
spellcheckToggle.addClass('active')
|
|
|
|
|
|
|
|
Cookies.set('spellcheck', lang, {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
|
|
|
|
self.editor.setOption('mode', 'spell-checker')
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
|
|
|
|
2019-11-26 12:29:21 +00:00
|
|
|
self.setSpellcheckLang(lang)
|
2017-03-09 07:39:42 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
checkSpellcheck()
|
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
|
2019-08-16 16:17:22 +00:00
|
|
|
toggleLinter (enable) {
|
|
|
|
const gutters = this.editor.getOption('gutters')
|
|
|
|
const lintGutter = 'CodeMirror-lint-markers'
|
|
|
|
|
|
|
|
if (enable) {
|
|
|
|
if (!gutters.includes(lintGutter)) {
|
|
|
|
this.editor.setOption('gutters', [lintGutter, ...gutters])
|
|
|
|
}
|
|
|
|
Cookies.set('linter', true, {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
this.editor.setOption('gutters', gutters.filter(g => g !== lintGutter))
|
|
|
|
Cookies.remove('linter')
|
|
|
|
}
|
|
|
|
this.editor.setOption('lint', enable)
|
|
|
|
}
|
|
|
|
|
|
|
|
setLinter () {
|
|
|
|
const linterToggle = this.statusLinter.find('.ui-linter-toggle')
|
|
|
|
|
|
|
|
const updateLinterStatus = (enable) => {
|
|
|
|
linterToggle.toggleClass('active', enable)
|
|
|
|
}
|
|
|
|
|
|
|
|
linterToggle.click(() => {
|
|
|
|
const lintEnable = this.editor.getOption('lint')
|
|
|
|
this.toggleLinter.bind(this)(!lintEnable)
|
|
|
|
updateLinterStatus(!lintEnable)
|
|
|
|
})
|
|
|
|
|
|
|
|
const enable = !!Cookies.get('linter')
|
|
|
|
this.toggleLinter.bind(this)(enable)
|
|
|
|
updateLinterStatus(enable)
|
|
|
|
}
|
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
resetEditorKeymapToBrowserKeymap () {
|
|
|
|
var keymap = this.editor.getOption('keyMap')
|
|
|
|
if (!this.jumpToAddressBarKeymapValue) {
|
|
|
|
this.jumpToAddressBarKeymapValue = CodeMirror.keyMap[keymap][jumpToAddressBarKeymapName]
|
|
|
|
delete CodeMirror.keyMap[keymap][jumpToAddressBarKeymapName]
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
restoreOverrideEditorKeymap () {
|
|
|
|
var keymap = this.editor.getOption('keyMap')
|
|
|
|
if (this.jumpToAddressBarKeymapValue) {
|
|
|
|
CodeMirror.keyMap[keymap][jumpToAddressBarKeymapName] = this.jumpToAddressBarKeymapValue
|
|
|
|
this.jumpToAddressBarKeymapValue = null
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
2019-08-04 15:20:44 +00:00
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
setOverrideBrowserKeymap () {
|
|
|
|
var overrideBrowserKeymap = $(
|
2018-11-14 13:10:14 +00:00
|
|
|
'.ui-preferences-override-browser-keymap label > input[type="checkbox"]'
|
2017-03-09 07:39:42 +00:00
|
|
|
)
|
|
|
|
if (overrideBrowserKeymap.is(':checked')) {
|
|
|
|
Cookies.set('preferences-override-browser-keymap', true, {
|
|
|
|
expires: 365
|
|
|
|
})
|
|
|
|
this.restoreOverrideEditorKeymap()
|
|
|
|
} else {
|
|
|
|
Cookies.remove('preferences-override-browser-keymap')
|
|
|
|
this.resetEditorKeymapToBrowserKeymap()
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
setPreferences () {
|
|
|
|
var overrideBrowserKeymap = $(
|
2018-11-14 13:10:14 +00:00
|
|
|
'.ui-preferences-override-browser-keymap label > input[type="checkbox"]'
|
2017-03-09 07:39:42 +00:00
|
|
|
)
|
|
|
|
var cookieOverrideBrowserKeymap = Cookies.get(
|
2018-11-14 13:10:14 +00:00
|
|
|
'preferences-override-browser-keymap'
|
2017-03-09 07:39:42 +00:00
|
|
|
)
|
|
|
|
if (cookieOverrideBrowserKeymap && cookieOverrideBrowserKeymap === 'true') {
|
|
|
|
overrideBrowserKeymap.prop('checked', true)
|
|
|
|
} else {
|
|
|
|
overrideBrowserKeymap.prop('checked', false)
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|
2017-03-09 07:39:42 +00:00
|
|
|
this.setOverrideBrowserKeymap()
|
|
|
|
|
|
|
|
overrideBrowserKeymap.change(() => {
|
|
|
|
this.setOverrideBrowserKeymap()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
init (textit) {
|
|
|
|
this.editor = CodeMirror.fromTextArea(textit, {
|
|
|
|
mode: defaultEditorMode,
|
|
|
|
backdrop: defaultEditorMode,
|
|
|
|
keyMap: 'sublime',
|
|
|
|
viewportMargin: viewportMargin,
|
|
|
|
styleActiveLine: true,
|
|
|
|
lineNumbers: true,
|
|
|
|
lineWrapping: true,
|
|
|
|
showCursorWhenSelecting: true,
|
|
|
|
highlightSelectionMatches: true,
|
|
|
|
indentUnit: 4,
|
|
|
|
continueComments: 'Enter',
|
|
|
|
theme: 'one-dark',
|
|
|
|
inputStyle: 'textarea',
|
|
|
|
matchBrackets: true,
|
|
|
|
autoCloseBrackets: true,
|
|
|
|
matchTags: {
|
|
|
|
bothTags: true
|
|
|
|
},
|
|
|
|
autoCloseTags: true,
|
|
|
|
foldGutter: true,
|
|
|
|
gutters: [
|
|
|
|
'CodeMirror-linenumbers',
|
|
|
|
'authorship-gutters',
|
2019-08-16 02:30:34 +00:00
|
|
|
'CodeMirror-foldgutter'
|
2017-03-09 07:39:42 +00:00
|
|
|
],
|
|
|
|
extraKeys: this.defaultExtraKeys,
|
|
|
|
flattenSpans: true,
|
|
|
|
addModeClass: true,
|
|
|
|
readOnly: true,
|
|
|
|
autoRefresh: true,
|
|
|
|
otherCursors: true,
|
|
|
|
placeholder: "← Start by entering a title here\n===\nVisit /features if you don't know what to do.\nHappy hacking :)"
|
|
|
|
})
|
|
|
|
|
2020-02-05 10:18:29 +00:00
|
|
|
this.spellchecker = new CodeMirrorSpellChecker(CodeMirror, this.getExistingSpellcheckLang(), this.editor)
|
2019-10-14 05:20:44 +00:00
|
|
|
this.tableEditor = initTableEditor(this.editor)
|
|
|
|
|
2017-03-09 07:39:42 +00:00
|
|
|
return this.editor
|
|
|
|
}
|
|
|
|
|
|
|
|
getEditor () {
|
|
|
|
return this.editor
|
|
|
|
}
|
2017-03-07 14:35:54 +00:00
|
|
|
}
|