mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-11-26 15:21:51 +00:00
cb10316a7b
Change-Id: I0d719905203d24a545d9b7dcea5bb9d032f10a76
285 lines
8.2 KiB
JavaScript
285 lines
8.2 KiB
JavaScript
/*!
|
|
* VisualEditor MediaWiki Initialization CollabTarget class.
|
|
*
|
|
* @copyright 2011-2016 VisualEditor Team and others; see AUTHORS.txt
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
*/
|
|
|
|
/**
|
|
* MediaWiki mobile article target.
|
|
*
|
|
* @class
|
|
* @extends ve.init.mw.Target
|
|
*
|
|
* @constructor
|
|
* @param {mw.Title} title Page sub-title
|
|
* @param {rebaserUrl} string Rebaser server URL
|
|
* @param {Object} [config] Configuration options
|
|
* @cfg {mw.Title} [importTitle] Title to import
|
|
*/
|
|
ve.init.mw.CollabTarget = function VeInitMwCollabTarget( title, rebaserUrl, config ) {
|
|
config = config || {};
|
|
config.toolbarConfig = $.extend( {
|
|
shadow: true,
|
|
actions: true,
|
|
floatable: true
|
|
}, config.toolbarConfig );
|
|
|
|
this.title = title;
|
|
this.rebaserUrl = rebaserUrl;
|
|
this.importTitle = config.importTitle || null;
|
|
|
|
// Parent constructor
|
|
ve.init.mw.CollabTarget.super.call( this, config );
|
|
|
|
// HACK: Disable history commands until supported (T185706)
|
|
ve.ui.commandRegistry.unregister( 'undo' );
|
|
ve.ui.commandRegistry.unregister( 'redo' );
|
|
|
|
// HACK: Disable references until supported (T194838)
|
|
ve.ui.commandRegistry.unregister( 'reference' );
|
|
ve.ui.commandRegistry.unregister( 'referencesList' );
|
|
ve.ui.commandRegistry.unregister( 'citefromid' );
|
|
|
|
this.$originalContent = $( '<div>' ).addClass( 've-init-mw-desktopArticleTarget-originalContent' );
|
|
this.$editableContent = $( '#mw-content-text' );
|
|
|
|
// Initialization
|
|
this.$element.addClass( 've-init-mw-articleTarget ve-init-mw-desktopArticleTarget ve-init-mw-collabTarget' ).append( this.$originalContent );
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
OO.inheritClass( ve.init.mw.CollabTarget, ve.init.mw.Target );
|
|
|
|
/* Static Properties */
|
|
|
|
ve.init.mw.CollabTarget.static.name = 'collab';
|
|
|
|
ve.init.mw.CollabTarget.static.trackingName = 'collab';
|
|
|
|
ve.init.mw.CollabTarget.static.toolbarGroups = ve.copy( ve.init.mw.CollabTarget.static.toolbarGroups );
|
|
ve.init.mw.CollabTarget.static.toolbarGroups.splice( 4, 0, {
|
|
name: 'commentAnnotation',
|
|
include: [ 'commentAnnotation' ]
|
|
} );
|
|
// HACK: Disable references until supported (T194838)
|
|
ve.init.mw.CollabTarget.static.toolbarGroups = ve.init.mw.CollabTarget.static.toolbarGroups.filter( function ( group ) {
|
|
return group.name !== 'reference';
|
|
} );
|
|
|
|
ve.init.mw.CollabTarget.static.actionGroups = [
|
|
{
|
|
name: 'help',
|
|
include: [ 'help' ]
|
|
},
|
|
{
|
|
name: 'pageMenu',
|
|
type: 'list',
|
|
icon: 'menu',
|
|
indicator: null,
|
|
title: ve.msg( 'visualeditor-pagemenu-tooltip' ),
|
|
include: [ 'changeDirectionality', 'findAndReplace' ]
|
|
},
|
|
{
|
|
name: 'authorList',
|
|
include: [ 'authorList' ]
|
|
}
|
|
];
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* Page modifications after editor load.
|
|
*/
|
|
ve.init.mw.CollabTarget.prototype.transformPage = function () {
|
|
this.$originalContent.append( this.$element.siblings() );
|
|
};
|
|
|
|
/**
|
|
* Page modifications after editor teardown.
|
|
*/
|
|
ve.init.mw.CollabTarget.prototype.restorePage = function () {
|
|
this.$element.parent().append( this.$originalContent.children() );
|
|
$( '#contentSub' ).empty();
|
|
};
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.init.mw.CollabTarget.prototype.surfaceReady = function () {
|
|
var exportButton,
|
|
surfaceView = this.getSurface().getView(),
|
|
toolbar = this.getToolbar();
|
|
|
|
// Parent method
|
|
ve.init.mw.CollabTarget.super.prototype.surfaceReady.apply( this, arguments );
|
|
|
|
this.getSurface().getView().focus();
|
|
|
|
exportButton = new OO.ui.ButtonWidget( {
|
|
label: ve.msg( 'visualeditor-rebase-client-export' ),
|
|
flags: [ 'progressive', 'primary' ]
|
|
} );
|
|
exportButton.connect( this, { click: 'onExportButtonClick' } );
|
|
|
|
toolbar.$actions.append( exportButton.$element );
|
|
toolbar.initialize();
|
|
|
|
surfaceView.focus();
|
|
};
|
|
|
|
/**
|
|
* Handle click events from the export button
|
|
*/
|
|
ve.init.mw.CollabTarget.prototype.onExportButtonClick = function () {
|
|
var surface = this.getSurface(),
|
|
windowAction = ve.ui.actionFactory.create( 'window', surface );
|
|
windowAction.open( 'mwExportWikitext', { surface: surface } );
|
|
};
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.init.mw.CollabTarget.prototype.attachToolbar = function () {
|
|
this.toolbar.$element.addClass(
|
|
've-init-mw-desktopArticleTarget-toolbar ve-init-mw-desktopArticleTarget-toolbar-open ve-init-mw-desktopArticleTarget-toolbar-opened'
|
|
);
|
|
this.$element.prepend( this.toolbar.$element );
|
|
this.toolbar.initialize();
|
|
};
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.init.mw.CollabTarget.prototype.setSurface = function ( surface ) {
|
|
var synchronizer, surfaceView,
|
|
importDeferred = $.Deferred(),
|
|
target = this;
|
|
|
|
if ( surface !== this.surface ) {
|
|
this.$editableContent.after( surface.$element );
|
|
|
|
surfaceView = surface.getView();
|
|
|
|
synchronizer = new ve.dm.SurfaceSynchronizer(
|
|
surface.getModel(),
|
|
this.title.toString(),
|
|
{
|
|
server: this.rebaserUrl,
|
|
// TODO: server could communicate with MW (via oauth?) to know the
|
|
// current-user's name. Disable changing name if logged in?
|
|
// Communicate an I-am-a-valid-user flag to other clients?
|
|
defaultName: mw.user.isAnon() ? mw.user.getName() : undefined
|
|
}
|
|
);
|
|
|
|
synchronizer.once( 'initDoc', function () {
|
|
var surfaceModel, initPromise, title;
|
|
|
|
if ( target.importTitle && !surface.getModel().getDocument().getCompleteHistoryLength() ) {
|
|
initPromise = mw.libs.ve.targetLoader.requestParsoidData( target.importTitle.toString(), { targetName: 'collabpad' } ).then( function ( response ) {
|
|
var doc, dmDoc, fragment,
|
|
data = response.visualeditor;
|
|
|
|
if ( data && data.content ) {
|
|
doc = target.constructor.static.parseDocument( data.content );
|
|
dmDoc = target.constructor.static.createModelFromDom( doc );
|
|
fragment = surface.getModel().getLinearFragment( new ve.Range( 0, 2 ) );
|
|
fragment.insertDocument( dmDoc );
|
|
|
|
target.etag = data.etag;
|
|
target.baseTimeStamp = data.basetimestamp;
|
|
target.startTimeStamp = data.starttimestamp;
|
|
target.revid = data.oldid;
|
|
|
|
// Store the document metadata as a hidden meta item
|
|
fragment.collapseToEnd().insertContent( [
|
|
{
|
|
type: 'alienMeta',
|
|
attributes: {
|
|
importedDocument: {
|
|
title: target.importTitle.toString(),
|
|
etag: target.etag,
|
|
baseTimeStamp: target.baseTimeStamp,
|
|
startTimeStamp: target.startTimeStamp,
|
|
revid: target.revid
|
|
}
|
|
}
|
|
},
|
|
{ type: '/alienMeta' }
|
|
] );
|
|
surface.getModel().selectFirstContentOffset();
|
|
} else {
|
|
// Import failed
|
|
return $.Deferred().reject( 'No content for ' + target.importTitle ).promise();
|
|
}
|
|
} );
|
|
} else {
|
|
// No import, or history already exists
|
|
initPromise = $.Deferred().resolve().promise();
|
|
|
|
// Look for import metadata in document
|
|
surfaceModel = target.getSurface().getModel();
|
|
surfaceModel.getMetaList().getItemsInGroup( 'misc' ).some( function ( item ) {
|
|
var importedDocument = item.getAttribute( 'importedDocument' );
|
|
if ( importedDocument ) {
|
|
target.importTitle = mw.Title.newFromText( importedDocument.title );
|
|
target.etag = importedDocument.etag;
|
|
target.baseTimeStamp = importedDocument.baseTimeStamp;
|
|
target.startTimeStamp = importedDocument.startTimeStamp;
|
|
target.revid = importedDocument.revid;
|
|
return true;
|
|
}
|
|
} );
|
|
}
|
|
initPromise.fail( function ( err ) {
|
|
setTimeout( function () {
|
|
throw new Error( err );
|
|
} );
|
|
} );
|
|
initPromise.always( function () {
|
|
surface.getModel().selectFirstContentOffset();
|
|
// Resolve progress bar
|
|
importDeferred.resolve();
|
|
if ( ( title = target.getImportTitle() ) ) {
|
|
$( '#contentSub' ).html(
|
|
ve.htmlMsg(
|
|
'collabpad-import-subtitle',
|
|
$( '<a>' ).attr( 'href', title.getUrl() ).text( title.getMainText() )
|
|
)
|
|
);
|
|
ve.targetLinksToNewWindow( $( '#contentSub' )[ 0 ] );
|
|
} else {
|
|
$( '#contentSub' ).empty();
|
|
}
|
|
} );
|
|
} );
|
|
|
|
surfaceView.setSynchronizer( synchronizer, importDeferred.promise() );
|
|
}
|
|
|
|
// Parent method
|
|
ve.init.mw.CollabTarget.super.prototype.setSurface.apply( this, arguments );
|
|
};
|
|
|
|
/**
|
|
* Get the title of the imported document, if there was one
|
|
*
|
|
* @return {mw.Title|null} Title of imported document
|
|
*/
|
|
ve.init.mw.CollabTarget.prototype.getImportTitle = function () {
|
|
return this.importTitle;
|
|
};
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.init.mw.CollabTarget.prototype.getPageName = function () {
|
|
return this.getImportTitle() || this.pageName;
|
|
};
|
|
|
|
/* Registration */
|
|
|
|
ve.init.mw.targetFactory.register( ve.init.mw.CollabTarget );
|