mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-11-29 16:44:51 +00:00
a9233045c7
Logically depends on I575d92c4fa. Should really use a registry, and also doesn't update when format is changed by deleting text, but is an improvement for now. Bug: T147218 Change-Id: I0cb87dbad2ab4269c4f89bdff8a4ccd9b35c8635
152 lines
4.2 KiB
JavaScript
152 lines
4.2 KiB
JavaScript
/*!
|
|
* VisualEditor DataModel MWWikitextSurfaceFragment class.
|
|
*
|
|
* @copyright 2011-2016 VisualEditor Team and others; see http://ve.mit-license.org
|
|
*/
|
|
|
|
/**
|
|
* DataModel MWWikitextSurfaceFragment.
|
|
*
|
|
* @class
|
|
* @extends ve.dm.SourceSurfaceFragment
|
|
*
|
|
* @constructor
|
|
* @param {ve.dm.Document} doc
|
|
*/
|
|
ve.dm.MWWikitextSurfaceFragment = function VeDmMwWikitextSurfaceFragment() {
|
|
// Parent constructors
|
|
ve.dm.MWWikitextSurfaceFragment.super.apply( this, arguments );
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
OO.inheritClass( ve.dm.MWWikitextSurfaceFragment, ve.dm.SourceSurfaceFragment );
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.dm.MWWikitextSurfaceFragment.prototype.hasMatchingAncestor = function ( type, attributes ) {
|
|
var i, len, text, command,
|
|
nodes = this.getSelectedLeafNodes(),
|
|
all = !!nodes.length;
|
|
|
|
nodes = this.getSelectedLeafNodes();
|
|
all = !!nodes.length;
|
|
for ( i = 0, len = nodes.length; i < len; i++ ) {
|
|
text = this.document.data.getText( false, nodes[ i ].getRange() );
|
|
// TODO: Use a registry to do this matching
|
|
switch ( type ) {
|
|
case 'paragraph':
|
|
all = !text.match( /^ |^=|^<blockquote>/ );
|
|
break;
|
|
case 'mwPreformatted':
|
|
all = text.slice( 0, 1 ) === ' ';
|
|
break;
|
|
case 'blockquote':
|
|
all = text.slice( 0, 12 ) === '<blockquote>';
|
|
break;
|
|
case 'mwHeading':
|
|
command = ve.ui.wikitextCommandRegistry.lookup( 'heading' + attributes.level );
|
|
if ( text.indexOf( command.args[ 0 ] ) !== 0 || text.indexOf( command.args[ 1 ] ) !== text.length - command.args[ 1 ].length ) {
|
|
all = false;
|
|
}
|
|
break;
|
|
default:
|
|
all = false;
|
|
break;
|
|
}
|
|
if ( !all ) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return all;
|
|
};
|
|
|
|
/**
|
|
* Wrap a text selection.
|
|
*
|
|
* If the selection is already identically wrapped it will be unwrapped.
|
|
*
|
|
* @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 {boolean} forceWrap Force wrapping, even if matching wrapping exists
|
|
* @chainable
|
|
*/
|
|
ve.dm.MWWikitextSurfaceFragment.prototype.wrapText = function ( before, after, placeholder, forceWrap ) {
|
|
var wrappedFragment, wasExcludingInsertions;
|
|
|
|
placeholder = OO.ui.resolveMsg( placeholder );
|
|
|
|
function unwrap( fragment ) {
|
|
var text = fragment.getText();
|
|
if (
|
|
( !before || text.slice( 0, before.length ) === before ) &&
|
|
( !after || text.slice( -after.length ) === after )
|
|
) {
|
|
fragment.unwrapText( before.length, after.length );
|
|
// Just the placeholder left, nothing meaningful was selected so just remove it
|
|
if ( fragment.getText() === placeholder ) {
|
|
fragment.removeContent();
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if ( !forceWrap && ( unwrap( this ) || unwrap( this.adjustLinearSelection( -before.length, after.length ) ) ) ) {
|
|
return this;
|
|
} else {
|
|
if ( placeholder && this.getSelection().isCollapsed() ) {
|
|
this.insertContent( placeholder );
|
|
}
|
|
wrappedFragment = this.clone();
|
|
wasExcludingInsertions = this.willExcludeInsertions();
|
|
this.setExcludeInsertions( true );
|
|
this.collapseToStart().insertContent( before );
|
|
this.collapseToEnd().insertContent( after );
|
|
this.setExcludeInsertions( wasExcludingInsertions );
|
|
return wrappedFragment;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Unwrap a fixed amount of text
|
|
*
|
|
* @param {number} before Amount of text to remove from start
|
|
* @param {number} after Amount of text to remove from end
|
|
* @chainable
|
|
*/
|
|
ve.dm.MWWikitextSurfaceFragment.prototype.unwrapText = function ( before, after ) {
|
|
this.collapseToStart().adjustLinearSelection( 0, before ).removeContent();
|
|
this.collapseToEnd().adjustLinearSelection( -after, 0 ).removeContent();
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.dm.MWWikitextSurfaceFragment.prototype.convertDocument = function ( doc ) {
|
|
var wikitextPromise;
|
|
|
|
if ( !doc.data.hasContent() ) {
|
|
return $.Deferred().reject().promise();
|
|
}
|
|
|
|
wikitextPromise = ve.init.target.getWikitextFragment( doc, false );
|
|
|
|
// TODO: Emit an event to trigger the progress bar
|
|
ve.init.target.getSurface().createProgress(
|
|
wikitextPromise, ve.msg( 'visualeditor-generating-wikitext-progress' )
|
|
).done( function ( progressBar, cancelPromise ) {
|
|
cancelPromise.fail( function () {
|
|
wikitextPromise.abort();
|
|
} );
|
|
} );
|
|
|
|
return wikitextPromise;
|
|
};
|