2013-10-15 12:18:11 +00:00
|
|
|
/*!
|
|
|
|
* VisualEditor DataModel MWImageNode class.
|
|
|
|
*
|
2017-01-03 16:58:33 +00:00
|
|
|
* @copyright 2011-2017 VisualEditor Team and others; see AUTHORS.txt
|
2013-10-15 12:18:11 +00:00
|
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2014-08-19 12:25:40 +00:00
|
|
|
* DataModel MediaWiki image node.
|
2013-10-15 12:18:11 +00:00
|
|
|
*
|
|
|
|
* @class
|
|
|
|
* @abstract
|
|
|
|
* @extends ve.dm.GeneratedContentNode
|
2014-09-17 00:33:39 +00:00
|
|
|
* @mixins ve.dm.FocusableNode
|
2014-04-10 00:26:48 +00:00
|
|
|
* @mixins ve.dm.ResizableNode
|
|
|
|
*
|
2013-10-15 12:18:11 +00:00
|
|
|
* @constructor
|
|
|
|
*/
|
|
|
|
ve.dm.MWImageNode = function VeDmMWImageNode() {
|
|
|
|
// Parent constructor
|
|
|
|
ve.dm.GeneratedContentNode.call( this );
|
2014-09-17 00:33:39 +00:00
|
|
|
|
|
|
|
// Mixin constructors
|
|
|
|
ve.dm.FocusableNode.call( this );
|
2017-04-13 10:32:30 +00:00
|
|
|
// ve.dm.MWResizableNode doesn't exist
|
|
|
|
ve.dm.ResizableNode.call( this );
|
2014-04-10 00:26:48 +00:00
|
|
|
|
|
|
|
this.scalablePromise = null;
|
|
|
|
|
|
|
|
// Use 'bitmap' as default media type until we can
|
|
|
|
// fetch the actual media type from the API
|
|
|
|
this.mediaType = 'BITMAP';
|
|
|
|
|
|
|
|
// Initialize
|
2014-07-15 18:43:54 +00:00
|
|
|
this.constructor.static.syncScalableToType(
|
|
|
|
this.getAttribute( 'type' ),
|
|
|
|
this.mediaType,
|
|
|
|
this.getScalable()
|
|
|
|
);
|
2014-04-10 00:26:48 +00:00
|
|
|
|
|
|
|
// Events
|
2014-08-22 20:50:48 +00:00
|
|
|
this.connect( this, { attributeChange: 'onAttributeChange' } );
|
2013-10-15 12:18:11 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Inheritance */
|
|
|
|
|
2013-10-11 21:44:09 +00:00
|
|
|
OO.inheritClass( ve.dm.MWImageNode, ve.dm.GeneratedContentNode );
|
2013-10-15 12:18:11 +00:00
|
|
|
|
2014-09-17 00:33:39 +00:00
|
|
|
OO.mixinClass( ve.dm.MWImageNode, ve.dm.FocusableNode );
|
|
|
|
|
2014-04-10 00:26:48 +00:00
|
|
|
OO.mixinClass( ve.dm.MWImageNode, ve.dm.ResizableNode );
|
|
|
|
|
2014-08-19 12:25:40 +00:00
|
|
|
/* Static methods */
|
|
|
|
|
|
|
|
ve.dm.MWImageNode.static.getHashObject = function ( dataElement ) {
|
|
|
|
return {
|
2014-08-22 20:50:48 +00:00
|
|
|
type: dataElement.type,
|
|
|
|
resource: dataElement.attributes.resource,
|
|
|
|
width: dataElement.attributes.width,
|
|
|
|
height: dataElement.attributes.height
|
2014-08-19 12:25:40 +00:00
|
|
|
};
|
|
|
|
};
|
2014-01-26 15:02:07 +00:00
|
|
|
|
2017-03-16 15:32:59 +00:00
|
|
|
ve.dm.MWImageNode.static.describeChanges = function ( attributeChanges, attributes ) {
|
|
|
|
var key, sizeFrom, sizeTo, change,
|
|
|
|
customKeys = [ 'width', 'height', 'defaultSize', 'src', 'href' ],
|
|
|
|
descriptions = [];
|
|
|
|
|
|
|
|
function describeSize( width, height ) {
|
|
|
|
return width + ve.msg( 'visualeditor-dimensionswidget-times' ) + height + ve.msg( 'visualeditor-dimensionswidget-px' );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( 'width' in attributeChanges || 'height' in attributeChanges ) {
|
|
|
|
if ( attributeChanges.defaultSize && attributeChanges.defaultSize.from === true ) {
|
|
|
|
sizeFrom = ve.msg( 'visualeditor-mediasizewidget-sizeoptions-default' );
|
|
|
|
} else {
|
|
|
|
sizeFrom = describeSize(
|
|
|
|
'width' in attributeChanges ? attributeChanges.width.from : attributes.width,
|
|
|
|
'height' in attributeChanges ? attributeChanges.height.from : attributes.height
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if ( attributeChanges.defaultSize && attributeChanges.defaultSize.to === true ) {
|
|
|
|
sizeTo = ve.msg( 'visualeditor-mediasizewidget-sizeoptions-default' );
|
|
|
|
} else {
|
|
|
|
sizeTo = describeSize(
|
|
|
|
'width' in attributeChanges ? attributeChanges.width.to : attributes.width,
|
|
|
|
'height' in attributeChanges ? attributeChanges.height.to : attributes.height
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
descriptions.push( ve.msg( 'visualeditor-changedesc-image-size', sizeFrom, sizeTo ) );
|
|
|
|
}
|
|
|
|
for ( key in attributeChanges ) {
|
|
|
|
if ( customKeys.indexOf( key ) === -1 ) {
|
2017-03-23 00:28:41 +00:00
|
|
|
if ( key === 'borderImage' && !attributeChanges.borderImage.from && !attributeChanges.borderImage.to ) {
|
|
|
|
// Skip noise from the data model
|
|
|
|
continue;
|
|
|
|
}
|
2017-03-16 15:32:59 +00:00
|
|
|
change = this.describeChange( key, attributeChanges[ key ] );
|
|
|
|
descriptions.push( change );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return descriptions;
|
|
|
|
};
|
|
|
|
|
|
|
|
ve.dm.MWImageNode.static.describeChange = function ( key, change ) {
|
|
|
|
if ( key === 'align' ) {
|
|
|
|
return ve.msg( 'visualeditor-changedesc-align',
|
2017-03-23 00:28:41 +00:00
|
|
|
// Messages used:
|
|
|
|
// visualeditor-align-widget-left, visualeditor-align-widget-right,
|
|
|
|
// visualeditor-align-widget-center, visualeditor-align-widget-default
|
2017-03-16 15:32:59 +00:00
|
|
|
ve.msg( 'visualeditor-align-widget-' + change.from ),
|
|
|
|
ve.msg( 'visualeditor-align-widget-' + change.to )
|
|
|
|
);
|
|
|
|
}
|
|
|
|
// Parent method
|
|
|
|
return ve.dm.Node.static.describeChange.apply( this, arguments );
|
|
|
|
};
|
|
|
|
|
2014-07-25 02:28:07 +00:00
|
|
|
/**
|
|
|
|
* Take the given dimensions and scale them to thumbnail size.
|
2015-08-19 18:21:01 +00:00
|
|
|
*
|
2014-07-25 02:28:07 +00:00
|
|
|
* @param {Object} dimensions Width and height of the image
|
|
|
|
* @param {string} [mediaType] Media type 'DRAWING' or 'BITMAP'
|
|
|
|
* @return {Object} The new width and height of the scaled image
|
|
|
|
*/
|
|
|
|
ve.dm.MWImageNode.static.scaleToThumbnailSize = function ( dimensions, mediaType ) {
|
|
|
|
var defaultThumbSize = mw.config.get( 'wgVisualEditorConfig' ).defaultUserOptions.defaultthumbsize;
|
|
|
|
|
|
|
|
mediaType = mediaType || 'BITMAP';
|
|
|
|
|
|
|
|
if ( dimensions.width && dimensions.height ) {
|
|
|
|
// Use dimensions
|
|
|
|
// Resize to default thumbnail size, but only if the image itself
|
|
|
|
// isn't smaller than the default size
|
|
|
|
// For svg/drawings, the default wiki size is always applied
|
|
|
|
if ( dimensions.width > defaultThumbSize || mediaType === 'DRAWING' ) {
|
|
|
|
return ve.dm.Scalable.static.getDimensionsFromValue( {
|
|
|
|
width: defaultThumbSize
|
|
|
|
}, dimensions.width / dimensions.height );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return dimensions;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Translate the image dimensions into new ones according to the bounding box.
|
2015-08-19 18:21:01 +00:00
|
|
|
*
|
2015-08-19 18:33:59 +00:00
|
|
|
* @param {Object} imageDimensions Width and height of the image
|
2014-07-25 02:28:07 +00:00
|
|
|
* @param {Object} boundingBox The limit of the bounding box
|
2015-08-19 18:09:34 +00:00
|
|
|
* @return {Object} The new width and height of the scaled image.
|
2014-07-25 02:28:07 +00:00
|
|
|
*/
|
2015-01-15 18:50:30 +00:00
|
|
|
ve.dm.MWImageNode.static.resizeToBoundingBox = function ( imageDimensions, boundingBox ) {
|
|
|
|
var newDimensions = ve.copy( imageDimensions ),
|
|
|
|
scale = Math.min(
|
|
|
|
boundingBox.height / imageDimensions.height,
|
|
|
|
boundingBox.width / imageDimensions.width
|
2014-07-25 02:28:07 +00:00
|
|
|
);
|
2015-01-15 18:50:30 +00:00
|
|
|
|
|
|
|
if ( scale < 1 ) {
|
|
|
|
// Scale down
|
|
|
|
newDimensions = {
|
2015-02-12 01:07:44 +00:00
|
|
|
width: Math.floor( newDimensions.width * scale ),
|
|
|
|
height: Math.floor( newDimensions.height * scale )
|
2015-01-15 18:50:30 +00:00
|
|
|
};
|
2014-07-25 02:28:07 +00:00
|
|
|
}
|
2015-01-15 18:50:30 +00:00
|
|
|
return newDimensions;
|
2014-07-25 02:28:07 +00:00
|
|
|
};
|
|
|
|
|
2014-01-26 15:02:07 +00:00
|
|
|
/**
|
2014-06-04 18:20:37 +00:00
|
|
|
* Update image scalable properties according to the image type.
|
2014-01-26 15:02:07 +00:00
|
|
|
*
|
2014-04-10 00:26:48 +00:00
|
|
|
* @param {string} type The new image type
|
2014-07-15 18:43:54 +00:00
|
|
|
* @param {string} mediaType Image media type 'DRAWING' or 'BITMAP'
|
|
|
|
* @param {ve.dm.Scalable} scalable The scalable object to update
|
2014-01-26 15:02:07 +00:00
|
|
|
*/
|
2014-07-15 18:43:54 +00:00
|
|
|
ve.dm.MWImageNode.static.syncScalableToType = function ( type, mediaType, scalable ) {
|
2014-04-10 00:26:48 +00:00
|
|
|
var originalDimensions, dimensions,
|
2014-07-15 18:43:54 +00:00
|
|
|
defaultThumbSize = mw.config.get( 'wgVisualEditorConfig' ).defaultUserOptions.defaultthumbsize;
|
2014-03-25 16:01:04 +00:00
|
|
|
|
2014-04-10 00:26:48 +00:00
|
|
|
originalDimensions = scalable.getOriginalDimensions();
|
|
|
|
|
2014-09-04 15:29:51 +00:00
|
|
|
// We can only set default dimensions if we have the original ones
|
|
|
|
if ( originalDimensions ) {
|
|
|
|
if ( type === 'thumb' || type === 'frameless' ) {
|
|
|
|
// Set the default size to that in the wiki configuration if
|
|
|
|
// 1. The original image width is not smaller than the default
|
|
|
|
// 2. If the image is an SVG drawing
|
|
|
|
if ( originalDimensions.width >= defaultThumbSize || mediaType === 'DRAWING' ) {
|
|
|
|
dimensions = ve.dm.Scalable.static.getDimensionsFromValue( {
|
|
|
|
width: defaultThumbSize
|
|
|
|
}, scalable.getRatio() );
|
|
|
|
} else {
|
|
|
|
dimensions = ve.dm.Scalable.static.getDimensionsFromValue(
|
|
|
|
originalDimensions,
|
|
|
|
scalable.getRatio()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
scalable.setDefaultDimensions( dimensions );
|
2014-04-10 00:26:48 +00:00
|
|
|
} else {
|
|
|
|
scalable.setDefaultDimensions( originalDimensions );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deal with maximum dimensions for images and drawings
|
2014-07-15 18:43:54 +00:00
|
|
|
if ( mediaType !== 'DRAWING' ) {
|
2014-04-10 00:26:48 +00:00
|
|
|
if ( originalDimensions ) {
|
|
|
|
scalable.setMaxDimensions( originalDimensions );
|
|
|
|
scalable.setEnforcedMax( true );
|
|
|
|
} else {
|
|
|
|
scalable.setEnforcedMax( false );
|
|
|
|
}
|
2016-11-02 18:12:02 +00:00
|
|
|
} else {
|
|
|
|
// EnforcedMax may have previously been set to true
|
|
|
|
scalable.setEnforcedMax( false );
|
2014-04-10 00:26:48 +00:00
|
|
|
}
|
2014-06-26 19:40:20 +00:00
|
|
|
// TODO: Some day, when svgMaxSize works properly in MediaWiki
|
2016-11-02 18:12:02 +00:00
|
|
|
// we can add it back as max dimension consideration:
|
|
|
|
// mw.config.get( 'wgVisualEditorConfig' ).svgMaxSize
|
2014-04-10 00:26:48 +00:00
|
|
|
};
|
2014-07-17 19:55:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the scalable promise which fetches original dimensions from the API
|
2015-08-19 18:21:01 +00:00
|
|
|
*
|
2014-07-17 19:55:24 +00:00
|
|
|
* @param {string} filename The image filename whose details the scalable will represent
|
2015-08-19 18:09:34 +00:00
|
|
|
* @return {jQuery.Promise} Promise which resolves after the image size details are fetched from the API
|
2014-07-17 19:55:24 +00:00
|
|
|
*/
|
|
|
|
ve.dm.MWImageNode.static.getScalablePromise = function ( filename ) {
|
|
|
|
// On the first call set off an async call to update the scalable's
|
|
|
|
// original dimensions from the API.
|
2014-12-16 01:06:05 +00:00
|
|
|
if ( ve.init.platform.imageInfoCache ) {
|
|
|
|
return ve.init.platform.imageInfoCache.get( filename ).then( function ( info ) {
|
|
|
|
if ( !info ) {
|
|
|
|
return $.Deferred().reject().promise();
|
2014-07-17 19:55:24 +00:00
|
|
|
}
|
2014-12-16 01:06:05 +00:00
|
|
|
return info;
|
2014-07-17 19:55:24 +00:00
|
|
|
} );
|
|
|
|
} else {
|
2014-12-16 01:06:05 +00:00
|
|
|
return $.Deferred().reject().promise();
|
2014-07-17 19:55:24 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-08-19 12:25:40 +00:00
|
|
|
/* Methods */
|
|
|
|
|
2014-04-10 00:26:48 +00:00
|
|
|
/**
|
|
|
|
* Respond to attribute change.
|
|
|
|
* Update the rendering of the 'align', src', 'width' and 'height' attributes
|
|
|
|
* when they change in the model.
|
|
|
|
*
|
|
|
|
* @method
|
|
|
|
* @param {string} key Attribute key
|
|
|
|
* @param {string} from Old value
|
|
|
|
* @param {string} to New value
|
|
|
|
*/
|
|
|
|
ve.dm.MWImageNode.prototype.onAttributeChange = function ( key, from, to ) {
|
|
|
|
if ( key === 'type' ) {
|
2014-07-15 18:43:54 +00:00
|
|
|
this.constructor.static.syncScalableToType( to, this.mediaType, this.getScalable() );
|
2014-01-26 15:02:07 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the normalised filename of the image
|
|
|
|
*
|
2015-08-19 18:09:34 +00:00
|
|
|
* @return {string} Filename
|
2014-01-26 15:02:07 +00:00
|
|
|
*/
|
|
|
|
ve.dm.MWImageNode.prototype.getFilename = function () {
|
2014-05-28 19:21:06 +00:00
|
|
|
// Strip ./ stuff and decode URI encoding
|
2014-09-24 19:43:57 +00:00
|
|
|
var resource = this.getAttribute( 'resource' ) || '',
|
2014-08-21 01:16:13 +00:00
|
|
|
filename = resource.replace( /^(\.+\/)*/, '' );
|
2014-05-28 19:21:06 +00:00
|
|
|
|
2016-06-14 22:27:44 +00:00
|
|
|
return ve.decodeURIComponentIntoArticleTitle( filename, true );
|
2014-01-26 15:02:07 +00:00
|
|
|
};
|
|
|
|
|
2014-04-10 00:26:48 +00:00
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
ve.dm.MWImageNode.prototype.getScalable = function () {
|
2014-10-29 01:43:03 +00:00
|
|
|
var imageNode = this;
|
2014-07-17 19:55:24 +00:00
|
|
|
if ( !this.scalablePromise ) {
|
2014-12-16 01:06:05 +00:00
|
|
|
this.scalablePromise = ve.dm.MWImageNode.static.getScalablePromise( this.getFilename() );
|
|
|
|
// If the promise was already resolved before getScalablePromise returned, then jQuery will execute the done straight away.
|
|
|
|
// So don't just do getScalablePromise( ... ).done because we need to make sure that this.scalablePromise gets set first.
|
|
|
|
this.scalablePromise.done( function ( info ) {
|
2015-02-17 12:30:27 +00:00
|
|
|
if ( info ) {
|
|
|
|
imageNode.getScalable().setOriginalDimensions( {
|
|
|
|
width: info.width,
|
|
|
|
height: info.height
|
|
|
|
} );
|
|
|
|
// Update media type
|
|
|
|
imageNode.mediaType = info.mediatype;
|
|
|
|
// Update according to type
|
|
|
|
imageNode.constructor.static.syncScalableToType(
|
|
|
|
imageNode.getAttribute( 'type' ),
|
|
|
|
imageNode.mediaType,
|
|
|
|
imageNode.getScalable()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} );
|
2014-07-17 19:55:24 +00:00
|
|
|
}
|
2016-08-22 21:44:59 +00:00
|
|
|
// Mixin method
|
2014-04-10 00:26:48 +00:00
|
|
|
return ve.dm.ResizableNode.prototype.getScalable.call( this );
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
ve.dm.MWImageNode.prototype.createScalable = function () {
|
|
|
|
return new ve.dm.Scalable( {
|
2014-08-22 20:50:48 +00:00
|
|
|
currentDimensions: {
|
|
|
|
width: this.getAttribute( 'width' ),
|
|
|
|
height: this.getAttribute( 'height' )
|
2014-04-10 00:26:48 +00:00
|
|
|
},
|
2014-08-22 20:50:48 +00:00
|
|
|
minDimensions: {
|
|
|
|
width: 1,
|
|
|
|
height: 1
|
2014-04-10 00:26:48 +00:00
|
|
|
}
|
|
|
|
} );
|
|
|
|
};
|
2014-06-04 18:20:37 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get symbolic name of media type.
|
|
|
|
*
|
|
|
|
* Example values: "BITMAP" for JPEG or PNG images; "DRAWING" for SVG graphics
|
|
|
|
*
|
|
|
|
* @return {string|undefined} Symbolic media type name, or undefined if empty
|
|
|
|
*/
|
|
|
|
ve.dm.MWImageNode.prototype.getMediaType = function () {
|
|
|
|
return this.mediaType;
|
|
|
|
};
|