/*! * VisualEditor DataModel MWExtensionNode class. * * @copyright 2011-2017 VisualEditor Team and others; see AUTHORS.txt * @license The MIT License (MIT); see LICENSE.txt */ /** * DataModel MediaWiki extension node. * * @class * @abstract * @extends ve.dm.LeafNode * @mixins ve.dm.FocusableNode * @mixins ve.dm.GeneratedContentNode * * @constructor */ ve.dm.MWExtensionNode = function VeDmMWExtensionNode() { // Parent constructor ve.dm.MWExtensionNode.super.apply( this, arguments ); // Mixin constructors ve.dm.GeneratedContentNode.call( this ); ve.dm.FocusableNode.call( this ); }; /* Inheritance */ OO.inheritClass( ve.dm.MWExtensionNode, ve.dm.LeafNode ); OO.mixinClass( ve.dm.MWExtensionNode, ve.dm.FocusableNode ); OO.mixinClass( ve.dm.MWExtensionNode, ve.dm.GeneratedContentNode ); /* Static members */ ve.dm.MWExtensionNode.static.enableAboutGrouping = true; ve.dm.MWExtensionNode.static.matchTagNames = null; ve.dm.MWExtensionNode.static.childNodeTypes = []; /** * HTML tag name. * @static * @property {string} * @inheritable */ ve.dm.MWExtensionNode.static.tagName = null; /** * Name of the extension and the parser tag name. * @static * @property {string} * @inheritable */ ve.dm.MWExtensionNode.static.extensionName = null; ve.dm.MWExtensionNode.static.getMatchRdfaTypes = function () { return [ 'mw:Extension/' + this.extensionName ]; }; ve.dm.MWExtensionNode.static.toDataElement = function ( domElements, converter ) { var dataElement, mwDataJSON = domElements[ 0 ].getAttribute( 'data-mw' ), mwData = mwDataJSON ? JSON.parse( mwDataJSON ) : {}; dataElement = { type: this.name, attributes: { mw: mwData, originalMw: mwDataJSON } }; this.storeGeneratedContents( dataElement, domElements, converter.getStore() ); return dataElement; }; /** */ ve.dm.MWExtensionNode.static.cloneElement = function () { // Parent method var clone = ve.dm.MWExtensionNode.super.static.cloneElement.apply( this, arguments ); delete clone.attributes.originalMw; return clone; }; ve.dm.MWExtensionNode.static.toDomElements = function ( dataElement, doc, converter ) { var el, els, value, store = converter.getStore(), originalMw = dataElement.attributes.originalMw; // If the transclusion is unchanged just send back the // original DOM elements so selser can skip over it if ( dataElement.originalDomElementsIndex && originalMw && ve.compare( dataElement.attributes.mw, JSON.parse( originalMw ) ) ) { // originalDomElements is also used for CE rendering so return a copy els = ve.copyDomElements( converter.getStore().value( dataElement.originalDomElementsIndex ), doc ); } else { if ( converter.isForClipboard() && // Use getHashObjectForRendering to get the rendering from the store ( value = store.value( store.indexOfValue( null, OO.getHash( [ this.getHashObjectForRendering( dataElement ), undefined ] ) ) ) ) ) { // For the clipboard use the current DOM contents so the user has something // meaningful to paste into external applications els = ve.copyDomElements( value, doc ); } else { el = doc.createElement( this.tagName ); el.setAttribute( 'typeof', 'mw:Extension/' + this.getExtensionName( dataElement ) ); el.setAttribute( 'data-mw', JSON.stringify( dataElement.attributes.mw ) ); els = [ el ]; } } if ( converter.isForClipboard() ) { // Resolve attributes ve.resolveAttributes( $( els ), doc, ve.dm.Converter.static.computedAttributes ); } return els; }; ve.dm.MWExtensionNode.static.getHashObject = function ( dataElement ) { return { type: dataElement.type, mw: ve.copy( dataElement.attributes.mw ) }; }; /** * Get the extension's name * * Static version for toDomElements * * @static * @param {Object} dataElement Data element * @return {string} Extension name */ ve.dm.MWExtensionNode.static.getExtensionName = function () { return this.extensionName; }; ve.dm.MWExtensionNode.static.describeChanges = function ( attributeChanges, change, element ) { // HACK: Try to generate an ' has changed' message using associated tool's title // Extensions should provide more detailed change descriptions var tools = ve.ui.toolFactory.getRelatedItems( [ ve.dm.nodeFactory.createFromElement( element ) ] ); if ( tools.length ) { return [ ve.msg( 'visualeditor-changedesc-unknown', OO.ui.resolveMsg( ve.ui.toolFactory.lookup( tools[ 0 ].name ).static.title ) ) ]; } // Parent method return ve.dm.MWExtensionNode.super.static.describeChanges.apply( this, arguments ); }; ve.dm.MWExtensionNode.static.describeChange = function ( key ) { if ( key === 'originalMw' ) { return null; } // Parent method return ve.dm.MWExtensionNode.super.static.describeChange.apply( this, arguments ); }; /* Methods */ /** * Get the extension's name * * @method * @return {string} Extension name */ ve.dm.MWExtensionNode.prototype.getExtensionName = function () { return this.constructor.static.getExtensionName( this.element ); }; /** * DataModel MediaWiki inline extension node. * * @class * @abstract * @extends ve.dm.MWExtensionNode * * @constructor * @param {Object} [element] Reference to element in linear model */ ve.dm.MWInlineExtensionNode = function VeDmMWInlineExtensionNode() { // Parent constructor ve.dm.MWInlineExtensionNode.super.apply( this, arguments ); }; /* Inheritance */ OO.inheritClass( ve.dm.MWInlineExtensionNode, ve.dm.MWExtensionNode ); /* Static members */ ve.dm.MWInlineExtensionNode.static.isContent = true; ve.dm.MWInlineExtensionNode.static.tagName = 'span'; /** * DataModel MediaWiki block extension node. * * @class * @abstract * @extends ve.dm.MWExtensionNode * * @constructor * @param {Object} [element] Reference to element in linear model * @param {ve.dm.Node[]} [children] */ ve.dm.MWBlockExtensionNode = function VeDmMWBlockExtensionNode() { // Parent constructor ve.dm.MWBlockExtensionNode.super.apply( this, arguments ); }; /* Inheritance */ OO.inheritClass( ve.dm.MWBlockExtensionNode, ve.dm.MWExtensionNode ); /* Static members */ ve.dm.MWBlockExtensionNode.static.tagName = 'div';