mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2025-01-11 10:39:56 -05:00
Comment items in Scratch3VideoSensingBlocks
This commit is contained in:
parent
c291a04cc7
commit
b46c956203
1 changed files with 125 additions and 29 deletions
|
@ -1,3 +1,5 @@
|
|||
const Runtime = require('../../engine/runtime');
|
||||
|
||||
const ArgumentType = require('../../extension-support/argument-type');
|
||||
const BlockType = require('../../extension-support/block-type');
|
||||
const Clone = require('../../util/clone');
|
||||
|
@ -32,32 +34,123 @@ class Scratch3VideoSensingBlocks {
|
|||
*/
|
||||
this.runtime = runtime;
|
||||
|
||||
/**
|
||||
* The motion detection algoritm used to power the motion amount and
|
||||
* direction values.
|
||||
* @type {VideoMotion}
|
||||
*/
|
||||
this.detect = new VideoMotion();
|
||||
|
||||
/**
|
||||
* The last millisecond epoch timestamp that the video stream was
|
||||
* analyzed.
|
||||
* @type {number}
|
||||
*/
|
||||
this._lastUpdate = null;
|
||||
|
||||
/**
|
||||
* Id representing a Scratch Renderer skin the video is rendered to for
|
||||
* previewing.
|
||||
* @type {number}
|
||||
*/
|
||||
this._skinId = -1;
|
||||
|
||||
/**
|
||||
* The Scratch Renderer Skin object.
|
||||
* @type {Skin}
|
||||
*/
|
||||
this._skin = null;
|
||||
|
||||
/**
|
||||
* Id for a drawable using the video's skin that will render as a video
|
||||
* preview.
|
||||
* @type {Drawable}
|
||||
*/
|
||||
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._setupSampleCanvas();
|
||||
this._setupPreview();
|
||||
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 () {
|
||||
return [480, 360];
|
||||
}
|
||||
|
||||
/**
|
||||
* Order preview drawable is inserted at in the renderer.
|
||||
* @type {number}
|
||||
*/
|
||||
static get ORDER () {
|
||||
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 () {
|
||||
this._video = document.createElement('video');
|
||||
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 () {
|
||||
// Create low-resolution image to sample video for analysis and preview
|
||||
const canvas = this._sampleCanvas = document.createElement('canvas');
|
||||
|
@ -88,6 +186,11 @@ class Scratch3VideoSensingBlocks {
|
|||
this._sampleContext = canvas.getContext('2d');
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a Scratch Renderer Skin and Drawable to preview the user media
|
||||
* video stream.
|
||||
* @private
|
||||
*/
|
||||
_setupPreview () {
|
||||
if (this._skinId !== -1) 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 () {
|
||||
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
|
||||
* value properties. The text is a translated string, and the value is one-indexed.
|
||||
* @param {object[]} info - An array of info objects each having a name property.
|
||||
* Create data for a menu in scratch-blocks format, consisting of an array
|
||||
* of objects with text and value properties. The text is a translated
|
||||
* 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.
|
||||
* @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.
|
||||
* @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
|
||||
*/
|
||||
_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.
|
||||
* @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.
|
||||
|
@ -231,8 +325,10 @@ class Scratch3VideoSensingBlocks {
|
|||
/**
|
||||
* An array of info about each drum.
|
||||
* @type {object[]} an array of objects.
|
||||
* @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} name - the translatable name to display in the drums
|
||||
* menu.
|
||||
* @param {string} fileName - the name of the audio file containing the
|
||||
* drum sound.
|
||||
*/
|
||||
get STAGE_SPRITE_INFO () {
|
||||
return [
|
||||
|
|
Loading…
Reference in a new issue