mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/CodeMirror
synced 2025-01-05 17:24:13 +00:00
13c9eae26e
This is toggled by pressing Mod-Shift-, (or Command-Shift-, on MacOS), which then puts focus on the preferences panel. It can be closed with the Escape key, just like other CM panels. The CodeMirror class comes with these extension which can be toggled in preferences: * Bracket matching * Line numbering * Line wrapping * Highlight the active line * Show special characters Only bracket matching, line numbering, and line wrapping are available in the 2017 editor. The bidi isolation and template folding extensions are registered in CodeMirrorModeMediaWiki as they are MW-specific. CodeMirrorPreferences' new registerExtension() method allows any consumer of CodeMirror to add any arbitrary extensions to the preferences panel. This is expected to be called *after* CodeMirror has finished initializing. The 'ext.CodeMirror.ready' hook now passes the CodeMirror instance to accommodate this. The preferences are stored as a single user option in the database, called 'codemirror-preferences'. The defaults can be configured with the $wgCodeMirrorDefaultPreferences configuration setting. The sysadmin-facing values are the familiar boolean, but since CodeMirror is widely used, we make extra efforts to reduce the storage footprint (see T54777). This includes only storing preferences that differ from the defaults, and using binary representation instead of boolean values, since the user option is stored as a string. For now, all preferences are ignored in the 2017 editor. In a future patch, we may add some as toggleable Tools in the VE toolbar. Other changes: * Refactor CSS to use a .darkmode() mixin * Add a method to create a CSS-only fieldset in CodeMirrorPanel * Fix Jest tests now that there are more calls to mw.user.options.get() * Adjust Selenium tests to always use CM6 * Adjust Selenium tests to delete test pages (useful for local dev) * Remove unused code Bug: T359498 Change-Id: I70dcf2f49418cea632c452c1266440effad634f3
225 lines
6.5 KiB
JavaScript
225 lines
6.5 KiB
JavaScript
const { EditorView, Extension, Panel } = require( 'ext.CodeMirror.v6.lib' );
|
|
|
|
/**
|
|
* Abstract class for a panel that can be used with CodeMirror.
|
|
* This class provides methods to create CSS-only Codex components.
|
|
*
|
|
* @see https://codemirror.net/docs/ref/#h_panels
|
|
* @abstract
|
|
*/
|
|
class CodeMirrorPanel {
|
|
/**
|
|
* @constructor
|
|
*/
|
|
constructor() {
|
|
/** @type {EditorView} */
|
|
this.view = undefined;
|
|
}
|
|
|
|
/**
|
|
* Get the panel and any associated keymaps as a CodeMirror Extension.
|
|
*
|
|
* @abstract
|
|
* @type {Extension}
|
|
*/
|
|
// eslint-disable-next-line getter-return
|
|
get extension() {}
|
|
|
|
/**
|
|
* Get the Panel object.
|
|
*
|
|
* @abstract
|
|
* @type {Panel}
|
|
*/
|
|
// eslint-disable-next-line getter-return
|
|
get panel() {}
|
|
|
|
/**
|
|
* Get a CSS-only Codex TextInput.
|
|
*
|
|
* @param {string} name
|
|
* @param {string} [value='']
|
|
* @param {string} placeholder
|
|
* @return {Array<HTMLElement>} [HTMLDivElement, HTMLInputElement]
|
|
* @internal
|
|
*/
|
|
getTextInput( name, value = '', placeholder = '' ) {
|
|
const wrapper = document.createElement( 'div' );
|
|
wrapper.className = 'cdx-text-input cm-mw-panel--text-input';
|
|
const input = document.createElement( 'input' );
|
|
input.className = 'cdx-text-input__input';
|
|
input.type = 'text';
|
|
input.name = name;
|
|
// The following messages may be used here:
|
|
// * codemirror-find
|
|
// * codemirror-replace-placeholder
|
|
input.placeholder = placeholder ? mw.msg( placeholder ) : '';
|
|
input.value = value;
|
|
wrapper.appendChild( input );
|
|
return [ wrapper, input ];
|
|
}
|
|
|
|
/**
|
|
* Get a CSS-only Codex Button.
|
|
*
|
|
* @param {string} label
|
|
* @param {string|null} [icon=null]
|
|
* @param {boolean} [iconOnly=false]
|
|
* @return {HTMLButtonElement}
|
|
* @internal
|
|
*/
|
|
getButton( label, icon = null, iconOnly = false ) {
|
|
const button = document.createElement( 'button' );
|
|
button.className = 'cdx-button cm-mw-panel--button';
|
|
button.type = 'button';
|
|
|
|
if ( icon ) {
|
|
const iconSpan = document.createElement( 'span' );
|
|
// The following CSS classes may be used here:
|
|
// * cm-mw-icon--previous
|
|
// * cm-mw-icon--next
|
|
// * cm-mw-icon--all
|
|
// * cm-mw-icon--replace
|
|
// * cm-mw-icon--replace-all
|
|
// * cm-mw-icon--done
|
|
// * cm-mw-icon--goto-line-go
|
|
iconSpan.className = 'cdx-button__icon cm-mw-icon--' + icon;
|
|
|
|
if ( !iconOnly ) {
|
|
iconSpan.setAttribute( 'aria-hidden', 'true' );
|
|
}
|
|
|
|
button.appendChild( iconSpan );
|
|
}
|
|
|
|
// The following messages may be used here:
|
|
// * codemirror-next
|
|
// * codemirror-previous
|
|
// * codemirror-all
|
|
// * codemirror-replace
|
|
// * codemirror-replace-all
|
|
const message = mw.msg( label );
|
|
if ( iconOnly ) {
|
|
button.classList.add( 'cdx-button--icon-only' );
|
|
button.title = message;
|
|
button.setAttribute( 'aria-label', message );
|
|
} else {
|
|
button.append( message );
|
|
}
|
|
|
|
return button;
|
|
}
|
|
|
|
/**
|
|
* Get a CSS-only Codex Checkbox.
|
|
*
|
|
* @param {string} name
|
|
* @param {string} label
|
|
* @param {boolean} [checked=false]
|
|
* @return {Array<HTMLElement>} [HTMLSpanElement, HTMLInputElement]
|
|
* @internal
|
|
*/
|
|
getCheckbox( name, label, checked = false ) {
|
|
const wrapper = document.createElement( 'span' );
|
|
wrapper.className = 'cdx-checkbox cdx-checkbox--inline cm-mw-panel--checkbox';
|
|
const input = document.createElement( 'input' );
|
|
input.className = 'cdx-checkbox__input';
|
|
input.id = `cm-mw-panel--checkbox-${ name }`;
|
|
input.type = 'checkbox';
|
|
input.name = name;
|
|
input.checked = checked;
|
|
wrapper.appendChild( input );
|
|
const emptyIcon = document.createElement( 'span' );
|
|
emptyIcon.className = 'cdx-checkbox__icon';
|
|
wrapper.appendChild( emptyIcon );
|
|
const labelWrapper = document.createElement( 'div' );
|
|
labelWrapper.className = 'cdx-checkbox__label cdx-label';
|
|
const labelElement = document.createElement( 'label' );
|
|
labelElement.className = 'cdx-label__label';
|
|
labelElement.htmlFor = input.id;
|
|
const innerSpan = document.createElement( 'span' );
|
|
innerSpan.className = 'cdx-label__label__text';
|
|
// The following messages may be used here:
|
|
// * codemirror-match-case
|
|
// * codemirror-regexp
|
|
// * codemirror-by-word
|
|
innerSpan.textContent = mw.msg( label );
|
|
labelElement.appendChild( innerSpan );
|
|
labelWrapper.appendChild( labelElement );
|
|
wrapper.appendChild( labelWrapper );
|
|
return [ wrapper, input ];
|
|
}
|
|
|
|
/**
|
|
* Get a CSS-only Codex ToggleButton.
|
|
*
|
|
* @param {string} name
|
|
* @param {string} label
|
|
* @param {string} icon
|
|
* @param {boolean} [checked=false]
|
|
* @return {HTMLButtonElement}
|
|
* @internal
|
|
*/
|
|
getToggleButton( name, label, icon, checked = false ) {
|
|
const btn = document.createElement( 'button' );
|
|
// The following CSS classes may be used here:
|
|
// * cdx-toggle-button--toggled-on
|
|
// * cdx-toggle-button--toggled-off
|
|
btn.className = 'cdx-toggle-button cdx-toggle-button--framed ' +
|
|
`cdx-toggle-button--toggled-${ checked ? 'on' : 'off' } cm-mw-panel--toggle-button`;
|
|
btn.dataset.checked = String( checked );
|
|
btn.setAttribute( 'aria-pressed', checked );
|
|
// The following messages may be used here:
|
|
// * codemirror-match-case
|
|
// * codemirror-regexp
|
|
// * codemirror-by-word
|
|
const message = mw.msg( label );
|
|
btn.title = message;
|
|
btn.setAttribute( 'aria-label', message );
|
|
|
|
// Add the icon.
|
|
const iconWrapper = document.createElement( 'span' );
|
|
// The following CSS classes may be used here:
|
|
// * cm-mw-icon--match-case
|
|
// * cm-mw-icon--regexp
|
|
// * cm-mw-icon--quotes
|
|
iconWrapper.className = 'cdx-icon cdx-icon--medium cm-mw-icon--' + icon;
|
|
btn.appendChild( iconWrapper );
|
|
|
|
// Add the click handler.
|
|
btn.addEventListener( 'click', ( e ) => {
|
|
e.preventDefault();
|
|
const toggled = btn.dataset.checked === 'true';
|
|
btn.dataset.checked = String( !toggled );
|
|
btn.setAttribute( 'aria-pressed', String( !toggled ) );
|
|
btn.classList.toggle( 'cdx-toggle-button--toggled-on', !toggled );
|
|
btn.classList.toggle( 'cdx-toggle-button--toggled-off', toggled );
|
|
} );
|
|
|
|
return btn;
|
|
}
|
|
|
|
/**
|
|
* Get a CSS-only Codex Fieldset.
|
|
*
|
|
* @param {string} legendText
|
|
* @param {...HTMLElement[]} fields
|
|
* @return {Element}
|
|
*/
|
|
getFieldset( legendText, ...fields ) {
|
|
const fieldset = document.createElement( 'fieldset' );
|
|
fieldset.className = 'cm-mw-panel--fieldset cdx-field';
|
|
const legend = document.createElement( 'legend' );
|
|
legend.className = 'cdx-label';
|
|
const innerSpan = document.createElement( 'span' );
|
|
innerSpan.className = 'cdx-label__label__text';
|
|
innerSpan.textContent = legendText;
|
|
legend.appendChild( innerSpan );
|
|
fieldset.appendChild( legend );
|
|
fieldset.append( ...fields );
|
|
return fieldset;
|
|
}
|
|
}
|
|
|
|
module.exports = CodeMirrorPanel;
|