handsfree
Version:
Quickly integrate face, hand, and/or pose tracking to your frontend projects in a snap ✨👌
1,553 lines (1,340 loc) • 337 kB
JavaScript
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
typeof define === 'function' && define.amd ? define(factory) :
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.Handsfree = factory());
}(this, (function () { 'use strict';
class BaseModel {
constructor(handsfree, config) {
this.handsfree = handsfree;
this.config = config;
this.data = {}; // Whether we've loaded dependencies or not
this.dependenciesLoaded = false; // Whether the model is enabled or not
this.enabled = config.enabled; // Collection of plugins and gestures
this.plugins = [];
this.gestures = [];
this.gestureEstimator = null;
setTimeout(() => {
// Get data
const getData = this.getData;
this.getData = async () => {
let data = (await getData.apply(this, arguments)) || {};
data.gesture = this.getGesture();
this.runPlugins();
return data;
}; // Get gesture
let getGesture = this.getGesture;
this.getGesture = () => {
if (!getGesture) {
getGesture = function () {};
}
return getGesture.apply(this, arguments);
};
}, 0);
} // Implement in the model class
loadDependencies() {}
updateData() {}
updateGestureEstimator() {}
/**
* Enable model
* @param {*} handleLoad If true then it'll also attempt to load,
* otherwise you'll need to handle it yourself. This is mostly used internally
* to prevent the .update() method from double loading
*/
enable(handleLoad = true) {
this.handsfree.config[this.name] = this.config;
this.handsfree.config[this.name].enabled = this.enabled = true;
document.body.classList.add(`handsfree-model-${this.name}`);
if (handleLoad && !this.dependenciesLoaded) {
this.loadDependencies();
} // Weboji uses a webgl context
if (this.name === 'weboji') {
this.handsfree.debug.$canvas.weboji.style.display = 'block';
}
}
disable() {
this.handsfree.config[this.name] = this.config;
this.handsfree.config[this.name].enabled = this.enabled = false;
document.body.classList.remove(`handsfree-model-${this.name}`);
setTimeout(() => {
// Weboji uses a webgl context so let's just hide it
if (this.name === 'weboji') {
this.handsfree.debug.$canvas.weboji.style.display = 'none';
} else {
var _this$handsfree$debug;
((_this$handsfree$debug = this.handsfree.debug.context[this.name]) === null || _this$handsfree$debug === void 0 ? void 0 : _this$handsfree$debug.clearRect) && this.handsfree.debug.context[this.name].clearRect(0, 0, this.handsfree.debug.$canvas[this.name].width, this.handsfree.debug.$canvas[this.name].height);
} // Stop if all models have been stopped
let hasRunningModels = Object.keys(this.handsfree.model).some(model => this.handsfree.model[model].enabled);
if (!hasRunningModels) {
this.handsfree.stop();
}
}, 0);
}
/**
* Loads a script and runs a callback
* @param {string} src The absolute path of the source file
* @param {*} callback The callback to call after the file is loaded
* @param {boolean} skip Whether to skip loading the dependency and just call the callback
*/
loadDependency(src, callback, skip = false) {
// Skip and run callback
if (skip) {
callback && callback();
return;
} // Inject script into DOM
const $script = document.createElement('script');
$script.async = true;
$script.onload = () => {
callback && callback();
};
$script.onerror = () => {
this.handsfree.emit('modelError', `Error loading ${src}`);
};
$script.src = src;
document.body.appendChild($script);
}
/**
* Run all the plugins attached to this model
*/
runPlugins() {
// Exit if no data
if (!this.data || this.name === 'handpose' && !this.data.annotations) {
return;
}
if (Object.keys(this.data).length) {
this.plugins.forEach(name => {
var _this$handsfree$plugi;
this.handsfree.plugin[name].enabled && ((_this$handsfree$plugi = this.handsfree.plugin[name]) === null || _this$handsfree$plugi === void 0 ? void 0 : _this$handsfree$plugi.onFrame(this.handsfree.data));
});
}
}
}
var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};
function getDefaultExportFromCjs (x) {
return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
}
function createCommonjsModule(fn, basedir, module) {
return module = {
path: basedir,
exports: {},
require: function (path, base) {
return commonjsRequire(path, (base === undefined || base === null) ? module.path : base);
}
}, fn(module, module.exports), module.exports;
}
function commonjsRequire () {
throw new Error('Dynamic requires are not currently supported by @rollup/plugin-commonjs');
}
var fingerpose = createCommonjsModule(function (module, exports) {
!function (t, e) {
module.exports = e() ;
}("undefined" != typeof self ? self : commonjsGlobal, function () {
return function (t) {
var e = {};
function n(r) {
if (e[r]) return e[r].exports;
var i = e[r] = {
i: r,
l: !1,
exports: {}
};
return t[r].call(i.exports, i, i.exports, n), i.l = !0, i.exports;
}
return n.m = t, n.c = e, n.d = function (t, e, r) {
n.o(t, e) || Object.defineProperty(t, e, {
enumerable: !0,
get: r
});
}, n.r = function (t) {
"undefined" != typeof Symbol && Symbol.toStringTag && Object.defineProperty(t, Symbol.toStringTag, {
value: "Module"
}), Object.defineProperty(t, "__esModule", {
value: !0
});
}, n.t = function (t, e) {
if (1 & e && (t = n(t)), 8 & e) return t;
if (4 & e && "object" == typeof t && t && t.__esModule) return t;
var r = Object.create(null);
if (n.r(r), Object.defineProperty(r, "default", {
enumerable: !0,
value: t
}), 2 & e && "string" != typeof t) for (var i in t) n.d(r, i, function (e) {
return t[e];
}.bind(null, i));
return r;
}, n.n = function (t) {
var e = t && t.__esModule ? function () {
return t.default;
} : function () {
return t;
};
return n.d(e, "a", e), e;
}, n.o = function (t, e) {
return Object.prototype.hasOwnProperty.call(t, e);
}, n.p = "", n(n.s = 0);
}([function (t, e, n) {
n.r(e);
var r = {};
function i(t) {
return (i = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (t) {
return typeof t;
} : function (t) {
return t && "function" == typeof Symbol && t.constructor === Symbol && t !== Symbol.prototype ? "symbol" : typeof t;
})(t);
}
n.r(r), n.d(r, "VictoryGesture", function () {
return C;
}), n.d(r, "ThumbsUpGesture", function () {
return j;
});
var o = {
Thumb: 0,
Index: 1,
Middle: 2,
Ring: 3,
Pinky: 4,
all: [0, 1, 2, 3, 4],
nameMapping: {
0: "Thumb",
1: "Index",
2: "Middle",
3: "Ring",
4: "Pinky"
},
pointsMapping: {
0: [[0, 1], [1, 2], [2, 3], [3, 4]],
1: [[0, 5], [5, 6], [6, 7], [7, 8]],
2: [[0, 9], [9, 10], [10, 11], [11, 12]],
3: [[0, 13], [13, 14], [14, 15], [15, 16]],
4: [[0, 17], [17, 18], [18, 19], [19, 20]]
},
getName: function (t) {
return void 0 !== i(this.nameMapping[t]) && this.nameMapping[t];
},
getPoints: function (t) {
return void 0 !== i(this.pointsMapping[t]) && this.pointsMapping[t];
}
},
a = {
NoCurl: 0,
HalfCurl: 1,
FullCurl: 2,
nameMapping: {
0: "No Curl",
1: "Half Curl",
2: "Full Curl"
},
getName: function (t) {
return void 0 !== i(this.nameMapping[t]) && this.nameMapping[t];
}
},
l = {
VerticalUp: 0,
VerticalDown: 1,
HorizontalLeft: 2,
HorizontalRight: 3,
DiagonalUpRight: 4,
DiagonalUpLeft: 5,
DiagonalDownRight: 6,
DiagonalDownLeft: 7,
nameMapping: {
0: "Vertical Up",
1: "Vertical Down",
2: "Horizontal Left",
3: "Horizontal Right",
4: "Diagonal Up Right",
5: "Diagonal Up Left",
6: "Diagonal Down Right",
7: "Diagonal Down Left"
},
getName: function (t) {
return void 0 !== i(this.nameMapping[t]) && this.nameMapping[t];
}
};
function u(t) {
if ("undefined" == typeof Symbol || null == t[Symbol.iterator]) {
if (Array.isArray(t) || (t = function (t, e) {
if (!t) return;
if ("string" == typeof t) return c(t, e);
var n = Object.prototype.toString.call(t).slice(8, -1);
"Object" === n && t.constructor && (n = t.constructor.name);
if ("Map" === n || "Set" === n) return Array.from(n);
if ("Arguments" === n || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return c(t, e);
}(t))) {
var e = 0,
n = function () {};
return {
s: n,
n: function () {
return e >= t.length ? {
done: !0
} : {
done: !1,
value: t[e++]
};
},
e: function (t) {
throw t;
},
f: n
};
}
throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
}
var r,
i,
o = !0,
a = !1;
return {
s: function () {
r = t[Symbol.iterator]();
},
n: function () {
var t = r.next();
return o = t.done, t;
},
e: function (t) {
a = !0, i = t;
},
f: function () {
try {
o || null == r.return || r.return();
} finally {
if (a) throw i;
}
}
};
}
function c(t, e) {
(null == e || e > t.length) && (e = t.length);
for (var n = 0, r = new Array(e); n < e; n++) r[n] = t[n];
return r;
}
function f(t, e) {
var n = Object.keys(t);
if (Object.getOwnPropertySymbols) {
var r = Object.getOwnPropertySymbols(t);
e && (r = r.filter(function (e) {
return Object.getOwnPropertyDescriptor(t, e).enumerable;
})), n.push.apply(n, r);
}
return n;
}
function s(t, e, n) {
return e in t ? Object.defineProperty(t, e, {
value: n,
enumerable: !0,
configurable: !0,
writable: !0
}) : t[e] = n, t;
}
function h(t, e) {
for (var n = 0; n < e.length; n++) {
var r = e[n];
r.enumerable = r.enumerable || !1, r.configurable = !0, "value" in r && (r.writable = !0), Object.defineProperty(t, r.key, r);
}
}
var d = function () {
function t(e) {
!function (t, e) {
if (!(t instanceof e)) throw new TypeError("Cannot call a class as a function");
}(this, t), this.options = function (t) {
for (var e = 1; e < arguments.length; e++) {
var n = null != arguments[e] ? arguments[e] : {};
e % 2 ? f(Object(n), !0).forEach(function (e) {
s(t, e, n[e]);
}) : Object.getOwnPropertyDescriptors ? Object.defineProperties(t, Object.getOwnPropertyDescriptors(n)) : f(Object(n)).forEach(function (e) {
Object.defineProperty(t, e, Object.getOwnPropertyDescriptor(n, e));
});
}
return t;
}({}, {
HALF_CURL_START_LIMIT: 60,
NO_CURL_START_LIMIT: 130,
DISTANCE_VOTE_POWER: 1.1,
SINGLE_ANGLE_VOTE_POWER: .9,
TOTAL_ANGLE_VOTE_POWER: 1.6
}, {}, e);
}
var e, n;
return e = t, (n = [{
key: "estimate",
value: function (t) {
var e,
n = [],
r = [],
i = u(o.all);
try {
for (i.s(); !(e = i.n()).done;) {
var a,
l = e.value,
c = o.getPoints(l),
f = [],
s = [],
h = u(c);
try {
for (h.s(); !(a = h.n()).done;) {
var d = a.value,
p = t[d[0]],
y = t[d[1]],
g = this.getSlopes(p, y),
v = g[0],
m = g[1];
f.push(v), s.push(m);
}
} catch (t) {
h.e(t);
} finally {
h.f();
}
n.push(f), r.push(s);
}
} catch (t) {
i.e(t);
} finally {
i.f();
}
var b,
D = [],
w = [],
O = u(o.all);
try {
for (O.s(); !(b = O.n()).done;) {
var M = b.value,
S = M == o.Thumb ? 1 : 0,
T = o.getPoints(M),
C = t[T[S][0]],
R = t[T[S + 1][1]],
A = t[T[3][1]],
L = this.estimateFingerCurl(C, R, A),
_ = this.calculateFingerDirection(C, R, A, n[M].slice(S));
D[M] = L, w[M] = _;
}
} catch (t) {
O.e(t);
} finally {
O.f();
}
return {
curls: D,
directions: w
};
}
}, {
key: "getSlopes",
value: function (t, e) {
var n = this.calculateSlope(t[0], t[1], e[0], e[1]);
return 2 == t.length ? n : [n, this.calculateSlope(t[1], t[2], e[1], e[2])];
}
}, {
key: "angleOrientationAt",
value: function (t) {
var e = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : 1,
n = 0,
r = 0,
i = 0;
return t >= 75 && t <= 105 ? n = 1 * e : t >= 25 && t <= 155 ? r = 1 * e : i = 1 * e, [n, r, i];
}
}, {
key: "estimateFingerCurl",
value: function (t, e, n) {
var r = t[0] - e[0],
i = t[0] - n[0],
o = e[0] - n[0],
l = t[1] - e[1],
u = t[1] - n[1],
c = e[1] - n[1],
f = t[2] - e[2],
s = t[2] - n[2],
h = e[2] - n[2],
d = Math.sqrt(r * r + l * l + f * f),
p = Math.sqrt(i * i + u * u + s * s),
y = Math.sqrt(o * o + c * c + h * h),
g = (y * y + d * d - p * p) / (2 * y * d);
g > 1 ? g = 1 : g < -1 && (g = -1);
var v = Math.acos(g);
return (v = 57.2958 * v % 180) > this.options.NO_CURL_START_LIMIT ? a.NoCurl : v > this.options.HALF_CURL_START_LIMIT ? a.HalfCurl : a.FullCurl;
}
}, {
key: "estimateHorizontalDirection",
value: function (t, e, n, r) {
return r == Math.abs(t) ? t > 0 ? l.HorizontalLeft : l.HorizontalRight : r == Math.abs(e) ? e > 0 ? l.HorizontalLeft : l.HorizontalRight : n > 0 ? l.HorizontalLeft : l.HorizontalRight;
}
}, {
key: "estimateVerticalDirection",
value: function (t, e, n, r) {
return r == Math.abs(t) ? t < 0 ? l.VerticalDown : l.VerticalUp : r == Math.abs(e) ? e < 0 ? l.VerticalDown : l.VerticalUp : n < 0 ? l.VerticalDown : l.VerticalUp;
}
}, {
key: "estimateDiagonalDirection",
value: function (t, e, n, r, i, o, a, u) {
var c = this.estimateVerticalDirection(t, e, n, r),
f = this.estimateHorizontalDirection(i, o, a, u);
return c == l.VerticalUp ? f == l.HorizontalLeft ? l.DiagonalUpLeft : l.DiagonalUpRight : f == l.HorizontalLeft ? l.DiagonalDownLeft : l.DiagonalDownRight;
}
}, {
key: "calculateFingerDirection",
value: function (t, e, n, r) {
var i = t[0] - e[0],
o = t[0] - n[0],
a = e[0] - n[0],
l = t[1] - e[1],
c = t[1] - n[1],
f = e[1] - n[1],
s = Math.max(Math.abs(i), Math.abs(o), Math.abs(a)),
h = Math.max(Math.abs(l), Math.abs(c), Math.abs(f)),
d = 0,
p = 0,
y = 0,
g = h / (s + 1e-5);
g > 1.5 ? d += this.options.DISTANCE_VOTE_POWER : g > .66 ? p += this.options.DISTANCE_VOTE_POWER : y += this.options.DISTANCE_VOTE_POWER;
var v = Math.sqrt(i * i + l * l),
m = Math.sqrt(o * o + c * c),
b = Math.sqrt(a * a + f * f),
D = Math.max(v, m, b),
w = t[0],
O = t[1],
M = n[0],
S = n[1];
D == v ? (M = n[0], S = n[1]) : D == b && (w = e[0], O = e[1]);
var T = [w, O],
C = [M, S],
R = this.getSlopes(T, C),
A = this.angleOrientationAt(R, this.options.TOTAL_ANGLE_VOTE_POWER);
d += A[0], p += A[1], y += A[2];
var L,
_ = u(r);
try {
for (_.s(); !(L = _.n()).done;) {
var j = L.value,
E = this.angleOrientationAt(j, this.options.SINGLE_ANGLE_VOTE_POWER);
d += E[0], p += E[1], y += E[2];
}
} catch (t) {
_.e(t);
} finally {
_.f();
}
return d == Math.max(d, p, y) ? this.estimateVerticalDirection(c, l, f, h) : y == Math.max(p, y) ? this.estimateHorizontalDirection(o, i, a, s) : this.estimateDiagonalDirection(c, l, f, h, o, i, a, s);
}
}, {
key: "calculateSlope",
value: function (t, e, n, r) {
var i = (e - r) / (t - n),
o = 180 * Math.atan(i) / Math.PI;
return o <= 0 ? o = -o : o > 0 && (o = 180 - o), o;
}
}]) && h(e.prototype, n), t;
}();
function p(t) {
if ("undefined" == typeof Symbol || null == t[Symbol.iterator]) {
if (Array.isArray(t) || (t = function (t, e) {
if (!t) return;
if ("string" == typeof t) return y(t, e);
var n = Object.prototype.toString.call(t).slice(8, -1);
"Object" === n && t.constructor && (n = t.constructor.name);
if ("Map" === n || "Set" === n) return Array.from(n);
if ("Arguments" === n || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return y(t, e);
}(t))) {
var e = 0,
n = function () {};
return {
s: n,
n: function () {
return e >= t.length ? {
done: !0
} : {
done: !1,
value: t[e++]
};
},
e: function (t) {
throw t;
},
f: n
};
}
throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
}
var r,
i,
o = !0,
a = !1;
return {
s: function () {
r = t[Symbol.iterator]();
},
n: function () {
var t = r.next();
return o = t.done, t;
},
e: function (t) {
a = !0, i = t;
},
f: function () {
try {
o || null == r.return || r.return();
} finally {
if (a) throw i;
}
}
};
}
function y(t, e) {
(null == e || e > t.length) && (e = t.length);
for (var n = 0, r = new Array(e); n < e; n++) r[n] = t[n];
return r;
}
function g(t, e) {
if (!(t instanceof e)) throw new TypeError("Cannot call a class as a function");
}
function v(t, e) {
for (var n = 0; n < e.length; n++) {
var r = e[n];
r.enumerable = r.enumerable || !1, r.configurable = !0, "value" in r && (r.writable = !0), Object.defineProperty(t, r.key, r);
}
}
var m = function () {
function t(e) {
var n = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : {};
g(this, t), this.estimator = new d(n), this.gestures = e;
}
var e, n;
return e = t, (n = [{
key: "estimate",
value: function (t, e) {
var n,
r = [],
i = this.estimator.estimate(t),
u = [],
c = p(o.all);
try {
for (c.s(); !(n = c.n()).done;) {
var f = n.value;
u.push([o.getName(f), a.getName(i.curls[f]), l.getName(i.directions[f])]);
}
} catch (t) {
c.e(t);
} finally {
c.f();
}
var s,
h = p(this.gestures);
try {
for (h.s(); !(s = h.n()).done;) {
var d = s.value,
y = d.matchAgainst(i.curls, i.directions);
y >= e && r.push({
name: d.name,
confidence: y
});
}
} catch (t) {
h.e(t);
} finally {
h.f();
}
return {
poseData: u,
gestures: r
};
}
}]) && v(e.prototype, n), t;
}();
function b(t, e) {
return function (t) {
if (Array.isArray(t)) return t;
}(t) || function (t, e) {
if ("undefined" == typeof Symbol || !(Symbol.iterator in Object(t))) return;
var n = [],
r = !0,
i = !1,
o = void 0;
try {
for (var a, l = t[Symbol.iterator](); !(r = (a = l.next()).done) && (n.push(a.value), !e || n.length !== e); r = !0);
} catch (t) {
i = !0, o = t;
} finally {
try {
r || null == l.return || l.return();
} finally {
if (i) throw o;
}
}
return n;
}(t, e) || w(t, e) || function () {
throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
}();
}
function D(t) {
if ("undefined" == typeof Symbol || null == t[Symbol.iterator]) {
if (Array.isArray(t) || (t = w(t))) {
var e = 0,
n = function () {};
return {
s: n,
n: function () {
return e >= t.length ? {
done: !0
} : {
done: !1,
value: t[e++]
};
},
e: function (t) {
throw t;
},
f: n
};
}
throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
}
var r,
i,
o = !0,
a = !1;
return {
s: function () {
r = t[Symbol.iterator]();
},
n: function () {
var t = r.next();
return o = t.done, t;
},
e: function (t) {
a = !0, i = t;
},
f: function () {
try {
o || null == r.return || r.return();
} finally {
if (a) throw i;
}
}
};
}
function w(t, e) {
if (t) {
if ("string" == typeof t) return O(t, e);
var n = Object.prototype.toString.call(t).slice(8, -1);
return "Object" === n && t.constructor && (n = t.constructor.name), "Map" === n || "Set" === n ? Array.from(n) : "Arguments" === n || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n) ? O(t, e) : void 0;
}
}
function O(t, e) {
(null == e || e > t.length) && (e = t.length);
for (var n = 0, r = new Array(e); n < e; n++) r[n] = t[n];
return r;
}
function M(t, e) {
for (var n = 0; n < e.length; n++) {
var r = e[n];
r.enumerable = r.enumerable || !1, r.configurable = !0, "value" in r && (r.writable = !0), Object.defineProperty(t, r.key, r);
}
}
var S = function () {
function t(e) {
!function (t, e) {
if (!(t instanceof e)) throw new TypeError("Cannot call a class as a function");
}(this, t), this.name = e, this.curls = {}, this.directions = {}, this.weights = [1, 1, 1, 1, 1], this.weightsRelative = [1, 1, 1, 1, 1];
}
var e, n;
return e = t, (n = [{
key: "addCurl",
value: function (t, e, n) {
void 0 === this.curls[t] && (this.curls[t] = []), this.curls[t].push([e, n]);
}
}, {
key: "addDirection",
value: function (t, e, n) {
void 0 === this.directions[t] && (this.directions[t] = []), this.directions[t].push([e, n]);
}
}, {
key: "setWeight",
value: function (t, e) {
this.weights[t] = e;
var n = this.weights.reduce(function (t, e) {
return t + e;
}, 0);
this.weightsRelative = this.weights.map(function (t) {
return 5 * t / n;
});
}
}, {
key: "matchAgainst",
value: function (t, e) {
var n = 0;
for (var r in t) {
var i = t[r],
o = this.curls[r];
if (void 0 !== o) {
var a,
l = D(o);
try {
for (l.s(); !(a = l.n()).done;) {
var u = b(a.value, 2),
c = u[0],
f = u[1];
if (i == c) {
n += f * this.weightsRelative[r];
break;
}
}
} catch (t) {
l.e(t);
} finally {
l.f();
}
} else n += this.weightsRelative[r];
}
for (var s in e) {
var h = e[s],
d = this.directions[s];
if (void 0 !== d) {
var p,
y = D(d);
try {
for (y.s(); !(p = y.n()).done;) {
var g = b(p.value, 2),
v = g[0],
m = g[1];
if (h == v) {
n += m * this.weightsRelative[s];
break;
}
}
} catch (t) {
y.e(t);
} finally {
y.f();
}
} else n += this.weightsRelative[s];
}
return n;
}
}]) && M(e.prototype, n), t;
}(),
T = new S("victory");
T.addCurl(o.Thumb, a.HalfCurl, .5), T.addCurl(o.Thumb, a.NoCurl, .5), T.addDirection(o.Thumb, l.VerticalUp, 1), T.addDirection(o.Thumb, l.DiagonalUpLeft, 1), T.addCurl(o.Index, a.NoCurl, 1), T.addDirection(o.Index, l.VerticalUp, .75), T.addDirection(o.Index, l.DiagonalUpLeft, 1), T.addCurl(o.Middle, a.NoCurl, 1), T.addDirection(o.Middle, l.VerticalUp, 1), T.addDirection(o.Middle, l.DiagonalUpLeft, .75), T.addCurl(o.Ring, a.FullCurl, 1), T.addDirection(o.Ring, l.VerticalUp, .2), T.addDirection(o.Ring, l.DiagonalUpLeft, 1), T.addDirection(o.Ring, l.HorizontalLeft, .2), T.addCurl(o.Pinky, a.FullCurl, 1), T.addDirection(o.Pinky, l.VerticalUp, .2), T.addDirection(o.Pinky, l.DiagonalUpLeft, 1), T.addDirection(o.Pinky, l.HorizontalLeft, .2), T.setWeight(o.Index, 2), T.setWeight(o.Middle, 2);
var C = T,
R = new S("thumbs_up");
R.addCurl(o.Thumb, a.NoCurl, 1), R.addDirection(o.Thumb, l.VerticalUp, 1), R.addDirection(o.Thumb, l.DiagonalUpLeft, .25), R.addDirection(o.Thumb, l.DiagonalUpRight, .25);
for (var A = 0, L = [o.Index, o.Middle, o.Ring, o.Pinky]; A < L.length; A++) {
var _ = L[A];
R.addCurl(_, a.FullCurl, 1), R.addDirection(_, l.HorizontalLeft, 1), R.addDirection(_, l.HorizontalRight, 1);
}
var j = R;
e.default = {
GestureEstimator: m,
GestureDescription: S,
Finger: o,
FingerCurl: a,
FingerDirection: l,
Gestures: r
};
}]).default;
});
});
var fingerpose$1 = /*@__PURE__*/getDefaultExportFromCjs(fingerpose);
class HandsModel extends BaseModel {
constructor(handsfree, config) {
super(handsfree, config);
this.name = 'hands';
this.palmPoints = [0, 5, 9, 13, 17];
this.gestureEstimator = new fingerpose$1.GestureEstimator([]);
}
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 hands
this.loadDependency(`${this.handsfree.config.assetsPath}/@mediapipe/hands/hands.js`, () => {
// Configure model
this.api = new window.Hands({
locateFile: file => {
return `${this.handsfree.config.assetsPath}/@mediapipe/hands/${file}`;
}
});
this.api.setOptions(this.handsfree.config.hands);
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-hands');
this.handsfree.emit('modelReady', this);
this.handsfree.emit('handsModelReady', this);
this.handsfree.emit('mediapipeWarmedUp', this);
callback && callback(this);
}
/**
* Get data
*/
async getData() {
this.dependenciesLoaded && (await this.api.send({
image: this.handsfree.debug.$video
}));
return this.data;
} // Called through this.api.onResults
dataReceived(results) {
// Get center of palm
if (results.multiHandLandmarks) {
results = this.getCenterOfPalm(results);
} // Force handedness
results = this.forceHandedness(results); // Update and debug
this.data = results;
this.handsfree.data.hands = results;
if (this.handsfree.isDebugging) {
this.debug(results);
}
}
/**
* Forces the hands to always be in the same index
*/
forceHandedness(results) {
// Empty landmarks
results.landmarks = [[], [], [], []];
results.landmarksVisible = [false, false, false, false];
if (!results.multiHandLandmarks) {
return results;
} // Store landmarks in the correct index
results.multiHandLandmarks.forEach((landmarks, n) => {
let hand;
if (n < 2) {
hand = results.multiHandedness[n].label === 'Right' ? 0 : 1;
} else {
hand = results.multiHandedness[n].label === 'Right' ? 2 : 3;
}
results.landmarks[hand] = landmarks;
results.landmarksVisible[hand] = true;
});
return results;
}
/**
* Calculates the center of the palm
*/
getCenterOfPalm(results) {
results.multiHandLandmarks.forEach((hand, n) => {
let x = 0;
let y = 0;
this.palmPoints.forEach(i => {
x += hand[i].x;
y += hand[i].y;
});
x /= this.palmPoints.length;
y /= this.palmPoints.length;
results.multiHandLandmarks[n][21] = {
x,
y
};
});
return results;
}
/**
* Debugs the hands model
*/
debug(results) {
// Bail if drawing helpers haven't loaded
if (typeof drawConnectors === 'undefined') return; // Clear the canvas
this.handsfree.debug.context.hands.clearRect(0, 0, this.handsfree.debug.$canvas.hands.width, this.handsfree.debug.$canvas.hands.height); // Draw skeletons
if (results.multiHandLandmarks) {
for (const landmarks of results.multiHandLandmarks) {
drawConnectors(this.handsfree.debug.context.hands, landmarks, HAND_CONNECTIONS, {
color: '#00FF00',
lineWidth: 5
});
drawLandmarks(this.handsfree.debug.context.hands, landmarks, {
color: '#FF0000',
lineWidth: 2
});
}
}
}
/**
* Updates the gesture estimator
*/
updateGestureEstimator() {
const activeGestures = [];
const gestureDescriptions = []; // Build the gesture descriptions
this.gestures.forEach(name => {
if (!this.handsfree.gesture[name].enabled) return;
activeGestures.push(name); // Loop through the description and compile it
if (!this.handsfree.gesture[name].compiledDescription && this.handsfree.gesture[name].enabled) {
const description = new fingerpose$1.GestureDescription(name);
this.handsfree.gesture[name].description.forEach(pose => {
// Build the description
switch (pose[0]) {
case 'addCurl':
description[pose[0]](fingerpose$1.Finger[pose[1]], fingerpose$1.FingerCurl[pose[2]], pose[3]);
break;
case 'addDirection':
description[pose[0]](fingerpose$1.Finger[pose[1]], fingerpose$1.FingerDirection[pose[2]], pose[3]);
break;
case 'setWeight':
description[pose[0]](fingerpose$1.Finger[pose[1]], pose[2]);
break;
}
});
this.handsfree.gesture[name].compiledDescription = description;
}
}); // Create the gesture estimator
activeGestures.forEach(gesture => {
gestureDescriptions.push(this.handsfree.gesture[gesture].compiledDescription);
});
if (activeGestures.length) {
this.gestureEstimator = new fingerpose$1.GestureEstimator(gestureDescriptions);
}
}
/**
* Gets current gesture
*/
getGesture() {
let gestures = [null, null, null, null];
this.data.landmarks.forEach((landmarksObj, hand) => {
if (this.data.landmarksVisible[hand]) {
// Convert object to array
const landmarks = [];
for (let i = 0; i < 21; i++) {
landmarks.push([landmarksObj[i].x * window.outerWidth, landmarksObj[i].y * window.outerHeight, 0]);
} // Estimate
const estimate = this.gestureEstimator.estimate(landmarks, 7.5);
if (estimate.gestures.length) {
gestures[hand] = estimate.gestures.reduce((p, c) => {
const requiredConfidence = this.handsfree.gesture[c.name].confidence;
return c.confidence >= requiredConfidence && c.confidence > p.confidence ? c : p;
});
} else {
gestures[hand] = {
name: '',
confidence: 0
};
} // Must pass confidence
if (gestures[hand].name) {
const requiredConfidence = this.handsfree.gesture[gestures[hand].name].confidence;
if (gestures[hand].confidence < requiredConfidence) {
gestures[hand] = {
name: '',
confidence: 0
};
}
}
gestures[hand].pose = estimate.poseData;
}
});
return gestures;
}
}
class FacemeshModel extends BaseModel {
constructor(handsfree, config) {
super(handsfree, config);
this.name = 'facemesh';
this.isWarmedUp = false;
}
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 facemesh
this.loadDependency(`${this.handsfree.config.assetsPath}/@mediapipe/face_mesh/face_mesh.js`, () => {
// Configure model
this.api = new window.FaceMesh({
locateFile: file => {
return `${this.handsfree.config.assetsPath}/@mediapipe/face_mesh/${file}`;
}
});
this.api.setOptions(this.handsfree.config.facemesh);
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-facemesh');
this.handsfree.emit('modelReady', this);
this.handsfree.emit('facemeshModelReady', 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.facemesh = results;
if (this.handsfree.isDebugging) {
this.debug(results);
}
}
/**
* Debugs the facemesh model
*/
debug(results) {
// Bail if drawing helpers haven't loaded
if (typeof drawConnectors === 'undefined') return;
this.handsfree.debug.context.facemesh.clearRect(0, 0, this.handsfree.debug.$canvas.facemesh.width, this.handsfree.debug.$canvas.facemesh.height);
if (results.multiFaceLandmarks) {
for (const landmarks of results.multiFaceLandmarks) {
drawConnectors(this.handsfree.debug.context.facemesh, landmarks, FACEMESH_TESSELATION, {
color: '#C0C0C070',
lineWidth: 1
});
drawConnectors(this.handsfree.debug.context.facemesh, landmarks, FACEMESH_RIGHT_EYE, {
color: '#FF3030'
});
drawConnectors(this.handsfree.debug.context.facemesh, landmarks, FACEMESH_RIGHT_EYEBROW, {
color: '#FF3030'
});
drawConnectors(this.handsfree.debug.context.facemesh, landmarks, FACEMESH_LEFT_EYE, {
color: '#30FF30'
});
drawConnectors(this.handsfree.debug.context.facemesh, landmarks, FACEMESH_LEFT_EYEBROW, {
color: '#30FF30'
});
drawConnectors(this.handsfree.debug.context.facemesh, landmarks, FACEMESH_FACE_OVAL, {
color: '#E0E0E0'
});
drawConnectors(this.handsfree.debug.context.facemesh, landmarks, FACEMESH_LIPS, {
color: '#E0E0E0'
});
}
}
}
}
class PoseModel extends BaseModel {
constructor(handsfree, config) {
super(handsfree, config);
this.name = 'pose'; // 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 pose
this.loadDependency(`${this.handsfree.config.assetsPath}/@mediapipe/pose/pose.js`, () => {
this.api = new window.Pose({
locateFile: file => {
return `${this.handsfree.config.assetsPath}/@mediapipe/pose/${file}`;
}
});
this.api.setOptions(this.handsfree.config.pose);
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-pose');
this.handsfree.emit('modelReady', this);
this.handsfree.emit('poseModelReady', 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.pose = results;
if (this.handsfree.isDebugging) {
this.debug(results);
}
}
/**
* Debugs the pose model
*/
debug(results) {
this.handsfree.debug.context.pose.clearRect(0, 0, this.handsfree.debug.$canvas.pose.width, this.handsfree.debug.$canvas.pose.height);
if (results.poseLandmarks) {
drawConnectors(this.handsfree.debug.context.pose, results.poseLandmarks, POSE_CONNECTIONS, {
color: '#00FF00',
lineWidth: 4
});
drawLandmarks(this.handsfree.debug.context.pose, results.poseLandmarks, {
color: '#FF0000',
lineWidth: 2
});
}
}
}
/**
* 🚨 This model is not currently active
*/
class HandposeModel extends BaseModel {
constructor(handsfree, config) {
super(handsfree, config);
this.name = 'handpose'; // Various THREE variables
this.three = {
scene: null,
camera: null,
renderer: null,
meshes: []
};
this.normalized = []; // landmark indices that represent the palm
// 8 = Index finger tip
// 12 = Middle finger tip
this.palmPoints = [0, 1, 2, 5, 9, 13, 17];
this.gestureEstimator = new fingerpose$1.GestureEstimator([]);
}
loadDependencies(callback) {
this.loadDependency(`${this.handsfree.config.assetsPath}/three/three.min.js`, () => {
this.loadDependency(`${this.handsfree.config.assetsPath}/@tensorflow/tf-core.js`, () => {
this.loadDependency(`${this.handsfree.config.assetsPath}/@tensorflow/tf-converter.js`, () => {
this.loadDependency(`${this.handsfree.config.assetsPath}/@tensorflow/tf-backend-${this.handsfree.config.handpose.backend}.js`, () => {
this.loadDependency(`${this.handsfree.config.assetsPath}/@tensorflow-models/handpose/handpose.js`, () => {
this.handsfree.getUserMedia(async () => {
await window.tf.setBackend(this.handsfree.config.handpose.backend);
this.api = await handpose.load(this.handsfree.config.handpose.model);
this.setup3D();
callback && callback(this);
this.dependenciesLoaded = true;
this.handsfree.emit('modelReady', this);
this.handsfree.emit('handposeModelReady', this);
document.body.classList.add('handsfree-model-handpose');
});
});
});
});
}, !!window.tf);
}, !!window.THREE);
}
/**
* Runs inference and sets up other data
*/
async getData() {
if (!this.handsfree.debug.$video) return;
const predictions = await this.api.estimateHands(this.handsfree.debug.$video);
this.handsfree.data.handpose = this.data = { ...predictions[0],
normalized: this.normalized,
meshes: this.three.meshes
};
if (predictions[0]) {
this.updateMeshes(this.data);
}
this.three.renderer.render(this.three.scene, this.three.camera);
return this.data;
}
/**
* Sets up the 3D environment
*/
setup3D() {
// Setup Three
this.three = {
scene: new window.THREE.Scene(),
camera: new window.THREE.PerspectiveCamera(90, window.outerWidth / window.outerHeight, 0.1, 1000),
renderer: new THREE.WebGLRenderer({
alpha: true,
canvas: this.handsfree.debug.$canvas.handpose
}),
meshes: []
};
this.three.renderer.setSize(window.outerWidth, window.outerHeight);
this.three.camera.position.z = this.handsfree.debug.$video.videoWidth / 4;
this.three.camera.lookAt(new window.THREE.Vector3(0, 0, 0)); // Camera plane
this.three.screen = new window.THREE.Mesh(new window.THREE.BoxGeometry(window.outerWidth, window.outerHeight, 1), new window.THREE.