mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-11-25 06:46:26 +00:00
280c85d8e7
This resolves a TODO * Added logic to support passing an argument into annotation constructor which is used as the data property (reusing the element argument) * Updated documentation * Simplified instantiation of annotations Change-Id: I142b8fa3883bf70c896a2a568088d833814ef2dc
178 lines
5.5 KiB
JavaScript
178 lines
5.5 KiB
JavaScript
/*!
|
|
* VisualEditor DataModel Annotation class.
|
|
*
|
|
* @copyright 2011-2012 VisualEditor Team and others; see AUTHORS.txt
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
*/
|
|
|
|
/**
|
|
* Generic DataModel annotation.
|
|
*
|
|
* This is an abstract class, annotations should extend this and call this constructor from their
|
|
* constructor. You should not instantiate this class directly.
|
|
*
|
|
* Annotations in the linear model are instances of subclasses of this class. Subclasses are
|
|
* required to have a constructor with the same signature.
|
|
*
|
|
* this.htmlTagName and this.htmlAttributes are private to the base class, subclasses must not
|
|
* use them. Any information from the HTML element that is needed later should be extracted into
|
|
* this.data by overriding getAnnotationData(). Subclasses can read from this.data but must not
|
|
* write to it directly.
|
|
*
|
|
* @class
|
|
* @constructor
|
|
* @param {HTMLElement|Object} [element] HTML element the annotation was converted from, if any, or
|
|
* an object to copy into the annotation's data property
|
|
*/
|
|
ve.dm.Annotation = function VeDmAnnotation( element ) {
|
|
this.name = this.constructor.static.name; // Needed for proper hashing
|
|
this.data = {};
|
|
if ( ve.isPlainObject( element ) ) {
|
|
this.data = ve.copyObject( element );
|
|
} else if ( element && element.nodeType === Node.ELEMENT_NODE ) {
|
|
this.htmlTagName = element.nodeName.toLowerCase();
|
|
this.htmlAttributes = ve.getDomAttributes( element );
|
|
this.data = this.getAnnotationData( element );
|
|
}
|
|
};
|
|
|
|
/* Static properties */
|
|
|
|
/**
|
|
* Object containing static properties.
|
|
*
|
|
* ve#inheritClass contains special logic to make sure these properties are inherited by subclasses.
|
|
* @static
|
|
* @property
|
|
* @inheritable
|
|
*/
|
|
ve.dm.Annotation.static = {};
|
|
|
|
/**
|
|
* Symbolic name for the annotation class.
|
|
*
|
|
* Must be set to a unique string by every subclass.
|
|
*
|
|
* @static
|
|
* @property {string} [static.name=null]
|
|
* @inheritable
|
|
*/
|
|
ve.dm.Annotation.static.name = null;
|
|
|
|
/**
|
|
* Array of HTML tag names that the annotation should be a match candidate for.
|
|
*
|
|
* Empty array means none, null means any.
|
|
*
|
|
* @see ve.dm.ModelRegistry
|
|
*
|
|
* @static
|
|
* @property {Array} static.matchTagNames
|
|
* @inheritable
|
|
*/
|
|
ve.dm.Annotation.static.matchTagNames = null;
|
|
|
|
/**
|
|
* Array of RDFa types that the annotation should be a match candidate for.
|
|
*
|
|
* Empty array means none, null means any.
|
|
*
|
|
* @see ve.dm.ModelRegistry
|
|
*
|
|
* @static
|
|
* @property {Array} static.matchRdfaType
|
|
* @inheritable
|
|
*/
|
|
ve.dm.Annotation.static.matchRdfaTypes = null;
|
|
|
|
/**
|
|
* Optional function to determine whether the annotation should match a given element.
|
|
*
|
|
* Takes an HTMLElement and returns true or false.
|
|
*
|
|
* This function is only called if this annotation has a chance of "winning"; see
|
|
* ve.dm.ModelRegistry for more information about element matching.
|
|
* If set to null, this property is ignored. Setting this to null is not the same as unconditionally
|
|
* returning true, because the presence or absence of a matchFunction affects the annotation's
|
|
* specificity.
|
|
*
|
|
* NOTE: This function is NOT a method, within this function "this" will not refer to an instance
|
|
* of this class (or to anything reasonable, for that matter).
|
|
* @static
|
|
* @property {Function} static.matchFunction
|
|
* @inheritable
|
|
*/
|
|
ve.dm.Annotation.static.matchFunction = null;
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* Get annotation data for the linear model.
|
|
*
|
|
* Called when building a new annotation from an HTML element.
|
|
*
|
|
* This annotation data object is completely free-form. It's stored in the linear model, it can be
|
|
* manipulated by UI widgets, and you access it as this.data in toHTML() on the way out and in
|
|
* renderHTML() for rendering. It is also the ONLY data you can reliably use in those contexts, so
|
|
* any information from the HTML element that you'll need later should be extracted into the data
|
|
* object here.
|
|
*
|
|
* @method
|
|
* @param {HTMLElement} element HTML element the annotation will represent
|
|
* @returns {Object} Annotation data
|
|
*/
|
|
ve.dm.Annotation.prototype.getAnnotationData = function () {
|
|
return {};
|
|
};
|
|
|
|
/**
|
|
* Convert the annotation back to HTML for output purposes.
|
|
*
|
|
* You should only use this.data here, you cannot reliably use any of the other properties.
|
|
* The default action is to restore the original HTML element's tag name and attributes (if this
|
|
* annotation was created based on an element). If a subclass wants to do this too (this is common),
|
|
* it should call its parent's implementation first, then manipulate the return value.
|
|
*
|
|
* @method
|
|
* @returns {Object} Object with 'tag' (tag name) and 'attributes' (object with attribute key/values)
|
|
*/
|
|
ve.dm.Annotation.prototype.toHTML = function () {
|
|
return {
|
|
'tag': this.htmlTagName || '',
|
|
'attributes': this.htmlAttributes || {}
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Convert the annotation to HTML for rendering purposes.
|
|
*
|
|
* By default, this just calls #toHTML, but it may be customized if the rendering should be
|
|
* different from the output.
|
|
*
|
|
* @see #toHTML
|
|
*
|
|
* @method
|
|
* @returns {Object} Object with 'tag' (tag name) and 'attributes' (object with attribute key/values)
|
|
*/
|
|
ve.dm.Annotation.prototype.renderHTML = function () {
|
|
return this.toHTML();
|
|
};
|
|
|
|
/**
|
|
* Get a hash of the annotation.
|
|
*
|
|
* This is a custom hash function for ve#getHash and should not be overridden by subclasses.
|
|
*
|
|
* @method
|
|
* @returns {string} Hash string
|
|
*/
|
|
ve.dm.Annotation.prototype.getHash = function () {
|
|
var keys = [ 'name', 'data' ], obj = {}, i;
|
|
for ( i = 0; i < keys.length; i++ ) {
|
|
if ( this[keys[i]] !== undefined ) {
|
|
obj[keys[i]] = this[keys[i]];
|
|
}
|
|
}
|
|
return ve.getHash( obj );
|
|
};
|