From 3f0816bac8523551c97f8d3bc8688d9506d9db14 Mon Sep 17 00:00:00 2001 From: Kevin Andersen Date: Fri, 12 Apr 2019 11:33:10 -0400 Subject: [PATCH] This commit addresses point 1 from the discussion with @ericrosenbaum around moving the setting of motor-state into the BoostMotor-class rather than having it in the opcodes. - turnOn() renamed to _turnOn() and marked as a private function, i.e. it should only be called by BoostMotor-functions, not opcodes. - New function turnOnForever() to be called by opcodes. - turnOff() now sets the motor state. - _clearRotationState now does a check for null rather than a truthy value - all motor state setting removed from Boost-class and opcodes: stopAllMotors(), motorOnFor(), motorOnForRotation(), motorOn(), motorOff() - turnOnForever(), turnOnFor() and turnOnForDegrees() now have a resetState-parameter with the default value of true. This allows the setMotorPower() and setMotorDirection()-functions to not reset state, to avoid them resolving the promises of the original motor commands that they are affecting. --- src/extensions/scratch3_boost/index.js | 53 +++++++++++++++----------- 1 file changed, 31 insertions(+), 22 deletions(-) diff --git a/src/extensions/scratch3_boost/index.js b/src/extensions/scratch3_boost/index.js index 3049d1ee4..ea2d7c7be 100644 --- a/src/extensions/scratch3_boost/index.js +++ b/src/extensions/scratch3_boost/index.js @@ -468,9 +468,10 @@ class BoostMotor { } /** - * Turn this motor on indefinitely. + * Turn this motor on indefinitely + * @private */ - turnOn () { + _turnOn () { if (this._power === 0) return; const cmd = this._parent.generateOutputCommand( this._index, @@ -483,19 +484,29 @@ class BoostMotor { ]); this._parent.send(BoostBLE.characteristic, cmd); + } - this._clearTimeout(); + /** + * Turn this motor on indefinitely + * @param {boolean} [resetState=true] - whether to reset the state of the motor when running this command. + */ + turnOnForever (resetState = true){ + if (this.power === 0) return; + if (resetState) this.status = BoostMotorState.ON_FOREVER; + this._turnOn(); } /** * Turn this motor on for a specific duration. * @param {number} milliseconds - run the motor for this long. + * @param {boolean} [resetState=true] - whether to reset the state of the motor when running this command. */ - turnOnFor (milliseconds) { + turnOnFor (milliseconds, resetState = true) { if (this.power === 0) return; milliseconds = Math.max(0, milliseconds); - this.turnOn(); + if (resetState) this.status = BoostMotorState.ON_FOR_TIME; + this._turnOn(); this._setNewTimeout(this.turnOff, milliseconds); } @@ -503,10 +514,11 @@ class BoostMotor { * Turn this motor on for a specific rotation in degrees. * @param {number} degrees - run the motor for this amount of degrees. * @param {number} direction - rotate in this direction + * @param {boolean} [resetState=true] - whether to reset the state of the motor when running this command. */ - turnOnForDegrees (degrees, direction) { + turnOnForDegrees (degrees, direction, resetState = true) { if (this.power === 0) { - this.pendingPromiseFunction(); + this._clearRotationState(); return; } @@ -524,7 +536,8 @@ class BoostMotor { BoostMotorProfile.DO_NOT_USE ] ); - + + if (resetState) this.status = BoostMotorState.ON_FOR_ROTATION; this._pendingPositionDestination = this.position + (degrees * this.direction * direction); this._parent.send(BoostBLE.characteristic, cmd); } @@ -547,6 +560,7 @@ class BoostMotor { ] ); + this.status = BoostMotorState.OFF; this._parent.send(BoostBLE.characteristic, cmd, useLimiter); } @@ -590,7 +604,7 @@ class BoostMotor { * @private */ _clearRotationState () { - if (this._pendingPromiseFunction) { + if (this._pendingPromiseFunction !== null) { this._pendingPromiseFunction(); this._pendingPromiseFunction = null; } @@ -719,7 +733,6 @@ class Boost { // Send the motor off command without using the rate limiter. // This allows the stop button to stop motors even if we are // otherwise flooded with commands. - motor.status = BoostMotorState.OFF; motor.turnOff(false); } }); @@ -1660,7 +1673,6 @@ class Scratch3BoostBlocks { this._forEachMotor(args.MOTOR_ID, motorIndex => { const motor = this._peripheral.motor(motorIndex); if (motor) { - motor.status = BoostMotorState.ON_FOR_TIME; motor.turnOnFor(durationMS); } }); @@ -1697,9 +1709,8 @@ class Scratch3BoostBlocks { const motor = this._peripheral.motor(portID); if (motor) { return new Promise(resolve => { - motor.status = BoostMotorState.ON_FOR_ROTATION; - motor.pendingPromiseFunction = resolve; motor.turnOnForDegrees(degrees, sign); + motor.pendingPromiseFunction = resolve; }); } return null; @@ -1722,8 +1733,7 @@ class Scratch3BoostBlocks { this._forEachMotor(args.MOTOR_ID, motorIndex => { const motor = this._peripheral.motor(motorIndex); if (motor) { - motor.status = BoostMotorState.ON_FOREVER; - motor.turnOn(); + motor.turnOnForever(); } }); @@ -1745,7 +1755,6 @@ class Scratch3BoostBlocks { this._forEachMotor(args.MOTOR_ID, motorIndex => { const motor = this._peripheral.motor(motorIndex); if (motor) { - motor.status = BoostMotorState.OFF; motor.turnOff(); } }); @@ -1771,13 +1780,13 @@ class Scratch3BoostBlocks { motor.power = MathUtil.clamp(Cast.toNumber(args.POWER), 0, 100); switch (motor.status) { case BoostMotorState.ON_FOREVER: - motor.turnOn(); + motor.turnOnForever(false); break; case BoostMotorState.ON_FOR_TIME: - motor.turnOnFor(motor.pendingTimeoutStartTime + motor.pendingTimeoutDelay - Date.now()); + motor.turnOnFor(motor.pendingTimeoutStartTime + motor.pendingTimeoutDelay - Date.now(), false); break; case BoostMotorState.ON_FOR_ROTATION: { - const p = Math.abs(motor.pendingPositionDestination - motor.position); + const p = Math.abs(motor.pendingPositionDestination - motor.position, false); motor.turnOnForDegrees(p, Math.sign(p)); break; } @@ -1816,14 +1825,14 @@ class Scratch3BoostBlocks { if (motor) { switch (motor.status) { case BoostMotorState.ON_FOREVER: - motor.turnOn(); + motor.turnOnForever(false); break; case BoostMotorState.ON_FOR_TIME: - motor.turnOnFor(motor.pendingTimeoutStartTime + motor.pendingTimeoutDelay - Date.now()); + motor.turnOnFor(motor.pendingTimeoutStartTime + motor.pendingTimeoutDelay - Date.now(), false); break; case BoostMotorState.ON_FOR_ROTATION: { const p = Math.abs(motor.pendingPositionDestination - motor.position); - motor.turnOnForDegrees(p, Math.sign(p)); + motor.turnOnForDegrees(p, Math.sign(p), false); break; } }