mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2025-01-01 07:53:17 +00:00
59a665c424
Abortable promises are definitely among my least favorite things. It takes all of this bookkeeping to make .abort() work consistently (so that it always aborts a request if one is in flight, and always causes the final promise to be rejected even if we didn't start a request yet or it has already finished). But, if you squint and ignore every line with the word "abort", it's like a normal promise chain. Depends-On: Iec8a15dadd595bed0f7e54f907fbb8e192b45cf3 Bug: T331397 Change-Id: I67309c79e6d211d69630fe89cbf5402f8fbd350c
170 lines
4.7 KiB
JavaScript
170 lines
4.7 KiB
JavaScript
/*!
|
|
* VisualEditor ContentEditable MWSignatureNode class.
|
|
*
|
|
* @copyright 2011-2020 VisualEditor Team and others; see AUTHORS.txt
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
*/
|
|
|
|
/**
|
|
* ContentEditable MediaWiki signature node. This defines the behavior of the signature node
|
|
* inserted into the ContentEditable document.
|
|
*
|
|
* @class
|
|
* @extends ve.ce.LeafNode
|
|
*
|
|
* @constructor
|
|
* @param {ve.dm.MWSignatureNode} model Model to observe
|
|
* @param {Object} [config] Configuration options
|
|
*/
|
|
ve.ce.MWSignatureNode = function VeCeMWSignatureNode() {
|
|
// Parent constructor
|
|
ve.ce.MWSignatureNode.super.apply( this, arguments );
|
|
|
|
// Mixin constructors
|
|
ve.ce.GeneratedContentNode.call( this );
|
|
ve.ce.FocusableNode.call( this );
|
|
|
|
// DOM changes
|
|
this.$element.addClass( 've-ce-mwSignatureNode' );
|
|
|
|
if ( this.isGenerating() ) {
|
|
// Use an initial rendering of '~~~~' as a placeholder to avoid
|
|
// the width changing when using the Sequence.
|
|
this.$element.text( '~~~~' );
|
|
}
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
OO.inheritClass( ve.ce.MWSignatureNode, ve.ce.LeafNode );
|
|
OO.mixinClass( ve.ce.MWSignatureNode, ve.ce.GeneratedContentNode );
|
|
OO.mixinClass( ve.ce.MWSignatureNode, ve.ce.FocusableNode );
|
|
|
|
/* Static Properties */
|
|
|
|
ve.ce.MWSignatureNode.static.name = 'mwSignature';
|
|
|
|
ve.ce.MWSignatureNode.static.tagName = 'span';
|
|
|
|
ve.ce.MWSignatureNode.static.primaryCommandName = 'mwSignature';
|
|
|
|
ve.ce.MWSignatureNode.static.liveSignatures = [];
|
|
|
|
// Set a description for focusable node tooltip
|
|
ve.ce.MWSignatureNode.static.getDescription = function () {
|
|
return ve.msg( 'visualeditor-mwsignature-tool' );
|
|
};
|
|
|
|
// Update the timestamp on inserted signatures every minute.
|
|
setInterval( function () {
|
|
var liveSignatures = ve.ce.MWSignatureNode.static.liveSignatures;
|
|
|
|
var updatedSignatures = [];
|
|
for ( var i = 0; i < liveSignatures.length; i++ ) {
|
|
var sig = liveSignatures[ i ];
|
|
try {
|
|
sig.forceUpdate();
|
|
updatedSignatures.push( sig );
|
|
} catch ( er ) {
|
|
// Do nothing
|
|
}
|
|
}
|
|
// Stop updating signatures that failed once
|
|
ve.ce.MWSignatureNode.static.liveSignatures = updatedSignatures;
|
|
}, 60 * 1000 );
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.ce.MWSignatureNode.prototype.onSetup = function () {
|
|
// Parent method
|
|
ve.ce.MWSignatureNode.super.prototype.onSetup.call( this );
|
|
|
|
// Keep track for regular updating of timestamp
|
|
this.constructor.static.liveSignatures.push( this );
|
|
};
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.ce.MWSignatureNode.prototype.onTeardown = function () {
|
|
var liveSignatures = this.constructor.static.liveSignatures;
|
|
|
|
// Parent method
|
|
ve.ce.MWSignatureNode.super.prototype.onTeardown.call( this );
|
|
|
|
// Stop tracking
|
|
var index = liveSignatures.indexOf( this );
|
|
if ( index !== -1 ) {
|
|
liveSignatures.splice( index, 1 );
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @inheritdoc ve.ce.GeneratedContentNode
|
|
*/
|
|
ve.ce.MWSignatureNode.prototype.generateContents = function () {
|
|
var doc = this.getModel().getDocument();
|
|
var abortable, aborted;
|
|
var abortedPromise = ve.createDeferred().reject( 'http',
|
|
{ textStatus: 'abort', exception: 'abort' } ).promise();
|
|
|
|
function abort() {
|
|
aborted = true;
|
|
if ( abortable && abortable.abort ) {
|
|
abortable.abort();
|
|
}
|
|
}
|
|
|
|
// Acquire a temporary user username before previewing, so that signatures
|
|
// display the temp user instead of IP user. (T331397)
|
|
return mw.user.acquireTempUserName()
|
|
.then( function () {
|
|
if ( aborted ) {
|
|
return abortedPromise;
|
|
}
|
|
|
|
// We must have only one top-level node, this is the easiest way.
|
|
var wikitext = '<span>~~~~</span>';
|
|
|
|
// Parsoid doesn't support pre-save transforms. PHP parser doesn't support Parsoid's
|
|
// meta attributes (that may or may not be required).
|
|
// We could try hacking up one (or even both) of these, but just calling the two parsers
|
|
// in order seems slightly saner.
|
|
return ( abortable = ve.init.target.getContentApi( doc ).post( {
|
|
action: 'parse',
|
|
text: wikitext,
|
|
contentmodel: 'wikitext',
|
|
prop: 'text',
|
|
onlypst: true
|
|
} ) );
|
|
} )
|
|
.then( function ( pstResponse ) {
|
|
if ( aborted ) {
|
|
return abortedPromise;
|
|
}
|
|
var wikitext = ve.getProp( pstResponse, 'parse', 'text' );
|
|
if ( !wikitext ) {
|
|
return ve.createDeferred().reject();
|
|
}
|
|
return ( abortable = ve.init.target.parseWikitextFragment( wikitext, true, doc ) );
|
|
} )
|
|
.then( function ( parseResponse ) {
|
|
if ( aborted ) {
|
|
return abortedPromise;
|
|
}
|
|
if ( ve.getProp( parseResponse, 'visualeditor', 'result' ) !== 'success' ) {
|
|
return ve.createDeferred().reject();
|
|
}
|
|
// Simplified case of template rendering, don't need to worry about filtering etc
|
|
return $( parseResponse.visualeditor.content ).contents().toArray();
|
|
} )
|
|
.promise( { abort: abort } );
|
|
};
|
|
|
|
/* Registration */
|
|
|
|
ve.ce.nodeFactory.register( ve.ce.MWSignatureNode );
|