mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-11-15 10:35:48 +00:00
289 lines
6.7 KiB
JavaScript
289 lines
6.7 KiB
JavaScript
/**
|
|
* Creates an ve.dm.Node object.
|
|
*
|
|
* @class
|
|
* @abstract
|
|
* @constructor
|
|
* @extends {ve.Node}
|
|
* @param {String} type Symbolic name of node type
|
|
* @param {Object} element Element object in document data
|
|
* @param {Integer} [length] Length of content data in document
|
|
*/
|
|
ve.dm.Node = function( type, element, length ) {
|
|
// Inheritance
|
|
ve.Node.call( this );
|
|
|
|
// Properties
|
|
this.type = type;
|
|
this.parent = null;
|
|
this.root = this;
|
|
this.element = element || null;
|
|
this.contentLength = length || 0;
|
|
};
|
|
|
|
/* Abstract Methods */
|
|
|
|
/**
|
|
* Creates a view for this node.
|
|
*
|
|
* @abstract
|
|
* @method
|
|
* @returns {ve.ce.Node} New item view associated with this model
|
|
*/
|
|
ve.dm.Node.prototype.createView = function() {
|
|
throw 'DocumentModelNode.createView not implemented in this subclass:' + this.constructor;
|
|
};
|
|
|
|
/**
|
|
* Gets a plain object representation of the document's data.
|
|
*
|
|
* @method
|
|
* @returns {Object} Plain object representation
|
|
*/
|
|
ve.dm.Node.prototype.getPlainObject = function() {
|
|
throw 'DocumentModelNode.getPlainObject not implemented in this subclass:' + this.constructor;
|
|
};
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* Gets the content length.
|
|
*
|
|
* @method
|
|
* @see {ve.Node.prototype.getContentLength}
|
|
* @returns {Integer} Length of content
|
|
*/
|
|
ve.dm.Node.prototype.getContentLength = function() {
|
|
return this.contentLength;
|
|
};
|
|
|
|
/**
|
|
* Gets the element length.
|
|
*
|
|
* @method
|
|
* @see {ve.Node.prototype.getElementLength}
|
|
* @returns {Integer} Length of content
|
|
*/
|
|
ve.dm.Node.prototype.getElementLength = function() {
|
|
return this.contentLength + 2;
|
|
};
|
|
|
|
/**
|
|
* Sets the content length.
|
|
*
|
|
* @method
|
|
* @param {Integer} contentLength Length of content
|
|
* @throws Invalid content length error if contentLength is less than 0
|
|
*/
|
|
ve.dm.Node.prototype.setContentLength = function( contentLength ) {
|
|
if ( contentLength < 0 ) {
|
|
throw 'Invalid content length error. Content length can not be less than 0.';
|
|
}
|
|
var diff = contentLength - this.contentLength;
|
|
this.contentLength = contentLength;
|
|
if ( this.parent ) {
|
|
this.parent.adjustContentLength( diff );
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Adjust the content length.
|
|
*
|
|
* @method
|
|
* @param {Integer} adjustment Amount to adjust content length by
|
|
* @throws Invalid adjustment error if resulting length is less than 0
|
|
*/
|
|
ve.dm.Node.prototype.adjustContentLength = function( adjustment, quiet ) {
|
|
this.contentLength += adjustment;
|
|
// Make sure the adjustment was sane
|
|
if ( this.contentLength < 0 ) {
|
|
// Reverse the adjustment
|
|
this.contentLength -= adjustment;
|
|
// Complain about it
|
|
throw 'Invalid adjustment error. Content length can not be less than 0.';
|
|
}
|
|
if ( this.parent ) {
|
|
this.parent.adjustContentLength( adjustment, true );
|
|
}
|
|
if ( !quiet ) {
|
|
this.emit( 'update' );
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Attaches this node to another as a child.
|
|
*
|
|
* @method
|
|
* @param {ve.dm.Node} parent Node to attach to
|
|
* @emits attach (parent)
|
|
*/
|
|
ve.dm.Node.prototype.attach = function( parent ) {
|
|
this.emit( 'beforeAttach', parent );
|
|
this.parent = parent;
|
|
this.setRoot( parent.getRoot() );
|
|
this.emit( 'afterAttach', parent );
|
|
};
|
|
|
|
/**
|
|
* Detaches this node from it's parent.
|
|
*
|
|
* @method
|
|
* @emits detach
|
|
*/
|
|
ve.dm.Node.prototype.detach = function() {
|
|
this.emit( 'beforeDetach' );
|
|
this.parent = null;
|
|
this.clearRoot();
|
|
this.emit( 'afterDetach' );
|
|
};
|
|
|
|
/**
|
|
* Gets a reference to this node's parent.
|
|
*
|
|
* @method
|
|
* @returns {ve.dm.Node} Reference to this node's parent
|
|
*/
|
|
ve.dm.Node.prototype.getParent = function() {
|
|
return this.parent;
|
|
};
|
|
|
|
/**
|
|
* Gets the root node in the tree this node is currently attached to.
|
|
*
|
|
* @method
|
|
* @returns {ve.dm.Node} Root node
|
|
*/
|
|
ve.dm.Node.prototype.getRoot = function() {
|
|
return this.root;
|
|
};
|
|
|
|
/**
|
|
* Sets the root node to this and all of it's children.
|
|
*
|
|
* This method is overridden by nodes with children.
|
|
*
|
|
* @method
|
|
* @param {ve.dm.Node} root Node to use as root
|
|
*/
|
|
ve.dm.Node.prototype.setRoot = function( root ) {
|
|
this.root = root;
|
|
};
|
|
|
|
/**
|
|
* Clears the root node from this and all of it's children.
|
|
*
|
|
* This method is overridden by nodes with children.
|
|
*
|
|
* @method
|
|
*/
|
|
ve.dm.Node.prototype.clearRoot = function() {
|
|
this.root = null;
|
|
};
|
|
|
|
/**
|
|
* Gets the element object.
|
|
*
|
|
* @method
|
|
* @returns {Object} Element object in linear data model
|
|
*/
|
|
ve.dm.Node.prototype.getElement = function() {
|
|
return this.element;
|
|
};
|
|
|
|
/**
|
|
* Gets the symbolic element type name.
|
|
*
|
|
* @method
|
|
* @returns {String} Symbolic name of element type
|
|
*/
|
|
ve.dm.Node.prototype.getElementType = function() {
|
|
//return this.element.type;
|
|
// We can't use this.element.type because this.element may be null
|
|
// So this function now returns this.type and should really be called
|
|
// getType()
|
|
// TODO: Do we care?
|
|
return this.type;
|
|
};
|
|
|
|
/**
|
|
* Gets an element attribute value.
|
|
*
|
|
* @method
|
|
* @returns {Mixed} Value of attribute, or null if no such attribute exists
|
|
*/
|
|
ve.dm.Node.prototype.getElementAttribute = function( key ) {
|
|
if ( this.element && this.element.attributes && key in this.element.attributes ) {
|
|
return this.element.attributes[key];
|
|
}
|
|
return null;
|
|
};
|
|
|
|
/**
|
|
* Gets all element data, including the element opening, closing and it's contents.
|
|
*
|
|
* @method
|
|
* @returns {Array} Element data
|
|
*/
|
|
ve.dm.Node.prototype.getElementData = function() {
|
|
// Get reference to the document, which might be this node but otherwise should be this.root
|
|
var root = this.type === 'document' ?
|
|
this : ( this.root && this.root.type === 'document' ? this.root : null );
|
|
if ( root ) {
|
|
return root.getElementDataFromNode( this );
|
|
}
|
|
return [];
|
|
};
|
|
|
|
/**
|
|
* Gets content data within a given range.
|
|
*
|
|
* @method
|
|
* @param {ve.Range} [range] Range of content to get
|
|
* @returns {Array} Content data
|
|
*/
|
|
ve.dm.Node.prototype.getContentData = function( range ) {
|
|
// Get reference to the document, which might be this node but otherwise should be this.root
|
|
var root = this.type === 'document' ?
|
|
this : ( this.root && this.root.type === 'document' ? this.root : null );
|
|
if ( root ) {
|
|
return root.getContentDataFromNode( this, range );
|
|
}
|
|
return [];
|
|
};
|
|
|
|
/**
|
|
* Gets plain text version of the content within a specific range.
|
|
*
|
|
* Two newlines are inserted between leaf nodes.
|
|
*
|
|
* TODO: Maybe do something more adaptive with newlines
|
|
*
|
|
* @method
|
|
* @param {ve.Range} [range] Range of text to get
|
|
* @returns {String} Text within given range
|
|
*/
|
|
ve.dm.Node.prototype.getContentText = function( range ) {
|
|
var content = this.getContentData( range );
|
|
// Copy characters
|
|
var text = '',
|
|
element = false;
|
|
for ( var i = 0, length = content.length; i < length; i++ ) {
|
|
if ( typeof content[i].type === 'string' ) {
|
|
if ( i ) {
|
|
element = true;
|
|
}
|
|
} else {
|
|
if ( element ) {
|
|
text += '\n\n';
|
|
element = false;
|
|
}
|
|
text += typeof content[i] === 'string' ? content[i] : content[i][0];
|
|
}
|
|
}
|
|
return text;
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
ve.extendClass( ve.dm.Node, ve.Node );
|