UNPKG

handsfree

Version:

A Face Pointer and Pose Estimator for interacting with pages, desktops, robots, and more via gestures

132 lines (113 loc) 4.21 kB
import BaseModel from './base.js' export default class HolisticModel extends BaseModel { constructor (handsfree, config) { super(handsfree, config) this.name = 'holistic' // Without this the loading event will happen before the first frame this.hasLoadedAndRun = false this.palmPoints = [0, 1, 2, 5, 9, 13, 17] } loadDependencies (callback) { // Just load utils on client if (this.handsfree.config.isClient) { this.loadDependency(`${this.handsfree.config.assetsPath}/@mediapipe/drawing_utils.js`, () => { this.onWarmUp(callback) }, !!window.drawConnectors) return } // Load holistic this.loadDependency(`${this.handsfree.config.assetsPath}/@mediapipe/holistic/holistic.js`, () => { this.api = new window.Holistic({locateFile: file => { return `${this.handsfree.config.assetsPath}/@mediapipe/holistic/${file}` }}) this.api.setOptions(this.handsfree.config.holistic) this.api.onResults(results => this.dataReceived(results)) // Load the media stream this.handsfree.getUserMedia(() => { // Warm up before using in loop if (!this.handsfree.mediapipeWarmups.isWarmingUp) { this.warmUp(callback) } else { this.handsfree.on('mediapipeWarmedUp', () => { if (!this.handsfree.mediapipeWarmups.isWarmingUp && !this.handsfree.mediapipeWarmups[this.name]) { this.warmUp(callback) } }) } }) // Load the hands camera module this.loadDependency(`${this.handsfree.config.assetsPath}/@mediapipe/drawing_utils.js`, null, !!window.drawConnectors) }) } /** * Warms up the model */ warmUp (callback) { this.handsfree.mediapipeWarmups[this.name] = true this.handsfree.mediapipeWarmups.isWarmingUp = true this.api.send({image: this.handsfree.debug.$video}).then(() => { this.handsfree.mediapipeWarmups.isWarmingUp = false this.onWarmUp(callback) }) } /** * Called after the model has been warmed up * - If we don't do this there will be too many initial hits and cause an error */ onWarmUp (callback) { this.dependenciesLoaded = true document.body.classList.add('handsfree-model-holistic') this.handsfree.emit('modelReady', this) this.handsfree.emit('holisticModelReady', this) this.handsfree.emit('mediapipeWarmedUp', this) callback && callback(this) } /** * Get data */ async getData () { this.dependenciesLoaded && await this.api.send({image: this.handsfree.debug.$video}) } // Called through this.api.onResults dataReceived (results) { this.data = results this.handsfree.data.holistic = results if (this.handsfree.isDebugging) { this.debug(results) } } /** * Debugs the holistic model */ debug (results) { this.handsfree.debug.context.holistic.clearRect(0, 0, this.handsfree.debug.$canvas.holistic.width, this.handsfree.debug.$canvas.holistic.height) drawConnectors(this.handsfree.debug.context.holistic, results.poseLandmarks, POSE_CONNECTIONS, { color: '#0f0', lineWidth: 4 }) drawLandmarks(this.handsfree.debug.context.holistic, results.poseLandmarks, { color: '#f00', lineWidth: 2 }) drawConnectors(this.handsfree.debug.context.holistic, results.faceLandmarks, FACEMESH_TESSELATION, { color: '#f0f', lineWidth: 1 }) drawConnectors(this.handsfree.debug.context.holistic, results.leftHandLandmarks, HAND_CONNECTIONS, { color: '#0f0', lineWidth: 5 }) drawLandmarks(this.handsfree.debug.context.holistic, results.leftHandLandmarks, { color: '#f0f', lineWidth: 2 }) drawConnectors(this.handsfree.debug.context.holistic, results.rightHandLandmarks, HAND_CONNECTIONS, { color: '#0f0', lineWidth: 5 }) drawLandmarks(this.handsfree.debug.context.holistic, results.rightHandLandmarks, { color: '#f0f', lineWidth: 2 }) } }