diff --git a/.github/workflows/pxt-buildmain.yml b/.github/workflows/pxt-buildmain.yml index 60c501d2..2abb2cf1 100644 --- a/.github/workflows/pxt-buildmain.yml +++ b/.github/workflows/pxt-buildmain.yml @@ -14,7 +14,7 @@ jobs: strategy: matrix: - node-version: [8.x] + node-version: [14.x] steps: - uses: actions/checkout@v1 diff --git a/.github/workflows/pxt-buildpr.yml b/.github/workflows/pxt-buildpr.yml index 22907593..45f93266 100644 --- a/.github/workflows/pxt-buildpr.yml +++ b/.github/workflows/pxt-buildpr.yml @@ -9,7 +9,7 @@ jobs: strategy: matrix: - node-version: [8.x] + node-version: [14.x] steps: - uses: actions/checkout@v1 diff --git a/.github/workflows/pxt-buildpush.yml b/.github/workflows/pxt-buildpush.yml index cf699974..1a87ca6a 100644 --- a/.github/workflows/pxt-buildpush.yml +++ b/.github/workflows/pxt-buildpush.yml @@ -14,7 +14,7 @@ jobs: strategy: matrix: - node-version: [8.x] + node-version: [14.x] steps: - uses: actions/checkout@v1 diff --git a/editor/deploy.ts b/editor/deploy.ts index 4a2d937d..e16715c8 100644 --- a/editor/deploy.ts +++ b/editor/deploy.ts @@ -49,7 +49,7 @@ declare interface Serial extends EventTarget { requestPort(options: SerialPortRequestOptions): Promise; } -class WebSerialPackageIO implements pxt.HF2.PacketIO { +class WebSerialPackageIO implements pxt.packetio.PacketIO { onData: (v: Uint8Array) => void; onError: (e: Error) => void; onEvent: (v: Uint8Array) => void; @@ -87,7 +87,7 @@ class WebSerialPackageIO implements pxt.HF2.PacketIO { } static portIos: WebSerialPackageIO[] = []; - static async mkPacketIOAsync(): Promise { + static async mkPacketIOAsync(): Promise { const serial = (navigator).serial; if (serial) { try { @@ -130,7 +130,7 @@ class WebSerialPackageIO implements pxt.HF2.PacketIO { private async closeAsync() { // don't close port - return Promise.delay(500); + return pxt.U.delay(500); } reconnectAsync(): Promise { @@ -146,11 +146,33 @@ class WebSerialPackageIO implements pxt.HF2.PacketIO { this._writer = this.port.writable.getWriter(); return this._writer.write(pkt); } + + onDeviceConnectionChanged(connect: boolean) { + throw new Error("onDeviceConnectionChanged not implemented"); + } + + onConnectionChanged() { + throw new Error("onConnectionChanged not implemented"); + } + + isConnecting() { + throw new Error("isConnecting not implemented"); + return false; + } + + isConnected() { + throw new Error("isConnected not implemented"); + return false; + } + + disposeAsync() { + return Promise.reject("disposeAsync not implemented") + } } function hf2Async() { - const pktIOAsync: Promise = useWebSerial - ? WebSerialPackageIO.mkPacketIOAsync() : pxt.HF2.mkPacketIOAsync() + const pktIOAsync: Promise = useWebSerial + ? WebSerialPackageIO.mkPacketIOAsync() : pxt.packetio.mkPacketIOAsync() return pktIOAsync.then(h => { let w = new Ev3Wrapper(h) ev3 = w @@ -190,14 +212,19 @@ export function enableWebSerialAsync() { else return Promise.resolve(); } -function cleanupAsync() { +async function cleanupAsync() { if (ev3) { console.log('cleanup previous port') - return ev3.disconnectAsync() - .catch(e => { }) - .finally(() => { ev3 = undefined; }); + try { + await ev3.disconnectAsync() + } + catch (e) { + + } + finally { + ev3 = undefined; + } } - return Promise.resolve(); } let initPromise: Promise @@ -207,7 +234,7 @@ function initHidAsync() { // needs to run within a click handler if (useHID) { initPromise = cleanupAsync() .then(() => hf2Async()) - .catch(err => { + .catch((err: any) => { console.error(err); initPromise = null useHID = false; @@ -284,7 +311,7 @@ export function deployCoreAsync(resp: pxtc.CompileResult) { .catch(e => { // user easily forgets to stop robot bluetoothTryAgainAsync().then(() => w.disconnectAsync()) - .then(() => Promise.delay(1000)) + .then(() => pxt.U.delay(1000)) .then(() => w.reconnectAsync()); // nothing we can do @@ -296,7 +323,7 @@ export function deployCoreAsync(resp: pxtc.CompileResult) { .then(() => w.flashAsync(elfPath, UF2.readBytes(origElfUF2, 0, origElfUF2.length * 256))) .then(() => w.flashAsync(rbfPath, rbfBIN)) .then(() => w.runAsync(rbfPath)) - .then(() => Promise.delay(500)) + .then(() => pxt.U.delay(500)) .then(() => { pxt.tickEvent("webserial.success"); return w.disconnectAsync() diff --git a/editor/dialogs.tsx b/editor/dialogs.tsx index cafedf34..8f329201 100644 --- a/editor/dialogs.tsx +++ b/editor/dialogs.tsx @@ -21,7 +21,7 @@ export function bluetoothTryAgainAsync(): Promise { function enableWebSerialAndCompileAsync() { return enableWebSerialAsync() - .then(() => Promise.delay(500)) + .then(() => pxt.U.delay(500)) .then(() => projectView.compile()); } @@ -53,7 +53,7 @@ function explainWebSerialPairingAsync(): Promise { export function showUploadDialogAsync(fn: string, url: string, _confirmAsync: (options: any) => Promise): Promise { confirmAsync = _confirmAsync; // https://msdn.microsoft.com/en-us/library/cc848897.aspx - // "For security reasons, data URIs are restricted to downloaded resources. + // "For security reasons, data URIs are restricted to downloaded resources. // Data URIs cannot be used for navigation, for scripting, or to populate frame or iframe elements" const downloadAgain = !pxt.BrowserUtils.isIE() && !pxt.BrowserUtils.isEdge(); const docUrl = pxt.appTarget.appTheme.usbDocs; @@ -126,7 +126,6 @@ export function showUploadDialogAsync(fn: string, url: string, _confirmAsync: (o pxt.tickEvent("bluetooth.enable"); explainWebSerialPairingAsync() .then(() => enableWebSerialAndCompileAsync()) - .done(); } } : undefined, downloadAgain ? { label: fn, diff --git a/editor/wrap.ts b/editor/wrap.ts index 018358ff..630ddaa9 100644 --- a/editor/wrap.ts +++ b/editor/wrap.ts @@ -26,7 +26,7 @@ export class Ev3Wrapper { isStreaming = false; dataDump = /talkdbg=1/.test(window.location.href); - constructor(public io: pxt.HF2.PacketIO) { + constructor(public io: pxt.packetio.PacketIO) { io.onData = buf => { buf = buf.slice(0, HF2.read16(buf, 0) + 2) if (HF2.read16(buf, 4) == usbMagic) { @@ -81,7 +81,7 @@ export class Ev3Wrapper { log(`stopping PXT app`) let buf = this.allocCustom(2) return this.justSendAsync(buf) - .then(() => Promise.delay(500)) + .then(() => pxt.U.delay(500)) }) } @@ -236,7 +236,7 @@ export class Ev3Wrapper { let contFileReq = this.allocSystem(1 + 2, 0x97) HF2.write16(contFileReq, 7, 1000) // maxRead contFileReq[6] = handle - return Promise.delay(data.length > 0 ? 0 : 500) + return pxt.U.delay(data.length > 0 ? 0 : 500) .then(() => this.talkAsync(contFileReq, -1)) .then(resp) } @@ -251,7 +251,7 @@ export class Ev3Wrapper { let loop = (): Promise => this.lock.enqueue("file", () => this.streamFileOnceAsync(path, cb)) - .then(() => Promise.delay(500)) + .then(() => pxt.U.delay(500)) .then(loop) return loop() } diff --git a/fieldeditors/extension.ts b/fieldeditors/extension.ts index 10944a73..3f20b741 100644 --- a/fieldeditors/extension.ts +++ b/fieldeditors/extension.ts @@ -2,7 +2,6 @@ /// import { FieldPorts } from "./field_ports"; -import { FieldMotors } from "./field_motors"; import { FieldBrickButtons } from "./field_brickbuttons"; import { FieldColorEnum } from "./field_color"; import { FieldMusic } from "./field_music"; @@ -14,9 +13,6 @@ pxt.editor.initFieldExtensionsAsync = function (opts: pxt.editor.FieldExtensionO fieldEditors: [{ selector: "ports", editor: FieldPorts - }, { - selector: "motors", - editor: FieldMotors }, { selector: "brickbuttons", editor: FieldBrickButtons diff --git a/fieldeditors/field_color.ts b/fieldeditors/field_color.ts index 857f0677..8c6a2f14 100644 --- a/fieldeditors/field_color.ts +++ b/fieldeditors/field_color.ts @@ -64,7 +64,7 @@ export class FieldColorEnum extends pxtblockly.FieldColorNumber implements Block } this.value_ = colour; if (this.sourceBlock_) { - this.sourceBlock_.setColour(colour, colour, colour); + this.sourceBlock_.setColour(colour); } } } \ No newline at end of file diff --git a/fieldeditors/field_motors.ts b/fieldeditors/field_motors.ts deleted file mode 100644 index 4e5587ad..00000000 --- a/fieldeditors/field_motors.ts +++ /dev/null @@ -1,583 +0,0 @@ -/// -/// - -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_ = pxt.toolbox.fadeColor(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.dom.createSvgElement('image', { - 'height': (this as any).arrowSize_ + 'px', - 'width': (this as any).arrowSize_ + 'px' - }, null); - this.arrow_.setAttributeNS('http://www.w3.org/1999/xlink', - 'xlink:href', (Blockly.FieldDropdown as any).DROPDOWN_SVG_DATAURI); - - this.arrow2_ = Blockly.utils.dom.createSvgElement('image', { - 'height': (this as any).arrowSize_ + 'px', - 'width': (this as any).arrowSize_ + 'px' - }, null); - 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.dom.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. - let size = this.getSize(); - let fieldX = (this.sourceBlock_.RTL) ? -size.width / 2 : size.width / 2; - /** @type {!Element} */ - this.textElement_ = Blockly.utils.dom.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.dom.createSvgElement('text', - { - 'class': (this as any).className_, - 'x': fieldX, - 'dy': '0.7ex', - 'y': this.size_.height / 2 - }, - this.fieldGroup_); - - this.updateEditable(); - (this.sourceBlock_ as Blockly.BlockSvg).getSvgRoot().appendChild(this.fieldGroup_); - // Force a render. - this.render_(); - this.isDirty_ = true; - (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.dom.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.dom.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. - let text = this.text_; - this.text_ = null; - this.setText(text); - } - - getFirstValue(value: string) { - const typeValue = value.indexOf('large') != -1 ? 'large' : 'medium'; - const portValue = this.getSecondValue(value); - const isDual = portValue.length > 1; - return `${typeValue} motor${isDual ? 's' : ''}`; - } - - getSecondValue(value: string) { - return (value.indexOf('large') != -1) ? - value.substring(value.indexOf('large') + 5) : - value.substring(value.indexOf('medium') + 6); - } - - getFirstValueI11n(value: string) { - const firstValue = this.getFirstValue(value); - const motorOptions = { - 'medium motor': lf("medium motor"), - 'large motor': lf("large motor"), - 'large motors': lf("large motors") - } - return motorOptions[firstValue]; - } - - 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; - } - let 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; - } - let textNode = document.createTextNode(text); - this.textElement2_.appendChild(textNode); - - // Cached width is obsolete. Clear it. - this.isDirty_ = true; - }; - - patchDualMotorText(text: string) { - if (text === null) { - return text; - } - if (text.indexOf('|') == -1) { - text = this.sourceBlock_.RTL ? `${text}|${lf("large motors")}` : `${lf("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; - } - - let 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; - }; - - updateSize_() { - // Calculate width of field - let width = Blockly.Field.getCachedWidth(this.textElement_); - let 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_)); - - // First dropdown - // Use one of the following options, medium motor, large motor or large motors (translated) - const textNode1 = document.createTextNode(this.getFirstValueI11n(this.value_)); - this.textElement_.appendChild(textNode1); - - // Second dropdown, no need to translate. Only port numbers - if (this.textElement2_) { - const textNode2 = document.createTextNode(this.getSecondValue(this.value_)); - this.textElement2_.appendChild(textNode2); - } - this.updateSize_(); - - // 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. - let width = Blockly.Field.getCachedWidth(this.textElement_); - let 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. - let width2 = Blockly.Field.getCachedWidth(this.textElement2_); - let 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'); - const foptions = this.getOptions(); // [img info, text] - - let opts = {}; - let conts = {}; - let vals = {}; - // Go through all option values and split them into groups - for (let opt = 0; opt < foptions.length; opt++) { - const value = foptions[opt][1]; - const motorValue = value.substring(value.indexOf('.') + 1); - const typeValue = motorValue.indexOf('large') == 0 ? 'large' : 'medium'; - const portValue = motorValue.indexOf('large') == 0 ? motorValue.substring(5) : motorValue.substring(6); - const isDual = portValue.length > 1; - - const text = `${typeValue} motor${isDual ? 's' : ''}|${portValue}`; - const key = `${typeValue} motor${isDual ? 's' : ''}`; - if (!opts[key]) opts[key] = []; - opts[key].push(portValue); - - conts[text] = foptions[opt][0]; - vals[text] = value; - } - - const currentFirst = this.getFirstValue(this.value_); - //const currentSecond = this.getSecondValue(this.value_); - - let options: string[]; - 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) { - options = [lf("medium motor"), lf("large motor"), lf("large motors")]; - } 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 + '|' + (option.indexOf('motors') != -1 ? 'BC' : 'A') : 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.getFirstValueI11n(value) : this.getSecondValue(value); - 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'); - let imgSrc = content.src; - if (this.isFirst_) { - const motorValue = value.substring(value.indexOf('.') + 1); - // Find out what kind of motor this is based on it's value, possible values: mediumX, largeX, and largeXY - if (motorValue.indexOf('medium') == 0) imgSrc = isFirstUrl['medium motor']; - else if (motorValue.length == 6) imgSrc = isFirstUrl['large motor']; - else imgSrc = isFirstUrl['large motors']; - } - buttonImg.src = imgSrc; - //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_() { - const content = Blockly.DropDownDiv.getContentDiv(); - content.removeAttribute('role'); - content.removeAttribute('aria-haspopup'); - content.removeAttribute('aria-activedescendant'); - (content as HTMLElement).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/fieldeditors/field_music.ts b/fieldeditors/field_music.ts index 53ffddf6..42c3e463 100644 --- a/fieldeditors/field_music.ts +++ b/fieldeditors/field_music.ts @@ -27,7 +27,6 @@ export class FieldMusic extends pxtblockly.FieldImages implements Blockly.FieldC this.setText = Blockly.FieldDropdown.prototype.setText; this.updateSize_ = (Blockly.Field as any).prototype.updateSize_; - this.updateTextNode_ = Blockly.Field.prototype.updateTextNode_; if (!pxt.BrowserUtils.isIE() && !soundCache) { soundCache = JSON.parse(pxtTargetBundle.bundledpkgs['music']['sounds.jres']); @@ -68,7 +67,8 @@ export class FieldMusic extends pxtblockly.FieldImages implements Blockly.FieldC // Accessibility properties categoriesDiv.setAttribute('role', 'menu'); categoriesDiv.setAttribute('aria-haspopup', 'true'); - categoriesDiv.style.backgroundColor = this.sourceBlock_.getColourTertiary(); + // FIXME: tertiary color? + categoriesDiv.style.backgroundColor = this.sourceBlock_.getColour(); categoriesDiv.className = 'blocklyMusicFieldCategories'; this.refreshCategories(categoriesDiv, categories); @@ -82,7 +82,9 @@ export class FieldMusic extends pxtblockly.FieldImages implements Blockly.FieldC dropdownDiv.appendChild(categoriesDiv); dropdownDiv.appendChild(contentDiv); - Blockly.DropDownDiv.setColour(this.sourceBlock_.getColour(), this.sourceBlock_.getColourTertiary()); + Blockly.DropDownDiv.setColour(this.sourceBlock_.getColour(), + // FIXME: tertiary color? + this.sourceBlock_.getColour()); // Calculate positioning based on the field position. let scale = (this.sourceBlock_.workspace).scale; @@ -102,10 +104,9 @@ export class FieldMusic extends pxtblockly.FieldImages implements Blockly.FieldC // Update colour to look selected. if (this.sourceBlock_.isShadow()) { this.savedPrimary_ = this.sourceBlock_.getColour(); - this.sourceBlock_.setColour(this.sourceBlock_.getColourTertiary(), - this.sourceBlock_.getColourSecondary(), this.sourceBlock_.getColourTertiary()); - } else if (this.box_) { - this.box_.setAttribute('fill', this.sourceBlock_.getColourTertiary()); + // FIXME + // this.sourceBlock_.setColour(this.sourceBlock_.getColourTertiary(), + // this.sourceBlock_.getColourSecondary(), this.sourceBlock_.getColourTertiary()); } } @@ -122,7 +123,7 @@ export class FieldMusic extends pxtblockly.FieldImages implements Blockly.FieldC } refreshCategories(categoriesDiv: Element, categories: string[]) { - // Show category dropdown. + // Show category dropdown. for (let i = 0; i < categories.length; i++) { const category = categories[i]; @@ -186,11 +187,13 @@ export class FieldMusic extends pxtblockly.FieldImages implements Blockly.FieldC let backgroundColor = this.savedPrimary_ || this.sourceBlock_.getColour(); if (value == this.getValue()) { // This icon is selected, show it in a different colour - backgroundColor = this.sourceBlock_.getColourTertiary(); + // FIXME: tertiary color? + backgroundColor = this.sourceBlock_.getColour(); button.setAttribute('aria-selected', 'true'); } button.style.backgroundColor = backgroundColor; - button.style.borderColor = this.sourceBlock_.getColourTertiary(); + // FIXME: tertiary color? + button.style.borderColor = this.sourceBlock_.getColour(); Blockly.bindEvent_(button, 'click', this, this.buttonClick_); Blockly.bindEvent_(button, 'mouseup', this, this.buttonClick_); // These are applied manually instead of using the :hover pseudoclass diff --git a/fieldeditors/field_ports.ts b/fieldeditors/field_ports.ts index 31e65cc3..9af3acae 100644 --- a/fieldeditors/field_ports.ts +++ b/fieldeditors/field_ports.ts @@ -18,7 +18,6 @@ export class FieldPorts extends pxtblockly.FieldImages implements Blockly.FieldC this.setText = Blockly.FieldDropdown.prototype.setText; this.updateSize_ = (Blockly.Field as any).prototype.updateSize_; - this.updateTextNode_ = Blockly.Field.prototype.updateTextNode_; } trimOptions_() { diff --git a/libs/base/shims.d.ts b/libs/base/shims.d.ts index d54c32f0..c765da27 100644 --- a/libs/base/shims.d.ts +++ b/libs/base/shims.d.ts @@ -87,6 +87,12 @@ declare interface Buffer { */ //% shim=BufferMethods::write write(dstOffset: int32, src: Buffer): void; + + /** + * Compute k-bit FNV-1 non-cryptographic hash of the buffer. + */ + //% shim=BufferMethods::hash + hash(bits: int32): uint32; } declare namespace control { @@ -94,14 +100,14 @@ declare namespace control { * Create a new zero-initialized buffer. * @param size number of bytes in the buffer */ - //% shim=control::createBuffer + //% deprecated=1 shim=control::createBuffer function createBuffer(size: int32): Buffer; /** * Create a new buffer with UTF8-encoded string * @param str the string to put in the buffer */ - //% shim=control::createBufferFromUTF8 + //% deprecated=1 shim=control::createBufferFromUTF8 function createBufferFromUTF8(str: string): Buffer; } declare namespace loops { @@ -198,6 +204,8 @@ declare namespace control { */ //% shim=control::dmesgValue function dmesgValue(v: any): void; +} +declare namespace control { /** * Force GC and dump basic information about heap. diff --git a/libs/core/output.ts b/libs/core/output.ts index e5371f13..d9923329 100644 --- a/libs/core/output.ts +++ b/libs/core/output.ts @@ -195,7 +195,8 @@ 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" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=60 blockGap=8 //% group="Properties" //% help=motors/motor/set-brake @@ -209,7 +210,8 @@ namespace motors { * @param value true to pause; false to continue the program execution */ //% blockId=outputMotorSetPauseMode block="set %motor|pause on run %brake=toggleOnOff" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=60 blockGap=8 //% group="Properties" setPauseOnRun(value: boolean) { @@ -217,11 +219,12 @@ namespace motors { this._pauseOnRun = value; } - /** + /** * Inverts the motor polarity */ //% blockId=motorSetInverted block="set %motor|inverted %reversed=toggleOnOff" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=59 blockGap=8 //% group="Properties" //% help=motors/motor/set-inverted @@ -234,11 +237,12 @@ namespace motors { return this._inverted ? -1 : 1; } - /** + /** * Set the settle time after braking in milliseconds (default is 10ms). */ //% blockId=motorSetBrakeSettleTime block="set %motor|brake settle time %millis|ms" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=1 blockGap=8 //% group="Properties" //% millis.defl=200 millis.min=0 millis.max=500 @@ -343,7 +347,8 @@ namespace motors { //% blockId=motorRun block="run %motor at %speed=motorSpeedPicker|\\%||for %value %unit" //% weight=100 blockGap=8 //% group="Move" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% expandableArgumentMode=toggle //% help=motors/motor/run run(speed: number, value: number = 0, unit: MoveUnit = MoveUnit.MilliSeconds) { @@ -392,7 +397,8 @@ namespace motors { //% blockId=motorSchedule block="ramp %motor at %speed=motorSpeedPicker|\\%|for %value|%unit||accelerate %acceleration|decelerate %deceleration" //% weight=99 blockGap=8 //% group="Move" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% help=motors/motor/ramp //% inlineInputMode=inline //% expandableArgumentMode=toggle @@ -421,7 +427,8 @@ namespace motors { * of run commands. */ //% blockId=outputMotorsetRunRamp block="set %motor|run %ramp to $value||$unit" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=21 blockGap=8 //% group="Properties" //% help=motors/motor/set-run-phase @@ -492,7 +499,8 @@ namespace motors { * @param value true for regulated motor */ //% blockId=outputMotorSetRegulated block="set %motor|regulated %value=toggleOnOff" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=58 blockGap=8 //% group="Properties" //% help=motors/motor/set-regulated @@ -517,7 +525,8 @@ namespace motors { * @param timeOut optional maximum pausing time in milliseconds */ //% blockId=motorPauseUntilRead block="pause until %motor|ready" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=90 blockGap=8 //% group="Move" pauseUntilReady(timeOut?: number) { @@ -575,8 +584,9 @@ namespace motors { * @param motor the port which connects to the motor */ //% blockId=motorSpeed block="%motor|speed" - //% motor.fieldEditor="motors" - //% weight=72 + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 + //% weight=72 //% blockGap=8 //% group="Counters" //% help=motors/motor/speed @@ -590,7 +600,8 @@ namespace motors { * @param motor the port which connects to the motor */ //% blockId=motorAngle block="%motor|angle" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=70 //% blockGap=8 //% group="Counters" @@ -604,7 +615,8 @@ namespace motors { * Clears the motor count */ //% blockId=motorClearCount block="clear %motor|counters" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=68 //% blockGap=8 //% group="Counters" @@ -632,7 +644,8 @@ namespace motors { * Pauses the program until the motor is stalled. */ //% blockId=motorPauseUntilStall block="pause until %motor|stalled" - //% motor.fieldEditor="motors" + //% motor.fieldEditor="speed" + //% motor.fieldOptions.decompileLiterals=1 //% weight=89 //% group="Move" //% help=motors/motor/pause-until-stalled @@ -697,10 +710,10 @@ namespace motors { } /** - * The Move Tank block can make a robot drive forward, backward, turn, or stop. - * Use the Move Tank block for robot vehicles that have two Large Motors, - * with one motor driving the left side of the vehicle and the other the right side. - * You can make the two motors go at different speeds or in different directions + * The Move Tank block can make a robot drive forward, backward, turn, or stop. + * Use the Move Tank block for robot vehicles that have two Large Motors, + * with one motor driving the left side of the vehicle and the other the right side. + * You can make the two motors go at different speeds or in different directions * to make your robot turn. * @param speedLeft the speed on the left motor, eg: 50 * @param speedRight the speed on the right motor, eg: 50 diff --git a/package.json b/package.json index 241ab8f4..6f9134f8 100644 --- a/package.json +++ b/package.json @@ -32,21 +32,20 @@ "docs/*/*/*.md" ], "devDependencies": { - "typescript": "2.6.1", - "react": "16.8.3", - "semantic-ui-less": "2.2.14", - "@types/bluebird": "2.0.33", - "@types/marked": "0.3.0", - "@types/node": "8.0.53", - "webfonts-generator": "^0.4.0", - "@types/jquery": "3.2.16", - "@types/react": "16.0.25", + "@types/marked": "^0.3.0", + "@types/node": "^9.3.0", + "@types/react": "16.8.25", "@types/react-dom": "16.0.3", - "@types/web-bluetooth": "0.0.4" + "@types/web-bluetooth": "0.0.4", + "@vusion/webfonts-generator": "^0.7.1", + "react": "16.8.3", + "react-dom": "16.11.0", + "semantic-ui-less": "2.4.1", + "typescript": "^4.2.3" }, "dependencies": { - "pxt-common-packages": "6.18.2", - "pxt-core": "5.32.3" + "pxt-common-packages": "9.2.7", + "pxt-core": "7.2.16" }, "scripts": { "test": "node node_modules/pxt-core/built/pxt.js travis" diff --git a/sim/public/sim.manifest b/sim/public/sim.manifest index bfa592ca..ccc97b63 100644 --- a/sim/public/sim.manifest +++ b/sim/public/sim.manifest @@ -1,7 +1,6 @@ CACHE MANIFEST CACHE: -/cdn/bluebird.min.js /cdn/pxtsim.js /sim/common-sim.js /sim/sim.js diff --git a/sim/public/simulator.html b/sim/public/simulator.html index ae445bae..bfc33461 100644 --- a/sim/public/simulator.html +++ b/sim/public/simulator.html @@ -23,7 +23,6 @@ body { transition: none !important; } - diff --git a/sim/tsconfig.json b/sim/tsconfig.json index d043b21a..67bd8125 100644 --- a/sim/tsconfig.json +++ b/sim/tsconfig.json @@ -9,7 +9,7 @@ "newLine": "LF", "sourceMap": false, "lib": ["dom", "dom.iterable", "scripthost", "es6"], - "types": ["bluebird"], + "types": [], "typeRoots": ["../node_modules/@types"] } } diff --git a/sim/visuals/board.ts b/sim/visuals/board.ts index 45079740..fcd049bc 100644 --- a/sim/visuals/board.ts +++ b/sim/visuals/board.ts @@ -19,7 +19,7 @@ namespace pxsim.visuals { -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; - user-select: none; + user-select: none; } .sim-button { cursor: pointer; @@ -381,7 +381,7 @@ namespace pxsim.visuals { this.screenCanvas = document.createElement("canvas"); this.screenCanvas.id = "board-screen-canvas"; this.screenCanvas.style.userSelect = "none"; - this.screenCanvas.style.msUserSelect = "none"; + (this.screenCanvas.style as any).msUserSelect = "none"; this.screenCanvas.style.webkitUserSelect = "none"; (this.screenCanvas.style as any).MozUserSelect = "none"; this.screenCanvas.style.position = "absolute"; diff --git a/theme/theme.config b/theme/theme.config index 02e51279..f8930f54 100644 --- a/theme/theme.config +++ b/theme/theme.config @@ -17,6 +17,8 @@ specify theme name below */ +@placeholder: 'default'; + /* Global */ @site : 'pxt'; @reset : 'default'; @@ -87,7 +89,7 @@ Import Theme *******************************/ -@import "theme.less"; +@import (multiple) "theme.less"; @fontPath : 'fonts';