diff --git a/editor/extension.ts b/editor/extension.ts index 9a822927..957a41da 100644 --- a/editor/extension.ts +++ b/editor/extension.ts @@ -3,7 +3,7 @@ import { deployCoreAsync, initAsync } from "./deploy"; import { FieldPorts } from "./field_ports"; -import { FieldImages } from "./field_images"; +import { FieldMotors } from "./field_motors"; import { FieldSpeed } from "./field_speed"; import { FieldBrickButtons } from "./field_brickbuttons"; import { FieldTurnRatio } from "./field_turnratio"; @@ -17,8 +17,8 @@ pxt.editor.initExtensionsAsync = function (opts: pxt.editor.ExtensionOptions): P selector: "ports", editor: FieldPorts }, { - selector: "images", - editor: FieldImages + selector: "motors", + editor: FieldMotors }, { selector: "speed", editor: FieldSpeed diff --git a/editor/field_brickbuttons.ts b/editor/field_brickbuttons.ts index 94f2bc2b..f55ed62b 100644 --- a/editor/field_brickbuttons.ts +++ b/editor/field_brickbuttons.ts @@ -155,5 +155,6 @@ export class FieldBrickButtons extends Blockly.FieldDropdown implements Blockly. Blockly.DropDownDiv.content_.removeAttribute('role'); Blockly.DropDownDiv.content_.removeAttribute('aria-haspopup'); Blockly.DropDownDiv.content_.removeAttribute('aria-activedescendant'); + Blockly.DropDownDiv.getContentDiv().style.width = ''; }; } \ No newline at end of file diff --git a/editor/field_images.ts b/editor/field_images.ts deleted file mode 100644 index 2357a580..00000000 --- a/editor/field_images.ts +++ /dev/null @@ -1,114 +0,0 @@ -/// -/// -/// - -export interface FieldImagesOptions extends pxtblockly.FieldImageDropdownOptions { -} - -export class FieldImages extends pxtblockly.FieldImageDropdown implements Blockly.FieldCustom { - public isFieldCustom_ = true; - - constructor(text: string, options: FieldImagesOptions, validator?: Function) { - super(text, options, validator); - } - - /** - * Create a dropdown menu under the text. - * @private - */ - public showEditor_() { - // If there is an existing drop-down we own, this is a request to hide the drop-down. - if (Blockly.DropDownDiv.hideIfOwner(this)) { - return; - } - // 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'); - const options = this.getOptions(); - for (let i = 0, option: any; option = options[i]; i++) { - let content = (options[i] as any)[0]; // Human-readable text or image. - const value = (options[i] as any)[1]; // Language-neutral value. - // 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 = content.alt; - if ((this as any).columns_) { - button.style.width = (((this as any).width_ / (this as any).columns_) - 8) + 'px'; - //button.style.height = ((this.width_ / this.columns_) - 8) + 'px'; - } else { - button.style.width = content.width + 'px'; - button.style.height = content.height + 'px'; - } - let backgroundColor = this.sourceBlock_.getColour(); - 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.sourceBlock_.getColourTertiary(); - Blockly.bindEvent_(button, 'click', this, (this as any).buttonClick_); - Blockly.bindEvent_(button, 'mouseup', this, (this as any).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 = 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 as any).width_ + 'px'; - dropdownDiv.appendChild(contentDiv); - - Blockly.DropDownDiv.setColour(this.sourceBlock_.getColour(), this.sourceBlock_.getColourTertiary()); - - // Calculate positioning based on the field position. - var scale = this.sourceBlock_.workspace.scale; - var bBox = { width: this.size_.width, height: this.size_.height }; - bBox.width *= scale; - bBox.height *= scale; - var position = this.fieldGroup_.getBoundingClientRect(); - var primaryX = position.left + bBox.width / 2; - var primaryY = position.top + bBox.height; - var secondaryX = primaryX; - var 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 as any).onHide_.bind(this)); - } -} \ No newline at end of file diff --git a/editor/field_motors.ts b/editor/field_motors.ts new file mode 100644 index 00000000..64984eee --- /dev/null +++ b/editor/field_motors.ts @@ -0,0 +1,565 @@ +/// +/// + +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)"; + this.borderColour_ = Blockly.PXTUtils.fadeColour(this.backgroundColour_, 0.4, false); + } + + 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 = {}; + for (let opt in options) { + 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 = ''; +} \ No newline at end of file diff --git a/editor/field_ports.ts b/editor/field_ports.ts index ff6a51b9..53778ecf 100644 --- a/editor/field_ports.ts +++ b/editor/field_ports.ts @@ -1,4 +1,5 @@ /// +/// /// export interface FieldPortsOptions extends Blockly.FieldCustomDropdownOptions { @@ -6,146 +7,31 @@ export interface FieldPortsOptions extends Blockly.FieldCustomDropdownOptions { width?: string; } -export class FieldPorts extends Blockly.FieldDropdown implements Blockly.FieldCustom { +export class FieldPorts extends pxtblockly.FieldImages implements Blockly.FieldCustom { public isFieldCustom_ = true; - // Width in pixels - private width_: number; - - // Columns in grid - private columns_: number; - - private savedPrimary_: string; - constructor(text: string, options: FieldPortsOptions, validator?: Function) { - super(options.data); + super(text, options, validator); this.columns_ = parseInt(options.columns) || 4; this.width_ = parseInt(options.width) || 300; + + this.setText = Blockly.FieldDropdown.prototype.setText; + this.updateWidth = (Blockly.Field as any).prototype.updateWidth; + this.updateTextNode_ = Blockly.Field.prototype.updateTextNode_; } trimOptions_() { } - /** - * Create a dropdown menu under the text. - * @private - */ - public showEditor_() { - // If there is an existing drop-down we own, this is a request to hide the drop-down. - if (Blockly.DropDownDiv.hideIfOwner(this)) { - return; - } - // 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(); - options = options.sort(); - for (let i = 0, option: any; option = options[i]; i++) { - let content = (options[i] as any)[0]; // Human-readable text or image. - const value = (options[i] as any)[1]; // Language-neutral value. - // 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 = content.alt; - if (this.columns_) { - button.style.width = ((this.width_ / this.columns_) - 8) + 'px'; - button.style.height = ((this.width_ / this.columns_) - 8) + 'px'; - } else { - button.style.width = content.width + 'px'; - button.style.height = content.height + 'px'; - } - let backgroundColor = this.sourceBlock_.getColour(); - 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.sourceBlock_.getColourTertiary(); - 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 = 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.width_ + 'px'; - dropdownDiv.appendChild(contentDiv); - - Blockly.DropDownDiv.setColour(this.sourceBlock_.getColour(), this.sourceBlock_.getColourTertiary()); - - // Calculate positioning based on the field position. - var scale = this.sourceBlock_.workspace.scale; - var bBox = { width: this.size_.width, height: this.size_.height }; - bBox.width *= scale; - bBox.height *= scale; - var position = this.fieldGroup_.getBoundingClientRect(); - var primaryX = position.left + bBox.width / 2; - var primaryY = position.top + bBox.height; - var secondaryX = primaryX; - var 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)); + getOptions() { + const options = super.getOptions(); + return options ? options.sort() : undefined; } - /** - * Callback for when a button is clicked inside the drop-down. - * Should be bound to the FieldIconMenu. - * @param {Event} e DOM event for the click/touch - * @private - */ - private buttonClick_ = function (e: any) { + 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. - */ - private onHide_ = function () { - Blockly.DropDownDiv.content_.removeAttribute('role'); - Blockly.DropDownDiv.content_.removeAttribute('aria-haspopup'); - Blockly.DropDownDiv.content_.removeAttribute('aria-activedescendant'); - }; } \ No newline at end of file diff --git a/legoresources/SVGassets/Large Motor.svg b/legoresources/SVGassets/Large Motor.svg index ee3d43a7..351e4e41 100644 --- a/legoresources/SVGassets/Large Motor.svg +++ b/legoresources/SVGassets/Large Motor.svg @@ -13,7 +13,7 @@ sodipodi:docname="Large Motor.svg" width="153.634" height="153.634" - inkscape:version="0.92.1 r15371" + inkscape:version="0.91 r13725" inkscape:export-filename="C:\gh\pxt-ev3\libs\core\jres\icons\motorLarge-icon.png" inkscape:export-xdpi="74.983398" inkscape:export-ydpi="74.983398"> @@ -25,7 +25,7 @@ image/svg+xml - + @@ -38,15 +38,15 @@ guidetolerance="10" inkscape:pageopacity="0" inkscape:pageshadow="2" - inkscape:window-width="1600" - inkscape:window-height="837" + inkscape:window-width="2560" + inkscape:window-height="1395" id="namedview7026" showgrid="false" - inkscape:zoom="1.5361183" - inkscape:cx="39.260502" - inkscape:cy="76.817001" - inkscape:window-x="-8" - inkscape:window-y="-8" + inkscape:zoom="3.0722366" + inkscape:cx="-73.361033" + inkscape:cy="76.817002" + inkscape:window-x="0" + inkscape:window-y="1" inkscape:window-maximized="1" inkscape:current-layer="svg7024" /> + transform="matrix(-1,0,0,-1,3003.735,6879.9214)"> + transform="translate(-1.623,0)"> @@ -358,8 +358,8 @@ id="Path_36" data-name="Path 36" class="cls-1" - d="m 0.607,0 h 26.262 a 0.607,0.607 0 0 1 0.607,0.607 V 5.9 A 0.607,0.607 0 0 1 26.869,6.507 H 0.607 A 0.607,0.607 0 0 1 0,5.9 V 0.607 A 0.607,0.607 0 0 1 0.607,0 Z" - transform="translate(41.1)" + d="m 0.607,0 26.262,0 a 0.607,0.607 0 0 1 0.607,0.607 l 0,5.293 a 0.607,0.607 0 0 1 -0.607,0.607 l -26.262,0 A 0.607,0.607 0 0 1 0,5.9 L 0,0.607 A 0.607,0.607 0 0 1 0.607,0 Z" + transform="translate(41.1,0)" inkscape:connector-curvature="0" style="fill:#a8a9a8" /> @@ -380,7 +380,7 @@ id="Path_2-2" data-name="Path 2" class="cls-3" - d="m 5610.158,939.234 a 4.151,4.151 0 0 1 -2.342,0.705 5.131,5.131 0 0 1 -2.988,-0.705 c -1.257,-0.931 -1.229,0.1 -1.229,0.1 v 1.8 c 0,0.45 0.553,0.846 1.229,0.459 0.676,-0.387 1,-0.581 1,-0.581 0,0 0.361,-0.184 0.361,0.335 v 2.62 c 0,0 0.021,0.786 -0.73,0.768 -0.751,-0.018 -1.38,0 -1.38,0 0,0 -0.293,0.248 -0.293,1.385 0,1.137 0.293,1.322 0.293,1.322 h 1.38 c 0,0 0.73,-0.024 0.73,0.649 v 1.3 c 0,0 0.227,0.359 1.169,0.306 a 3.6,3.6 0 0 0 1.4,-0.306 v -1.3 a 0.67,0.67 0 0 1 0.66,-0.649 c 0.631,-0.01 1.257,0 1.257,0 0,0 0.252,-0.191 0.287,-1.322 0.035,-1.131 -0.287,-1.385 -0.287,-1.385 h -1.257 c 0,0 -0.805,-0.052 -0.817,-0.768 -0.012,-0.716 0,-2.62 0,-2.62 0,0 -0.047,-0.515 0.332,-0.335 0.379,0.18 1.215,0.581 1.215,0.581 0,0 0.83,0.407 0.814,-0.459 -0.016,-0.866 0,-1.8 0,-1.8 0,0 -0.072,-0.534 -0.804,-0.1 z" + d="m 5610.158,939.234 a 4.151,4.151 0 0 1 -2.342,0.705 5.131,5.131 0 0 1 -2.988,-0.705 c -1.257,-0.931 -1.229,0.1 -1.229,0.1 l 0,1.8 c 0,0.45 0.553,0.846 1.229,0.459 0.676,-0.387 1,-0.581 1,-0.581 0,0 0.361,-0.184 0.361,0.335 l 0,2.62 c 0,0 0.021,0.786 -0.73,0.768 -0.751,-0.018 -1.38,0 -1.38,0 0,0 -0.293,0.248 -0.293,1.385 0,1.137 0.293,1.322 0.293,1.322 l 1.38,0 c 0,0 0.73,-0.024 0.73,0.649 l 0,1.3 c 0,0 0.227,0.359 1.169,0.306 a 3.6,3.6 0 0 0 1.4,-0.306 l 0,-1.3 a 0.67,0.67 0 0 1 0.66,-0.649 c 0.631,-0.01 1.257,0 1.257,0 0,0 0.252,-0.191 0.287,-1.322 0.035,-1.131 -0.287,-1.385 -0.287,-1.385 l -1.257,0 c 0,0 -0.805,-0.052 -0.817,-0.768 -0.012,-0.716 0,-2.62 0,-2.62 0,0 -0.047,-0.515 0.332,-0.335 0.379,0.18 1.215,0.581 1.215,0.581 0,0 0.83,0.407 0.814,-0.459 -0.016,-0.866 0,-1.8 0,-1.8 0,0 -0.072,-0.534 -0.804,-0.1 z" transform="translate(-5553.893,-845.301)" inkscape:connector-curvature="0" style="fill:#ffffff" /> @@ -388,7 +388,7 @@ id="Path_3" data-name="Path 3" class="cls-3" - d="m 5601.9,1015.336 a 0.83,0.83 0 0 1 1.2,-0.594 6.516,6.516 0 0 0 2.85,0.772 5.05,5.05 0 0 0 2.768,-0.772 c 0.829,-0.4 1.067,0.594 1.067,0.594 v 1.76 a 0.71,0.71 0 0 1 -1.067,0.561 5.791,5.791 0 0 0 -2.768,-0.837 7.06,7.06 0 0 0 -2.85,0.837 c 0,0 -1.178,0.239 -1.2,-0.561 -0.022,-0.8 0,-1.76 0,-1.76 z" + d="m 5601.9,1015.336 a 0.83,0.83 0 0 1 1.2,-0.594 6.516,6.516 0 0 0 2.85,0.772 5.05,5.05 0 0 0 2.768,-0.772 c 0.829,-0.4 1.067,0.594 1.067,0.594 l 0,1.76 a 0.71,0.71 0 0 1 -1.067,0.561 5.791,5.791 0 0 0 -2.768,-0.837 7.06,7.06 0 0 0 -2.85,0.837 c 0,0 -1.178,0.239 -1.2,-0.561 -0.022,-0.8 0,-1.76 0,-1.76 z" transform="translate(-5552.485,-909.604)" inkscape:connector-curvature="0" style="fill:#ffffff" /> @@ -415,7 +415,7 @@ id="Path_2-3" data-name="Path 2" class="cls-1" - d="m 1484.037,521.285 v 17.253 c 0,0 0.019,0.845 0.3,1.121 0.281,0.276 3.516,3.626 3.516,3.626 a 1.566,1.566 0 0 0 1.157,0.4 c 0.758,-0.025 11.627,0 11.627,0 l 4.429,-4.7 v -7.963 l -10.662,-0.567 z" + d="m 1484.037,521.285 0,17.253 c 0,0 0.019,0.845 0.3,1.121 0.281,0.276 3.516,3.626 3.516,3.626 a 1.566,1.566 0 0 0 1.157,0.4 c 0.758,-0.025 11.627,0 11.627,0 l 4.429,-4.7 0,-7.963 -10.662,-0.567 z" transform="translate(676.31,5680.52)" inkscape:connector-curvature="0" style="fill:#a8a9a8" /> @@ -428,7 +428,7 @@ id="Path_44" data-name="Path 44" class="cls-1" - d="m 5.461,0 h 5.612 a 5.461,5.461 0 0 1 5.461,5.461 v 22.9 a 5.461,5.461 0 0 1 -5.461,5.461 H 5.461 A 5.461,5.461 0 0 1 0,28.365 V 5.461 A 5.461,5.461 0 0 1 5.461,0 Z" + d="m 5.461,0 5.612,0 a 5.461,5.461 0 0 1 5.461,5.461 l 0,22.9 a 5.461,5.461 0 0 1 -5.461,5.461 l -5.612,0 A 5.461,5.461 0 0 1 0,28.365 L 0,5.461 A 5.461,5.461 0 0 1 5.461,0 Z" inkscape:connector-curvature="0" style="fill:#a8a9a8" /> @@ -461,7 +461,7 @@ id="Path_6" data-name="Path 6" class="cls-3" - d="m 5601.9,1015.336 a 0.83,0.83 0 0 1 1.2,-0.594 6.516,6.516 0 0 0 2.85,0.772 5.05,5.05 0 0 0 2.768,-0.772 c 0.829,-0.4 1.067,0.594 1.067,0.594 v 1.76 a 0.71,0.71 0 0 1 -1.067,0.561 5.791,5.791 0 0 0 -2.768,-0.837 7.06,7.06 0 0 0 -2.85,0.837 c 0,0 -1.178,0.239 -1.2,-0.561 -0.022,-0.8 0,-1.76 0,-1.76 z" + d="m 5601.9,1015.336 a 0.83,0.83 0 0 1 1.2,-0.594 6.516,6.516 0 0 0 2.85,0.772 5.05,5.05 0 0 0 2.768,-0.772 c 0.829,-0.4 1.067,0.594 1.067,0.594 l 0,1.76 a 0.71,0.71 0 0 1 -1.067,0.561 5.791,5.791 0 0 0 -2.768,-0.837 7.06,7.06 0 0 0 -2.85,0.837 c 0,0 -1.178,0.239 -1.2,-0.561 -0.022,-0.8 0,-1.76 0,-1.76 z" transform="translate(-5600.979,-993.616)" inkscape:connector-curvature="0" style="fill:#ffffff" /> @@ -475,7 +475,7 @@ id="Path_1-4" data-name="Path 1" class="cls-1" - d="m 5.784,0 h 42.66 a 5.784,5.784 0 0 1 5.784,5.784 v 29.64 C 43.86,49.487 36.1,59.29 36.1,59.29 h -0.595 c 0,0 -19.462,-0.147 -23.864,-0.147 A 2.28,2.28 0 0 1 9.263,58.158 L 0,48.582 V 5.784 A 5.784,5.784 0 0 1 5.784,0 Z" + d="m 5.784,0 42.66,0 a 5.784,5.784 0 0 1 5.784,5.784 l 0,29.64 C 43.86,49.487 36.1,59.29 36.1,59.29 l -0.595,0 c 0,0 -19.462,-0.147 -23.864,-0.147 A 2.28,2.28 0 0 1 9.263,58.158 L 0,48.582 0,5.784 A 5.784,5.784 0 0 1 5.784,0 Z" transform="translate(2159.6,6153.16)" inkscape:connector-curvature="0" style="fill:#a8a9a8" /> @@ -492,7 +492,7 @@ id="hvid-2" data-name="hvid" class="cls-5" - d="m 1498.578,449.051 7.155,7.37 23.789,31.08 v 14.857 l 13.063,-16.131 v -31.221 l -0.993,-7.691 z" + d="m 1498.578,449.051 7.155,7.37 23.789,31.08 0,14.857 13.063,-16.131 0,-31.221 -0.993,-7.691 z" transform="translate(-1487.315,-449.455)" inkscape:connector-curvature="0" style="fill:#f2f2f2" /> @@ -507,7 +507,7 @@ id="Path_4-2" data-name="Path 4" class="cls-1" - d="M 5.784,0 H 36.875 A 5.784,5.784 0 0 1 42.66,5.784 V 28.032 C 34.013,39.3 27.1,48.444 27.1,48.444 H 5.784 A 5.784,5.784 0 0 1 0,42.66 V 5.784 A 5.784,5.784 0 0 1 5.784,0 Z" + d="m 5.784,0 31.091,0 a 5.784,5.784 0 0 1 5.785,5.784 l 0,22.248 C 34.013,39.3 27.1,48.444 27.1,48.444 l -21.316,0 A 5.784,5.784 0 0 1 0,42.66 L 0,5.784 A 5.784,5.784 0 0 1 5.784,0 Z" transform="translate(2165.38,6158.22)" inkscape:connector-curvature="0" style="fill:#a8a9a8" /> @@ -525,7 +525,7 @@ id="Union_1" data-name="Union 1" class="cls-7" - d="M 14.461,28.56 V 1.085 a 1.085,1.085 0 1 1 2.169,0 V 28.56 a 1.085,1.085 0 1 1 -2.169,0 z m -3.615,0 V 1.085 a 1.085,1.085 0 1 1 2.169,0 V 28.56 a 1.085,1.085 0 1 1 -2.169,0 z m -3.615,0 V 1.085 a 1.085,1.085 0 1 1 2.169,0 V 28.56 a 1.085,1.085 0 1 1 -2.169,0 z m -3.615,0 V 1.085 a 1.085,1.085 0 1 1 2.169,0 V 28.56 a 1.085,1.085 0 1 1 -2.169,0 z M 0,28.56 V 1.085 a 1.085,1.085 0 1 1 2.169,0 V 28.56 A 1.085,1.085 0 1 1 0,28.56 Z" + d="m 14.461,28.56 0,-27.475 a 1.085,1.085 0 1 1 2.169,0 l 0,27.475 a 1.085,1.085 0 1 1 -2.169,0 z m -3.615,0 0,-27.475 a 1.085,1.085 0 1 1 2.169,0 l 0,27.475 a 1.085,1.085 0 1 1 -2.169,0 z m -3.615,0 0,-27.475 a 1.085,1.085 0 1 1 2.169,0 l 0,27.475 a 1.085,1.085 0 1 1 -2.169,0 z m -3.615,0 0,-27.475 a 1.085,1.085 0 1 1 2.169,0 l 0,27.475 a 1.085,1.085 0 1 1 -2.169,0 z M 0,28.56 0,1.085 a 1.085,1.085 0 1 1 2.169,0 l 0,27.475 A 1.085,1.085 0 1 1 0,28.56 Z" transform="translate(15.907,-5.365)" inkscape:connector-curvature="0" style="fill:#6a6a6a" /> @@ -573,7 +573,7 @@ class="cls-10" cx="3.7920001" cy="3.7920001" - transform="translate(11.528)" + transform="translate(11.528,0)" r="3.7920001" style="fill:#3c3c3c" /> @@ -623,28 +623,361 @@ id="Path_7" data-name="Path 7" class="cls-10" - d="M 0.455,0.155 2.016,0 3.64,0.155 A 0.455,0.455 0 0 1 4.1,0.61 V 2.734 A 0.455,0.455 0 0 1 3.645,3.189 L 2.094,2.979 0.46,3.189 A 0.455,0.455 0 0 1 0,2.734 V 0.61 A 0.455,0.455 0 0 1 0.455,0.155 Z" - transform="rotate(-90,12.71,4.628)" + d="M 0.455,0.155 2.016,0 3.64,0.155 A 0.455,0.455 0 0 1 4.1,0.61 l 0,2.124 A 0.455,0.455 0 0 1 3.645,3.189 L 2.094,2.979 0.46,3.189 A 0.455,0.455 0 0 1 0,2.734 L 0,0.61 A 0.455,0.455 0 0 1 0.455,0.155 Z" + transform="matrix(0,-1,1,0,8.082,17.338)" inkscape:connector-curvature="0" style="fill:#3c3c3c" /> + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/legoresources/SVGassets/Large Motors.svg b/legoresources/SVGassets/Large Motors.svg new file mode 100644 index 00000000..93fd9212 --- /dev/null +++ b/legoresources/SVGassets/Large Motors.svg @@ -0,0 +1,983 @@ + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/legoresources/SVGassets/icn_port.svg b/legoresources/SVGassets/icn_port.svg index 64b2aad2..05ff8074 100644 --- a/legoresources/SVGassets/icn_port.svg +++ b/legoresources/SVGassets/icn_port.svg @@ -13,10 +13,10 @@ sodipodi:docname="icn_port.svg" width="58.928001" height="58.928001" - inkscape:export-filename="C:\gh\pxt-ev3\libs\core\jres\icons\portB.png" - inkscape:export-xdpi="156.39424" - inkscape:export-ydpi="156.39424" - inkscape:version="0.92.1 r15371"> + inkscape:export-filename="/Users/sam/pxt-ev3/libs/core/jres/icons/portCD-icon.png" + inkscape:export-xdpi="146.62" + inkscape:export-ydpi="146.62" + inkscape:version="0.91 r13725"> @@ -25,7 +25,7 @@ image/svg+xml - + @@ -38,15 +38,15 @@ guidetolerance="10" inkscape:pageopacity="0" inkscape:pageshadow="2" - inkscape:window-width="1600" - inkscape:window-height="837" + inkscape:window-width="2560" + inkscape:window-height="1395" id="namedview6350" showgrid="false" inkscape:zoom="4.0048872" - inkscape:cx="24.98" + inkscape:cx="-18.217222" inkscape:cy="29.464001" - inkscape:window-x="-8" - inkscape:window-y="-8" + inkscape:window-x="0" + inkscape:window-y="1" inkscape:window-maximized="1" inkscape:current-layer="svg6348" /> B + x="1457.2511" + y="6184.4844" + style="font-size:27.5px">CD diff --git a/libs/core/icons.jres b/libs/core/icons.jres index 03a5e0a5..e71a83f9 100644 --- a/libs/core/icons.jres +++ b/libs/core/icons.jres @@ -12,8 +12,11 @@ "motorMedium": { "icon": "" }, + "dualMotorLarge": { + "icon": "" + }, "motorLarge": { - "icon": "" + "icon": "" }, "touchSensor": { "icon": "" @@ -43,12 +46,27 @@ "icon": "" }, "portA": { - "icon": "" + "icon": "" }, "portB": { - "icon": "" + "icon": "" }, "portC": { - "icon": "" + "icon": "" + }, + "portD": { + "icon": "" + }, + "portAB": { + "icon": "" + }, + "portBC": { + "icon": "" + }, + "portCD": { + "icon": "" + }, + "portAD": { + "icon": "" } } \ No newline at end of file diff --git a/libs/core/jres/icons/dualMotorLarge-icon.png b/libs/core/jres/icons/dualMotorLarge-icon.png new file mode 100644 index 00000000..0cd76b09 Binary files /dev/null and b/libs/core/jres/icons/dualMotorLarge-icon.png differ diff --git a/libs/core/jres/icons/motorLarge-icon.png b/libs/core/jres/icons/motorLarge-icon.png index 32bf7796..3ed0fca4 100644 Binary files a/libs/core/jres/icons/motorLarge-icon.png and b/libs/core/jres/icons/motorLarge-icon.png differ diff --git a/libs/core/jres/icons/portA-icon.png b/libs/core/jres/icons/portA-icon.png index c9550307..97887a58 100644 Binary files a/libs/core/jres/icons/portA-icon.png and b/libs/core/jres/icons/portA-icon.png differ diff --git a/libs/core/jres/icons/portAB-icon.png b/libs/core/jres/icons/portAB-icon.png new file mode 100644 index 00000000..d79814e6 Binary files /dev/null and b/libs/core/jres/icons/portAB-icon.png differ diff --git a/libs/core/jres/icons/portAD-icon.png b/libs/core/jres/icons/portAD-icon.png new file mode 100644 index 00000000..59fe830b Binary files /dev/null and b/libs/core/jres/icons/portAD-icon.png differ diff --git a/libs/core/jres/icons/portB-icon.png b/libs/core/jres/icons/portB-icon.png index d5112f5d..f12e7830 100644 Binary files a/libs/core/jres/icons/portB-icon.png and b/libs/core/jres/icons/portB-icon.png differ diff --git a/libs/core/jres/icons/portBC-icon.png b/libs/core/jres/icons/portBC-icon.png new file mode 100644 index 00000000..15b1b5f4 Binary files /dev/null and b/libs/core/jres/icons/portBC-icon.png differ diff --git a/libs/core/jres/icons/portC-icon.png b/libs/core/jres/icons/portC-icon.png index 628b2410..4baa4c41 100644 Binary files a/libs/core/jres/icons/portC-icon.png and b/libs/core/jres/icons/portC-icon.png differ diff --git a/libs/core/jres/icons/portCD-icon.png b/libs/core/jres/icons/portCD-icon.png new file mode 100644 index 00000000..f5b69733 Binary files /dev/null and b/libs/core/jres/icons/portCD-icon.png differ diff --git a/libs/core/jres/icons/portD-icon.png b/libs/core/jres/icons/portD-icon.png index 3122fd14..20b26828 100644 Binary files a/libs/core/jres/icons/portD-icon.png and b/libs/core/jres/icons/portD-icon.png differ diff --git a/libs/core/output.ts b/libs/core/output.ts index 861f959d..4481f05e 100644 --- a/libs/core/output.ts +++ b/libs/core/output.ts @@ -163,6 +163,7 @@ namespace motors { * @param brake a value indicating if the motor should break when off */ //% blockId=outputMotorSetBrakeMode block="set %motor|brake %brake=toggleOnOff" + //% motor.fieldEditor="motors" //% weight=60 blockGap=8 //% group="Properties" //% help=motors/motor/set-brake @@ -175,6 +176,7 @@ namespace motors { * Inverts the motor polarity */ //% blockId=motorSetInverted block="set %motor|inverted %reversed=toggleOnOff" + //% motor.fieldEditor="motors" //% weight=59 blockGap=8 //% group="Properties" //% help=motors/motor/set-inverted @@ -192,6 +194,7 @@ namespace motors { //% group="Move" //% help=motors/motor/stop //% blockId=motorStop block="stop %motors|" + //% motors.fieldEditor="motors" stop() { this.init(); stop(this._port, this._brake); @@ -212,6 +215,7 @@ namespace motors { //% group="Move" //% help=motors/motor/reset //% blockId=motorReset block="reset %motors|" + //% motors.fieldEditor="motors" reset() { this.init(); reset(this._port); @@ -226,6 +230,7 @@ namespace motors { //% blockId=motorRun block="run %motor at %speed=motorSpeedPicker|\\%||for %value %unit" //% weight=100 blockGap=8 //% group="Move" + //% motor.fieldEditor="motors" //% expandableArgumentMode=toggle //% help=motors/motor/run run(speed: number, value: number = 0, unit: MoveUnit = MoveUnit.MilliSeconds) { @@ -287,6 +292,7 @@ namespace motors { * @param timeOut optional maximum pausing time in milliseconds */ //% blockId=motorPauseUntilRead block="pause until %motor|ready" + //% motor.fieldEditor="motors" //% weight=90 //% group="Move" pauseUntilReady(timeOut?: number) { @@ -343,6 +349,7 @@ namespace motors { * @param value true for regulated motor */ //% blockId=outputMotorSetRegulated block="set %motor|regulated %value=toggleOnOff" + //% motor.fieldEditor="motors" //% weight=58 //% group="Properties" //% help=motors/motor/set-regulated @@ -355,6 +362,7 @@ namespace motors { * @param motor the port which connects to the motor */ //% blockId=motorSpeed block="%motor|speed" + //% motor.fieldEditor="motors" //% weight=72 //% blockGap=8 //% group="Counters" @@ -369,6 +377,7 @@ namespace motors { * @param motor the port which connects to the motor */ //% blockId=motorAngle block="%motor|angle" + //% motor.fieldEditor="motors" //% weight=70 //% blockGap=8 //% group="Counters" @@ -382,6 +391,7 @@ namespace motors { * Clears the motor count */ //% blockId=motorClearCount block="clear %motor|counters" + //% motor.fieldEditor="motors" //% weight=68 //% blockGap=8 //% group="Counters" @@ -406,28 +416,28 @@ namespace motors { } } - //% whenUsed fixedInstance block="large motor A" + //% whenUsed fixedInstance block="large motor A" jres=icons.portA export const largeA = new Motor(Output.A, true); - //% whenUsed fixedInstance block="large motor B" + //% whenUsed fixedInstance block="large motor B" jres=icons.portB export const largeB = new Motor(Output.B, true); - //% whenUsed fixedInstance block="large motor C" + //% whenUsed fixedInstance block="large motor C" jres=icons.portC export const largeC = new Motor(Output.C, true); - //% whenUsed fixedInstance block="large motor D" + //% whenUsed fixedInstance block="large motor D" jres=icons.portD export const largeD = new Motor(Output.D, true); - //% whenUsed fixedInstance block="medium motor A" + //% whenUsed fixedInstance block="medium motor A" jres=icons.portA export const mediumA = new Motor(Output.A, false); - //% whenUsed fixedInstance block="medium motor B" + //% whenUsed fixedInstance block="medium motor B" jres=icons.portB export const mediumB = new Motor(Output.B, false); - //% whenUsed fixedInstance block="medium motor C" + //% whenUsed fixedInstance block="medium motor C" jres=icons.portC export const mediumC = new Motor(Output.C, false); - //% whenUsed fixedInstance block="medium motor D" + //% whenUsed fixedInstance block="medium motor D" jres=icons.portD export const mediumD = new Motor(Output.D, false); //% fixedInstances @@ -481,7 +491,8 @@ namespace motors { * @param value (optional) move duration or rotation * @param unit (optional) unit of the value */ - //% blockId=motorPairTank block="tank %motors %speedLeft=motorSpeedPicker|\\% %speedRight=motorSpeedPicker|\\%||for %value %unit" + //% blockId=motorPairTank block="tank **motors** %motors %speedLeft=motorSpeedPicker|\\% %speedRight=motorSpeedPicker|\\%||for %value %unit" + //% motors.fieldEditor="ports" //% weight=96 blockGap=8 //% inlineInputMode=inline //% group="Move" @@ -508,7 +519,8 @@ namespace motors { * @param value (optional) move duration or rotation * @param unit (optional) unit of the value */ - //% blockId=motorPairSteer block="steer %chassis turn ratio %turnRatio=motorTurnRatioPicker speed %speed=motorSpeedPicker|\\%||for %value %unit" + //% blockId=motorPairSteer block="steer **motors** %chassis turn ratio %turnRatio=motorTurnRatioPicker speed %speed=motorSpeedPicker|\\%||for %value %unit" + //% chassis.fieldEditor="ports" //% weight=95 //% turnRatio.min=-200 turnRatio=200 //% inlineInputMode=inline @@ -571,16 +583,16 @@ namespace motors { } } - //% whenUsed fixedInstance block="large motors B+C" + //% whenUsed fixedInstance block="B+C" jres=icons.portBC export const largeBC = new SynchedMotorPair(Output.BC); - //% whenUsed fixedInstance block="large motors A+D" + //% whenUsed fixedInstance block="A+D" jres=icons.portAD export const largeAD = new SynchedMotorPair(Output.AD); - //% whenUsed fixedInstance block="large motors A+B" + //% whenUsed fixedInstance block="A+B" jres=icons.portAB export const largeAB = new SynchedMotorPair(Output.AB); - //% whenUsed fixedInstance block="large motors C+D" + //% whenUsed fixedInstance block="C+D" jres=icons.portCD export const largeCD = new SynchedMotorPair(Output.CD); function reset(out: Output) {