human/dist/human.node-wasm.js

15242 lines
493 KiB
JavaScript

/*
Human
homepage: <https://github.com/vladmandic/human>
author: <https://github.com/vladmandic>'
*/
"use strict";
var __create = Object.create;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __getProtoOf = Object.getPrototypeOf;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
var __commonJS = (cb, mod3) => function __require() {
return mod3 || (0, cb[__getOwnPropNames(cb)[0]])((mod3 = { exports: {} }).exports, mod3), mod3.exports;
};
var __export = (target, all2) => {
for (var name in all2)
__defProp(target, name, { get: all2[name], enumerable: true });
};
var __copyProps = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
}
return to;
};
var __toESM = (mod3, isNodeMode, target) => (target = mod3 != null ? __create(__getProtoOf(mod3)) : {}, __copyProps(
isNodeMode || !mod3 || !mod3.__esModule ? __defProp(target, "default", { value: mod3, enumerable: true }) : target,
mod3
));
var __toCommonJS = (mod3) => __copyProps(__defProp({}, "__esModule", { value: true }), mod3);
var __publicField = (obj, key, value) => {
__defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
return value;
};
var __accessCheck = (obj, member, msg) => {
if (!member.has(obj))
throw TypeError("Cannot " + msg);
};
var __privateGet = (obj, member, getter) => {
__accessCheck(obj, member, "read from private field");
return getter ? getter.call(obj) : member.get(obj);
};
var __privateAdd = (obj, member, value) => {
if (member.has(obj))
throw TypeError("Cannot add the same private member more than once");
member instanceof WeakSet ? member.add(obj) : member.set(obj, value);
};
var __privateSet = (obj, member, value, setter) => {
__accessCheck(obj, member, "write to private field");
setter ? setter.call(obj, value) : member.set(obj, value);
return value;
};
// dist/tfjs.esm.js
var require_tfjs_esm = __commonJS({
"dist/tfjs.esm.js"(exports, module2) {
"use strict";
var __defProp2 = Object.defineProperty;
var __getOwnPropDesc2 = Object.getOwnPropertyDescriptor;
var __getOwnPropNames2 = Object.getOwnPropertyNames;
var __hasOwnProp2 = Object.prototype.hasOwnProperty;
var __export2 = (target, all2) => {
for (var name in all2)
__defProp2(target, name, { get: all2[name], enumerable: true });
};
var __copyProps2 = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames2(from))
if (!__hasOwnProp2.call(to, key) && key !== except)
__defProp2(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc2(from, key)) || desc.enumerable });
}
return to;
};
var __reExport = (target, mod3, secondTarget) => (__copyProps2(target, mod3, "default"), secondTarget && __copyProps2(secondTarget, mod3, "default"));
var __toCommonJS2 = (mod3) => __copyProps2(__defProp2({}, "__esModule", { value: true }), mod3);
var tf_node_wasm_exports = {};
__export2(tf_node_wasm_exports, {
version: () => version7
});
module2.exports = __toCommonJS2(tf_node_wasm_exports);
__reExport(tf_node_wasm_exports, require("@tensorflow/tfjs-core"), module2.exports);
__reExport(tf_node_wasm_exports, require("@tensorflow/tfjs-converter"), module2.exports);
__reExport(tf_node_wasm_exports, require("@tensorflow/tfjs-backend-wasm"), module2.exports);
var version4 = "4.1.0";
var version22 = "4.1.0";
var version32 = "4.1.0";
var version42 = "4.1.0";
var version5 = "4.1.0";
var version6 = "0.0.1-alpha.16";
var version7 = {
tfjs: version4,
"tfjs-core": version4,
"tfjs-converter": version22,
"tfjs-backend-cpu": version32,
"tfjs-backend-webgl": version42,
"tfjs-backend-wasm": version5,
"tfjs-backend-webgpu": version6
};
}
});
// src/human.ts
var human_exports = {};
__export(human_exports, {
Env: () => Env,
Human: () => Human,
default: () => Human,
defaults: () => config,
draw: () => draw_exports,
empty: () => empty,
env: () => env,
match: () => match_exports,
models: () => models_exports2
});
module.exports = __toCommonJS(human_exports);
var tf38 = __toESM(require_tfjs_esm());
// src/util/util.ts
function log(...msg) {
const dt = new Date();
const ts = `${dt.getHours().toString().padStart(2, "0")}:${dt.getMinutes().toString().padStart(2, "0")}:${dt.getSeconds().toString().padStart(2, "0")}.${dt.getMilliseconds().toString().padStart(3, "0")}`;
if (msg)
console.log(ts, "Human:", ...msg);
}
function join(folder, file) {
const separator = folder.endsWith("/") ? "" : "/";
const skipJoin = file.startsWith(".") || file.startsWith("/") || file.startsWith("http:") || file.startsWith("https:") || file.startsWith("file:");
const path = skipJoin ? `${file}` : `${folder}${separator}${file}`;
if (!path.toLocaleLowerCase().includes(".json"))
throw new Error(`modelpath error: expecting json file: ${path}`);
return path;
}
var now = () => {
if (typeof performance !== "undefined")
return performance.now();
return parseInt((Number(process.hrtime.bigint()) / 1e3 / 1e3).toString());
};
function validate(defaults, config3, parent = "config", msgs = []) {
for (const key of Object.keys(config3)) {
if (typeof config3[key] === "object") {
validate(defaults[key], config3[key], key, msgs);
} else {
const defined = defaults && typeof defaults[key] !== "undefined";
if (!defined)
msgs.push({ reason: "unknown property", where: `${parent}.${key} = ${config3[key]}` });
const same = defaults && typeof defaults[key] === typeof config3[key];
if (defined && !same)
msgs.push({ reason: "property type mismatch", where: `${parent}.${key} = ${config3[key]}`, expected: typeof defaults[key] });
}
}
if (config3.debug && parent === "config" && msgs.length > 0)
log("invalid configuration", msgs);
return msgs;
}
function mergeDeep(...objects) {
const isObject = (obj) => obj && typeof obj === "object";
return objects.reduce((prev, obj) => {
Object.keys(obj || {}).forEach((key) => {
const pVal = prev[key];
const oVal = obj[key];
if (Array.isArray(pVal) && Array.isArray(oVal))
prev[key] = pVal.concat(...oVal);
else if (isObject(pVal) && isObject(oVal))
prev[key] = mergeDeep(pVal, oVal);
else
prev[key] = oVal;
});
return prev;
}, {});
}
// src/config.ts
var config = {
backend: "",
modelBasePath: "",
cacheModels: true,
validateModels: true,
wasmPath: "",
wasmPlatformFetch: false,
debug: false,
async: true,
warmup: "full",
cacheSensitivity: 0.7,
skipAllowed: false,
deallocate: false,
flags: {},
softwareKernels: false,
filter: {
enabled: true,
equalization: false,
width: 0,
height: 0,
flip: false,
return: true,
autoBrightness: true,
brightness: 0,
contrast: 0,
sharpness: 0,
blur: 0,
saturation: 0,
hue: 0,
negative: false,
sepia: false,
vintage: false,
kodachrome: false,
technicolor: false,
polaroid: false,
pixelate: 0
},
gesture: {
enabled: true
},
face: {
enabled: true,
detector: {
modelPath: "blazeface.json",
rotation: true,
maxDetected: 1,
skipFrames: 99,
skipTime: 2500,
minConfidence: 0.2,
iouThreshold: 0.1,
mask: false,
return: false
},
mesh: {
enabled: true,
modelPath: "facemesh.json",
keepInvalid: false
},
attention: {
enabled: false,
modelPath: "facemesh-attention.json"
},
iris: {
enabled: true,
modelPath: "iris.json"
},
emotion: {
enabled: true,
minConfidence: 0.1,
skipFrames: 99,
skipTime: 1500,
modelPath: "emotion.json"
},
description: {
enabled: true,
modelPath: "faceres.json",
skipFrames: 99,
skipTime: 3e3,
minConfidence: 0.1
},
antispoof: {
enabled: false,
skipFrames: 99,
skipTime: 4e3,
modelPath: "antispoof.json"
},
liveness: {
enabled: false,
skipFrames: 99,
skipTime: 4e3,
modelPath: "liveness.json"
}
},
body: {
enabled: true,
modelPath: "movenet-lightning.json",
maxDetected: -1,
minConfidence: 0.3,
skipFrames: 1,
skipTime: 200
},
hand: {
enabled: true,
rotation: true,
skipFrames: 99,
skipTime: 1e3,
minConfidence: 0.5,
iouThreshold: 0.2,
maxDetected: -1,
landmarks: true,
detector: {
modelPath: "handtrack.json"
},
skeleton: {
modelPath: "handlandmark-full.json"
}
},
object: {
enabled: false,
modelPath: "centernet.json",
minConfidence: 0.2,
iouThreshold: 0.4,
maxDetected: 10,
skipFrames: 99,
skipTime: 2e3
},
segmentation: {
enabled: false,
modelPath: "rvm.json",
ratio: 0.5,
mode: "default"
}
};
// src/util/env.ts
var tf3 = __toESM(require_tfjs_esm());
// src/image/image.ts
var tf2 = __toESM(require_tfjs_esm());
// src/image/imagefxshaders.ts
var vertexIdentity = `
precision highp float;
attribute vec2 pos;
attribute vec2 uv;
varying vec2 vUv;
uniform float flipY;
void main(void) {
vUv = uv;
gl_Position = vec4(pos.x, pos.y*flipY, 0.0, 1.);
}
`;
var colorMatrixWithAlpha = `
precision highp float;
varying vec2 vUv;
uniform sampler2D texture;
uniform float m[20];
void main(void) {
vec4 c = texture2D(texture, vUv);
gl_FragColor.r = m[0] * c.r + m[1] * c.g + m[2] * c.b + m[3] * c.a + m[4];
gl_FragColor.g = m[5] * c.r + m[6] * c.g + m[7] * c.b + m[8] * c.a + m[9];
gl_FragColor.b = m[10] * c.r + m[11] * c.g + m[12] * c.b + m[13] * c.a + m[14];
gl_FragColor.a = m[15] * c.r + m[16] * c.g + m[17] * c.b + m[18] * c.a + m[19];
}
`;
var colorMatrixWithoutAlpha = `
precision highp float;
varying vec2 vUv;
uniform sampler2D texture;
uniform float m[20];
void main(void) {
vec4 c = texture2D(texture, vUv);
gl_FragColor.r = m[0] * c.r + m[1] * c.g + m[2] * c.b + m[4];
gl_FragColor.g = m[5] * c.r + m[6] * c.g + m[7] * c.b + m[9];
gl_FragColor.b = m[10] * c.r + m[11] * c.g + m[12] * c.b + m[14];
gl_FragColor.a = c.a;
}
`;
var pixelate = `
precision highp float;
varying vec2 vUv;
uniform vec2 size;
uniform sampler2D texture;
vec2 pixelate(vec2 coord, vec2 size) {
return floor( coord / size ) * size;
}
void main(void) {
gl_FragColor = vec4(0.0);
vec2 coord = pixelate(vUv, size);
gl_FragColor += texture2D(texture, coord);
}
`;
var blur = `
precision highp float;
varying vec2 vUv;
uniform sampler2D texture;
uniform vec2 px;
void main(void) {
gl_FragColor = vec4(0.0);
gl_FragColor += texture2D(texture, vUv + vec2(-7.0*px.x, -7.0*px.y))*0.0044299121055113265;
gl_FragColor += texture2D(texture, vUv + vec2(-6.0*px.x, -6.0*px.y))*0.00895781211794;
gl_FragColor += texture2D(texture, vUv + vec2(-5.0*px.x, -5.0*px.y))*0.0215963866053;
gl_FragColor += texture2D(texture, vUv + vec2(-4.0*px.x, -4.0*px.y))*0.0443683338718;
gl_FragColor += texture2D(texture, vUv + vec2(-3.0*px.x, -3.0*px.y))*0.0776744219933;
gl_FragColor += texture2D(texture, vUv + vec2(-2.0*px.x, -2.0*px.y))*0.115876621105;
gl_FragColor += texture2D(texture, vUv + vec2(-1.0*px.x, -1.0*px.y))*0.147308056121;
gl_FragColor += texture2D(texture, vUv )*0.159576912161;
gl_FragColor += texture2D(texture, vUv + vec2( 1.0*px.x, 1.0*px.y))*0.147308056121;
gl_FragColor += texture2D(texture, vUv + vec2( 2.0*px.x, 2.0*px.y))*0.115876621105;
gl_FragColor += texture2D(texture, vUv + vec2( 3.0*px.x, 3.0*px.y))*0.0776744219933;
gl_FragColor += texture2D(texture, vUv + vec2( 4.0*px.x, 4.0*px.y))*0.0443683338718;
gl_FragColor += texture2D(texture, vUv + vec2( 5.0*px.x, 5.0*px.y))*0.0215963866053;
gl_FragColor += texture2D(texture, vUv + vec2( 6.0*px.x, 6.0*px.y))*0.00895781211794;
gl_FragColor += texture2D(texture, vUv + vec2( 7.0*px.x, 7.0*px.y))*0.0044299121055113265;
}
`;
var convolution = `
precision highp float;
varying vec2 vUv;
uniform sampler2D texture;
uniform vec2 px;
uniform float m[9];
void main(void) {
vec4 c11 = texture2D(texture, vUv - px); // top left
vec4 c12 = texture2D(texture, vec2(vUv.x, vUv.y - px.y)); // top center
vec4 c13 = texture2D(texture, vec2(vUv.x + px.x, vUv.y - px.y)); // top right
vec4 c21 = texture2D(texture, vec2(vUv.x - px.x, vUv.y) ); // mid left
vec4 c22 = texture2D(texture, vUv); // mid center
vec4 c23 = texture2D(texture, vec2(vUv.x + px.x, vUv.y) ); // mid right
vec4 c31 = texture2D(texture, vec2(vUv.x - px.x, vUv.y + px.y) ); // bottom left
vec4 c32 = texture2D(texture, vec2(vUv.x, vUv.y + px.y) ); // bottom center
vec4 c33 = texture2D(texture, vUv + px ); // bottom right
gl_FragColor =
c11 * m[0] + c12 * m[1] + c22 * m[2] +
c21 * m[3] + c22 * m[4] + c23 * m[5] +
c31 * m[6] + c32 * m[7] + c33 * m[8];
gl_FragColor.a = c22.a;
}
`;
// src/image/imagefx.ts
var collect = (source, prefix, collection) => {
const r = new RegExp("\\b" + prefix + " \\w+ (\\w+)", "ig");
source.replace(r, (match2, name) => {
collection[name] = 0;
return match2;
});
};
var GLProgram = class {
constructor(gl, vertexSource, fragmentSource) {
__publicField(this, "uniform", {});
__publicField(this, "attribute", {});
__publicField(this, "gl");
__publicField(this, "id");
__publicField(this, "compile", (source, type) => {
const shader = this.gl.createShader(type);
if (!shader) {
log("filter: could not create shader");
return null;
}
this.gl.shaderSource(shader, source);
this.gl.compileShader(shader);
if (!this.gl.getShaderParameter(shader, this.gl.COMPILE_STATUS)) {
log(`filter: gl compile failed: ${this.gl.getShaderInfoLog(shader) || "unknown"}`);
return null;
}
return shader;
});
this.gl = gl;
const vertexShader = this.compile(vertexSource, this.gl.VERTEX_SHADER);
const fragmentShader = this.compile(fragmentSource, this.gl.FRAGMENT_SHADER);
this.id = this.gl.createProgram();
if (!vertexShader || !fragmentShader)
return;
if (!this.id) {
log("filter: could not create webgl program");
return;
}
this.gl.attachShader(this.id, vertexShader);
this.gl.attachShader(this.id, fragmentShader);
this.gl.linkProgram(this.id);
if (!this.gl.getProgramParameter(this.id, this.gl.LINK_STATUS)) {
log(`filter: gl link failed: ${this.gl.getProgramInfoLog(this.id) || "unknown"}`);
return;
}
this.gl.useProgram(this.id);
collect(vertexSource, "attribute", this.attribute);
for (const a in this.attribute)
this.attribute[a] = this.gl.getAttribLocation(this.id, a);
collect(vertexSource, "uniform", this.uniform);
collect(fragmentSource, "uniform", this.uniform);
for (const u in this.uniform)
this.uniform[u] = this.gl.getUniformLocation(this.id, u);
}
};
function GLImageFilter() {
let drawCount = 0;
let sourceTexture = null;
let lastInChain = false;
let currentFramebufferIndex = -1;
let tempFramebuffers = [null, null];
let filterChain = [];
let vertexBuffer = null;
let currentProgram = null;
const fxcanvas = canvas(100, 100);
const shaderProgramCache = {};
const DRAW = { INTERMEDIATE: 1 };
const gl = fxcanvas.getContext("webgl");
if (!gl) {
log("filter: cannot get webgl context");
return;
}
this.gl = gl;
function resize(width, height) {
if (width === fxcanvas.width && height === fxcanvas.height)
return;
fxcanvas.width = width;
fxcanvas.height = height;
if (!vertexBuffer) {
const vertices = new Float32Array([-1, -1, 0, 1, 1, -1, 1, 1, -1, 1, 0, 0, -1, 1, 0, 0, 1, -1, 1, 1, 1, 1, 1, 0]);
vertexBuffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, vertexBuffer);
gl.bufferData(gl.ARRAY_BUFFER, vertices, gl.STATIC_DRAW);
gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, true);
}
gl.viewport(0, 0, fxcanvas.width, fxcanvas.height);
tempFramebuffers = [null, null];
}
function createFramebufferTexture(width, height) {
const fbo = gl.createFramebuffer();
gl.bindFramebuffer(gl.FRAMEBUFFER, fbo);
const renderbuffer = gl.createRenderbuffer();
gl.bindRenderbuffer(gl.RENDERBUFFER, renderbuffer);
const texture = gl.createTexture();
gl.bindTexture(gl.TEXTURE_2D, texture);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, width, height, 0, gl.RGBA, gl.UNSIGNED_BYTE, null);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, texture, 0);
gl.bindTexture(gl.TEXTURE_2D, null);
gl.bindFramebuffer(gl.FRAMEBUFFER, null);
return { fbo, texture };
}
function getTempFramebuffer(index2) {
tempFramebuffers[index2] = tempFramebuffers[index2] || createFramebufferTexture(fxcanvas.width, fxcanvas.height);
return tempFramebuffers[index2];
}
function draw(flags = 0) {
if (!currentProgram)
return;
let source = null;
let target = null;
let flipY = false;
if (drawCount === 0)
source = sourceTexture;
else
source = getTempFramebuffer(currentFramebufferIndex).texture || null;
drawCount++;
if (lastInChain && !(flags & DRAW.INTERMEDIATE)) {
target = null;
flipY = drawCount % 2 === 0;
} else {
currentFramebufferIndex = (currentFramebufferIndex + 1) % 2;
target = getTempFramebuffer(currentFramebufferIndex).fbo || null;
}
gl.bindTexture(gl.TEXTURE_2D, source);
gl.bindFramebuffer(gl.FRAMEBUFFER, target);
gl.uniform1f(currentProgram.uniform["flipY"], flipY ? -1 : 1);
gl.drawArrays(gl.TRIANGLES, 0, 6);
}
function compileShader(fragmentSource) {
if (shaderProgramCache[fragmentSource]) {
currentProgram = shaderProgramCache[fragmentSource];
gl.useProgram((currentProgram ? currentProgram.id : null) || null);
return currentProgram;
}
currentProgram = new GLProgram(gl, vertexIdentity, fragmentSource);
if (!currentProgram) {
log("filter: could not get webgl program");
return null;
}
const floatSize = Float32Array.BYTES_PER_ELEMENT;
const vertSize = 4 * floatSize;
gl.enableVertexAttribArray(currentProgram.attribute["pos"]);
gl.vertexAttribPointer(currentProgram.attribute["pos"], 2, gl.FLOAT, false, vertSize, 0 * floatSize);
gl.enableVertexAttribArray(currentProgram.attribute["uv"]);
gl.vertexAttribPointer(currentProgram.attribute["uv"], 2, gl.FLOAT, false, vertSize, 2 * floatSize);
shaderProgramCache[fragmentSource] = currentProgram;
return currentProgram;
}
const filter = {
colorMatrix: (matrix) => {
const m = new Float32Array(matrix);
m[4] /= 255;
m[9] /= 255;
m[14] /= 255;
m[19] /= 255;
const shader = m[18] === 1 && m[3] === 0 && m[8] === 0 && m[13] === 0 && m[15] === 0 && m[16] === 0 && m[17] === 0 && m[19] === 0 ? colorMatrixWithoutAlpha : colorMatrixWithAlpha;
const program = compileShader(shader);
if (!program)
return;
gl.uniform1fv(program.uniform["m"], m);
draw();
},
brightness: (brightness) => {
const b = (brightness || 0) + 1;
filter.colorMatrix([
b,
0,
0,
0,
0,
0,
b,
0,
0,
0,
0,
0,
b,
0,
0,
0,
0,
0,
1,
0
]);
},
saturation: (amount) => {
const x = (amount || 0) * 2 / 3 + 1;
const y = (x - 1) * -0.5;
filter.colorMatrix([
x,
y,
y,
0,
0,
y,
x,
y,
0,
0,
y,
y,
x,
0,
0,
0,
0,
0,
1,
0
]);
},
desaturate: () => {
filter.saturation(-1);
},
contrast: (amount) => {
const v = (amount || 0) + 1;
const o = -128 * (v - 1);
filter.colorMatrix([
v,
0,
0,
0,
o,
0,
v,
0,
0,
o,
0,
0,
v,
0,
o,
0,
0,
0,
1,
0
]);
},
negative: () => {
filter.contrast(-2);
},
hue: (rotation) => {
rotation = (rotation || 0) / 180 * Math.PI;
const cos = Math.cos(rotation);
const sin = Math.sin(rotation);
const lumR = 0.213;
const lumG = 0.715;
const lumB = 0.072;
filter.colorMatrix([
lumR + cos * (1 - lumR) + sin * -lumR,
lumG + cos * -lumG + sin * -lumG,
lumB + cos * -lumB + sin * (1 - lumB),
0,
0,
lumR + cos * -lumR + sin * 0.143,
lumG + cos * (1 - lumG) + sin * 0.14,
lumB + cos * -lumB + sin * -0.283,
0,
0,
lumR + cos * -lumR + sin * -(1 - lumR),
lumG + cos * -lumG + sin * lumG,
lumB + cos * (1 - lumB) + sin * lumB,
0,
0,
0,
0,
0,
1,
0
]);
},
desaturateLuminance: () => {
filter.colorMatrix([
0.2764723,
0.929708,
0.0938197,
0,
-37.1,
0.2764723,
0.929708,
0.0938197,
0,
-37.1,
0.2764723,
0.929708,
0.0938197,
0,
-37.1,
0,
0,
0,
1,
0
]);
},
sepia: () => {
filter.colorMatrix([
0.393,
0.7689999,
0.18899999,
0,
0,
0.349,
0.6859999,
0.16799999,
0,
0,
0.272,
0.5339999,
0.13099999,
0,
0,
0,
0,
0,
1,
0
]);
},
brownie: () => {
filter.colorMatrix([
0.5997023498159715,
0.34553243048391263,
-0.2708298674538042,
0,
47.43192855600873,
-0.037703249837783157,
0.8609577587992641,
0.15059552388459913,
0,
-36.96841498319127,
0.24113635128153335,
-0.07441037908422492,
0.44972182064877153,
0,
-7.562075277591283,
0,
0,
0,
1,
0
]);
},
vintagePinhole: () => {
filter.colorMatrix([
0.6279345635605994,
0.3202183420819367,
-0.03965408211312453,
0,
9.651285835294123,
0.02578397704808868,
0.6441188644374771,
0.03259127616149294,
0,
7.462829176470591,
0.0466055556782719,
-0.0851232987247891,
0.5241648018700465,
0,
5.159190588235296,
0,
0,
0,
1,
0
]);
},
kodachrome: () => {
filter.colorMatrix([
1.1285582396593525,
-0.3967382283601348,
-0.03992559172921793,
0,
63.72958762196502,
-0.16404339962244616,
1.0835251566291304,
-0.05498805115633132,
0,
24.732407896706203,
-0.16786010706155763,
-0.5603416277695248,
1.6014850761964943,
0,
35.62982807460946,
0,
0,
0,
1,
0
]);
},
technicolor: () => {
filter.colorMatrix([
1.9125277891456083,
-0.8545344976951645,
-0.09155508482755585,
0,
11.793603434377337,
-0.3087833385928097,
1.7658908555458428,
-0.10601743074722245,
0,
-70.35205161461398,
-0.231103377548616,
-0.7501899197440212,
1.847597816108189,
0,
30.950940869491138,
0,
0,
0,
1,
0
]);
},
polaroid: () => {
filter.colorMatrix([
1.438,
-0.062,
-0.062,
0,
0,
-0.122,
1.378,
-0.122,
0,
0,
-0.016,
-0.016,
1.483,
0,
0,
0,
0,
0,
1,
0
]);
},
shiftToBGR: () => {
filter.colorMatrix([
0,
0,
1,
0,
0,
0,
1,
0,
0,
0,
1,
0,
0,
0,
0,
0,
0,
0,
1,
0
]);
},
convolution: (matrix) => {
const m = new Float32Array(matrix);
const pixelSizeX = 1 / fxcanvas.width;
const pixelSizeY = 1 / fxcanvas.height;
const program = compileShader(convolution);
if (!program)
return;
gl.uniform1fv(program.uniform["m"], m);
gl.uniform2f(program.uniform["px"], pixelSizeX, pixelSizeY);
draw();
},
detectEdges: () => {
filter.convolution.call(this, [
0,
1,
0,
1,
-4,
1,
0,
1,
0
]);
},
sobelX: () => {
filter.convolution.call(this, [
-1,
0,
1,
-2,
0,
2,
-1,
0,
1
]);
},
sobelY: () => {
filter.convolution.call(this, [
-1,
-2,
-1,
0,
0,
0,
1,
2,
1
]);
},
sharpen: (amount) => {
const a = amount || 1;
filter.convolution.call(this, [
0,
-1 * a,
0,
-1 * a,
1 + 4 * a,
-1 * a,
0,
-1 * a,
0
]);
},
emboss: (size2) => {
const s = size2 || 1;
filter.convolution.call(this, [
-2 * s,
-1 * s,
0,
-1 * s,
1,
1 * s,
0,
1 * s,
2 * s
]);
},
blur: (size2) => {
const blurSizeX = size2 / 7 / fxcanvas.width;
const blurSizeY = size2 / 7 / fxcanvas.height;
const program = compileShader(blur);
if (!program)
return;
gl.uniform2f(program.uniform["px"], 0, blurSizeY);
draw(DRAW.INTERMEDIATE);
gl.uniform2f(program.uniform["px"], blurSizeX, 0);
draw();
},
pixelate: (size2) => {
const blurSizeX = size2 / fxcanvas.width;
const blurSizeY = size2 / fxcanvas.height;
const program = compileShader(pixelate);
if (!program)
return;
gl.uniform2f(program.uniform["size"], blurSizeX, blurSizeY);
draw();
}
};
this.add = function(name) {
const args = Array.prototype.slice.call(arguments, 1);
const func = filter[name];
filterChain.push({ func, args });
};
this.reset = function() {
filterChain = [];
};
this.get = function() {
return filterChain;
};
this.apply = function(image28) {
resize(image28.width, image28.height);
drawCount = 0;
if (!sourceTexture)
sourceTexture = gl.createTexture();
gl.bindTexture(gl.TEXTURE_2D, sourceTexture);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, image28);
for (let i = 0; i < filterChain.length; i++) {
lastInChain = i === filterChain.length - 1;
const f = filterChain[i];
f.func.apply(this, f.args || []);
}
return fxcanvas;
};
this.draw = function(image28) {
this.add("brightness", 0);
return this.apply(image28);
};
}
// src/image/enhance.ts
var tf = __toESM(require_tfjs_esm());
async function histogramEqualization(inputImage) {
const squeeze14 = inputImage.shape.length === 4 ? tf.squeeze(inputImage) : inputImage;
const rgb2 = tf.split(squeeze14, 3, 2);
const min2 = [tf.min(rgb2[0]), tf.min(rgb2[1]), tf.min(rgb2[2])];
const max5 = [tf.max(rgb2[0]), tf.max(rgb2[1]), tf.max(rgb2[2])];
const absMax = await Promise.all(max5.map((channel) => channel.data()));
const maxValue = Math.max(absMax[0][0], absMax[1][0], absMax[2][0]);
const maxRange = maxValue > 1 ? 255 : 1;
const factor = maxRange / maxValue;
let final;
if (factor > 1) {
const sub11 = [tf.sub(rgb2[0], min2[0]), tf.sub(rgb2[1], min2[1]), tf.sub(rgb2[2], min2[2])];
const range = [tf.sub(max5[0], min2[0]), tf.sub(max5[1], min2[1]), tf.sub(max5[2], min2[2])];
const enh = [tf.mul(sub11[0], factor), tf.mul(sub11[1], factor), tf.mul(sub11[2], factor)];
const stack5 = tf.stack([enh[0], enh[1], enh[2]], 2);
final = tf.reshape(stack5, [1, squeeze14.shape[0] || 0, squeeze14.shape[1] || 0, 3]);
tf.dispose([...sub11, ...range, ...enh]);
} else {
final = tf.expandDims(squeeze14, 0);
}
tf.dispose([...rgb2, ...min2, ...max5, rgb2, squeeze14, inputImage]);
return final;
}
// src/image/image.ts
var maxSize = 3840;
var inCanvas = null;
var outCanvas = null;
var tmpCanvas = null;
var fx;
var last = {
inputSum: 0,
cacheDiff: 1,
sumMethod: 0,
inputTensor: void 0
};
function reset() {
last.inputSum = 0;
last.cacheDiff = 1;
last.sumMethod = 0;
last.inputTensor = void 0;
}
function canvas(width, height) {
let c;
if (env.browser) {
if (env.worker) {
if (typeof OffscreenCanvas === "undefined")
throw new Error("canvas error: attempted to run in web worker but OffscreenCanvas is not supported");
c = new OffscreenCanvas(width, height);
} else {
if (typeof document === "undefined")
throw new Error("canvas error: attempted to run in browser but DOM is not defined");
c = document.createElement("canvas");
c.width = width;
c.height = height;
}
} else {
if (typeof env.Canvas !== "undefined")
c = new env.Canvas(width, height);
else if (typeof globalThis.Canvas !== "undefined")
c = new globalThis.Canvas(width, height);
}
return c;
}
function copy(input, output) {
const outputCanvas = output || canvas(input.width, input.height);
const ctx = outputCanvas.getContext("2d");
ctx.drawImage(input, 0, 0);
return outputCanvas;
}
async function process2(input, config3, getTensor = true) {
var _a, _b, _c;
if (!input) {
if (config3.debug)
log("input error: input is missing");
return { tensor: null, canvas: null };
}
if (!(input instanceof tf2.Tensor) && !(typeof Image !== "undefined" && input instanceof Image) && !(typeof env.Canvas !== "undefined" && input instanceof env.Canvas) && !(typeof globalThis.Canvas !== "undefined" && input instanceof globalThis.Canvas) && !(typeof ImageData !== "undefined" && input instanceof ImageData) && !(typeof ImageBitmap !== "undefined" && input instanceof ImageBitmap) && !(typeof HTMLImageElement !== "undefined" && input instanceof HTMLImageElement) && !(typeof HTMLMediaElement !== "undefined" && input instanceof HTMLMediaElement) && !(typeof HTMLVideoElement !== "undefined" && input instanceof HTMLVideoElement) && !(typeof HTMLCanvasElement !== "undefined" && input instanceof HTMLCanvasElement) && !(typeof OffscreenCanvas !== "undefined" && input instanceof OffscreenCanvas)) {
throw new Error("input error: type is not recognized");
}
if (input instanceof tf2.Tensor) {
let tensor7 = null;
if (input["isDisposedInternal"])
throw new Error("input error: attempted to use tensor but it is disposed");
if (!input.shape)
throw new Error("input error: attempted to use tensor without a shape");
if (input.shape.length === 3) {
if (input.shape[2] === 3) {
tensor7 = tf2.expandDims(input, 0);
} else if (input.shape[2] === 4) {
const rgb2 = tf2.slice3d(input, [0, 0, 0], [-1, -1, 3]);
tensor7 = tf2.expandDims(rgb2, 0);
tf2.dispose(rgb2);
}
} else if (input.shape.length === 4) {
if (input.shape[3] === 3) {
tensor7 = tf2.clone(input);
} else if (input.shape[3] === 4) {
tensor7 = tf2.slice4d(input, [0, 0, 0, 0], [-1, -1, -1, 3]);
}
}
if (tensor7 == null || tensor7.shape.length !== 4 || tensor7.shape[0] !== 1 || tensor7.shape[3] !== 3)
throw new Error(`input error: attempted to use tensor with unrecognized shape: ${input.shape.toString()}`);
if (tensor7.dtype === "int32") {
const cast8 = tf2.cast(tensor7, "float32");
tf2.dispose(tensor7);
tensor7 = cast8;
}
return { tensor: tensor7, canvas: config3.filter.return ? outCanvas : null };
}
if (typeof input["readyState"] !== "undefined" && input.readyState <= 2) {
if (config3.debug)
log("input stream is not ready");
return { tensor: null, canvas: inCanvas };
}
const originalWidth = input["naturalWidth"] || input["videoWidth"] || input["width"] || input["shape"] && input["shape"][1] > 0;
const originalHeight = input["naturalHeight"] || input["videoHeight"] || input["height"] || input["shape"] && input["shape"][2] > 0;
if (!originalWidth || !originalHeight) {
if (config3.debug)
log("cannot determine input dimensions");
return { tensor: null, canvas: inCanvas };
}
let targetWidth = originalWidth;
let targetHeight = originalHeight;
if (targetWidth > maxSize) {
targetWidth = maxSize;
targetHeight = Math.trunc(targetWidth * originalHeight / originalWidth);
}
if (targetHeight > maxSize) {
targetHeight = maxSize;
targetWidth = Math.trunc(targetHeight * originalWidth / originalHeight);
}
if ((((_a = config3.filter) == null ? void 0 : _a.width) || 0) > 0)
targetWidth = config3.filter.width;
else if ((((_b = config3.filter) == null ? void 0 : _b.height) || 0) > 0)
targetWidth = originalWidth * ((config3.filter.height || 0) / originalHeight);
if ((config3.filter.height || 0) > 0)
targetHeight = config3.filter.height;
else if ((config3.filter.width || 0) > 0)
targetHeight = originalHeight * ((config3.filter.width || 0) / originalWidth);
if (!targetWidth || !targetHeight)
throw new Error("input error: cannot determine dimension");
if (!inCanvas || inCanvas.width !== targetWidth || inCanvas.height !== targetHeight)
inCanvas = canvas(targetWidth, targetHeight);
const inCtx = inCanvas.getContext("2d");
if (typeof ImageData !== "undefined" && input instanceof ImageData) {
inCtx.putImageData(input, 0, 0);
} else {
if (config3.filter.flip && typeof inCtx.translate !== "undefined") {
inCtx.translate(originalWidth, 0);
inCtx.scale(-1, 1);
inCtx.drawImage(input, 0, 0, originalWidth, originalHeight, 0, 0, inCanvas.width, inCanvas.height);
inCtx.setTransform(1, 0, 0, 1, 0, 0);
} else {
inCtx.drawImage(input, 0, 0, originalWidth, originalHeight, 0, 0, inCanvas.width, inCanvas.height);
}
}
if (!outCanvas || inCanvas.width !== outCanvas.width || inCanvas.height !== outCanvas.height)
outCanvas = canvas(inCanvas.width, inCanvas.height);
if (config3.filter.enabled && env.webgl.supported) {
if (!fx)
fx = env.browser ? new GLImageFilter() : null;
env.filter = !!fx;
if (!(fx == null ? void 0 : fx.add)) {
if (config3.debug)
log("input process error: cannot initialize filters");
env.webgl.supported = false;
config3.filter.enabled = false;
copy(inCanvas, outCanvas);
} else {
fx.reset();
if (config3.filter.brightness !== 0)
fx.add("brightness", config3.filter.brightness);
if (config3.filter.contrast !== 0)
fx.add("contrast", config3.filter.contrast);
if (config3.filter.sharpness !== 0)
fx.add("sharpen", config3.filter.sharpness);
if (config3.filter.blur !== 0)
fx.add("blur", config3.filter.blur);
if (config3.filter.saturation !== 0)
fx.add("saturation", config3.filter.saturation);
if (config3.filter.hue !== 0)
fx.add("hue", config3.filter.hue);
if (config3.filter.negative)
fx.add("negative");
if (config3.filter.sepia)
fx.add("sepia");
if (config3.filter.vintage)
fx.add("brownie");
if (config3.filter.sepia)
fx.add("sepia");
if (config3.filter.kodachrome)
fx.add("kodachrome");
if (config3.filter.technicolor)
fx.add("technicolor");
if (config3.filter.polaroid)
fx.add("polaroid");
if (config3.filter.pixelate !== 0)
fx.add("pixelate", config3.filter.pixelate);
if (((_c = fx.get()) == null ? void 0 : _c.length) > 1)
outCanvas = fx.apply(inCanvas);
else
outCanvas = fx.draw(inCanvas);
}
} else {
copy(inCanvas, outCanvas);
if (fx)
fx = null;
env.filter = !!fx;
}
if (!getTensor)
return { tensor: null, canvas: outCanvas };
if (!outCanvas)
throw new Error("canvas error: cannot create output");
let pixels;
let depth = 3;
if (typeof ImageData !== "undefined" && input instanceof ImageData || input.data && input.width && input.height) {
if (env.browser && tf2.browser) {
pixels = tf2.browser ? tf2.browser.fromPixels(input) : null;
} else {
depth = input.data.length / input.height / input.width;
const arr = new Uint8Array(input.data.buffer);
pixels = tf2.tensor(arr, [input.height, input.width, depth], "int32");
}
} else {
if (!tmpCanvas || outCanvas.width !== tmpCanvas.width || outCanvas.height !== tmpCanvas.height)
tmpCanvas = canvas(outCanvas.width, outCanvas.height);
if (tf2.browser && env.browser) {
if (config3.backend === "webgl" || config3.backend === "humangl" || config3.backend === "webgpu") {
pixels = tf2.browser.fromPixels(outCanvas);
} else {
tmpCanvas = copy(outCanvas);
pixels = tf2.browser.fromPixels(tmpCanvas);
}
} else {
const tempCanvas = copy(outCanvas);
const tempCtx = tempCanvas.getContext("2d");
const tempData = tempCtx.getImageData(0, 0, targetWidth, targetHeight);
depth = tempData.data.length / targetWidth / targetHeight;
const arr = new Uint8Array(tempData.data.buffer);
pixels = tf2.tensor(arr, [targetWidth, targetHeight, depth]);
}
}
if (depth === 4) {
const rgb2 = tf2.slice3d(pixels, [0, 0, 0], [-1, -1, 3]);
tf2.dispose(pixels);
pixels = rgb2;
}
if (!pixels)
throw new Error("input error: cannot create tensor");
const casted = tf2.cast(pixels, "float32");
const tensor6 = config3.filter.equalization ? await histogramEqualization(casted) : tf2.expandDims(casted, 0);
tf2.dispose([pixels, casted]);
if (config3.filter.autoBrightness) {
const max5 = tf2.max(tensor6);
const maxVal = await max5.data();
config3.filter.brightness = maxVal[0] > 1 ? 1 - maxVal[0] / 255 : 1 - maxVal[0];
tf2.dispose(max5);
}
return { tensor: tensor6, canvas: config3.filter.return ? outCanvas : null };
}
async function skip(config3, input) {
let skipFrame = false;
if (config3.cacheSensitivity === 0 || !input.shape || input.shape.length !== 4 || input.shape[1] > 3840 || input.shape[2] > 2160)
return skipFrame;
if (!last.inputTensor) {
last.inputTensor = tf2.clone(input);
} else if (last.inputTensor.shape[1] !== input.shape[1] || last.inputTensor.shape[2] !== input.shape[2]) {
tf2.dispose(last.inputTensor);
last.inputTensor = tf2.clone(input);
} else {
const t2 = {};
t2.diff = tf2.sub(input, last.inputTensor);
t2.squared = tf2.mul(t2.diff, t2.diff);
t2.sum = tf2.sum(t2.squared);
const diffSum = await t2.sum.data();
const diffRelative = diffSum[0] / (input.shape[1] || 1) / (input.shape[2] || 1) / 255 / 3;
tf2.dispose([last.inputTensor, t2.diff, t2.squared, t2.sum]);
last.inputTensor = tf2.clone(input);
skipFrame = diffRelative <= (config3.cacheSensitivity || 0);
}
return skipFrame;
}
async function compare(config3, input1, input2) {
const t2 = {};
if (!input1 || !input2 || input1.shape.length !== 4 || input1.shape.length !== input2.shape.length) {
if (!config3.debug)
log("invalid input tensor or tensor shapes do not match:", input1.shape, input2.shape);
return 0;
}
if (input1.shape[0] !== 1 || input2.shape[0] !== 1 || input1.shape[3] !== 3 || input2.shape[3] !== 3) {
if (!config3.debug)
log("input tensors must be of shape [1, height, width, 3]:", input1.shape, input2.shape);
return 0;
}
t2.input1 = tf2.clone(input1);
t2.input2 = input1.shape[1] !== input2.shape[1] || input1.shape[2] !== input2.shape[2] ? tf2.image.resizeBilinear(input2, [input1.shape[1], input1.shape[2]]) : tf2.clone(input2);
t2.diff = tf2.sub(t2.input1, t2.input2);
t2.squared = tf2.mul(t2.diff, t2.diff);
t2.sum = tf2.sum(t2.squared);
const diffSum = await t2.sum.data();
const diffRelative = diffSum[0] / (input1.shape[1] || 1) / (input1.shape[2] || 1) / 255 / 3;
tf2.dispose([t2.input1, t2.input2, t2.diff, t2.squared, t2.sum]);
return diffRelative;
}
// src/util/env.ts
var Env = class {
constructor() {
__publicField(this, "browser");
__publicField(this, "node");
__publicField(this, "worker");
__publicField(this, "platform", "");
__publicField(this, "agent", "");
__publicField(this, "backends", []);
__publicField(this, "initial");
__publicField(this, "filter");
__publicField(this, "tfjs");
__publicField(this, "offscreen");
__publicField(this, "perfadd", false);
__publicField(this, "tensorflow", {
version: void 0,
gpu: void 0
});
__publicField(this, "wasm", {
supported: void 0,
backend: void 0,
simd: void 0,
multithread: void 0
});
__publicField(this, "webgl", {
supported: void 0,
backend: void 0,
version: void 0,
renderer: void 0,
shader: void 0,
vendor: void 0
});
__publicField(this, "webgpu", {
supported: void 0,
backend: void 0,
adapter: void 0
});
__publicField(this, "cpu", {
model: void 0,
flags: []
});
__publicField(this, "kernels", []);
__publicField(this, "Canvas");
__publicField(this, "Image");
__publicField(this, "ImageData");
this.browser = typeof navigator !== "undefined";
this.node = typeof process !== "undefined" && typeof process.versions !== "undefined" && typeof process.versions.node !== "undefined";
this.tfjs = { version: tf3.version["tfjs-core"] };
this.offscreen = typeof OffscreenCanvas !== "undefined";
this.initial = true;
this.worker = this.browser && this.offscreen ? typeof WorkerGlobalScope !== "undefined" : void 0;
if (typeof navigator !== "undefined") {
const raw = navigator.userAgent.match(/\(([^()]+)\)/g);
if (raw == null ? void 0 : raw[0]) {
const platformMatch = raw[0].match(/\(([^()]+)\)/g);
this.platform = (platformMatch == null ? void 0 : platformMatch[0]) ? platformMatch[0].replace(/\(|\)/g, "") : "";
this.agent = navigator.userAgent.replace(raw[0], "");
if (this.platform[1])
this.agent = this.agent.replace(raw[1], "");
this.agent = this.agent.replace(/ /g, " ");
}
} else if (typeof process !== "undefined") {
this.platform = `${process.platform} ${process.arch}`;
this.agent = `NodeJS ${process.version}`;
}
}
async updateBackend() {
this.backends = Object.keys(tf3.engine().registryFactory);
try {
this.tensorflow = {
version: tf3.backend()["binding"] ? tf3.backend()["binding"].TF_Version : void 0,
gpu: tf3.backend()["binding"] ? tf3.backend()["binding"].isUsingGpuDevice() : void 0
};
} catch (e) {
}
this.wasm.supported = typeof WebAssembly !== "undefined";
this.wasm.backend = this.backends.includes("wasm");
if (this.wasm.supported && this.wasm.backend) {
this.wasm.simd = await tf3.env().getAsync("WASM_HAS_SIMD_SUPPORT");
this.wasm.multithread = await tf3.env().getAsync("WASM_HAS_MULTITHREAD_SUPPORT");
}
const c = canvas(100, 100);
const gl = c ? c.getContext("webgl2") : void 0;
this.webgl.supported = typeof gl !== "undefined";
this.webgl.backend = this.backends.includes("webgl");
if (this.webgl.supported && this.webgl.backend && gl) {
this.webgl.version = gl.getParameter(gl.VERSION);
this.webgl.vendor = gl.getParameter(gl.VENDOR);
this.webgl.renderer = gl.getParameter(gl.RENDERER);
this.webgl.shader = gl.getParameter(gl.SHADING_LANGUAGE_VERSION);
}
this.webgpu.supported = this.browser && typeof navigator.gpu !== "undefined";
this.webgpu.backend = this.backends.includes("webgpu");
try {
if (this.webgpu.supported) {
const adapter = await navigator.gpu.requestAdapter();
this.webgpu.adapter = await (adapter == null ? void 0 : adapter.requestAdapterInfo());
}
} catch (e) {
this.webgpu.supported = false;
}
try {
this.kernels = tf3.getKernelsForBackend(tf3.getBackend()).map((kernel) => kernel.kernelName.toLowerCase());
} catch (e) {
}
}
updateCPU() {
const cpu = { model: "", flags: [] };
if (this.node && this.platform.startsWith("linux")) {
}
if (!this.cpu)
Object.defineProperty(this, "cpu", { value: cpu });
else
this.cpu = cpu;
}
};
var env = new Env();
// src/util/webcam.ts
var WebCam = class {
constructor() {
__publicField(this, "config");
__publicField(this, "element");
__publicField(this, "stream");
__publicField(this, "devices", []);
__publicField(this, "enumerate", async () => {
try {
const devices = await navigator.mediaDevices.enumerateDevices();
this.devices = devices.filter((device) => device.kind === "videoinput");
} catch (e) {
this.devices = [];
}
return this.devices;
});
__publicField(this, "start", async (webcamConfig) => {
if (webcamConfig == null ? void 0 : webcamConfig.debug)
this.config.debug = webcamConfig == null ? void 0 : webcamConfig.debug;
if (webcamConfig == null ? void 0 : webcamConfig.crop)
this.config.crop = webcamConfig == null ? void 0 : webcamConfig.crop;
if (webcamConfig == null ? void 0 : webcamConfig.mode)
this.config.mode = webcamConfig == null ? void 0 : webcamConfig.mode;
if (webcamConfig == null ? void 0 : webcamConfig.width)
this.config.width = webcamConfig == null ? void 0 : webcamConfig.width;
if (webcamConfig == null ? void 0 : webcamConfig.height)
this.config.height = webcamConfig == null ? void 0 : webcamConfig.height;
if (webcamConfig == null ? void 0 : webcamConfig.id)
this.config.id = webcamConfig == null ? void 0 : webcamConfig.id;
if (webcamConfig == null ? void 0 : webcamConfig.element) {
if (typeof webcamConfig.element === "string") {
const el = document.getElementById(webcamConfig.element);
if (el && el instanceof HTMLVideoElement) {
this.element = el;
} else {
if (this.config.debug)
log("webcam", "cannot get dom element", webcamConfig.element);
return;
}
} else if (webcamConfig.element instanceof HTMLVideoElement) {
this.element = webcamConfig.element;
} else {
if (this.config.debug)
log("webcam", "unknown dom element", webcamConfig.element);
return;
}
} else {
this.element = document.createElement("video");
}
const requestedConstraints = {
audio: false,
video: {
facingMode: this.config.mode === "front" ? "user" : "environment",
resizeMode: this.config.crop ? "crop-and-scale" : "none",
width: { ideal: this.config.width > 0 ? this.config.width : window.innerWidth },
height: { ideal: this.config.height > 0 ? this.config.height : window.innerHeight }
}
};
if (this.config.id)
requestedConstraints.video.deviceId = this.config.id;
this.element.addEventListener("play", () => {
if (this.config.debug)
log("webcam", "play");
});
this.element.addEventListener("pause", () => {
if (this.config.debug)
log("webcam", "pause");
});
this.element.addEventListener("click", async () => {
if (!this.element || !this.stream)
return;
if (this.element.paused)
await this.element.play();
else
this.element.pause();
});
if (!(navigator == null ? void 0 : navigator.mediaDevices)) {
if (this.config.debug)
log("webcam", "no devices");
return;
}
try {
this.stream = await navigator.mediaDevices.getUserMedia(requestedConstraints);
} catch (err) {
log("webcam", err);
return;
}
if (!this.stream) {
if (this.config.debug)
log("webcam", "no stream");
return;
}
this.element.srcObject = this.stream;
const ready3 = new Promise((resolve) => {
if (!this.element)
resolve(false);
else
this.element.onloadeddata = () => resolve(true);
});
await ready3;
await this.element.play();
if (this.config.debug) {
log("webcam", {
width: this.width,
height: this.height,
label: this.label,
stream: this.stream,
track: this.track,
settings: this.settings,
constraints: this.constraints,
capabilities: this.capabilities
});
}
});
__publicField(this, "pause", () => {
if (this.element)
this.element.pause();
});
__publicField(this, "play", async () => {
if (this.element)
await this.element.play();
});
__publicField(this, "stop", () => {
if (this.config.debug)
log("webcam", "stop");
if (this.track)
this.track.stop();
});
this.config = {
element: void 0,
debug: true,
mode: "front",
crop: false,
width: 0,
height: 0
};
}
get track() {
if (!this.stream)
return void 0;
return this.stream.getVideoTracks()[0];
}
get capabilities() {
if (!this.track)
return void 0;
return this.track.getCapabilities ? this.track.getCapabilities() : void 0;
}
get constraints() {
if (!this.track)
return void 0;
return this.track.getConstraints ? this.track.getConstraints() : void 0;
}
get settings() {
if (!this.stream)
return void 0;
const track = this.stream.getVideoTracks()[0];
return track.getSettings ? track.getSettings() : void 0;
}
get label() {
if (!this.track)
return "";
return this.track.label;
}
get paused() {
var _a;
return ((_a = this.element) == null ? void 0 : _a.paused) || false;
}
get width() {
var _a;
return ((_a = this.element) == null ? void 0 : _a.videoWidth) || 0;
}
get height() {
var _a;
return ((_a = this.element) == null ? void 0 : _a.videoHeight) || 0;
}
};
// src/tfjs/load.ts
var tf4 = __toESM(require_tfjs_esm());
// models/models.json
var models_exports = {};
__export(models_exports, {
age: () => age,
"anti-spoofing": () => anti_spoofing,
antispoof: () => antispoof,
blazeface: () => blazeface,
"blazeface-back": () => blazeface_back,
"blazeface-front": () => blazeface_front,
"blazepose-detector": () => blazepose_detector,
"blazepose-detector2d": () => blazepose_detector2d,
"blazepose-detector3d": () => blazepose_detector3d,
"blazepose-full": () => blazepose_full,
"blazepose-heavy": () => blazepose_heavy,
"blazepose-lite": () => blazepose_lite,
centernet: () => centernet,
default: () => models_default,
efficientpose: () => efficientpose,
"efficientpose-i-lite": () => efficientpose_i_lite,
"efficientpose-ii-lite": () => efficientpose_ii_lite,
"efficientpose-iv": () => efficientpose_iv,
emotion: () => emotion,
faceboxes: () => faceboxes,
facemesh: () => facemesh,
"facemesh-attention": () => facemesh_attention,
"facemesh-attention-alt": () => facemesh_attention_alt,
"facemesh-detection-full": () => facemesh_detection_full,
"facemesh-detection-short": () => facemesh_detection_short,
"facemesh-orig": () => facemesh_orig,
faceres: () => faceres,
"faceres-deep": () => faceres_deep,
gear: () => gear,
gender: () => gender,
"gender-ssrnet-imdb": () => gender_ssrnet_imdb,
handdetect: () => handdetect,
"handlandmark-full": () => handlandmark_full,
"handlandmark-lite": () => handlandmark_lite,
"handlandmark-sparse": () => handlandmark_sparse,
handskeleton: () => handskeleton,
handtrack: () => handtrack,
"insightface-efficientnet-b0": () => insightface_efficientnet_b0,
"insightface-ghostnet-strides1": () => insightface_ghostnet_strides1,
"insightface-ghostnet-strides2": () => insightface_ghostnet_strides2,
"insightface-mobilenet-emore": () => insightface_mobilenet_emore,
"insightface-mobilenet-swish": () => insightface_mobilenet_swish,
iris: () => iris,
liveness: () => liveness,
meet: () => meet,
mobileface: () => mobileface,
mobilefacenet: () => mobilefacenet,
models: () => models,
"movenet-lightning": () => movenet_lightning,
"movenet-multipose": () => movenet_multipose,
"movenet-thunder": () => movenet_thunder,
nanodet: () => nanodet,
"nanodet-e": () => nanodet_e,
"nanodet-g": () => nanodet_g,
"nanodet-m": () => nanodet_m,
"nanodet-t": () => nanodet_t,
posenet: () => posenet,
rvm: () => rvm,
selfie: () => selfie
});
var antispoof = 853098;
var blazeface = 538928;
var centernet = 4030290;
var emotion = 820516;
var facemesh = 1477958;
var faceres = 6978814;
var handlandmark_full = 5431368;
var handtrack = 2964837;
var iris = 2599092;
var liveness = 592976;
var models = 0;
var movenet_lightning = 4650216;
var age = 161240;
var blazeface_back = 538928;
var blazeface_front = 402048;
var blazepose_detector2d = 7499400;
var blazepose_detector3d = 5928856;
var blazepose_full = 6338290;
var blazepose_heavy = 27501554;
var blazepose_lite = 2725490;
var efficientpose = 5651240;
var faceboxes = 2013002;
var facemesh_attention_alt = 2387598;
var facemesh_attention = 2382414;
var facemesh_detection_full = 1026192;
var facemesh_detection_short = 201268;
var facemesh_orig = 2955780;
var faceres_deep = 13957620;
var gear = 1498916;
var gender_ssrnet_imdb = 161236;
var gender = 201808;
var handdetect = 3515612;
var handlandmark_lite = 2023432;
var handlandmark_sparse = 5286322;
var handskeleton = 5502280;
var meet = 372228;
var mobileface = 2183192;
var mobilefacenet = 5171976;
var movenet_multipose = 9448838;
var movenet_thunder = 12477112;
var nanodet = 7574558;
var posenet = 5032780;
var rvm = 3739355;
var selfie = 212886;
var blazepose_detector = 5928856;
var anti_spoofing = 853098;
var efficientpose_i_lite = 2269064;
var efficientpose_ii_lite = 5651240;
var efficientpose_iv = 25643252;
var insightface_efficientnet_b0 = 13013224;
var insightface_ghostnet_strides1 = 8093408;
var insightface_ghostnet_strides2 = 8049584;
var insightface_mobilenet_emore = 6938536;
var insightface_mobilenet_swish = 12168584;
var nanodet_e = 12319156;
var nanodet_g = 7574558;
var nanodet_m = 1887474;
var nanodet_t = 5294216;
var models_default = {
antispoof,
blazeface,
centernet,
emotion,
facemesh,
faceres,
"handlandmark-full": handlandmark_full,
handtrack,
iris,
liveness,
models,
"movenet-lightning": movenet_lightning,
age,
"blazeface-back": blazeface_back,
"blazeface-front": blazeface_front,
"blazepose-detector2d": blazepose_detector2d,
"blazepose-detector3d": blazepose_detector3d,
"blazepose-full": blazepose_full,
"blazepose-heavy": blazepose_heavy,
"blazepose-lite": blazepose_lite,
efficientpose,
faceboxes,
"facemesh-attention-alt": facemesh_attention_alt,
"facemesh-attention": facemesh_attention,
"facemesh-detection-full": facemesh_detection_full,
"facemesh-detection-short": facemesh_detection_short,
"facemesh-orig": facemesh_orig,
"faceres-deep": faceres_deep,
gear,
"gender-ssrnet-imdb": gender_ssrnet_imdb,
gender,
handdetect,
"handlandmark-lite": handlandmark_lite,
"handlandmark-sparse": handlandmark_sparse,
handskeleton,
meet,
mobileface,
mobilefacenet,
"movenet-multipose": movenet_multipose,
"movenet-thunder": movenet_thunder,
nanodet,
posenet,
rvm,
selfie,
"blazepose-detector": blazepose_detector,
"anti-spoofing": anti_spoofing,
"efficientpose-i-lite": efficientpose_i_lite,
"efficientpose-ii-lite": efficientpose_ii_lite,
"efficientpose-iv": efficientpose_iv,
"insightface-efficientnet-b0": insightface_efficientnet_b0,
"insightface-ghostnet-strides1": insightface_ghostnet_strides1,
"insightface-ghostnet-strides2": insightface_ghostnet_strides2,
"insightface-mobilenet-emore": insightface_mobilenet_emore,
"insightface-mobilenet-swish": insightface_mobilenet_swish,
"nanodet-e": nanodet_e,
"nanodet-g": nanodet_g,
"nanodet-m": nanodet_m,
"nanodet-t": nanodet_t
};
// src/tfjs/load.ts
var options = {
cacheModels: true,
cacheSupported: true,
verbose: true,
debug: false,
modelBasePath: ""
};
var modelStats = {};
async function httpHandler(url, init4) {
if (options.debug)
log("load model fetch:", url, init4);
return fetch(url, init4);
}
function setModelLoadOptions(config3) {
options.cacheModels = config3.cacheModels;
options.verbose = config3.debug;
options.modelBasePath = config3.modelBasePath;
}
async function loadModel(modelPath) {
var _a, _b, _c, _d;
let modelUrl = join(options.modelBasePath, modelPath || "");
if (!modelUrl.toLowerCase().endsWith(".json"))
modelUrl += ".json";
const modelPathSegments = modelUrl.includes("/") ? modelUrl.split("/") : modelUrl.split("\\");
const shortModelName = modelPathSegments[modelPathSegments.length - 1].replace(".json", "");
const cachedModelName = "indexeddb://" + shortModelName;
modelStats[shortModelName] = {
name: shortModelName,
sizeFromManifest: 0,
sizeLoadedWeights: 0,
sizeDesired: models_exports[shortModelName],
inCache: false,
url: ""
};
options.cacheSupported = typeof indexedDB !== "undefined";
let cachedModels = {};
try {
cachedModels = options.cacheSupported && options.cacheModels ? await tf4.io.listModels() : {};
} catch (e) {
options.cacheSupported = false;
}
modelStats[shortModelName].inCache = options.cacheSupported && options.cacheModels && Object.keys(cachedModels).includes(cachedModelName);
modelStats[shortModelName].url = modelStats[shortModelName].inCache ? cachedModelName : modelUrl;
const tfLoadOptions = typeof fetch === "undefined" ? {} : { fetchFunc: (url, init4) => httpHandler(url, init4) };
let model23 = new tf4.GraphModel(modelStats[shortModelName].url, tfLoadOptions);
let loaded = false;
try {
model23.findIOHandler();
if (options.debug)
log("model load handler:", model23["handler"]);
} catch (err) {
log("error finding model i/o handler:", modelUrl, err);
}
try {
const artifacts = await ((_a = model23.handler) == null ? void 0 : _a.load()) || null;
modelStats[shortModelName].sizeFromManifest = ((_b = artifacts == null ? void 0 : artifacts.weightData) == null ? void 0 : _b.byteLength) || 0;
if (artifacts)
model23.loadSync(artifacts);
else
model23 = await tf4.loadGraphModel(modelStats[shortModelName].inCache ? cachedModelName : modelUrl, tfLoadOptions);
modelStats[shortModelName].sizeLoadedWeights = ((_d = (_c = model23.artifacts) == null ? void 0 : _c.weightData) == null ? void 0 : _d.byteLength) || 0;
if (options.verbose)
log("load:", { model: shortModelName, url: model23["modelUrl"], bytes: modelStats[shortModelName].sizeLoadedWeights });
loaded = true;
} catch (err) {
log("error loading model:", modelUrl, err);
}
if (loaded && options.cacheModels && options.cacheSupported && !modelStats[shortModelName].inCache) {
try {
const saveResult = await model23.save(cachedModelName);
if (options.debug)
log("model saved:", cachedModelName, saveResult);
} catch (err) {
log("error saving model:", modelUrl, err);
}
}
return model23;
}
// package.json
var version2 = "3.0.0";
// src/tfjs/backend.ts
var tf7 = __toESM(require_tfjs_esm());
// src/tfjs/humangl.ts
var tf5 = __toESM(require_tfjs_esm());
var config2 = {
name: "humangl",
priority: 999,
canvas: null,
gl: null,
extensions: [],
webGLattr: {
alpha: false,
antialias: false,
premultipliedAlpha: false,
preserveDrawingBuffer: false,
depth: false,
stencil: false,
failIfMajorPerformanceCaveat: false,
desynchronized: true
}
};
function extensions() {
const gl = config2.gl;
if (!gl)
return;
config2.extensions = gl.getSupportedExtensions();
}
function register(instance) {
var _a;
if (instance.config.backend !== "humangl")
return;
if (config2.name in tf5.engine().registry && !((_a = config2 == null ? void 0 : config2.gl) == null ? void 0 : _a.getParameter(config2.gl.VERSION))) {
log("humangl error: backend invalid context");
instance.models.reset();
}
if (!tf5.findBackend(config2.name)) {
try {
config2.canvas = canvas(100, 100);
} catch (err) {
log("humangl error: cannot create canvas:", err);
return;
}
try {
config2.gl = config2.canvas.getContext("webgl2", config2.webGLattr);
if (!config2.gl) {
log("humangl error: cannot get webgl context");
return;
}
const glv2 = config2.gl.getParameter(config2.gl.VERSION).includes("2.0");
if (!glv2) {
log("backend override: using fallback webgl backend as webgl 2.0 is not detected");
instance.config.backend = "webgl";
return;
}
if (config2.canvas) {
config2.canvas.addEventListener("webglcontextlost", (e) => {
log("humangl error:", e.type);
log("possible browser memory leak using webgl or conflict with multiple backend registrations");
instance.emit("error");
throw new Error("backend error: webgl context lost");
});
config2.canvas.addEventListener("webglcontextrestored", (e) => {
log("humangl error: context restored:", e);
});
config2.canvas.addEventListener("webglcontextcreationerror", (e) => {
log("humangl error: context create:", e);
});
}
} catch (err) {
log("humangl error: cannot get webgl context:", err);
return;
}
try {
tf5.setWebGLContext(2, config2.gl);
} catch (err) {
log("humangl error: cannot set webgl context:", err);
return;
}
try {
const ctx = new tf5.GPGPUContext(config2.gl);
tf5.registerBackend(config2.name, () => new tf5.MathBackendWebGL(ctx), config2.priority);
} catch (err) {
log("humangl error: cannot register webgl backend:", err);
return;
}
try {
const kernels = tf5.getKernelsForBackend("webgl");
kernels.forEach((kernelConfig) => {
const newKernelConfig = { ...kernelConfig, backendName: config2.name };
tf5.registerKernel(newKernelConfig);
});
} catch (err) {
log("humangl error: cannot update webgl backend registration:", err);
return;
}
try {
if (tf5.env().flagRegistry.WEBGL_VERSION)
tf5.env().set("WEBGL_VERSION", 2);
} catch (err) {
log("humangl error: cannot set WebGL backend flags:", err);
return;
}
extensions();
const backend4 = tf5.backend();
const current = typeof backend4["gpgpu"] !== "undefined" ? backend4["getGPGPUContext"]().gl : null;
if (current) {
if (instance.config.debug)
log("humangl backend registered:", { webgl: current.getParameter(current.VERSION), renderer: current.getParameter(current.RENDERER) });
} else {
log("humangl error: no current gl context:", current, config2.gl);
}
}
}
// src/tfjs/constants.ts
var tf6 = __toESM(require_tfjs_esm());
var constants = {
tf255: 255,
tf1: 1,
tf2: 2,
tf05: 0.5,
tf127: 127.5,
rgb: [0.2989, 0.587, 0.114]
};
function init() {
constants.tf255 = tf6.scalar(255, "float32");
constants.tf1 = tf6.scalar(1, "float32");
constants.tf2 = tf6.scalar(2, "float32");
constants.tf05 = tf6.scalar(0.5, "float32");
constants.tf127 = tf6.scalar(127.5, "float32");
constants.rgb = tf6.tensor1d([0.2989, 0.587, 0.114], "float32");
}
// src/tfjs/backend.ts
async function getBestBackend() {
await env.updateBackend();
if (!env.browser)
return "tensorflow";
if (env.webgpu.supported && env.webgpu.backend)
return "webgpu";
if (env.webgl.supported && env.webgl.backend)
return "webgl";
if (env.wasm.supported && env.wasm.backend)
return "wasm";
return "cpu";
}
function registerCustomOps(config3) {
const newKernels = [];
if (!env.kernels.includes("mod")) {
const kernelMod = {
kernelName: "Mod",
backendName: tf7.getBackend(),
kernelFunc: (op) => tf7.tidy(() => tf7.sub(op.inputs.a, tf7.mul(tf7.div(op.inputs.a, op.inputs.b), op.inputs.b)))
};
tf7.registerKernel(kernelMod);
env.kernels.push("mod");
newKernels.push("mod");
}
if (!env.kernels.includes("floormod")) {
const kernelFloorMod = {
kernelName: "FloorMod",
backendName: tf7.getBackend(),
kernelFunc: (op) => tf7.tidy(() => tf7.add(tf7.mul(tf7.floorDiv(op.inputs.a, op.inputs.b), op.inputs.b), tf7.mod(op.inputs.a, op.inputs.b)))
};
tf7.registerKernel(kernelFloorMod);
env.kernels.push("floormod");
newKernels.push("floormod");
}
if (!env.kernels.includes("rotatewithoffset") && config3.softwareKernels) {
const kernelRotateWithOffset = {
kernelName: "RotateWithOffset",
backendName: tf7.getBackend(),
kernelFunc: (op) => tf7.tidy(() => {
const backend4 = tf7.getBackend();
tf7.setBackend("cpu");
const t2 = tf7.image.rotateWithOffset(op.inputs.image, op.attrs.radians, op.attrs.fillValue, op.attrs.center);
tf7.setBackend(backend4);
return t2;
})
};
tf7.registerKernel(kernelRotateWithOffset);
env.kernels.push("rotatewithoffset");
newKernels.push("rotatewithoffset");
}
if (newKernels.length > 0 && config3.debug)
log("registered kernels:", newKernels);
}
var defaultFlags = {};
async function check(instance, force = false) {
var _a;
instance.state = "backend";
if (((_a = instance.config.backend) == null ? void 0 : _a.length) === 0)
instance.config.backend = await getBestBackend();
if (force || env.initial || instance.config.backend && instance.config.backend.length > 0 && tf7.getBackend() !== instance.config.backend) {
const timeStamp = now();
if (instance.config.backend && instance.config.backend.length > 0) {
if (typeof window === "undefined" && typeof WorkerGlobalScope !== "undefined" && instance.config.debug) {
if (instance.config.debug)
log("running inside web worker");
}
if (env.browser && instance.config.backend === "tensorflow") {
if (instance.config.debug)
log("override: backend set to tensorflow while running in browser");
instance.config.backend = "webgl";
}
if (env.node && (instance.config.backend === "webgl" || instance.config.backend === "humangl")) {
if (instance.config.debug)
log(`override: backend set to ${instance.config.backend} while running in nodejs`);
instance.config.backend = "tensorflow";
}
if (env.browser && instance.config.backend === "webgpu") {
if (typeof navigator === "undefined" || typeof navigator.gpu === "undefined") {
log("override: backend set to webgpu but browser does not support webgpu");
instance.config.backend = "webgl";
} else {
const adapter = await navigator.gpu.requestAdapter();
if (instance.config.debug)
log("enumerated webgpu adapter:", adapter);
if (!adapter) {
log("override: backend set to webgpu but browser reports no available gpu");
instance.config.backend = "webgl";
} else {
const adapterInfo = "requestAdapterInfo" in adapter ? await adapter.requestAdapterInfo() : void 0;
log("webgpu adapter info:", adapterInfo);
}
}
}
let available = Object.keys(tf7.engine().registryFactory);
if (instance.config.backend === "humangl" && !available.includes("humangl")) {
register(instance);
available = Object.keys(tf7.engine().registryFactory);
}
if (instance.config.debug)
log("available backends:", available);
if (!available.includes(instance.config.backend)) {
log(`error: backend ${instance.config.backend} not found in registry`);
instance.config.backend = env.node ? "tensorflow" : "webgl";
if (instance.config.debug)
log(`override: setting backend ${instance.config.backend}`);
}
if (instance.config.debug)
log("setting backend:", [instance.config.backend]);
if (instance.config.backend === "wasm") {
if (tf7.env().flagRegistry.CANVAS2D_WILL_READ_FREQUENTLY)
tf7.env().set("CANVAS2D_WILL_READ_FREQUENTLY", true);
if (instance.config.debug)
log("wasm path:", instance.config.wasmPath);
if (typeof tf7.setWasmPaths !== "undefined")
tf7.setWasmPaths(instance.config.wasmPath, instance.config.wasmPlatformFetch);
else
throw new Error("backend error: attempting to use wasm backend but wasm path is not set");
let mt = false;
let simd = false;
try {
mt = await tf7.env().getAsync("WASM_HAS_MULTITHREAD_SUPPORT");
simd = await tf7.env().getAsync("WASM_HAS_SIMD_SUPPORT");
if (instance.config.debug)
log(`wasm execution: ${simd ? "simd" : "no simd"} ${mt ? "multithreaded" : "singlethreaded"}`);
if (instance.config.debug && !simd)
log("warning: wasm simd support is not enabled");
} catch (e) {
log("wasm detection failed");
}
}
try {
await tf7.setBackend(instance.config.backend);
await tf7.ready();
} catch (err) {
log("error: cannot set backend:", instance.config.backend, err);
return false;
}
if (instance.config.debug)
defaultFlags = JSON.parse(JSON.stringify(tf7.env().flags));
}
if (tf7.getBackend() === "humangl" || tf7.getBackend() === "webgl") {
if (tf7.env().flagRegistry.WEBGL_USE_SHAPES_UNIFORMS)
tf7.env().set("WEBGL_USE_SHAPES_UNIFORMS", true);
if (tf7.env().flagRegistry.WEBGL_EXP_CONV)
tf7.env().set("WEBGL_EXP_CONV", true);
if (instance.config.debug && typeof instance.config.deallocate !== "undefined" && instance.config.deallocate) {
log("changing webgl: WEBGL_DELETE_TEXTURE_THRESHOLD:", true);
tf7.env().set("WEBGL_DELETE_TEXTURE_THRESHOLD", 0);
}
}
if (tf7.getBackend() === "webgpu") {
}
if (instance.config.debug) {
const newFlags = tf7.env().flags;
const updatedFlags = {};
for (const key of Object.keys(newFlags)) {
if (defaultFlags[key] === newFlags[key])
continue;
updatedFlags[key] = newFlags[key];
}
if (instance.config.debug && Object.keys(updatedFlags).length > 0)
log("backend:", tf7.getBackend(), "flags:", updatedFlags);
}
if (instance.config.flags && Object.keys(instance.config.flags).length > 0) {
if (instance.config.debug)
log("flags:", instance.config["flags"]);
for (const [key, val] of Object.entries(instance.config.flags)) {
tf7.env().set(key, val);
}
}
tf7.enableProdMode();
init();
instance.performance.initBackend = Math.trunc(now() - timeStamp);
instance.config.backend = tf7.getBackend();
await env.updateBackend();
registerCustomOps(instance.config);
env.initial = false;
}
return true;
}
function fakeOps(kernelNames, config3) {
for (const kernelName of kernelNames) {
const kernelConfig = {
kernelName,
backendName: config3.backend,
kernelFunc: (param) => {
var _a;
if (config3.debug)
log("kernelFunc", kernelName, config3.backend, param);
return (_a = param == null ? void 0 : param.inputs) == null ? void 0 : _a.info;
}
};
tf7.registerKernel(kernelConfig);
}
env.kernels = tf7.getKernelsForBackend(tf7.getBackend()).map((kernel) => kernel.kernelName.toLowerCase());
}
// src/draw/draw.ts
var draw_exports = {};
__export(draw_exports, {
all: () => all,
body: () => body,
canvas: () => canvas2,
face: () => face,
gesture: () => gesture,
hand: () => hand,
init: () => init2,
object: () => object,
options: () => options2,
person: () => person
});
// src/draw/primitives.ts
var getCanvasContext = (input) => {
if (!input)
log("draw error: invalid canvas");
else if (!input.getContext)
log("draw error: canvas context not defined");
else {
const ctx = input.getContext("2d");
if (!ctx)
log("draw error: cannot get canvas context");
else
return ctx;
}
return null;
};
var rad2deg = (theta) => Math.round(theta * 180 / Math.PI);
var replace = (str, source, target) => str.replace(source, typeof target === "number" ? target.toFixed(1) : target);
var colorDepth = (z, opt) => {
if (!opt.useDepth || typeof z === "undefined")
return opt.color;
const rgb2 = Uint8ClampedArray.from([127 + 2 * z, 127 - 2 * z, 255]);
return `rgba(${rgb2[0]}, ${rgb2[1]}, ${rgb2[2]}, ${opt.alpha})`;
};
function labels(ctx, str, startX, startY, localOptions2) {
const line = str.replace(/\[.*\]/g, "").split("\n").map((l) => l.trim());
const x = Math.max(0, startX);
for (let i = line.length - 1; i >= 0; i--) {
const y = i * localOptions2.lineHeight + startY;
if (localOptions2.shadowColor && localOptions2.shadowColor !== "") {
ctx.fillStyle = localOptions2.shadowColor;
ctx.fillText(line[i], x + 5, y + 16);
}
ctx.fillStyle = localOptions2.labelColor;
ctx.fillText(line[i], x + 4, y + 15);
}
}
function point(ctx, x, y, z, localOptions2) {
ctx.fillStyle = colorDepth(z, localOptions2);
ctx.beginPath();
ctx.arc(x, y, localOptions2.pointSize, 0, 2 * Math.PI);
ctx.fill();
}
function rect(ctx, x, y, width, height, localOptions2) {
ctx.beginPath();
ctx.lineWidth = localOptions2.lineWidth;
if (localOptions2.useCurves) {
const cx = (x + x + width) / 2;
const cy = (y + y + height) / 2;
ctx.ellipse(cx, cy, width / 2, height / 2, 0, 0, 2 * Math.PI);
} else {
ctx.moveTo(x + localOptions2.roundRect, y);
ctx.lineTo(x + width - localOptions2.roundRect, y);
ctx.quadraticCurveTo(x + width, y, x + width, y + localOptions2.roundRect);
ctx.lineTo(x + width, y + height - localOptions2.roundRect);
ctx.quadraticCurveTo(x + width, y + height, x + width - localOptions2.roundRect, y + height);
ctx.lineTo(x + localOptions2.roundRect, y + height);
ctx.quadraticCurveTo(x, y + height, x, y + height - localOptions2.roundRect);
ctx.lineTo(x, y + localOptions2.roundRect);
ctx.quadraticCurveTo(x, y, x + localOptions2.roundRect, y);
ctx.closePath();
}
ctx.stroke();
}
function lines(ctx, points, localOptions2) {
if (points.length < 2)
return;
ctx.beginPath();
ctx.moveTo(points[0][0], points[0][1]);
for (const pt of points) {
ctx.strokeStyle = colorDepth(pt[2] || 0, localOptions2);
ctx.lineTo(Math.trunc(pt[0]), Math.trunc(pt[1]));
}
ctx.stroke();
if (localOptions2.fillPolygons) {
ctx.closePath();
ctx.fill();
}
}
function curves(ctx, points, localOptions2) {
if (points.length < 2)
return;
ctx.lineWidth = localOptions2.lineWidth;
if (!localOptions2.useCurves || points.length <= 2) {
lines(ctx, points, localOptions2);
return;
}
ctx.moveTo(points[0][0], points[0][1]);
for (let i = 0; i < points.length - 2; i++) {
const xc = (points[i][0] + points[i + 1][0]) / 2;
const yc = (points[i][1] + points[i + 1][1]) / 2;
ctx.quadraticCurveTo(points[i][0], points[i][1], xc, yc);
}
ctx.quadraticCurveTo(points[points.length - 2][0], points[points.length - 2][1], points[points.length - 1][0], points[points.length - 1][1]);
ctx.stroke();
if (localOptions2.fillPolygons) {
ctx.closePath();
ctx.fill();
}
}
function arrow(ctx, from, to, radius = 5) {
let angle;
let x;
let y;
ctx.beginPath();
ctx.moveTo(from[0], from[1]);
ctx.lineTo(to[0], to[1]);
angle = Math.atan2(to[1] - from[1], to[0] - from[0]);
x = radius * Math.cos(angle) + to[0];
y = radius * Math.sin(angle) + to[1];
ctx.moveTo(x, y);
angle += 1 / 3 * (2 * Math.PI);
x = radius * Math.cos(angle) + to[0];
y = radius * Math.sin(angle) + to[1];
ctx.lineTo(x, y);
angle += 1 / 3 * (2 * Math.PI);
x = radius * Math.cos(angle) + to[0];
y = radius * Math.sin(angle) + to[1];
ctx.lineTo(x, y);
ctx.closePath();
ctx.stroke();
ctx.fill();
}
// src/draw/options.ts
var options2 = {
color: "rgba(173, 216, 230, 0.6)",
labelColor: "rgba(173, 216, 230, 1)",
shadowColor: "black",
alpha: 0.5,
font: 'small-caps 16px "Segoe UI"',
lineHeight: 18,
lineWidth: 4,
pointSize: 2,
roundRect: 8,
drawPoints: false,
drawLabels: true,
drawBoxes: true,
drawAttention: true,
drawGestures: true,
drawPolygons: true,
drawGaze: true,
fillPolygons: false,
useDepth: true,
useCurves: false,
faceLabels: "",
bodyLabels: "",
bodyPartLabels: "",
objectLabels: "",
handLabels: "",
fingerLabels: "",
gestureLabels: ""
};
// src/face/facemeshcoords.ts
var meshAnnotations = {
silhouette: [
10,
338,
297,
332,
284,
251,
389,
356,
454,
323,
361,
288,
397,
365,
379,
378,
400,
377,
152,
148,
176,
149,
150,
136,
172,
58,
132,
93,
234,
127,
162,
21,
54,
103,
67,
109
],
lipsUpperOuter: [185, 40, 39, 37, 0, 267, 269, 270, 409],
lipsLowerOuter: [61, 146, 91, 181, 84, 17, 314, 405, 321, 375, 291],
lipsUpperInner: [191, 80, 81, 82, 13, 312, 311, 310, 415],
lipsLowerInner: [78, 95, 88, 178, 87, 14, 317, 402, 318, 324, 308],
lipsLowerSemiOuter: [76, 77, 90, 180, 85, 16, 315, 404, 320, 307, 306],
lipsUpperSemiOuter: [184, 74, 73, 72, 11, 302, 303, 304, 408],
lipsLowerSemiInner: [62, 96, 89, 179, 86, 15, 316, 403, 319, 325, 292],
lipsUpperSemiInner: [183, 42, 41, 38, 12, 268, 271, 272, 407],
rightEyeUpper0: [246, 161, 160, 159, 158, 157, 173],
rightEyeLower0: [33, 7, 163, 144, 145, 153, 154, 155, 133],
rightEyeUpper1: [247, 30, 29, 27, 28, 56, 190],
rightEyeLower1: [130, 25, 110, 24, 23, 22, 26, 112, 243],
rightEyeUpper2: [113, 225, 224, 223, 222, 221, 189],
rightEyeLower2: [226, 31, 228, 229, 230, 231, 232, 233, 244],
rightEyeLower3: [143, 111, 117, 118, 119, 120, 121, 128, 245],
rightEyebrowUpper: [156, 70, 63, 105, 66, 107, 55, 193],
rightEyebrowLower: [35, 124, 46, 53, 52, 65],
rightEyeIris: [473, 474, 475, 476, 477],
leftEyeUpper0: [466, 388, 387, 386, 385, 384, 398],
leftEyeLower0: [263, 249, 390, 373, 374, 380, 381, 382, 362],
leftEyeUpper1: [467, 260, 259, 257, 258, 286, 414],
leftEyeLower1: [359, 255, 339, 254, 253, 252, 256, 341, 463],
leftEyeUpper2: [342, 445, 444, 443, 442, 441, 413],
leftEyeLower2: [446, 261, 448, 449, 450, 451, 452, 453, 464],
leftEyeLower3: [372, 340, 346, 347, 348, 349, 350, 357, 465],
leftEyebrowUpper: [383, 300, 293, 334, 296, 336, 285, 417],
leftEyebrowLower: [265, 353, 276, 283, 282, 295],
leftEyeIris: [468, 469, 470, 471, 472],
midwayBetweenEyes: [168],
noseTip: [1],
noseBottom: [2],
noseRightCorner: [98],
noseLeftCorner: [327],
rightCheek: [205],
leftCheek: [425]
};
var meshLandmarks = {
count: 468,
mouth: 13,
symmetryLine: [13, meshAnnotations.midwayBetweenEyes[0]]
};
var blazeFaceLandmarks = {
leftEye: 0,
rightEye: 1,
nose: 2,
mouth: 3,
leftEar: 4,
rightEar: 5,
symmetryLine: [3, 2]
};
var irisIndices = [
{ key: "EyeUpper0", indices: [9, 10, 11, 12, 13, 14, 15] },
{ key: "EyeUpper1", indices: [25, 26, 27, 28, 29, 30, 31] },
{ key: "EyeUpper2", indices: [41, 42, 43, 44, 45, 46, 47] },
{ key: "EyeLower0", indices: [0, 1, 2, 3, 4, 5, 6, 7, 8] },
{ key: "EyeLower1", indices: [16, 17, 18, 19, 20, 21, 22, 23, 24] },
{ key: "EyeLower2", indices: [32, 33, 34, 35, 36, 37, 38, 39, 40] },
{ key: "EyeLower3", indices: [54, 55, 56, 57, 58, 59, 60, 61, 62] },
{ key: "EyebrowUpper", indices: [63, 64, 65, 66, 67, 68, 69, 70] },
{ key: "EyebrowLower", indices: [48, 49, 50, 51, 52, 53] }
];
var UV468 = [
[0.499976992607117, 0.652534008026123],
[0.500025987625122, 0.547487020492554],
[0.499974012374878, 0.602371990680695],
[0.482113003730774, 0.471979022026062],
[0.500150978565216, 0.527155995368958],
[0.499909996986389, 0.498252987861633],
[0.499523013830185, 0.40106201171875],
[0.289712011814117, 0.380764007568359],
[0.499954998493195, 0.312398016452789],
[0.499987006187439, 0.269918978214264],
[0.500023007392883, 0.107050001621246],
[0.500023007392883, 0.666234016418457],
[0.5000159740448, 0.679224014282227],
[0.500023007392883, 0.692348003387451],
[0.499976992607117, 0.695277988910675],
[0.499976992607117, 0.70593398809433],
[0.499976992607117, 0.719385027885437],
[0.499976992607117, 0.737019002437592],
[0.499967992305756, 0.781370997428894],
[0.499816000461578, 0.562981009483337],
[0.473773002624512, 0.573909997940063],
[0.104906998574734, 0.254140973091125],
[0.365929991006851, 0.409575998783112],
[0.338757991790771, 0.41302502155304],
[0.311120003461838, 0.409460008144379],
[0.274657994508743, 0.389131009578705],
[0.393361985683441, 0.403706014156342],
[0.345234006643295, 0.344011008739471],
[0.370094001293182, 0.346076011657715],
[0.319321990013123, 0.347265005111694],
[0.297903001308441, 0.353591024875641],
[0.24779200553894, 0.410809993743896],
[0.396889001131058, 0.842755019664764],
[0.280097991228104, 0.375599980354309],
[0.106310002505779, 0.399955987930298],
[0.2099249958992, 0.391353011131287],
[0.355807989835739, 0.534406006336212],
[0.471751004457474, 0.65040397644043],
[0.474155008792877, 0.680191993713379],
[0.439785003662109, 0.657229006290436],
[0.414617002010345, 0.66654098033905],
[0.450374007225037, 0.680860996246338],
[0.428770989179611, 0.682690978050232],
[0.374971002340317, 0.727805018424988],
[0.486716985702515, 0.547628998756409],
[0.485300987958908, 0.527395009994507],
[0.257764995098114, 0.314490020275116],
[0.401223003864288, 0.455172002315521],
[0.429818987846375, 0.548614978790283],
[0.421351999044418, 0.533740997314453],
[0.276895999908447, 0.532056987285614],
[0.483370006084442, 0.499586999416351],
[0.33721199631691, 0.282882988452911],
[0.296391993761063, 0.293242990970612],
[0.169294998049736, 0.193813979625702],
[0.447580009698868, 0.302609980106354],
[0.392390012741089, 0.353887975215912],
[0.354490011930466, 0.696784019470215],
[0.067304998636246, 0.730105042457581],
[0.442739009857178, 0.572826027870178],
[0.457098007202148, 0.584792017936707],
[0.381974011659622, 0.694710969924927],
[0.392388999462128, 0.694203019142151],
[0.277076005935669, 0.271932005882263],
[0.422551989555359, 0.563233017921448],
[0.385919004678726, 0.281364023685455],
[0.383103013038635, 0.255840003490448],
[0.331431001424789, 0.119714021682739],
[0.229923993349075, 0.232002973556519],
[0.364500999450684, 0.189113974571228],
[0.229622006416321, 0.299540996551514],
[0.173287004232407, 0.278747975826263],
[0.472878992557526, 0.666198015213013],
[0.446828007698059, 0.668527007102966],
[0.422762006521225, 0.673889994621277],
[0.445307999849319, 0.580065965652466],
[0.388103008270264, 0.693961024284363],
[0.403039008378983, 0.706539988517761],
[0.403629004955292, 0.693953037261963],
[0.460041999816895, 0.557139039039612],
[0.431158006191254, 0.692366003990173],
[0.452181994915009, 0.692366003990173],
[0.475387006998062, 0.692366003990173],
[0.465828001499176, 0.779190003871918],
[0.472328990697861, 0.736225962638855],
[0.473087012767792, 0.717857003211975],
[0.473122000694275, 0.704625964164734],
[0.473033010959625, 0.695277988910675],
[0.427942007780075, 0.695277988910675],
[0.426479011774063, 0.703539967536926],
[0.423162013292313, 0.711845993995667],
[0.4183090031147, 0.720062971115112],
[0.390094995498657, 0.639572978019714],
[0.013953999616206, 0.560034036636353],
[0.499913990497589, 0.58014702796936],
[0.413199990987778, 0.69539999961853],
[0.409626007080078, 0.701822996139526],
[0.468080013990402, 0.601534962654114],
[0.422728985548019, 0.585985004901886],
[0.463079988956451, 0.593783974647522],
[0.37211999297142, 0.47341400384903],
[0.334562003612518, 0.496073007583618],
[0.411671012639999, 0.546965003013611],
[0.242175996303558, 0.14767599105835],
[0.290776997804642, 0.201445996761322],
[0.327338010072708, 0.256527006626129],
[0.399509996175766, 0.748921036720276],
[0.441727995872498, 0.261676013469696],
[0.429764986038208, 0.187834024429321],
[0.412198007106781, 0.108901023864746],
[0.288955003023148, 0.398952007293701],
[0.218936994671822, 0.435410976409912],
[0.41278201341629, 0.398970007896423],
[0.257135003805161, 0.355440020561218],
[0.427684992551804, 0.437960982322693],
[0.448339998722076, 0.536936044692993],
[0.178560003638268, 0.45755398273468],
[0.247308000922203, 0.457193970680237],
[0.286267012357712, 0.467674970626831],
[0.332827985286713, 0.460712015628815],
[0.368755996227264, 0.447206974029541],
[0.398963987827301, 0.432654976844788],
[0.476410001516342, 0.405806005001068],
[0.189241006970406, 0.523923993110657],
[0.228962004184723, 0.348950982093811],
[0.490725994110107, 0.562400996685028],
[0.404670000076294, 0.485132992267609],
[0.019469000399113, 0.401564002037048],
[0.426243007183075, 0.420431017875671],
[0.396993011236191, 0.548797011375427],
[0.266469985246658, 0.376977026462555],
[0.439121007919312, 0.51895797252655],
[0.032313998788595, 0.644356966018677],
[0.419054001569748, 0.387154996395111],
[0.462783008813858, 0.505746960639954],
[0.238978996872902, 0.779744982719421],
[0.198220998048782, 0.831938028335571],
[0.107550002634525, 0.540755033493042],
[0.183610007166862, 0.740257024765015],
[0.134409993886948, 0.333683013916016],
[0.385764002799988, 0.883153975009918],
[0.490967005491257, 0.579378008842468],
[0.382384985685349, 0.508572995662689],
[0.174399003386497, 0.397670984268188],
[0.318785011768341, 0.39623498916626],
[0.343364000320435, 0.400596976280212],
[0.396100014448166, 0.710216999053955],
[0.187885001301765, 0.588537991046906],
[0.430987000465393, 0.944064974784851],
[0.318993002176285, 0.898285031318665],
[0.266247987747192, 0.869701027870178],
[0.500023007392883, 0.190576016902924],
[0.499976992607117, 0.954452991485596],
[0.366169989109039, 0.398822009563446],
[0.393207013607025, 0.39553701877594],
[0.410373002290726, 0.391080021858215],
[0.194993004202843, 0.342101991176605],
[0.388664990663528, 0.362284004688263],
[0.365961998701096, 0.355970978736877],
[0.343364000320435, 0.355356991291046],
[0.318785011768341, 0.35834002494812],
[0.301414996385574, 0.363156020641327],
[0.058132998645306, 0.319076001644135],
[0.301414996385574, 0.387449026107788],
[0.499987989664078, 0.618434011936188],
[0.415838003158569, 0.624195992946625],
[0.445681989192963, 0.566076993942261],
[0.465844005346298, 0.620640993118286],
[0.49992299079895, 0.351523995399475],
[0.288718998432159, 0.819945991039276],
[0.335278987884521, 0.852819979190826],
[0.440512001514435, 0.902418971061707],
[0.128294005990028, 0.791940987110138],
[0.408771991729736, 0.373893976211548],
[0.455606997013092, 0.451801002025604],
[0.499877005815506, 0.908990025520325],
[0.375436991453171, 0.924192011356354],
[0.11421000212431, 0.615022003650665],
[0.448662012815475, 0.695277988910675],
[0.4480200111866, 0.704632043838501],
[0.447111994028091, 0.715808033943176],
[0.444831997156143, 0.730794012546539],
[0.430011987686157, 0.766808986663818],
[0.406787008047104, 0.685672998428345],
[0.400738000869751, 0.681069016456604],
[0.392399996519089, 0.677703022956848],
[0.367855995893478, 0.663918972015381],
[0.247923001646996, 0.601333022117615],
[0.452769994735718, 0.420849978923798],
[0.43639200925827, 0.359887003898621],
[0.416164010763168, 0.368713974952698],
[0.413385987281799, 0.692366003990173],
[0.228018000721931, 0.683571994304657],
[0.468268007040024, 0.352671027183533],
[0.411361992359161, 0.804327011108398],
[0.499989002943039, 0.469825029373169],
[0.479153990745544, 0.442654013633728],
[0.499974012374878, 0.439637005329132],
[0.432112008333206, 0.493588984012604],
[0.499886006116867, 0.866917014122009],
[0.49991300702095, 0.821729004383087],
[0.456548988819122, 0.819200992584229],
[0.344549000263214, 0.745438992977142],
[0.37890899181366, 0.574010014533997],
[0.374292999505997, 0.780184984207153],
[0.319687992334366, 0.570737957954407],
[0.357154995203018, 0.604269981384277],
[0.295284003019333, 0.621580958366394],
[0.447750002145767, 0.862477004528046],
[0.410986006259918, 0.508723020553589],
[0.31395098567009, 0.775308012962341],
[0.354128003120422, 0.812552988529205],
[0.324548006057739, 0.703992962837219],
[0.189096003770828, 0.646299958229065],
[0.279776990413666, 0.71465802192688],
[0.1338230073452, 0.682700991630554],
[0.336768001317978, 0.644733011722565],
[0.429883986711502, 0.466521978378296],
[0.455527991056442, 0.548622965812683],
[0.437114000320435, 0.558896005153656],
[0.467287987470627, 0.529924988746643],
[0.414712011814117, 0.335219979286194],
[0.37704598903656, 0.322777986526489],
[0.344107985496521, 0.320150971412659],
[0.312875986099243, 0.32233202457428],
[0.283526003360748, 0.333190023899078],
[0.241245999932289, 0.382785975933075],
[0.102986000478268, 0.468762993812561],
[0.267612010240555, 0.424560010433197],
[0.297879010438919, 0.433175981044769],
[0.333433985710144, 0.433878004550934],
[0.366427004337311, 0.426115989685059],
[0.396012008190155, 0.416696012020111],
[0.420121014118195, 0.41022801399231],
[0.007561000064015, 0.480777025222778],
[0.432949006557465, 0.569517970085144],
[0.458638995885849, 0.479089021682739],
[0.473466008901596, 0.545744001865387],
[0.476087987422943, 0.563830018043518],
[0.468472003936768, 0.555056989192963],
[0.433990985155106, 0.582361996173859],
[0.483518004417419, 0.562983989715576],
[0.482482999563217, 0.57784903049469],
[0.42645001411438, 0.389798998832703],
[0.438998997211456, 0.39649498462677],
[0.450067013502121, 0.400434017181396],
[0.289712011814117, 0.368252992630005],
[0.276670008897781, 0.363372981548309],
[0.517862021923065, 0.471948027610779],
[0.710287988185883, 0.380764007568359],
[0.526226997375488, 0.573909997940063],
[0.895093023777008, 0.254140973091125],
[0.634069979190826, 0.409575998783112],
[0.661242008209229, 0.41302502155304],
[0.688880026340485, 0.409460008144379],
[0.725341975688934, 0.389131009578705],
[0.606630027294159, 0.40370500087738],
[0.654766023159027, 0.344011008739471],
[0.629905998706818, 0.346076011657715],
[0.680678009986877, 0.347265005111694],
[0.702096998691559, 0.353591024875641],
[0.75221198797226, 0.410804986953735],
[0.602918028831482, 0.842862963676453],
[0.719901978969574, 0.375599980354309],
[0.893692970275879, 0.399959981441498],
[0.790081977844238, 0.391354024410248],
[0.643998026847839, 0.534487962722778],
[0.528249025344849, 0.65040397644043],
[0.525849997997284, 0.680191040039062],
[0.560214996337891, 0.657229006290436],
[0.585384011268616, 0.66654098033905],
[0.549625992774963, 0.680860996246338],
[0.57122802734375, 0.682691991329193],
[0.624852001667023, 0.72809898853302],
[0.513050019741058, 0.547281980514526],
[0.51509702205658, 0.527251958847046],
[0.742246985435486, 0.314507007598877],
[0.598631024360657, 0.454979002475739],
[0.570338010787964, 0.548575043678284],
[0.578631997108459, 0.533622980117798],
[0.723087012767792, 0.532054007053375],
[0.516445994377136, 0.499638974666595],
[0.662801027297974, 0.282917976379395],
[0.70362401008606, 0.293271005153656],
[0.830704987049103, 0.193813979625702],
[0.552385985851288, 0.302568018436432],
[0.607609987258911, 0.353887975215912],
[0.645429015159607, 0.696707010269165],
[0.932694971561432, 0.730105042457581],
[0.557260990142822, 0.572826027870178],
[0.542901992797852, 0.584792017936707],
[0.6180260181427, 0.694710969924927],
[0.607590973377228, 0.694203019142151],
[0.722943007946014, 0.271963000297546],
[0.577413976192474, 0.563166975975037],
[0.614082992076874, 0.281386971473694],
[0.616907000541687, 0.255886018276215],
[0.668509006500244, 0.119913995265961],
[0.770092010498047, 0.232020974159241],
[0.635536015033722, 0.189248979091644],
[0.77039098739624, 0.299556016921997],
[0.826722025871277, 0.278755009174347],
[0.527121007442474, 0.666198015213013],
[0.553171992301941, 0.668527007102966],
[0.577238023281097, 0.673889994621277],
[0.554691970348358, 0.580065965652466],
[0.611896991729736, 0.693961024284363],
[0.59696102142334, 0.706539988517761],
[0.596370995044708, 0.693953037261963],
[0.539958000183105, 0.557139039039612],
[0.568841993808746, 0.692366003990173],
[0.547818005084991, 0.692366003990173],
[0.52461302280426, 0.692366003990173],
[0.534089982509613, 0.779141008853912],
[0.527670979499817, 0.736225962638855],
[0.526912987232208, 0.717857003211975],
[0.526877999305725, 0.704625964164734],
[0.526966989040375, 0.695277988910675],
[0.572058022022247, 0.695277988910675],
[0.573521018028259, 0.703539967536926],
[0.57683801651001, 0.711845993995667],
[0.581691026687622, 0.720062971115112],
[0.609944999217987, 0.639909982681274],
[0.986046016216278, 0.560034036636353],
[0.5867999792099, 0.69539999961853],
[0.590372025966644, 0.701822996139526],
[0.531915009021759, 0.601536989212036],
[0.577268004417419, 0.585934996604919],
[0.536915004253387, 0.593786001205444],
[0.627542972564697, 0.473352015018463],
[0.665585994720459, 0.495950996875763],
[0.588353991508484, 0.546862006187439],
[0.757824003696442, 0.14767599105835],
[0.709249973297119, 0.201507985591888],
[0.672684013843536, 0.256581008434296],
[0.600408971309662, 0.74900496006012],
[0.55826598405838, 0.261672019958496],
[0.570303976535797, 0.187870979309082],
[0.588165998458862, 0.109044015407562],
[0.711045026779175, 0.398952007293701],
[0.781069993972778, 0.435405015945435],
[0.587247014045715, 0.398931980133057],
[0.742869973182678, 0.355445981025696],
[0.572156012058258, 0.437651991844177],
[0.55186802148819, 0.536570012569427],
[0.821442008018494, 0.457556009292603],
[0.752701997756958, 0.457181990146637],
[0.71375697851181, 0.467626988887787],
[0.66711300611496, 0.460672974586487],
[0.631101012229919, 0.447153985500336],
[0.6008620262146, 0.432473003864288],
[0.523481011390686, 0.405627012252808],
[0.810747981071472, 0.523926019668579],
[0.771045982837677, 0.348959028720856],
[0.509127020835876, 0.562718033790588],
[0.595292985439301, 0.485023975372314],
[0.980530977249146, 0.401564002037048],
[0.573499977588654, 0.420000016689301],
[0.602994978427887, 0.548687994480133],
[0.733529984951019, 0.376977026462555],
[0.560611009597778, 0.519016981124878],
[0.967685997486115, 0.644356966018677],
[0.580985009670258, 0.387160003185272],
[0.537728011608124, 0.505385041236877],
[0.760966002941132, 0.779752969741821],
[0.801778972148895, 0.831938028335571],
[0.892440974712372, 0.54076099395752],
[0.816350996494293, 0.740260004997253],
[0.865594983100891, 0.333687007427216],
[0.614073991775513, 0.883246004581451],
[0.508952975273132, 0.579437971115112],
[0.617941975593567, 0.508316040039062],
[0.825608015060425, 0.397674977779388],
[0.681214988231659, 0.39623498916626],
[0.656635999679565, 0.400596976280212],
[0.603900015354156, 0.710216999053955],
[0.81208598613739, 0.588539004325867],
[0.56801301240921, 0.944564998149872],
[0.681007981300354, 0.898285031318665],
[0.733752012252808, 0.869701027870178],
[0.633830010890961, 0.398822009563446],
[0.606792986392975, 0.39553701877594],
[0.589659988880157, 0.391062021255493],
[0.805015981197357, 0.342108011245728],
[0.611334979534149, 0.362284004688263],
[0.634037971496582, 0.355970978736877],
[0.656635999679565, 0.355356991291046],
[0.681214988231659, 0.35834002494812],
[0.698584973812103, 0.363156020641327],
[0.941866993904114, 0.319076001644135],
[0.698584973812103, 0.387449026107788],
[0.584177017211914, 0.624107003211975],
[0.554318010807037, 0.566076993942261],
[0.534153997898102, 0.62064003944397],
[0.711217999458313, 0.819975018501282],
[0.664629995822906, 0.852871000766754],
[0.559099972248077, 0.902631998062134],
[0.871706008911133, 0.791940987110138],
[0.591234028339386, 0.373893976211548],
[0.544341027736664, 0.451583981513977],
[0.624562978744507, 0.924192011356354],
[0.88577002286911, 0.615028977394104],
[0.551338016986847, 0.695277988910675],
[0.551980018615723, 0.704632043838501],
[0.552887976169586, 0.715808033943176],
[0.555167973041534, 0.730794012546539],
[0.569944024085999, 0.767035007476807],
[0.593203008174896, 0.685675978660583],
[0.599261999130249, 0.681069016456604],
[0.607599973678589, 0.677703022956848],
[0.631937980651855, 0.663500010967255],
[0.752032995223999, 0.601315021514893],
[0.547226011753082, 0.420395016670227],
[0.563543975353241, 0.359827995300293],
[0.583841025829315, 0.368713974952698],
[0.586614012718201, 0.692366003990173],
[0.771915018558502, 0.683578014373779],
[0.531597018241882, 0.352482974529266],
[0.588370978832245, 0.804440975189209],
[0.52079701423645, 0.442565023899078],
[0.567984998226166, 0.493479013442993],
[0.543282985687256, 0.819254994392395],
[0.655317008495331, 0.745514988899231],
[0.621008992195129, 0.574018001556396],
[0.625559985637665, 0.78031200170517],
[0.680198013782501, 0.570719003677368],
[0.64276397228241, 0.604337990283966],
[0.704662978649139, 0.621529996395111],
[0.552012026309967, 0.862591981887817],
[0.589071989059448, 0.508637011051178],
[0.685944974422455, 0.775357007980347],
[0.645735025405884, 0.812640011310577],
[0.675342977046967, 0.703978002071381],
[0.810858011245728, 0.646304965019226],
[0.72012197971344, 0.714666962623596],
[0.866151988506317, 0.682704985141754],
[0.663187026977539, 0.644596993923187],
[0.570082008838654, 0.466325998306274],
[0.544561982154846, 0.548375964164734],
[0.562758982181549, 0.558784961700439],
[0.531987011432648, 0.530140042304993],
[0.585271000862122, 0.335177004337311],
[0.622952997684479, 0.32277899980545],
[0.655896008014679, 0.320163011550903],
[0.687132000923157, 0.322345972061157],
[0.716481983661652, 0.333200991153717],
[0.758756995201111, 0.382786989212036],
[0.897013008594513, 0.468769013881683],
[0.732392013072968, 0.424547016620636],
[0.70211398601532, 0.433162987232208],
[0.66652500629425, 0.433866024017334],
[0.633504986763, 0.426087975502014],
[0.603875994682312, 0.416586995124817],
[0.579657971858978, 0.409945011138916],
[0.992439985275269, 0.480777025222778],
[0.567192018032074, 0.569419980049133],
[0.54136598110199, 0.478899002075195],
[0.526564002037048, 0.546118021011353],
[0.523913025856018, 0.563830018043518],
[0.531529009342194, 0.555056989192963],
[0.566035985946655, 0.582329034805298],
[0.51631098985672, 0.563053965568542],
[0.5174720287323, 0.577877044677734],
[0.573594987392426, 0.389806985855103],
[0.560697972774506, 0.395331978797913],
[0.549755990505219, 0.399751007556915],
[0.710287988185883, 0.368252992630005],
[0.723330020904541, 0.363372981548309]
];
var TRI468 = [
127,
34,
139,
11,
0,
37,
232,
231,
120,
72,
37,
39,
128,
121,
47,
232,
121,
128,
104,
69,
67,
175,
171,
148,
157,
154,
155,
118,
50,
101,
73,
39,
40,
9,
151,
108,
48,
115,
131,
194,
204,
211,
74,
40,
185,
80,
42,
183,
40,
92,
186,
230,
229,
118,
202,
212,
214,
83,
18,
17,
76,
61,
146,
160,
29,
30,
56,
157,
173,
106,
204,
194,
135,
214,
192,
203,
165,
98,
21,
71,
68,
51,
45,
4,
144,
24,
23,
77,
146,
91,
205,
50,
187,
201,
200,
18,
91,
106,
182,
90,
91,
181,
85,
84,
17,
206,
203,
36,
148,
171,
140,
92,
40,
39,
193,
189,
244,
159,
158,
28,
247,
246,
161,
236,
3,
196,
54,
68,
104,
193,
168,
8,
117,
228,
31,
189,
193,
55,
98,
97,
99,
126,
47,
100,
166,
79,
218,
155,
154,
26,
209,
49,
131,
135,
136,
150,
47,
126,
217,
223,
52,
53,
45,
51,
134,
211,
170,
140,
67,
69,
108,
43,
106,
91,
230,
119,
120,
226,
130,
247,
63,
53,
52,
238,
20,
242,
46,
70,
156,
78,
62,
96,
46,
53,
63,
143,
34,
227,
173,
155,
133,
123,
117,
111,
44,
125,
19,
236,
134,
51,
216,
206,
205,
154,
153,
22,
39,
37,
167,
200,
201,
208,
36,
142,
100,
57,
212,
202,
20,
60,
99,
28,
158,
157,
35,
226,
113,
160,
159,
27,
204,
202,
210,
113,
225,
46,
43,
202,
204,
62,
76,
77,
137,
123,
116,
41,
38,
72,
203,
129,
142,
64,
98,
240,
49,
102,
64,
41,
73,
74,
212,
216,
207,
42,
74,
184,
169,
170,
211,
170,
149,
176,
105,
66,
69,
122,
6,
168,
123,
147,
187,
96,
77,
90,
65,
55,
107,
89,
90,
180,
101,
100,
120,
63,
105,
104,
93,
137,
227,
15,
86,
85,
129,
102,
49,
14,
87,
86,
55,
8,
9,
100,
47,
121,
145,
23,
22,
88,
89,
179,
6,
122,
196,
88,
95,
96,
138,
172,
136,
215,
58,
172,
115,
48,
219,
42,
80,
81,
195,
3,
51,
43,
146,
61,
171,
175,
199,
81,
82,
38,
53,
46,
225,
144,
163,
110,
246,
33,
7,
52,
65,
66,
229,
228,
117,
34,
127,
234,
107,
108,
69,
109,
108,
151,
48,
64,
235,
62,
78,
191,
129,
209,
126,
111,
35,
143,
163,
161,
246,
117,
123,
50,
222,
65,
52,
19,
125,
141,
221,
55,
65,
3,
195,
197,
25,
7,
33,
220,
237,
44,
70,
71,
139,
122,
193,
245,
247,
130,
33,
71,
21,
162,
153,
158,
159,
170,
169,
150,
188,
174,
196,
216,
186,
92,
144,
160,
161,
2,
97,
167,
141,
125,
241,
164,
167,
37,
72,
38,
12,
145,
159,
160,
38,
82,
13,
63,
68,
71,
226,
35,
111,
158,
153,
154,
101,
50,
205,
206,
92,
165,
209,
198,
217,
165,
167,
97,
220,
115,
218,
133,
112,
243,
239,
238,
241,
214,
135,
169,
190,
173,
133,
171,
208,
32,
125,
44,
237,
86,
87,
178,
85,
86,
179,
84,
85,
180,
83,
84,
181,
201,
83,
182,
137,
93,
132,
76,
62,
183,
61,
76,
184,
57,
61,
185,
212,
57,
186,
214,
207,
187,
34,
143,
156,
79,
239,
237,
123,
137,
177,
44,
1,
4,
201,
194,
32,
64,
102,
129,
213,
215,
138,
59,
166,
219,
242,
99,
97,
2,
94,
141,
75,
59,
235,
24,
110,
228,
25,
130,
226,
23,
24,
229,
22,
23,
230,
26,
22,
231,
112,
26,
232,
189,
190,
243,
221,
56,
190,
28,
56,
221,
27,
28,
222,
29,
27,
223,
30,
29,
224,
247,
30,
225,
238,
79,
20,
166,
59,
75,
60,
75,
240,
147,
177,
215,
20,
79,
166,
187,
147,
213,
112,
233,
244,
233,
128,
245,
128,
114,
188,
114,
217,
174,
131,
115,
220,
217,
198,
236,
198,
131,
134,
177,
132,
58,
143,
35,
124,
110,
163,
7,
228,
110,
25,
356,
389,
368,
11,
302,
267,
452,
350,
349,
302,
303,
269,
357,
343,
277,
452,
453,
357,
333,
332,
297,
175,
152,
377,
384,
398,
382,
347,
348,
330,
303,
304,
270,
9,
336,
337,
278,
279,
360,
418,
262,
431,
304,
408,
409,
310,
415,
407,
270,
409,
410,
450,
348,
347,
422,
430,
434,
313,
314,
17,
306,
307,
375,
387,
388,
260,
286,
414,
398,
335,
406,
418,
364,
367,
416,
423,
358,
327,
251,
284,
298,
281,
5,
4,
373,
374,
253,
307,
320,
321,
425,
427,
411,
421,
313,
18,
321,
405,
406,
320,
404,
405,
315,
16,
17,
426,
425,
266,
377,
400,
369,
322,
391,
269,
417,
465,
464,
386,
257,
258,
466,
260,
388,
456,
399,
419,
284,
332,
333,
417,
285,
8,
346,
340,
261,
413,
441,
285,
327,
460,
328,
355,
371,
329,
392,
439,
438,
382,
341,
256,
429,
420,
360,
364,
394,
379,
277,
343,
437,
443,
444,
283,
275,
440,
363,
431,
262,
369,
297,
338,
337,
273,
375,
321,
450,
451,
349,
446,
342,
467,
293,
334,
282,
458,
461,
462,
276,
353,
383,
308,
324,
325,
276,
300,
293,
372,
345,
447,
382,
398,
362,
352,
345,
340,
274,
1,
19,
456,
248,
281,
436,
427,
425,
381,
256,
252,
269,
391,
393,
200,
199,
428,
266,
330,
329,
287,
273,
422,
250,
462,
328,
258,
286,
384,
265,
353,
342,
387,
259,
257,
424,
431,
430,
342,
353,
276,
273,
335,
424,
292,
325,
307,
366,
447,
345,
271,
303,
302,
423,
266,
371,
294,
455,
460,
279,
278,
294,
271,
272,
304,
432,
434,
427,
272,
407,
408,
394,
430,
431,
395,
369,
400,
334,
333,
299,
351,
417,
168,
352,
280,
411,
325,
319,
320,
295,
296,
336,
319,
403,
404,
330,
348,
349,
293,
298,
333,
323,
454,
447,
15,
16,
315,
358,
429,
279,
14,
15,
316,
285,
336,
9,
329,
349,
350,
374,
380,
252,
318,
402,
403,
6,
197,
419,
318,
319,
325,
367,
364,
365,
435,
367,
397,
344,
438,
439,
272,
271,
311,
195,
5,
281,
273,
287,
291,
396,
428,
199,
311,
271,
268,
283,
444,
445,
373,
254,
339,
263,
466,
249,
282,
334,
296,
449,
347,
346,
264,
447,
454,
336,
296,
299,
338,
10,
151,
278,
439,
455,
292,
407,
415,
358,
371,
355,
340,
345,
372,
390,
249,
466,
346,
347,
280,
442,
443,
282,
19,
94,
370,
441,
442,
295,
248,
419,
197,
263,
255,
359,
440,
275,
274,
300,
383,
368,
351,
412,
465,
263,
467,
466,
301,
368,
389,
380,
374,
386,
395,
378,
379,
412,
351,
419,
436,
426,
322,
373,
390,
388,
2,
164,
393,
370,
462,
461,
164,
0,
267,
302,
11,
12,
374,
373,
387,
268,
12,
13,
293,
300,
301,
446,
261,
340,
385,
384,
381,
330,
266,
425,
426,
423,
391,
429,
355,
437,
391,
327,
326,
440,
457,
438,
341,
382,
362,
459,
457,
461,
434,
430,
394,
414,
463,
362,
396,
369,
262,
354,
461,
457,
316,
403,
402,
315,
404,
403,
314,
405,
404,
313,
406,
405,
421,
418,
406,
366,
401,
361,
306,
408,
407,
291,
409,
408,
287,
410,
409,
432,
436,
410,
434,
416,
411,
264,
368,
383,
309,
438,
457,
352,
376,
401,
274,
275,
4,
421,
428,
262,
294,
327,
358,
433,
416,
367,
289,
455,
439,
462,
370,
326,
2,
326,
370,
305,
460,
455,
254,
449,
448,
255,
261,
446,
253,
450,
449,
252,
451,
450,
256,
452,
451,
341,
453,
452,
413,
464,
463,
441,
413,
414,
258,
442,
441,
257,
443,
442,
259,
444,
443,
260,
445,
444,
467,
342,
445,
459,
458,
250,
289,
392,
290,
290,
328,
460,
376,
433,
435,
250,
290,
392,
411,
416,
433,
341,
463,
464,
453,
464,
465,
357,
465,
412,
343,
412,
399,
360,
363,
440,
437,
399,
456,
420,
456,
363,
401,
435,
288,
372,
383,
353,
339,
255,
249,
448,
261,
255,
133,
243,
190,
133,
155,
112,
33,
246,
247,
33,
130,
25,
398,
384,
286,
362,
398,
414,
362,
463,
341,
263,
359,
467,
263,
249,
255,
466,
467,
260,
75,
60,
166,
238,
239,
79,
162,
127,
139,
72,
11,
37,
121,
232,
120,
73,
72,
39,
114,
128,
47,
233,
232,
128,
103,
104,
67,
152,
175,
148,
173,
157,
155,
119,
118,
101,
74,
73,
40,
107,
9,
108,
49,
48,
131,
32,
194,
211,
184,
74,
185,
191,
80,
183,
185,
40,
186,
119,
230,
118,
210,
202,
214,
84,
83,
17,
77,
76,
146,
161,
160,
30,
190,
56,
173,
182,
106,
194,
138,
135,
192,
129,
203,
98,
54,
21,
68,
5,
51,
4,
145,
144,
23,
90,
77,
91,
207,
205,
187,
83,
201,
18,
181,
91,
182,
180,
90,
181,
16,
85,
17,
205,
206,
36,
176,
148,
140,
165,
92,
39,
245,
193,
244,
27,
159,
28,
30,
247,
161,
174,
236,
196,
103,
54,
104,
55,
193,
8,
111,
117,
31,
221,
189,
55,
240,
98,
99,
142,
126,
100,
219,
166,
218,
112,
155,
26,
198,
209,
131,
169,
135,
150,
114,
47,
217,
224,
223,
53,
220,
45,
134,
32,
211,
140,
109,
67,
108,
146,
43,
91,
231,
230,
120,
113,
226,
247,
105,
63,
52,
241,
238,
242,
124,
46,
156,
95,
78,
96,
70,
46,
63,
116,
143,
227,
116,
123,
111,
1,
44,
19,
3,
236,
51,
207,
216,
205,
26,
154,
22,
165,
39,
167,
199,
200,
208,
101,
36,
100,
43,
57,
202,
242,
20,
99,
56,
28,
157,
124,
35,
113,
29,
160,
27,
211,
204,
210,
124,
113,
46,
106,
43,
204,
96,
62,
77,
227,
137,
116,
73,
41,
72,
36,
203,
142,
235,
64,
240,
48,
49,
64,
42,
41,
74,
214,
212,
207,
183,
42,
184,
210,
169,
211,
140,
170,
176,
104,
105,
69,
193,
122,
168,
50,
123,
187,
89,
96,
90,
66,
65,
107,
179,
89,
180,
119,
101,
120,
68,
63,
104,
234,
93,
227,
16,
15,
85,
209,
129,
49,
15,
14,
86,
107,
55,
9,
120,
100,
121,
153,
145,
22,
178,
88,
179,
197,
6,
196,
89,
88,
96,
135,
138,
136,
138,
215,
172,
218,
115,
219,
41,
42,
81,
5,
195,
51,
57,
43,
61,
208,
171,
199,
41,
81,
38,
224,
53,
225,
24,
144,
110,
105,
52,
66,
118,
229,
117,
227,
34,
234,
66,
107,
69,
10,
109,
151,
219,
48,
235,
183,
62,
191,
142,
129,
126,
116,
111,
143,
7,
163,
246,
118,
117,
50,
223,
222,
52,
94,
19,
141,
222,
221,
65,
196,
3,
197,
45,
220,
44,
156,
70,
139,
188,
122,
245,
139,
71,
162,
145,
153,
159,
149,
170,
150,
122,
188,
196,
206,
216,
92,
163,
144,
161,
164,
2,
167,
242,
141,
241,
0,
164,
37,
11,
72,
12,
144,
145,
160,
12,
38,
13,
70,
63,
71,
31,
226,
111,
157,
158,
154,
36,
101,
205,
203,
206,
165,
126,
209,
217,
98,
165,
97,
237,
220,
218,
237,
239,
241,
210,
214,
169,
140,
171,
32,
241,
125,
237,
179,
86,
178,
180,
85,
179,
181,
84,
180,
182,
83,
181,
194,
201,
182,
177,
137,
132,
184,
76,
183,
185,
61,
184,
186,
57,
185,
216,
212,
186,
192,
214,
187,
139,
34,
156,
218,
79,
237,
147,
123,
177,
45,
44,
4,
208,
201,
32,
98,
64,
129,
192,
213,
138,
235,
59,
219,
141,
242,
97,
97,
2,
141,
240,
75,
235,
229,
24,
228,
31,
25,
226,
230,
23,
229,
231,
22,
230,
232,
26,
231,
233,
112,
232,
244,
189,
243,
189,
221,
190,
222,
28,
221,
223,
27,
222,
224,
29,
223,
225,
30,
224,
113,
247,
225,
99,
60,
240,
213,
147,
215,
60,
20,
166,
192,
187,
213,
243,
112,
244,
244,
233,
245,
245,
128,
188,
188,
114,
174,
134,
131,
220,
174,
217,
236,
236,
198,
134,
215,
177,
58,
156,
143,
124,
25,
110,
7,
31,
228,
25,
264,
356,
368,
0,
11,
267,
451,
452,
349,
267,
302,
269,
350,
357,
277,
350,
452,
357,
299,
333,
297,
396,
175,
377,
381,
384,
382,
280,
347,
330,
269,
303,
270,
151,
9,
337,
344,
278,
360,
424,
418,
431,
270,
304,
409,
272,
310,
407,
322,
270,
410,
449,
450,
347,
432,
422,
434,
18,
313,
17,
291,
306,
375,
259,
387,
260,
424,
335,
418,
434,
364,
416,
391,
423,
327,
301,
251,
298,
275,
281,
4,
254,
373,
253,
375,
307,
321,
280,
425,
411,
200,
421,
18,
335,
321,
406,
321,
320,
405,
314,
315,
17,
423,
426,
266,
396,
377,
369,
270,
322,
269,
413,
417,
464,
385,
386,
258,
248,
456,
419,
298,
284,
333,
168,
417,
8,
448,
346,
261,
417,
413,
285,
326,
327,
328,
277,
355,
329,
309,
392,
438,
381,
382,
256,
279,
429,
360,
365,
364,
379,
355,
277,
437,
282,
443,
283,
281,
275,
363,
395,
431,
369,
299,
297,
337,
335,
273,
321,
348,
450,
349,
359,
446,
467,
283,
293,
282,
250,
458,
462,
300,
276,
383,
292,
308,
325,
283,
276,
293,
264,
372,
447,
346,
352,
340,
354,
274,
19,
363,
456,
281,
426,
436,
425,
380,
381,
252,
267,
269,
393,
421,
200,
428,
371,
266,
329,
432,
287,
422,
290,
250,
328,
385,
258,
384,
446,
265,
342,
386,
387,
257,
422,
424,
430,
445,
342,
276,
422,
273,
424,
306,
292,
307,
352,
366,
345,
268,
271,
302,
358,
423,
371,
327,
294,
460,
331,
279,
294,
303,
271,
304,
436,
432,
427,
304,
272,
408,
395,
394,
431,
378,
395,
400,
296,
334,
299,
6,
351,
168,
376,
352,
411,
307,
325,
320,
285,
295,
336,
320,
319,
404,
329,
330,
349,
334,
293,
333,
366,
323,
447,
316,
15,
315,
331,
358,
279,
317,
14,
316,
8,
285,
9,
277,
329,
350,
253,
374,
252,
319,
318,
403,
351,
6,
419,
324,
318,
325,
397,
367,
365,
288,
435,
397,
278,
344,
439,
310,
272,
311,
248,
195,
281,
375,
273,
291,
175,
396,
199,
312,
311,
268,
276,
283,
445,
390,
373,
339,
295,
282,
296,
448,
449,
346,
356,
264,
454,
337,
336,
299,
337,
338,
151,
294,
278,
455,
308,
292,
415,
429,
358,
355,
265,
340,
372,
388,
390,
466,
352,
346,
280,
295,
442,
282,
354,
19,
370,
285,
441,
295,
195,
248,
197,
457,
440,
274,
301,
300,
368,
417,
351,
465,
251,
301,
389,
385,
380,
386,
394,
395,
379,
399,
412,
419,
410,
436,
322,
387,
373,
388,
326,
2,
393,
354,
370,
461,
393,
164,
267,
268,
302,
12,
386,
374,
387,
312,
268,
13,
298,
293,
301,
265,
446,
340,
380,
385,
381,
280,
330,
425,
322,
426,
391,
420,
429,
437,
393,
391,
326,
344,
440,
438,
458,
459,
461,
364,
434,
394,
428,
396,
262,
274,
354,
457,
317,
316,
402,
316,
315,
403,
315,
314,
404,
314,
313,
405,
313,
421,
406,
323,
366,
361,
292,
306,
407,
306,
291,
408,
291,
287,
409,
287,
432,
410,
427,
434,
411,
372,
264,
383,
459,
309,
457,
366,
352,
401,
1,
274,
4,
418,
421,
262,
331,
294,
358,
435,
433,
367,
392,
289,
439,
328,
462,
326,
94,
2,
370,
289,
305,
455,
339,
254,
448,
359,
255,
446,
254,
253,
449,
253,
252,
450,
252,
256,
451,
256,
341,
452,
414,
413,
463,
286,
441,
414,
286,
258,
441,
258,
257,
442,
257,
259,
443,
259,
260,
444,
260,
467,
445,
309,
459,
250,
305,
289,
290,
305,
290,
460,
401,
376,
435,
309,
250,
392,
376,
411,
433,
453,
341,
464,
357,
453,
465,
343,
357,
412,
437,
343,
399,
344,
360,
440,
420,
437,
456,
360,
420,
363,
361,
401,
288,
265,
372,
353,
390,
339,
249,
339,
448,
255
];
var VTX68 = [
127,
234,
132,
58,
172,
150,
149,
148,
152,
377,
378,
379,
397,
288,
361,
454,
356,
70,
63,
105,
66,
107,
336,
296,
334,
293,
300,
168,
6,
195,
4,
98,
97,
2,
326,
327,
33,
160,
158,
133,
153,
144,
362,
385,
387,
263,
373,
380,
57,
40,
37,
0,
267,
270,
287,
321,
314,
17,
84,
91,
78,
81,
13,
311,
308,
402,
14,
178
];
var VTX33 = [33, 133, 362, 263, 1, 62, 308, 159, 145, 386, 374, 6, 102, 331, 2, 13, 14, 70, 105, 107, 336, 334, 300, 54, 10, 284, 50, 280, 234, 454, 58, 288, 152];
var VTX7 = [33, 133, 362, 263, 1, 78, 308];
var UV68 = VTX68.map((x) => UV468[x]);
var UV33 = VTX33.map((x) => UV468[x]);
var UV7 = VTX7.map((x) => UV468[x]);
function connectionsToIndices(connections) {
const indices = connections.map((connection) => connection[0]);
indices.push(connections[connections.length - 1][1]);
return indices;
}
var pairsLips = [
[61, 146],
[146, 91],
[91, 181],
[181, 84],
[84, 17],
[17, 314],
[314, 405],
[405, 321],
[321, 375],
[375, 291],
[61, 185],
[185, 40],
[40, 39],
[39, 37],
[37, 0],
[0, 267],
[267, 269],
[269, 270],
[270, 409],
[409, 291],
[78, 95],
[95, 88],
[88, 178],
[178, 87],
[87, 14],
[14, 317],
[317, 402],
[402, 318],
[318, 324],
[324, 308],
[78, 191],
[191, 80],
[80, 81],
[81, 82],
[82, 13],
[13, 312],
[312, 311],
[311, 310],
[310, 415],
[415, 308]
];
var pairsLeftEye = [[263, 249], [249, 390], [390, 373], [373, 374], [374, 380], [380, 381], [381, 382], [382, 362], [263, 466], [466, 388], [388, 387], [387, 386], [386, 385], [385, 384], [384, 398], [398, 362]];
var pairsLeftEyebrow = [[276, 283], [283, 282], [282, 295], [295, 285], [300, 293], [293, 334], [334, 296], [296, 336]];
var pairsLeftIris = [[474, 475], [475, 476], [476, 477], [477, 474]];
var pairsRightEye = [[33, 7], [7, 163], [163, 144], [144, 145], [145, 153], [153, 154], [154, 155], [155, 133], [33, 246], [246, 161], [161, 160], [160, 159], [159, 158], [158, 157], [157, 173], [173, 133]];
var pairsRightEyebrow = [[46, 53], [53, 52], [52, 65], [65, 55], [70, 63], [63, 105], [105, 66], [66, 107]];
var pairsRightIris = [[469, 470], [470, 471], [471, 472], [472, 469]];
var pairsFaceContour = [
[10, 338],
[338, 297],
[297, 332],
[332, 284],
[284, 251],
[251, 389],
[389, 356],
[356, 454],
[454, 323],
[323, 361],
[361, 288],
[288, 397],
[397, 365],
[365, 379],
[379, 378],
[378, 400],
[400, 377],
[377, 152],
[152, 148],
[148, 176],
[176, 149],
[149, 150],
[150, 136],
[136, 172],
[172, 58],
[58, 132],
[132, 93],
[93, 234],
[234, 127],
[127, 162],
[162, 21],
[21, 54],
[54, 103],
[103, 67],
[67, 109],
[109, 10]
];
var contourKeypoints = {
lips: connectionsToIndices(pairsLips),
leftEye: connectionsToIndices(pairsLeftEye),
leftEyebrow: connectionsToIndices(pairsLeftEyebrow),
leftIris: connectionsToIndices(pairsLeftIris),
rightEye: connectionsToIndices(pairsRightEye),
rightEyebrow: connectionsToIndices(pairsRightEyebrow),
rightIris: connectionsToIndices(pairsRightIris),
faceOval: connectionsToIndices(pairsFaceContour)
};
// src/face/constants.ts
var LIPS_CONNECTIONS = [
[61, 146],
[146, 91],
[91, 181],
[181, 84],
[84, 17],
[17, 314],
[314, 405],
[405, 321],
[321, 375],
[375, 291],
[61, 185],
[185, 40],
[40, 39],
[39, 37],
[37, 0],
[0, 267],
[267, 269],
[269, 270],
[270, 409],
[409, 291],
[78, 95],
[95, 88],
[88, 178],
[178, 87],
[87, 14],
[14, 317],
[317, 402],
[402, 318],
[318, 324],
[324, 308],
[78, 191],
[191, 80],
[80, 81],
[81, 82],
[82, 13],
[13, 312],
[312, 311],
[311, 310],
[310, 415],
[415, 308]
];
var LEFT_EYE_CONNECTIONS = [[263, 249], [249, 390], [390, 373], [373, 374], [374, 380], [380, 381], [381, 382], [382, 362], [263, 466], [466, 388], [388, 387], [387, 386], [386, 385], [385, 384], [384, 398], [398, 362]];
var LEFT_EYEBROW_CONNECTIONS = [[276, 283], [283, 282], [282, 295], [295, 285], [300, 293], [293, 334], [334, 296], [296, 336]];
var LEFT_IRIS_CONNECTIONS = [[474, 475], [475, 476], [476, 477], [477, 474]];
var RIGHT_EYE_CONNECTIONS = [[33, 7], [7, 163], [163, 144], [144, 145], [145, 153], [153, 154], [154, 155], [155, 133], [33, 246], [246, 161], [161, 160], [160, 159], [159, 158], [158, 157], [157, 173], [173, 133]];
var RIGHT_EYEBROW_CONNECTIONS = [[46, 53], [53, 52], [52, 65], [65, 55], [70, 63], [63, 105], [105, 66], [66, 107]];
var RIGHT_IRIS_CONNECTIONS = [[469, 470], [470, 471], [471, 472], [472, 469]];
var FACE_OVAL_CONNECTIONS = [
[10, 338],
[338, 297],
[297, 332],
[332, 284],
[284, 251],
[251, 389],
[389, 356],
[356, 454],
[454, 323],
[323, 361],
[361, 288],
[288, 397],
[397, 365],
[365, 379],
[379, 378],
[378, 400],
[400, 377],
[377, 152],
[152, 148],
[148, 176],
[176, 149],
[149, 150],
[150, 136],
[136, 172],
[172, 58],
[58, 132],
[132, 93],
[93, 234],
[234, 127],
[127, 162],
[162, 21],
[21, 54],
[54, 103],
[103, 67],
[67, 109],
[109, 10]
];
function connectionsToIndices2(connections) {
const indices = connections.map((connection) => connection[0]);
indices.push(connections[connections.length - 1][1]);
return indices;
}
var MEDIAPIPE_FACE_MESH_KEYPOINTS_BY_CONTOUR = {
lips: connectionsToIndices2(LIPS_CONNECTIONS),
leftEye: connectionsToIndices2(LEFT_EYE_CONNECTIONS),
leftEyebrow: connectionsToIndices2(LEFT_EYEBROW_CONNECTIONS),
leftIris: connectionsToIndices2(LEFT_IRIS_CONNECTIONS),
rightEye: connectionsToIndices2(RIGHT_EYE_CONNECTIONS),
rightEyebrow: connectionsToIndices2(RIGHT_EYEBROW_CONNECTIONS),
rightIris: connectionsToIndices2(RIGHT_IRIS_CONNECTIONS),
faceOval: connectionsToIndices2(FACE_OVAL_CONNECTIONS)
};
var indexLabelPairs = Object.entries(MEDIAPIPE_FACE_MESH_KEYPOINTS_BY_CONTOUR).map(([label, indices]) => indices.map((index2) => [index2, label])).flat();
var MEDIAPIPE_FACE_MESH_KEYPOINTS = new Map(indexLabelPairs);
var LANDMARKS_REFINEMENT_LIPS_CONFIG = [
61,
146,
91,
181,
84,
17,
314,
405,
321,
375,
291,
185,
40,
39,
37,
0,
267,
269,
270,
409,
78,
95,
88,
178,
87,
14,
317,
402,
318,
324,
308,
191,
80,
81,
82,
13,
312,
311,
310,
415,
76,
77,
90,
180,
85,
16,
315,
404,
320,
307,
306,
184,
74,
73,
72,
11,
302,
303,
304,
408,
62,
96,
89,
179,
86,
15,
316,
403,
319,
325,
292,
183,
42,
41,
38,
12,
268,
271,
272,
407
];
var LANDMARKS_REFINEMENT_LEFT_EYE_CONFIG = [
33,
7,
163,
144,
145,
153,
154,
155,
133,
246,
161,
160,
159,
158,
157,
173,
130,
25,
110,
24,
23,
22,
26,
112,
243,
247,
30,
29,
27,
28,
56,
190,
226,
31,
228,
229,
230,
231,
232,
233,
244,
113,
225,
224,
223,
222,
221,
189,
35,
124,
46,
53,
52,
65,
143,
111,
117,
118,
119,
120,
121,
128,
245,
156,
70,
63,
105,
66,
107,
55,
193
];
var LANDMARKS_REFINEMENT_RIGHT_EYE_CONFIG = [
263,
249,
390,
373,
374,
380,
381,
382,
362,
466,
388,
387,
386,
385,
384,
398,
359,
255,
339,
254,
253,
252,
256,
341,
463,
467,
260,
259,
257,
258,
286,
414,
446,
261,
448,
449,
450,
451,
452,
453,
464,
342,
445,
444,
443,
442,
441,
413,
265,
353,
276,
283,
282,
295,
372,
340,
346,
347,
348,
349,
350,
357,
465,
383,
300,
293,
334,
296,
336,
285,
417
];
// src/draw/face.ts
var localOptions;
function drawLabels(f, ctx) {
var _a, _b, _c, _d, _e, _f, _g, _h, _i;
if (!localOptions.drawLabels || ((_a = localOptions.faceLabels) == null ? void 0 : _a.length) === 0)
return;
let l = localOptions.faceLabels.slice();
if (f.score)
l = replace(l, "[score]", 100 * f.score);
if (f.gender)
l = replace(l, "[gender]", f.gender);
if (f.genderScore)
l = replace(l, "[genderScore]", 100 * f.genderScore);
if (f.age)
l = replace(l, "[age]", f.age);
if (f.distance)
l = replace(l, "[distance]", 100 * f.distance);
if (f.real)
l = replace(l, "[real]", 100 * f.real);
if (f.live)
l = replace(l, "[live]", 100 * f.live);
if (f.emotion && f.emotion.length > 0) {
const emotion2 = f.emotion.map((a) => `${Math.trunc(100 * a.score)}% ${a.emotion}`);
if (emotion2.length > 3)
emotion2.length = 3;
l = replace(l, "[emotions]", emotion2.join(" "));
}
if ((_c = (_b = f.rotation) == null ? void 0 : _b.angle) == null ? void 0 : _c.roll)
l = replace(l, "[roll]", rad2deg(f.rotation.angle.roll));
if ((_e = (_d = f.rotation) == null ? void 0 : _d.angle) == null ? void 0 : _e.yaw)
l = replace(l, "[yaw]", rad2deg(f.rotation.angle.yaw));
if ((_g = (_f = f.rotation) == null ? void 0 : _f.angle) == null ? void 0 : _g.pitch)
l = replace(l, "[pitch]", rad2deg(f.rotation.angle.pitch));
if ((_i = (_h = f.rotation) == null ? void 0 : _h.gaze) == null ? void 0 : _i.bearing)
l = replace(l, "[gaze]", rad2deg(f.rotation.gaze.bearing));
labels(ctx, l, f.box[0], f.box[1], localOptions);
}
function drawIrisElipse(f, ctx) {
var _a, _b, _c, _d;
if (((_a = f.annotations) == null ? void 0 : _a.leftEyeIris) && ((_b = f.annotations) == null ? void 0 : _b.leftEyeIris[0])) {
ctx.strokeStyle = localOptions.useDepth ? "rgba(255, 200, 255, 0.3)" : localOptions.color;
ctx.beginPath();
const sizeX = Math.abs(f.annotations.leftEyeIris[3][0] - f.annotations.leftEyeIris[1][0]) / 2;
const sizeY = Math.abs(f.annotations.leftEyeIris[4][1] - f.annotations.leftEyeIris[2][1]) / 2;
ctx.ellipse(f.annotations.leftEyeIris[0][0], f.annotations.leftEyeIris[0][1], sizeX, sizeY, 0, 0, 2 * Math.PI);
ctx.stroke();
if (localOptions.fillPolygons) {
ctx.fillStyle = localOptions.useDepth ? "rgba(255, 255, 200, 0.3)" : localOptions.color;
ctx.fill();
}
}
if (((_c = f.annotations) == null ? void 0 : _c.rightEyeIris) && ((_d = f.annotations) == null ? void 0 : _d.rightEyeIris[0])) {
ctx.strokeStyle = localOptions.useDepth ? "rgba(255, 200, 255, 0.3)" : localOptions.color;
ctx.beginPath();
const sizeX = Math.abs(f.annotations.rightEyeIris[3][0] - f.annotations.rightEyeIris[1][0]) / 2;
const sizeY = Math.abs(f.annotations.rightEyeIris[4][1] - f.annotations.rightEyeIris[2][1]) / 2;
ctx.ellipse(f.annotations.rightEyeIris[0][0], f.annotations.rightEyeIris[0][1], sizeX, sizeY, 0, 0, 2 * Math.PI);
ctx.stroke();
if (localOptions.fillPolygons) {
ctx.fillStyle = localOptions.useDepth ? "rgba(255, 255, 200, 0.3)" : localOptions.color;
ctx.fill();
}
}
}
function drawGazeSpheres(f, ctx) {
var _a;
if (localOptions.drawGaze && ((_a = f.rotation) == null ? void 0 : _a.angle) && typeof Path2D !== "undefined") {
ctx.strokeStyle = "pink";
const valX = f.box[0] + f.box[2] / 2 - f.box[3] * rad2deg(f.rotation.angle.yaw) / 90;
const valY = f.box[1] + f.box[3] / 2 + f.box[2] * rad2deg(f.rotation.angle.pitch) / 90;
const pathV = new Path2D(`
M ${f.box[0] + f.box[2] / 2} ${f.box[1]}
C
${valX} ${f.box[1]},
${valX} ${f.box[1] + f.box[3]},
${f.box[0] + f.box[2] / 2} ${f.box[1] + f.box[3]}
`);
const pathH = new Path2D(`
M ${f.box[0]} ${f.box[1] + f.box[3] / 2}
C
${f.box[0]} ${valY},
${f.box[0] + f.box[2]} ${valY},
${f.box[0] + f.box[2]} ${f.box[1] + f.box[3] / 2}
`);
ctx.stroke(pathH);
ctx.stroke(pathV);
}
}
function drawGazeArrows(f, ctx) {
var _a;
if (localOptions.drawGaze && ((_a = f.rotation) == null ? void 0 : _a.gaze.strength) && f.rotation.gaze.bearing && f.annotations.leftEyeIris && f.annotations.rightEyeIris && f.annotations.leftEyeIris[0] && f.annotations.rightEyeIris[0]) {
ctx.strokeStyle = "pink";
ctx.fillStyle = "pink";
const leftGaze = [
f.annotations.leftEyeIris[0][0] + Math.sin(f.rotation.gaze.bearing) * f.rotation.gaze.strength * f.box[3],
f.annotations.leftEyeIris[0][1] + Math.cos(f.rotation.gaze.bearing) * f.rotation.gaze.strength * f.box[2]
];
arrow(ctx, [f.annotations.leftEyeIris[0][0], f.annotations.leftEyeIris[0][1]], [leftGaze[0], leftGaze[1]], 4);
const rightGaze = [
f.annotations.rightEyeIris[0][0] + Math.sin(f.rotation.gaze.bearing) * f.rotation.gaze.strength * f.box[3],
f.annotations.rightEyeIris[0][1] + Math.cos(f.rotation.gaze.bearing) * f.rotation.gaze.strength * f.box[2]
];
arrow(ctx, [f.annotations.rightEyeIris[0][0], f.annotations.rightEyeIris[0][1]], [rightGaze[0], rightGaze[1]], 4);
}
}
function drawFacePolygons(f, ctx) {
if (localOptions.drawPolygons && f.mesh.length >= 468) {
ctx.lineWidth = 1;
for (let i = 0; i < TRI468.length / 3; i++) {
const points = [TRI468[i * 3 + 0], TRI468[i * 3 + 1], TRI468[i * 3 + 2]].map((index2) => f.mesh[index2]);
lines(ctx, points, localOptions);
}
drawIrisElipse(f, ctx);
}
}
function drawFacePoints(f, ctx) {
if (localOptions.drawPoints && f.mesh.length >= 468) {
for (let i = 0; i < f.mesh.length; i++) {
point(ctx, f.mesh[i][0], f.mesh[i][1], f.mesh[i][2], localOptions);
if (localOptions.drawAttention) {
if (LANDMARKS_REFINEMENT_LIPS_CONFIG.includes(i))
point(ctx, f.mesh[i][0], f.mesh[i][1], f.mesh[i][2] + 127, localOptions);
if (LANDMARKS_REFINEMENT_LEFT_EYE_CONFIG.includes(i))
point(ctx, f.mesh[i][0], f.mesh[i][1], f.mesh[i][2] - 127, localOptions);
if (LANDMARKS_REFINEMENT_RIGHT_EYE_CONFIG.includes(i))
point(ctx, f.mesh[i][0], f.mesh[i][1], f.mesh[i][2] - 127, localOptions);
}
}
}
}
function drawFaceBoxes(f, ctx) {
if (localOptions.drawBoxes) {
rect(ctx, f.box[0], f.box[1], f.box[2], f.box[3], localOptions);
}
}
function face(inCanvas2, result, drawOptions) {
localOptions = mergeDeep(options2, drawOptions);
if (!result || !inCanvas2)
return;
const ctx = getCanvasContext(inCanvas2);
if (!ctx)
return;
ctx.font = localOptions.font;
ctx.strokeStyle = localOptions.color;
ctx.fillStyle = localOptions.color;
for (const f of result) {
drawFaceBoxes(f, ctx);
drawLabels(f, ctx);
if (f.mesh && f.mesh.length > 0) {
drawFacePoints(f, ctx);
drawFacePolygons(f, ctx);
drawGazeSpheres(f, ctx);
drawGazeArrows(f, ctx);
}
}
}
// src/draw/body.ts
function body(inCanvas2, result, drawOptions) {
var _a, _b;
const localOptions2 = mergeDeep(options2, drawOptions);
if (!result || !inCanvas2)
return;
const ctx = getCanvasContext(inCanvas2);
if (!ctx)
return;
ctx.lineJoin = "round";
for (let i = 0; i < result.length; i++) {
ctx.strokeStyle = localOptions2.color;
ctx.fillStyle = localOptions2.color;
ctx.lineWidth = localOptions2.lineWidth;
ctx.font = localOptions2.font;
if (localOptions2.drawBoxes && result[i].box && result[i].box.length === 4) {
rect(ctx, result[i].box[0], result[i].box[1], result[i].box[2], result[i].box[3], localOptions2);
if (localOptions2.drawLabels && ((_a = localOptions2.bodyLabels) == null ? void 0 : _a.length) > 0) {
let l = localOptions2.bodyLabels.slice();
l = replace(l, "[score]", 100 * result[i].score);
labels(ctx, l, result[i].box[0], result[i].box[1], localOptions2);
}
}
if (localOptions2.drawPoints && result[i].keypoints) {
for (let pt = 0; pt < result[i].keypoints.length; pt++) {
if (!result[i].keypoints[pt].score || result[i].keypoints[pt].score === 0)
continue;
ctx.fillStyle = colorDepth(result[i].keypoints[pt].position[2], localOptions2);
point(ctx, result[i].keypoints[pt].position[0], result[i].keypoints[pt].position[1], 0, localOptions2);
}
}
if (localOptions2.drawLabels && ((_b = localOptions2.bodyPartLabels) == null ? void 0 : _b.length) > 0 && result[i].keypoints) {
ctx.font = localOptions2.font;
for (const pt of result[i].keypoints) {
if (!pt.score || pt.score === 0)
continue;
let l = localOptions2.bodyPartLabels.slice();
l = replace(l, "[label]", pt.part);
l = replace(l, "[score]", 100 * pt.score);
labels(ctx, l, pt.position[0], pt.position[1], localOptions2);
}
}
if (localOptions2.drawPolygons && result[i].keypoints && result[i].annotations) {
for (const part of Object.values(result[i].annotations)) {
for (const connected4 of part)
curves(ctx, connected4, localOptions2);
}
}
}
}
// src/draw/hand.ts
function hand(inCanvas2, result, drawOptions) {
var _a, _b;
const localOptions2 = mergeDeep(options2, drawOptions);
if (!result || !inCanvas2)
return;
const ctx = getCanvasContext(inCanvas2);
if (!ctx)
return;
ctx.lineJoin = "round";
ctx.font = localOptions2.font;
for (const h of result) {
if (localOptions2.drawBoxes) {
ctx.strokeStyle = localOptions2.color;
ctx.fillStyle = localOptions2.color;
rect(ctx, h.box[0], h.box[1], h.box[2], h.box[3], localOptions2);
if (localOptions2.drawLabels && ((_a = localOptions2.handLabels) == null ? void 0 : _a.length) > 0) {
let l = localOptions2.handLabels.slice();
l = replace(l, "[label]", h.label);
l = replace(l, "[score]", 100 * h.score);
labels(ctx, l, h.box[0], h.box[1], localOptions2);
}
ctx.stroke();
}
if (localOptions2.drawPoints) {
if (h.keypoints && h.keypoints.length > 0) {
for (const pt of h.keypoints) {
ctx.fillStyle = colorDepth(pt[2], localOptions2);
point(ctx, pt[0], pt[1], 0, localOptions2);
}
}
}
if (localOptions2.drawLabels && h.annotations && ((_b = localOptions2.fingerLabels) == null ? void 0 : _b.length) > 0) {
for (const [part, pt] of Object.entries(h.annotations)) {
let l = localOptions2.fingerLabels.slice();
l = replace(l, "[label]", part);
labels(ctx, l, pt[pt.length - 1][0], pt[pt.length - 1][1], localOptions2);
}
}
if (localOptions2.drawPolygons && h.annotations) {
const addHandLine = (part) => {
if (!part || part.length === 0 || !part[0])
return;
for (let i = 0; i < part.length; i++) {
ctx.beginPath();
const z = part[i][2] || 0;
ctx.strokeStyle = colorDepth(i * z, localOptions2);
ctx.moveTo(part[i > 0 ? i - 1 : 0][0], part[i > 0 ? i - 1 : 0][1]);
ctx.lineTo(part[i][0], part[i][1]);
ctx.stroke();
}
};
ctx.lineWidth = localOptions2.lineWidth;
addHandLine(h.annotations.index);
addHandLine(h.annotations.middle);
addHandLine(h.annotations.ring);
addHandLine(h.annotations.pinky);
addHandLine(h.annotations.thumb);
}
}
}
// src/draw/object.ts
function object(inCanvas2, result, drawOptions) {
var _a;
const localOptions2 = mergeDeep(options2, drawOptions);
if (!result || !inCanvas2)
return;
const ctx = getCanvasContext(inCanvas2);
if (!ctx)
return;
ctx.lineJoin = "round";
ctx.font = localOptions2.font;
for (const h of result) {
if (localOptions2.drawBoxes) {
ctx.strokeStyle = localOptions2.color;
ctx.fillStyle = localOptions2.color;
rect(ctx, h.box[0], h.box[1], h.box[2], h.box[3], localOptions2);
if (localOptions2.drawLabels && ((_a = localOptions2.objectLabels) == null ? void 0 : _a.length) > 0) {
let l = localOptions2.objectLabels.slice();
l = replace(l, "[label]", h.label);
l = replace(l, "[score]", 100 * h.score);
labels(ctx, l, h.box[0], h.box[1], localOptions2);
}
ctx.stroke();
}
}
}
// src/draw/gesture.ts
function gesture(inCanvas2, result, drawOptions) {
var _a;
const localOptions2 = mergeDeep(options2, drawOptions);
if (!result || !inCanvas2)
return;
if (localOptions2.drawGestures && ((_a = localOptions2.gestureLabels) == null ? void 0 : _a.length) > 0) {
const ctx = getCanvasContext(inCanvas2);
if (!ctx)
return;
ctx.font = localOptions2.font;
ctx.fillStyle = localOptions2.color;
let i = 1;
for (let j = 0; j < result.length; j++) {
const [where, what] = Object.entries(result[j]);
if (what.length > 1 && what[1].length > 0) {
const who = where[1] > 0 ? `#${where[1]}` : "";
let l = localOptions2.gestureLabels.slice();
l = replace(l, "[where]", where[0]);
l = replace(l, "[who]", who);
l = replace(l, "[what]", what[1]);
labels(ctx, l, 8, 2 + i * localOptions2.lineHeight, localOptions2);
i += 1;
}
}
}
}
// src/draw/labels.ts
var defaultLabels = {
face: `face
confidence: [score]%
[gender] [genderScore]%
age: [age] years
distance: [distance]cm
real: [real]%
live: [live]%
[emotions]
roll: [roll]\xB0 yaw:[yaw]\xB0 pitch:[pitch]\xB0
gaze: [gaze]\xB0`,
body: "body [score]%",
bodyPart: "[label] [score]%",
object: "[label] [score]%",
hand: "[label] [score]%",
finger: "[label]",
gesture: "[where] [who]: [what]"
};
// src/draw/draw.ts
var drawTime = 0;
function person(inCanvas2, result, drawOptions) {
const localOptions2 = mergeDeep(options2, drawOptions);
if (!result || !inCanvas2)
return;
const ctx = getCanvasContext(inCanvas2);
if (!ctx)
return;
ctx.lineJoin = "round";
ctx.font = localOptions2.font;
for (let i = 0; i < result.length; i++) {
if (localOptions2.drawBoxes) {
ctx.strokeStyle = localOptions2.color;
ctx.fillStyle = localOptions2.color;
rect(ctx, result[i].box[0], result[i].box[1], result[i].box[2], result[i].box[3], localOptions2);
if (localOptions2.drawLabels) {
const label = `person #${i}`;
if (localOptions2.shadowColor && localOptions2.shadowColor !== "") {
ctx.fillStyle = localOptions2.shadowColor;
ctx.fillText(label, result[i].box[0] + 3, 1 + result[i].box[1] + localOptions2.lineHeight, result[i].box[2]);
}
ctx.fillStyle = localOptions2.labelColor;
ctx.fillText(label, result[i].box[0] + 2, 0 + result[i].box[1] + localOptions2.lineHeight, result[i].box[2]);
}
ctx.stroke();
}
}
}
function canvas2(input, output) {
if (!input || !output)
return;
const ctx = getCanvasContext(output);
if (!ctx)
return;
ctx.drawImage(input, 0, 0);
}
async function all(inCanvas2, result, drawOptions) {
if (!(result == null ? void 0 : result.performance) || !inCanvas2)
return null;
const timeStamp = now();
const localOptions2 = mergeDeep(options2, drawOptions);
const promise = Promise.all([
face(inCanvas2, result.face, localOptions2),
body(inCanvas2, result.body, localOptions2),
hand(inCanvas2, result.hand, localOptions2),
object(inCanvas2, result.object, localOptions2),
gesture(inCanvas2, result.gesture, localOptions2)
]);
drawTime = env.perfadd ? drawTime + Math.round(now() - timeStamp) : Math.round(now() - timeStamp);
result.performance.draw = drawTime;
return promise;
}
function init2() {
options2.faceLabels = defaultLabels.face;
options2.bodyLabels = defaultLabels.body;
options2.bodyPartLabels = defaultLabels.bodyPart;
options2.handLabels = defaultLabels.hand;
options2.fingerLabels = defaultLabels.finger;
options2.objectLabels = defaultLabels.object;
options2.gestureLabels = defaultLabels.gesture;
}
// src/body/blazepose.ts
var tf9 = __toESM(require_tfjs_esm());
// src/body/blazeposecoords.ts
var blazeposecoords_exports = {};
__export(blazeposecoords_exports, {
connected: () => connected,
kpt: () => kpt
});
var kpt = [
"nose",
"leftEyeInside",
"leftEye",
"leftEyeOutside",
"rightEyeInside",
"rightEye",
"rightEyeOutside",
"leftEar",
"rightEar",
"leftMouth",
"rightMouth",
"leftShoulder",
"rightShoulder",
"leftElbow",
"rightElbow",
"leftWrist",
"rightWrist",
"leftPinky",
"rightPinky",
"leftIndex",
"rightIndex",
"leftThumb",
"rightThumb",
"leftHip",
"rightHip",
"leftKnee",
"rightKnee",
"leftAnkle",
"rightAnkle",
"leftHeel",
"rightHeel",
"leftFoot",
"rightFoot",
"bodyCenter",
"bodyTop",
"leftPalm",
"leftHand",
"rightPalm",
"rightHand"
];
var connected = {
shoulders: ["leftShoulder", "rightShoulder"],
hips: ["rightHip", "leftHip"],
mouth: ["leftMouth", "rightMouth"],
leftLegUpper: ["leftHip", "leftKnee"],
leftLegLower: ["leftKnee", "leftAnkle"],
leftFoot: ["leftAnkle", "leftHeel", "leftFoot"],
leftTorso: ["leftShoulder", "leftHip"],
leftArmUpper: ["leftShoulder", "leftElbow"],
leftArmLower: ["leftElbow", "leftWrist"],
leftHand: ["leftWrist", "leftPalm"],
leftHandPinky: ["leftPalm", "leftPinky"],
leftHandIndex: ["leftPalm", "leftIndex"],
leftHandThumb: ["leftPalm", "leftThumb"],
leftEyeOutline: ["leftEyeInside", "leftEyeOutside"],
rightLegUpper: ["rightHip", "rightKnee"],
rightLegLower: ["rightKnee", "rightAnkle"],
rightFoot: ["rightAnkle", "rightHeel", "rightFoot"],
rightTorso: ["rightShoulder", "rightHip"],
rightArmUpper: ["rightShoulder", "rightElbow"],
rightArmLower: ["rightElbow", "rightWrist"],
rightHand: ["rightWrist", "rightPalm"],
rightHandPinky: ["rightPalm", "rightPinky"],
rightHandIndex: ["rightPalm", "rightIndex"],
rightHandThumb: ["rightPalm", "rightThumb"],
rightEyeOutline: ["rightEyeInside", "rightEyeOutside"]
};
// src/body/blazeposedetector.ts
var tf8 = __toESM(require_tfjs_esm());
var model;
var inputSize = 224;
var anchorTensor;
var numLayers = 5;
var strides = [8, 16, 32, 32, 32];
function createAnchors() {
const anchors3 = [];
let layerId = 0;
while (layerId < numLayers) {
let anchorCount = 0;
let lastSameStrideLayer = layerId;
while (lastSameStrideLayer < strides.length && strides[lastSameStrideLayer] === strides[layerId]) {
anchorCount += 2;
lastSameStrideLayer++;
}
const stride = strides[layerId];
const featureMapHeight = Math.ceil(inputSize / stride);
const featureMapWidth = Math.ceil(inputSize / stride);
for (let y = 0; y < featureMapHeight; ++y) {
for (let x = 0; x < featureMapWidth; ++x) {
for (let anchorId = 0; anchorId < anchorCount; ++anchorId) {
anchors3.push({ x: (x + 0.5) / featureMapWidth, y: (y + 0.5) / featureMapHeight });
}
}
}
layerId = lastSameStrideLayer;
}
anchorTensor = { x: tf8.tensor1d(anchors3.map((a) => a.x)), y: tf8.tensor1d(anchors3.map((a) => a.y)) };
}
async function loadDetector(config3) {
if (env.initial)
model = null;
if (!model && config3.body["detector"] && config3.body["detector"].modelPath || "") {
model = await loadModel(config3.body["detector"].modelPath);
const inputs = (model == null ? void 0 : model["executor"]) ? Object.values(model.modelSignature["inputs"]) : void 0;
inputSize = Array.isArray(inputs) ? parseInt(inputs[0].tensorShape.dim[1].size) : 0;
} else if (config3.debug && model)
log("cached model:", model["modelUrl"]);
createAnchors();
return model;
}
var cropFactor = [5, 5];
function decodeBoxes(boxesTensor, anchor) {
return tf8.tidy(() => {
const split6 = tf8.split(boxesTensor, 12, 1);
let xCenter = tf8.squeeze(split6[0]);
let yCenter = tf8.squeeze(split6[1]);
let width = tf8.squeeze(split6[2]);
let height = tf8.squeeze(split6[3]);
xCenter = tf8.add(tf8.div(xCenter, inputSize), anchor.x);
yCenter = tf8.add(tf8.div(yCenter, inputSize), anchor.y);
width = tf8.mul(tf8.div(width, inputSize), cropFactor[0]);
height = tf8.mul(tf8.div(height, inputSize), cropFactor[1]);
const xMin = tf8.sub(xCenter, tf8.div(width, 2));
const yMin = tf8.sub(yCenter, tf8.div(height, 2));
const xMax = tf8.add(xMin, width);
const yMax = tf8.add(yMin, height);
const boxes = tf8.stack([xMin, yMin, xMax, yMax], 1);
return boxes;
});
}
async function decodeResults(boxesTensor, logitsTensor, config3, outputSize2) {
var _a, _b;
const detectedBoxes = [];
const t2 = {};
t2.boxes = decodeBoxes(boxesTensor, anchorTensor);
t2.scores = tf8.sigmoid(logitsTensor);
t2.nms = await tf8.image.nonMaxSuppressionAsync(t2.boxes, t2.scores, 1, ((_a = config3.body["detector"]) == null ? void 0 : _a.minConfidence) || 0.1, ((_b = config3.body["detector"]) == null ? void 0 : _b.iouThreshold) || 0.1);
const nms = await t2.nms.data();
const scores = await t2.scores.data();
const boxes = await t2.boxes.array();
for (const i of Array.from(nms)) {
const score = scores[i];
const boxRaw = boxes[i];
const box = [Math.round(boxRaw[0] * outputSize2[0]), Math.round(boxRaw[1] * outputSize2[1]), Math.round(boxRaw[2] * outputSize2[0]), Math.round(boxRaw[3] * outputSize2[1])];
const detectedBox = { score, boxRaw, box };
detectedBoxes.push(detectedBox);
}
Object.keys(t2).forEach((tensor6) => tf8.dispose(t2[tensor6]));
return detectedBoxes;
}
async function detectBoxes(input, config3, outputSize2) {
const t2 = {};
t2.res = model == null ? void 0 : model.execute(input, ["Identity"]);
t2.logitsRaw = tf8.slice(t2.res, [0, 0, 0], [1, -1, 1]);
t2.boxesRaw = tf8.slice(t2.res, [0, 0, 1], [1, -1, -1]);
t2.logits = tf8.squeeze(t2.logitsRaw);
t2.boxes = tf8.squeeze(t2.boxesRaw);
const boxes = await decodeResults(t2.boxes, t2.logits, config3, outputSize2);
Object.keys(t2).forEach((tensor6) => tf8.dispose(t2[tensor6]));
return boxes;
}
// src/util/box.ts
function calc(keypoints, outputSize2 = [1, 1]) {
const coords = [keypoints.map((pt) => pt[0]), keypoints.map((pt) => pt[1])];
const min2 = [Math.min(...coords[0]), Math.min(...coords[1])];
const max5 = [Math.max(...coords[0]), Math.max(...coords[1])];
const box = [min2[0], min2[1], max5[0] - min2[0], max5[1] - min2[1]];
const boxRaw = [box[0] / outputSize2[0], box[1] / outputSize2[1], box[2] / outputSize2[0], box[3] / outputSize2[1]];
return { box, boxRaw };
}
function square(keypoints, outputSize2 = [1, 1]) {
const coords = [keypoints.map((pt) => pt[0]), keypoints.map((pt) => pt[1])];
const min2 = [Math.min(...coords[0]), Math.min(...coords[1])];
const max5 = [Math.max(...coords[0]), Math.max(...coords[1])];
const center = [(min2[0] + max5[0]) / 2, (min2[1] + max5[1]) / 2];
const dist = Math.max(center[0] - min2[0], center[1] - min2[1], -center[0] + max5[0], -center[1] + max5[1]);
const box = [Math.trunc(center[0] - dist), Math.trunc(center[1] - dist), Math.trunc(2 * dist), Math.trunc(2 * dist)];
const boxRaw = [box[0] / outputSize2[0], box[1] / outputSize2[1], box[2] / outputSize2[0], box[3] / outputSize2[1]];
return { box, boxRaw };
}
function scale(box, scaleFact) {
const dist = [box[2] * scaleFact, box[3] * scaleFact];
const newBox = [
box[0] - (dist[0] - box[2]) / 2,
box[1] - (dist[1] - box[3]) / 2,
dist[0],
dist[1]
];
return newBox;
}
// src/body/blazepose.ts
var model2;
var inputSize2 = 256;
var skipped = Number.MAX_SAFE_INTEGER;
var outputNodes = {
landmarks: ["ld_3d", "activation_segmentation", "activation_heatmap", "world_3d", "output_poseflag"],
detector: []
};
var cache = [];
var padding = [[0, 0], [0, 0], [0, 0], [0, 0]];
var lastTime = 0;
var sigmoid2 = (x) => 1 - 1 / (1 + Math.exp(x));
var loadDetect = (config3) => loadDetector(config3);
async function loadPose(config3) {
if (env.initial)
model2 = null;
if (!model2) {
model2 = await loadModel(config3.body.modelPath);
const inputs = (model2 == null ? void 0 : model2["executor"]) ? Object.values(model2.modelSignature["inputs"]) : void 0;
inputSize2 = Array.isArray(inputs) ? parseInt(inputs[0].tensorShape.dim[1].size) : 0;
} else if (config3.debug)
log("cached model:", model2["modelUrl"]);
return model2;
}
function prepareImage(input, size2, cropBox) {
var _a, _b;
const t2 = {};
if (!((_a = input == null ? void 0 : input.shape) == null ? void 0 : _a[1]) || !((_b = input == null ? void 0 : input.shape) == null ? void 0 : _b[2]))
return input;
let final;
if (cropBox) {
t2.cropped = tf9.image.cropAndResize(input, [cropBox], [0], [input.shape[1], input.shape[2]]);
}
if (input.shape[1] !== input.shape[2]) {
const height = [
input.shape[2] > input.shape[1] ? Math.trunc((input.shape[2] - input.shape[1]) / 2) : 0,
input.shape[2] > input.shape[1] ? Math.trunc((input.shape[2] - input.shape[1]) / 2) : 0
];
const width = [
input.shape[1] > input.shape[2] ? Math.trunc((input.shape[1] - input.shape[2]) / 2) : 0,
input.shape[1] > input.shape[2] ? Math.trunc((input.shape[1] - input.shape[2]) / 2) : 0
];
padding = [
[0, 0],
height,
width,
[0, 0]
];
t2.pad = tf9.pad(t2.cropped || input, padding);
t2.resize = tf9.image.resizeBilinear(t2.pad, [size2, size2]);
final = tf9.div(t2.resize, constants.tf255);
} else if (input.shape[1] !== size2) {
t2.resize = tf9.image.resizeBilinear(t2.cropped || input, [size2, size2]);
final = tf9.div(t2.resize, constants.tf255);
} else {
final = tf9.div(t2.cropped || input, constants.tf255);
}
Object.keys(t2).forEach((tensor6) => tf9.dispose(t2[tensor6]));
return final;
}
function rescaleKeypoints(keypoints, outputSize2, cropBox) {
for (const kpt4 of keypoints) {
kpt4.position = [
Math.trunc(kpt4.position[0] * (outputSize2[0] + padding[2][0] + padding[2][1]) / outputSize2[0] - padding[2][0]),
Math.trunc(kpt4.position[1] * (outputSize2[1] + padding[1][0] + padding[1][1]) / outputSize2[1] - padding[1][0]),
kpt4.position[2]
];
kpt4.positionRaw = [kpt4.position[0] / outputSize2[0], kpt4.position[1] / outputSize2[1], 2 * kpt4.position[2] / (outputSize2[0] + outputSize2[1])];
}
if (cropBox) {
const width = cropBox[2] - cropBox[0];
const height = cropBox[3] - cropBox[1];
for (const kpt4 of keypoints) {
kpt4.positionRaw = [
kpt4.positionRaw[0] / height + cropBox[1],
kpt4.positionRaw[1] / width + cropBox[0],
kpt4.positionRaw[2]
];
kpt4.position = [
Math.trunc(kpt4.positionRaw[0] * outputSize2[0]),
Math.trunc(kpt4.positionRaw[1] * outputSize2[1]),
kpt4.positionRaw[2]
];
}
}
return keypoints;
}
function fixKeypoints(keypoints) {
const leftPalm = keypoints.find((k) => k.part === "leftPalm");
const leftWrist = keypoints.find((k) => k.part === "leftWrist");
const leftIndex = keypoints.find((k) => k.part === "leftIndex");
leftPalm.position[2] = ((leftWrist.position[2] || 0) + (leftIndex.position[2] || 0)) / 2;
const rightPalm = keypoints.find((k) => k.part === "rightPalm");
const rightWrist = keypoints.find((k) => k.part === "rightWrist");
const rightIndex = keypoints.find((k) => k.part === "rightIndex");
rightPalm.position[2] = ((rightWrist.position[2] || 0) + (rightIndex.position[2] || 0)) / 2;
}
async function detectLandmarks(input, config3, outputSize2) {
if (!(model2 == null ? void 0 : model2["executor"]))
return null;
const t2 = {};
[t2.ld, t2.segmentation, t2.heatmap, t2.world, t2.poseflag] = model2 == null ? void 0 : model2.execute(input, outputNodes.landmarks);
const poseScore = (await t2.poseflag.data())[0];
const points = await t2.ld.data();
const distances = await t2.world.data();
Object.keys(t2).forEach((tensor6) => tf9.dispose(t2[tensor6]));
const keypointsRelative = [];
const depth = 5;
for (let i = 0; i < points.length / depth; i++) {
const score = sigmoid2(points[depth * i + 3]);
const presence = sigmoid2(points[depth * i + 4]);
const adjScore = Math.trunc(100 * score * presence * poseScore) / 100;
const positionRaw = [points[depth * i + 0] / inputSize2, points[depth * i + 1] / inputSize2, points[depth * i + 2] + 0];
const position = [Math.trunc(outputSize2[0] * positionRaw[0]), Math.trunc(outputSize2[1] * positionRaw[1]), positionRaw[2]];
const distance2 = [distances[depth * i + 0], distances[depth * i + 1], distances[depth * i + 2] + 0];
keypointsRelative.push({ part: kpt[i], positionRaw, position, distance: distance2, score: adjScore });
}
if (poseScore < (config3.body.minConfidence || 0))
return null;
fixKeypoints(keypointsRelative);
const keypoints = rescaleKeypoints(keypointsRelative, outputSize2);
const kpts = keypoints.map((k) => k.position);
const boxes = calc(kpts, [outputSize2[0], outputSize2[1]]);
const annotations2 = {};
for (const [name, indexes] of Object.entries(connected)) {
const pt = [];
for (let i = 0; i < indexes.length - 1; i++) {
const pt0 = keypoints.find((kpt4) => kpt4.part === indexes[i]);
const pt1 = keypoints.find((kpt4) => kpt4.part === indexes[i + 1]);
if (pt0 && pt1)
pt.push([pt0.position, pt1.position]);
}
annotations2[name] = pt;
}
const body4 = { id: 0, score: Math.trunc(100 * poseScore) / 100, box: boxes.box, boxRaw: boxes.boxRaw, keypoints, annotations: annotations2 };
return body4;
}
async function predict(input, config3) {
var _a, _b, _c;
const outputSize2 = [input.shape[2] || 0, input.shape[1] || 0];
const skipTime = (config3.body.skipTime || 0) > now() - lastTime;
const skipFrame = skipped < (config3.body.skipFrames || 0);
if (config3.skipAllowed && skipTime && skipFrame && cache !== null) {
skipped++;
} else {
let boxes = [];
if ((_b = (_a = config3.body) == null ? void 0 : _a["detector"]) == null ? void 0 : _b["enabled"]) {
const preparedImage = prepareImage(input, 224);
boxes = await detectBoxes(preparedImage, config3, outputSize2);
tf9.dispose(preparedImage);
} else {
boxes = [{ box: [0, 0, 0, 0], boxRaw: [0, 0, 1, 1], score: 0 }];
}
for (let i = 0; i < boxes.length; i++) {
const preparedBox = prepareImage(input, 256, (_c = boxes[i]) == null ? void 0 : _c.boxRaw);
cache.length = 0;
const bodyResult = await detectLandmarks(preparedBox, config3, outputSize2);
tf9.dispose(preparedBox);
if (!bodyResult)
continue;
bodyResult.id = i;
cache.push(bodyResult);
}
lastTime = now();
skipped = 0;
}
return cache;
}
// src/object/centernet.ts
var tf10 = __toESM(require_tfjs_esm());
// src/object/labels.ts
var labels2 = [
{ class: 1, label: "person" },
{ class: 2, label: "bicycle" },
{ class: 3, label: "car" },
{ class: 4, label: "motorcycle" },
{ class: 5, label: "airplane" },
{ class: 6, label: "bus" },
{ class: 7, label: "train" },
{ class: 8, label: "truck" },
{ class: 9, label: "boat" },
{ class: 10, label: "traffic light" },
{ class: 11, label: "fire hydrant" },
{ class: 12, label: "stop sign" },
{ class: 13, label: "parking meter" },
{ class: 14, label: "bench" },
{ class: 15, label: "bird" },
{ class: 16, label: "cat" },
{ class: 17, label: "dog" },
{ class: 18, label: "horse" },
{ class: 19, label: "sheep" },
{ class: 20, label: "cow" },
{ class: 21, label: "elephant" },
{ class: 22, label: "bear" },
{ class: 23, label: "zebra" },
{ class: 24, label: "giraffe" },
{ class: 25, label: "backpack" },
{ class: 26, label: "umbrella" },
{ class: 27, label: "handbag" },
{ class: 28, label: "tie" },
{ class: 29, label: "suitcase" },
{ class: 30, label: "frisbee" },
{ class: 31, label: "skis" },
{ class: 32, label: "snowboard" },
{ class: 33, label: "sports ball" },
{ class: 34, label: "kite" },
{ class: 35, label: "baseball bat" },
{ class: 36, label: "baseball glove" },
{ class: 37, label: "skateboard" },
{ class: 38, label: "surfboard" },
{ class: 39, label: "tennis racket" },
{ class: 40, label: "bottle" },
{ class: 41, label: "wine glass" },
{ class: 42, label: "cup" },
{ class: 43, label: "fork" },
{ class: 44, label: "knife" },
{ class: 45, label: "spoon" },
{ class: 46, label: "bowl" },
{ class: 47, label: "banana" },
{ class: 48, label: "apple" },
{ class: 49, label: "sandwich" },
{ class: 50, label: "orange" },
{ class: 51, label: "broccoli" },
{ class: 52, label: "carrot" },
{ class: 53, label: "hot dog" },
{ class: 54, label: "pizza" },
{ class: 55, label: "donut" },
{ class: 56, label: "cake" },
{ class: 57, label: "chair" },
{ class: 58, label: "couch" },
{ class: 59, label: "potted plant" },
{ class: 60, label: "bed" },
{ class: 61, label: "dining table" },
{ class: 62, label: "toilet" },
{ class: 63, label: "tv" },
{ class: 64, label: "laptop" },
{ class: 65, label: "mouse" },
{ class: 66, label: "remote" },
{ class: 67, label: "keyboard" },
{ class: 68, label: "cell phone" },
{ class: 69, label: "microwave" },
{ class: 70, label: "oven" },
{ class: 71, label: "toaster" },
{ class: 72, label: "sink" },
{ class: 73, label: "refrigerator" },
{ class: 74, label: "book" },
{ class: 75, label: "clock" },
{ class: 76, label: "vase" },
{ class: 77, label: "scissors" },
{ class: 78, label: "teddy bear" },
{ class: 79, label: "hair drier" },
{ class: 80, label: "toothbrush" }
];
// src/object/centernet.ts
var model3;
var inputSize3 = 0;
var last2 = [];
var lastTime2 = 0;
var skipped2 = Number.MAX_SAFE_INTEGER;
async function load(config3) {
if (env.initial)
model3 = null;
if (!model3) {
model3 = await loadModel(config3.object.modelPath);
const inputs = (model3 == null ? void 0 : model3["executor"]) ? Object.values(model3.modelSignature["inputs"]) : void 0;
inputSize3 = Array.isArray(inputs) ? parseInt(inputs[0].tensorShape.dim[2].size) : 0;
} else if (config3.debug)
log("cached model:", model3["modelUrl"]);
return model3;
}
async function process3(res, outputShape, config3) {
if (!res)
return [];
const t2 = {};
const results = [];
const detections = await res.array();
t2.squeeze = tf10.squeeze(res);
const arr = tf10.split(t2.squeeze, 6, 1);
t2.stack = tf10.stack([arr[1], arr[0], arr[3], arr[2]], 1);
t2.boxes = tf10.squeeze(t2.stack);
t2.scores = tf10.squeeze(arr[4]);
t2.classes = tf10.squeeze(arr[5]);
tf10.dispose([res, ...arr]);
t2.nms = await tf10.image.nonMaxSuppressionAsync(t2.boxes, t2.scores, config3.object.maxDetected || 0, config3.object.iouThreshold, config3.object.minConfidence || 0);
const nms = await t2.nms.data();
let i = 0;
for (const id of Array.from(nms)) {
const score = Math.trunc(100 * detections[0][id][4]) / 100;
const classVal = detections[0][id][5];
if (Number.isNaN(classVal))
continue;
const label = labels2[classVal].label;
const [x, y] = [
detections[0][id][0] / inputSize3,
detections[0][id][1] / inputSize3
];
const boxRaw = [
x,
y,
detections[0][id][2] / inputSize3 - x,
detections[0][id][3] / inputSize3 - y
];
const box = [
Math.trunc(boxRaw[0] * outputShape[0]),
Math.trunc(boxRaw[1] * outputShape[1]),
Math.trunc(boxRaw[2] * outputShape[0]),
Math.trunc(boxRaw[3] * outputShape[1])
];
results.push({ id: i++, score, class: classVal, label, box, boxRaw });
}
Object.keys(t2).forEach((tensor6) => tf10.dispose(t2[tensor6]));
return results;
}
async function predict2(input, config3) {
if (!(model3 == null ? void 0 : model3["executor"]))
return [];
const skipTime = (config3.object.skipTime || 0) > now() - lastTime2;
const skipFrame = skipped2 < (config3.object.skipFrames || 0);
if (config3.skipAllowed && skipTime && skipFrame && last2.length > 0) {
skipped2++;
return last2;
}
skipped2 = 0;
return new Promise(async (resolve) => {
const outputSize2 = [input.shape[2] || 0, input.shape[1] || 0];
const resize = tf10.image.resizeBilinear(input, [inputSize3, inputSize3]);
const objectT = config3.object.enabled ? model3 == null ? void 0 : model3.execute(resize, ["tower_0/detections"]) : null;
lastTime2 = now();
tf10.dispose(resize);
const obj = await process3(objectT, outputSize2, config3);
last2 = obj;
resolve(obj);
});
}
// src/body/efficientpose.ts
var tf11 = __toESM(require_tfjs_esm());
// src/body/efficientposecoords.ts
var efficientposecoords_exports = {};
__export(efficientposecoords_exports, {
connected: () => connected2,
kpt: () => kpt2
});
var kpt2 = [
"head",
"neck",
"rightShoulder",
"rightElbow",
"rightWrist",
"chest",
"leftShoulder",
"leftElbow",
"leftWrist",
"bodyCenter",
"rightHip",
"rightKnee",
"rightAnkle",
"leftHip",
"leftKnee",
"leftAnkle"
];
var connected2 = {
leftLeg: ["leftHip", "leftKnee", "leftAnkle"],
rightLeg: ["rightHip", "rightKnee", "rightAnkle"],
torso: ["leftShoulder", "rightShoulder", "rightHip", "leftHip", "leftShoulder"],
leftArm: ["leftShoulder", "leftElbow", "leftWrist"],
rightArm: ["rightShoulder", "rightElbow", "rightWrist"],
head: []
};
// src/body/efficientpose.ts
var model4;
var lastTime3 = 0;
var cache2 = { id: 0, keypoints: [], box: [0, 0, 0, 0], boxRaw: [0, 0, 0, 0], score: 0, annotations: {} };
var skipped3 = Number.MAX_SAFE_INTEGER;
async function load2(config3) {
if (env.initial)
model4 = null;
if (!model4)
model4 = await loadModel(config3.body.modelPath);
else if (config3.debug)
log("cached model:", model4["modelUrl"]);
return model4;
}
async function max2d(inputs, minScore) {
const [width, height] = inputs.shape;
const reshaped = tf11.reshape(inputs, [height * width]);
const max5 = tf11.max(reshaped, 0);
const newScore = (await max5.data())[0];
if (newScore > minScore) {
const coordinates = tf11.argMax(reshaped, 0);
const mod3 = tf11.mod(coordinates, width);
const x = (await mod3.data())[0];
const div15 = tf11.div(coordinates, width);
const y = (await div15.data())[0];
tf11.dispose([reshaped, max5, coordinates, mod3, div15]);
return [x, y, newScore];
}
tf11.dispose([reshaped, max5]);
return [0, 0, newScore];
}
async function predict3(image28, config3) {
if (!(model4 == null ? void 0 : model4["executor"]) || !(model4 == null ? void 0 : model4.inputs[0].shape))
return [];
const skipTime = (config3.body.skipTime || 0) > now() - lastTime3;
const skipFrame = skipped3 < (config3.body.skipFrames || 0);
if (config3.skipAllowed && skipTime && skipFrame && Object.keys(cache2.keypoints).length > 0) {
skipped3++;
return [cache2];
}
skipped3 = 0;
return new Promise(async (resolve) => {
const tensor6 = tf11.tidy(() => {
var _a, _b;
const resize = tf11.image.resizeBilinear(image28, [((_a = model4 == null ? void 0 : model4.inputs[0].shape) == null ? void 0 : _a[2]) || 0, ((_b = model4 == null ? void 0 : model4.inputs[0].shape) == null ? void 0 : _b[1]) || 0], false);
const enhance2 = tf11.mul(resize, constants.tf2);
const norm = tf11.sub(enhance2, constants.tf1);
return norm;
});
let resT;
if (config3.body.enabled)
resT = model4 == null ? void 0 : model4.execute(tensor6);
lastTime3 = now();
tf11.dispose(tensor6);
if (resT) {
cache2.keypoints.length = 0;
const squeeze14 = tf11.squeeze(resT);
tf11.dispose(resT);
const stack5 = tf11.unstack(squeeze14, 2);
tf11.dispose(squeeze14);
for (let id = 0; id < stack5.length; id++) {
const [x2, y2, partScore] = await max2d(stack5[id], config3.body.minConfidence);
if (partScore > (config3.body.minConfidence || 0)) {
cache2.keypoints.push({
score: Math.round(100 * partScore) / 100,
part: kpt2[id],
positionRaw: [
x2 / model4.inputs[0].shape[2],
y2 / model4.inputs[0].shape[1]
],
position: [
Math.round(image28.shape[2] * x2 / model4.inputs[0].shape[2]),
Math.round(image28.shape[1] * y2 / model4.inputs[0].shape[1])
]
});
}
}
stack5.forEach((s) => tf11.dispose(s));
}
cache2.score = cache2.keypoints.reduce((prev, curr) => curr.score > prev ? curr.score : prev, 0);
const x = cache2.keypoints.map((a) => a.position[0]);
const y = cache2.keypoints.map((a) => a.position[1]);
cache2.box = [
Math.min(...x),
Math.min(...y),
Math.max(...x) - Math.min(...x),
Math.max(...y) - Math.min(...y)
];
const xRaw = cache2.keypoints.map((a) => a.positionRaw[0]);
const yRaw = cache2.keypoints.map((a) => a.positionRaw[1]);
cache2.boxRaw = [
Math.min(...xRaw),
Math.min(...yRaw),
Math.max(...xRaw) - Math.min(...xRaw),
Math.max(...yRaw) - Math.min(...yRaw)
];
for (const [name, indexes] of Object.entries(connected2)) {
const pt = [];
for (let i = 0; i < indexes.length - 1; i++) {
const pt0 = cache2.keypoints.find((kpt4) => kpt4.part === indexes[i]);
const pt1 = cache2.keypoints.find((kpt4) => kpt4.part === indexes[i + 1]);
if (pt0 && pt1 && pt0.score > (config3.body.minConfidence || 0) && pt1.score > (config3.body.minConfidence || 0))
pt.push([pt0.position, pt1.position]);
}
cache2.annotations[name] = pt;
}
resolve([cache2]);
});
}
// src/face/face.ts
var tf25 = __toESM(require_tfjs_esm());
// src/face/facemesh.ts
var tf15 = __toESM(require_tfjs_esm());
// src/face/blazeface.ts
var tf13 = __toESM(require_tfjs_esm());
// src/face/facemeshutil.ts
var tf12 = __toESM(require_tfjs_esm());
var getBoxSize = (box) => [Math.abs(box.endPoint[0] - box.startPoint[0]), Math.abs(box.endPoint[1] - box.startPoint[1])];
var getBoxCenter = (box) => [box.startPoint[0] + (box.endPoint[0] - box.startPoint[0]) / 2, box.startPoint[1] + (box.endPoint[1] - box.startPoint[1]) / 2, 1];
var clampBox = (box, input) => box ? [
Math.trunc(Math.max(0, box.startPoint[0])),
Math.trunc(Math.max(0, box.startPoint[1])),
Math.trunc(Math.min(input.shape[2] || 0, box.endPoint[0]) - Math.max(0, box.startPoint[0])),
Math.trunc(Math.min(input.shape[1] || 0, box.endPoint[1]) - Math.max(0, box.startPoint[1]))
] : [0, 0, 0, 0];
var getRawBox = (box, input) => box ? [
box.startPoint[0] / (input.shape[2] || 0),
box.startPoint[1] / (input.shape[1] || 0),
(box.endPoint[0] - box.startPoint[0]) / (input.shape[2] || 0),
(box.endPoint[1] - box.startPoint[1]) / (input.shape[1] || 0)
] : [0, 0, 0, 0];
var scaleBoxCoordinates = (box, factor) => {
const startPoint = [box.startPoint[0] * factor[0], box.startPoint[1] * factor[1]];
const endPoint = [box.endPoint[0] * factor[0], box.endPoint[1] * factor[1]];
return { startPoint, endPoint, landmarks: box.landmarks, confidence: box.confidence };
};
var cutAndResize = (box, image28, cropSize) => {
const h = image28.shape[1];
const w = image28.shape[2];
const cutBox = [box.startPoint[1] / h, box.startPoint[0] / w, box.endPoint[1] / h, box.endPoint[0] / w];
const crop = tf12.image.cropAndResize(image28, [cutBox], [0], cropSize);
const norm = tf12.div(crop, constants.tf255);
tf12.dispose(crop);
return norm;
};
var enlargeBox = (box, factor) => {
const center = getBoxCenter(box);
const size2 = getBoxSize(box);
const halfSize = [factor * size2[0] / 2, factor * size2[1] / 2];
return { startPoint: [center[0] - halfSize[0], center[1] - halfSize[1]], endPoint: [center[0] + halfSize[0], center[1] + halfSize[1]], landmarks: box.landmarks, confidence: box.confidence };
};
var squarifyBox = (box) => {
const centers = getBoxCenter(box);
const size2 = getBoxSize(box);
const halfSize = Math.max(...size2) / 2;
return { startPoint: [Math.round(centers[0] - halfSize), Math.round(centers[1] - halfSize)], endPoint: [Math.round(centers[0] + halfSize), Math.round(centers[1] + halfSize)], landmarks: box.landmarks, confidence: box.confidence };
};
var calculateLandmarksBoundingBox = (landmarks) => {
const x = landmarks.map((d) => d[0]);
const y = landmarks.map((d) => d[1]);
return { startPoint: [Math.min(...x), Math.min(...y)], endPoint: [Math.max(...x), Math.max(...y)], landmarks };
};
var fixedRotationMatrix = [[1, 0, 0], [0, 1, 0], [0, 0, 1]];
var normalizeRadians = (angle) => angle - 2 * Math.PI * Math.floor((angle + Math.PI) / (2 * Math.PI));
var computeRotation = (point1, point2) => normalizeRadians(Math.PI / 2 - Math.atan2(-(point2[1] - point1[1]), point2[0] - point1[0]));
var buildTranslationMatrix = (x, y) => [[1, 0, x], [0, 1, y], [0, 0, 1]];
var dot = (v1, v2) => {
let product = 0;
for (let i = 0; i < v1.length; i++)
product += v1[i] * v2[i];
return product;
};
var getColumnFrom2DArr = (arr, columnIndex) => {
const column = [];
for (let i = 0; i < arr.length; i++)
column.push(arr[i][columnIndex]);
return column;
};
var multiplyTransformMatrices = (mat1, mat2) => {
const product = [];
const size2 = mat1.length;
for (let row = 0; row < size2; row++) {
product.push([]);
for (let col = 0; col < size2; col++)
product[row].push(dot(mat1[row], getColumnFrom2DArr(mat2, col)));
}
return product;
};
var buildRotationMatrix = (rotation, center) => {
const cosA = Math.cos(rotation);
const sinA = Math.sin(rotation);
const rotationMatrix = [[cosA, -sinA, 0], [sinA, cosA, 0], [0, 0, 1]];
const translationMatrix = buildTranslationMatrix(center[0], center[1]);
const translationTimesRotation = multiplyTransformMatrices(translationMatrix, rotationMatrix);
const negativeTranslationMatrix = buildTranslationMatrix(-center[0], -center[1]);
return multiplyTransformMatrices(translationTimesRotation, negativeTranslationMatrix);
};
var invertTransformMatrix = (matrix) => {
const rotationComponent = [[matrix[0][0], matrix[1][0]], [matrix[0][1], matrix[1][1]]];
const translationComponent = [matrix[0][2], matrix[1][2]];
const invertedTranslation = [-dot(rotationComponent[0], translationComponent), -dot(rotationComponent[1], translationComponent)];
return [rotationComponent[0].concat(invertedTranslation[0]), rotationComponent[1].concat(invertedTranslation[1]), [0, 0, 1]];
};
var rotatePoint = (homogeneousCoordinate, rotationMatrix) => [dot(homogeneousCoordinate, rotationMatrix[0]), dot(homogeneousCoordinate, rotationMatrix[1])];
function generateAnchors(inputSize10) {
const spec = inputSize10 === 192 ? { strides: [4], anchors: [1] } : { strides: [inputSize10 / 16, inputSize10 / 8], anchors: [2, 6] };
const anchors3 = [];
for (let i = 0; i < spec.strides.length; i++) {
const stride = spec.strides[i];
const gridRows = Math.floor((inputSize10 + stride - 1) / stride);
const gridCols = Math.floor((inputSize10 + stride - 1) / stride);
const anchorsNum = spec.anchors[i];
for (let gridY = 0; gridY < gridRows; gridY++) {
const anchorY = stride * (gridY + 0.5);
for (let gridX = 0; gridX < gridCols; gridX++) {
const anchorX = stride * (gridX + 0.5);
for (let n = 0; n < anchorsNum; n++)
anchors3.push([anchorX, anchorY]);
}
}
}
return anchors3;
}
function transformRawCoords(coordsRaw, box, angle, rotationMatrix, inputSize10) {
const boxSize = getBoxSize(box);
const coordsScaled = coordsRaw.map((coord) => [
boxSize[0] / inputSize10 * (coord[0] - inputSize10 / 2),
boxSize[1] / inputSize10 * (coord[1] - inputSize10 / 2),
coord[2] || 0
]);
const largeAngle = angle && angle !== 0 && Math.abs(angle) > 0.2;
const coordsRotationMatrix = largeAngle ? buildRotationMatrix(angle, [0, 0]) : fixedRotationMatrix;
const coordsRotated = largeAngle ? coordsScaled.map((coord) => [...rotatePoint(coord, coordsRotationMatrix), coord[2]]) : coordsScaled;
const inverseRotationMatrix = largeAngle ? invertTransformMatrix(rotationMatrix) : fixedRotationMatrix;
const boxCenter = getBoxCenter(box);
const offsets = [dot(boxCenter, inverseRotationMatrix[0]), dot(boxCenter, inverseRotationMatrix[1])];
return coordsRotated.map((coord) => [
Math.trunc(coord[0] + offsets[0]),
Math.trunc(coord[1] + offsets[1]),
Math.trunc(coord[2] || 0)
]);
}
function correctFaceRotation(rotate, box, input, inputSize10) {
const symmetryLine = box.landmarks.length >= meshLandmarks.count ? meshLandmarks.symmetryLine : blazeFaceLandmarks.symmetryLine;
let angle = 0;
let rotationMatrix = fixedRotationMatrix;
let face4;
if (rotate && env.kernels.includes("rotatewithoffset")) {
angle = computeRotation(box.landmarks[symmetryLine[0]], box.landmarks[symmetryLine[1]]);
const largeAngle = angle && angle !== 0 && Math.abs(angle) > 0.2;
if (largeAngle) {
const center = getBoxCenter(box);
const centerRaw = [center[0] / input.shape[2], center[1] / input.shape[1]];
const rotated = tf12.image.rotateWithOffset(input, angle, 0, [centerRaw[0], centerRaw[1]]);
rotationMatrix = buildRotationMatrix(-angle, center);
face4 = cutAndResize(box, rotated, [inputSize10, inputSize10]);
tf12.dispose(rotated);
} else {
face4 = cutAndResize(box, input, [inputSize10, inputSize10]);
}
} else {
face4 = cutAndResize(box, input, [inputSize10, inputSize10]);
}
return [angle, rotationMatrix, face4];
}
var findFaceCenter = (mesh) => {
const x = mesh.map((m) => m[0]);
const y = mesh.map((m) => m[1]);
return [Math.min(...x) + (Math.max(...x) - Math.min(...x)) / 2, Math.min(...y) + (Math.max(...y) - Math.min(...y)) / 2];
};
var calculateFaceBox = (mesh, previousBox) => {
const center = findFaceCenter(mesh);
const boxSize = getBoxSize(previousBox);
const calculatedBox = {
startPoint: [center[0] - boxSize[0] / 2, center[1] - boxSize[1] / 2],
endPoint: [center[0] + boxSize[0] / 2, center[1] + boxSize[1] / 2]
};
return calculatedBox;
};
// src/face/blazeface.ts
var keypointsCount = 6;
var faceBoxScaleFactor = 1.4;
var model5;
var anchors = null;
var inputSize4 = 0;
var inputSizeT = null;
var size = () => inputSize4;
async function load3(config3) {
var _a;
if (env.initial)
model5 = null;
if (!model5)
model5 = await loadModel((_a = config3.face.detector) == null ? void 0 : _a.modelPath);
else if (config3.debug)
log("cached model:", model5["modelUrl"]);
inputSize4 = model5["executor"] && model5.inputs[0].shape ? model5.inputs[0].shape[2] : 256;
inputSizeT = tf13.scalar(inputSize4, "int32");
anchors = tf13.tensor2d(generateAnchors(inputSize4));
return model5;
}
function decodeBoxes2(boxOutputs) {
if (!anchors || !inputSizeT)
return tf13.zeros([0, 0]);
const t2 = {};
t2.boxStarts = tf13.slice(boxOutputs, [0, 1], [-1, 2]);
t2.centers = tf13.add(t2.boxStarts, anchors);
t2.boxSizes = tf13.slice(boxOutputs, [0, 3], [-1, 2]);
t2.boxSizesNormalized = tf13.div(t2.boxSizes, inputSizeT);
t2.centersNormalized = tf13.div(t2.centers, inputSizeT);
t2.halfBoxSize = tf13.div(t2.boxSizesNormalized, constants.tf2);
t2.starts = tf13.sub(t2.centersNormalized, t2.halfBoxSize);
t2.ends = tf13.add(t2.centersNormalized, t2.halfBoxSize);
t2.startNormalized = tf13.mul(t2.starts, inputSizeT);
t2.endNormalized = tf13.mul(t2.ends, inputSizeT);
const boxes = tf13.concat2d([t2.startNormalized, t2.endNormalized], 1);
Object.keys(t2).forEach((tensor6) => tf13.dispose(t2[tensor6]));
return boxes;
}
async function getBoxes(inputImage, config3) {
var _a, _b, _c, _d;
if (!inputImage || inputImage["isDisposedInternal"] || inputImage.shape.length !== 4 || inputImage.shape[1] < 1 || inputImage.shape[2] < 1)
return [];
const t2 = {};
t2.resized = tf13.image.resizeBilinear(inputImage, [inputSize4, inputSize4]);
t2.div = tf13.div(t2.resized, constants.tf127);
t2.normalized = tf13.sub(t2.div, constants.tf05);
const res = model5 == null ? void 0 : model5.execute(t2.normalized);
if (Array.isArray(res) && res.length > 2) {
const sorted = res.sort((a, b) => a.size - b.size);
t2.concat384 = tf13.concat([sorted[0], sorted[2]], 2);
t2.concat512 = tf13.concat([sorted[1], sorted[3]], 2);
t2.concat = tf13.concat([t2.concat512, t2.concat384], 1);
t2.batch = tf13.squeeze(t2.concat, [0]);
} else if (Array.isArray(res)) {
t2.batch = tf13.squeeze(res[0]);
} else {
t2.batch = tf13.squeeze(res);
}
tf13.dispose(res);
t2.boxes = decodeBoxes2(t2.batch);
t2.logits = tf13.slice(t2.batch, [0, 0], [-1, 1]);
t2.sigmoid = tf13.sigmoid(t2.logits);
t2.scores = tf13.squeeze(t2.sigmoid);
t2.nms = await tf13.image.nonMaxSuppressionAsync(t2.boxes, t2.scores, ((_a = config3.face.detector) == null ? void 0 : _a.maxDetected) || 0, ((_b = config3.face.detector) == null ? void 0 : _b.iouThreshold) || 0, ((_c = config3.face.detector) == null ? void 0 : _c.minConfidence) || 0);
const nms = await t2.nms.array();
const boxes = [];
const scores = await t2.scores.data();
for (let i = 0; i < nms.length; i++) {
const confidence = scores[nms[i]];
if (confidence > (((_d = config3.face.detector) == null ? void 0 : _d.minConfidence) || 0)) {
const b = {};
b.bbox = tf13.slice(t2.boxes, [nms[i], 0], [1, -1]);
b.slice = tf13.slice(t2.batch, [nms[i], keypointsCount - 1], [1, -1]);
b.squeeze = tf13.squeeze(b.slice);
b.landmarks = tf13.reshape(b.squeeze, [keypointsCount, -1]);
const points = await b.bbox.data();
const rawBox = {
startPoint: [points[0], points[1]],
endPoint: [points[2], points[3]],
landmarks: await b.landmarks.array(),
confidence
};
const scaledBox = scaleBoxCoordinates(rawBox, [(inputImage.shape[2] || 0) / inputSize4, (inputImage.shape[1] || 0) / inputSize4]);
const enlargedBox = enlargeBox(scaledBox, config3.face["scale"] || faceBoxScaleFactor);
const squaredBox = squarifyBox(enlargedBox);
boxes.push(squaredBox);
Object.keys(b).forEach((tensor6) => tf13.dispose(b[tensor6]));
}
}
Object.keys(t2).forEach((tensor6) => tf13.dispose(t2[tensor6]));
return boxes;
}
// src/face/iris.ts
var tf14 = __toESM(require_tfjs_esm());
var model6;
var inputSize5 = 0;
var irisEnlarge = 2.3;
var leftOutline = meshAnnotations.leftEyeLower0;
var rightOutline = meshAnnotations.rightEyeLower0;
var eyeLandmarks = {
leftBounds: [leftOutline[0], leftOutline[leftOutline.length - 1]],
rightBounds: [rightOutline[0], rightOutline[rightOutline.length - 1]]
};
var irisLandmarks = {
upperCenter: 3,
lowerCenter: 4,
index: 71,
numCoordinates: 76
};
async function load4(config3) {
var _a, _b;
if (env.initial)
model6 = null;
if (!model6)
model6 = await loadModel((_a = config3.face.iris) == null ? void 0 : _a.modelPath);
else if (config3.debug)
log("cached model:", model6["modelUrl"]);
inputSize5 = (model6 == null ? void 0 : model6["executor"]) && ((_b = model6.inputs) == null ? void 0 : _b[0].shape) ? model6.inputs[0].shape[2] : 0;
if (inputSize5 === -1)
inputSize5 = 64;
return model6;
}
function replaceIrisCoords(rawCoords, newCoords, prefix, keys) {
for (let i = 0; i < irisIndices.length; i++) {
const { key, indices } = irisIndices[i];
const originalIndices = meshAnnotations[`${prefix}${key}`];
if (!keys || keys.includes(key)) {
for (let j = 0; j < indices.length; j++) {
const index2 = indices[j];
rawCoords[originalIndices[j]] = [
newCoords[index2][0],
newCoords[index2][1],
(newCoords[index2][2] + rawCoords[originalIndices[j]][2]) / 2
];
}
}
}
}
var getLeftToRightEyeDepthDifference = (rawCoords) => {
const leftEyeZ = rawCoords[eyeLandmarks.leftBounds[0]][2];
const rightEyeZ = rawCoords[eyeLandmarks.rightBounds[0]][2];
return leftEyeZ - rightEyeZ;
};
var getEyeBox = (rawCoords, face4, eyeInnerCornerIndex, eyeOuterCornerIndex, meshSize, flip = false) => {
const box = squarifyBox(enlargeBox(calculateLandmarksBoundingBox([rawCoords[eyeInnerCornerIndex], rawCoords[eyeOuterCornerIndex]]), irisEnlarge));
const boxSize = getBoxSize(box);
let crop = tf14.image.cropAndResize(face4, [[
box.startPoint[1] / meshSize,
box.startPoint[0] / meshSize,
box.endPoint[1] / meshSize,
box.endPoint[0] / meshSize
]], [0], [inputSize5, inputSize5]);
if (flip && env.kernels.includes("flipleftright")) {
const flipped = tf14.image.flipLeftRight(crop);
tf14.dispose(crop);
crop = flipped;
}
return { box, boxSize, crop };
};
var getEyeCoords = (eyeData, eyeBox, eyeBoxSize, flip = false) => {
const eyeRawCoords = [];
for (let i = 0; i < irisLandmarks.numCoordinates; i++) {
const x = eyeData[i * 3];
const y = eyeData[i * 3 + 1];
const z = eyeData[i * 3 + 2];
eyeRawCoords.push([
(flip ? 1 - x / inputSize5 : x / inputSize5) * eyeBoxSize[0] + eyeBox.startPoint[0],
y / inputSize5 * eyeBoxSize[1] + eyeBox.startPoint[1],
z
]);
}
return { rawCoords: eyeRawCoords, iris: eyeRawCoords.slice(irisLandmarks.index) };
};
var getAdjustedIrisCoords = (rawCoords, irisCoords, direction) => {
const upperCenterZ = rawCoords[meshAnnotations[`${direction}EyeUpper0`][irisLandmarks.upperCenter]][2];
const lowerCenterZ = rawCoords[meshAnnotations[`${direction}EyeLower0`][irisLandmarks.lowerCenter]][2];
const averageZ = (upperCenterZ + lowerCenterZ) / 2;
return irisCoords.map((coord, i) => {
let z = averageZ;
if (i === 2) {
z = upperCenterZ;
} else if (i === 4) {
z = lowerCenterZ;
}
return [coord[0], coord[1], z];
});
};
async function augmentIris(rawCoords, face4, meshSize) {
if (!(model6 == null ? void 0 : model6["executor"]))
return rawCoords;
const { box: leftEyeBox, boxSize: leftEyeBoxSize, crop: leftEyeCrop } = getEyeBox(rawCoords, face4, eyeLandmarks.leftBounds[0], eyeLandmarks.leftBounds[1], meshSize, true);
const { box: rightEyeBox, boxSize: rightEyeBoxSize, crop: rightEyeCrop } = getEyeBox(rawCoords, face4, eyeLandmarks.rightBounds[0], eyeLandmarks.rightBounds[1], meshSize, true);
const combined = tf14.concat([leftEyeCrop, rightEyeCrop]);
tf14.dispose(leftEyeCrop);
tf14.dispose(rightEyeCrop);
const eyePredictions = model6.execute(combined);
tf14.dispose(combined);
const eyePredictionsData = await eyePredictions.data();
tf14.dispose(eyePredictions);
const leftEyeData = eyePredictionsData.slice(0, irisLandmarks.numCoordinates * 3);
const { rawCoords: leftEyeRawCoords, iris: leftIrisRawCoords } = getEyeCoords(leftEyeData, leftEyeBox, leftEyeBoxSize, true);
const rightEyeData = eyePredictionsData.slice(irisLandmarks.numCoordinates * 3);
const { rawCoords: rightEyeRawCoords, iris: rightIrisRawCoords } = getEyeCoords(rightEyeData, rightEyeBox, rightEyeBoxSize, false);
const leftToRightEyeDepthDifference = getLeftToRightEyeDepthDifference(rawCoords);
if (Math.abs(leftToRightEyeDepthDifference) < 30) {
replaceIrisCoords(rawCoords, leftEyeRawCoords, "left", null);
replaceIrisCoords(rawCoords, rightEyeRawCoords, "right", null);
} else if (leftToRightEyeDepthDifference < 1) {
replaceIrisCoords(rawCoords, leftEyeRawCoords, "left", ["EyeUpper0", "EyeLower0"]);
} else {
replaceIrisCoords(rawCoords, rightEyeRawCoords, "right", ["EyeUpper0", "EyeLower0"]);
}
const adjustedLeftIrisCoords = getAdjustedIrisCoords(rawCoords, leftIrisRawCoords, "left");
const adjustedRightIrisCoords = getAdjustedIrisCoords(rawCoords, rightIrisRawCoords, "right");
const newCoords = rawCoords.concat(adjustedLeftIrisCoords).concat(adjustedRightIrisCoords);
return newCoords;
}
// src/face/attention.ts
async function augment(rawCoords, results) {
var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j;
const t2 = {
lips: await ((_b = (_a = results.filter((r) => r.size === 160)) == null ? void 0 : _a[0]) == null ? void 0 : _b.data()),
irisL: await ((_d = (_c = results.filter((r) => r.size === 10)) == null ? void 0 : _c[0]) == null ? void 0 : _d.data()),
eyeL: await ((_f = (_e = results.filter((r) => r.size === 142)) == null ? void 0 : _e[0]) == null ? void 0 : _f.data()),
irisR: await ((_h = (_g = results.filter((r) => r.size === 10)) == null ? void 0 : _g[1]) == null ? void 0 : _h.data()),
eyeR: await ((_j = (_i = results.filter((r) => r.size === 142)) == null ? void 0 : _i[1]) == null ? void 0 : _j.data())
};
for (const val of Object.values(t2)) {
if (!val)
return rawCoords;
}
const irisLDepth = LANDMARKS_REFINEMENT_LEFT_EYE_CONFIG.reduce((prev, curr) => prev += rawCoords[curr][2], 0) / LANDMARKS_REFINEMENT_LEFT_EYE_CONFIG.length;
for (let i = 0; i < t2.irisL.length / 2; i++)
rawCoords.push([t2.irisL[2 * i + 0], t2.irisL[2 * i + 1], irisLDepth]);
const irisRDepth = LANDMARKS_REFINEMENT_RIGHT_EYE_CONFIG.reduce((prev, curr) => prev += rawCoords[curr][2], 0) / LANDMARKS_REFINEMENT_RIGHT_EYE_CONFIG.length;
for (let i = 0; i < t2.irisR.length / 2; i++)
rawCoords.push([t2.irisR[2 * i + 0], t2.irisR[2 * i + 1], irisRDepth]);
for (let i = 0; i < t2.eyeL.length / 2; i++)
rawCoords[LANDMARKS_REFINEMENT_LEFT_EYE_CONFIG[i]] = [t2.eyeL[2 * i + 0], t2.eyeL[2 * i + 1], rawCoords[LANDMARKS_REFINEMENT_LEFT_EYE_CONFIG[i]][2]];
for (let i = 0; i < t2.eyeR.length / 2; i++)
rawCoords[LANDMARKS_REFINEMENT_RIGHT_EYE_CONFIG[i]] = [t2.eyeR[2 * i + 0], t2.eyeR[2 * i + 1], rawCoords[LANDMARKS_REFINEMENT_RIGHT_EYE_CONFIG[i]][2]];
for (let i = 0; i < t2.lips.length / 2; i++)
rawCoords[LANDMARKS_REFINEMENT_LIPS_CONFIG[i]] = [t2.lips[2 * i + 0], t2.lips[2 * i + 1], rawCoords[LANDMARKS_REFINEMENT_LIPS_CONFIG[i]][2]];
return rawCoords;
}
// src/face/facemesh.ts
var cache3 = {
boxes: [],
skipped: Number.MAX_SAFE_INTEGER,
timestamp: 0
};
var model7 = null;
var inputSize6 = 0;
async function predict4(input, config3) {
var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j;
if (!(model7 == null ? void 0 : model7["executor"]))
return [];
const skipTime = (((_a = config3.face.detector) == null ? void 0 : _a.skipTime) || 0) > now() - cache3.timestamp;
const skipFrame = cache3.skipped < (((_b = config3.face.detector) == null ? void 0 : _b.skipFrames) || 0);
if (!config3.skipAllowed || !skipTime || !skipFrame || cache3.boxes.length === 0) {
cache3.boxes = await getBoxes(input, config3);
cache3.timestamp = now();
cache3.skipped = 0;
} else {
cache3.skipped++;
}
const faces = [];
const newCache = [];
let id = 0;
const size2 = inputSize6;
for (let i = 0; i < cache3.boxes.length; i++) {
const box = cache3.boxes[i];
let angle = 0;
let rotationMatrix;
const face4 = {
id: id++,
mesh: [],
meshRaw: [],
box: [0, 0, 0, 0],
boxRaw: [0, 0, 0, 0],
score: 0,
boxScore: 0,
faceScore: 0,
annotations: {}
};
[angle, rotationMatrix, face4.tensor] = correctFaceRotation((_c = config3.face.detector) == null ? void 0 : _c.rotation, box, input, ((_d = config3.face.mesh) == null ? void 0 : _d.enabled) ? inputSize6 : size());
if (config3.filter.equalization) {
const equilized = face4.tensor ? await histogramEqualization(face4.tensor) : void 0;
tf15.dispose(face4.tensor);
if (equilized)
face4.tensor = equilized;
}
face4.boxScore = Math.round(100 * box.confidence) / 100;
if (!((_e = config3.face.mesh) == null ? void 0 : _e.enabled)) {
face4.box = clampBox(box, input);
face4.boxRaw = getRawBox(box, input);
face4.score = face4.boxScore;
face4.mesh = box.landmarks.map((pt) => [
(box.startPoint[0] + box.endPoint[0]) / 2 + (box.endPoint[0] + box.startPoint[0]) * pt[0] / size(),
(box.startPoint[1] + box.endPoint[1]) / 2 + (box.endPoint[1] + box.startPoint[1]) * pt[1] / size()
]);
face4.meshRaw = face4.mesh.map((pt) => [pt[0] / (input.shape[2] || 0), pt[1] / (input.shape[1] || 0), (pt[2] || 0) / size2]);
for (const key of Object.keys(blazeFaceLandmarks)) {
face4.annotations[key] = [face4.mesh[blazeFaceLandmarks[key]]];
}
} else if (!model7) {
if (config3.debug)
log("face mesh detection requested, but model is not loaded");
} else {
if (((_f = config3.face.attention) == null ? void 0 : _f.enabled) && !env.kernels.includes("atan2")) {
config3.face.attention.enabled = false;
tf15.dispose(face4.tensor);
return faces;
}
const results = model7.execute(face4.tensor);
const confidenceT = results.find((t2) => t2.shape[t2.shape.length - 1] === 1);
const faceConfidence = await confidenceT.data();
face4.faceScore = Math.round(100 * faceConfidence[0]) / 100;
if (face4.faceScore < (((_g = config3.face.detector) == null ? void 0 : _g.minConfidence) || 1)) {
box.confidence = face4.faceScore;
if (config3.face.mesh.keepInvalid) {
face4.box = clampBox(box, input);
face4.boxRaw = getRawBox(box, input);
face4.score = face4.boxScore;
face4.mesh = box.landmarks.map((pt) => [
(box.startPoint[0] + box.endPoint[0]) / 2 + (box.endPoint[0] + box.startPoint[0]) * pt[0] / size(),
(box.startPoint[1] + box.endPoint[1]) / 2 + (box.endPoint[1] + box.startPoint[1]) * pt[1] / size()
]);
face4.meshRaw = face4.mesh.map((pt) => [pt[0] / (input.shape[2] || 1), pt[1] / (input.shape[1] || 1), (pt[2] || 0) / size2]);
for (const key of Object.keys(blazeFaceLandmarks)) {
face4.annotations[key] = [face4.mesh[blazeFaceLandmarks[key]]];
}
}
} else {
const meshT = results.find((t2) => t2.shape[t2.shape.length - 1] === 1404);
const coordsReshaped = tf15.reshape(meshT, [-1, 3]);
let rawCoords = await coordsReshaped.array();
tf15.dispose(coordsReshaped);
if ((_h = config3.face.attention) == null ? void 0 : _h.enabled) {
rawCoords = await augment(rawCoords, results);
} else if ((_i = config3.face.iris) == null ? void 0 : _i.enabled) {
rawCoords = await augmentIris(rawCoords, face4.tensor, inputSize6);
}
face4.mesh = transformRawCoords(rawCoords, box, angle, rotationMatrix, inputSize6);
face4.meshRaw = face4.mesh.map((pt) => [pt[0] / (input.shape[2] || 0), pt[1] / (input.shape[1] || 0), (pt[2] || 0) / size2]);
for (const key of Object.keys(meshAnnotations))
face4.annotations[key] = meshAnnotations[key].map((index2) => face4.mesh[index2]);
face4.score = face4.faceScore;
const calculatedBox = { ...calculateFaceBox(face4.mesh, box), confidence: box.confidence, landmarks: box.landmarks };
face4.box = clampBox(calculatedBox, input);
face4.boxRaw = getRawBox(calculatedBox, input);
newCache.push(calculatedBox);
}
tf15.dispose(results);
}
if (face4.score > (((_j = config3.face.detector) == null ? void 0 : _j.minConfidence) || 1))
faces.push(face4);
else
tf15.dispose(face4.tensor);
}
cache3.boxes = newCache;
return faces;
}
async function load5(config3) {
var _a, _b, _c, _d, _e, _f;
if (env.initial)
model7 = null;
if (((_a = config3.face.attention) == null ? void 0 : _a.enabled) && (model7 == null ? void 0 : model7["signature"])) {
if (Object.keys(((_b = model7 == null ? void 0 : model7["signature"]) == null ? void 0 : _b.outputs) || {}).length < 6)
model7 = null;
}
if (!model7) {
if ((_c = config3.face.attention) == null ? void 0 : _c.enabled)
model7 = await loadModel(config3.face.attention.modelPath);
else
model7 = await loadModel((_d = config3.face.mesh) == null ? void 0 : _d.modelPath);
} else if (config3.debug) {
log("cached model:", model7["modelUrl"]);
}
inputSize6 = model7["executor"] && ((_e = model7 == null ? void 0 : model7.inputs) == null ? void 0 : _e[0].shape) ? (_f = model7 == null ? void 0 : model7.inputs) == null ? void 0 : _f[0].shape[2] : 256;
return model7;
}
var triangulation = TRI468;
var uvmap = UV468;
// src/gear/emotion.ts
var tf16 = __toESM(require_tfjs_esm());
var annotations = ["angry", "disgust", "fear", "happy", "sad", "surprise", "neutral"];
var model8;
var last3 = [];
var lastCount = 0;
var lastTime4 = 0;
var skipped4 = Number.MAX_SAFE_INTEGER;
async function load6(config3) {
var _a;
if (env.initial)
model8 = null;
if (!model8)
model8 = await loadModel((_a = config3.face.emotion) == null ? void 0 : _a.modelPath);
else if (config3.debug)
log("cached model:", model8["modelUrl"]);
return model8;
}
async function predict5(image28, config3, idx, count2) {
var _a, _b;
if (!model8)
return [];
const skipFrame = skipped4 < (((_a = config3.face.emotion) == null ? void 0 : _a.skipFrames) || 0);
const skipTime = (((_b = config3.face.emotion) == null ? void 0 : _b.skipTime) || 0) > now() - lastTime4;
if (config3.skipAllowed && skipTime && skipFrame && lastCount === count2 && last3[idx] && last3[idx].length > 0) {
skipped4++;
return last3[idx];
}
skipped4 = 0;
return new Promise(async (resolve) => {
var _a2;
const obj = [];
if ((_a2 = config3.face.emotion) == null ? void 0 : _a2.enabled) {
const t2 = {};
const inputSize10 = (model8 == null ? void 0 : model8.inputs[0].shape) ? model8.inputs[0].shape[2] : 0;
t2.resize = tf16.image.resizeBilinear(image28, [inputSize10, inputSize10], false);
t2.channels = tf16.mul(t2.resize, constants.rgb);
t2.grayscale = tf16.sum(t2.channels, 3, true);
t2.grayscaleSub = tf16.sub(t2.grayscale, constants.tf05);
t2.grayscaleMul = tf16.mul(t2.grayscaleSub, constants.tf2);
t2.emotion = model8 == null ? void 0 : model8.execute(t2.grayscaleMul);
lastTime4 = now();
const data = await t2.emotion.data();
for (let i = 0; i < data.length; i++) {
if (data[i] > (config3.face.emotion.minConfidence || 0))
obj.push({ score: Math.min(0.99, Math.trunc(100 * data[i]) / 100), emotion: annotations[i] });
}
obj.sort((a, b) => b.score - a.score);
Object.keys(t2).forEach((tensor6) => tf16.dispose(t2[tensor6]));
}
last3[idx] = obj;
lastCount = count2;
resolve(obj);
});
}
// src/face/faceres.ts
var tf17 = __toESM(require_tfjs_esm());
var model9;
var last4 = [];
var lastTime5 = 0;
var lastCount2 = 0;
var skipped5 = Number.MAX_SAFE_INTEGER;
async function load7(config3) {
var _a;
if (env.initial)
model9 = null;
if (!model9)
model9 = await loadModel((_a = config3.face.description) == null ? void 0 : _a.modelPath);
else if (config3.debug)
log("cached model:", model9["modelUrl"]);
return model9;
}
function enhance(input) {
const tensor6 = input.image || input.tensor || input;
if (!(model9 == null ? void 0 : model9.inputs[0].shape))
return tensor6;
const crop = tf17.image.resizeBilinear(tensor6, [model9.inputs[0].shape[2], model9.inputs[0].shape[1]], false);
const norm = tf17.mul(crop, constants.tf255);
tf17.dispose(crop);
return norm;
}
async function predict6(image28, config3, idx, count2) {
var _a, _b, _c, _d;
const obj = {
age: 0,
gender: "unknown",
genderScore: 0,
descriptor: []
};
if (!(model9 == null ? void 0 : model9["executor"]))
return obj;
const skipFrame = skipped5 < (((_a = config3.face.description) == null ? void 0 : _a.skipFrames) || 0);
const skipTime = (((_b = config3.face.description) == null ? void 0 : _b.skipTime) || 0) > now() - lastTime5;
if (config3.skipAllowed && skipFrame && skipTime && lastCount2 === count2 && ((_c = last4 == null ? void 0 : last4[idx]) == null ? void 0 : _c.age) > 0 && ((_d = last4 == null ? void 0 : last4[idx]) == null ? void 0 : _d.genderScore) > 0) {
skipped5++;
return last4[idx];
}
skipped5 = 0;
return new Promise(async (resolve) => {
var _a2;
if ((_a2 = config3.face.description) == null ? void 0 : _a2.enabled) {
const enhanced = enhance(image28);
const resT = model9 == null ? void 0 : model9.execute(enhanced);
lastTime5 = now();
tf17.dispose(enhanced);
const genderT = resT.find((t2) => t2.shape[1] === 1);
const gender2 = await genderT.data();
const confidence = Math.trunc(200 * Math.abs(gender2[0] - 0.5)) / 100;
if (confidence > (config3.face.description.minConfidence || 0)) {
obj.gender = gender2[0] <= 0.5 ? "female" : "male";
obj.genderScore = Math.min(0.99, confidence);
}
const argmax = tf17.argMax(resT.find((t2) => t2.shape[1] === 100), 1);
const ageIdx = (await argmax.data())[0];
tf17.dispose(argmax);
const ageT = resT.find((t2) => t2.shape[1] === 100);
const all2 = await ageT.data();
obj.age = Math.round(all2[ageIdx - 1] > all2[ageIdx + 1] ? 10 * ageIdx - 100 * all2[ageIdx - 1] : 10 * ageIdx + 100 * all2[ageIdx + 1]) / 10;
if (Number.isNaN(gender2[0]) || Number.isNaN(all2[0]))
log("faceres error:", { model: model9, result: resT });
const desc = resT.find((t2) => t2.shape[1] === 1024);
const descriptor = desc ? await desc.data() : [];
obj.descriptor = Array.from(descriptor);
resT.forEach((t2) => tf17.dispose(t2));
}
last4[idx] = obj;
lastCount2 = count2;
resolve(obj);
});
}
// src/face/mask.ts
var expandFact = 0.1;
var alpha = 0.5;
function insidePoly(x, y, polygon) {
let inside = false;
let j = polygon.length - 1;
for (let i = 0; i < polygon.length; j = i++) {
if (polygon[i].y > y !== polygon[j].y > y && x < (polygon[j].x - polygon[i].x) * (y - polygon[i].y) / (polygon[j].y - polygon[i].y) + polygon[i].x)
inside = !inside;
}
return inside;
}
async function mask(face4) {
if (!face4.tensor)
return face4.tensor;
if (!face4.mesh || face4.mesh.length < 100)
return face4.tensor;
const width = face4.tensor.shape[2] || 0;
const height = face4.tensor.shape[1] || 0;
const buffer = await face4.tensor.buffer();
let silhouette = [];
for (const pt of meshAnnotations.silhouette)
silhouette.push({ x: (face4.mesh[pt][0] - face4.box[0]) / face4.box[2], y: (face4.mesh[pt][1] - face4.box[1]) / face4.box[3] });
if (expandFact && expandFact > 0)
silhouette = silhouette.map((pt) => ({ x: pt.x > 0.5 ? pt.x + expandFact : pt.x - expandFact, y: pt.y > 0.5 ? pt.y + expandFact : pt.y - expandFact }));
for (let x = 0; x < width; x++) {
for (let y = 0; y < height; y++) {
const inside = insidePoly(x / width, y / width, silhouette);
if (!inside) {
buffer.set(alpha * buffer.get(0, y, x, 0), 0, y, x, 0);
buffer.set(alpha * buffer.get(0, y, x, 1), 0, y, x, 1);
buffer.set(alpha * buffer.get(0, y, x, 2), 0, y, x, 2);
}
}
}
const output = buffer.toTensor();
return output;
}
// src/face/antispoof.ts
var tf18 = __toESM(require_tfjs_esm());
var model10;
var cached = [];
var skipped6 = Number.MAX_SAFE_INTEGER;
var lastCount3 = 0;
var lastTime6 = 0;
async function load8(config3) {
var _a;
if (env.initial)
model10 = null;
if (!model10)
model10 = await loadModel((_a = config3.face.antispoof) == null ? void 0 : _a.modelPath);
else if (config3.debug)
log("cached model:", model10["modelUrl"]);
return model10;
}
async function predict7(image28, config3, idx, count2) {
var _a, _b;
if (!(model10 == null ? void 0 : model10["executor"]))
return 0;
const skipTime = (((_a = config3.face.antispoof) == null ? void 0 : _a.skipTime) || 0) > now() - lastTime6;
const skipFrame = skipped6 < (((_b = config3.face.antispoof) == null ? void 0 : _b.skipFrames) || 0);
if (config3.skipAllowed && skipTime && skipFrame && lastCount3 === count2 && cached[idx]) {
skipped6++;
return cached[idx];
}
skipped6 = 0;
return new Promise(async (resolve) => {
const resize = tf18.image.resizeBilinear(image28, [(model10 == null ? void 0 : model10.inputs[0].shape) ? model10.inputs[0].shape[2] : 0, (model10 == null ? void 0 : model10.inputs[0].shape) ? model10.inputs[0].shape[1] : 0], false);
const res = model10 == null ? void 0 : model10.execute(resize);
const num = (await res.data())[0];
cached[idx] = Math.round(100 * num) / 100;
lastCount3 = count2;
lastTime6 = now();
tf18.dispose([resize, res]);
resolve(cached[idx]);
});
}
// src/face/liveness.ts
var tf19 = __toESM(require_tfjs_esm());
var model11;
var cached2 = [];
var skipped7 = Number.MAX_SAFE_INTEGER;
var lastCount4 = 0;
var lastTime7 = 0;
async function load9(config3) {
var _a;
if (env.initial)
model11 = null;
if (!model11)
model11 = await loadModel((_a = config3.face.liveness) == null ? void 0 : _a.modelPath);
else if (config3.debug)
log("cached model:", model11["modelUrl"]);
return model11;
}
async function predict8(image28, config3, idx, count2) {
var _a, _b;
if (!(model11 == null ? void 0 : model11["executor"]))
return 0;
const skipTime = (((_a = config3.face.liveness) == null ? void 0 : _a.skipTime) || 0) > now() - lastTime7;
const skipFrame = skipped7 < (((_b = config3.face.liveness) == null ? void 0 : _b.skipFrames) || 0);
if (config3.skipAllowed && skipTime && skipFrame && lastCount4 === count2 && cached2[idx]) {
skipped7++;
return cached2[idx];
}
skipped7 = 0;
return new Promise(async (resolve) => {
const resize = tf19.image.resizeBilinear(image28, [(model11 == null ? void 0 : model11.inputs[0].shape) ? model11.inputs[0].shape[2] : 0, (model11 == null ? void 0 : model11.inputs[0].shape) ? model11.inputs[0].shape[1] : 0], false);
const res = model11 == null ? void 0 : model11.execute(resize);
const num = (await res.data())[0];
cached2[idx] = Math.round(100 * num) / 100;
lastCount4 = count2;
lastTime7 = now();
tf19.dispose([resize, res]);
resolve(cached2[idx]);
});
}
// src/gear/gear.ts
var tf20 = __toESM(require_tfjs_esm());
var model12;
var last5 = [];
var raceNames = ["white", "black", "asian", "indian", "other"];
var ageWeights = [15, 23, 28, 35.5, 45.5, 55.5, 65];
var lastCount5 = 0;
var lastTime8 = 0;
var skipped8 = Number.MAX_SAFE_INTEGER;
async function load10(config3) {
var _a;
if (env.initial)
model12 = null;
if (!model12)
model12 = await loadModel((_a = config3.face.gear) == null ? void 0 : _a.modelPath);
else if (config3.debug)
log("cached model:", model12["modelUrl"]);
return model12;
}
async function predict9(image28, config3, idx, count2) {
var _a, _b;
if (!model12)
return { age: 0, gender: "unknown", genderScore: 0, race: [] };
const skipFrame = skipped8 < (((_a = config3.face.gear) == null ? void 0 : _a.skipFrames) || 0);
const skipTime = (((_b = config3.face.gear) == null ? void 0 : _b.skipTime) || 0) > now() - lastTime8;
if (config3.skipAllowed && skipTime && skipFrame && lastCount5 === count2 && last5[idx]) {
skipped8++;
return last5[idx];
}
skipped8 = 0;
return new Promise(async (resolve) => {
var _a2, _b2;
if (!(model12 == null ? void 0 : model12.inputs[0].shape))
return;
const t2 = {};
const box = [[0, 0.1, 0.9, 0.9]];
t2.resize = tf20.image.cropAndResize(image28, box, [0], [model12.inputs[0].shape[2], model12.inputs[0].shape[1]]);
const obj = { age: 0, gender: "unknown", genderScore: 0, race: [] };
if ((_a2 = config3.face.gear) == null ? void 0 : _a2.enabled)
[t2.age, t2.gender, t2.race] = model12.execute(t2.resize, ["age_output", "gender_output", "race_output"]);
const gender2 = await t2.gender.data();
obj.gender = gender2[0] > gender2[1] ? "male" : "female";
obj.genderScore = Math.round(100 * (gender2[0] > gender2[1] ? gender2[0] : gender2[1])) / 100;
const race = await t2.race.data();
for (let i = 0; i < race.length; i++) {
if (race[i] > (((_b2 = config3.face.gear) == null ? void 0 : _b2.minConfidence) || 0.2))
obj.race.push({ score: Math.round(100 * race[i]) / 100, race: raceNames[i] });
}
obj.race.sort((a, b) => b.score - a.score);
const ageDistribution = Array.from(await t2.age.data());
const ageSorted = ageDistribution.map((a, i) => [ageWeights[i], a]).sort((a, b) => b[1] - a[1]);
let age2 = ageSorted[0][0];
for (let i = 1; i < ageSorted.length; i++)
age2 += ageSorted[i][1] * (ageSorted[i][0] - age2);
obj.age = Math.round(10 * age2) / 10;
Object.keys(t2).forEach((tensor6) => tf20.dispose(t2[tensor6]));
last5[idx] = obj;
lastCount5 = count2;
lastTime8 = now();
resolve(obj);
});
}
// src/gear/ssrnet-age.ts
var tf21 = __toESM(require_tfjs_esm());
var model13;
var last6 = [];
var lastCount6 = 0;
var lastTime9 = 0;
var skipped9 = Number.MAX_SAFE_INTEGER;
async function load11(config3) {
if (env.initial)
model13 = null;
if (!model13)
model13 = await loadModel(config3.face["ssrnet"].modelPathAge);
else if (config3.debug)
log("cached model:", model13["modelUrl"]);
return model13;
}
async function predict10(image28, config3, idx, count2) {
var _a, _b, _c, _d;
if (!model13)
return { age: 0 };
const skipFrame = skipped9 < (((_a = config3.face["ssrnet"]) == null ? void 0 : _a.skipFrames) || 0);
const skipTime = (((_b = config3.face["ssrnet"]) == null ? void 0 : _b.skipTime) || 0) > now() - lastTime9;
if (config3.skipAllowed && skipFrame && skipTime && lastCount6 === count2 && ((_c = last6[idx]) == null ? void 0 : _c.age) && ((_d = last6[idx]) == null ? void 0 : _d.age) > 0) {
skipped9++;
return last6[idx];
}
skipped9 = 0;
return new Promise(async (resolve) => {
var _a2;
if (!(model13 == null ? void 0 : model13.inputs) || !model13.inputs[0] || !model13.inputs[0].shape)
return;
const t2 = {};
t2.resize = tf21.image.resizeBilinear(image28, [model13.inputs[0].shape[2], model13.inputs[0].shape[1]], false);
t2.enhance = tf21.mul(t2.resize, constants.tf255);
const obj = { age: 0 };
if ((_a2 = config3.face["ssrnet"]) == null ? void 0 : _a2.enabled)
t2.age = model13.execute(t2.enhance);
if (t2.age) {
const data = await t2.age.data();
obj.age = Math.trunc(10 * data[0]) / 10;
}
Object.keys(t2).forEach((tensor6) => tf21.dispose(t2[tensor6]));
last6[idx] = obj;
lastCount6 = count2;
lastTime9 = now();
resolve(obj);
});
}
// src/gear/ssrnet-gender.ts
var tf22 = __toESM(require_tfjs_esm());
var model14;
var last7 = [];
var lastCount7 = 0;
var lastTime10 = 0;
var skipped10 = Number.MAX_SAFE_INTEGER;
var rgb = [0.2989, 0.587, 0.114];
async function load12(config3) {
var _a;
if (env.initial)
model14 = null;
if (!model14)
model14 = await loadModel((_a = config3.face["ssrnet"]) == null ? void 0 : _a.modelPathGender);
else if (config3.debug)
log("cached model:", model14["modelUrl"]);
return model14;
}
async function predict11(image28, config3, idx, count2) {
var _a, _b, _c, _d;
if (!model14)
return { gender: "unknown", genderScore: 0 };
const skipFrame = skipped10 < (((_a = config3.face["ssrnet"]) == null ? void 0 : _a.skipFrames) || 0);
const skipTime = (((_b = config3.face["ssrnet"]) == null ? void 0 : _b.skipTime) || 0) > now() - lastTime10;
if (config3.skipAllowed && skipFrame && skipTime && lastCount7 === count2 && ((_c = last7[idx]) == null ? void 0 : _c.gender) && ((_d = last7[idx]) == null ? void 0 : _d.genderScore) > 0) {
skipped10++;
return last7[idx];
}
skipped10 = 0;
return new Promise(async (resolve) => {
var _a2;
if (!(model14 == null ? void 0 : model14.inputs[0].shape))
return;
const t2 = {};
t2.resize = tf22.image.resizeBilinear(image28, [model14.inputs[0].shape[2], model14.inputs[0].shape[1]], false);
t2.enhance = tf22.tidy(() => {
const [red, green, blue] = tf22.split(t2.resize, 3, 3);
const redNorm = tf22.mul(red, rgb[0]);
const greenNorm = tf22.mul(green, rgb[1]);
const blueNorm = tf22.mul(blue, rgb[2]);
const grayscale = tf22.addN([redNorm, greenNorm, blueNorm]);
const normalize2 = tf22.mul(tf22.sub(grayscale, constants.tf05), 2);
return normalize2;
});
const obj = { gender: "unknown", genderScore: 0 };
if ((_a2 = config3.face["ssrnet"]) == null ? void 0 : _a2.enabled)
t2.gender = model14.execute(t2.enhance);
const data = await t2.gender.data();
obj.gender = data[0] > data[1] ? "female" : "male";
obj.genderScore = data[0] > data[1] ? Math.trunc(100 * data[0]) / 100 : Math.trunc(100 * data[1]) / 100;
Object.keys(t2).forEach((tensor6) => tf22.dispose(t2[tensor6]));
last7[idx] = obj;
lastCount7 = count2;
lastTime10 = now();
resolve(obj);
});
}
// src/face/mobilefacenet.ts
var tf23 = __toESM(require_tfjs_esm());
var model15;
var last8 = [];
var lastCount8 = 0;
var lastTime11 = 0;
var skipped11 = Number.MAX_SAFE_INTEGER;
async function load13(config3) {
var _a;
if (env.initial)
model15 = null;
if (!model15)
model15 = await loadModel((_a = config3.face["mobilefacenet"]) == null ? void 0 : _a.modelPath);
else if (config3.debug)
log("cached model:", model15["modelUrl"]);
return model15;
}
async function predict12(input, config3, idx, count2) {
var _a, _b;
if (!(model15 == null ? void 0 : model15["executor"]))
return [];
const skipFrame = skipped11 < (((_a = config3.face["mobilefacenet"]) == null ? void 0 : _a.skipFrames) || 0);
const skipTime = (((_b = config3.face["mobilefacenet"]) == null ? void 0 : _b.skipTime) || 0) > now() - lastTime11;
if (config3.skipAllowed && skipTime && skipFrame && lastCount8 === count2 && last8[idx]) {
skipped11++;
return last8[idx];
}
return new Promise(async (resolve) => {
var _a2;
let data = [];
if (((_a2 = config3.face["mobilefacenet"]) == null ? void 0 : _a2.enabled) && (model15 == null ? void 0 : model15.inputs[0].shape)) {
const t2 = {};
t2.crop = tf23.image.resizeBilinear(input, [model15.inputs[0].shape[2], model15.inputs[0].shape[1]], false);
t2.data = model15.execute(t2.crop);
const output = await t2.data.data();
data = Array.from(output);
Object.keys(t2).forEach((tensor6) => tf23.dispose(t2[tensor6]));
}
last8[idx] = data;
lastCount8 = count2;
lastTime11 = now();
resolve(data);
});
}
// src/face/insightface.ts
var tf24 = __toESM(require_tfjs_esm());
var model16;
var last9 = [];
var lastCount9 = 0;
var lastTime12 = 0;
var skipped12 = Number.MAX_SAFE_INTEGER;
async function load14(config3) {
if (env.initial)
model16 = null;
if (!model16)
model16 = await loadModel(config3.face["insightface"].modelPath);
else if (config3.debug)
log("cached model:", model16["modelUrl"]);
return model16;
}
async function predict13(input, config3, idx, count2) {
var _a, _b;
if (!(model16 == null ? void 0 : model16["executor"]))
return [];
const skipFrame = skipped12 < (((_a = config3.face["insightface"]) == null ? void 0 : _a.skipFrames) || 0);
const skipTime = (((_b = config3.face["insightface"]) == null ? void 0 : _b.skipTime) || 0) > now() - lastTime12;
if (config3.skipAllowed && skipTime && skipFrame && lastCount9 === count2 && last9[idx]) {
skipped12++;
return last9[idx];
}
return new Promise(async (resolve) => {
var _a2;
let data = [];
if (((_a2 = config3.face["insightface"]) == null ? void 0 : _a2.enabled) && (model16 == null ? void 0 : model16.inputs[0].shape)) {
const t2 = {};
t2.crop = tf24.image.resizeBilinear(input, [model16.inputs[0].shape[2], model16.inputs[0].shape[1]], false);
t2.data = model16.execute(t2.crop);
const output = await t2.data.data();
data = Array.from(output);
Object.keys(t2).forEach((tensor6) => tf24.dispose(t2[tensor6]));
}
last9[idx] = data;
lastCount9 = count2;
lastTime12 = now();
resolve(data);
});
}
// src/face/angles.ts
var calculateGaze = (face4) => {
const radians = (pt1, pt2) => Math.atan2(pt1[1] - pt2[1], pt1[0] - pt2[0]);
if (!face4.annotations.rightEyeIris || !face4.annotations.leftEyeIris)
return { bearing: 0, strength: 0 };
const offsetIris = [0, -0.1];
const eyeRatio = 1;
const left = (face4.mesh[33][2] || 0) > (face4.mesh[263][2] || 0);
const irisCenter = left ? face4.mesh[473] : face4.mesh[468];
const eyeCenter = left ? [(face4.mesh[133][0] + face4.mesh[33][0]) / 2, (face4.mesh[133][1] + face4.mesh[33][1]) / 2] : [(face4.mesh[263][0] + face4.mesh[362][0]) / 2, (face4.mesh[263][1] + face4.mesh[362][1]) / 2];
const eyeSize = left ? [face4.mesh[133][0] - face4.mesh[33][0], face4.mesh[23][1] - face4.mesh[27][1]] : [face4.mesh[263][0] - face4.mesh[362][0], face4.mesh[253][1] - face4.mesh[257][1]];
const eyeDiff = [
(eyeCenter[0] - irisCenter[0]) / eyeSize[0] - offsetIris[0],
eyeRatio * (irisCenter[1] - eyeCenter[1]) / eyeSize[1] - offsetIris[1]
];
let strength = Math.sqrt(eyeDiff[0] * eyeDiff[0] + eyeDiff[1] * eyeDiff[1]);
strength = Math.min(strength, face4.boxRaw[2] / 2, face4.boxRaw[3] / 2);
const bearing = (radians([0, 0], eyeDiff) + Math.PI / 2) % Math.PI;
return { bearing, strength };
};
var calculateFaceAngle = (face4, imageSize) => {
const normalize2 = (v) => {
const length = Math.sqrt(v[0] * v[0] + v[1] * v[1] + v[2] * v[2]);
v[0] /= length;
v[1] /= length;
v[2] /= length;
return v;
};
const subVectors = (a, b) => {
const x = a[0] - b[0];
const y = a[1] - b[1];
const z = a[2] - b[2];
return [x, y, z];
};
const crossVectors = (a, b) => {
const x = a[1] * b[2] - a[2] * b[1];
const y = a[2] * b[0] - a[0] * b[2];
const z = a[0] * b[1] - a[1] * b[0];
return [x, y, z];
};
const rotationMatrixToEulerAngle = (r) => {
const [r00, _r01, _r02, r10, r11, r12, r20, r21, r22] = r;
let thetaX;
let thetaY;
let thetaZ;
if (r10 < 1) {
if (r10 > -1) {
thetaZ = Math.asin(r10);
thetaY = Math.atan2(-r20, r00);
thetaX = Math.atan2(-r12, r11);
} else {
thetaZ = -Math.PI / 2;
thetaY = -Math.atan2(r21, r22);
thetaX = 0;
}
} else {
thetaZ = Math.PI / 2;
thetaY = Math.atan2(r21, r22);
thetaX = 0;
}
if (Number.isNaN(thetaX))
thetaX = 0;
if (Number.isNaN(thetaY))
thetaY = 0;
if (Number.isNaN(thetaZ))
thetaZ = 0;
return { pitch: 2 * -thetaX, yaw: 2 * -thetaY, roll: 2 * -thetaZ };
};
const mesh = face4.meshRaw;
if (!mesh || mesh.length < 300)
return { angle: { pitch: 0, yaw: 0, roll: 0 }, matrix: [1, 0, 0, 0, 1, 0, 0, 0, 1], gaze: { bearing: 0, strength: 0 } };
const size2 = Math.max(face4.boxRaw[2] * imageSize[0], face4.boxRaw[3] * imageSize[1]) / 1.5;
const pts = [mesh[10], mesh[152], mesh[234], mesh[454]].map((pt) => [pt[0] * imageSize[0] / size2, pt[1] * imageSize[1] / size2, pt[2]]);
const yAxis = normalize2(subVectors(pts[1], pts[0]));
let xAxis = normalize2(subVectors(pts[3], pts[2]));
const zAxis = normalize2(crossVectors(xAxis, yAxis));
xAxis = crossVectors(yAxis, zAxis);
const matrix = [
xAxis[0],
xAxis[1],
xAxis[2],
yAxis[0],
yAxis[1],
yAxis[2],
zAxis[0],
zAxis[1],
zAxis[2]
];
const angle = rotationMatrixToEulerAngle(matrix);
const gaze = mesh.length === 478 ? calculateGaze(face4) : { bearing: 0, strength: 0 };
return { angle, matrix, gaze };
};
// src/face/anthropometry.ts
function calculateCameraDistance(face4, width) {
const f = face4 == null ? void 0 : face4.annotations;
if (!f)
return 0;
const irisSize = Math.max(Math.abs(f.leftEyeIris[3][0] - f.leftEyeIris[1][0]), Math.abs(f.rightEyeIris[3][0] - f.rightEyeIris[1][0])) / width;
const cameraDistance = Math.round(1.17 / irisSize) / 100;
return cameraDistance;
}
// src/face/face.ts
var detectFace = async (instance, input) => {
var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l, _m, _n, _o, _p, _q, _r, _s, _t, _u, _v, _w;
let timeStamp = now();
let ageRes;
let gearRes;
let genderRes;
let emotionRes;
let mobilefacenetRes;
let insightfaceRes;
let antispoofRes;
let livenessRes;
let descRes;
const faceRes = [];
instance.state = "run:face";
const faces = await predict4(input, instance.config);
instance.performance.face = env.perfadd ? (instance.performance.face || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
if (!input.shape || input.shape.length !== 4)
return [];
if (!faces)
return [];
for (let i = 0; i < faces.length; i++) {
instance.analyze("Get Face");
if (!faces[i].tensor || faces[i].tensor.isDisposedInternal) {
log("Face object is disposed:", faces[i].tensor);
continue;
}
if ((_a = instance.config.face.detector) == null ? void 0 : _a.mask) {
const masked = await mask(faces[i]);
tf25.dispose(faces[i].tensor);
if (masked)
faces[i].tensor = masked;
}
const rotation = faces[i].mesh && faces[i].mesh.length > 200 ? calculateFaceAngle(faces[i], [input.shape[2], input.shape[1]]) : null;
instance.analyze("Start Emotion:");
if (instance.config.async) {
emotionRes = ((_b = instance.config.face.emotion) == null ? void 0 : _b.enabled) ? predict5(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : [];
} else {
instance.state = "run:emotion";
timeStamp = now();
emotionRes = ((_c = instance.config.face.emotion) == null ? void 0 : _c.enabled) ? await predict5(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : [];
instance.performance.emotion = env.perfadd ? (instance.performance.emotion || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
}
instance.analyze("End Emotion:");
instance.analyze("Start AntiSpoof:");
if (instance.config.async) {
antispoofRes = ((_d = instance.config.face.antispoof) == null ? void 0 : _d.enabled) ? predict7(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : 0;
} else {
instance.state = "run:antispoof";
timeStamp = now();
antispoofRes = ((_e = instance.config.face.antispoof) == null ? void 0 : _e.enabled) ? await predict7(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : 0;
instance.performance.antispoof = env.perfadd ? (instance.performance.antispoof || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
}
instance.analyze("End AntiSpoof:");
instance.analyze("Start Liveness:");
if (instance.config.async) {
livenessRes = ((_f = instance.config.face.liveness) == null ? void 0 : _f.enabled) ? predict8(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : 0;
} else {
instance.state = "run:liveness";
timeStamp = now();
livenessRes = ((_g = instance.config.face.liveness) == null ? void 0 : _g.enabled) ? await predict8(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : 0;
instance.performance.liveness = env.perfadd ? (instance.performance.antispoof || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
}
instance.analyze("End Liveness:");
instance.analyze("Start GEAR:");
if (instance.config.async) {
gearRes = ((_h = instance.config.face.gear) == null ? void 0 : _h.enabled) ? predict9(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
} else {
instance.state = "run:gear";
timeStamp = now();
gearRes = ((_i = instance.config.face.gear) == null ? void 0 : _i.enabled) ? await predict9(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
instance.performance.gear = Math.trunc(now() - timeStamp);
}
instance.analyze("End GEAR:");
instance.analyze("Start SSRNet:");
if (instance.config.async) {
ageRes = ((_j = instance.config.face["ssrnet"]) == null ? void 0 : _j.enabled) ? predict10(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
genderRes = ((_k = instance.config.face["ssrnet"]) == null ? void 0 : _k.enabled) ? predict11(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
} else {
instance.state = "run:ssrnet";
timeStamp = now();
ageRes = ((_l = instance.config.face["ssrnet"]) == null ? void 0 : _l.enabled) ? await predict10(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
genderRes = ((_m = instance.config.face["ssrnet"]) == null ? void 0 : _m.enabled) ? await predict11(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
instance.performance.ssrnet = Math.trunc(now() - timeStamp);
}
instance.analyze("End SSRNet:");
instance.analyze("Start MobileFaceNet:");
if (instance.config.async) {
mobilefacenetRes = ((_n = instance.config.face["mobilefacenet"]) == null ? void 0 : _n.enabled) ? predict12(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
} else {
instance.state = "run:mobilefacenet";
timeStamp = now();
mobilefacenetRes = ((_o = instance.config.face["mobilefacenet"]) == null ? void 0 : _o.enabled) ? await predict12(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
instance.performance.mobilefacenet = Math.trunc(now() - timeStamp);
}
instance.analyze("End MobileFaceNet:");
instance.analyze("Start InsightFace:");
if (instance.config.async) {
insightfaceRes = ((_p = instance.config.face["insightface"]) == null ? void 0 : _p.enabled) ? predict13(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
} else {
instance.state = "run:mobilefacenet";
timeStamp = now();
insightfaceRes = ((_q = instance.config.face["insightface"]) == null ? void 0 : _q.enabled) ? await predict13(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length) : null;
instance.performance.mobilefacenet = Math.trunc(now() - timeStamp);
}
instance.analyze("End InsightFace:");
instance.analyze("Start Description:");
if (instance.config.async) {
descRes = predict6(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length);
} else {
instance.state = "run:description";
timeStamp = now();
descRes = await predict6(faces[i].tensor || tf25.tensor([]), instance.config, i, faces.length);
instance.performance.description = env.perfadd ? (instance.performance.description || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
}
instance.analyze("End Description:");
if (instance.config.async) {
[ageRes, genderRes, emotionRes, mobilefacenetRes, insightfaceRes, descRes, gearRes, antispoofRes, livenessRes] = await Promise.all([ageRes, genderRes, emotionRes, mobilefacenetRes, insightfaceRes, descRes, gearRes, antispoofRes, livenessRes]);
}
instance.analyze("Finish Face:");
if (((_r = instance.config.face["ssrnet"]) == null ? void 0 : _r.enabled) && ageRes && genderRes) {
descRes = {
...descRes,
age: ageRes.age,
gender: genderRes.gender,
genderScore: genderRes.genderScore
};
}
if (((_s = instance.config.face.gear) == null ? void 0 : _s.enabled) && gearRes) {
descRes = {
...descRes,
age: gearRes.age,
gender: gearRes.gender,
genderScore: gearRes.genderScore,
race: gearRes.race
};
}
if (((_t = instance.config.face["mobilefacenet"]) == null ? void 0 : _t.enabled) && mobilefacenetRes) {
descRes.descriptor = mobilefacenetRes;
}
if (((_u = instance.config.face["insightface"]) == null ? void 0 : _u.enabled) && insightfaceRes) {
descRes.descriptor = insightfaceRes;
}
const irisSize = ((_v = instance.config.face.iris) == null ? void 0 : _v.enabled) ? calculateCameraDistance(faces[i], input.shape[2]) : 0;
const tensor6 = ((_w = instance.config.face.detector) == null ? void 0 : _w.return) ? tf25.squeeze(faces[i].tensor) : null;
tf25.dispose(faces[i].tensor);
if (faces[i].tensor)
delete faces[i].tensor;
const res = {
...faces[i],
id: i
};
if (descRes.age)
res.age = descRes.age;
if (descRes.gender)
res.gender = descRes.gender;
if (descRes.genderScore)
res.genderScore = descRes.genderScore;
if (descRes.descriptor)
res.embedding = descRes.descriptor;
if (descRes.race)
res.race = descRes.race;
if (emotionRes)
res.emotion = emotionRes;
if (antispoofRes)
res.real = antispoofRes;
if (livenessRes)
res.live = livenessRes;
if (irisSize > 0)
res.distance = irisSize;
if (rotation)
res.rotation = rotation;
if (tensor6)
res.tensor = tensor6;
faceRes.push(res);
instance.analyze("End Face");
}
instance.analyze("End FaceMesh:");
if (instance.config.async) {
if (instance.performance.face)
delete instance.performance.face;
if (instance.performance.age)
delete instance.performance.age;
if (instance.performance.gender)
delete instance.performance.gender;
if (instance.performance.emotion)
delete instance.performance.emotion;
}
return faceRes;
};
// src/hand/fingerdef.ts
var Finger = {
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: (value) => Finger.nameMapping[value],
getPoints: (value) => Finger.pointsMapping[value]
};
var FingerCurl = {
none: 0,
half: 1,
full: 2,
nameMapping: { 0: "none", 1: "half", 2: "full" },
getName: (value) => FingerCurl.nameMapping[value]
};
var FingerDirection = {
verticalUp: 0,
verticalDown: 1,
horizontalLeft: 2,
horizontalRight: 3,
diagonalUpRight: 4,
diagonalUpLeft: 5,
diagonalDownRight: 6,
diagonalDownLeft: 7,
nameMapping: { 0: "verticalUp", 1: "verticalDown", 2: "horizontalLeft", 3: "horizontalRight", 4: "diagonalUpRight", 5: "diagonalUpLeft", 6: "diagonalDownRight", 7: "diagonalDownLeft" },
getName: (value) => FingerDirection.nameMapping[value]
};
var FingerGesture = class {
constructor(name) {
__publicField(this, "name");
__publicField(this, "curls");
__publicField(this, "directions");
__publicField(this, "weights");
__publicField(this, "weightsRelative");
this.name = name;
this.curls = {};
this.directions = {};
this.weights = [1, 1, 1, 1, 1];
this.weightsRelative = [1, 1, 1, 1, 1];
}
curl(finger, curl, confidence) {
if (typeof this.curls[finger] === "undefined")
this.curls[finger] = [];
this.curls[finger].push([curl, confidence]);
}
direction(finger, position, confidence) {
if (!this.directions[finger])
this.directions[finger] = [];
this.directions[finger].push([position, confidence]);
}
weight(finger, weight) {
this.weights[finger] = weight;
const total = this.weights.reduce((a, b) => a + b, 0);
this.weightsRelative = this.weights.map((el) => el * 5 / total);
}
matchAgainst(detectedCurls, detectedDirections) {
let confidence = 0;
for (const fingerIdx in detectedCurls) {
const detectedCurl = detectedCurls[fingerIdx];
const expectedCurls = this.curls[fingerIdx];
if (typeof expectedCurls === "undefined") {
confidence += this.weightsRelative[fingerIdx];
continue;
}
for (const [expectedCurl, score] of expectedCurls) {
if (detectedCurl === expectedCurl) {
confidence += score * this.weightsRelative[fingerIdx];
break;
}
}
}
for (const fingerIdx in detectedDirections) {
const detectedDirection = detectedDirections[fingerIdx];
const expectedDirections = this.directions[fingerIdx];
if (typeof expectedDirections === "undefined") {
confidence += this.weightsRelative[fingerIdx];
continue;
}
for (const [expectedDirection, score] of expectedDirections) {
if (detectedDirection === expectedDirection) {
confidence += score * this.weightsRelative[fingerIdx];
break;
}
}
}
return confidence / 10;
}
};
// src/hand/fingergesture.ts
var { thumb, index, middle, ring, pinky } = Finger;
var { none, half, full } = FingerCurl;
var { verticalUp, verticalDown, horizontalLeft, horizontalRight, diagonalUpRight, diagonalUpLeft, diagonalDownRight, diagonalDownLeft } = FingerDirection;
var ThumbsUp = new FingerGesture("thumbs up");
ThumbsUp.curl(thumb, none, 1);
ThumbsUp.direction(thumb, verticalUp, 1);
ThumbsUp.direction(thumb, diagonalUpLeft, 0.25);
ThumbsUp.direction(thumb, diagonalUpRight, 0.25);
for (const finger of [Finger.index, Finger.middle, Finger.ring, Finger.pinky]) {
ThumbsUp.curl(finger, full, 1);
ThumbsUp.direction(finger, horizontalLeft, 1);
ThumbsUp.direction(finger, horizontalRight, 1);
}
var Victory = new FingerGesture("victory");
Victory.curl(thumb, half, 0.5);
Victory.curl(thumb, none, 0.5);
Victory.direction(thumb, verticalUp, 1);
Victory.direction(thumb, diagonalUpLeft, 1);
Victory.curl(index, none, 1);
Victory.direction(index, verticalUp, 0.75);
Victory.direction(index, diagonalUpLeft, 1);
Victory.curl(middle, none, 1);
Victory.direction(middle, verticalUp, 1);
Victory.direction(middle, diagonalUpLeft, 0.75);
Victory.curl(ring, full, 1);
Victory.direction(ring, verticalUp, 0.2);
Victory.direction(ring, diagonalUpLeft, 1);
Victory.direction(ring, horizontalLeft, 0.2);
Victory.curl(pinky, full, 1);
Victory.direction(pinky, verticalUp, 0.2);
Victory.direction(pinky, diagonalUpLeft, 1);
Victory.direction(pinky, horizontalLeft, 0.2);
Victory.weight(index, 2);
Victory.weight(middle, 2);
var Point = new FingerGesture("point");
Point.curl(thumb, full, 1);
Point.curl(index, none, 0.5);
Point.curl(middle, full, 0.5);
Point.curl(ring, full, 0.5);
Point.curl(pinky, full, 0.5);
Point.weight(index, 2);
Point.weight(middle, 2);
var MiddleFinger = new FingerGesture("middle finger");
MiddleFinger.curl(thumb, none, 1);
MiddleFinger.curl(index, full, 0.5);
MiddleFinger.curl(middle, full, 0.5);
MiddleFinger.curl(ring, full, 0.5);
MiddleFinger.curl(pinky, full, 0.5);
MiddleFinger.weight(index, 2);
MiddleFinger.weight(middle, 2);
var OpenPalm = new FingerGesture("open palm");
OpenPalm.curl(thumb, none, 0.75);
OpenPalm.curl(index, none, 0.75);
OpenPalm.curl(middle, none, 0.75);
OpenPalm.curl(ring, none, 0.75);
OpenPalm.curl(pinky, none, 0.75);
var fingergesture_default = [ThumbsUp, Victory, Point, MiddleFinger, OpenPalm];
// src/hand/fingerpose.ts
var minConfidence = 0.7;
var options3 = {
HALF_CURL_START_LIMIT: 60,
NO_CURL_START_LIMIT: 130,
DISTANCE_VOTE_POWER: 1.1,
SINGLE_ANGLE_VOTE_POWER: 0.9,
TOTAL_ANGLE_VOTE_POWER: 1.6
};
function calculateSlope(point1x, point1y, point2x, point2y) {
const value = (point1y - point2y) / (point1x - point2x);
let slope = Math.atan(value) * 180 / Math.PI;
if (slope <= 0)
slope = -slope;
else if (slope > 0)
slope = 180 - slope;
return slope;
}
function getSlopes(point1, point2) {
if (!point1 || !point2)
return [0, 0];
const slopeXY = calculateSlope(point1[0], point1[1], point2[0], point2[1]);
if (point1.length === 2)
return slopeXY;
const slopeYZ = calculateSlope(point1[1], point1[2], point2[1], point2[2]);
return [slopeXY, slopeYZ];
}
function angleOrientationAt(angle, weightageAt = 1) {
let isVertical = 0;
let isDiagonal = 0;
let isHorizontal = 0;
if (angle >= 75 && angle <= 105)
isVertical = 1 * weightageAt;
else if (angle >= 25 && angle <= 155)
isDiagonal = 1 * weightageAt;
else
isHorizontal = 1 * weightageAt;
return [isVertical, isDiagonal, isHorizontal];
}
function estimateFingerCurl(startPoint, midPoint, endPoint) {
const start_mid_x_dist = startPoint[0] - midPoint[0];
const start_end_x_dist = startPoint[0] - endPoint[0];
const mid_end_x_dist = midPoint[0] - endPoint[0];
const start_mid_y_dist = startPoint[1] - midPoint[1];
const start_end_y_dist = startPoint[1] - endPoint[1];
const mid_end_y_dist = midPoint[1] - endPoint[1];
const start_mid_z_dist = startPoint[2] - midPoint[2];
const start_end_z_dist = startPoint[2] - endPoint[2];
const mid_end_z_dist = midPoint[2] - endPoint[2];
const start_mid_dist = Math.sqrt(start_mid_x_dist * start_mid_x_dist + start_mid_y_dist * start_mid_y_dist + start_mid_z_dist * start_mid_z_dist);
const start_end_dist = Math.sqrt(start_end_x_dist * start_end_x_dist + start_end_y_dist * start_end_y_dist + start_end_z_dist * start_end_z_dist);
const mid_end_dist = Math.sqrt(mid_end_x_dist * mid_end_x_dist + mid_end_y_dist * mid_end_y_dist + mid_end_z_dist * mid_end_z_dist);
let cos_in = (mid_end_dist * mid_end_dist + start_mid_dist * start_mid_dist - start_end_dist * start_end_dist) / (2 * mid_end_dist * start_mid_dist);
if (cos_in > 1)
cos_in = 1;
else if (cos_in < -1)
cos_in = -1;
let angleOfCurve = Math.acos(cos_in);
angleOfCurve = 57.2958 * angleOfCurve % 180;
let fingerCurl;
if (angleOfCurve > options3.NO_CURL_START_LIMIT)
fingerCurl = FingerCurl.none;
else if (angleOfCurve > options3.HALF_CURL_START_LIMIT)
fingerCurl = FingerCurl.half;
else
fingerCurl = FingerCurl.full;
return fingerCurl;
}
function estimateHorizontalDirection(start_end_x_dist, start_mid_x_dist, mid_end_x_dist, max_dist_x) {
let estimatedDirection;
if (max_dist_x === Math.abs(start_end_x_dist)) {
if (start_end_x_dist > 0)
estimatedDirection = FingerDirection.horizontalLeft;
else
estimatedDirection = FingerDirection.horizontalRight;
} else if (max_dist_x === Math.abs(start_mid_x_dist)) {
if (start_mid_x_dist > 0)
estimatedDirection = FingerDirection.horizontalLeft;
else
estimatedDirection = FingerDirection.horizontalRight;
} else {
if (mid_end_x_dist > 0)
estimatedDirection = FingerDirection.horizontalLeft;
else
estimatedDirection = FingerDirection.horizontalRight;
}
return estimatedDirection;
}
function estimateVerticalDirection(start_end_y_dist, start_mid_y_dist, mid_end_y_dist, max_dist_y) {
let estimatedDirection;
if (max_dist_y === Math.abs(start_end_y_dist)) {
if (start_end_y_dist < 0)
estimatedDirection = FingerDirection.verticalDown;
else
estimatedDirection = FingerDirection.verticalUp;
} else if (max_dist_y === Math.abs(start_mid_y_dist)) {
if (start_mid_y_dist < 0)
estimatedDirection = FingerDirection.verticalDown;
else
estimatedDirection = FingerDirection.verticalUp;
} else {
if (mid_end_y_dist < 0)
estimatedDirection = FingerDirection.verticalDown;
else
estimatedDirection = FingerDirection.verticalUp;
}
return estimatedDirection;
}
function estimateDiagonalDirection(start_end_y_dist, start_mid_y_dist, mid_end_y_dist, max_dist_y, start_end_x_dist, start_mid_x_dist, mid_end_x_dist, max_dist_x) {
let estimatedDirection;
const reqd_vertical_direction = estimateVerticalDirection(start_end_y_dist, start_mid_y_dist, mid_end_y_dist, max_dist_y);
const reqd_horizontal_direction = estimateHorizontalDirection(start_end_x_dist, start_mid_x_dist, mid_end_x_dist, max_dist_x);
if (reqd_vertical_direction === FingerDirection.verticalUp) {
if (reqd_horizontal_direction === FingerDirection.horizontalLeft)
estimatedDirection = FingerDirection.diagonalUpLeft;
else
estimatedDirection = FingerDirection.diagonalUpRight;
} else {
if (reqd_horizontal_direction === FingerDirection.horizontalLeft)
estimatedDirection = FingerDirection.diagonalDownLeft;
else
estimatedDirection = FingerDirection.diagonalDownRight;
}
return estimatedDirection;
}
function calculateFingerDirection(startPoint, midPoint, endPoint, fingerSlopes) {
const start_mid_x_dist = startPoint[0] - midPoint[0];
const start_end_x_dist = startPoint[0] - endPoint[0];
const mid_end_x_dist = midPoint[0] - endPoint[0];
const start_mid_y_dist = startPoint[1] - midPoint[1];
const start_end_y_dist = startPoint[1] - endPoint[1];
const mid_end_y_dist = midPoint[1] - endPoint[1];
const max_dist_x = Math.max(Math.abs(start_mid_x_dist), Math.abs(start_end_x_dist), Math.abs(mid_end_x_dist));
const max_dist_y = Math.max(Math.abs(start_mid_y_dist), Math.abs(start_end_y_dist), Math.abs(mid_end_y_dist));
let voteVertical = 0;
let voteDiagonal = 0;
let voteHorizontal = 0;
const start_end_x_y_dist_ratio = max_dist_y / (max_dist_x + 1e-5);
if (start_end_x_y_dist_ratio > 1.5)
voteVertical += options3.DISTANCE_VOTE_POWER;
else if (start_end_x_y_dist_ratio > 0.66)
voteDiagonal += options3.DISTANCE_VOTE_POWER;
else
voteHorizontal += options3.DISTANCE_VOTE_POWER;
const start_mid_dist = Math.sqrt(start_mid_x_dist * start_mid_x_dist + start_mid_y_dist * start_mid_y_dist);
const start_end_dist = Math.sqrt(start_end_x_dist * start_end_x_dist + start_end_y_dist * start_end_y_dist);
const mid_end_dist = Math.sqrt(mid_end_x_dist * mid_end_x_dist + mid_end_y_dist * mid_end_y_dist);
const max_dist = Math.max(start_mid_dist, start_end_dist, mid_end_dist);
let calc_start_point_x = startPoint[0];
let calc_start_point_y = startPoint[1];
let calc_end_point_x = endPoint[0];
let calc_end_point_y = endPoint[1];
if (max_dist === start_mid_dist) {
calc_end_point_x = endPoint[0];
calc_end_point_y = endPoint[1];
} else if (max_dist === mid_end_dist) {
calc_start_point_x = midPoint[0];
calc_start_point_y = midPoint[1];
}
const calcStartPoint = [calc_start_point_x, calc_start_point_y];
const calcEndPoint = [calc_end_point_x, calc_end_point_y];
const totalAngle = getSlopes(calcStartPoint, calcEndPoint);
const votes = angleOrientationAt(totalAngle, options3.TOTAL_ANGLE_VOTE_POWER);
voteVertical += votes[0];
voteDiagonal += votes[1];
voteHorizontal += votes[2];
for (const fingerSlope of fingerSlopes) {
const fingerVotes = angleOrientationAt(fingerSlope, options3.SINGLE_ANGLE_VOTE_POWER);
voteVertical += fingerVotes[0];
voteDiagonal += fingerVotes[1];
voteHorizontal += fingerVotes[2];
}
let estimatedDirection;
if (voteVertical === Math.max(voteVertical, voteDiagonal, voteHorizontal)) {
estimatedDirection = estimateVerticalDirection(start_end_y_dist, start_mid_y_dist, mid_end_y_dist, max_dist_y);
} else if (voteHorizontal === Math.max(voteDiagonal, voteHorizontal)) {
estimatedDirection = estimateHorizontalDirection(start_end_x_dist, start_mid_x_dist, mid_end_x_dist, max_dist_x);
} else {
estimatedDirection = estimateDiagonalDirection(start_end_y_dist, start_mid_y_dist, mid_end_y_dist, max_dist_y, start_end_x_dist, start_mid_x_dist, mid_end_x_dist, max_dist_x);
}
return estimatedDirection;
}
function estimate(landmarks) {
const slopesXY = [];
const slopesYZ = [];
const fingerCurls = [];
const fingerDirections = [];
if (!landmarks)
return { curls: fingerCurls, directions: fingerDirections };
for (const finger of Finger.all) {
const points = Finger.getPoints(finger);
const slopeAtXY = [];
const slopeAtYZ = [];
for (const point2 of points) {
const point1 = landmarks[point2[0]];
const point22 = landmarks[point2[1]];
const slopes = getSlopes(point1, point22);
const slopeXY = slopes[0];
const slopeYZ = slopes[1];
slopeAtXY.push(slopeXY);
slopeAtYZ.push(slopeYZ);
}
slopesXY.push(slopeAtXY);
slopesYZ.push(slopeAtYZ);
}
for (const finger of Finger.all) {
const pointIndexAt = finger === Finger.thumb ? 1 : 0;
const fingerPointsAt = Finger.getPoints(finger);
const startPoint = landmarks[fingerPointsAt[pointIndexAt][0]];
const midPoint = landmarks[fingerPointsAt[pointIndexAt + 1][1]];
const endPoint = landmarks[fingerPointsAt[3][1]];
const fingerCurled = estimateFingerCurl(startPoint, midPoint, endPoint);
const fingerPosition = calculateFingerDirection(startPoint, midPoint, endPoint, slopesXY[finger].slice(pointIndexAt));
fingerCurls[finger] = fingerCurled;
fingerDirections[finger] = fingerPosition;
}
return { curls: fingerCurls, directions: fingerDirections };
}
function analyze(keypoints) {
if (!keypoints || keypoints.length === 0)
return null;
const estimatorRes = estimate(keypoints);
const landmarks = {};
for (const fingerIdx of Finger.all) {
landmarks[Finger.getName(fingerIdx)] = {
curl: FingerCurl.getName(estimatorRes.curls[fingerIdx]),
direction: FingerDirection.getName(estimatorRes.directions[fingerIdx])
};
}
return landmarks;
}
function match(keypoints) {
const poses = [];
if (!keypoints || keypoints.length === 0)
return poses;
const estimatorRes = estimate(keypoints);
for (const gesture2 of fingergesture_default) {
const confidence = gesture2.matchAgainst(estimatorRes.curls, estimatorRes.directions);
if (confidence >= minConfidence)
poses.push({ name: gesture2.name, confidence });
}
return poses;
}
// src/gesture/gesture.ts
var body2 = (res) => {
if (!res)
return [];
const gestures = [];
for (let i = 0; i < res.length; i++) {
const leftWrist = res[i].keypoints.find((a) => a.part === "leftWrist");
const rightWrist = res[i].keypoints.find((a) => a.part === "rightWrist");
const nose = res[i].keypoints.find((a) => a.part === "nose");
if (nose && leftWrist && rightWrist && leftWrist.position[1] < nose.position[1] && rightWrist.position[1] < nose.position[1])
gestures.push({ body: i, gesture: "i give up" });
else if (nose && leftWrist && leftWrist.position[1] < nose.position[1])
gestures.push({ body: i, gesture: "raise left hand" });
else if (nose && rightWrist && rightWrist.position[1] < nose.position[1])
gestures.push({ body: i, gesture: "raise right hand" });
const leftShoulder = res[i].keypoints.find((a) => a.part === "leftShoulder");
const rightShoulder = res[i].keypoints.find((a) => a.part === "rightShoulder");
if (leftShoulder && rightShoulder && Math.abs(leftShoulder.positionRaw[1] - rightShoulder.positionRaw[1]) > 0.1) {
gestures.push({ body: i, gesture: `leaning ${leftShoulder.position[1] > rightShoulder.position[1] ? "left" : "right"}` });
}
}
return gestures;
};
var face2 = (res) => {
if (!res)
return [];
const gestures = [];
for (let i = 0; i < res.length; i++) {
if (res[i].mesh && res[i].mesh.length > 450) {
const zDiff = (res[i].mesh[33][2] || 0) - (res[i].mesh[263][2] || 0);
const xDiff = res[i].mesh[33][0] - res[i].mesh[263][0];
if (Math.abs(zDiff / xDiff) <= 0.15)
gestures.push({ face: i, gesture: "facing center" });
else
gestures.push({ face: i, gesture: `facing ${zDiff < 0 ? "left" : "right"}` });
const openLeft = Math.abs(res[i].mesh[374][1] - res[i].mesh[386][1]) / Math.abs(res[i].mesh[443][1] - res[i].mesh[450][1]);
if (openLeft < 0.2)
gestures.push({ face: i, gesture: "blink left eye" });
const openRight = Math.abs(res[i].mesh[145][1] - res[i].mesh[159][1]) / Math.abs(res[i].mesh[223][1] - res[i].mesh[230][1]);
if (openRight < 0.2)
gestures.push({ face: i, gesture: "blink right eye" });
const mouthOpen = Math.min(100, 500 * Math.abs(res[i].mesh[13][1] - res[i].mesh[14][1]) / Math.abs(res[i].mesh[10][1] - res[i].mesh[152][1]));
if (mouthOpen > 10)
gestures.push({ face: i, gesture: `mouth ${Math.trunc(mouthOpen)}% open` });
const chinDepth = res[i].mesh[152][2] || 0;
if (Math.abs(chinDepth) > 10)
gestures.push({ face: i, gesture: `head ${chinDepth < 0 ? "up" : "down"}` });
}
}
return gestures;
};
var iris2 = (res) => {
var _a, _b, _c, _d;
if (!res)
return [];
const gestures = [];
for (let i = 0; i < res.length; i++) {
if (!((_b = (_a = res[i].annotations) == null ? void 0 : _a.leftEyeIris) == null ? void 0 : _b[0]) || !((_d = (_c = res[i].annotations) == null ? void 0 : _c.rightEyeIris) == null ? void 0 : _d[0]))
continue;
const sizeXLeft = res[i].annotations.leftEyeIris[3][0] - res[i].annotations.leftEyeIris[1][0];
const sizeYLeft = res[i].annotations.leftEyeIris[4][1] - res[i].annotations.leftEyeIris[2][1];
const areaLeft = Math.abs(sizeXLeft * sizeYLeft);
const sizeXRight = res[i].annotations.rightEyeIris[3][0] - res[i].annotations.rightEyeIris[1][0];
const sizeYRight = res[i].annotations.rightEyeIris[4][1] - res[i].annotations.rightEyeIris[2][1];
const areaRight = Math.abs(sizeXRight * sizeYRight);
let center = false;
const difference = Math.abs(areaLeft - areaRight) / Math.max(areaLeft, areaRight);
if (difference < 0.25) {
center = true;
gestures.push({ iris: i, gesture: "facing center" });
}
const leftIrisCenterX = Math.abs(res[i].mesh[263][0] - res[i].annotations.leftEyeIris[0][0]) / res[i].box[2];
const rightIrisCenterX = Math.abs(res[i].mesh[33][0] - res[i].annotations.rightEyeIris[0][0]) / res[i].box[2];
if (leftIrisCenterX > 0.06 || rightIrisCenterX > 0.06)
center = false;
if (leftIrisCenterX > rightIrisCenterX) {
if (leftIrisCenterX > 0.05)
gestures.push({ iris: i, gesture: "looking right" });
} else {
if (rightIrisCenterX > 0.05)
gestures.push({ iris: i, gesture: "looking left" });
}
const rightIrisCenterY = Math.abs(res[i].mesh[145][1] - res[i].annotations.rightEyeIris[0][1]) / res[i].box[3];
const leftIrisCenterY = Math.abs(res[i].mesh[374][1] - res[i].annotations.leftEyeIris[0][1]) / res[i].box[3];
if (leftIrisCenterY < 0.01 || rightIrisCenterY < 0.01 || leftIrisCenterY > 0.022 || rightIrisCenterY > 0.022)
center = false;
if (leftIrisCenterY < 0.01 || rightIrisCenterY < 0.01)
gestures.push({ iris: i, gesture: "looking down" });
if (leftIrisCenterY > 0.022 || rightIrisCenterY > 0.022)
gestures.push({ iris: i, gesture: "looking up" });
if (center)
gestures.push({ iris: i, gesture: "looking center" });
}
return gestures;
};
var hand2 = (res) => {
if (!res)
return [];
const gestures = [];
for (let i = 0; i < res.length; i++) {
const fingers = [];
if (res[i].annotations) {
for (const [finger, pos] of Object.entries(res[i].annotations)) {
if (finger !== "palmBase" && Array.isArray(pos) && pos[0])
fingers.push({ name: finger.toLowerCase(), position: pos[0] });
}
}
if (fingers && fingers.length > 0) {
const closest = fingers.reduce((best, a) => (best.position[2] || 0) < (a.position[2] || 0) ? best : a);
gestures.push({ hand: i, gesture: `${closest.name} forward` });
const highest = fingers.reduce((best, a) => best.position[1] < a.position[1] ? best : a);
gestures.push({ hand: i, gesture: `${highest.name} up` });
}
if (res[i].keypoints) {
const poses = match(res[i].keypoints);
for (const pose of poses)
gestures.push({ hand: i, gesture: pose.name });
}
}
return gestures;
};
// src/hand/handposedetector.ts
var tf27 = __toESM(require_tfjs_esm());
// src/hand/handposeutil.ts
var tf26 = __toESM(require_tfjs_esm());
function getBoxSize2(box) {
return [
Math.abs(box.endPoint[0] - box.startPoint[0]),
Math.abs(box.endPoint[1] - box.startPoint[1])
];
}
function getBoxCenter2(box) {
return [
box.startPoint[0] + (box.endPoint[0] - box.startPoint[0]) / 2,
box.startPoint[1] + (box.endPoint[1] - box.startPoint[1]) / 2
];
}
function cutBoxFromImageAndResize(box, image28, cropSize) {
const h = image28.shape[1];
const w = image28.shape[2];
const boxes = [[
box.startPoint[1] / h,
box.startPoint[0] / w,
box.endPoint[1] / h,
box.endPoint[0] / w
]];
return tf26.image.cropAndResize(image28, boxes, [0], cropSize);
}
function scaleBoxCoordinates2(box, factor) {
const startPoint = [box.startPoint[0] * factor[0], box.startPoint[1] * factor[1]];
const endPoint = [box.endPoint[0] * factor[0], box.endPoint[1] * factor[1]];
const palmLandmarks = box.palmLandmarks.map((coord) => {
const scaledCoord = [coord[0] * factor[0], coord[1] * factor[1]];
return scaledCoord;
});
return { startPoint, endPoint, palmLandmarks, confidence: box.confidence };
}
function enlargeBox2(box, factor = 1.5) {
const center = getBoxCenter2(box);
const size2 = getBoxSize2(box);
const newHalfSize = [factor * size2[0] / 2, factor * size2[1] / 2];
const startPoint = [center[0] - newHalfSize[0], center[1] - newHalfSize[1]];
const endPoint = [center[0] + newHalfSize[0], center[1] + newHalfSize[1]];
return { startPoint, endPoint, palmLandmarks: box.palmLandmarks };
}
function squarifyBox2(box) {
const centers = getBoxCenter2(box);
const size2 = getBoxSize2(box);
const maxEdge = Math.max(...size2);
const halfSize = maxEdge / 2;
const startPoint = [centers[0] - halfSize, centers[1] - halfSize];
const endPoint = [centers[0] + halfSize, centers[1] + halfSize];
return { startPoint, endPoint, palmLandmarks: box.palmLandmarks };
}
function normalizeRadians2(angle) {
return angle - 2 * Math.PI * Math.floor((angle + Math.PI) / (2 * Math.PI));
}
function computeRotation2(point1, point2) {
const radians = Math.PI / 2 - Math.atan2(-(point2[1] - point1[1]), point2[0] - point1[0]);
return normalizeRadians2(radians);
}
var buildTranslationMatrix2 = (x, y) => [[1, 0, x], [0, 1, y], [0, 0, 1]];
function dot2(v1, v2) {
let product = 0;
for (let i = 0; i < v1.length; i++) {
product += v1[i] * v2[i];
}
return product;
}
function getColumnFrom2DArr2(arr, columnIndex) {
const column = [];
for (let i = 0; i < arr.length; i++) {
column.push(arr[i][columnIndex]);
}
return column;
}
function multiplyTransformMatrices2(mat1, mat2) {
const product = [];
const size2 = mat1.length;
for (let row = 0; row < size2; row++) {
product.push([]);
for (let col = 0; col < size2; col++) {
product[row].push(dot2(mat1[row], getColumnFrom2DArr2(mat2, col)));
}
}
return product;
}
function buildRotationMatrix2(rotation, center) {
const cosA = Math.cos(rotation);
const sinA = Math.sin(rotation);
const rotationMatrix = [[cosA, -sinA, 0], [sinA, cosA, 0], [0, 0, 1]];
const translationMatrix = buildTranslationMatrix2(center[0], center[1]);
const translationTimesRotation = multiplyTransformMatrices2(translationMatrix, rotationMatrix);
const negativeTranslationMatrix = buildTranslationMatrix2(-center[0], -center[1]);
return multiplyTransformMatrices2(translationTimesRotation, negativeTranslationMatrix);
}
function invertTransformMatrix2(matrix) {
const rotationComponent = [[matrix[0][0], matrix[1][0]], [matrix[0][1], matrix[1][1]]];
const translationComponent = [matrix[0][2], matrix[1][2]];
const invertedTranslation = [
-dot2(rotationComponent[0], translationComponent),
-dot2(rotationComponent[1], translationComponent)
];
return [
rotationComponent[0].concat(invertedTranslation[0]),
rotationComponent[1].concat(invertedTranslation[1]),
[0, 0, 1]
];
}
function rotatePoint2(homogeneousCoordinate, rotationMatrix) {
return [
dot2(homogeneousCoordinate, rotationMatrix[0]),
dot2(homogeneousCoordinate, rotationMatrix[1])
];
}
// src/hand/handposeanchors.ts
var anchors2 = [
{ x: 0.015625, y: 0.015625 },
{ x: 0.015625, y: 0.015625 },
{ x: 0.046875, y: 0.015625 },
{ x: 0.046875, y: 0.015625 },
{ x: 0.078125, y: 0.015625 },
{ x: 0.078125, y: 0.015625 },
{ x: 0.109375, y: 0.015625 },
{ x: 0.109375, y: 0.015625 },
{ x: 0.140625, y: 0.015625 },
{ x: 0.140625, y: 0.015625 },
{ x: 0.171875, y: 0.015625 },
{ x: 0.171875, y: 0.015625 },
{ x: 0.203125, y: 0.015625 },
{ x: 0.203125, y: 0.015625 },
{ x: 0.234375, y: 0.015625 },
{ x: 0.234375, y: 0.015625 },
{ x: 0.265625, y: 0.015625 },
{ x: 0.265625, y: 0.015625 },
{ x: 0.296875, y: 0.015625 },
{ x: 0.296875, y: 0.015625 },
{ x: 0.328125, y: 0.015625 },
{ x: 0.328125, y: 0.015625 },
{ x: 0.359375, y: 0.015625 },
{ x: 0.359375, y: 0.015625 },
{ x: 0.390625, y: 0.015625 },
{ x: 0.390625, y: 0.015625 },
{ x: 0.421875, y: 0.015625 },
{ x: 0.421875, y: 0.015625 },
{ x: 0.453125, y: 0.015625 },
{ x: 0.453125, y: 0.015625 },
{ x: 0.484375, y: 0.015625 },
{ x: 0.484375, y: 0.015625 },
{ x: 0.515625, y: 0.015625 },
{ x: 0.515625, y: 0.015625 },
{ x: 0.546875, y: 0.015625 },
{ x: 0.546875, y: 0.015625 },
{ x: 0.578125, y: 0.015625 },
{ x: 0.578125, y: 0.015625 },
{ x: 0.609375, y: 0.015625 },
{ x: 0.609375, y: 0.015625 },
{ x: 0.640625, y: 0.015625 },
{ x: 0.640625, y: 0.015625 },
{ x: 0.671875, y: 0.015625 },
{ x: 0.671875, y: 0.015625 },
{ x: 0.703125, y: 0.015625 },
{ x: 0.703125, y: 0.015625 },
{ x: 0.734375, y: 0.015625 },
{ x: 0.734375, y: 0.015625 },
{ x: 0.765625, y: 0.015625 },
{ x: 0.765625, y: 0.015625 },
{ x: 0.796875, y: 0.015625 },
{ x: 0.796875, y: 0.015625 },
{ x: 0.828125, y: 0.015625 },
{ x: 0.828125, y: 0.015625 },
{ x: 0.859375, y: 0.015625 },
{ x: 0.859375, y: 0.015625 },
{ x: 0.890625, y: 0.015625 },
{ x: 0.890625, y: 0.015625 },
{ x: 0.921875, y: 0.015625 },
{ x: 0.921875, y: 0.015625 },
{ x: 0.953125, y: 0.015625 },
{ x: 0.953125, y: 0.015625 },
{ x: 0.984375, y: 0.015625 },
{ x: 0.984375, y: 0.015625 },
{ x: 0.015625, y: 0.046875 },
{ x: 0.015625, y: 0.046875 },
{ x: 0.046875, y: 0.046875 },
{ x: 0.046875, y: 0.046875 },
{ x: 0.078125, y: 0.046875 },
{ x: 0.078125, y: 0.046875 },
{ x: 0.109375, y: 0.046875 },
{ x: 0.109375, y: 0.046875 },
{ x: 0.140625, y: 0.046875 },
{ x: 0.140625, y: 0.046875 },
{ x: 0.171875, y: 0.046875 },
{ x: 0.171875, y: 0.046875 },
{ x: 0.203125, y: 0.046875 },
{ x: 0.203125, y: 0.046875 },
{ x: 0.234375, y: 0.046875 },
{ x: 0.234375, y: 0.046875 },
{ x: 0.265625, y: 0.046875 },
{ x: 0.265625, y: 0.046875 },
{ x: 0.296875, y: 0.046875 },
{ x: 0.296875, y: 0.046875 },
{ x: 0.328125, y: 0.046875 },
{ x: 0.328125, y: 0.046875 },
{ x: 0.359375, y: 0.046875 },
{ x: 0.359375, y: 0.046875 },
{ x: 0.390625, y: 0.046875 },
{ x: 0.390625, y: 0.046875 },
{ x: 0.421875, y: 0.046875 },
{ x: 0.421875, y: 0.046875 },
{ x: 0.453125, y: 0.046875 },
{ x: 0.453125, y: 0.046875 },
{ x: 0.484375, y: 0.046875 },
{ x: 0.484375, y: 0.046875 },
{ x: 0.515625, y: 0.046875 },
{ x: 0.515625, y: 0.046875 },
{ x: 0.546875, y: 0.046875 },
{ x: 0.546875, y: 0.046875 },
{ x: 0.578125, y: 0.046875 },
{ x: 0.578125, y: 0.046875 },
{ x: 0.609375, y: 0.046875 },
{ x: 0.609375, y: 0.046875 },
{ x: 0.640625, y: 0.046875 },
{ x: 0.640625, y: 0.046875 },
{ x: 0.671875, y: 0.046875 },
{ x: 0.671875, y: 0.046875 },
{ x: 0.703125, y: 0.046875 },
{ x: 0.703125, y: 0.046875 },
{ x: 0.734375, y: 0.046875 },
{ x: 0.734375, y: 0.046875 },
{ x: 0.765625, y: 0.046875 },
{ x: 0.765625, y: 0.046875 },
{ x: 0.796875, y: 0.046875 },
{ x: 0.796875, y: 0.046875 },
{ x: 0.828125, y: 0.046875 },
{ x: 0.828125, y: 0.046875 },
{ x: 0.859375, y: 0.046875 },
{ x: 0.859375, y: 0.046875 },
{ x: 0.890625, y: 0.046875 },
{ x: 0.890625, y: 0.046875 },
{ x: 0.921875, y: 0.046875 },
{ x: 0.921875, y: 0.046875 },
{ x: 0.953125, y: 0.046875 },
{ x: 0.953125, y: 0.046875 },
{ x: 0.984375, y: 0.046875 },
{ x: 0.984375, y: 0.046875 },
{ x: 0.015625, y: 0.078125 },
{ x: 0.015625, y: 0.078125 },
{ x: 0.046875, y: 0.078125 },
{ x: 0.046875, y: 0.078125 },
{ x: 0.078125, y: 0.078125 },
{ x: 0.078125, y: 0.078125 },
{ x: 0.109375, y: 0.078125 },
{ x: 0.109375, y: 0.078125 },
{ x: 0.140625, y: 0.078125 },
{ x: 0.140625, y: 0.078125 },
{ x: 0.171875, y: 0.078125 },
{ x: 0.171875, y: 0.078125 },
{ x: 0.203125, y: 0.078125 },
{ x: 0.203125, y: 0.078125 },
{ x: 0.234375, y: 0.078125 },
{ x: 0.234375, y: 0.078125 },
{ x: 0.265625, y: 0.078125 },
{ x: 0.265625, y: 0.078125 },
{ x: 0.296875, y: 0.078125 },
{ x: 0.296875, y: 0.078125 },
{ x: 0.328125, y: 0.078125 },
{ x: 0.328125, y: 0.078125 },
{ x: 0.359375, y: 0.078125 },
{ x: 0.359375, y: 0.078125 },
{ x: 0.390625, y: 0.078125 },
{ x: 0.390625, y: 0.078125 },
{ x: 0.421875, y: 0.078125 },
{ x: 0.421875, y: 0.078125 },
{ x: 0.453125, y: 0.078125 },
{ x: 0.453125, y: 0.078125 },
{ x: 0.484375, y: 0.078125 },
{ x: 0.484375, y: 0.078125 },
{ x: 0.515625, y: 0.078125 },
{ x: 0.515625, y: 0.078125 },
{ x: 0.546875, y: 0.078125 },
{ x: 0.546875, y: 0.078125 },
{ x: 0.578125, y: 0.078125 },
{ x: 0.578125, y: 0.078125 },
{ x: 0.609375, y: 0.078125 },
{ x: 0.609375, y: 0.078125 },
{ x: 0.640625, y: 0.078125 },
{ x: 0.640625, y: 0.078125 },
{ x: 0.671875, y: 0.078125 },
{ x: 0.671875, y: 0.078125 },
{ x: 0.703125, y: 0.078125 },
{ x: 0.703125, y: 0.078125 },
{ x: 0.734375, y: 0.078125 },
{ x: 0.734375, y: 0.078125 },
{ x: 0.765625, y: 0.078125 },
{ x: 0.765625, y: 0.078125 },
{ x: 0.796875, y: 0.078125 },
{ x: 0.796875, y: 0.078125 },
{ x: 0.828125, y: 0.078125 },
{ x: 0.828125, y: 0.078125 },
{ x: 0.859375, y: 0.078125 },
{ x: 0.859375, y: 0.078125 },
{ x: 0.890625, y: 0.078125 },
{ x: 0.890625, y: 0.078125 },
{ x: 0.921875, y: 0.078125 },
{ x: 0.921875, y: 0.078125 },
{ x: 0.953125, y: 0.078125 },
{ x: 0.953125, y: 0.078125 },
{ x: 0.984375, y: 0.078125 },
{ x: 0.984375, y: 0.078125 },
{ x: 0.015625, y: 0.109375 },
{ x: 0.015625, y: 0.109375 },
{ x: 0.046875, y: 0.109375 },
{ x: 0.046875, y: 0.109375 },
{ x: 0.078125, y: 0.109375 },
{ x: 0.078125, y: 0.109375 },
{ x: 0.109375, y: 0.109375 },
{ x: 0.109375, y: 0.109375 },
{ x: 0.140625, y: 0.109375 },
{ x: 0.140625, y: 0.109375 },
{ x: 0.171875, y: 0.109375 },
{ x: 0.171875, y: 0.109375 },
{ x: 0.203125, y: 0.109375 },
{ x: 0.203125, y: 0.109375 },
{ x: 0.234375, y: 0.109375 },
{ x: 0.234375, y: 0.109375 },
{ x: 0.265625, y: 0.109375 },
{ x: 0.265625, y: 0.109375 },
{ x: 0.296875, y: 0.109375 },
{ x: 0.296875, y: 0.109375 },
{ x: 0.328125, y: 0.109375 },
{ x: 0.328125, y: 0.109375 },
{ x: 0.359375, y: 0.109375 },
{ x: 0.359375, y: 0.109375 },
{ x: 0.390625, y: 0.109375 },
{ x: 0.390625, y: 0.109375 },
{ x: 0.421875, y: 0.109375 },
{ x: 0.421875, y: 0.109375 },
{ x: 0.453125, y: 0.109375 },
{ x: 0.453125, y: 0.109375 },
{ x: 0.484375, y: 0.109375 },
{ x: 0.484375, y: 0.109375 },
{ x: 0.515625, y: 0.109375 },
{ x: 0.515625, y: 0.109375 },
{ x: 0.546875, y: 0.109375 },
{ x: 0.546875, y: 0.109375 },
{ x: 0.578125, y: 0.109375 },
{ x: 0.578125, y: 0.109375 },
{ x: 0.609375, y: 0.109375 },
{ x: 0.609375, y: 0.109375 },
{ x: 0.640625, y: 0.109375 },
{ x: 0.640625, y: 0.109375 },
{ x: 0.671875, y: 0.109375 },
{ x: 0.671875, y: 0.109375 },
{ x: 0.703125, y: 0.109375 },
{ x: 0.703125, y: 0.109375 },
{ x: 0.734375, y: 0.109375 },
{ x: 0.734375, y: 0.109375 },
{ x: 0.765625, y: 0.109375 },
{ x: 0.765625, y: 0.109375 },
{ x: 0.796875, y: 0.109375 },
{ x: 0.796875, y: 0.109375 },
{ x: 0.828125, y: 0.109375 },
{ x: 0.828125, y: 0.109375 },
{ x: 0.859375, y: 0.109375 },
{ x: 0.859375, y: 0.109375 },
{ x: 0.890625, y: 0.109375 },
{ x: 0.890625, y: 0.109375 },
{ x: 0.921875, y: 0.109375 },
{ x: 0.921875, y: 0.109375 },
{ x: 0.953125, y: 0.109375 },
{ x: 0.953125, y: 0.109375 },
{ x: 0.984375, y: 0.109375 },
{ x: 0.984375, y: 0.109375 },
{ x: 0.015625, y: 0.140625 },
{ x: 0.015625, y: 0.140625 },
{ x: 0.046875, y: 0.140625 },
{ x: 0.046875, y: 0.140625 },
{ x: 0.078125, y: 0.140625 },
{ x: 0.078125, y: 0.140625 },
{ x: 0.109375, y: 0.140625 },
{ x: 0.109375, y: 0.140625 },
{ x: 0.140625, y: 0.140625 },
{ x: 0.140625, y: 0.140625 },
{ x: 0.171875, y: 0.140625 },
{ x: 0.171875, y: 0.140625 },
{ x: 0.203125, y: 0.140625 },
{ x: 0.203125, y: 0.140625 },
{ x: 0.234375, y: 0.140625 },
{ x: 0.234375, y: 0.140625 },
{ x: 0.265625, y: 0.140625 },
{ x: 0.265625, y: 0.140625 },
{ x: 0.296875, y: 0.140625 },
{ x: 0.296875, y: 0.140625 },
{ x: 0.328125, y: 0.140625 },
{ x: 0.328125, y: 0.140625 },
{ x: 0.359375, y: 0.140625 },
{ x: 0.359375, y: 0.140625 },
{ x: 0.390625, y: 0.140625 },
{ x: 0.390625, y: 0.140625 },
{ x: 0.421875, y: 0.140625 },
{ x: 0.421875, y: 0.140625 },
{ x: 0.453125, y: 0.140625 },
{ x: 0.453125, y: 0.140625 },
{ x: 0.484375, y: 0.140625 },
{ x: 0.484375, y: 0.140625 },
{ x: 0.515625, y: 0.140625 },
{ x: 0.515625, y: 0.140625 },
{ x: 0.546875, y: 0.140625 },
{ x: 0.546875, y: 0.140625 },
{ x: 0.578125, y: 0.140625 },
{ x: 0.578125, y: 0.140625 },
{ x: 0.609375, y: 0.140625 },
{ x: 0.609375, y: 0.140625 },
{ x: 0.640625, y: 0.140625 },
{ x: 0.640625, y: 0.140625 },
{ x: 0.671875, y: 0.140625 },
{ x: 0.671875, y: 0.140625 },
{ x: 0.703125, y: 0.140625 },
{ x: 0.703125, y: 0.140625 },
{ x: 0.734375, y: 0.140625 },
{ x: 0.734375, y: 0.140625 },
{ x: 0.765625, y: 0.140625 },
{ x: 0.765625, y: 0.140625 },
{ x: 0.796875, y: 0.140625 },
{ x: 0.796875, y: 0.140625 },
{ x: 0.828125, y: 0.140625 },
{ x: 0.828125, y: 0.140625 },
{ x: 0.859375, y: 0.140625 },
{ x: 0.859375, y: 0.140625 },
{ x: 0.890625, y: 0.140625 },
{ x: 0.890625, y: 0.140625 },
{ x: 0.921875, y: 0.140625 },
{ x: 0.921875, y: 0.140625 },
{ x: 0.953125, y: 0.140625 },
{ x: 0.953125, y: 0.140625 },
{ x: 0.984375, y: 0.140625 },
{ x: 0.984375, y: 0.140625 },
{ x: 0.015625, y: 0.171875 },
{ x: 0.015625, y: 0.171875 },
{ x: 0.046875, y: 0.171875 },
{ x: 0.046875, y: 0.171875 },
{ x: 0.078125, y: 0.171875 },
{ x: 0.078125, y: 0.171875 },
{ x: 0.109375, y: 0.171875 },
{ x: 0.109375, y: 0.171875 },
{ x: 0.140625, y: 0.171875 },
{ x: 0.140625, y: 0.171875 },
{ x: 0.171875, y: 0.171875 },
{ x: 0.171875, y: 0.171875 },
{ x: 0.203125, y: 0.171875 },
{ x: 0.203125, y: 0.171875 },
{ x: 0.234375, y: 0.171875 },
{ x: 0.234375, y: 0.171875 },
{ x: 0.265625, y: 0.171875 },
{ x: 0.265625, y: 0.171875 },
{ x: 0.296875, y: 0.171875 },
{ x: 0.296875, y: 0.171875 },
{ x: 0.328125, y: 0.171875 },
{ x: 0.328125, y: 0.171875 },
{ x: 0.359375, y: 0.171875 },
{ x: 0.359375, y: 0.171875 },
{ x: 0.390625, y: 0.171875 },
{ x: 0.390625, y: 0.171875 },
{ x: 0.421875, y: 0.171875 },
{ x: 0.421875, y: 0.171875 },
{ x: 0.453125, y: 0.171875 },
{ x: 0.453125, y: 0.171875 },
{ x: 0.484375, y: 0.171875 },
{ x: 0.484375, y: 0.171875 },
{ x: 0.515625, y: 0.171875 },
{ x: 0.515625, y: 0.171875 },
{ x: 0.546875, y: 0.171875 },
{ x: 0.546875, y: 0.171875 },
{ x: 0.578125, y: 0.171875 },
{ x: 0.578125, y: 0.171875 },
{ x: 0.609375, y: 0.171875 },
{ x: 0.609375, y: 0.171875 },
{ x: 0.640625, y: 0.171875 },
{ x: 0.640625, y: 0.171875 },
{ x: 0.671875, y: 0.171875 },
{ x: 0.671875, y: 0.171875 },
{ x: 0.703125, y: 0.171875 },
{ x: 0.703125, y: 0.171875 },
{ x: 0.734375, y: 0.171875 },
{ x: 0.734375, y: 0.171875 },
{ x: 0.765625, y: 0.171875 },
{ x: 0.765625, y: 0.171875 },
{ x: 0.796875, y: 0.171875 },
{ x: 0.796875, y: 0.171875 },
{ x: 0.828125, y: 0.171875 },
{ x: 0.828125, y: 0.171875 },
{ x: 0.859375, y: 0.171875 },
{ x: 0.859375, y: 0.171875 },
{ x: 0.890625, y: 0.171875 },
{ x: 0.890625, y: 0.171875 },
{ x: 0.921875, y: 0.171875 },
{ x: 0.921875, y: 0.171875 },
{ x: 0.953125, y: 0.171875 },
{ x: 0.953125, y: 0.171875 },
{ x: 0.984375, y: 0.171875 },
{ x: 0.984375, y: 0.171875 },
{ x: 0.015625, y: 0.203125 },
{ x: 0.015625, y: 0.203125 },
{ x: 0.046875, y: 0.203125 },
{ x: 0.046875, y: 0.203125 },
{ x: 0.078125, y: 0.203125 },
{ x: 0.078125, y: 0.203125 },
{ x: 0.109375, y: 0.203125 },
{ x: 0.109375, y: 0.203125 },
{ x: 0.140625, y: 0.203125 },
{ x: 0.140625, y: 0.203125 },
{ x: 0.171875, y: 0.203125 },
{ x: 0.171875, y: 0.203125 },
{ x: 0.203125, y: 0.203125 },
{ x: 0.203125, y: 0.203125 },
{ x: 0.234375, y: 0.203125 },
{ x: 0.234375, y: 0.203125 },
{ x: 0.265625, y: 0.203125 },
{ x: 0.265625, y: 0.203125 },
{ x: 0.296875, y: 0.203125 },
{ x: 0.296875, y: 0.203125 },
{ x: 0.328125, y: 0.203125 },
{ x: 0.328125, y: 0.203125 },
{ x: 0.359375, y: 0.203125 },
{ x: 0.359375, y: 0.203125 },
{ x: 0.390625, y: 0.203125 },
{ x: 0.390625, y: 0.203125 },
{ x: 0.421875, y: 0.203125 },
{ x: 0.421875, y: 0.203125 },
{ x: 0.453125, y: 0.203125 },
{ x: 0.453125, y: 0.203125 },
{ x: 0.484375, y: 0.203125 },
{ x: 0.484375, y: 0.203125 },
{ x: 0.515625, y: 0.203125 },
{ x: 0.515625, y: 0.203125 },
{ x: 0.546875, y: 0.203125 },
{ x: 0.546875, y: 0.203125 },
{ x: 0.578125, y: 0.203125 },
{ x: 0.578125, y: 0.203125 },
{ x: 0.609375, y: 0.203125 },
{ x: 0.609375, y: 0.203125 },
{ x: 0.640625, y: 0.203125 },
{ x: 0.640625, y: 0.203125 },
{ x: 0.671875, y: 0.203125 },
{ x: 0.671875, y: 0.203125 },
{ x: 0.703125, y: 0.203125 },
{ x: 0.703125, y: 0.203125 },
{ x: 0.734375, y: 0.203125 },
{ x: 0.734375, y: 0.203125 },
{ x: 0.765625, y: 0.203125 },
{ x: 0.765625, y: 0.203125 },
{ x: 0.796875, y: 0.203125 },
{ x: 0.796875, y: 0.203125 },
{ x: 0.828125, y: 0.203125 },
{ x: 0.828125, y: 0.203125 },
{ x: 0.859375, y: 0.203125 },
{ x: 0.859375, y: 0.203125 },
{ x: 0.890625, y: 0.203125 },
{ x: 0.890625, y: 0.203125 },
{ x: 0.921875, y: 0.203125 },
{ x: 0.921875, y: 0.203125 },
{ x: 0.953125, y: 0.203125 },
{ x: 0.953125, y: 0.203125 },
{ x: 0.984375, y: 0.203125 },
{ x: 0.984375, y: 0.203125 },
{ x: 0.015625, y: 0.234375 },
{ x: 0.015625, y: 0.234375 },
{ x: 0.046875, y: 0.234375 },
{ x: 0.046875, y: 0.234375 },
{ x: 0.078125, y: 0.234375 },
{ x: 0.078125, y: 0.234375 },
{ x: 0.109375, y: 0.234375 },
{ x: 0.109375, y: 0.234375 },
{ x: 0.140625, y: 0.234375 },
{ x: 0.140625, y: 0.234375 },
{ x: 0.171875, y: 0.234375 },
{ x: 0.171875, y: 0.234375 },
{ x: 0.203125, y: 0.234375 },
{ x: 0.203125, y: 0.234375 },
{ x: 0.234375, y: 0.234375 },
{ x: 0.234375, y: 0.234375 },
{ x: 0.265625, y: 0.234375 },
{ x: 0.265625, y: 0.234375 },
{ x: 0.296875, y: 0.234375 },
{ x: 0.296875, y: 0.234375 },
{ x: 0.328125, y: 0.234375 },
{ x: 0.328125, y: 0.234375 },
{ x: 0.359375, y: 0.234375 },
{ x: 0.359375, y: 0.234375 },
{ x: 0.390625, y: 0.234375 },
{ x: 0.390625, y: 0.234375 },
{ x: 0.421875, y: 0.234375 },
{ x: 0.421875, y: 0.234375 },
{ x: 0.453125, y: 0.234375 },
{ x: 0.453125, y: 0.234375 },
{ x: 0.484375, y: 0.234375 },
{ x: 0.484375, y: 0.234375 },
{ x: 0.515625, y: 0.234375 },
{ x: 0.515625, y: 0.234375 },
{ x: 0.546875, y: 0.234375 },
{ x: 0.546875, y: 0.234375 },
{ x: 0.578125, y: 0.234375 },
{ x: 0.578125, y: 0.234375 },
{ x: 0.609375, y: 0.234375 },
{ x: 0.609375, y: 0.234375 },
{ x: 0.640625, y: 0.234375 },
{ x: 0.640625, y: 0.234375 },
{ x: 0.671875, y: 0.234375 },
{ x: 0.671875, y: 0.234375 },
{ x: 0.703125, y: 0.234375 },
{ x: 0.703125, y: 0.234375 },
{ x: 0.734375, y: 0.234375 },
{ x: 0.734375, y: 0.234375 },
{ x: 0.765625, y: 0.234375 },
{ x: 0.765625, y: 0.234375 },
{ x: 0.796875, y: 0.234375 },
{ x: 0.796875, y: 0.234375 },
{ x: 0.828125, y: 0.234375 },
{ x: 0.828125, y: 0.234375 },
{ x: 0.859375, y: 0.234375 },
{ x: 0.859375, y: 0.234375 },
{ x: 0.890625, y: 0.234375 },
{ x: 0.890625, y: 0.234375 },
{ x: 0.921875, y: 0.234375 },
{ x: 0.921875, y: 0.234375 },
{ x: 0.953125, y: 0.234375 },
{ x: 0.953125, y: 0.234375 },
{ x: 0.984375, y: 0.234375 },
{ x: 0.984375, y: 0.234375 },
{ x: 0.015625, y: 0.265625 },
{ x: 0.015625, y: 0.265625 },
{ x: 0.046875, y: 0.265625 },
{ x: 0.046875, y: 0.265625 },
{ x: 0.078125, y: 0.265625 },
{ x: 0.078125, y: 0.265625 },
{ x: 0.109375, y: 0.265625 },
{ x: 0.109375, y: 0.265625 },
{ x: 0.140625, y: 0.265625 },
{ x: 0.140625, y: 0.265625 },
{ x: 0.171875, y: 0.265625 },
{ x: 0.171875, y: 0.265625 },
{ x: 0.203125, y: 0.265625 },
{ x: 0.203125, y: 0.265625 },
{ x: 0.234375, y: 0.265625 },
{ x: 0.234375, y: 0.265625 },
{ x: 0.265625, y: 0.265625 },
{ x: 0.265625, y: 0.265625 },
{ x: 0.296875, y: 0.265625 },
{ x: 0.296875, y: 0.265625 },
{ x: 0.328125, y: 0.265625 },
{ x: 0.328125, y: 0.265625 },
{ x: 0.359375, y: 0.265625 },
{ x: 0.359375, y: 0.265625 },
{ x: 0.390625, y: 0.265625 },
{ x: 0.390625, y: 0.265625 },
{ x: 0.421875, y: 0.265625 },
{ x: 0.421875, y: 0.265625 },
{ x: 0.453125, y: 0.265625 },
{ x: 0.453125, y: 0.265625 },
{ x: 0.484375, y: 0.265625 },
{ x: 0.484375, y: 0.265625 },
{ x: 0.515625, y: 0.265625 },
{ x: 0.515625, y: 0.265625 },
{ x: 0.546875, y: 0.265625 },
{ x: 0.546875, y: 0.265625 },
{ x: 0.578125, y: 0.265625 },
{ x: 0.578125, y: 0.265625 },
{ x: 0.609375, y: 0.265625 },
{ x: 0.609375, y: 0.265625 },
{ x: 0.640625, y: 0.265625 },
{ x: 0.640625, y: 0.265625 },
{ x: 0.671875, y: 0.265625 },
{ x: 0.671875, y: 0.265625 },
{ x: 0.703125, y: 0.265625 },
{ x: 0.703125, y: 0.265625 },
{ x: 0.734375, y: 0.265625 },
{ x: 0.734375, y: 0.265625 },
{ x: 0.765625, y: 0.265625 },
{ x: 0.765625, y: 0.265625 },
{ x: 0.796875, y: 0.265625 },
{ x: 0.796875, y: 0.265625 },
{ x: 0.828125, y: 0.265625 },
{ x: 0.828125, y: 0.265625 },
{ x: 0.859375, y: 0.265625 },
{ x: 0.859375, y: 0.265625 },
{ x: 0.890625, y: 0.265625 },
{ x: 0.890625, y: 0.265625 },
{ x: 0.921875, y: 0.265625 },
{ x: 0.921875, y: 0.265625 },
{ x: 0.953125, y: 0.265625 },
{ x: 0.953125, y: 0.265625 },
{ x: 0.984375, y: 0.265625 },
{ x: 0.984375, y: 0.265625 },
{ x: 0.015625, y: 0.296875 },
{ x: 0.015625, y: 0.296875 },
{ x: 0.046875, y: 0.296875 },
{ x: 0.046875, y: 0.296875 },
{ x: 0.078125, y: 0.296875 },
{ x: 0.078125, y: 0.296875 },
{ x: 0.109375, y: 0.296875 },
{ x: 0.109375, y: 0.296875 },
{ x: 0.140625, y: 0.296875 },
{ x: 0.140625, y: 0.296875 },
{ x: 0.171875, y: 0.296875 },
{ x: 0.171875, y: 0.296875 },
{ x: 0.203125, y: 0.296875 },
{ x: 0.203125, y: 0.296875 },
{ x: 0.234375, y: 0.296875 },
{ x: 0.234375, y: 0.296875 },
{ x: 0.265625, y: 0.296875 },
{ x: 0.265625, y: 0.296875 },
{ x: 0.296875, y: 0.296875 },
{ x: 0.296875, y: 0.296875 },
{ x: 0.328125, y: 0.296875 },
{ x: 0.328125, y: 0.296875 },
{ x: 0.359375, y: 0.296875 },
{ x: 0.359375, y: 0.296875 },
{ x: 0.390625, y: 0.296875 },
{ x: 0.390625, y: 0.296875 },
{ x: 0.421875, y: 0.296875 },
{ x: 0.421875, y: 0.296875 },
{ x: 0.453125, y: 0.296875 },
{ x: 0.453125, y: 0.296875 },
{ x: 0.484375, y: 0.296875 },
{ x: 0.484375, y: 0.296875 },
{ x: 0.515625, y: 0.296875 },
{ x: 0.515625, y: 0.296875 },
{ x: 0.546875, y: 0.296875 },
{ x: 0.546875, y: 0.296875 },
{ x: 0.578125, y: 0.296875 },
{ x: 0.578125, y: 0.296875 },
{ x: 0.609375, y: 0.296875 },
{ x: 0.609375, y: 0.296875 },
{ x: 0.640625, y: 0.296875 },
{ x: 0.640625, y: 0.296875 },
{ x: 0.671875, y: 0.296875 },
{ x: 0.671875, y: 0.296875 },
{ x: 0.703125, y: 0.296875 },
{ x: 0.703125, y: 0.296875 },
{ x: 0.734375, y: 0.296875 },
{ x: 0.734375, y: 0.296875 },
{ x: 0.765625, y: 0.296875 },
{ x: 0.765625, y: 0.296875 },
{ x: 0.796875, y: 0.296875 },
{ x: 0.796875, y: 0.296875 },
{ x: 0.828125, y: 0.296875 },
{ x: 0.828125, y: 0.296875 },
{ x: 0.859375, y: 0.296875 },
{ x: 0.859375, y: 0.296875 },
{ x: 0.890625, y: 0.296875 },
{ x: 0.890625, y: 0.296875 },
{ x: 0.921875, y: 0.296875 },
{ x: 0.921875, y: 0.296875 },
{ x: 0.953125, y: 0.296875 },
{ x: 0.953125, y: 0.296875 },
{ x: 0.984375, y: 0.296875 },
{ x: 0.984375, y: 0.296875 },
{ x: 0.015625, y: 0.328125 },
{ x: 0.015625, y: 0.328125 },
{ x: 0.046875, y: 0.328125 },
{ x: 0.046875, y: 0.328125 },
{ x: 0.078125, y: 0.328125 },
{ x: 0.078125, y: 0.328125 },
{ x: 0.109375, y: 0.328125 },
{ x: 0.109375, y: 0.328125 },
{ x: 0.140625, y: 0.328125 },
{ x: 0.140625, y: 0.328125 },
{ x: 0.171875, y: 0.328125 },
{ x: 0.171875, y: 0.328125 },
{ x: 0.203125, y: 0.328125 },
{ x: 0.203125, y: 0.328125 },
{ x: 0.234375, y: 0.328125 },
{ x: 0.234375, y: 0.328125 },
{ x: 0.265625, y: 0.328125 },
{ x: 0.265625, y: 0.328125 },
{ x: 0.296875, y: 0.328125 },
{ x: 0.296875, y: 0.328125 },
{ x: 0.328125, y: 0.328125 },
{ x: 0.328125, y: 0.328125 },
{ x: 0.359375, y: 0.328125 },
{ x: 0.359375, y: 0.328125 },
{ x: 0.390625, y: 0.328125 },
{ x: 0.390625, y: 0.328125 },
{ x: 0.421875, y: 0.328125 },
{ x: 0.421875, y: 0.328125 },
{ x: 0.453125, y: 0.328125 },
{ x: 0.453125, y: 0.328125 },
{ x: 0.484375, y: 0.328125 },
{ x: 0.484375, y: 0.328125 },
{ x: 0.515625, y: 0.328125 },
{ x: 0.515625, y: 0.328125 },
{ x: 0.546875, y: 0.328125 },
{ x: 0.546875, y: 0.328125 },
{ x: 0.578125, y: 0.328125 },
{ x: 0.578125, y: 0.328125 },
{ x: 0.609375, y: 0.328125 },
{ x: 0.609375, y: 0.328125 },
{ x: 0.640625, y: 0.328125 },
{ x: 0.640625, y: 0.328125 },
{ x: 0.671875, y: 0.328125 },
{ x: 0.671875, y: 0.328125 },
{ x: 0.703125, y: 0.328125 },
{ x: 0.703125, y: 0.328125 },
{ x: 0.734375, y: 0.328125 },
{ x: 0.734375, y: 0.328125 },
{ x: 0.765625, y: 0.328125 },
{ x: 0.765625, y: 0.328125 },
{ x: 0.796875, y: 0.328125 },
{ x: 0.796875, y: 0.328125 },
{ x: 0.828125, y: 0.328125 },
{ x: 0.828125, y: 0.328125 },
{ x: 0.859375, y: 0.328125 },
{ x: 0.859375, y: 0.328125 },
{ x: 0.890625, y: 0.328125 },
{ x: 0.890625, y: 0.328125 },
{ x: 0.921875, y: 0.328125 },
{ x: 0.921875, y: 0.328125 },
{ x: 0.953125, y: 0.328125 },
{ x: 0.953125, y: 0.328125 },
{ x: 0.984375, y: 0.328125 },
{ x: 0.984375, y: 0.328125 },
{ x: 0.015625, y: 0.359375 },
{ x: 0.015625, y: 0.359375 },
{ x: 0.046875, y: 0.359375 },
{ x: 0.046875, y: 0.359375 },
{ x: 0.078125, y: 0.359375 },
{ x: 0.078125, y: 0.359375 },
{ x: 0.109375, y: 0.359375 },
{ x: 0.109375, y: 0.359375 },
{ x: 0.140625, y: 0.359375 },
{ x: 0.140625, y: 0.359375 },
{ x: 0.171875, y: 0.359375 },
{ x: 0.171875, y: 0.359375 },
{ x: 0.203125, y: 0.359375 },
{ x: 0.203125, y: 0.359375 },
{ x: 0.234375, y: 0.359375 },
{ x: 0.234375, y: 0.359375 },
{ x: 0.265625, y: 0.359375 },
{ x: 0.265625, y: 0.359375 },
{ x: 0.296875, y: 0.359375 },
{ x: 0.296875, y: 0.359375 },
{ x: 0.328125, y: 0.359375 },
{ x: 0.328125, y: 0.359375 },
{ x: 0.359375, y: 0.359375 },
{ x: 0.359375, y: 0.359375 },
{ x: 0.390625, y: 0.359375 },
{ x: 0.390625, y: 0.359375 },
{ x: 0.421875, y: 0.359375 },
{ x: 0.421875, y: 0.359375 },
{ x: 0.453125, y: 0.359375 },
{ x: 0.453125, y: 0.359375 },
{ x: 0.484375, y: 0.359375 },
{ x: 0.484375, y: 0.359375 },
{ x: 0.515625, y: 0.359375 },
{ x: 0.515625, y: 0.359375 },
{ x: 0.546875, y: 0.359375 },
{ x: 0.546875, y: 0.359375 },
{ x: 0.578125, y: 0.359375 },
{ x: 0.578125, y: 0.359375 },
{ x: 0.609375, y: 0.359375 },
{ x: 0.609375, y: 0.359375 },
{ x: 0.640625, y: 0.359375 },
{ x: 0.640625, y: 0.359375 },
{ x: 0.671875, y: 0.359375 },
{ x: 0.671875, y: 0.359375 },
{ x: 0.703125, y: 0.359375 },
{ x: 0.703125, y: 0.359375 },
{ x: 0.734375, y: 0.359375 },
{ x: 0.734375, y: 0.359375 },
{ x: 0.765625, y: 0.359375 },
{ x: 0.765625, y: 0.359375 },
{ x: 0.796875, y: 0.359375 },
{ x: 0.796875, y: 0.359375 },
{ x: 0.828125, y: 0.359375 },
{ x: 0.828125, y: 0.359375 },
{ x: 0.859375, y: 0.359375 },
{ x: 0.859375, y: 0.359375 },
{ x: 0.890625, y: 0.359375 },
{ x: 0.890625, y: 0.359375 },
{ x: 0.921875, y: 0.359375 },
{ x: 0.921875, y: 0.359375 },
{ x: 0.953125, y: 0.359375 },
{ x: 0.953125, y: 0.359375 },
{ x: 0.984375, y: 0.359375 },
{ x: 0.984375, y: 0.359375 },
{ x: 0.015625, y: 0.390625 },
{ x: 0.015625, y: 0.390625 },
{ x: 0.046875, y: 0.390625 },
{ x: 0.046875, y: 0.390625 },
{ x: 0.078125, y: 0.390625 },
{ x: 0.078125, y: 0.390625 },
{ x: 0.109375, y: 0.390625 },
{ x: 0.109375, y: 0.390625 },
{ x: 0.140625, y: 0.390625 },
{ x: 0.140625, y: 0.390625 },
{ x: 0.171875, y: 0.390625 },
{ x: 0.171875, y: 0.390625 },
{ x: 0.203125, y: 0.390625 },
{ x: 0.203125, y: 0.390625 },
{ x: 0.234375, y: 0.390625 },
{ x: 0.234375, y: 0.390625 },
{ x: 0.265625, y: 0.390625 },
{ x: 0.265625, y: 0.390625 },
{ x: 0.296875, y: 0.390625 },
{ x: 0.296875, y: 0.390625 },
{ x: 0.328125, y: 0.390625 },
{ x: 0.328125, y: 0.390625 },
{ x: 0.359375, y: 0.390625 },
{ x: 0.359375, y: 0.390625 },
{ x: 0.390625, y: 0.390625 },
{ x: 0.390625, y: 0.390625 },
{ x: 0.421875, y: 0.390625 },
{ x: 0.421875, y: 0.390625 },
{ x: 0.453125, y: 0.390625 },
{ x: 0.453125, y: 0.390625 },
{ x: 0.484375, y: 0.390625 },
{ x: 0.484375, y: 0.390625 },
{ x: 0.515625, y: 0.390625 },
{ x: 0.515625, y: 0.390625 },
{ x: 0.546875, y: 0.390625 },
{ x: 0.546875, y: 0.390625 },
{ x: 0.578125, y: 0.390625 },
{ x: 0.578125, y: 0.390625 },
{ x: 0.609375, y: 0.390625 },
{ x: 0.609375, y: 0.390625 },
{ x: 0.640625, y: 0.390625 },
{ x: 0.640625, y: 0.390625 },
{ x: 0.671875, y: 0.390625 },
{ x: 0.671875, y: 0.390625 },
{ x: 0.703125, y: 0.390625 },
{ x: 0.703125, y: 0.390625 },
{ x: 0.734375, y: 0.390625 },
{ x: 0.734375, y: 0.390625 },
{ x: 0.765625, y: 0.390625 },
{ x: 0.765625, y: 0.390625 },
{ x: 0.796875, y: 0.390625 },
{ x: 0.796875, y: 0.390625 },
{ x: 0.828125, y: 0.390625 },
{ x: 0.828125, y: 0.390625 },
{ x: 0.859375, y: 0.390625 },
{ x: 0.859375, y: 0.390625 },
{ x: 0.890625, y: 0.390625 },
{ x: 0.890625, y: 0.390625 },
{ x: 0.921875, y: 0.390625 },
{ x: 0.921875, y: 0.390625 },
{ x: 0.953125, y: 0.390625 },
{ x: 0.953125, y: 0.390625 },
{ x: 0.984375, y: 0.390625 },
{ x: 0.984375, y: 0.390625 },
{ x: 0.015625, y: 0.421875 },
{ x: 0.015625, y: 0.421875 },
{ x: 0.046875, y: 0.421875 },
{ x: 0.046875, y: 0.421875 },
{ x: 0.078125, y: 0.421875 },
{ x: 0.078125, y: 0.421875 },
{ x: 0.109375, y: 0.421875 },
{ x: 0.109375, y: 0.421875 },
{ x: 0.140625, y: 0.421875 },
{ x: 0.140625, y: 0.421875 },
{ x: 0.171875, y: 0.421875 },
{ x: 0.171875, y: 0.421875 },
{ x: 0.203125, y: 0.421875 },
{ x: 0.203125, y: 0.421875 },
{ x: 0.234375, y: 0.421875 },
{ x: 0.234375, y: 0.421875 },
{ x: 0.265625, y: 0.421875 },
{ x: 0.265625, y: 0.421875 },
{ x: 0.296875, y: 0.421875 },
{ x: 0.296875, y: 0.421875 },
{ x: 0.328125, y: 0.421875 },
{ x: 0.328125, y: 0.421875 },
{ x: 0.359375, y: 0.421875 },
{ x: 0.359375, y: 0.421875 },
{ x: 0.390625, y: 0.421875 },
{ x: 0.390625, y: 0.421875 },
{ x: 0.421875, y: 0.421875 },
{ x: 0.421875, y: 0.421875 },
{ x: 0.453125, y: 0.421875 },
{ x: 0.453125, y: 0.421875 },
{ x: 0.484375, y: 0.421875 },
{ x: 0.484375, y: 0.421875 },
{ x: 0.515625, y: 0.421875 },
{ x: 0.515625, y: 0.421875 },
{ x: 0.546875, y: 0.421875 },
{ x: 0.546875, y: 0.421875 },
{ x: 0.578125, y: 0.421875 },
{ x: 0.578125, y: 0.421875 },
{ x: 0.609375, y: 0.421875 },
{ x: 0.609375, y: 0.421875 },
{ x: 0.640625, y: 0.421875 },
{ x: 0.640625, y: 0.421875 },
{ x: 0.671875, y: 0.421875 },
{ x: 0.671875, y: 0.421875 },
{ x: 0.703125, y: 0.421875 },
{ x: 0.703125, y: 0.421875 },
{ x: 0.734375, y: 0.421875 },
{ x: 0.734375, y: 0.421875 },
{ x: 0.765625, y: 0.421875 },
{ x: 0.765625, y: 0.421875 },
{ x: 0.796875, y: 0.421875 },
{ x: 0.796875, y: 0.421875 },
{ x: 0.828125, y: 0.421875 },
{ x: 0.828125, y: 0.421875 },
{ x: 0.859375, y: 0.421875 },
{ x: 0.859375, y: 0.421875 },
{ x: 0.890625, y: 0.421875 },
{ x: 0.890625, y: 0.421875 },
{ x: 0.921875, y: 0.421875 },
{ x: 0.921875, y: 0.421875 },
{ x: 0.953125, y: 0.421875 },
{ x: 0.953125, y: 0.421875 },
{ x: 0.984375, y: 0.421875 },
{ x: 0.984375, y: 0.421875 },
{ x: 0.015625, y: 0.453125 },
{ x: 0.015625, y: 0.453125 },
{ x: 0.046875, y: 0.453125 },
{ x: 0.046875, y: 0.453125 },
{ x: 0.078125, y: 0.453125 },
{ x: 0.078125, y: 0.453125 },
{ x: 0.109375, y: 0.453125 },
{ x: 0.109375, y: 0.453125 },
{ x: 0.140625, y: 0.453125 },
{ x: 0.140625, y: 0.453125 },
{ x: 0.171875, y: 0.453125 },
{ x: 0.171875, y: 0.453125 },
{ x: 0.203125, y: 0.453125 },
{ x: 0.203125, y: 0.453125 },
{ x: 0.234375, y: 0.453125 },
{ x: 0.234375, y: 0.453125 },
{ x: 0.265625, y: 0.453125 },
{ x: 0.265625, y: 0.453125 },
{ x: 0.296875, y: 0.453125 },
{ x: 0.296875, y: 0.453125 },
{ x: 0.328125, y: 0.453125 },
{ x: 0.328125, y: 0.453125 },
{ x: 0.359375, y: 0.453125 },
{ x: 0.359375, y: 0.453125 },
{ x: 0.390625, y: 0.453125 },
{ x: 0.390625, y: 0.453125 },
{ x: 0.421875, y: 0.453125 },
{ x: 0.421875, y: 0.453125 },
{ x: 0.453125, y: 0.453125 },
{ x: 0.453125, y: 0.453125 },
{ x: 0.484375, y: 0.453125 },
{ x: 0.484375, y: 0.453125 },
{ x: 0.515625, y: 0.453125 },
{ x: 0.515625, y: 0.453125 },
{ x: 0.546875, y: 0.453125 },
{ x: 0.546875, y: 0.453125 },
{ x: 0.578125, y: 0.453125 },
{ x: 0.578125, y: 0.453125 },
{ x: 0.609375, y: 0.453125 },
{ x: 0.609375, y: 0.453125 },
{ x: 0.640625, y: 0.453125 },
{ x: 0.640625, y: 0.453125 },
{ x: 0.671875, y: 0.453125 },
{ x: 0.671875, y: 0.453125 },
{ x: 0.703125, y: 0.453125 },
{ x: 0.703125, y: 0.453125 },
{ x: 0.734375, y: 0.453125 },
{ x: 0.734375, y: 0.453125 },
{ x: 0.765625, y: 0.453125 },
{ x: 0.765625, y: 0.453125 },
{ x: 0.796875, y: 0.453125 },
{ x: 0.796875, y: 0.453125 },
{ x: 0.828125, y: 0.453125 },
{ x: 0.828125, y: 0.453125 },
{ x: 0.859375, y: 0.453125 },
{ x: 0.859375, y: 0.453125 },
{ x: 0.890625, y: 0.453125 },
{ x: 0.890625, y: 0.453125 },
{ x: 0.921875, y: 0.453125 },
{ x: 0.921875, y: 0.453125 },
{ x: 0.953125, y: 0.453125 },
{ x: 0.953125, y: 0.453125 },
{ x: 0.984375, y: 0.453125 },
{ x: 0.984375, y: 0.453125 },
{ x: 0.015625, y: 0.484375 },
{ x: 0.015625, y: 0.484375 },
{ x: 0.046875, y: 0.484375 },
{ x: 0.046875, y: 0.484375 },
{ x: 0.078125, y: 0.484375 },
{ x: 0.078125, y: 0.484375 },
{ x: 0.109375, y: 0.484375 },
{ x: 0.109375, y: 0.484375 },
{ x: 0.140625, y: 0.484375 },
{ x: 0.140625, y: 0.484375 },
{ x: 0.171875, y: 0.484375 },
{ x: 0.171875, y: 0.484375 },
{ x: 0.203125, y: 0.484375 },
{ x: 0.203125, y: 0.484375 },
{ x: 0.234375, y: 0.484375 },
{ x: 0.234375, y: 0.484375 },
{ x: 0.265625, y: 0.484375 },
{ x: 0.265625, y: 0.484375 },
{ x: 0.296875, y: 0.484375 },
{ x: 0.296875, y: 0.484375 },
{ x: 0.328125, y: 0.484375 },
{ x: 0.328125, y: 0.484375 },
{ x: 0.359375, y: 0.484375 },
{ x: 0.359375, y: 0.484375 },
{ x: 0.390625, y: 0.484375 },
{ x: 0.390625, y: 0.484375 },
{ x: 0.421875, y: 0.484375 },
{ x: 0.421875, y: 0.484375 },
{ x: 0.453125, y: 0.484375 },
{ x: 0.453125, y: 0.484375 },
{ x: 0.484375, y: 0.484375 },
{ x: 0.484375, y: 0.484375 },
{ x: 0.515625, y: 0.484375 },
{ x: 0.515625, y: 0.484375 },
{ x: 0.546875, y: 0.484375 },
{ x: 0.546875, y: 0.484375 },
{ x: 0.578125, y: 0.484375 },
{ x: 0.578125, y: 0.484375 },
{ x: 0.609375, y: 0.484375 },
{ x: 0.609375, y: 0.484375 },
{ x: 0.640625, y: 0.484375 },
{ x: 0.640625, y: 0.484375 },
{ x: 0.671875, y: 0.484375 },
{ x: 0.671875, y: 0.484375 },
{ x: 0.703125, y: 0.484375 },
{ x: 0.703125, y: 0.484375 },
{ x: 0.734375, y: 0.484375 },
{ x: 0.734375, y: 0.484375 },
{ x: 0.765625, y: 0.484375 },
{ x: 0.765625, y: 0.484375 },
{ x: 0.796875, y: 0.484375 },
{ x: 0.796875, y: 0.484375 },
{ x: 0.828125, y: 0.484375 },
{ x: 0.828125, y: 0.484375 },
{ x: 0.859375, y: 0.484375 },
{ x: 0.859375, y: 0.484375 },
{ x: 0.890625, y: 0.484375 },
{ x: 0.890625, y: 0.484375 },
{ x: 0.921875, y: 0.484375 },
{ x: 0.921875, y: 0.484375 },
{ x: 0.953125, y: 0.484375 },
{ x: 0.953125, y: 0.484375 },
{ x: 0.984375, y: 0.484375 },
{ x: 0.984375, y: 0.484375 },
{ x: 0.015625, y: 0.515625 },
{ x: 0.015625, y: 0.515625 },
{ x: 0.046875, y: 0.515625 },
{ x: 0.046875, y: 0.515625 },
{ x: 0.078125, y: 0.515625 },
{ x: 0.078125, y: 0.515625 },
{ x: 0.109375, y: 0.515625 },
{ x: 0.109375, y: 0.515625 },
{ x: 0.140625, y: 0.515625 },
{ x: 0.140625, y: 0.515625 },
{ x: 0.171875, y: 0.515625 },
{ x: 0.171875, y: 0.515625 },
{ x: 0.203125, y: 0.515625 },
{ x: 0.203125, y: 0.515625 },
{ x: 0.234375, y: 0.515625 },
{ x: 0.234375, y: 0.515625 },
{ x: 0.265625, y: 0.515625 },
{ x: 0.265625, y: 0.515625 },
{ x: 0.296875, y: 0.515625 },
{ x: 0.296875, y: 0.515625 },
{ x: 0.328125, y: 0.515625 },
{ x: 0.328125, y: 0.515625 },
{ x: 0.359375, y: 0.515625 },
{ x: 0.359375, y: 0.515625 },
{ x: 0.390625, y: 0.515625 },
{ x: 0.390625, y: 0.515625 },
{ x: 0.421875, y: 0.515625 },
{ x: 0.421875, y: 0.515625 },
{ x: 0.453125, y: 0.515625 },
{ x: 0.453125, y: 0.515625 },
{ x: 0.484375, y: 0.515625 },
{ x: 0.484375, y: 0.515625 },
{ x: 0.515625, y: 0.515625 },
{ x: 0.515625, y: 0.515625 },
{ x: 0.546875, y: 0.515625 },
{ x: 0.546875, y: 0.515625 },
{ x: 0.578125, y: 0.515625 },
{ x: 0.578125, y: 0.515625 },
{ x: 0.609375, y: 0.515625 },
{ x: 0.609375, y: 0.515625 },
{ x: 0.640625, y: 0.515625 },
{ x: 0.640625, y: 0.515625 },
{ x: 0.671875, y: 0.515625 },
{ x: 0.671875, y: 0.515625 },
{ x: 0.703125, y: 0.515625 },
{ x: 0.703125, y: 0.515625 },
{ x: 0.734375, y: 0.515625 },
{ x: 0.734375, y: 0.515625 },
{ x: 0.765625, y: 0.515625 },
{ x: 0.765625, y: 0.515625 },
{ x: 0.796875, y: 0.515625 },
{ x: 0.796875, y: 0.515625 },
{ x: 0.828125, y: 0.515625 },
{ x: 0.828125, y: 0.515625 },
{ x: 0.859375, y: 0.515625 },
{ x: 0.859375, y: 0.515625 },
{ x: 0.890625, y: 0.515625 },
{ x: 0.890625, y: 0.515625 },
{ x: 0.921875, y: 0.515625 },
{ x: 0.921875, y: 0.515625 },
{ x: 0.953125, y: 0.515625 },
{ x: 0.953125, y: 0.515625 },
{ x: 0.984375, y: 0.515625 },
{ x: 0.984375, y: 0.515625 },
{ x: 0.015625, y: 0.546875 },
{ x: 0.015625, y: 0.546875 },
{ x: 0.046875, y: 0.546875 },
{ x: 0.046875, y: 0.546875 },
{ x: 0.078125, y: 0.546875 },
{ x: 0.078125, y: 0.546875 },
{ x: 0.109375, y: 0.546875 },
{ x: 0.109375, y: 0.546875 },
{ x: 0.140625, y: 0.546875 },
{ x: 0.140625, y: 0.546875 },
{ x: 0.171875, y: 0.546875 },
{ x: 0.171875, y: 0.546875 },
{ x: 0.203125, y: 0.546875 },
{ x: 0.203125, y: 0.546875 },
{ x: 0.234375, y: 0.546875 },
{ x: 0.234375, y: 0.546875 },
{ x: 0.265625, y: 0.546875 },
{ x: 0.265625, y: 0.546875 },
{ x: 0.296875, y: 0.546875 },
{ x: 0.296875, y: 0.546875 },
{ x: 0.328125, y: 0.546875 },
{ x: 0.328125, y: 0.546875 },
{ x: 0.359375, y: 0.546875 },
{ x: 0.359375, y: 0.546875 },
{ x: 0.390625, y: 0.546875 },
{ x: 0.390625, y: 0.546875 },
{ x: 0.421875, y: 0.546875 },
{ x: 0.421875, y: 0.546875 },
{ x: 0.453125, y: 0.546875 },
{ x: 0.453125, y: 0.546875 },
{ x: 0.484375, y: 0.546875 },
{ x: 0.484375, y: 0.546875 },
{ x: 0.515625, y: 0.546875 },
{ x: 0.515625, y: 0.546875 },
{ x: 0.546875, y: 0.546875 },
{ x: 0.546875, y: 0.546875 },
{ x: 0.578125, y: 0.546875 },
{ x: 0.578125, y: 0.546875 },
{ x: 0.609375, y: 0.546875 },
{ x: 0.609375, y: 0.546875 },
{ x: 0.640625, y: 0.546875 },
{ x: 0.640625, y: 0.546875 },
{ x: 0.671875, y: 0.546875 },
{ x: 0.671875, y: 0.546875 },
{ x: 0.703125, y: 0.546875 },
{ x: 0.703125, y: 0.546875 },
{ x: 0.734375, y: 0.546875 },
{ x: 0.734375, y: 0.546875 },
{ x: 0.765625, y: 0.546875 },
{ x: 0.765625, y: 0.546875 },
{ x: 0.796875, y: 0.546875 },
{ x: 0.796875, y: 0.546875 },
{ x: 0.828125, y: 0.546875 },
{ x: 0.828125, y: 0.546875 },
{ x: 0.859375, y: 0.546875 },
{ x: 0.859375, y: 0.546875 },
{ x: 0.890625, y: 0.546875 },
{ x: 0.890625, y: 0.546875 },
{ x: 0.921875, y: 0.546875 },
{ x: 0.921875, y: 0.546875 },
{ x: 0.953125, y: 0.546875 },
{ x: 0.953125, y: 0.546875 },
{ x: 0.984375, y: 0.546875 },
{ x: 0.984375, y: 0.546875 },
{ x: 0.015625, y: 0.578125 },
{ x: 0.015625, y: 0.578125 },
{ x: 0.046875, y: 0.578125 },
{ x: 0.046875, y: 0.578125 },
{ x: 0.078125, y: 0.578125 },
{ x: 0.078125, y: 0.578125 },
{ x: 0.109375, y: 0.578125 },
{ x: 0.109375, y: 0.578125 },
{ x: 0.140625, y: 0.578125 },
{ x: 0.140625, y: 0.578125 },
{ x: 0.171875, y: 0.578125 },
{ x: 0.171875, y: 0.578125 },
{ x: 0.203125, y: 0.578125 },
{ x: 0.203125, y: 0.578125 },
{ x: 0.234375, y: 0.578125 },
{ x: 0.234375, y: 0.578125 },
{ x: 0.265625, y: 0.578125 },
{ x: 0.265625, y: 0.578125 },
{ x: 0.296875, y: 0.578125 },
{ x: 0.296875, y: 0.578125 },
{ x: 0.328125, y: 0.578125 },
{ x: 0.328125, y: 0.578125 },
{ x: 0.359375, y: 0.578125 },
{ x: 0.359375, y: 0.578125 },
{ x: 0.390625, y: 0.578125 },
{ x: 0.390625, y: 0.578125 },
{ x: 0.421875, y: 0.578125 },
{ x: 0.421875, y: 0.578125 },
{ x: 0.453125, y: 0.578125 },
{ x: 0.453125, y: 0.578125 },
{ x: 0.484375, y: 0.578125 },
{ x: 0.484375, y: 0.578125 },
{ x: 0.515625, y: 0.578125 },
{ x: 0.515625, y: 0.578125 },
{ x: 0.546875, y: 0.578125 },
{ x: 0.546875, y: 0.578125 },
{ x: 0.578125, y: 0.578125 },
{ x: 0.578125, y: 0.578125 },
{ x: 0.609375, y: 0.578125 },
{ x: 0.609375, y: 0.578125 },
{ x: 0.640625, y: 0.578125 },
{ x: 0.640625, y: 0.578125 },
{ x: 0.671875, y: 0.578125 },
{ x: 0.671875, y: 0.578125 },
{ x: 0.703125, y: 0.578125 },
{ x: 0.703125, y: 0.578125 },
{ x: 0.734375, y: 0.578125 },
{ x: 0.734375, y: 0.578125 },
{ x: 0.765625, y: 0.578125 },
{ x: 0.765625, y: 0.578125 },
{ x: 0.796875, y: 0.578125 },
{ x: 0.796875, y: 0.578125 },
{ x: 0.828125, y: 0.578125 },
{ x: 0.828125, y: 0.578125 },
{ x: 0.859375, y: 0.578125 },
{ x: 0.859375, y: 0.578125 },
{ x: 0.890625, y: 0.578125 },
{ x: 0.890625, y: 0.578125 },
{ x: 0.921875, y: 0.578125 },
{ x: 0.921875, y: 0.578125 },
{ x: 0.953125, y: 0.578125 },
{ x: 0.953125, y: 0.578125 },
{ x: 0.984375, y: 0.578125 },
{ x: 0.984375, y: 0.578125 },
{ x: 0.015625, y: 0.609375 },
{ x: 0.015625, y: 0.609375 },
{ x: 0.046875, y: 0.609375 },
{ x: 0.046875, y: 0.609375 },
{ x: 0.078125, y: 0.609375 },
{ x: 0.078125, y: 0.609375 },
{ x: 0.109375, y: 0.609375 },
{ x: 0.109375, y: 0.609375 },
{ x: 0.140625, y: 0.609375 },
{ x: 0.140625, y: 0.609375 },
{ x: 0.171875, y: 0.609375 },
{ x: 0.171875, y: 0.609375 },
{ x: 0.203125, y: 0.609375 },
{ x: 0.203125, y: 0.609375 },
{ x: 0.234375, y: 0.609375 },
{ x: 0.234375, y: 0.609375 },
{ x: 0.265625, y: 0.609375 },
{ x: 0.265625, y: 0.609375 },
{ x: 0.296875, y: 0.609375 },
{ x: 0.296875, y: 0.609375 },
{ x: 0.328125, y: 0.609375 },
{ x: 0.328125, y: 0.609375 },
{ x: 0.359375, y: 0.609375 },
{ x: 0.359375, y: 0.609375 },
{ x: 0.390625, y: 0.609375 },
{ x: 0.390625, y: 0.609375 },
{ x: 0.421875, y: 0.609375 },
{ x: 0.421875, y: 0.609375 },
{ x: 0.453125, y: 0.609375 },
{ x: 0.453125, y: 0.609375 },
{ x: 0.484375, y: 0.609375 },
{ x: 0.484375, y: 0.609375 },
{ x: 0.515625, y: 0.609375 },
{ x: 0.515625, y: 0.609375 },
{ x: 0.546875, y: 0.609375 },
{ x: 0.546875, y: 0.609375 },
{ x: 0.578125, y: 0.609375 },
{ x: 0.578125, y: 0.609375 },
{ x: 0.609375, y: 0.609375 },
{ x: 0.609375, y: 0.609375 },
{ x: 0.640625, y: 0.609375 },
{ x: 0.640625, y: 0.609375 },
{ x: 0.671875, y: 0.609375 },
{ x: 0.671875, y: 0.609375 },
{ x: 0.703125, y: 0.609375 },
{ x: 0.703125, y: 0.609375 },
{ x: 0.734375, y: 0.609375 },
{ x: 0.734375, y: 0.609375 },
{ x: 0.765625, y: 0.609375 },
{ x: 0.765625, y: 0.609375 },
{ x: 0.796875, y: 0.609375 },
{ x: 0.796875, y: 0.609375 },
{ x: 0.828125, y: 0.609375 },
{ x: 0.828125, y: 0.609375 },
{ x: 0.859375, y: 0.609375 },
{ x: 0.859375, y: 0.609375 },
{ x: 0.890625, y: 0.609375 },
{ x: 0.890625, y: 0.609375 },
{ x: 0.921875, y: 0.609375 },
{ x: 0.921875, y: 0.609375 },
{ x: 0.953125, y: 0.609375 },
{ x: 0.953125, y: 0.609375 },
{ x: 0.984375, y: 0.609375 },
{ x: 0.984375, y: 0.609375 },
{ x: 0.015625, y: 0.640625 },
{ x: 0.015625, y: 0.640625 },
{ x: 0.046875, y: 0.640625 },
{ x: 0.046875, y: 0.640625 },
{ x: 0.078125, y: 0.640625 },
{ x: 0.078125, y: 0.640625 },
{ x: 0.109375, y: 0.640625 },
{ x: 0.109375, y: 0.640625 },
{ x: 0.140625, y: 0.640625 },
{ x: 0.140625, y: 0.640625 },
{ x: 0.171875, y: 0.640625 },
{ x: 0.171875, y: 0.640625 },
{ x: 0.203125, y: 0.640625 },
{ x: 0.203125, y: 0.640625 },
{ x: 0.234375, y: 0.640625 },
{ x: 0.234375, y: 0.640625 },
{ x: 0.265625, y: 0.640625 },
{ x: 0.265625, y: 0.640625 },
{ x: 0.296875, y: 0.640625 },
{ x: 0.296875, y: 0.640625 },
{ x: 0.328125, y: 0.640625 },
{ x: 0.328125, y: 0.640625 },
{ x: 0.359375, y: 0.640625 },
{ x: 0.359375, y: 0.640625 },
{ x: 0.390625, y: 0.640625 },
{ x: 0.390625, y: 0.640625 },
{ x: 0.421875, y: 0.640625 },
{ x: 0.421875, y: 0.640625 },
{ x: 0.453125, y: 0.640625 },
{ x: 0.453125, y: 0.640625 },
{ x: 0.484375, y: 0.640625 },
{ x: 0.484375, y: 0.640625 },
{ x: 0.515625, y: 0.640625 },
{ x: 0.515625, y: 0.640625 },
{ x: 0.546875, y: 0.640625 },
{ x: 0.546875, y: 0.640625 },
{ x: 0.578125, y: 0.640625 },
{ x: 0.578125, y: 0.640625 },
{ x: 0.609375, y: 0.640625 },
{ x: 0.609375, y: 0.640625 },
{ x: 0.640625, y: 0.640625 },
{ x: 0.640625, y: 0.640625 },
{ x: 0.671875, y: 0.640625 },
{ x: 0.671875, y: 0.640625 },
{ x: 0.703125, y: 0.640625 },
{ x: 0.703125, y: 0.640625 },
{ x: 0.734375, y: 0.640625 },
{ x: 0.734375, y: 0.640625 },
{ x: 0.765625, y: 0.640625 },
{ x: 0.765625, y: 0.640625 },
{ x: 0.796875, y: 0.640625 },
{ x: 0.796875, y: 0.640625 },
{ x: 0.828125, y: 0.640625 },
{ x: 0.828125, y: 0.640625 },
{ x: 0.859375, y: 0.640625 },
{ x: 0.859375, y: 0.640625 },
{ x: 0.890625, y: 0.640625 },
{ x: 0.890625, y: 0.640625 },
{ x: 0.921875, y: 0.640625 },
{ x: 0.921875, y: 0.640625 },
{ x: 0.953125, y: 0.640625 },
{ x: 0.953125, y: 0.640625 },
{ x: 0.984375, y: 0.640625 },
{ x: 0.984375, y: 0.640625 },
{ x: 0.015625, y: 0.671875 },
{ x: 0.015625, y: 0.671875 },
{ x: 0.046875, y: 0.671875 },
{ x: 0.046875, y: 0.671875 },
{ x: 0.078125, y: 0.671875 },
{ x: 0.078125, y: 0.671875 },
{ x: 0.109375, y: 0.671875 },
{ x: 0.109375, y: 0.671875 },
{ x: 0.140625, y: 0.671875 },
{ x: 0.140625, y: 0.671875 },
{ x: 0.171875, y: 0.671875 },
{ x: 0.171875, y: 0.671875 },
{ x: 0.203125, y: 0.671875 },
{ x: 0.203125, y: 0.671875 },
{ x: 0.234375, y: 0.671875 },
{ x: 0.234375, y: 0.671875 },
{ x: 0.265625, y: 0.671875 },
{ x: 0.265625, y: 0.671875 },
{ x: 0.296875, y: 0.671875 },
{ x: 0.296875, y: 0.671875 },
{ x: 0.328125, y: 0.671875 },
{ x: 0.328125, y: 0.671875 },
{ x: 0.359375, y: 0.671875 },
{ x: 0.359375, y: 0.671875 },
{ x: 0.390625, y: 0.671875 },
{ x: 0.390625, y: 0.671875 },
{ x: 0.421875, y: 0.671875 },
{ x: 0.421875, y: 0.671875 },
{ x: 0.453125, y: 0.671875 },
{ x: 0.453125, y: 0.671875 },
{ x: 0.484375, y: 0.671875 },
{ x: 0.484375, y: 0.671875 },
{ x: 0.515625, y: 0.671875 },
{ x: 0.515625, y: 0.671875 },
{ x: 0.546875, y: 0.671875 },
{ x: 0.546875, y: 0.671875 },
{ x: 0.578125, y: 0.671875 },
{ x: 0.578125, y: 0.671875 },
{ x: 0.609375, y: 0.671875 },
{ x: 0.609375, y: 0.671875 },
{ x: 0.640625, y: 0.671875 },
{ x: 0.640625, y: 0.671875 },
{ x: 0.671875, y: 0.671875 },
{ x: 0.671875, y: 0.671875 },
{ x: 0.703125, y: 0.671875 },
{ x: 0.703125, y: 0.671875 },
{ x: 0.734375, y: 0.671875 },
{ x: 0.734375, y: 0.671875 },
{ x: 0.765625, y: 0.671875 },
{ x: 0.765625, y: 0.671875 },
{ x: 0.796875, y: 0.671875 },
{ x: 0.796875, y: 0.671875 },
{ x: 0.828125, y: 0.671875 },
{ x: 0.828125, y: 0.671875 },
{ x: 0.859375, y: 0.671875 },
{ x: 0.859375, y: 0.671875 },
{ x: 0.890625, y: 0.671875 },
{ x: 0.890625, y: 0.671875 },
{ x: 0.921875, y: 0.671875 },
{ x: 0.921875, y: 0.671875 },
{ x: 0.953125, y: 0.671875 },
{ x: 0.953125, y: 0.671875 },
{ x: 0.984375, y: 0.671875 },
{ x: 0.984375, y: 0.671875 },
{ x: 0.015625, y: 0.703125 },
{ x: 0.015625, y: 0.703125 },
{ x: 0.046875, y: 0.703125 },
{ x: 0.046875, y: 0.703125 },
{ x: 0.078125, y: 0.703125 },
{ x: 0.078125, y: 0.703125 },
{ x: 0.109375, y: 0.703125 },
{ x: 0.109375, y: 0.703125 },
{ x: 0.140625, y: 0.703125 },
{ x: 0.140625, y: 0.703125 },
{ x: 0.171875, y: 0.703125 },
{ x: 0.171875, y: 0.703125 },
{ x: 0.203125, y: 0.703125 },
{ x: 0.203125, y: 0.703125 },
{ x: 0.234375, y: 0.703125 },
{ x: 0.234375, y: 0.703125 },
{ x: 0.265625, y: 0.703125 },
{ x: 0.265625, y: 0.703125 },
{ x: 0.296875, y: 0.703125 },
{ x: 0.296875, y: 0.703125 },
{ x: 0.328125, y: 0.703125 },
{ x: 0.328125, y: 0.703125 },
{ x: 0.359375, y: 0.703125 },
{ x: 0.359375, y: 0.703125 },
{ x: 0.390625, y: 0.703125 },
{ x: 0.390625, y: 0.703125 },
{ x: 0.421875, y: 0.703125 },
{ x: 0.421875, y: 0.703125 },
{ x: 0.453125, y: 0.703125 },
{ x: 0.453125, y: 0.703125 },
{ x: 0.484375, y: 0.703125 },
{ x: 0.484375, y: 0.703125 },
{ x: 0.515625, y: 0.703125 },
{ x: 0.515625, y: 0.703125 },
{ x: 0.546875, y: 0.703125 },
{ x: 0.546875, y: 0.703125 },
{ x: 0.578125, y: 0.703125 },
{ x: 0.578125, y: 0.703125 },
{ x: 0.609375, y: 0.703125 },
{ x: 0.609375, y: 0.703125 },
{ x: 0.640625, y: 0.703125 },
{ x: 0.640625, y: 0.703125 },
{ x: 0.671875, y: 0.703125 },
{ x: 0.671875, y: 0.703125 },
{ x: 0.703125, y: 0.703125 },
{ x: 0.703125, y: 0.703125 },
{ x: 0.734375, y: 0.703125 },
{ x: 0.734375, y: 0.703125 },
{ x: 0.765625, y: 0.703125 },
{ x: 0.765625, y: 0.703125 },
{ x: 0.796875, y: 0.703125 },
{ x: 0.796875, y: 0.703125 },
{ x: 0.828125, y: 0.703125 },
{ x: 0.828125, y: 0.703125 },
{ x: 0.859375, y: 0.703125 },
{ x: 0.859375, y: 0.703125 },
{ x: 0.890625, y: 0.703125 },
{ x: 0.890625, y: 0.703125 },
{ x: 0.921875, y: 0.703125 },
{ x: 0.921875, y: 0.703125 },
{ x: 0.953125, y: 0.703125 },
{ x: 0.953125, y: 0.703125 },
{ x: 0.984375, y: 0.703125 },
{ x: 0.984375, y: 0.703125 },
{ x: 0.015625, y: 0.734375 },
{ x: 0.015625, y: 0.734375 },
{ x: 0.046875, y: 0.734375 },
{ x: 0.046875, y: 0.734375 },
{ x: 0.078125, y: 0.734375 },
{ x: 0.078125, y: 0.734375 },
{ x: 0.109375, y: 0.734375 },
{ x: 0.109375, y: 0.734375 },
{ x: 0.140625, y: 0.734375 },
{ x: 0.140625, y: 0.734375 },
{ x: 0.171875, y: 0.734375 },
{ x: 0.171875, y: 0.734375 },
{ x: 0.203125, y: 0.734375 },
{ x: 0.203125, y: 0.734375 },
{ x: 0.234375, y: 0.734375 },
{ x: 0.234375, y: 0.734375 },
{ x: 0.265625, y: 0.734375 },
{ x: 0.265625, y: 0.734375 },
{ x: 0.296875, y: 0.734375 },
{ x: 0.296875, y: 0.734375 },
{ x: 0.328125, y: 0.734375 },
{ x: 0.328125, y: 0.734375 },
{ x: 0.359375, y: 0.734375 },
{ x: 0.359375, y: 0.734375 },
{ x: 0.390625, y: 0.734375 },
{ x: 0.390625, y: 0.734375 },
{ x: 0.421875, y: 0.734375 },
{ x: 0.421875, y: 0.734375 },
{ x: 0.453125, y: 0.734375 },
{ x: 0.453125, y: 0.734375 },
{ x: 0.484375, y: 0.734375 },
{ x: 0.484375, y: 0.734375 },
{ x: 0.515625, y: 0.734375 },
{ x: 0.515625, y: 0.734375 },
{ x: 0.546875, y: 0.734375 },
{ x: 0.546875, y: 0.734375 },
{ x: 0.578125, y: 0.734375 },
{ x: 0.578125, y: 0.734375 },
{ x: 0.609375, y: 0.734375 },
{ x: 0.609375, y: 0.734375 },
{ x: 0.640625, y: 0.734375 },
{ x: 0.640625, y: 0.734375 },
{ x: 0.671875, y: 0.734375 },
{ x: 0.671875, y: 0.734375 },
{ x: 0.703125, y: 0.734375 },
{ x: 0.703125, y: 0.734375 },
{ x: 0.734375, y: 0.734375 },
{ x: 0.734375, y: 0.734375 },
{ x: 0.765625, y: 0.734375 },
{ x: 0.765625, y: 0.734375 },
{ x: 0.796875, y: 0.734375 },
{ x: 0.796875, y: 0.734375 },
{ x: 0.828125, y: 0.734375 },
{ x: 0.828125, y: 0.734375 },
{ x: 0.859375, y: 0.734375 },
{ x: 0.859375, y: 0.734375 },
{ x: 0.890625, y: 0.734375 },
{ x: 0.890625, y: 0.734375 },
{ x: 0.921875, y: 0.734375 },
{ x: 0.921875, y: 0.734375 },
{ x: 0.953125, y: 0.734375 },
{ x: 0.953125, y: 0.734375 },
{ x: 0.984375, y: 0.734375 },
{ x: 0.984375, y: 0.734375 },
{ x: 0.015625, y: 0.765625 },
{ x: 0.015625, y: 0.765625 },
{ x: 0.046875, y: 0.765625 },
{ x: 0.046875, y: 0.765625 },
{ x: 0.078125, y: 0.765625 },
{ x: 0.078125, y: 0.765625 },
{ x: 0.109375, y: 0.765625 },
{ x: 0.109375, y: 0.765625 },
{ x: 0.140625, y: 0.765625 },
{ x: 0.140625, y: 0.765625 },
{ x: 0.171875, y: 0.765625 },
{ x: 0.171875, y: 0.765625 },
{ x: 0.203125, y: 0.765625 },
{ x: 0.203125, y: 0.765625 },
{ x: 0.234375, y: 0.765625 },
{ x: 0.234375, y: 0.765625 },
{ x: 0.265625, y: 0.765625 },
{ x: 0.265625, y: 0.765625 },
{ x: 0.296875, y: 0.765625 },
{ x: 0.296875, y: 0.765625 },
{ x: 0.328125, y: 0.765625 },
{ x: 0.328125, y: 0.765625 },
{ x: 0.359375, y: 0.765625 },
{ x: 0.359375, y: 0.765625 },
{ x: 0.390625, y: 0.765625 },
{ x: 0.390625, y: 0.765625 },
{ x: 0.421875, y: 0.765625 },
{ x: 0.421875, y: 0.765625 },
{ x: 0.453125, y: 0.765625 },
{ x: 0.453125, y: 0.765625 },
{ x: 0.484375, y: 0.765625 },
{ x: 0.484375, y: 0.765625 },
{ x: 0.515625, y: 0.765625 },
{ x: 0.515625, y: 0.765625 },
{ x: 0.546875, y: 0.765625 },
{ x: 0.546875, y: 0.765625 },
{ x: 0.578125, y: 0.765625 },
{ x: 0.578125, y: 0.765625 },
{ x: 0.609375, y: 0.765625 },
{ x: 0.609375, y: 0.765625 },
{ x: 0.640625, y: 0.765625 },
{ x: 0.640625, y: 0.765625 },
{ x: 0.671875, y: 0.765625 },
{ x: 0.671875, y: 0.765625 },
{ x: 0.703125, y: 0.765625 },
{ x: 0.703125, y: 0.765625 },
{ x: 0.734375, y: 0.765625 },
{ x: 0.734375, y: 0.765625 },
{ x: 0.765625, y: 0.765625 },
{ x: 0.765625, y: 0.765625 },
{ x: 0.796875, y: 0.765625 },
{ x: 0.796875, y: 0.765625 },
{ x: 0.828125, y: 0.765625 },
{ x: 0.828125, y: 0.765625 },
{ x: 0.859375, y: 0.765625 },
{ x: 0.859375, y: 0.765625 },
{ x: 0.890625, y: 0.765625 },
{ x: 0.890625, y: 0.765625 },
{ x: 0.921875, y: 0.765625 },
{ x: 0.921875, y: 0.765625 },
{ x: 0.953125, y: 0.765625 },
{ x: 0.953125, y: 0.765625 },
{ x: 0.984375, y: 0.765625 },
{ x: 0.984375, y: 0.765625 },
{ x: 0.015625, y: 0.796875 },
{ x: 0.015625, y: 0.796875 },
{ x: 0.046875, y: 0.796875 },
{ x: 0.046875, y: 0.796875 },
{ x: 0.078125, y: 0.796875 },
{ x: 0.078125, y: 0.796875 },
{ x: 0.109375, y: 0.796875 },
{ x: 0.109375, y: 0.796875 },
{ x: 0.140625, y: 0.796875 },
{ x: 0.140625, y: 0.796875 },
{ x: 0.171875, y: 0.796875 },
{ x: 0.171875, y: 0.796875 },
{ x: 0.203125, y: 0.796875 },
{ x: 0.203125, y: 0.796875 },
{ x: 0.234375, y: 0.796875 },
{ x: 0.234375, y: 0.796875 },
{ x: 0.265625, y: 0.796875 },
{ x: 0.265625, y: 0.796875 },
{ x: 0.296875, y: 0.796875 },
{ x: 0.296875, y: 0.796875 },
{ x: 0.328125, y: 0.796875 },
{ x: 0.328125, y: 0.796875 },
{ x: 0.359375, y: 0.796875 },
{ x: 0.359375, y: 0.796875 },
{ x: 0.390625, y: 0.796875 },
{ x: 0.390625, y: 0.796875 },
{ x: 0.421875, y: 0.796875 },
{ x: 0.421875, y: 0.796875 },
{ x: 0.453125, y: 0.796875 },
{ x: 0.453125, y: 0.796875 },
{ x: 0.484375, y: 0.796875 },
{ x: 0.484375, y: 0.796875 },
{ x: 0.515625, y: 0.796875 },
{ x: 0.515625, y: 0.796875 },
{ x: 0.546875, y: 0.796875 },
{ x: 0.546875, y: 0.796875 },
{ x: 0.578125, y: 0.796875 },
{ x: 0.578125, y: 0.796875 },
{ x: 0.609375, y: 0.796875 },
{ x: 0.609375, y: 0.796875 },
{ x: 0.640625, y: 0.796875 },
{ x: 0.640625, y: 0.796875 },
{ x: 0.671875, y: 0.796875 },
{ x: 0.671875, y: 0.796875 },
{ x: 0.703125, y: 0.796875 },
{ x: 0.703125, y: 0.796875 },
{ x: 0.734375, y: 0.796875 },
{ x: 0.734375, y: 0.796875 },
{ x: 0.765625, y: 0.796875 },
{ x: 0.765625, y: 0.796875 },
{ x: 0.796875, y: 0.796875 },
{ x: 0.796875, y: 0.796875 },
{ x: 0.828125, y: 0.796875 },
{ x: 0.828125, y: 0.796875 },
{ x: 0.859375, y: 0.796875 },
{ x: 0.859375, y: 0.796875 },
{ x: 0.890625, y: 0.796875 },
{ x: 0.890625, y: 0.796875 },
{ x: 0.921875, y: 0.796875 },
{ x: 0.921875, y: 0.796875 },
{ x: 0.953125, y: 0.796875 },
{ x: 0.953125, y: 0.796875 },
{ x: 0.984375, y: 0.796875 },
{ x: 0.984375, y: 0.796875 },
{ x: 0.015625, y: 0.828125 },
{ x: 0.015625, y: 0.828125 },
{ x: 0.046875, y: 0.828125 },
{ x: 0.046875, y: 0.828125 },
{ x: 0.078125, y: 0.828125 },
{ x: 0.078125, y: 0.828125 },
{ x: 0.109375, y: 0.828125 },
{ x: 0.109375, y: 0.828125 },
{ x: 0.140625, y: 0.828125 },
{ x: 0.140625, y: 0.828125 },
{ x: 0.171875, y: 0.828125 },
{ x: 0.171875, y: 0.828125 },
{ x: 0.203125, y: 0.828125 },
{ x: 0.203125, y: 0.828125 },
{ x: 0.234375, y: 0.828125 },
{ x: 0.234375, y: 0.828125 },
{ x: 0.265625, y: 0.828125 },
{ x: 0.265625, y: 0.828125 },
{ x: 0.296875, y: 0.828125 },
{ x: 0.296875, y: 0.828125 },
{ x: 0.328125, y: 0.828125 },
{ x: 0.328125, y: 0.828125 },
{ x: 0.359375, y: 0.828125 },
{ x: 0.359375, y: 0.828125 },
{ x: 0.390625, y: 0.828125 },
{ x: 0.390625, y: 0.828125 },
{ x: 0.421875, y: 0.828125 },
{ x: 0.421875, y: 0.828125 },
{ x: 0.453125, y: 0.828125 },
{ x: 0.453125, y: 0.828125 },
{ x: 0.484375, y: 0.828125 },
{ x: 0.484375, y: 0.828125 },
{ x: 0.515625, y: 0.828125 },
{ x: 0.515625, y: 0.828125 },
{ x: 0.546875, y: 0.828125 },
{ x: 0.546875, y: 0.828125 },
{ x: 0.578125, y: 0.828125 },
{ x: 0.578125, y: 0.828125 },
{ x: 0.609375, y: 0.828125 },
{ x: 0.609375, y: 0.828125 },
{ x: 0.640625, y: 0.828125 },
{ x: 0.640625, y: 0.828125 },
{ x: 0.671875, y: 0.828125 },
{ x: 0.671875, y: 0.828125 },
{ x: 0.703125, y: 0.828125 },
{ x: 0.703125, y: 0.828125 },
{ x: 0.734375, y: 0.828125 },
{ x: 0.734375, y: 0.828125 },
{ x: 0.765625, y: 0.828125 },
{ x: 0.765625, y: 0.828125 },
{ x: 0.796875, y: 0.828125 },
{ x: 0.796875, y: 0.828125 },
{ x: 0.828125, y: 0.828125 },
{ x: 0.828125, y: 0.828125 },
{ x: 0.859375, y: 0.828125 },
{ x: 0.859375, y: 0.828125 },
{ x: 0.890625, y: 0.828125 },
{ x: 0.890625, y: 0.828125 },
{ x: 0.921875, y: 0.828125 },
{ x: 0.921875, y: 0.828125 },
{ x: 0.953125, y: 0.828125 },
{ x: 0.953125, y: 0.828125 },
{ x: 0.984375, y: 0.828125 },
{ x: 0.984375, y: 0.828125 },
{ x: 0.015625, y: 0.859375 },
{ x: 0.015625, y: 0.859375 },
{ x: 0.046875, y: 0.859375 },
{ x: 0.046875, y: 0.859375 },
{ x: 0.078125, y: 0.859375 },
{ x: 0.078125, y: 0.859375 },
{ x: 0.109375, y: 0.859375 },
{ x: 0.109375, y: 0.859375 },
{ x: 0.140625, y: 0.859375 },
{ x: 0.140625, y: 0.859375 },
{ x: 0.171875, y: 0.859375 },
{ x: 0.171875, y: 0.859375 },
{ x: 0.203125, y: 0.859375 },
{ x: 0.203125, y: 0.859375 },
{ x: 0.234375, y: 0.859375 },
{ x: 0.234375, y: 0.859375 },
{ x: 0.265625, y: 0.859375 },
{ x: 0.265625, y: 0.859375 },
{ x: 0.296875, y: 0.859375 },
{ x: 0.296875, y: 0.859375 },
{ x: 0.328125, y: 0.859375 },
{ x: 0.328125, y: 0.859375 },
{ x: 0.359375, y: 0.859375 },
{ x: 0.359375, y: 0.859375 },
{ x: 0.390625, y: 0.859375 },
{ x: 0.390625, y: 0.859375 },
{ x: 0.421875, y: 0.859375 },
{ x: 0.421875, y: 0.859375 },
{ x: 0.453125, y: 0.859375 },
{ x: 0.453125, y: 0.859375 },
{ x: 0.484375, y: 0.859375 },
{ x: 0.484375, y: 0.859375 },
{ x: 0.515625, y: 0.859375 },
{ x: 0.515625, y: 0.859375 },
{ x: 0.546875, y: 0.859375 },
{ x: 0.546875, y: 0.859375 },
{ x: 0.578125, y: 0.859375 },
{ x: 0.578125, y: 0.859375 },
{ x: 0.609375, y: 0.859375 },
{ x: 0.609375, y: 0.859375 },
{ x: 0.640625, y: 0.859375 },
{ x: 0.640625, y: 0.859375 },
{ x: 0.671875, y: 0.859375 },
{ x: 0.671875, y: 0.859375 },
{ x: 0.703125, y: 0.859375 },
{ x: 0.703125, y: 0.859375 },
{ x: 0.734375, y: 0.859375 },
{ x: 0.734375, y: 0.859375 },
{ x: 0.765625, y: 0.859375 },
{ x: 0.765625, y: 0.859375 },
{ x: 0.796875, y: 0.859375 },
{ x: 0.796875, y: 0.859375 },
{ x: 0.828125, y: 0.859375 },
{ x: 0.828125, y: 0.859375 },
{ x: 0.859375, y: 0.859375 },
{ x: 0.859375, y: 0.859375 },
{ x: 0.890625, y: 0.859375 },
{ x: 0.890625, y: 0.859375 },
{ x: 0.921875, y: 0.859375 },
{ x: 0.921875, y: 0.859375 },
{ x: 0.953125, y: 0.859375 },
{ x: 0.953125, y: 0.859375 },
{ x: 0.984375, y: 0.859375 },
{ x: 0.984375, y: 0.859375 },
{ x: 0.015625, y: 0.890625 },
{ x: 0.015625, y: 0.890625 },
{ x: 0.046875, y: 0.890625 },
{ x: 0.046875, y: 0.890625 },
{ x: 0.078125, y: 0.890625 },
{ x: 0.078125, y: 0.890625 },
{ x: 0.109375, y: 0.890625 },
{ x: 0.109375, y: 0.890625 },
{ x: 0.140625, y: 0.890625 },
{ x: 0.140625, y: 0.890625 },
{ x: 0.171875, y: 0.890625 },
{ x: 0.171875, y: 0.890625 },
{ x: 0.203125, y: 0.890625 },
{ x: 0.203125, y: 0.890625 },
{ x: 0.234375, y: 0.890625 },
{ x: 0.234375, y: 0.890625 },
{ x: 0.265625, y: 0.890625 },
{ x: 0.265625, y: 0.890625 },
{ x: 0.296875, y: 0.890625 },
{ x: 0.296875, y: 0.890625 },
{ x: 0.328125, y: 0.890625 },
{ x: 0.328125, y: 0.890625 },
{ x: 0.359375, y: 0.890625 },
{ x: 0.359375, y: 0.890625 },
{ x: 0.390625, y: 0.890625 },
{ x: 0.390625, y: 0.890625 },
{ x: 0.421875, y: 0.890625 },
{ x: 0.421875, y: 0.890625 },
{ x: 0.453125, y: 0.890625 },
{ x: 0.453125, y: 0.890625 },
{ x: 0.484375, y: 0.890625 },
{ x: 0.484375, y: 0.890625 },
{ x: 0.515625, y: 0.890625 },
{ x: 0.515625, y: 0.890625 },
{ x: 0.546875, y: 0.890625 },
{ x: 0.546875, y: 0.890625 },
{ x: 0.578125, y: 0.890625 },
{ x: 0.578125, y: 0.890625 },
{ x: 0.609375, y: 0.890625 },
{ x: 0.609375, y: 0.890625 },
{ x: 0.640625, y: 0.890625 },
{ x: 0.640625, y: 0.890625 },
{ x: 0.671875, y: 0.890625 },
{ x: 0.671875, y: 0.890625 },
{ x: 0.703125, y: 0.890625 },
{ x: 0.703125, y: 0.890625 },
{ x: 0.734375, y: 0.890625 },
{ x: 0.734375, y: 0.890625 },
{ x: 0.765625, y: 0.890625 },
{ x: 0.765625, y: 0.890625 },
{ x: 0.796875, y: 0.890625 },
{ x: 0.796875, y: 0.890625 },
{ x: 0.828125, y: 0.890625 },
{ x: 0.828125, y: 0.890625 },
{ x: 0.859375, y: 0.890625 },
{ x: 0.859375, y: 0.890625 },
{ x: 0.890625, y: 0.890625 },
{ x: 0.890625, y: 0.890625 },
{ x: 0.921875, y: 0.890625 },
{ x: 0.921875, y: 0.890625 },
{ x: 0.953125, y: 0.890625 },
{ x: 0.953125, y: 0.890625 },
{ x: 0.984375, y: 0.890625 },
{ x: 0.984375, y: 0.890625 },
{ x: 0.015625, y: 0.921875 },
{ x: 0.015625, y: 0.921875 },
{ x: 0.046875, y: 0.921875 },
{ x: 0.046875, y: 0.921875 },
{ x: 0.078125, y: 0.921875 },
{ x: 0.078125, y: 0.921875 },
{ x: 0.109375, y: 0.921875 },
{ x: 0.109375, y: 0.921875 },
{ x: 0.140625, y: 0.921875 },
{ x: 0.140625, y: 0.921875 },
{ x: 0.171875, y: 0.921875 },
{ x: 0.171875, y: 0.921875 },
{ x: 0.203125, y: 0.921875 },
{ x: 0.203125, y: 0.921875 },
{ x: 0.234375, y: 0.921875 },
{ x: 0.234375, y: 0.921875 },
{ x: 0.265625, y: 0.921875 },
{ x: 0.265625, y: 0.921875 },
{ x: 0.296875, y: 0.921875 },
{ x: 0.296875, y: 0.921875 },
{ x: 0.328125, y: 0.921875 },
{ x: 0.328125, y: 0.921875 },
{ x: 0.359375, y: 0.921875 },
{ x: 0.359375, y: 0.921875 },
{ x: 0.390625, y: 0.921875 },
{ x: 0.390625, y: 0.921875 },
{ x: 0.421875, y: 0.921875 },
{ x: 0.421875, y: 0.921875 },
{ x: 0.453125, y: 0.921875 },
{ x: 0.453125, y: 0.921875 },
{ x: 0.484375, y: 0.921875 },
{ x: 0.484375, y: 0.921875 },
{ x: 0.515625, y: 0.921875 },
{ x: 0.515625, y: 0.921875 },
{ x: 0.546875, y: 0.921875 },
{ x: 0.546875, y: 0.921875 },
{ x: 0.578125, y: 0.921875 },
{ x: 0.578125, y: 0.921875 },
{ x: 0.609375, y: 0.921875 },
{ x: 0.609375, y: 0.921875 },
{ x: 0.640625, y: 0.921875 },
{ x: 0.640625, y: 0.921875 },
{ x: 0.671875, y: 0.921875 },
{ x: 0.671875, y: 0.921875 },
{ x: 0.703125, y: 0.921875 },
{ x: 0.703125, y: 0.921875 },
{ x: 0.734375, y: 0.921875 },
{ x: 0.734375, y: 0.921875 },
{ x: 0.765625, y: 0.921875 },
{ x: 0.765625, y: 0.921875 },
{ x: 0.796875, y: 0.921875 },
{ x: 0.796875, y: 0.921875 },
{ x: 0.828125, y: 0.921875 },
{ x: 0.828125, y: 0.921875 },
{ x: 0.859375, y: 0.921875 },
{ x: 0.859375, y: 0.921875 },
{ x: 0.890625, y: 0.921875 },
{ x: 0.890625, y: 0.921875 },
{ x: 0.921875, y: 0.921875 },
{ x: 0.921875, y: 0.921875 },
{ x: 0.953125, y: 0.921875 },
{ x: 0.953125, y: 0.921875 },
{ x: 0.984375, y: 0.921875 },
{ x: 0.984375, y: 0.921875 },
{ x: 0.015625, y: 0.953125 },
{ x: 0.015625, y: 0.953125 },
{ x: 0.046875, y: 0.953125 },
{ x: 0.046875, y: 0.953125 },
{ x: 0.078125, y: 0.953125 },
{ x: 0.078125, y: 0.953125 },
{ x: 0.109375, y: 0.953125 },
{ x: 0.109375, y: 0.953125 },
{ x: 0.140625, y: 0.953125 },
{ x: 0.140625, y: 0.953125 },
{ x: 0.171875, y: 0.953125 },
{ x: 0.171875, y: 0.953125 },
{ x: 0.203125, y: 0.953125 },
{ x: 0.203125, y: 0.953125 },
{ x: 0.234375, y: 0.953125 },
{ x: 0.234375, y: 0.953125 },
{ x: 0.265625, y: 0.953125 },
{ x: 0.265625, y: 0.953125 },
{ x: 0.296875, y: 0.953125 },
{ x: 0.296875, y: 0.953125 },
{ x: 0.328125, y: 0.953125 },
{ x: 0.328125, y: 0.953125 },
{ x: 0.359375, y: 0.953125 },
{ x: 0.359375, y: 0.953125 },
{ x: 0.390625, y: 0.953125 },
{ x: 0.390625, y: 0.953125 },
{ x: 0.421875, y: 0.953125 },
{ x: 0.421875, y: 0.953125 },
{ x: 0.453125, y: 0.953125 },
{ x: 0.453125, y: 0.953125 },
{ x: 0.484375, y: 0.953125 },
{ x: 0.484375, y: 0.953125 },
{ x: 0.515625, y: 0.953125 },
{ x: 0.515625, y: 0.953125 },
{ x: 0.546875, y: 0.953125 },
{ x: 0.546875, y: 0.953125 },
{ x: 0.578125, y: 0.953125 },
{ x: 0.578125, y: 0.953125 },
{ x: 0.609375, y: 0.953125 },
{ x: 0.609375, y: 0.953125 },
{ x: 0.640625, y: 0.953125 },
{ x: 0.640625, y: 0.953125 },
{ x: 0.671875, y: 0.953125 },
{ x: 0.671875, y: 0.953125 },
{ x: 0.703125, y: 0.953125 },
{ x: 0.703125, y: 0.953125 },
{ x: 0.734375, y: 0.953125 },
{ x: 0.734375, y: 0.953125 },
{ x: 0.765625, y: 0.953125 },
{ x: 0.765625, y: 0.953125 },
{ x: 0.796875, y: 0.953125 },
{ x: 0.796875, y: 0.953125 },
{ x: 0.828125, y: 0.953125 },
{ x: 0.828125, y: 0.953125 },
{ x: 0.859375, y: 0.953125 },
{ x: 0.859375, y: 0.953125 },
{ x: 0.890625, y: 0.953125 },
{ x: 0.890625, y: 0.953125 },
{ x: 0.921875, y: 0.953125 },
{ x: 0.921875, y: 0.953125 },
{ x: 0.953125, y: 0.953125 },
{ x: 0.953125, y: 0.953125 },
{ x: 0.984375, y: 0.953125 },
{ x: 0.984375, y: 0.953125 },
{ x: 0.015625, y: 0.984375 },
{ x: 0.015625, y: 0.984375 },
{ x: 0.046875, y: 0.984375 },
{ x: 0.046875, y: 0.984375 },
{ x: 0.078125, y: 0.984375 },
{ x: 0.078125, y: 0.984375 },
{ x: 0.109375, y: 0.984375 },
{ x: 0.109375, y: 0.984375 },
{ x: 0.140625, y: 0.984375 },
{ x: 0.140625, y: 0.984375 },
{ x: 0.171875, y: 0.984375 },
{ x: 0.171875, y: 0.984375 },
{ x: 0.203125, y: 0.984375 },
{ x: 0.203125, y: 0.984375 },
{ x: 0.234375, y: 0.984375 },
{ x: 0.234375, y: 0.984375 },
{ x: 0.265625, y: 0.984375 },
{ x: 0.265625, y: 0.984375 },
{ x: 0.296875, y: 0.984375 },
{ x: 0.296875, y: 0.984375 },
{ x: 0.328125, y: 0.984375 },
{ x: 0.328125, y: 0.984375 },
{ x: 0.359375, y: 0.984375 },
{ x: 0.359375, y: 0.984375 },
{ x: 0.390625, y: 0.984375 },
{ x: 0.390625, y: 0.984375 },
{ x: 0.421875, y: 0.984375 },
{ x: 0.421875, y: 0.984375 },
{ x: 0.453125, y: 0.984375 },
{ x: 0.453125, y: 0.984375 },
{ x: 0.484375, y: 0.984375 },
{ x: 0.484375, y: 0.984375 },
{ x: 0.515625, y: 0.984375 },
{ x: 0.515625, y: 0.984375 },
{ x: 0.546875, y: 0.984375 },
{ x: 0.546875, y: 0.984375 },
{ x: 0.578125, y: 0.984375 },
{ x: 0.578125, y: 0.984375 },
{ x: 0.609375, y: 0.984375 },
{ x: 0.609375, y: 0.984375 },
{ x: 0.640625, y: 0.984375 },
{ x: 0.640625, y: 0.984375 },
{ x: 0.671875, y: 0.984375 },
{ x: 0.671875, y: 0.984375 },
{ x: 0.703125, y: 0.984375 },
{ x: 0.703125, y: 0.984375 },
{ x: 0.734375, y: 0.984375 },
{ x: 0.734375, y: 0.984375 },
{ x: 0.765625, y: 0.984375 },
{ x: 0.765625, y: 0.984375 },
{ x: 0.796875, y: 0.984375 },
{ x: 0.796875, y: 0.984375 },
{ x: 0.828125, y: 0.984375 },
{ x: 0.828125, y: 0.984375 },
{ x: 0.859375, y: 0.984375 },
{ x: 0.859375, y: 0.984375 },
{ x: 0.890625, y: 0.984375 },
{ x: 0.890625, y: 0.984375 },
{ x: 0.921875, y: 0.984375 },
{ x: 0.921875, y: 0.984375 },
{ x: 0.953125, y: 0.984375 },
{ x: 0.953125, y: 0.984375 },
{ x: 0.984375, y: 0.984375 },
{ x: 0.984375, y: 0.984375 },
{ x: 0.03125, y: 0.03125 },
{ x: 0.03125, y: 0.03125 },
{ x: 0.09375, y: 0.03125 },
{ x: 0.09375, y: 0.03125 },
{ x: 0.15625, y: 0.03125 },
{ x: 0.15625, y: 0.03125 },
{ x: 0.21875, y: 0.03125 },
{ x: 0.21875, y: 0.03125 },
{ x: 0.28125, y: 0.03125 },
{ x: 0.28125, y: 0.03125 },
{ x: 0.34375, y: 0.03125 },
{ x: 0.34375, y: 0.03125 },
{ x: 0.40625, y: 0.03125 },
{ x: 0.40625, y: 0.03125 },
{ x: 0.46875, y: 0.03125 },
{ x: 0.46875, y: 0.03125 },
{ x: 0.53125, y: 0.03125 },
{ x: 0.53125, y: 0.03125 },
{ x: 0.59375, y: 0.03125 },
{ x: 0.59375, y: 0.03125 },
{ x: 0.65625, y: 0.03125 },
{ x: 0.65625, y: 0.03125 },
{ x: 0.71875, y: 0.03125 },
{ x: 0.71875, y: 0.03125 },
{ x: 0.78125, y: 0.03125 },
{ x: 0.78125, y: 0.03125 },
{ x: 0.84375, y: 0.03125 },
{ x: 0.84375, y: 0.03125 },
{ x: 0.90625, y: 0.03125 },
{ x: 0.90625, y: 0.03125 },
{ x: 0.96875, y: 0.03125 },
{ x: 0.96875, y: 0.03125 },
{ x: 0.03125, y: 0.09375 },
{ x: 0.03125, y: 0.09375 },
{ x: 0.09375, y: 0.09375 },
{ x: 0.09375, y: 0.09375 },
{ x: 0.15625, y: 0.09375 },
{ x: 0.15625, y: 0.09375 },
{ x: 0.21875, y: 0.09375 },
{ x: 0.21875, y: 0.09375 },
{ x: 0.28125, y: 0.09375 },
{ x: 0.28125, y: 0.09375 },
{ x: 0.34375, y: 0.09375 },
{ x: 0.34375, y: 0.09375 },
{ x: 0.40625, y: 0.09375 },
{ x: 0.40625, y: 0.09375 },
{ x: 0.46875, y: 0.09375 },
{ x: 0.46875, y: 0.09375 },
{ x: 0.53125, y: 0.09375 },
{ x: 0.53125, y: 0.09375 },
{ x: 0.59375, y: 0.09375 },
{ x: 0.59375, y: 0.09375 },
{ x: 0.65625, y: 0.09375 },
{ x: 0.65625, y: 0.09375 },
{ x: 0.71875, y: 0.09375 },
{ x: 0.71875, y: 0.09375 },
{ x: 0.78125, y: 0.09375 },
{ x: 0.78125, y: 0.09375 },
{ x: 0.84375, y: 0.09375 },
{ x: 0.84375, y: 0.09375 },
{ x: 0.90625, y: 0.09375 },
{ x: 0.90625, y: 0.09375 },
{ x: 0.96875, y: 0.09375 },
{ x: 0.96875, y: 0.09375 },
{ x: 0.03125, y: 0.15625 },
{ x: 0.03125, y: 0.15625 },
{ x: 0.09375, y: 0.15625 },
{ x: 0.09375, y: 0.15625 },
{ x: 0.15625, y: 0.15625 },
{ x: 0.15625, y: 0.15625 },
{ x: 0.21875, y: 0.15625 },
{ x: 0.21875, y: 0.15625 },
{ x: 0.28125, y: 0.15625 },
{ x: 0.28125, y: 0.15625 },
{ x: 0.34375, y: 0.15625 },
{ x: 0.34375, y: 0.15625 },
{ x: 0.40625, y: 0.15625 },
{ x: 0.40625, y: 0.15625 },
{ x: 0.46875, y: 0.15625 },
{ x: 0.46875, y: 0.15625 },
{ x: 0.53125, y: 0.15625 },
{ x: 0.53125, y: 0.15625 },
{ x: 0.59375, y: 0.15625 },
{ x: 0.59375, y: 0.15625 },
{ x: 0.65625, y: 0.15625 },
{ x: 0.65625, y: 0.15625 },
{ x: 0.71875, y: 0.15625 },
{ x: 0.71875, y: 0.15625 },
{ x: 0.78125, y: 0.15625 },
{ x: 0.78125, y: 0.15625 },
{ x: 0.84375, y: 0.15625 },
{ x: 0.84375, y: 0.15625 },
{ x: 0.90625, y: 0.15625 },
{ x: 0.90625, y: 0.15625 },
{ x: 0.96875, y: 0.15625 },
{ x: 0.96875, y: 0.15625 },
{ x: 0.03125, y: 0.21875 },
{ x: 0.03125, y: 0.21875 },
{ x: 0.09375, y: 0.21875 },
{ x: 0.09375, y: 0.21875 },
{ x: 0.15625, y: 0.21875 },
{ x: 0.15625, y: 0.21875 },
{ x: 0.21875, y: 0.21875 },
{ x: 0.21875, y: 0.21875 },
{ x: 0.28125, y: 0.21875 },
{ x: 0.28125, y: 0.21875 },
{ x: 0.34375, y: 0.21875 },
{ x: 0.34375, y: 0.21875 },
{ x: 0.40625, y: 0.21875 },
{ x: 0.40625, y: 0.21875 },
{ x: 0.46875, y: 0.21875 },
{ x: 0.46875, y: 0.21875 },
{ x: 0.53125, y: 0.21875 },
{ x: 0.53125, y: 0.21875 },
{ x: 0.59375, y: 0.21875 },
{ x: 0.59375, y: 0.21875 },
{ x: 0.65625, y: 0.21875 },
{ x: 0.65625, y: 0.21875 },
{ x: 0.71875, y: 0.21875 },
{ x: 0.71875, y: 0.21875 },
{ x: 0.78125, y: 0.21875 },
{ x: 0.78125, y: 0.21875 },
{ x: 0.84375, y: 0.21875 },
{ x: 0.84375, y: 0.21875 },
{ x: 0.90625, y: 0.21875 },
{ x: 0.90625, y: 0.21875 },
{ x: 0.96875, y: 0.21875 },
{ x: 0.96875, y: 0.21875 },
{ x: 0.03125, y: 0.28125 },
{ x: 0.03125, y: 0.28125 },
{ x: 0.09375, y: 0.28125 },
{ x: 0.09375, y: 0.28125 },
{ x: 0.15625, y: 0.28125 },
{ x: 0.15625, y: 0.28125 },
{ x: 0.21875, y: 0.28125 },
{ x: 0.21875, y: 0.28125 },
{ x: 0.28125, y: 0.28125 },
{ x: 0.28125, y: 0.28125 },
{ x: 0.34375, y: 0.28125 },
{ x: 0.34375, y: 0.28125 },
{ x: 0.40625, y: 0.28125 },
{ x: 0.40625, y: 0.28125 },
{ x: 0.46875, y: 0.28125 },
{ x: 0.46875, y: 0.28125 },
{ x: 0.53125, y: 0.28125 },
{ x: 0.53125, y: 0.28125 },
{ x: 0.59375, y: 0.28125 },
{ x: 0.59375, y: 0.28125 },
{ x: 0.65625, y: 0.28125 },
{ x: 0.65625, y: 0.28125 },
{ x: 0.71875, y: 0.28125 },
{ x: 0.71875, y: 0.28125 },
{ x: 0.78125, y: 0.28125 },
{ x: 0.78125, y: 0.28125 },
{ x: 0.84375, y: 0.28125 },
{ x: 0.84375, y: 0.28125 },
{ x: 0.90625, y: 0.28125 },
{ x: 0.90625, y: 0.28125 },
{ x: 0.96875, y: 0.28125 },
{ x: 0.96875, y: 0.28125 },
{ x: 0.03125, y: 0.34375 },
{ x: 0.03125, y: 0.34375 },
{ x: 0.09375, y: 0.34375 },
{ x: 0.09375, y: 0.34375 },
{ x: 0.15625, y: 0.34375 },
{ x: 0.15625, y: 0.34375 },
{ x: 0.21875, y: 0.34375 },
{ x: 0.21875, y: 0.34375 },
{ x: 0.28125, y: 0.34375 },
{ x: 0.28125, y: 0.34375 },
{ x: 0.34375, y: 0.34375 },
{ x: 0.34375, y: 0.34375 },
{ x: 0.40625, y: 0.34375 },
{ x: 0.40625, y: 0.34375 },
{ x: 0.46875, y: 0.34375 },
{ x: 0.46875, y: 0.34375 },
{ x: 0.53125, y: 0.34375 },
{ x: 0.53125, y: 0.34375 },
{ x: 0.59375, y: 0.34375 },
{ x: 0.59375, y: 0.34375 },
{ x: 0.65625, y: 0.34375 },
{ x: 0.65625, y: 0.34375 },
{ x: 0.71875, y: 0.34375 },
{ x: 0.71875, y: 0.34375 },
{ x: 0.78125, y: 0.34375 },
{ x: 0.78125, y: 0.34375 },
{ x: 0.84375, y: 0.34375 },
{ x: 0.84375, y: 0.34375 },
{ x: 0.90625, y: 0.34375 },
{ x: 0.90625, y: 0.34375 },
{ x: 0.96875, y: 0.34375 },
{ x: 0.96875, y: 0.34375 },
{ x: 0.03125, y: 0.40625 },
{ x: 0.03125, y: 0.40625 },
{ x: 0.09375, y: 0.40625 },
{ x: 0.09375, y: 0.40625 },
{ x: 0.15625, y: 0.40625 },
{ x: 0.15625, y: 0.40625 },
{ x: 0.21875, y: 0.40625 },
{ x: 0.21875, y: 0.40625 },
{ x: 0.28125, y: 0.40625 },
{ x: 0.28125, y: 0.40625 },
{ x: 0.34375, y: 0.40625 },
{ x: 0.34375, y: 0.40625 },
{ x: 0.40625, y: 0.40625 },
{ x: 0.40625, y: 0.40625 },
{ x: 0.46875, y: 0.40625 },
{ x: 0.46875, y: 0.40625 },
{ x: 0.53125, y: 0.40625 },
{ x: 0.53125, y: 0.40625 },
{ x: 0.59375, y: 0.40625 },
{ x: 0.59375, y: 0.40625 },
{ x: 0.65625, y: 0.40625 },
{ x: 0.65625, y: 0.40625 },
{ x: 0.71875, y: 0.40625 },
{ x: 0.71875, y: 0.40625 },
{ x: 0.78125, y: 0.40625 },
{ x: 0.78125, y: 0.40625 },
{ x: 0.84375, y: 0.40625 },
{ x: 0.84375, y: 0.40625 },
{ x: 0.90625, y: 0.40625 },
{ x: 0.90625, y: 0.40625 },
{ x: 0.96875, y: 0.40625 },
{ x: 0.96875, y: 0.40625 },
{ x: 0.03125, y: 0.46875 },
{ x: 0.03125, y: 0.46875 },
{ x: 0.09375, y: 0.46875 },
{ x: 0.09375, y: 0.46875 },
{ x: 0.15625, y: 0.46875 },
{ x: 0.15625, y: 0.46875 },
{ x: 0.21875, y: 0.46875 },
{ x: 0.21875, y: 0.46875 },
{ x: 0.28125, y: 0.46875 },
{ x: 0.28125, y: 0.46875 },
{ x: 0.34375, y: 0.46875 },
{ x: 0.34375, y: 0.46875 },
{ x: 0.40625, y: 0.46875 },
{ x: 0.40625, y: 0.46875 },
{ x: 0.46875, y: 0.46875 },
{ x: 0.46875, y: 0.46875 },
{ x: 0.53125, y: 0.46875 },
{ x: 0.53125, y: 0.46875 },
{ x: 0.59375, y: 0.46875 },
{ x: 0.59375, y: 0.46875 },
{ x: 0.65625, y: 0.46875 },
{ x: 0.65625, y: 0.46875 },
{ x: 0.71875, y: 0.46875 },
{ x: 0.71875, y: 0.46875 },
{ x: 0.78125, y: 0.46875 },
{ x: 0.78125, y: 0.46875 },
{ x: 0.84375, y: 0.46875 },
{ x: 0.84375, y: 0.46875 },
{ x: 0.90625, y: 0.46875 },
{ x: 0.90625, y: 0.46875 },
{ x: 0.96875, y: 0.46875 },
{ x: 0.96875, y: 0.46875 },
{ x: 0.03125, y: 0.53125 },
{ x: 0.03125, y: 0.53125 },
{ x: 0.09375, y: 0.53125 },
{ x: 0.09375, y: 0.53125 },
{ x: 0.15625, y: 0.53125 },
{ x: 0.15625, y: 0.53125 },
{ x: 0.21875, y: 0.53125 },
{ x: 0.21875, y: 0.53125 },
{ x: 0.28125, y: 0.53125 },
{ x: 0.28125, y: 0.53125 },
{ x: 0.34375, y: 0.53125 },
{ x: 0.34375, y: 0.53125 },
{ x: 0.40625, y: 0.53125 },
{ x: 0.40625, y: 0.53125 },
{ x: 0.46875, y: 0.53125 },
{ x: 0.46875, y: 0.53125 },
{ x: 0.53125, y: 0.53125 },
{ x: 0.53125, y: 0.53125 },
{ x: 0.59375, y: 0.53125 },
{ x: 0.59375, y: 0.53125 },
{ x: 0.65625, y: 0.53125 },
{ x: 0.65625, y: 0.53125 },
{ x: 0.71875, y: 0.53125 },
{ x: 0.71875, y: 0.53125 },
{ x: 0.78125, y: 0.53125 },
{ x: 0.78125, y: 0.53125 },
{ x: 0.84375, y: 0.53125 },
{ x: 0.84375, y: 0.53125 },
{ x: 0.90625, y: 0.53125 },
{ x: 0.90625, y: 0.53125 },
{ x: 0.96875, y: 0.53125 },
{ x: 0.96875, y: 0.53125 },
{ x: 0.03125, y: 0.59375 },
{ x: 0.03125, y: 0.59375 },
{ x: 0.09375, y: 0.59375 },
{ x: 0.09375, y: 0.59375 },
{ x: 0.15625, y: 0.59375 },
{ x: 0.15625, y: 0.59375 },
{ x: 0.21875, y: 0.59375 },
{ x: 0.21875, y: 0.59375 },
{ x: 0.28125, y: 0.59375 },
{ x: 0.28125, y: 0.59375 },
{ x: 0.34375, y: 0.59375 },
{ x: 0.34375, y: 0.59375 },
{ x: 0.40625, y: 0.59375 },
{ x: 0.40625, y: 0.59375 },
{ x: 0.46875, y: 0.59375 },
{ x: 0.46875, y: 0.59375 },
{ x: 0.53125, y: 0.59375 },
{ x: 0.53125, y: 0.59375 },
{ x: 0.59375, y: 0.59375 },
{ x: 0.59375, y: 0.59375 },
{ x: 0.65625, y: 0.59375 },
{ x: 0.65625, y: 0.59375 },
{ x: 0.71875, y: 0.59375 },
{ x: 0.71875, y: 0.59375 },
{ x: 0.78125, y: 0.59375 },
{ x: 0.78125, y: 0.59375 },
{ x: 0.84375, y: 0.59375 },
{ x: 0.84375, y: 0.59375 },
{ x: 0.90625, y: 0.59375 },
{ x: 0.90625, y: 0.59375 },
{ x: 0.96875, y: 0.59375 },
{ x: 0.96875, y: 0.59375 },
{ x: 0.03125, y: 0.65625 },
{ x: 0.03125, y: 0.65625 },
{ x: 0.09375, y: 0.65625 },
{ x: 0.09375, y: 0.65625 },
{ x: 0.15625, y: 0.65625 },
{ x: 0.15625, y: 0.65625 },
{ x: 0.21875, y: 0.65625 },
{ x: 0.21875, y: 0.65625 },
{ x: 0.28125, y: 0.65625 },
{ x: 0.28125, y: 0.65625 },
{ x: 0.34375, y: 0.65625 },
{ x: 0.34375, y: 0.65625 },
{ x: 0.40625, y: 0.65625 },
{ x: 0.40625, y: 0.65625 },
{ x: 0.46875, y: 0.65625 },
{ x: 0.46875, y: 0.65625 },
{ x: 0.53125, y: 0.65625 },
{ x: 0.53125, y: 0.65625 },
{ x: 0.59375, y: 0.65625 },
{ x: 0.59375, y: 0.65625 },
{ x: 0.65625, y: 0.65625 },
{ x: 0.65625, y: 0.65625 },
{ x: 0.71875, y: 0.65625 },
{ x: 0.71875, y: 0.65625 },
{ x: 0.78125, y: 0.65625 },
{ x: 0.78125, y: 0.65625 },
{ x: 0.84375, y: 0.65625 },
{ x: 0.84375, y: 0.65625 },
{ x: 0.90625, y: 0.65625 },
{ x: 0.90625, y: 0.65625 },
{ x: 0.96875, y: 0.65625 },
{ x: 0.96875, y: 0.65625 },
{ x: 0.03125, y: 0.71875 },
{ x: 0.03125, y: 0.71875 },
{ x: 0.09375, y: 0.71875 },
{ x: 0.09375, y: 0.71875 },
{ x: 0.15625, y: 0.71875 },
{ x: 0.15625, y: 0.71875 },
{ x: 0.21875, y: 0.71875 },
{ x: 0.21875, y: 0.71875 },
{ x: 0.28125, y: 0.71875 },
{ x: 0.28125, y: 0.71875 },
{ x: 0.34375, y: 0.71875 },
{ x: 0.34375, y: 0.71875 },
{ x: 0.40625, y: 0.71875 },
{ x: 0.40625, y: 0.71875 },
{ x: 0.46875, y: 0.71875 },
{ x: 0.46875, y: 0.71875 },
{ x: 0.53125, y: 0.71875 },
{ x: 0.53125, y: 0.71875 },
{ x: 0.59375, y: 0.71875 },
{ x: 0.59375, y: 0.71875 },
{ x: 0.65625, y: 0.71875 },
{ x: 0.65625, y: 0.71875 },
{ x: 0.71875, y: 0.71875 },
{ x: 0.71875, y: 0.71875 },
{ x: 0.78125, y: 0.71875 },
{ x: 0.78125, y: 0.71875 },
{ x: 0.84375, y: 0.71875 },
{ x: 0.84375, y: 0.71875 },
{ x: 0.90625, y: 0.71875 },
{ x: 0.90625, y: 0.71875 },
{ x: 0.96875, y: 0.71875 },
{ x: 0.96875, y: 0.71875 },
{ x: 0.03125, y: 0.78125 },
{ x: 0.03125, y: 0.78125 },
{ x: 0.09375, y: 0.78125 },
{ x: 0.09375, y: 0.78125 },
{ x: 0.15625, y: 0.78125 },
{ x: 0.15625, y: 0.78125 },
{ x: 0.21875, y: 0.78125 },
{ x: 0.21875, y: 0.78125 },
{ x: 0.28125, y: 0.78125 },
{ x: 0.28125, y: 0.78125 },
{ x: 0.34375, y: 0.78125 },
{ x: 0.34375, y: 0.78125 },
{ x: 0.40625, y: 0.78125 },
{ x: 0.40625, y: 0.78125 },
{ x: 0.46875, y: 0.78125 },
{ x: 0.46875, y: 0.78125 },
{ x: 0.53125, y: 0.78125 },
{ x: 0.53125, y: 0.78125 },
{ x: 0.59375, y: 0.78125 },
{ x: 0.59375, y: 0.78125 },
{ x: 0.65625, y: 0.78125 },
{ x: 0.65625, y: 0.78125 },
{ x: 0.71875, y: 0.78125 },
{ x: 0.71875, y: 0.78125 },
{ x: 0.78125, y: 0.78125 },
{ x: 0.78125, y: 0.78125 },
{ x: 0.84375, y: 0.78125 },
{ x: 0.84375, y: 0.78125 },
{ x: 0.90625, y: 0.78125 },
{ x: 0.90625, y: 0.78125 },
{ x: 0.96875, y: 0.78125 },
{ x: 0.96875, y: 0.78125 },
{ x: 0.03125, y: 0.84375 },
{ x: 0.03125, y: 0.84375 },
{ x: 0.09375, y: 0.84375 },
{ x: 0.09375, y: 0.84375 },
{ x: 0.15625, y: 0.84375 },
{ x: 0.15625, y: 0.84375 },
{ x: 0.21875, y: 0.84375 },
{ x: 0.21875, y: 0.84375 },
{ x: 0.28125, y: 0.84375 },
{ x: 0.28125, y: 0.84375 },
{ x: 0.34375, y: 0.84375 },
{ x: 0.34375, y: 0.84375 },
{ x: 0.40625, y: 0.84375 },
{ x: 0.40625, y: 0.84375 },
{ x: 0.46875, y: 0.84375 },
{ x: 0.46875, y: 0.84375 },
{ x: 0.53125, y: 0.84375 },
{ x: 0.53125, y: 0.84375 },
{ x: 0.59375, y: 0.84375 },
{ x: 0.59375, y: 0.84375 },
{ x: 0.65625, y: 0.84375 },
{ x: 0.65625, y: 0.84375 },
{ x: 0.71875, y: 0.84375 },
{ x: 0.71875, y: 0.84375 },
{ x: 0.78125, y: 0.84375 },
{ x: 0.78125, y: 0.84375 },
{ x: 0.84375, y: 0.84375 },
{ x: 0.84375, y: 0.84375 },
{ x: 0.90625, y: 0.84375 },
{ x: 0.90625, y: 0.84375 },
{ x: 0.96875, y: 0.84375 },
{ x: 0.96875, y: 0.84375 },
{ x: 0.03125, y: 0.90625 },
{ x: 0.03125, y: 0.90625 },
{ x: 0.09375, y: 0.90625 },
{ x: 0.09375, y: 0.90625 },
{ x: 0.15625, y: 0.90625 },
{ x: 0.15625, y: 0.90625 },
{ x: 0.21875, y: 0.90625 },
{ x: 0.21875, y: 0.90625 },
{ x: 0.28125, y: 0.90625 },
{ x: 0.28125, y: 0.90625 },
{ x: 0.34375, y: 0.90625 },
{ x: 0.34375, y: 0.90625 },
{ x: 0.40625, y: 0.90625 },
{ x: 0.40625, y: 0.90625 },
{ x: 0.46875, y: 0.90625 },
{ x: 0.46875, y: 0.90625 },
{ x: 0.53125, y: 0.90625 },
{ x: 0.53125, y: 0.90625 },
{ x: 0.59375, y: 0.90625 },
{ x: 0.59375, y: 0.90625 },
{ x: 0.65625, y: 0.90625 },
{ x: 0.65625, y: 0.90625 },
{ x: 0.71875, y: 0.90625 },
{ x: 0.71875, y: 0.90625 },
{ x: 0.78125, y: 0.90625 },
{ x: 0.78125, y: 0.90625 },
{ x: 0.84375, y: 0.90625 },
{ x: 0.84375, y: 0.90625 },
{ x: 0.90625, y: 0.90625 },
{ x: 0.90625, y: 0.90625 },
{ x: 0.96875, y: 0.90625 },
{ x: 0.96875, y: 0.90625 },
{ x: 0.03125, y: 0.96875 },
{ x: 0.03125, y: 0.96875 },
{ x: 0.09375, y: 0.96875 },
{ x: 0.09375, y: 0.96875 },
{ x: 0.15625, y: 0.96875 },
{ x: 0.15625, y: 0.96875 },
{ x: 0.21875, y: 0.96875 },
{ x: 0.21875, y: 0.96875 },
{ x: 0.28125, y: 0.96875 },
{ x: 0.28125, y: 0.96875 },
{ x: 0.34375, y: 0.96875 },
{ x: 0.34375, y: 0.96875 },
{ x: 0.40625, y: 0.96875 },
{ x: 0.40625, y: 0.96875 },
{ x: 0.46875, y: 0.96875 },
{ x: 0.46875, y: 0.96875 },
{ x: 0.53125, y: 0.96875 },
{ x: 0.53125, y: 0.96875 },
{ x: 0.59375, y: 0.96875 },
{ x: 0.59375, y: 0.96875 },
{ x: 0.65625, y: 0.96875 },
{ x: 0.65625, y: 0.96875 },
{ x: 0.71875, y: 0.96875 },
{ x: 0.71875, y: 0.96875 },
{ x: 0.78125, y: 0.96875 },
{ x: 0.78125, y: 0.96875 },
{ x: 0.84375, y: 0.96875 },
{ x: 0.84375, y: 0.96875 },
{ x: 0.90625, y: 0.96875 },
{ x: 0.90625, y: 0.96875 },
{ x: 0.96875, y: 0.96875 },
{ x: 0.96875, y: 0.96875 },
{ x: 0.0625, y: 0.0625 },
{ x: 0.0625, y: 0.0625 },
{ x: 0.0625, y: 0.0625 },
{ x: 0.0625, y: 0.0625 },
{ x: 0.0625, y: 0.0625 },
{ x: 0.0625, y: 0.0625 },
{ x: 0.1875, y: 0.0625 },
{ x: 0.1875, y: 0.0625 },
{ x: 0.1875, y: 0.0625 },
{ x: 0.1875, y: 0.0625 },
{ x: 0.1875, y: 0.0625 },
{ x: 0.1875, y: 0.0625 },
{ x: 0.3125, y: 0.0625 },
{ x: 0.3125, y: 0.0625 },
{ x: 0.3125, y: 0.0625 },
{ x: 0.3125, y: 0.0625 },
{ x: 0.3125, y: 0.0625 },
{ x: 0.3125, y: 0.0625 },
{ x: 0.4375, y: 0.0625 },
{ x: 0.4375, y: 0.0625 },
{ x: 0.4375, y: 0.0625 },
{ x: 0.4375, y: 0.0625 },
{ x: 0.4375, y: 0.0625 },
{ x: 0.4375, y: 0.0625 },
{ x: 0.5625, y: 0.0625 },
{ x: 0.5625, y: 0.0625 },
{ x: 0.5625, y: 0.0625 },
{ x: 0.5625, y: 0.0625 },
{ x: 0.5625, y: 0.0625 },
{ x: 0.5625, y: 0.0625 },
{ x: 0.6875, y: 0.0625 },
{ x: 0.6875, y: 0.0625 },
{ x: 0.6875, y: 0.0625 },
{ x: 0.6875, y: 0.0625 },
{ x: 0.6875, y: 0.0625 },
{ x: 0.6875, y: 0.0625 },
{ x: 0.8125, y: 0.0625 },
{ x: 0.8125, y: 0.0625 },
{ x: 0.8125, y: 0.0625 },
{ x: 0.8125, y: 0.0625 },
{ x: 0.8125, y: 0.0625 },
{ x: 0.8125, y: 0.0625 },
{ x: 0.9375, y: 0.0625 },
{ x: 0.9375, y: 0.0625 },
{ x: 0.9375, y: 0.0625 },
{ x: 0.9375, y: 0.0625 },
{ x: 0.9375, y: 0.0625 },
{ x: 0.9375, y: 0.0625 },
{ x: 0.0625, y: 0.1875 },
{ x: 0.0625, y: 0.1875 },
{ x: 0.0625, y: 0.1875 },
{ x: 0.0625, y: 0.1875 },
{ x: 0.0625, y: 0.1875 },
{ x: 0.0625, y: 0.1875 },
{ x: 0.1875, y: 0.1875 },
{ x: 0.1875, y: 0.1875 },
{ x: 0.1875, y: 0.1875 },
{ x: 0.1875, y: 0.1875 },
{ x: 0.1875, y: 0.1875 },
{ x: 0.1875, y: 0.1875 },
{ x: 0.3125, y: 0.1875 },
{ x: 0.3125, y: 0.1875 },
{ x: 0.3125, y: 0.1875 },
{ x: 0.3125, y: 0.1875 },
{ x: 0.3125, y: 0.1875 },
{ x: 0.3125, y: 0.1875 },
{ x: 0.4375, y: 0.1875 },
{ x: 0.4375, y: 0.1875 },
{ x: 0.4375, y: 0.1875 },
{ x: 0.4375, y: 0.1875 },
{ x: 0.4375, y: 0.1875 },
{ x: 0.4375, y: 0.1875 },
{ x: 0.5625, y: 0.1875 },
{ x: 0.5625, y: 0.1875 },
{ x: 0.5625, y: 0.1875 },
{ x: 0.5625, y: 0.1875 },
{ x: 0.5625, y: 0.1875 },
{ x: 0.5625, y: 0.1875 },
{ x: 0.6875, y: 0.1875 },
{ x: 0.6875, y: 0.1875 },
{ x: 0.6875, y: 0.1875 },
{ x: 0.6875, y: 0.1875 },
{ x: 0.6875, y: 0.1875 },
{ x: 0.6875, y: 0.1875 },
{ x: 0.8125, y: 0.1875 },
{ x: 0.8125, y: 0.1875 },
{ x: 0.8125, y: 0.1875 },
{ x: 0.8125, y: 0.1875 },
{ x: 0.8125, y: 0.1875 },
{ x: 0.8125, y: 0.1875 },
{ x: 0.9375, y: 0.1875 },
{ x: 0.9375, y: 0.1875 },
{ x: 0.9375, y: 0.1875 },
{ x: 0.9375, y: 0.1875 },
{ x: 0.9375, y: 0.1875 },
{ x: 0.9375, y: 0.1875 },
{ x: 0.0625, y: 0.3125 },
{ x: 0.0625, y: 0.3125 },
{ x: 0.0625, y: 0.3125 },
{ x: 0.0625, y: 0.3125 },
{ x: 0.0625, y: 0.3125 },
{ x: 0.0625, y: 0.3125 },
{ x: 0.1875, y: 0.3125 },
{ x: 0.1875, y: 0.3125 },
{ x: 0.1875, y: 0.3125 },
{ x: 0.1875, y: 0.3125 },
{ x: 0.1875, y: 0.3125 },
{ x: 0.1875, y: 0.3125 },
{ x: 0.3125, y: 0.3125 },
{ x: 0.3125, y: 0.3125 },
{ x: 0.3125, y: 0.3125 },
{ x: 0.3125, y: 0.3125 },
{ x: 0.3125, y: 0.3125 },
{ x: 0.3125, y: 0.3125 },
{ x: 0.4375, y: 0.3125 },
{ x: 0.4375, y: 0.3125 },
{ x: 0.4375, y: 0.3125 },
{ x: 0.4375, y: 0.3125 },
{ x: 0.4375, y: 0.3125 },
{ x: 0.4375, y: 0.3125 },
{ x: 0.5625, y: 0.3125 },
{ x: 0.5625, y: 0.3125 },
{ x: 0.5625, y: 0.3125 },
{ x: 0.5625, y: 0.3125 },
{ x: 0.5625, y: 0.3125 },
{ x: 0.5625, y: 0.3125 },
{ x: 0.6875, y: 0.3125 },
{ x: 0.6875, y: 0.3125 },
{ x: 0.6875, y: 0.3125 },
{ x: 0.6875, y: 0.3125 },
{ x: 0.6875, y: 0.3125 },
{ x: 0.6875, y: 0.3125 },
{ x: 0.8125, y: 0.3125 },
{ x: 0.8125, y: 0.3125 },
{ x: 0.8125, y: 0.3125 },
{ x: 0.8125, y: 0.3125 },
{ x: 0.8125, y: 0.3125 },
{ x: 0.8125, y: 0.3125 },
{ x: 0.9375, y: 0.3125 },
{ x: 0.9375, y: 0.3125 },
{ x: 0.9375, y: 0.3125 },
{ x: 0.9375, y: 0.3125 },
{ x: 0.9375, y: 0.3125 },
{ x: 0.9375, y: 0.3125 },
{ x: 0.0625, y: 0.4375 },
{ x: 0.0625, y: 0.4375 },
{ x: 0.0625, y: 0.4375 },
{ x: 0.0625, y: 0.4375 },
{ x: 0.0625, y: 0.4375 },
{ x: 0.0625, y: 0.4375 },
{ x: 0.1875, y: 0.4375 },
{ x: 0.1875, y: 0.4375 },
{ x: 0.1875, y: 0.4375 },
{ x: 0.1875, y: 0.4375 },
{ x: 0.1875, y: 0.4375 },
{ x: 0.1875, y: 0.4375 },
{ x: 0.3125, y: 0.4375 },
{ x: 0.3125, y: 0.4375 },
{ x: 0.3125, y: 0.4375 },
{ x: 0.3125, y: 0.4375 },
{ x: 0.3125, y: 0.4375 },
{ x: 0.3125, y: 0.4375 },
{ x: 0.4375, y: 0.4375 },
{ x: 0.4375, y: 0.4375 },
{ x: 0.4375, y: 0.4375 },
{ x: 0.4375, y: 0.4375 },
{ x: 0.4375, y: 0.4375 },
{ x: 0.4375, y: 0.4375 },
{ x: 0.5625, y: 0.4375 },
{ x: 0.5625, y: 0.4375 },
{ x: 0.5625, y: 0.4375 },
{ x: 0.5625, y: 0.4375 },
{ x: 0.5625, y: 0.4375 },
{ x: 0.5625, y: 0.4375 },
{ x: 0.6875, y: 0.4375 },
{ x: 0.6875, y: 0.4375 },
{ x: 0.6875, y: 0.4375 },
{ x: 0.6875, y: 0.4375 },
{ x: 0.6875, y: 0.4375 },
{ x: 0.6875, y: 0.4375 },
{ x: 0.8125, y: 0.4375 },
{ x: 0.8125, y: 0.4375 },
{ x: 0.8125, y: 0.4375 },
{ x: 0.8125, y: 0.4375 },
{ x: 0.8125, y: 0.4375 },
{ x: 0.8125, y: 0.4375 },
{ x: 0.9375, y: 0.4375 },
{ x: 0.9375, y: 0.4375 },
{ x: 0.9375, y: 0.4375 },
{ x: 0.9375, y: 0.4375 },
{ x: 0.9375, y: 0.4375 },
{ x: 0.9375, y: 0.4375 },
{ x: 0.0625, y: 0.5625 },
{ x: 0.0625, y: 0.5625 },
{ x: 0.0625, y: 0.5625 },
{ x: 0.0625, y: 0.5625 },
{ x: 0.0625, y: 0.5625 },
{ x: 0.0625, y: 0.5625 },
{ x: 0.1875, y: 0.5625 },
{ x: 0.1875, y: 0.5625 },
{ x: 0.1875, y: 0.5625 },
{ x: 0.1875, y: 0.5625 },
{ x: 0.1875, y: 0.5625 },
{ x: 0.1875, y: 0.5625 },
{ x: 0.3125, y: 0.5625 },
{ x: 0.3125, y: 0.5625 },
{ x: 0.3125, y: 0.5625 },
{ x: 0.3125, y: 0.5625 },
{ x: 0.3125, y: 0.5625 },
{ x: 0.3125, y: 0.5625 },
{ x: 0.4375, y: 0.5625 },
{ x: 0.4375, y: 0.5625 },
{ x: 0.4375, y: 0.5625 },
{ x: 0.4375, y: 0.5625 },
{ x: 0.4375, y: 0.5625 },
{ x: 0.4375, y: 0.5625 },
{ x: 0.5625, y: 0.5625 },
{ x: 0.5625, y: 0.5625 },
{ x: 0.5625, y: 0.5625 },
{ x: 0.5625, y: 0.5625 },
{ x: 0.5625, y: 0.5625 },
{ x: 0.5625, y: 0.5625 },
{ x: 0.6875, y: 0.5625 },
{ x: 0.6875, y: 0.5625 },
{ x: 0.6875, y: 0.5625 },
{ x: 0.6875, y: 0.5625 },
{ x: 0.6875, y: 0.5625 },
{ x: 0.6875, y: 0.5625 },
{ x: 0.8125, y: 0.5625 },
{ x: 0.8125, y: 0.5625 },
{ x: 0.8125, y: 0.5625 },
{ x: 0.8125, y: 0.5625 },
{ x: 0.8125, y: 0.5625 },
{ x: 0.8125, y: 0.5625 },
{ x: 0.9375, y: 0.5625 },
{ x: 0.9375, y: 0.5625 },
{ x: 0.9375, y: 0.5625 },
{ x: 0.9375, y: 0.5625 },
{ x: 0.9375, y: 0.5625 },
{ x: 0.9375, y: 0.5625 },
{ x: 0.0625, y: 0.6875 },
{ x: 0.0625, y: 0.6875 },
{ x: 0.0625, y: 0.6875 },
{ x: 0.0625, y: 0.6875 },
{ x: 0.0625, y: 0.6875 },
{ x: 0.0625, y: 0.6875 },
{ x: 0.1875, y: 0.6875 },
{ x: 0.1875, y: 0.6875 },
{ x: 0.1875, y: 0.6875 },
{ x: 0.1875, y: 0.6875 },
{ x: 0.1875, y: 0.6875 },
{ x: 0.1875, y: 0.6875 },
{ x: 0.3125, y: 0.6875 },
{ x: 0.3125, y: 0.6875 },
{ x: 0.3125, y: 0.6875 },
{ x: 0.3125, y: 0.6875 },
{ x: 0.3125, y: 0.6875 },
{ x: 0.3125, y: 0.6875 },
{ x: 0.4375, y: 0.6875 },
{ x: 0.4375, y: 0.6875 },
{ x: 0.4375, y: 0.6875 },
{ x: 0.4375, y: 0.6875 },
{ x: 0.4375, y: 0.6875 },
{ x: 0.4375, y: 0.6875 },
{ x: 0.5625, y: 0.6875 },
{ x: 0.5625, y: 0.6875 },
{ x: 0.5625, y: 0.6875 },
{ x: 0.5625, y: 0.6875 },
{ x: 0.5625, y: 0.6875 },
{ x: 0.5625, y: 0.6875 },
{ x: 0.6875, y: 0.6875 },
{ x: 0.6875, y: 0.6875 },
{ x: 0.6875, y: 0.6875 },
{ x: 0.6875, y: 0.6875 },
{ x: 0.6875, y: 0.6875 },
{ x: 0.6875, y: 0.6875 },
{ x: 0.8125, y: 0.6875 },
{ x: 0.8125, y: 0.6875 },
{ x: 0.8125, y: 0.6875 },
{ x: 0.8125, y: 0.6875 },
{ x: 0.8125, y: 0.6875 },
{ x: 0.8125, y: 0.6875 },
{ x: 0.9375, y: 0.6875 },
{ x: 0.9375, y: 0.6875 },
{ x: 0.9375, y: 0.6875 },
{ x: 0.9375, y: 0.6875 },
{ x: 0.9375, y: 0.6875 },
{ x: 0.9375, y: 0.6875 },
{ x: 0.0625, y: 0.8125 },
{ x: 0.0625, y: 0.8125 },
{ x: 0.0625, y: 0.8125 },
{ x: 0.0625, y: 0.8125 },
{ x: 0.0625, y: 0.8125 },
{ x: 0.0625, y: 0.8125 },
{ x: 0.1875, y: 0.8125 },
{ x: 0.1875, y: 0.8125 },
{ x: 0.1875, y: 0.8125 },
{ x: 0.1875, y: 0.8125 },
{ x: 0.1875, y: 0.8125 },
{ x: 0.1875, y: 0.8125 },
{ x: 0.3125, y: 0.8125 },
{ x: 0.3125, y: 0.8125 },
{ x: 0.3125, y: 0.8125 },
{ x: 0.3125, y: 0.8125 },
{ x: 0.3125, y: 0.8125 },
{ x: 0.3125, y: 0.8125 },
{ x: 0.4375, y: 0.8125 },
{ x: 0.4375, y: 0.8125 },
{ x: 0.4375, y: 0.8125 },
{ x: 0.4375, y: 0.8125 },
{ x: 0.4375, y: 0.8125 },
{ x: 0.4375, y: 0.8125 },
{ x: 0.5625, y: 0.8125 },
{ x: 0.5625, y: 0.8125 },
{ x: 0.5625, y: 0.8125 },
{ x: 0.5625, y: 0.8125 },
{ x: 0.5625, y: 0.8125 },
{ x: 0.5625, y: 0.8125 },
{ x: 0.6875, y: 0.8125 },
{ x: 0.6875, y: 0.8125 },
{ x: 0.6875, y: 0.8125 },
{ x: 0.6875, y: 0.8125 },
{ x: 0.6875, y: 0.8125 },
{ x: 0.6875, y: 0.8125 },
{ x: 0.8125, y: 0.8125 },
{ x: 0.8125, y: 0.8125 },
{ x: 0.8125, y: 0.8125 },
{ x: 0.8125, y: 0.8125 },
{ x: 0.8125, y: 0.8125 },
{ x: 0.8125, y: 0.8125 },
{ x: 0.9375, y: 0.8125 },
{ x: 0.9375, y: 0.8125 },
{ x: 0.9375, y: 0.8125 },
{ x: 0.9375, y: 0.8125 },
{ x: 0.9375, y: 0.8125 },
{ x: 0.9375, y: 0.8125 },
{ x: 0.0625, y: 0.9375 },
{ x: 0.0625, y: 0.9375 },
{ x: 0.0625, y: 0.9375 },
{ x: 0.0625, y: 0.9375 },
{ x: 0.0625, y: 0.9375 },
{ x: 0.0625, y: 0.9375 },
{ x: 0.1875, y: 0.9375 },
{ x: 0.1875, y: 0.9375 },
{ x: 0.1875, y: 0.9375 },
{ x: 0.1875, y: 0.9375 },
{ x: 0.1875, y: 0.9375 },
{ x: 0.1875, y: 0.9375 },
{ x: 0.3125, y: 0.9375 },
{ x: 0.3125, y: 0.9375 },
{ x: 0.3125, y: 0.9375 },
{ x: 0.3125, y: 0.9375 },
{ x: 0.3125, y: 0.9375 },
{ x: 0.3125, y: 0.9375 },
{ x: 0.4375, y: 0.9375 },
{ x: 0.4375, y: 0.9375 },
{ x: 0.4375, y: 0.9375 },
{ x: 0.4375, y: 0.9375 },
{ x: 0.4375, y: 0.9375 },
{ x: 0.4375, y: 0.9375 },
{ x: 0.5625, y: 0.9375 },
{ x: 0.5625, y: 0.9375 },
{ x: 0.5625, y: 0.9375 },
{ x: 0.5625, y: 0.9375 },
{ x: 0.5625, y: 0.9375 },
{ x: 0.5625, y: 0.9375 },
{ x: 0.6875, y: 0.9375 },
{ x: 0.6875, y: 0.9375 },
{ x: 0.6875, y: 0.9375 },
{ x: 0.6875, y: 0.9375 },
{ x: 0.6875, y: 0.9375 },
{ x: 0.6875, y: 0.9375 },
{ x: 0.8125, y: 0.9375 },
{ x: 0.8125, y: 0.9375 },
{ x: 0.8125, y: 0.9375 },
{ x: 0.8125, y: 0.9375 },
{ x: 0.8125, y: 0.9375 },
{ x: 0.8125, y: 0.9375 },
{ x: 0.9375, y: 0.9375 },
{ x: 0.9375, y: 0.9375 },
{ x: 0.9375, y: 0.9375 },
{ x: 0.9375, y: 0.9375 },
{ x: 0.9375, y: 0.9375 },
{ x: 0.9375, y: 0.9375 }
];
// src/hand/handposedetector.ts
var HandDetector = class {
constructor(model23) {
__publicField(this, "model");
__publicField(this, "anchors");
__publicField(this, "anchorsTensor");
__publicField(this, "inputSize");
__publicField(this, "inputSizeTensor");
__publicField(this, "doubleInputSizeTensor");
var _a, _b, _c, _d;
this.model = model23;
this.anchors = anchors2.map((anchor) => [anchor.x, anchor.y]);
this.anchorsTensor = tf27.tensor2d(this.anchors);
this.inputSize = ((_d = (_c = (_b = (_a = this == null ? void 0 : this.model) == null ? void 0 : _a.inputs) == null ? void 0 : _b[0]) == null ? void 0 : _c.shape) == null ? void 0 : _d[2]) || 0;
this.inputSizeTensor = tf27.tensor1d([this.inputSize, this.inputSize]);
this.doubleInputSizeTensor = tf27.tensor1d([this.inputSize * 2, this.inputSize * 2]);
}
normalizeBoxes(boxes) {
const t2 = {};
t2.boxOffsets = tf27.slice(boxes, [0, 0], [-1, 2]);
t2.boxSizes = tf27.slice(boxes, [0, 2], [-1, 2]);
t2.div = tf27.div(t2.boxOffsets, this.inputSizeTensor);
t2.boxCenterPoints = tf27.add(t2.div, this.anchorsTensor);
t2.halfBoxSizes = tf27.div(t2.boxSizes, this.doubleInputSizeTensor);
t2.sub = tf27.sub(t2.boxCenterPoints, t2.halfBoxSizes);
t2.startPoints = tf27.mul(t2.sub, this.inputSizeTensor);
t2.add = tf27.add(t2.boxCenterPoints, t2.halfBoxSizes);
t2.endPoints = tf27.mul(t2.add, this.inputSizeTensor);
const res = tf27.concat2d([t2.startPoints, t2.endPoints], 1);
Object.keys(t2).forEach((tensor6) => tf27.dispose(t2[tensor6]));
return res;
}
normalizeLandmarks(rawPalmLandmarks, index2) {
const t2 = {};
t2.reshape = tf27.reshape(rawPalmLandmarks, [-1, 7, 2]);
t2.div = tf27.div(t2.reshape, this.inputSizeTensor);
t2.landmarks = tf27.add(t2.div, this.anchors[index2] ? this.anchors[index2] : 0);
const res = tf27.mul(t2.landmarks, this.inputSizeTensor);
Object.keys(t2).forEach((tensor6) => tf27.dispose(t2[tensor6]));
return res;
}
async predict(input, config3) {
var _a;
const t2 = {};
t2.resize = tf27.image.resizeBilinear(input, [this.inputSize, this.inputSize]);
t2.div = tf27.div(t2.resize, constants.tf127);
t2.image = tf27.sub(t2.div, constants.tf1);
t2.batched = this.model.execute(t2.image);
t2.predictions = tf27.squeeze(t2.batched);
t2.slice = tf27.slice(t2.predictions, [0, 0], [-1, 1]);
t2.sigmoid = tf27.sigmoid(t2.slice);
t2.scores = tf27.squeeze(t2.sigmoid);
const scores = await t2.scores.data();
t2.boxes = tf27.slice(t2.predictions, [0, 1], [-1, 4]);
t2.norm = this.normalizeBoxes(t2.boxes);
t2.nms = await tf27.image.nonMaxSuppressionAsync(t2.norm, t2.scores, 3 * (((_a = config3.hand) == null ? void 0 : _a.maxDetected) || 1), config3.hand.iouThreshold, config3.hand.minConfidence);
const nms = await t2.nms.array();
const hands = [];
for (const index2 of nms) {
const p = {};
p.box = tf27.slice(t2.norm, [index2, 0], [1, -1]);
p.slice = tf27.slice(t2.predictions, [index2, 5], [1, 14]);
p.norm = this.normalizeLandmarks(p.slice, index2);
p.palmLandmarks = tf27.reshape(p.norm, [-1, 2]);
const box = await p.box.data();
const startPoint = box.slice(0, 2);
const endPoint = box.slice(2, 4);
const palmLandmarks = await p.palmLandmarks.array();
const hand3 = { startPoint, endPoint, palmLandmarks, confidence: scores[index2] };
const scaled = scaleBoxCoordinates2(hand3, [(input.shape[2] || 1) / this.inputSize, (input.shape[1] || 0) / this.inputSize]);
hands.push(scaled);
Object.keys(p).forEach((tensor6) => tf27.dispose(p[tensor6]));
}
Object.keys(t2).forEach((tensor6) => tf27.dispose(t2[tensor6]));
return hands;
}
};
// src/hand/handposepipeline.ts
var tf28 = __toESM(require_tfjs_esm());
var palmBoxEnlargeFactor = 5;
var handBoxEnlargeFactor = 1.65;
var palmLandmarkIds = [0, 5, 9, 13, 17, 1, 2];
var palmLandmarksPalmBase = 0;
var palmLandmarksMiddleFingerBase = 2;
var lastTime13 = 0;
var HandPipeline = class {
constructor(handDetector, handPoseModel2) {
__publicField(this, "handDetector");
__publicField(this, "handPoseModel");
__publicField(this, "inputSize");
__publicField(this, "storedBoxes");
__publicField(this, "skipped");
__publicField(this, "detectedHands");
var _a, _b, _c;
this.handDetector = handDetector;
this.handPoseModel = handPoseModel2;
this.inputSize = ((_c = (_b = (_a = this.handPoseModel) == null ? void 0 : _a.inputs) == null ? void 0 : _b[0].shape) == null ? void 0 : _c[2]) || 0;
this.storedBoxes = [];
this.skipped = Number.MAX_SAFE_INTEGER;
this.detectedHands = 0;
}
calculateLandmarksBoundingBox(landmarks) {
const xs = landmarks.map((d) => d[0]);
const ys = landmarks.map((d) => d[1]);
const startPoint = [Math.min(...xs), Math.min(...ys)];
const endPoint = [Math.max(...xs), Math.max(...ys)];
return { startPoint, endPoint };
}
getBoxForPalmLandmarks(palmLandmarks, rotationMatrix) {
const rotatedPalmLandmarks = palmLandmarks.map((coord) => rotatePoint2([...coord, 1], rotationMatrix));
const boxAroundPalm = this.calculateLandmarksBoundingBox(rotatedPalmLandmarks);
return enlargeBox2(squarifyBox2(boxAroundPalm), palmBoxEnlargeFactor);
}
getBoxForHandLandmarks(landmarks) {
const boundingBox = this.calculateLandmarksBoundingBox(landmarks);
const boxAroundHand = enlargeBox2(squarifyBox2(boundingBox), handBoxEnlargeFactor);
boxAroundHand.palmLandmarks = [];
for (let i = 0; i < palmLandmarkIds.length; i++) {
boxAroundHand.palmLandmarks.push(landmarks[palmLandmarkIds[i]].slice(0, 2));
}
return boxAroundHand;
}
transformRawCoords(rawCoords, box2, angle, rotationMatrix) {
const boxSize = getBoxSize2(box2);
const scaleFactor = [boxSize[0] / this.inputSize, boxSize[1] / this.inputSize, (boxSize[0] + boxSize[1]) / this.inputSize / 2];
const coordsScaled = rawCoords.map((coord) => [
scaleFactor[0] * (coord[0] - this.inputSize / 2),
scaleFactor[1] * (coord[1] - this.inputSize / 2),
scaleFactor[2] * coord[2]
]);
const coordsRotationMatrix = buildRotationMatrix2(angle, [0, 0]);
const coordsRotated = coordsScaled.map((coord) => {
const rotated = rotatePoint2(coord, coordsRotationMatrix);
return [...rotated, coord[2]];
});
const inverseRotationMatrix = invertTransformMatrix2(rotationMatrix);
const boxCenter = [...getBoxCenter2(box2), 1];
const originalBoxCenter = [
dot2(boxCenter, inverseRotationMatrix[0]),
dot2(boxCenter, inverseRotationMatrix[1])
];
return coordsRotated.map((coord) => [
Math.trunc(coord[0] + originalBoxCenter[0]),
Math.trunc(coord[1] + originalBoxCenter[1]),
Math.trunc(coord[2])
]);
}
async estimateHands(image28, config3) {
let useFreshBox = false;
let boxes;
const skipTime = (config3.hand.skipTime || 0) > now() - lastTime13;
const skipFrame = this.skipped < (config3.hand.skipFrames || 0);
if (config3.skipAllowed && skipTime && skipFrame) {
boxes = await this.handDetector.predict(image28, config3);
this.skipped = 0;
}
if (config3.skipAllowed)
this.skipped++;
if (boxes && boxes.length > 0 && (boxes.length !== this.detectedHands && this.detectedHands !== config3.hand.maxDetected || !config3.hand.landmarks)) {
this.detectedHands = 0;
this.storedBoxes = [...boxes];
if (this.storedBoxes.length > 0)
useFreshBox = true;
}
const hands = [];
for (let i = 0; i < this.storedBoxes.length; i++) {
const currentBox = this.storedBoxes[i];
if (!currentBox)
continue;
if (config3.hand.landmarks) {
const angle = config3.hand.rotation ? computeRotation2(currentBox.palmLandmarks[palmLandmarksPalmBase], currentBox.palmLandmarks[palmLandmarksMiddleFingerBase]) : 0;
const palmCenter = getBoxCenter2(currentBox);
const palmCenterNormalized = [palmCenter[0] / image28.shape[2], palmCenter[1] / image28.shape[1]];
const rotatedImage = config3.hand.rotation && env.kernels.includes("rotatewithoffset") ? tf28.image.rotateWithOffset(image28, angle, 0, palmCenterNormalized) : image28.clone();
const rotationMatrix = buildRotationMatrix2(-angle, palmCenter);
const newBox = useFreshBox ? this.getBoxForPalmLandmarks(currentBox.palmLandmarks, rotationMatrix) : currentBox;
const croppedInput = cutBoxFromImageAndResize(newBox, rotatedImage, [this.inputSize, this.inputSize]);
const handImage = tf28.div(croppedInput, constants.tf255);
tf28.dispose(croppedInput);
tf28.dispose(rotatedImage);
const [confidenceT, keypoints] = this.handPoseModel.execute(handImage);
lastTime13 = now();
tf28.dispose(handImage);
const confidence = (await confidenceT.data())[0];
tf28.dispose(confidenceT);
if (confidence >= config3.hand.minConfidence / 4) {
const keypointsReshaped = tf28.reshape(keypoints, [-1, 3]);
const rawCoords = await keypointsReshaped.array();
tf28.dispose(keypoints);
tf28.dispose(keypointsReshaped);
const coords = this.transformRawCoords(rawCoords, newBox, angle, rotationMatrix);
const nextBoundingBox = this.getBoxForHandLandmarks(coords);
this.storedBoxes[i] = { ...nextBoundingBox, confidence };
const result = {
landmarks: coords,
confidence,
boxConfidence: currentBox.confidence,
fingerConfidence: confidence,
box: { topLeft: nextBoundingBox.startPoint, bottomRight: nextBoundingBox.endPoint }
};
hands.push(result);
} else {
this.storedBoxes[i] = null;
}
tf28.dispose(keypoints);
} else {
const enlarged = enlargeBox2(squarifyBox2(currentBox), handBoxEnlargeFactor);
const result = {
confidence: currentBox.confidence,
boxConfidence: currentBox.confidence,
fingerConfidence: 0,
box: { topLeft: enlarged.startPoint, bottomRight: enlarged.endPoint },
landmarks: []
};
hands.push(result);
}
}
this.storedBoxes = this.storedBoxes.filter((a) => a !== null);
this.detectedHands = hands.length;
if (hands.length > config3.hand.maxDetected)
hands.length = config3.hand.maxDetected;
return hands;
}
};
// src/hand/handpose.ts
var meshAnnotations2 = {
thumb: [1, 2, 3, 4],
index: [5, 6, 7, 8],
middle: [9, 10, 11, 12],
ring: [13, 14, 15, 16],
pinky: [17, 18, 19, 20],
palm: [0]
};
var handDetectorModel;
var handPoseModel;
var handPipeline;
async function predict14(input, config3) {
const predictions = await handPipeline.estimateHands(input, config3);
if (!predictions)
return [];
const hands = [];
for (let i = 0; i < predictions.length; i++) {
const annotations2 = {};
if (predictions[i].landmarks) {
for (const key of Object.keys(meshAnnotations2)) {
annotations2[key] = meshAnnotations2[key].map((index2) => predictions[i].landmarks[index2]);
}
}
const keypoints = predictions[i].landmarks;
let box = [Number.MAX_SAFE_INTEGER, Number.MAX_SAFE_INTEGER, 0, 0];
let boxRaw = [0, 0, 0, 0];
if (keypoints && keypoints.length > 0) {
for (const pt of keypoints) {
if (pt[0] < box[0])
box[0] = pt[0];
if (pt[1] < box[1])
box[1] = pt[1];
if (pt[0] > box[2])
box[2] = pt[0];
if (pt[1] > box[3])
box[3] = pt[1];
}
box[2] -= box[0];
box[3] -= box[1];
boxRaw = [box[0] / (input.shape[2] || 0), box[1] / (input.shape[1] || 0), box[2] / (input.shape[2] || 0), box[3] / (input.shape[1] || 0)];
} else {
box = predictions[i].box ? [
Math.trunc(Math.max(0, predictions[i].box.topLeft[0])),
Math.trunc(Math.max(0, predictions[i].box.topLeft[1])),
Math.trunc(Math.min(input.shape[2] || 0, predictions[i].box.bottomRight[0]) - Math.max(0, predictions[i].box.topLeft[0])),
Math.trunc(Math.min(input.shape[1] || 0, predictions[i].box.bottomRight[1]) - Math.max(0, predictions[i].box.topLeft[1]))
] : [0, 0, 0, 0];
boxRaw = [
predictions[i].box.topLeft[0] / (input.shape[2] || 0),
predictions[i].box.topLeft[1] / (input.shape[1] || 0),
(predictions[i].box.bottomRight[0] - predictions[i].box.topLeft[0]) / (input.shape[2] || 0),
(predictions[i].box.bottomRight[1] - predictions[i].box.topLeft[1]) / (input.shape[1] || 0)
];
}
const landmarks = analyze(keypoints);
hands.push({
id: i,
score: Math.round(100 * predictions[i].confidence) / 100,
boxScore: Math.round(100 * predictions[i].boxConfidence) / 100,
fingerScore: Math.round(100 * predictions[i].fingerConfidence) / 100,
label: "hand",
box,
boxRaw,
keypoints,
annotations: annotations2,
landmarks
});
}
return hands;
}
async function load15(config3) {
var _a, _b;
if (env.initial) {
handDetectorModel = null;
handPoseModel = null;
}
if (!handDetectorModel || !handPoseModel) {
[handDetectorModel, handPoseModel] = await Promise.all([
config3.hand.enabled ? loadModel((_a = config3.hand.detector) == null ? void 0 : _a.modelPath) : null,
config3.hand.landmarks ? loadModel((_b = config3.hand.skeleton) == null ? void 0 : _b.modelPath) : null
]);
} else {
if (config3.debug)
log("cached model:", handDetectorModel["modelUrl"]);
if (config3.debug)
log("cached model:", handPoseModel["modelUrl"]);
}
const handDetector = handDetectorModel ? new HandDetector(handDetectorModel) : void 0;
if (handDetector && handPoseModel)
handPipeline = new HandPipeline(handDetector, handPoseModel);
return [handDetectorModel, handPoseModel];
}
// src/hand/handtrack.ts
var tf29 = __toESM(require_tfjs_esm());
var models2 = [null, null];
var modelOutputNodes = ["StatefulPartitionedCall/Postprocessor/Slice", "StatefulPartitionedCall/Postprocessor/ExpandDims_1"];
var inputSize7 = [[0, 0], [0, 0]];
var classes = ["hand", "fist", "pinch", "point", "face", "tip", "pinchtip"];
var faceIndex = 4;
var boxExpandFact = 1.6;
var maxDetectorResolution = 512;
var detectorExpandFact = 1.4;
var skipped13 = Number.MAX_SAFE_INTEGER;
var lastTime14 = 0;
var outputSize = [0, 0];
var cache4 = {
boxes: [],
hands: []
};
var fingerMap = {
thumb: [1, 2, 3, 4],
index: [5, 6, 7, 8],
middle: [9, 10, 11, 12],
ring: [13, 14, 15, 16],
pinky: [17, 18, 19, 20],
base: [0],
palm: [0, 17, 13, 9, 5, 1, 0]
};
async function loadDetect2(config3) {
var _a;
if (env.initial)
models2[0] = null;
if (!models2[0]) {
fakeOps(["tensorlistreserve", "enter", "tensorlistfromtensor", "merge", "loopcond", "switch", "exit", "tensorliststack", "nextiteration", "tensorlistsetitem", "tensorlistgetitem", "reciprocal", "shape", "split", "where"], config3);
models2[0] = await loadModel((_a = config3.hand.detector) == null ? void 0 : _a.modelPath);
const inputs = models2[0]["executor"] ? Object.values(models2[0].modelSignature["inputs"]) : void 0;
inputSize7[0][0] = Array.isArray(inputs) ? parseInt(inputs[0].tensorShape.dim[1].size) : 0;
inputSize7[0][1] = Array.isArray(inputs) ? parseInt(inputs[0].tensorShape.dim[2].size) : 0;
} else if (config3.debug)
log("cached model:", models2[0]["modelUrl"]);
return models2[0];
}
async function loadSkeleton(config3) {
var _a;
if (env.initial)
models2[1] = null;
if (!models2[1]) {
models2[1] = await loadModel((_a = config3.hand.skeleton) == null ? void 0 : _a.modelPath);
const inputs = models2[1]["executor"] ? Object.values(models2[1].modelSignature["inputs"]) : void 0;
inputSize7[1][0] = Array.isArray(inputs) ? parseInt(inputs[0].tensorShape.dim[1].size) : 0;
inputSize7[1][1] = Array.isArray(inputs) ? parseInt(inputs[0].tensorShape.dim[2].size) : 0;
} else if (config3.debug)
log("cached model:", models2[1]["modelUrl"]);
return models2[1];
}
async function detectHands(input, config3) {
const hands = [];
if (!input || !models2[0])
return hands;
const t2 = {};
const ratio2 = (input.shape[2] || 1) / (input.shape[1] || 1);
const height = Math.min(Math.round((input.shape[1] || 0) / 8) * 8, maxDetectorResolution);
const width = Math.round(height * ratio2 / 8) * 8;
t2.resize = tf29.image.resizeBilinear(input, [height, width]);
t2.cast = tf29.cast(t2.resize, "int32");
[t2.rawScores, t2.rawBoxes] = await models2[0].executeAsync(t2.cast, modelOutputNodes);
t2.boxes = tf29.squeeze(t2.rawBoxes, [0, 2]);
t2.scores = tf29.squeeze(t2.rawScores, [0]);
const classScores = tf29.unstack(t2.scores, 1);
tf29.dispose(classScores[faceIndex]);
classScores.splice(faceIndex, 1);
t2.filtered = tf29.stack(classScores, 1);
tf29.dispose(classScores);
t2.max = tf29.max(t2.filtered, 1);
t2.argmax = tf29.argMax(t2.filtered, 1);
let id = 0;
t2.nms = await tf29.image.nonMaxSuppressionAsync(t2.boxes, t2.max, (config3.hand.maxDetected || 0) + 1, config3.hand.iouThreshold || 0, config3.hand.minConfidence || 1);
const nms = await t2.nms.data();
const scores = await t2.max.data();
const classNum = await t2.argmax.data();
for (const nmsIndex of Array.from(nms)) {
const boxSlice = tf29.slice(t2.boxes, nmsIndex, 1);
const boxYX = await boxSlice.data();
tf29.dispose(boxSlice);
const boxData = [boxYX[1], boxYX[0], boxYX[3] - boxYX[1], boxYX[2] - boxYX[0]];
const boxRaw = scale(boxData, detectorExpandFact);
const boxFull = [Math.trunc(boxData[0] * outputSize[0]), Math.trunc(boxData[1] * outputSize[1]), Math.trunc(boxData[2] * outputSize[0]), Math.trunc(boxData[3] * outputSize[1])];
const score = scores[nmsIndex];
const label = classes[classNum[nmsIndex]];
const hand3 = { id: id++, score, box: boxFull, boxRaw, label };
hands.push(hand3);
}
Object.keys(t2).forEach((tensor6) => tf29.dispose(t2[tensor6]));
hands.sort((a, b) => b.score - a.score);
if (hands.length > (config3.hand.maxDetected || 1))
hands.length = config3.hand.maxDetected || 1;
return hands;
}
async function detectFingers(input, h, config3) {
const hand3 = {
id: h.id,
score: Math.round(100 * h.score) / 100,
boxScore: Math.round(100 * h.score) / 100,
fingerScore: 0,
box: h.box,
boxRaw: h.boxRaw,
label: h.label,
keypoints: [],
landmarks: {},
annotations: {}
};
if (input && models2[1] && config3.hand.landmarks && h.score > (config3.hand.minConfidence || 0)) {
const t2 = {};
const boxCrop = [h.boxRaw[1], h.boxRaw[0], h.boxRaw[3] + h.boxRaw[1], h.boxRaw[2] + h.boxRaw[0]];
t2.crop = tf29.image.cropAndResize(input, [boxCrop], [0], [inputSize7[1][0], inputSize7[1][1]], "bilinear");
t2.div = tf29.div(t2.crop, constants.tf255);
[t2.score, t2.keypoints] = models2[1].execute(t2.div, ["Identity_1", "Identity"]);
const rawScore = (await t2.score.data())[0];
const score = (100 - Math.trunc(100 / (1 + Math.exp(rawScore)))) / 100;
if (score >= (config3.hand.minConfidence || 0)) {
hand3.fingerScore = score;
t2.reshaped = tf29.reshape(t2.keypoints, [-1, 3]);
const coordsData = await t2.reshaped.array();
const coordsRaw = coordsData.map((kpt4) => [kpt4[0] / inputSize7[1][1], kpt4[1] / inputSize7[1][0], kpt4[2] || 0]);
const coordsNorm = coordsRaw.map((kpt4) => [kpt4[0] * h.boxRaw[2], kpt4[1] * h.boxRaw[3], kpt4[2] || 0]);
hand3.keypoints = coordsNorm.map((kpt4) => [outputSize[0] * (kpt4[0] + h.boxRaw[0]), outputSize[1] * (kpt4[1] + h.boxRaw[1]), kpt4[2] || 0]);
hand3.landmarks = analyze(hand3.keypoints);
for (const key of Object.keys(fingerMap)) {
hand3.annotations[key] = fingerMap[key].map((index2) => hand3.landmarks && hand3.keypoints[index2] ? hand3.keypoints[index2] : null);
}
}
Object.keys(t2).forEach((tensor6) => tf29.dispose(t2[tensor6]));
}
return hand3;
}
async function predict15(input, config3) {
var _a, _b;
if (!((_a = models2[0]) == null ? void 0 : _a["executor"]) || !((_b = models2[1]) == null ? void 0 : _b["executor"]) || !models2[0].inputs[0].shape || !models2[1].inputs[0].shape)
return [];
outputSize = [input.shape[2] || 0, input.shape[1] || 0];
skipped13++;
const skipTime = (config3.hand.skipTime || 0) > now() - lastTime14;
const skipFrame = skipped13 < (config3.hand.skipFrames || 0);
if (config3.skipAllowed && skipTime && skipFrame) {
return cache4.hands;
}
return new Promise(async (resolve) => {
const skipTimeExtended = 3 * (config3.hand.skipTime || 0) > now() - lastTime14;
const skipFrameExtended = skipped13 < 3 * (config3.hand.skipFrames || 0);
if (config3.skipAllowed && cache4.hands.length === config3.hand.maxDetected) {
cache4.hands = await Promise.all(cache4.boxes.map((handBox) => detectFingers(input, handBox, config3)));
} else if (config3.skipAllowed && skipTimeExtended && skipFrameExtended && cache4.hands.length > 0) {
cache4.hands = await Promise.all(cache4.boxes.map((handBox) => detectFingers(input, handBox, config3)));
} else {
cache4.boxes = await detectHands(input, config3);
lastTime14 = now();
cache4.hands = await Promise.all(cache4.boxes.map((handBox) => detectFingers(input, handBox, config3)));
skipped13 = 0;
}
const oldCache = [...cache4.boxes];
cache4.boxes.length = 0;
if (config3.cacheSensitivity > 0) {
for (let i = 0; i < cache4.hands.length; i++) {
const boxKpt = square(cache4.hands[i].keypoints, outputSize);
if (boxKpt.box[2] / (input.shape[2] || 1) > 0.05 && boxKpt.box[3] / (input.shape[1] || 1) > 0.05 && cache4.hands[i].fingerScore && cache4.hands[i].fingerScore > (config3.hand.minConfidence || 0)) {
const boxScale = scale(boxKpt.box, boxExpandFact);
const boxScaleRaw = scale(boxKpt.boxRaw, boxExpandFact);
cache4.boxes.push({ ...oldCache[i], box: boxScale, boxRaw: boxScaleRaw });
}
}
}
for (let i = 0; i < cache4.hands.length; i++) {
const bbox = calc(cache4.hands[i].keypoints, outputSize);
cache4.hands[i].box = bbox.box;
cache4.hands[i].boxRaw = bbox.boxRaw;
}
resolve(cache4.hands);
});
}
// src/result.ts
var empty = (error = null) => ({ face: [], body: [], hand: [], gesture: [], object: [], persons: [], performance: {}, timestamp: 0, width: 0, height: 0, error });
// src/body/movenetcoords.ts
var movenetcoords_exports = {};
__export(movenetcoords_exports, {
connected: () => connected3,
horizontal: () => horizontal,
kpt: () => kpt3,
relative: () => relative,
vertical: () => vertical
});
var kpt3 = [
"nose",
"leftEye",
"rightEye",
"leftEar",
"rightEar",
"leftShoulder",
"rightShoulder",
"leftElbow",
"rightElbow",
"leftWrist",
"rightWrist",
"leftHip",
"rightHip",
"leftKnee",
"rightKnee",
"leftAnkle",
"rightAnkle"
];
var horizontal = [
["leftEye", "rightEye"],
["leftEar", "rightEar"],
["leftShoulder", "rightShoulder"],
["leftElbow", "rightElbow"],
["leftWrist", "rightWrist"],
["leftHip", "rightHip"],
["leftKnee", "rightKnee"],
["leftAnkle", "rightAnkle"]
];
var vertical = [
["leftKnee", "leftShoulder"],
["rightKnee", "rightShoulder"],
["leftAnkle", "leftKnee"],
["rightAnkle", "rightKnee"]
];
var relative = [
[["leftHip", "rightHip"], ["leftShoulder", "rightShoulder"]],
[["leftElbow", "rightElbow"], ["leftShoulder", "rightShoulder"]]
];
var connected3 = {
leftLeg: ["leftHip", "leftKnee", "leftAnkle"],
rightLeg: ["rightHip", "rightKnee", "rightAnkle"],
torso: ["leftShoulder", "rightShoulder", "rightHip", "leftHip", "leftShoulder"],
leftArm: ["leftShoulder", "leftElbow", "leftWrist"],
rightArm: ["rightShoulder", "rightElbow", "rightWrist"],
head: []
};
// src/util/interpolate.ts
var bufferedResult = empty();
var interpolateTime = 0;
function calc2(newResult, config3) {
var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l, _m, _n, _o, _p, _q, _r, _s, _t, _u, _v, _w;
const t0 = now();
if (!newResult)
return empty();
const elapsed = Date.now() - newResult.timestamp;
const bufferedFactor = elapsed < 1e3 ? 8 - Math.log(elapsed + 1) : 1;
if (newResult.canvas)
bufferedResult.canvas = newResult.canvas;
if (newResult.error)
bufferedResult.error = newResult.error;
if (!bufferedResult.body || newResult.body.length !== bufferedResult.body.length) {
bufferedResult.body = JSON.parse(JSON.stringify(newResult.body));
} else {
for (let i = 0; i < newResult.body.length; i++) {
const box = newResult.body[i].box.map((newBoxCoord, j) => ((bufferedFactor - 1) * bufferedResult.body[i].box[j] + newBoxCoord) / bufferedFactor);
const boxRaw = newResult.body[i].boxRaw.map((newBoxCoord, j) => ((bufferedFactor - 1) * bufferedResult.body[i].boxRaw[j] + newBoxCoord) / bufferedFactor);
const keypoints = newResult.body[i].keypoints.map((newKpt, j) => {
var _a2, _b2, _c2, _d2, _e2, _f2, _g2, _h2, _i2;
return {
score: newKpt.score,
part: newKpt.part,
position: [
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (bufferedResult.body[i].keypoints[j].position[0] || 0) + (newKpt.position[0] || 0)) / bufferedFactor : newKpt.position[0],
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (bufferedResult.body[i].keypoints[j].position[1] || 0) + (newKpt.position[1] || 0)) / bufferedFactor : newKpt.position[1],
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (bufferedResult.body[i].keypoints[j].position[2] || 0) + (newKpt.position[2] || 0)) / bufferedFactor : newKpt.position[2]
],
positionRaw: [
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (bufferedResult.body[i].keypoints[j].positionRaw[0] || 0) + (newKpt.positionRaw[0] || 0)) / bufferedFactor : newKpt.positionRaw[0],
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (bufferedResult.body[i].keypoints[j].positionRaw[1] || 0) + (newKpt.positionRaw[1] || 0)) / bufferedFactor : newKpt.positionRaw[1],
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (bufferedResult.body[i].keypoints[j].positionRaw[2] || 0) + (newKpt.positionRaw[2] || 0)) / bufferedFactor : newKpt.positionRaw[2]
],
distance: [
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (((_a2 = bufferedResult.body[i].keypoints[j].distance) == null ? void 0 : _a2[0]) || 0) + (((_b2 = newKpt.distance) == null ? void 0 : _b2[0]) || 0)) / bufferedFactor : (_c2 = newKpt.distance) == null ? void 0 : _c2[0],
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (((_d2 = bufferedResult.body[i].keypoints[j].distance) == null ? void 0 : _d2[1]) || 0) + (((_e2 = newKpt.distance) == null ? void 0 : _e2[1]) || 0)) / bufferedFactor : (_f2 = newKpt.distance) == null ? void 0 : _f2[1],
bufferedResult.body[i].keypoints[j] ? ((bufferedFactor - 1) * (((_g2 = bufferedResult.body[i].keypoints[j].distance) == null ? void 0 : _g2[2]) || 0) + (((_h2 = newKpt.distance) == null ? void 0 : _h2[2]) || 0)) / bufferedFactor : (_i2 = newKpt.distance) == null ? void 0 : _i2[2]
]
};
});
const annotations2 = {};
let coords = { connected: {} };
if ((_a = config3.body.modelPath) == null ? void 0 : _a.includes("efficientpose"))
coords = efficientposecoords_exports;
else if ((_b = config3.body.modelPath) == null ? void 0 : _b.includes("blazepose"))
coords = blazeposecoords_exports;
else if ((_c = config3.body.modelPath) == null ? void 0 : _c.includes("movenet"))
coords = movenetcoords_exports;
for (const [name, indexes] of Object.entries(coords.connected)) {
const pt = [];
for (let j = 0; j < indexes.length - 1; j++) {
const pt0 = keypoints.find((kp) => kp.part === indexes[j]);
const pt1 = keypoints.find((kp) => kp.part === indexes[j + 1]);
if (pt0 && pt1)
pt.push([pt0.position, pt1.position]);
}
annotations2[name] = pt;
}
bufferedResult.body[i] = { ...newResult.body[i], box, boxRaw, keypoints, annotations: annotations2 };
}
}
if (!bufferedResult.hand || newResult.hand.length !== bufferedResult.hand.length) {
bufferedResult.hand = JSON.parse(JSON.stringify(newResult.hand));
} else {
for (let i = 0; i < newResult.hand.length; i++) {
const box = newResult.hand[i].box.map((b, j) => ((bufferedFactor - 1) * bufferedResult.hand[i].box[j] + b) / bufferedFactor);
const boxRaw = newResult.hand[i].boxRaw.map((b, j) => ((bufferedFactor - 1) * bufferedResult.hand[i].boxRaw[j] + b) / bufferedFactor);
if (bufferedResult.hand[i].keypoints.length !== newResult.hand[i].keypoints.length)
bufferedResult.hand[i].keypoints = newResult.hand[i].keypoints;
const keypoints = newResult.hand[i].keypoints && newResult.hand[i].keypoints.length > 0 ? newResult.hand[i].keypoints.map((landmark, j) => landmark.map((coord, k) => ((bufferedFactor - 1) * (bufferedResult.hand[i].keypoints[j][k] || 1) + (coord || 0)) / bufferedFactor)) : [];
let annotations2 = {};
if (Object.keys(bufferedResult.hand[i].annotations).length !== Object.keys(newResult.hand[i].annotations).length) {
bufferedResult.hand[i].annotations = newResult.hand[i].annotations;
annotations2 = bufferedResult.hand[i].annotations;
} else if (newResult.hand[i].annotations) {
for (const key of Object.keys(newResult.hand[i].annotations)) {
annotations2[key] = ((_f = (_e = (_d = newResult.hand[i]) == null ? void 0 : _d.annotations) == null ? void 0 : _e[key]) == null ? void 0 : _f[0]) ? newResult.hand[i].annotations[key].map((val, j) => val.map((coord, k) => ((bufferedFactor - 1) * bufferedResult.hand[i].annotations[key][j][k] + coord) / bufferedFactor)) : null;
}
}
bufferedResult.hand[i] = { ...newResult.hand[i], box, boxRaw, keypoints, annotations: annotations2 };
}
}
if (!bufferedResult.face || newResult.face.length !== bufferedResult.face.length) {
bufferedResult.face = JSON.parse(JSON.stringify(newResult.face));
} else {
for (let i = 0; i < newResult.face.length; i++) {
const box = newResult.face[i].box.map((b, j) => ((bufferedFactor - 1) * bufferedResult.face[i].box[j] + b) / bufferedFactor);
const boxRaw = newResult.face[i].boxRaw.map((b, j) => ((bufferedFactor - 1) * bufferedResult.face[i].boxRaw[j] + b) / bufferedFactor);
if (newResult.face[i].rotation) {
const rotation = { matrix: [0, 0, 0, 0, 0, 0, 0, 0, 0], angle: { roll: 0, yaw: 0, pitch: 0 }, gaze: { bearing: 0, strength: 0 } };
rotation.matrix = (_g = newResult.face[i].rotation) == null ? void 0 : _g.matrix;
rotation.angle = {
roll: ((bufferedFactor - 1) * (((_i = (_h = bufferedResult.face[i].rotation) == null ? void 0 : _h.angle) == null ? void 0 : _i.roll) || 0) + (((_k = (_j = newResult.face[i].rotation) == null ? void 0 : _j.angle) == null ? void 0 : _k.roll) || 0)) / bufferedFactor,
yaw: ((bufferedFactor - 1) * (((_m = (_l = bufferedResult.face[i].rotation) == null ? void 0 : _l.angle) == null ? void 0 : _m.yaw) || 0) + (((_o = (_n = newResult.face[i].rotation) == null ? void 0 : _n.angle) == null ? void 0 : _o.yaw) || 0)) / bufferedFactor,
pitch: ((bufferedFactor - 1) * (((_q = (_p = bufferedResult.face[i].rotation) == null ? void 0 : _p.angle) == null ? void 0 : _q.pitch) || 0) + (((_s = (_r = newResult.face[i].rotation) == null ? void 0 : _r.angle) == null ? void 0 : _s.pitch) || 0)) / bufferedFactor
};
rotation.gaze = {
bearing: ((bufferedFactor - 1) * (((_t = bufferedResult.face[i].rotation) == null ? void 0 : _t.gaze.bearing) || 0) + (((_u = newResult.face[i].rotation) == null ? void 0 : _u.gaze.bearing) || 0)) / bufferedFactor,
strength: ((bufferedFactor - 1) * (((_v = bufferedResult.face[i].rotation) == null ? void 0 : _v.gaze.strength) || 0) + (((_w = newResult.face[i].rotation) == null ? void 0 : _w.gaze.strength) || 0)) / bufferedFactor
};
bufferedResult.face[i] = { ...newResult.face[i], rotation, box, boxRaw };
} else {
bufferedResult.face[i] = { ...newResult.face[i], box, boxRaw };
}
}
}
if (!bufferedResult.object || newResult.object.length !== bufferedResult.object.length) {
bufferedResult.object = JSON.parse(JSON.stringify(newResult.object));
} else {
for (let i = 0; i < newResult.object.length; i++) {
const box = newResult.object[i].box.map((b, j) => ((bufferedFactor - 1) * bufferedResult.object[i].box[j] + b) / bufferedFactor);
const boxRaw = newResult.object[i].boxRaw.map((b, j) => ((bufferedFactor - 1) * bufferedResult.object[i].boxRaw[j] + b) / bufferedFactor);
bufferedResult.object[i] = { ...newResult.object[i], box, boxRaw };
}
}
if (newResult.persons) {
const newPersons = newResult.persons;
if (!bufferedResult.persons || newPersons.length !== bufferedResult.persons.length) {
bufferedResult.persons = JSON.parse(JSON.stringify(newPersons));
} else {
for (let i = 0; i < newPersons.length; i++) {
bufferedResult.persons[i].box = newPersons[i].box.map((box, j) => ((bufferedFactor - 1) * bufferedResult.persons[i].box[j] + box) / bufferedFactor);
}
}
}
if (newResult.gesture)
bufferedResult.gesture = newResult.gesture;
bufferedResult.width = newResult.width;
bufferedResult.height = newResult.height;
const t1 = now();
interpolateTime = env.perfadd ? interpolateTime + Math.round(t1 - t0) : Math.round(t1 - t0);
if (newResult.performance)
bufferedResult.performance = { ...newResult.performance, interpolate: interpolateTime };
return bufferedResult;
}
// src/segmentation/meet.ts
var tf30 = __toESM(require_tfjs_esm());
var model17;
async function load16(config3) {
if (!model17 || env.initial)
model17 = await loadModel(config3.segmentation.modelPath);
else if (config3.debug)
log("cached model:", model17["modelUrl"]);
return model17;
}
async function predict16(input, config3) {
var _a;
if (!model17)
model17 = await load16(config3);
if (!(model17 == null ? void 0 : model17["executor"]) || !((_a = model17 == null ? void 0 : model17.inputs) == null ? void 0 : _a[0].shape))
return null;
const t2 = {};
t2.resize = tf30.image.resizeBilinear(input, [model17.inputs[0].shape ? model17.inputs[0].shape[1] : 0, model17.inputs[0].shape ? model17.inputs[0].shape[2] : 0], false);
t2.norm = tf30.div(t2.resize, constants.tf255);
t2.res = model17.execute(t2.norm);
t2.squeeze = tf30.squeeze(t2.res, [0]);
[t2.bgRaw, t2.fgRaw] = tf30.unstack(t2.squeeze, 2);
t2.fg = tf30.softmax(t2.fgRaw);
t2.mul = tf30.mul(t2.fg, constants.tf255);
t2.expand = tf30.expandDims(t2.mul, 2);
t2.output = tf30.image.resizeBilinear(t2.expand, [input.shape[1] || 0, input.shape[2] || 0]);
let rgba;
switch (config3.segmentation.mode || "default") {
case "default":
t2.input = tf30.squeeze(input);
t2.concat = tf30.concat([t2.input, t2.output], -1);
rgba = tf30.cast(t2.concat, "int32");
break;
case "alpha":
rgba = tf30.cast(t2.output, "int32");
break;
default:
rgba = tf30.tensor(0);
}
Object.keys(t2).forEach((tensor6) => tf30.dispose(t2[tensor6]));
return rgba;
}
// src/face/match.ts
var match_exports = {};
__export(match_exports, {
distance: () => distance,
find: () => find,
similarity: () => similarity
});
function distance(descriptor1, descriptor2, options4 = { order: 2, multiplier: 25 }) {
if (!descriptor1 || !descriptor1)
return Number.MAX_SAFE_INTEGER;
let sum3 = 0;
for (let i = 0; i < descriptor1.length; i++) {
const diff = !options4.order || options4.order === 2 ? descriptor1[i] - descriptor2[i] : Math.abs(descriptor1[i] - descriptor2[i]);
sum3 += !options4.order || options4.order === 2 ? diff * diff : diff ** options4.order;
}
return (options4.multiplier || 20) * sum3;
}
var normalizeDistance = (dist, order, min2, max5) => {
if (dist === 0)
return 1;
const root = order === 2 ? Math.sqrt(dist) : dist ** (1 / order);
const norm = (1 - root / 100 - min2) / (max5 - min2);
const clamp2 = Math.max(Math.min(norm, 1), 0);
return clamp2;
};
function similarity(descriptor1, descriptor2, options4 = { order: 2, multiplier: 25, min: 0.2, max: 0.8 }) {
const dist = distance(descriptor1, descriptor2, options4);
return normalizeDistance(dist, options4.order || 2, options4.min || 0, options4.max || 1);
}
function find(descriptor, descriptors, options4 = { order: 2, multiplier: 25, threshold: 0, min: 0.2, max: 0.8 }) {
if (!Array.isArray(descriptor) || !Array.isArray(descriptors) || descriptor.length < 64 || descriptors.length === 0) {
return { index: -1, distance: Number.POSITIVE_INFINITY, similarity: 0 };
}
let lowestDistance = Number.MAX_SAFE_INTEGER;
let index2 = -1;
for (let i = 0; i < descriptors.length; i++) {
const res = descriptors[i].length === descriptor.length ? distance(descriptor, descriptors[i], options4) : Number.MAX_SAFE_INTEGER;
if (res < lowestDistance) {
lowestDistance = res;
index2 = i;
}
if (lowestDistance < (options4.threshold || 0))
break;
}
const normalizedSimilarity = normalizeDistance(lowestDistance, options4.order || 2, options4.min || 0, options4.max || 1);
return { index: index2, distance: lowestDistance, similarity: normalizedSimilarity };
}
// src/models.ts
var models_exports2 = {};
__export(models_exports2, {
Models: () => Models,
validateModel: () => validateModel
});
// src/body/movenet.ts
var tf32 = __toESM(require_tfjs_esm());
// src/body/movenetfix.ts
var tf31 = __toESM(require_tfjs_esm());
var maxJitter = 5e-3;
var cache5 = {
keypoints: [],
padding: [[0, 0], [0, 0], [0, 0], [0, 0]]
};
function bodyParts(body4) {
for (const pair of horizontal) {
const left = body4.keypoints.findIndex((kp) => kp.part === pair[0]);
const right = body4.keypoints.findIndex((kp) => kp.part === pair[1]);
if (body4.keypoints[left] && body4.keypoints[right]) {
if (body4.keypoints[left].position[0] < body4.keypoints[right].position[0]) {
const tmp = body4.keypoints[left];
body4.keypoints[left] = body4.keypoints[right];
body4.keypoints[right] = tmp;
}
}
}
for (const pair of vertical) {
const lower = body4.keypoints.findIndex((kp) => kp && kp.part === pair[0]);
const higher = body4.keypoints.findIndex((kp) => kp && kp.part === pair[1]);
if (body4.keypoints[lower] && body4.keypoints[higher]) {
if (body4.keypoints[lower].position[1] < body4.keypoints[higher].position[1]) {
body4.keypoints.splice(lower, 1);
}
}
}
for (const [pair, compare2] of relative) {
const left = body4.keypoints.findIndex((kp) => kp && kp.part === pair[0]);
const right = body4.keypoints.findIndex((kp) => kp && kp.part === pair[1]);
const leftTo = body4.keypoints.findIndex((kp) => kp && kp.part === compare2[0]);
const rightTo = body4.keypoints.findIndex((kp) => kp && kp.part === compare2[1]);
if (!body4.keypoints[leftTo] || !body4.keypoints[rightTo])
continue;
const distanceLeft = body4.keypoints[left] ? [
Math.abs(body4.keypoints[leftTo].position[0] - body4.keypoints[left].position[0]),
Math.abs(body4.keypoints[rightTo].position[0] - body4.keypoints[left].position[0])
] : [0, 0];
const distanceRight = body4.keypoints[right] ? [
Math.abs(body4.keypoints[rightTo].position[0] - body4.keypoints[right].position[0]),
Math.abs(body4.keypoints[leftTo].position[0] - body4.keypoints[right].position[0])
] : [0, 0];
if (distanceLeft[0] > distanceLeft[1] || distanceRight[0] > distanceRight[1]) {
const tmp = body4.keypoints[left];
body4.keypoints[left] = body4.keypoints[right];
body4.keypoints[right] = tmp;
}
}
}
function jitter(keypoints) {
for (let i = 0; i < keypoints.length; i++) {
if (keypoints[i] && cache5.keypoints[i]) {
const diff = [Math.abs(keypoints[i].positionRaw[0] - cache5.keypoints[i].positionRaw[0]), Math.abs(keypoints[i].positionRaw[1] - cache5.keypoints[i].positionRaw[1])];
if (diff[0] < maxJitter && diff[1] < maxJitter) {
keypoints[i] = cache5.keypoints[i];
} else {
cache5.keypoints[i] = keypoints[i];
}
} else {
cache5.keypoints[i] = keypoints[i];
}
}
return keypoints;
}
function padInput(input, inputSize10) {
var _a, _b;
const t2 = {};
if (!((_a = input == null ? void 0 : input.shape) == null ? void 0 : _a[1]) || !((_b = input == null ? void 0 : input.shape) == null ? void 0 : _b[2]))
return input;
cache5.padding = [
[0, 0],
[input.shape[2] > input.shape[1] ? Math.trunc((input.shape[2] - input.shape[1]) / 2) : 0, input.shape[2] > input.shape[1] ? Math.trunc((input.shape[2] - input.shape[1]) / 2) : 0],
[input.shape[1] > input.shape[2] ? Math.trunc((input.shape[1] - input.shape[2]) / 2) : 0, input.shape[1] > input.shape[2] ? Math.trunc((input.shape[1] - input.shape[2]) / 2) : 0],
[0, 0]
];
t2.pad = tf31.pad(input, cache5.padding);
t2.resize = tf31.image.resizeBilinear(t2.pad, [inputSize10, inputSize10]);
const final = tf31.cast(t2.resize, "int32");
Object.keys(t2).forEach((tensor6) => tf31.dispose(t2[tensor6]));
return final;
}
function rescaleBody(body4, outputSize2) {
body4.keypoints = body4.keypoints.filter((kpt4) => kpt4 == null ? void 0 : kpt4.position);
for (const kpt4 of body4.keypoints) {
kpt4.position = [
kpt4.position[0] * (outputSize2[0] + cache5.padding[2][0] + cache5.padding[2][1]) / outputSize2[0] - cache5.padding[2][0],
kpt4.position[1] * (outputSize2[1] + cache5.padding[1][0] + cache5.padding[1][1]) / outputSize2[1] - cache5.padding[1][0]
];
kpt4.positionRaw = [
kpt4.position[0] / outputSize2[0],
kpt4.position[1] / outputSize2[1]
];
}
const rescaledBoxes = calc(body4.keypoints.map((pt) => pt.position), outputSize2);
body4.box = rescaledBoxes.box;
body4.boxRaw = rescaledBoxes.boxRaw;
return body4;
}
// src/body/movenet.ts
var model18;
var inputSize8 = 0;
var skipped14 = Number.MAX_SAFE_INTEGER;
var cache6 = {
boxes: [],
bodies: [],
last: 0
};
async function load17(config3) {
var _a;
if (env.initial)
model18 = null;
if (!model18) {
fakeOps(["size"], config3);
model18 = await loadModel(config3.body.modelPath);
} else if (config3.debug)
log("cached model:", model18["modelUrl"]);
inputSize8 = (model18 == null ? void 0 : model18["executor"]) && ((_a = model18 == null ? void 0 : model18.inputs) == null ? void 0 : _a[0].shape) ? model18.inputs[0].shape[2] : 0;
if (inputSize8 < 64)
inputSize8 = 256;
return model18;
}
function parseSinglePose(res, config3, image28) {
const kpt4 = res[0][0];
const keypoints = [];
let score = 0;
for (let id = 0; id < kpt4.length; id++) {
score = kpt4[id][2];
if (score > config3.body.minConfidence) {
const positionRaw = [kpt4[id][1], kpt4[id][0]];
keypoints.push({
score: Math.round(100 * score) / 100,
part: kpt3[id],
positionRaw,
position: [
Math.round((image28.shape[2] || 0) * positionRaw[0]),
Math.round((image28.shape[1] || 0) * positionRaw[1])
]
});
}
}
score = keypoints.reduce((prev, curr) => curr.score > prev ? curr.score : prev, 0);
const bodies = [];
const newBox = calc(keypoints.map((pt) => pt.position), [image28.shape[2], image28.shape[1]]);
const annotations2 = {};
for (const [name, indexes] of Object.entries(connected3)) {
const pt = [];
for (let i = 0; i < indexes.length - 1; i++) {
const pt0 = keypoints.find((kp) => kp.part === indexes[i]);
const pt1 = keypoints.find((kp) => kp.part === indexes[i + 1]);
if (pt0 && pt1 && pt0.score > (config3.body.minConfidence || 0) && pt1.score > (config3.body.minConfidence || 0))
pt.push([pt0.position, pt1.position]);
}
annotations2[name] = pt;
}
const body4 = { id: 0, score, box: newBox.box, boxRaw: newBox.boxRaw, keypoints, annotations: annotations2 };
bodyParts(body4);
bodies.push(body4);
return bodies;
}
function parseMultiPose(res, config3, image28) {
const bodies = [];
for (let id = 0; id < res[0].length; id++) {
const kpt4 = res[0][id];
const totalScore = Math.round(100 * kpt4[51 + 4]) / 100;
if (totalScore > config3.body.minConfidence) {
const keypoints = [];
for (let i = 0; i < 17; i++) {
const score = kpt4[3 * i + 2];
if (score > config3.body.minConfidence) {
const positionRaw = [kpt4[3 * i + 1], kpt4[3 * i + 0]];
keypoints.push({
part: kpt3[i],
score: Math.round(100 * score) / 100,
positionRaw,
position: [Math.round((image28.shape[2] || 0) * positionRaw[0]), Math.round((image28.shape[1] || 0) * positionRaw[1])]
});
}
}
const newBox = calc(keypoints.map((pt) => pt.position), [image28.shape[2], image28.shape[1]]);
const annotations2 = {};
for (const [name, indexes] of Object.entries(connected3)) {
const pt = [];
for (let i = 0; i < indexes.length - 1; i++) {
const pt0 = keypoints.find((kp) => kp.part === indexes[i]);
const pt1 = keypoints.find((kp) => kp.part === indexes[i + 1]);
if (pt0 && pt1 && pt0.score > (config3.body.minConfidence || 0) && pt1.score > (config3.body.minConfidence || 0))
pt.push([pt0.position, pt1.position]);
}
annotations2[name] = pt;
}
const body4 = { id, score: totalScore, box: newBox.box, boxRaw: newBox.boxRaw, keypoints: [...keypoints], annotations: annotations2 };
bodyParts(body4);
bodies.push(body4);
}
}
bodies.sort((a, b) => b.score - a.score);
if (bodies.length > config3.body.maxDetected)
bodies.length = config3.body.maxDetected;
return bodies;
}
async function predict17(input, config3) {
var _a;
if (!(model18 == null ? void 0 : model18["executor"]) || !((_a = model18 == null ? void 0 : model18.inputs) == null ? void 0 : _a[0].shape))
return [];
if (!config3.skipAllowed)
cache6.boxes.length = 0;
skipped14++;
const skipTime = (config3.body.skipTime || 0) > now() - cache6.last;
const skipFrame = skipped14 < (config3.body.skipFrames || 0);
if (config3.skipAllowed && skipTime && skipFrame) {
return cache6.bodies;
}
return new Promise(async (resolve) => {
const t2 = {};
skipped14 = 0;
t2.input = padInput(input, inputSize8);
t2.res = model18 == null ? void 0 : model18.execute(t2.input);
cache6.last = now();
const res = await t2.res.array();
cache6.bodies = t2.res.shape[2] === 17 ? parseSinglePose(res, config3, input) : parseMultiPose(res, config3, input);
for (const body4 of cache6.bodies) {
rescaleBody(body4, [input.shape[2] || 1, input.shape[1] || 1]);
jitter(body4.keypoints);
}
Object.keys(t2).forEach((tensor6) => tf32.dispose(t2[tensor6]));
resolve(cache6.bodies);
});
}
// src/object/nanodet.ts
var tf33 = __toESM(require_tfjs_esm());
var model19;
var last10 = [];
var lastTime15 = 0;
var skipped15 = Number.MAX_SAFE_INTEGER;
var inputSize9 = 0;
var scaleBox = 2.5;
async function load18(config3) {
if (!model19 || env.initial) {
model19 = await loadModel(config3.object.modelPath);
const inputs = (model19 == null ? void 0 : model19["executor"]) ? Object.values(model19.modelSignature["inputs"]) : void 0;
inputSize9 = Array.isArray(inputs) ? parseInt(inputs[0].tensorShape.dim[2].size) : 416;
} else if (config3.debug)
log("cached model:", model19["modelUrl"]);
return model19;
}
async function process4(res, outputShape, config3) {
var _a, _b;
let id = 0;
let results = [];
const size2 = inputSize9;
for (const strideSize of [1, 2, 4]) {
const baseSize = strideSize * 13;
const scoresT = tf33.squeeze(res.find((a) => a.shape[1] === baseSize ** 2 && (a.shape[2] || 0) === labels2.length));
const scores = await scoresT.array();
const featuresT = tf33.squeeze(res.find((a) => a.shape[1] === baseSize ** 2 && (a.shape[2] || 0) < labels2.length));
const boxesMaxT = tf33.reshape(featuresT, [-1, 4, (((_a = featuresT.shape) == null ? void 0 : _a[1]) || 0) / 4]);
const boxIdxT = tf33.argMax(boxesMaxT, 2);
const boxIdx = await boxIdxT.array();
for (let i = 0; i < scoresT.shape[0]; i++) {
for (let j = 0; j < (((_b = scoresT.shape) == null ? void 0 : _b[1]) || 0); j++) {
const score = scores[i][j];
if (score > (config3.object.minConfidence || 0) && j !== 61) {
const cx = (0.5 + Math.trunc(i % baseSize)) / baseSize;
const cy = (0.5 + Math.trunc(i / baseSize)) / baseSize;
const boxOffset = boxIdx[i].map((a) => a * (baseSize / strideSize / size2));
const [x, y] = [
cx - scaleBox / strideSize * boxOffset[0],
cy - scaleBox / strideSize * boxOffset[1]
];
const [w, h] = [
cx + scaleBox / strideSize * boxOffset[2] - x,
cy + scaleBox / strideSize * boxOffset[3] - y
];
let boxRaw = [x, y, w, h];
boxRaw = boxRaw.map((a) => Math.max(0, Math.min(a, 1)));
const box = [
boxRaw[0] * outputShape[0],
boxRaw[1] * outputShape[1],
boxRaw[2] * outputShape[0],
boxRaw[3] * outputShape[1]
];
const result = {
id: id++,
score: Math.round(100 * score) / 100,
class: j + 1,
label: labels2[j].label,
box: box.map((a) => Math.trunc(a)),
boxRaw
};
results.push(result);
}
}
}
tf33.dispose([scoresT, featuresT, boxesMaxT, boxIdxT]);
}
const nmsBoxes = results.map((a) => [a.boxRaw[1], a.boxRaw[0], a.boxRaw[3], a.boxRaw[2]]);
const nmsScores = results.map((a) => a.score);
let nmsIdx = [];
if (nmsBoxes && nmsBoxes.length > 0) {
const nms = await tf33.image.nonMaxSuppressionAsync(nmsBoxes, nmsScores, config3.object.maxDetected || 0, config3.object.iouThreshold, config3.object.minConfidence);
nmsIdx = Array.from(await nms.data());
tf33.dispose(nms);
}
results = results.filter((_val, idx) => nmsIdx.includes(idx)).sort((a, b) => b.score - a.score);
return results;
}
async function predict18(image28, config3) {
if (!(model19 == null ? void 0 : model19["executor"]))
return [];
const skipTime = (config3.object.skipTime || 0) > now() - lastTime15;
const skipFrame = skipped15 < (config3.object.skipFrames || 0);
if (config3.skipAllowed && skipTime && skipFrame && last10.length > 0) {
skipped15++;
return last10;
}
skipped15 = 0;
if (!env.kernels.includes("mod") || !env.kernels.includes("sparsetodense"))
return last10;
return new Promise(async (resolve) => {
const outputSize2 = [image28.shape[2] || 0, image28.shape[1] || 0];
const resizeT = tf33.image.resizeBilinear(image28, [inputSize9, inputSize9], false);
const normT = tf33.div(resizeT, constants.tf255);
const transposeT = tf33.transpose(normT, [0, 3, 1, 2]);
let objectT;
if (config3.object.enabled)
objectT = model19.execute(transposeT);
lastTime15 = now();
const obj = await process4(objectT, outputSize2, config3);
last10 = obj;
tf33.dispose([resizeT, normT, transposeT, ...objectT]);
resolve(obj);
});
}
// src/body/posenet.ts
var tf34 = __toESM(require_tfjs_esm());
// src/body/posenetutils.ts
var partNames = [
"nose",
"leftEye",
"rightEye",
"leftEar",
"rightEar",
"leftShoulder",
"rightShoulder",
"leftElbow",
"rightElbow",
"leftWrist",
"rightWrist",
"leftHip",
"rightHip",
"leftKnee",
"rightKnee",
"leftAnkle",
"rightAnkle"
];
var count = partNames.length;
var partIds = partNames.reduce((result, jointName, i) => {
result[jointName] = i;
return result;
}, {});
var connectedPartNames = [
["leftHip", "leftShoulder"],
["leftElbow", "leftShoulder"],
["leftElbow", "leftWrist"],
["leftHip", "leftKnee"],
["leftKnee", "leftAnkle"],
["rightHip", "rightShoulder"],
["rightElbow", "rightShoulder"],
["rightElbow", "rightWrist"],
["rightHip", "rightKnee"],
["rightKnee", "rightAnkle"],
["leftShoulder", "rightShoulder"],
["leftHip", "rightHip"]
];
var connectedPartIndices = connectedPartNames.map(([jointNameA, jointNameB]) => [partIds[jointNameA], partIds[jointNameB]]);
var poseChain = [
["nose", "leftEye"],
["leftEye", "leftEar"],
["nose", "rightEye"],
["rightEye", "rightEar"],
["nose", "leftShoulder"],
["leftShoulder", "leftElbow"],
["leftElbow", "leftWrist"],
["leftShoulder", "leftHip"],
["leftHip", "leftKnee"],
["leftKnee", "leftAnkle"],
["nose", "rightShoulder"],
["rightShoulder", "rightElbow"],
["rightElbow", "rightWrist"],
["rightShoulder", "rightHip"],
["rightHip", "rightKnee"],
["rightKnee", "rightAnkle"]
];
function getBoundingBox(keypoints) {
const coord = keypoints.reduce(({ maxX, maxY, minX, minY }, { position: { x, y } }) => ({
maxX: Math.max(maxX, x),
maxY: Math.max(maxY, y),
minX: Math.min(minX, x),
minY: Math.min(minY, y)
}), {
maxX: Number.NEGATIVE_INFINITY,
maxY: Number.NEGATIVE_INFINITY,
minX: Number.POSITIVE_INFINITY,
minY: Number.POSITIVE_INFINITY
});
return [coord.minX, coord.minY, coord.maxX - coord.minX, coord.maxY - coord.minY];
}
function scalePoses(poses, [height, width], [inputResolutionHeight, inputResolutionWidth]) {
const scaleY = height / inputResolutionHeight;
const scaleX = width / inputResolutionWidth;
const scalePose = (pose, i) => ({
id: i,
score: pose.score,
boxRaw: [pose.box[0] / inputResolutionWidth, pose.box[1] / inputResolutionHeight, pose.box[2] / inputResolutionWidth, pose.box[3] / inputResolutionHeight],
box: [Math.trunc(pose.box[0] * scaleX), Math.trunc(pose.box[1] * scaleY), Math.trunc(pose.box[2] * scaleX), Math.trunc(pose.box[3] * scaleY)],
keypoints: pose.keypoints.map(({ score, part, position }) => ({
score,
part,
position: [Math.trunc(position.x * scaleX), Math.trunc(position.y * scaleY)],
positionRaw: [position.x / inputResolutionHeight, position.y / inputResolutionHeight]
})),
annotations: {}
});
const scaledPoses = poses.map((pose, i) => scalePose(pose, i));
return scaledPoses;
}
var MaxHeap = class {
constructor(maxSize2, getElementValue) {
__publicField(this, "priorityQueue");
__publicField(this, "numberOfElements");
__publicField(this, "getElementValue");
this.priorityQueue = new Array(maxSize2);
this.numberOfElements = -1;
this.getElementValue = getElementValue;
}
enqueue(x) {
this.priorityQueue[++this.numberOfElements] = x;
this.swim(this.numberOfElements);
}
dequeue() {
const max5 = this.priorityQueue[0];
this.exchange(0, this.numberOfElements--);
this.sink(0);
this.priorityQueue[this.numberOfElements + 1] = null;
return max5;
}
empty() {
return this.numberOfElements === -1;
}
size() {
return this.numberOfElements + 1;
}
all() {
return this.priorityQueue.slice(0, this.numberOfElements + 1);
}
max() {
return this.priorityQueue[0];
}
swim(k) {
while (k > 0 && this.less(Math.floor(k / 2), k)) {
this.exchange(k, Math.floor(k / 2));
k = Math.floor(k / 2);
}
}
sink(k) {
while (2 * k <= this.numberOfElements) {
let j = 2 * k;
if (j < this.numberOfElements && this.less(j, j + 1))
j++;
if (!this.less(k, j))
break;
this.exchange(k, j);
k = j;
}
}
getValueAt(i) {
return this.getElementValue(this.priorityQueue[i]);
}
less(i, j) {
return this.getValueAt(i) < this.getValueAt(j);
}
exchange(i, j) {
const t2 = this.priorityQueue[i];
this.priorityQueue[i] = this.priorityQueue[j];
this.priorityQueue[j] = t2;
}
};
function getOffsetPoint(y, x, keypoint, offsets) {
return {
y: offsets.get(y, x, keypoint),
x: offsets.get(y, x, keypoint + count)
};
}
function getImageCoords(part, outputStride2, offsets) {
const { heatmapY, heatmapX, id: keypoint } = part;
const { y, x } = getOffsetPoint(heatmapY, heatmapX, keypoint, offsets);
return {
x: part.heatmapX * outputStride2 + x,
y: part.heatmapY * outputStride2 + y
};
}
function clamp(a, min2, max5) {
if (a < min2)
return min2;
if (a > max5)
return max5;
return a;
}
function squaredDistance(y1, x1, y2, x2) {
const dy = y2 - y1;
const dx = x2 - x1;
return dy * dy + dx * dx;
}
function addVectors(a, b) {
return { x: a.x + b.x, y: a.y + b.y };
}
// src/body/posenet.ts
var model20;
var poseNetOutputs = ["MobilenetV1/offset_2/BiasAdd", "MobilenetV1/heatmap_2/BiasAdd", "MobilenetV1/displacement_fwd_2/BiasAdd", "MobilenetV1/displacement_bwd_2/BiasAdd"];
var localMaximumRadius = 1;
var outputStride = 16;
var squaredNmsRadius = 50 ** 2;
function traverse(edgeId, sourceKeypoint, targetId, scores, offsets, displacements, offsetRefineStep = 2) {
const getDisplacement = (point2) => ({
y: displacements.get(point2.y, point2.x, edgeId),
x: displacements.get(point2.y, point2.x, displacements.shape[2] / 2 + edgeId)
});
const getStridedIndexNearPoint = (point2, height2, width2) => ({
y: clamp(Math.round(point2.y / outputStride), 0, height2 - 1),
x: clamp(Math.round(point2.x / outputStride), 0, width2 - 1)
});
const [height, width] = scores.shape;
const sourceKeypointIndices = getStridedIndexNearPoint(sourceKeypoint.position, height, width);
const displacement = getDisplacement(sourceKeypointIndices);
const displacedPoint = addVectors(sourceKeypoint.position, displacement);
let targetKeypoint = displacedPoint;
for (let i = 0; i < offsetRefineStep; i++) {
const targetKeypointIndices = getStridedIndexNearPoint(targetKeypoint, height, width);
const offsetPoint = getOffsetPoint(targetKeypointIndices.y, targetKeypointIndices.x, targetId, offsets);
targetKeypoint = addVectors(
{ x: targetKeypointIndices.x * outputStride, y: targetKeypointIndices.y * outputStride },
{ x: offsetPoint.x, y: offsetPoint.y }
);
}
const targetKeyPointIndices = getStridedIndexNearPoint(targetKeypoint, height, width);
const score = scores.get(targetKeyPointIndices.y, targetKeyPointIndices.x, targetId);
return { position: targetKeypoint, part: partNames[targetId], score };
}
function decodePose(root, scores, offsets, displacementsFwd, displacementsBwd) {
const tuples = poseChain.map(([parentJoinName, childJoinName]) => [partIds[parentJoinName], partIds[childJoinName]]);
const edgesFwd = tuples.map(([, childJointId]) => childJointId);
const edgesBwd = tuples.map(([parentJointId]) => parentJointId);
const numParts = scores.shape[2];
const numEdges = edgesFwd.length;
const keypoints = new Array(numParts);
const rootPoint = getImageCoords(root.part, outputStride, offsets);
keypoints[root.part.id] = {
score: root.score,
part: partNames[root.part.id],
position: rootPoint
};
for (let edge = numEdges - 1; edge >= 0; --edge) {
const sourceId = edgesFwd[edge];
const targetId = edgesBwd[edge];
if (keypoints[sourceId] && !keypoints[targetId]) {
keypoints[targetId] = traverse(edge, keypoints[sourceId], targetId, scores, offsets, displacementsBwd);
}
}
for (let edge = 0; edge < numEdges; ++edge) {
const sourceId = edgesBwd[edge];
const targetId = edgesFwd[edge];
if (keypoints[sourceId] && !keypoints[targetId]) {
keypoints[targetId] = traverse(edge, keypoints[sourceId], targetId, scores, offsets, displacementsFwd);
}
}
return keypoints;
}
function scoreIsMaximumInLocalWindow(keypointId, score, heatmapY, heatmapX, scores) {
const [height, width] = scores.shape;
let localMaximum = true;
const yStart = Math.max(heatmapY - localMaximumRadius, 0);
const yEnd = Math.min(heatmapY + localMaximumRadius + 1, height);
for (let yCurrent = yStart; yCurrent < yEnd; ++yCurrent) {
const xStart = Math.max(heatmapX - localMaximumRadius, 0);
const xEnd = Math.min(heatmapX + localMaximumRadius + 1, width);
for (let xCurrent = xStart; xCurrent < xEnd; ++xCurrent) {
if (scores.get(yCurrent, xCurrent, keypointId) > score) {
localMaximum = false;
break;
}
}
if (!localMaximum)
break;
}
return localMaximum;
}
function buildPartWithScoreQueue(minConfidence2, scores) {
const [height, width, numKeypoints] = scores.shape;
const queue = new MaxHeap(height * width * numKeypoints, ({ score }) => score);
for (let heatmapY = 0; heatmapY < height; ++heatmapY) {
for (let heatmapX = 0; heatmapX < width; ++heatmapX) {
for (let keypointId = 0; keypointId < numKeypoints; ++keypointId) {
const score = scores.get(heatmapY, heatmapX, keypointId);
if (score < minConfidence2)
continue;
if (scoreIsMaximumInLocalWindow(keypointId, score, heatmapY, heatmapX, scores))
queue.enqueue({ score, part: { heatmapY, heatmapX, id: keypointId } });
}
}
}
return queue;
}
function withinRadius(poses, { x, y }, keypointId) {
return poses.some(({ keypoints }) => {
var _a;
const correspondingKeypoint = (_a = keypoints[keypointId]) == null ? void 0 : _a.position;
if (!correspondingKeypoint)
return false;
return squaredDistance(y, x, correspondingKeypoint.y, correspondingKeypoint.x) <= squaredNmsRadius;
});
}
function getInstanceScore(existingPoses, keypoints) {
const notOverlappedKeypointScores = keypoints.reduce((result, { position, score }, keypointId) => {
if (!withinRadius(existingPoses, position, keypointId))
result += score;
return result;
}, 0);
return notOverlappedKeypointScores / keypoints.length;
}
function decode(offsets, scores, displacementsFwd, displacementsBwd, maxDetected, minConfidence2) {
const poses = [];
const queue = buildPartWithScoreQueue(minConfidence2, scores);
while (poses.length < maxDetected && !queue.empty()) {
const root = queue.dequeue();
const rootImageCoords = getImageCoords(root.part, outputStride, offsets);
if (withinRadius(poses, rootImageCoords, root.part.id))
continue;
let keypoints = decodePose(root, scores, offsets, displacementsFwd, displacementsBwd);
keypoints = keypoints.filter((a) => a.score > minConfidence2);
const score = getInstanceScore(poses, keypoints);
const box = getBoundingBox(keypoints);
if (score > minConfidence2)
poses.push({ keypoints, box, score: Math.round(100 * score) / 100 });
}
return poses;
}
async function predict19(input, config3) {
if (!(model20 == null ? void 0 : model20["executor"]))
return [];
const res = tf34.tidy(() => {
if (!model20.inputs[0].shape)
return [];
const resized = tf34.image.resizeBilinear(input, [model20.inputs[0].shape[2], model20.inputs[0].shape[1]]);
const normalized = tf34.sub(tf34.div(tf34.cast(resized, "float32"), 127.5), 1);
const results = model20.execute(normalized, poseNetOutputs);
const results3d = results.map((y) => tf34.squeeze(y, [0]));
results3d[1] = tf34.sigmoid(results3d[1]);
return results3d;
});
const buffers = await Promise.all(res.map((tensor6) => tensor6.buffer()));
for (const t2 of res)
tf34.dispose(t2);
const decoded = decode(buffers[0], buffers[1], buffers[2], buffers[3], config3.body.maxDetected, config3.body.minConfidence);
if (!model20.inputs[0].shape)
return [];
const scaled = scalePoses(decoded, [input.shape[1], input.shape[2]], [model20.inputs[0].shape[2], model20.inputs[0].shape[1]]);
return scaled;
}
async function load19(config3) {
if (!model20 || env.initial)
model20 = await loadModel(config3.body.modelPath);
else if (config3.debug)
log("cached model:", model20["modelUrl"]);
return model20;
}
// src/segmentation/rvm.ts
var tf35 = __toESM(require_tfjs_esm());
var model21;
var outputNodes2 = ["fgr", "pha", "r1o", "r2o", "r3o", "r4o"];
var t = {};
var ratio = 0;
function init3(config3) {
tf35.dispose([t.r1i, t.r2i, t.r3i, t.r4i, t.downsample_ratio]);
t.r1i = tf35.tensor(0);
t.r2i = tf35.tensor(0);
t.r3i = tf35.tensor(0);
t.r4i = tf35.tensor(0);
ratio = config3.segmentation.ratio || 0.5;
t.downsample_ratio = tf35.tensor(ratio);
}
async function load20(config3) {
if (!model21 || env.initial)
model21 = await loadModel(config3.segmentation.modelPath);
else if (config3.debug)
log("cached model:", model21["modelUrl"]);
init3(config3);
return model21;
}
var normalize = (r) => tf35.tidy(() => {
const squeeze14 = tf35.squeeze(r, [0]);
const mul15 = tf35.mul(squeeze14, constants.tf255);
const cast8 = tf35.cast(mul15, "int32");
return cast8;
});
function getRGBA(fgr, pha) {
const rgb2 = fgr ? normalize(fgr) : tf35.fill([pha.shape[1] || 0, pha.shape[2] || 0, 3], 255, "int32");
const a = pha ? normalize(pha) : tf35.fill([fgr.shape[1] || 0, fgr.shape[2] || 0, 1], 255, "int32");
const rgba = tf35.concat([rgb2, a], -1);
tf35.dispose([rgb2, a]);
return rgba;
}
function getState(state) {
return tf35.tidy(() => {
const r = {};
r.unstack = tf35.unstack(state, -1);
r.concat = tf35.concat(r.unstack, 1);
r.split = tf35.split(r.concat, 4, 1);
r.stack = tf35.concat(r.split, 2);
r.squeeze = tf35.squeeze(r.stack, [0]);
r.expand = tf35.expandDims(r.squeeze, -1);
r.add = tf35.add(r.expand, 1);
r.mul = tf35.mul(r.add, 127.5);
r.cast = tf35.cast(r.mul, "int32");
r.tile = tf35.tile(r.cast, [1, 1, 3]);
r.alpha = tf35.fill([r.tile.shape[0] || 0, r.tile.shape[1] || 0, 1], 255, "int32");
return tf35.concat([r.tile, r.alpha], -1);
});
}
async function predict20(input, config3) {
if (!model21)
model21 = await load20(config3);
if (!(model21 == null ? void 0 : model21["executor"]))
return null;
t.src = tf35.div(input, 255);
if (ratio !== config3.segmentation.ratio)
init3(config3);
const [fgr, pha, r1o, r2o, r3o, r4o] = await model21.executeAsync(t, outputNodes2);
let rgba;
switch (config3.segmentation.mode || "default") {
case "default":
rgba = getRGBA(fgr, pha);
break;
case "alpha":
rgba = getRGBA(null, pha);
break;
case "foreground":
rgba = getRGBA(fgr, null);
break;
case "state":
rgba = getState(r1o);
break;
default:
rgba = tf35.tensor(0);
}
tf35.dispose([t.src, fgr, pha, t.r1i, t.r2i, t.r3i, t.r4i]);
[t.r1i, t.r2i, t.r3i, t.r4i] = [r1o, r2o, r3o, r4o];
return rgba;
}
// src/segmentation/selfie.ts
var tf36 = __toESM(require_tfjs_esm());
var model22;
async function load21(config3) {
if (!model22 || env.initial)
model22 = await loadModel(config3.segmentation.modelPath);
else if (config3.debug)
log("cached model:", model22["modelUrl"]);
return model22;
}
async function predict21(input, config3) {
var _a;
if (!model22)
model22 = await load21(config3);
if (!(model22 == null ? void 0 : model22["executor"]) || !((_a = model22 == null ? void 0 : model22.inputs) == null ? void 0 : _a[0].shape))
return null;
const t2 = {};
t2.resize = tf36.image.resizeBilinear(input, [model22.inputs[0].shape ? model22.inputs[0].shape[1] : 0, model22.inputs[0].shape ? model22.inputs[0].shape[2] : 0], false);
t2.norm = tf36.div(t2.resize, constants.tf255);
t2.res = model22.execute(t2.norm);
t2.squeeze = tf36.squeeze(t2.res, [0]);
t2.alpha = tf36.image.resizeBilinear(t2.squeeze, [input.shape[1] || 0, input.shape[2] || 0]);
t2.mul = tf36.mul(t2.alpha, constants.tf255);
let rgba;
switch (config3.segmentation.mode || "default") {
case "default":
t2.input = tf36.squeeze(input);
t2.concat = tf36.concat([t2.input, t2.mul], -1);
rgba = tf36.cast(t2.concat, "int32");
break;
case "alpha":
rgba = tf36.cast(t2.mul, "int32");
break;
default:
rgba = tf36.tensor(0);
}
Object.keys(t2).forEach((tensor6) => tf36.dispose(t2[tensor6]));
return rgba;
}
// src/models.ts
function validateModel(instance, model23, name) {
var _a, _b;
if (!model23)
return null;
if (!((_a = instance == null ? void 0 : instance.config) == null ? void 0 : _a.validateModels))
return null;
const simpleOps = ["const", "placeholder", "noop", "pad", "squeeze", "add", "sub", "mul", "div"];
const ignoreOps = ["biasadd", "fusedbatchnormv3", "matmul", "switch", "shape", "merge", "split", "broadcastto"];
const ops = [];
const missing = [];
const url = model23["modelUrl"];
const executor = model23["executor"];
if ((_b = executor == null ? void 0 : executor.graph) == null ? void 0 : _b.nodes) {
for (const kernel of Object.values(executor.graph.nodes)) {
const op = kernel.op.toLowerCase();
if (!ops.includes(op))
ops.push(op);
}
} else {
if (!executor && instance.config.debug) {
log("model not loaded", name);
}
}
for (const op of ops) {
if (!simpleOps.includes(op) && !ignoreOps.includes(op) && !instance.env.kernels.includes(op) && !instance.env.kernels.includes(op.replace("_", "")) && !instance.env.kernels.includes(op.replace("native", "")) && !instance.env.kernels.includes(op.replace("v2", ""))) {
missing.push(op);
}
}
if (instance.config.debug && missing.length > 0)
log("model validation failed:", name, missing);
return missing.length > 0 ? { name, missing, ops, url } : null;
}
var Models = class {
constructor(currentInstance) {
__publicField(this, "instance");
__publicField(this, "models", {});
this.models = {};
this.instance = currentInstance;
}
stats() {
let totalSizeFromManifest = 0;
let totalSizeWeights = 0;
let totalSizeLoading = 0;
for (const m of Object.values(modelStats)) {
totalSizeFromManifest += m.sizeFromManifest;
totalSizeWeights += m.sizeLoadedWeights;
totalSizeLoading += m.sizeDesired;
}
const percentageLoaded = totalSizeLoading > 0 ? totalSizeWeights / totalSizeLoading : 0;
return {
numLoadedModels: Object.values(modelStats).length,
numDefinedModels: Object.keys(this.models).length,
percentageLoaded,
totalSizeFromManifest,
totalSizeWeights,
totalSizeLoading,
modelStats: Object.values(modelStats)
};
}
reset() {
for (const model23 of Object.keys(this.models))
this.models[model23] = null;
}
async load(instance) {
var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l, _m, _n, _o, _p, _q, _r, _s, _t, _u, _v, _w, _x, _y, _z, _A;
if (env.initial)
this.reset();
if (instance)
this.instance = instance;
const m = {};
m.blazeface = this.instance.config.face.enabled && !this.models.blazeface ? load3(this.instance.config) : null;
m.antispoof = this.instance.config.face.enabled && ((_a = this.instance.config.face.antispoof) == null ? void 0 : _a.enabled) && !this.models.antispoof ? load8(this.instance.config) : null;
m.liveness = this.instance.config.face.enabled && ((_b = this.instance.config.face.liveness) == null ? void 0 : _b.enabled) && !this.models.liveness ? load9(this.instance.config) : null;
m.faceres = this.instance.config.face.enabled && ((_c = this.instance.config.face.description) == null ? void 0 : _c.enabled) && !this.models.faceres ? load7(this.instance.config) : null;
m.emotion = this.instance.config.face.enabled && ((_d = this.instance.config.face.emotion) == null ? void 0 : _d.enabled) && !this.models.emotion ? load6(this.instance.config) : null;
m.iris = this.instance.config.face.enabled && ((_e = this.instance.config.face.iris) == null ? void 0 : _e.enabled) && !((_f = this.instance.config.face.attention) == null ? void 0 : _f.enabled) && !this.models.iris ? load4(this.instance.config) : null;
m.facemesh = this.instance.config.face.enabled && ((_g = this.instance.config.face.mesh) == null ? void 0 : _g.enabled) && !this.models.facemesh ? load5(this.instance.config) : null;
m.gear = this.instance.config.face.enabled && ((_h = this.instance.config.face["gear"]) == null ? void 0 : _h.enabled) && !this.models.gear ? load10(this.instance.config) : null;
m.ssrnetage = this.instance.config.face.enabled && ((_i = this.instance.config.face["ssrnet"]) == null ? void 0 : _i.enabled) && !this.models.ssrnetage ? load11(this.instance.config) : null;
m.ssrnetgender = this.instance.config.face.enabled && ((_j = this.instance.config.face["ssrnet"]) == null ? void 0 : _j.enabled) && !this.models.ssrnetgender ? load12(this.instance.config) : null;
m.mobilefacenet = this.instance.config.face.enabled && ((_k = this.instance.config.face["mobilefacenet"]) == null ? void 0 : _k.enabled) && !this.models.mobilefacenet ? load13(this.instance.config) : null;
m.insightface = this.instance.config.face.enabled && ((_l = this.instance.config.face["insightface"]) == null ? void 0 : _l.enabled) && !this.models.insightface ? load14(this.instance.config) : null;
m.blazepose = this.instance.config.body.enabled && !this.models.blazepose && ((_m = this.instance.config.body.modelPath) == null ? void 0 : _m.includes("blazepose")) ? loadPose(this.instance.config) : null;
m.blazeposedetect = this.instance.config.body.enabled && !this.models.blazeposedetect && this.instance.config.body["detector"] && this.instance.config.body["detector"].modelPath ? loadDetect(this.instance.config) : null;
m.efficientpose = this.instance.config.body.enabled && !this.models.efficientpose && ((_n = this.instance.config.body.modelPath) == null ? void 0 : _n.includes("efficientpose")) ? load2(this.instance.config) : null;
m.movenet = this.instance.config.body.enabled && !this.models.movenet && ((_o = this.instance.config.body.modelPath) == null ? void 0 : _o.includes("movenet")) ? load17(this.instance.config) : null;
m.posenet = this.instance.config.body.enabled && !this.models.posenet && ((_p = this.instance.config.body.modelPath) == null ? void 0 : _p.includes("posenet")) ? load19(this.instance.config) : null;
m.handtrack = this.instance.config.hand.enabled && !this.models.handtrack && ((_r = (_q = this.instance.config.hand.detector) == null ? void 0 : _q.modelPath) == null ? void 0 : _r.includes("handtrack")) ? loadDetect2(this.instance.config) : null;
m.handskeleton = this.instance.config.hand.enabled && this.instance.config.hand.landmarks && !this.models.handskeleton && ((_t = (_s = this.instance.config.hand.detector) == null ? void 0 : _s.modelPath) == null ? void 0 : _t.includes("handtrack")) ? loadSkeleton(this.instance.config) : null;
if ((_v = (_u = this.instance.config.hand.detector) == null ? void 0 : _u.modelPath) == null ? void 0 : _v.includes("handdetect"))
[m.handpose, m.handskeleton] = !this.models.handpose ? await load15(this.instance.config) : [null, null];
m.centernet = this.instance.config.object.enabled && !this.models.centernet && ((_w = this.instance.config.object.modelPath) == null ? void 0 : _w.includes("centernet")) ? load(this.instance.config) : null;
m.nanodet = this.instance.config.object.enabled && !this.models.nanodet && ((_x = this.instance.config.object.modelPath) == null ? void 0 : _x.includes("nanodet")) ? load18(this.instance.config) : null;
m.selfie = this.instance.config.segmentation.enabled && !this.models.selfie && ((_y = this.instance.config.segmentation.modelPath) == null ? void 0 : _y.includes("selfie")) ? load21(this.instance.config) : null;
m.meet = this.instance.config.segmentation.enabled && !this.models.meet && ((_z = this.instance.config.segmentation.modelPath) == null ? void 0 : _z.includes("meet")) ? load16(this.instance.config) : null;
m.rvm = this.instance.config.segmentation.enabled && !this.models.rvm && ((_A = this.instance.config.segmentation.modelPath) == null ? void 0 : _A.includes("rvm")) ? load20(this.instance.config) : null;
await Promise.all([...Object.values(m)]);
for (const model23 of Object.keys(m))
this.models[model23] = m[model23] || this.models[model23] || null;
}
list() {
const models3 = Object.keys(this.models).map((model23) => {
var _a;
return { name: model23, loaded: this.models[model23] !== null, size: 0, url: this.models[model23] ? (_a = this.models[model23]) == null ? void 0 : _a["modelUrl"] : null };
});
for (const m of models3) {
const stats = Object.keys(modelStats).find((s) => s.startsWith(m.name));
if (!stats)
continue;
m.size = modelStats[stats].sizeLoadedWeights;
m.url = modelStats[stats].url;
}
return models3;
}
loaded() {
const list = this.list();
const loaded = list.filter((model23) => model23.loaded).map((model23) => model23.name);
return loaded;
}
validate() {
const missing = [];
for (const defined of Object.keys(this.models)) {
const model23 = this.models[defined];
if (!model23)
continue;
const res = validateModel(this.instance, model23, defined);
if (res)
missing.push(res);
}
return missing;
}
};
// src/util/persons.ts
function join2(faces, bodies, hands, gestures, shape) {
var _a, _b, _c, _d, _e, _f;
let id = 0;
const persons = [];
for (const face4 of faces) {
const person2 = { id: id++, face: face4, body: null, hands: { left: null, right: null }, gestures: [], box: [0, 0, 0, 0] };
for (const body4 of bodies) {
if (face4.box[0] > body4.box[0] && face4.box[0] < body4.box[0] + body4.box[2] && face4.box[1] + face4.box[3] > body4.box[1] && face4.box[1] + face4.box[3] < body4.box[1] + body4.box[3]) {
person2.body = body4;
}
}
if (person2.body) {
for (const hand3 of hands) {
if (hand3.box[0] + hand3.box[2] > person2.body.box[0] && hand3.box[0] + hand3.box[2] < person2.body.box[0] + person2.body.box[2] && hand3.box[1] + hand3.box[3] > person2.body.box[1] && hand3.box[1] + hand3.box[3] < person2.body.box[1] + person2.body.box[3]) {
if (person2.hands)
person2.hands.left = hand3;
}
if (hand3.box[0] < person2.body.box[0] + person2.body.box[2] && hand3.box[0] > person2.body.box[0] && hand3.box[1] + hand3.box[3] > person2.body.box[1] && hand3.box[1] + hand3.box[3] < person2.body.box[1] + person2.body.box[3]) {
if (person2.hands)
person2.hands.right = hand3;
}
}
}
for (const gesture2 of gestures) {
if (gesture2["face"] !== void 0 && gesture2["face"] === face4.id)
person2.gestures.push(gesture2);
else if (gesture2["iris"] !== void 0 && gesture2["iris"] === face4.id)
person2.gestures.push(gesture2);
else if (gesture2["body"] !== void 0 && gesture2["body"] === ((_a = person2.body) == null ? void 0 : _a.id))
person2.gestures.push(gesture2);
else if (gesture2["hand"] !== void 0 && gesture2["hand"] === ((_b = person2.hands.left) == null ? void 0 : _b.id))
person2.gestures.push(gesture2);
else if (gesture2["hand"] !== void 0 && gesture2["hand"] === ((_c = person2.hands.right) == null ? void 0 : _c.id))
person2.gestures.push(gesture2);
}
const x = [];
const y = [];
const extractXY = (box) => {
if (box && box.length === 4) {
x.push(box[0], box[0] + box[2]);
y.push(box[1], box[1] + box[3]);
}
};
extractXY(person2.face.box);
extractXY((_d = person2.body) == null ? void 0 : _d.box);
extractXY((_e = person2.hands.left) == null ? void 0 : _e.box);
extractXY((_f = person2.hands.right) == null ? void 0 : _f.box);
const minX = Math.min(...x);
const minY = Math.min(...y);
person2.box = [minX, minY, Math.max(...x) - minX, Math.max(...y) - minY];
if ((shape == null ? void 0 : shape[1]) && (shape == null ? void 0 : shape[2]))
person2.boxRaw = [person2.box[0] / shape[2], person2.box[1] / shape[1], person2.box[2] / shape[2], person2.box[3] / shape[1]];
persons.push(person2);
}
return persons;
}
// src/warmup.ts
var tf37 = __toESM(require_tfjs_esm());
// src/sample.ts
var face3 = `
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`;
var body3 = `
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
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAa2p1UYo5adbz+xbxMlvqJ1OqjDHLXree0ejmzNrWm953tPmTPWbWneZ7yoy5YhjrXXTjH8s75N
mtkyxt0VZM2/m175N1V03yTKubMLXVXybeYLLX2VXy7eam+b0bOg4VquJW+rry4/O9uyZOq3UjVm
9r25axMzPaIdvhns1kzbZddM0p5Y47z8/R2+HcF03Doi1a8+Xzvbv+TotJnjDXkt+K8ODHp8cY8N
IpSO0RCwF2YAAAAAAAAACvUZYw6fJkntWN3k8dfHz2vLucdz8mkjFE9bz1+UOZosX1UzPm0nqI/W
MYo9FlcPNklfFGeH/NshLGun+Cz6PtHZtVZWlRLS+jxPkRpIn7rdoupHTdA5s6SI+7H6Mfo+32Y2
+To3neSIiZ7A0IjPXpXLePlMotGW3272t85datKzHZjbTVnsDj+FG/2Y/RlGP4R+jo20u7H6N1Ql
o+H8I/REY957R+jpfReiK6eOYHLtj2tttH6KrY/6Y/R2c+kjeJiFVtLG24hxpw7/AHY/RRkw9O37
O99Hrt1YX0tfOBLjcGp4XF8c+u8fs9c4dcVcGemSI61nd3IneN1orQAAAAAAAAAAAAABFqxes1tE
TE9JiUgPKcX4RbRXnNgiZwWnrH4XPi28PdXpW9JraImsxtMS8pxXhF9DecuGJtgmf+1TWW2N/la1
L7N7T5e3Vy6W3hsYcvLbqzbO9jvvCzvDR0+XeO7crO6FmGSvRThy/RtVXJ92elvk2rRvDUzU7pl4
izsd2J3jeBpcNz+Lg5LT7+Pp+Xk3W7js5eAAgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADs0NTrN96Yp6edkW8Wzm6+LNTq4pvTHO9vOfRoWtt
1mes95YWvs1s2fZldddOczLPLn2ju0MmebT3YZc2/mpm3qqllN1drsbZIhr3yzvtHf4AsvlYYseb
V5Yx4KTe0+UQ6nDvZ3UazbJqd8OKeu33peq0eh0+hxcmnxxWPOfOfm0mP+steT/ji8N9mKY9suum
L37+HHaPm9DSlaVitKxWsdohI0Y22gAgAAAAAAAAAABXnyRhw3yT92Nwef4xm8bVzET0rPJH5d12
CvLhho3rN9RWs9Z23n5y6O21YhrVYbdGOCfrrLPJRpv863zVS6FS09SvZj3lVZZRdPSqmnSWdrIE
ebOkK4ldTsgW1WKqd1oMZhEVZyRAImOjGI6rJ7IiATNd46qL02bHkiaxaoNGY2n4ImPgtyV2n0Vo
Gvlx7x2beiyTk08RPevSVUxux00+Fn2n7N+n5rRFb4AAAAAAAAAAAAAAACLVres1tETWekxKQHlu
L8InR2nPp43wz3j8P/s5dLveWrFqzW0bxPeJeV4xwmdFec+CJnDM9Y/CrY1xv8qvTZ+WYdbDk5oh
5zHk283U0eo3jaZZ2N5XYjrCnLSJhOK+8d1kxvCqzSwZvousrb7k9LfJ3nB1OLeJdLhufx9LEWn3
6e7LXN9Ofy5/W4AuxAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAETaKxMzO0Qi9646Ta07RDmZ9VbPbaOlI7Qi3i+c3TPUaqcu9adKfy0722ZXvFa9
XO1OrjrESxt66ZJmcjPUanlidmhkzTZVfLN5VWvsC2b7R3U3yqrZZtO1esz2h2+F+zWTUcuXXTNM
feKR3n5+iZLVbqRzNJo9TxHLyaekz62ntD1fDOA6fQbZL7Zc/wCKY6R8odLBgxabFGPDSKUjyiFj
SZkYa3aALKAAAAAAAAAAAAAADQ4pl2pTFH3p3n5Q33E12Tn1eSfKscsLZ+orS00eJqbW+Lfnu1tF
XaJnZsz3WpCfsyp00fWSvmPdVYOmSUDd8kR3InoQosy7JmUX7MdwZ17ro7KKT1XRPRAsrO0rYndr
79V1ZBaQiJ6JgCSIJASwrO07MpV2nqBlrv1a1o2bf2qtfLXaQUTO0sb05o3jv3ZXhjS20xEphW5h
yeJjjf7UdJWNKLziyRePsz0lux1SgAQAAAAAAAAAAAAAADG9K5KTS8Rato2mJZAPIcU4ZbQZuekT
OC3afT4NXFkmlntc2GmoxWx5K71tG0vHa/RX0GpmlutJ61t6wrY2xr8dXS5uesN+tt4ef0eaa223
2dnHk3juyreM81OaFGiy/RtZET9jJ7s/2bdutd2jqKeic3iNTsd8a2h1H0jTVtP2o6W+bZbOO+gA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABje9cdJt
adohGTLXFTmvO0fy52bJfU23t0pHaqLeL5xdK9Rnvqb+cUjtCi94xxvK3JetKuHrdZvaa1ljb10y
cnIs1Wt3naJc++TmVWvMz1YWybfMGdsm3eWek0mo4jm8PT0mfW3lDf4V7P5tdMZdRviwfvZ6/TaX
DpMMYsFIpWPTzXmf+steT8jn8L4Dp+HxF77Zc/4pjpHydYGjC3oAAAAAAAAAAAAAAAAADG9opS1p
7RG7zszN6WtPe0zLua+3Joss/wBOzhzG2OsL5+IrY09dsSyYRijbHEMvOChb7KjF0yS2LQ169Mso
S24noyrPVXWejNVKbTuw3T3REdQWU6LYlVvsyiUDPfqupPRr79VuOQX1lZEqoZxIMksd0gT2VT0l
bPZVbuCaW8i8bwr32WxbcGnkjaZa9p2ndv5qbw5+aNugLItF6TEtvTX5sMb969HMpfazc0d9stqe
vVZDdAQAAAAAAAAAAAAAAAADV1+iprtPOO/2u9bektoB4TJTJpNRbHkja1Z6uto8viVht+0HDvpG
H6Tjj6zHHvbecONw7Ltfkmeqmo6Ma69DXbbZTkr1mGWO3RneOaGbZRoM30fVzSelMnT83aef1FZ7
x3h1tBqfpGnjmn369LNc3sc3kzy9bQCzIAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAa+q1dNNXr7157VhGp1Xh70x+9f9ocy283m1p5rz3mVbrjXHjt91lz
5c9+fJ1nyjyhdM8lZlOOIiqrUXikd+kMreunnI5XEdX4dZiZcG+XmtNl/F83PeeWWHDOGanieSKY
q+5H2rz2hMzWd1Iqx1yajJXHhrNrW6REeb1nCPZumn2z62Ivl7xTyr/6uhwzhGn4Zj2xxzZJ+1kn
vLoNJnjHW7TbbsAszAAAAAAAAAAAAAAAAAAAAaPFrbaSK/itEOXt0rDf4xb/ACa/GZacRvaF58Q2
IjasQnzPIhCU92tMbZGzHmotG10C6nZkwpPRmipIllEbMIZIE7solgmJBnCyk9VMM6z1BtVllEqK
z0WRILYlluriWcSDJVbusV27gwInaSWM9ECyZ3hqamnSWxFmOSOaqRx725bNnSZNs9J+OynVY+WZ
YYr7TE+nVaIr0Ais81Yn1hKAAAAAAAAAAAAAAAAAABExvG09peU4nov9n66L0j6q/WPg9Y1OJaON
ZpL0+9HWs/EWzeVz9PbmrEtnyc3h9reHy26TWdnSr2YX6657ijLXpLX0+onSamL/AHJ6W+Tbv2aW
ekTv16JzeI1Ox6KJiYiY7Slz+E6jxdN4dp3vj6fl5Og2clnKACAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACZ2jeQRMxEbzO0Q08uqtkma4ulfO3r8lefUePMxWf
cjy9WvlzVxV6T1Z61/x0Y8f7Wc7Ur1lqVy+LqOWJ2hp6rXddon5rOF1tfmz5OkT0qzb8dWbxjp1c
biuuilJ5Z6r+IcQrixzEy8zl1E6rNt1tMztFY81sztU1eRucN4ffi2p5esRM72n0h7rS6XFo8FcO
CkVpX082nwXh3+z9FWLxHi36328vg6TZyW9ABAAAAAAAAAAAAAAAAAAAAAADj8Unm1tK/hqppHvw
y1k8/EMk+m0GOPeafiFpCZYwolnXspvHvLa9mF46gmnZmwozRUiUCBKYYsoBLOFbKAX0llEqqyzi
QXRLOJVRLOOwLIljZMEgrlhKyYYTAK5nZPN0RZjugUanHzVlz6xtLq361c+9eXItPpXX0dubTU+E
bL2lw2++O1fSW6m/VYAISAAAAAAAAAAAAAAAAAp1GbwcfTreelYEydcuMcRrM/L9nnlsV6wqpi2r
tv133mfWVkRyRtEdGFva7MzkYZNoamWN4bV4mYa9qztKIujhVppxGI8r1mJegeZpknBqKZY+7L0t
LRekWrO8TG8Ns/HJ5ZypAWZAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAADS12fp4VJ6z9qVuq1HgUiI+3bpDl589cOKZmevqprXPTbx477rDJlrhr1nq4+s182tMRP
RqaziXiZJrWekNG17ZbxWJ336M5LXRbI3dLTJrs07RMY6fan1dHLrowY+X7MVjt6N3R6Kul0EbWm
s7bz8Z+LnabQX43r7Y53php/mXj+Dnv0f1JO1x/8ZxbUzj02O15mfLtD13AvZqnDds+pmMmo26el
XX0Wh0/D8EYtNjilY7+s/NstpOOTW7QBKgAAAAAAAAAAAAAAAAAAAAAADG88tLW9I3BwJtz6nNf1
vK/DHVqYJ3pzT5y3MPZeojOWMQylEKpTVjZnDCwkqzYQyRRICATCITAJZQxhMAshnEq4ZQC2srKq
qrIBZCWNZZgwswmFloVyCu0dFcx1WyrtCBhv5NTPHXds2U5o3hIz4ffbPt+KHUcTSW5c9Jme0u2v
VYAKpAAAAAAAAAAAAAAAAYZctcVOa35R6tLrltN795/YvknNqrfhpPLH92V5isd9mWq6fHjk6rn0
ZxG8KK5Jm/wbVZiYZtqrmkqL023bkxvCiY3lJHNyRG81mHS4Rn5sNsNp64+3yaWaNrzOzHBl+i6q
mT7s9J+S+ay8mex6EIneN47SNXKAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAImYiJme0JafEs3h6fkidrZOn5eaLeJk7eOdm1Hi2vmtPTry/CHmOJcUvmvOPF1n09Pm
6HF9ZGm01qxO3R5vSY7XwzmzTy47zzTEd7en5Mfvt2/PURWdo3tvPrPlKymbktFqTtMTvHzbOLDG
f63JXbFX7FdnoODcDprZpq9TjiMMTvSn4vj8l5fxnrk91saPSa7i2hpOfbTVt5x1m0fLydzR6PDo
dPGHBXasd585n1lsRERG0dIF5OOe6tAEqgAAAAAAAAAAAAAAAAAAAAAAADX11+TRZrf0y2Gjxe22
gtH4piP3TPpXKwxtjhuYo9xq442iIblI2pC1RET2ILd9kxCqRjZmwlCSEohIJAQAAJZISDKGUd2M
MoBnVbVVCyAWVWeSuqyOwIlXZZKue4MJV2WWYT2QKbKL9YlfdRdIo35b7/Hd3KTzUrPrDh27uxpb
c2mpPwX/ABX9XAKpAAAAAAAAAAAAAACekTIp1eTwtJmv+GkyJn1oafeazbfpMzLR4jq/o8b823zX
6XNF8ERCvTcNpxLV5LauvPhx9Irv3lhztdtv8TtaWLicXrt03jzjzb2k1nid56ty3s/w+a7Uwzjn
1raejlarhmbhl/FpbxMO/fzj5p/ixSeXOvTtRfeI280ZI26tfDm3pWe63LaZx7qtGvniJ6tPLvOK
fOa9WzbJvTbza02jl3n5SSljscK1MajSxWZ96nSW88xw/VfQ9XMT9nfa3yemid43jtLeXsce88qQ
EqAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADia3UTm1l4j7OP3Y/u
7Vp2rM+kPJW1PhYcmS0+9MzKm/jbwz31weMzbV8UppazPL9q0/BF4rk1GLDSNqxPWPhCnHmnNrtT
qPKteWPm6U6OdHaZvO+SaRNvhv12Ub/q3FhtrNVj0uKOt56z6R5y9zix1w4qY6RtWsREOJ7L6OKa
S2rvX6zNM7T6Vh3mmZyOfya7eACzIAAAAAAAAAAAAAAAAAAAAAAAAAAczjVvqMVfW/8AZ03I41bf
Lp6/OVs/UVrY47NyOzUxd4bUJpEbb3Z7IiOrKIVSjZhMLJYyhKIgmGUQSDESIEbJEgQmCITEAmGU
IiGUAyhZVhDOoM4Wx2VQtqBKuyyWEgqlhKyyuyBVaGtkbNmvk7A15l1eH2300R6TMORPSXT4ZO+O
8fFefEX63gEAAAAAAAAAAAAAAAq1WPxdLlp+Kkx+y1Fvsz8gjhaDauGK8sx07y3OE3m1tT6RaP4c
vU6yMNKUx73zT0ilY3l2eF6a+m0kRl/zbzz3+Ez5M8z26fJruW6wzYq5sV8d43raNpZjRzPPaTmx
5b6bJ9rHO3zb2WJ8GWPEscY9bgzxH2t62n19GWW0eHOzHU5XbjXZ1x8WTnz2iZ7S2M1IjH2+LX0V
KTqs8zO9ot0j8nUthi1J3UaOFMTfLFo6xMbS9BwHWTqdHOO8+/hnln5eTjYMFo1WTH5VnePzXcIm
2k4zlpPSmXy/hfF5eMfJns69OA2cgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAADG/2LfJ874rW845mubliY7bPoto5qzHrDz0+yePNF41OotaJ7RWNtpV1OtfHqZ715fhu
j8adNpcVfeyzE2/vLuanhOu1nEctIxTTFa/+ZPbZ3eHcF0vDbTfFE2yzG03t32+DokynXl9+leDB
TTYKYccbUpWIhYCzEAAAAAAAAAAAAAAAAAAAAAAAAAAAAcXjE/4zDH9M/wAu04XF5/3jj/0f3Wz9
RUYmzDWxS2I7FSyjuzY1ZKpRKEygEwiWUIkGIk2QJNhKQhMIhkCYZQxhlAMoZwwZwgWQshVCyATL
CWc9ldpBhZXLOVdpQK7NfJPRdaWvknoDVvPvOnwuel4+TlXn3nS4VPvXj4QtEV0wAAAAAAAAAAAA
AAAAAVV02CmTxK4qRf8AFFeq0AAAanEsfPpZmO9Ji0NDLfkwdOsulrumiyzHlVzJrz4Ovoy26vB8
cTBa9NffLtMY77Rv8Yegx5ImkKdJoY1HC81Y+3OSbVn0mGGkmbY45u6tnrrTOu2xGO0RxCd+nNVj
qKxTV1vH2pjaGtnyzXXYdo96ZmGXEMk15b7/AGZiVerWPTYckZcNbx5wzc7hGbnxXxzPWk7x8pdF
0S9jh1OXgAlUAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAcPjEf4/FP9H93ccXjMf4vDP9Mx+62fqKrx+S+GvibEFSsqyYwlVK
ZYsmIMoRKYJQIPIEiQ2ATCUQygCGUIhMAyhnDCGUIFkLIV1ZxIMpVWWSrsCuyqyyyq09ECq8tfJK
66jJ2Bp5J6upwn7dv9Lk5J951uE/av8AJaIrqAAAAAAAAAAAAAAAAAAAAAAq1Mc2myxPnWf4cmtu
XT9fR0tffk0WSe28bfq5Wbamm3326MtunwfK6PCv/AxPraZ/dz9PO97/AOqf5dHhdZrw7Dv3mOb9
XOxRFM+avpe38mvkPHf/AFWlrKba7Tzt99ZxKkfR7euyNXMTrtPHfa0z+zPiM/UR8Zj+Wbdu8HpN
M2bfzrV13M4dO2pyR61dNvj44/J/oAWZgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADj8bj63BPzdhyeNx0wz8ZWz9RWri7Nmv
VrYu0NmqaRZHZlDGGSiwxZSgCEkCBCQSCQBMJRCYgEsoYx3Z17AlMIhlCBnDOGEM4AlhZZKq4KrK
7LLKrIFN2vdfZReAaObu6/CO9vk5OePR1uEd7fJeIrqAIAAAAAAAAAAAAAAAAAAAAGtxCk5NFliI
3mI32+XVyNTyZOHTee946PQKPoeDffw4777eW/yVs60xv+ZxOnr4Okx1t05KRv8Ao41Z5q3yed5m
XY1szXRZ5jvFJ/hxItP0aOSN9q7yrtr4f2tHFM5+KT16Yq/vK/iGSbXw4vO14UcPx5MGfNbPG18m
1oj4THRsTw7VanPXVYpi3gzMcnrvCnG11JOupwuN8+a3pEQ6jT4divjxWnJExa09pbjbM5HHu90A
JUAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAHM41H1GOf6nTc/jEf4Ws+lls/UX45uGekNujTwdm5RNIthKIZKLDFlsiQIShIC
EgCUJ7AmGTGO7IDzZQhMSDJMMYZQgZwzhhDOATuqssmVdgVWVWWyqtCBTeVF19lF+wNLNG7q8I+9
8nLyupwnt+S8RXUAQAAAAAAAAAAAAAAAAAAAAAAItWL1mto3iY2lyrcLyUxzix2ia2nvPeK+jrCL
OrTVnxpanhuPPemSs8l6RtE7dJj0ldpNP9GwRSZ3neZmV4cR/Vs4AJQAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAANHi1d9H
M+kt5ra+vPoskfDdOfqK4mn7Q3aNHBPZu0W0RdDOGFWcKLCJZeTGQQlCQSgASBsCYZQxhlAJTAmA
TsmAgGcM4YQyjsgRLC3VnaVcgwsrt3Z2V2QK7tbJ1bN5a9waeWO7p8Knt8nNyebpcK8vkvlFdQBA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAK9RXmwZI+ErEWjesx6wQeZwejeo0cccuW8
elpblJaaRGxVnCuss4ZrMvJEgCAASISCQIBlCYYpieoM0wx8k7gzIRueYM4Z79FcSy3QEsLJmWFp
BjaVVpZWlXMoGNmvkXXlr3kGtknu6XCf7OXkl1OEdl8orqgIAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAHmskcmtzV/rls0U62OXiWX4zErcc9GmkRfWVkSqqziWayxCPIANwBIhIJSxS
CRG6dwZwlhEs4BluMdzfqgZxLLdXuy3AmVdpZTKuZBjaVVpWWV2QlhZRdfZRcGpl7urwfrzfJy8r
rcH61vPyWitdMAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAHA4nHLxKZ9awnH2ZcY
jbW459aq8fZpfiI2IZwrqzhmsz3Ebm4JN0AMhCQSIASndiAziWUSriWcAyRujc80DM3RCfIETLCW
UsZEsJYSslXZAwlTddPZTkBp5e7r8Gj6rJPxhx8k9Xa4PG2C8/FaK10QAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAcfjcbZMFvnDWx9m5x2PqcNvS+zSxT7sNPxH62YZQwqzhRZO6UCB
KUAJTux3SDIRuAncQAmJZRLBMSgZ7iIAZRKd2DICUSlAljLCYWMLIFVukNfI2bNbIDTyT7zu8Ijb
Sz/qcG/2nf4T/wCE/wD2WnxWt4AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAHL9oL
+Hw2cm28VvEuPptfgyVj6yIn0no7/FtJfW8NzYMe3PaPd39d3iMug1WktNc2C9dvPbeP1aZ9xF+v
T471tHu2iflK2HkqWmvaZj5Surqc9Ps5bx+alTHqYHm68S1Vf/NmfnC2vGNTXvyT84Ql6A3cSvHM
sfaxVn5Ssrxyv3sM/lKB1xza8bwT3pePyWV4tpZ+/MfOEjfGrXiGlt2zV/PotrqcN/s5aT/+wLRj
FontMSlAlKEgndO6IAZQljDIEgeQljLCzOVdkCu/SGrkbF56NPNeKxMzMRHxENe0+89DwuNtHHzl
5PJr8NcnLW3Pbf7r1nCZm2gpae8zMrz4i/W6AgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAETETG0xukB4HVaeMHEtRi26RedvkyjBSfX9W77QYvC4xz7dMlYlrU7M929dWJLFc6aPK0q
7YLxPS0S22FlP6q38Zac0yR92s/KVc3tHfFf8tpbcsLRvB/dR/8ALLVnU0r9uL1+dZI1mnmdvGpv
6TOy6ym+Oto2tWJ+cJ/tW+KLK5KW+zes/KU7tG+h01p64qx8Y6NXNo6Y+uPJlp8rLf0rfG7MXtHa
0x8pZxqs9e2a8f8A7Oj7HaTHn0+f6RWM23LETfr6vRW4PoL99NT8ui7F4+vEdXXtnt+fVbXjGsr/
AOZE/OsPS29nuH27YrV+VpeV9pdPXhOtw49NG9Mld55+vXcTPd42I47qo7xSfyWV9oM8d8VJ/VxM
d8l46xWF9cV7en6o/qLfxp2I9ob+eCv/AHMo9op89P8A/wBORGmyT5R+qfo2X8P7n9Q/jTsx7RR5
6ef+4/8AuHftg/8A6cWcOSO9J/WEbWr3pY7Efzp2Lcfv5YK/9zWy8d1E/ZpSv5Oba1/+Hb9lc+LP
bFt87I7E/wAabWbiurvEx4nL/pjZzc2bJkn372t85ZXx55/BX85lucC0vPxnTxlnnjm32mOiZqUu
LJ2p4TwnVavNWaYbRTfre0bQ99pcH0bT0xb78vmtiIiNojaErMwAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAHnfarF7umzRHaZrLjYrdIen9ocPi8JyTt1xzF4eUw23rCm3R4r6bMy
wt6kdTaWLdjswmNoZontsCm0K5XWjopnuDC0dGpqG5bs08/daKV672MjbSaif6oh6Z5f2LtvptRX
0tEvUN3Jfo8f7cYve0eX4zV7B5z20xc/C8eSPuZIRficfXlcPaG7ino08HWIbePpLF2NuiyOyrHK
3fZFSwuovHVfaVF4QK5YWTM9UT0EKry6Ps1Tn4zjn8NZn9nOtLseydObiWW34cf918fWfk+PYANn
KAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAq1WKM+ly4p+/WYeBxTNd6zG0xO0
vobw3FcP0bi2em20Tbmj5Srr418V9sa2Z7qKyzi07MXUylhaU7yjqhLCeiq3ddaFNxFYW7NLNG8t
zya+WO6Va9J7FW66mvwidnrXiPY3Ny8RyUn71Jj9Ht3RPjk19HK9pMHj8D1ER3rHN+jqqtTjjNps
uOe16zAifXzfTz7kNyndpYazS9qT0mszDdoxrsi6m8LazMq6zDOsq1ZEyrt1WWlXaUCqyq0rbKbi
Fdp6PReyFd8uqv8ACsfy83aXrPZHHto89/xX2/SP/dpj6y8vx6EBq5gAAAAAAAAAAAAAAAAAAAAA
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
NO67NbkhzNVnmInqzaOZrL93JyZeV0M1++7S02jvxDWxhxx033tPpC8Z6rrezWjmZyazJG2/u03h
2vFibTHoqvamiwVwY+nLGzV0+SZ1Mx8G0/45tOhzJ5lXMc3UVXRdlF1HP+iYsDPLPPy49/tz1+Te
pSIr0ho6ak5Ms5J8o2q6NImOrHV7XX488ypzTtHXo0s9t6zG7c1G1qz6ubeZiZ3UatXJG3yauSO7
cvMTEx5tPLb3prPRMVr0HB8vicNxf0+7+kt+LOJwTJyY/Bnz3tH93X36N58cWvq6LSyiyndMSlC7
mZcymLJiwLosmJVRLKLAtiU7q4lMSCzc3YxJuDMRuAlKAEgAAAlAkAAAAAABKAEgAAAAAJAAAAAA
AAAAAAAEgAAAAAAAAAAAAAkAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAhIAAACAAAASgAAAAAAEAAAA
hGzJAImGMwzQDDZjNVuyNgUTVhNGxysZqDVmiu1G5NN2M4waM0+DCaN2cbGcQNGaMZq3JxMJxA1J
qx2bU4kU09slorWNwa20z02RXHbJbl26QvtFovbHWkxEdJt5y2MOHlr2U1W3jx+1hiw8vSO63lmI
XRTaEWmtY6snRHO1VpmJ+DjavpSZl2s8b7y4HFcnh0n0gha5ebJN55KRM2mdoiPN6fh+kpwXh0Wy
RHj5Otp/s5Ps1p62y31+em9aTMYt/OfVfxTiPjZ52naI7fBrI5t66xz5+a1rW7yx0eSL6iZjtEOX
qNbSletom3lENjh2fbHzbbWt3iVozruc+5ztWubf4M4ybpQ2Oboyrva0Vjza8WdDR4OkXt3n9ldX
kaePP9VtYqctYhdvt5oivTeCZ2YOxXk6ubqMfV0b9mrljfqlFcq88k7z2U5axeItDa1OPessuC8P
ya7XRWYnwqdbT/ZMilvIu4dpslNdixXja8Y5tt85djZdbDWnGOesRtXFtuw6T27No5Kx2OrKYQlC
ExKJgBnEpiyvdlEgsizKLKollFgWxLKJVRLKJBbEp3VxLKJBnuMWQJEbpBIAAAJAAAABIAAAAAAA
lAJAAAAAAAAAAAAAASAAAAAAAAAAAAAJAAAABAJABAlAAAAAAAAAAAAAAAAAAAAAAAAIAAAAAAAA
AAABAJQAAAAgAABAAI2EoBGyJhkgGPKxmqxAKpownHC+YRMdN5BrTj67R3bOn01o7p01Iv71u89o
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
2Q==`;
// src/warmup.ts
async function warmupBitmap(instance) {
const b64toBlob = (base64, type = "application/octet-stream") => fetch(`data:${type};base64,${base64}`).then((res2) => res2.blob());
let blob;
let res;
switch (instance.config.warmup) {
case "face":
blob = await b64toBlob(face3);
break;
case "body":
case "full":
blob = await b64toBlob(body3);
break;
default:
blob = null;
}
if (blob) {
const bitmap = await createImageBitmap(blob);
res = await instance.detect(bitmap, instance.config);
bitmap.close();
}
return res;
}
async function warmupCanvas(instance) {
return new Promise((resolve) => {
let src;
switch (instance.config.warmup) {
case "face":
src = "data:image/jpeg;base64," + face3;
break;
case "full":
case "body":
src = "data:image/jpeg;base64," + body3;
break;
default:
src = "";
}
let img;
if (typeof Image !== "undefined")
img = new Image();
else if (env.Image)
img = new env.Image();
else
return;
img.onload = async () => {
const canvas3 = canvas(img.naturalWidth, img.naturalHeight);
if (!canvas3) {
log("Warmup: Canvas not found");
resolve(void 0);
} else {
const ctx = canvas3.getContext("2d");
if (ctx)
ctx.drawImage(img, 0, 0);
const tensor6 = await instance.image(canvas3, true);
const res = tensor6.tensor ? await instance.detect(tensor6.tensor, instance.config) : void 0;
resolve(res);
}
};
if (src)
img.src = src;
else
resolve(void 0);
});
}
async function warmupNode(instance) {
const atob = (str) => Buffer.from(str, "base64");
let img;
if (instance.config.warmup === "face")
img = atob(face3);
else
img = atob(body3);
let res;
if ("node" in tf37 && tf37.getBackend() === "tensorflow") {
const data = tf37["node"].decodeJpeg(img);
const expanded = tf37.expandDims(data, 0);
instance.tf.dispose(data);
res = await instance.detect(expanded, instance.config);
instance.tf.dispose(expanded);
} else {
if (instance.config.debug)
log("Warmup tfjs-node not loaded");
}
return res;
}
async function runInference(instance) {
let res;
if (typeof createImageBitmap === "function")
res = await warmupBitmap(instance);
else if (typeof Image !== "undefined" || env.Canvas !== void 0)
res = await warmupCanvas(instance);
else
res = await warmupNode(instance);
return res;
}
async function runCompile(instance) {
var _a, _b, _c, _d;
if (!tf37.env().flagRegistry.ENGINE_COMPILE_ONLY)
return;
const backendType = tf37.getBackend();
const webGLBackend = tf37.backend();
if (backendType !== "webgl" && backendType !== "humangl" || !(webGLBackend == null ? void 0 : webGLBackend["checkCompileCompletion"])) {
return;
}
tf37.env().set("ENGINE_COMPILE_ONLY", true);
const numTensorsStart = tf37.engine().state.numTensors;
const compiledModels = [];
for (const [modelName, model23] of Object.entries(instance.models).filter(([key, val]) => key !== null && val !== null)) {
const shape = (model23 == null ? void 0 : model23.modelSignature) && ((_b = (_a = model23 == null ? void 0 : model23.inputs) == null ? void 0 : _a[0]) == null ? void 0 : _b.shape) ? [...model23.inputs[0].shape] : [1, 64, 64, 3];
const dtype = (model23 == null ? void 0 : model23.modelSignature) && ((_d = (_c = model23 == null ? void 0 : model23.inputs) == null ? void 0 : _c[0]) == null ? void 0 : _d.dtype) ? model23.inputs[0].dtype : "float32";
for (let dim = 0; dim < shape.length; dim++) {
if (shape[dim] === -1)
shape[dim] = dim === 0 ? 1 : 64;
}
const tensor6 = tf37.zeros(shape, dtype);
try {
const res = model23.execute(tensor6);
compiledModels.push(modelName);
if (Array.isArray(res))
res.forEach((t2) => tf37.dispose(t2));
else
tf37.dispose(res);
} catch (e) {
if (instance.config.debug)
log("compile fail model:", modelName);
}
tf37.dispose(tensor6);
}
const kernels = await webGLBackend["checkCompileCompletionAsync"]();
webGLBackend["getUniformLocations"]();
if (instance.config.debug)
log("compile pass:", { models: compiledModels, kernels: kernels.length });
tf37.env().set("ENGINE_COMPILE_ONLY", false);
const numTensorsEnd = tf37.engine().state.numTensors;
if (numTensorsEnd - numTensorsStart > 0)
log("tensor leak:", numTensorsEnd - numTensorsStart);
}
async function warmup(instance, userConfig) {
await check(instance, false);
const t0 = now();
instance.state = "warmup";
if (userConfig)
instance.config = mergeDeep(instance.config, userConfig);
if (!instance.config.warmup || instance.config.warmup.length === 0 || instance.config.warmup === "none") {
return empty();
}
return new Promise(async (resolve) => {
await instance.models.load();
await runCompile(instance);
const res = await runInference(instance);
const t1 = now();
if (instance.config.debug)
log("warmup", instance.config.warmup, Math.round(t1 - t0), "ms");
instance.emit("warmup");
resolve(res);
});
}
// src/human.ts
var _numTensors, _analyzeMemoryLeaks, _checkSanity, _sanity, _loops;
var Human = class {
constructor(userConfig) {
__publicField(this, "version");
__publicField(this, "config");
__publicField(this, "result");
__publicField(this, "state");
__publicField(this, "process");
__publicField(this, "tf");
__publicField(this, "env", env);
__publicField(this, "draw", draw_exports);
__publicField(this, "match", match_exports);
__publicField(this, "models");
__publicField(this, "events");
__publicField(this, "faceTriangulation");
__publicField(this, "faceUVMap");
__publicField(this, "performance");
__privateAdd(this, _numTensors, void 0);
__privateAdd(this, _analyzeMemoryLeaks, void 0);
__privateAdd(this, _checkSanity, void 0);
__publicField(this, "analyze", (...msg) => {
if (!__privateGet(this, _analyzeMemoryLeaks))
return;
const currentTensors = this.tf.engine().state.numTensors;
const previousTensors = __privateGet(this, _numTensors);
__privateSet(this, _numTensors, currentTensors);
const leaked = currentTensors - previousTensors;
if (leaked !== 0)
log(...msg, leaked);
});
__privateAdd(this, _sanity, (input) => {
if (!__privateGet(this, _checkSanity))
return null;
if (!input)
return "input is not defined";
if (this.env.node && !(input instanceof tf38.Tensor))
return "input must be a tensor";
try {
this.tf.getBackend();
} catch (e) {
return "backend not loaded";
}
return null;
});
__publicField(this, "webcam", new WebCam());
__publicField(this, "emit", (event) => {
var _a;
if ((_a = this.events) == null ? void 0 : _a.dispatchEvent)
this.events.dispatchEvent(new Event(event));
});
__privateAdd(this, _loops, {});
const tfVersion = (tf38.version.tfjs || tf38.version_core).replace(/-(.*)/, "");
config.wasmPath = `https://cdn.jsdelivr.net/npm/@tensorflow/tfjs-backend-wasm@${tfVersion}/dist/`;
config.modelBasePath = env.browser ? "../models/" : "file://models/";
this.version = version2;
Object.defineProperty(this, "version", { value: version2 });
this.config = JSON.parse(JSON.stringify(config));
Object.seal(this.config);
this.config.cacheModels = typeof indexedDB !== "undefined";
if (userConfig)
this.config = mergeDeep(this.config, userConfig);
setModelLoadOptions(this.config);
this.tf = tf38;
this.state = "idle";
__privateSet(this, _numTensors, 0);
__privateSet(this, _analyzeMemoryLeaks, false);
__privateSet(this, _checkSanity, false);
this.performance = {};
this.events = typeof EventTarget !== "undefined" ? new EventTarget() : void 0;
this.models = new Models(this);
init2();
this.result = empty();
this.process = { tensor: null, canvas: null };
this.faceTriangulation = triangulation;
this.faceUVMap = uvmap;
validateModel(this, null, "");
this.emit("create");
if (this.config.debug || this.env.browser)
log(`version: ${this.version}`);
if (this.config.debug)
log(`tfjs version: ${this.tf.version["tfjs-core"]}`);
const envTemp = JSON.parse(JSON.stringify(this.env));
delete envTemp.kernels;
delete envTemp.initial;
delete envTemp.perfadd;
if (this.config.debug)
log("environment:", envTemp);
}
reset() {
const currentBackend = this.config.backend;
this.config = JSON.parse(JSON.stringify(config));
this.config.backend = currentBackend;
reset();
env.initial = true;
}
validate(userConfig) {
const msgs = validate(config, userConfig || this.config);
if (msgs.length === 0)
this.config = mergeDeep(this.config, userConfig);
return msgs;
}
now() {
return now();
}
image(input, getTensor = false) {
return process2(input, this.config, getTensor);
}
async segmentation(input, userConfig) {
var _a, _b, _c;
if (userConfig)
this.config = mergeDeep(this.config, userConfig);
if (!this.config.segmentation.enabled)
return null;
const processed = await process2(input, this.config);
if (!processed.tensor)
return null;
let tensor6 = null;
if ((_a = this.config.segmentation.modelPath) == null ? void 0 : _a.includes("rvm"))
tensor6 = await predict20(processed.tensor, this.config);
if ((_b = this.config.segmentation.modelPath) == null ? void 0 : _b.includes("meet"))
tensor6 = await predict16(processed.tensor, this.config);
if ((_c = this.config.segmentation.modelPath) == null ? void 0 : _c.includes("selfie"))
tensor6 = await predict21(processed.tensor, this.config);
tf38.dispose(processed.tensor);
return tensor6;
}
compare(firstImageTensor, secondImageTensor) {
return compare(this.config, firstImageTensor, secondImageTensor);
}
async init() {
await check(this, true);
await this.tf.ready();
reset();
}
async load(userConfig) {
this.state = "load";
const timeStamp = now();
const count2 = Object.values(this.models.models).filter((model23) => model23).length;
if (userConfig)
this.config = mergeDeep(this.config, userConfig);
if (this.env.initial) {
if (!await check(this, false))
log("error: backend check failed");
await tf38.ready();
if (this.env.browser) {
if (this.config.debug)
log("configuration:", this.config);
if (this.config.debug)
log("tf flags:", this.tf.ENV.flags);
}
}
await this.models.load(this);
if (this.env.initial && this.config.debug)
log("tf engine state:", this.tf.engine().state.numBytes, "bytes", this.tf.engine().state.numTensors, "tensors");
this.env.initial = false;
const loaded = Object.values(this.models.models).filter((model23) => model23).length;
if (loaded !== count2) {
this.models.validate();
this.emit("load");
}
const current = Math.trunc(now() - timeStamp);
if (current > (this.performance.loadModels || 0))
this.performance.loadModels = this.env.perfadd ? (this.performance.loadModels || 0) + current : current;
}
next(result = this.result) {
return calc2(result, this.config);
}
async warmup(userConfig) {
const t0 = now();
const res = await warmup(this, userConfig);
const t1 = now();
this.performance.warmup = Math.trunc(t1 - t0);
return res;
}
async profile(input, userConfig) {
const profile = await this.tf.profile(() => this.detect(input, userConfig));
const kernels = {};
let total = 0;
for (const kernel of profile.kernels) {
const ms = Number(kernel.kernelTimeMs) || 0;
if (kernels[kernel.name])
kernels[kernel.name] += ms;
else
kernels[kernel.name] = ms;
total += ms;
}
const kernelArr = [];
Object.entries(kernels).forEach((key) => kernelArr.push({ kernel: key[0], time: key[1], perc: 0 }));
for (const kernel of kernelArr) {
kernel.perc = Math.round(1e3 * kernel.time / total) / 1e3;
kernel.time = Math.round(1e3 * kernel.time) / 1e3;
}
kernelArr.sort((a, b) => b.time - a.time);
kernelArr.length = 20;
return kernelArr;
}
async detect(input, userConfig) {
this.state = "detect";
return new Promise(async (resolve) => {
var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l, _m, _n, _o, _p, _q, _r, _s, _t, _u;
this.state = "config";
let timeStamp;
this.config = mergeDeep(this.config, userConfig);
this.state = "check";
const error = __privateGet(this, _sanity).call(this, input);
if (error) {
log(error, input);
this.emit("error");
resolve(empty(error));
}
const timeStart = now();
await this.load();
timeStamp = now();
this.state = "image";
const img = await process2(input, this.config);
this.process = img;
this.performance.inputProcess = this.env.perfadd ? (this.performance.inputProcess || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
this.analyze("Get Image:");
if (!img.tensor) {
if (this.config.debug)
log("could not convert input to tensor");
this.emit("error");
resolve(empty("could not convert input to tensor"));
return;
}
this.emit("image");
timeStamp = now();
this.config.skipAllowed = await skip(this.config, img.tensor);
this.config.filter.autoBrightness = (this.config.filter.autoBrightness || false) && this.config.skipAllowed;
if (!this.performance.totalFrames)
this.performance.totalFrames = 0;
if (!this.performance.cachedFrames)
this.performance.cachedFrames = 0;
this.performance.totalFrames++;
if (this.config.skipAllowed)
this.performance.cachedFrames++;
this.performance.cacheCheck = this.env.perfadd ? (this.performance.cacheCheck || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
this.analyze("Check Changed:");
let faceRes = [];
let bodyRes = [];
let handRes = [];
let objectRes = [];
this.state = "detect:face";
if (this.config.async) {
faceRes = this.config.face.enabled ? detectFace(this, img.tensor) : [];
if (this.performance.face)
delete this.performance.face;
} else {
timeStamp = now();
faceRes = this.config.face.enabled ? await detectFace(this, img.tensor) : [];
this.performance.face = this.env.perfadd ? (this.performance.face || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
}
if (this.config.async && (this.config.body.maxDetected === -1 || this.config.hand.maxDetected === -1))
faceRes = await faceRes;
this.analyze("Start Body:");
this.state = "detect:body";
const bodyConfig = this.config.body.maxDetected === -1 ? mergeDeep(this.config, { body: { maxDetected: this.config.face.enabled ? 1 * faceRes.length : 1 } }) : this.config;
if (this.config.async) {
if ((_a = this.config.body.modelPath) == null ? void 0 : _a.includes("posenet"))
bodyRes = this.config.body.enabled ? predict19(img.tensor, bodyConfig) : [];
else if ((_b = this.config.body.modelPath) == null ? void 0 : _b.includes("blazepose"))
bodyRes = this.config.body.enabled ? predict(img.tensor, bodyConfig) : [];
else if ((_c = this.config.body.modelPath) == null ? void 0 : _c.includes("efficientpose"))
bodyRes = this.config.body.enabled ? predict3(img.tensor, bodyConfig) : [];
else if ((_d = this.config.body.modelPath) == null ? void 0 : _d.includes("movenet"))
bodyRes = this.config.body.enabled ? predict17(img.tensor, bodyConfig) : [];
if (this.performance.body)
delete this.performance.body;
} else {
timeStamp = now();
if ((_e = this.config.body.modelPath) == null ? void 0 : _e.includes("posenet"))
bodyRes = this.config.body.enabled ? await predict19(img.tensor, bodyConfig) : [];
else if ((_f = this.config.body.modelPath) == null ? void 0 : _f.includes("blazepose"))
bodyRes = this.config.body.enabled ? await predict(img.tensor, bodyConfig) : [];
else if ((_g = this.config.body.modelPath) == null ? void 0 : _g.includes("efficientpose"))
bodyRes = this.config.body.enabled ? await predict3(img.tensor, bodyConfig) : [];
else if ((_h = this.config.body.modelPath) == null ? void 0 : _h.includes("movenet"))
bodyRes = this.config.body.enabled ? await predict17(img.tensor, bodyConfig) : [];
this.performance.body = this.env.perfadd ? (this.performance.body || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
}
this.analyze("End Body:");
this.analyze("Start Hand:");
this.state = "detect:hand";
const handConfig = this.config.hand.maxDetected === -1 ? mergeDeep(this.config, { hand: { maxDetected: this.config.face.enabled ? 2 * faceRes.length : 1 } }) : this.config;
if (this.config.async) {
if ((_j = (_i = this.config.hand.detector) == null ? void 0 : _i.modelPath) == null ? void 0 : _j.includes("handdetect"))
handRes = this.config.hand.enabled ? predict14(img.tensor, handConfig) : [];
else if ((_l = (_k = this.config.hand.detector) == null ? void 0 : _k.modelPath) == null ? void 0 : _l.includes("handtrack"))
handRes = this.config.hand.enabled ? predict15(img.tensor, handConfig) : [];
if (this.performance.hand)
delete this.performance.hand;
} else {
timeStamp = now();
if ((_n = (_m = this.config.hand.detector) == null ? void 0 : _m.modelPath) == null ? void 0 : _n.includes("handdetect"))
handRes = this.config.hand.enabled ? await predict14(img.tensor, handConfig) : [];
else if ((_p = (_o = this.config.hand.detector) == null ? void 0 : _o.modelPath) == null ? void 0 : _p.includes("handtrack"))
handRes = this.config.hand.enabled ? await predict15(img.tensor, handConfig) : [];
this.performance.hand = this.env.perfadd ? (this.performance.hand || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
}
this.analyze("End Hand:");
this.analyze("Start Object:");
this.state = "detect:object";
if (this.config.async) {
if ((_q = this.config.object.modelPath) == null ? void 0 : _q.includes("nanodet"))
objectRes = this.config.object.enabled ? predict18(img.tensor, this.config) : [];
else if ((_r = this.config.object.modelPath) == null ? void 0 : _r.includes("centernet"))
objectRes = this.config.object.enabled ? predict2(img.tensor, this.config) : [];
if (this.performance.object)
delete this.performance.object;
} else {
timeStamp = now();
if ((_s = this.config.object.modelPath) == null ? void 0 : _s.includes("nanodet"))
objectRes = this.config.object.enabled ? await predict18(img.tensor, this.config) : [];
else if ((_t = this.config.object.modelPath) == null ? void 0 : _t.includes("centernet"))
objectRes = this.config.object.enabled ? await predict2(img.tensor, this.config) : [];
this.performance.object = this.env.perfadd ? (this.performance.object || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
}
this.analyze("End Object:");
this.state = "detect:await";
if (this.config.async)
[faceRes, bodyRes, handRes, objectRes] = await Promise.all([faceRes, bodyRes, handRes, objectRes]);
this.state = "detect:gesture";
let gestureRes = [];
if (this.config.gesture.enabled) {
timeStamp = now();
gestureRes = [...face2(faceRes), ...body2(bodyRes), ...hand2(handRes), ...iris2(faceRes)];
if (!this.config.async)
this.performance.gesture = this.env.perfadd ? (this.performance.gesture || 0) + Math.trunc(now() - timeStamp) : Math.trunc(now() - timeStamp);
else if (this.performance.gesture)
delete this.performance.gesture;
}
this.performance.total = this.env.perfadd ? (this.performance.total || 0) + Math.trunc(now() - timeStart) : Math.trunc(now() - timeStart);
const shape = ((_u = this.process.tensor) == null ? void 0 : _u.shape) || [0, 0, 0, 0];
this.result = {
face: faceRes,
body: bodyRes,
hand: handRes,
gesture: gestureRes,
object: objectRes,
performance: this.performance,
canvas: this.process.canvas,
timestamp: Date.now(),
error: null,
width: shape[2],
height: shape[1],
get persons() {
return join2(faceRes, bodyRes, handRes, gestureRes, shape);
}
};
tf38.dispose(img.tensor);
this.emit("detect");
this.state = "idle";
resolve(this.result);
});
}
async sleep(ms) {
return new Promise((resolve) => {
setTimeout(resolve, ms);
});
}
async video(element, run = true, delay = 0) {
if (run) {
if (!__privateGet(this, _loops)[element.id]) {
if (this.config.debug)
log("video start", element.id);
__privateGet(this, _loops)[element.id] = true;
}
if (!element.paused && __privateGet(this, _loops)[element.id] && element.readyState >= 2)
await this.detect(element);
if (delay > 0)
await this.sleep(delay);
if (__privateGet(this, _loops)[element.id])
requestAnimationFrame(() => this.video(element, run, delay));
} else {
if (this.config.debug)
log("video stop", element.id);
__privateGet(this, _loops)[element.id] = false;
}
}
};
_numTensors = new WeakMap();
_analyzeMemoryLeaks = new WeakMap();
_checkSanity = new WeakMap();
_sanity = new WeakMap();
_loops = new WeakMap();
// Annotate the CommonJS export names for ESM import in node:
0 && (module.exports = {
Env,
Human,
defaults,
draw,
empty,
env,
match,
models
});