mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-12-25 12:33:19 +00:00
425e581613
Once MW images are registered, we should remove core image support so we don't accidentally match to them (e.g. an MW inline image with an unsupported extra RDFa type). Change-Id: I1c8567346c371fe338f95b232c9ac53e009c5a46
216 lines
6.2 KiB
JavaScript
216 lines
6.2 KiB
JavaScript
/*!
|
|
* VisualEditor DataModel MWInlineImage class.
|
|
*
|
|
* @copyright 2011-2015 VisualEditor Team and others; see AUTHORS.txt
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
*/
|
|
|
|
/**
|
|
* DataModel MediaWiki image node.
|
|
*
|
|
* @class
|
|
* @extends ve.dm.LeafNode
|
|
* @mixins ve.dm.MWImageNode
|
|
*
|
|
* @constructor
|
|
* @param {Object} [element] Reference to element in linear model
|
|
*/
|
|
ve.dm.MWInlineImageNode = function VeDmMWInlineImageNode() {
|
|
// Parent constructor
|
|
ve.dm.LeafNode.apply( this, arguments );
|
|
|
|
// Mixin constructors
|
|
ve.dm.MWImageNode.call( this );
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
OO.inheritClass( ve.dm.MWInlineImageNode, ve.dm.LeafNode );
|
|
|
|
// Need to mixin base class as well
|
|
OO.mixinClass( ve.dm.MWInlineImageNode, ve.dm.GeneratedContentNode );
|
|
|
|
OO.mixinClass( ve.dm.MWInlineImageNode, ve.dm.MWImageNode );
|
|
|
|
/* Static Properties */
|
|
|
|
ve.dm.MWInlineImageNode.static.rdfaToType = {
|
|
'mw:Image': 'none',
|
|
'mw:Image/Frameless': 'frameless'
|
|
};
|
|
|
|
ve.dm.MWInlineImageNode.static.isContent = true;
|
|
|
|
ve.dm.MWInlineImageNode.static.name = 'mwInlineImage';
|
|
|
|
ve.dm.MWInlineImageNode.static.preserveHtmlAttributes = function ( attribute ) {
|
|
var attributes = [ 'typeof', 'class', 'src', 'resource', 'width', 'height', 'href' ];
|
|
return attributes.indexOf( attribute ) === -1;
|
|
};
|
|
|
|
ve.dm.MWInlineImageNode.static.matchTagNames = [ 'span' ];
|
|
|
|
ve.dm.MWInlineImageNode.static.blacklistedAnnotationTypes = [ 'link' ];
|
|
|
|
ve.dm.MWInlineImageNode.static.getMatchRdfaTypes = function () {
|
|
return Object.keys( this.rdfaToType );
|
|
};
|
|
|
|
ve.dm.MWInlineImageNode.static.allowedRdfaTypes = [ 'mw:Error' ];
|
|
|
|
ve.dm.MWInlineImageNode.static.toDataElement = function ( domElements, converter ) {
|
|
var dataElement, attributes,
|
|
$span = $( domElements[0] ),
|
|
$firstChild = $span.children().first(), // could be <span> or <a>
|
|
$img = $firstChild.children().first(),
|
|
typeofAttrs = $span.attr( 'typeof' ).split( ' ' ),
|
|
classes = $span.attr( 'class' ),
|
|
recognizedClasses = [],
|
|
errorIndex = typeofAttrs.indexOf( 'mw:Error' ),
|
|
width = $img.attr( 'width' ),
|
|
height = $img.attr( 'height' );
|
|
|
|
if ( errorIndex !== -1 ) {
|
|
typeofAttrs.splice( errorIndex, 1 );
|
|
}
|
|
|
|
attributes = {
|
|
type: this.rdfaToType[typeofAttrs[0]],
|
|
src: $img.attr( 'src' ),
|
|
resource: $img.attr( 'resource' ),
|
|
originalClasses: classes
|
|
};
|
|
|
|
if ( errorIndex !== -1 ) {
|
|
attributes.isError = true;
|
|
}
|
|
|
|
attributes.width = width !== undefined && width !== '' ? Number( width ) : null;
|
|
attributes.height = height !== undefined && height !== '' ? Number( height ) : null;
|
|
|
|
attributes.isLinked = $firstChild.is( 'a' );
|
|
if ( attributes.isLinked ) {
|
|
attributes.href = $firstChild.attr( 'href' );
|
|
}
|
|
|
|
// Extract individual classes
|
|
classes = typeof classes === 'string' ? classes.trim().split( /\s+/ ) : [];
|
|
|
|
// Deal with border flag
|
|
if ( classes.indexOf( 'mw-image-border' ) !== -1 ) {
|
|
attributes.borderImage = true;
|
|
recognizedClasses.push( 'mw-image-border' );
|
|
}
|
|
// Vertical alignment
|
|
if ( classes.indexOf( 'mw-valign-middle' ) !== -1 ) {
|
|
attributes.valign = 'middle';
|
|
recognizedClasses.push( 'mw-valign-middle' );
|
|
} else if ( classes.indexOf( 'mw-valign-baseline' ) !== -1 ) {
|
|
attributes.valign = 'baseline';
|
|
recognizedClasses.push( 'mw-valign-baseline' );
|
|
} else if ( classes.indexOf( 'mw-valign-sub' ) !== -1 ) {
|
|
attributes.valign = 'sub';
|
|
recognizedClasses.push( 'mw-valign-sub' );
|
|
} else if ( classes.indexOf( 'mw-valign-super' ) !== -1 ) {
|
|
attributes.valign = 'super';
|
|
recognizedClasses.push( 'mw-valign-super' );
|
|
} else if ( classes.indexOf( 'mw-valign-top' ) !== -1 ) {
|
|
attributes.valign = 'top';
|
|
recognizedClasses.push( 'mw-valign-top' );
|
|
} else if ( classes.indexOf( 'mw-valign-text-top' ) !== -1 ) {
|
|
attributes.valign = 'text-top';
|
|
recognizedClasses.push( 'mw-valign-text-top' );
|
|
} else if ( classes.indexOf( 'mw-valign-bottom' ) !== -1 ) {
|
|
attributes.valign = 'bottom';
|
|
recognizedClasses.push( 'mw-valign-bottom' );
|
|
} else if ( classes.indexOf( 'mw-valign-text-bottom' ) !== -1 ) {
|
|
attributes.valign = 'text-bottom';
|
|
recognizedClasses.push( 'mw-valign-text-bottom' );
|
|
} else {
|
|
attributes.valign = 'default';
|
|
}
|
|
|
|
// Border
|
|
if ( classes.indexOf( 'mw-image-border' ) !== -1 ) {
|
|
attributes.borderImage = true;
|
|
recognizedClasses.push( 'mw-image-border' );
|
|
}
|
|
|
|
// Default-size
|
|
if ( classes.indexOf( 'mw-default-size' ) !== -1 ) {
|
|
attributes.defaultSize = true;
|
|
recognizedClasses.push( 'mw-default-size' );
|
|
}
|
|
|
|
// Store unrecognized classes so we can restore them on the way out
|
|
attributes.unrecognizedClasses = OO.simpleArrayDifference( classes, recognizedClasses );
|
|
|
|
dataElement = { type: this.name, attributes: attributes };
|
|
|
|
this.storeGeneratedContents( dataElement, dataElement.attributes.src, converter.getStore() );
|
|
|
|
return dataElement;
|
|
};
|
|
|
|
ve.dm.MWInlineImageNode.static.toDomElements = function ( data, doc ) {
|
|
var firstChild,
|
|
span = doc.createElement( 'span' ),
|
|
img = doc.createElement( 'img' ),
|
|
classes = [],
|
|
originalClasses = data.attributes.originalClasses,
|
|
rdfa;
|
|
|
|
ve.setDomAttributes( img, data.attributes, [ 'src', 'width', 'height', 'resource' ] );
|
|
|
|
if ( !this.typeToRdfa ) {
|
|
this.typeToRdfa = {};
|
|
for ( rdfa in this.rdfaToType ) {
|
|
this.typeToRdfa[this.rdfaToType[rdfa]] = rdfa;
|
|
}
|
|
}
|
|
|
|
span.setAttribute( 'typeof', this.typeToRdfa[data.attributes.type] );
|
|
|
|
if ( data.attributes.defaultSize ) {
|
|
classes.push( 'mw-default-size' );
|
|
}
|
|
|
|
if ( data.attributes.borderImage ) {
|
|
classes.push( 'mw-image-border' );
|
|
}
|
|
|
|
if ( data.attributes.valign && data.attributes.valign !== 'default' ) {
|
|
classes.push( 'mw-valign-' + data.attributes.valign );
|
|
}
|
|
|
|
if ( data.attributes.unrecognizedClasses ) {
|
|
classes = OO.simpleArrayUnion( classes, data.attributes.unrecognizedClasses );
|
|
}
|
|
|
|
if (
|
|
originalClasses &&
|
|
ve.compare( originalClasses.trim().split( /\s+/ ).sort(), classes.sort() )
|
|
) {
|
|
span.className = originalClasses;
|
|
} else if ( classes.length > 0 ) {
|
|
span.className = classes.join( ' ' );
|
|
}
|
|
|
|
if ( data.attributes.isLinked ) {
|
|
firstChild = doc.createElement( 'a' );
|
|
firstChild.setAttribute( 'href', data.attributes.href );
|
|
} else {
|
|
firstChild = doc.createElement( 'span' );
|
|
}
|
|
|
|
span.appendChild( firstChild );
|
|
firstChild.appendChild( img );
|
|
|
|
return [ span ];
|
|
};
|
|
|
|
/* Registration */
|
|
|
|
ve.dm.modelRegistry.unregister( ve.dm.InlineImageNode );
|
|
ve.dm.modelRegistry.register( ve.dm.MWInlineImageNode );
|