mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-12-24 20:13:08 +00:00
137 lines
5 KiB
JavaScript
137 lines
5 KiB
JavaScript
|
/*!
|
||
|
* VisualEditor ContentEditable MWWikitextClipboardHandler class.
|
||
|
*
|
||
|
* @copyright See AUTHORS.txt
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* @param {ve.ce.Surface} surface
|
||
|
*/
|
||
|
ve.ce.MWWikitextClipboardHandler = function VeCeMwWikitextClipboardHandler() {
|
||
|
// Parent constructor
|
||
|
ve.ce.MWWikitextClipboardHandler.super.apply( this, arguments );
|
||
|
|
||
|
this.plainInput = new OO.ui.MultilineTextInputWidget();
|
||
|
};
|
||
|
|
||
|
/* Inheritance */
|
||
|
|
||
|
OO.inheritClass( ve.ce.MWWikitextClipboardHandler, ve.ce.ClipboardHandler );
|
||
|
|
||
|
/* Methods */
|
||
|
|
||
|
/**
|
||
|
* @inheritdoc
|
||
|
*/
|
||
|
ve.ce.MWWikitextClipboardHandler.prototype.onCopy = function ( e ) {
|
||
|
const clipboardData = e.originalEvent.clipboardData,
|
||
|
surface = this.getSurface(),
|
||
|
text = surface.getModel().getFragment().getText( true ).replace( /\n\n/g, '\n' );
|
||
|
|
||
|
if ( !text ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if ( clipboardData ) {
|
||
|
// Disable the default event so we can override the data
|
||
|
e.preventDefault();
|
||
|
clipboardData.setData( 'text/plain', text );
|
||
|
// We're not going to set HTML, but for browsers that support custom data, set a clipboard key
|
||
|
if ( ve.isClipboardDataFormatsSupported( e, true ) ) {
|
||
|
const slice = surface.getModel().getDocument().shallowCloneFromSelection( surface.getModel().getSelection() );
|
||
|
this.clipboardIndex++;
|
||
|
const clipboardKey = this.clipboardId + '-' + this.clipboardIndex;
|
||
|
this.clipboard = { slice: slice, hash: null };
|
||
|
// Clone the elements in the slice
|
||
|
slice.data.cloneElements( true );
|
||
|
clipboardData.setData( this.constructor.static.clipboardKeyMimeType, clipboardKey );
|
||
|
|
||
|
// Explicitly store wikitext as text/x-wiki, so that wikitext-aware paste
|
||
|
// contexts can accept it without having to do any content-
|
||
|
// sniffing.
|
||
|
clipboardData.setData( 'text/x-wiki', text );
|
||
|
}
|
||
|
} else {
|
||
|
const originalSelection = new ve.SelectionState( surface.nativeSelection );
|
||
|
|
||
|
// Save scroll position before changing focus to "offscreen" clipboard target
|
||
|
const scrollTop = surface.getSurface().$scrollContainer.scrollTop();
|
||
|
|
||
|
// Prevent surface observation due to native range changing
|
||
|
surface.surfaceObserver.disable();
|
||
|
this.$element.empty().append( this.plainInput.$element );
|
||
|
this.plainInput.setValue( text ).select();
|
||
|
|
||
|
// Restore scroll position after changing focus
|
||
|
surface.getSurface().$scrollContainer.scrollTop( scrollTop );
|
||
|
|
||
|
// setTimeout: postpone until after the default copy action
|
||
|
setTimeout( () => {
|
||
|
// Change focus back
|
||
|
surface.$attachedRootNode[ 0 ].focus();
|
||
|
surface.showSelectionState( originalSelection );
|
||
|
// Restore scroll position
|
||
|
surface.getSurface().$scrollContainer.scrollTop( scrollTop );
|
||
|
surface.surfaceObserver.clear();
|
||
|
surface.surfaceObserver.enable();
|
||
|
// Detach input
|
||
|
this.plainInput.$element.detach();
|
||
|
} );
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @inheritdoc
|
||
|
*/
|
||
|
ve.ce.MWWikitextClipboardHandler.prototype.afterPasteInsertExternalData = function ( targetFragment, pastedDocumentModel, contextRange ) {
|
||
|
const wasSpecial = this.isPasteSpecial(),
|
||
|
// TODO: This check returns true if the paste contains meaningful structure (tables, lists etc.)
|
||
|
// but no annotations (bold, links etc.).
|
||
|
wasPlain = wasSpecial || pastedDocumentModel.data.isPlainText( contextRange, true, undefined, true );
|
||
|
|
||
|
const plainPastedDocumentModel = pastedDocumentModel.shallowCloneFromRange( contextRange );
|
||
|
plainPastedDocumentModel.data.sanitize( { plainText: true, keepEmptyContentBranches: true } );
|
||
|
// We just turned this into plaintext, which probably
|
||
|
// affected the content-length. Luckily, because of
|
||
|
// the earlier clone, we know we just want the whole
|
||
|
// document, and because of the major change to
|
||
|
// plaintext, the difference between originalRange and
|
||
|
// balancedRange don't really apply. As such, clear
|
||
|
// out newDocRange. (Can't just make it undefined;
|
||
|
// need to exclude the internal list, and since we're
|
||
|
// from a paste we also have to exclude the
|
||
|
// opening/closing paragraph.)
|
||
|
const plainContextRange = new ve.Range( plainPastedDocumentModel.getDocumentRange().from + 1, plainPastedDocumentModel.getDocumentRange().to - 1 );
|
||
|
this.prepareForPasteSpecial();
|
||
|
|
||
|
// isPlainText is true but we still need sanitize (e.g. remove lists)
|
||
|
const promise = ve.ce.MWWikitextClipboardHandler.super.prototype.afterPasteInsertExternalData.call( this, targetFragment, plainPastedDocumentModel, plainContextRange );
|
||
|
if ( !wasPlain ) {
|
||
|
promise.then( () => {
|
||
|
// We need to wait for the selection change after paste as that triggers
|
||
|
// a contextChange event. Really we should wait for the afterPaste promise to resolve.
|
||
|
setTimeout( () => {
|
||
|
const surface = this.getSurface(),
|
||
|
context = surface.getSurface().getContext();
|
||
|
// Ensure surface is deactivated on mobile so context can be shown (T336073)
|
||
|
if ( context.isMobile() ) {
|
||
|
surface.deactivate();
|
||
|
}
|
||
|
context.addPersistentSource( {
|
||
|
embeddable: false,
|
||
|
name: 'wikitextPaste',
|
||
|
data: {
|
||
|
doc: pastedDocumentModel,
|
||
|
contextRange: contextRange,
|
||
|
fragment: targetFragment
|
||
|
}
|
||
|
} );
|
||
|
surface.getModel().once( 'select', () => {
|
||
|
context.removePersistentSource( 'wikitextPaste' );
|
||
|
} );
|
||
|
} );
|
||
|
} );
|
||
|
}
|
||
|
return promise;
|
||
|
};
|