2016-05-26 12:08:26 +00:00
|
|
|
/*!
|
|
|
|
* VisualEditor DataModel Surface class.
|
|
|
|
*
|
2020-01-08 17:13:04 +00:00
|
|
|
* @copyright 2011-2020 VisualEditor Team and others; see http://ve.mit-license.org
|
2016-05-26 12:08:26 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* DataModel surface.
|
|
|
|
*
|
|
|
|
* @class
|
|
|
|
* @extends ve.ce.Surface
|
|
|
|
*
|
|
|
|
* @constructor
|
|
|
|
* @param {ve.dm.Surface} model
|
|
|
|
* @param {ve.ui.Surface} ui
|
|
|
|
* @param {Object} [config]
|
|
|
|
*/
|
|
|
|
ve.ce.MWWikitextSurface = function VeCeMwWikitextSurface() {
|
|
|
|
// Parent constructors
|
|
|
|
ve.ce.MWWikitextSurface.super.apply( this, arguments );
|
|
|
|
|
2017-07-03 21:02:00 +00:00
|
|
|
this.pasteTargetInput = new OO.ui.MultilineTextInputWidget();
|
2016-05-26 12:08:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Inheritance */
|
|
|
|
|
|
|
|
OO.inheritClass( ve.ce.MWWikitextSurface, ve.ce.Surface );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
ve.ce.MWWikitextSurface.prototype.onCopy = function ( e ) {
|
2021-10-13 12:57:45 +00:00
|
|
|
var view = this,
|
2016-05-26 12:08:26 +00:00
|
|
|
clipboardData = e.originalEvent.clipboardData,
|
|
|
|
text = this.getModel().getFragment().getText( true ).replace( /\n\n/g, '\n' );
|
|
|
|
|
2017-11-07 15:35:14 +00:00
|
|
|
if ( !text ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-05-26 12:08:26 +00:00
|
|
|
if ( clipboardData ) {
|
|
|
|
// Disable the default event so we can override the data
|
|
|
|
e.preventDefault();
|
|
|
|
clipboardData.setData( 'text/plain', text );
|
2018-01-26 01:40:26 +00:00
|
|
|
// We're not going to set HTML, but for browsers that support custom data, set a clipboard key
|
|
|
|
if ( ve.isClipboardDataFormatsSupported( e, true ) ) {
|
2021-10-13 12:57:45 +00:00
|
|
|
var slice = this.model.documentModel.shallowCloneFromSelection( this.getModel().getSelection() );
|
2018-01-26 01:40:26 +00:00
|
|
|
this.clipboardIndex++;
|
2021-10-13 12:57:45 +00:00
|
|
|
var clipboardKey = this.clipboardId + '-' + this.clipboardIndex;
|
2018-01-26 01:40:26 +00:00
|
|
|
this.clipboard = { slice: slice, hash: null };
|
|
|
|
// Clone the elements in the slice
|
|
|
|
slice.data.cloneElements( true );
|
|
|
|
clipboardData.setData( 'text/xcustom', clipboardKey );
|
2018-04-23 21:04:49 +00:00
|
|
|
|
|
|
|
// 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 );
|
2018-01-26 01:40:26 +00:00
|
|
|
}
|
2016-05-26 12:08:26 +00:00
|
|
|
} else {
|
2021-10-13 12:57:45 +00:00
|
|
|
var originalSelection = new ve.SelectionState( this.nativeSelection );
|
2016-05-26 12:08:26 +00:00
|
|
|
|
|
|
|
// Save scroll position before changing focus to "offscreen" paste target
|
2021-10-13 12:57:45 +00:00
|
|
|
var scrollTop = this.$window.scrollTop();
|
2016-05-26 12:08:26 +00:00
|
|
|
|
|
|
|
// Prevent surface observation due to native range changing
|
|
|
|
this.surfaceObserver.disable();
|
2018-03-29 12:58:00 +00:00
|
|
|
this.$pasteTarget.empty().append( this.pasteTargetInput.$element );
|
2016-05-26 12:08:26 +00:00
|
|
|
this.pasteTargetInput.setValue( text ).select();
|
|
|
|
|
|
|
|
// Restore scroll position after changing focus
|
|
|
|
this.$window.scrollTop( scrollTop );
|
|
|
|
|
|
|
|
// setTimeout: postpone until after the default copy action
|
|
|
|
setTimeout( function () {
|
|
|
|
// Change focus back
|
2019-02-13 13:20:46 +00:00
|
|
|
view.$attachedRootNode[ 0 ].focus();
|
2016-05-26 12:08:26 +00:00
|
|
|
view.showSelectionState( originalSelection );
|
|
|
|
// Restore scroll position
|
|
|
|
view.$window.scrollTop( scrollTop );
|
|
|
|
view.surfaceObserver.clear();
|
|
|
|
view.surfaceObserver.enable();
|
|
|
|
// Detach input
|
2018-03-29 12:58:00 +00:00
|
|
|
view.pasteTargetInput.$element.detach();
|
2016-05-26 12:08:26 +00:00
|
|
|
} );
|
|
|
|
}
|
|
|
|
};
|
2018-04-11 04:23:47 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
ve.ce.MWWikitextSurface.prototype.afterPasteInsertExternalData = function ( targetFragment, pastedDocumentModel, contextRange ) {
|
2021-10-13 12:57:45 +00:00
|
|
|
var wasSpecial = this.pasteSpecial,
|
2019-11-16 15:20:44 +00:00
|
|
|
// 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 ),
|
2018-06-13 12:28:15 +00:00
|
|
|
view = this;
|
2018-04-11 04:23:47 +00:00
|
|
|
|
2021-10-13 12:57:45 +00:00
|
|
|
var plainPastedDocumentModel = pastedDocumentModel.shallowCloneFromRange( contextRange );
|
2019-11-16 15:20:44 +00:00
|
|
|
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.)
|
2021-10-13 12:57:45 +00:00
|
|
|
var plainContextRange = new ve.Range( plainPastedDocumentModel.getDocumentRange().from + 1, plainPastedDocumentModel.getDocumentRange().to - 1 );
|
2019-11-16 15:20:44 +00:00
|
|
|
view.pasteSpecial = true;
|
2019-11-07 14:49:57 +00:00
|
|
|
|
2019-11-16 15:20:44 +00:00
|
|
|
// isPlainText is true but we still need sanitize (e.g. remove lists)
|
2021-10-13 12:57:45 +00:00
|
|
|
var promise = ve.ce.MWWikitextSurface.super.prototype.afterPasteInsertExternalData.call( this, targetFragment, plainPastedDocumentModel, plainContextRange );
|
2019-11-16 15:20:44 +00:00
|
|
|
if ( ve.init.target.constructor.static.convertToWikitextOnPaste && !wasPlain ) {
|
|
|
|
promise.then( function () {
|
|
|
|
// 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( function () {
|
|
|
|
var surface = view.getSurface(),
|
|
|
|
context = surface.getContext();
|
|
|
|
// HACK: Directly set the 'relatedSources' result in the context to trick it
|
|
|
|
// into showing a context at the end of the paste. This context will disappear
|
|
|
|
// as soon as the selection change as a contextChange will fire.
|
|
|
|
// TODO: Come up witha method to store this context on the surface model then
|
|
|
|
// have the LinearContext read it from there.
|
|
|
|
context.relatedSources = [ {
|
|
|
|
embeddable: false,
|
|
|
|
// HACK²: Pass the rich text document and original fragment (which should now cover
|
|
|
|
// the pasted text) to the context via the otherwise-unused 'model' property.
|
|
|
|
model: {
|
|
|
|
doc: pastedDocumentModel,
|
|
|
|
contextRange: contextRange,
|
|
|
|
fragment: targetFragment
|
|
|
|
},
|
|
|
|
name: 'wikitextPaste',
|
|
|
|
type: 'item'
|
|
|
|
} ];
|
|
|
|
context.afterContextChange();
|
|
|
|
surface.getModel().once( 'select', function () {
|
|
|
|
context.relatedSources = [];
|
|
|
|
context.afterContextChange();
|
2019-11-15 22:13:54 +00:00
|
|
|
} );
|
2019-11-16 15:20:44 +00:00
|
|
|
} );
|
2018-06-13 12:28:15 +00:00
|
|
|
} );
|
2018-04-11 04:23:47 +00:00
|
|
|
}
|
2019-11-16 15:20:44 +00:00
|
|
|
return promise;
|
2018-04-11 04:23:47 +00:00
|
|
|
};
|