mirror of
https://github.com/google/blockly.git
synced 2026-01-09 01:50:11 +01:00
An upcoming change to emit private properties in Google's internal version of https://github.com/angular/clutz will otherwise break any TypeScript depending on these types. This is because TypeScript errors on overriding protected with private (Closure Compiler does not).
288 lines
7.7 KiB
JavaScript
288 lines
7.7 KiB
JavaScript
/**
|
|
* @license
|
|
* Copyright 2012 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/**
|
|
* @fileoverview Image field. Used for pictures, icons, etc.
|
|
* @author fraser@google.com (Neil Fraser)
|
|
*/
|
|
'use strict';
|
|
|
|
goog.provide('Blockly.FieldImage');
|
|
|
|
goog.require('Blockly.Field');
|
|
goog.require('Blockly.fieldRegistry');
|
|
goog.require('Blockly.utils');
|
|
goog.require('Blockly.utils.dom');
|
|
goog.require('Blockly.utils.object');
|
|
goog.require('Blockly.utils.Size');
|
|
goog.require('Blockly.utils.Svg');
|
|
|
|
|
|
/**
|
|
* Class for an image on a block.
|
|
* @param {string} src The URL of the image.
|
|
* @param {!(string|number)} width Width of the image.
|
|
* @param {!(string|number)} height Height of the image.
|
|
* @param {string=} opt_alt Optional alt text for when block is collapsed.
|
|
* @param {function(!Blockly.FieldImage)=} opt_onClick Optional function to be
|
|
* called when the image is clicked. If opt_onClick is defined, opt_alt must
|
|
* also be defined.
|
|
* @param {boolean=} opt_flipRtl Whether to flip the icon in RTL.
|
|
* @param {Object=} opt_config A map of options used to configure the field.
|
|
* See the [field creation documentation]{@link https://developers.google.com/blockly/guides/create-custom-blocks/fields/built-in-fields/image#creation}
|
|
* for a list of properties this parameter supports.
|
|
* @extends {Blockly.Field}
|
|
* @constructor
|
|
*/
|
|
Blockly.FieldImage = function(src, width, height,
|
|
opt_alt, opt_onClick, opt_flipRtl, opt_config) {
|
|
// Return early.
|
|
if (!src) {
|
|
throw Error('Src value of an image field is required');
|
|
}
|
|
src = Blockly.utils.replaceMessageReferences(src);
|
|
var imageHeight = Number(Blockly.utils.replaceMessageReferences(height));
|
|
var imageWidth = Number(Blockly.utils.replaceMessageReferences(width));
|
|
if (isNaN(imageHeight) || isNaN(imageWidth)) {
|
|
throw Error('Height and width values of an image field must cast to' +
|
|
' numbers.');
|
|
}
|
|
if (imageHeight <= 0 || imageWidth <= 0) {
|
|
throw Error('Height and width values of an image field must be greater' +
|
|
' than 0.');
|
|
}
|
|
|
|
// Initialize configurable properties.
|
|
/**
|
|
* Whether to flip this image in RTL.
|
|
* @type {boolean}
|
|
* @private
|
|
*/
|
|
this.flipRtl_ = false;
|
|
|
|
/**
|
|
* Alt text of this image.
|
|
* @type {string}
|
|
* @private
|
|
*/
|
|
this.altText_ = '';
|
|
|
|
Blockly.FieldImage.superClass_.constructor.call(
|
|
this, src, null, opt_config);
|
|
|
|
if (!opt_config) { // If the config wasn't passed, do old configuration.
|
|
this.flipRtl_ = !!opt_flipRtl;
|
|
this.altText_ = Blockly.utils.replaceMessageReferences(opt_alt) || '';
|
|
}
|
|
|
|
// Initialize other properties.
|
|
/**
|
|
* The size of the area rendered by the field.
|
|
* @type {Blockly.utils.Size}
|
|
* @protected
|
|
* @override
|
|
*/
|
|
this.size_ = new Blockly.utils.Size(imageWidth,
|
|
imageHeight + Blockly.FieldImage.Y_PADDING);
|
|
|
|
/**
|
|
* Store the image height, since it is different from the field height.
|
|
* @type {number}
|
|
* @private
|
|
*/
|
|
this.imageHeight_ = imageHeight;
|
|
|
|
/**
|
|
* The function to be called when this field is clicked.
|
|
* @type {?function(!Blockly.FieldImage)}
|
|
* @private
|
|
*/
|
|
this.clickHandler_ = null;
|
|
|
|
if (typeof opt_onClick == 'function') {
|
|
this.clickHandler_ = opt_onClick;
|
|
}
|
|
|
|
/**
|
|
* The rendered field's image element.
|
|
* @type {SVGImageElement}
|
|
* @private
|
|
*/
|
|
this.imageElement_ = null;
|
|
};
|
|
Blockly.utils.object.inherits(Blockly.FieldImage, Blockly.Field);
|
|
|
|
/**
|
|
* The default value for this field.
|
|
* @type {*}
|
|
* @protected
|
|
*/
|
|
Blockly.FieldImage.prototype.DEFAULT_VALUE = '';
|
|
|
|
/**
|
|
* Construct a FieldImage from a JSON arg object,
|
|
* dereferencing any string table references.
|
|
* @param {!Object} options A JSON object with options (src, width, height,
|
|
* alt, and flipRtl).
|
|
* @return {!Blockly.FieldImage} The new field instance.
|
|
* @package
|
|
* @nocollapse
|
|
*/
|
|
Blockly.FieldImage.fromJson = function(options) {
|
|
return new Blockly.FieldImage(
|
|
options['src'], options['width'], options['height'],
|
|
undefined, undefined, undefined, options);
|
|
};
|
|
|
|
/**
|
|
* Vertical padding below the image, which is included in the reported height of
|
|
* the field.
|
|
* @type {number}
|
|
* @private
|
|
*/
|
|
Blockly.FieldImage.Y_PADDING = 1;
|
|
|
|
/**
|
|
* Editable fields usually show some sort of UI indicating they are
|
|
* editable. This field should not.
|
|
* @type {boolean}
|
|
*/
|
|
Blockly.FieldImage.prototype.EDITABLE = false;
|
|
|
|
/**
|
|
* Used to tell if the field needs to be rendered the next time the block is
|
|
* rendered. Image fields are statically sized, and only need to be
|
|
* rendered at initialization.
|
|
* @type {boolean}
|
|
* @protected
|
|
*/
|
|
Blockly.FieldImage.prototype.isDirty_ = false;
|
|
|
|
/**
|
|
* Configure the field based on the given map of options.
|
|
* @param {!Object} config A map of options to configure the field based on.
|
|
* @protected
|
|
* @override
|
|
*/
|
|
Blockly.FieldImage.prototype.configure_ = function(config) {
|
|
Blockly.FieldImage.superClass_.configure_.call(this, config);
|
|
this.flipRtl_ = !!config['flipRtl'];
|
|
this.altText_ = Blockly.utils.replaceMessageReferences(config['alt']) || '';
|
|
};
|
|
|
|
/**
|
|
* Create the block UI for this image.
|
|
* @package
|
|
*/
|
|
Blockly.FieldImage.prototype.initView = function() {
|
|
this.imageElement_ = Blockly.utils.dom.createSvgElement(
|
|
Blockly.utils.Svg.IMAGE,
|
|
{
|
|
'height': this.imageHeight_ + 'px',
|
|
'width': this.size_.width + 'px',
|
|
'alt': this.altText_
|
|
},
|
|
this.fieldGroup_);
|
|
this.imageElement_.setAttributeNS(Blockly.utils.dom.XLINK_NS,
|
|
'xlink:href', /** @type {string} */ (this.value_));
|
|
|
|
if (this.clickHandler_) {
|
|
this.imageElement_.style.cursor = 'pointer';
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.FieldImage.prototype.updateSize_ = function() {
|
|
// NOP
|
|
};
|
|
|
|
/**
|
|
* Ensure that the input value (the source URL) is a string.
|
|
* @param {*=} opt_newValue The input value.
|
|
* @return {?string} A string, or null if invalid.
|
|
* @protected
|
|
*/
|
|
Blockly.FieldImage.prototype.doClassValidation_ = function(opt_newValue) {
|
|
if (typeof opt_newValue != 'string') {
|
|
return null;
|
|
}
|
|
return opt_newValue;
|
|
};
|
|
|
|
/**
|
|
* Update the value of this image field, and update the displayed image.
|
|
* @param {*} newValue The value to be saved. The default validator guarantees
|
|
* that this is a string.
|
|
* @protected
|
|
*/
|
|
Blockly.FieldImage.prototype.doValueUpdate_ = function(newValue) {
|
|
this.value_ = newValue;
|
|
if (this.imageElement_) {
|
|
this.imageElement_.setAttributeNS(Blockly.utils.dom.XLINK_NS,
|
|
'xlink:href', String(this.value_));
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Get whether to flip this image in RTL
|
|
* @return {boolean} True if we should flip in RTL.
|
|
* @override
|
|
*/
|
|
Blockly.FieldImage.prototype.getFlipRtl = function() {
|
|
return this.flipRtl_;
|
|
};
|
|
|
|
/**
|
|
* Set the alt text of this image.
|
|
* @param {?string} alt New alt text.
|
|
* @public
|
|
*/
|
|
Blockly.FieldImage.prototype.setAlt = function(alt) {
|
|
if (alt == this.altText_) {
|
|
return;
|
|
}
|
|
this.altText_ = alt || '';
|
|
if (this.imageElement_) {
|
|
this.imageElement_.setAttribute('alt', this.altText_);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* If field click is called, and click handler defined,
|
|
* call the handler.
|
|
* @protected
|
|
*/
|
|
Blockly.FieldImage.prototype.showEditor_ = function() {
|
|
if (this.clickHandler_) {
|
|
this.clickHandler_(this);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Set the function that is called when this image is clicked.
|
|
* @param {?function(!Blockly.FieldImage)} func The function that is called
|
|
* when the image is clicked, or null to remove.
|
|
*/
|
|
Blockly.FieldImage.prototype.setOnClickHandler = function(func) {
|
|
this.clickHandler_ = func;
|
|
};
|
|
|
|
/**
|
|
* Use the `getText_` developer hook to override the field's text
|
|
* representation.
|
|
* Return the image alt text instead.
|
|
* @return {?string} The image alt text.
|
|
* @protected
|
|
* @override
|
|
*/
|
|
Blockly.FieldImage.prototype.getText_ = function() {
|
|
return this.altText_;
|
|
};
|
|
|
|
Blockly.fieldRegistry.register('field_image', Blockly.FieldImage);
|