From a4339889294dc8d0eeaaae278f045c4d468222e7 Mon Sep 17 00:00:00 2001 From: Sam El-Husseini <16690124+samelhusseini@users.noreply.github.com> Date: Mon, 9 Apr 2018 15:59:28 -0700 Subject: [PATCH] New motor field editor (#470) * New motor field editor showing two dropdown, first to select the type and the second to select the port. --- editor/extension.ts | 6 +- editor/field_brickbuttons.ts | 1 + editor/field_images.ts | 114 --- editor/field_motors.ts | 565 +++++++++++ editor/field_ports.ts | 136 +-- legoresources/SVGassets/Large Motor.svg | 403 +++++++- legoresources/SVGassets/Large Motors.svg | 983 +++++++++++++++++++ legoresources/SVGassets/icn_port.svg | 34 +- libs/core/icons.jres | 26 +- libs/core/jres/icons/dualMotorLarge-icon.png | Bin 0 -> 9574 bytes libs/core/jres/icons/motorLarge-icon.png | Bin 5003 -> 5690 bytes libs/core/jres/icons/portA-icon.png | Bin 1752 -> 1821 bytes libs/core/jres/icons/portAB-icon.png | Bin 0 -> 2479 bytes libs/core/jres/icons/portAD-icon.png | Bin 0 -> 2420 bytes libs/core/jres/icons/portB-icon.png | Bin 1754 -> 1696 bytes libs/core/jres/icons/portBC-icon.png | Bin 0 -> 2517 bytes libs/core/jres/icons/portC-icon.png | Bin 1861 -> 1935 bytes libs/core/jres/icons/portCD-icon.png | Bin 0 -> 2484 bytes libs/core/jres/icons/portD-icon.png | Bin 1684 -> 1653 bytes libs/core/output.ts | 40 +- 20 files changed, 1998 insertions(+), 310 deletions(-) delete mode 100644 editor/field_images.ts create mode 100644 editor/field_motors.ts create mode 100644 legoresources/SVGassets/Large Motors.svg create mode 100644 libs/core/jres/icons/dualMotorLarge-icon.png create mode 100644 libs/core/jres/icons/portAB-icon.png create mode 100644 libs/core/jres/icons/portAD-icon.png create mode 100644 libs/core/jres/icons/portBC-icon.png create mode 100644 libs/core/jres/icons/portCD-icon.png 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 0000000000000000000000000000000000000000..0cd76b09e1f93a5582adeb91e7ab0a9593403f9d GIT binary patch literal 9574 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 literal 0 HcmV?d00001 diff --git a/libs/core/jres/icons/motorLarge-icon.png b/libs/core/jres/icons/motorLarge-icon.png index 32bf77962adf1b514b2f274d52eabcb3a1343b23..3ed0fca4e0f9f89b8e15160433495aaec3cf3558 100644 GIT binary patch literal 5690 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 V{+n(fm4g5P002ovPDHLkV1lzMu0Q|) diff --git a/libs/core/jres/icons/portA-icon.png b/libs/core/jres/icons/portA-icon.png index c9550307e628f10ed4abcf2ce17fe67fd1e7ecae..97887a58121c899e072386f4921ccab0454d07b7 100644 GIT binary patch delta 1746 zcmaKsc`(}vAI6h3T@}(M?jF3ADq#_-?ofTk5DgJE@c+@ST6s$hc4t;guG1z z)ivjh#{Ee`i&*Uqc-sf4sl zF{`H?AelXK5|WbXDwtI<%^yBCybadk22VDAUO0nZ7q4|4^_dd3o{0tihxeZ%htve2 zL`WNOTz!MMoH=1IRm@sAHhj0MeeJ$X>krA_`Sw=?h=kHhEAco6kT>owC zrEfC%?88&%x@4p#Q6gOCQ%Q{5C8D;WA%$d_U0X-Bt4z4185MZ2xv9|>2ORm69^X<{ zA?>ayYh27PIJ}WLoH!&YnH1M#$-^R1ZK_l_tAQC%HNScSf zjyt+QB+g}J#>dAGWu-lf4)?v3+v#8xtgK+9c$AWWs~BbI-vQ8ySy@?Vu{w96gOH^l z!S?4__M`x+Jexv|uhUdhGvdG!A?1AfV14Q^0Z)~)7@eM0@We|~nyh5u=da1vq=1}q zD+2&(+gw#Ci^YoZi|Sq&D9XT>X$DV+5^x<%+oGBmFHAVqt}Qrtvr_F?pwxNULjk8; zb!6c$WvKw3fTIqIrwrC%wf6YU=5I^Y7`@w_B9K38%e(vS5l~_Hx#6R#=?(MfXnmyp z^81~WRNnnVUV7_2H%%?Aqmcy`4?%MA4HN|Ll*=q5UtcU;7e&Jtw>!4~{K2-E0~3yc z%(*X=oude8x1S|jTU$-7{P5UwRv4p9V9}y(X2xR~(=1c9E=pS#%=h2j+Y@JI98z`| zy0I=Bjc3!}VPhyB9$+q)tA!yCd-{Vner;uc`SQIuusI;wmxj#ky!&;}t8barHKf@6 zNK74>|I1E2xm>)asVQmHs*0bc)Wh%f++C}p?9d&JG)1%m^^ywPHF>*WyaF))p~QS~ zeaZ30#(KYMpgGUU9)S0IOkAmH%1 zL)o~m`_H&0ZahI+exb28K)>o)j4p0(cYo>C=%NaRfR$8JEeoV)5j?@aFUK4n9%=#Y zF9$J77g9a%+_}k{rQwa_vy2U7n7=!h7hxgyKqu28xeHq-v=7_5u`a{FHWRk+1L=>?WO|9FXXqgp;bTq1o%Y=^UlzUip+oay`L-) zKh7+|Ha9o_wsr7`HX;-m+zu9nL>)P&z>!dUghgolIxHCy3LcG$;tj@nIc7svFo-wktwCsz8 z8Us%q6@mi;5jyrUDuyQmfQ(T2mmT%?kK#syBD$YsT)*V{KwC$LJ!>dc=@Uop)!{L0 z6#5$qF&lX{OzwuLI20TtIo?a|amb|Y?cFsPl5@Gn`R23f;mr~}l{;_pG_SDkx%q04 zaQ^!bqN{nTt|kEi--0gJXQFT|(*M#Y0Syzf+*0Z*p`gaQpHb9!_0DJ16CH<#hiC6a zAM<0Ks3#KY9T^N59UD^&c5jW!L3l$APOJYvBMWC;)SxubX_SiB%$1#}^jOon&8w`O ze`tG_5L8)hezRdsz2xC7?P7E%#X$edMKS11sk@ohbBaaS5ZfN{--168GVK`L6(^U} SCj}1xot%Z4jcK(B>HZ(v2}u$F delta 1676 zcmZ{kYh2QY9>)KWX5mmVQURx^smIIFO{=eBO1830 znQ36I=B1psHpPw|&4qbEBvYH0#DTZeMQW0VozMB4S3AGY@73?}yn3E5Pv<4P#7)}> z4gf}Qh~UJjliXu}149CQ|3ONrSmH$x(V<4YJB{3%<{Gt*Lp3F?`*=S%b!N)_tU5#? zspaAP_x)M{UP0@qK#h@S#RbKmW6ih){^k)6IQ}h$yICId&z?I}ET^(Bx&NG|Y!gn# z4`5E)AYm|#iN>}1F4plqQ^k5+Wtd;=?pU|$9$8#oR#0S1pechM6XPp&a*Vcx+z8av z($acdO7^H%P+rIS!e-PTC^L?;4-lUXyx0S&KtNqAG|nIUkJNt`xI;!#cnrCRz|L!* zWVF(!K3uu_{EO=VxpY<5kC7=~1>oN^4T}7bMSil2!T~Ttqhcf|xc>U((4(PG-N3&T zuiAo8f{~g7A5(HwstlRuz)t$rhYuf`j4IF`kn-~KP-;4m-#hDZzyHYX?>UsZT%6V2 z(1_dBv0TzQo6+P-fxy!J@*01ZnVe|_v9`7j)gh^9YQX~EYB@V+68dP&etDu@tP0Ek zXDR$U{r&j#%{74_W8bQ*`j0o1J$L4mLqjKAcm#8kqA!9%QlnELh1J0;iZtOww@CB} zVqoRX>S}i{l?3lPr1w_2bSnMhc3J*$b40d9l;ZP_bQsPNYhd9h>zV>GpJ3zSw zG42fHrx!QJmwGlfHpb@P`WT(GyNt>PzG-x&X-{kVf19ivV~X$U4`a*w6(2L4(KY+K z0=>BBv-5Mhw}(p-GzTeOI(fT@RW{}(x z-azXy24c0Y?(d4=T@yPdG_S!M%bA{@E`5FrhCgxYOAH26vco+jxt5!t!51eWFKYW% zNIK)q0`yt;*A}D=VdMyT8`~ad!6{=?L_2lH;%^Sd6oP_+!fLFEh2zx&0r|)7syp#U z=ntgYRp~-QQc6lnY+ii?&MY!54U+>cV=LadQn%mvw62dVuzs{}n0nHnn`|dW+SJ~N z%}cGigm%i|P#x|^&}h9wLtMrjit26koxE=(Dobrqi6mt5SLbav4(q*F`i34`NF$D| zFu@gssyzLQ^~~N0A*i0fiyb+`TS@KVDO(?ik~eAA*8?qoDela!t_6Q{FT+)bI}oL% zEk#wtL8bDrXU3NA)9&d;S&=S4Z7{sF^4fBynF8-?=F8R~S=y1T&am$md4o&)_vIRR zfBB~ms)GGnnf9+w@q47Bqrr7&c0e+{8x|?3Y;QlJ_%)A!LZf}Kde|Asnf-ly<-wRR z$d7+CwfPI_&0Xk0F}+6A2t6(DjcbaED>d#Lfi3XX6cRgQQ`57y<0XHl>W~T-6x7XV zU0`>5W4%?T(76j_<~>n~iW4ZvJTCMwpJ6hY5pMIdz1O%e_ysPBU@Xfm?5Ax*!G-o> z!OL%JU`C$`2ZDHA3Hy)MV4cnW#~OSc_JIUeIUr50e7=kR$o_#W!ox&&Ld5{QqZ&Ar zrS11#T3WlKn#S@I9d2sBF2DESrOq{-6SlmDaLlKuoPR|_0ORd6fb2;lIJvtQV}|9G z9kf@Wc(DHExt5msv=5W2#3c}ghb~@pi;IsN_heW!SDoqAg~QF>&L&)s$ooX2w9X*5dUOq~&j9k3;uqo;d$yT60wZ8e;rt@aAgkk=b z-v6iPMKxE?mm!!g#^-;YjIYGdJ%Fh#f` z3mHcz0lzOSWQmhHANMoFplcKAcHC2qR|uQ@zW}ued?d1E4~(-80O|_xBly<)L}vaQ Duh|;V diff --git a/libs/core/jres/icons/portAB-icon.png b/libs/core/jres/icons/portAB-icon.png new file mode 100644 index 0000000000000000000000000000000000000000..d79814e603e2f1efd8f232ccd6507ee19ee09ab7 GIT binary patch literal 2479 zcmZ`*c`zI577tA;q1Kk*wuE$5lv1>|(nL@@rCeLI)V@n>Z7sELmsB?))RuBXwdA^j z*!QI=L8GW84H|nbwZ3%by+7W2bHADMojKpknK|e8{pOrsl9>sTms^}0008hB=)*18 z-uah-PO$Hpml!p+0|x0DSb^9l67(RE9fSS#ZG!*+kWsyz@>06($065jp3btoY(~%G+l4Y~$dKmym!zJF+5jcr1o?73Vt|g2 zgx-0O1rScW;3^2!h{=tf6Kc&p-X4eLYW{-7@rw-tK)-1U$Th z8KPa~KyW^7MWu#yy8m2vY#~GcQ`d%nu#=OduGF=33TDO8d$QivoIHrvPMBJmGIP)o z$RGV)F#0_ckLNF>61uj?O;%A03m(;uUUBC#E8o^cQe&P!7eV+;{T;+rAz@?K`HjJt zZi+tUepoA(ot>S$uYY1lz6*tE!|iXbE48y&vh?xsfX$w})g+Qjc=$~p(n~PZK!}Bo z_;L+D$hDoX-YU_~^jNX9Z;TW>HG9g#18)maR#tx6-QE3^p+e@}E88~zU}J3^KSa^a zeP@3fks8sdsHT?q;;NGn<}0gDw!(>$zH+psIqZ4;Ivxf#wU*#870~7epO0R1a{+M+%0>1m~4d0!oGEPm6je*kiV9bov)F$0^s1_03O-luKcE? zB%V-cT}~YyhR;>a$NL^_s6=A5ZpeW+O>k;D6DE304l4gtM~U78*Y-N+8oPj%Wz8E) zivdyspdw9u6>V+)zsA$ZM~x{31yXNVbKTgnh6vS=9I;bS0Z{%Z?amxxd;7|>?CgT- z>eH1@loNpZM|&a;xejWZ3vZt-@$>P`%(SO~dt-T;Yo}hBy|vn5Ver#^uaqv2J7%PU zKhi$O`K!s6rlh3Yj}P;zvLn_$+9TmepS_cT0*K6ci!3&e*Sh_*o7e7lW>Pe@UCG^D zV@gmrmgMz?e*nc~gwZ)xFNo@huQkyMO~iY^LpJ(%Z9)C!A%c3~^w zFO4p+rYMqZTmtj@-CF++Vp^;>yDPge2XfifHABBKnTm>vQv6tiIXogloz0bim{2iO|mS}xN>Xywz$UvUvZ`;kesmaOQkq695>YQ=!O^i^uB>g&3U#YA^^s6bS2HeF^^ zkGc^2?w)MzNH}i|mlq?_Aj|*zh+Jqx@NJj}q=^wISja=PPTAevyLN#_*wW3#gGUH)w(qRgsZL^&j!Zoi_`H)=!R!qtEX2>)8SQZ{~td zbenQ3+dcz3D~qp9NAoAu2ls=eGeOH!l#Cozzm8{@!JrP9++fsv73H(Ho{moR3|h`u zL}TfZOzA|duaU7aYMkU_kiiS&1Se3k&Y@5!4sFU$F^LSW3P;Gd{UF87t-C9+?Jfof zTy4>H04x@(m?CCK#tScMmBpHrVlkF?vVi3N^ygQz{N^65%^Yvs`;NIhu9DMV8@%x8 z>J#%xUl?TKu1t3Qcb&_2M3In?kfH_x;xU~-^Th4`INjK=FIVFb?o|-ZrJn^%h_w%{ ztfji0dwmcN3kw^jr|X%mOJ0k< z1uzY_#_&DWmHM%{DHvjCXlQ0(kyV#q-ZoA&v$s$E^vObc#li6-;JIm?Vb3RPn?RW?tf5xLSVGMzSMn9^BTLfn5A;5 z%(bKdm*y^Kg*I;k(|9y?D|ZoDs2{~*c@K{aU#JvCtPDB!^%Fbnl2cL`p#yDmz^boR z54R-2hqVUon_1t^fuI6L^KLbIiX0nbfWd literal 0 HcmV?d00001 diff --git a/libs/core/jres/icons/portAD-icon.png b/libs/core/jres/icons/portAD-icon.png new file mode 100644 index 0000000000000000000000000000000000000000..59fe830baf2a77d993f61a65f071a79af28aceb0 GIT binary patch literal 2420 zcmaJ@doQH9NF9&@ zfj}}i8;kRRCH+*G1h9Y3Vj2JjN;k(5V8DukUAzVOk~AAPItZkA=cj^U@jhWda6iM6 z$Z(?iF_`2KUl5bYLZoQgioU$-i-b$_;e3n;gKR!;Qv7?94h0`FJHlNklJ<8Qol!ZAOb5Vn!DqNhg@Y z4gTz}-Sf#aQJS+41@c)Y?lMh}TwfrgFxCUpk?e3;WO`C*U&dA`^j2?&T(Z?Op&c@v zAl49^k@I;8x+8#S-~9XJ|8}yxoOMx%c=N-7+QBw%UKi1EZQgG6+x51wB<)+)sVYy( zQ_*jpt#7_7${#FezVcX10VO?wPK~dWw-!jTcm1E_$g&KYkxIvSM`}RZvQ=R%L)KFa z@3f7zIC8xt-2sT-dYHDyvJ3Bj) zw>P9v-CPc;kHLJtVGYWM&p{#JH{)N8 zCx0|YGmPh63se)i{r#p|b*`5*O&yrB_cJpmYxwi+z->%Tdo8R;K)`oWQ+;~wYY~Kc z*>;zLg6zxYmzQ-VcI~?IV=H|3p55kY7fi6q$yNeUiL6)XHOp%@ZyB)jdrU;pY}tyS zi9yrVFz5{N&0?RFt^B>KuV!@g9vreco6aFCF^Vp!oc1orJ2Y>Kg^FU+tr3 z&m@(Vl`T%Q$5Nj>Ia=}BJwqsr*oeXsTwLI5op#U?d|u&jrXNJTDloNUhnbwCt7~FS zjqZ?2l12X*ugaOIp`m_)gNWHQ51pD8zL+J*|DFS@uC9)~%&Xjsp(Y|YJqg^he6PEd z;>RcPCGv-&zaJ*2t4ShI9tDGmNXW>`D`%Ry5K~LbofHaXqJ}q{fGvEFqfH5d0X7eDp} z7#es^O?4v7f{XvpcjSsA9v%bA??buP^rOmV|Quj(#A zeSCc4d-=_4-)iBL=Jt&zrUGF$iH_=zzp|gcRJWY{v$HEYcKIs;3O?rW&`z}S2l0-2 zVs!MYte(|1lBP1pC}g1C@U2hLRY_kms=dt2i~VCa=k$uZlqFZq^lc8(ta5nlMUeN zl9WGi-!>>!R_ts1n}bOoITgjm|>XLZOvr<_G0vDhD}=h_3L0@z?4KHoyM#P zv-by+t(34KS)I<@*kNPS@Kt^w27n~LdEWTar$o9*=OhAsfkdOp1=ov`^F2L0JVJ#B zQ#(=o`+Vb_2p{gRq9;%p(s&%?cE$$S*aU$FST>3iT0G?w#q1 z*2$2T8Cs=iWoy6Ae+dQ=tK$J`zj#unx2MO+7*}W^2OaQ>GElNQvLy7dzjE%6sBZzf z2Dmx4VIKZVXoSu0w{{uLg`F|Z>YP|!j$b(@);mMJbSo>)(kiUV1{I#Vf) zm4=5OEgQ&L`j~L#hECC%Z4I0gh$-qCy|*+i6pFNs#lnqn0Lg66o&|&YqNA-IKfla; zZ(hYalV$@}n%omO{9KaLqt)sP$+AOw?={fZSCaR?fqI!|M4o`lJrH1z%0eo5cFD)O34 u&4>GcQqHEO8%M#scmI*St-SQ2 zBxNF8V)Ihwih%>VSZ7C{*urU6_mM3mXj?-B6Mu&5;I+vICl67RudYtiYVTJvu9$zhD6RV(wyvIRQ=AF$YaB}>L zuCc+kEFy(c9aP7IZD&V?P(p^k0ja5}!$t=6g_xuGM;6VVlRJLC^RF<>1C{xYu2fa+ zDGmRVE>oEH@N;rYcj6w>F*E4 zFhK~j;<7U3f!|-$oNl;&UCY@}o%e}Ko0%aC1cDFM>(^9Z$Z5nCWDMLk<_e!zKpa)m zHuUxNp%#a&hnAnE~K)xr+0j8e#P0Z8>_O0NhCIRm(0~c zFFxu7+X|vQXaSr|IT=RG%F5!78-e|8fwdj0tE>4->036*r=A}VY$?Lo1qEeTf67hvJ9pli4w-Vdi zv9@c{IWF*ic2O)~?hX6SRoP$A+1OEi{Agq7C`#27IKpoLF3AC3Py(-wnouR`2XjA|fc;%_Q@V<5Mi;+FjXgq$UtUt=et?1FP%J!QX)W30 z-M~QF)D)>%dKIKAmjps(^MAHJ@Zi*TcBcGfqtrIRzQ|;4v0mdN3X)+aFM|3^2J`gkq9gp zbN)%Uen6pwcGcr7n9jhVhr!|A? z>;|L)fjiKZz-K?QQHvJTd^7Qv&k}~la4jlLKF$^1Nv+2++U&y;>Ph=c(<@Ho^@btU4 zS(EycZ4$}b`IElBTD+4;P|&{e9W@k+f~w`b?&J+Q?8#9ceY*-OAA!_5MAH%~`T z2)aYO34}tKHMJ`>W{z32viM9XnNH#tJsq`=(~B0peQ%I?ea{Jmw)cS<3`0RT8R{hW aPo(1jW~dgOgrPK2CTTLpqunN? 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 literal 0 HcmV?d00001 diff --git a/libs/core/jres/icons/portC-icon.png b/libs/core/jres/icons/portC-icon.png index 628b24105945d32a86a1a167b4cb7f8eb041dac6..4baa4c4119cdab6ed67fcab486be1bb7abead779 100644 GIT binary patch delta 1828 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 KMNUMnLSTZIgGq$| diff --git a/libs/core/jres/icons/portCD-icon.png b/libs/core/jres/icons/portCD-icon.png new file mode 100644 index 0000000000000000000000000000000000000000..f5b69733a5f03e6e1da818c1a339d9563f27e33a GIT binary patch literal 2484 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 delta 1609 zcmaKsYd8}M7{|9oa#;tV6+@c)tYPlbT+SGp6mtnBs#$Z3B&XU;l(Dd)+?ohw?#m^S z=(sGGof1>I#BwP_HH5>?&UwzKp68tR`F;66@B8Wfzwdup<~BIpN*V$N03cvsvHNj+ z21c8_T@LGD>k*r`kk5776$0sM9SD4f3#mI@OjX(|z7K+y=}5?P2$$u@-J&adMe{oj z`>jlO1|PQO$3+Sgma7~0w%omYvKEtSaB#+t|fb`1B7ROy!^fp;m~Ox}Fmw^*43#H8-y{D#+h|Ojz!@lYlS{^+^4Y~t#4$zg z0~W@*dcaqSI(GNdOQH}61kv&YB3f!vM)Je|@-2@%JzsPv7b5PKPqv#P&jin z*7}~O+5wtI?-}EAy#9QQRUS_J^XIwhs2LnYW_wa*Wp9(hQZXEX(16H@I7K8%g*7Mx z3V3rhtkB&%Rm>?vL--xk>ghT~n7UF*Ad%=yX8Gb_3PUlkJT)>X8}Z6tB*IAJ&!CR2 zhiD}wT8E>`9c)1i4$OLNkSVprO-)PV{~12>YIKyozr|6VHktF$Am>;*yysz66_=(< zqdLK;PAF=GPAUP_G_OJ{{4m{@yA*rSJZf4-uwMMMz}FoGDr2!CS9t6mRpS}v3-U1A z&|3Moq=|{ZYcB!^`4~q`Rvx=%} zbNNPFo9(no=52~A4BOt8AQ3syE>0>Nc@WoI*IG3($ntWX)A0c^Bi@)td>OIuhG#X` zc<3eSlLG>CT0e>dD5GIrkB*gkkh?G#%+j|hmorbW&mG_SFAzdPI4Sxxs+Il4yHlFL zJpzH?Vdo-vJfX09&Gdj|cfkcO{c`-d0BfWgkZU_dN6s(LGu&|6bzl1NZ67h zVHpJjA;Oppii*f11}8Y6dBm?DrY70FPk_5JmDuw79}#%UZd2JXGc(imn`km1we<*{ z_vQ_f;|Mw*Oa5BbM1fkx{H=_3^hzv!*m73NDusg7bjkfwUDefb>EvumfCO>Z=v z@TwfV=_@kU5Q!{!M~pF78Rpfcg={etWz)LIhCg`fVIz(A*QZ?=*9H#dEFKtbaNHjwY9n7la*US5bN#lpPYQ&L7lGlhMTYPvGZ@)jzwj7QAdH+L!Po5 zPqrB*)tmGke+`PZgj!l!M$EpqAFdBozO&aEN9j0aYj3~J53yTx8^hIs+jwz9#gxiQ zgxGjRGtSZUB@zNoryGfWQN?5@X{0d{tmk)d?kM`Ao7ti~D#2GSZy)eqbZ6}w%gad} z%DdeazAK^nxZJZ9E#yblp6IO80LsPA+(8vqWn~iX58HT9)oPS-8?ml%L=aekQ-`2o z-zH{QFzicTyUEp!fsJHZ;f=CB&rdbkJmXm5)|K@OmoBaCZny_tZ?Au<>qLR~43E}t zqzJ=;qhTz$`U`VIg4#45yNazUZEA|LD;=LAg4ny{YV>9meisGtPh<-L XFMlcJuKPZ#0Ei0vE6$