mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-12-02 01:46:47 +00:00
51e096d6f2
Use OOJS-UI's newly-extended paged dialogs (in e08eb2a03b) to refactor how the Transclusion and Meta dialogs work, splitting out the code for each of the panels into its own file and simplifying extensibility. The Meta dialog (ve.ui.MWMetaDialog) now has two self-managing panels: * ve.ui.MWCategoriesPage for categories and the default sort key * ve.ui.MWLanguagesPage for language links The Transclusion dialog (ve.ui.MWTransclusionDialog) now has four: * ve.ui.MWTemplatePage for a template's primary panel * ve.ui.MWTemplateParameterPage for each parameter of a template * ve.ui.MWTemplatePlaceholderPage for a placeholder to insert a template * ve.ui.MWTransclusionContentPage for non-template transclusion Additionally, the Transclusion dialog has been slightly cleaned up: * Replace add/remove events with replace events in transclusion model * Actually return and resolve a promise (as documented) * Get rid of "origin" info in template models * Add method for adding required parts TODO: * Decide how and when we will choose between advanced transclusion and template dialogs * Work out design issues with how template descriptions will be visible and how adding parameters will work if only showing parameters in outline * Add preview to template dialog * Consider ways to further improve pages for use in continuous mode WARNING: * Right now the template dialog gets overridden by the advanced transclusion dialog because they have the same symbolic name and the latter is registered later than the former. To test the template dialog, just change the symbolic name of the advanced transclusion dialog. Change-Id: I51e74b322aec9a4c3918e6f792bdb3d318060979
285 lines
6.7 KiB
JavaScript
285 lines
6.7 KiB
JavaScript
/*!
|
|
* VisualEditor DataModel MWTemplateModel class.
|
|
*
|
|
* @copyright 2011-2013 VisualEditor Team and others; see AUTHORS.txt
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
*/
|
|
|
|
/*global mw*/
|
|
|
|
/**
|
|
* MediaWiki template model.
|
|
*
|
|
* @class
|
|
* @extends ve.dm.MWTransclusionPartModel
|
|
*
|
|
* @constructor
|
|
* @param {ve.dm.MWTransclusionModel} transclusion Transclusion
|
|
* @param {Object} target Template target
|
|
* @param {string} target.wt Original wikitext of target
|
|
* @param {string} [target.href] Hypertext reference to target
|
|
*/
|
|
ve.dm.MWTemplateModel = function VeDmMWTemplateModel( transclusion, target ) {
|
|
// Parent constructor
|
|
ve.dm.MWTransclusionPartModel.call( this, transclusion );
|
|
|
|
// Properties
|
|
this.target = target;
|
|
|
|
// TODO: Either here or in uses of this constructor we need to validate the title
|
|
this.title = ( target.href && target.href.replace( /^(\.\.?\/)*/, '' ) ) || null;
|
|
this.sequence = null;
|
|
this.params = {};
|
|
this.spec = new ve.dm.MWTemplateSpecModel( this );
|
|
this.originalData = null;
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
OO.inheritClass( ve.dm.MWTemplateModel, ve.dm.MWTransclusionPartModel );
|
|
|
|
/* Events */
|
|
|
|
/**
|
|
* @event add
|
|
* @param {ve.dm.MWTemplateParameterModel} param Added param
|
|
*/
|
|
|
|
/**
|
|
* @event remove
|
|
* @param {ve.dm.MWTemplateParameterModel} param Removed param
|
|
*/
|
|
|
|
/* Static Methods */
|
|
|
|
/**
|
|
* Create from data.
|
|
*
|
|
* Data is in the format provided by Parsoid.
|
|
*
|
|
* @param {ve.dm.MWTransclusionModel} transclusion Transclusion template is in
|
|
* @param {Object} data Template data
|
|
* @returns {ve.dm.MWTemplateModel} New template model
|
|
*/
|
|
ve.dm.MWTemplateModel.newFromData = function ( transclusion, data ) {
|
|
var key,
|
|
template = new ve.dm.MWTemplateModel( transclusion, data.target, 'data' );
|
|
|
|
for ( key in data.params ) {
|
|
template.addParameter(
|
|
new ve.dm.MWTemplateParameterModel( template, key, data.params[key].wt, 'data' )
|
|
);
|
|
}
|
|
|
|
template.setOriginalData( data );
|
|
|
|
return template;
|
|
};
|
|
|
|
/**
|
|
* Create from name.
|
|
*
|
|
* Name is equivalent to what would be entered between double brackets, defaulting to the Template
|
|
* namespace, using a leading colon to access other namespaces.
|
|
*
|
|
* @param {ve.dm.MWTransclusionModel} transclusion Transclusion template is in
|
|
* @param {string} name Template name
|
|
* @returns {ve.dm.MWTemplateModel} New template model
|
|
*/
|
|
ve.dm.MWTemplateModel.newFromName = function ( transclusion, name ) {
|
|
var href = name;
|
|
|
|
if ( href.charAt( 0 ) !== ':' ) {
|
|
href = mw.config.get( 'wgFormattedNamespaces' )[10] + ':' + href;
|
|
}
|
|
|
|
// TODO: Do we need to account for the title being invalid?
|
|
href = new mw.Title( href ).getPrefixedText();
|
|
|
|
return new ve.dm.MWTemplateModel( transclusion, { 'href': href, 'wt': name }, 'user' );
|
|
};
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* Get template target.
|
|
*
|
|
* @returns {Object} Template target
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.getTarget = function () {
|
|
return this.target;
|
|
};
|
|
|
|
/**
|
|
* Get template title.
|
|
*
|
|
* @returns {string|null} Template title, if available
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.getTitle = function () {
|
|
return this.title;
|
|
};
|
|
|
|
/**
|
|
* Get template specification.
|
|
*
|
|
* @returns {ve.dm.MWTemplateSpecModel} Template specification
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.getSpec = function () {
|
|
return this.spec;
|
|
};
|
|
|
|
/**
|
|
* Get all params.
|
|
*
|
|
* @returns {Object.<string,ve.dm.MWTemplateParameterModel>} Parameters keyed by name
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.getParameters = function () {
|
|
return this.params;
|
|
};
|
|
|
|
/**
|
|
* Get a parameter.
|
|
*
|
|
* @param {string} name Parameter name
|
|
* @returns {ve.dm.MWTemplateParameterModel} Parameter
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.getParameter = function ( name ) {
|
|
return this.params[name];
|
|
};
|
|
|
|
/**
|
|
* Check if a parameter exists.
|
|
*
|
|
* @param {string} name Parameter name
|
|
* @returns {boolean} Parameter exists
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.hasParameter = function ( name ) {
|
|
var i, len, primaryName, names;
|
|
|
|
// Check if name (which may be an alias) is present in the template
|
|
if ( this.params[name] ) {
|
|
return true;
|
|
}
|
|
|
|
// Check if the name is known at all
|
|
if ( this.spec.isParameterKnown( name ) ) {
|
|
primaryName = this.spec.getParameterName( name );
|
|
// Check for primary name (may be the same as name)
|
|
if ( this.params[primaryName] ) {
|
|
return true;
|
|
}
|
|
// Check for other aliases (may include name)
|
|
names = this.spec.getParameterAliases( primaryName );
|
|
for ( i = 0, len = names.length; i < len; i++ ) {
|
|
if ( this.params[names[i]] ) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
/**
|
|
* Get ordered list of parameter names.
|
|
*
|
|
* Numeric names, whether strings or real numbers, are placed at the begining, followed by
|
|
* alphabetically sorted names.
|
|
*
|
|
* @returns {string[]} List of parameter names
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.getParameterNames = function () {
|
|
if ( !this.sequence ) {
|
|
this.sequence = ve.getObjectKeys( this.params ).sort( function ( a, b ) {
|
|
var aIsNaN = isNaN( a ),
|
|
bIsNaN = isNaN( b );
|
|
if ( aIsNaN && bIsNaN ) {
|
|
// Two strings
|
|
return a < b ? -1 : a === b ? 0 : 1;
|
|
}
|
|
if ( aIsNaN ) {
|
|
// A is a string
|
|
return 1;
|
|
}
|
|
if ( bIsNaN ) {
|
|
// B is a string
|
|
return -1;
|
|
}
|
|
// Two numbers
|
|
return a - b;
|
|
} );
|
|
}
|
|
return this.sequence;
|
|
};
|
|
|
|
/**
|
|
* Add a parameter to template.
|
|
*
|
|
* @param {ve.dm.MWTemplateParameterModel} param Parameter to add
|
|
* @fires add
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.addParameter = function ( param ) {
|
|
var name = param.getName();
|
|
this.sequence = null;
|
|
this.params[name] = param;
|
|
this.spec.fill();
|
|
this.emit( 'add', param );
|
|
};
|
|
|
|
/**
|
|
* Remove parameter from template.
|
|
*
|
|
* @param {ve.dm.MWTemplateParameterModel} param Parameter to remove
|
|
* @fires remove
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.removeParameter = function ( param ) {
|
|
if ( param ) {
|
|
this.sequence = null;
|
|
delete this.params[param.getName()];
|
|
this.emit( 'remove', param );
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Add all non-existing required parameters, if any.
|
|
*
|
|
* @method
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.addRequiredParameters = function () {
|
|
var i, len,
|
|
spec = this.getSpec(),
|
|
names = spec.getParameterNames();
|
|
|
|
for ( i = 0, len = names.length; i < len; i++ ) {
|
|
if ( !this.params[name] && spec.isParameterRequired( names[i] ) ) {
|
|
this.addParameter( new ve.dm.MWTemplateParameterModel( this, names[i] ) );
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Set original data, to be used as a base for serialization.
|
|
*
|
|
* @returns {Object} Template data
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.setOriginalData = function ( data ) {
|
|
this.originalData = data;
|
|
};
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ve.dm.MWTemplateModel.prototype.serialize = function () {
|
|
var name,
|
|
template = ve.extendObject(
|
|
this.originalData || {}, { 'target': this.getTarget(), 'params': {} }
|
|
),
|
|
params = this.getParameters();
|
|
|
|
for ( name in params ) {
|
|
template.params[params[name].getOriginalName()] = { 'wt': params[name].getValue() };
|
|
}
|
|
|
|
return { 'template': template };
|
|
};
|