Comment items in Scratch3VideoSensingBlocks

This commit is contained in:
Michael "Z" Goddard 2018-03-30 14:50:53 -04:00
parent c291a04cc7
commit b46c956203
No known key found for this signature in database
GPG key ID: 762CD40DD5349872

View file

@ -1,3 +1,5 @@
const Runtime = require('../../engine/runtime');
const ArgumentType = require('../../extension-support/argument-type'); const ArgumentType = require('../../extension-support/argument-type');
const BlockType = require('../../extension-support/block-type'); const BlockType = require('../../extension-support/block-type');
const Clone = require('../../util/clone'); const Clone = require('../../util/clone');
@ -32,32 +34,123 @@ class Scratch3VideoSensingBlocks {
*/ */
this.runtime = runtime; this.runtime = runtime;
/**
* The motion detection algoritm used to power the motion amount and
* direction values.
* @type {VideoMotion}
*/
this.detect = new VideoMotion(); this.detect = new VideoMotion();
/**
* The last millisecond epoch timestamp that the video stream was
* analyzed.
* @type {number}
*/
this._lastUpdate = null; this._lastUpdate = null;
/**
* Id representing a Scratch Renderer skin the video is rendered to for
* previewing.
* @type {number}
*/
this._skinId = -1; this._skinId = -1;
/**
* The Scratch Renderer Skin object.
* @type {Skin}
*/
this._skin = null; this._skin = null;
/**
* Id for a drawable using the video's skin that will render as a video
* preview.
* @type {Drawable}
*/
this._drawable = -1; this._drawable = -1;
/**
* Canvas DOM element video is rendered to down or up sample to the
* expected resolution.
* @type {HTMLCanvasElement}
*/
this._sampleCanvas = null;
/**
* Canvas 2D Context to render to the _sampleCanvas member.
* @type {CanvasRenderingContext2D}
*/
this._sampleContext = null;
// Clear target motion state values when the project starts.
this.runtime.on(Runtime.PROJECT_RUN_START, this.reset.bind(this));
// Boot up the video, canvas to down/up sample the video stream, the
// preview skin and drawable, and kick off looping the analysis logic.
this._setupVideo(); this._setupVideo();
this._setupSampleCanvas(); this._setupSampleCanvas();
this._setupPreview(); this._setupPreview();
this._loop(); this._loop();
} }
static get INTERVAL () { /**
return 33; * Dimensions the video stream is analyzed at after its rendered to the
} * sample canvas.
* @type {Array.<number>}
*/
static get DIMENSIONS () { static get DIMENSIONS () {
return [480, 360]; return [480, 360];
} }
/**
* Order preview drawable is inserted at in the renderer.
* @type {number}
*/
static get ORDER () { static get ORDER () {
return 1; return 1;
} }
/**
* The key to load & store a target's motion-related state.
* @type {string}
*/
static get STATE_KEY () {
return 'Scratch.videoSensing';
}
/**
* The default motion-related state, to be used when a target has no existing motion state.
* @type {MotionState}
*/
static get DEFAULT_MOTION_STATE () {
return {
motionFrameNumber: 0,
motionAmount: 0,
motionDirection: 0
};
}
/**
* Reset the extension's data motion detection data. This will clear out
* for example old frames, so the first analyzed frame will not be compared
* against a frame from before reset was called.
*/
reset () {
this.detect.reset();
const targets = this.runtime.targets;
for (let i = 0; i < targets.length; i++) {
const state = targets[i].getCustomState(Scratch3VideoSensingBlocks.STATE_KEY);
if (state) {
state.motionAmount = 0;
state.motionDirection = 0;
}
}
}
/**
* Setup a video element connected to a user media stream.
* @private
*/
_setupVideo () { _setupVideo () {
this._video = document.createElement('video'); this._video = document.createElement('video');
navigator.getUserMedia({ navigator.getUserMedia({
@ -80,6 +173,11 @@ class Scratch3VideoSensingBlocks {
}); });
} }
/**
* Create a campus to render the user media video to down/up sample to the
* needed resolution.
* @private
*/
_setupSampleCanvas () { _setupSampleCanvas () {
// Create low-resolution image to sample video for analysis and preview // Create low-resolution image to sample video for analysis and preview
const canvas = this._sampleCanvas = document.createElement('canvas'); const canvas = this._sampleCanvas = document.createElement('canvas');
@ -88,6 +186,11 @@ class Scratch3VideoSensingBlocks {
this._sampleContext = canvas.getContext('2d'); this._sampleContext = canvas.getContext('2d');
} }
/**
* Create a Scratch Renderer Skin and Drawable to preview the user media
* video stream.
* @private
*/
_setupPreview () { _setupPreview () {
if (this._skinId !== -1) return; if (this._skinId !== -1) return;
if (this._skin !== null) return; if (this._skin !== null) return;
@ -106,6 +209,11 @@ class Scratch3VideoSensingBlocks {
}); });
} }
/**
* Occasionally step a loop to sample the video, stamp it to the preview
* skin, and add a TypedArray copy of the canvas's pixel data.
* @private
*/
_loop () { _loop () {
setTimeout(this._loop.bind(this), this.runtime.currentStepTime); setTimeout(this._loop.bind(this), this.runtime.currentStepTime);
@ -164,9 +272,11 @@ class Scratch3VideoSensingBlocks {
} }
/** /**
* Create data for a menu in scratch-blocks format, consisting of an array of objects with text and * Create data for a menu in scratch-blocks format, consisting of an array
* value properties. The text is a translated string, and the value is one-indexed. * of objects with text and value properties. The text is a translated
* @param {object[]} info - An array of info objects each having a name property. * string, and the value is one-indexed.
* @param {object[]} info - An array of info objects each having a name
* property.
* @return {array} - An array of objects with text and value properties. * @return {array} - An array of objects with text and value properties.
* @private * @private
*/ */
@ -179,27 +289,10 @@ class Scratch3VideoSensingBlocks {
}); });
} }
/**
* The key to load & store a target's motion-related state.
* @type {string}
*/
static get STATE_KEY () {
return 'Scratch.videoSensing';
}
/**
* The default music-related state, to be used when a target has no existing music state.
* @type {MusicState}
*/
static get DEFAULT_MOTION_STATE () {
return {
currentInstrument: 0
};
}
/** /**
* @param {Target} target - collect motion state for this target. * @param {Target} target - collect motion state for this target.
* @returns {MotionState} the mutable motion state associated with that target. This will be created if necessary. * @returns {MotionState} the mutable motion state associated with that
* target. This will be created if necessary.
* @private * @private
*/ */
_getMotionState (target) { _getMotionState (target) {
@ -212,7 +305,8 @@ class Scratch3VideoSensingBlocks {
} }
/** /**
* An array of info about each drum. * An array of choices of whether a reporter should return the frame's
* motion amount or direction.
* @type {object[]} an array of objects. * @type {object[]} an array of objects.
* @param {string} name - the translatable name to display in the drums menu. * @param {string} name - the translatable name to display in the drums menu.
* @param {string} fileName - the name of the audio file containing the drum sound. * @param {string} fileName - the name of the audio file containing the drum sound.
@ -231,8 +325,10 @@ class Scratch3VideoSensingBlocks {
/** /**
* An array of info about each drum. * An array of info about each drum.
* @type {object[]} an array of objects. * @type {object[]} an array of objects.
* @param {string} name - the translatable name to display in the drums menu. * @param {string} name - the translatable name to display in the drums
* @param {string} fileName - the name of the audio file containing the drum sound. * menu.
* @param {string} fileName - the name of the audio file containing the
* drum sound.
*/ */
get STAGE_SPRITE_INFO () { get STAGE_SPRITE_INFO () {
return [ return [