mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2025-01-11 10:39:56 -05:00
Merge pull request #2008 from ericrosenbaum/feature/vernier-block-updates
Vernier Force & Accel extension updates
This commit is contained in:
commit
ab6a60357e
1 changed files with 135 additions and 67 deletions
|
@ -12,7 +12,14 @@ const ScratchLinkDeviceAdapter = require('./scratch-link-device-adapter');
|
||||||
* @type {string}
|
* @type {string}
|
||||||
*/
|
*/
|
||||||
// eslint-disable-next-line max-len
|
// eslint-disable-next-line max-len
|
||||||
const blockIconURI = '';
|
const blockIconURI = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Icon png to be displayed in the blocks category menu, encoded as a data URI.
|
||||||
|
* @type {string}
|
||||||
|
*/
|
||||||
|
// eslint-disable-next-line max-len
|
||||||
|
const menuIconURI = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Enum for Vernier godirect protocol.
|
* Enum for Vernier godirect protocol.
|
||||||
|
@ -53,7 +60,7 @@ const GDXFOR_SENSOR = {
|
||||||
/**
|
/**
|
||||||
* The update rate, in milliseconds, for sensor data input from the peripheral.
|
* The update rate, in milliseconds, for sensor data input from the peripheral.
|
||||||
*/
|
*/
|
||||||
const GDXFOR_UPDATE_RATE = 100;
|
const GDXFOR_UPDATE_RATE = 80;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Threshold for pushing and pulling force, for the whenForcePushedOrPulled hat block.
|
* Threshold for pushing and pulling force, for the whenForcePushedOrPulled hat block.
|
||||||
|
@ -61,12 +68,6 @@ const GDXFOR_UPDATE_RATE = 100;
|
||||||
*/
|
*/
|
||||||
const FORCE_THRESHOLD = 5;
|
const FORCE_THRESHOLD = 5;
|
||||||
|
|
||||||
/**
|
|
||||||
* Threshold for acceleration magnitude, for the "moved" gesture.
|
|
||||||
* @type {number}
|
|
||||||
*/
|
|
||||||
const MOVED_THRESHOLD = 3;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Threshold for acceleration magnitude, for the "shaken" gesture.
|
* Threshold for acceleration magnitude, for the "shaken" gesture.
|
||||||
* @type {number}
|
* @type {number}
|
||||||
|
@ -91,6 +92,12 @@ const FREEFALL_THRESHOLD = 0.5;
|
||||||
*/
|
*/
|
||||||
const FREEFALL_ROTATION_FACTOR = 0.3;
|
const FREEFALL_ROTATION_FACTOR = 0.3;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Threshold in degrees for reporting that the sensor is tilted.
|
||||||
|
* @type {number}
|
||||||
|
*/
|
||||||
|
const TILT_THRESHOLD = 15;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Acceleration due to gravity, in m/s^2.
|
* Acceleration due to gravity, in m/s^2.
|
||||||
* @type {number}
|
* @type {number}
|
||||||
|
@ -409,7 +416,6 @@ const PushPullValues = {
|
||||||
* @enum {string}
|
* @enum {string}
|
||||||
*/
|
*/
|
||||||
const GestureValues = {
|
const GestureValues = {
|
||||||
MOVED: 'moved',
|
|
||||||
SHAKEN: 'shaken',
|
SHAKEN: 'shaken',
|
||||||
STARTED_FALLING: 'started falling'
|
STARTED_FALLING: 'started falling'
|
||||||
};
|
};
|
||||||
|
@ -423,7 +429,8 @@ const TiltAxisValues = {
|
||||||
FRONT: 'front',
|
FRONT: 'front',
|
||||||
BACK: 'back',
|
BACK: 'back',
|
||||||
LEFT: 'left',
|
LEFT: 'left',
|
||||||
RIGHT: 'right'
|
RIGHT: 'right',
|
||||||
|
ANY: 'any'
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -520,6 +527,20 @@ class Scratch3GdxForBlocks {
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get TILT_MENU_ANY () {
|
||||||
|
return [
|
||||||
|
...this.TILT_MENU,
|
||||||
|
{
|
||||||
|
text: formatMessage({
|
||||||
|
id: 'gdxfor.tiltDirectionMenu.any',
|
||||||
|
default: 'any',
|
||||||
|
description: 'label for any direction element in tilt direction picker for gdxfor extension'
|
||||||
|
}),
|
||||||
|
value: TiltAxisValues.ANY
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
get FACE_MENU () {
|
get FACE_MENU () {
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
|
@ -564,14 +585,6 @@ class Scratch3GdxForBlocks {
|
||||||
|
|
||||||
get GESTURE_MENU () {
|
get GESTURE_MENU () {
|
||||||
return [
|
return [
|
||||||
{
|
|
||||||
text: formatMessage({
|
|
||||||
id: 'gdxfor.moved',
|
|
||||||
default: 'moved',
|
|
||||||
description: 'the sensor was moved'
|
|
||||||
}),
|
|
||||||
value: GestureValues.MOVED
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
text: formatMessage({
|
text: formatMessage({
|
||||||
id: 'gdxfor.shaken',
|
id: 'gdxfor.shaken',
|
||||||
|
@ -614,8 +627,25 @@ class Scratch3GdxForBlocks {
|
||||||
id: Scratch3GdxForBlocks.EXTENSION_ID,
|
id: Scratch3GdxForBlocks.EXTENSION_ID,
|
||||||
name: Scratch3GdxForBlocks.EXTENSION_NAME,
|
name: Scratch3GdxForBlocks.EXTENSION_NAME,
|
||||||
blockIconURI: blockIconURI,
|
blockIconURI: blockIconURI,
|
||||||
|
menuIconURI: menuIconURI,
|
||||||
showStatusButton: true,
|
showStatusButton: true,
|
||||||
blocks: [
|
blocks: [
|
||||||
|
{
|
||||||
|
opcode: 'whenGesture',
|
||||||
|
text: formatMessage({
|
||||||
|
id: 'gdxfor.whenGesture',
|
||||||
|
default: 'when [GESTURE]',
|
||||||
|
description: 'when the sensor detects a gesture'
|
||||||
|
}),
|
||||||
|
blockType: BlockType.HAT,
|
||||||
|
arguments: {
|
||||||
|
GESTURE: {
|
||||||
|
type: ArgumentType.STRING,
|
||||||
|
menu: 'gestureOptions',
|
||||||
|
defaultValue: GestureValues.SHAKEN
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
opcode: 'whenForcePushedOrPulled',
|
opcode: 'whenForcePushedOrPulled',
|
||||||
text: formatMessage({
|
text: formatMessage({
|
||||||
|
@ -643,18 +673,34 @@ class Scratch3GdxForBlocks {
|
||||||
},
|
},
|
||||||
'---',
|
'---',
|
||||||
{
|
{
|
||||||
opcode: 'whenGesture',
|
opcode: 'whenTilted',
|
||||||
text: formatMessage({
|
text: formatMessage({
|
||||||
id: 'gdxfor.whenGesture',
|
id: 'gdxfor.whenTilted',
|
||||||
default: 'when [GESTURE]',
|
default: 'when tilted [TILT]',
|
||||||
description: 'when the sensor detects a gesture'
|
description: 'when the sensor detects tilt'
|
||||||
}),
|
}),
|
||||||
blockType: BlockType.HAT,
|
blockType: BlockType.HAT,
|
||||||
arguments: {
|
arguments: {
|
||||||
GESTURE: {
|
TILT: {
|
||||||
type: ArgumentType.STRING,
|
type: ArgumentType.STRING,
|
||||||
menu: 'gestureOptions',
|
menu: 'tiltAnyOptions',
|
||||||
defaultValue: GestureValues.MOVED
|
defaultValue: TiltAxisValues.ANY
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
opcode: 'isTilted',
|
||||||
|
text: formatMessage({
|
||||||
|
id: 'gdxfor.isTilted',
|
||||||
|
default: 'tilted [TILT]?',
|
||||||
|
description: 'is the device tilted?'
|
||||||
|
}),
|
||||||
|
blockType: BlockType.BOOLEAN,
|
||||||
|
arguments: {
|
||||||
|
TILT: {
|
||||||
|
type: ArgumentType.STRING,
|
||||||
|
menu: 'tiltAnyOptions',
|
||||||
|
defaultValue: TiltAxisValues.ANY
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -662,7 +708,7 @@ class Scratch3GdxForBlocks {
|
||||||
opcode: 'getTilt',
|
opcode: 'getTilt',
|
||||||
text: formatMessage({
|
text: formatMessage({
|
||||||
id: 'gdxfor.getTilt',
|
id: 'gdxfor.getTilt',
|
||||||
default: 'tilt [TILT]',
|
default: 'tilt angle [TILT]',
|
||||||
description: 'gets tilt'
|
description: 'gets tilt'
|
||||||
}),
|
}),
|
||||||
blockType: BlockType.REPORTER,
|
blockType: BlockType.REPORTER,
|
||||||
|
@ -674,38 +720,6 @@ class Scratch3GdxForBlocks {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
|
||||||
opcode: 'getSpinSpeed',
|
|
||||||
text: formatMessage({
|
|
||||||
id: 'gdxfor.getSpin',
|
|
||||||
default: 'spin [DIRECTION]',
|
|
||||||
description: 'gets spin speed'
|
|
||||||
}),
|
|
||||||
blockType: BlockType.REPORTER,
|
|
||||||
arguments: {
|
|
||||||
DIRECTION: {
|
|
||||||
type: ArgumentType.STRING,
|
|
||||||
menu: 'axisOptions',
|
|
||||||
defaultValue: AxisValues.Z
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
opcode: 'getAcceleration',
|
|
||||||
text: formatMessage({
|
|
||||||
id: 'gdxfor.getAcceleration',
|
|
||||||
default: 'acceleration [DIRECTION]',
|
|
||||||
description: 'gets acceleration'
|
|
||||||
}),
|
|
||||||
blockType: BlockType.REPORTER,
|
|
||||||
arguments: {
|
|
||||||
DIRECTION: {
|
|
||||||
type: ArgumentType.STRING,
|
|
||||||
menu: 'axisOptions',
|
|
||||||
defaultValue: AxisValues.X
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
'---',
|
'---',
|
||||||
{
|
{
|
||||||
opcode: 'isFacing',
|
opcode: 'isFacing',
|
||||||
|
@ -731,7 +745,38 @@ class Scratch3GdxForBlocks {
|
||||||
description: 'is the device in free fall?'
|
description: 'is the device in free fall?'
|
||||||
}),
|
}),
|
||||||
blockType: BlockType.BOOLEAN
|
blockType: BlockType.BOOLEAN
|
||||||
|
},
|
||||||
|
{
|
||||||
|
opcode: 'getSpinSpeed',
|
||||||
|
text: formatMessage({
|
||||||
|
id: 'gdxfor.getSpin',
|
||||||
|
default: 'spin speed [DIRECTION]',
|
||||||
|
description: 'gets spin speed'
|
||||||
|
}),
|
||||||
|
blockType: BlockType.REPORTER,
|
||||||
|
arguments: {
|
||||||
|
DIRECTION: {
|
||||||
|
type: ArgumentType.STRING,
|
||||||
|
menu: 'axisOptions',
|
||||||
|
defaultValue: AxisValues.Z
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
opcode: 'getAcceleration',
|
||||||
|
text: formatMessage({
|
||||||
|
id: 'gdxfor.getAcceleration',
|
||||||
|
default: 'acceleration [DIRECTION]',
|
||||||
|
description: 'gets acceleration'
|
||||||
|
}),
|
||||||
|
blockType: BlockType.REPORTER,
|
||||||
|
arguments: {
|
||||||
|
DIRECTION: {
|
||||||
|
type: ArgumentType.STRING,
|
||||||
|
menu: 'axisOptions',
|
||||||
|
defaultValue: AxisValues.X
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
menus: {
|
menus: {
|
||||||
|
@ -739,6 +784,7 @@ class Scratch3GdxForBlocks {
|
||||||
gestureOptions: this.GESTURE_MENU,
|
gestureOptions: this.GESTURE_MENU,
|
||||||
axisOptions: this.AXIS_MENU,
|
axisOptions: this.AXIS_MENU,
|
||||||
tiltOptions: this.TILT_MENU,
|
tiltOptions: this.TILT_MENU,
|
||||||
|
tiltAnyOptions: this.TILT_MENU_ANY,
|
||||||
faceOptions: this.FACE_MENU
|
faceOptions: this.FACE_MENU
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -762,8 +808,6 @@ class Scratch3GdxForBlocks {
|
||||||
|
|
||||||
whenGesture (args) {
|
whenGesture (args) {
|
||||||
switch (args.GESTURE) {
|
switch (args.GESTURE) {
|
||||||
case GestureValues.MOVED:
|
|
||||||
return this.gestureMagnitude() > MOVED_THRESHOLD;
|
|
||||||
case GestureValues.SHAKEN:
|
case GestureValues.SHAKEN:
|
||||||
return this.gestureMagnitude() > SHAKEN_THRESHOLD;
|
return this.gestureMagnitude() > SHAKEN_THRESHOLD;
|
||||||
case GestureValues.STARTED_FALLING:
|
case GestureValues.STARTED_FALLING:
|
||||||
|
@ -774,24 +818,48 @@ class Scratch3GdxForBlocks {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
whenTilted (args) {
|
||||||
|
return this._isTilted(args.TILT);
|
||||||
|
}
|
||||||
|
|
||||||
|
isTilted (args) {
|
||||||
|
return this._isTilted(args.TILT);
|
||||||
|
}
|
||||||
|
|
||||||
getTilt (args) {
|
getTilt (args) {
|
||||||
|
return this._getTiltAngle(args.TILT);
|
||||||
|
}
|
||||||
|
|
||||||
|
_isTilted (direction) {
|
||||||
|
switch (direction) {
|
||||||
|
case TiltAxisValues.ANY:
|
||||||
|
return this._getTiltAngle(TiltAxisValues.FRONT) > TILT_THRESHOLD ||
|
||||||
|
this._getTiltAngle(TiltAxisValues.BACK) > TILT_THRESHOLD ||
|
||||||
|
this._getTiltAngle(TiltAxisValues.LEFT) > TILT_THRESHOLD ||
|
||||||
|
this._getTiltAngle(TiltAxisValues.RIGHT) > TILT_THRESHOLD;
|
||||||
|
default:
|
||||||
|
return this._getTiltAngle(direction) > TILT_THRESHOLD;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_getTiltAngle (direction) {
|
||||||
// Tilt values are calculated using acceleration due to gravity,
|
// Tilt values are calculated using acceleration due to gravity,
|
||||||
// so we need to return 0 when the peripheral is not connected.
|
// so we need to return 0 when the peripheral is not connected.
|
||||||
if (!this._peripheral.isConnected()) {
|
if (!this._peripheral.isConnected()) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (args.TILT) {
|
switch (direction) {
|
||||||
case TiltAxisValues.FRONT:
|
case TiltAxisValues.FRONT:
|
||||||
return Math.round(this._peripheral.getTiltFrontBack(false));
|
|
||||||
case TiltAxisValues.BACK:
|
|
||||||
return Math.round(this._peripheral.getTiltFrontBack(true));
|
return Math.round(this._peripheral.getTiltFrontBack(true));
|
||||||
|
case TiltAxisValues.BACK:
|
||||||
|
return Math.round(this._peripheral.getTiltFrontBack(false));
|
||||||
case TiltAxisValues.LEFT:
|
case TiltAxisValues.LEFT:
|
||||||
return Math.round(this._peripheral.getTiltLeftRight(false));
|
|
||||||
case TiltAxisValues.RIGHT:
|
|
||||||
return Math.round(this._peripheral.getTiltLeftRight(true));
|
return Math.round(this._peripheral.getTiltLeftRight(true));
|
||||||
|
case TiltAxisValues.RIGHT:
|
||||||
|
return Math.round(this._peripheral.getTiltLeftRight(false));
|
||||||
default:
|
default:
|
||||||
log.warn(`Unknown direction in getTilt: ${args.TILT}`);
|
log.warn(`Unknown direction in getTilt: ${direction}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue