177 lines
5.5 KiB
JavaScript
177 lines
5.5 KiB
JavaScript
'use strict';
|
|
|
|
define(require => {
|
|
const {
|
|
$,
|
|
$create,
|
|
getEventKeyName,
|
|
messageBoxProxy,
|
|
moveFocus,
|
|
} = require('/js/dom');
|
|
const t = require('/js/localization');
|
|
const prefs = require('/js/prefs');
|
|
|
|
let CodeMirror;
|
|
|
|
const util = {
|
|
|
|
get CodeMirror() {
|
|
return CodeMirror;
|
|
},
|
|
set CodeMirror(val) {
|
|
CodeMirror = val;
|
|
},
|
|
|
|
helpPopup: {
|
|
show(title = '', body) {
|
|
const div = $('#help-popup');
|
|
const contents = $('.contents', div);
|
|
div.className = '';
|
|
contents.textContent = '';
|
|
if (body) {
|
|
contents.appendChild(typeof body === 'string' ? t.HTML(body) : body);
|
|
}
|
|
$('.title', div).textContent = title;
|
|
$('.dismiss', div).onclick = util.helpPopup.close;
|
|
window.on('keydown', util.helpPopup.close, true);
|
|
// reset any inline styles
|
|
div.style = 'display: block';
|
|
util.helpPopup.originalFocus = document.activeElement;
|
|
return div;
|
|
},
|
|
close(event) {
|
|
const canClose =
|
|
!event ||
|
|
event.type === 'click' || (
|
|
getEventKeyName(event) === 'Escape' &&
|
|
!$('.CodeMirror-hints, #message-box') && (
|
|
!document.activeElement ||
|
|
!document.activeElement.closest('#search-replace-dialog') &&
|
|
document.activeElement.matches(':not(input), .can-close-on-esc')
|
|
)
|
|
);
|
|
const div = $('#help-popup');
|
|
if (!canClose || !div) {
|
|
return;
|
|
}
|
|
if (event && div.codebox && !div.codebox.options.readOnly && !div.codebox.isClean()) {
|
|
setTimeout(async () => {
|
|
const ok = await messageBoxProxy.confirm(t('confirmDiscardChanges'));
|
|
return ok && util.helpPopup.close();
|
|
});
|
|
return;
|
|
}
|
|
if (div.contains(document.activeElement) && util.helpPopup.originalFocus) {
|
|
util.helpPopup.originalFocus.focus();
|
|
}
|
|
const contents = $('.contents', div);
|
|
div.style.display = '';
|
|
contents.textContent = '';
|
|
window.off('keydown', util.helpPopup.close, true);
|
|
window.dispatchEvent(new Event('closeHelp'));
|
|
},
|
|
},
|
|
|
|
clipString(str, limit = 100) {
|
|
return str.length <= limit ? str : str.substr(0, limit) + '...';
|
|
},
|
|
|
|
createHotkeyInput(prefId, onDone = () => {}) {
|
|
return $create('input', {
|
|
type: 'search',
|
|
spellcheck: false,
|
|
value: prefs.get(prefId),
|
|
onkeydown(event) {
|
|
const key = CodeMirror.keyName(event);
|
|
if (key === 'Tab' || key === 'Shift-Tab') {
|
|
return;
|
|
}
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
switch (key) {
|
|
case 'Enter':
|
|
if (this.checkValidity()) onDone(true);
|
|
return;
|
|
case 'Esc':
|
|
onDone(false);
|
|
return;
|
|
default:
|
|
// disallow: [Shift?] characters, modifiers-only, [modifiers?] + Esc, Tab, nav keys
|
|
if (!key || new RegExp('^(' + [
|
|
'(Back)?Space',
|
|
'(Shift-)?.', // a single character
|
|
'(Shift-?|Ctrl-?|Alt-?|Cmd-?){0,2}(|Esc|Tab|(Page)?(Up|Down)|Left|Right|Home|End|Insert|Delete)',
|
|
].join('|') + ')$', 'i').test(key)) {
|
|
this.value = key || this.value;
|
|
this.setCustomValidity('Not allowed');
|
|
return;
|
|
}
|
|
}
|
|
this.value = key;
|
|
this.setCustomValidity('');
|
|
prefs.set(prefId, key);
|
|
},
|
|
oninput() {
|
|
// fired on pressing "x" to clear the field
|
|
prefs.set(prefId, '');
|
|
},
|
|
onpaste(event) {
|
|
event.preventDefault();
|
|
},
|
|
});
|
|
},
|
|
|
|
async rerouteHotkeys(...args) {
|
|
require(['./reroute-hotkeys'], res => res(...args));
|
|
},
|
|
|
|
showCodeMirrorPopup(title, html, options) {
|
|
const popup = util.helpPopup.show(title, html);
|
|
popup.classList.add('big');
|
|
|
|
let cm = popup.codebox = CodeMirror($('.contents', popup), Object.assign({
|
|
mode: 'css',
|
|
lineNumbers: true,
|
|
lineWrapping: prefs.get('editor.lineWrapping'),
|
|
foldGutter: true,
|
|
gutters: ['CodeMirror-linenumbers', 'CodeMirror-foldgutter', 'CodeMirror-lint-markers'],
|
|
matchBrackets: true,
|
|
styleActiveLine: true,
|
|
theme: prefs.get('editor.theme'),
|
|
keyMap: prefs.get('editor.keyMap'),
|
|
}, options));
|
|
cm.focus();
|
|
util.rerouteHotkeys(false);
|
|
|
|
document.documentElement.style.pointerEvents = 'none';
|
|
popup.style.pointerEvents = 'auto';
|
|
|
|
const onKeyDown = event => {
|
|
if (event.key === 'Tab' && !event.ctrlKey && !event.altKey && !event.metaKey) {
|
|
const search = $('#search-replace-dialog');
|
|
const area = search && search.contains(document.activeElement) ? search : popup;
|
|
moveFocus(area, event.shiftKey ? -1 : 1);
|
|
event.preventDefault();
|
|
}
|
|
};
|
|
window.on('keydown', onKeyDown, true);
|
|
|
|
window.on('closeHelp', () => {
|
|
window.off('keydown', onKeyDown, true);
|
|
document.documentElement.style.removeProperty('pointer-events');
|
|
util.rerouteHotkeys(true);
|
|
cm = popup.codebox = null;
|
|
}, {once: true});
|
|
|
|
return popup;
|
|
},
|
|
|
|
trimCommentLabel(str, limit = 1000) {
|
|
// stripping /*** foo ***/ to foo
|
|
return util.clipString(str.replace(/^[!-/:;=\s]*|[-#$&(+,./:;<=>\s*]*$/g, ''), limit);
|
|
},
|
|
};
|
|
|
|
return util;
|
|
});
|