Skip to content
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
100 changes: 70 additions & 30 deletions packages/scratch-vm/src/extensions/scratch3_face_sensing/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,32 @@ const menuIconURI = '
// eslint-disable-next-line max-len
const blockIconURI = '';

/**
* Face detection keypoints from TensorFlow's face sensing model.
* @readonly
* @enum {string}
*/
const PARTS = {
NOSE: '2',
MOUTH: '3',
LEFT_EYE: '0',
RIGHT_EYE: '1',
BETWEEN_EYES: '6',
LEFT_EAR: '4',
RIGHT_EAR: '5',
TOP_OF_HEAD: '7'
};

/**
* Possible tilt directions.
* @readonly
* @enum {string}
*/
const TILT = {
LEFT: 'left',
RIGHT: 'right'
};

/**
* Class for the Face sensing blocks in Scratch 3.0
* @param {Runtime} runtime - the runtime instantiating this block package.
Expand Down Expand Up @@ -116,6 +142,22 @@ class Scratch3FaceSensingBlocks {
return 5;
}

/**
* Minimum tilt (in degrees) needed before counting as a tilt event.
* @type {number}
*/
static get TILT_THRESHOLD () {
return 10;
}

/**
* Default face part position when no facial keypoints are detected.
* @type {{x: number, y: number}}
*/
static get DEFAULT_PART_POSITION () {
return {x: 0, y: 0};
}

