mirror of
https://github.com/google/blockly.git
synced 2026-01-08 17:40:09 +01:00
262 lines
7.4 KiB
JavaScript
262 lines
7.4 KiB
JavaScript
/**
|
|
* @license
|
|
* Copyright 2019 Google LLC
|
|
*
|
|
* 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 An object that owns a block's rendering SVG elements.
|
|
* @author samelh@google.com (Sam El-Husseini)
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
goog.provide('Blockly.zelos.PathObject');
|
|
|
|
goog.require('Blockly.blockRendering.PathObject');
|
|
goog.require('Blockly.zelos.ConstantProvider');
|
|
goog.require('Blockly.utils.dom');
|
|
goog.require('Blockly.utils.object');
|
|
|
|
|
|
/**
|
|
* An object that handles creating and setting each of the SVG elements
|
|
* used by the renderer.
|
|
* @param {!SVGElement} root The root SVG element.
|
|
* @param {!Blockly.Theme.BlockStyle} style The style object to use for
|
|
* colouring.
|
|
* @param {!Blockly.zelos.ConstantProvider} constants The renderer's constants.
|
|
* @constructor
|
|
* @extends {Blockly.blockRendering.PathObject}
|
|
* @package
|
|
*/
|
|
Blockly.zelos.PathObject = function(root, style, constants) {
|
|
Blockly.zelos.PathObject.superClass_.constructor.call(this, root, style,
|
|
constants);
|
|
|
|
/**
|
|
* The renderer's constant provider.
|
|
* @type {!Blockly.zelos.ConstantProvider}
|
|
*/
|
|
this.constants_ = constants;
|
|
|
|
/**
|
|
* The selected path of the block.
|
|
* @type {SVGElement}
|
|
* @private
|
|
*/
|
|
this.svgPathSelected_ = null;
|
|
|
|
/**
|
|
* The outline paths on the block.
|
|
* @type {!Object.<string,!SVGElement>}
|
|
* @private
|
|
*/
|
|
this.outlines_ = {};
|
|
|
|
/**
|
|
* A set used to determine which outlines were used during a draw pass. The
|
|
* set is initialized with a reference to all the outlines in
|
|
* `this.outlines_`. Everytime we use an outline during the draw pass, the
|
|
* reference is removed from this set.
|
|
* @type {Object.<string, number>}
|
|
* @private
|
|
*/
|
|
this.remainingOutlines_ = null;
|
|
};
|
|
Blockly.utils.object.inherits(Blockly.zelos.PathObject,
|
|
Blockly.blockRendering.PathObject);
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.setPath = function(pathString) {
|
|
Blockly.zelos.PathObject.superClass_.setPath.call(this, pathString);
|
|
if (this.svgPathSelected_) {
|
|
this.svgPathSelected_.setAttribute('d', pathString);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.applyColour = function(block) {
|
|
Blockly.zelos.PathObject.superClass_.applyColour.call(this, block);
|
|
// Set shadow stroke colour.
|
|
if (block.isShadow() && block.getParent()) {
|
|
this.svgPath.setAttribute('stroke', block.getParent().style.colourTertiary);
|
|
}
|
|
|
|
// Apply colour to outlines.
|
|
for (var i = 0, keys = Object.keys(this.outlines_),
|
|
key; (key = keys[i]); i++) {
|
|
this.outlines_[key].setAttribute('fill', this.style.colourTertiary);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.flipRTL = function() {
|
|
Blockly.zelos.PathObject.superClass_.flipRTL.call(this);
|
|
// Mirror each input outline path.
|
|
for (var i = 0, keys = Object.keys(this.outlines_),
|
|
key; (key = keys[i]); i++) {
|
|
this.outlines_[key].setAttribute('transform', 'scale(-1 1)');
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.updateDisabled_ = function(
|
|
disabled) {
|
|
Blockly.zelos.PathObject.superClass_.updateDisabled_.call(this, disabled);
|
|
for (var i = 0, keys = Object.keys(this.outlines_),
|
|
key; (key = keys[i]); i++) {
|
|
if (disabled) {
|
|
this.outlines_[key].setAttribute('fill',
|
|
'url(#' + this.constants_.disabledPatternId + ')');
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.updateSelected = function(enable) {
|
|
this.setClass_('blocklySelected', enable);
|
|
if (enable) {
|
|
if (!this.svgPathSelected_) {
|
|
this.svgPathSelected_ =
|
|
/** @type {!SVGElement} */ (this.svgPath.cloneNode(true));
|
|
this.svgPathSelected_.setAttribute('fill', 'none');
|
|
this.svgPathSelected_.setAttribute('filter',
|
|
'url(#' + this.constants_.highlightGlowFilterId + ')');
|
|
this.svgRoot.appendChild(this.svgPathSelected_);
|
|
}
|
|
} else {
|
|
if (this.svgPathSelected_) {
|
|
this.svgRoot.removeChild(this.svgPathSelected_);
|
|
this.svgPathSelected_ = null;
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.updateReplacementHighlight = function(
|
|
enable) {
|
|
this.setClass_('blocklyReplaceable', enable);
|
|
if (enable) {
|
|
this.svgPath.setAttribute('filter',
|
|
'url(#' + this.constants_.replacementGlowFilterId + ')');
|
|
} else {
|
|
this.svgPath.removeAttribute('filter');
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @override
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.updateShapeForInputHighlight = function(
|
|
conn, enable) {
|
|
var name = conn.getParentInput().name;
|
|
var outlinePath = this.getOutlinePath_(name);
|
|
if (!outlinePath) {
|
|
return;
|
|
}
|
|
if (enable) {
|
|
outlinePath.setAttribute('filter',
|
|
'url(#' + this.constants_.replacementGlowFilterId + ')');
|
|
} else {
|
|
outlinePath.removeAttribute('filter');
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Method that's called when the drawer is about to draw the block.
|
|
* @package
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.beginDrawing = function() {
|
|
this.remainingOutlines_ = {};
|
|
for (var i = 0, keys = Object.keys(this.outlines_),
|
|
key; (key = keys[i]); i++) {
|
|
// The value set here isn't used anywhere, we are just using the
|
|
// object as a Set data structure.
|
|
this.remainingOutlines_[key] = 1;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Method that's called when the drawer is done drawing.
|
|
* @package
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.endDrawing = function() {
|
|
// Go through all remaining outlines that were not used this draw pass, and
|
|
// remove them.
|
|
if (this.remainingOutlines_) {
|
|
for (var i = 0, keys = Object.keys(this.remainingOutlines_),
|
|
key; (key = keys[i]); i++) {
|
|
this.removeOutlinePath_(key);
|
|
}
|
|
}
|
|
this.remainingOutlines_ = null;
|
|
};
|
|
|
|
/**
|
|
* Set the path generated by the renderer for an outline path on the respective
|
|
* outline path SVG element.
|
|
* @param {string} name The input name.
|
|
* @param {string} pathString The path.
|
|
* @package
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.setOutlinePath = function(name, pathString) {
|
|
var outline = this.getOutlinePath_(name);
|
|
outline.setAttribute('d', pathString);
|
|
outline.setAttribute('fill', this.style.colourTertiary);
|
|
};
|
|
|
|
/**
|
|
* Create's an outline path for the specified input.
|
|
* @param {string} name The input name.
|
|
* @return {!SVGElement} The SVG outline path.
|
|
* @private
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.getOutlinePath_ = function(name) {
|
|
if (!this.outlines_[name]) {
|
|
this.outlines_[name] = Blockly.utils.dom.createSvgElement('path', {
|
|
'class': 'blocklyOutlinePath',
|
|
// IE doesn't like paths without the data definition, set empty default
|
|
'd': ''
|
|
},
|
|
this.svgRoot);
|
|
}
|
|
if (this.remainingOutlines_) {
|
|
delete this.remainingOutlines_[name];
|
|
}
|
|
return this.outlines_[name];
|
|
};
|
|
|
|
/**
|
|
* Remove an outline path that is associated with the specified input.
|
|
* @param {string} name The input name.
|
|
* @private
|
|
*/
|
|
Blockly.zelos.PathObject.prototype.removeOutlinePath_ = function(name) {
|
|
this.outlines_[name].parentNode.removeChild(this.outlines_[name]);
|
|
delete this.outlines_[name];
|
|
};
|