Merge pull request #2100 from knandersen/bugfix/2089,2088,2087

Fix 2087, 2088, and 2089 by properly setting parameters in rotation-commands and not using command feedback when relying on scratch-vm for motor timing
This commit is contained in:
Kevin Nørby Andersen 2019-04-09 14:25:44 -04:00 committed by GitHub
commit 6e976f854e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 73 additions and 44 deletions

View file

@ -33,10 +33,13 @@ const BoostBLE = {
};
/**
* Boost Motor Max Power.
* Boost Motor Max Power Add. Defines how much more power than the target speed
* the motors may supply to reach the target speed faster.
* Lower number == softer, slower reached target speed.
* Higher number == harder, faster reached target speed.
* @constant {number}
*/
const BoostMotorMaxPower = 100;
const BoostMotorMaxPowerAdd = 10;
/**
* A time interval to wait (in milliseconds) in between battery check calls.
@ -280,7 +283,7 @@ class BoostMotor {
* @type {number}
* @private
*/
this._power = 100;
this._power = 50;
/**
* This motor's current relative position
@ -362,16 +365,16 @@ class BoostMotor {
}
/**
* @param {int} value - this motor's new power level, in the range [0,100].
* @param {int} value - this motor's new power level, in the range [10,100].
*/
set power (value) {
const p = Math.max(0, Math.min(value, 100));
// The Boost motors barely move at speed 1 - solution is to step up to 2.
if (p === 1) {
this._power = 2;
} else {
this._power = p;
}
/**
* Scale the motor power to a range between 10 and 100,
* to make sure the motors will run with something built onto them.
*/
const p = MathUtil.scale(value, 0, 100, 10, 100);
this._power = p;
}
/**
@ -440,13 +443,14 @@ class BoostMotor {
if (this._power === 0) return;
const cmd = this._parent.generateOutputCommand(
this._index,
BoostOutputExecution.EXECUTE_IMMEDIATELY ^ BoostOutputExecution.COMMAND_FEEDBACK,
BoostOutputExecution.EXECUTE_IMMEDIATELY,
BoostOutputSubCommand.START_SPEED,
[
this._power * this._direction,
BoostMotorMaxPower,
this.power * this.direction,
MathUtil.clamp(this.power + BoostMotorMaxPowerAdd, 0, 100),
BoostMotorProfile.DO_NOT_USE
]);
this._status = BoostPortFeedback.BUSY_OR_FULL;
this._parent.send(BoostBLE.characteristic, cmd);
@ -458,7 +462,7 @@ class BoostMotor {
* @param {number} milliseconds - run the motor for this long.
*/
turnOnFor (milliseconds) {
if (this._power === 0) return;
if (this.power === 0) return;
milliseconds = Math.max(0, milliseconds);
this.turnOn();
@ -471,24 +475,24 @@ class BoostMotor {
* @param {number} direction - rotate in this direction
*/
turnOnForDegrees (degrees, direction) {
if (this._power === 0) return;
if (this.power === 0) return;
degrees = Math.max(0, degrees);
const cmd = this._parent.generateOutputCommand(
this._index,
BoostOutputExecution.EXECUTE_IMMEDIATELY ^ BoostOutputExecution.COMMAND_FEEDBACK,
(BoostOutputExecution.EXECUTE_IMMEDIATELY ^ BoostOutputExecution.COMMAND_FEEDBACK),
BoostOutputSubCommand.START_SPEED_FOR_DEGREES,
[
...numberToInt32Array(degrees),
this._power * this._direction * direction,
BoostMotorMaxPower,
this.power * this.direction * direction,
MathUtil.clamp(this.power + BoostMotorMaxPowerAdd, 0, 100),
BoostMotorEndState.BRAKE,
BoostMotorProfile.DO_NOT_USE
]
);
this._pendingPositionOrigin = this._position;
this._pendingPositionDestination = this._position + (degrees * this._direction * direction);
this._pendingPositionOrigin = this.position;
this._pendingPositionDestination = this.position + (degrees * this.direction * direction);
this._parent.send(BoostBLE.characteristic, cmd);
}
@ -497,7 +501,7 @@ class BoostMotor {
* @param {boolean} [useLimiter=true] - if true, use the rate limiter
*/
turnOff (useLimiter = true) {
if (this._power === 0) return;
if (this.power === 0) return;
const cmd = this._parent.generateOutputCommand(
this._index,
@ -590,7 +594,8 @@ class Boost {
this._sensors = {
tiltX: 0,
tiltY: 0,
color: BoostColor.NONE
color: BoostColor.NONE,
previousColor: BoostColor.NONE
};
/**
@ -598,7 +603,7 @@ class Boost {
* @type {Array}
* @private
*/
this._colorBucket = [];
this._colorSamples = [];
/**
* The Bluetooth connection socket for reading/writing peripheral data.
@ -734,7 +739,7 @@ class Boost {
dataPrefix: [0x97, 0x03, 0x00, 0x40],
mask: [0xFF, 0xFF, 0, 0xFF]
}
} */
} commented out until feature is enabled in scratch-link */
}],
optionalServices: []
}, this._onConnect, this.disconnect);
@ -760,7 +765,7 @@ class Boost {
tiltX: 0,
tiltY: 0,
color: BoostColor.NONE,
oldColor: BoostColor.NONE
previousColor: BoostColor.NONE
};
if (this._ble) {
@ -879,7 +884,7 @@ class Boost {
/**
* First three bytes are the common header:
* 0: Length of message
* 1: Hub ID (always 0x00 at the moment)
* 1: Hub ID (always 0x00 at the moment, unused)
* 2: Message Type
* 3: Port ID
* We base our switch-case on Message Type
@ -914,11 +919,11 @@ class Boost {
this._sensors.tiltY = data[5];
break;
case BoostIO.COLOR:
this._colorBucket.unshift(data[4]);
if (this._colorBucket.length > BoostColorSampleSize) {
this._colorBucket.pop();
if (this._colorBucket.every((v, i, arr) => v === arr[0])) {
this._sensors.color = this._colorBucket[0];
this._colorSamples.unshift(data[4]);
if (this._colorSamples.length > BoostColorSampleSize) {
this._colorSamples.pop();
if (this._colorSamples.every((v, i, arr) => v === arr[0])) {
this._sensors.color = this._colorSamples[0];
} else {
this._sensors.color = BoostColor.NONE;
}
@ -944,9 +949,11 @@ class Boost {
if (motor) {
motor._status = feedback;
// Makes sure that commands resolve both when they actually complete and when they fail
const isBusy = feedback & BoostPortFeedback.IN_PROGRESS;
const commandCompleted = feedback & (BoostPortFeedback.COMPLETED ^ BoostPortFeedback.DISCARDED);
if (commandCompleted && motor.pendingPromiseFunction) {
if (!isBusy && commandCompleted && motor.pendingPromiseFunction) {
motor.pendingPromiseFunction();
motor.pendingPromiseFunction = null;
}
}
break;
@ -954,7 +961,6 @@ class Boost {
case BoostMessage.ERROR:
log.warn(`Error reported by hub: ${data}`);
break;
default:
}
}
@ -1642,6 +1648,11 @@ class Scratch3BoostBlocks {
const promises = motors.map(portID => {
const motor = this._peripheral.motor(portID);
if (motor) {
if (motor.pendingPromiseFunction) {
// If there's already a promise for this motor it must be resolved to avoid hanging blocks.
motor.pendingPromiseFunction();
motor.pendingPromiseFunction = null;
}
return new Promise(resolve => {
motor.turnOnForDegrees(degrees, sign);
motor.pendingPromiseFunction = resolve;
@ -1715,9 +1726,11 @@ class Scratch3BoostBlocks {
if (motor.isOn) {
if (motor.pendingTimeoutDelay) {
motor.turnOnFor(motor.pendingTimeoutStartTime + motor.pendingTimeoutDelay - Date.now());
} else if (motor.pendingPositionDestination) {
const p = motor.pendingPositionDestination - motor.position;
motor.turnOnForDegrees(p, Math.sign(p) * motor.direction);
} else if (motor.pendingPromiseFunction) {
const p = Math.abs(motor.pendingPositionDestination - motor.position);
motor.turnOnForDegrees(p, Math.sign(p));
} else {
motor.turnOn();
}
}
}
@ -1754,9 +1767,11 @@ class Scratch3BoostBlocks {
if (motor.isOn) {
if (motor.pendingTimeoutDelay) {
motor.turnOnFor(motor.pendingTimeoutStartTime + motor.pendingTimeoutDelay - Date.now());
} else if (motor.pendingPositionDestination) {
const p = motor.pendingPositionDestination - motor.position;
motor.turnOnForDegrees(p, Math.sign(p) * motor.direction);
} else if (motor.pendingPromiseFunction) {
const p = Math.abs(motor.pendingPositionDestination - motor.position);
motor.turnOnForDegrees(p, Math.sign(p));
} else {
motor.turnOn();
}
}
}
@ -1787,8 +1802,8 @@ class Scratch3BoostBlocks {
log.warn('Asked for a motor position that doesnt exist!');
return false;
}
if (portID && this._peripheral._motors[portID]) {
return MathUtil.wrapClamp(this._peripheral._motors[portID].position, 0, 360);
if (portID && this.motor(portID)) {
return MathUtil.wrapClamp(this.motor(portID).position, 0, 360);
}
return 0;
}
@ -1927,10 +1942,10 @@ class Scratch3BoostBlocks {
case BoostColorLabel.ANY:
if (Object.keys(BoostColor).find(key => BoostColor[key])
.toLowerCase() !== this.getColor()) {
if (this.getColor() === this._peripheral.oldColor) {
if (this.getColor() === this._peripheral._sensors.previousColor) {
return false;
}
this._peripheral.oldColor = this.getColor();
this._peripheral._sensors.previousColor = this.getColor();
return true;
}
break;

View file

@ -77,6 +77,20 @@ class MathUtil {
const sorted = elts.slice(0).sort((a, b) => a - b);
return elts.map(e => sorted.indexOf(e));
}
/**
* Scales a number from one range to another.
* @param {number} i number to be scaled
* @param {number} iMin input range minimum
* @param {number} iMax input range maximum
* @param {number} oMin output range minimum
* @param {number} oMax output range maximum
* @return {number} scaled number
*/
static scale (i, iMin, iMax, oMin, oMax) {
const p = (i - iMin) / (iMax - iMin);
return (p * (oMax - oMin)) + oMin;
}
}
module.exports = MathUtil;