/**
* An array of info about the face part menu choices.
* @type {object[]}
Expand All @@ -128,63 +170,63 @@ class Scratch3FaceSensingBlocks {
description: 'Option for the "go to [PART]" and "when sprite touches [PART] blocks'

}),
value: '2'
value: PARTS.NOSE
}, {
text: formatMessage({
id: 'faceSensing.mouth',
default: 'mouth',
description: 'Option for the "go to [PART]" and "when sprite touches [PART] blocks'

}),
value: '3'
value: PARTS.MOUTH
}, {
text: formatMessage({
id: 'faceSensing.leftEye',
default: 'left eye',
description: 'Option for the "go to [PART]" and "when sprite touches [PART] blocks'

}),
value: '0'
value: PARTS.LEFT_EYE
}, {
text: formatMessage({
id: 'faceSensing.rightEye',
default: 'right eye',
description: 'Option for the "go to [PART]" and "when sprite touches [PART] blocks'

}),
value: '1'
value: PARTS.RIGHT_EYE
}, {
text: formatMessage({
id: 'faceSensing.betweenEyes',
default: 'between eyes',
description: 'Option for the "go to [PART]" and "when sprite touches [PART] blocks'

}),
value: '6'
value: PARTS.BETWEEN_EYES
}, {
text: formatMessage({
id: 'faceSensing.leftEar',
default: 'left ear',
description: 'Option for the "go to [PART]" and "when sprite touches [PART] blocks'

}),
value: '4'
value: PARTS.LEFT_EAR
}, {
text: formatMessage({
id: 'faceSensing.rightEar',
default: 'right ear',
description: 'Option for the "go to [PART]" and "when sprite touches [PART] blocks'

}),
value: '5'
value: PARTS.RIGHT_EAR
}, {
text: formatMessage({
id: 'faceSensing.topOfHead',
default: 'top of head',
description: 'Option for the "go to [PART]" and "when sprite touches [PART] blocks'

}),
value: '7'
value: PARTS.TOP_OF_HEAD
}];
}

Expand All @@ -200,15 +242,15 @@ class Scratch3FaceSensingBlocks {
description: 'Argument for the "when face tilts [DIRECTION]" block'

}),
value: 'left'
value: TILT.LEFT
}, {
text: formatMessage({
id: 'faceSensing.right',
default: 'right',
description: 'Argument for the "when face tilts [DIRECTION]" block'

}),
value: 'right'
value: TILT.RIGHT
}];
}

Expand Down Expand Up @@ -304,7 +346,7 @@ class Scratch3FaceSensingBlocks {
PART: {
type: ArgumentType.STRING,
menu: 'PART',
defaultValue: '2'
defaultValue: PARTS.NOSE
}
},
filter: [TargetType.SPRITE]
Expand Down Expand Up @@ -342,7 +384,7 @@ class Scratch3FaceSensingBlocks {
DIRECTION: {
type: ArgumentType.STRING,
menu: 'TILT',
defaultValue: 'left'
defaultValue: TILT.LEFT
}
}
},
Expand All @@ -357,7 +399,7 @@ class Scratch3FaceSensingBlocks {
PART: {
type: ArgumentType.STRING,
menu: 'PART',
defaultValue: '2'
defaultValue: PARTS.NOSE
}
},
blockType: BlockType.HAT,
Expand Down Expand Up @@ -415,8 +457,8 @@ class Scratch3FaceSensingBlocks {
* @private
*/
_getBetweenEyesPosition () {
const leftEye = this._getPartPosition(0);
const rightEye = this._getPartPosition(1);
const leftEye = this._getPartPosition(PARTS.LEFT_EYE);
const rightEye = this._getPartPosition(PARTS.RIGHT_EYE);
const betweenEyes = {x: 0, y: 0};
betweenEyes.x = leftEye.x + ((rightEye.x - leftEye.x) / 2);
betweenEyes.y = leftEye.y + ((rightEye.y - leftEye.y) / 2);
Expand All @@ -433,9 +475,9 @@ class Scratch3FaceSensingBlocks {
* @private
*/
_getTopOfHeadPosition () {
const leftEyePos = this._getPartPosition(0);
const rightEyePos = this._getPartPosition(1);
const mouthPos = this._getPartPosition(3);
const leftEyePos = this._getPartPosition(PARTS.LEFT_EYE);
const rightEyePos = this._getPartPosition(PARTS.RIGHT_EYE);
const mouthPos = this._getPartPosition(PARTS.MOUTH);
const dx = rightEyePos.x - leftEyePos.x;
const dy = rightEyePos.y - leftEyePos.y;
const directionRads = Math.atan2(dy, dx) + (Math.PI / 2);
Expand All @@ -453,20 +495,20 @@ class Scratch3FaceSensingBlocks {
* Get the position of a given facial keypoint.
* Returns {0,0} if no face or keypoints are available.
*
* @param {number} part - Part of the face to be detected
* @param {string} part - Part of the face to be detected
* @returns {{x: number, y: number}} Coordinates of the detected keypoint.
* @private
*/
_getPartPosition (part) {
const defaultPos = {x: 0, y: 0};
const defaultPos = Scratch3FaceSensingBlocks.DEFAULT_PART_POSITION;

if (!this._currentFace) return defaultPos;
if (!this._currentFace.keypoints) return defaultPos;

if (Number(part) === 6) {
if (part === PARTS.BETWEEN_EYES) {
return this._getBetweenEyesPosition();
}
if (Number(part) === 7) {
if (part === PARTS.TOP_OF_HEAD) {
return this._getTopOfHeadPosition();
}

Expand Down Expand Up @@ -547,8 +589,8 @@ class Scratch3FaceSensingBlocks {
faceTilt () {
if (!this._currentFace) return this._cachedTilt;

const leftEyePos = this._getPartPosition(0);
const rightEyePos = this._getPartPosition(1);
const leftEyePos = this._getPartPosition(PARTS.LEFT_EYE);
const rightEyePos = this._getPartPosition(PARTS.RIGHT_EYE);
const dx = rightEyePos.x - leftEyePos.x;
const dy = rightEyePos.y - leftEyePos.y;
const direction = 90 - MathUtil.radToDeg(Math.atan2(dy, dx));
Expand All @@ -567,13 +609,11 @@ class Scratch3FaceSensingBlocks {
* @returns {boolean} - true if the face is tilted
*/
whenTilted (args) {
const TILT_THRESHOLD = 10;

if (args.DIRECTION === 'left') {
return this.faceTilt() < (90 - TILT_THRESHOLD);
if (args.DIRECTION === TILT.LEFT) {
return this.faceTilt() < (90 - Scratch3FaceSensingBlocks.TILT_THRESHOLD);
}
if (args.DIRECTION === 'right') {
return this.faceTilt() > (90 + TILT_THRESHOLD);
if (args.DIRECTION === TILT.RIGHT) {
return this.faceTilt() > (90 + Scratch3FaceSensingBlocks.TILT_THRESHOLD);
}
return false;
}
Expand Down
Loading