@sage-rsc/talking-head-react 1.8.8 → 1.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -1,21 +1,21 @@
1
- import { jsxs as Be, jsx as re } from "react/jsx-runtime";
2
- import { forwardRef as Ke, useRef as V, useState as ye, useEffect as Ce, useCallback as U, useImperativeHandle as $e, useLayoutEffect as yt } from "react";
1
+ import { jsxs as Ee, jsx as le } from "react/jsx-runtime";
2
+ import { forwardRef as Ke, useRef as V, useState as ge, useEffect as ze, useCallback as W, useImperativeHandle as $e, useLayoutEffect as ft } from "react";
3
3
  import * as A from "three";
4
- import { OrbitControls as ft } from "three/addons/controls/OrbitControls.js";
5
- import { GLTFLoader as xt } from "three/addons/loaders/GLTFLoader.js";
6
- import { DRACOLoader as bt } from "three/addons/loaders/DRACOLoader.js";
7
- import { FBXLoader as st } from "three/addons/loaders/FBXLoader.js";
8
- import { RoomEnvironment as At } from "three/addons/environments/RoomEnvironment.js";
9
- import Rt from "three/addons/libs/stats.module.js";
10
- let p, we, Me;
11
- const z = [0, 0, 0, 0], N = new A.Vector3(), Qe = new A.Vector3(), Ae = new A.Vector3(), Ye = new A.Vector3();
4
+ import { OrbitControls as xt } from "three/addons/controls/OrbitControls.js";
5
+ import { GLTFLoader as bt } from "three/addons/loaders/GLTFLoader.js";
6
+ import { DRACOLoader as At } from "three/addons/loaders/DRACOLoader.js";
7
+ import { FBXLoader as ot } from "three/addons/loaders/FBXLoader.js";
8
+ import { RoomEnvironment as Rt } from "three/addons/environments/RoomEnvironment.js";
9
+ import vt from "three/addons/libs/stats.module.js";
10
+ let m, Ce, He;
11
+ const T = [0, 0, 0, 0], U = new A.Vector3(), Qe = new A.Vector3(), ve = new A.Vector3(), Ye = new A.Vector3();
12
12
  new A.Plane();
13
13
  new A.Ray();
14
14
  new A.Euler();
15
- const Re = new A.Quaternion(), rt = new A.Quaternion(), Oe = new A.Matrix4(), De = new A.Matrix4();
15
+ const Ie = new A.Quaternion(), st = new A.Quaternion(), Pe = new A.Matrix4(), Be = new A.Matrix4();
16
16
  new A.Vector3();
