stylus/edit/source-editor.js

329 lines
9.7 KiB
JavaScript
Raw Normal View History

2017-11-22 00:12:05 +00:00
/* global CodeMirror dirtyReporter initLint beautify showKeyMapHelp */
2017-09-11 16:09:25 +00:00
/* global showToggleStyleHelp goBackToManage updateLintReportIfEnabled */
2017-11-23 05:18:55 +00:00
/* global hotkeyRerouter setupAutocomplete */
/* global editors linterConfig updateLinter regExpTester mozParser */
/* global makeLink createAppliesToLineWidget messageBox */
2017-09-11 16:09:25 +00:00
'use strict';
function createSourceEditor(style) {
2017-11-09 05:56:12 +00:00
// a flag for isTouched()
let hadBeenSaved = false;
let savedGeneration = 0;
2017-09-11 16:09:25 +00:00
$('#name').disabled = true;
$('#mozilla-format-container').remove();
$('#sections').textContent = '';
2017-09-12 12:06:00 +00:00
$('#sections').appendChild(
2017-11-22 13:28:50 +00:00
$element({className: 'single-editor'})
2017-09-12 12:06:00 +00:00
);
2017-09-11 16:09:25 +00:00
const dirty = dirtyReporter();
dirty.onChange(() => {
document.body.classList.toggle('dirty', dirty.isDirty());
$('#save-button').disabled = !dirty.isDirty();
2017-09-13 09:33:32 +00:00
updateTitle();
2017-09-11 16:09:25 +00:00
});
// normalize style
if (!style.id) {
setupNewStyle(style);
} else {
// style might be an object reference to background page
style = deepCopy(style);
}
2017-11-22 13:28:50 +00:00
const cm = CodeMirror($('.single-editor'));
editors.push(cm);
updateMeta().then(() => {
initLint();
initLinterSwitch();
cm.setValue(style.sourceCode);
cm.clearHistory();
cm.markClean();
savedGeneration = cm.changeGeneration();
initHooks();
initAppliesToLineWidget();
// focus must be the last action, otherwise the style is duplicated on saving
cm.focus();
});
2017-09-13 08:58:03 +00:00
function initAppliesToLineWidget() {
const PREF_NAME = 'editor.appliesToLineWidget';
const widget = createAppliesToLineWidget(cm);
widget.toggle(prefs.get(PREF_NAME));
prefs.subscribe([PREF_NAME], (key, value) => widget.toggle(value));
}
function initLinterSwitch() {
const linterEl = $('#editor.linter');
let prevMode = NaN;
cm.on('optionChange', (cm, option) => {
if (option !== 'mode') {
return;
}
const mode = cm.doc.mode;
if (mode === prevMode || mode && mode.name === prevMode) {
return;
}
prevMode = mode;
updateLinter();
update();
});
linterEl.addEventListener('change', update);
update();
function update() {
linterEl.value = linterConfig.getName();
const cssLintOption = linterEl.querySelector('[value="csslint"]');
if (cm.getOption('mode') !== 'css') {
cssLintOption.disabled = true;
cssLintOption.title = t('linterCSSLintIncompatible', cm.getOption('mode'));
} else {
cssLintOption.disabled = false;
cssLintOption.title = '';
}
}
2017-10-15 19:58:02 +00:00
}
function setupNewStyle(style) {
style.sections[0].code = ' '.repeat(prefs.get('editor.tabSize')) + '/* Insert code here... */';
let section = mozParser.format(style);
if (!section.includes('@-moz-document')) {
style.sections[0].domains = ['example.com'];
section = mozParser.format(style);
}
const DEFAULT_CODE = `
/* ==UserStyle==
@name ${t('usercssReplaceTemplateName') + ' - ' + new Date().toLocaleString()}
@namespace github.com/openstyles/stylus
@version 0.1.0
@description A new userstyle
@author Me
==/UserStyle== */
${section}
`.replace(/^\s+/gm, '');
dirty.clear('sourceGeneration');
style.sourceCode = '';
BG.chromeSync.getLZValue('usercssTemplate').then(code => {
style.sourceCode = code || DEFAULT_CODE;
cm.startOperation();
cm.setValue(style.sourceCode);
cm.clearHistory();
cm.markClean();
cm.endOperation();
dirty.clear('sourceGeneration');
savedGeneration = cm.changeGeneration();
});
}
2017-09-11 16:09:25 +00:00
function initHooks() {
$('#save-button').onclick = save;
$('#beautify').onclick = beautify;
$('#keyMap-help').onclick = showKeyMapHelp;
$('#toggle-style-help').onclick = showToggleStyleHelp;
$('#cancel-button').onclick = goBackToManage;
$('#enabled').onchange = function () {
const value = this.checked;
2017-09-11 16:09:25 +00:00
dirty.modify('enabled', style.enabled, value);
style.enabled = value;
};
cm.on('changes', () => {
dirty.modify('sourceGeneration', savedGeneration, cm.changeGeneration());
2017-09-11 16:09:25 +00:00
updateLintReportIfEnabled(cm);
});
cm.on('focus', () => hotkeyRerouter.setState(false));
cm.on('blur', () => hotkeyRerouter.setState(true));
2017-09-11 16:09:25 +00:00
CodeMirror.commands.prevEditor = cm => nextPrevMozDocument(cm, -1);
CodeMirror.commands.nextEditor = cm => nextPrevMozDocument(cm, 1);
2017-09-11 16:09:25 +00:00
}
2017-11-08 23:26:51 +00:00
function updateMeta() {
2017-09-11 16:09:25 +00:00
$('#name').value = style.name;
$('#enabled').checked = style.enabled;
$('#url').href = style.url;
const {usercssData: {preprocessor} = {}} = style;
2017-09-11 16:09:25 +00:00
// beautify only works with regular CSS
2017-09-24 08:54:21 +00:00
$('#beautify').disabled = cm.getOption('mode') !== 'css';
2017-09-13 09:33:32 +00:00
updateTitle();
return cm.setPreprocessor(preprocessor);
2017-09-13 09:33:32 +00:00
}
function updateTitle() {
const newTitle = (dirty.isDirty() ? '* ' : '') +
(style.id ? t('editStyleTitle', [style.name]) : t('addStyleTitle'));
if (document.title !== newTitle) {
document.title = newTitle;
}
2017-09-11 16:09:25 +00:00
}
function replaceStyle(newStyle, codeIsUpdated) {
const sameCode = newStyle.sourceCode === cm.getValue();
hadBeenSaved = sameCode;
if (sameCode) {
savedGeneration = cm.changeGeneration();
dirty.clear('sourceGeneration');
}
if (codeIsUpdated === false || sameCode) {
updateEnvironment();
dirty.clear('enabled');
return;
2017-09-11 16:09:25 +00:00
}
Promise.resolve(messageBox.confirm(t('styleUpdateDiscardChanges'))).then(ok => {
if (!ok) {
return;
}
updateEnvironment();
if (!sameCode) {
const cursor = cm.getCursor();
cm.setValue(style.sourceCode);
cm.setCursor(cursor);
savedGeneration = cm.changeGeneration();
}
dirty.clear();
});
function updateEnvironment() {
if (style.id !== newStyle.id) {
history.replaceState({}, '', `?id=${newStyle.id}`);
}
sessionStorage.justEditedStyleId = newStyle.id;
style = deepCopy(newStyle);
updateMeta();
}
}
2017-09-11 16:09:25 +00:00
function toggleStyle() {
const value = !style.enabled;
dirty.modify('enabled', style.enabled, value);
style.enabled = value;
2017-11-08 23:26:51 +00:00
updateMeta();
2017-09-11 16:09:25 +00:00
// save when toggle enable state?
save();
}
function save() {
if (!dirty.isDirty()) {
return;
}
return onBackgroundReady()
.then(() => BG.usercssHelper.save({
reason: 'editSave',
id: style.id,
enabled: style.enabled,
sourceCode: cm.getValue(),
}))
.then(replaceStyle)
.then(() => cm.setOption('mode', cm.doc.mode))
2017-09-11 16:09:25 +00:00
.catch(err => {
if (err.message === t('styleMissingMeta', 'name')) {
messageBox.confirm(t('usercssReplaceTemplateConfirmation')).then(ok => ok &&
BG.chromeSync.setLZValue('usercssTemplate', style.sourceCode)
.then(() => BG.chromeSync.getLZValue('usercssTemplate'))
.then(saved => {
if (saved !== style.sourceCode) {
messageBox.alert(t('syncStorageErrorSaving'));
}
}));
return;
}
const contents = Array.isArray(err) ?
$element({tag: 'pre', textContent: err.join('\n')}) :
[String(err)];
if (Number.isInteger(err.index)) {
const pos = cm.posFromIndex(err.index);
contents[0] += ` (line ${pos.line + 1} col ${pos.ch + 1})`;
contents.push($element({
tag: 'pre',
textContent: drawLinePointer(pos)
}));
}
messageBox.alert(contents);
2017-09-11 16:09:25 +00:00
});
function drawLinePointer(pos) {
const SIZE = 60;
const line = cm.getLine(pos.line);
const numTabs = pos.ch + 1 - line.slice(0, pos.ch + 1).replace(/\t/g, '').length;
const pointer = ' '.repeat(pos.ch) + '^';
const start = Math.max(Math.min(pos.ch - SIZE / 2, line.length - SIZE), 0);
const end = Math.min(Math.max(pos.ch + SIZE / 2, SIZE), line.length);
const leftPad = start !== 0 ? '...' : '';
const rightPad = end !== line.length ? '...' : '';
return (
leftPad +
line.slice(start, end).replace(/\t/g, ' '.repeat(cm.options.tabSize)) +
rightPad +
'\n' +
' '.repeat(leftPad.length + numTabs * cm.options.tabSize) +
pointer.slice(start, end)
);
}
2017-09-11 16:09:25 +00:00
}
2017-11-09 05:56:12 +00:00
function isTouched() {
// indicate that the editor had been touched by the user
return dirty.isDirty() || hadBeenSaved;
}
function nextPrevMozDocument(cm, dir) {
const cursor = cm.getCursor();
let line = cursor.line;
let found;
if (dir > 0) {
cm.doc.iter(cursor.line + 1, cm.doc.size, ({text}) => ++line && goFind(text));
if (!found && cursor.line > 0) {
line = -1;
cm.doc.iter(0, cursor.line, ({text}) => ++line && goFind(text));
}
} else {
let handle, parentLines;
let passesRemain = line < cm.doc.size - 1 ? 2 : 1;
while (passesRemain--) {
let indexInParent = 0;
while (line--) {
if (!indexInParent--) {
handle = cm.getLineHandle(line);
parentLines = handle.parent.lines;
indexInParent = parentLines.indexOf(handle);
} else {
handle = parentLines[indexInParent];
}
if (goFind(handle.text)) {
return true;
}
}
line = cm.doc.size;
}
}
function goFind(text) {
const ch = text.indexOf('@-moz-document');
if (ch >= 0 && cm.getTokenTypeAt({line, ch}) === 'def') {
cm.scrollIntoView({line: line + 1, ch}, Math.min(50, cm.display.scroller.clientHeight / 4));
cm.setCursor(line, ch);
found = true;
return true;
}
}
}
return {
replaceStyle,
save,
toggleStyle,
isDirty: dirty.isDirty,
getStyle: () => style,
2017-11-09 05:56:12 +00:00
isTouched
};
2017-09-11 16:09:25 +00:00
}