123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500 |
- /**
- * @license
- * Visual Blocks Editor
- *
- * Copyright 2012 Google Inc.
- * https://developers.google.com/blockly/
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- /**
- * @fileoverview Field. Used for editable titles, variables, etc.
- * This is an abstract class that defines the UI on the block. Actual
- * instances would be Blockly.FieldTextInput, Blockly.FieldDropdown, etc.
- * @author fraser@google.com (Neil Fraser)
- */
- 'use strict';
- goog.provide('Blockly.Field');
- goog.require('goog.asserts');
- goog.require('goog.dom');
- goog.require('goog.math.Size');
- goog.require('goog.style');
- goog.require('goog.userAgent');
- /**
- * Abstract class for an editable field.
- * @param {string} text The initial content of the field.
- * @param {Function=} opt_validator An optional function that is called
- * to validate any constraints on what the user entered. Takes the new
- * text as an argument and returns either the accepted text, a replacement
- * text, or null to abort the change.
- * @constructor
- */
- Blockly.Field = function(text, opt_validator) {
- this.size_ = new goog.math.Size(0, 25);
- this.setValue(text);
- this.setValidator(opt_validator);
- };
- /**
- * Temporary cache of text widths.
- * @type {Object}
- * @private
- */
- Blockly.Field.cacheWidths_ = null;
- /**
- * Number of current references to cache.
- * @type {number}
- * @private
- */
- Blockly.Field.cacheReference_ = 0;
- /**
- * Name of field. Unique within each block.
- * Static labels are usually unnamed.
- * @type {string=}
- */
- Blockly.Field.prototype.name = undefined;
- /**
- * Maximum characters of text to display before adding an ellipsis.
- * @type {number}
- */
- Blockly.Field.prototype.maxDisplayLength = 50;
- /**
- * Visible text to display.
- * @type {string}
- * @private
- */
- Blockly.Field.prototype.text_ = '';
- /**
- * Block this field is attached to. Starts as null, then in set in init.
- * @type {Blockly.Block}
- * @private
- */
- Blockly.Field.prototype.sourceBlock_ = null;
- /**
- * Is the field visible, or hidden due to the block being collapsed?
- * @type {boolean}
- * @private
- */
- Blockly.Field.prototype.visible_ = true;
- /**
- * Validation function called when user edits an editable field.
- * @type {Function}
- * @private
- */
- Blockly.Field.prototype.validator_ = null;
- /**
- * Non-breaking space.
- * @const
- */
- Blockly.Field.NBSP = '\u00A0';
- /**
- * Editable fields are saved by the XML renderer, non-editable fields are not.
- */
- Blockly.Field.prototype.EDITABLE = true;
- /**
- * Attach this field to a block.
- * @param {!Blockly.Block} block The block containing this field.
- */
- Blockly.Field.prototype.setSourceBlock = function(block) {
- goog.asserts.assert(!this.sourceBlock_, 'Field already bound to a block.');
- this.sourceBlock_ = block;
- };
- /**
- * Install this field on a block.
- */
- Blockly.Field.prototype.init = function() {
- if (this.fieldGroup_) {
- // Field has already been initialized once.
- return;
- }
- // Build the DOM.
- this.fieldGroup_ = Blockly.createSvgElement('g', {}, null);
- if (!this.visible_) {
- this.fieldGroup_.style.display = 'none';
- }
- this.borderRect_ = Blockly.createSvgElement('rect',
- {'rx': 4,
- 'ry': 4,
- 'x': -Blockly.BlockSvg.SEP_SPACE_X / 2,
- 'y': 0,
- 'height': 16}, this.fieldGroup_, this.sourceBlock_.workspace);
- /** @type {!Element} */
- this.textElement_ = Blockly.createSvgElement('text',
- {'class': 'blocklyText', 'y': this.size_.height - 12.5},
- this.fieldGroup_);
- this.updateEditable();
- this.sourceBlock_.getSvgRoot().appendChild(this.fieldGroup_);
- this.mouseUpWrapper_ =
- Blockly.bindEventWithChecks_(this.fieldGroup_, 'mouseup', this,
- this.onMouseUp_);
- // Force a render.
- this.updateTextNode_();
- };
- /**
- * Dispose of all DOM objects belonging to this editable field.
- */
- Blockly.Field.prototype.dispose = function() {
- if (this.mouseUpWrapper_) {
- Blockly.unbindEvent_(this.mouseUpWrapper_);
- this.mouseUpWrapper_ = null;
- }
- this.sourceBlock_ = null;
- goog.dom.removeNode(this.fieldGroup_);
- this.fieldGroup_ = null;
- this.textElement_ = null;
- this.borderRect_ = null;
- this.validator_ = null;
- };
- /**
- * Add or remove the UI indicating if this field is editable or not.
- */
- Blockly.Field.prototype.updateEditable = function() {
- var group = this.fieldGroup_;
- if (!this.EDITABLE || !group) {
- return;
- }
- if (this.sourceBlock_.isEditable()) {
- Blockly.addClass_(group, 'blocklyEditableText');
- Blockly.removeClass_(group, 'blocklyNonEditableText');
- this.fieldGroup_.style.cursor = this.CURSOR;
- } else {
- Blockly.addClass_(group, 'blocklyNonEditableText');
- Blockly.removeClass_(group, 'blocklyEditableText');
- this.fieldGroup_.style.cursor = '';
- }
- };
- /**
- * Gets whether this editable field is visible or not.
- * @return {boolean} True if visible.
- */
- Blockly.Field.prototype.isVisible = function() {
- return this.visible_;
- };
- /**
- * Sets whether this editable field is visible or not.
- * @param {boolean} visible True if visible.
- */
- Blockly.Field.prototype.setVisible = function(visible) {
- if (this.visible_ == visible) {
- return;
- }
- this.visible_ = visible;
- var root = this.getSvgRoot();
- if (root) {
- root.style.display = visible ? 'block' : 'none';
- this.render_();
- }
- };
- /**
- * Sets a new validation function for editable fields.
- * @param {Function} handler New validation function, or null.
- */
- Blockly.Field.prototype.setValidator = function(handler) {
- this.validator_ = handler;
- };
- /**
- * Gets the validation function for editable fields.
- * @return {Function} Validation function, or null.
- */
- Blockly.Field.prototype.getValidator = function() {
- return this.validator_;
- };
- /**
- * Validates a change. Does nothing. Subclasses may override this.
- * @param {string} text The user's text.
- * @return {string} No change needed.
- */
- Blockly.Field.prototype.classValidator = function(text) {
- return text;
- };
- /**
- * Calls the validation function for this field, as well as all the validation
- * function for the field's class and its parents.
- * @param {string} text Proposed text.
- * @return {?string} Revised text, or null if invalid.
- */
- Blockly.Field.prototype.callValidator = function(text) {
- var classResult = this.classValidator(text);
- if (classResult === null) {
- // Class validator rejects value. Game over.
- return null;
- } else if (classResult !== undefined) {
- text = classResult;
- }
- var userValidator = this.getValidator();
- if (userValidator) {
- var userResult = userValidator.call(this, text);
- if (userResult === null) {
- // User validator rejects value. Game over.
- return null;
- } else if (userResult !== undefined) {
- text = userResult;
- }
- }
- return text;
- };
- /**
- * Gets the group element for this editable field.
- * Used for measuring the size and for positioning.
- * @return {!Element} The group element.
- */
- Blockly.Field.prototype.getSvgRoot = function() {
- return /** @type {!Element} */ (this.fieldGroup_);
- };
- /**
- * Draws the border with the correct width.
- * Saves the computed width in a property.
- * @private
- */
- Blockly.Field.prototype.render_ = function() {
- if (this.visible_ && this.textElement_) {
- var key = this.textElement_.textContent + '\n' +
- this.textElement_.className.baseVal;
- if (Blockly.Field.cacheWidths_ && Blockly.Field.cacheWidths_[key]) {
- var width = Blockly.Field.cacheWidths_[key];
- } else {
- try {
- var width = this.textElement_.getComputedTextLength();
- } catch (e) {
- // MSIE 11 is known to throw "Unexpected call to method or property
- // access." if Blockly is hidden.
- var width = this.textElement_.textContent.length * 8;
- }
- if (Blockly.Field.cacheWidths_) {
- Blockly.Field.cacheWidths_[key] = width;
- }
- }
- if (this.borderRect_) {
- this.borderRect_.setAttribute('width',
- width + Blockly.BlockSvg.SEP_SPACE_X);
- }
- } else {
- var width = 0;
- }
- this.size_.width = width;
- };
- /**
- * Start caching field widths. Every call to this function MUST also call
- * stopCache. Caches must not survive between execution threads.
- */
- Blockly.Field.startCache = function() {
- Blockly.Field.cacheReference_++;
- if (!Blockly.Field.cacheWidths_) {
- Blockly.Field.cacheWidths_ = {};
- }
- };
- /**
- * Stop caching field widths. Unless caching was already on when the
- * corresponding call to startCache was made.
- */
- Blockly.Field.stopCache = function() {
- Blockly.Field.cacheReference_--;
- if (!Blockly.Field.cacheReference_) {
- Blockly.Field.cacheWidths_ = null;
- }
- };
- /**
- * Returns the height and width of the field.
- * @return {!goog.math.Size} Height and width.
- */
- Blockly.Field.prototype.getSize = function() {
- if (!this.size_.width) {
- this.render_();
- }
- return this.size_;
- };
- /**
- * Returns the height and width of the field,
- * accounting for the workspace scaling.
- * @return {!goog.math.Size} Height and width.
- * @private
- */
- Blockly.Field.prototype.getScaledBBox_ = function() {
- var bBox = this.borderRect_.getBBox();
- // Create new object, as getBBox can return an uneditable SVGRect in IE.
- return new goog.math.Size(bBox.width * this.sourceBlock_.workspace.scale,
- bBox.height * this.sourceBlock_.workspace.scale);
- };
- /**
- * Get the text from this field.
- * @return {string} Current text.
- */
- Blockly.Field.prototype.getText = function() {
- return this.text_;
- };
- /**
- * Set the text in this field. Trigger a rerender of the source block.
- * @param {*} text New text.
- */
- Blockly.Field.prototype.setText = function(text) {
- if (text === null) {
- // No change if null.
- return;
- }
- text = String(text);
- if (text === this.text_) {
- // No change.
- return;
- }
- this.text_ = text;
- this.updateTextNode_();
- if (this.sourceBlock_ && this.sourceBlock_.rendered) {
- this.sourceBlock_.render();
- this.sourceBlock_.bumpNeighbours_();
- }
- };
- /**
- * Update the text node of this field to display the current text.
- * @private
- */
- Blockly.Field.prototype.updateTextNode_ = function() {
- if (!this.textElement_) {
- // Not rendered yet.
- return;
- }
- var text = this.text_;
- if (text.length > this.maxDisplayLength) {
- // Truncate displayed string and add an ellipsis ('...').
- text = text.substring(0, this.maxDisplayLength - 2) + '\u2026';
- }
- // Replace whitespace with non-breaking spaces so the text doesn't collapse.
- text = text.replace(/\s/g, Blockly.Field.NBSP);
- if (this.sourceBlock_.RTL && text) {
- // The SVG is LTR, force text to be RTL.
- text += '\u200F';
- }
- if (!text) {
- // Prevent the field from disappearing if empty.
- text = Blockly.Field.NBSP;
- }
- // Replace the text.
- goog.dom.removeChildren(/** @type {!Element} */ (this.textElement_));
- var textNode = document.createTextNode(text);
- this.textElement_.appendChild(textNode);
- // Cached width is obsolete. Clear it.
- this.size_.width = 0;
- };
- /**
- * By default there is no difference between the human-readable text and
- * the language-neutral values. Subclasses (such as dropdown) may define this.
- * @return {string} Current text.
- */
- Blockly.Field.prototype.getValue = function() {
- return this.getText();
- };
- /**
- * By default there is no difference between the human-readable text and
- * the language-neutral values. Subclasses (such as dropdown) may define this.
- * @param {string} newText New text.
- */
- Blockly.Field.prototype.setValue = function(newText) {
- if (newText === null) {
- // No change if null.
- return;
- }
- var oldText = this.getValue();
- if (oldText == newText) {
- return;
- }
- if (this.sourceBlock_ && Blockly.Events.isEnabled()) {
- Blockly.Events.fire(new Blockly.Events.Change(
- this.sourceBlock_, 'field', this.name, oldText, newText));
- }
- this.setText(newText);
- };
- /**
- * Handle a mouse up event on an editable field.
- * @param {!Event} e Mouse up event.
- * @private
- */
- Blockly.Field.prototype.onMouseUp_ = function(e) {
- if ((goog.userAgent.IPHONE || goog.userAgent.IPAD) &&
- !goog.userAgent.isVersionOrHigher('537.51.2') &&
- e.layerX !== 0 && e.layerY !== 0) {
- // Old iOS spawns a bogus event on the next touch after a 'prompt()' edit.
- // Unlike the real events, these have a layerX and layerY set.
- return;
- } else if (Blockly.isRightButton(e)) {
- // Right-click.
- return;
- } else if (this.sourceBlock_.workspace.isDragging()) {
- // Drag operation is concluding. Don't open the editor.
- return;
- } else if (this.sourceBlock_.isEditable()) {
- // Non-abstract sub-classes must define a showEditor_ method.
- this.showEditor_(e);
- // The field is handling the touch, but we also want the blockSvg onMouseUp
- // handler to fire, so we will leave the touch identifier as it is.
- // The next onMouseUp is responsible for nulling it out.
- }
- };
- /**
- * Change the tooltip text for this field.
- * @param {string|!Element} newTip Text for tooltip or a parent element to
- * link to for its tooltip.
- */
- Blockly.Field.prototype.setTooltip = function(newTip) {
- // Non-abstract sub-classes may wish to implement this. See FieldLabel.
- };
- /**
- * Return the absolute coordinates of the top-left corner of this field.
- * The origin (0,0) is the top-left corner of the page body.
- * @return {!goog.math.Coordinate} Object with .x and .y properties.
- * @private
- */
- Blockly.Field.prototype.getAbsoluteXY_ = function() {
- return goog.style.getPageOffset(this.borderRect_);
- };
|