17
- const qe = new A.Vector3(0, 0, 1), vt = new A.Vector3(1, 0, 0), It = new A.Vector3(0, 1, 0), Lt = new A.Vector3(0, 0, 1);
18
- class St {
17
+ const qe = new A.Vector3(0, 0, 1), It = new A.Vector3(1, 0, 0), Lt = new A.Vector3(0, 1, 0), St = new A.Vector3(0, 0, 1);
18
+ class kt {
19
19
  constructor(n = null) {
20
20
  this.opt = Object.assign({
21
21
  warmupMs: 2e3,
@@ -264,7 +264,7 @@ class St {
264
264
  "pivot",
265
265
  "helper"
266
266
  ].forEach((t) => {
267
- p = this.getValue(n.name, t), p && (e[t] = p);
267
+ m = this.getValue(n.name, t), m && (e[t] = m);
268
268
  }), e;
269
269
  });
270
270
  }
@@ -279,7 +279,7 @@ class St {
279
279
  this.armature.traverse((i) => {
280
280
  e.has(i) || (e.set(i, n), n++);
281
281
  }), this.data.sort((i, a) => e.get(i.bone) - e.get(a.bone)), this.data.forEach((i) => {
282
- p = this.dict[i.boneParent.name], p && (p.children || (p.children = []), p.children.push(i));
282
+ m = this.dict[i.boneParent.name], m && (m.children || (m.children = []), m.children.push(i));
283
283
  }), this.objectsUpdate = [];
284
284
  const t = /* @__PURE__ */ new WeakSet(), o = (i) => i.parent?.isBone ? [i, ...o(i.parent)] : [i], s = (i) => {
285
285
  o(i).forEach((u) => {
@@ -321,7 +321,7 @@ class St {
321
321
  /// Bone's parent object
322
322
  vBasis: l.position.clone(),
323
323
  // Original local position
324
- vWorld: l.parent.getWorldPosition(N).clone(),
324
+ vWorld: l.parent.getWorldPosition(U).clone(),
325
325
  // World position, parent
326
326
  qBasis: l.parent.quaternion.clone(),
327
327
  // Original quaternion, parent
@@ -338,7 +338,7 @@ class St {
338
338
  ea: [0, 0, 0, 0]
339
339
  // External acceleration [m/s^2]
340
340
  };
341
- c.boneParent.matrixWorld.decompose(N, Re, Ae), N.copy(qe).applyQuaternion(Re).setY(0).normalize(), Re.premultiply(rt.setFromUnitVectors(qe, N).invert()).normalize(), c.qWorldInverseYaw = Re.clone().normalize(), this.data.push(c), this.dict[u] = c;
341
+ c.boneParent.matrixWorld.decompose(U, Ie, ve), U.copy(qe).applyQuaternion(Ie).setY(0).normalize(), Ie.premultiply(st.setFromUnitVectors(qe, U).invert()).normalize(), c.qWorldInverseYaw = Ie.clone().normalize(), this.data.push(c), this.dict[u] = c;
342
342
  try {
343
343
  this.setValue(u, "type", s.type), this.setValue(u, "stiffness", s.stiffness), this.setValue(u, "damping", s.damping), this.setValue(u, "external", s.external), this.setValue(u, "limits", s.limits), this.setValue(u, "excludes", s.excludes), this.setValue(u, "deltaLocal", s.deltaLocal), this.setValue(u, "deltaWorld", s.deltaWorld), this.setValue(u, "pivot", s.pivot), this.setValue(u, "helper", s.helper);
344
344
  } catch (r) {
@@ -356,22 +356,22 @@ class St {
356
356
  for (this.timerMs += n, n > 1e3 && (this.timerMs = 0), n /= 1e3, e = 0, o = this.objectsUpdate.length; e < o; e++)
357
357
  i = this.objectsUpdate[e], i.updateMatrix(), i.parent === null ? i.matrixWorld.copy(i.matrix) : i.matrixWorld.multiplyMatrices(i.parent.matrixWorld, i.matrix), i.matrixWorldNeedsUpdate = !1;
358
358
  for (e = 0, o = this.data.length; e < o; e++) {
359
- if (i = this.data[e], N.copy(i.vWorld), Oe.copy(i.boneParent.matrixWorld), De.copy(Oe).invert(), i.vWorld.setFromMatrixPosition(Oe), N.applyMatrix4(De), N.length() > 0.5 && (console.info("Info: Unrealistic jump of " + N.length().toFixed(2) + " meters."), N.setLength(0.5)), N.applyQuaternion(i.bone.quaternion), z[0] = N.x, z[1] = N.y, z[2] = -N.z, z[3] = N.length() / 3, i.children)
359
+ if (i = this.data[e], U.copy(i.vWorld), Pe.copy(i.boneParent.matrixWorld), Be.copy(Pe).invert(), i.vWorld.setFromMatrixPosition(Pe), U.applyMatrix4(Be), U.length() > 0.5 && (console.info("Info: Unrealistic jump of " + U.length().toFixed(2) + " meters."), U.setLength(0.5)), U.applyQuaternion(i.bone.quaternion), T[0] = U.x, T[1] = U.y, T[2] = -U.z, T[3] = U.length() / 3, i.children)
360
360
  for (t = 0, s = i.children.length; t < s; t++)
361
- p = i.children[t], z[0] -= p.v[0] * n / 3, z[1] -= p.v[1] * n / 3, z[2] += p.v[2] * n / 3, z[3] -= p.v[3] * n / 3;
362
- if (p = this.opt.sensitivityFactor, z[0] *= i.ext * p, z[1] *= i.ext * p, z[2] *= i.ext * p, z[3] *= i.ext * p, i.isX && (p = z[0] / n, i.ea[0] = (p - i.ev[0]) / n, i.ev[0] = p, i.a[0] = -i.k[0] * i.p[0] - i.c[0] * i.v[0] - i.ea[0], i.p[0] += i.v[0] * n + i.a[0] * n * n / 2 + z[0], p = i.v[0] + i.a[0] * n / 2, p = -i.k[0] * i.p[0] - i.c[0] * p - i.ea[0], i.v[0] = i.v[0] + (p + i.a[0]) * n / 2), i.isY && (p = z[1] / n, i.ea[1] = (p - i.ev[1]) / n, i.ev[1] = p, i.a[1] = -i.k[1] * i.p[1] - i.c[1] * i.v[1] - i.ea[1], i.p[1] += i.v[1] * n + i.a[1] * n * n / 2 + z[1], p = i.v[1] + i.a[1] * n / 2, p = -i.k[1] * i.p[1] - i.c[1] * p - i.ea[1], i.v[1] = i.v[1] + (p + i.a[1]) * n / 2), i.isZ && (p = z[2] / n, i.ea[2] = (p - i.ev[2]) / n, i.ev[2] = p, i.a[2] = -i.k[2] * i.p[2] - i.c[2] * i.v[2] - i.ea[2], i.p[2] += i.v[2] * n + i.a[2] * n * n / 2 + z[2], p = i.v[2] + i.a[2] * n / 2, p = -i.k[2] * i.p[2] - i.c[2] * p - i.ea[2], i.v[2] = i.v[2] + (p + i.a[2]) * n / 2), i.isT && (p = z[3] / n, i.ea[3] = (p - i.ev[3]) / n, i.ev[3] = p, i.a[3] = -i.k[3] * i.p[3] - i.c[3] * i.v[3] - i.ea[3], i.p[3] += i.v[3] * n + i.a[3] * n * n / 2 + z[3], p = i.v[3] + i.a[3] * n / 2, p = -i.k[3] * i.p[3] - i.c[3] * p - i.ea[3], i.v[3] = i.v[3] + (p + i.a[3]) * n / 2), this.timerMs < this.opt.warmupMs && (i.v[0] *= 1e-4, i.p[0] *= 1e-4, i.v[1] *= 1e-4, i.p[1] *= 1e-4, i.v[2] *= 1e-4, i.p[2] *= 1e-4, i.v[3] *= 1e-4, i.p[3] *= 1e-4), z[0] = i.p[0], z[1] = i.p[1], z[2] = i.p[2], z[3] = i.p[3], p = this.opt.movementFactor, z[0] *= p, z[1] *= p, z[2] *= p, z[3] *= p, i.dl && (p = i.dl, z[0] += p[0], z[1] += p[1], z[2] += p[2]), i.dw && (p = i.dw, N.set(
363
- i.vBasis.x + z[0],
364
- i.vBasis.y + z[1],
365
- i.vBasis.z + z[2]
366
- ), N.applyMatrix4(Oe), N.x += p[0], N.y += p[1], N.z += p[2], N.applyMatrix4(De), z[0] += N.x - i.vBasis.x, z[1] += N.y - i.vBasis.y, z[2] += N.z - i.vBasis.z), i.limits && this.opt.isLimits && (p = i.limits, p[0] && (p[0][0] !== null && z[0] < p[0][0] && (z[0] = p[0][0]), p[0][1] !== null && z[0] > p[0][1] && (z[0] = p[0][1])), p[1] && (p[1][0] !== null && z[1] < p[1][0] && (z[1] = p[1][0]), p[1][1] !== null && z[1] > p[1][1] && (z[1] = p[1][1])), p[2] && (p[2][0] !== null && z[2] < p[2][0] && (z[2] = p[2][0]), p[2][1] !== null && z[2] > p[2][1] && (z[2] = p[2][1])), p[3] && (p[3][0] !== null && z[3] < p[3][0] && (z[3] = p[3][0]), p[3][1] !== null && z[3] > p[3][1] && (z[3] = p[3][1]))), i.isPoint)
361
+ m = i.children[t], T[0] -= m.v[0] * n / 3, T[1] -= m.v[1] * n / 3, T[2] += m.v[2] * n / 3, T[3] -= m.v[3] * n / 3;
362
+ if (m = this.opt.sensitivityFactor, T[0] *= i.ext * m, T[1] *= i.ext * m, T[2] *= i.ext * m, T[3] *= i.ext * m, i.isX && (m = T[0] / n, i.ea[0] = (m - i.ev[0]) / n, i.ev[0] = m, i.a[0] = -i.k[0] * i.p[0] - i.c[0] * i.v[0] - i.ea[0], i.p[0] += i.v[0] * n + i.a[0] * n * n / 2 + T[0], m = i.v[0] + i.a[0] * n / 2, m = -i.k[0] * i.p[0] - i.c[0] * m - i.ea[0], i.v[0] = i.v[0] + (m + i.a[0]) * n / 2), i.isY && (m = T[1] / n, i.ea[1] = (m - i.ev[1]) / n, i.ev[1] = m, i.a[1] = -i.k[1] * i.p[1] - i.c[1] * i.v[1] - i.ea[1], i.p[1] += i.v[1] * n + i.a[1] * n * n / 2 + T[1], m = i.v[1] + i.a[1] * n / 2, m = -i.k[1] * i.p[1] - i.c[1] * m - i.ea[1], i.v[1] = i.v[1] + (m + i.a[1]) * n / 2), i.isZ && (m = T[2] / n, i.ea[2] = (m - i.ev[2]) / n, i.ev[2] = m, i.a[2] = -i.k[2] * i.p[2] - i.c[2] * i.v[2] - i.ea[2], i.p[2] += i.v[2] * n + i.a[2] * n * n / 2 + T[2], m = i.v[2] + i.a[2] * n / 2, m = -i.k[2] * i.p[2] - i.c[2] * m - i.ea[2], i.v[2] = i.v[2] + (m + i.a[2]) * n / 2), i.isT && (m = T[3] / n, i.ea[3] = (m - i.ev[3]) / n, i.ev[3] = m, i.a[3] = -i.k[3] * i.p[3] - i.c[3] * i.v[3] - i.ea[3], i.p[3] += i.v[3] * n + i.a[3] * n * n / 2 + T[3], m = i.v[3] + i.a[3] * n / 2, m = -i.k[3] * i.p[3] - i.c[3] * m - i.ea[3], i.v[3] = i.v[3] + (m + i.a[3]) * n / 2), this.timerMs < this.opt.warmupMs && (i.v[0] *= 1e-4, i.p[0] *= 1e-4, i.v[1] *= 1e-4, i.p[1] *= 1e-4, i.v[2] *= 1e-4, i.p[2] *= 1e-4, i.v[3] *= 1e-4, i.p[3] *= 1e-4), T[0] = i.p[0], T[1] = i.p[1], T[2] = i.p[2], T[3] = i.p[3], m = this.opt.movementFactor, T[0] *= m, T[1] *= m, T[2] *= m, T[3] *= m, i.dl && (m = i.dl, T[0] += m[0], T[1] += m[1], T[2] += m[2]), i.dw && (m = i.dw, U.set(
363
+ i.vBasis.x + T[0],
364
+ i.vBasis.y + T[1],
365
+ i.vBasis.z + T[2]
366
+ ), U.applyMatrix4(Pe), U.x += m[0], U.y += m[1], U.z += m[2], U.applyMatrix4(Be), T[0] += U.x - i.vBasis.x, T[1] += U.y - i.vBasis.y, T[2] += U.z - i.vBasis.z), i.limits && this.opt.isLimits && (m = i.limits, m[0] && (m[0][0] !== null && T[0] < m[0][0] && (T[0] = m[0][0]), m[0][1] !== null && T[0] > m[0][1] && (T[0] = m[0][1])), m[1] && (m[1][0] !== null && T[1] < m[1][0] && (T[1] = m[1][0]), m[1][1] !== null && T[1] > m[1][1] && (T[1] = m[1][1])), m[2] && (m[2][0] !== null && T[2] < m[2][0] && (T[2] = m[2][0]), m[2][1] !== null && T[2] > m[2][1] && (T[2] = m[2][1])), m[3] && (m[3][0] !== null && T[3] < m[3][0] && (T[3] = m[3][0]), m[3][1] !== null && T[3] > m[3][1] && (T[3] = m[3][1]))), i.isPoint)
367
367
  i.bone.position.set(
368
- i.vBasis.x + z[0],
369
- i.vBasis.y + z[1],
370
- i.vBasis.z - z[2]
368
+ i.vBasis.x + T[0],
369
+ i.vBasis.y + T[1],
370
+ i.vBasis.z - T[2]
371
371
  );
372
- else if (i.boneParent.quaternion.copy(i.qBasis), i.pivot && this.opt.isPivots && (i.boneParent.updateWorldMatrix(!1, !1), i.boneParent.matrixWorld.decompose(N, Re, Ae), N.copy(qe).applyQuaternion(Re).setY(0).normalize(), Re.premultiply(rt.setFromUnitVectors(qe, N).invert()).normalize(), i.boneParent.quaternion.multiply(Re.invert()), i.boneParent.quaternion.multiply(i.qWorldInverseYaw)), i.isZ && (p = Math.atan(z[0] / i.l), Re.setFromAxisAngle(Lt, -p), i.boneParent.quaternion.multiply(Re)), i.isY && (p = i.l / 3, p = p * Math.tanh(z[1] / p), i.bone.position.setLength(i.l + p)), i.isX && (p = Math.atan(z[2] / i.l), Re.setFromAxisAngle(vt, -p), i.boneParent.quaternion.multiply(Re)), i.isT && (p = 1.5 * Math.tanh(z[3] * 1.5), Re.setFromAxisAngle(It, -p), i.boneParent.quaternion.multiply(Re)), i.boneParent.updateWorldMatrix(!1, !0), i.excludes && this.opt.isExcludes)
372
+ else if (i.boneParent.quaternion.copy(i.qBasis), i.pivot && this.opt.isPivots && (i.boneParent.updateWorldMatrix(!1, !1), i.boneParent.matrixWorld.decompose(U, Ie, ve), U.copy(qe).applyQuaternion(Ie).setY(0).normalize(), Ie.premultiply(st.setFromUnitVectors(qe, U).invert()).normalize(), i.boneParent.quaternion.multiply(Ie.invert()), i.boneParent.quaternion.multiply(i.qWorldInverseYaw)), i.isZ && (m = Math.atan(T[0] / i.l), Ie.setFromAxisAngle(St, -m), i.boneParent.quaternion.multiply(Ie)), i.isY && (m = i.l / 3, m = m * Math.tanh(T[1] / m), i.bone.position.setLength(i.l + m)), i.isX && (m = Math.atan(T[2] / i.l), Ie.setFromAxisAngle(It, -m), i.boneParent.quaternion.multiply(Ie)), i.isT && (m = 1.5 * Math.tanh(T[3] * 1.5), Ie.setFromAxisAngle(Lt, -m), i.boneParent.quaternion.multiply(Ie)), i.boneParent.updateWorldMatrix(!1, !0), i.excludes && this.opt.isExcludes)
373
373
  for (t = 0, s = i.excludes.length; t < s; t++)
374
- p = i.excludes[t], Ae.set(0, 0, 0), p.deltaLocal && (Ae.x += p.deltaLocal[0], Ae.y += p.deltaLocal[1], Ae.z += p.deltaLocal[2]), Ae.applyMatrix4(p.bone.matrixWorld), De.copy(i.boneParent.matrixWorld).invert(), Ae.applyMatrix4(De), N.copy(i.bone.position), !(N.distanceToSquared(Ae) >= p.radiusSq) && (Me = N.length(), we = Ae.length(), !(we > p.radius + Me) && (we < Math.abs(p.radius - Me) || (we = (we * we + Me * Me - p.radiusSq) / (2 * we), Ae.normalize(), Ye.copy(Ae).multiplyScalar(we), we = Math.sqrt(Me * Me - we * we), N.subVectors(N, Ye).projectOnPlane(Ae).normalize().multiplyScalar(we), Qe.subVectors(i.vBasis, Ye).projectOnPlane(Ae).normalize(), Me = Qe.dot(N), Me < 0 && (Me = Math.sqrt(we * we - Me * Me), Qe.multiplyScalar(Me), N.add(Qe)), N.add(Ye).normalize(), Ae.copy(i.bone.position).normalize(), Re.setFromUnitVectors(Ae, N), i.boneParent.quaternion.premultiply(Re), i.boneParent.updateWorldMatrix(!1, !0))));
374
+ m = i.excludes[t], ve.set(0, 0, 0), m.deltaLocal && (ve.x += m.deltaLocal[0], ve.y += m.deltaLocal[1], ve.z += m.deltaLocal[2]), ve.applyMatrix4(m.bone.matrixWorld), Be.copy(i.boneParent.matrixWorld).invert(), ve.applyMatrix4(Be), U.copy(i.bone.position), !(U.distanceToSquared(ve) >= m.radiusSq) && (He = U.length(), Ce = ve.length(), !(Ce > m.radius + He) && (Ce < Math.abs(m.radius - He) || (Ce = (Ce * Ce + He * He - m.radiusSq) / (2 * Ce), ve.normalize(), Ye.copy(ve).multiplyScalar(Ce), Ce = Math.sqrt(He * He - Ce * Ce), U.subVectors(U, Ye).projectOnPlane(ve).normalize().multiplyScalar(Ce), Qe.subVectors(i.vBasis, Ye).projectOnPlane(ve).normalize(), He = Qe.dot(U), He < 0 && (He = Math.sqrt(Ce * Ce - He * He), Qe.multiplyScalar(He), U.add(Qe)), U.add(Ye).normalize(), ve.copy(i.bone.position).normalize(), Ie.setFromUnitVectors(ve, U), i.boneParent.quaternion.premultiply(Ie), i.boneParent.updateWorldMatrix(!1, !0))));
375
375
  }
376
376
  this.helpers.isActive && this.updateHelpers();
377
377
  }
@@ -382,18 +382,18 @@ class St {
382
382
  * @param {boolean} [keepSetting=false] If true, keep the previos all setting
383
383
  */
384
384
  showHelpers(n) {
385
- if (this.hideHelpers(), this.helpers.isShowAll = n === void 0 ? this.helpers.isShowAll : n === !0, p = this.helpers, this.data.forEach((e) => {
386
- (this.helpers.isShowAll || e.helper === !0) && (p.points.bones.push(e.bone), p.points.pivots.push(e.pivot), e.type !== 0 && p.lines.bones.push(e.bone), e.excludes && e.excludes.forEach((t) => {
385
+ if (this.hideHelpers(), this.helpers.isShowAll = n === void 0 ? this.helpers.isShowAll : n === !0, m = this.helpers, this.data.forEach((e) => {
386
+ (this.helpers.isShowAll || e.helper === !0) && (m.points.bones.push(e.bone), m.points.pivots.push(e.pivot), e.type !== 0 && m.lines.bones.push(e.bone), e.excludes && e.excludes.forEach((t) => {
387
387
  let o = !1;
388
- for (let s = 0; s < p.excludes.bones.length; s++)
389
- if (p.excludes.bones[s] === t.bone && p.excludes.radii[s] === t.radius && !(p.excludes.deltaLocals[s] === null && t.deltaLocal !== null) && !(p.excludes.deltaLocals[s] !== null && t.deltaLocal === null) && !(p.excludes.deltaLocals[s] !== null && p.excludes.deltaLocals[s].some((i, a) => i !== t.deltaLocal[a]))) {
388
+ for (let s = 0; s < m.excludes.bones.length; s++)
389
+ if (m.excludes.bones[s] === t.bone && m.excludes.radii[s] === t.radius && !(m.excludes.deltaLocals[s] === null && t.deltaLocal !== null) && !(m.excludes.deltaLocals[s] !== null && t.deltaLocal === null) && !(m.excludes.deltaLocals[s] !== null && m.excludes.deltaLocals[s].some((i, a) => i !== t.deltaLocal[a]))) {
390
390
  o = !0;
391
391
  break;
392
392
  }
393
- o || (p.excludes.bones.push(t.bone), p.excludes.radii.push(t.radius), p.excludes.deltaLocals.push(t.deltaLocal ? [...t.deltaLocal] : null), p.excludes.objects.push(null));
393
+ o || (m.excludes.bones.push(t.bone), m.excludes.radii.push(t.radius), m.excludes.deltaLocals.push(t.deltaLocal ? [...t.deltaLocal] : null), m.excludes.objects.push(null));
394
394
  }));
395
- }), p = this.helpers.excludes, this.opt.isExcludes && p.bones.length && p.bones.forEach((e, t) => {
396
- const o = new A.SphereGeometry(p.radii[t], 6, 6), s = new A.MeshBasicMaterial({
395
+ }), m = this.helpers.excludes, this.opt.isExcludes && m.bones.length && m.bones.forEach((e, t) => {
396
+ const o = new A.SphereGeometry(m.radii[t], 6, 6), s = new A.MeshBasicMaterial({
397
397
  depthTest: !1,
398
398
  depthWrite: !1,
399
399
  toneMapped: !1,
@@ -401,16 +401,16 @@ class St {
401
401
  wireframe: !0,
402
402
  color: this.opt.helperExcludesColor
403
403
  });
404
- p.objects[t] = new A.Mesh(o, s), p.objects[t].renderOrder = 997, e.add(p.objects[t]), p.deltaLocals[t] && p.objects[t].position.set(
405
- p.deltaLocals[t][0],
406
- p.deltaLocals[t][1],
407
- p.deltaLocals[t][2]
404
+ m.objects[t] = new A.Mesh(o, s), m.objects[t].renderOrder = 997, e.add(m.objects[t]), m.deltaLocals[t] && m.objects[t].position.set(
405
+ m.deltaLocals[t][0],
406
+ m.deltaLocals[t][1],
407
+ m.deltaLocals[t][2]
408
408
  );
409
- }), p = this.helpers.points, p.bones.length) {
409
+ }), m = this.helpers.points, m.bones.length) {
410
410
  this.helpers.isActive = !0;
411
- const e = new A.BufferGeometry(), t = p.bones.map((u) => [0, 0, 0]).flat();
411
+ const e = new A.BufferGeometry(), t = m.bones.map((u) => [0, 0, 0]).flat();
412
412
  e.setAttribute("position", new A.Float32BufferAttribute(t, 3));
413
- const o = new A.Color(this.opt.helperBoneColor1), s = new A.Color(this.opt.helperBoneColor2), i = p.pivots.map((u) => u && this.opt.isPivots ? [s.r, s.g, s.b] : [o.r, o.g, o.b]).flat();
413
+ const o = new A.Color(this.opt.helperBoneColor1), s = new A.Color(this.opt.helperBoneColor2), i = m.pivots.map((u) => u && this.opt.isPivots ? [s.r, s.g, s.b] : [o.r, o.g, o.b]).flat();
414
414
  e.setAttribute("color", new A.Float32BufferAttribute(i, 3));
415
415
  const a = new A.PointsMaterial({
416
416
  depthTest: !1,
@@ -420,12 +420,12 @@ class St {
420
420
  size: 0.2,
421
421
  vertexColors: !0
422
422
  });
423
- p.object = new A.Points(e, a), p.object.renderOrder = 998, p.object.matrix = this.armature.matrixWorld, p.object.matrixAutoUpdate = !1, this.scene.add(p.object);
423
+ m.object = new A.Points(e, a), m.object.renderOrder = 998, m.object.matrix = this.armature.matrixWorld, m.object.matrixAutoUpdate = !1, this.scene.add(m.object);
424
424
  }
425
- if (p = this.helpers.lines, p.bones.length) {
426
- const e = new A.BufferGeometry(), t = p.bones.map((u) => [0, 0, 0, 0, 0, 0]).flat();
425
+ if (m = this.helpers.lines, m.bones.length) {
426
+ const e = new A.BufferGeometry(), t = m.bones.map((u) => [0, 0, 0, 0, 0, 0]).flat();
427
427
  e.setAttribute("position", new A.Float32BufferAttribute(t, 3));
428
- const o = new A.Color(this.opt.helperLinkColor1), s = new A.Color(this.opt.helperLinkColor2), i = p.bones.map((u) => [o.r, o.g, o.b, s.r, s.g, s.b]).flat();
428
+ const o = new A.Color(this.opt.helperLinkColor1), s = new A.Color(this.opt.helperLinkColor2), i = m.bones.map((u) => [o.r, o.g, o.b, s.r, s.g, s.b]).flat();
429
429
  e.setAttribute("color", new A.Float32BufferAttribute(i, 3));
430
430
  const a = new A.LineBasicMaterial({
431
431
  vertexColors: !0,
@@ -434,26 +434,26 @@ class St {
434
434
  toneMapped: !1,
435
435
  transparent: !0
436
436
  });
437
- p.object = new A.LineSegments(e, a), p.object.renderOrder = 999, p.object.matrix = this.armature.matrixWorld, p.object.matrixAutoUpdate = !1, this.scene.add(p.object);
437
+ m.object = new A.LineSegments(e, a), m.object.renderOrder = 999, m.object.matrix = this.armature.matrixWorld, m.object.matrixAutoUpdate = !1, this.scene.add(m.object);
438
438
  }
439
439
  }
440
440
  /**
441
441
  * Update the positions of dynamic bone helpers.
442
442
  */
443
443
  updateHelpers() {
444
- if (p = this.helpers.points, p.bones.length) {
445
- De.copy(this.armature.matrixWorld).invert();
446
- const n = p.object.geometry.getAttribute("position");
447
- for (let e = 0, t = p.bones.length; e < t; e++)
448
- Oe.multiplyMatrices(De, p.bones[e].matrixWorld), N.setFromMatrixPosition(Oe), n.setXYZ(e, N.x, N.y, N.z);
449
- n.needsUpdate = !0, p.object.updateMatrixWorld();
444
+ if (m = this.helpers.points, m.bones.length) {
445
+ Be.copy(this.armature.matrixWorld).invert();
446
+ const n = m.object.geometry.getAttribute("position");
447
+ for (let e = 0, t = m.bones.length; e < t; e++)
448
+ Pe.multiplyMatrices(Be, m.bones[e].matrixWorld), U.setFromMatrixPosition(Pe), n.setXYZ(e, U.x, U.y, U.z);
449
+ n.needsUpdate = !0, m.object.updateMatrixWorld();
450
450
  }
451
- if (p = this.helpers.lines, p.bones.length) {
452
- De.copy(this.armature.matrixWorld).invert();
453
- const n = p.object.geometry.getAttribute("position");
454
- for (let e = 0, t = 0, o = p.bones.length; e < o; e++, t += 2)
455
- Oe.multiplyMatrices(De, p.bones[e].matrixWorld), N.setFromMatrixPosition(Oe), n.setXYZ(t, N.x, N.y, N.z), Oe.multiplyMatrices(De, p.bones[e].parent.matrixWorld), N.setFromMatrixPosition(Oe), n.setXYZ(t + 1, N.x, N.y, N.z);
456
- n.needsUpdate = !0, p.object.updateMatrixWorld();
451
+ if (m = this.helpers.lines, m.bones.length) {
452
+ Be.copy(this.armature.matrixWorld).invert();
453
+ const n = m.object.geometry.getAttribute("position");
454
+ for (let e = 0, t = 0, o = m.bones.length; e < o; e++, t += 2)
455
+ Pe.multiplyMatrices(Be, m.bones[e].matrixWorld), U.setFromMatrixPosition(Pe), n.setXYZ(t, U.x, U.y, U.z), Pe.multiplyMatrices(Be, m.bones[e].parent.matrixWorld), U.setFromMatrixPosition(Pe), n.setXYZ(t + 1, U.x, U.y, U.z);
456
+ n.needsUpdate = !0, m.object.updateMatrixWorld();
457
457
  }
458
458
  }
459
459
  /**
@@ -462,9 +462,9 @@ class St {
462
462
  hideHelpers() {
463
463
  [this.helpers.points, this.helpers.lines].forEach((n) => {
464
464
  n.bones = [], n.object && (this.scene.remove(n.object), n.object.geometry.dispose(), n.object.material.dispose(), n.object = null);
465
- }), p = this.helpers.excludes, p.objects.forEach((n, e) => {
466
- n && (p.bones[e].remove(n), n.geometry.dispose(), n.material.dispose());
467
- }), p.bones = [], p.deltaLocals = [], p.radii = [], p.objects = [], this.helpers.isActive = !1;
465
+ }), m = this.helpers.excludes, m.objects.forEach((n, e) => {
466
+ n && (m.bones[e].remove(n), n.geometry.dispose(), n.material.dispose());
467
+ }), m.bones = [], m.deltaLocals = [], m.radii = [], m.objects = [], this.helpers.isActive = !1;
468
468
  }
469
469
  /**
470
470
  * Start dynamic bones.
@@ -489,7 +489,7 @@ class St {
489
489
  this.stop(), this.scene = null, this.armature = null, this.config = [], this.data = [], this.dict = {}, this.objectsUpdate = [], this.timerMs = 0;
490
490
  }
491
491
  }
492
- class kt {
492
+ class wt {
493
493
  constructor(n) {
494
494
  this.audioContext = n, this.analyzer = null, this.dataArray = null, this.bufferLength = 0;
495
495
  }
@@ -525,8 +525,8 @@ class kt {
525
525
  t.spectralCentroid.push(h);
526
526
  const d = this.calculateZeroCrossingRate(c);
527
527
  t.zeroCrossingRate.push(d);
528
- const g = this.calculateMFCC(c);
529
- t.mfcc.push(g);
528
+ const p = this.calculateMFCC(c);
529
+ t.mfcc.push(p);
530
530
  }
531
531
  return t.onsets = this.detectOnsets(t.energy), t.phonemeBoundaries = this.detectPhonemeBoundaries(t), t;
532
532
  }
@@ -606,10 +606,10 @@ class kt {
606
606
  for (let u = 0; u < e; u += o) {
607
607
  let l = 1, c = 0;
608
608
  for (let r = 0; r < o / 2; r++) {
609
- const h = t[(u + r) * 2], d = t[(u + r) * 2 + 1], g = t[(u + r + o / 2) * 2] * l - t[(u + r + o / 2) * 2 + 1] * c, b = t[(u + r + o / 2) * 2] * c + t[(u + r + o / 2) * 2 + 1] * l;
610
- t[(u + r) * 2] = h + g, t[(u + r) * 2 + 1] = d + b, t[(u + r + o / 2) * 2] = h - g, t[(u + r + o / 2) * 2 + 1] = d - b;
611
- const x = l * i - c * a, k = l * a + c * i;
612
- l = x, c = k;
609
+ const h = t[(u + r) * 2], d = t[(u + r) * 2 + 1], p = t[(u + r + o / 2) * 2] * l - t[(u + r + o / 2) * 2 + 1] * c, b = t[(u + r + o / 2) * 2] * c + t[(u + r + o / 2) * 2 + 1] * l;
610
+ t[(u + r) * 2] = h + p, t[(u + r) * 2 + 1] = d + b, t[(u + r + o / 2) * 2] = h - p, t[(u + r + o / 2) * 2 + 1] = d - b;
611
+ const f = l * i - c * a, L = l * a + c * i;
612
+ l = f, c = L;
613
613
  }
614
614
  }
615
615
  }
@@ -814,7 +814,7 @@ class kt {
814
814
  return o * s;
815
815
  }
816
816
  }
817
- class wt {
817
+ class Ct {
818
818
  /**
819
819
  * @constructor
820
820
  */
@@ -1208,10 +1208,10 @@ class wt {
1208
1208
  this.rules[e] = this.rules[e].map((t) => {
1209
1209
  const o = t.indexOf("["), s = t.indexOf("]"), i = t.indexOf("="), a = t.substring(0, o), u = t.substring(o + 1, s), l = t.substring(s + 1, i), c = t.substring(i + 1), r = { regex: "", move: 0, visemes: [] };
1210
1210
  let h = "";
1211
- h += [...a].map((g) => n[g] || g).join("");
1211
+ h += [...a].map((p) => n[p] || p).join("");
1212
1212
  const d = [...u];
1213
- return d[0] = d[0].toLowerCase(), h += d.join(""), r.move = d.length, h += [...l].map((g) => n[g] || g).join(""), r.regex = new RegExp(h), c.length && c.split(" ").forEach((g) => {
1214
- r.visemes.push(g);
1213
+ return d[0] = d[0].toLowerCase(), h += d.join(""), r.move = d.length, h += [...l].map((p) => n[p] || p).join(""), r.regex = new RegExp(h), c.length && c.split(" ").forEach((p) => {
1214
+ r.visemes.push(p);
1215
1215
  }), r;
1216
1216
  });
1217
1217
  }), this.visemeDurations = {
@@ -1396,11 +1396,11 @@ class wt {
1396
1396
  return e;
1397
1397
  }
1398
1398
  }
1399
- const Ct = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
1399
+ const zt = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
1400
1400
  __proto__: null,
1401
- LipsyncEn: wt
1401
+ LipsyncEn: Ct
1402
1402
  }, Symbol.toStringTag, { value: "Module" }));
1403
- class zt {
1403
+ class Ht {
1404
1404
  /**
1405
1405
  * @constructor
1406
1406
  */
@@ -1618,10 +1618,10 @@ class zt {
1618
1618
  this.rules[e] = this.rules[e].map((t) => {
1619
1619
  const o = t.indexOf("["), s = t.indexOf("]"), i = t.indexOf("="), a = t.substring(0, o), u = t.substring(o + 1, s), l = t.substring(s + 1, i), c = t.substring(i + 1), r = { regex: "", move: 0, visemes: [] };
1620
1620
  let h = "";
1621
- h += [...a].map((g) => n[g] || g).join("");
1621
+ h += [...a].map((p) => n[p] || p).join("");
1622
1622
  const d = [...u];
1623
- return d[0] = d[0].toLowerCase(), h += d.join(""), r.move = d.length, h += [...l].map((g) => n[g] || g).join(""), r.regex = new RegExp(h), c.length && c.split(" ").forEach((g) => {
1624
- r.visemes.push(g);
1623
+ return d[0] = d[0].toLowerCase(), h += d.join(""), r.move = d.length, h += [...l].map((p) => n[p] || p).join(""), r.regex = new RegExp(h), c.length && c.split(" ").forEach((p) => {
1624
+ r.visemes.push(p);
1625
1625
  }), r;
1626
1626
  });
1627
1627
  }), this.visemeDurations = {
@@ -1754,11 +1754,11 @@ class zt {
1754
1754
  return e;
1755
1755
  }
1756
1756
  }
1757
- const Ht = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
1757
+ const Tt = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
1758
1758
  __proto__: null,
1759
- LipsyncDe: zt
1759
+ LipsyncDe: Ht
1760
1760
  }, Symbol.toStringTag, { value: "Module" }));
1761
- class Tt {
1761
+ class Ft {
1762
1762
  /**
1763
1763
  * @constructor
1764
1764
  */
@@ -2133,10 +2133,10 @@ class Tt {
2133
2133
  this.rules[e] = this.rules[e].map((t) => {
2134
2134
  const o = t.indexOf("["), s = t.indexOf("]"), i = t.indexOf("="), a = t.substring(0, o), u = t.substring(o + 1, s), l = t.substring(s + 1, i), c = t.substring(i + 1), r = { regex: "", move: 0, visemes: [] };
2135
2135
  let h = "";
2136
- h += [...a].map((g) => n[g] || g).join("");
2136
+ h += [...a].map((p) => n[p] || p).join("");
2137
2137
  const d = [...u];
2138
- return d[0] = d[0].toLowerCase(), h += d.join(""), r.move = d.length, h += [...l].map((g) => n[g] || g).join(""), r.regex = new RegExp(h, "i"), c.length && c.split(" ").forEach((g) => {
2139
- g && r.visemes.push(g);
2138
+ return d[0] = d[0].toLowerCase(), h += d.join(""), r.move = d.length, h += [...l].map((p) => n[p] || p).join(""), r.regex = new RegExp(h, "i"), c.length && c.split(" ").forEach((p) => {
2139
+ p && r.visemes.push(p);
2140
2140
  }), r;
2141
2141
  });
2142
2142
  }), this.visemeDurations = {
@@ -2289,11 +2289,11 @@ class Tt {
2289
2289
  return e;
2290
2290
  }
2291
2291
  }
2292
- const Ft = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
2292
+ const Mt = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
2293
2293
  __proto__: null,
2294
- LipsyncFr: Tt
2294
+ LipsyncFr: Ft
2295
2295
  }, Symbol.toStringTag, { value: "Module" }));
2296
- class Mt {
2296
+ class Et {
2297
2297
  /**
2298
2298
  * @constructor
2299
2299
  */
@@ -2436,11 +2436,11 @@ class Mt {
2436
2436
  return e;
2437
2437
  }
2438
2438
  }
2439
- const Et = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
2439
+ const Pt = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
2440
2440
  __proto__: null,
2441
- LipsyncFi: Mt
2441
+ LipsyncFi: Et
2442
2442
  }, Symbol.toStringTag, { value: "Module" }));
2443
- class Pt {
2443
+ class Bt {
2444
2444
  /**
2445
2445
  * @constructor
2446
2446
  */
@@ -2620,21 +2620,21 @@ class Pt {
2620
2620
  return e;
2621
2621
  }
2622
2622
  }
2623
- const Bt = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
2623
+ const Ot = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
2624
2624
  __proto__: null,
2625
- LipsyncLt: Pt
2626
- }, Symbol.toStringTag, { value: "Module" })), Ot = new URL("data:text/javascript;base64,class PlaybackWorklet extends AudioWorkletProcessor {
  static FSM = {
    IDLE: 0,
    PLAYING: 1,
  };

  constructor(options) {
    super();
    this.port.onmessage = this.handleMessage.bind(this);

    this._sampleRate = options?.processorOptions?.sampleRate || sampleRate;
    this._scale = 1 / 32768; // PCM16 -> float

    // Silence detection threshold (1 second) as a fallback safety net
    const silenceDurationSeconds = 1.0;
    this._silenceThresholdBlocks = Math.ceil((this._sampleRate * silenceDurationSeconds) / 128);

    // Metrics configuration via options
    const metricsCfg = options?.processorOptions?.metrics || {};
    this._metricsEnabled = metricsCfg.enabled !== false;
    const intervalHz = (typeof metricsCfg.intervalHz === "number" && metricsCfg.intervalHz > 0)
      ? metricsCfg.intervalHz : 2;
    // Metrics state (low-overhead)
    this._framesProcessed = 0;
    this._underrunBlocks = 0;
    this._maxQueueSamples = 0;
    this._lastMetricsSentAtFrame = 0;
    // Convert to frames between reports
    this._metricsIntervalFrames = Math.max(128, Math.round(this._sampleRate / intervalHz));

    this.reset();
  }

  /**
   * Resets the worklet to its initial IDLE state.
   */
  reset() {
    this._bufferQueue = [];
    this._currentChunk = null;
    this._currentChunkOffset = 0;
    this._state = PlaybackWorklet.FSM.IDLE;

    this._noMoreDataReceived = false;
    this._silenceFramesCount = 0;
    this._hasSentEnded = false;
    // Reset max queue tracker only when going idle
    this._maxQueueSamples = 0;
  }

  handleMessage(event) {
    const { type, data } = event.data;

    // INTERRUPT: The main thread wants to stop immediately.
    if (type === "stop") {
      this.reset();
      // Send final metrics showing cleared state
      if (this._metricsEnabled) {
        try {
          this.port.postMessage({
            type: "metrics",
            data: {
              state: PlaybackWorklet.FSM.IDLE,
              queuedSamples: 0,
              queuedMs: 0,
              maxQueuedMs: Math.round((this._maxQueueSamples / this._sampleRate) * 1000),
              underrunBlocks: this._underrunBlocks,
              framesProcessed: this._framesProcessed
            }
          });
        } catch (_) { }
      }
      return;
    }

    // Main thread has signaled that no more audio chunks will be sent for this utterance.
    if (type === "no-more-data") {
      this._noMoreDataReceived = true;
      return;
    }

    // Update metrics configuration at runtime
    if (type === "config-metrics" && data && typeof data === "object") {
      if ("enabled" in data) this._metricsEnabled = !!data.enabled;
      if (typeof data.intervalHz === "number" && data.intervalHz > 0) {
        const intervalHz = data.intervalHz;
        this._metricsIntervalFrames = Math.max(128, Math.round(this._sampleRate / intervalHz));
      }
      // Reset pacing so the next report aligns with new interval
      this._lastMetricsSentAtFrame = this._framesProcessed;
      return;
    }

    // New audio data has arrived.
    if (type === "audioData" && data instanceof ArrayBuffer) {
      this._noMoreDataReceived = false;
      // If we were idle, this new data kicks off the playback.
      if (this._state === PlaybackWorklet.FSM.IDLE) {
        this._state = PlaybackWorklet.FSM.PLAYING;
        this.port.postMessage({ type: "playback-started" });
      }

      // We only queue data if we are in the PLAYING state. This prevents
      // data from a previous, interrupted stream from lingering.
      if (this._state === PlaybackWorklet.FSM.PLAYING) {
        // Store as Int16Array view to avoid constructing it in process()
        this._bufferQueue.push(new Int16Array(data));
        this._silenceFramesCount = 0; // Reset silence counter on new data
      }
    }
  }

  process(inputs, outputs, parameters) {
    const outputChannel = outputs[0]?.[0];
    if (!outputChannel) {
      return true; // Keep alive even if output is temporarily disconnected
    }

    // If we are not playing, just output silence and wait.
    if (this._state !== PlaybackWorklet.FSM.PLAYING) {
      outputChannel.fill(0);
      return true; // Always return true to keep the processor alive
    }

    // Core PLAYING Logic
    const blockSize = outputChannel.length;
    let samplesCopied = 0;

    while (samplesCopied < blockSize) {
      if (!this._currentChunk || this._currentChunkOffset >= this._currentChunk.length) {
        if (this._bufferQueue.length > 0) {
          this._currentChunk = this._bufferQueue.shift();
          this._currentChunkOffset = 0;
        } else {
          // Buffer is empty. Check for end conditions.
          const isTimedOut = this._silenceFramesCount > this._silenceThresholdBlocks;

          if (this._noMoreDataReceived || isTimedOut) {
            // END OF PLAYBACK: Either explicitly signaled or timed out.
            if (!this._hasSentEnded) {
              this.port.postMessage({ type: "playback-ended" });
              this._hasSentEnded = true;
            }
            // Send final metrics showing cleared state
            if (this._metricsEnabled) {
              try {
                this.port.postMessage({
                  type: "metrics",
                  data: {
                    state: PlaybackWorklet.FSM.IDLE,
                    queuedSamples: 0,
                    queuedMs: 0,
                    maxQueuedMs: Math.round((this._maxQueueSamples / this._sampleRate) * 1000),
                    underrunBlocks: this._underrunBlocks,
                    framesProcessed: this._framesProcessed
                  }
                });
              } catch (_) { }
            }
            this.reset(); // Reset to IDLE state for reuse
            break; // Exit while loop
          } else {
            // BUFFER UNDERRUN (LAG): Play silence and wait for more data.
            this._silenceFramesCount++;
            if (this._metricsEnabled) this._underrunBlocks++;
            break; // Exit while loop
          }
        }
      }

      // If we have a chunk (could be a new one from the logic above), process it.
      if (this._currentChunk) {
        const samplesToCopy = Math.min(
          blockSize - samplesCopied,
          this._currentChunk.length - this._currentChunkOffset
        );
        // Directly write to outputChannel to avoid extra copy
        const src = this._currentChunk;
        const baseSrc = this._currentChunkOffset;
        const baseDst = samplesCopied;
        const scale = this._scale;
        for (let i = 0; i < samplesToCopy; i++) {
          outputChannel[baseDst + i] = src[baseSrc + i] * scale;
        }

        this._currentChunkOffset += samplesToCopy;
        samplesCopied += samplesToCopy;
      }
    }

    // Zero-fill the remainder, if any, once per block
    if (samplesCopied < blockSize) {
      outputChannel.fill(0, samplesCopied);
    }

    // Update metrics (optional)
    if (this._metricsEnabled) {
      this._framesProcessed += blockSize;

      // Track queue depth in samples (approximate)
      let queuedSamples = 0;
      if (this._currentChunk) queuedSamples += Math.max(0, this._currentChunk.length - this._currentChunkOffset);
      for (let i = 0; i < this._bufferQueue.length; i++) queuedSamples += this._bufferQueue[i].length;
      if (queuedSamples > this._maxQueueSamples) this._maxQueueSamples = queuedSamples;

      // Periodically send metrics to main thread
      if (this._framesProcessed - this._lastMetricsSentAtFrame >= this._metricsIntervalFrames) {
        this._lastMetricsSentAtFrame = this._framesProcessed;
        try {
          this.port.postMessage({
            type: "metrics",
            data: {
              state: this._state,
              queuedSamples,
              queuedMs: Math.round((queuedSamples / this._sampleRate) * 1000),
              maxQueuedMs: Math.round((this._maxQueueSamples / this._sampleRate) * 1000),
              underrunBlocks: this._underrunBlocks,
              framesProcessed: this._framesProcessed
            }
          });
        } catch (_) { }
        // Don't reset max tracker - keep session peak until idle
      }
    }

    // ALWAYS return true to keep the processor alive for reuse.
    return true;
  }
}

registerProcessor("playback-worklet", PlaybackWorklet);
", import.meta.url), at = {
2627
- en: Ct,
2628
- de: Ht,
2629
- fr: Ft,
2630
- fi: Et,
2631
- lt: Bt
2632
- }, ue = new A.Quaternion(), oe = new A.Euler(), Ne = new A.Vector3(), Xe = new A.Vector3(), lt = new A.Box3();
2625
+ LipsyncLt: Bt
2626
+ }, Symbol.toStringTag, { value: "Module" })), Dt = new URL("data:text/javascript;base64,class PlaybackWorklet extends AudioWorkletProcessor {
  static FSM = {
    IDLE: 0,
    PLAYING: 1,
  };

  constructor(options) {
    super();
    this.port.onmessage = this.handleMessage.bind(this);

    this._sampleRate = options?.processorOptions?.sampleRate || sampleRate;
    this._scale = 1 / 32768; // PCM16 -> float

    // Silence detection threshold (1 second) as a fallback safety net
    const silenceDurationSeconds = 1.0;
    this._silenceThresholdBlocks = Math.ceil((this._sampleRate * silenceDurationSeconds) / 128);

    // Metrics configuration via options
    const metricsCfg = options?.processorOptions?.metrics || {};
    this._metricsEnabled = metricsCfg.enabled !== false;
    const intervalHz = (typeof metricsCfg.intervalHz === "number" && metricsCfg.intervalHz > 0)
      ? metricsCfg.intervalHz : 2;
    // Metrics state (low-overhead)
    this._framesProcessed = 0;
    this._underrunBlocks = 0;
    this._maxQueueSamples = 0;
    this._lastMetricsSentAtFrame = 0;
    // Convert to frames between reports
    this._metricsIntervalFrames = Math.max(128, Math.round(this._sampleRate / intervalHz));

    this.reset();
  }

  /**
   * Resets the worklet to its initial IDLE state.
   */
  reset() {
    this._bufferQueue = [];
    this._currentChunk = null;
    this._currentChunkOffset = 0;
    this._state = PlaybackWorklet.FSM.IDLE;

    this._noMoreDataReceived = false;
    this._silenceFramesCount = 0;
    this._hasSentEnded = false;
    // Reset max queue tracker only when going idle
    this._maxQueueSamples = 0;
  }

  handleMessage(event) {
    const { type, data } = event.data;

    // INTERRUPT: The main thread wants to stop immediately.
    if (type === "stop") {
      this.reset();
      // Send final metrics showing cleared state
      if (this._metricsEnabled) {
        try {
          this.port.postMessage({
            type: "metrics",
            data: {
              state: PlaybackWorklet.FSM.IDLE,
              queuedSamples: 0,
              queuedMs: 0,
              maxQueuedMs: Math.round((this._maxQueueSamples / this._sampleRate) * 1000),
              underrunBlocks: this._underrunBlocks,
              framesProcessed: this._framesProcessed
            }
          });
        } catch (_) { }
      }
      return;
    }

    // Main thread has signaled that no more audio chunks will be sent for this utterance.
    if (type === "no-more-data") {
      this._noMoreDataReceived = true;
      return;
    }

    // Update metrics configuration at runtime
    if (type === "config-metrics" && data && typeof data === "object") {
      if ("enabled" in data) this._metricsEnabled = !!data.enabled;
      if (typeof data.intervalHz === "number" && data.intervalHz > 0) {
        const intervalHz = data.intervalHz;
        this._metricsIntervalFrames = Math.max(128, Math.round(this._sampleRate / intervalHz));
      }
      // Reset pacing so the next report aligns with new interval
      this._lastMetricsSentAtFrame = this._framesProcessed;
      return;
    }

    // New audio data has arrived.
    if (type === "audioData" && data instanceof ArrayBuffer) {
      this._noMoreDataReceived = false;
      // If we were idle, this new data kicks off the playback.
      if (this._state === PlaybackWorklet.FSM.IDLE) {
        this._state = PlaybackWorklet.FSM.PLAYING;
        this.port.postMessage({ type: "playback-started" });
      }

      // We only queue data if we are in the PLAYING state. This prevents
      // data from a previous, interrupted stream from lingering.
      if (this._state === PlaybackWorklet.FSM.PLAYING) {
        // Store as Int16Array view to avoid constructing it in process()
        this._bufferQueue.push(new Int16Array(data));
        this._silenceFramesCount = 0; // Reset silence counter on new data
      }
    }
  }

  process(inputs, outputs, parameters) {
    const outputChannel = outputs[0]?.[0];
    if (!outputChannel) {
      return true; // Keep alive even if output is temporarily disconnected
    }

    // If we are not playing, just output silence and wait.
    if (this._state !== PlaybackWorklet.FSM.PLAYING) {
      outputChannel.fill(0);
      return true; // Always return true to keep the processor alive
    }

    // Core PLAYING Logic
    const blockSize = outputChannel.length;
    let samplesCopied = 0;

    while (samplesCopied < blockSize) {
      if (!this._currentChunk || this._currentChunkOffset >= this._currentChunk.length) {
        if (this._bufferQueue.length > 0) {
          this._currentChunk = this._bufferQueue.shift();
          this._currentChunkOffset = 0;
        } else {
          // Buffer is empty. Check for end conditions.
          const isTimedOut = this._silenceFramesCount > this._silenceThresholdBlocks;

          if (this._noMoreDataReceived || isTimedOut) {
            // END OF PLAYBACK: Either explicitly signaled or timed out.
            if (!this._hasSentEnded) {
              this.port.postMessage({ type: "playback-ended" });
              this._hasSentEnded = true;
            }
            // Send final metrics showing cleared state
            if (this._metricsEnabled) {
              try {
                this.port.postMessage({
                  type: "metrics",
                  data: {
                    state: PlaybackWorklet.FSM.IDLE,
                    queuedSamples: 0,
                    queuedMs: 0,
                    maxQueuedMs: Math.round((this._maxQueueSamples / this._sampleRate) * 1000),
                    underrunBlocks: this._underrunBlocks,
                    framesProcessed: this._framesProcessed
                  }
                });
              } catch (_) { }
            }
            this.reset(); // Reset to IDLE state for reuse
            break; // Exit while loop
          } else {
            // BUFFER UNDERRUN (LAG): Play silence and wait for more data.
            this._silenceFramesCount++;
            if (this._metricsEnabled) this._underrunBlocks++;
            break; // Exit while loop
          }
        }
      }

      // If we have a chunk (could be a new one from the logic above), process it.
      if (this._currentChunk) {
        const samplesToCopy = Math.min(
          blockSize - samplesCopied,
          this._currentChunk.length - this._currentChunkOffset
        );
        // Directly write to outputChannel to avoid extra copy
        const src = this._currentChunk;
        const baseSrc = this._currentChunkOffset;
        const baseDst = samplesCopied;
        const scale = this._scale;
        for (let i = 0; i < samplesToCopy; i++) {
          outputChannel[baseDst + i] = src[baseSrc + i] * scale;
        }

        this._currentChunkOffset += samplesToCopy;
        samplesCopied += samplesToCopy;
      }
    }

    // Zero-fill the remainder, if any, once per block
    if (samplesCopied < blockSize) {
      outputChannel.fill(0, samplesCopied);
    }

    // Update metrics (optional)
    if (this._metricsEnabled) {
      this._framesProcessed += blockSize;

      // Track queue depth in samples (approximate)
      let queuedSamples = 0;
      if (this._currentChunk) queuedSamples += Math.max(0, this._currentChunk.length - this._currentChunkOffset);
      for (let i = 0; i < this._bufferQueue.length; i++) queuedSamples += this._bufferQueue[i].length;
      if (queuedSamples > this._maxQueueSamples) this._maxQueueSamples = queuedSamples;

      // Periodically send metrics to main thread
      if (this._framesProcessed - this._lastMetricsSentAtFrame >= this._metricsIntervalFrames) {
        this._lastMetricsSentAtFrame = this._framesProcessed;
        try {
          this.port.postMessage({
            type: "metrics",
            data: {
              state: this._state,
              queuedSamples,
              queuedMs: Math.round((queuedSamples / this._sampleRate) * 1000),
              maxQueuedMs: Math.round((this._maxQueueSamples / this._sampleRate) * 1000),
              underrunBlocks: this._underrunBlocks,
              framesProcessed: this._framesProcessed
            }
          });
        } catch (_) { }
        // Don't reset max tracker - keep session peak until idle
      }
    }

    // ALWAYS return true to keep the processor alive for reuse.
    return true;
  }
}

registerProcessor("playback-worklet", PlaybackWorklet);
", import.meta.url), rt = {
2627
+ en: zt,
2628
+ de: Tt,
2629
+ fr: Mt,
2630
+ fi: Pt,
2631
+ lt: Ot
2632
+ }, he = new A.Quaternion(), re = new A.Euler(), Ne = new A.Vector3(), We = new A.Vector3(), at = new A.Box3();
2633
2633
  new A.Matrix4();
2634
2634
  new A.Matrix4();
2635
2635
  new A.Vector3();
2636
2636
  new A.Vector3(0, 0, 1);
2637
- const Dt = new A.Vector3(1, 0, 0);
2637
+ const Nt = new A.Vector3(1, 0, 0);
2638
2638
  new A.Vector3(0, 1, 0);
2639
2639
  new A.Vector3(0, 0, 1);
2640
2640
  class it {
@@ -2763,7 +2763,7 @@ class it {
2763
2763
  avatarOnlyCamera: null,
2764
2764
  statsNode: null,
2765
2765
  statsStyle: null
2766
- }, Object.assign(this.opt, e || {}), this.opt.statsNode && (this.stats = new Rt(), this.opt.statsStyle && (this.stats.dom.style.cssText = this.opt.statsStyle), this.opt.statsNode.appendChild(this.stats.dom)), this.poseTemplates = {
2766
+ }, Object.assign(this.opt, e || {}), this.opt.statsNode && (this.stats = new vt(), this.opt.statsStyle && (this.stats.dom.style.cssText = this.opt.statsStyle), this.opt.statsNode.appendChild(this.stats.dom)), this.poseTemplates = {
2767
2767
  side: {
2768
2768
  standing: !0,
2769
2769
  props: {
@@ -4086,7 +4086,7 @@ class it {
4086
4086
  this.opt.lightSpotDispersion
4087
4087
  ), this.setLighting(this.opt);
4088
4088
  const a = new A.PMREMGenerator(this.renderer);
4089
- a.compileEquirectangularShader(), this.scene.environment = a.fromScene(new At()).texture, this.resizeobserver = new ResizeObserver(this.onResize.bind(this)), this.resizeobserver.observe(this.nodeAvatar), this.controls = new ft(this.camera, this.renderer.domElement), this.controls.enableZoom = this.opt.cameraZoomEnable, this.controls.enableRotate = this.opt.cameraRotateEnable, this.controls.enablePan = this.opt.cameraPanEnable, this.controls.minDistance = 2, this.controls.maxDistance = 2e3, this.controls.autoRotateSpeed = 0, this.controls.autoRotate = !1, this.controls.update(), this.cameraClock = null;
4089
+ a.compileEquirectangularShader(), this.scene.environment = a.fromScene(new Rt()).texture, this.resizeobserver = new ResizeObserver(this.onResize.bind(this)), this.resizeobserver.observe(this.nodeAvatar), this.controls = new xt(this.camera, this.renderer.domElement), this.controls.enableZoom = this.opt.cameraZoomEnable, this.controls.enableRotate = this.opt.cameraRotateEnable, this.controls.enablePan = this.opt.cameraPanEnable, this.controls.minDistance = 2, this.controls.maxDistance = 2e3, this.controls.autoRotateSpeed = 0, this.controls.autoRotate = !1, this.controls.update(), this.cameraClock = null;
4090
4090
  }
4091
4091
  this.ikMesh = new A.SkinnedMesh();
4092
4092
  const s = {
@@ -4104,14 +4104,14 @@ class it {
4104
4104
  Object.entries(s).forEach((a, u) => {
4105
4105
  const l = new A.Bone();
4106
4106
  l.name = a[0], a[1] ? this.ikMesh.getObjectByName(a[1]).add(l) : this.ikMesh.add(l), i.push(l);
4107
- }), this.ikMesh.bind(new A.Skeleton(i)), this.dynamicbones = new St(), this.isStreaming = !1, this.streamWorkletNode = null, this.streamAudioStartTime = null, this.streamWaitForAudioChunks = !0, this.streamLipsyncLang = null, this.streamLipsyncType = "visemes", this.streamLipsyncQueue = [];
4107
+ }), this.ikMesh.bind(new A.Skeleton(i)), this.dynamicbones = new kt(), this.isStreaming = !1, this.streamWorkletNode = null, this.streamAudioStartTime = null, this.streamWaitForAudioChunks = !0, this.streamLipsyncLang = null, this.streamLipsyncType = "visemes", this.streamLipsyncQueue = [];
4108
4108
  }
4109
4109
  /**
4110
4110
  * Helper that re/creates the audio context and the other nodes.
4111
4111
  * @param {number} sampleRate
4112
4112
  */
4113
4113
  initAudioGraph(n = null) {
4114
- if (this.audioCtx && this.audioCtx.state !== "closed" && this.audioCtx.close(), n ? this.audioCtx = new AudioContext({ sampleRate: n }) : this.audioCtx = new AudioContext(), this.audioSpeechSource = this.audioCtx.createBufferSource(), this.audioBackgroundSource = this.audioCtx.createBufferSource(), this.audioBackgroundGainNode = this.audioCtx.createGain(), this.audioSpeechGainNode = this.audioCtx.createGain(), this.audioStreamGainNode = this.audioCtx.createGain(), this.audioAnalyzerNode = this.audioCtx.createAnalyser(), this.audioAnalyzerNode.fftSize = 256, this.audioAnalyzerNode.smoothingTimeConstant = 0.1, this.audioAnalyzerNode.minDecibels = -70, this.audioAnalyzerNode.maxDecibels = -10, this.audioAnalyzer = new kt(this.audioCtx), this.audioReverbNode = this.audioCtx.createConvolver(), this.audioBackgroundGainNode.connect(this.audioReverbNode), this.audioAnalyzerNode.connect(this.audioSpeechGainNode), this.audioSpeechGainNode.connect(this.audioReverbNode), this.audioStreamGainNode.connect(this.audioReverbNode), this.audioReverbNode.connect(this.audioCtx.destination), this.setReverb(this.currentReverb || null), this.setMixerGain(
4114
+ if (this.audioCtx && this.audioCtx.state !== "closed" && this.audioCtx.close(), n ? this.audioCtx = new AudioContext({ sampleRate: n }) : this.audioCtx = new AudioContext(), this.audioSpeechSource = this.audioCtx.createBufferSource(), this.audioBackgroundSource = this.audioCtx.createBufferSource(), this.audioBackgroundGainNode = this.audioCtx.createGain(), this.audioSpeechGainNode = this.audioCtx.createGain(), this.audioStreamGainNode = this.audioCtx.createGain(), this.audioAnalyzerNode = this.audioCtx.createAnalyser(), this.audioAnalyzerNode.fftSize = 256, this.audioAnalyzerNode.smoothingTimeConstant = 0.1, this.audioAnalyzerNode.minDecibels = -70, this.audioAnalyzerNode.maxDecibels = -10, this.audioAnalyzer = new wt(this.audioCtx), this.audioReverbNode = this.audioCtx.createConvolver(), this.audioBackgroundGainNode.connect(this.audioReverbNode), this.audioAnalyzerNode.connect(this.audioSpeechGainNode), this.audioSpeechGainNode.connect(this.audioReverbNode), this.audioStreamGainNode.connect(this.audioReverbNode), this.audioReverbNode.connect(this.audioCtx.destination), this.setReverb(this.currentReverb || null), this.setMixerGain(
4115
4115
  this.opt.mixerGainSpeech,
4116
4116
  this.opt.mixerGainBackground
4117
4117
  ), this.workletLoaded = !1, this.streamWorkletNode) {
@@ -4227,14 +4227,14 @@ class it {
4227
4227
  if (s.morphTargetDictionary.hasOwnProperty(l)) {
4228
4228
  const r = s.morphTargetDictionary[l], h = i.morphAttributes.position[r], d = i.morphAttributes.normal?.[r];
4229
4229
  a || (a = new A.Float32BufferAttribute(h.count * 3, 3), d && (u = new A.Float32BufferAttribute(h.count * 3, 3)));
4230
- for (let g = 0; g < h.count; g++) {
4231
- const b = a.getX(g) + h.getX(g) * c, x = a.getY(g) + h.getY(g) * c, k = a.getZ(g) + h.getZ(g) * c;
4232
- a.setXYZ(g, b, x, k);
4230
+ for (let p = 0; p < h.count; p++) {
4231
+ const b = a.getX(p) + h.getX(p) * c, f = a.getY(p) + h.getY(p) * c, L = a.getZ(p) + h.getZ(p) * c;
4232
+ a.setXYZ(p, b, f, L);
4233
4233
  }
4234
4234
  if (d)
4235
- for (let g = 0; g < h.count; g++) {
4236
- const b = u.getX(g) + d.getX(g) * c, x = u.getY(g) + d.getY(g) * c, k = u.getZ(g) + d.getZ(g) * c;
4237
- u.setXYZ(g, b, x, k);
4235
+ for (let p = 0; p < h.count; p++) {
4236
+ const b = u.getX(p) + d.getX(p) * c, f = u.getY(p) + d.getY(p) * c, L = u.getZ(p) + d.getZ(p) * c;
4237
+ u.setXYZ(p, b, f, L);
4238
4238
  }
4239
4239
  }
4240
4240
  if (a) {
@@ -4252,9 +4252,9 @@ class it {
4252
4252
  async showAvatar(n, e = null) {
4253
4253
  if (!n || !n.hasOwnProperty("url"))
4254
4254
  throw new Error("Invalid parameter. The avatar must have at least 'url' specified.");
4255
- const t = new xt();
4255
+ const t = new bt();
4256
4256
  if (this.dracoEnabled) {
4257
- const l = new bt();
4257
+ const l = new At();
4258
4258
  l.setDecoderPath(this.dracoDecoderPath), t.setDRACOLoader(l);
4259
4259
  }
4260
4260
  let o = await t.loadAsync(n.url, e);
@@ -4412,9 +4412,9 @@ class it {
4412
4412
  updatePoseDelta() {
4413
4413
  for (const [n, e] of Object.entries(this.poseDelta.props)) {
4414
4414
  if (e.x === 0 && e.y === 0 && e.z === 0) continue;
4415
- oe.set(e.x, e.y, e.z);
4415
+ re.set(e.x, e.y, e.z);
4416
4416
  const t = this.poseAvatar.props[n];
4417
- t.isQuaternion ? (ue.setFromEuler(oe), t.multiply(ue)) : t.isVector3 && t.add(oe);
4417
+ t.isQuaternion ? (he.setFromEuler(re), t.multiply(he)) : t.isVector3 && t.add(re);
4418
4418
  }
4419
4419
  }
4420
4420
  /**
@@ -5109,14 +5109,14 @@ class it {
5109
5109
  for (let [h, d] of Object.entries(r.vs))
5110
5110
  if (this.mtAvatar.hasOwnProperty(h)) {
5111
5111
  if (d[o + 1] === null) continue;
5112
- const g = this.mtAvatar[h];
5113
- if (d[o] === null && (d[o] = g.value), o === i - 1)
5114
- g.newvalue = d[o];
5112
+ const p = this.mtAvatar[h];
5113
+ if (d[o] === null && (d[o] = p.value), o === i - 1)
5114
+ p.newvalue = d[o];
5115
5115
  else {
5116
- g.newvalue = d[o + 1];
5116
+ p.newvalue = d[o + 1];
5117
5117
  const b = r.ts[o + 1] - r.ts[o];
5118
- let x = 1;
5119
- b > 1e-4 && (x = (this.animClock - r.ts[o]) / b), x < 1 && (g.easing && (x = g.easing(x)), g.newvalue = (1 - x) * d[o] + x * g.newvalue), g.ref && g.ref !== r.vs && g.ref.hasOwnProperty(h) && delete g.ref[h], g.ref = r.vs;
5118
+ let f = 1;
5119
+ b > 1e-4 && (f = (this.animClock - r.ts[o]) / b), f < 1 && (p.easing && (f = p.easing(f)), p.newvalue = (1 - f) * d[o] + f * p.newvalue), p.ref && p.ref !== r.vs && p.ref.hasOwnProperty(h) && delete p.ref[h], p.ref = r.vs;
5120
5120
  }
5121
5121
  if (a)
5122
5122
  switch (h) {
@@ -5125,9 +5125,9 @@ class it {
5125
5125
  case "viseme_I":
5126
5126
  case "viseme_O":
5127
5127
  case "viseme_U":
5128
- g.newvalue *= 1 + a / 255 - 0.5;
5128
+ p.newvalue *= 1 + a / 255 - 0.5;
5129
5129
  }
5130
- g.needsUpdate = !0;
5130
+ p.needsUpdate = !0;
5131
5131
  } else h === "eyeContact" && d[o] !== null && u !== !1 ? u = !!d[o] : h === "headMove" && d[o] !== null && l !== !1 ? d[o] === 0 ? l = !1 : (Math.random() < d[o] && (l = !0), d[o] = null) : d[o] !== null && (c.push({ mt: h, val: d[o] }), d[o] = null);
5132
5132
  o === i ? (r.hasOwnProperty("mood") && this.setMood(r.mood), r.loop ? (i = this.isSpeaking && (r.template.name === "head" || r.template.name === "eyes") ? 4 : 1, this.animQueue[t] = this.animFactory(r.template, r.loop > 0 ? r.loop - 1 : r.loop, 1, 1 / i)) : (this.animQueue.splice(t--, 1), s--)) : r.ndx = o - 1;
5133
5133
  }
@@ -5179,7 +5179,7 @@ class it {
5179
5179
  }, o.x ? new A.Vector3(o.x, o.y, o.z) : null, !0, o.d);
5180
5180
  break;
5181
5181
  }
5182
- if ((u || l) && (oe.setFromQuaternion(this.poseAvatar.props["Head.quaternion"]), oe.x = Math.max(-0.9, Math.min(0.9, 2 * oe.x - 0.5)), oe.y = Math.max(-0.9, Math.min(0.9, -2.5 * oe.y)), u ? (Object.assign(this.mtAvatar.eyesLookDown, { system: oe.x < 0 ? -oe.x : 0, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyesLookUp, { system: oe.x < 0 ? 0 : oe.x, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyeLookInLeft, { system: oe.y < 0 ? -oe.y : 0, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyeLookOutLeft, { system: oe.y < 0 ? 0 : oe.y, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyeLookInRight, { system: oe.y < 0 ? 0 : oe.y, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyeLookOutRight, { system: oe.y < 0 ? -oe.y : 0, needsUpdate: !0 }), l && (t = -this.mtAvatar.bodyRotateY.value, o = this.gaussianRandom(-0.2, 0.2), this.animQueue.push(this.animFactory({
5182
+ if ((u || l) && (re.setFromQuaternion(this.poseAvatar.props["Head.quaternion"]), re.x = Math.max(-0.9, Math.min(0.9, 2 * re.x - 0.5)), re.y = Math.max(-0.9, Math.min(0.9, -2.5 * re.y)), u ? (Object.assign(this.mtAvatar.eyesLookDown, { system: re.x < 0 ? -re.x : 0, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyesLookUp, { system: re.x < 0 ? 0 : re.x, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyeLookInLeft, { system: re.y < 0 ? -re.y : 0, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyeLookOutLeft, { system: re.y < 0 ? 0 : re.y, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyeLookInRight, { system: re.y < 0 ? 0 : re.y, needsUpdate: !0 }), Object.assign(this.mtAvatar.eyeLookOutRight, { system: re.y < 0 ? -re.y : 0, needsUpdate: !0 }), l && (t = -this.mtAvatar.bodyRotateY.value, o = this.gaussianRandom(-0.2, 0.2), this.animQueue.push(this.animFactory({
5183
5183
  name: "headmove",
5184
5184
  dt: [[1e3, 2e3], [1e3, 2e3, 1, 2], [1e3, 2e3], [1e3, 2e3, 1, 2]],
5185
5185
  vs: {
@@ -5208,7 +5208,7 @@ class it {
5208
5208
  this.animationFinishedCallback = null, this.currentFBXActionCallback = null, this.currentFBXActionForCallback = null, this.currentFBXActionStartTime = null, this.currentFBXActionClipDuration = null, b();
5209
5209
  } else this.currentFBXActionCallback && (this.currentFBXActionCallback = null, this.currentFBXActionForCallback = null, this.currentFBXActionStartTime = null, this.currentFBXActionClipDuration = null);
5210
5210
  }
5211
- if (this.updatePoseDelta(), (this.isSpeaking || this.isListening) && u ? a > this.volumeMax ? (this.volumeHeadBase = 0.05, Math.random() > 0.6 && (this.volumeHeadTarget = -0.05 - Math.random() / 15), this.volumeMax = a) : (this.volumeMax *= 0.92, this.volumeHeadTarget = this.volumeHeadBase - 0.9 * (this.volumeHeadBase - this.volumeHeadTarget)) : (this.volumeHeadTarget = 0, this.volumeMax = 0), t = this.volumeHeadTarget - this.volumeHeadCurrent, o = Math.abs(t), o > 1e-4 && (i = o * (this.volumeHeadEasing(Math.min(1, this.volumeHeadVelocity * e / 1e3 / o) / 2 + 0.5) - 0.5), this.volumeHeadCurrent += Math.sign(t) * Math.min(o, i)), Math.abs(this.volumeHeadCurrent) > 1e-4 && (ue.setFromAxisAngle(Dt, this.volumeHeadCurrent), this.objectNeck.quaternion.multiply(ue)), lt.setFromObject(this.armature), this.objectLeftToeBase.getWorldPosition(Ne), Ne.sub(this.armature.position), this.objectRightToeBase.getWorldPosition(Xe), Xe.sub(this.armature.position), this.objectHips.position.y -= lt.min.y / 2, this.objectHips.position.x -= (Ne.x + Xe.x) / 4, this.objectHips.position.z -= (Ne.z + Xe.z) / 2, this.dynamicbones.update(e), this.fbxAnimationLoader && this.fbxAnimationLoader.update(), this.opt.update && this.opt.update(e), this.updateMorphTargets(e), this.isAvatarOnly)
5211
+ if (this.updatePoseDelta(), (this.isSpeaking || this.isListening) && u ? a > this.volumeMax ? (this.volumeHeadBase = 0.05, Math.random() > 0.6 && (this.volumeHeadTarget = -0.05 - Math.random() / 15), this.volumeMax = a) : (this.volumeMax *= 0.92, this.volumeHeadTarget = this.volumeHeadBase - 0.9 * (this.volumeHeadBase - this.volumeHeadTarget)) : (this.volumeHeadTarget = 0, this.volumeMax = 0), t = this.volumeHeadTarget - this.volumeHeadCurrent, o = Math.abs(t), o > 1e-4 && (i = o * (this.volumeHeadEasing(Math.min(1, this.volumeHeadVelocity * e / 1e3 / o) / 2 + 0.5) - 0.5), this.volumeHeadCurrent += Math.sign(t) * Math.min(o, i)), Math.abs(this.volumeHeadCurrent) > 1e-4 && (he.setFromAxisAngle(Nt, this.volumeHeadCurrent), this.objectNeck.quaternion.multiply(he)), at.setFromObject(this.armature), this.objectLeftToeBase.getWorldPosition(Ne), Ne.sub(this.armature.position), this.objectRightToeBase.getWorldPosition(We), We.sub(this.armature.position), this.objectHips.position.y -= at.min.y / 2, this.objectHips.position.x -= (Ne.x + We.x) / 4, this.objectHips.position.z -= (Ne.z + We.z) / 2, this.dynamicbones.update(e), this.fbxAnimationLoader && this.fbxAnimationLoader.update(), this.opt.update && this.opt.update(e), this.updateMorphTargets(e), this.isAvatarOnly)
5212
5212
  this.stats && this.stats.end();
5213
5213
  else {
5214
5214
  if (this.cameraClock !== null && this.cameraClock < 1e3) {
@@ -5239,8 +5239,8 @@ class it {
5239
5239
  if (!this.lipsync.hasOwnProperty(n)) {
5240
5240
  const t = n.toLowerCase(), o = "Lipsync" + n.charAt(0).toUpperCase() + n.slice(1);
5241
5241
  try {
5242
- const s = at[t];
5243
- s && s[o] ? this.lipsync[n] = new s[o]() : console.warn(`Lip-sync module for ${n} not found. Available modules:`, Object.keys(at));
5242
+ const s = rt[t];
5243
+ s && s[o] ? this.lipsync[n] = new s[o]() : console.warn(`Lip-sync module for ${n} not found. Available modules:`, Object.keys(rt));
5244
5244
  } catch (s) {
5245
5245
  console.warn(`Failed to load lip-sync module for ${n}:`, s);
5246
5246
  }
@@ -5276,16 +5276,16 @@ class it {
5276
5276
  speakText(n, e = null, t = null, o = null) {
5277
5277
  e = e || {};
5278
5278
  const s = /[!\.\?\n\p{Extended_Pictographic}]/ug, i = /[ ]/ug, a = /[\p{L}\p{N},\.\p{Quotation_Mark}!€\$\+\p{Dash_Punctuation}%&\?]/ug, u = /[\p{Extended_Pictographic}]/ug, l = e.lipsyncLang || this.avatar.lipsyncLang || this.opt.lipsyncLang;
5279
- let c = "", r = "", h = 0, d = [], g = [];
5280
- const b = Array.from(this.segmenter.segment(n), (x) => x.segment);
5281
- for (let x = 0; x < b.length; x++) {
5282
- const k = x === b.length - 1, G = b[x].match(a);
5283
- let m = b[x].match(s);
5284
- const D = b[x].match(u), T = b[x].match(i);
5285
- if (m && !k && !D && b[x + 1].match(s) && (m = !1), t && (c += b[x]), G && (!o || o.every((f) => x < f[0] || x > f[1])) && (r += b[x]), (T || m || k) && (r.length && (r = this.lipsyncPreProcessText(r, l), r.length && d.push({
5279
+ let c = "", r = "", h = 0, d = [], p = [];
5280
+ const b = Array.from(this.segmenter.segment(n), (f) => f.segment);
5281
+ for (let f = 0; f < b.length; f++) {
5282
+ const L = f === b.length - 1, Q = b[f].match(a);
5283
+ let g = b[f].match(s);
5284
+ const H = b[f].match(u), M = b[f].match(i);
5285
+ if (g && !L && !H && b[f + 1].match(s) && (g = !1), t && (c += b[f]), Q && (!o || o.every((y) => f < y[0] || f > y[1])) && (r += b[f]), (M || g || L) && (r.length && (r = this.lipsyncPreProcessText(r, l), r.length && d.push({
5286
5286
  mark: h,
5287
5287
  word: r
5288
- })), c.length && (g.push({
5288
+ })), c.length && (p.push({
5289
5289
  mark: h,
5290
5290
  template: { name: "subtitles" },
5291
5291
  ts: [0],
@@ -5293,31 +5293,31 @@ class it {
5293
5293
  subtitles: [c]
5294
5294
  }
5295
5295
  }), c = ""), r.length)) {
5296
- const f = this.lipsyncWordsToVisemes(r, l);
5297
- if (f && f.visemes && f.visemes.length) {
5298
- const B = f.times[f.visemes.length - 1] + f.durations[f.visemes.length - 1];
5299
- for (let L = 0; L < f.visemes.length; L++)
5300
- g.push({
5296
+ const y = this.lipsyncWordsToVisemes(r, l);
5297
+ if (y && y.visemes && y.visemes.length) {
5298
+ const E = y.times[y.visemes.length - 1] + y.durations[y.visemes.length - 1];
5299
+ for (let S = 0; S < y.visemes.length; S++)
5300
+ p.push({
5301
5301
  mark: h,
5302
5302
  template: { name: "viseme" },
5303
- ts: [(f.times[L] - 0.6) / B, (f.times[L] + 0.5) / B, (f.times[L] + f.durations[L] + 0.5) / B],
5303
+ ts: [(y.times[S] - 0.6) / E, (y.times[S] + 0.5) / E, (y.times[S] + y.durations[S] + 0.5) / E],
5304
5304
  vs: {
5305
- ["viseme_" + f.visemes[L]]: [null, f.visemes[L] === "PP" || f.visemes[L] === "FF" ? 0.9 : 0.6, 0]
5305
+ ["viseme_" + y.visemes[S]]: [null, y.visemes[S] === "PP" || y.visemes[S] === "FF" ? 0.9 : 0.6, 0]
5306
5306
  }
5307
5307
  });
5308
5308
  }
5309
5309
  r = "", h++;
5310
5310
  }
5311
- if (m || k) {
5312
- if (d.length || k && g.length) {
5313
- const f = {
5314
- anim: g
5311
+ if (g || L) {
5312
+ if (d.length || L && p.length) {
5313
+ const y = {
5314
+ anim: p
5315
5315
  };
5316
- t && (f.onSubtitles = t), d.length && !e.avatarMute && (f.text = d, e.avatarMood && (f.mood = e.avatarMood), e.ttsLang && (f.lang = e.ttsLang), e.ttsVoice && (f.voice = e.ttsVoice), e.ttsRate && (f.rate = e.ttsRate), e.ttsVoice && (f.pitch = e.ttsPitch), e.ttsVolume && (f.volume = e.ttsVolume)), this.speechQueue.push(f), d = [], r = "", h = 0, g = [];
5316
+ t && (y.onSubtitles = t), d.length && !e.avatarMute && (y.text = d, e.avatarMood && (y.mood = e.avatarMood), e.ttsLang && (y.lang = e.ttsLang), e.ttsVoice && (y.voice = e.ttsVoice), e.ttsRate && (y.rate = e.ttsRate), e.ttsVoice && (y.pitch = e.ttsPitch), e.ttsVolume && (y.volume = e.ttsVolume)), this.speechQueue.push(y), d = [], r = "", h = 0, p = [];
5317
5317
  }
5318
- if (D) {
5319
- let f = this.animEmojis[b[x]];
5320
- f && f.link && (f = this.animEmojis[f.link]), f && this.speechQueue.push({ emoji: f });
5318
+ if (H) {
5319
+ let y = this.animEmojis[b[f]];
5320
+ y && y.link && (y = this.animEmojis[y.link]), y && this.speechQueue.push({ emoji: y });
5321
5321
  }
5322
5322
  this.speechQueue.push({ break: 100 });
5323
5323
  }
@@ -5409,16 +5409,16 @@ class it {
5409
5409
  }), !n.visemes)) {
5410
5410
  const r = this.lipsyncPreProcessText(u, o), h = this.lipsyncWordsToVisemes(r, o);
5411
5411
  if (h && h.visemes && h.visemes.length) {
5412
- const d = h.times[h.visemes.length - 1] + h.durations[h.visemes.length - 1], g = Math.min(c, Math.max(0, c - h.visemes.length * 150));
5413
- let b = 0.6 + this.convertRange(g, [0, c], [0, 0.4]);
5412
+ const d = h.times[h.visemes.length - 1] + h.durations[h.visemes.length - 1], p = Math.min(c, Math.max(0, c - h.visemes.length * 150));
5413
+ let b = 0.6 + this.convertRange(p, [0, c], [0, 0.4]);
5414
5414
  if (c = Math.min(c, h.visemes.length * 200), d > 0)
5415
- for (let x = 0; x < h.visemes.length; x++) {
5416
- const k = l + h.times[x] / d * c, G = h.durations[x] / d * c;
5415
+ for (let f = 0; f < h.visemes.length; f++) {
5416
+ const L = l + h.times[f] / d * c, Q = h.durations[f] / d * c;
5417
5417
  i.push({
5418
5418
  template: { name: "viseme" },
5419
- ts: [k - Math.min(60, 2 * G / 3), k + Math.min(25, G / 2), k + G + Math.min(60, G / 2)],
5419
+ ts: [L - Math.min(60, 2 * Q / 3), L + Math.min(25, Q / 2), L + Q + Math.min(60, Q / 2)],
5420
5420
  vs: {
5421
- ["viseme_" + h.visemes[x]]: [null, h.visemes[x] === "PP" || h.visemes[x] === "FF" ? 0.9 : b, 0]
5421
+ ["viseme_" + h.visemes[f]]: [null, h.visemes[f] === "PP" || h.visemes[f] === "FF" ? 0.9 : b, 0]
5422
5422
  }
5423
5423
  });
5424
5424
  }
@@ -5547,32 +5547,32 @@ class it {
5547
5547
  */
5548
5548
  async synthesizeWithBrowserTTS(n) {
5549
5549
  return new Promise((e, t) => {
5550
- const o = Array.isArray(n.text) ? n.text.map((m) => m.word).join(" ") : typeof n.text == "string" ? n.text : "", s = new SpeechSynthesisUtterance(o), i = n.lang || this.avatar.ttsLang || this.opt.ttsLang || "en-US", a = (n.rate || this.avatar.ttsRate || this.opt.ttsRate || 1) + this.mood.speech.deltaRate, u = (n.pitch || this.avatar.ttsPitch || this.opt.ttsPitch || 1) + this.mood.speech.deltaPitch, l = (n.volume || this.avatar.ttsVolume || this.opt.ttsVolume || 1) + this.mood.speech.deltaVolume;
5550
+ const o = Array.isArray(n.text) ? n.text.map((g) => g.word).join(" ") : typeof n.text == "string" ? n.text : "", s = new SpeechSynthesisUtterance(o), i = n.lang || this.avatar.ttsLang || this.opt.ttsLang || "en-US", a = (n.rate || this.avatar.ttsRate || this.opt.ttsRate || 1) + this.mood.speech.deltaRate, u = (n.pitch || this.avatar.ttsPitch || this.opt.ttsPitch || 1) + this.mood.speech.deltaPitch, l = (n.volume || this.avatar.ttsVolume || this.opt.ttsVolume || 1) + this.mood.speech.deltaVolume;
5551
5551
  s.lang = i, s.rate = Math.max(0.1, Math.min(10, a)), s.pitch = Math.max(0, Math.min(2, u)), s.volume = Math.max(0, Math.min(1, l));
5552
5552
  const c = speechSynthesis.getVoices(), r = n.voice || this.avatar.ttsVoice || this.opt.ttsVoice;
5553
5553
  if (r && c.length > 0) {
5554
- const m = c.find((D) => D.name.includes(r) || D.lang === i);
5555
- m && (s.voice = m);
5554
+ const g = c.find((H) => H.name.includes(r) || H.lang === i);
5555
+ g && (s.voice = g);
5556
5556
  }
5557
- const h = o.length * 100 / s.rate, d = this.audioCtx.createBuffer(1, this.audioCtx.sampleRate * (h / 1e3), this.audioCtx.sampleRate), g = this.avatar.lipsyncLang || this.opt.lipsyncLang || "en", b = this.lipsyncPreProcessText(o, g), x = this.lipsyncWordsToVisemes(b, g), k = [];
5558
- if (x && x.visemes && x.visemes.length > 0) {
5559
- const m = x.times[x.visemes.length - 1] + x.durations[x.visemes.length - 1];
5560
- for (let D = 0; D < x.visemes.length; D++) {
5561
- const T = x.visemes[D], f = x.times[D] / m, B = x.durations[D] / m, L = f * h, y = B * h;
5562
- k.push({
5557
+ const h = o.length * 100 / s.rate, d = this.audioCtx.createBuffer(1, this.audioCtx.sampleRate * (h / 1e3), this.audioCtx.sampleRate), p = this.avatar.lipsyncLang || this.opt.lipsyncLang || "en", b = this.lipsyncPreProcessText(o, p), f = this.lipsyncWordsToVisemes(b, p), L = [];
5558
+ if (f && f.visemes && f.visemes.length > 0) {
5559
+ const g = f.times[f.visemes.length - 1] + f.durations[f.visemes.length - 1];
5560
+ for (let H = 0; H < f.visemes.length; H++) {
5561
+ const M = f.visemes[H], y = f.times[H] / g, E = f.durations[H] / g, S = y * h, C = E * h;
5562
+ L.push({
5563
5563
  template: { name: "viseme" },
5564
- ts: [L - Math.min(60, 2 * y / 3), L + Math.min(25, y / 2), L + y + Math.min(60, y / 2)],
5564
+ ts: [S - Math.min(60, 2 * C / 3), S + Math.min(25, C / 2), S + C + Math.min(60, C / 2)],
5565
5565
  vs: {
5566
- ["viseme_" + T]: [null, T === "PP" || T === "FF" ? 0.9 : 0.6, 0]
5566
+ ["viseme_" + M]: [null, M === "PP" || M === "FF" ? 0.9 : 0.6, 0]
5567
5567
  }
5568
5568
  });
5569
5569
  }
5570
5570
  }
5571
- const G = [...n.anim, ...k];
5572
- this.audioPlaylist.push({ anim: G, audio: d }), this.onSubtitles = n.onSubtitles || null, this.resetLips(), n.mood && this.setMood(n.mood), this.playAudio(), s.onend = () => {
5571
+ const Q = [...n.anim, ...L];
5572
+ this.audioPlaylist.push({ anim: Q, audio: d }), this.onSubtitles = n.onSubtitles || null, this.resetLips(), n.mood && this.setMood(n.mood), this.playAudio(), s.onend = () => {
5573
5573
  e();
5574
- }, s.onerror = (m) => {
5575
- console.error("Speech synthesis error:", m.error), t(m.error);
5574
+ }, s.onerror = (g) => {
5575
+ console.error("Speech synthesis error:", g.error), t(g.error);
5576
5576
  }, speechSynthesis.speak(s);
5577
5577
  });
5578
5578
  }
@@ -5616,8 +5616,8 @@ class it {
5616
5616
  hasVisemes: d && d.visemes && d.visemes.length > 0
5617
5617
  }), d && d.visemes && d.visemes.length > 0)
5618
5618
  l = {
5619
- visemes: d.visemes.map((g, b) => ({
5620
- viseme: g,
5619
+ visemes: d.visemes.map((p, b) => ({
5620
+ viseme: p,
5621
5621
  startTime: b * a.duration / d.visemes.length,
5622
5622
  endTime: (b + 1) * a.duration / d.visemes.length,
5623
5623
  duration: a.duration / d.visemes.length,
@@ -5631,18 +5631,18 @@ class it {
5631
5631
  throw new Error("No visemes generated from text");
5632
5632
  } catch (h) {
5633
5633
  console.error("Text-based lip-sync failed, using fallback:", h);
5634
- const d = e.toLowerCase().split(/\s+/), g = [];
5634
+ const d = e.toLowerCase().split(/\s+/), p = [];
5635
5635
  for (const b of d)
5636
- for (const x of b) {
5637
- let k = "aa";
5638
- "aeiou".includes(x) ? k = "aa" : "bp".includes(x) ? k = "PP" : "fv".includes(x) ? k = "FF" : "st".includes(x) ? k = "SS" : "dln".includes(x) ? k = "DD" : "kg".includes(x) ? k = "kk" : "rw".includes(x) && (k = "RR"), g.push(k);
5636
+ for (const f of b) {
5637
+ let L = "aa";
5638
+ "aeiou".includes(f) ? L = "aa" : "bp".includes(f) ? L = "PP" : "fv".includes(f) ? L = "FF" : "st".includes(f) ? L = "SS" : "dln".includes(f) ? L = "DD" : "kg".includes(f) ? L = "kk" : "rw".includes(f) && (L = "RR"), p.push(L);
5639
5639
  }
5640
5640
  l = {
5641
- visemes: g.map((b, x) => ({
5641
+ visemes: p.map((b, f) => ({
5642
5642
  viseme: b,
5643
- startTime: x * a.duration / g.length,
5644
- endTime: (x + 1) * a.duration / g.length,
5645
- duration: a.duration / g.length,
5643
+ startTime: f * a.duration / p.length,
5644
+ endTime: (f + 1) * a.duration / p.length,
5645
+ duration: a.duration / p.length,
5646
5646
  intensity: 0.6
5647
5647
  })),
5648
5648
  words: [],
@@ -5653,12 +5653,12 @@ class it {
5653
5653
  const c = [];
5654
5654
  if (l.visemes && l.visemes.length > 0)
5655
5655
  for (let h = 0; h < l.visemes.length; h++) {
5656
- const d = l.visemes[h], g = d.startTime * 1e3, b = d.duration * 1e3, x = d.intensity;
5656
+ const d = l.visemes[h], p = d.startTime * 1e3, b = d.duration * 1e3, f = d.intensity;
5657
5657
  c.push({
5658
5658
  template: { name: "viseme" },
5659
- ts: [g - Math.min(60, 2 * b / 3), g + Math.min(25, b / 2), g + b + Math.min(60, b / 2)],
5659
+ ts: [p - Math.min(60, 2 * b / 3), p + Math.min(25, b / 2), p + b + Math.min(60, b / 2)],
5660
5660
  vs: {
5661
- ["viseme_" + d.viseme]: [null, x, 0]
5661
+ ["viseme_" + d.viseme]: [null, f, 0]
5662
5662
  }
5663
5663
  });
5664
5664
  }
@@ -5698,8 +5698,8 @@ class it {
5698
5698
  hasVisemes: d && d.visemes && d.visemes.length > 0
5699
5699
  }), d && d.visemes && d.visemes.length > 0)
5700
5700
  l = {
5701
- visemes: d.visemes.map((g, b) => ({
5702
- viseme: g,
5701
+ visemes: d.visemes.map((p, b) => ({
5702
+ viseme: p,
5703
5703
  startTime: b * a.duration / d.visemes.length,
5704
5704
  endTime: (b + 1) * a.duration / d.visemes.length,
5705
5705
  duration: a.duration / d.visemes.length,
@@ -5713,18 +5713,18 @@ class it {
5713
5713
  throw new Error("No visemes generated from text");
5714
5714
  } catch (h) {
5715
5715
  console.error("Text-based lip-sync failed, using fallback:", h);
5716
- const d = e.toLowerCase().split(/\s+/), g = [];
5716
+ const d = e.toLowerCase().split(/\s+/), p = [];
5717
5717
  for (const b of d)
5718
- for (const x of b) {
5719
- let k = "aa";
5720
- "aeiou".includes(x) ? k = "aa" : "bp".includes(x) ? k = "PP" : "fv".includes(x) ? k = "FF" : "st".includes(x) ? k = "SS" : "dln".includes(x) ? k = "DD" : "kg".includes(x) ? k = "kk" : "rw".includes(x) && (k = "RR"), g.push(k);
5718
+ for (const f of b) {
5719
+ let L = "aa";
5720
+ "aeiou".includes(f) ? L = "aa" : "bp".includes(f) ? L = "PP" : "fv".includes(f) ? L = "FF" : "st".includes(f) ? L = "SS" : "dln".includes(f) ? L = "DD" : "kg".includes(f) ? L = "kk" : "rw".includes(f) && (L = "RR"), p.push(L);
5721
5721
  }
5722
5722
  l = {
5723
- visemes: g.map((b, x) => ({
5723
+ visemes: p.map((b, f) => ({
5724
5724
  viseme: b,
5725
- startTime: x * a.duration / g.length,
5726
- endTime: (x + 1) * a.duration / g.length,
5727
- duration: a.duration / g.length,
5725
+ startTime: f * a.duration / p.length,
5726
+ endTime: (f + 1) * a.duration / p.length,
5727
+ duration: a.duration / p.length,
5728
5728
  intensity: 0.6
5729
5729
  })),
5730
5730
  words: [],
@@ -5735,12 +5735,12 @@ class it {
5735
5735
  const c = [];
5736
5736
  if (l.visemes && l.visemes.length > 0)
5737
5737
  for (let h = 0; h < l.visemes.length; h++) {
5738
- const d = l.visemes[h], g = d.startTime * 1e3, b = d.duration * 1e3, x = d.intensity;
5738
+ const d = l.visemes[h], p = d.startTime * 1e3, b = d.duration * 1e3, f = d.intensity;
5739
5739
  c.push({
5740
5740
  template: { name: "viseme" },
5741
- ts: [g - Math.min(60, 2 * b / 3), g + Math.min(25, b / 2), g + b + Math.min(60, b / 2)],
5741
+ ts: [p - Math.min(60, 2 * b / 3), p + Math.min(25, b / 2), p + b + Math.min(60, b / 2)],
5742
5742
  vs: {
5743
- ["viseme_" + d.viseme]: [null, x, 0]
5743
+ ["viseme_" + d.viseme]: [null, f, 0]
5744
5744
  }
5745
5745
  });
5746
5746
  }
@@ -5773,10 +5773,10 @@ class it {
5773
5773
  throw new Error(`Azure TTS error: ${s.status} ${s.statusText}`);
5774
5774
  const i = await s.arrayBuffer(), a = await this.audioCtx.decodeAudioData(i), u = await this.audioAnalyzer.analyzeAudio(a, e), l = [];
5775
5775
  for (let r = 0; r < u.visemes.length; r++) {
5776
- const h = u.visemes[r], d = h.startTime * 1e3, g = h.duration * 1e3, b = h.intensity;
5776
+ const h = u.visemes[r], d = h.startTime * 1e3, p = h.duration * 1e3, b = h.intensity;
5777
5777
  l.push({
5778
5778
  template: { name: "viseme" },
5779
- ts: [d - Math.min(60, 2 * g / 3), d + Math.min(25, g / 2), d + g + Math.min(60, g / 2)],
5779
+ ts: [d - Math.min(60, 2 * p / 3), d + Math.min(25, p / 2), d + p + Math.min(60, p / 2)],
5780
5780
  vs: {
5781
5781
  ["viseme_" + h.viseme]: [null, b, 0]
5782
5782
  }
@@ -5826,23 +5826,23 @@ class it {
5826
5826
  let l = 0;
5827
5827
  n.text.forEach((h, d) => {
5828
5828
  if (d > 0) {
5829
- let g = u[u.length - 1];
5830
- s.timepoints[l] && (g = s.timepoints[l].timeSeconds * 1e3, s.timepoints[l].markName === "" + h.mark && l++), u.push(g);
5829
+ let p = u[u.length - 1];
5830
+ s.timepoints[l] && (p = s.timepoints[l].timeSeconds * 1e3, s.timepoints[l].markName === "" + h.mark && l++), u.push(p);
5831
5831
  }
5832
5832
  });
5833
5833
  const c = [{ mark: 0, time: 0 }];
5834
5834
  u.forEach((h, d) => {
5835
5835
  if (d > 0) {
5836
- let g = h - u[d - 1];
5837
- c[d - 1].duration = g, c.push({ mark: d, time: h });
5836
+ let p = h - u[d - 1];
5837
+ c[d - 1].duration = p, c.push({ mark: d, time: h });
5838
5838
  }
5839
5839
  });
5840
5840
  let r = 1e3 * a.duration;
5841
5841
  r > this.opt.ttsTrimEnd && (r = r - this.opt.ttsTrimEnd), c[c.length - 1].duration = r - c[c.length - 1].time, n.anim.forEach((h) => {
5842
5842
  const d = c[h.mark];
5843
5843
  if (d)
5844
- for (let g = 0; g < h.ts.length; g++)
5845
- h.ts[g] = d.time + h.ts[g] * d.duration + this.opt.ttsTrimStart;
5844
+ for (let p = 0; p < h.ts.length; p++)
5845
+ h.ts[p] = d.time + h.ts[p] * d.duration + this.opt.ttsTrimStart;
5846
5846
  }), this.audioPlaylist.push({ anim: n.anim, audio: a }), this.onSubtitles = n.onSubtitles || null, this.resetLips(), n.mood && this.setMood(n.mood), this.playAudio();
5847
5847
  } else
5848
5848
  this.startSpeaking(!0);
@@ -5866,10 +5866,17 @@ class it {
5866
5866
  e.isRaw || (this.lookAtCamera(500), this.speakWithHands(), this.resetLips()), this.audioPlaylist.push({ anim: e.anim, audio: e.audio, isRaw: e.isRaw }), this.onSubtitles = e.onSubtitles || null, e.mood && this.setMood(e.mood), this.playAudio();
5867
5867
  else if (e.text) {
5868
5868
  this.lookAtCamera(500);
5869
+ let t = "";
5870
+ if (Array.isArray(e.text) ? t = e.text.map((o) => o.word).join(" ") : typeof e.text == "string" && (t = e.text), t && this.onSubtitles && typeof this.onSubtitles == "function")
5871
+ try {
5872
+ this.onSubtitles({ text: t, subtitles: [t] });
5873
+ } catch (o) {
5874
+ console.warn("Error in subtitle callback:", o);
5875
+ }
5869
5876
  try {
5870
5877
  !this.opt.ttsEndpoint || this.opt.ttsEndpoint === "" ? await this.synthesizeWithBrowserTTS(e) : this.opt.ttsService === "elevenlabs" ? await this.synthesizeWithElevenLabsTTS(e) : this.opt.ttsService === "deepgram" ? await this.synthesizeWithDeepgramTTS(e) : this.opt.ttsService === "azure" ? await this.synthesizeWithAzureTTS(e) : await this.synthesizeWithExternalTTS(e);
5871
- } catch (t) {
5872
- console.error("Error:", t), this.startSpeaking(!0);
5878
+ } catch (o) {
5879
+ console.error("Error:", o), this.startSpeaking(!0);
5873
5880
  }
5874
5881
  } else e.anim ? (this.onSubtitles = e.onSubtitles || null, this.resetLips(), e.mood && this.setMood(e.mood), e.anim.forEach((t, o) => {
5875
5882
  for (let s = 0; s < t.ts.length; s++)
@@ -5895,27 +5902,27 @@ class it {
5895
5902
  a
5896
5903
  );
5897
5904
  for (let h = 0; h < i.numberOfChannels; h++) {
5898
- const d = i.getChannelData(h), g = c.getChannelData(h);
5905
+ const d = i.getChannelData(h), p = c.getChannelData(h);
5899
5906
  for (let b = 0; b < l; b++)
5900
- g[b] = d[u + b];
5907
+ p[b] = d[u + b];
5901
5908
  }
5902
5909
  let r = null;
5903
5910
  if (this.currentAudioItem.anim) {
5904
- const h = this.animClock + this.currentAudioItem.delay, d = t * 1e3, g = h + d;
5911
+ const h = this.animClock + this.currentAudioItem.delay, d = t * 1e3, p = h + d;
5905
5912
  r = this.currentAudioItem.anim.map((b) => {
5906
- const x = {
5913
+ const f = {
5907
5914
  template: b.template,
5908
5915
  ts: [],
5909
5916
  vs: []
5910
5917
  };
5911
- for (let k = 0; k < b.ts.length; k++) {
5912
- const G = b.ts[k];
5913
- if (G > g) {
5914
- const m = G - g;
5915
- x.ts.push(m), x.vs.push(b.vs[k]);
5918
+ for (let L = 0; L < b.ts.length; L++) {
5919
+ const Q = b.ts[L];
5920
+ if (Q > p) {
5921
+ const g = Q - p;
5922
+ f.ts.push(g), f.vs.push(b.vs[L]);
5916
5923
  }
5917
5924
  }
5918
- return x.ts.length > 0 ? x : null;
5925
+ return f.ts.length > 0 ? f : null;
5919
5926
  }).filter((b) => b !== null);
5920
5927
  }
5921
5928
  n = {
@@ -5974,7 +5981,7 @@ class it {
5974
5981
  }
5975
5982
  if (!this.workletLoaded)
5976
5983
  try {
5977
- const a = this.audioCtx.audioWorklet.addModule(Ot.href), u = new Promise(
5984
+ const a = this.audioCtx.audioWorklet.addModule(Dt.href), u = new Promise(
5978
5985
  (l, c) => setTimeout(() => c(new Error("Worklet loading timed out")), 5e3)
5979
5986
  );
5980
5987
  await Promise.race([a, u]), this.workletLoaded = !0;
@@ -6116,10 +6123,10 @@ class it {
6116
6123
  let h = 0.6 + this.convertRange(r, [0, i], [0, 0.4]);
6117
6124
  if (i = Math.min(i, l.visemes.length * 200), c > 0)
6118
6125
  for (let d = 0; d < l.visemes.length; d++) {
6119
- const g = e + s + l.times[d] / c * i, b = l.durations[d] / c * i;
6126
+ const p = e + s + l.times[d] / c * i, b = l.durations[d] / c * i;
6120
6127
  this.animQueue.push({
6121
6128
  template: { name: "viseme" },
6122
- ts: [g - Math.min(60, 2 * b / 3), g + Math.min(25, b / 2), g + b + Math.min(60, b / 2)],
6129
+ ts: [p - Math.min(60, 2 * b / 3), p + Math.min(25, b / 2), p + b + Math.min(60, b / 2)],
6123
6130
  vs: {
6124
6131
  ["viseme_" + l.visemes[d]]: [null, l.visemes[d] === "PP" || l.visemes[d] === "FF" ? 0.9 : h, 0]
6125
6132
  }
@@ -6212,7 +6219,7 @@ class it {
6212
6219
  */
6213
6220
  lookAtCamera(n) {
6214
6221
  let e;
6215
- if (this.speakTo && (e = new A.Vector3(), this.speakTo.objectLeftEye?.isObject3D ? (this.speakTo.armature.objectHead, this.speakTo.objectLeftEye.updateMatrixWorld(!0), this.speakTo.objectRightEye.updateMatrixWorld(!0), Ne.setFromMatrixPosition(this.speakTo.objectLeftEye.matrixWorld), Xe.setFromMatrixPosition(this.speakTo.objectRightEye.matrixWorld), e.addVectors(Ne, Xe).divideScalar(2)) : this.speakTo.isObject3D ? this.speakTo.getWorldPosition(e) : this.speakTo.isVector3 ? e.set(this.speakTo) : this.speakTo.x && this.speakTo.y && this.speakTo.z && e.set(this.speakTo.x, this.speakTo.y, this.speakTo.z)), !e) {
6222
+ if (this.speakTo && (e = new A.Vector3(), this.speakTo.objectLeftEye?.isObject3D ? (this.speakTo.armature.objectHead, this.speakTo.objectLeftEye.updateMatrixWorld(!0), this.speakTo.objectRightEye.updateMatrixWorld(!0), Ne.setFromMatrixPosition(this.speakTo.objectLeftEye.matrixWorld), We.setFromMatrixPosition(this.speakTo.objectRightEye.matrixWorld), e.addVectors(Ne, We).divideScalar(2)) : this.speakTo.isObject3D ? this.speakTo.getWorldPosition(e) : this.speakTo.isVector3 ? e.set(this.speakTo) : this.speakTo.x && this.speakTo.y && this.speakTo.z && e.set(this.speakTo.x, this.speakTo.y, this.speakTo.z)), !e) {
6216
6223
  if (this.avatar.hasOwnProperty("avatarIgnoreCamera")) {
6217
6224
  if (this.avatar.avatarIgnoreCamera) {
6218
6225
  this.lookAhead(n);
@@ -6225,23 +6232,23 @@ class it {
6225
6232
  this.lookAt(null, null, n);
6226
6233
  return;
6227
6234
  }
6228
- this.objectLeftEye.updateMatrixWorld(!0), this.objectRightEye.updateMatrixWorld(!0), Ne.setFromMatrixPosition(this.objectLeftEye.matrixWorld), Xe.setFromMatrixPosition(this.objectRightEye.matrixWorld), Ne.add(Xe).divideScalar(2), ue.copy(this.armature.quaternion), ue.multiply(this.poseTarget.props["Hips.quaternion"]), ue.multiply(this.poseTarget.props["Spine.quaternion"]), ue.multiply(this.poseTarget.props["Spine1.quaternion"]), ue.multiply(this.poseTarget.props["Spine2.quaternion"]), ue.multiply(this.poseTarget.props["Neck.quaternion"]), ue.multiply(this.poseTarget.props["Head.quaternion"]);
6235
+ this.objectLeftEye.updateMatrixWorld(!0), this.objectRightEye.updateMatrixWorld(!0), Ne.setFromMatrixPosition(this.objectLeftEye.matrixWorld), We.setFromMatrixPosition(this.objectRightEye.matrixWorld), Ne.add(We).divideScalar(2), he.copy(this.armature.quaternion), he.multiply(this.poseTarget.props["Hips.quaternion"]), he.multiply(this.poseTarget.props["Spine.quaternion"]), he.multiply(this.poseTarget.props["Spine1.quaternion"]), he.multiply(this.poseTarget.props["Spine2.quaternion"]), he.multiply(this.poseTarget.props["Neck.quaternion"]), he.multiply(this.poseTarget.props["Head.quaternion"]);
6229
6236
  const t = new A.Vector3().subVectors(e, Ne).normalize(), o = Math.atan2(t.x, t.z), s = Math.asin(-t.y);
6230
- oe.set(s, o, 0, "YXZ");
6231
- const a = new A.Quaternion().setFromEuler(oe), u = new A.Quaternion().copy(a).multiply(ue.clone().invert());
6232
- oe.setFromQuaternion(u, "YXZ");
6233
- let l = oe.x / (40 / 24) + 0.2, c = oe.y / (9 / 4), r = Math.min(0.6, Math.max(-0.3, l)), h = Math.min(0.8, Math.max(-0.8, c)), d = (Math.random() - 0.5) / 4, g = (Math.random() - 0.5) / 4;
6237
+ re.set(s, o, 0, "YXZ");
6238
+ const a = new A.Quaternion().setFromEuler(re), u = new A.Quaternion().copy(a).multiply(he.clone().invert());
6239
+ re.setFromQuaternion(u, "YXZ");
6240
+ let l = re.x / (40 / 24) + 0.2, c = re.y / (9 / 4), r = Math.min(0.6, Math.max(-0.3, l)), h = Math.min(0.8, Math.max(-0.8, c)), d = (Math.random() - 0.5) / 4, p = (Math.random() - 0.5) / 4;
6234
6241
  if (n) {
6235
- let b = this.animQueue.findIndex((k) => k.template.name === "lookat");
6242
+ let b = this.animQueue.findIndex((L) => L.template.name === "lookat");
6236
6243
  b !== -1 && this.animQueue.splice(b, 1);
6237
- const x = {
6244
+ const f = {
6238
6245
  name: "lookat",
6239
6246
  dt: [750, n],
6240
6247
  vs: {
6241
6248
  bodyRotateX: [r + d],
6242
- bodyRotateY: [h + g],
6249
+ bodyRotateY: [h + p],
6243
6250
  eyesRotateX: [-3 * d + 0.1],
6244
- eyesRotateY: [-5 * g],
6251
+ eyesRotateY: [-5 * p],
6245
6252
  browInnerUp: [[0, 0.7]],
6246
6253
  mouthLeft: [[0, 0.7]],
6247
6254
  mouthRight: [[0, 0.7]],
@@ -6249,7 +6256,7 @@ class it {
6249
6256
  headMove: [0]
6250
6257
  }
6251
6258
  };
6252
- this.animQueue.push(this.animFactory(x));
6259
+ this.animQueue.push(this.animFactory(f));
6253
6260
  }
6254
6261
  }
6255
6262
  /**
@@ -6265,21 +6272,21 @@ class it {
6265
6272
  const s = new A.Vector3().setFromMatrixPosition(this.objectLeftEye.matrixWorld), i = new A.Vector3().setFromMatrixPosition(this.objectRightEye.matrixWorld), a = new A.Vector3().addVectors(s, i).divideScalar(2);
6266
6273
  a.project(this.camera);
6267
6274
  let u = (a.x + 1) / 2 * o.width + o.left, l = -(a.y - 1) / 2 * o.height + o.top;
6268
- n === null && (n = u), e === null && (e = l), ue.copy(this.armature.quaternion), ue.multiply(this.poseTarget.props["Hips.quaternion"]), ue.multiply(this.poseTarget.props["Spine.quaternion"]), ue.multiply(this.poseTarget.props["Spine1.quaternion"]), ue.multiply(this.poseTarget.props["Spine2.quaternion"]), ue.multiply(this.poseTarget.props["Neck.quaternion"]), ue.multiply(this.poseTarget.props["Head.quaternion"]), oe.setFromQuaternion(ue);
6269
- let c = oe.x / (40 / 24), r = oe.y / (9 / 4), h = Math.min(0.4, Math.max(-0.4, this.camera.rotation.x)), d = Math.min(0.4, Math.max(-0.4, this.camera.rotation.y)), g = Math.max(window.innerWidth - u, u), b = Math.max(window.innerHeight - l, l), x = this.convertRange(e, [l - b, l + b], [-0.3, 0.6]) - c + h, k = this.convertRange(n, [u - g, u + g], [-0.8, 0.8]) - r + d;
6270
- x = Math.min(0.6, Math.max(-0.3, x)), k = Math.min(0.8, Math.max(-0.8, k));
6271
- let G = (Math.random() - 0.5) / 4, m = (Math.random() - 0.5) / 4;
6275
+ n === null && (n = u), e === null && (e = l), he.copy(this.armature.quaternion), he.multiply(this.poseTarget.props["Hips.quaternion"]), he.multiply(this.poseTarget.props["Spine.quaternion"]), he.multiply(this.poseTarget.props["Spine1.quaternion"]), he.multiply(this.poseTarget.props["Spine2.quaternion"]), he.multiply(this.poseTarget.props["Neck.quaternion"]), he.multiply(this.poseTarget.props["Head.quaternion"]), re.setFromQuaternion(he);
6276
+ let c = re.x / (40 / 24), r = re.y / (9 / 4), h = Math.min(0.4, Math.max(-0.4, this.camera.rotation.x)), d = Math.min(0.4, Math.max(-0.4, this.camera.rotation.y)), p = Math.max(window.innerWidth - u, u), b = Math.max(window.innerHeight - l, l), f = this.convertRange(e, [l - b, l + b], [-0.3, 0.6]) - c + h, L = this.convertRange(n, [u - p, u + p], [-0.8, 0.8]) - r + d;
6277
+ f = Math.min(0.6, Math.max(-0.3, f)), L = Math.min(0.8, Math.max(-0.8, L));
6278
+ let Q = (Math.random() - 0.5) / 4, g = (Math.random() - 0.5) / 4;
6272
6279
  if (t) {
6273
- let D = this.animQueue.findIndex((f) => f.template.name === "lookat");
6274
- D !== -1 && this.animQueue.splice(D, 1);
6275
- const T = {
6280
+ let H = this.animQueue.findIndex((y) => y.template.name === "lookat");
6281
+ H !== -1 && this.animQueue.splice(H, 1);
6282
+ const M = {
6276
6283
  name: "lookat",
6277
6284
  dt: [750, t],
6278
6285
  vs: {
6279
- bodyRotateX: [x + G],
6280
- bodyRotateY: [k + m],
6281
- eyesRotateX: [-3 * G + 0.1],
6282
- eyesRotateY: [-5 * m],
6286
+ bodyRotateX: [f + Q],
6287
+ bodyRotateY: [L + g],
6288
+ eyesRotateX: [-3 * Q + 0.1],
6289
+ eyesRotateY: [-5 * g],
6283
6290
  browInnerUp: [[0, 0.7]],
6284
6291
  mouthLeft: [[0, 0.7]],
6285
6292
  mouthRight: [[0, 0.7]],
@@ -6287,7 +6294,7 @@ class it {
6287
6294
  headMove: [0]
6288
6295
  }
6289
6296
  };
6290
- this.animQueue.push(this.animFactory(T));
6297
+ this.animQueue.push(this.animFactory(M));
6291
6298
  }
6292
6299
  }
6293
6300
  /**
@@ -6511,7 +6518,7 @@ class it {
6511
6518
  } catch (d) {
6512
6519
  console.warn(`Could not verify file existence for ${n}, attempting to load anyway:`, d);
6513
6520
  }
6514
- const r = new st();
6521
+ const r = new ot();
6515
6522
  let h;
6516
6523
  try {
6517
6524
  h = await r.loadAsync(n, e);
@@ -6522,54 +6529,54 @@ class it {
6522
6529
  suggestion: "Make sure the file is a valid FBX file and the path is correct"
6523
6530
  }), d.message && d.message.includes("version number") && (console.error("FBX Loader Error: Cannot find version number"), console.error("This error usually means:"), console.error("1. The file is not a valid FBX file (might be GLB, corrupted, or wrong format)"), console.error("2. The file might be corrupted"), console.error("3. The file path might be incorrect"), console.error("4. The server returned an HTML error page instead of the FBX file"), console.error("5. The file might not exist at that path"), console.error(""), console.error("Solution: Please verify:"), console.error(` - File exists at: ${n}`), console.error(" - File is a valid FBX binary file"), console.error(" - File path matches your public folder structure"), console.error(" - File is not corrupted"));
6524
6531
  try {
6525
- const g = await fetch(n), b = g.headers.get("content-type"), x = await g.text();
6532
+ const p = await fetch(n), b = p.headers.get("content-type"), f = await p.text();
6526
6533
  console.error("Response details:", {
6527
- status: g.status,
6534
+ status: p.status,
6528
6535
  contentType: b,
6529
- firstBytes: x.substring(0, 100),
6530
- isHTML: x.trim().startsWith("<!DOCTYPE") || x.trim().startsWith("<html")
6531
- }), (x.trim().startsWith("<!DOCTYPE") || x.trim().startsWith("<html")) && console.error("The server returned an HTML page instead of an FBX file. The file path is likely incorrect.");
6532
- } catch (g) {
6533
- console.error("Could not fetch file for debugging:", g);
6536
+ firstBytes: f.substring(0, 100),
6537
+ isHTML: f.trim().startsWith("<!DOCTYPE") || f.trim().startsWith("<html")
6538
+ }), (f.trim().startsWith("<!DOCTYPE") || f.trim().startsWith("<html")) && console.error("The server returned an HTML page instead of an FBX file. The file path is likely incorrect.");
6539
+ } catch (p) {
6540
+ console.error("Could not fetch file for debugging:", p);
6534
6541
  }
6535
6542
  return;
6536
6543
  }
6537
6544
  if (h && h.animations && h.animations[o]) {
6538
6545
  let d = h.animations[o];
6539
- const g = /* @__PURE__ */ new Set();
6540
- this.armature && this.armature.traverse((f) => {
6541
- (f.isBone || f.type === "Bone") && g.add(f.name);
6546
+ const p = /* @__PURE__ */ new Set();
6547
+ this.armature && this.armature.traverse((y) => {
6548
+ (y.isBone || y.type === "Bone") && p.add(y.name);
6542
6549
  });
6543
- const b = /* @__PURE__ */ new Map(), x = (f) => {
6544
- if (g.has(f))
6545
- return f;
6546
- let B = f.replace(/^mixamorig/i, "").replace(/^CC_Base_/i, "").replace(/^RPM_/i, "");
6547
- if (g.has(B))
6548
- return B;
6549
- const L = B.toLowerCase();
6550
- if (L.includes("left") && L.includes("arm")) {
6551
- if (L.includes("fore") || L.includes("lower")) {
6552
- if (g.has("LeftForeArm")) return "LeftForeArm";
6553
- if (g.has("LeftForearm")) return "LeftForearm";
6554
- } else if (!L.includes("fore") && !L.includes("hand") && g.has("LeftArm"))
6550
+ const b = /* @__PURE__ */ new Map(), f = (y) => {
6551
+ if (p.has(y))
6552
+ return y;
6553
+ let E = y.replace(/^mixamorig/i, "").replace(/^CC_Base_/i, "").replace(/^RPM_/i, "");
6554
+ if (p.has(E))
6555
+ return E;
6556
+ const S = E.toLowerCase();
6557
+ if (S.includes("left") && S.includes("arm")) {
6558
+ if (S.includes("fore") || S.includes("lower")) {
6559
+ if (p.has("LeftForeArm")) return "LeftForeArm";
6560
+ if (p.has("LeftForearm")) return "LeftForearm";
6561
+ } else if (!S.includes("fore") && !S.includes("hand") && p.has("LeftArm"))
6555
6562
  return "LeftArm";
6556
6563
  }
6557
- if (L.includes("right") && L.includes("arm")) {
6558
- if (L.includes("fore") || L.includes("lower")) {
6559
- if (g.has("RightForeArm")) return "RightForeArm";
6560
- if (g.has("RightForearm")) return "RightForearm";
6561
- } else if (!L.includes("fore") && !L.includes("hand") && g.has("RightArm"))
6564
+ if (S.includes("right") && S.includes("arm")) {
6565
+ if (S.includes("fore") || S.includes("lower")) {
6566
+ if (p.has("RightForeArm")) return "RightForeArm";
6567
+ if (p.has("RightForearm")) return "RightForearm";
6568
+ } else if (!S.includes("fore") && !S.includes("hand") && p.has("RightArm"))
6562
6569
  return "RightArm";
6563
6570
  }
6564
- if (L.includes("left") && L.includes("hand") && !L.includes("index") && !L.includes("thumb") && !L.includes("middle") && !L.includes("ring") && !L.includes("pinky") && g.has("LeftHand"))
6571
+ if (S.includes("left") && S.includes("hand") && !S.includes("index") && !S.includes("thumb") && !S.includes("middle") && !S.includes("ring") && !S.includes("pinky") && p.has("LeftHand"))
6565
6572
  return "LeftHand";
6566
- if (L.includes("right") && L.includes("hand") && !L.includes("index") && !L.includes("thumb") && !L.includes("middle") && !L.includes("ring") && !L.includes("pinky") && g.has("RightHand"))
6573
+ if (S.includes("right") && S.includes("hand") && !S.includes("index") && !S.includes("thumb") && !S.includes("middle") && !S.includes("ring") && !S.includes("pinky") && p.has("RightHand"))
6567
6574
  return "RightHand";
6568
- if (L.includes("left") && (L.includes("shoulder") || L.includes("clavicle")) && g.has("LeftShoulder"))
6575
+ if (S.includes("left") && (S.includes("shoulder") || S.includes("clavicle")) && p.has("LeftShoulder"))
6569
6576
  return "LeftShoulder";
6570
- if (L.includes("right") && (L.includes("shoulder") || L.includes("clavicle")) && g.has("RightShoulder"))
6577
+ if (S.includes("right") && (S.includes("shoulder") || S.includes("clavicle")) && p.has("RightShoulder"))
6571
6578
  return "RightShoulder";
6572
- const y = {
6579
+ const C = {
6573
6580
  // Arm bones - exact matches
6574
6581
  LeftArm: "LeftArm",
6575
6582
  leftArm: "LeftArm",
@@ -6609,54 +6616,54 @@ class it {
6609
6616
  Root: "Hips",
6610
6617
  root: "Hips"
6611
6618
  };
6612
- if (y[B]) {
6613
- const M = y[B];
6614
- if (g.has(M))
6615
- return M;
6619
+ if (C[E]) {
6620
+ const x = C[E];
6621
+ if (p.has(x))
6622
+ return x;
6616
6623
  }
6617
- for (const M of g)
6618
- if (M.toLowerCase() === L)
6619
- return M;
6620
- for (const M of g) {
6621
- const I = M.toLowerCase();
6622
- if ((L.includes("left") && I.includes("left") || L.includes("right") && I.includes("right")) && (L.includes("arm") && I.includes("arm") && !I.includes("fore") || L.includes("forearm") && I.includes("forearm") || L.includes("hand") && I.includes("hand") && !I.includes("index") && !I.includes("thumb") || L.includes("shoulder") && I.includes("shoulder")))
6623
- return M;
6624
+ for (const x of p)
6625
+ if (x.toLowerCase() === S)
6626
+ return x;
6627
+ for (const x of p) {
6628
+ const I = x.toLowerCase();
6629
+ if ((S.includes("left") && I.includes("left") || S.includes("right") && I.includes("right")) && (S.includes("arm") && I.includes("arm") && !I.includes("fore") || S.includes("forearm") && I.includes("forearm") || S.includes("hand") && I.includes("hand") && !I.includes("index") && !I.includes("thumb") || S.includes("shoulder") && I.includes("shoulder")))
6630
+ return x;
6624
6631
  }
6625
6632
  return null;
6626
- }, k = /* @__PURE__ */ new Set();
6627
- d.tracks.forEach((f) => {
6628
- const B = f.name.split(".");
6629
- k.add(B[0]);
6630
- }), Array.from(k).filter(
6631
- (f) => f.toLowerCase().includes("arm") || f.toLowerCase().includes("hand") || f.toLowerCase().includes("shoulder")
6632
- ), Array.from(g).filter(
6633
- (f) => f.includes("Arm") || f.includes("Hand") || f.includes("Shoulder")
6633
+ }, L = /* @__PURE__ */ new Set();
6634
+ d.tracks.forEach((y) => {
6635
+ const E = y.name.split(".");
6636
+ L.add(E[0]);
6637
+ }), Array.from(L).filter(
6638
+ (y) => y.toLowerCase().includes("arm") || y.toLowerCase().includes("hand") || y.toLowerCase().includes("shoulder")
6639
+ ), Array.from(p).filter(
6640
+ (y) => y.includes("Arm") || y.includes("Hand") || y.includes("Shoulder")
6634
6641
  );
6635
- const G = [], m = /* @__PURE__ */ new Set();
6636
- d.tracks.forEach((f) => {
6637
- const L = f.name.replaceAll("mixamorig", "").split("."), y = L[0], M = L[1], I = x(y);
6638
- if (!(I && (I === "LeftShoulder" || I === "RightShoulder") && (M === "quaternion" || M === "rotation")))
6639
- if (I && M) {
6640
- const E = `${I}.${M}`, P = f.clone();
6641
- P.name = E, G.push(P), y !== I && b.set(y, I);
6642
+ const Q = [], g = /* @__PURE__ */ new Set();
6643
+ d.tracks.forEach((y) => {
6644
+ const S = y.name.replaceAll("mixamorig", "").split("."), C = S[0], x = S[1], I = f(C);
6645
+ if (!(I && (I === "LeftShoulder" || I === "RightShoulder") && (x === "quaternion" || x === "rotation")))
6646
+ if (I && x) {
6647
+ const G = `${I}.${x}`, P = y.clone();
6648
+ P.name = G, Q.push(P), C !== I && b.set(C, I);
6642
6649
  } else
6643
- m.add(y), (y.toLowerCase().includes("arm") || y.toLowerCase().includes("hand") || y.toLowerCase().includes("shoulder")) && console.warn(`⚠️ Arm bone "${y}" could not be mapped to avatar skeleton`);
6644
- }), G.length > 0 ? d = new A.AnimationClip(d.name, d.duration, G) : console.error("No tracks could be mapped! Animation may not work correctly.");
6645
- const D = {};
6646
- d.tracks.forEach((f) => {
6647
- f.name = f.name.replaceAll("mixamorig", "");
6648
- const B = f.name.split(".");
6649
- if (B[1] === "position") {
6650
- for (let L = 0; L < f.values.length; L++)
6651
- f.values[L] = f.values[L] * s;
6652
- D[f.name] = new A.Vector3(f.values[0], f.values[1], f.values[2]);
6653
- } else B[1] === "quaternion" ? D[f.name] = new A.Quaternion(f.values[0], f.values[1], f.values[2], f.values[3]) : B[1] === "rotation" && (D[B[0] + ".quaternion"] = new A.Quaternion().setFromEuler(new A.Euler(f.values[0], f.values[1], f.values[2], "XYZ")).normalize());
6650
+ g.add(C), (C.toLowerCase().includes("arm") || C.toLowerCase().includes("hand") || C.toLowerCase().includes("shoulder")) && console.warn(`⚠️ Arm bone "${C}" could not be mapped to avatar skeleton`);
6651
+ }), Q.length > 0 ? d = new A.AnimationClip(d.name, d.duration, Q) : console.error("No tracks could be mapped! Animation may not work correctly.");
6652
+ const H = {};
6653
+ d.tracks.forEach((y) => {
6654
+ y.name = y.name.replaceAll("mixamorig", "");
6655
+ const E = y.name.split(".");
6656
+ if (E[1] === "position") {
6657
+ for (let S = 0; S < y.values.length; S++)
6658
+ y.values[S] = y.values[S] * s;
6659
+ H[y.name] = new A.Vector3(y.values[0], y.values[1], y.values[2]);
6660
+ } else E[1] === "quaternion" ? H[y.name] = new A.Quaternion(y.values[0], y.values[1], y.values[2], y.values[3]) : E[1] === "rotation" && (H[E[0] + ".quaternion"] = new A.Quaternion().setFromEuler(new A.Euler(y.values[0], y.values[1], y.values[2], "XYZ")).normalize());
6654
6661
  });
6655
- const T = { props: D };
6656
- D["Hips.position"] && (D["Hips.position"].y < 0.5 ? T.lying = !0 : T.standing = !0), this.animClips.push({
6662
+ const M = { props: H };
6663
+ H["Hips.position"] && (H["Hips.position"].y < 0.5 ? M.lying = !0 : M.standing = !0), this.animClips.push({
6657
6664
  url: n + "-" + o,
6658
6665
  clip: d,
6659
- pose: T
6666
+ pose: M
6660
6667
  }), this.playAnimation(n, e, t, o, s);
6661
6668
  } else {
6662
6669
  const d = "Animation " + n + " (ndx=" + o + ") not found";
@@ -6694,7 +6701,7 @@ class it {
6694
6701
  let a = this.animQueue.find((u) => u.template.name === "pose");
6695
6702
  a && (a.ts[0] = this.animClock + t * 1e3 + 2e3), this.setPoseFromTemplate(i);
6696
6703
  } else {
6697
- let u = await new st().loadAsync(n, e);
6704
+ let u = await new ot().loadAsync(n, e);
6698
6705
  if (u && u.animations && u.animations[o]) {
6699
6706
  let l = u.animations[o];
6700
6707
  const c = {};
@@ -6748,8 +6755,8 @@ class it {
6748
6755
  if (e * 1e3 - c > 0) {
6749
6756
  const h = [];
6750
6757
  for (let b = 1; b < a.ts.length; b++) h.push(a.ts[b] - a.ts[b - 1]);
6751
- const d = i.template?.rescale || h.map((b) => b / c), g = e * 1e3 - c;
6752
- a.ts = a.ts.map((b, x, k) => x === 0 ? u : k[x - 1] + h[x - 1] + d[x - 1] * g);
6758
+ const d = i.template?.rescale || h.map((b) => b / c), p = e * 1e3 - c;
6759
+ a.ts = a.ts.map((b, f, L) => f === 0 ? u : L[f - 1] + h[f - 1] + d[f - 1] * p);
6753
6760
  } else {
6754
6761
  const h = e * 1e3 / c;
6755
6762
  a.ts = a.ts.map((d) => u + h * (d - u));
@@ -6784,32 +6791,32 @@ class it {
6784
6791
  ikSolve(n, e = null, t = !1, o = null) {
6785
6792
  const s = new A.Vector3(), i = new A.Vector3(), a = new A.Vector3(), u = new A.Vector3(), l = new A.Quaternion(), c = new A.Vector3(), r = new A.Vector3(), h = new A.Vector3(), d = this.ikMesh.getObjectByName(n.root);
6786
6793
  d.position.setFromMatrixPosition(this.armature.getObjectByName(n.root).matrixWorld), d.quaternion.setFromRotationMatrix(this.armature.getObjectByName(n.root).matrixWorld), e && t && e.applyQuaternion(this.armature.quaternion).add(d.position);
6787
- const g = this.ikMesh.getObjectByName(n.effector), b = n.links;
6788
- b.forEach((k) => {
6789
- k.bone = this.ikMesh.getObjectByName(k.link), k.bone.quaternion.copy(this.getPoseTemplateProp(k.link + ".quaternion"));
6794
+ const p = this.ikMesh.getObjectByName(n.effector), b = n.links;
6795
+ b.forEach((L) => {
6796
+ L.bone = this.ikMesh.getObjectByName(L.link), L.bone.quaternion.copy(this.getPoseTemplateProp(L.link + ".quaternion"));
6790
6797
  }), d.updateMatrixWorld(!0);
6791
- const x = n.iterations || 10;
6798
+ const f = n.iterations || 10;
6792
6799
  if (e)
6793
- for (let k = 0; k < x; k++) {
6794
- let G = !1;
6795
- for (let m = 0, D = b.length; m < D; m++) {
6796
- const T = b[m].bone;
6797
- T.matrixWorld.decompose(u, l, c), l.invert(), i.setFromMatrixPosition(g.matrixWorld), a.subVectors(i, u), a.applyQuaternion(l), a.normalize(), s.subVectors(e, u), s.applyQuaternion(l), s.normalize();
6798
- let f = s.dot(a);
6799
- f > 1 ? f = 1 : f < -1 && (f = -1), f = Math.acos(f), !(f < 1e-5) && (b[m].minAngle !== void 0 && f < b[m].minAngle && (f = b[m].minAngle), b[m].maxAngle !== void 0 && f > b[m].maxAngle && (f = b[m].maxAngle), r.crossVectors(a, s), r.normalize(), ue.setFromAxisAngle(r, f), T.quaternion.multiply(ue), T.rotation.setFromVector3(h.setFromEuler(T.rotation).clamp(new A.Vector3(
6800
- b[m].minx !== void 0 ? b[m].minx : -1 / 0,
6801
- b[m].miny !== void 0 ? b[m].miny : -1 / 0,
6802
- b[m].minz !== void 0 ? b[m].minz : -1 / 0
6800
+ for (let L = 0; L < f; L++) {
6801
+ let Q = !1;
6802
+ for (let g = 0, H = b.length; g < H; g++) {
6803
+ const M = b[g].bone;
6804
+ M.matrixWorld.decompose(u, l, c), l.invert(), i.setFromMatrixPosition(p.matrixWorld), a.subVectors(i, u), a.applyQuaternion(l), a.normalize(), s.subVectors(e, u), s.applyQuaternion(l), s.normalize();
6805
+ let y = s.dot(a);
6806
+ y > 1 ? y = 1 : y < -1 && (y = -1), y = Math.acos(y), !(y < 1e-5) && (b[g].minAngle !== void 0 && y < b[g].minAngle && (y = b[g].minAngle), b[g].maxAngle !== void 0 && y > b[g].maxAngle && (y = b[g].maxAngle), r.crossVectors(a, s), r.normalize(), he.setFromAxisAngle(r, y), M.quaternion.multiply(he), M.rotation.setFromVector3(h.setFromEuler(M.rotation).clamp(new A.Vector3(
6807
+ b[g].minx !== void 0 ? b[g].minx : -1 / 0,
6808
+ b[g].miny !== void 0 ? b[g].miny : -1 / 0,
6809
+ b[g].minz !== void 0 ? b[g].minz : -1 / 0
6803
6810
  ), new A.Vector3(
6804
- b[m].maxx !== void 0 ? b[m].maxx : 1 / 0,
6805
- b[m].maxy !== void 0 ? b[m].maxy : 1 / 0,
6806
- b[m].maxz !== void 0 ? b[m].maxz : 1 / 0
6807
- ))), T.updateMatrixWorld(!0), G = !0);
6811
+ b[g].maxx !== void 0 ? b[g].maxx : 1 / 0,
6812
+ b[g].maxy !== void 0 ? b[g].maxy : 1 / 0,
6813
+ b[g].maxz !== void 0 ? b[g].maxz : 1 / 0
6814
+ ))), M.updateMatrixWorld(!0), Q = !0);
6808
6815
  }
6809
- if (!G) break;
6816
+ if (!Q) break;
6810
6817
  }
6811
- o && b.forEach((k) => {
6812
- this.poseTarget.props[k.link + ".quaternion"].copy(k.bone.quaternion), this.poseTarget.props[k.link + ".quaternion"].t = this.animClock, this.poseTarget.props[k.link + ".quaternion"].d = o;
6818
+ o && b.forEach((L) => {
6819
+ this.poseTarget.props[L.link + ".quaternion"].copy(L.bone.quaternion), this.poseTarget.props[L.link + ".quaternion"].t = this.animClock, this.poseTarget.props[L.link + ".quaternion"].d = o;
6813
6820
  });
6814
6821
  }
6815
6822
  /**
@@ -6819,7 +6826,7 @@ class it {
6819
6826
  this.isRunning = !1, this.stop(), this.stopSpeaking(), this.streamStop(), this.isAvatarOnly ? this.armature && (this.armature.parent && this.armature.parent.remove(this.armature), this.clearThree(this.armature)) : (this.clearThree(this.scene), this.resizeobserver.disconnect(), this.renderer && (this.renderer.dispose(), this.renderer.domElement && this.renderer.domElement.parentNode && this.renderer.domElement.parentNode.removeChild(this.renderer.domElement), this.renderer = null)), this.clearThree(this.ikMesh), this.dynamicbones.dispose();
6820
6827
  }
6821
6828
  }
6822
- const Ge = {
6829
+ const Ve = {
6823
6830
  apiKey: "sk_ace57ef3ef65a92b9d3bee2a00183b78ca790bc3e10964f2",
6824
6831
  // Replace with your actual API key (should start with sk_)
6825
6832
  endpoint: "https://api.elevenlabs.io/v1/text-to-speech",
@@ -6862,23 +6869,23 @@ const Ge = {
6862
6869
  function Je() {
6863
6870
  return {
6864
6871
  service: "elevenlabs",
6865
- endpoint: Ge.endpoint,
6866
- apiKey: Ge.apiKey,
6867
- defaultVoice: Ge.defaultVoice,
6868
- voices: Ge.voices
6872
+ endpoint: Ve.endpoint,
6873
+ apiKey: Ve.apiKey,
6874
+ defaultVoice: Ve.defaultVoice,
6875
+ voices: Ve.voices
6869
6876
  };
6870
6877
  }
6871
- function Kt() {
6872
- const j = Je(), n = [];
6873
- return Object.entries(j.voices).forEach(([e, t]) => {
6878
+ function $t() {
6879
+ const Y = Je(), n = [];
6880
+ return Object.entries(Y.voices).forEach(([e, t]) => {
6874
6881
  n.push({
6875
6882
  value: t,
6876
- label: `${e.charAt(0).toUpperCase() + e.slice(1)} (${j.service})`
6883
+ label: `${e.charAt(0).toUpperCase() + e.slice(1)} (${Y.service})`
6877
6884
  });
6878
6885
  }), n;
6879
6886
  }
6880
- const ut = Ke(({
6881
- avatarUrl: j = "/avatars/brunette.glb",
6887
+ const ct = Ke(({
6888
+ avatarUrl: Y = "/avatars/brunette.glb",
6882
6889
  avatarBody: n = "F",
6883
6890
  mood: e = "neutral",
6884
6891
  ttsLang: t = "en",
@@ -6895,320 +6902,320 @@ const ut = Ke(({
6895
6902
  },
6896
6903
  onError: d = () => {
6897
6904
  },
6898
- className: g = "",
6905
+ className: p = "",
6899
6906
  style: b = {},
6900
- animations: x = {}
6901
- }, k) => {
6902
- const G = V(null), m = V(null), D = V(l), T = V(null), f = V(null), B = V(!1), L = V({ remainingText: null, originalText: null, options: null }), y = V([]), M = V(0), [I, E] = ye(!0), [P, Y] = ye(null), [$, Le] = ye(!1), [pe, Pe] = ye(!1);
6903
- Ce(() => {
6904
- B.current = pe;
6905
- }, [pe]), Ce(() => {
6906
- D.current = l;
6907
+ animations: f = {}
6908
+ }, L) => {
6909
+ const Q = V(null), g = V(null), H = V(l), M = V(null), y = V(null), E = V(!1), S = V({ remainingText: null, originalText: null, options: null }), C = V([]), x = V(0), [I, G] = ge(!0), [P, D] = ge(null), [K, fe] = ge(!1), [xe, Te] = ge(!1);
6910
+ ze(() => {
6911
+ E.current = xe;
6912
+ }, [xe]), ze(() => {
6913
+ H.current = l;
6907
6914
  }, [l]);
6908
- const be = Je(), ve = o || be.service;
6909
- let he;
6910
- ve === "browser" ? he = {
6915
+ const Le = Je(), Me = o || Le.service;
6916
+ let ae;
6917
+ Me === "browser" ? ae = {
6911
6918
  service: "browser",
6912
6919
  endpoint: "",
6913
6920
  apiKey: null,
6914
6921
  defaultVoice: "Google US English"
6915
- } : ve === "elevenlabs" ? he = {
6922
+ } : Me === "elevenlabs" ? ae = {
6916
6923
  service: "elevenlabs",
6917
6924
  endpoint: "https://api.elevenlabs.io/v1/text-to-speech",
6918
- apiKey: i || be.apiKey,
6919
- defaultVoice: s || be.defaultVoice || Ge.defaultVoice,
6920
- voices: be.voices || Ge.voices
6921
- } : ve === "deepgram" ? he = {
6925
+ apiKey: i || Le.apiKey,
6926
+ defaultVoice: s || Le.defaultVoice || Ve.defaultVoice,
6927
+ voices: Le.voices || Ve.voices
6928
+ } : Me === "deepgram" ? ae = {
6922
6929
  service: "deepgram",
6923
6930
  endpoint: "https://api.deepgram.com/v1/speak",
6924
- apiKey: i || be.apiKey,
6925
- defaultVoice: s || be.defaultVoice || _e.defaultVoice,
6926
- voices: be.voices || _e.voices
6927
- } : he = {
6928
- ...be,
6931
+ apiKey: i || Le.apiKey,
6932
+ defaultVoice: s || Le.defaultVoice || _e.defaultVoice,
6933
+ voices: Le.voices || _e.voices
6934
+ } : ae = {
6935
+ ...Le,
6929
6936
  // Override API key if provided via props
6930
- apiKey: i !== null ? i : be.apiKey
6937
+ apiKey: i !== null ? i : Le.apiKey
6931
6938
  };
6932
6939
  const R = {
6933
- url: j,
6940
+ url: Y,
6934
6941
  body: n,
6935
6942
  avatarMood: e,
6936
- ttsLang: ve === "browser" ? "en-US" : t,
6937
- ttsVoice: s || he.defaultVoice,
6943
+ ttsLang: Me === "browser" ? "en-US" : t,
6944
+ ttsVoice: s || ae.defaultVoice,
6938
6945
  lipsyncLang: "en",
6939
6946
  showFullAvatar: l,
6940
6947
  bodyMovement: a,
6941
6948
  movementIntensity: u
6942
6949
  }, v = {
6943
- ttsEndpoint: he.endpoint,
6944
- ttsApikey: he.apiKey,
6945
- ttsService: ve,
6950
+ ttsEndpoint: ae.endpoint,
6951
+ ttsApikey: ae.apiKey,
6952
+ ttsService: Me,
6946
6953
  lipsyncModules: ["en"],
6947
6954
  cameraView: c
6948
- }, F = U(async () => {
6949
- if (!(!G.current || m.current))
6955
+ }, N = W(async () => {
6956
+ if (!(!Q.current || g.current))
6950
6957
  try {
6951
- if (E(!0), Y(null), m.current = new it(G.current, v), m.current.controls && (m.current.controls.enableRotate = !1, m.current.controls.enableZoom = !1, m.current.controls.enablePan = !1, m.current.controls.enableDamping = !1), x && Object.keys(x).length > 0 && (m.current.customAnimations = x), await m.current.showAvatar(R, (K) => {
6952
- if (K.lengthComputable) {
6953
- const ae = Math.min(100, Math.round(K.loaded / K.total * 100));
6954
- h(ae);
6958
+ if (G(!0), D(null), g.current = new it(Q.current, v), g.current.controls && (g.current.controls.enableRotate = !1, g.current.controls.enableZoom = !1, g.current.controls.enablePan = !1, g.current.controls.enableDamping = !1), f && Object.keys(f).length > 0 && (g.current.customAnimations = f), await g.current.showAvatar(R, (te) => {
6959
+ if (te.lengthComputable) {
6960
+ const ue = Math.min(100, Math.round(te.loaded / te.total * 100));
6961
+ h(ue);
6955
6962
  }
6956
- }), await new Promise((K) => {
6957
- const ae = () => {
6958
- m.current.lipsync && Object.keys(m.current.lipsync).length > 0 ? K() : setTimeout(ae, 100);
6963
+ }), await new Promise((te) => {
6964
+ const ue = () => {
6965
+ g.current.lipsync && Object.keys(g.current.lipsync).length > 0 ? te() : setTimeout(ue, 100);
6959
6966
  };
6960
- ae();
6961
- }), m.current && m.current.setShowFullAvatar)
6967
+ ue();
6968
+ }), g.current && g.current.setShowFullAvatar)
6962
6969
  try {
6963
- m.current.setShowFullAvatar(l);
6964
- } catch (K) {
6965
- console.warn("Error setting full body mode on initialization:", K);
6970
+ g.current.setShowFullAvatar(l);
6971
+ } catch (te) {
6972
+ console.warn("Error setting full body mode on initialization:", te);
6966
6973
  }
6967
- m.current && m.current.controls && (m.current.controls.enableRotate = !1, m.current.controls.enableZoom = !1, m.current.controls.enablePan = !1, m.current.controls.enableDamping = !1, m.current.controls.update()), E(!1), Le(!0), r(m.current);
6974
+ g.current && g.current.controls && (g.current.controls.enableRotate = !1, g.current.controls.enableZoom = !1, g.current.controls.enablePan = !1, g.current.controls.enableDamping = !1, g.current.controls.update()), G(!1), fe(!0), r(g.current);
6968
6975
  const _ = () => {
6969
- document.visibilityState === "visible" ? m.current?.start() : m.current?.stop();
6976
+ document.visibilityState === "visible" ? g.current?.start() : g.current?.stop();
6970
6977
  };
6971
6978
  return document.addEventListener("visibilitychange", _), () => {
6972
6979
  document.removeEventListener("visibilitychange", _);
6973
6980
  };
6974
- } catch (C) {
6975
- console.error("Error initializing TalkingHead:", C), Y(C.message || "Failed to initialize avatar"), E(!1), d(C);
6981
+ } catch (w) {
6982
+ console.error("Error initializing TalkingHead:", w), D(w.message || "Failed to initialize avatar"), G(!1), d(w);
6976
6983
  }
6977
- }, [j, n, e, t, o, s, i, l, a, u, c]);
6978
- Ce(() => (F(), () => {
6979
- m.current && (m.current.stop(), m.current.dispose(), m.current = null);
6980
- }), [F]), Ce(() => {
6981
- if (!G.current || !m.current) return;
6982
- const C = new ResizeObserver((K) => {
6983
- for (const ae of K)
6984
- m.current && m.current.onResize && m.current.onResize();
6984
+ }, [Y, n, e, t, o, s, i, l, a, u, c]);
6985
+ ze(() => (N(), () => {
6986
+ g.current && (g.current.stop(), g.current.dispose(), g.current = null);
6987
+ }), [N]), ze(() => {
6988
+ if (!Q.current || !g.current) return;
6989
+ const w = new ResizeObserver((te) => {
6990
+ for (const ue of te)
6991
+ g.current && g.current.onResize && g.current.onResize();
6985
6992
  });
6986
- C.observe(G.current);
6993
+ w.observe(Q.current);
6987
6994
  const _ = () => {
6988
- m.current && m.current.onResize && m.current.onResize();
6995
+ g.current && g.current.onResize && g.current.onResize();
6989
6996
  };
6990
6997
  return window.addEventListener("resize", _), () => {
6991
- C.disconnect(), window.removeEventListener("resize", _);
6998
+ w.disconnect(), window.removeEventListener("resize", _);
6992
6999
  };
6993
- }, [$]);
6994
- const O = U(async () => {
6995
- if (m.current && m.current.audioCtx)
7000
+ }, [K]);
7001
+ const B = W(async () => {
7002
+ if (g.current && g.current.audioCtx)
6996
7003
  try {
6997
- (m.current.audioCtx.state === "suspended" || m.current.audioCtx.state === "interrupted") && (await m.current.audioCtx.resume(), console.log("Audio context resumed"));
6998
- } catch (C) {
6999
- console.warn("Failed to resume audio context:", C);
7004
+ (g.current.audioCtx.state === "suspended" || g.current.audioCtx.state === "interrupted") && (await g.current.audioCtx.resume(), console.log("Audio context resumed"));
7005
+ } catch (w) {
7006
+ console.warn("Failed to resume audio context:", w);
7000
7007
  }
7001
- }, []), W = U(async (C, _ = {}) => {
7002
- if (m.current && $)
7008
+ }, []), j = W(async (w, _ = {}) => {
7009
+ if (g.current && K)
7003
7010
  try {
7004
- f.current && (clearInterval(f.current), f.current = null), T.current = { text: C, options: _ }, L.current = { remainingText: null, originalText: null, options: null };
7005
- const K = /[!\.\?\n\p{Extended_Pictographic}]/ug, ae = C.split(K).map((se) => se.trim()).filter((se) => se.length > 0);
7006
- y.current = ae, M.current = 0, Pe(!1), B.current = !1, await O();
7007
- const ge = {
7011
+ y.current && (clearInterval(y.current), y.current = null), M.current = { text: w, options: _ }, S.current = { remainingText: null, originalText: null, options: null };
7012
+ const te = /[!\.\?\n\p{Extended_Pictographic}]/ug, ue = w.split(te).map((ne) => ne.trim()).filter((ne) => ne.length > 0);
7013
+ C.current = ue, x.current = 0, Te(!1), E.current = !1, await B();
7014
+ const de = {
7008
7015
  ..._,
7009
7016
  lipsyncLang: _.lipsyncLang || R.lipsyncLang || "en"
7010
7017
  };
7011
- if (_.onSpeechEnd && m.current) {
7012
- const se = m.current;
7013
- let ke = null, Ue = 0;
7014
- const ze = 1200;
7015
- let Ee = !1;
7016
- ke = setInterval(() => {
7017
- if (Ue++, B.current)
7018
+ if (_.onSpeechEnd && g.current) {
7019
+ const ne = g.current;
7020
+ let me = null, Xe = 0;
7021
+ const Oe = 1200;
7022
+ let Fe = !1;
7023
+ me = setInterval(() => {
7024
+ if (Xe++, E.current)
7018
7025
  return;
7019
- if (Ue > ze) {
7020
- if (ke && (clearInterval(ke), ke = null, f.current = null), !Ee && !B.current) {
7021
- Ee = !0;
7026
+ if (Xe > Oe) {
7027
+ if (me && (clearInterval(me), me = null, y.current = null), !Fe && !E.current) {
7028
+ Fe = !0;
7022
7029
  try {
7023
7030
  _.onSpeechEnd();
7024
- } catch (Ie) {
7025
- console.error("Error in onSpeechEnd callback (timeout):", Ie);
7031
+ } catch (Ge) {
7032
+ console.error("Error in onSpeechEnd callback (timeout):", Ge);
7026
7033
  }
7027
7034
  }
7028
7035
  return;
7029
7036
  }
7030
- const He = !se.speechQueue || se.speechQueue.length === 0, We = !se.audioPlaylist || se.audioPlaylist.length === 0;
7031
- se && se.isSpeaking === !1 && He && We && se.isAudioPlaying === !1 && !Ee && !B.current && setTimeout(() => {
7032
- if (se && !B.current && se.isSpeaking === !1 && (!se.speechQueue || se.speechQueue.length === 0) && (!se.audioPlaylist || se.audioPlaylist.length === 0) && se.isAudioPlaying === !1 && !Ee && !B.current) {
7033
- Ee = !0, ke && (clearInterval(ke), ke = null, f.current = null);
7037
+ const Re = !ne.speechQueue || ne.speechQueue.length === 0, Ue = !ne.audioPlaylist || ne.audioPlaylist.length === 0;
7038
+ ne && ne.isSpeaking === !1 && Re && Ue && ne.isAudioPlaying === !1 && !Fe && !E.current && setTimeout(() => {
7039
+ if (ne && !E.current && ne.isSpeaking === !1 && (!ne.speechQueue || ne.speechQueue.length === 0) && (!ne.audioPlaylist || ne.audioPlaylist.length === 0) && ne.isAudioPlaying === !1 && !Fe && !E.current) {
7040
+ Fe = !0, me && (clearInterval(me), me = null, y.current = null);
7034
7041
  try {
7035
7042
  _.onSpeechEnd();
7036
- } catch (nt) {
7037
- console.error("Error in onSpeechEnd callback:", nt);
7043
+ } catch (we) {
7044
+ console.error("Error in onSpeechEnd callback:", we);
7038
7045
  }
7039
7046
  }
7040
7047
  }, 100);
7041
- }, 100), f.current = ke;
7048
+ }, 100), y.current = me;
7042
7049
  }
7043
- m.current.lipsync && Object.keys(m.current.lipsync).length > 0 ? (m.current.setSlowdownRate && m.current.setSlowdownRate(1.05), m.current.speakText(C, ge)) : setTimeout(async () => {
7044
- await O(), m.current && m.current.lipsync && (m.current.setSlowdownRate && m.current.setSlowdownRate(1.05), m.current.speakText(C, ge));
7050
+ g.current.lipsync && Object.keys(g.current.lipsync).length > 0 ? (g.current.setSlowdownRate && g.current.setSlowdownRate(1.05), g.current.speakText(w, de)) : setTimeout(async () => {
7051
+ await B(), g.current && g.current.lipsync && (g.current.setSlowdownRate && g.current.setSlowdownRate(1.05), g.current.speakText(w, de));
7045
7052
  }, 100);
7046
- } catch (K) {
7047
- console.error("Error speaking text:", K), Y(K.message || "Failed to speak text");
7053
+ } catch (te) {
7054
+ console.error("Error speaking text:", te), D(te.message || "Failed to speak text");
7048
7055
  }
7049
- }, [$, O, R.lipsyncLang]), Z = U(() => {
7050
- m.current && (m.current.stopSpeaking(), m.current.setSlowdownRate && m.current.setSlowdownRate(1), T.current = null, Pe(!1));
7051
- }, []), q = U(() => {
7052
- if (m.current && m.current.pauseSpeaking) {
7053
- const C = m.current;
7054
- if (C.isSpeaking || C.audioPlaylist && C.audioPlaylist.length > 0 || C.speechQueue && C.speechQueue.length > 0) {
7055
- f.current && (clearInterval(f.current), f.current = null);
7056
- let K = "";
7057
- if (T.current && y.current.length > 0) {
7058
- const ae = y.current.length, ge = C.speechQueue ? C.speechQueue.filter((ze) => ze && ze.text && Array.isArray(ze.text) && ze.text.length > 0).length : 0, se = C.audioPlaylist && C.audioPlaylist.length > 0, ke = ge + (se ? 1 : 0), Ue = ae - ke;
7059
- if (ke > 0 && Ue < ae && (K = y.current.slice(Ue).join(". ").trim(), !K && ge > 0 && C.speechQueue)) {
7060
- const Ee = C.speechQueue.filter((He) => He && He.text && Array.isArray(He.text) && He.text.length > 0).map((He) => He.text.map((We) => We.word || "").filter((We) => We.length > 0).join(" ")).filter((He) => He.length > 0).join(" ");
7061
- Ee && Ee.trim() && (K = Ee.trim());
7056
+ }, [K, B, R.lipsyncLang]), oe = W(() => {
7057
+ g.current && (g.current.stopSpeaking(), g.current.setSlowdownRate && g.current.setSlowdownRate(1), M.current = null, Te(!1));
7058
+ }, []), X = W(() => {
7059
+ if (g.current && g.current.pauseSpeaking) {
7060
+ const w = g.current;
7061
+ if (w.isSpeaking || w.audioPlaylist && w.audioPlaylist.length > 0 || w.speechQueue && w.speechQueue.length > 0) {
7062
+ y.current && (clearInterval(y.current), y.current = null);
7063
+ let te = "";
7064
+ if (M.current && C.current.length > 0) {
7065
+ const ue = C.current.length, de = w.speechQueue ? w.speechQueue.filter((Oe) => Oe && Oe.text && Array.isArray(Oe.text) && Oe.text.length > 0).length : 0, ne = w.audioPlaylist && w.audioPlaylist.length > 0, me = de + (ne ? 1 : 0), Xe = ue - me;
7066
+ if (me > 0 && Xe < ue && (te = C.current.slice(Xe).join(". ").trim(), !te && de > 0 && w.speechQueue)) {
7067
+ const Fe = w.speechQueue.filter((Re) => Re && Re.text && Array.isArray(Re.text) && Re.text.length > 0).map((Re) => Re.text.map((Ue) => Ue.word || "").filter((Ue) => Ue.length > 0).join(" ")).filter((Re) => Re.length > 0).join(" ");
7068
+ Fe && Fe.trim() && (te = Fe.trim());
7062
7069
  }
7063
7070
  }
7064
- T.current && (L.current = {
7065
- remainingText: K || null,
7066
- originalText: T.current.text,
7067
- options: T.current.options
7068
- }), C.speechQueue && (C.speechQueue.length = 0), m.current.pauseSpeaking(), B.current = !0, Pe(!0);
7071
+ M.current && (S.current = {
7072
+ remainingText: te || null,
7073
+ originalText: M.current.text,
7074
+ options: M.current.options
7075
+ }), w.speechQueue && (w.speechQueue.length = 0), g.current.pauseSpeaking(), E.current = !0, Te(!0);
7069
7076
  }
7070
7077
  }
7071
- }, []), ne = U(async () => {
7072
- if (!m.current || !pe)
7078
+ }, []), q = W(async () => {
7079
+ if (!g.current || !xe)
7073
7080
  return;
7074
- let C = "", _ = {};
7075
- if (L.current && L.current.remainingText)
7076
- C = L.current.remainingText, _ = L.current.options || {}, L.current = { remainingText: null, originalText: null, options: null };
7077
- else if (T.current && T.current.text)
7078
- C = T.current.text, _ = T.current.options || {};
7081
+ let w = "", _ = {};
7082
+ if (S.current && S.current.remainingText)
7083
+ w = S.current.remainingText, _ = S.current.options || {}, S.current = { remainingText: null, originalText: null, options: null };
7084
+ else if (M.current && M.current.text)
7085
+ w = M.current.text, _ = M.current.options || {};
7079
7086
  else {
7080
- console.warn("Resume called but no paused speech found"), Pe(!1), B.current = !1;
7087
+ console.warn("Resume called but no paused speech found"), Te(!1), E.current = !1;
7081
7088
  return;
7082
7089
  }
7083
- Pe(!1), B.current = !1, await O();
7084
- const K = {
7090
+ Te(!1), E.current = !1, await B();
7091
+ const te = {
7085
7092
  ..._,
7086
7093
  lipsyncLang: _.lipsyncLang || R.lipsyncLang || "en"
7087
7094
  };
7088
7095
  try {
7089
- await W(C, K);
7090
- } catch (ae) {
7091
- console.error("Error resuming speech:", ae), Pe(!1), B.current = !1;
7096
+ await j(w, te);
7097
+ } catch (ue) {
7098
+ console.error("Error resuming speech:", ue), Te(!1), E.current = !1;
7092
7099
  }
7093
- }, [O, pe, W, R]), me = U((C) => {
7094
- m.current && m.current.setMood(C);
7095
- }, []), fe = U((C) => {
7096
- m.current && m.current.setSlowdownRate && m.current.setSlowdownRate(C);
7097
- }, []), et = U((C, _ = !1) => {
7098
- if (m.current && m.current.playAnimation) {
7099
- if (x && x[C] && (C = x[C]), m.current.setShowFullAvatar)
7100
+ }, [B, xe, j, R]), Se = W((w) => {
7101
+ g.current && g.current.setMood(w);
7102
+ }, []), be = W((w) => {
7103
+ g.current && g.current.setSlowdownRate && g.current.setSlowdownRate(w);
7104
+ }, []), Ae = W((w, _ = !1) => {
7105
+ if (g.current && g.current.playAnimation) {
7106
+ if (f && f[w] && (w = f[w]), g.current.setShowFullAvatar)
7100
7107
  try {
7101
- m.current.setShowFullAvatar(D.current);
7102
- } catch (ae) {
7103
- console.warn("Error setting full body mode:", ae);
7108
+ g.current.setShowFullAvatar(H.current);
7109
+ } catch (ue) {
7110
+ console.warn("Error setting full body mode:", ue);
7104
7111
  }
7105
- if (C.includes("."))
7112
+ if (w.includes("."))
7106
7113
  try {
7107
- m.current.playAnimation(C, null, 10, 0, 0.01, _);
7108
- } catch (ae) {
7109
- console.warn(`Failed to play ${C}:`, ae);
7114
+ g.current.playAnimation(w, null, 10, 0, 0.01, _);
7115
+ } catch (ue) {
7116
+ console.warn(`Failed to play ${w}:`, ue);
7110
7117
  try {
7111
- m.current.setBodyMovement("idle");
7112
- } catch (ge) {
7113
- console.warn("Fallback animation also failed:", ge);
7118
+ g.current.setBodyMovement("idle");
7119
+ } catch (de) {
7120
+ console.warn("Fallback animation also failed:", de);
7114
7121
  }
7115
7122
  }
7116
7123
  else {
7117
- const ae = [".fbx", ".glb", ".gltf"];
7118
- let ge = !1;
7119
- for (const se of ae)
7124
+ const ue = [".fbx", ".glb", ".gltf"];
7125
+ let de = !1;
7126
+ for (const ne of ue)
7120
7127
  try {
7121
- m.current.playAnimation(C + se, null, 10, 0, 0.01, _), ge = !0;
7128
+ g.current.playAnimation(w + ne, null, 10, 0, 0.01, _), de = !0;
7122
7129
  break;
7123
7130
  } catch {
7124
7131
  }
7125
- if (!ge) {
7126
- console.warn("Animation not found:", C);
7132
+ if (!de) {
7133
+ console.warn("Animation not found:", w);
7127
7134
  try {
7128
- m.current.setBodyMovement("idle");
7129
- } catch (se) {
7130
- console.warn("Fallback animation also failed:", se);
7135
+ g.current.setBodyMovement("idle");
7136
+ } catch (ne) {
7137
+ console.warn("Fallback animation also failed:", ne);
7131
7138
  }
7132
7139
  }
7133
7140
  }
7134
7141
  }
7135
- }, [x]), Se = U(() => {
7136
- m.current && m.current.onResize && m.current.onResize();
7142
+ }, [f]), et = W(() => {
7143
+ g.current && g.current.onResize && g.current.onResize();
7137
7144
  }, []);
7138
- return $e(k, () => ({
7139
- speakText: W,
7140
- stopSpeaking: Z,
7141
- pauseSpeaking: q,
7142
- resumeSpeaking: ne,
7143
- resumeAudioContext: O,
7144
- setMood: me,
7145
- setTimingAdjustment: fe,
7146
- playAnimation: et,
7147
- isReady: $,
7148
- isPaused: pe,
7149
- talkingHead: m.current,
7150
- handleResize: Se,
7151
- setBodyMovement: (C) => {
7152
- if (m.current && m.current.setShowFullAvatar && m.current.setBodyMovement)
7145
+ return $e(L, () => ({
7146
+ speakText: j,
7147
+ stopSpeaking: oe,
7148
+ pauseSpeaking: X,
7149
+ resumeSpeaking: q,
7150
+ resumeAudioContext: B,
7151
+ setMood: Se,
7152
+ setTimingAdjustment: be,
7153
+ playAnimation: Ae,
7154
+ isReady: K,
7155
+ isPaused: xe,
7156
+ talkingHead: g.current,
7157
+ handleResize: et,
7158
+ setBodyMovement: (w) => {
7159
+ if (g.current && g.current.setShowFullAvatar && g.current.setBodyMovement)
7153
7160
  try {
7154
- m.current.setShowFullAvatar(D.current), m.current.setBodyMovement(C);
7161
+ g.current.setShowFullAvatar(H.current), g.current.setBodyMovement(w);
7155
7162
  } catch (_) {
7156
7163
  console.warn("Error setting body movement:", _);
7157
7164
  }
7158
7165
  },
7159
- setMovementIntensity: (C) => m.current?.setMovementIntensity(C),
7166
+ setMovementIntensity: (w) => g.current?.setMovementIntensity(w),
7160
7167
  playRandomDance: () => {
7161
- if (m.current && m.current.setShowFullAvatar && m.current.playRandomDance)
7168
+ if (g.current && g.current.setShowFullAvatar && g.current.playRandomDance)
7162
7169
  try {
7163
- m.current.setShowFullAvatar(D.current), m.current.playRandomDance();
7164
- } catch (C) {
7165
- console.warn("Error playing random dance:", C);
7170
+ g.current.setShowFullAvatar(H.current), g.current.playRandomDance();
7171
+ } catch (w) {
7172
+ console.warn("Error playing random dance:", w);
7166
7173
  }
7167
7174
  },
7168
- playReaction: (C) => {
7169
- if (m.current && m.current.setShowFullAvatar && m.current.playReaction)
7175
+ playReaction: (w) => {
7176
+ if (g.current && g.current.setShowFullAvatar && g.current.playReaction)
7170
7177
  try {
7171
- m.current.setShowFullAvatar(D.current), m.current.playReaction(C);
7178
+ g.current.setShowFullAvatar(H.current), g.current.playReaction(w);
7172
7179
  } catch (_) {
7173
7180
  console.warn("Error playing reaction:", _);
7174
7181
  }
7175
7182
  },
7176
7183
  playCelebration: () => {
7177
- if (m.current && m.current.setShowFullAvatar && m.current.playCelebration)
7184
+ if (g.current && g.current.setShowFullAvatar && g.current.playCelebration)
7178
7185
  try {
7179
- m.current.setShowFullAvatar(D.current), m.current.playCelebration();
7180
- } catch (C) {
7181
- console.warn("Error playing celebration:", C);
7186
+ g.current.setShowFullAvatar(H.current), g.current.playCelebration();
7187
+ } catch (w) {
7188
+ console.warn("Error playing celebration:", w);
7182
7189
  }
7183
7190
  },
7184
- setShowFullAvatar: (C) => {
7185
- if (m.current && m.current.setShowFullAvatar)
7191
+ setShowFullAvatar: (w) => {
7192
+ if (g.current && g.current.setShowFullAvatar)
7186
7193
  try {
7187
- D.current = C, m.current.setShowFullAvatar(C);
7194
+ H.current = w, g.current.setShowFullAvatar(w);
7188
7195
  } catch (_) {
7189
7196
  console.warn("Error setting showFullAvatar:", _);
7190
7197
  }
7191
7198
  },
7192
7199
  lockAvatarPosition: () => {
7193
- if (m.current && m.current.lockAvatarPosition)
7200
+ if (g.current && g.current.lockAvatarPosition)
7194
7201
  try {
7195
- m.current.lockAvatarPosition();
7196
- } catch (C) {
7197
- console.warn("Error locking avatar position:", C);
7202
+ g.current.lockAvatarPosition();
7203
+ } catch (w) {
7204
+ console.warn("Error locking avatar position:", w);
7198
7205
  }
7199
7206
  },
7200
7207
  unlockAvatarPosition: () => {
7201
- if (m.current && m.current.unlockAvatarPosition)
7208
+ if (g.current && g.current.unlockAvatarPosition)
7202
7209
  try {
7203
- m.current.unlockAvatarPosition();
7204
- } catch (C) {
7205
- console.warn("Error unlocking avatar position:", C);
7210
+ g.current.unlockAvatarPosition();
7211
+ } catch (w) {
7212
+ console.warn("Error unlocking avatar position:", w);
7206
7213
  }
7207
7214
  }
7208
- })), /* @__PURE__ */ Be(
7215
+ })), /* @__PURE__ */ Ee(
7209
7216
  "div",
7210
7217
  {
7211
- className: `talking-head-avatar ${g}`,
7218
+ className: `talking-head-avatar ${p}`,
7212
7219
  style: {
7213
7220
  width: "100%",
7214
7221
  height: "100%",
@@ -7216,10 +7223,10 @@ const ut = Ke(({
7216
7223
  ...b
7217
7224
  },
7218
7225
  children: [
7219
- /* @__PURE__ */ re(
7226
+ /* @__PURE__ */ le(
7220
7227
  "div",
7221
7228
  {
7222
- ref: G,
7229
+ ref: Q,
7223
7230
  className: "talking-head-viewer",
7224
7231
  style: {
7225
7232
  width: "100%",
@@ -7228,7 +7235,7 @@ const ut = Ke(({
7228
7235
  }
7229
7236
  }
7230
7237
  ),
7231
- I && /* @__PURE__ */ re("div", { className: "loading-overlay", style: {
7238
+ I && /* @__PURE__ */ le("div", { className: "loading-overlay", style: {
7232
7239
  position: "absolute",
7233
7240
  top: "50%",
7234
7241
  left: "50%",
@@ -7237,7 +7244,7 @@ const ut = Ke(({
7237
7244
  fontSize: "18px",
7238
7245
  zIndex: 10
7239
7246
  }, children: "Loading avatar..." }),
7240
- P && /* @__PURE__ */ re("div", { className: "error-overlay", style: {
7247
+ P && /* @__PURE__ */ le("div", { className: "error-overlay", style: {
7241
7248
  position: "absolute",
7242
7249
  top: "50%",
7243
7250
  left: "50%",
@@ -7253,9 +7260,9 @@ const ut = Ke(({
7253
7260
  }
7254
7261
  );
7255
7262
  });
7256
- ut.displayName = "TalkingHeadAvatar";
7257
- const Nt = Ke(({
7258
- text: j = "Hello! I'm a talking avatar. How are you today?",
7263
+ ct.displayName = "TalkingHeadAvatar";
7264
+ const Ut = Ke(({
7265
+ text: Y = "Hello! I'm a talking avatar. How are you today?",
7259
7266
  onLoading: n = () => {
7260
7267
  },
7261
7268
  onError: e = () => {
@@ -7266,23 +7273,23 @@ const Nt = Ke(({
7266
7273
  style: s = {},
7267
7274
  avatarConfig: i = {}
7268
7275
  }, a) => {
7269
- const u = V(null), l = V(null), [c, r] = ye(!0), [h, d] = ye(null), [g, b] = ye(!1), x = Je(), k = i.ttsService || x.service, G = k === "browser" ? {
7276
+ const u = V(null), l = V(null), [c, r] = ge(!0), [h, d] = ge(null), [p, b] = ge(!1), f = Je(), L = i.ttsService || f.service, Q = L === "browser" ? {
7270
7277
  endpoint: "",
7271
7278
  apiKey: null,
7272
7279
  defaultVoice: "Google US English"
7273
7280
  } : {
7274
- ...x,
7281
+ ...f,
7275
7282
  // Override API key if provided via avatarConfig
7276
- apiKey: i.ttsApiKey !== void 0 && i.ttsApiKey !== null ? i.ttsApiKey : x.apiKey,
7283
+ apiKey: i.ttsApiKey !== void 0 && i.ttsApiKey !== null ? i.ttsApiKey : f.apiKey,
7277
7284
  // Override endpoint for ElevenLabs if service is explicitly set
7278
- endpoint: k === "elevenlabs" && i.ttsApiKey ? "https://api.elevenlabs.io/v1/text-to-speech" : x.endpoint
7279
- }, m = {
7285
+ endpoint: L === "elevenlabs" && i.ttsApiKey ? "https://api.elevenlabs.io/v1/text-to-speech" : f.endpoint
7286
+ }, g = {
7280
7287
  url: "/avatars/brunette.glb",
7281
7288
  // Use brunette avatar (working glTF file)
7282
7289
  body: "F",
7283
7290
  avatarMood: "neutral",
7284
- ttsLang: k === "browser" ? "en-US" : "en",
7285
- ttsVoice: i.ttsVoice || G.defaultVoice,
7291
+ ttsLang: L === "browser" ? "en-US" : "en",
7292
+ ttsVoice: i.ttsVoice || Q.defaultVoice,
7286
7293
  lipsyncLang: "en",
7287
7294
  // English lip-sync
7288
7295
  showFullAvatar: !0,
@@ -7290,31 +7297,31 @@ const Nt = Ke(({
7290
7297
  bodyMovement: "idle",
7291
7298
  movementIntensity: 0.5,
7292
7299
  ...i
7293
- }, D = {
7294
- ttsEndpoint: G.endpoint,
7295
- ttsApikey: G.apiKey,
7296
- ttsService: k,
7300
+ }, H = {
7301
+ ttsEndpoint: Q.endpoint,
7302
+ ttsApikey: Q.apiKey,
7303
+ ttsService: L,
7297
7304
  lipsyncModules: ["en"],
7298
7305
  cameraView: "upper"
7299
- }, T = U(async () => {
7306
+ }, M = W(async () => {
7300
7307
  if (!(!u.current || l.current))
7301
7308
  try {
7302
- if (r(!0), d(null), l.current = new it(u.current, D), await l.current.showAvatar(m, (P) => {
7309
+ if (r(!0), d(null), l.current = new it(u.current, H), await l.current.showAvatar(g, (P) => {
7303
7310
  if (P.lengthComputable) {
7304
- const Y = Math.min(100, Math.round(P.loaded / P.total * 100));
7305
- n(Y);
7311
+ const D = Math.min(100, Math.round(P.loaded / P.total * 100));
7312
+ n(D);
7306
7313
  }
7307
7314
  }), l.current.morphs && l.current.morphs.length > 0) {
7308
7315
  const P = l.current.morphs[0].morphTargetDictionary;
7309
7316
  console.log("Available morph targets:", Object.keys(P));
7310
- const Y = Object.keys(P).filter(($) => $.startsWith("viseme_"));
7311
- console.log("Viseme morph targets found:", Y), Y.length === 0 && (console.warn("No viseme morph targets found! Lip-sync will not work properly."), console.log("Expected viseme targets: viseme_aa, viseme_E, viseme_I, viseme_O, viseme_U, viseme_PP, viseme_SS, viseme_TH, viseme_DD, viseme_FF, viseme_kk, viseme_nn, viseme_RR, viseme_CH, viseme_sil"));
7317
+ const D = Object.keys(P).filter((K) => K.startsWith("viseme_"));
7318
+ console.log("Viseme morph targets found:", D), D.length === 0 && (console.warn("No viseme morph targets found! Lip-sync will not work properly."), console.log("Expected viseme targets: viseme_aa, viseme_E, viseme_I, viseme_O, viseme_U, viseme_PP, viseme_SS, viseme_TH, viseme_DD, viseme_FF, viseme_kk, viseme_nn, viseme_RR, viseme_CH, viseme_sil"));
7312
7319
  }
7313
7320
  if (await new Promise((P) => {
7314
- const Y = () => {
7315
- l.current.lipsync && Object.keys(l.current.lipsync).length > 0 ? (console.log("Lip-sync modules loaded:", Object.keys(l.current.lipsync)), P()) : (console.log("Waiting for lip-sync modules to load..."), setTimeout(Y, 100));
7321
+ const D = () => {
7322
+ l.current.lipsync && Object.keys(l.current.lipsync).length > 0 ? (console.log("Lip-sync modules loaded:", Object.keys(l.current.lipsync)), P()) : (console.log("Waiting for lip-sync modules to load..."), setTimeout(D, 100));
7316
7323
  };
7317
- Y();
7324
+ D();
7318
7325
  }), l.current && l.current.setShowFullAvatar)
7319
7326
  try {
7320
7327
  l.current.setShowFullAvatar(!0), console.log("Avatar initialized in full body mode");
@@ -7322,71 +7329,71 @@ const Nt = Ke(({
7322
7329
  console.warn("Error setting full body mode on initialization:", P);
7323
7330
  }
7324
7331
  r(!1), b(!0), t(l.current);
7325
- const E = () => {
7332
+ const G = () => {
7326
7333
  document.visibilityState === "visible" ? l.current?.start() : l.current?.stop();
7327
7334
  };
7328
- return document.addEventListener("visibilitychange", E), () => {
7329
- document.removeEventListener("visibilitychange", E);
7335
+ return document.addEventListener("visibilitychange", G), () => {
7336
+ document.removeEventListener("visibilitychange", G);
7330
7337
  };
7331
7338
  } catch (I) {
7332
7339
  console.error("Error initializing TalkingHead:", I), d(I.message || "Failed to initialize avatar"), r(!1), e(I);
7333
7340
  }
7334
7341
  }, []);
7335
- Ce(() => (T(), () => {
7342
+ ze(() => (M(), () => {
7336
7343
  l.current && (l.current.stop(), l.current.dispose(), l.current = null);
7337
- }), [T]);
7338
- const f = U((I) => {
7339
- if (l.current && g)
7344
+ }), [M]);
7345
+ const y = W((I) => {
7346
+ if (l.current && p)
7340
7347
  try {
7341
- console.log("Speaking text:", I), console.log("Avatar config:", m), console.log("TalkingHead instance:", l.current), l.current.lipsync && Object.keys(l.current.lipsync).length > 0 ? (console.log("Lip-sync modules loaded:", Object.keys(l.current.lipsync)), l.current.setSlowdownRate && (l.current.setSlowdownRate(1.05), console.log("Applied timing adjustment for better lip-sync")), l.current.speakText(I)) : (console.warn("Lip-sync modules not ready, waiting..."), setTimeout(() => {
7348
+ console.log("Speaking text:", I), console.log("Avatar config:", g), console.log("TalkingHead instance:", l.current), l.current.lipsync && Object.keys(l.current.lipsync).length > 0 ? (console.log("Lip-sync modules loaded:", Object.keys(l.current.lipsync)), l.current.setSlowdownRate && (l.current.setSlowdownRate(1.05), console.log("Applied timing adjustment for better lip-sync")), l.current.speakText(I)) : (console.warn("Lip-sync modules not ready, waiting..."), setTimeout(() => {
7342
7349
  l.current && l.current.lipsync ? (console.log("Lip-sync now ready, speaking..."), l.current.setSlowdownRate && (l.current.setSlowdownRate(1.05), console.log("Applied timing adjustment for better lip-sync")), l.current.speakText(I)) : console.error("Lip-sync still not ready after waiting");
7343
7350
  }, 500));
7344
- } catch (E) {
7345
- console.error("Error speaking text:", E), d(E.message || "Failed to speak text");
7351
+ } catch (G) {
7352
+ console.error("Error speaking text:", G), d(G.message || "Failed to speak text");
7346
7353
  }
7347
7354
  else
7348
- console.warn("Avatar not ready for speaking. isReady:", g, "talkingHeadRef:", !!l.current);
7349
- }, [g, m]), B = U(() => {
7355
+ console.warn("Avatar not ready for speaking. isReady:", p, "talkingHeadRef:", !!l.current);
7356
+ }, [p, g]), E = W(() => {
7350
7357
  l.current && (l.current.stopSpeaking(), l.current.setSlowdownRate && (l.current.setSlowdownRate(1), console.log("Reset timing to normal")));
7351
- }, []), L = U((I) => {
7358
+ }, []), S = W((I) => {
7352
7359
  l.current && l.current.setMood(I);
7353
- }, []), y = U((I) => {
7360
+ }, []), C = W((I) => {
7354
7361
  l.current && l.current.setSlowdownRate && (l.current.setSlowdownRate(I), console.log("Timing adjustment set to:", I));
7355
- }, []), M = U((I, E = !1) => {
7362
+ }, []), x = W((I, G = !1) => {
7356
7363
  if (l.current && l.current.playAnimation) {
7357
7364
  if (l.current.setShowFullAvatar)
7358
7365
  try {
7359
7366
  l.current.setShowFullAvatar(!0);
7360
- } catch (Y) {
7361
- console.warn("Error setting full body mode:", Y);
7367
+ } catch (D) {
7368
+ console.warn("Error setting full body mode:", D);
7362
7369
  }
7363
7370
  if (I.includes("."))
7364
7371
  try {
7365
- l.current.playAnimation(I, null, 10, 0, 0.01, E), console.log("Playing animation:", I);
7366
- } catch (Y) {
7367
- console.log(`Failed to play ${I}:`, Y);
7372
+ l.current.playAnimation(I, null, 10, 0, 0.01, G), console.log("Playing animation:", I);
7373
+ } catch (D) {
7374
+ console.log(`Failed to play ${I}:`, D);
7368
7375
  try {
7369
7376
  l.current.setBodyMovement("idle"), console.log("Fallback to idle animation");
7370
- } catch ($) {
7371
- console.warn("Fallback animation also failed:", $);
7377
+ } catch (K) {
7378
+ console.warn("Fallback animation also failed:", K);
7372
7379
  }
7373
7380
  }
7374
7381
  else {
7375
- const Y = [".fbx", ".glb", ".gltf"];
7376
- let $ = !1;
7377
- for (const Le of Y)
7382
+ const D = [".fbx", ".glb", ".gltf"];
7383
+ let K = !1;
7384
+ for (const fe of D)
7378
7385
  try {
7379
- l.current.playAnimation(I + Le, null, 10, 0, 0.01, E), console.log("Playing animation:", I + Le), $ = !0;
7386
+ l.current.playAnimation(I + fe, null, 10, 0, 0.01, G), console.log("Playing animation:", I + fe), K = !0;
7380
7387
  break;
7381
7388
  } catch {
7382
- console.log(`Failed to play ${I}${Le}, trying next format...`);
7389
+ console.log(`Failed to play ${I}${fe}, trying next format...`);
7383
7390
  }
7384
- if (!$) {
7391
+ if (!K) {
7385
7392
  console.warn("Animation system not available or animation not found:", I);
7386
7393
  try {
7387
7394
  l.current.setBodyMovement("idle"), console.log("Fallback to idle animation");
7388
- } catch (Le) {
7389
- console.warn("Fallback animation also failed:", Le);
7395
+ } catch (fe) {
7396
+ console.warn("Fallback animation also failed:", fe);
7390
7397
  }
7391
7398
  }
7392
7399
  }
@@ -7394,19 +7401,19 @@ const Nt = Ke(({
7394
7401
  console.warn("Animation system not available or animation not found:", I);
7395
7402
  }, []);
7396
7403
  return $e(a, () => ({
7397
- speakText: f,
7398
- stopSpeaking: B,
7399
- setMood: L,
7400
- setTimingAdjustment: y,
7401
- playAnimation: M,
7402
- isReady: g,
7404
+ speakText: y,
7405
+ stopSpeaking: E,
7406
+ setMood: S,
7407
+ setTimingAdjustment: C,
7408
+ playAnimation: x,
7409
+ isReady: p,
7403
7410
  talkingHead: l.current,
7404
7411
  setBodyMovement: (I) => {
7405
7412
  if (l.current && l.current.setShowFullAvatar && l.current.setBodyMovement)
7406
7413
  try {
7407
7414
  l.current.setShowFullAvatar(!0), l.current.setBodyMovement(I), console.log("Body movement set with full body mode:", I);
7408
- } catch (E) {
7409
- console.warn("Error setting body movement:", E);
7415
+ } catch (G) {
7416
+ console.warn("Error setting body movement:", G);
7410
7417
  }
7411
7418
  },
7412
7419
  setMovementIntensity: (I) => l.current?.setMovementIntensity(I),
@@ -7422,8 +7429,8 @@ const Nt = Ke(({
7422
7429
  if (l.current && l.current.setShowFullAvatar && l.current.playReaction)
7423
7430
  try {
7424
7431
  l.current.setShowFullAvatar(!0), l.current.playReaction(I), console.log("Reaction played with full body mode:", I);
7425
- } catch (E) {
7426
- console.warn("Error playing reaction:", E);
7432
+ } catch (G) {
7433
+ console.warn("Error playing reaction:", G);
7427
7434
  }
7428
7435
  },
7429
7436
  playCelebration: () => {
@@ -7438,8 +7445,8 @@ const Nt = Ke(({
7438
7445
  if (l.current && l.current.setShowFullAvatar)
7439
7446
  try {
7440
7447
  l.current.setShowFullAvatar(I), console.log("Show full avatar set to:", I);
7441
- } catch (E) {
7442
- console.warn("Error setting showFullAvatar:", E);
7448
+ } catch (G) {
7449
+ console.warn("Error setting showFullAvatar:", G);
7443
7450
  }
7444
7451
  },
7445
7452
  lockAvatarPosition: () => {
@@ -7458,8 +7465,8 @@ const Nt = Ke(({
7458
7465
  console.warn("Error unlocking avatar position:", I);
7459
7466
  }
7460
7467
  }
7461
- })), /* @__PURE__ */ Be("div", { className: `talking-head-container ${o}`, style: s, children: [
7462
- /* @__PURE__ */ re(
7468
+ })), /* @__PURE__ */ Ee("div", { className: `talking-head-container ${o}`, style: s, children: [
7469
+ /* @__PURE__ */ le(
7463
7470
  "div",
7464
7471
  {
7465
7472
  ref: u,
@@ -7471,7 +7478,7 @@ const Nt = Ke(({
7471
7478
  }
7472
7479
  }
7473
7480
  ),
7474
- c && /* @__PURE__ */ re("div", { className: "loading-overlay", style: {
7481
+ c && /* @__PURE__ */ le("div", { className: "loading-overlay", style: {
7475
7482
  position: "absolute",
7476
7483
  top: "50%",
7477
7484
  left: "50%",
@@ -7480,7 +7487,7 @@ const Nt = Ke(({
7480
7487
  fontSize: "18px",
7481
7488
  zIndex: 10
7482
7489
  }, children: "Loading avatar..." }),
7483
- h && /* @__PURE__ */ re("div", { className: "error-overlay", style: {
7490
+ h && /* @__PURE__ */ le("div", { className: "error-overlay", style: {
7484
7491
  position: "absolute",
7485
7492
  top: "50%",
7486
7493
  left: "50%",
@@ -7494,10 +7501,10 @@ const Nt = Ke(({
7494
7501
  }, children: h })
7495
7502
  ] });
7496
7503
  });
7497
- Nt.displayName = "TalkingHeadComponent";
7498
- async function je(j) {
7504
+ Ut.displayName = "TalkingHeadComponent";
7505
+ async function je(Y) {
7499
7506
  try {
7500
- const n = await fetch(j);
7507
+ const n = await fetch(Y);
7501
7508
  if (!n.ok) {
7502
7509
  if (n.status === 404)
7503
7510
  return {};
@@ -7509,11 +7516,11 @@ async function je(j) {
7509
7516
  const t = await n.text();
7510
7517
  return t.trim().startsWith("<!") ? {} : JSON.parse(t).animations || {};
7511
7518
  } catch (n) {
7512
- return n instanceof SyntaxError || console.warn("⚠️ Could not load animation manifest (this is optional):", j), {};
7519
+ return n instanceof SyntaxError || console.warn("⚠️ Could not load animation manifest (this is optional):", Y), {};
7513
7520
  }
7514
7521
  }
7515
- async function Ut(j, n = "F") {
7516
- const e = [], t = j.replace(/\/$/, "");
7522
+ async function Wt(Y, n = "F") {
7523
+ const e = [], t = Y.replace(/\/$/, "");
7517
7524
  try {
7518
7525
  const i = [
7519
7526
  `${t}/.list.json`,
@@ -7556,19 +7563,19 @@ async function Ut(j, n = "F") {
7556
7563
  }
7557
7564
  return e.length > 0, e;
7558
7565
  }
7559
- async function ct(j, n = "F") {
7566
+ async function lt(Y, n = "F") {
7560
7567
  const e = {};
7561
- for (const [t, o] of Object.entries(j))
7568
+ for (const [t, o] of Object.entries(Y))
7562
7569
  try {
7563
- const s = await Ut(o, n);
7570
+ const s = await Wt(o, n);
7564
7571
  s.length > 0 && (e[t] = s);
7565
7572
  } catch (s) {
7566
7573
  console.error(`❌ Failed to auto-load animations from ${o}:`, s);
7567
7574
  }
7568
7575
  return e;
7569
7576
  }
7570
- const Wt = Ke(({
7571
- text: j = null,
7577
+ const Vt = Ke(({
7578
+ text: Y = null,
7572
7579
  avatarUrl: n = "/avatars/brunette.glb",
7573
7580
  avatarBody: e = "F",
7574
7581
  mood: t = "neutral",
@@ -7584,467 +7591,472 @@ const Wt = Ke(({
7584
7591
  },
7585
7592
  onLoading: d = () => {
7586
7593
  },
7587
- onError: g = () => {
7594
+ onError: p = () => {
7588
7595
  },
7589
7596
  onSpeechStart: b = () => {
7590
7597
  },
7591
- onSpeechEnd: x = () => {
7598
+ onSpeechEnd: f = () => {
7592
7599
  },
7593
- className: k = "",
7594
- style: G = {},
7595
- animations: m = {},
7596
- autoAnimationGroup: D = null,
7597
- // e.g., "talking" - will randomly select from this group when speaking
7598
- autoIdleGroup: T = null,
7599
- // e.g., "idle" - will randomly select from this group when idle
7600
- autoSpeak: f = !1
7601
- }, B) => {
7602
- const L = V(null), y = V(null), M = V(c), I = V(null), E = V(null), P = V(!1), Y = V({ remainingText: null, originalText: null, options: null }), $ = V([]), [Le, pe] = ye(!0), [Pe, be] = ye(null), [ve, he] = ye(!1), [R, v] = ye(!1), [F, O] = ye(m), W = V(null), Z = V(!1), q = V(null), ne = V(!1), me = V([]), fe = V([]), et = V(0), Se = V(null), C = V(null);
7603
- Ce(() => {
7604
- P.current = R;
7605
- }, [R]);
7606
- const _ = U((S) => {
7607
- let w = "F";
7608
- if (S) {
7609
- const H = S.toString().toUpperCase().trim();
7610
- H === "M" || H === "MALE" ? w = "M" : (H === "F" || H === "FEMALE") && (w = "F");
7600
+ onSubtitle: L = null,
7601
+ className: Q = "",
7602
+ style: g = {},
7603
+ animations: H = {},
7604
+ autoAnimationGroup: M = null,
7605
+ // e.g., "talking" - will select from this group in order when speaking
7606
+ autoIdleGroup: y = null,
7607
+ // e.g., "idle" - will select from this group in order when idle
7608
+ autoSpeak: E = !1
7609
+ }, S) => {
7610
+ const C = V(null), x = V(null), I = V(c), G = V(null), P = V(null), D = V(!1), K = V({ remainingText: null, originalText: null, options: null }), fe = V([]), [xe, Te] = ge(!0), [Le, Me] = ge(null), [ae, R] = ge(!1), [v, N] = ge(!1), [B, j] = ge(H), oe = V(null), X = V(!1), q = V(null), Se = V(!1), be = V([]), Ae = V([]), et = V(0), w = V(null), _ = V(null), te = V("");
7611
+ ze(() => {
7612
+ D.current = v;
7613
+ }, [v]);
7614
+ const ue = W((k) => {
7615
+ let z = "F";
7616
+ if (k) {
7617
+ const F = k.toString().toUpperCase().trim();
7618
+ F === "M" || F === "MALE" ? z = "M" : (F === "F" || F === "FEMALE") && (z = "F");
7611
7619
  }
7612
- return w === "M" ? "male" : "female";
7620
+ return z === "M" ? "male" : "female";
7613
7621
  }, []);
7614
- Ce(() => {
7622
+ ze(() => {
7615
7623
  (async () => {
7616
- if (m.manifest && m.auto)
7624
+ if (H.manifest && H.auto)
7617
7625
  try {
7618
- const w = await je(m.manifest);
7619
- O(w);
7626
+ const z = await je(H.manifest);
7627
+ j(z);
7620
7628
  return;
7621
7629
  } catch {
7622
7630
  }
7623
- if (m.manifest && !m.auto) {
7624
- const w = await je(m.manifest), H = Object.keys(w).length > 0;
7625
- O(H ? w : m);
7626
- } else if (m.auto)
7631
+ if (H.manifest && !H.auto) {
7632
+ const z = await je(H.manifest), F = Object.keys(z).length > 0;
7633
+ j(F ? z : H);
7634
+ } else if (H.auto)
7627
7635
  try {
7628
- let w = null;
7629
- if (m.manifest)
7636
+ let z = null;
7637
+ if (H.manifest)
7630
7638
  try {
7631
- w = await je(m.manifest);
7639
+ z = await je(H.manifest);
7632
7640
  } catch {
7633
7641
  }
7634
- if (typeof m.auto == "string") {
7635
- const H = m.auto, J = {
7636
- talking: `${H}/talking`,
7637
- idle: `${H}/idle`
7638
- }, X = _(e);
7639
- J[`${X}_talking`] = `${H}/${X}/talking`, J[`${X}_idle`] = `${H}/${X}/idle`, J.shared_talking = `${H}/shared/talking`, J.shared_idle = `${H}/shared/idle`;
7640
- const ie = await ct(J, e);
7641
- if (!Object.values(ie).some((Q) => Array.isArray(Q) && Q.length > 0) && w) {
7642
- O(w);
7642
+ if (typeof H.auto == "string") {
7643
+ const F = H.auto, se = {
7644
+ talking: `${F}/talking`,
7645
+ idle: `${F}/idle`
7646
+ }, Z = ue(e);
7647
+ se[`${Z}_talking`] = `${F}/${Z}/talking`, se[`${Z}_idle`] = `${F}/${Z}/idle`, se.shared_talking = `${F}/shared/talking`, se.shared_idle = `${F}/shared/idle`;
7648
+ const ke = await lt(se, e);
7649
+ if (!Object.values(ke).some(($) => Array.isArray($) && $.length > 0) && z) {
7650
+ j(z);
7643
7651
  return;
7644
7652
  }
7645
- const ce = {
7653
+ const ee = {
7646
7654
  _genderSpecific: {
7647
- [X]: {},
7655
+ [Z]: {},
7648
7656
  shared: {}
7649
7657
  }
7650
7658
  };
7651
- Object.entries(ie).forEach(([Q, ee]) => {
7652
- if (Q.includes("_")) {
7653
- const [te, ...xe] = Q.split("_"), de = xe.join("_");
7654
- te === "shared" ? (ce._genderSpecific.shared[de] || (ce._genderSpecific.shared[de] = []), ce._genderSpecific.shared[de].push(...ee)) : te === X && (ce._genderSpecific[X][de] || (ce._genderSpecific[X][de] = []), ce._genderSpecific[X][de].push(...ee));
7659
+ Object.entries(ke).forEach(([$, ie]) => {
7660
+ if ($.includes("_")) {
7661
+ const [J, ...ye] = $.split("_"), pe = ye.join("_");
7662
+ J === "shared" ? (ee._genderSpecific.shared[pe] || (ee._genderSpecific.shared[pe] = []), ee._genderSpecific.shared[pe].push(...ie)) : J === Z && (ee._genderSpecific[Z][pe] || (ee._genderSpecific[Z][pe] = []), ee._genderSpecific[Z][pe].push(...ie));
7655
7663
  } else
7656
- ce[Q] = ee;
7657
- }), w && (w._genderSpecific && Object.keys(w._genderSpecific).forEach((Q) => {
7658
- ce._genderSpecific[Q] || (ce._genderSpecific[Q] = {}), Object.entries(w._genderSpecific[Q]).forEach(([ee, te]) => {
7659
- ce._genderSpecific[Q][ee] || (ce._genderSpecific[Q][ee] = te);
7664
+ ee[$] = ie;
7665
+ }), z && (z._genderSpecific && Object.keys(z._genderSpecific).forEach(($) => {
7666
+ ee._genderSpecific[$] || (ee._genderSpecific[$] = {}), Object.entries(z._genderSpecific[$]).forEach(([ie, J]) => {
7667
+ ee._genderSpecific[$][ie] || (ee._genderSpecific[$][ie] = J);
7660
7668
  });
7661
- }), Object.entries(w).forEach(([Q, ee]) => {
7662
- Q !== "_genderSpecific" && !ce[Q] && (ce[Q] = ee);
7663
- })), O(ce);
7664
- } else if (typeof m.auto == "object") {
7665
- const H = await ct(m.auto, e), J = Object.values(H).some((X) => Array.isArray(X) && X.length > 0);
7666
- O(!J && w ? w : H);
7669
+ }), Object.entries(z).forEach(([$, ie]) => {
7670
+ $ !== "_genderSpecific" && !ee[$] && (ee[$] = ie);
7671
+ })), j(ee);
7672
+ } else if (typeof H.auto == "object") {
7673
+ const F = await lt(H.auto, e), se = Object.values(F).some((Z) => Array.isArray(Z) && Z.length > 0);
7674
+ j(!se && z ? z : F);
7667
7675
  }
7668
- } catch (w) {
7669
- if (console.error("Failed to auto-discover animations:", w), m.manifest)
7676
+ } catch (z) {
7677
+ if (console.error("Failed to auto-discover animations:", z), H.manifest)
7670
7678
  try {
7671
- const H = await je(m.manifest);
7672
- O(H);
7679
+ const F = await je(H.manifest);
7680
+ j(F);
7673
7681
  } catch {
7674
- O(m);
7682
+ j(H);
7675
7683
  }
7676
7684
  else
7677
- O(m);
7685
+ j(H);
7678
7686
  }
7679
7687
  else
7680
- O(m);
7688
+ j(H);
7681
7689
  })();
7682
- }, [m, e, _]), Ce(() => {
7683
- M.current = c;
7690
+ }, [H, e, ue]), ze(() => {
7691
+ I.current = c;
7684
7692
  }, [c]);
7685
- const K = Je(), ae = s || K.service;
7686
- let ge;
7687
- ae === "browser" ? ge = {
7693
+ const de = Je(), ne = s || de.service;
7694
+ let me;
7695
+ ne === "browser" ? me = {
7688
7696
  service: "browser",
7689
7697
  endpoint: "",
7690
7698
  apiKey: null,
7691
7699
  defaultVoice: "Google US English"
7692
- } : ae === "elevenlabs" ? ge = {
7700
+ } : ne === "elevenlabs" ? me = {
7693
7701
  service: "elevenlabs",
7694
7702
  endpoint: "https://api.elevenlabs.io/v1/text-to-speech",
7695
- apiKey: a || K.apiKey,
7696
- defaultVoice: i || K.defaultVoice || Ge.defaultVoice,
7697
- voices: K.voices || Ge.voices
7698
- } : ae === "deepgram" ? ge = {
7703
+ apiKey: a || de.apiKey,
7704
+ defaultVoice: i || de.defaultVoice || Ve.defaultVoice,
7705
+ voices: de.voices || Ve.voices
7706
+ } : ne === "deepgram" ? me = {
7699
7707
  service: "deepgram",
7700
7708
  endpoint: "https://api.deepgram.com/v1/speak",
7701
- apiKey: a || K.apiKey,
7702
- defaultVoice: i || K.defaultVoice || _e.defaultVoice,
7703
- voices: K.voices || _e.voices
7704
- } : ge = {
7705
- ...K,
7706
- apiKey: a !== null ? a : K.apiKey
7709
+ apiKey: a || de.apiKey,
7710
+ defaultVoice: i || de.defaultVoice || _e.defaultVoice,
7711
+ voices: de.voices || _e.voices
7712
+ } : me = {
7713
+ ...de,
7714
+ apiKey: a !== null ? a : de.apiKey
7707
7715
  };
7708
- const se = {
7716
+ const Xe = {
7709
7717
  url: n,
7710
7718
  body: e,
7711
7719
  avatarMood: t,
7712
- ttsLang: ae === "browser" ? "en-US" : o,
7713
- ttsVoice: i || ge.defaultVoice,
7720
+ ttsLang: ne === "browser" ? "en-US" : o,
7721
+ ttsVoice: i || me.defaultVoice,
7714
7722
  lipsyncLang: "en",
7715
7723
  showFullAvatar: c,
7716
7724
  bodyMovement: u,
7717
7725
  movementIntensity: l
7718
- }, ke = {
7719
- ttsEndpoint: ge.endpoint,
7720
- ttsApikey: ge.apiKey,
7721
- ttsService: ae,
7726
+ }, Oe = {
7727
+ ttsEndpoint: me.endpoint,
7728
+ ttsApikey: me.apiKey,
7729
+ ttsService: ne,
7722
7730
  lipsyncModules: ["en"],
7723
7731
  cameraView: r
7724
- }, Ue = U(async () => {
7725
- if (!(!L.current || y.current))
7732
+ }, Fe = W(async () => {
7733
+ if (!(!C.current || x.current))
7726
7734
  try {
7727
- pe(!0), be(null), y.current = new it(L.current, ke), await y.current.showAvatar(se, (w) => {
7728
- if (w.lengthComputable) {
7729
- const H = Math.min(100, Math.round(w.loaded / w.total * 100));
7730
- d(H);
7735
+ Te(!0), Me(null), x.current = new it(C.current, Oe), await x.current.showAvatar(Xe, (z) => {
7736
+ if (z.lengthComputable) {
7737
+ const F = Math.min(100, Math.round(z.loaded / z.total * 100));
7738
+ d(F);
7731
7739
  }
7732
- }), pe(!1), he(!0), h(y.current);
7733
- const S = () => {
7734
- document.visibilityState === "visible" ? y.current?.start() : y.current?.stop();
7740
+ }), Te(!1), R(!0), h(x.current);
7741
+ const k = () => {
7742
+ document.visibilityState === "visible" ? x.current?.start() : x.current?.stop();
7735
7743
  };
7736
- return document.addEventListener("visibilitychange", S), () => {
7737
- document.removeEventListener("visibilitychange", S);
7744
+ return document.addEventListener("visibilitychange", k), () => {
7745
+ document.removeEventListener("visibilitychange", k);
7738
7746
  };
7739
- } catch (S) {
7740
- console.error("Error initializing TalkingHead:", S), be(S.message || "Failed to initialize avatar"), pe(!1), g(S);
7747
+ } catch (k) {
7748
+ console.error("Error initializing TalkingHead:", k), Me(k.message || "Failed to initialize avatar"), Te(!1), p(k);
7741
7749
  }
7742
7750
  }, []);
7743
- Ce(() => (Ue(), () => {
7744
- y.current && (y.current.stop(), y.current.dispose(), y.current = null);
7745
- }), [Ue]);
7746
- const ze = U(async () => {
7747
- if (y.current)
7751
+ ze(() => (Fe(), () => {
7752
+ x.current && (x.current.stop(), x.current.dispose(), x.current = null);
7753
+ }), [Fe]);
7754
+ const Re = W(async () => {
7755
+ if (x.current)
7748
7756
  try {
7749
- const S = y.current.audioCtx || y.current.audioContext;
7750
- S && (S.state === "suspended" || S.state === "interrupted") && await S.resume();
7751
- } catch (S) {
7752
- console.warn("Failed to resume audio context:", S);
7757
+ const k = x.current.audioCtx || x.current.audioContext;
7758
+ k && (k.state === "suspended" || k.state === "interrupted") && await k.resume();
7759
+ } catch (k) {
7760
+ console.warn("Failed to resume audio context:", k);
7753
7761
  }
7754
- }, []), Ee = U((S) => {
7755
- if (!F)
7762
+ }, []), Ue = W((k) => {
7763
+ if (!B)
7756
7764
  return [];
7757
- let w = null;
7758
- if (F._genderSpecific) {
7759
- const H = _(e), J = F._genderSpecific[H];
7760
- if (J && J[S] && Array.isArray(J[S]) && J[S].length > 0 && (w = J[S]), !w && F._genderSpecific.shared && F._genderSpecific.shared[S]) {
7761
- const X = F._genderSpecific.shared[S];
7762
- Array.isArray(X) && X.length > 0 && (w = X);
7765
+ let z = null;
7766
+ if (B._genderSpecific) {
7767
+ const F = ue(e), se = B._genderSpecific[F];
7768
+ if (se && se[k] && Array.isArray(se[k]) && se[k].length > 0 && (z = se[k]), !z && B._genderSpecific.shared && B._genderSpecific.shared[k]) {
7769
+ const Z = B._genderSpecific.shared[k];
7770
+ Array.isArray(Z) && Z.length > 0 && (z = Z);
7763
7771
  }
7764
7772
  }
7765
- if (!w && F[S]) {
7766
- const H = F[S];
7767
- Array.isArray(H) && H.length > 0 ? w = H : typeof H == "string" && (w = [H]);
7768
- }
7769
- return !w || Array.isArray(w) && w.length === 0 ? ((Object.keys(F).length > 0 || F._genderSpecific && Object.keys(F._genderSpecific).length > 0) && console.warn(`⚠️ No animations found for group "${S}" (gender: ${_(e)}). Make sure animations are configured correctly.`), []) : Array.isArray(w) ? [...w] : typeof w == "string" ? [w] : [];
7770
- }, [F, e, _]), He = U((S) => {
7771
- const w = [...S];
7772
- for (let H = w.length - 1; H > 0; H--) {
7773
- const J = Math.floor(Math.random() * (H + 1));
7774
- [w[H], w[J]] = [w[J], w[H]];
7773
+ if (!z && B[k]) {
7774
+ const F = B[k];
7775
+ Array.isArray(F) && F.length > 0 ? z = F : typeof F == "string" && (z = [F]);
7775
7776
  }
7776
- return w;
7777
- }, []), We = U((S) => {
7778
- if (fe.current.length === 0) {
7779
- const H = Ee(S);
7780
- if (H.length === 0)
7777
+ return !z || Array.isArray(z) && z.length === 0 ? ((Object.keys(B).length > 0 || B._genderSpecific && Object.keys(B._genderSpecific).length > 0) && console.warn(`⚠️ No animations found for group "${k}" (gender: ${ue(e)}). Make sure animations are configured correctly.`), []) : Array.isArray(z) ? [...z] : typeof z == "string" ? [z] : [];
7778
+ }, [B, e, ue]), tt = W((k) => {
7779
+ if (Ae.current.length === 0) {
7780
+ const F = Ue(k);
7781
+ if (F.length === 0)
7781
7782
  return null;
7782
- fe.current = He(H), me.current = [];
7783
+ Ae.current = [...F], be.current = [];
7783
7784
  }
7784
- const w = fe.current.shift();
7785
- return w && me.current.push(w), w;
7786
- }, [Ee, He]), tt = U((S) => S ? S.split("/").pop().replace(".fbx", "").replace(/[-_]/g, " ") : "Unknown", []), Ie = U((S, w = !1, H = null) => {
7787
- if (!y.current || !Z.current || q.current !== S)
7785
+ const z = Ae.current.shift();
7786
+ return z && be.current.push(z), z;
7787
+ }, [Ue]), Ge = W((k) => k ? k.split("/").pop().replace(".fbx", "").replace(/[-_]/g, " ") : "Unknown", []), we = W((k, z = !1, F = null) => {
7788
+ if (!x.current || !X.current || q.current !== k)
7788
7789
  return null;
7789
- const J = We(S);
7790
- if (J)
7790
+ const se = tt(k);
7791
+ if (se)
7791
7792
  try {
7792
- const X = tt(J);
7793
- console.log(`🎬 Playing animation: "${X}"`);
7794
- const ie = () => {
7793
+ const Z = Ge(se);
7794
+ console.log(`🎬 Playing animation: "${Z}"`);
7795
+ const ke = () => {
7795
7796
  console.log("🎬 Animation finished, checking if should continue...");
7796
- const le = () => {
7797
- if (!y.current)
7797
+ const O = () => {
7798
+ if (!x.current)
7798
7799
  return console.log("🎬 No talkingHeadRef, stopping animations"), !1;
7799
- if (!Z.current)
7800
+ if (!X.current)
7800
7801
  return console.log("🎬 isSpeakingRef is false, stopping animations"), !1;
7801
- if (q.current !== S)
7802
- return console.log(`🎬 Animation group mismatch (${q.current} !== ${S}), stopping animations`), !1;
7803
- const Q = y.current, ee = Q.isAudioPlaying === !0, te = Q.audioPlaylist && Q.audioPlaylist.length > 0, xe = Q.speechQueue && Q.speechQueue.length > 0, de = Q.isSpeaking === !0, Ve = de || ee || te || xe;
7802
+ if (q.current !== k)
7803
+ return console.log(`🎬 Animation group mismatch (${q.current} !== ${k}), stopping animations`), !1;
7804
+ const $ = x.current, ie = $.isAudioPlaying === !0, J = $.audioPlaylist && $.audioPlaylist.length > 0, ye = $.speechQueue && $.speechQueue.length > 0, pe = $.isSpeaking === !0, De = pe || ie || J || ye;
7804
7805
  return console.log("🎬 Still speaking check:", {
7805
- isSpeakingRef: Z.current,
7806
- isTalkingHeadSpeaking: de,
7807
- hasAudioPlaying: ee,
7808
- hasAudioInPlaylist: te,
7809
- hasItemsInQueue: xe,
7810
- shouldContinue: Ve
7811
- }), Ve;
7806
+ isSpeakingRef: X.current,
7807
+ isTalkingHeadSpeaking: pe,
7808
+ hasAudioPlaying: ie,
7809
+ hasAudioInPlaylist: J,
7810
+ hasItemsInQueue: ye,
7811
+ shouldContinue: De
7812
+ }), De;
7812
7813
  };
7813
- le() ? (console.log("🎬 Still speaking, continuing to next animation..."), requestAnimationFrame(() => {
7814
- le() ? (console.log("🎬 Playing next animation..."), Ie(S, w, H)) : (console.log("🎬 Speech ended, stopping animations"), Z.current = !1, q.current = null, y.current && y.current.stopAnimation(), H && H());
7815
- })) : (console.log("🎬 Speech has ended, stopping animations"), Z.current = !1, q.current = null, y.current && y.current.stopAnimation(), H && H());
7814
+ O() ? (console.log("🎬 Still speaking, continuing to next animation..."), requestAnimationFrame(() => {
7815
+ O() ? (console.log("🎬 Playing next animation..."), we(k, z, F)) : (console.log("🎬 Speech ended, stopping animations"), X.current = !1, q.current = null, x.current && x.current.stopAnimation(), F && F());
7816
+ })) : (console.log("🎬 Speech has ended, stopping animations"), X.current = !1, q.current = null, x.current && x.current.stopAnimation(), F && F());
7816
7817
  };
7817
- return y.current.playAnimation(J, null, 0, 0, 0.01, w, ie), J;
7818
- } catch (X) {
7819
- return console.error("Failed to play animation:", X), null;
7818
+ return x.current.playAnimation(se, null, 0, 0, 0.01, z, ke), se;
7819
+ } catch (Z) {
7820
+ return console.error("Failed to play animation:", Z), null;
7820
7821
  }
7821
7822
  else {
7822
- const X = () => {
7823
- if (!y.current || !Z.current || q.current !== S)
7823
+ const Z = () => {
7824
+ if (!x.current || !X.current || q.current !== k)
7824
7825
  return !1;
7825
- const le = y.current, ce = le.isAudioPlaying === !0, Q = le.audioPlaylist && le.audioPlaylist.length > 0, ee = le.speechQueue && le.speechQueue.length > 0;
7826
- return le.isSpeaking === !0 || ce || Q || ee;
7826
+ const O = x.current, ee = O.isAudioPlaying === !0, $ = O.audioPlaylist && O.audioPlaylist.length > 0, ie = O.speechQueue && O.speechQueue.length > 0;
7827
+ return O.isSpeaking === !0 || ee || $ || ie;
7827
7828
  };
7828
- X() && (fe.current = [], me.current = [], X() ? Ie(S, w, H) : (Z.current = !1, q.current = null, y.current && y.current.stopAnimation()));
7829
+ Z() && (Ae.current = [], be.current = [], Z() ? we(k, z, F) : (X.current = !1, q.current = null, x.current && x.current.stopAnimation()));
7829
7830
  }
7830
7831
  return null;
7831
- }, [We, tt]), nt = U(() => {
7832
- T && y.current && !Z.current && (W.current && (clearInterval(W.current), W.current = null), fe.current = [], me.current = [], Ie(T, !1, null), W.current = setInterval(() => {
7833
- !Z.current && !P.current && T && (fe.current = [], me.current = [], Ie(T, !1, null));
7832
+ }, [tt, Ge]), ht = W(() => {
7833
+ y && x.current && !X.current && (oe.current && (clearInterval(oe.current), oe.current = null), Ae.current = [], be.current = [], we(y, !1, null), oe.current = setInterval(() => {
7834
+ !X.current && !D.current && y && (Ae.current = [], be.current = [], we(y, !1, null));
7834
7835
  }, 12e3 + Math.random() * 3e3));
7835
- }, [T, Ie]);
7836
- C.current = nt;
7837
- const Ze = U(async (S, w = {}) => {
7838
- if (!y.current || !ve || !S || S.trim() === "")
7836
+ }, [y, we]);
7837
+ _.current = ht;
7838
+ const Ze = W(async (k, z = {}) => {
7839
+ if (!x.current || !ae || !k || k.trim() === "")
7839
7840
  return;
7840
- await ze(), W.current && (clearInterval(W.current), W.current = null);
7841
- const H = w.animationGroup || D;
7842
- H && !w.skipAnimation && (Z.current = !0, q.current = H, fe.current = [], me.current = [], Ie(H));
7841
+ await Re(), oe.current && (clearInterval(oe.current), oe.current = null);
7842
+ const F = z.animationGroup || M;
7843
+ F && !z.skipAnimation && (X.current = !0, q.current = F, Ae.current = [], be.current = [], we(F));
7843
7844
  try {
7844
- b(S), w.onSpeechStart && w.onSpeechStart(S);
7845
- } catch (ie) {
7846
- console.warn("Error in onSpeechStart callback:", ie);
7845
+ b(k), z.onSpeechStart && z.onSpeechStart(k);
7846
+ } catch (O) {
7847
+ console.warn("Error in onSpeechStart callback:", O);
7847
7848
  }
7848
- Y.current = { remainingText: null, originalText: null, options: null }, $.current = [], I.current = { text: S, options: w }, E.current && (clearInterval(E.current), E.current = null), v(!1), P.current = !1, ne.current = !1;
7849
- const J = S.split(/[.!?]+/).filter((ie) => ie.trim().length > 0);
7850
- $.current = J, et.current = 0;
7851
- const X = {
7852
- lipsyncLang: w.lipsyncLang || "en"
7849
+ K.current = { remainingText: null, originalText: null, options: null }, fe.current = [], G.current = { text: k, options: z }, P.current && (clearInterval(P.current), P.current = null), N(!1), D.current = !1, Se.current = !1;
7850
+ const se = k.split(/[.!?]+/).filter((O) => O.trim().length > 0);
7851
+ fe.current = se, et.current = 0, te.current = "";
7852
+ const Z = (O) => {
7853
+ let ee = "";
7854
+ if (O && (O.text ? ee = O.text : O.vs && O.vs.subtitles ? Array.isArray(O.vs.subtitles) ? ee = O.vs.subtitles.join("") : typeof O.vs.subtitles == "string" && (ee = O.vs.subtitles) : O.subtitles && (Array.isArray(O.subtitles) ? ee = O.subtitles.join("") : typeof O.subtitles == "string" && (ee = O.subtitles))), ee.trim() && (te.current += ee, L && typeof L == "function"))
7855
+ try {
7856
+ L(te.current.trim());
7857
+ } catch ($) {
7858
+ console.warn("Error in onSubtitle callback:", $);
7859
+ }
7860
+ }, ke = {
7861
+ lipsyncLang: z.lipsyncLang || "en",
7862
+ onSubtitles: L ? Z : null
7853
7863
  };
7854
- if (y.current) {
7855
- const ie = y.current;
7856
- let le = 0;
7857
- const ce = 1200;
7858
- E.current && (clearInterval(E.current), E.current = null);
7859
- const Q = { current: !1 }, ee = setInterval(() => {
7860
- if (le++, P.current)
7864
+ if (x.current) {
7865
+ const O = x.current;
7866
+ let ee = 0;
7867
+ const $ = 1200;
7868
+ P.current && (clearInterval(P.current), P.current = null);
7869
+ const ie = { current: !1 }, J = setInterval(() => {
7870
+ if (ee++, D.current)
7861
7871
  return;
7862
- if (le > ce) {
7863
- if (ee && (clearInterval(ee), E.current = null), !Q.current && !P.current) {
7864
- Q.current = !0, Z.current = !1, q.current = null, fe.current = [], me.current = [], y.current && (y.current.stopAnimation(), y.current.isSpeaking = !1);
7872
+ if (ee > $) {
7873
+ if (J && (clearInterval(J), P.current = null), !ie.current && !D.current) {
7874
+ ie.current = !0, X.current = !1, q.current = null, Ae.current = [], be.current = [], x.current && (x.current.stopAnimation(), x.current.isSpeaking = !1);
7865
7875
  try {
7866
- w.onSpeechEnd && w.onSpeechEnd(), x();
7867
- } catch (Te) {
7868
- console.error("Error in onSpeechEnd callback (timeout):", Te);
7876
+ z.onSpeechEnd && z.onSpeechEnd(), f();
7877
+ } catch (ce) {
7878
+ console.error("Error in onSpeechEnd callback (timeout):", ce);
7869
7879
  }
7870
7880
  }
7871
7881
  return;
7872
7882
  }
7873
- const te = !ie.speechQueue || ie.speechQueue.length === 0, xe = !ie.audioPlaylist || ie.audioPlaylist.length === 0, de = ie.isAudioPlaying === !1;
7874
- ie && !P.current && te && xe && de && ie.isSpeaking === !1 && !Q.current && !P.current && setTimeout(() => {
7875
- const Te = y.current;
7876
- if (!Te)
7883
+ const ye = !O.speechQueue || O.speechQueue.length === 0, pe = !O.audioPlaylist || O.audioPlaylist.length === 0, De = O.isAudioPlaying === !1;
7884
+ O && !D.current && ye && pe && De && O.isSpeaking === !1 && !ie.current && !D.current && setTimeout(() => {
7885
+ const ce = x.current;
7886
+ if (!ce)
7877
7887
  return;
7878
- !P.current && (!Te.speechQueue || Te.speechQueue.length === 0) && (!Te.audioPlaylist || Te.audioPlaylist.length === 0) && Te.isAudioPlaying === !1 && Te.isSpeaking === !1 && !Q.current && !P.current && (Q.current = !0, ee && (clearInterval(ee), E.current = null), Z.current = !1, q.current = null, fe.current = [], me.current = [], Te && (Te.stopAnimation(), Te.isSpeaking = !1), ne.current = !0, setTimeout(() => {
7879
- C.current && C.current();
7888
+ !D.current && (!ce.speechQueue || ce.speechQueue.length === 0) && (!ce.audioPlaylist || ce.audioPlaylist.length === 0) && ce.isAudioPlaying === !1 && ce.isSpeaking === !1 && !ie.current && !D.current && (ie.current = !0, J && (clearInterval(J), P.current = null), X.current = !1, q.current = null, Ae.current = [], be.current = [], te.current = "", ce && (ce.stopAnimation(), ce.isSpeaking = !1), Se.current = !0, setTimeout(() => {
7889
+ _.current && _.current();
7880
7890
  }, 500), setTimeout(() => {
7881
7891
  try {
7882
- w.onSpeechEnd && w.onSpeechEnd(), x(), setTimeout(() => {
7883
- ne.current = !1;
7892
+ z.onSpeechEnd && z.onSpeechEnd(), f(), setTimeout(() => {
7893
+ Se.current = !1;
7884
7894
  }, 500);
7885
- } catch (ot) {
7886
- console.error("Error in onSpeechEnd callback:", ot), ne.current = !1;
7895
+ } catch (nt) {
7896
+ console.error("Error in onSpeechEnd callback:", nt), Se.current = !1;
7887
7897
  }
7888
7898
  }, 100));
7889
7899
  }, 200);
7890
7900
  }, 50);
7891
- E.current = ee;
7901
+ P.current = J;
7892
7902
  }
7893
7903
  try {
7894
- y.current.speakText(S, X);
7895
- } catch (ie) {
7896
- console.error("Error speaking text:", ie), be(ie.message || "Failed to speak text");
7904
+ x.current.speakText(k, ke);
7905
+ } catch (O) {
7906
+ console.error("Error speaking text:", O), Me(O.message || "Failed to speak text");
7897
7907
  }
7898
- }, [ve, x, ze, D, Ie]);
7899
- Ce(() => {
7900
- if (!ve || !T || !y.current)
7908
+ }, [ae, f, Re, M, we]);
7909
+ ze(() => {
7910
+ if (!ae || !y || !x.current)
7901
7911
  return;
7902
- W.current && (clearInterval(W.current), W.current = null);
7903
- const S = () => {
7904
- y.current && !P.current && !Z.current && (fe.current = [], me.current = [], Ie(T, !1, null));
7912
+ oe.current && (clearInterval(oe.current), oe.current = null);
7913
+ const k = () => {
7914
+ x.current && !D.current && !X.current && (Ae.current = [], be.current = [], we(y, !1, null));
7905
7915
  };
7906
- return Z.current || S(), W.current = setInterval(() => {
7907
- Z.current || S();
7916
+ return X.current || k(), oe.current = setInterval(() => {
7917
+ X.current || k();
7908
7918
  }, 12e3 + Math.random() * 3e3), () => {
7909
- W.current && (clearInterval(W.current), W.current = null);
7919
+ oe.current && (clearInterval(oe.current), oe.current = null);
7910
7920
  };
7911
- }, [ve, T, Ie, Z]), Ce(() => {
7912
- ve && j && f && y.current && !Z.current && !P.current && !ne.current && Ze(j);
7913
- }, [ve, j, f, Ze]);
7914
- const dt = U(() => {
7915
- if (y.current)
7921
+ }, [ae, y, we, X]), ze(() => {
7922
+ ae && Y && E && x.current && !X.current && !D.current && !Se.current && Ze(Y);
7923
+ }, [ae, Y, E, Ze]);
7924
+ const dt = W(() => {
7925
+ if (x.current)
7916
7926
  try {
7917
- const S = y.current.isSpeaking || !1, w = [...y.current.audioPlaylist || []], H = [...y.current.speechQueue || []];
7918
- if (S || w.length > 0 || H.length > 0) {
7919
- E.current && (clearInterval(E.current), E.current = null);
7920
- const J = $.current;
7921
- let X = [];
7922
- const ie = y.current.isAudioPlaying || !1;
7923
- if (J.length > 0) {
7924
- const te = w.length + H.length, xe = ie ? 1 : 0, de = J.length - te - xe, Ve = ie ? de : de + xe;
7925
- Ve < J.length && (X = J.slice(Ve));
7927
+ const k = x.current.isSpeaking || !1, z = [...x.current.audioPlaylist || []], F = [...x.current.speechQueue || []];
7928
+ if (k || z.length > 0 || F.length > 0) {
7929
+ P.current && (clearInterval(P.current), P.current = null);
7930
+ const se = fe.current;
7931
+ let Z = [];
7932
+ const ke = x.current.isAudioPlaying || !1;
7933
+ if (se.length > 0) {
7934
+ const J = z.length + F.length, ye = ke ? 1 : 0, pe = se.length - J - ye, De = ke ? pe : pe + ye;
7935
+ De < se.length && (Z = se.slice(De));
7926
7936
  } else
7927
- w.length > 0 && w.forEach((te) => {
7928
- if (te.text)
7929
- if (Array.isArray(te.text)) {
7930
- const xe = te.text.map((de) => de.word).join(" ");
7931
- xe.trim() && X.push(xe);
7932
- } else te.text.trim() && X.push(te.text);
7933
- }), H.length > 0 && H.forEach((te) => {
7934
- if (te.text)
7935
- if (Array.isArray(te.text)) {
7936
- const xe = te.text.map((de) => de.word).join(" ");
7937
- xe.trim() && X.push(xe);
7938
- } else te.text.trim() && X.push(te.text);
7937
+ z.length > 0 && z.forEach((J) => {
7938
+ if (J.text)
7939
+ if (Array.isArray(J.text)) {
7940
+ const ye = J.text.map((pe) => pe.word).join(" ");
7941
+ ye.trim() && Z.push(ye);
7942
+ } else J.text.trim() && Z.push(J.text);
7943
+ }), F.length > 0 && F.forEach((J) => {
7944
+ if (J.text)
7945
+ if (Array.isArray(J.text)) {
7946
+ const ye = J.text.map((pe) => pe.word).join(" ");
7947
+ ye.trim() && Z.push(ye);
7948
+ } else J.text.trim() && Z.push(J.text);
7939
7949
  });
7940
- const le = X.join(" ");
7941
- Y.current = {
7942
- remainingText: le || null,
7943
- originalText: I.current?.text || null,
7944
- options: I.current?.options || null,
7950
+ const O = Z.join(" ");
7951
+ K.current = {
7952
+ remainingText: O || null,
7953
+ originalText: G.current?.text || null,
7954
+ options: G.current?.options || null,
7945
7955
  // Track if we're pausing mid-sentence (has currently playing audio)
7946
- isMidSentence: w.length > 0
7956
+ isMidSentence: z.length > 0
7947
7957
  };
7948
- const Q = y.current.isAudioPlaying || !1 ? [...y.current.speechQueue || []] : null;
7949
- y.current.speechQueue.length = 0;
7950
- const ee = y.current.pauseSpeaking();
7951
- y.current.isSpeaking = !1, Z.current = !1, q.current = null, ee && ee.audio && Q ? (Se.current = ee, Se.current.savedSpeechQueue = Q) : Se.current = ee, v(!0), P.current = !0;
7958
+ const $ = x.current.isAudioPlaying || !1 ? [...x.current.speechQueue || []] : null;
7959
+ x.current.speechQueue.length = 0;
7960
+ const ie = x.current.pauseSpeaking();
7961
+ x.current.isSpeaking = !1, X.current = !1, q.current = null, ie && ie.audio && $ ? (w.current = ie, w.current.savedSpeechQueue = $) : w.current = ie, N(!0), D.current = !0;
7952
7962
  }
7953
- } catch (S) {
7954
- console.warn("Error pausing speech:", S);
7963
+ } catch (k) {
7964
+ console.warn("Error pausing speech:", k);
7955
7965
  }
7956
- }, []), mt = U(async () => {
7957
- if (!(!y.current || !R))
7966
+ }, []), mt = W(async () => {
7967
+ if (!(!x.current || !v))
7958
7968
  try {
7959
- if (await ze(), v(!1), P.current = !1, Se.current && Se.current.audio) {
7960
- Z.current = !0;
7961
- const X = Y.current?.options || I.current?.options || {}, ie = X.animationGroup || D;
7962
- if (ie && (q.current = ie), Se.current.savedSpeechQueue && (y.current.speechQueue.length = 0, y.current.speechQueue.push(...Se.current.savedSpeechQueue)), y.current.isSpeaking = !0, y.current) {
7963
- const le = y.current;
7964
- let ce = 0;
7965
- const Q = 1200, ee = { current: !1 };
7966
- E.current && (clearInterval(E.current), E.current = null);
7967
- const te = setInterval(() => {
7968
- if (ce++, P.current)
7969
+ if (await Re(), N(!1), D.current = !1, w.current && w.current.audio) {
7970
+ X.current = !0;
7971
+ const Z = K.current?.options || G.current?.options || {}, ke = Z.animationGroup || M;
7972
+ if (ke && (q.current = ke), w.current.savedSpeechQueue && (x.current.speechQueue.length = 0, x.current.speechQueue.push(...w.current.savedSpeechQueue)), x.current.isSpeaking = !0, x.current) {
7973
+ const O = x.current;
7974
+ let ee = 0;
7975
+ const $ = 1200, ie = { current: !1 };
7976
+ P.current && (clearInterval(P.current), P.current = null);
7977
+ const J = setInterval(() => {
7978
+ if (ee++, D.current)
7969
7979
  return;
7970
- if (ce > Q) {
7971
- if (te && (clearInterval(te), E.current = null), !ee.current && !P.current) {
7972
- ee.current = !0, Z.current = !1, q.current = null, fe.current = [], me.current = [], y.current && (y.current.stopAnimation(), y.current.isSpeaking = !1);
7980
+ if (ee > $) {
7981
+ if (J && (clearInterval(J), P.current = null), !ie.current && !D.current) {
7982
+ ie.current = !0, X.current = !1, q.current = null, Ae.current = [], be.current = [], x.current && (x.current.stopAnimation(), x.current.isSpeaking = !1);
7973
7983
  try {
7974
- X.onSpeechEnd && X.onSpeechEnd(), x();
7975
- } catch (Fe) {
7976
- console.error("Error in onSpeechEnd callback (timeout):", Fe);
7984
+ Z.onSpeechEnd && Z.onSpeechEnd(), f();
7985
+ } catch (ce) {
7986
+ console.error("Error in onSpeechEnd callback (timeout):", ce);
7977
7987
  }
7978
7988
  }
7979
7989
  return;
7980
7990
  }
7981
- const xe = !le.speechQueue || le.speechQueue.length === 0, de = !le.audioPlaylist || le.audioPlaylist.length === 0, Ve = le.isAudioPlaying === !1;
7982
- le && !P.current && xe && de && Ve && le.isSpeaking === !1 && !ee.current && !P.current && setTimeout(() => {
7983
- const Fe = y.current;
7984
- if (!Fe) return;
7985
- !P.current && (!Fe.speechQueue || Fe.speechQueue.length === 0) && (!Fe.audioPlaylist || Fe.audioPlaylist.length === 0) && Fe.isAudioPlaying === !1 && Fe.isSpeaking === !1 && !ee.current && !P.current && (ee.current = !0, te && (clearInterval(te), E.current = null), Z.current = !1, q.current = null, fe.current = [], me.current = [], Fe && (Fe.stopAnimation(), Fe.isSpeaking = !1), ne.current = !0, setTimeout(() => {
7986
- C.current && C.current();
7991
+ const ye = !O.speechQueue || O.speechQueue.length === 0, pe = !O.audioPlaylist || O.audioPlaylist.length === 0, De = O.isAudioPlaying === !1;
7992
+ O && !D.current && ye && pe && De && O.isSpeaking === !1 && !ie.current && !D.current && setTimeout(() => {
7993
+ const ce = x.current;
7994
+ if (!ce) return;
7995
+ !D.current && (!ce.speechQueue || ce.speechQueue.length === 0) && (!ce.audioPlaylist || ce.audioPlaylist.length === 0) && ce.isAudioPlaying === !1 && ce.isSpeaking === !1 && !ie.current && !D.current && (ie.current = !0, J && (clearInterval(J), P.current = null), X.current = !1, q.current = null, Ae.current = [], be.current = [], te.current = "", ce && (ce.stopAnimation(), ce.isSpeaking = !1), Se.current = !0, setTimeout(() => {
7996
+ _.current && _.current();
7987
7997
  }, 500), setTimeout(() => {
7988
7998
  try {
7989
- X.onSpeechEnd && X.onSpeechEnd(), x(), setTimeout(() => {
7990
- ne.current = !1;
7999
+ Z.onSpeechEnd && Z.onSpeechEnd(), f(), setTimeout(() => {
8000
+ Se.current = !1;
7991
8001
  }, 500);
7992
- } catch (gt) {
7993
- console.error("Error in onSpeechEnd callback:", gt), ne.current = !1;
8002
+ } catch (nt) {
8003
+ console.error("Error in onSpeechEnd callback:", nt), Se.current = !1;
7994
8004
  }
7995
8005
  }, 100));
7996
8006
  }, 200);
7997
8007
  }, 50);
7998
- E.current = te;
8008
+ P.current = J;
7999
8009
  }
8000
- await y.current.playAudio(!1, Se.current), ie && !X.skipAnimation && (fe.current = [], me.current = [], Ie(ie)), Se.current = null;
8010
+ await x.current.playAudio(!1, w.current), ke && !Z.skipAnimation && (Ae.current = [], be.current = [], we(ke)), w.current = null;
8001
8011
  return;
8002
8012
  }
8003
- const S = Y.current?.remainingText, w = Y.current?.originalText || I.current?.text, H = Y.current?.options || I.current?.options || {}, J = S || w;
8004
- J && Ze(J, H), Se.current = null;
8005
- } catch (S) {
8006
- console.warn("Error resuming speech:", S), v(!1), P.current = !1, Se.current = null;
8013
+ const k = K.current?.remainingText, z = K.current?.originalText || G.current?.text, F = K.current?.options || G.current?.options || {}, se = k || z;
8014
+ se && Ze(se, F), w.current = null;
8015
+ } catch (k) {
8016
+ console.warn("Error resuming speech:", k), N(!1), D.current = !1, w.current = null;
8007
8017
  }
8008
- }, [R, Ze, ze, D, Ie]), pt = U(() => {
8009
- if (y.current) {
8010
- y.current.stopAnimation(), y.current.stopSpeaking(), E.current && (clearInterval(E.current), E.current = null), Z.current = !1, q.current = null, fe.current = [], me.current = [], v(!1), P.current = !1, Se.current = null, ne.current = !1, setTimeout(() => {
8011
- C.current && C.current();
8018
+ }, [v, Ze, Re, M, we]), pt = W(() => {
8019
+ if (x.current) {
8020
+ x.current.stopAnimation(), x.current.stopSpeaking(), P.current && (clearInterval(P.current), P.current = null), X.current = !1, q.current = null, Ae.current = [], be.current = [], te.current = "", N(!1), D.current = !1, w.current = null, Se.current = !1, setTimeout(() => {
8021
+ _.current && _.current();
8012
8022
  }, 500);
8013
8023
  try {
8014
- x();
8015
- } catch (S) {
8016
- console.warn("Error in onSpeechEnd callback (stopSpeaking):", S);
8024
+ f();
8025
+ } catch (k) {
8026
+ console.warn("Error in onSpeechEnd callback (stopSpeaking):", k);
8017
8027
  }
8018
8028
  }
8019
- }, [x]);
8020
- return $e(B, () => ({
8029
+ }, [f]);
8030
+ return $e(S, () => ({
8021
8031
  speakText: Ze,
8022
8032
  pauseSpeaking: dt,
8023
8033
  resumeSpeaking: mt,
8024
8034
  stopSpeaking: pt,
8025
- resumeAudioContext: ze,
8026
- isPaused: () => R,
8027
- setMood: (S) => y.current?.setMood(S),
8028
- setBodyMovement: (S) => {
8029
- y.current && y.current.setBodyMovement(S);
8035
+ resumeAudioContext: Re,
8036
+ isPaused: () => v,
8037
+ getCurrentSubtitle: () => te.current,
8038
+ // Get current subtitle text
8039
+ setMood: (k) => x.current?.setMood(k),
8040
+ setBodyMovement: (k) => {
8041
+ x.current && x.current.setBodyMovement(k);
8030
8042
  },
8031
- playAnimation: (S, w = !1) => {
8032
- y.current && y.current.playAnimation && y.current.playAnimation(S, null, 10, 0, 0.01, w);
8043
+ playAnimation: (k, z = !1) => {
8044
+ x.current && x.current.playAnimation && x.current.playAnimation(k, null, 10, 0, 0.01, z);
8033
8045
  },
8034
- playRandomAnimation: (S, w = !1) => Ie(S, w),
8035
- getRandomAnimation: (S) => getRandomAnimation(S),
8036
- playReaction: (S) => y.current?.playReaction(S),
8037
- playCelebration: () => y.current?.playCelebration(),
8038
- setShowFullAvatar: (S) => {
8039
- y.current && (M.current = S, y.current.setShowFullAvatar(S));
8046
+ playRandomAnimation: (k, z = !1) => we(k, z),
8047
+ getRandomAnimation: (k) => getRandomAnimation(k),
8048
+ playReaction: (k) => x.current?.playReaction(k),
8049
+ playCelebration: () => x.current?.playCelebration(),
8050
+ setShowFullAvatar: (k) => {
8051
+ x.current && (I.current = k, x.current.setShowFullAvatar(k));
8040
8052
  },
8041
- isReady: ve,
8042
- talkingHead: y.current
8043
- })), /* @__PURE__ */ Be("div", { className: `simple-talking-avatar-container ${k}`, style: G, children: [
8044
- /* @__PURE__ */ re(
8053
+ isReady: ae,
8054
+ talkingHead: x.current
8055
+ })), /* @__PURE__ */ Ee("div", { className: `simple-talking-avatar-container ${Q}`, style: g, children: [
8056
+ /* @__PURE__ */ le(
8045
8057
  "div",
8046
8058
  {
8047
- ref: L,
8059
+ ref: C,
8048
8060
  className: "talking-head-viewer",
8049
8061
  style: {
8050
8062
  width: "100%",
@@ -8053,7 +8065,7 @@ const Wt = Ke(({
8053
8065
  }
8054
8066
  }
8055
8067
  ),
8056
- Le && /* @__PURE__ */ re("div", { className: "loading-overlay", style: {
8068
+ xe && /* @__PURE__ */ le("div", { className: "loading-overlay", style: {
8057
8069
  position: "absolute",
8058
8070
  top: "50%",
8059
8071
  left: "50%",
@@ -8062,7 +8074,7 @@ const Wt = Ke(({
8062
8074
  fontSize: "18px",
8063
8075
  zIndex: 10
8064
8076
  }, children: "Loading avatar..." }),
8065
- Pe && /* @__PURE__ */ re("div", { className: "error-overlay", style: {
8077
+ Le && /* @__PURE__ */ le("div", { className: "error-overlay", style: {
8066
8078
  position: "absolute",
8067
8079
  top: "50%",
8068
8080
  left: "50%",
@@ -8073,12 +8085,12 @@ const Wt = Ke(({
8073
8085
  zIndex: 10,
8074
8086
  padding: "20px",
8075
8087
  borderRadius: "8px"
8076
- }, children: Pe })
8088
+ }, children: Le })
8077
8089
  ] });
8078
8090
  });
8079
- Wt.displayName = "SimpleTalkingAvatar";
8080
- const Vt = Ke(({
8081
- curriculumData: j = null,
8091
+ Vt.displayName = "SimpleTalkingAvatar";
8092
+ const Xt = Ke(({
8093
+ curriculumData: Y = null,
8082
8094
  avatarConfig: n = {},
8083
8095
  animations: e = {},
8084
8096
  onLessonStart: t = () => {
@@ -8109,12 +8121,12 @@ const Vt = Ke(({
8109
8121
  onQuestionAnswer: s,
8110
8122
  onCurriculumComplete: i,
8111
8123
  onCustomAction: a
8112
- }), d = V(null), g = V(null), b = V(null), x = V(null), k = V(null), G = V(null), m = V(null), D = V(j?.curriculum || {
8124
+ }), d = V(null), p = V(null), b = V(null), f = V(null), L = V(null), Q = V(null), g = V(null), H = V(Y?.curriculum || {
8113
8125
  title: "Default Curriculum",
8114
8126
  description: "No curriculum data provided",
8115
8127
  language: "en",
8116
8128
  modules: []
8117
- }), T = V({
8129
+ }), M = V({
8118
8130
  avatarUrl: n.avatarUrl || "/avatars/brunette.glb",
8119
8131
  avatarBody: n.avatarBody || "F",
8120
8132
  mood: n.mood || "happy",
@@ -8128,7 +8140,7 @@ const Vt = Ke(({
8128
8140
  animations: e,
8129
8141
  lipsyncLang: "en"
8130
8142
  });
8131
- Ce(() => {
8143
+ ze(() => {
8132
8144
  h.current = {
8133
8145
  onLessonStart: t,
8134
8146
  onLessonComplete: o,
@@ -8136,13 +8148,13 @@ const Vt = Ke(({
8136
8148
  onCurriculumComplete: i,
8137
8149
  onCustomAction: a
8138
8150
  };
8139
- }, [t, o, s, i, a]), Ce(() => {
8140
- D.current = j?.curriculum || {
8151
+ }, [t, o, s, i, a]), ze(() => {
8152
+ H.current = Y?.curriculum || {
8141
8153
  title: "Default Curriculum",
8142
8154
  description: "No curriculum data provided",
8143
8155
  language: "en",
8144
8156
  modules: []
8145
- }, T.current = {
8157
+ }, M.current = {
8146
8158
  avatarUrl: n.avatarUrl || "/avatars/brunette.glb",
8147
8159
  avatarBody: n.avatarBody || "F",
8148
8160
  mood: n.mood || "happy",
@@ -8156,8 +8168,8 @@ const Vt = Ke(({
8156
8168
  animations: e,
8157
8169
  lipsyncLang: "en"
8158
8170
  };
8159
- }, [j, n, e]);
8160
- const f = U(() => (D.current || { modules: [] }).modules[r.current.currentModuleIndex]?.lessons[r.current.currentLessonIndex], []), B = U(() => f()?.questions[r.current.currentQuestionIndex], [f]), L = U((R, v) => v.type === "multiple_choice" || v.type === "true_false" ? R === v.answer : v.type === "code_test" && typeof R == "object" && R !== null ? R.passed === !0 : !1, []), y = U(() => {
8171
+ }, [Y, n, e]);
8172
+ const y = W(() => (H.current || { modules: [] }).modules[r.current.currentModuleIndex]?.lessons[r.current.currentLessonIndex], []), E = W(() => y()?.questions[r.current.currentQuestionIndex], [y]), S = W((R, v) => v.type === "multiple_choice" || v.type === "true_false" ? R === v.answer : v.type === "code_test" && typeof R == "object" && R !== null ? R.passed === !0 : !1, []), C = W(() => {
8161
8173
  r.current.lessonCompleted = !0, r.current.isQuestionMode = !1;
8162
8174
  const R = r.current.totalQuestions > 0 ? Math.round(r.current.score / r.current.totalQuestions * 100) : 100;
8163
8175
  let v = "Congratulations! You've completed this lesson";
@@ -8181,9 +8193,9 @@ const Vt = Ke(({
8181
8193
  } catch {
8182
8194
  c.current.playCelebration();
8183
8195
  }
8184
- const F = D.current || { modules: [] }, O = F.modules[r.current.currentModuleIndex], W = r.current.currentLessonIndex < (O?.lessons?.length || 0) - 1, Z = r.current.currentModuleIndex < (F.modules?.length || 0) - 1, q = W || Z, ne = T.current || { lipsyncLang: "en" };
8196
+ const N = H.current || { modules: [] }, B = N.modules[r.current.currentModuleIndex], j = r.current.currentLessonIndex < (B?.lessons?.length || 0) - 1, oe = r.current.currentModuleIndex < (N.modules?.length || 0) - 1, X = j || oe, q = M.current || { lipsyncLang: "en" };
8185
8197
  c.current.speakText(v, {
8186
- lipsyncLang: ne.lipsyncLang,
8198
+ lipsyncLang: q.lipsyncLang,
8187
8199
  onSpeechEnd: () => {
8188
8200
  h.current.onCustomAction({
8189
8201
  type: "lessonCompleteFeedbackDone",
@@ -8192,17 +8204,17 @@ const Vt = Ke(({
8192
8204
  score: r.current.score,
8193
8205
  totalQuestions: r.current.totalQuestions,
8194
8206
  percentage: R,
8195
- hasNextLesson: q
8207
+ hasNextLesson: X
8196
8208
  });
8197
8209
  }
8198
8210
  });
8199
8211
  }
8200
- }, [e.lessonComplete]), M = U(() => {
8212
+ }, [e.lessonComplete]), x = W(() => {
8201
8213
  r.current.curriculumCompleted = !0;
8202
- const R = D.current || { modules: [] };
8214
+ const R = H.current || { modules: [] };
8203
8215
  if (h.current.onCurriculumComplete({
8204
8216
  modules: R.modules.length,
8205
- totalLessons: R.modules.reduce((v, F) => v + F.lessons.length, 0)
8217
+ totalLessons: R.modules.reduce((v, N) => v + N.lessons.length, 0)
8206
8218
  }), c.current) {
8207
8219
  if (c.current.setMood("celebrating"), e.curriculumComplete)
8208
8220
  try {
@@ -8210,13 +8222,13 @@ const Vt = Ke(({
8210
8222
  } catch {
8211
8223
  c.current.playCelebration();
8212
8224
  }
8213
- const v = T.current || { lipsyncLang: "en" };
8225
+ const v = M.current || { lipsyncLang: "en" };
8214
8226
  c.current.speakText("Amazing! You've completed the entire curriculum! You're now ready to move on to more advanced topics. Well done!", { lipsyncLang: v.lipsyncLang });
8215
8227
  }
8216
- }, [e.curriculumComplete]), I = U(() => {
8217
- const R = f();
8228
+ }, [e.curriculumComplete]), I = W(() => {
8229
+ const R = y();
8218
8230
  r.current.isQuestionMode = !0, r.current.currentQuestionIndex = 0, r.current.totalQuestions = R?.questions?.length || 0, r.current.score = 0;
8219
- const v = B();
8231
+ const v = E();
8220
8232
  v && h.current.onCustomAction({
8221
8233
  type: "questionStart",
8222
8234
  moduleIndex: r.current.currentModuleIndex,
@@ -8226,35 +8238,35 @@ const Vt = Ke(({
8226
8238
  question: v,
8227
8239
  score: r.current.score
8228
8240
  });
8229
- const F = () => {
8241
+ const N = () => {
8230
8242
  if (!c.current || !v) return;
8231
8243
  if (c.current.setMood("happy"), e.questionStart)
8232
8244
  try {
8233
8245
  c.current.playAnimation(e.questionStart, !0);
8234
- } catch (W) {
8235
- console.warn("Failed to play questionStart animation:", W);
8246
+ } catch (j) {
8247
+ console.warn("Failed to play questionStart animation:", j);
8236
8248
  }
8237
- const O = T.current || { lipsyncLang: "en" };
8238
- v.type === "code_test" ? c.current.speakText(`Let's test your coding skills! Here's your first challenge: ${v.question}`, { lipsyncLang: O.lipsyncLang }) : v.type === "multiple_choice" ? c.current.speakText(`Now let me ask you some questions. Here's the first one: ${v.question}`, { lipsyncLang: O.lipsyncLang }) : v.type === "true_false" ? c.current.speakText(`Let's start with some true or false questions. First question: ${v.question}`, { lipsyncLang: O.lipsyncLang }) : c.current.speakText(`Now let me ask you some questions. Here's the first one: ${v.question}`, { lipsyncLang: O.lipsyncLang });
8249
+ const B = M.current || { lipsyncLang: "en" };
8250
+ v.type === "code_test" ? c.current.speakText(`Let's test your coding skills! Here's your first challenge: ${v.question}`, { lipsyncLang: B.lipsyncLang }) : v.type === "multiple_choice" ? c.current.speakText(`Now let me ask you some questions. Here's the first one: ${v.question}`, { lipsyncLang: B.lipsyncLang }) : v.type === "true_false" ? c.current.speakText(`Let's start with some true or false questions. First question: ${v.question}`, { lipsyncLang: B.lipsyncLang }) : c.current.speakText(`Now let me ask you some questions. Here's the first one: ${v.question}`, { lipsyncLang: B.lipsyncLang });
8239
8251
  };
8240
8252
  if (c.current && c.current.isReady && v)
8241
- F();
8253
+ N();
8242
8254
  else if (c.current && c.current.isReady) {
8243
- const O = T.current || { lipsyncLang: "en" };
8244
- c.current.speakText("Now let me ask you some questions to test your understanding.", { lipsyncLang: O.lipsyncLang });
8255
+ const B = M.current || { lipsyncLang: "en" };
8256
+ c.current.speakText("Now let me ask you some questions to test your understanding.", { lipsyncLang: B.lipsyncLang });
8245
8257
  } else {
8246
- const O = setInterval(() => {
8247
- c.current && c.current.isReady && (clearInterval(O), v && F());
8258
+ const B = setInterval(() => {
8259
+ c.current && c.current.isReady && (clearInterval(B), v && N());
8248
8260
  }, 100);
8249
8261
  setTimeout(() => {
8250
- clearInterval(O);
8262
+ clearInterval(B);
8251
8263
  }, 5e3);
8252
8264
  }
8253
- }, [e.questionStart, f, B]), E = U(() => {
8254
- const R = f();
8265
+ }, [e.questionStart, y, E]), G = W(() => {
8266
+ const R = y();
8255
8267
  if (r.current.currentQuestionIndex < (R?.questions?.length || 0) - 1) {
8256
8268
  c.current && c.current.stopSpeaking && c.current.stopSpeaking(), r.current.currentQuestionIndex += 1;
8257
- const v = B();
8269
+ const v = E();
8258
8270
  v && h.current.onCustomAction({
8259
8271
  type: "nextQuestion",
8260
8272
  moduleIndex: r.current.currentModuleIndex,
@@ -8264,45 +8276,45 @@ const Vt = Ke(({
8264
8276
  question: v,
8265
8277
  score: r.current.score
8266
8278
  });
8267
- const F = () => {
8279
+ const N = () => {
8268
8280
  if (!c.current || !v) return;
8269
8281
  if (c.current.setMood("happy"), c.current.setBodyMovement("idle"), e.nextQuestion)
8270
8282
  try {
8271
8283
  c.current.playAnimation(e.nextQuestion, !0);
8272
- } catch (ne) {
8273
- console.warn("Failed to play nextQuestion animation:", ne);
8284
+ } catch (q) {
8285
+ console.warn("Failed to play nextQuestion animation:", q);
8274
8286
  }
8275
- const O = T.current || { lipsyncLang: "en" }, Z = f()?.questions?.length || 0, q = r.current.currentQuestionIndex >= Z - 1;
8287
+ const B = M.current || { lipsyncLang: "en" }, oe = y()?.questions?.length || 0, X = r.current.currentQuestionIndex >= oe - 1;
8276
8288
  if (v.type === "code_test") {
8277
- const ne = q ? `Great! Here's your final coding challenge: ${v.question}` : `Great! Now let's move on to your next coding challenge: ${v.question}`;
8278
- c.current.speakText(ne, {
8279
- lipsyncLang: O.lipsyncLang
8289
+ const q = X ? `Great! Here's your final coding challenge: ${v.question}` : `Great! Now let's move on to your next coding challenge: ${v.question}`;
8290
+ c.current.speakText(q, {
8291
+ lipsyncLang: B.lipsyncLang
8280
8292
  });
8281
8293
  } else if (v.type === "multiple_choice") {
8282
- const ne = q ? `Alright! Here's your final question: ${v.question}` : `Alright! Here's your next question: ${v.question}`;
8283
- c.current.speakText(ne, {
8284
- lipsyncLang: O.lipsyncLang
8294
+ const q = X ? `Alright! Here's your final question: ${v.question}` : `Alright! Here's your next question: ${v.question}`;
8295
+ c.current.speakText(q, {
8296
+ lipsyncLang: B.lipsyncLang
8285
8297
  });
8286
8298
  } else if (v.type === "true_false") {
8287
- const ne = q ? `Now let's try this final one: ${v.question}` : `Now let's try this one: ${v.question}`;
8288
- c.current.speakText(ne, {
8289
- lipsyncLang: O.lipsyncLang
8299
+ const q = X ? `Now let's try this final one: ${v.question}` : `Now let's try this one: ${v.question}`;
8300
+ c.current.speakText(q, {
8301
+ lipsyncLang: B.lipsyncLang
8290
8302
  });
8291
8303
  } else {
8292
- const ne = q ? `Here's your final question: ${v.question}` : `Here's the next question: ${v.question}`;
8293
- c.current.speakText(ne, {
8294
- lipsyncLang: O.lipsyncLang
8304
+ const q = X ? `Here's your final question: ${v.question}` : `Here's the next question: ${v.question}`;
8305
+ c.current.speakText(q, {
8306
+ lipsyncLang: B.lipsyncLang
8295
8307
  });
8296
8308
  }
8297
8309
  };
8298
8310
  if (c.current && c.current.isReady && v)
8299
- F();
8311
+ N();
8300
8312
  else if (v) {
8301
- const O = setInterval(() => {
8302
- c.current && c.current.isReady && (clearInterval(O), F());
8313
+ const B = setInterval(() => {
8314
+ c.current && c.current.isReady && (clearInterval(B), N());
8303
8315
  }, 100);
8304
8316
  setTimeout(() => {
8305
- clearInterval(O);
8317
+ clearInterval(B);
8306
8318
  }, 5e3);
8307
8319
  }
8308
8320
  } else
@@ -8313,55 +8325,55 @@ const Vt = Ke(({
8313
8325
  totalQuestions: r.current.totalQuestions,
8314
8326
  score: r.current.score
8315
8327
  });
8316
- }, [e.nextQuestion, f, B]), P = U(() => {
8317
- const R = D.current || { modules: [] }, v = R.modules[r.current.currentModuleIndex];
8328
+ }, [e.nextQuestion, y, E]), P = W(() => {
8329
+ const R = H.current || { modules: [] }, v = R.modules[r.current.currentModuleIndex];
8318
8330
  if (r.current.currentLessonIndex < (v?.lessons?.length || 0) - 1) {
8319
8331
  r.current.currentLessonIndex += 1, r.current.currentQuestionIndex = 0, r.current.lessonCompleted = !1, r.current.isQuestionMode = !1, r.current.isTeaching = !1, r.current.score = 0, r.current.totalQuestions = 0;
8320
- const O = R.modules[r.current.currentModuleIndex], W = r.current.currentLessonIndex < (O?.lessons?.length || 0) - 1, Z = r.current.currentModuleIndex < (R.modules?.length || 0) - 1, q = W || Z;
8332
+ const B = R.modules[r.current.currentModuleIndex], j = r.current.currentLessonIndex < (B?.lessons?.length || 0) - 1, oe = r.current.currentModuleIndex < (R.modules?.length || 0) - 1, X = j || oe;
8321
8333
  h.current.onCustomAction({
8322
8334
  type: "lessonStart",
8323
8335
  moduleIndex: r.current.currentModuleIndex,
8324
8336
  lessonIndex: r.current.currentLessonIndex,
8325
- hasNextLesson: q
8337
+ hasNextLesson: X
8326
8338
  }), h.current.onLessonStart({
8327
8339
  moduleIndex: r.current.currentModuleIndex,
8328
8340
  lessonIndex: r.current.currentLessonIndex,
8329
- lesson: f()
8341
+ lesson: y()
8330
8342
  }), c.current && (c.current.setMood("happy"), c.current.setBodyMovement("idle"));
8331
8343
  } else if (r.current.currentModuleIndex < (R.modules?.length || 0) - 1) {
8332
8344
  r.current.currentModuleIndex += 1, r.current.currentLessonIndex = 0, r.current.currentQuestionIndex = 0, r.current.lessonCompleted = !1, r.current.isQuestionMode = !1, r.current.isTeaching = !1, r.current.score = 0, r.current.totalQuestions = 0;
8333
- const W = R.modules[r.current.currentModuleIndex], Z = r.current.currentLessonIndex < (W?.lessons?.length || 0) - 1, q = r.current.currentModuleIndex < (R.modules?.length || 0) - 1, ne = Z || q;
8345
+ const j = R.modules[r.current.currentModuleIndex], oe = r.current.currentLessonIndex < (j?.lessons?.length || 0) - 1, X = r.current.currentModuleIndex < (R.modules?.length || 0) - 1, q = oe || X;
8334
8346
  h.current.onCustomAction({
8335
8347
  type: "lessonStart",
8336
8348
  moduleIndex: r.current.currentModuleIndex,
8337
8349
  lessonIndex: r.current.currentLessonIndex,
8338
- hasNextLesson: ne
8350
+ hasNextLesson: q
8339
8351
  }), h.current.onLessonStart({
8340
8352
  moduleIndex: r.current.currentModuleIndex,
8341
8353
  lessonIndex: r.current.currentLessonIndex,
8342
- lesson: f()
8354
+ lesson: y()
8343
8355
  }), c.current && (c.current.setMood("happy"), c.current.setBodyMovement("idle"));
8344
8356
  } else
8345
- k.current && k.current();
8346
- }, []), Y = U(() => {
8347
- const R = f();
8357
+ L.current && L.current();
8358
+ }, []), D = W(() => {
8359
+ const R = y();
8348
8360
  let v = null;
8349
8361
  if (R?.avatar_script && R?.body) {
8350
- const F = R.avatar_script.trim(), O = R.body.trim(), W = F.match(/[.!?]$/) ? " " : ". ";
8351
- v = `${F}${W}${O}`;
8362
+ const N = R.avatar_script.trim(), B = R.body.trim(), j = N.match(/[.!?]$/) ? " " : ". ";
8363
+ v = `${N}${j}${B}`;
8352
8364
  } else
8353
8365
  v = R?.avatar_script || R?.body || null;
8354
8366
  if (c.current && c.current.isReady && v) {
8355
8367
  r.current.isTeaching = !0, r.current.isQuestionMode = !1, r.current.score = 0, r.current.totalQuestions = 0, c.current.setMood("happy");
8356
- let F = !1;
8368
+ let N = !1;
8357
8369
  if (e.teaching)
8358
8370
  try {
8359
- c.current.playAnimation(e.teaching, !0), F = !0;
8360
- } catch (W) {
8361
- console.warn("Failed to play teaching animation:", W);
8371
+ c.current.playAnimation(e.teaching, !0), N = !0;
8372
+ } catch (j) {
8373
+ console.warn("Failed to play teaching animation:", j);
8362
8374
  }
8363
- F || c.current.setBodyMovement("gesturing");
8364
- const O = T.current || { lipsyncLang: "en" };
8375
+ N || c.current.setBodyMovement("gesturing");
8376
+ const B = M.current || { lipsyncLang: "en" };
8365
8377
  h.current.onLessonStart({
8366
8378
  moduleIndex: r.current.currentModuleIndex,
8367
8379
  lessonIndex: r.current.currentLessonIndex,
@@ -8372,7 +8384,7 @@ const Vt = Ke(({
8372
8384
  lessonIndex: r.current.currentLessonIndex,
8373
8385
  lesson: R
8374
8386
  }), c.current.speakText(v, {
8375
- lipsyncLang: O.lipsyncLang,
8387
+ lipsyncLang: B.lipsyncLang,
8376
8388
  onSpeechEnd: () => {
8377
8389
  r.current.isTeaching = !1, h.current.onCustomAction({
8378
8390
  type: "teachingComplete",
@@ -8390,17 +8402,17 @@ const Vt = Ke(({
8390
8402
  }
8391
8403
  });
8392
8404
  }
8393
- }, [e.teaching, f]), $ = U((R) => {
8394
- const v = B(), F = L(R, v);
8395
- if (F && (r.current.score += 1), h.current.onQuestionAnswer({
8405
+ }, [e.teaching, y]), K = W((R) => {
8406
+ const v = E(), N = S(R, v);
8407
+ if (N && (r.current.score += 1), h.current.onQuestionAnswer({
8396
8408
  moduleIndex: r.current.currentModuleIndex,
8397
8409
  lessonIndex: r.current.currentLessonIndex,
8398
8410
  questionIndex: r.current.currentQuestionIndex,
8399
8411
  answer: R,
8400
- isCorrect: F,
8412
+ isCorrect: N,
8401
8413
  question: v
8402
8414
  }), c.current)
8403
- if (F) {
8415
+ if (N) {
8404
8416
  if (c.current.setMood("happy"), e.correct)
8405
8417
  try {
8406
8418
  c.current.playReaction("happy");
@@ -8408,13 +8420,13 @@ const Vt = Ke(({
8408
8420
  c.current.setBodyMovement("happy");
8409
8421
  }
8410
8422
  c.current.setBodyMovement("gesturing");
8411
- const W = f()?.questions?.length || 0;
8412
- r.current.currentQuestionIndex >= W - 1;
8413
- const Z = r.current.currentQuestionIndex < W - 1;
8414
- console.log("[CurriculumLearning] Answer feedback - questionIndex:", r.current.currentQuestionIndex, "totalQuestions:", W, "hasNextQuestion:", Z);
8415
- const q = v.type === "code_test" ? `Great job! Your code passed all the tests! ${v.explanation || ""}` : `Excellent! That's correct! ${v.explanation || ""}`, ne = T.current || { lipsyncLang: "en" };
8416
- c.current.speakText(q, {
8417
- lipsyncLang: ne.lipsyncLang,
8423
+ const j = y()?.questions?.length || 0;
8424
+ r.current.currentQuestionIndex >= j - 1;
8425
+ const oe = r.current.currentQuestionIndex < j - 1;
8426
+ console.log("[CurriculumLearning] Answer feedback - questionIndex:", r.current.currentQuestionIndex, "totalQuestions:", j, "hasNextQuestion:", oe);
8427
+ const X = v.type === "code_test" ? `Great job! Your code passed all the tests! ${v.explanation || ""}` : `Excellent! That's correct! ${v.explanation || ""}`, q = M.current || { lipsyncLang: "en" };
8428
+ c.current.speakText(X, {
8429
+ lipsyncLang: q.lipsyncLang,
8418
8430
  onSpeechEnd: () => {
8419
8431
  h.current.onCustomAction({
8420
8432
  type: "answerFeedbackComplete",
@@ -8422,7 +8434,7 @@ const Vt = Ke(({
8422
8434
  lessonIndex: r.current.currentLessonIndex,
8423
8435
  questionIndex: r.current.currentQuestionIndex,
8424
8436
  isCorrect: !0,
8425
- hasNextQuestion: Z,
8437
+ hasNextQuestion: oe,
8426
8438
  score: r.current.score,
8427
8439
  totalQuestions: r.current.totalQuestions
8428
8440
  });
@@ -8436,11 +8448,11 @@ const Vt = Ke(({
8436
8448
  c.current.setBodyMovement("idle");
8437
8449
  }
8438
8450
  c.current.setBodyMovement("gesturing");
8439
- const W = f()?.questions?.length || 0, Z = r.current.currentQuestionIndex >= W - 1, q = r.current.currentQuestionIndex < W - 1;
8440
- console.log("[CurriculumLearning] Answer feedback (incorrect) - questionIndex:", r.current.currentQuestionIndex, "totalQuestions:", W, "hasNextQuestion:", q);
8441
- const ne = v.type === "code_test" ? `Your code didn't pass all the tests. ${v.explanation || "Try again!"}` : `Not quite right, but don't worry! ${v.explanation || ""}${Z ? "" : " Let's move on to the next question."}`, me = T.current || { lipsyncLang: "en" };
8442
- c.current.speakText(ne, {
8443
- lipsyncLang: me.lipsyncLang,
8451
+ const j = y()?.questions?.length || 0, oe = r.current.currentQuestionIndex >= j - 1, X = r.current.currentQuestionIndex < j - 1;
8452
+ console.log("[CurriculumLearning] Answer feedback (incorrect) - questionIndex:", r.current.currentQuestionIndex, "totalQuestions:", j, "hasNextQuestion:", X);
8453
+ const q = v.type === "code_test" ? `Your code didn't pass all the tests. ${v.explanation || "Try again!"}` : `Not quite right, but don't worry! ${v.explanation || ""}${oe ? "" : " Let's move on to the next question."}`, Se = M.current || { lipsyncLang: "en" };
8454
+ c.current.speakText(q, {
8455
+ lipsyncLang: Se.lipsyncLang,
8444
8456
  onSpeechEnd: () => {
8445
8457
  h.current.onCustomAction({
8446
8458
  type: "answerFeedbackComplete",
@@ -8448,7 +8460,7 @@ const Vt = Ke(({
8448
8460
  lessonIndex: r.current.currentLessonIndex,
8449
8461
  questionIndex: r.current.currentQuestionIndex,
8450
8462
  isCorrect: !1,
8451
- hasNextQuestion: q,
8463
+ hasNextQuestion: X,
8452
8464
  score: r.current.score,
8453
8465
  totalQuestions: r.current.totalQuestions
8454
8466
  });
@@ -8456,21 +8468,21 @@ const Vt = Ke(({
8456
8468
  });
8457
8469
  }
8458
8470
  else {
8459
- const W = f()?.questions?.length || 0;
8471
+ const j = y()?.questions?.length || 0;
8460
8472
  h.current.onCustomAction({
8461
8473
  type: "answerFeedbackComplete",
8462
8474
  moduleIndex: r.current.currentModuleIndex,
8463
8475
  lessonIndex: r.current.currentLessonIndex,
8464
8476
  questionIndex: r.current.currentQuestionIndex,
8465
- isCorrect: F,
8466
- hasNextQuestion: r.current.currentQuestionIndex < W - 1,
8477
+ isCorrect: N,
8478
+ hasNextQuestion: r.current.currentQuestionIndex < j - 1,
8467
8479
  score: r.current.score,
8468
8480
  totalQuestions: r.current.totalQuestions,
8469
8481
  avatarNotReady: !0
8470
8482
  });
8471
8483
  }
8472
- }, [e.correct, e.incorrect, B, f, L]), Le = U((R) => {
8473
- const v = B();
8484
+ }, [e.correct, e.incorrect, E, y, S]), fe = W((R) => {
8485
+ const v = E();
8474
8486
  if (!R || typeof R != "object") {
8475
8487
  console.error("Invalid code test result format. Expected object with {passed: boolean, ...}");
8476
8488
  return;
@@ -8479,7 +8491,7 @@ const Vt = Ke(({
8479
8491
  console.warn("Current question is not a code test. Use handleAnswerSelect for other question types.");
8480
8492
  return;
8481
8493
  }
8482
- const F = {
8494
+ const N = {
8483
8495
  passed: R.passed === !0,
8484
8496
  results: R.results || [],
8485
8497
  output: R.output || "",
@@ -8494,13 +8506,13 @@ const Vt = Ke(({
8494
8506
  moduleIndex: r.current.currentModuleIndex,
8495
8507
  lessonIndex: r.current.currentLessonIndex,
8496
8508
  questionIndex: r.current.currentQuestionIndex,
8497
- testResult: F,
8509
+ testResult: N,
8498
8510
  question: v
8499
- }), m.current && m.current(F);
8500
- }, [B, L]), pe = U(() => {
8511
+ }), g.current && g.current(N);
8512
+ }, [E, S]), xe = W(() => {
8501
8513
  if (r.current.currentQuestionIndex > 0) {
8502
8514
  r.current.currentQuestionIndex -= 1;
8503
- const R = B();
8515
+ const R = E();
8504
8516
  R && h.current.onCustomAction({
8505
8517
  type: "questionStart",
8506
8518
  moduleIndex: r.current.currentModuleIndex,
@@ -8513,26 +8525,26 @@ const Vt = Ke(({
8513
8525
  const v = () => {
8514
8526
  if (!c.current || !R) return;
8515
8527
  c.current.setMood("happy"), c.current.setBodyMovement("idle");
8516
- const F = T.current || { lipsyncLang: "en" };
8528
+ const N = M.current || { lipsyncLang: "en" };
8517
8529
  R.type === "code_test" ? c.current.speakText(`Let's go back to this coding challenge: ${R.question}`, {
8518
- lipsyncLang: F.lipsyncLang
8530
+ lipsyncLang: N.lipsyncLang
8519
8531
  }) : c.current.speakText(`Going back to: ${R.question}`, {
8520
- lipsyncLang: F.lipsyncLang
8532
+ lipsyncLang: N.lipsyncLang
8521
8533
  });
8522
8534
  };
8523
8535
  if (c.current && c.current.isReady && R)
8524
8536
  v();
8525
8537
  else if (R) {
8526
- const F = setInterval(() => {
8527
- c.current && c.current.isReady && (clearInterval(F), v());
8538
+ const N = setInterval(() => {
8539
+ c.current && c.current.isReady && (clearInterval(N), v());
8528
8540
  }, 100);
8529
8541
  setTimeout(() => {
8530
- clearInterval(F);
8542
+ clearInterval(N);
8531
8543
  }, 5e3);
8532
8544
  }
8533
8545
  }
8534
- }, [B]), Pe = U(() => {
8535
- const R = D.current || { modules: [] };
8546
+ }, [E]), Te = W(() => {
8547
+ const R = H.current || { modules: [] };
8536
8548
  if (R.modules[r.current.currentModuleIndex], r.current.currentLessonIndex > 0)
8537
8549
  r.current.currentLessonIndex -= 1, r.current.currentQuestionIndex = 0, r.current.lessonCompleted = !1, r.current.isQuestionMode = !1, r.current.isTeaching = !1, r.current.score = 0, r.current.totalQuestions = 0, h.current.onCustomAction({
8538
8550
  type: "lessonStart",
@@ -8541,54 +8553,54 @@ const Vt = Ke(({
8541
8553
  }), h.current.onLessonStart({
8542
8554
  moduleIndex: r.current.currentModuleIndex,
8543
8555
  lessonIndex: r.current.currentLessonIndex,
8544
- lesson: f()
8556
+ lesson: y()
8545
8557
  }), c.current && (c.current.setMood("happy"), c.current.setBodyMovement("idle"));
8546
8558
  else if (r.current.currentModuleIndex > 0) {
8547
- const O = R.modules[r.current.currentModuleIndex - 1];
8548
- r.current.currentModuleIndex -= 1, r.current.currentLessonIndex = (O?.lessons?.length || 1) - 1, r.current.currentQuestionIndex = 0, r.current.lessonCompleted = !1, r.current.isQuestionMode = !1, r.current.isTeaching = !1, r.current.score = 0, r.current.totalQuestions = 0, h.current.onCustomAction({
8559
+ const B = R.modules[r.current.currentModuleIndex - 1];
8560
+ r.current.currentModuleIndex -= 1, r.current.currentLessonIndex = (B?.lessons?.length || 1) - 1, r.current.currentQuestionIndex = 0, r.current.lessonCompleted = !1, r.current.isQuestionMode = !1, r.current.isTeaching = !1, r.current.score = 0, r.current.totalQuestions = 0, h.current.onCustomAction({
8549
8561
  type: "lessonStart",
8550
8562
  moduleIndex: r.current.currentModuleIndex,
8551
8563
  lessonIndex: r.current.currentLessonIndex
8552
8564
  }), h.current.onLessonStart({
8553
8565
  moduleIndex: r.current.currentModuleIndex,
8554
8566
  lessonIndex: r.current.currentLessonIndex,
8555
- lesson: f()
8567
+ lesson: y()
8556
8568
  }), c.current && (c.current.setMood("happy"), c.current.setBodyMovement("idle"));
8557
8569
  }
8558
- }, [f]), be = U(() => {
8570
+ }, [y]), Le = W(() => {
8559
8571
  r.current.currentModuleIndex = 0, r.current.currentLessonIndex = 0, r.current.currentQuestionIndex = 0, r.current.isTeaching = !1, r.current.isQuestionMode = !1, r.current.lessonCompleted = !1, r.current.curriculumCompleted = !1, r.current.score = 0, r.current.totalQuestions = 0;
8560
- }, []), ve = U((R) => {
8572
+ }, []), Me = W((R) => {
8561
8573
  console.log("Avatar is ready!", R);
8562
- const v = f(), F = v?.avatar_script || v?.body;
8563
- u && F && setTimeout(() => {
8574
+ const v = y(), N = v?.avatar_script || v?.body;
8575
+ u && N && setTimeout(() => {
8564
8576
  d.current && d.current();
8565
8577
  }, 10);
8566
- }, [u, f]);
8567
- yt(() => {
8568
- d.current = Y, g.current = P, b.current = y, x.current = E, k.current = M, G.current = I, m.current = $;
8578
+ }, [u, y]);
8579
+ ft(() => {
8580
+ d.current = D, p.current = P, b.current = C, f.current = G, L.current = x, Q.current = I, g.current = K;
8569
8581
  }), $e(l, () => ({
8570
8582
  // Curriculum control methods
8571
- startTeaching: Y,
8583
+ startTeaching: D,
8572
8584
  startQuestions: I,
8573
- handleAnswerSelect: $,
8574
- handleCodeTestResult: Le,
8575
- nextQuestion: E,
8576
- previousQuestion: pe,
8585
+ handleAnswerSelect: K,
8586
+ handleCodeTestResult: fe,
8587
+ nextQuestion: G,
8588
+ previousQuestion: xe,
8577
8589
  nextLesson: P,
8578
- previousLesson: Pe,
8579
- completeLesson: y,
8580
- completeCurriculum: M,
8581
- resetCurriculum: be,
8590
+ previousLesson: Te,
8591
+ completeLesson: C,
8592
+ completeCurriculum: x,
8593
+ resetCurriculum: Le,
8582
8594
  getState: () => ({ ...r.current }),
8583
- getCurrentQuestion: () => B(),
8584
- getCurrentLesson: () => f(),
8595
+ getCurrentQuestion: () => E(),
8596
+ getCurrentLesson: () => y(),
8585
8597
  // Direct access to avatar ref (always returns current value)
8586
8598
  getAvatarRef: () => c.current,
8587
8599
  // Convenience methods that delegate to avatar (always check current ref)
8588
8600
  speakText: async (R, v = {}) => {
8589
8601
  await c.current?.resumeAudioContext?.();
8590
- const F = T.current || { lipsyncLang: "en" };
8591
- c.current?.speakText(R, { ...v, lipsyncLang: v.lipsyncLang || F.lipsyncLang });
8602
+ const N = M.current || { lipsyncLang: "en" };
8603
+ c.current?.speakText(R, { ...v, lipsyncLang: v.lipsyncLang || N.lipsyncLang });
8592
8604
  },
8593
8605
  resumeAudioContext: async () => {
8594
8606
  if (c.current?.resumeAudioContext)
@@ -8599,8 +8611,8 @@ const Vt = Ke(({
8599
8611
  if (v.state === "suspended" || v.state === "interrupted")
8600
8612
  try {
8601
8613
  await v.resume(), console.log("Audio context resumed via talkingHead");
8602
- } catch (F) {
8603
- console.warn("Failed to resume audio context:", F);
8614
+ } catch (N) {
8615
+ console.warn("Failed to resume audio context:", N);
8604
8616
  }
8605
8617
  } else
8606
8618
  console.warn("Audio context not available yet");
@@ -8632,8 +8644,8 @@ const Vt = Ke(({
8632
8644
  handleResize: () => c.current?.handleResize(),
8633
8645
  // Avatar readiness check (always returns current value)
8634
8646
  isAvatarReady: () => c.current?.isReady || !1
8635
- }), [Y, I, $, Le, E, P, y, M, be, B, f]);
8636
- const he = T.current || {
8647
+ }), [D, I, K, fe, G, P, C, x, Le, E, y]);
8648
+ const ae = M.current || {
8637
8649
  avatarUrl: "/avatars/brunette.glb",
8638
8650
  avatarBody: "F",
8639
8651
  mood: "happy",
@@ -8646,23 +8658,23 @@ const Vt = Ke(({
8646
8658
  showFullAvatar: !1,
8647
8659
  animations: e
8648
8660
  };
8649
- return /* @__PURE__ */ re("div", { style: { width: "100%", height: "100%" }, children: /* @__PURE__ */ re(
8650
- ut,
8661
+ return /* @__PURE__ */ le("div", { style: { width: "100%", height: "100%" }, children: /* @__PURE__ */ le(
8662
+ ct,
8651
8663
  {
8652
8664
  ref: c,
8653
- avatarUrl: he.avatarUrl,
8654
- avatarBody: he.avatarBody,
8655
- mood: he.mood,
8656
- ttsLang: he.ttsLang,
8657
- ttsService: he.ttsService,
8658
- ttsVoice: he.ttsVoice,
8659
- ttsApiKey: he.ttsApiKey,
8660
- bodyMovement: he.bodyMovement,
8661
- movementIntensity: he.movementIntensity,
8662
- showFullAvatar: he.showFullAvatar,
8665
+ avatarUrl: ae.avatarUrl,
8666
+ avatarBody: ae.avatarBody,
8667
+ mood: ae.mood,
8668
+ ttsLang: ae.ttsLang,
8669
+ ttsService: ae.ttsService,
8670
+ ttsVoice: ae.ttsVoice,
8671
+ ttsApiKey: ae.ttsApiKey,
8672
+ bodyMovement: ae.bodyMovement,
8673
+ movementIntensity: ae.movementIntensity,
8674
+ showFullAvatar: ae.showFullAvatar,
8663
8675
  cameraView: "upper",
8664
- animations: he.animations,
8665
- onReady: ve,
8676
+ animations: ae.animations,
8677
+ onReady: Me,
8666
8678
  onLoading: () => {
8667
8679
  },
8668
8680
  onError: (R) => {
@@ -8671,115 +8683,115 @@ const Vt = Ke(({
8671
8683
  }
8672
8684
  ) });
8673
8685
  });
8674
- Vt.displayName = "CurriculumLearning";
8675
- function $t({
8676
- manifestPath: j = "/animations/manifest.json",
8686
+ Xt.displayName = "CurriculumLearning";
8687
+ function Jt({
8688
+ manifestPath: Y = "/animations/manifest.json",
8677
8689
  avatarBody: n = "F",
8678
8690
  onAnimationPlay: e = null,
8679
8691
  onAnimationsSelected: t = null,
8680
8692
  onDeleteAnimations: o = null,
8681
8693
  style: s = {}
8682
8694
  }) {
8683
- const [i, a] = ye([]), [u, l] = ye(/* @__PURE__ */ new Set()), [c, r] = ye(!0), [h, d] = ye(null), [g, b] = ye("all"), [x, k] = ye("");
8684
- Ce(() => {
8695
+ const [i, a] = ge([]), [u, l] = ge(/* @__PURE__ */ new Set()), [c, r] = ge(!0), [h, d] = ge(null), [p, b] = ge("all"), [f, L] = ge("");
8696
+ ze(() => {
8685
8697
  (async () => {
8686
8698
  r(!0), d(null);
8687
8699
  try {
8688
- const M = await je(j), I = [];
8689
- if (M._genderSpecific) {
8700
+ const x = await je(Y), I = [];
8701
+ if (x._genderSpecific) {
8690
8702
  const P = (n?.toUpperCase() || "F") === "M" ? "male" : "female";
8691
- M._genderSpecific[P] && Object.entries(M._genderSpecific[P]).forEach(([Y, $]) => {
8692
- (Array.isArray($) ? $ : [$]).forEach((pe) => {
8703
+ x._genderSpecific[P] && Object.entries(x._genderSpecific[P]).forEach(([D, K]) => {
8704
+ (Array.isArray(K) ? K : [K]).forEach((xe) => {
8693
8705
  I.push({
8694
- path: pe,
8695
- group: Y,
8706
+ path: xe,
8707
+ group: D,
8696
8708
  gender: P,
8697
- name: pe.split("/").pop().replace(".fbx", "")
8709
+ name: xe.split("/").pop().replace(".fbx", "")
8698
8710
  });
8699
8711
  });
8700
- }), M._genderSpecific.shared && Object.entries(M._genderSpecific.shared).forEach(([Y, $]) => {
8701
- (Array.isArray($) ? $ : [$]).forEach((pe) => {
8712
+ }), x._genderSpecific.shared && Object.entries(x._genderSpecific.shared).forEach(([D, K]) => {
8713
+ (Array.isArray(K) ? K : [K]).forEach((xe) => {
8702
8714
  I.push({
8703
- path: pe,
8704
- group: Y,
8715
+ path: xe,
8716
+ group: D,
8705
8717
  gender: "shared",
8706
- name: pe.split("/").pop().replace(".fbx", "")
8718
+ name: xe.split("/").pop().replace(".fbx", "")
8707
8719
  });
8708
8720
  });
8709
8721
  });
8710
8722
  }
8711
- Object.entries(M).forEach(([E, P]) => {
8712
- E !== "_genderSpecific" && (Array.isArray(P) ? P : [P]).forEach(($) => {
8713
- typeof $ == "string" && I.push({
8714
- path: $,
8715
- group: E,
8723
+ Object.entries(x).forEach(([G, P]) => {
8724
+ G !== "_genderSpecific" && (Array.isArray(P) ? P : [P]).forEach((K) => {
8725
+ typeof K == "string" && I.push({
8726
+ path: K,
8727
+ group: G,
8716
8728
  gender: "root",
8717
- name: $.split("/").pop().replace(".fbx", "")
8729
+ name: K.split("/").pop().replace(".fbx", "")
8718
8730
  });
8719
8731
  });
8720
8732
  }), a(I), r(!1);
8721
- } catch (M) {
8722
- console.error("Failed to load animations:", M), d(M.message), r(!1);
8733
+ } catch (x) {
8734
+ console.error("Failed to load animations:", x), d(x.message), r(!1);
8723
8735
  }
8724
8736
  })();
8725
- }, [j, n]);
8726
- const G = ["all", ...new Set(i.map((y) => y.group))], m = i.filter((y) => {
8727
- const M = g === "all" || y.group === g, I = x === "" || y.name.toLowerCase().includes(x.toLowerCase()) || y.path.toLowerCase().includes(x.toLowerCase());
8728
- return M && I;
8729
- }), D = (y) => {
8730
- const M = new Set(u);
8731
- M.has(y) ? M.delete(y) : M.add(y), l(M), t && t(Array.from(M));
8732
- }, T = () => {
8733
- const y = new Set(u);
8734
- m.forEach((M) => {
8735
- y.add(M.path);
8736
- }), l(y), t && t(Array.from(y));
8737
- }, f = () => {
8738
- const y = new Set(u);
8739
- m.forEach((M) => {
8740
- y.delete(M.path);
8741
- }), l(y), t && t(Array.from(y));
8742
- }, B = () => {
8743
- const M = i.filter((I) => !u.has(I.path)).map((I) => I.path);
8744
- if (M.length === 0) {
8737
+ }, [Y, n]);
8738
+ const Q = ["all", ...new Set(i.map((C) => C.group))], g = i.filter((C) => {
8739
+ const x = p === "all" || C.group === p, I = f === "" || C.name.toLowerCase().includes(f.toLowerCase()) || C.path.toLowerCase().includes(f.toLowerCase());
8740
+ return x && I;
8741
+ }), H = (C) => {
8742
+ const x = new Set(u);
8743
+ x.has(C) ? x.delete(C) : x.add(C), l(x), t && t(Array.from(x));
8744
+ }, M = () => {
8745
+ const C = new Set(u);
8746
+ g.forEach((x) => {
8747
+ C.add(x.path);
8748
+ }), l(C), t && t(Array.from(C));
8749
+ }, y = () => {
8750
+ const C = new Set(u);
8751
+ g.forEach((x) => {
8752
+ C.delete(x.path);
8753
+ }), l(C), t && t(Array.from(C));
8754
+ }, E = () => {
8755
+ const x = i.filter((I) => !u.has(I.path)).map((I) => I.path);
8756
+ if (x.length === 0) {
8745
8757
  alert("No animations to delete. Select animations to keep, then delete will remove the unselected ones.");
8746
8758
  return;
8747
8759
  }
8748
- window.confirm(`Are you sure you want to delete ${M.length} animation(s)?
8760
+ window.confirm(`Are you sure you want to delete ${x.length} animation(s)?
8749
8761
 
8750
8762
  This will delete:
8751
- ${M.slice(0, 5).join(`
8752
- `)}${M.length > 5 ? `
8753
- ...` : ""}`) && (o && o(M), a(i.filter((I) => u.has(I.path))), l(/* @__PURE__ */ new Set()));
8754
- }, L = (y) => {
8755
- e && e(y);
8763
+ ${x.slice(0, 5).join(`
8764
+ `)}${x.length > 5 ? `
8765
+ ...` : ""}`) && (o && o(x), a(i.filter((I) => u.has(I.path))), l(/* @__PURE__ */ new Set()));
8766
+ }, S = (C) => {
8767
+ e && e(C);
8756
8768
  };
8757
- return c ? /* @__PURE__ */ re("div", { style: { padding: "20px", textAlign: "center", ...s }, children: /* @__PURE__ */ re("p", { children: "Loading animations..." }) }) : h ? /* @__PURE__ */ re("div", { style: { padding: "20px", color: "red", ...s }, children: /* @__PURE__ */ Be("p", { children: [
8769
+ return c ? /* @__PURE__ */ le("div", { style: { padding: "20px", textAlign: "center", ...s }, children: /* @__PURE__ */ le("p", { children: "Loading animations..." }) }) : h ? /* @__PURE__ */ le("div", { style: { padding: "20px", color: "red", ...s }, children: /* @__PURE__ */ Ee("p", { children: [
8758
8770
  "Error loading animations: ",
8759
8771
  h
8760
- ] }) }) : /* @__PURE__ */ Be("div", { style: {
8772
+ ] }) }) : /* @__PURE__ */ Ee("div", { style: {
8761
8773
  padding: "20px",
8762
8774
  backgroundColor: "#2a2a2a",
8763
8775
  borderRadius: "8px",
8764
8776
  color: "#fff",
8765
8777
  ...s
8766
8778
  }, children: [
8767
- /* @__PURE__ */ re("h2", { style: { marginTop: 0 }, children: "Animation Selector" }),
8768
- /* @__PURE__ */ re("p", { style: { color: "#aaa", fontSize: "14px" }, children: "Click buttons to play animations. Select animations to keep, then delete will remove the rest." }),
8769
- /* @__PURE__ */ Be("div", { style: {
8779
+ /* @__PURE__ */ le("h2", { style: { marginTop: 0 }, children: "Animation Selector" }),
8780
+ /* @__PURE__ */ le("p", { style: { color: "#aaa", fontSize: "14px" }, children: "Click buttons to play animations. Select animations to keep, then delete will remove the rest." }),
8781
+ /* @__PURE__ */ Ee("div", { style: {
8770
8782
  display: "flex",
8771
8783
  gap: "10px",
8772
8784
  marginBottom: "20px",
8773
8785
  flexWrap: "wrap",
8774
8786
  alignItems: "center"
8775
8787
  }, children: [
8776
- /* @__PURE__ */ re(
8788
+ /* @__PURE__ */ le(
8777
8789
  "input",
8778
8790
  {
8779
8791
  type: "text",
8780
8792
  placeholder: "Search animations...",
8781
- value: x,
8782
- onChange: (y) => k(y.target.value),
8793
+ value: f,
8794
+ onChange: (C) => L(C.target.value),
8783
8795
  style: {
8784
8796
  padding: "8px 12px",
8785
8797
  borderRadius: "6px",
@@ -8792,11 +8804,11 @@ ${M.slice(0, 5).join(`
8792
8804
  }
8793
8805
  }
8794
8806
  ),
8795
- /* @__PURE__ */ re(
8807
+ /* @__PURE__ */ le(
8796
8808
  "select",
8797
8809
  {
8798
- value: g,
8799
- onChange: (y) => b(y.target.value),
8810
+ value: p,
8811
+ onChange: (C) => b(C.target.value),
8800
8812
  style: {
8801
8813
  padding: "8px 12px",
8802
8814
  borderRadius: "6px",
@@ -8805,13 +8817,13 @@ ${M.slice(0, 5).join(`
8805
8817
  color: "#fff",
8806
8818
  fontSize: "14px"
8807
8819
  },
8808
- children: G.map((y) => /* @__PURE__ */ re("option", { value: y, children: y === "all" ? "All Groups" : y }, y))
8820
+ children: Q.map((C) => /* @__PURE__ */ le("option", { value: C, children: C === "all" ? "All Groups" : C }, C))
8809
8821
  }
8810
8822
  ),
8811
- /* @__PURE__ */ re(
8823
+ /* @__PURE__ */ le(
8812
8824
  "button",
8813
8825
  {
8814
- onClick: T,
8826
+ onClick: M,
8815
8827
  style: {
8816
8828
  padding: "8px 16px",
8817
8829
  backgroundColor: "#4CAF50",
@@ -8824,10 +8836,10 @@ ${M.slice(0, 5).join(`
8824
8836
  children: "Select All Visible"
8825
8837
  }
8826
8838
  ),
8827
- /* @__PURE__ */ re(
8839
+ /* @__PURE__ */ le(
8828
8840
  "button",
8829
8841
  {
8830
- onClick: f,
8842
+ onClick: y,
8831
8843
  style: {
8832
8844
  padding: "8px 16px",
8833
8845
  backgroundColor: "#666",
@@ -8840,10 +8852,10 @@ ${M.slice(0, 5).join(`
8840
8852
  children: "Deselect All Visible"
8841
8853
  }
8842
8854
  ),
8843
- /* @__PURE__ */ Be(
8855
+ /* @__PURE__ */ Ee(
8844
8856
  "button",
8845
8857
  {
8846
- onClick: B,
8858
+ onClick: E,
8847
8859
  style: {
8848
8860
  padding: "8px 16px",
8849
8861
  backgroundColor: "#f44336",
@@ -8862,7 +8874,7 @@ ${M.slice(0, 5).join(`
8862
8874
  }
8863
8875
  )
8864
8876
  ] }),
8865
- /* @__PURE__ */ Be("div", { style: {
8877
+ /* @__PURE__ */ Ee("div", { style: {
8866
8878
  marginBottom: "15px",
8867
8879
  fontSize: "14px",
8868
8880
  color: "#aaa"
@@ -8872,9 +8884,9 @@ ${M.slice(0, 5).join(`
8872
8884
  " | Selected: ",
8873
8885
  u.size,
8874
8886
  " | Showing: ",
8875
- m.length
8887
+ g.length
8876
8888
  ] }),
8877
- /* @__PURE__ */ re("div", { style: {
8889
+ /* @__PURE__ */ le("div", { style: {
8878
8890
  display: "grid",
8879
8891
  gridTemplateColumns: "repeat(auto-fill, minmax(200px, 1fr))",
8880
8892
  gap: "10px",
@@ -8883,9 +8895,9 @@ ${M.slice(0, 5).join(`
8883
8895
  padding: "10px",
8884
8896
  backgroundColor: "#1a1a1a",
8885
8897
  borderRadius: "6px"
8886
- }, children: m.map((y, M) => {
8887
- const I = u.has(y.path);
8888
- return /* @__PURE__ */ Be(
8898
+ }, children: g.map((C, x) => {
8899
+ const I = u.has(C.path);
8900
+ return /* @__PURE__ */ Ee(
8889
8901
  "div",
8890
8902
  {
8891
8903
  style: {
@@ -8896,39 +8908,39 @@ ${M.slice(0, 5).join(`
8896
8908
  cursor: "pointer",
8897
8909
  transition: "all 0.2s"
8898
8910
  },
8899
- onClick: () => D(y.path),
8911
+ onClick: () => H(C.path),
8900
8912
  children: [
8901
- /* @__PURE__ */ Be("div", { style: { marginBottom: "8px" }, children: [
8902
- /* @__PURE__ */ re(
8913
+ /* @__PURE__ */ Ee("div", { style: { marginBottom: "8px" }, children: [
8914
+ /* @__PURE__ */ le(
8903
8915
  "input",
8904
8916
  {
8905
8917
  type: "checkbox",
8906
8918
  checked: I,
8907
- onChange: () => D(y.path),
8908
- onClick: (E) => E.stopPropagation(),
8919
+ onChange: () => H(C.path),
8920
+ onClick: (G) => G.stopPropagation(),
8909
8921
  style: {
8910
8922
  marginRight: "8px",
8911
8923
  cursor: "pointer"
8912
8924
  }
8913
8925
  }
8914
8926
  ),
8915
- /* @__PURE__ */ Be("span", { style: { fontSize: "12px", color: "#aaa" }, children: [
8916
- y.group,
8927
+ /* @__PURE__ */ Ee("span", { style: { fontSize: "12px", color: "#aaa" }, children: [
8928
+ C.group,
8917
8929
  " ",
8918
- y.gender !== "root" && `(${y.gender})`
8930
+ C.gender !== "root" && `(${C.gender})`
8919
8931
  ] })
8920
8932
  ] }),
8921
- /* @__PURE__ */ re("div", { style: {
8933
+ /* @__PURE__ */ le("div", { style: {
8922
8934
  fontSize: "13px",
8923
8935
  fontWeight: "bold",
8924
8936
  marginBottom: "8px",
8925
8937
  wordBreak: "break-word"
8926
- }, children: y.name }),
8927
- /* @__PURE__ */ re(
8938
+ }, children: C.name }),
8939
+ /* @__PURE__ */ le(
8928
8940
  "button",
8929
8941
  {
8930
- onClick: (E) => {
8931
- E.stopPropagation(), L(y.path);
8942
+ onClick: (G) => {
8943
+ G.stopPropagation(), S(C.path);
8932
8944
  },
8933
8945
  style: {
8934
8946
  width: "100%",
@@ -8945,17 +8957,17 @@ ${M.slice(0, 5).join(`
8945
8957
  )
8946
8958
  ]
8947
8959
  },
8948
- `${y.path}-${M}`
8960
+ `${C.path}-${x}`
8949
8961
  );
8950
8962
  }) }),
8951
- m.length === 0 && /* @__PURE__ */ re("div", { style: {
8963
+ g.length === 0 && /* @__PURE__ */ le("div", { style: {
8952
8964
  padding: "40px",
8953
8965
  textAlign: "center",
8954
8966
  color: "#aaa"
8955
8967
  }, children: "No animations found matching your filters." })
8956
8968
  ] });
8957
8969
  }
8958
- const ht = {
8970
+ const ut = {
8959
8971
  // Code-based dance animations (no FBX required)
8960
8972
  dance: {
8961
8973
  name: "dance",
@@ -9058,16 +9070,16 @@ const ht = {
9058
9070
  duration: 5e3,
9059
9071
  description: "Excited, energetic movement"
9060
9072
  }
9061
- }, Jt = (j) => ht[j] || null, en = (j) => ht.hasOwnProperty(j);
9073
+ }, en = (Y) => ut[Y] || null, tn = (Y) => ut.hasOwnProperty(Y);
9062
9074
  export {
9063
- $t as AnimationSelector,
9064
- Vt as CurriculumLearning,
9065
- Wt as SimpleTalkingAvatar,
9066
- ut as TalkingHeadAvatar,
9067
- Nt as TalkingHeadComponent,
9068
- ht as animations,
9075
+ Jt as AnimationSelector,
9076
+ Xt as CurriculumLearning,
9077
+ Vt as SimpleTalkingAvatar,
9078
+ ct as TalkingHeadAvatar,
9079
+ Ut as TalkingHeadComponent,
9080
+ ut as animations,
9069
9081
  Je as getActiveTTSConfig,
9070
- Jt as getAnimation,
9071
- Kt as getVoiceOptions,
9072
- en as hasAnimation
9082
+ en as getAnimation,
9083
+ $t as getVoiceOptions,
9084
+ tn as hasAnimation
9073
9085
  };