2018-04-10 00:59:28 +02:00
|
|
|
/// <reference path="../node_modules/pxt-core/localtypings/blockly.d.ts"/>
|
|
|
|
/// <reference path="../node_modules/pxt-core/built/pxtsim.d.ts"/>
|
|
|
|
|
|
|
|
export interface FieldMotorsOptions extends Blockly.FieldCustomDropdownOptions {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
export class FieldMotors extends Blockly.FieldDropdown implements Blockly.FieldCustom {
|
|
|
|
public isFieldCustom_ = true;
|
|
|
|
|
|
|
|
private box2_: SVGRectElement;
|
|
|
|
private textElement2_: SVGTextElement;
|
|
|
|
private arrow2_: SVGImageElement;
|
|
|
|
|
|
|
|
// Width in pixels
|
|
|
|
protected itemWidth_: number;
|
|
|
|
|
|
|
|
// Number of rows to display (if there are extra rows, the picker will be scrollable)
|
|
|
|
protected maxRows_: number;
|
|
|
|
|
|
|
|
protected backgroundColour_: string;
|
|
|
|
protected itemColour_: string;
|
|
|
|
protected borderColour_: string;
|
|
|
|
|
|
|
|
private isFirst_: boolean; // which of the two dropdowns is selected
|
|
|
|
|
|
|
|
constructor(text: string, options: FieldMotorsOptions, validator?: Function) {
|
|
|
|
super(options.data, validator);
|
|
|
|
|
|
|
|
this.itemWidth_ = 75;
|
|
|
|
this.backgroundColour_ = pxtblockly.parseColour(options.colour);
|
|
|
|
this.itemColour_ = "rgba(255, 255, 255, 0.6)";
|
2018-04-14 06:46:19 +02:00
|
|
|
this.borderColour_ = pxt.toolbox.fadeColor(this.backgroundColour_, 0.4, false);
|
2018-04-10 00:59:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
init() {
|
|
|
|
if (this.fieldGroup_) {
|
|
|
|
// Field has already been initialized once.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Add dropdown arrow: "option ▾" (LTR) or "▾ אופציה" (RTL)
|
|
|
|
// Positioned on render, after text size is calculated.
|
|
|
|
/** @type {Number} */
|
|
|
|
(this as any).arrowSize_ = 12;
|
|
|
|
/** @type {Number} */
|
|
|
|
(this as any).arrowX_ = 0;
|
|
|
|
/** @type {Number} */
|
|
|
|
this.arrowY_ = 11;
|
|
|
|
this.arrow_ = Blockly.utils.createSvgElement('image', {
|
|
|
|
'height': (this as any).arrowSize_ + 'px',
|
|
|
|
'width': (this as any).arrowSize_ + 'px'
|
|
|
|
});
|
|
|
|
this.arrow_.setAttributeNS('http://www.w3.org/1999/xlink',
|
|
|
|
'xlink:href', (Blockly.FieldDropdown as any).DROPDOWN_SVG_DATAURI);
|
|
|
|
|
|
|
|
this.arrow2_ = Blockly.utils.createSvgElement('image', {
|
|
|
|
'height': (this as any).arrowSize_ + 'px',
|
|
|
|
'width': (this as any).arrowSize_ + 'px'
|
|
|
|
});
|
|
|
|
this.arrow2_.setAttributeNS('http://www.w3.org/1999/xlink',
|
|
|
|
'xlink:href', (Blockly.FieldDropdown as any).DROPDOWN_SVG_DATAURI);
|
|
|
|
(this as any).className_ += ' blocklyDropdownText';
|
|
|
|
|
|
|
|
// Build the DOM.
|
|
|
|
this.fieldGroup_ = Blockly.utils.createSvgElement('g', {}, null);
|
|
|
|
if (!this.visible_) {
|
|
|
|
(this.fieldGroup_ as any).style.display = 'none';
|
|
|
|
}
|
|
|
|
// Adjust X to be flipped for RTL. Position is relative to horizontal start of source block.
|
|
|
|
var size = this.getSize();
|
|
|
|
var fieldX = (this.sourceBlock_.RTL) ? -size.width / 2 : size.width / 2;
|
|
|
|
/** @type {!Element} */
|
|
|
|
this.textElement_ = Blockly.utils.createSvgElement('text',
|
|
|
|
{
|
|
|
|
'class': (this as any).className_,
|
|
|
|
'x': fieldX,
|
|
|
|
'dy': '0.7ex',
|
|
|
|
'y': size.height / 2
|
|
|
|
},
|
|
|
|
this.fieldGroup_);
|
|
|
|
fieldX += 10; // size of first group.
|
|
|
|
this.textElement2_ = Blockly.utils.createSvgElement('text',
|
|
|
|
{
|
|
|
|
'class': (this as any).className_,
|
|
|
|
'x': fieldX,
|
|
|
|
'dy': '0.7ex',
|
|
|
|
'y': this.size_.height / 2
|
|
|
|
},
|
|
|
|
this.fieldGroup_);
|
|
|
|
|
|
|
|
this.updateEditable();
|
|
|
|
this.sourceBlock_.getSvgRoot().appendChild(this.fieldGroup_);
|
|
|
|
// Force a render.
|
|
|
|
this.render_();
|
|
|
|
this.size_.width = 0;
|
|
|
|
(this as any).mouseDownWrapper_ =
|
|
|
|
Blockly.bindEventWithChecks_((this as any).getClickTarget_(), 'mousedown', this,
|
|
|
|
(this as any).onMouseDown_);
|
|
|
|
|
|
|
|
// Add second dropdown
|
|
|
|
if (this.shouldShowRect_()) {
|
|
|
|
this.box_ = Blockly.utils.createSvgElement('rect', {
|
|
|
|
'rx': (Blockly.BlockSvg as any).CORNER_RADIUS,
|
|
|
|
'ry': (Blockly.BlockSvg as any).CORNER_RADIUS,
|
|
|
|
'x': 0,
|
|
|
|
'y': 0,
|
|
|
|
'width': this.size_.width,
|
|
|
|
'height': this.size_.height,
|
|
|
|
'stroke': this.sourceBlock_.getColourTertiary(),
|
|
|
|
'fill': this.sourceBlock_.getColour(),
|
|
|
|
'class': 'blocklyBlockBackground',
|
|
|
|
'fill-opacity': 1
|
|
|
|
}, null);
|
|
|
|
this.fieldGroup_.insertBefore(this.box_, this.textElement_);
|
|
|
|
this.box2_ = Blockly.utils.createSvgElement('rect', {
|
|
|
|
'rx': (Blockly.BlockSvg as any).CORNER_RADIUS,
|
|
|
|
'ry': (Blockly.BlockSvg as any).CORNER_RADIUS,
|
|
|
|
'x': 0,
|
|
|
|
'y': 0,
|
|
|
|
'width': this.size_.width,
|
|
|
|
'height': this.size_.height,
|
|
|
|
'stroke': this.sourceBlock_.getColourTertiary(),
|
|
|
|
'fill': this.sourceBlock_.getColour(),
|
|
|
|
'class': 'blocklyBlockBackground',
|
|
|
|
'fill-opacity': 1
|
|
|
|
}, null);
|
|
|
|
this.fieldGroup_.insertBefore(this.box2_, this.textElement2_);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Force a reset of the text to add the arrow.
|
|
|
|
var text = this.text_;
|
|
|
|
this.text_ = null;
|
|
|
|
this.setText(text);
|
|
|
|
}
|
|
|
|
|
|
|
|
getFirstValue(text: string) {
|
|
|
|
// Get first set of words up until last space
|
|
|
|
return this.normalizeText_(text.substring(0, text.lastIndexOf(' ')));
|
|
|
|
}
|
|
|
|
|
|
|
|
getSecondValue(text: string) {
|
|
|
|
// Get last word
|
|
|
|
return this.normalizeText_(text.match(/\S*$/)[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
private normalizeText_(text: string) {
|
|
|
|
if (!text) {
|
|
|
|
// Prevent the field from disappearing if empty.
|
|
|
|
return Blockly.Field.NBSP;
|
|
|
|
}
|
|
|
|
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) {
|
|
|
|
// The SVG is LTR, force text to be RTL.
|
|
|
|
text += '\u200F';
|
|
|
|
}
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
|
|
|
|
updateTextNode2_() {
|
|
|
|
if (!this.textElement2_) {
|
|
|
|
// 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';
|
|
|
|
// Add special class for sizing font when truncated
|
|
|
|
this.textElement2_.setAttribute('class', (this as any).className_ + ' blocklyTextTruncated');
|
|
|
|
} else {
|
|
|
|
this.textElement2_.setAttribute('class', (this as any).className_);
|
|
|
|
}
|
|
|
|
// Empty the text element.
|
|
|
|
goog.dom.removeChildren(/** @type {!Element} */(this.textElement2_));
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
var textNode = document.createTextNode(text);
|
|
|
|
this.textElement2_.appendChild(textNode);
|
|
|
|
|
|
|
|
// Cached width is obsolete. Clear it.
|
|
|
|
this.size_.width = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
patchDualMotorText(text: string) {
|
|
|
|
if (text === null) {
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
if (text.indexOf(' ') == -1) {
|
|
|
|
text = `large motors ${text}`;
|
|
|
|
}
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
|
|
|
|
setText(text: string) {
|
|
|
|
if (text === null || text === this.text_) {
|
|
|
|
// No change if null.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
text = this.patchDualMotorText(text);
|
|
|
|
this.text_ = text;
|
|
|
|
this.updateTextNode_();
|
|
|
|
this.updateTextNode2_();
|
|
|
|
|
|
|
|
if (this.textElement_) {
|
|
|
|
this.textElement_.parentNode.appendChild(this.arrow_);
|
|
|
|
}
|
|
|
|
if (this.textElement2_) {
|
|
|
|
this.textElement2_.parentNode.appendChild(this.arrow2_);
|
|
|
|
}
|
|
|
|
if (this.sourceBlock_ && this.sourceBlock_.rendered) {
|
|
|
|
this.sourceBlock_.render();
|
|
|
|
this.sourceBlock_.bumpNeighbours_();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
positionArrow2(start: number, x: number) {
|
|
|
|
if (!this.arrow2_) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
var addedWidth = 0;
|
|
|
|
if (this.sourceBlock_.RTL) {
|
|
|
|
(this as any).arrow2X_ = (this as any).arrowSize_ - (Blockly.BlockSvg as any).DROPDOWN_ARROW_PADDING;
|
|
|
|
addedWidth = (this as any).arrowSize_ + (Blockly.BlockSvg as any).DROPDOWN_ARROW_PADDING;
|
|
|
|
} else {
|
|
|
|
(this as any).arrow2X_ = x + (Blockly.BlockSvg as any).DROPDOWN_ARROW_PADDING / 2;
|
|
|
|
addedWidth = (this as any).arrowSize_ + (Blockly.BlockSvg as any).DROPDOWN_ARROW_PADDING;
|
|
|
|
}
|
|
|
|
if (this.box_) {
|
|
|
|
// Bump positioning to the right for a box-type drop-down.
|
|
|
|
(this as any).arrow2X_ += Blockly.BlockSvg.BOX_FIELD_PADDING;
|
|
|
|
}
|
|
|
|
(this as any).arrow2X_ += start;
|
|
|
|
this.arrow2_.setAttribute('transform',
|
|
|
|
'translate(' + (this as any).arrow2X_ + ',' + this.arrowY_ + ')'
|
|
|
|
);
|
|
|
|
return addedWidth;
|
|
|
|
};
|
|
|
|
|
|
|
|
updateWidth() {
|
|
|
|
// Calculate width of field
|
|
|
|
var width = Blockly.Field.getCachedWidth(this.textElement_);
|
|
|
|
var width2 = Blockly.Field.getCachedWidth(this.textElement2_);
|
|
|
|
|
|
|
|
// Add padding to left and right of text.
|
|
|
|
if (this.EDITABLE) {
|
|
|
|
width += Blockly.BlockSvg.EDITABLE_FIELD_PADDING;
|
|
|
|
width2 += Blockly.BlockSvg.EDITABLE_FIELD_PADDING;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Adjust width for drop-down arrow.
|
|
|
|
this.arrowWidth_ = 0;
|
|
|
|
if (this.positionArrow) {
|
|
|
|
this.arrowWidth_ = this.positionArrow(width);
|
|
|
|
width += this.arrowWidth_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add padding to any drawn box.
|
|
|
|
if (this.box_) {
|
|
|
|
width += 2 * Blockly.BlockSvg.BOX_FIELD_PADDING;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Adjust width for second drop-down arrow.
|
|
|
|
(this as any).arrowWidth2_ = 0;
|
|
|
|
if (this.positionArrow2) {
|
|
|
|
(this as any).arrowWidth2_ = this.positionArrow2(width + Blockly.BlockSvg.BOX_FIELD_PADDING, width2);
|
|
|
|
width2 += (this as any).arrowWidth2_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add padding to any drawn box.
|
|
|
|
if (this.box2_) {
|
|
|
|
width2 += 2 * Blockly.BlockSvg.BOX_FIELD_PADDING;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set width of the field.
|
|
|
|
this.size_.width = width + Blockly.BlockSvg.BOX_FIELD_PADDING + width2;
|
|
|
|
(this as any).width1 = width;
|
|
|
|
(this as any).width2 = width2;
|
|
|
|
};
|
|
|
|
|
|
|
|
render_() {
|
|
|
|
if (this.visible_ && this.textElement_) {
|
|
|
|
goog.dom.removeChildren(/** @type {!Element} */(this.textElement_));
|
|
|
|
goog.dom.removeChildren(/** @type {!Element} */(this.textElement2_));
|
|
|
|
|
|
|
|
var text = this.text_;
|
|
|
|
text = this.patchDualMotorText(text);
|
|
|
|
|
|
|
|
// First dropdown
|
|
|
|
const textNode1 = document.createTextNode(this.getFirstValue(text));
|
|
|
|
this.textElement_.appendChild(textNode1);
|
|
|
|
|
|
|
|
// Second dropdown
|
|
|
|
if (this.textElement2_) {
|
|
|
|
const textNode2 = document.createTextNode(this.getSecondValue(text));
|
|
|
|
this.textElement2_.appendChild(textNode2);
|
|
|
|
}
|
|
|
|
this.updateWidth();
|
|
|
|
|
|
|
|
// Update text centering, based on newly calculated width.
|
|
|
|
let centerTextX = ((this as any).width1 - this.arrowWidth_) / 2;
|
|
|
|
if (this.sourceBlock_.RTL) {
|
|
|
|
centerTextX += this.arrowWidth_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// In a text-editing shadow block's field,
|
|
|
|
// if half the text length is not at least center of
|
|
|
|
// visible field (FIELD_WIDTH), center it there instead,
|
|
|
|
// unless there is a drop-down arrow.
|
|
|
|
if (this.sourceBlock_.isShadow() && !this.positionArrow) {
|
|
|
|
let minOffset = (Blockly.BlockSvg as any).FIELD_WIDTH / 2;
|
|
|
|
if (this.sourceBlock_.RTL) {
|
|
|
|
// X position starts at the left edge of the block, in both RTL and LTR.
|
|
|
|
// First offset by the width of the block to move to the right edge,
|
|
|
|
// and then subtract to move to the same position as LTR.
|
|
|
|
let minCenter = (this as any).width1 - minOffset;
|
|
|
|
centerTextX = Math.min(minCenter, centerTextX);
|
|
|
|
} else {
|
|
|
|
// (width / 2) should exceed Blockly.BlockSvg.FIELD_WIDTH / 2
|
|
|
|
// if the text is longer.
|
|
|
|
centerTextX = Math.max(minOffset, centerTextX);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply new text element x position.
|
|
|
|
var width = Blockly.Field.getCachedWidth(this.textElement_);
|
|
|
|
var newX = centerTextX - width / 2;
|
|
|
|
this.textElement_.setAttribute('x', `${newX}`);
|
|
|
|
|
|
|
|
// Update text centering, based on newly calculated width.
|
|
|
|
let centerTextX2 = ((this as any).width2 - (this as any).arrowWidth2_) / 2;
|
|
|
|
if (this.sourceBlock_.RTL) {
|
|
|
|
centerTextX2 += (this as any).arrowWidth2_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// In a text-editing shadow block's field,
|
|
|
|
// if half the text length is not at least center of
|
|
|
|
// visible field (FIELD_WIDTH), center it there instead,
|
|
|
|
// unless there is a drop-down arrow.
|
|
|
|
if (this.sourceBlock_.isShadow() && !this.positionArrow2) {
|
|
|
|
let minOffset = (Blockly.BlockSvg as any).FIELD_WIDTH / 2;
|
|
|
|
if (this.sourceBlock_.RTL) {
|
|
|
|
// X position starts at the left edge of the block, in both RTL and LTR.
|
|
|
|
// First offset by the width of the block to move to the right edge,
|
|
|
|
// and then subtract to move to the same position as LTR.
|
|
|
|
let minCenter = (this as any).width2 - minOffset;
|
|
|
|
centerTextX2 = Math.min(minCenter, centerTextX2);
|
|
|
|
} else {
|
|
|
|
// (width / 2) should exceed Blockly.BlockSvg.FIELD_WIDTH / 2
|
|
|
|
// if the text is longer.
|
|
|
|
centerTextX2 = Math.max(minOffset, centerTextX2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply new text element x position.
|
|
|
|
var width2 = Blockly.Field.getCachedWidth(this.textElement2_);
|
|
|
|
var newX2 = centerTextX2 - width2 / 2;
|
|
|
|
this.textElement2_.setAttribute('x', `${newX2 + (this as any).width1 + Blockly.BlockSvg.BOX_FIELD_PADDING}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update any drawn box to the correct width and height.
|
|
|
|
if (this.box_) {
|
|
|
|
this.box_.setAttribute('width', `${(this as any).width1}`);
|
|
|
|
this.box_.setAttribute('height', `${this.size_.height}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update any drawn box to the correct width and height.
|
|
|
|
if (this.box2_) {
|
|
|
|
this.box2_.setAttribute('x', `${(this as any).width1 + Blockly.BlockSvg.BOX_FIELD_PADDING}`);
|
|
|
|
this.box2_.setAttribute('width', `${(this as any).width2}`);
|
|
|
|
this.box2_.setAttribute('height', `${this.size_.height}`);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
showEditor_(e?: MouseEvent) {
|
|
|
|
// If there is an existing drop-down we own, this is a request to hide the drop-down.
|
|
|
|
if (Blockly.DropDownDiv.hideIfOwner(this)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.isFirst_ = e.clientX - this.getScaledBBox_().left < ((this as any).width1 * this.sourceBlock_.workspace.scale);
|
|
|
|
// If there is an existing drop-down someone else owns, hide it immediately and clear it.
|
|
|
|
Blockly.DropDownDiv.hideWithoutAnimation();
|
|
|
|
Blockly.DropDownDiv.clearContent();
|
|
|
|
// Populate the drop-down with the icons for this field.
|
|
|
|
let dropdownDiv = Blockly.DropDownDiv.getContentDiv();
|
|
|
|
let contentDiv = document.createElement('div');
|
|
|
|
// Accessibility properties
|
|
|
|
contentDiv.setAttribute('role', 'menu');
|
|
|
|
contentDiv.setAttribute('aria-haspopup', 'true');
|
|
|
|
let options = this.getOptions();
|
|
|
|
|
|
|
|
// Hashmap of options
|
|
|
|
let opts = {};
|
|
|
|
let conts = {};
|
|
|
|
let vals = {};
|
2018-04-14 00:21:41 +02:00
|
|
|
for (let opt = 0; opt < options.length; opt++) {
|
2018-04-10 00:59:28 +02:00
|
|
|
let text = options[opt][0].alt ? options[opt][0].alt : options[opt][0];
|
|
|
|
if (text.indexOf(' ') == -1) {
|
|
|
|
// Patch dual motors as they don't have prefixes.
|
|
|
|
text = this.patchDualMotorText(text);
|
|
|
|
if (options[opt][0].alt) options[opt][0].alt = text;
|
|
|
|
}
|
|
|
|
const value = options[opt][1];
|
|
|
|
const firstValue = this.getFirstValue(text);
|
|
|
|
const secondValue = this.getSecondValue(text);
|
|
|
|
if (!opts[firstValue]) opts[firstValue] = [secondValue];
|
|
|
|
else opts[firstValue].push(secondValue);
|
|
|
|
// Store a hash of the original key value pairs for later
|
|
|
|
conts[text] = options[opt][0];
|
|
|
|
vals[text] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
const currentFirst = this.getFirstValue(this.text_);
|
|
|
|
const currentSecond = this.getSecondValue(this.text_);
|
|
|
|
|
|
|
|
if (!this.isFirst_) {
|
|
|
|
options = opts[currentFirst];
|
|
|
|
} else {
|
|
|
|
options = Object.keys(opts);
|
|
|
|
// Flip the first and second options to make it sorted the way we want it (medium, large, dual)
|
|
|
|
if (options.length == 3) {
|
|
|
|
const temp = options[1];
|
|
|
|
options[1] = options[0];
|
|
|
|
options[0] = temp;
|
|
|
|
} else {
|
|
|
|
options.reverse();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const isFirstUrl = {
|
|
|
|
'large motors': FieldMotors.DUAL_MOTORS_DATAURI,
|
|
|
|
'large motor': FieldMotors.MOTORS_LARGE_DATAURI,
|
|
|
|
'medium motor': FieldMotors.MOTORS_MEDIUM_DATAURI
|
|
|
|
}
|
|
|
|
const columns = options.length;
|
|
|
|
|
|
|
|
for (let i = 0, option: any; option = options[i]; i++) {
|
|
|
|
let text = this.isFirst_ ? option + ' ' + opts[option][0] : currentFirst + ' ' + option;
|
|
|
|
text = text.replace(/\xA0/g, ' ');
|
|
|
|
const content: any = conts[text];
|
|
|
|
const value = vals[text];
|
|
|
|
// Icons with the type property placeholder take up space but don't have any functionality
|
|
|
|
// Use for special-case layouts
|
|
|
|
if (content.type == 'placeholder') {
|
|
|
|
let placeholder = document.createElement('span');
|
|
|
|
placeholder.setAttribute('class', 'blocklyDropDownPlaceholder');
|
|
|
|
placeholder.style.width = content.width + 'px';
|
|
|
|
placeholder.style.height = content.height + 'px';
|
|
|
|
contentDiv.appendChild(placeholder);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
let button = document.createElement('button');
|
|
|
|
button.setAttribute('id', ':' + i); // For aria-activedescendant
|
|
|
|
button.setAttribute('role', 'menuitem');
|
|
|
|
button.setAttribute('class', 'blocklyDropDownButton');
|
|
|
|
button.title = this.isFirst_ ? this.getFirstValue(content.alt) : content.alt;
|
|
|
|
button.style.width = ((this.itemWidth_) - 8) + 'px';
|
|
|
|
button.style.height = ((this.itemWidth_) - 8) + 'px';
|
|
|
|
let backgroundColor = this.backgroundColour_;
|
|
|
|
if (value == this.getValue()) {
|
|
|
|
// This icon is selected, show it in a different colour
|
|
|
|
backgroundColor = this.sourceBlock_.getColourTertiary();
|
|
|
|
button.setAttribute('aria-selected', 'true');
|
|
|
|
}
|
|
|
|
button.style.backgroundColor = backgroundColor;
|
|
|
|
button.style.borderColor = this.borderColour_;
|
|
|
|
Blockly.bindEvent_(button, 'click', this, this.buttonClick_);
|
|
|
|
Blockly.bindEvent_(button, 'mouseup', this, this.buttonClick_);
|
|
|
|
// These are applied manually instead of using the :hover pseudoclass
|
|
|
|
// because Android has a bad long press "helper" menu and green highlight
|
|
|
|
// that we must prevent with ontouchstart preventDefault
|
|
|
|
Blockly.bindEvent_(button, 'mousedown', button, function (e) {
|
|
|
|
this.setAttribute('class', 'blocklyDropDownButton blocklyDropDownButtonHover');
|
|
|
|
e.preventDefault();
|
|
|
|
});
|
|
|
|
Blockly.bindEvent_(button, 'mouseover', button, function () {
|
|
|
|
this.setAttribute('class', 'blocklyDropDownButton blocklyDropDownButtonHover');
|
|
|
|
contentDiv.setAttribute('aria-activedescendant', this.id);
|
|
|
|
});
|
|
|
|
Blockly.bindEvent_(button, 'mouseout', button, function () {
|
|
|
|
this.setAttribute('class', 'blocklyDropDownButton');
|
|
|
|
contentDiv.removeAttribute('aria-activedescendant');
|
|
|
|
});
|
|
|
|
let buttonImg = document.createElement('img');
|
|
|
|
buttonImg.src = this.isFirst_ ? isFirstUrl[option.replace(/\xA0/g, ' ')] : content.src;
|
|
|
|
//buttonImg.alt = icon.alt;
|
|
|
|
// Upon click/touch, we will be able to get the clicked element as e.target
|
|
|
|
// Store a data attribute on all possible click targets so we can match it to the icon.
|
|
|
|
button.setAttribute('data-value', value);
|
|
|
|
buttonImg.setAttribute('data-value', value);
|
|
|
|
button.appendChild(buttonImg);
|
|
|
|
contentDiv.appendChild(button);
|
|
|
|
}
|
|
|
|
contentDiv.style.width = (this.itemWidth_ * columns) + 'px';
|
|
|
|
dropdownDiv.appendChild(contentDiv);
|
|
|
|
|
|
|
|
Blockly.DropDownDiv.setColour(this.backgroundColour_, this.borderColour_);
|
|
|
|
|
|
|
|
// Calculate positioning based on the field position.
|
|
|
|
let scale = this.sourceBlock_.workspace.scale;
|
|
|
|
let width = this.isFirst_ ? (this as any).width1 : (this as any).width2;
|
|
|
|
let bBox = { width: this.size_.width, height: this.size_.height };
|
|
|
|
width *= scale;
|
|
|
|
bBox.height *= scale;
|
|
|
|
let position = this.fieldGroup_.getBoundingClientRect();
|
|
|
|
let leftPosition = this.isFirst_ ? position.left : position.left + (scale * (this as any).width1) + Blockly.BlockSvg.BOX_FIELD_PADDING;
|
|
|
|
let primaryX = leftPosition + width / 2;
|
|
|
|
let primaryY = position.top + bBox.height;
|
|
|
|
let secondaryX = primaryX;
|
|
|
|
let secondaryY = position.top;
|
|
|
|
// Set bounds to workspace; show the drop-down.
|
|
|
|
(Blockly.DropDownDiv as any).setBoundsElement(this.sourceBlock_.workspace.getParentSvg().parentNode);
|
|
|
|
(Blockly.DropDownDiv as any).show(this, primaryX, primaryY, secondaryX, secondaryY,
|
|
|
|
this.onHide_.bind(this));
|
|
|
|
|
|
|
|
// Update colour to look selected.
|
|
|
|
if (this.isFirst_ && this.box_) {
|
|
|
|
this.box_.setAttribute('fill', this.sourceBlock_.getColourTertiary());
|
|
|
|
} else if (!this.isFirst_ && this.box2_) {
|
|
|
|
this.box2_.setAttribute('fill', this.sourceBlock_.getColourTertiary());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected buttonClick_ = function (e: any) {
|
|
|
|
let value = e.target.getAttribute('data-value');
|
|
|
|
this.setValue(value);
|
|
|
|
Blockly.DropDownDiv.hide();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback for when the drop-down is hidden.
|
|
|
|
*/
|
|
|
|
protected onHide_ = function () {
|
|
|
|
Blockly.DropDownDiv.content_.removeAttribute('role');
|
|
|
|
Blockly.DropDownDiv.content_.removeAttribute('aria-haspopup');
|
|
|
|
Blockly.DropDownDiv.content_.removeAttribute('aria-activedescendant');
|
|
|
|
Blockly.DropDownDiv.getContentDiv().style.width = '';
|
|
|
|
if (this.isFirst_ && this.box_) {
|
|
|
|
this.box_.setAttribute('fill', this.sourceBlock_.getColour());
|
|
|
|
} else if (!this.isFirst_ && this.box2_) {
|
|
|
|
this.box2_.setAttribute('fill', this.sourceBlock_.getColour());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static MOTORS_MEDIUM_DATAURI = '
|
|
|
|
|
|
|
|
|
|
|
|
static MOTORS_LARGE_DATAURI = '
|
|
|
|
|
|
|
|
static DUAL_MOTORS_DATAURI = '
|
|
|
|
}
|