mirror of
https://github.com/google/blockly.git
synced 2026-01-10 10:27:08 +01:00
383 lines
13 KiB
JavaScript
383 lines
13 KiB
JavaScript
/**
|
|
* @license
|
|
* Visual Blocks Editor
|
|
*
|
|
* Copyright 2019 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 Old (compatibility) renderer.
|
|
* Geras: spirit of old age.
|
|
* @author fenichel@google.com (Rachel Fenichel)
|
|
*/
|
|
'use strict';
|
|
|
|
goog.provide('Blockly.geras');
|
|
goog.provide('Blockly.geras.RenderInfo');
|
|
|
|
goog.require('Blockly.blockRendering.RenderInfo');
|
|
goog.require('Blockly.blockRendering.constants');
|
|
goog.require('Blockly.blockRendering.Measurable');
|
|
goog.require('Blockly.blockRendering.BottomRow');
|
|
goog.require('Blockly.blockRendering.InputRow');
|
|
goog.require('Blockly.blockRendering.Row');
|
|
goog.require('Blockly.blockRendering.SpacerRow');
|
|
goog.require('Blockly.blockRendering.TopRow');
|
|
|
|
goog.require('Blockly.blockRendering.InlineInput');
|
|
goog.require('Blockly.blockRendering.ExternalValueInput');
|
|
goog.require('Blockly.blockRendering.StatementInput');
|
|
|
|
goog.require('Blockly.blockRendering.PreviousConnection');
|
|
goog.require('Blockly.blockRendering.NextConnection');
|
|
goog.require('Blockly.blockRendering.OutputConnection');
|
|
|
|
goog.require('Blockly.RenderedConnection');
|
|
|
|
/**
|
|
* An object containing all sizing information needed to draw this block.
|
|
*
|
|
* This measure pass does not propagate changes to the block (although fields
|
|
* may choose to rerender when getSize() is called). However, calling it
|
|
* repeatedly may be expensive.
|
|
*
|
|
* @param {!Blockly.BlockSvg} block The block to measure.
|
|
* @constructor
|
|
* @package
|
|
* @extends {Blockly.blockRendering.RenderInfo}
|
|
*/
|
|
Blockly.geras.RenderInfo = function(block) {
|
|
Blockly.geras.RenderInfo.superClass_.constructor.call(this, block);
|
|
};
|
|
goog.inherits(Blockly.geras.RenderInfo, Blockly.blockRendering.RenderInfo);
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.geras.RenderInfo.prototype.shouldStartNewRow_ = function(input, lastInput) {
|
|
// If this is the first input, just add to the existing row.
|
|
// That row is either empty or has some icons in it.
|
|
if (!lastInput) {
|
|
return false;
|
|
}
|
|
// A statement input always gets a new row.
|
|
if (input.type == Blockly.NEXT_STATEMENT) {
|
|
return true;
|
|
}
|
|
// Value and dummy inputs get new row if inputs are not inlined.
|
|
if (input.type == Blockly.INPUT_VALUE || input.type == Blockly.DUMMY_INPUT) {
|
|
return !this.isInline;
|
|
}
|
|
return false;
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.geras.RenderInfo.prototype.getInRowSpacing_ = function(prev, next) {
|
|
if (!prev) {
|
|
// Between an editable field and the beginning of the row.
|
|
if (next.isField() && next.isEditable) {
|
|
return Blockly.blockRendering.constants.MEDIUM_PADDING;
|
|
}
|
|
// Inline input at the beginning of the row.
|
|
if (next.isInput && next.isInlineInput()) {
|
|
return Blockly.blockRendering.constants.MEDIUM_LARGE_PADDING;
|
|
}
|
|
if (next.isStatementInput()) {
|
|
return Blockly.blockRendering.constants.STATEMENT_INPUT_PADDING_LEFT;
|
|
}
|
|
// Anything else at the beginning of the row.
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
|
|
// Spacing between a non-input and the end of the row.
|
|
if (!prev.isInput && !next) {
|
|
// Between an editable field and the end of the row.
|
|
if (prev.isField() && prev.isEditable) {
|
|
return Blockly.blockRendering.constants.MEDIUM_PADDING;
|
|
}
|
|
// Padding at the end of an icon-only row to make the block shape clearer.
|
|
if (prev.isIcon()) {
|
|
return (Blockly.blockRendering.constants.LARGE_PADDING * 2) + 1;
|
|
}
|
|
if (prev.isHat()) {
|
|
return Blockly.blockRendering.constants.NO_PADDING;
|
|
}
|
|
// Establish a minimum width for a block with a previous or next connection.
|
|
if (prev.isPreviousConnection() || prev.isNextConnection()) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
// Between rounded corner and the end of the row.
|
|
if (prev.isRoundedCorner()) {
|
|
return Blockly.blockRendering.constants.MIN_BLOCK_WIDTH;
|
|
}
|
|
// Between a jagged edge and the end of the row.
|
|
if (prev.isJaggedEdge()) {
|
|
return Blockly.blockRendering.constants.NO_PADDING;
|
|
}
|
|
// Between noneditable fields and icons and the end of the row.
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
|
|
// Between inputs and the end of the row.
|
|
if (prev.isInput && !next) {
|
|
if (prev.isExternalInput()) {
|
|
return Blockly.blockRendering.constants.NO_PADDING;
|
|
} else if (prev.isInlineInput()) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
} else if (prev.isStatementInput()) {
|
|
return Blockly.blockRendering.constants.NO_PADDING;
|
|
}
|
|
}
|
|
|
|
// Spacing between a non-input and an input.
|
|
if (!prev.isInput && next.isInput) {
|
|
// Between an editable field and an input.
|
|
if (prev.isEditable) {
|
|
if (next.isInlineInput()) {
|
|
return Blockly.blockRendering.constants.SMALL_PADDING;
|
|
} else if (next.isExternalInput()) {
|
|
return Blockly.blockRendering.constants.SMALL_PADDING;
|
|
}
|
|
} else {
|
|
if (next.isInlineInput()) {
|
|
return Blockly.blockRendering.constants.MEDIUM_LARGE_PADDING;
|
|
} else if (next.isExternalInput()) {
|
|
return Blockly.blockRendering.constants.MEDIUM_LARGE_PADDING;
|
|
} else if (next.isStatementInput()) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
}
|
|
return Blockly.blockRendering.constants.LARGE_PADDING - 1;
|
|
}
|
|
|
|
// Spacing between an icon and an icon or field.
|
|
if (prev.isIcon() && !next.isInput) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
|
|
// Spacing between an inline input and a field.
|
|
if (prev.isInlineInput() && !next.isInput) {
|
|
// Editable field after inline input.
|
|
if (next.isEditable) {
|
|
return Blockly.blockRendering.constants.MEDIUM_PADDING;
|
|
} else {
|
|
// Noneditable field after inline input.
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
}
|
|
|
|
if (prev.isSquareCorner()) {
|
|
// Spacing between a hat and a corner
|
|
if (next.isHat()) {
|
|
return Blockly.blockRendering.constants.NO_PADDING;
|
|
}
|
|
// Spacing between a square corner and a previous or next connection
|
|
if (next.isPreviousConnection()) {
|
|
return next.notchOffset;
|
|
} else if (next.isNextConnection()) {
|
|
// Next connections are shifted slightly to the left (in both LTR and RTL)
|
|
// to make the dark path under the previous connection show through.
|
|
var offset = (this.RTL ? 1 : -1) *
|
|
Blockly.blockRendering.constants.DARK_PATH_OFFSET / 2;
|
|
return next.notchOffset + offset;
|
|
}
|
|
}
|
|
|
|
// Spacing between a rounded corner and a previous or next connection.
|
|
if (prev.isRoundedCorner()) {
|
|
if (next.isPreviousConnection()) {
|
|
return next.notchOffset - Blockly.blockRendering.constants.CORNER_RADIUS;
|
|
} else if (next.isNextConnection()) {
|
|
// Next connections are shifted slightly to the left (in both LTR and RTL)
|
|
// to make the dark path under the previous connection show through.
|
|
var offset = (this.RTL ? 1 : -1) *
|
|
Blockly.blockRendering.constants.DARK_PATH_OFFSET / 2;
|
|
return next.notchOffset - Blockly.blockRendering.constants.CORNER_RADIUS + offset;
|
|
}
|
|
}
|
|
|
|
// Spacing between two fields of the same editability.
|
|
if (!prev.isInput && !next.isInput && (prev.isEditable == next.isEditable)) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
|
|
// Spacing between anything and a jagged edge.
|
|
if (next.isJaggedEdge()) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
|
|
return Blockly.blockRendering.constants.MEDIUM_PADDING;
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.geras.RenderInfo.prototype.alignRowElements_ = function() {
|
|
for (var i = 0, row; (row = this.rows[i]); i++) {
|
|
if (!row.hasInlineInput) {
|
|
if (row.hasStatement) {
|
|
var statementInput = row.getLastInput();
|
|
var currentWidth = row.width - statementInput.width;
|
|
var desiredWidth = this.statementEdge - this.startX;
|
|
} else {
|
|
var currentWidth = row.width;
|
|
var desiredWidth = this.width - this.startX;
|
|
}
|
|
if (row.type == 'bottom row' && row.hasFixedWidth) {
|
|
desiredWidth = Blockly.blockRendering.constants.MAX_BOTTOM_WIDTH;
|
|
}
|
|
var missingSpace = desiredWidth - currentWidth;
|
|
if (missingSpace) {
|
|
this.addAlignmentPadding_(row, missingSpace);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.geras.RenderInfo.prototype.addAlignmentPadding_ = function(row, missingSpace) {
|
|
var elems = row.elements;
|
|
var input = row.getLastInput();
|
|
if (input) {
|
|
var firstSpacer = row.getFirstSpacer();
|
|
var lastSpacer = row.getLastSpacer();
|
|
if (row.hasExternalInput || row.hasStatement) {
|
|
// Get the spacer right before the input socket.
|
|
lastSpacer = elems[elems.length - 3];
|
|
row.widthWithConnectedBlocks += missingSpace;
|
|
}
|
|
// Decide where the extra padding goes.
|
|
if (input.align == Blockly.ALIGN_LEFT) {
|
|
// Add padding to the end of the row.
|
|
lastSpacer.width += missingSpace;
|
|
} else if (input.align == Blockly.ALIGN_CENTRE) {
|
|
// Split the padding between the beginning and end of the row.
|
|
firstSpacer.width += missingSpace / 2;
|
|
lastSpacer.width += missingSpace / 2;
|
|
} else if (input.align == Blockly.ALIGN_RIGHT) {
|
|
// Add padding at the beginning of the row.
|
|
firstSpacer.width += missingSpace;
|
|
}
|
|
row.width += missingSpace;
|
|
// Top and bottom rows are always left aligned.
|
|
} else if (row.type == 'top row' || row.type == 'bottom row') {
|
|
row.getLastSpacer().width += missingSpace;
|
|
row.width += missingSpace;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.geras.RenderInfo.prototype.getSpacerRowWidth_ = function(prev, next) {
|
|
// The width of the spacer before the bottom row should be the same as the
|
|
// bottom row.
|
|
if (next.type == 'bottom row' && next.hasFixedWidth) {
|
|
return next.width;
|
|
}
|
|
return this.width - this.startX;
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.geras.RenderInfo.prototype.getSpacerRowHeight_ = function(prev, next) {
|
|
// If we have an empty block add a spacer to increase the height.
|
|
if (prev.type == 'top row' && next.type == 'bottom row') {
|
|
return Blockly.blockRendering.constants.EMPTY_BLOCK_SPACER_HEIGHT;
|
|
}
|
|
// Top and bottom rows act as a spacer so we don't need any extra padding.
|
|
if (prev.type == 'top row' || next.type == 'bottom row') {
|
|
return Blockly.blockRendering.constants.NO_PADDING;
|
|
}
|
|
if (prev.hasExternalInput && next.hasExternalInput) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
if (!prev.hasStatement && next.hasStatement) {
|
|
return Blockly.blockRendering.constants.BETWEEN_STATEMENT_PADDING_Y;
|
|
}
|
|
if (prev.hasStatement && next.hasStatement) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
if (next.hasDummyInput) {
|
|
return Blockly.blockRendering.constants.LARGE_PADDING;
|
|
}
|
|
return Blockly.blockRendering.constants.MEDIUM_PADDING;
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.geras.RenderInfo.prototype.getElemCenterline_ = function(row, elem) {
|
|
var result = row.yPos;
|
|
if (elem.isField() || elem.isIcon()) {
|
|
result += (elem.height / 2);
|
|
if (row.hasInlineInput || row.hasStatement) {
|
|
result += Blockly.blockRendering.constants.TALL_INPUT_FIELD_OFFSET_Y;
|
|
}
|
|
} else if (elem.isInlineInput()) {
|
|
result += elem.height / 2;
|
|
} else if (elem.isNextConnection()) {
|
|
result += (row.height - row.overhangY + elem.height / 2);
|
|
} else {
|
|
result += (row.height / 2);
|
|
}
|
|
return result;
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.geras.RenderInfo.prototype.finalize_ = function() {
|
|
// Performance note: this could be combined with the draw pass, if the time
|
|
// that this takes is excessive. But it shouldn't be, because it only
|
|
// accesses and sets properties that already exist on the objects.
|
|
var widestRowWithConnectedBlocks = 0;
|
|
var yCursor = 0;
|
|
for (var i = 0, row; (row = this.rows[i]); i++) {
|
|
row.yPos = yCursor;
|
|
row.xPos = this.startX;
|
|
yCursor += row.height;
|
|
|
|
widestRowWithConnectedBlocks =
|
|
Math.max(widestRowWithConnectedBlocks, row.widthWithConnectedBlocks);
|
|
// Add padding to the bottom row if block height is less than minimum
|
|
var heightWithoutHat = yCursor - this.topRow.startY;
|
|
if (row == this.bottomRow &&
|
|
heightWithoutHat < Blockly.blockRendering.constants.MIN_BLOCK_HEIGHT) {
|
|
// But the hat height shouldn't be part of this.
|
|
var diff = Blockly.blockRendering.constants.MIN_BLOCK_HEIGHT - heightWithoutHat;
|
|
this.bottomRow.height += diff;
|
|
yCursor += diff;
|
|
}
|
|
var xCursor = row.xPos;
|
|
for (var j = 0, elem; (elem = row.elements[j]); j++) {
|
|
elem.xPos = xCursor;
|
|
elem.centerline = this.getElemCenterline_(row, elem);
|
|
xCursor += elem.width;
|
|
}
|
|
}
|
|
|
|
this.widthWithChildren = widestRowWithConnectedBlocks + this.startX;
|
|
|
|
this.height = yCursor;
|
|
this.startY = this.topRow.startY;
|
|
};
|