mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-12-12 06:27:01 +00:00
84d548e7b1
A @method annotation is only necessary when the docblock is not directly followed by a function declaration (in which case JSDuck assumes it documents a property), e.g. when defining an abstract function or referencing a function from another library. I verified that JSDuck generates exactly the same output before and after this change (docs/data-<hash>.js files are identical). Change-Id: I7edf51a8560ab9978b42800ab1026f0b5555c3bf
153 lines
5.5 KiB
JavaScript
153 lines
5.5 KiB
JavaScript
/*!
|
|
* VisualEditor UserInterface MWWikitextAction class.
|
|
*
|
|
* @copyright 2011-2019 VisualEditor Team and others; see http://ve.mit-license.org
|
|
*/
|
|
|
|
/**
|
|
* Content action.
|
|
*
|
|
* @class
|
|
* @extends ve.ui.Action
|
|
*
|
|
* @constructor
|
|
* @param {ve.ui.Surface} surface Surface to act on
|
|
*/
|
|
ve.ui.MWWikitextAction = function VeUiMWWikitextAction() {
|
|
// Parent constructor
|
|
ve.ui.MWWikitextAction.super.apply( this, arguments );
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
OO.inheritClass( ve.ui.MWWikitextAction, ve.ui.Action );
|
|
|
|
/* Static Properties */
|
|
|
|
ve.ui.MWWikitextAction.static.name = 'mwWikitext';
|
|
|
|
/**
|
|
* List of allowed methods for the action.
|
|
*
|
|
* @static
|
|
* @property
|
|
*/
|
|
ve.ui.MWWikitextAction.static.methods = [ 'toggleWrapSelection', 'wrapSelection', 'wrapLine' ];
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* Wrap an selection inline
|
|
*
|
|
* @param {string} before Text to go before selection
|
|
* @param {string} after Text to go after selection
|
|
* @param {Function|string} placeholder Placeholder text to insert at an empty selection
|
|
* @param {Function} [expandOffsetsCallback] Function that returns a tuple of offsets to expand to selection to in order to get relevant text for unwrapping
|
|
* @param {Function} [unwrapOffsetsCallback] Function that returns a tuple of offsets to unwrap from the selected text,
|
|
* e.g. "''Foo'''" -> [2,3] to unwrap 2 from the left and 3 from the right
|
|
* @return {boolean} Action was executed
|
|
*/
|
|
ve.ui.MWWikitextAction.prototype.toggleWrapSelection = function ( before, after, placeholder, expandOffsetsCallback, unwrapOffsetsCallback ) {
|
|
var contextRange, data, range, textBefore, textAfter, expandOffsets, unwrapOffsets,
|
|
originalFragment = this.surface.getModel().getFragment( null, false, true /* excludeInsertions */ ),
|
|
fragment = originalFragment;
|
|
|
|
if ( expandOffsetsCallback ) {
|
|
contextRange = fragment.expandLinearSelection( 'siblings' ).getSelection().getCoveringRange();
|
|
data = fragment.getDocument().data;
|
|
range = fragment.getSelection().getCoveringRange();
|
|
textBefore = data.getText( true, new ve.Range( contextRange.start, range.start ) );
|
|
textAfter = data.getText( true, new ve.Range( range.end, contextRange.end ) );
|
|
expandOffsets = expandOffsetsCallback( textBefore, textAfter );
|
|
if ( expandOffsets ) {
|
|
fragment = originalFragment.adjustLinearSelection( expandOffsets[ 0 ], expandOffsets[ 1 ] );
|
|
}
|
|
}
|
|
|
|
if ( unwrapOffsetsCallback ) {
|
|
unwrapOffsets = unwrapOffsetsCallback( fragment.getText(), textBefore, textAfter );
|
|
if ( unwrapOffsets ) {
|
|
fragment.unwrapText( unwrapOffsets[ 0 ], unwrapOffsets[ 1 ] );
|
|
} else {
|
|
fragment.wrapText( before, after, placeholder, true );
|
|
}
|
|
originalFragment.select();
|
|
return true;
|
|
}
|
|
|
|
fragment.wrapText( before, after, placeholder ).select();
|
|
originalFragment.select();
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* Wrap an selection inline
|
|
*
|
|
* @param {string} before Text to go before selection
|
|
* @param {string} after Text to go after selection
|
|
* @param {Function|string} placeholder Placeholder text to insert at an empty selection
|
|
* @return {boolean} Action was executed
|
|
*/
|
|
ve.ui.MWWikitextAction.prototype.wrapSelection = function ( before, after, placeholder ) {
|
|
var fragment = this.surface.getModel().getFragment( null, false, true /* excludeInsertions */ );
|
|
fragment.wrapText( before, after, placeholder ).select();
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* Wrap an selection as a block element on its own line
|
|
*
|
|
* If the selection is collapsed, it expands to take the whole line, otherwise it splits
|
|
* the paragraph to make sure it is one line
|
|
*
|
|
* @param {string} before Text to go before each line
|
|
* @param {string} after Text to go after each line
|
|
* @param {Function|string} placeholder Placeholder text to insert at an empty selection
|
|
* @param {Function} [unwrapOffsetsCallback] Function that returns a tuple of offsets to unwrap from the selected text,
|
|
* e.g. '== Foo ===' -> [2,3] to unwrap 2 from the left and 3 from the right
|
|
* @return {boolean} Action was executed
|
|
*/
|
|
ve.ui.MWWikitextAction.prototype.wrapLine = function ( before, after, placeholder, unwrapOffsetsCallback ) {
|
|
var i, wrappedFragment, unwrapped,
|
|
fragment, unwrapOffsets,
|
|
originalFragment = this.surface.getModel().getFragment( null, false, true /* excludeInsertions */ ),
|
|
selectedNodes = originalFragment.getLeafNodes();
|
|
|
|
for ( i = selectedNodes.length - 1; i >= 0; i-- ) {
|
|
if ( selectedNodes.length > 1 && selectedNodes[ i ].nodeRange.isCollapsed() ) {
|
|
continue;
|
|
}
|
|
fragment = this.surface.getModel().getLinearFragment( selectedNodes[ i ].nodeRange, true );
|
|
unwrapOffsets = unwrapOffsetsCallback && unwrapOffsetsCallback( fragment.getText() );
|
|
|
|
if ( selectedNodes.length === 1 && originalFragment.getSelection().isCollapsed() ) {
|
|
originalFragment = fragment;
|
|
}
|
|
|
|
if ( unwrapOffsets ) {
|
|
fragment.unwrapText( unwrapOffsets[ 0 ], unwrapOffsets[ 1 ] );
|
|
unwrapped = true;
|
|
}
|
|
|
|
wrappedFragment = fragment.wrapText( before, after, placeholder );
|
|
if ( !unwrapped && wrappedFragment !== fragment ) {
|
|
if ( !ve.dm.LinearData.static.isElementData(
|
|
wrappedFragment.collapseToStart().adjustLinearSelection( -1, 0 ).getData()[ 0 ]
|
|
) ) {
|
|
wrappedFragment.collapseToStart().insertContent( [ { type: '/paragraph' }, { type: 'paragraph' } ] );
|
|
}
|
|
if ( !ve.dm.LinearData.static.isElementData(
|
|
wrappedFragment.collapseToEnd().adjustLinearSelection( 0, 1 ).getData()[ 0 ]
|
|
) ) {
|
|
wrappedFragment.collapseToEnd().insertContent( [ { type: '/paragraph' }, { type: 'paragraph' } ] );
|
|
}
|
|
}
|
|
}
|
|
originalFragment.select();
|
|
return true;
|
|
};
|
|
|
|
/* Registration */
|
|
|
|
ve.ui.actionFactory.register( ve.ui.MWWikitextAction );
|