@swan-admin/swan-web-component 1.0.115 → 1.0.116

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.
Files changed (322) hide show
  1. package/dist/BodyScan-266TM1IY.js +2 -0
  2. package/dist/BodyScan-266TM1IY.js.map +1 -0
  3. package/dist/BodyScan-6DkuHYY1.js +2 -0
  4. package/dist/BodyScan-6DkuHYY1.js.map +1 -0
  5. package/dist/BodyScan-9r5xhZtT.js +2 -0
  6. package/dist/BodyScan-9r5xhZtT.js.map +1 -0
  7. package/dist/BodyScan-B5rbj0lx.js +2 -0
  8. package/dist/BodyScan-B5rbj0lx.js.map +1 -0
  9. package/dist/BodyScan-BDIHkyr1.js +2 -0
  10. package/dist/BodyScan-BDIHkyr1.js.map +1 -0
  11. package/dist/BodyScan-BHAi6fgF.js +2 -0
  12. package/dist/BodyScan-BHAi6fgF.js.map +1 -0
  13. package/dist/BodyScan-BRHehCvZ.js +2 -0
  14. package/dist/BodyScan-BRHehCvZ.js.map +1 -0
  15. package/dist/BodyScan-BS1ERcCz.js +2 -0
  16. package/dist/BodyScan-BS1ERcCz.js.map +1 -0
  17. package/dist/BodyScan-BSyeQRam.js +2 -0
  18. package/dist/BodyScan-BSyeQRam.js.map +1 -0
  19. package/dist/BodyScan-BYiULEJt.js +2 -0
  20. package/dist/BodyScan-BYiULEJt.js.map +1 -0
  21. package/dist/BodyScan-Ba0_SdEv.js +2 -0
  22. package/dist/BodyScan-Ba0_SdEv.js.map +1 -0
  23. package/dist/BodyScan-BdDkcHs5.js +2 -0
  24. package/dist/BodyScan-BdDkcHs5.js.map +1 -0
  25. package/dist/BodyScan-BeRh1VEM.js +2 -0
  26. package/dist/BodyScan-BeRh1VEM.js.map +1 -0
  27. package/dist/BodyScan-C4uag-I5.js +2 -0
  28. package/dist/BodyScan-C4uag-I5.js.map +1 -0
  29. package/dist/BodyScan-C7ccmu4z.js +2 -0
  30. package/dist/BodyScan-C7ccmu4z.js.map +1 -0
  31. package/dist/BodyScan-C84F8j4s.js +2 -0
  32. package/dist/BodyScan-C84F8j4s.js.map +1 -0
  33. package/dist/BodyScan-CCQYp6d5.js +2 -0
  34. package/dist/BodyScan-CCQYp6d5.js.map +1 -0
  35. package/dist/BodyScan-CKWEpJua.js +2 -0
  36. package/dist/BodyScan-CKWEpJua.js.map +1 -0
  37. package/dist/BodyScan-CN98BKFZ.js +2 -0
  38. package/dist/BodyScan-CN98BKFZ.js.map +1 -0
  39. package/dist/BodyScan-CPP9J8Ct.js +2 -0
  40. package/dist/BodyScan-CPP9J8Ct.js.map +1 -0
  41. package/dist/BodyScan-CR5-H0UA.js +2 -0
  42. package/dist/BodyScan-CR5-H0UA.js.map +1 -0
  43. package/dist/BodyScan-CSP5yePt.js +2 -0
  44. package/dist/BodyScan-CSP5yePt.js.map +1 -0
  45. package/dist/BodyScan-C_ZAWWXZ.js +2 -0
  46. package/dist/BodyScan-C_ZAWWXZ.js.map +1 -0
  47. package/dist/BodyScan-CadnO5G7.js +2 -0
  48. package/dist/BodyScan-CadnO5G7.js.map +1 -0
  49. package/dist/BodyScan-CdGuDf3C.js +2 -0
  50. package/dist/BodyScan-CdGuDf3C.js.map +1 -0
  51. package/dist/BodyScan-CdL7FlA2.js +2 -0
  52. package/dist/BodyScan-CdL7FlA2.js.map +1 -0
  53. package/dist/BodyScan-Cw9EbZpT.js +2 -0
  54. package/dist/BodyScan-Cw9EbZpT.js.map +1 -0
  55. package/dist/BodyScan-CwRmpuqZ.js +2 -0
  56. package/dist/BodyScan-CwRmpuqZ.js.map +1 -0
  57. package/dist/BodyScan-D3adkbYi.js +2 -0
  58. package/dist/BodyScan-D3adkbYi.js.map +1 -0
  59. package/dist/BodyScan-D3sbCXE3.js +2 -0
  60. package/dist/BodyScan-D3sbCXE3.js.map +1 -0
  61. package/dist/BodyScan-DCprskz_.js +2 -0
  62. package/dist/BodyScan-DCprskz_.js.map +1 -0
  63. package/dist/BodyScan-DFXvZoV8.js +2 -0
  64. package/dist/BodyScan-DFXvZoV8.js.map +1 -0
  65. package/dist/BodyScan-DMVMpieA.js +2 -0
  66. package/dist/BodyScan-DMVMpieA.js.map +1 -0
  67. package/dist/BodyScan-DR6nx-J3.js +2 -0
  68. package/dist/BodyScan-DR6nx-J3.js.map +1 -0
  69. package/dist/BodyScan-DTpf6JPk.js +2 -0
  70. package/dist/BodyScan-DTpf6JPk.js.map +1 -0
  71. package/dist/BodyScan-DV7xfyrh.js +2 -0
  72. package/dist/BodyScan-DV7xfyrh.js.map +1 -0
  73. package/dist/BodyScan-DZ0v1aWY.js +2 -0
  74. package/dist/BodyScan-DZ0v1aWY.js.map +1 -0
  75. package/dist/BodyScan-DePDK-lV.js +2 -0
  76. package/dist/BodyScan-DePDK-lV.js.map +1 -0
  77. package/dist/BodyScan-Dh5XpuVl.js +2 -0
  78. package/dist/BodyScan-Dh5XpuVl.js.map +1 -0
  79. package/dist/BodyScan-DmleTPju.js +2 -0
  80. package/dist/BodyScan-DmleTPju.js.map +1 -0
  81. package/dist/BodyScan-DmsLdodt.js +2 -0
  82. package/dist/BodyScan-DmsLdodt.js.map +1 -0
  83. package/dist/BodyScan-Do7DMvoE.js +2 -0
  84. package/dist/BodyScan-Do7DMvoE.js.map +1 -0
  85. package/dist/BodyScan-DpSLSJkg.js +2 -0
  86. package/dist/BodyScan-DpSLSJkg.js.map +1 -0
  87. package/dist/BodyScan-DshLNWwS.js +2 -0
  88. package/dist/BodyScan-DshLNWwS.js.map +1 -0
  89. package/dist/BodyScan-GsUKrJbz.js +2 -0
  90. package/dist/BodyScan-GsUKrJbz.js.map +1 -0
  91. package/dist/BodyScan-L9522Lbf.js +2 -0
  92. package/dist/BodyScan-L9522Lbf.js.map +1 -0
  93. package/dist/BodyScan-MAcTx2L_.js +2 -0
  94. package/dist/BodyScan-MAcTx2L_.js.map +1 -0
  95. package/dist/BodyScan-OtYm2KGr.js +2 -0
  96. package/dist/BodyScan-OtYm2KGr.js.map +1 -0
  97. package/dist/BodyScan-UrOz9iX3.js +2 -0
  98. package/dist/BodyScan-UrOz9iX3.js.map +1 -0
  99. package/dist/BodyScan-YlIT9lwi.js +2 -0
  100. package/dist/BodyScan-YlIT9lwi.js.map +1 -0
  101. package/dist/BodyScan-Z0QgXmsy.js +2 -0
  102. package/dist/BodyScan-Z0QgXmsy.js.map +1 -0
  103. package/dist/BodyScan-grQSzLIy.js +2 -0
  104. package/dist/BodyScan-grQSzLIy.js.map +1 -0
  105. package/dist/BodyScan-hL7iBlIZ.js +2 -0
  106. package/dist/BodyScan-hL7iBlIZ.js.map +1 -0
  107. package/dist/BodyScan-i6cTFkQH.js +2 -0
  108. package/dist/BodyScan-i6cTFkQH.js.map +1 -0
  109. package/dist/BodyScan-iqP_d120.js +2 -0
  110. package/dist/BodyScan-iqP_d120.js.map +1 -0
  111. package/dist/BodyScan-ivkSAzu4.js +2 -0
  112. package/dist/BodyScan-ivkSAzu4.js.map +1 -0
  113. package/dist/BodyScan-rVGG8jfb.js +2 -0
  114. package/dist/BodyScan-rVGG8jfb.js.map +1 -0
  115. package/dist/FaceScan-3X6MQMEX.js +2 -0
  116. package/dist/FaceScan-3X6MQMEX.js.map +1 -0
  117. package/dist/FaceScan-B5Ig13YK.js +2 -0
  118. package/dist/FaceScan-B5Ig13YK.js.map +1 -0
  119. package/dist/FaceScan-BEYwgG5s.js +2 -0
  120. package/dist/FaceScan-BEYwgG5s.js.map +1 -0
  121. package/dist/FaceScan-BJwQ3zJy.js +2 -0
  122. package/dist/FaceScan-BJwQ3zJy.js.map +1 -0
  123. package/dist/FaceScan-BKNQASGz.js +2 -0
  124. package/dist/FaceScan-BKNQASGz.js.map +1 -0
  125. package/dist/FaceScan-BUCUq-Ky.js +2 -0
  126. package/dist/FaceScan-BUCUq-Ky.js.map +1 -0
  127. package/dist/FaceScan-BXeNhTPp.js +2 -0
  128. package/dist/FaceScan-BXeNhTPp.js.map +1 -0
  129. package/dist/FaceScan-BoeC5FYB.js +2 -0
  130. package/dist/FaceScan-BoeC5FYB.js.map +1 -0
  131. package/dist/FaceScan-BsdZKlOZ.js +2 -0
  132. package/dist/FaceScan-BsdZKlOZ.js.map +1 -0
  133. package/dist/FaceScan-C89mYHyf.js +2 -0
  134. package/dist/FaceScan-C89mYHyf.js.map +1 -0
  135. package/dist/FaceScan-CAYo5Sx3.js +2 -0
  136. package/dist/FaceScan-CAYo5Sx3.js.map +1 -0
  137. package/dist/FaceScan-CDd3-sHM.js +2 -0
  138. package/dist/FaceScan-CDd3-sHM.js.map +1 -0
  139. package/dist/FaceScan-CDup4cr6.js +2 -0
  140. package/dist/FaceScan-CDup4cr6.js.map +1 -0
  141. package/dist/FaceScan-CEKFYJ7H.js +2 -0
  142. package/dist/FaceScan-CEKFYJ7H.js.map +1 -0
  143. package/dist/FaceScan-CKosTv3D.js +2 -0
  144. package/dist/FaceScan-CKosTv3D.js.map +1 -0
  145. package/dist/FaceScan-CQvW2N4-.js +2 -0
  146. package/dist/FaceScan-CQvW2N4-.js.map +1 -0
  147. package/dist/FaceScan-CUOi23K0.js +2 -0
  148. package/dist/FaceScan-CUOi23K0.js.map +1 -0
  149. package/dist/FaceScan-CX9evcWd.js +2 -0
  150. package/dist/FaceScan-CX9evcWd.js.map +1 -0
  151. package/dist/FaceScan-CaUr_UkA.js +2 -0
  152. package/dist/FaceScan-CaUr_UkA.js.map +1 -0
  153. package/dist/FaceScan-CjN4rVmd.js +2 -0
  154. package/dist/FaceScan-CjN4rVmd.js.map +1 -0
  155. package/dist/FaceScan-CkIu9qjW.js +2 -0
  156. package/dist/FaceScan-CkIu9qjW.js.map +1 -0
  157. package/dist/FaceScan-CnRj1Yix.js +2 -0
  158. package/dist/FaceScan-CnRj1Yix.js.map +1 -0
  159. package/dist/FaceScan-CoYz9-ne.js +2 -0
  160. package/dist/FaceScan-CoYz9-ne.js.map +1 -0
  161. package/dist/FaceScan-CpOUXeSp.js +2 -0
  162. package/dist/FaceScan-CpOUXeSp.js.map +1 -0
  163. package/dist/FaceScan-Cthc-Mpu.js +2 -0
  164. package/dist/FaceScan-Cthc-Mpu.js.map +1 -0
  165. package/dist/FaceScan-Cxdni73m.js +2 -0
  166. package/dist/FaceScan-Cxdni73m.js.map +1 -0
  167. package/dist/FaceScan-Czg2RLuu.js +2 -0
  168. package/dist/FaceScan-Czg2RLuu.js.map +1 -0
  169. package/dist/FaceScan-D4uGzUfl.js +2 -0
  170. package/dist/FaceScan-D4uGzUfl.js.map +1 -0
  171. package/dist/FaceScan-DINxXMKv.js +2 -0
  172. package/dist/FaceScan-DINxXMKv.js.map +1 -0
  173. package/dist/FaceScan-DL21ykY_.js +2 -0
  174. package/dist/FaceScan-DL21ykY_.js.map +1 -0
  175. package/dist/FaceScan-DR_OKG51.js +2 -0
  176. package/dist/FaceScan-DR_OKG51.js.map +1 -0
  177. package/dist/FaceScan-DmjXFO5E.js +2 -0
  178. package/dist/FaceScan-DmjXFO5E.js.map +1 -0
  179. package/dist/FaceScan-ECW1q_Bc.js +2 -0
  180. package/dist/FaceScan-ECW1q_Bc.js.map +1 -0
  181. package/dist/FaceScan-HkWW9fIU.js +2 -0
  182. package/dist/FaceScan-HkWW9fIU.js.map +1 -0
  183. package/dist/FaceScan-KJ_k17JE.js +2 -0
  184. package/dist/FaceScan-KJ_k17JE.js.map +1 -0
  185. package/dist/FaceScan-MB0mPOC4.js +2 -0
  186. package/dist/FaceScan-MB0mPOC4.js.map +1 -0
  187. package/dist/FaceScan-O63Wbjt7.js +2 -0
  188. package/dist/FaceScan-O63Wbjt7.js.map +1 -0
  189. package/dist/FaceScan-UGdieN4m.js +2 -0
  190. package/dist/FaceScan-UGdieN4m.js.map +1 -0
  191. package/dist/FaceScan-X-iTekPu.js +2 -0
  192. package/dist/FaceScan-X-iTekPu.js.map +1 -0
  193. package/dist/FaceScan-YY0nhUc-.js +2 -0
  194. package/dist/FaceScan-YY0nhUc-.js.map +1 -0
  195. package/dist/FaceScan-aTY9dK-E.js +2 -0
  196. package/dist/FaceScan-aTY9dK-E.js.map +1 -0
  197. package/dist/FaceScan-gLauGMHv.js +2 -0
  198. package/dist/FaceScan-gLauGMHv.js.map +1 -0
  199. package/dist/FaceScan-gcS5xK0C.js +2 -0
  200. package/dist/FaceScan-gcS5xK0C.js.map +1 -0
  201. package/dist/FaceScan-k2iKowYL.js +2 -0
  202. package/dist/FaceScan-k2iKowYL.js.map +1 -0
  203. package/dist/FaceScan-ksOwbYeM.js +2 -0
  204. package/dist/FaceScan-ksOwbYeM.js.map +1 -0
  205. package/dist/FaceScan-sAMtjY_G.js +2 -0
  206. package/dist/FaceScan-sAMtjY_G.js.map +1 -0
  207. package/dist/FaceScan-srkhPOLY.js +2 -0
  208. package/dist/FaceScan-srkhPOLY.js.map +1 -0
  209. package/dist/FaceScan-tfgID8J-.js +2 -0
  210. package/dist/FaceScan-tfgID8J-.js.map +1 -0
  211. package/dist/LoadingScreen-6-Dztgm-.js +2 -0
  212. package/dist/LoadingScreen-6-Dztgm-.js.map +1 -0
  213. package/dist/LoadingScreen-B-Mv028i.js +2 -0
  214. package/dist/LoadingScreen-B-Mv028i.js.map +1 -0
  215. package/dist/LoadingScreen-B27DzMIt.js +2 -0
  216. package/dist/LoadingScreen-B27DzMIt.js.map +1 -0
  217. package/dist/LoadingScreen-B5iS2xWa.js +2 -0
  218. package/dist/LoadingScreen-B5iS2xWa.js.map +1 -0
  219. package/dist/LoadingScreen-Bj3uzfuT.js +2 -0
  220. package/dist/LoadingScreen-Bj3uzfuT.js.map +1 -0
  221. package/dist/LoadingScreen-BjhVLzbU.js +2 -0
  222. package/dist/LoadingScreen-BjhVLzbU.js.map +1 -0
  223. package/dist/LoadingScreen-BoqG9-bi.js +2 -0
  224. package/dist/LoadingScreen-BoqG9-bi.js.map +1 -0
  225. package/dist/LoadingScreen-Bpbe8TTf.js +2 -0
  226. package/dist/LoadingScreen-Bpbe8TTf.js.map +1 -0
  227. package/dist/LoadingScreen-BwJk45mz.js +2 -0
  228. package/dist/LoadingScreen-BwJk45mz.js.map +1 -0
  229. package/dist/LoadingScreen-C0x3_zHI.js +2 -0
  230. package/dist/LoadingScreen-C0x3_zHI.js.map +1 -0
  231. package/dist/LoadingScreen-C2p1rVUR.js +2 -0
  232. package/dist/LoadingScreen-C2p1rVUR.js.map +1 -0
  233. package/dist/LoadingScreen-C9yWL5Pc.js +2 -0
  234. package/dist/LoadingScreen-C9yWL5Pc.js.map +1 -0
  235. package/dist/LoadingScreen-CCaeHSK3.js +2 -0
  236. package/dist/LoadingScreen-CCaeHSK3.js.map +1 -0
  237. package/dist/LoadingScreen-CGbxyNyP.js +2 -0
  238. package/dist/LoadingScreen-CGbxyNyP.js.map +1 -0
  239. package/dist/LoadingScreen-CcAK_ftO.js +2 -0
  240. package/dist/LoadingScreen-CcAK_ftO.js.map +1 -0
  241. package/dist/LoadingScreen-CdA6xTi9.js +2 -0
  242. package/dist/LoadingScreen-CdA6xTi9.js.map +1 -0
  243. package/dist/LoadingScreen-CgzB83o8.js +2 -0
  244. package/dist/LoadingScreen-CgzB83o8.js.map +1 -0
  245. package/dist/LoadingScreen-Csi-kWWH.js +2 -0
  246. package/dist/LoadingScreen-Csi-kWWH.js.map +1 -0
  247. package/dist/LoadingScreen-CtyswN5E.js +2 -0
  248. package/dist/LoadingScreen-CtyswN5E.js.map +1 -0
  249. package/dist/LoadingScreen-CvTw4HCH.js +2 -0
  250. package/dist/LoadingScreen-CvTw4HCH.js.map +1 -0
  251. package/dist/LoadingScreen-CzLNKXb1.js +2 -0
  252. package/dist/LoadingScreen-CzLNKXb1.js.map +1 -0
  253. package/dist/LoadingScreen-D1nnEFNR.js +2 -0
  254. package/dist/LoadingScreen-D1nnEFNR.js.map +1 -0
  255. package/dist/LoadingScreen-DAMI7lHa.js +2 -0
  256. package/dist/LoadingScreen-DAMI7lHa.js.map +1 -0
  257. package/dist/LoadingScreen-DOyxaBGy.js +2 -0
  258. package/dist/LoadingScreen-DOyxaBGy.js.map +1 -0
  259. package/dist/LoadingScreen-DVjp9x4z.js +2 -0
  260. package/dist/LoadingScreen-DVjp9x4z.js.map +1 -0
  261. package/dist/LoadingScreen-DZBvMyMp.js +2 -0
  262. package/dist/LoadingScreen-DZBvMyMp.js.map +1 -0
  263. package/dist/LoadingScreen-DcZYWJaQ.js +2 -0
  264. package/dist/LoadingScreen-DcZYWJaQ.js.map +1 -0
  265. package/dist/LoadingScreen-DfhwWaDP.js +2 -0
  266. package/dist/LoadingScreen-DfhwWaDP.js.map +1 -0
  267. package/dist/LoadingScreen-DhbtuXrJ.js +2 -0
  268. package/dist/LoadingScreen-DhbtuXrJ.js.map +1 -0
  269. package/dist/LoadingScreen-Dib6A2Ve.js +2 -0
  270. package/dist/LoadingScreen-Dib6A2Ve.js.map +1 -0
  271. package/dist/LoadingScreen-DjYv3mcm.js +2 -0
  272. package/dist/LoadingScreen-DjYv3mcm.js.map +1 -0
  273. package/dist/LoadingScreen-DjbTg0B7.js +2 -0
  274. package/dist/LoadingScreen-DjbTg0B7.js.map +1 -0
  275. package/dist/LoadingScreen-DrHVz3VF.js +2 -0
  276. package/dist/LoadingScreen-DrHVz3VF.js.map +1 -0
  277. package/dist/LoadingScreen-DzHMcCLq.js +2 -0
  278. package/dist/LoadingScreen-DzHMcCLq.js.map +1 -0
  279. package/dist/LoadingScreen-HEWqrS0h.js +2 -0
  280. package/dist/LoadingScreen-HEWqrS0h.js.map +1 -0
  281. package/dist/LoadingScreen-NaDsFeSL.js +2 -0
  282. package/dist/LoadingScreen-NaDsFeSL.js.map +1 -0
  283. package/dist/LoadingScreen-QJ0014wx.js +2 -0
  284. package/dist/LoadingScreen-QJ0014wx.js.map +1 -0
  285. package/dist/LoadingScreen-XpTbgEdb.js +2 -0
  286. package/dist/LoadingScreen-XpTbgEdb.js.map +1 -0
  287. package/dist/LoadingScreen-mkx7zXm3.js +2 -0
  288. package/dist/LoadingScreen-mkx7zXm3.js.map +1 -0
  289. package/dist/LoadingScreen-xK5L8Cwd.js +2 -0
  290. package/dist/LoadingScreen-xK5L8Cwd.js.map +1 -0
  291. package/dist/bodyScan.d.ts +2 -2
  292. package/dist/bodyScan.js +1 -1
  293. package/dist/bodyScan.mjs +1 -1
  294. package/dist/faceScan.d.ts +2 -2
  295. package/dist/faceScan.js +1 -1
  296. package/dist/faceScan.mjs +1 -1
  297. package/dist/index.d.ts +2 -2
  298. package/dist/index.js +1 -1
  299. package/dist/index.mjs +1 -1
  300. package/dist/interfaces-9gH-zdwN.d.ts +172 -0
  301. package/dist/interfaces-B-8WwrSS.d.ts +205 -0
  302. package/dist/interfaces-BEjFALPD.d.ts +172 -0
  303. package/dist/interfaces-BIz9uFrG.d.ts +169 -0
  304. package/dist/interfaces-BdmYhSHQ.d.ts +178 -0
  305. package/dist/interfaces-BkHveLwI.d.ts +206 -0
  306. package/dist/interfaces-BsHu082_.d.ts +175 -0
  307. package/dist/interfaces-DDDU9Drs.d.ts +204 -0
  308. package/dist/interfaces-DQ9_eGDO.d.ts +206 -0
  309. package/dist/interfaces-DRQdRcUH.d.ts +167 -0
  310. package/dist/interfaces-DtTdp0w3.d.ts +180 -0
  311. package/dist/interfaces-LWgRZJ8r.d.ts +165 -0
  312. package/dist/interfaces-eRPH_aoH.d.ts +167 -0
  313. package/dist/interfaces-wwqeKDrE.d.ts +179 -0
  314. package/dist/pose-detection.esm-BLeo_q1f.js +18 -0
  315. package/dist/pose-detection.esm-BLeo_q1f.js.map +1 -0
  316. package/dist/pose-detection.esm-BaWguRmj.js +18 -0
  317. package/dist/pose-detection.esm-BaWguRmj.js.map +1 -0
  318. package/dist/pose-detection.esm-BpdMeLUd.js +18 -0
  319. package/dist/pose-detection.esm-BpdMeLUd.js.map +1 -0
  320. package/dist/pose-detection.esm-CUXej2pM.js +18 -0
  321. package/dist/pose-detection.esm-CUXej2pM.js.map +1 -0
  322. package/package.json +1 -1
@@ -0,0 +1 @@
1
+ {"version":3,"file":"FaceScan-D4uGzUfl.js","sources":["../src/customHooks/useFaceScan.ts","../src/components/faceScan/FaceScanErrorScreen.tsx","../src/components/faceScan/FaceScanGuide.tsx","../src/components/faceScan/FaceScanStep.tsx","../src/components/faceScan/FaceScan.tsx"],"sourcesContent":["import {\n useState,\n useRef,\n useEffect,\n useCallback,\n useMemo,\n RefObject,\n} from \"react\";\n// 1. Use TYPE-ONLY imports. These are erased at build time and won't crash SSR.\nimport type { PoseDetector } from \"@tensorflow-models/pose-detection\";\nimport { posthog } from \"posthog-js\";\nimport speechService from \"../utils/service/speechService\";\nimport { videoConstraints, voiceOverAssetsPath } from \"../utils/constants\";\n\n/**\n * useFaceScan Hook with PostHog Analytics\n * Refactored for Zero-Config SSR/CSR Compatibility\n */\n\n// Debounce utility\nfunction debounce(fn: (...args: any) => void, delay: number) {\n let timer: number | NodeJS.Timeout;\n return (...args: any) => {\n if (timer) clearTimeout(timer);\n timer = setTimeout(() => fn(...args), delay);\n };\n}\n\n// Global Singleton Cache (Preserves state across re-renders)\nlet preloadedDetector: PoseDetector | null = null;\nlet isPreloading = false;\n\n// 2. Async Preloader with Dynamic Imports\nexport const preloadDetector = async (): Promise<PoseDetector | null> => {\n // Guard: Never run on server\n if (typeof window === \"undefined\") return null;\n if (preloadedDetector) return preloadedDetector;\n if (isPreloading) {\n // Simple wait mechanism if already loading\n while (isPreloading) {\n await new Promise(r => setTimeout(r, 100));\n if (preloadedDetector) return preloadedDetector;\n }\n }\n\n isPreloading = true;\n\n try {\n // DYNAMIC IMPORTS: Only fetch these heavy bundles in the browser\n const [tf, poseDetection] = await Promise.all([\n import(\"@tensorflow/tfjs\"),\n import(\"@tensorflow-models/pose-detection\")\n ]);\n\n // Initialize Backend\n await tf.setBackend(\"webgl\");\n await tf.ready();\n\n const detector = await poseDetection.createDetector(\n poseDetection.SupportedModels.BlazePose,\n {\n runtime: \"tfjs\",\n modelType: \"full\",\n }\n );\n\n // Dummy video warmup (Performance Optimization)\n // This runs the model once on a blank canvas to \"wake up\" the GPU\n // so the first user frame detects instantly.\n const dummyCanvas = document.createElement(\"canvas\");\n dummyCanvas.width = videoConstraints.width;\n dummyCanvas.height = videoConstraints.height;\n const ctx = dummyCanvas.getContext(\"2d\");\n if (ctx) {\n ctx.fillStyle = \"black\";\n ctx.fillRect(0, 0, dummyCanvas.width, dummyCanvas.height);\n try {\n await detector.estimatePoses(dummyCanvas);\n } catch (e) {\n console.warn(\"Warmup frame failed (non-fatal):\", e);\n }\n }\n\n preloadedDetector = detector;\n return detector;\n } catch (err) {\n console.error(\"Failed to preload detector:\", err);\n return null;\n } finally {\n isPreloading = false;\n }\n};\n\ntype Point = [number, number, number];\n\nfunction useFaceScan({\n faceScanId,\n onValidPose,\n onScanComplete,\n onModelReady,\n onStartRecording,\n shopDomain,\n}: {\n faceScanId: string;\n onValidPose?: () => void;\n onScanComplete?: () => void;\n onModelReady?: () => void;\n onStartRecording: () => void;\n shopDomain: string;\n}) {\n const detectorRef = useRef<PoseDetector | null>(null);\n const [scanStage, setScanStage] = useState(0);\n const [consecutiveValid, setConsecutiveValid] = useState(0);\n const [isModelLoaded, setIsModelLoaded] = useState(false);\n const [isScanningActive, setIsScanningActive] = useState(false);\n \n const utteranceRef = useRef<SpeechSynthesisUtterance | null>(null);\n const scanStageRef = useRef(scanStage);\n const consecutiveValidRef = useRef(consecutiveValid);\n const validityBufferRef = useRef<boolean[]>([]);\n const lastSpokenMessageRef = useRef(\"\");\n const lastSpokenTimeRef = useRef(0);\n const voiceEnabledRef = useRef(false);\n \n const VALIDITY_BUFFER_LENGTH = 10;\n const CONSECUTIVE_VALID_LENGTH = 2;\n const TOTAL_STAGES = 4;\n const POSE_TRACKING_INTERVAL = 4000;\n const lastPoseTrackingTimeRef = useRef(0);\n\n // Safe iOS detection (SSR safe)\n const isIOS = typeof navigator !== 'undefined' && (\n /iPad|iPhone|iPod/.test(navigator.userAgent) ||\n (navigator.platform === \"MacIntel\" && navigator.maxTouchPoints > 1)\n );\n\n const directionMessages = useMemo(\n () => [\n \"Face front\",\n \"Turn head fully left\",\n \"Turn head fully right\",\n \"Smile facing front\",\n ],\n []\n );\n\n const debouncedTrackPoseDetection = useMemo(\n () =>\n debounce((payload) => {\n posthog.capture(`${shopDomain}/face_scan_pose_detection`, payload);\n }, 4000),\n [shopDomain]\n );\n\n const trackPoseDetection = useCallback(\n (\n faceKeypoints: Point[],\n bodyKeypoints: Point[],\n stage: number,\n headValid: boolean,\n bodyInvalid: boolean,\n consecutiveValidCount: number\n ) => {\n const now = Date.now();\n if (now - lastPoseTrackingTimeRef.current < POSE_TRACKING_INTERVAL) {\n return;\n }\n\n lastPoseTrackingTimeRef.current = now;\n\n try {\n const nose = faceKeypoints[0] || [0, 0, 0];\n const leftEye = faceKeypoints[2] || [0, 0, 0];\n const rightEye = faceKeypoints[5] || [0, 0, 0];\n const leftMouth = faceKeypoints[9] || [0, 0, 0];\n const rightMouth = faceKeypoints[10] || [0, 0, 0];\n\n const faceKeypointScores = faceKeypoints\n .map((kp: Point) => kp[2])\n .filter((score: number) => score > 0);\n const avgFaceScore =\n faceKeypointScores.length > 0\n ? faceKeypointScores.reduce((a, b) => a + b, 0) /\n faceKeypointScores.length\n : 0;\n\n const bodyKeypointScores = bodyKeypoints\n .map((kp: Point) => kp[2])\n .filter((score: number) => score > 0);\n const avgBodyScore =\n bodyKeypointScores.length > 0\n ? bodyKeypointScores.reduce((a: number, b: number) => a + b, 0) /\n bodyKeypointScores.length\n : 0;\n\n const eyeDistance = Math.abs(leftEye[0] - rightEye[0]);\n const noseToRight = nose[0] - rightEye[0];\n const noseToLeft = leftEye[0] - nose[0];\n\n let expectedPosition = \"frontal\";\n if (stage === 1) expectedPosition = \"left\";\n else if (stage === 2) expectedPosition = \"right\";\n else if (stage === 3) expectedPosition = \"frontal_smile\";\n\n const payload = {\n faceScanId,\n stage,\n timestamp: new Date().toISOString(),\n data: JSON.stringify({\n headValid,\n bodyInvalid,\n consecutiveValid: consecutiveValidCount,\n avgFaceScore: parseFloat(avgFaceScore.toFixed(3)),\n avgBodyScore: parseFloat(avgBodyScore.toFixed(3)),\n faceKeypointsDetected: faceKeypointScores.length,\n bodyKeypointsDetected: bodyKeypointScores.length,\n eyeDistance: parseFloat(eyeDistance.toFixed(2)),\n noseToRightRatio: parseFloat((noseToRight / eyeDistance).toFixed(3)),\n noseToLeftRatio: parseFloat((noseToLeft / eyeDistance).toFixed(3)),\n noseScore: parseFloat(nose[2].toFixed(3)),\n leftEyeScore: parseFloat(leftEye[2].toFixed(3)),\n rightEyeScore: parseFloat(rightEye[2].toFixed(3)),\n leftMouthScore: parseFloat(leftMouth[2].toFixed(3)),\n rightMouthScore: parseFloat(rightMouth[2].toFixed(3)),\n expectedPosition,\n }),\n };\n debouncedTrackPoseDetection(payload);\n } catch (error) {\n console.warn(\"Failed to track pose detection:\", error);\n }\n },\n [faceScanId, debouncedTrackPoseDetection]\n );\n\n const enableVoiceCommands = () => {\n voiceEnabledRef.current = true;\n if (typeof window !== 'undefined' && isIOS && \"speechSynthesis\" in window) {\n const silentUtterance = new SpeechSynthesisUtterance(\"\");\n silentUtterance.volume = 0;\n speechSynthesis.speak(silentUtterance);\n }\n };\n\n const startScanSequence = () => {\n posthog.capture(`${shopDomain}/face_scan_detection_started`, {\n faceScanId,\n stage: scanStageRef.current,\n timestamp: new Date().toISOString(),\n stageName: directionMessages[scanStageRef.current],\n });\n\n setTimeout(() => {\n setIsScanningActive(true);\n }, 1500);\n };\n\n const isHeadInPosition = (stage: number, keypoints: Point[]) => {\n // ... logic unchanged ...\n const nose = 0;\n const leftEye = 2;\n const rightEye = 5;\n const leftMouth = 9;\n const rightMouth = 10;\n\n const minScore = 0.2;\n if (\n !keypoints[nose] || !keypoints[leftEye] || !keypoints[rightEye] ||\n keypoints[nose][2] < minScore ||\n keypoints[leftEye][2] < minScore ||\n keypoints[rightEye][2] < minScore\n ) {\n return false;\n }\n\n const eyesToMouthVerticalDistance =\n 0.5 * (keypoints[rightMouth][1] + keypoints[leftMouth][1]) -\n 0.5 * (keypoints[rightEye][1] + keypoints[leftEye][1]);\n\n const faceTiltedness =\n (0.5 * (keypoints[rightMouth][1] + keypoints[leftMouth][1]) -\n keypoints[nose][1]) /\n eyesToMouthVerticalDistance;\n\n const faceIsTilted = faceTiltedness > 0.7 || faceTiltedness < 0.3;\n\n const faceIsVertical =\n Math.abs(keypoints[leftEye][1] - keypoints[rightEye][1]) <\n 0.5 * eyesToMouthVerticalDistance &&\n Math.abs(keypoints[leftMouth][1] - keypoints[rightMouth][1]) <\n 0.5 * eyesToMouthVerticalDistance;\n\n const eyeDistance = keypoints[leftEye][0] - keypoints[rightEye][0];\n const noseToRight = keypoints[nose][0] - keypoints[rightEye][0];\n const noseToLeft = keypoints[leftEye][0] - keypoints[nose][0];\n\n const frontalFace =\n noseToRight > 0.4 * eyeDistance && noseToRight < 0.6 * eyeDistance;\n const fullLeft = noseToRight > 0.85 * eyeDistance;\n const fullRight = noseToLeft > 0.85 * eyeDistance;\n\n switch (stage) {\n case 0:\n return !faceIsTilted && faceIsVertical && frontalFace;\n case 1:\n return !faceIsTilted && faceIsVertical && fullLeft;\n case 2:\n return !faceIsTilted && faceIsVertical && fullRight;\n case 3:\n return !faceIsTilted && faceIsVertical && frontalFace;\n default:\n return false;\n }\n };\n\n const faceScanDetector = async (\n webcamRef: RefObject<HTMLVideoElement | null>,\n canvasRef: RefObject<HTMLCanvasElement | null>\n ) => {\n if (\n !detectorRef.current ||\n !webcamRef.current ||\n !isModelLoaded ||\n !isScanningActive\n )\n return;\n\n // Safety: ensure video is actually playing with data\n if (webcamRef.current.readyState < 2) return;\n\n try {\n const poses = await detectorRef.current.estimatePoses(webcamRef.current);\n if (poses.length > 0) {\n const faceKeypoints: Point[] = [];\n const bodyKeypoints: Point[] = [];\n poses[0].keypoints.forEach((landmark, idx) => {\n const score = landmark.score ?? 0;\n const point: Point = [landmark.x ?? -1, landmark.y ?? -1, score];\n if (idx <= 10) faceKeypoints.push(point);\n else bodyKeypoints.push(point);\n });\n\n // Optional drawing\n if (canvasRef?.current) {\n const ctx = canvasRef.current.getContext(\"2d\");\n const { width, height } = canvasRef.current;\n if (ctx) {\n ctx.clearRect(0, 0, width, height);\n\n const progress = Math.min(\n consecutiveValidRef.current / CONSECUTIVE_VALID_LENGTH,\n 1\n );\n const radius = height * 0.35;\n ctx.beginPath();\n ctx.strokeStyle = \"#00ff00\";\n ctx.lineWidth = 6;\n ctx.arc(\n width / 2,\n height / 2,\n radius + 10,\n -Math.PI / 2,\n -Math.PI / 2 + progress * 2 * Math.PI\n );\n ctx.stroke();\n }\n }\n\n const headValid = isHeadInPosition(scanStageRef.current, faceKeypoints);\n const bodyInvalid =\n bodyKeypoints.slice(2).filter((p) => p[2] > 0.7).length <= 2;\n\n validityBufferRef.current.push(headValid && bodyInvalid);\n if (validityBufferRef.current.length > VALIDITY_BUFFER_LENGTH) {\n validityBufferRef.current.shift();\n }\n\n const validCount = validityBufferRef.current.filter(Boolean).length;\n if (validCount >= 2) {\n const nextValid = consecutiveValidRef.current + 1;\n setConsecutiveValid(nextValid);\n } else {\n setConsecutiveValid(Math.max(0, consecutiveValidRef.current - 1));\n }\n\n if (consecutiveValidRef.current >= CONSECUTIVE_VALID_LENGTH) {\n onValidPose?.();\n\n const nextStage = scanStageRef.current + 1;\n setConsecutiveValid(0);\n validityBufferRef.current = [];\n setScanStage(nextStage);\n setIsScanningActive(false);\n\n (async () => {\n await speechService.playAudio(\n `${voiceOverAssetsPath}face-scan-vos/Sound-effect.mp3`\n );\n posthog.capture(\n `${shopDomain}/face_scan_detection_stage_completed`,\n {\n faceScanId,\n completedStage: scanStageRef.current,\n nextStage,\n timestamp: new Date().toISOString(),\n totalStages: TOTAL_STAGES,\n stageName: directionMessages[scanStageRef.current],\n }\n );\n \n if (nextStage === 1 && onStartRecording) {\n onStartRecording();\n }\n if (nextStage >= TOTAL_STAGES) {\n onScanComplete?.();\n } else {\n let nextSound = null;\n switch (nextStage) {\n case 1: nextSound = \"Left.mp3\"; break;\n case 2: nextSound = \"Right.mp3\"; break;\n case 3: nextSound = \"Smile.mp3\"; break;\n }\n if (nextSound) {\n await speechService.playAudio(\n `${voiceOverAssetsPath}face-scan-vos/${nextSound}`\n );\n }\n startScanSequence();\n }\n })();\n }\n\n trackPoseDetection(\n faceKeypoints,\n bodyKeypoints,\n scanStageRef.current,\n headValid,\n bodyInvalid,\n consecutiveValidRef.current\n );\n }\n } catch (err) {\n console.error(\"Pose detection error:\", err);\n }\n };\n\n const initializeModels = async () => {\n // 3. Browser Guard\n if (typeof window === 'undefined') return;\n\n try {\n console.log(\"Initializing Face Scan...\");\n // Calls our safe, dynamic preloader\n const detector = await preloadDetector();\n \n if (detector) {\n detectorRef.current = detector;\n console.log(\"Face scan model loaded successfully\");\n setIsModelLoaded(true);\n onModelReady?.();\n }\n } catch (err) {\n console.error(\"Error initializing face scan:\", err);\n }\n };\n\n const disposeModelAndTf = async () => {\n // Optional: We might NOT want to dispose if we want to keep the cache \n // for re-mounting. But if you must dispose:\n try {\n setIsModelLoaded(false);\n // NOTE: We generally don't dispose the detector if we want to reuse it \n // via the 'preloadedDetector' global variable.\n // If you dispose here, make sure to set 'preloadedDetector = null' too.\n /* if (detectorRef.current) {\n await detectorRef.current.dispose();\n detectorRef.current = null;\n preloadedDetector = null; // Clear global cache if disposing\n }\n */\n } catch (err) {\n console.error(\"Error disposing resources:\", err);\n } finally {\n // Reset state\n setScanStage(0);\n setConsecutiveValid(0);\n scanStageRef.current = 0;\n consecutiveValidRef.current = 0;\n validityBufferRef.current = [];\n }\n };\n\n const resetScan = () => {\n posthog.capture(`${shopDomain}/face_scan_reset`, {\n faceScanId,\n stage: scanStageRef.current,\n timestamp: new Date().toISOString(),\n stageName: directionMessages[scanStageRef.current],\n consecutiveValid: consecutiveValidRef.current,\n });\n\n setScanStage(0);\n setConsecutiveValid(0);\n setIsScanningActive(false);\n scanStageRef.current = 0;\n consecutiveValidRef.current = 0;\n validityBufferRef.current = [];\n lastSpokenMessageRef.current = \"\";\n lastSpokenTimeRef.current = 0;\n voiceEnabledRef.current = false;\n if (utteranceRef.current) {\n speechSynthesis.cancel();\n utteranceRef.current = null;\n }\n };\n\n useEffect(() => {\n initializeModels();\n return () => {\n disposeModelAndTf();\n };\n }, []);\n\n useEffect(() => {\n scanStageRef.current = scanStage;\n }, [scanStage]);\n\n useEffect(() => {\n consecutiveValidRef.current = consecutiveValid;\n }, [consecutiveValid]);\n\n return {\n faceScanDetector,\n scanStage,\n setScanStage,\n consecutiveValid,\n isModelLoaded,\n resetScan,\n enableVoiceCommands,\n startScanSequence,\n isScanningActive,\n };\n}\n\nexport default useFaceScan;","import { ArrowRight } from \"lucide-react\";\nimport Header from \"../Header\";\nimport SpecificButton from \"../../atoms/specificButton/SpecificButton\";\nimport { useContext } from \"react\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { Config } from \"../../types/interfaces\";\n\nfunction FaceScanErrorScreen({ resetScanState, loading, config }: { resetScanState?: () => void; loading?: boolean; config?: Config }) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\n\treturn (\n\t\t<div className=\"fixed top-[0] left-[0] z-[999] flex justify-center items-center w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t<div className=\"flex flex-col w-full items-center p-[1rem] rounded-lg \">\n\t\t\t\t<Header noTitle resolvedConfig={config} />\n\t\t\t\t<div className=\"flex flex-col items-center w-full\">\n\t\t\t\t\t<h2\n\t\t\t\t\t\tclassName=\"text-xl font-semibold text-gray-800 \"\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.scanFailed)}\n\t\t\t\t\t</h2>\n\t\t\t\t\t<p\n\t\t\t\t\t\tclassName=\"mb-[1.5rem]\"\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\tcolor: config?.style?.subheading?.subheadingColor || \"#4b5563\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.clickToResetScan)}\n\t\t\t\t\t</p>\n\t\t\t\t\t<SpecificButton\n\t\t\t\t\t\tdisabled={loading}\n\t\t\t\t\t\tclassName=\"w-full h-[45px] shadow-[0px_1px_2px_0px_#00000040] text-[16px]\"\n\t\t\t\t\t\tbuttonText={translate?.(LanguageKeys.resetScan)}\n\t\t\t\t\t\tpostfixIcon={<ArrowRight />}\n\t\t\t\t\t\tbuttonFunc={() => resetScanState?.()}\n\t\t\t\t\t\tresolvedConfig={config}\n\t\t\t\t\t/>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t</div>\n\t);\n}\n\nexport default FaceScanErrorScreen;\n","import { FaceScanGuideProps } from \"../../types/interfaces\";\nimport Header from \"../Header\";\nimport { GenderType } from \"../../utils/enums\";\nimport { directionMessages, FACE_SCAN_HEADSHOT, glassesOffVideo, maleGlassesOffVideo } from \"../../utils/constants\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { useContext } from \"react\";\nimport SpecificButton from \"../../atoms/specificButton/SpecificButton\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\n\nfunction FaceScanGuide({ stage, videoLoading, setVideoLoading, videoError, setVideoError, gender, config, btnConfig }: FaceScanGuideProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\n\n\n\tif (stage === -1) {\n\t\treturn (\n\t\t\t<div className=\"text-center p-[16px] w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t\t<div className=\"w-full\">\n\t\t\t\t\t<Header noTitle resolvedConfig={config} />\n\t\t\t\t\t<h2\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.faceVisible)}\n\t\t\t\t\t</h2>\n\t\t\t\t\t{videoLoading && (\n\t\t\t\t\t\t<div\n\t\t\t\t\t\t\tclassName=\"mb-4 flex items-center justify-center\"\n\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\t\tcolor: config?.style?.subheading?.subheadingColor || \"#4b5563\",\n\t\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t>\n\t\t\t\t\t\t\t{translate?.(LanguageKeys.loadingVideo)}\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t\t{videoError && (\n\t\t\t\t\t\t<div\n\t\t\t\t\t\t\tclassName=\"mb-4 text-red-600\"\n\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\t\tcolor: \"#ff0000\",\n\t\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t>\n\t\t\t\t\t\t\t{translate?.(LanguageKeys.videoLoadFailed)}\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t\t{!videoError && (\n\t\t\t\t\t\t<div className={` w-[100px] mx-auto`}>\n\t\t\t\t\t\t\t<video\n\t\t\t\t\t\t\t\tsrc={gender === GenderType.Male ? maleGlassesOffVideo : glassesOffVideo}\n\t\t\t\t\t\t\t\tautoPlay\n\t\t\t\t\t\t\t\tloop\n\t\t\t\t\t\t\t\tcontrols={false}\n\t\t\t\t\t\t\t\tmuted\n\t\t\t\t\t\t\t\tplaysInline\n\t\t\t\t\t\t\t\tclassName=\"h-full w-full object-contain border-none\"\n\t\t\t\t\t\t\t\tonCanPlay={() => setVideoLoading(false)}\n\t\t\t\t\t\t\t\tonLoadStart={() => setVideoLoading(true)}\n\t\t\t\t\t\t\t\tonError={() => setVideoError(true)}\n\t\t\t\t\t\t\t\taria-label=\"Instructional video: Please remove your glasses before starting the scan.\"\n\t\t\t\t\t\t\t/>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t</div>\n\t\t\t\t<div className=\"flex justify-center w-full p-[1rem]\">\n\t\t\t\t\t<SpecificButton\n\t\t\t\t\t\tdisabled={btnConfig.isDisabled}\n\t\t\t\t\t\tclassName=\"!w-[60px] !h-[35px] !py-0 !px-0\"\n\t\t\t\t\t\tbuttonText={translate?.(btnConfig.label)}\n\t\t\t\t\t\tpostfixIcon={btnConfig?.icon}\n\t\t\t\t\t\tbuttonFunc={btnConfig.onClick}\n\t\t\t\t\t\tresolvedConfig={config}\n\t\t\t\t\t/>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t);\n\t}\n\n\treturn (\n\t\t<div className=\"text-center p-[16px] w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t<div className=\"w-full\">\n\t\t\t\t<Header noTitle resolvedConfig={config} />\n\t\t\t\t<h3\n\t\t\t\t\tclassName=\"mb-[0.5rem]\"\n\t\t\t\t\tstyle={{\n\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t{translate?.(directionMessages[stage])}\n\t\t\t\t</h3>\n\t\t\t\t{/* <p>We'll guide you to take 6 selfies </p> */}\n\t\t\t\t<div className=\"max-w-[400px] justify-center gap-1 mx-auto flex items-center\">\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 0 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.forward : FACE_SCAN_HEADSHOT.female.forward} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 1 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.left : FACE_SCAN_HEADSHOT.female.left} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 2 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.right : FACE_SCAN_HEADSHOT.female.right} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 3 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.smile : FACE_SCAN_HEADSHOT.female.smile} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t</div>\n\t);\n}\n\nexport default FaceScanGuide;\n","import React, { useContext, useEffect } from \"react\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\nimport FaceScanErrorScreen from \"./FaceScanErrorScreen\";\nimport LoadingScreen from \"../LoadingScreen\";\nimport { videoConstraints } from \"../../utils/constants\";\nimport { Drawer } from \"@mui/material\";\nimport FaceScanGuide from \"./FaceScanGuide\";\nimport { ArrowRight } from \"lucide-react\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { useLocalConfig } from \"../../config/useLocalConfig\";\ninterface FaceScanStepProps {\n\twebcamRef: React.RefObject<HTMLVideoElement | null>;\n\tcanvasRef: React.RefObject<HTMLCanvasElement | null>;\n}\n\nconst FaceScanStep: React.FC<FaceScanStepProps> = ({ webcamRef, canvasRef }) => {\n\tconst {\n\t\tisError,\n\t\tisSuccess,\n\t\tshowLoader,\n\t\thasError,\n\t\tresetScanState,\n\t\tshowGuideCard,\n\t\tscanStage,\n\t\tvideoLoading,\n\t\tsetVideoLoading,\n\t\tvideoError,\n\t\tsetVideoError,\n\t\tgender,\n\t\tgetButtonText,\n\t\tisScanning,\n\t\tstartScan,\n\t\tisModelLoaded,\n\t\tconfig,\n\t} = useContext(ParamsContext);\n\tconst resolvedConfig = useLocalConfig(config);\n\n\tconst { setPreferredLanguage } = useContext(LanguageContext) || {};\n\tuseEffect(() => {\n\t\tsetPreferredLanguage?.(resolvedConfig?.language);\n\t}, [resolvedConfig]);\n\tif (isError) {\n\t\treturn <FaceScanErrorScreen config={resolvedConfig} />;\n\t}\n\tif (isSuccess) {\n\t\treturn (\n\t\t\t<div className=\"fixed z-[9] w-full h-full\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t<LoadingScreen url={resolvedConfig?.loader} loaderType=\"black\" />\n\t\t\t</div>\n\t\t);\n\t}\n\n\treturn (\n\t\t<>\n\t\t\t<audio id=\"audioElement\" crossOrigin=\"anonymous\" preload=\"auto\" style={{ position: \"absolute\", zIndex: -99999 }} src={undefined} />\n\n\t\t\t{/* Error overlay */}\n\t\t\t{showLoader && !hasError && (\n\t\t\t\t<div className=\"fixed z-[9] w-full h-full\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t\t{/* <Asset genderType={gender} /> */}\n\t\t\t\t\t<LoadingScreen url={resolvedConfig?.loader} loaderType=\"black\" />\n\t\t\t\t</div>\n\t\t\t)}\n\n\t\t\t{/* Always show camera view */}\n\t\t\t<div className=\"h-full flex-col relative w-full flex justify-center items-center text-center rounded-t-[20px]\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t<div className=\"flex-1 w-full max-w-md overflow-hidden\">\n\t\t\t\t\t<div className=\"w-full h-full\">\n\t\t\t\t\t\t<video\n\t\t\t\t\t\t\tref={webcamRef}\n\t\t\t\t\t\t\tautoPlay\n\t\t\t\t\t\t\tplaysInline\n\t\t\t\t\t\t\tmuted\n\t\t\t\t\t\t\twidth={videoConstraints.width.ideal}\n\t\t\t\t\t\t\theight={videoConstraints.height.ideal}\n\t\t\t\t\t\t\tclassName=\"w-full h-full object-cover fixed left-0 top-0 z-0\"\n\t\t\t\t\t\t\tstyle={{ transform: \"scaleX(-1)\" }}\n\t\t\t\t\t\t/>\n\t\t\t\t\t\t<canvas ref={canvasRef} width={videoConstraints.width.ideal} height={videoConstraints.height.ideal} style={{ transform: \"scaleX(-1)\", opacity: \"0\" }} />\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\n\t\t\t{/* Error overlay */}\n\t\t\t{!showLoader && hasError && <FaceScanErrorScreen loading={showLoader} resetScanState={resetScanState} config={resolvedConfig} />}\n\n\t\t\t{/* Scan guide drawer - only show when scanning */}\n\t\t\t{showGuideCard && !hasError && !showLoader && (\n\t\t\t\t<Drawer\n\t\t\t\t\topen\n\t\t\t\t\tclassName=\"face-scan-small camera-draw\"\n\t\t\t\t\tanchor=\"bottom\"\n\t\t\t\t\tonClose={(event, reason) => {\n\t\t\t\t\t\tif (reason === \"backdropClick\") {\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t}\n\t\t\t\t\t}}\n\t\t\t\t\thideBackdrop\n\t\t\t\t>\n\t\t\t\t\t<FaceScanGuide\n\t\t\t\t\t\tstage={scanStage}\n\t\t\t\t\t\tvideoLoading={videoLoading}\n\t\t\t\t\t\tsetVideoLoading={setVideoLoading}\n\t\t\t\t\t\tvideoError={videoError}\n\t\t\t\t\t\tsetVideoError={setVideoError}\n\t\t\t\t\t\tgender={gender}\n\t\t\t\t\t\tconfig={resolvedConfig}\n\t\t\t\t\t\tbtnConfig={{\n\t\t\t\t\t\t\tlabel: getButtonText(),\n\t\t\t\t\t\t\tonClick: isScanning ? resetScanState : startScan,\n\t\t\t\t\t\t\tisDisabled: showLoader || !isModelLoaded,\n\t\t\t\t\t\t\ticon: isScanning ? <ArrowRight /> : <></>,\n\t\t\t\t\t\t}}\n\t\t\t\t\t/>\n\t\t\t\t</Drawer>\n\t\t\t)}\n\t\t</>\n\t);\n};\n\nexport default FaceScanStep;\n","\"use client\";\nimport { useEffect, useRef, useState, useCallback, useMemo } from \"react\";\nimport posthog from \"posthog-js\";\nimport { FaceScanProps } from \"../../types/interfaces\";\nimport {\n\tgenerateUuid,\n\tgetPreferredMediaRecorderTypes,\n\tgetRecordingExtension,\n\tgetRecordingMimeType,\n\tgetSupportedMediaRecorderTypes,\n\tgetVideoFPS,\n\thandleScanTimeCapture,\n\thandleWebSocketCapture,\n} from \"../../utils/utils\";\nimport { posthogPublicHost, posthogPublicKey, videoConstraintsExact, videoConstraintsFallback, videoTypes, voiceOverAssetsPath } from \"../../utils/constants\";\nimport { getSwanService } from \"../../utils/service/swanService\";\nimport speechService from \"../../utils/service/speechService\";\nimport useFaceScan from \"../../customHooks/useFaceScan\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport LanguageContextProvider from \"../../utils/context/languageContext\";\nimport FaceScanStep from \"./FaceScanStep\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\n\nexport const FaceScan: React.FC<FaceScanProps> = ({\n\tuserDetails,\n\ttoken,\n\tonScanSuccess,\n\tonScanError,\n\tonRetry,\n\tonScanStart,\n\tonUploadingStart,\n\tonUploadingEnd,\n\tonMeasurementSocketStart,\n\tonMeasurementSocketClose,\n\tconfig,\n\tisError,\n\tisSuccess,\n}) => {\n\tconst webcamRef = useRef<HTMLVideoElement | null>(null);\n\tconst canvasRef = useRef<HTMLCanvasElement | null>(null);\n\tconst mediaRecorderRef = useRef<MediaRecorder | null>(null);\n\tconst recordedBlobsRef = useRef<Blob[] | null>([]);\n\tconst streamRef = useRef<MediaStream | null>(null);\n\tconst recordedFpsRef = useRef<number | null>(null);\n\tconst [showLoader, setShowLoader] = useState(false);\n\tconst [modelReady, setModelReady] = useState(false);\n\tconst [isScanning, setIsScanning] = useState(false);\n\tconst [showGuideCard, setShowGuideCard] = useState(true);\n\tconst [videoLoading, setVideoLoading] = useState(false);\n\tconst [videoError, setVideoError] = useState(false);\n\tconst [hasError, setHasError] = useState(false);\n\tconst [faceScanId, setFaceScanId] = useState(generateUuid());\n\tconst swan = useMemo(() => getSwanService(token), [token]);\n\tconst { email, gender, deviceFocalLength, shopDomain, callbackUrl } = userDetails;\n\tconst [supportedTypes, setSupportedTypes] = useState<string[]>([]);\n\tconst [currentVideoConstraints, setCurrentVideoConstraints] = useState<MediaTrackConstraints>(videoConstraintsExact);\n\tconst preferredVideoTypes = useMemo(() => getPreferredMediaRecorderTypes(), []);\n\tconst recordingMimeType = getRecordingMimeType(supportedTypes);\n\tconst recordingExtension = getRecordingExtension(recordingMimeType);\n\n\tconst stopRecording = useCallback(() => {\n\t\tconsole.log(\"Stopping recording...\");\n\t\tif (mediaRecorderRef.current && mediaRecorderRef.current.state === \"recording\") {\n\t\t\tmediaRecorderRef.current.stop();\n\t\t}\n\t\tmediaRecorderRef.current = null;\n\t}, []);\n\n\tconst uploadFinalVideo = async () => {\n\t\tif (recordedBlobsRef.current) {\n\t\t\tif (recordedBlobsRef.current.length === 0) {\n\t\t\t\tconsole.error(\"No video data recorded\");\n\t\t\t\tsetHasError(true);\n\t\t\t\tsetShowLoader(false);\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tonUploadingStart?.();\n\t\t\tsetShowLoader(true);\n\t\t\tconst videoFile = new File(recordedBlobsRef.current, `${faceScanId}.${recordingExtension}`, {\n\t\t\t\ttype: recordingMimeType,\n\t\t\t});\n\t\t\t// Use FPS captured at recording time, fallback to calculation if not available\n\t\t\tconst videoFps = recordedFpsRef.current ?? (await getVideoFPS(videoFile));\n\n\t\t\tconst fileSize = videoFile.size;\n\t\t\tconst fileSizeMB = (fileSize / (1024 * 1024)).toFixed(2);\n\t\t\tconst estimatedDuration = Math.round(fileSize / 10000);\n\t\t\tconst videoData = {\n\t\t\t\tvideo_size_mb: parseFloat(fileSizeMB),\n\t\t\t\tvideo_size_bytes: fileSize,\n\t\t\t\tvideo_fps: videoFps || 0,\n\t\t\t\tblob_count: recordedBlobsRef.current.length,\n\t\t\t\testimated_duration_seconds: estimatedDuration,\n\t\t\t};\n\n\t\t\tconst metaData: any[] = [\n\t\t\t\t{ gender: gender },\n\t\t\t\t{ face_scan_id: faceScanId },\n\t\t\t\t{\n\t\t\t\t\tfocal_length: `${deviceFocalLength}`,\n\t\t\t\t},\n\t\t\t\t{ customer_store_url: shopDomain },\n\t\t\t\t{ scan_type: \"face_scan\" },\n\t\t\t];\n\t\t\tif (callbackUrl) {\n\t\t\t\tmetaData.push({ callback_url: callbackUrl });\n\t\t\t}\n\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/face_scan_meta_data`,\n\t\t\t\tfaceScanId,\n\t\t\t\temail,\n\t\t\t\tdata: JSON.stringify({ metaData, videoData }),\n\t\t\t});\n\n\t\t\tconst uploadStartTime = Date.now();\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/face_scan_upload_start`,\n\t\t\t\tfaceScanId,\n\t\t\t\temail,\n\t\t\t\tstartTime: uploadStartTime,\n\t\t\t});\n\n\t\t\ttry {\n\t\t\t\tconst res = await swan.fileUpload.faceScanFileUploader({\n\t\t\t\t\tfile: videoFile,\n\t\t\t\t\tarrayMetaData: metaData,\n\t\t\t\t\tobjectKey: faceScanId,\n\t\t\t\t\temail,\n\t\t\t\t\tcontentType: videoFile.type,\n\t\t\t\t});\n\n\t\t\t\tconst uploadEndTime = Date.now();\n\t\t\t\tconst uploadDuration = uploadEndTime - uploadStartTime;\n\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: `${shopDomain}/face_scan_upload_complete`,\n\t\t\t\t\tfaceScanId,\n\t\t\t\t\temail,\n\t\t\t\t\tcompletionTime: uploadEndTime,\n\t\t\t\t\tuploadDuration,\n\t\t\t\t});\n\t\t\t\tonUploadingEnd?.();\n\t\t\t\tconsole.log(\"✅ Upload successful\", res);\n\t\t\t\tswan.measurement.handlFaceScaneSocket({\n\t\t\t\t\tonPreopen: () => {\n\t\t\t\t\t\tonMeasurementSocketStart?.();\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"pre_open\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t},\n\t\t\t\t\tfaceScanId,\n\t\t\t\t\tonOpen: () => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"open\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tconsole.log(\"websocket connect open\");\n\t\t\t\t\t},\n\t\t\t\t\tonError: (err) => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"error\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tonError(err);\n\t\t\t\t\t},\n\t\t\t\t\tonSuccess: (data) => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"success\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tonSuccess(data);\n\t\t\t\t\t},\n\t\t\t\t\tonClose: () => {\n\t\t\t\t\t\tonMeasurementSocketClose?.();\n\t\t\t\t\t\tconsole.log(\"websocket connect close\");\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"close\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t},\n\t\t\t\t});\n\t\t\t} catch (error) {\n\t\t\t\tonUploadingEnd?.();\n\t\t\t\tonError(error);\n\t\t\t}\n\t\t}\n\t};\n\n\tconst startRecording = useCallback(() => {\n\t\tconst stream = webcamRef.current?.srcObject as MediaStream | null;\n\t\tif (!stream) return;\n\n\t\t// Create a canvas to normalize orientation\n\t\tconst videoTrack = stream.getVideoTracks()[0];\n\t\tconst settings = videoTrack.getSettings();\n\t\tconst defaultWidth = typeof currentVideoConstraints.width === \"number\" ? currentVideoConstraints.width : 720;\n\t\tconst defaultHeight = typeof currentVideoConstraints.height === \"number\" ? currentVideoConstraints.height : 1280;\n\t\tconst { width = defaultWidth, height = defaultHeight } = settings;\n\n\t\tconst canvas = document.createElement(\"canvas\");\n\t\tconst ctx = canvas.getContext(\"2d\");\n\n\t\t// Always force portrait (swap width/height if landscape)\n\t\tif (width > height) {\n\t\t\tcanvas.width = height;\n\t\t\tcanvas.height = width;\n\t\t} else {\n\t\t\tcanvas.width = width;\n\t\t\tcanvas.height = height;\n\t\t}\n\n\t\tconst videoEl = document.createElement(\"video\");\n\t\tvideoEl.srcObject = stream;\n\t\tvideoEl.muted = true;\n\t\tvideoEl.playsInline = true;\n\t\tvideoEl.play();\n\n\t\t// Draw video frames onto canvas\n\t\tconst drawFrame = () => {\n\t\t\t// Rotate if camera gives landscape frames\n\t\t\tif (width > height) {\n\t\t\t\tctx?.save();\n\t\t\t\tctx?.translate(canvas.width, 0);\n\t\t\t\tctx?.rotate(Math.PI / 2);\n\t\t\t\tctx?.drawImage(videoEl, 0, 0, height, width);\n\t\t\t\tctx?.restore();\n\t\t\t} else {\n\t\t\t\tctx?.drawImage(videoEl, 0, 0, width, height);\n\t\t\t}\n\t\t\trequestAnimationFrame(drawFrame);\n\t\t};\n\t\tdrawFrame();\n\n\t\tconst recordingFps = 30; // Canvas capture stream FPS\n\t\trecordedFpsRef.current = recordingFps; // Store FPS at recording time\n\t\tconst canvasStream = canvas.captureStream(recordingFps);\n\t\tconst mediaRecorderOptions: MediaRecorderOptions = {\n\t\t\tvideoBitsPerSecond: 1_500_000,\n\t\t};\n\t\tif (supportedTypes.length > 0) {\n\t\t\tmediaRecorderOptions.mimeType = supportedTypes[0];\n\t\t}\n\t\tlet mediaRecorder;\n\n\t\ttry {\n\t\t\tmediaRecorder = new MediaRecorder(canvasStream, mediaRecorderOptions);\n\t\t} catch (e) {\n\t\t\tconsole.error(\"MediaRecorder init failed:\", e);\n\t\t\tsetHasError(true);\n\t\t\tsetShowLoader(false);\n\t\t\treturn;\n\t\t}\n\n\t\trecordedBlobsRef.current = [];\n\t\tmediaRecorder.ondataavailable = (event) => {\n\t\t\tif (event?.data && event.data.size > 0 && recordedBlobsRef.current) {\n\t\t\t\trecordedBlobsRef.current.push(event.data);\n\t\t\t}\n\t\t};\n\t\tmediaRecorder.onstop = () => {\n\t\t\tconsole.log(\"Recording stopped, total blobs:\", recordedBlobsRef?.current?.length);\n\t\t\tuploadFinalVideo();\n\t\t};\n\n\t\tmediaRecorder.start(100); // 100ms chunks\n\t\tmediaRecorderRef.current = mediaRecorder;\n\t\tconsole.log(\"Recording started successfully (portrait normalized)\");\n\t}, [supportedTypes, uploadFinalVideo]);\n\n\tconst { faceScanDetector, scanStage, setScanStage, resetScan, isModelLoaded, startScanSequence } = useFaceScan({\n\t\tfaceScanId,\n\t\tshopDomain,\n\t\tonScanComplete: () => {\n\t\t\tstopRecording();\n\t\t},\n\t\tonModelReady: () => {\n\t\t\tsetModelReady(true);\n\t\t},\n\t\tonStartRecording: () => {\n\t\t\tconsole.log(\"Stage 0 completed - starting recording for stages 1-3\");\n\t\t\tstartRecording();\n\t\t},\n\t});\n\n\tconst startScan = useCallback(() => {\n\t\tif (!isModelLoaded) return;\n\t\tonScanStart?.();\n\t\tsetScanStage(0);\n\t\tspeechService.playAudio(`${voiceOverAssetsPath}face-scan-vos/Face-forward.mp3`);\n\t\tsetTimeout(() => {\n\t\t\tsetIsScanning(true);\n\t\t\tsetTimeout(() => {\n\t\t\t\tstartScanSequence();\n\t\t\t}, 200);\n\t\t}, 200);\n\t}, [setScanStage, setIsScanning, startScanSequence, isModelLoaded, onScanStart]);\n\n\tconst resetScanState = useCallback(() => {\n\t\tsetIsScanning(false);\n\t\tsetShowGuideCard(true);\n\t\tstopRecording();\n\t\tresetScan();\n\t\tonRetry?.();\n\t\trecordedBlobsRef.current = [];\n\t\trecordedFpsRef.current = null;\n\t\tif (mediaRecorderRef.current) {\n\t\t\tmediaRecorderRef.current = null;\n\t\t}\n\t\tsetHasError(false);\n\t\tsetScanStage(-1);\n\t\tsetFaceScanId(generateUuid());\n\t\tif (webcamRef.current && streamRef.current) {\n\t\t\twebcamRef.current.srcObject = streamRef.current;\n\t\t\tconsole.log(\"Camera stream restored after reset\");\n\t\t}\n\t}, [setScanStage, setIsScanning, setShowGuideCard, stopRecording, resetScan, setFaceScanId]);\n\n\tconst onError = (data: any) => {\n\t\tconsole.log(data, \"ws error\");\n\t\tonScanError?.(data);\n\t\tsetHasError(true);\n\t\tsetShowLoader(false);\n\t\tsetShowGuideCard(false);\n\t\thandleScanTimeCapture({\n\t\t\teventName: `${shopDomain}/faceScan`,\n\t\t\tfaceScanId,\n\t\t\tstatus: \"failed\",\n\t\t\temail,\n\t\t\tdata: JSON.stringify(data),\n\t\t});\n\t};\n\n\tconst onSuccess = (data: any) => {\n\t\tconsole.log(data, \"ws success\");\n\t\tif (data && data?.resultType === \"intermediate\") {\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/faceScan_success/intermediate`,\n\t\t\t\tfaceScanId,\n\t\t\t\tstatus: \"success\",\n\t\t\t\temail,\n\t\t\t\tdata: JSON.stringify(data),\n\t\t\t});\n\t\t}\n\t\tonScanSuccess?.(data);\n\t};\n\n\tuseEffect(() => {\n\t\tif (isError || isSuccess) return;\n\t\tif (navigator.mediaDevices.getUserMedia) {\n\t\t\tnavigator.mediaDevices\n\t\t\t\t.getUserMedia({ video: currentVideoConstraints })\n\t\t\t\t.then((stream) => {\n\t\t\t\t\tstreamRef.current = stream;\n\t\t\t\t\tif (webcamRef.current) {\n\t\t\t\t\t\twebcamRef.current.srcObject = stream;\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t\t\t.catch((err) => {\n\t\t\t\t\tconsole.error(\"Error accessing webcam:\", err);\n\t\t\t\t\tif (currentVideoConstraints === videoConstraintsExact) {\n\t\t\t\t\t\tsetCurrentVideoConstraints(videoConstraintsFallback);\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\t\t\t\t\tsetHasError(true);\n\t\t\t\t});\n\t\t}\n\t\treturn () => {\n\t\t\tif (streamRef.current) {\n\t\t\t\tstreamRef.current.getTracks().forEach((track) => track.stop());\n\t\t\t}\n\t\t};\n\t}, [isError, isSuccess, currentVideoConstraints]);\n\n\t// Face detection interval - only run when scanning is active\n\tuseEffect(() => {\n\t\tif (!modelReady || !isScanning) return;\n\t\tconst intervalId = setInterval(() => {\n\t\t\tfaceScanDetector(webcamRef, canvasRef);\n\t\t}, 500);\n\n\t\t// eslint-disable-next-line consistent-return\n\t\treturn () => clearInterval(intervalId);\n\t}, [faceScanDetector, modelReady, isScanning]);\n\n\tconst getButtonText = () => {\n\t\tif (isScanning) return LanguageKeys.reset;\n\t\tif (!isModelLoaded) return LanguageKeys.loading;\n\t\treturn LanguageKeys.start;\n\t};\n\n\tconsole.log(\"Model ready:\", modelReady, \"Has error:\", hasError, \"Is scanning:\", isScanning, \"showLoader\", showLoader);\n\tuseEffect(() => {\n\t\tconst supported = getSupportedMediaRecorderTypes(preferredVideoTypes, videoTypes);\n\t\tsetSupportedTypes(supported);\n\t}, [preferredVideoTypes]);\n\tuseEffect(() => {\n\t\tsetScanStage(-1);\n\t\tif (isError || isSuccess) return;\n\t}, [isError, isSuccess]);\n\n\tuseEffect(() => {\n\t\tposthog.init(posthogPublicKey, { api_host: posthogPublicHost });\n\t\tposthog.capture(\"$pageview\");\n\t}, []);\n\n\treturn (\n\t\t<LanguageContextProvider>\n\t\t\t<ParamsContext.Provider\n\t\t\t\tvalue={{\n\t\t\t\t\tisError,\n\t\t\t\t\tisSuccess,\n\t\t\t\t\tshowLoader,\n\t\t\t\t\thasError,\n\t\t\t\t\tresetScanState,\n\t\t\t\t\tshowGuideCard,\n\t\t\t\t\tscanStage,\n\t\t\t\t\tvideoLoading,\n\t\t\t\t\tsetVideoLoading,\n\t\t\t\t\tvideoError,\n\t\t\t\t\tsetVideoError,\n\t\t\t\t\tgender,\n\t\t\t\t\tgetButtonText,\n\t\t\t\t\tisScanning,\n\t\t\t\t\tstartScan,\n\t\t\t\t\tisModelLoaded,\n\t\t\t\t\tconfig,\n\t\t\t\t}}\n\t\t\t>\n\t\t\t\t<FaceScanStep webcamRef={webcamRef} canvasRef={canvasRef} />\n\t\t\t</ParamsContext.Provider>\n\t\t</LanguageContextProvider>\n\t);\n};\n"],"names":["preloadedDetector","isPreloading","useFaceScan","faceScanId","onValidPose","onScanComplete","onModelReady","onStartRecording","shopDomain","detectorRef","useRef","scanStage","setScanStage","useState","consecutiveValid","setConsecutiveValid","isModelLoaded","setIsModelLoaded","isScanningActive","setIsScanningActive","utteranceRef","scanStageRef","consecutiveValidRef","validityBufferRef","lastSpokenMessageRef","lastSpokenTimeRef","voiceEnabledRef","lastPoseTrackingTimeRef","isIOS","navigator","test","userAgent","platform","maxTouchPoints","directionMessages","useMemo","debouncedTrackPoseDetection","fn","delay","timer","args","clearTimeout","setTimeout","debounce","payload","posthog","capture","trackPoseDetection","useCallback","faceKeypoints","bodyKeypoints","stage","headValid","bodyInvalid","consecutiveValidCount","now","Date","current","nose","leftEye","rightEye","leftMouth","rightMouth","faceKeypointScores","map","kp","filter","score","avgFaceScore","length","reduce","a","b","bodyKeypointScores","avgBodyScore","eyeDistance","Math","abs","noseToRight","noseToLeft","expectedPosition","timestamp","toISOString","data","JSON","stringify","parseFloat","toFixed","faceKeypointsDetected","bodyKeypointsDetected","noseToRightRatio","noseToLeftRatio","noseScore","leftEyeScore","rightEyeScore","leftMouthScore","rightMouthScore","error","console","warn","startScanSequence","stageName","initializeModels","async","window","log","detector","Promise","r","tf","poseDetection","all","import","setBackend","ready","createDetector","SupportedModels","BlazePose","runtime","modelType","dummyCanvas","document","createElement","width","videoConstraints","height","ctx","getContext","fillStyle","fillRect","estimatePoses","e","err","preloadDetector","useEffect","disposeModelAndTf","faceScanDetector","webcamRef","canvasRef","readyState","poses","keypoints","forEach","landmark","idx","point","x","y","push","clearRect","progress","min","radius","beginPath","strokeStyle","lineWidth","arc","PI","stroke","eyesToMouthVerticalDistance","faceTiltedness","faceIsTilted","faceIsVertical","frontalFace","fullLeft","fullRight","isHeadInPosition","slice","p","shift","Boolean","nextValid","max","nextStage","speechService","playAudio","voiceOverAssetsPath","completedStage","totalStages","nextSound","resetScan","speechSynthesis","cancel","enableVoiceCommands","silentUtterance","SpeechSynthesisUtterance","volume","speak","FaceScanErrorScreen","resetScanState","loading","config","translate","useContext","LanguageContext","_jsx","className","style","background","base","backgroundColor","children","_jsxs","Header","noTitle","resolvedConfig","fontFamily","heading","headingFontFamily","fontSize","headingFontSize","color","headingColor","fontWeight","headingFontWeight","LanguageKeys","scanFailed","subheading","subheadingFontFamily","subheadingFontSize","subheadingColor","subheadingFontWeight","clickToResetScan","SpecificButton","disabled","buttonText","postfixIcon","ArrowRight","buttonFunc","FaceScanGuide","videoLoading","setVideoLoading","videoError","setVideoError","gender","btnConfig","faceVisible","loadingVideo","videoLoadFailed","src","GenderType","Male","maleGlassesOffVideo","glassesOffVideo","autoPlay","loop","controls","muted","playsInline","onCanPlay","onLoadStart","onError","isDisabled","label","icon","onClick","FACE_SCAN_HEADSHOT","male","forward","female","type","left","right","smile","FaceScanStep","isError","isSuccess","showLoader","hasError","showGuideCard","getButtonText","isScanning","startScan","ParamsContext","useLocalConfig","setPreferredLanguage","language","LoadingScreen","url","loader","loaderType","_Fragment","id","crossOrigin","preload","position","zIndex","undefined","ref","ideal","transform","opacity","Drawer","open","anchor","onClose","event","reason","hideBackdrop","userDetails","token","onScanSuccess","onScanError","onRetry","onScanStart","onUploadingStart","onUploadingEnd","onMeasurementSocketStart","onMeasurementSocketClose","mediaRecorderRef","recordedBlobsRef","streamRef","recordedFpsRef","setShowLoader","modelReady","setModelReady","setIsScanning","setShowGuideCard","setHasError","setFaceScanId","generateUuid","swan","getSwanService","email","deviceFocalLength","callbackUrl","supportedTypes","setSupportedTypes","currentVideoConstraints","setCurrentVideoConstraints","videoConstraintsExact","preferredVideoTypes","getPreferredMediaRecorderTypes","recordingMimeType","getRecordingMimeType","recordingExtension","getRecordingExtension","stopRecording","state","stop","uploadFinalVideo","videoFile","File","videoFps","getVideoFPS","fileSize","size","fileSizeMB","estimatedDuration","round","videoData","video_size_mb","video_size_bytes","video_fps","blob_count","estimated_duration_seconds","metaData","face_scan_id","focal_length","customer_store_url","scan_type","callback_url","handleScanTimeCapture","eventName","uploadStartTime","startTime","res","fileUpload","faceScanFileUploader","file","arrayMetaData","objectKey","contentType","uploadEndTime","uploadDuration","completionTime","measurement","handlFaceScaneSocket","onPreopen","handleWebSocketCapture","faceScanID","connection","onOpen","onSuccess","startRecording","stream","srcObject","settings","getVideoTracks","getSettings","defaultWidth","defaultHeight","canvas","videoEl","play","drawFrame","save","rotate","drawImage","restore","requestAnimationFrame","canvasStream","captureStream","mediaRecorderOptions","videoBitsPerSecond","mediaRecorder","mimeType","MediaRecorder","ondataavailable","onstop","start","status","resultType","mediaDevices","getUserMedia","video","then","catch","videoConstraintsFallback","getTracks","track","intervalId","setInterval","clearInterval","supported","getSupportedMediaRecorderTypes","videoTypes","init","posthogPublicKey","api_host","posthogPublicHost","LanguageContextProvider","Provider","value","reset"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ "use strict";var e=require("react/jsx-runtime"),t=require("react"),a=require("./LoadingScreen-DhbtuXrJ.js"),n=require("posthog-js"),o=require("lucide-react"),r=require("@mui/material");let s=null,c=!1;function i({faceScanId:e,onValidPose:o,onScanComplete:r,onModelReady:i,onStartRecording:l,shopDomain:d}){const u=t.useRef(null),[g,f]=t.useState(0),[h,m]=t.useState(0),[p,S]=t.useState(!1),[y,x]=t.useState(!1),v=t.useRef(null),b=t.useRef(g),w=t.useRef(h),F=t.useRef([]),C=t.useRef(""),j=t.useRef(0),_=t.useRef(!1),N=t.useRef(0),E="undefined"!=typeof navigator&&(/iPad|iPhone|iPod/.test(navigator.userAgent)||"MacIntel"===navigator.platform&&navigator.maxTouchPoints>1),k=t.useMemo(()=>["Face front","Turn head fully left","Turn head fully right","Smile facing front"],[]),T=t.useMemo(()=>function(e,t){let a;return(...n)=>{a&&clearTimeout(a),a=setTimeout(()=>e(...n),t)}}(e=>{n.posthog.capture(`${d}/face_scan_pose_detection`,e)},4e3),[d]),I=t.useCallback((t,a,n,o,r,s)=>{const c=Date.now();if(!(c-N.current<4e3)){N.current=c;try{const c=t[0]||[0,0,0],i=t[2]||[0,0,0],l=t[5]||[0,0,0],d=t[9]||[0,0,0],u=t[10]||[0,0,0],g=t.map(e=>e[2]).filter(e=>e>0),f=g.length>0?g.reduce((e,t)=>e+t,0)/g.length:0,h=a.map(e=>e[2]).filter(e=>e>0),m=h.length>0?h.reduce((e,t)=>e+t,0)/h.length:0,p=Math.abs(i[0]-l[0]),S=c[0]-l[0],y=i[0]-c[0];let x="frontal";1===n?x="left":2===n?x="right":3===n&&(x="frontal_smile");const v={faceScanId:e,stage:n,timestamp:(new Date).toISOString(),data:JSON.stringify({headValid:o,bodyInvalid:r,consecutiveValid:s,avgFaceScore:parseFloat(f.toFixed(3)),avgBodyScore:parseFloat(m.toFixed(3)),faceKeypointsDetected:g.length,bodyKeypointsDetected:h.length,eyeDistance:parseFloat(p.toFixed(2)),noseToRightRatio:parseFloat((S/p).toFixed(3)),noseToLeftRatio:parseFloat((y/p).toFixed(3)),noseScore:parseFloat(c[2].toFixed(3)),leftEyeScore:parseFloat(i[2].toFixed(3)),rightEyeScore:parseFloat(l[2].toFixed(3)),leftMouthScore:parseFloat(d[2].toFixed(3)),rightMouthScore:parseFloat(u[2].toFixed(3)),expectedPosition:x})};T(v)}catch(e){console.warn("Failed to track pose detection:",e)}}},[e,T]),R=()=>{n.posthog.capture(`${d}/face_scan_detection_started`,{faceScanId:e,stage:b.current,timestamp:(new Date).toISOString(),stageName:k[b.current]}),setTimeout(()=>{x(!0)},1500)},M=async()=>{if("undefined"!=typeof window)try{console.log("Initializing Face Scan...");const e=await(async()=>{if("undefined"==typeof window)return null;if(s)return s;if(c)for(;c;)if(await new Promise(e=>setTimeout(e,100)),s)return s;c=!0;try{const[e,t]=await Promise.all([import("@tensorflow/tfjs"),Promise.resolve().then(function(){return require("./pose-detection.esm-BaWguRmj.js")})]);await e.setBackend("webgl"),await e.ready();const n=await t.createDetector(t.SupportedModels.BlazePose,{runtime:"tfjs",modelType:"full"}),o=document.createElement("canvas");o.width=a.videoConstraints.width,o.height=a.videoConstraints.height;const r=o.getContext("2d");if(r){r.fillStyle="black",r.fillRect(0,0,o.width,o.height);try{await n.estimatePoses(o)}catch(e){console.warn("Warmup frame failed (non-fatal):",e)}}return s=n,n}catch(e){return console.error("Failed to preload detector:",e),null}finally{c=!1}})();e&&(u.current=e,console.log("Face scan model loaded successfully"),S(!0),i?.())}catch(e){console.error("Error initializing face scan:",e)}};return t.useEffect(()=>(M(),()=>{(async()=>{try{S(!1)}catch(e){console.error("Error disposing resources:",e)}finally{f(0),m(0),b.current=0,w.current=0,F.current=[]}})()}),[]),t.useEffect(()=>{b.current=g},[g]),t.useEffect(()=>{w.current=h},[h]),{faceScanDetector:async(t,s)=>{if(u.current&&t.current&&p&&y&&!(t.current.readyState<2))try{const c=await u.current.estimatePoses(t.current);if(c.length>0){const t=[],i=[];if(c[0].keypoints.forEach((e,a)=>{const n=e.score??0,o=[e.x??-1,e.y??-1,n];a<=10?t.push(o):i.push(o)}),s?.current){const e=s.current.getContext("2d"),{width:t,height:a}=s.current;if(e){e.clearRect(0,0,t,a);const n=Math.min(w.current/2,1),o=.35*a;e.beginPath(),e.strokeStyle="#00ff00",e.lineWidth=6,e.arc(t/2,a/2,o+10,-Math.PI/2,-Math.PI/2+2*n*Math.PI),e.stroke()}}const u=((e,t)=>{if(!t[0]||!t[2]||!t[5]||t[0][2]<.2||t[2][2]<.2||t[5][2]<.2)return!1;const a=.5*(t[10][1]+t[9][1])-.5*(t[5][1]+t[2][1]),n=(.5*(t[10][1]+t[9][1])-t[0][1])/a,o=n>.7||n<.3,r=Math.abs(t[2][1]-t[5][1])<.5*a&&Math.abs(t[9][1]-t[10][1])<.5*a,s=t[2][0]-t[5][0],c=t[0][0]-t[5][0],i=c>.4*s&&c<.6*s,l=c>.85*s,d=t[2][0]-t[0][0]>.85*s;switch(e){case 0:case 3:return!o&&r&&i;case 1:return!o&&r&&l;case 2:return!o&&r&&d;default:return!1}})(b.current,t),g=i.slice(2).filter(e=>e[2]>.7).length<=2;F.current.push(u&&g),F.current.length>10&&F.current.shift();if(F.current.filter(Boolean).length>=2){const e=w.current+1;m(e)}else m(Math.max(0,w.current-1));if(w.current>=2){o?.();const t=b.current+1;m(0),F.current=[],f(t),x(!1),(async()=>{if(await a.speechService.playAudio(`${a.voiceOverAssetsPath}face-scan-vos/Sound-effect.mp3`),n.posthog.capture(`${d}/face_scan_detection_stage_completed`,{faceScanId:e,completedStage:b.current,nextStage:t,timestamp:(new Date).toISOString(),totalStages:4,stageName:k[b.current]}),1===t&&l&&l(),t>=4)r?.();else{let e=null;switch(t){case 1:e="Left.mp3";break;case 2:e="Right.mp3";break;case 3:e="Smile.mp3"}e&&await a.speechService.playAudio(`${a.voiceOverAssetsPath}face-scan-vos/${e}`),R()}})()}I(t,i,b.current,u,g,w.current)}}catch(e){console.error("Pose detection error:",e)}},scanStage:g,setScanStage:f,consecutiveValid:h,isModelLoaded:p,resetScan:()=>{n.posthog.capture(`${d}/face_scan_reset`,{faceScanId:e,stage:b.current,timestamp:(new Date).toISOString(),stageName:k[b.current],consecutiveValid:w.current}),f(0),m(0),x(!1),b.current=0,w.current=0,F.current=[],C.current="",j.current=0,_.current=!1,v.current&&(speechSynthesis.cancel(),v.current=null)},enableVoiceCommands:()=>{if(_.current=!0,"undefined"!=typeof window&&E&&"speechSynthesis"in window){const e=new SpeechSynthesisUtterance("");e.volume=0,speechSynthesis.speak(e)}},startScanSequence:R,isScanningActive:y}}function l({resetScanState:n,loading:r,config:s}){const{translate:c}=t.useContext(a.LanguageContext)||{};return e.jsx("div",{className:"fixed top-[0] left-[0] z-[999] flex justify-center items-center w-full h-full",style:{background:s?.style?.base?.backgroundColor},children:e.jsxs("div",{className:"flex flex-col w-full items-center p-[1rem] rounded-lg ",children:[e.jsx(a.Header,{noTitle:!0,resolvedConfig:s}),e.jsxs("div",{className:"flex flex-col items-center w-full",children:[e.jsx("h2",{className:"text-xl font-semibold text-gray-800 ",style:{fontFamily:s?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:s?.style?.heading?.headingFontSize||"32px",color:s?.style?.heading?.headingColor||"#000",fontWeight:s?.style?.heading?.headingFontWeight||"normal"},children:c?.(a.LanguageKeys.scanFailed)}),e.jsx("p",{className:"mb-[1.5rem]",style:{fontFamily:s?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:s?.style?.subheading?.subheadingFontSize||"14px",color:s?.style?.subheading?.subheadingColor||"#4b5563",fontWeight:s?.style?.subheading?.subheadingFontWeight||"normal"},children:c?.(a.LanguageKeys.clickToResetScan)}),e.jsx(a.SpecificButton,{disabled:r,className:"w-full h-[45px] shadow-[0px_1px_2px_0px_#00000040] text-[16px]",buttonText:c?.(a.LanguageKeys.resetScan),postfixIcon:e.jsx(o.ArrowRight,{}),buttonFunc:()=>n?.(),resolvedConfig:s})]})]})})}function d({stage:n,videoLoading:o,setVideoLoading:r,videoError:s,setVideoError:c,gender:i,config:l,btnConfig:d}){const{translate:u}=t.useContext(a.LanguageContext)||{};return-1===n?e.jsxs("div",{className:"text-center p-[16px] w-full h-full",style:{background:l?.style?.base?.backgroundColor},children:[e.jsxs("div",{className:"w-full",children:[e.jsx(a.Header,{noTitle:!0,resolvedConfig:l}),e.jsx("h2",{style:{fontFamily:l?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:l?.style?.heading?.headingFontSize||"32px",color:l?.style?.heading?.headingColor||"#000",fontWeight:l?.style?.heading?.headingFontWeight||"normal"},children:u?.(a.LanguageKeys.faceVisible)}),o&&e.jsx("div",{className:"mb-4 flex items-center justify-center",style:{fontFamily:l?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:l?.style?.subheading?.subheadingFontSize||"14px",color:l?.style?.subheading?.subheadingColor||"#4b5563",fontWeight:l?.style?.subheading?.subheadingFontWeight||"normal"},children:u?.(a.LanguageKeys.loadingVideo)}),s&&e.jsx("div",{className:"mb-4 text-red-600",style:{fontFamily:l?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:l?.style?.subheading?.subheadingFontSize||"14px",color:"#ff0000",fontWeight:l?.style?.subheading?.subheadingFontWeight||"normal"},children:u?.(a.LanguageKeys.videoLoadFailed)}),!s&&e.jsx("div",{className:" w-[100px] mx-auto",children:e.jsx("video",{src:i===a.GenderType.Male?a.maleGlassesOffVideo:a.glassesOffVideo,autoPlay:!0,loop:!0,controls:!1,muted:!0,playsInline:!0,className:"h-full w-full object-contain border-none",onCanPlay:()=>r(!1),onLoadStart:()=>r(!0),onError:()=>c(!0),"aria-label":"Instructional video: Please remove your glasses before starting the scan."})})]}),e.jsx("div",{className:"flex justify-center w-full p-[1rem]",children:e.jsx(a.SpecificButton,{disabled:d.isDisabled,className:"!w-[60px] !h-[35px] !py-0 !px-0",buttonText:u?.(d.label),postfixIcon:d?.icon,buttonFunc:d.onClick,resolvedConfig:l})})]}):e.jsx("div",{className:"text-center p-[16px] w-full h-full",style:{background:l?.style?.base?.backgroundColor},children:e.jsxs("div",{className:"w-full",children:[e.jsx(a.Header,{noTitle:!0,resolvedConfig:l}),e.jsx("h3",{className:"mb-[0.5rem]",style:{fontFamily:l?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:l?.style?.heading?.headingFontSize||"32px",color:l?.style?.heading?.headingColor||"#000",fontWeight:l?.style?.heading?.headingFontWeight||"normal"},children:u?.(a.directionMessages[n])}),e.jsxs("div",{className:"max-w-[400px] justify-center gap-1 mx-auto flex items-center",children:[e.jsx("div",{className:` w-[120px] overflow-hidden ${0===n?"opacity-100":"opacity-0 hidden"} `,children:e.jsx("video",{className:"h-full w-full object-contain border-none",muted:!0,loop:!0,autoPlay:!0,playsInline:!0,children:e.jsx("source",{src:i===a.GenderType.Male?a.FACE_SCAN_HEADSHOT.male.forward:a.FACE_SCAN_HEADSHOT.female.forward,type:"video/mp4"})})}),e.jsx("div",{className:` w-[120px] overflow-hidden ${1===n?"opacity-100":"opacity-0 hidden"} `,children:e.jsx("video",{className:"h-full w-full object-contain border-none",muted:!0,loop:!0,autoPlay:!0,playsInline:!0,children:e.jsx("source",{src:i===a.GenderType.Male?a.FACE_SCAN_HEADSHOT.male.left:a.FACE_SCAN_HEADSHOT.female.left,type:"video/mp4"})})}),e.jsx("div",{className:` w-[120px] overflow-hidden ${2===n?"opacity-100":"opacity-0 hidden"} `,children:e.jsx("video",{className:"h-full w-full object-contain border-none",muted:!0,loop:!0,autoPlay:!0,playsInline:!0,children:e.jsx("source",{src:i===a.GenderType.Male?a.FACE_SCAN_HEADSHOT.male.right:a.FACE_SCAN_HEADSHOT.female.right,type:"video/mp4"})})}),e.jsx("div",{className:` w-[120px] overflow-hidden ${3===n?"opacity-100":"opacity-0 hidden"} `,children:e.jsx("video",{className:"h-full w-full object-contain border-none",muted:!0,loop:!0,autoPlay:!0,playsInline:!0,children:e.jsx("source",{src:i===a.GenderType.Male?a.FACE_SCAN_HEADSHOT.male.smile:a.FACE_SCAN_HEADSHOT.female.smile,type:"video/mp4"})})})]})]})})}const u=({webcamRef:n,canvasRef:s})=>{const{isError:c,isSuccess:i,showLoader:u,hasError:g,resetScanState:f,showGuideCard:h,scanStage:m,videoLoading:p,setVideoLoading:S,videoError:y,setVideoError:x,gender:v,getButtonText:b,isScanning:w,startScan:F,isModelLoaded:C,config:j,onRetry:_}=t.useContext(a.ParamsContext),N=a.useLocalConfig(j),{setPreferredLanguage:E}=t.useContext(a.LanguageContext)||{};return t.useEffect(()=>{E?.(N?.language)},[N]),c?e.jsx(l,{resetScanState:_,config:N}):i?e.jsx("div",{className:"fixed z-[9] w-full h-full",style:{background:N?.style?.base?.backgroundColor},children:e.jsx(a.LoadingScreen,{url:N?.loader,loaderType:"black"})}):e.jsxs(e.Fragment,{children:[e.jsx("audio",{id:"audioElement",crossOrigin:"anonymous",preload:"auto",style:{position:"absolute",zIndex:-99999},src:void 0}),u&&!g&&e.jsx("div",{className:"fixed z-[9] w-full h-full",style:{background:N?.style?.base?.backgroundColor},children:e.jsx(a.LoadingScreen,{url:N?.loader,loaderType:"black"})}),e.jsx("div",{className:"h-full flex-col relative w-full flex justify-center items-center text-center rounded-t-[20px]",style:{background:N?.style?.base?.backgroundColor},children:e.jsx("div",{className:"flex-1 w-full max-w-md overflow-hidden",children:e.jsxs("div",{className:"w-full h-full",children:[e.jsx("video",{ref:n,autoPlay:!0,playsInline:!0,muted:!0,width:a.videoConstraints.width.ideal,height:a.videoConstraints.height.ideal,className:"w-full h-full object-cover fixed left-0 top-0 z-0",style:{transform:"scaleX(-1)"}}),e.jsx("canvas",{ref:s,width:a.videoConstraints.width.ideal,height:a.videoConstraints.height.ideal,style:{transform:"scaleX(-1)",opacity:"0"}})]})})}),!u&&g&&e.jsx(l,{loading:u,resetScanState:f,config:N}),h&&!g&&!u&&e.jsx(r.Drawer,{open:!0,className:"face-scan-small camera-draw",anchor:"bottom",onClose:(e,t)=>{},hideBackdrop:!0,children:e.jsx(d,{stage:m,videoLoading:p,setVideoLoading:S,videoError:y,setVideoError:x,gender:v,config:N,btnConfig:{label:b(),onClick:w?f:F,isDisabled:u||!C,icon:w?e.jsx(o.ArrowRight,{}):e.jsx(e.Fragment,{})}})})]})};exports.FaceScan=n=>{const{onScanSuccess:o,onScanError:r,onRetry:s,onScanStart:c,onUploadingStart:l,onUploadingEnd:d,onMeasurementSocketStart:g,onMeasurementSocketClose:f,config:h,isError:m,isSuccess:p}=n,S=n.userDetails??{email:"",gender:"male",deviceFocalLength:0,shopDomain:""},y=n.token??"",x=t.useRef(null),v=t.useRef(null),b=t.useRef(null),w=t.useRef([]),F=t.useRef(null),C=t.useRef(null),[j,_]=t.useState(!1),[N,E]=t.useState(!1),[k,T]=t.useState(!1),[I,R]=t.useState(!0),[M,D]=t.useState(!1),[P,L]=t.useState(!1),[A,O]=t.useState(!1),[z,$]=t.useState(a.generateUuid()),H=t.useMemo(()=>a.getSwanService(y),[y]),{email:V,gender:W,deviceFocalLength:K,shopDomain:U,callbackUrl:q}=S,[B,G]=t.useState([]),[J,X]=t.useState(a.videoConstraintsExact),Q=t.useMemo(()=>a.getPreferredMediaRecorderTypes(),[]),Y=a.getRecordingMimeType(B),Z=a.getRecordingExtension(Y);a.usePosthogPageview();const ee=t.useCallback(()=>{console.log("Stopping recording..."),b.current&&"recording"===b.current.state&&b.current.stop(),b.current=null},[]),te=async()=>{if(w.current){if(0===w.current.length)return console.error("No video data recorded"),O(!0),void _(!1);l?.(),_(!0);const e=new File(w.current,`${z}.${Z}`,{type:Y}),t=C.current??await a.getVideoFPS(e),n=e.size,o=(n/1048576).toFixed(2),r=Math.round(n/1e4),s={video_size_mb:parseFloat(o),video_size_bytes:n,video_fps:t||0,blob_count:w.current.length,estimated_duration_seconds:r},c=[{gender:W},{face_scan_id:z},{focal_length:`${K}`},{customer_store_url:U},{scan_type:"face_scan"}];q&&c.push({callback_url:q}),a.handleScanTimeCapture({eventName:`${U}/face_scan_meta_data`,faceScanId:z,email:V,data:JSON.stringify({metaData:c,videoData:s})});const i=Date.now();a.handleScanTimeCapture({eventName:`${U}/face_scan_upload_start`,faceScanId:z,email:V,startTime:i});try{const t=await H.fileUpload.faceScanFileUploader({file:e,arrayMetaData:c,objectKey:z,email:V,contentType:e.type}),n=Date.now(),o=n-i;a.handleScanTimeCapture({eventName:`${U}/face_scan_upload_complete`,faceScanId:z,email:V,completionTime:n,uploadDuration:o}),d?.(),console.log("✅ Upload successful",t),H.measurement.handlFaceScaneSocket({onPreopen:()=>{g?.(),a.handleWebSocketCapture({eventName:`${U}/webSocket`,faceScanID:z,connection:"pre_open",type:"faceScan_recommendation",email:V})},faceScanId:z,onOpen:()=>{a.handleWebSocketCapture({eventName:`${U}/webSocket`,faceScanID:z,connection:"open",type:"faceScan_recommendation",email:V}),console.log("websocket connect open")},onError:e=>{a.handleWebSocketCapture({eventName:`${U}/webSocket`,faceScanID:z,connection:"error",type:"faceScan_recommendation",email:V}),ue(e)},onSuccess:e=>{a.handleWebSocketCapture({eventName:`${U}/webSocket`,faceScanID:z,connection:"success",type:"faceScan_recommendation",email:V}),ge(e)},onClose:()=>{f?.(),console.log("websocket connect close"),a.handleWebSocketCapture({eventName:`${U}/webSocket`,faceScanID:z,connection:"close",type:"faceScan_recommendation",email:V})}})}catch(e){d?.(),ue(e)}}},ae=t.useCallback(()=>{const e=x.current?.srcObject;if(!e)return;const t=e.getVideoTracks()[0].getSettings(),a="number"==typeof J.width?J.width:720,n="number"==typeof J.height?J.height:1280,{width:o=a,height:r=n}=t,s=document.createElement("canvas"),c=s.getContext("2d");o>r?(s.width=r,s.height=o):(s.width=o,s.height=r);const i=document.createElement("video");i.srcObject=e,i.muted=!0,i.playsInline=!0,i.play();const l=()=>{o>r?(c?.save(),c?.translate(s.width,0),c?.rotate(Math.PI/2),c?.drawImage(i,0,0,r,o),c?.restore()):c?.drawImage(i,0,0,o,r),requestAnimationFrame(l)};l();C.current=30;const d=s.captureStream(30),u={videoBitsPerSecond:15e5};let g;B.length>0&&(u.mimeType=B[0]);try{g=new MediaRecorder(d,u)}catch(e){return console.error("MediaRecorder init failed:",e),O(!0),void _(!1)}w.current=[],g.ondataavailable=e=>{e?.data&&e.data.size>0&&w.current&&w.current.push(e.data)},g.onstop=()=>{console.log("Recording stopped, total blobs:",w?.current?.length),te()},g.start(100),b.current=g,console.log("Recording started successfully (portrait normalized)")},[B,te]),{faceScanDetector:ne,scanStage:oe,setScanStage:re,resetScan:se,isModelLoaded:ce,startScanSequence:ie}=i({faceScanId:z,shopDomain:U,onScanComplete:()=>{ee()},onModelReady:()=>{E(!0)},onStartRecording:()=>{console.log("Stage 0 completed - starting recording for stages 1-3"),ae()}}),le=t.useCallback(()=>{ce&&(c?.(),re(0),a.speechService.playAudio(`${a.voiceOverAssetsPath}face-scan-vos/Face-forward.mp3`),setTimeout(()=>{T(!0),setTimeout(()=>{ie()},200)},200))},[re,T,ie,ce,c]),de=t.useCallback(()=>{T(!1),R(!0),ee(),se(),s?.(),w.current=[],C.current=null,b.current&&(b.current=null),O(!1),re(-1),$(a.generateUuid()),x.current&&F.current&&(x.current.srcObject=F.current,console.log("Camera stream restored after reset"))},[re,T,R,ee,se,$]),ue=e=>{console.log(e,"ws error"),r?.(e),O(!0),_(!1),R(!1),a.handleScanTimeCapture({eventName:`${U}/faceScan`,faceScanId:z,status:"failed",email:V,data:JSON.stringify(e)})},ge=e=>{console.log(e,"ws success"),e&&"intermediate"===e?.resultType&&a.handleScanTimeCapture({eventName:`${U}/faceScan_success/intermediate`,faceScanId:z,status:"success",email:V,data:JSON.stringify(e)}),o?.(e)};t.useEffect(()=>{if(!m&&!p)return navigator.mediaDevices.getUserMedia&&navigator.mediaDevices.getUserMedia({video:J}).then(e=>{F.current=e,x.current&&(x.current.srcObject=e)}).catch(e=>{console.error("Error accessing webcam:",e),J!==a.videoConstraintsExact?O(!0):X(a.videoConstraintsFallback)}),()=>{F.current&&F.current.getTracks().forEach(e=>e.stop())}},[m,p,J]),t.useEffect(()=>{if(!N||!k)return;const e=setInterval(()=>{ne(x,v)},500);return()=>clearInterval(e)},[ne,N,k]);return console.log("Model ready:",N,"Has error:",A,"Is scanning:",k,"showLoader",j),t.useEffect(()=>{const e=a.getSupportedMediaRecorderTypes(Q,a.videoTypes);G(e)},[Q]),t.useEffect(()=>{re(-1)},[]),e.jsx(a.LanguageContextProvider,{children:e.jsx(a.ParamsContext.Provider,{value:{isError:m,isSuccess:p,showLoader:j,hasError:A,resetScanState:de,showGuideCard:I,scanStage:oe,videoLoading:M,setVideoLoading:D,videoError:P,setVideoError:L,gender:W,getButtonText:()=>k?a.LanguageKeys.reset:ce?a.LanguageKeys.start:a.LanguageKeys.loading,isScanning:k,startScan:le,isModelLoaded:ce,config:h,onRetry:s},children:e.jsx(u,{webcamRef:x,canvasRef:v})})})};
2
+ //# sourceMappingURL=FaceScan-DINxXMKv.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"FaceScan-DINxXMKv.js","sources":["../src/customHooks/useFaceScan.ts","../src/components/faceScan/FaceScanErrorScreen.tsx","../src/components/faceScan/FaceScanGuide.tsx","../src/components/faceScan/FaceScanStep.tsx","../src/components/faceScan/FaceScan.tsx"],"sourcesContent":["import {\n useState,\n useRef,\n useEffect,\n useCallback,\n useMemo,\n RefObject,\n} from \"react\";\n// 1. Use TYPE-ONLY imports. These are erased at build time and won't crash SSR.\nimport type { PoseDetector } from \"@tensorflow-models/pose-detection\";\nimport { posthog } from \"posthog-js\";\nimport speechService from \"../utils/service/speechService\";\nimport { videoConstraints, voiceOverAssetsPath } from \"../utils/constants\";\n\n/**\n * useFaceScan Hook with PostHog Analytics\n * Refactored for Zero-Config SSR/CSR Compatibility\n */\n\n// Debounce utility\nfunction debounce(fn: (...args: any) => void, delay: number) {\n let timer: number | NodeJS.Timeout;\n return (...args: any) => {\n if (timer) clearTimeout(timer);\n timer = setTimeout(() => fn(...args), delay);\n };\n}\n\n// Global Singleton Cache (Preserves state across re-renders)\nlet preloadedDetector: PoseDetector | null = null;\nlet isPreloading = false;\n\n// 2. Async Preloader with Dynamic Imports\nexport const preloadDetector = async (): Promise<PoseDetector | null> => {\n // Guard: Never run on server\n if (typeof window === \"undefined\") return null;\n if (preloadedDetector) return preloadedDetector;\n if (isPreloading) {\n // Simple wait mechanism if already loading\n while (isPreloading) {\n await new Promise(r => setTimeout(r, 100));\n if (preloadedDetector) return preloadedDetector;\n }\n }\n\n isPreloading = true;\n\n try {\n // DYNAMIC IMPORTS: Only fetch these heavy bundles in the browser\n const [tf, poseDetection] = await Promise.all([\n import(\"@tensorflow/tfjs\"),\n import(\"@tensorflow-models/pose-detection\")\n ]);\n\n // Initialize Backend\n await tf.setBackend(\"webgl\");\n await tf.ready();\n\n const detector = await poseDetection.createDetector(\n poseDetection.SupportedModels.BlazePose,\n {\n runtime: \"tfjs\",\n modelType: \"full\",\n }\n );\n\n // Dummy video warmup (Performance Optimization)\n // This runs the model once on a blank canvas to \"wake up\" the GPU\n // so the first user frame detects instantly.\n const dummyCanvas = document.createElement(\"canvas\");\n dummyCanvas.width = videoConstraints.width;\n dummyCanvas.height = videoConstraints.height;\n const ctx = dummyCanvas.getContext(\"2d\");\n if (ctx) {\n ctx.fillStyle = \"black\";\n ctx.fillRect(0, 0, dummyCanvas.width, dummyCanvas.height);\n try {\n await detector.estimatePoses(dummyCanvas);\n } catch (e) {\n console.warn(\"Warmup frame failed (non-fatal):\", e);\n }\n }\n\n preloadedDetector = detector;\n return detector;\n } catch (err) {\n console.error(\"Failed to preload detector:\", err);\n return null;\n } finally {\n isPreloading = false;\n }\n};\n\ntype Point = [number, number, number];\n\nfunction useFaceScan({\n faceScanId,\n onValidPose,\n onScanComplete,\n onModelReady,\n onStartRecording,\n shopDomain,\n}: {\n faceScanId: string;\n onValidPose?: () => void;\n onScanComplete?: () => void;\n onModelReady?: () => void;\n onStartRecording: () => void;\n shopDomain: string;\n}) {\n const detectorRef = useRef<PoseDetector | null>(null);\n const [scanStage, setScanStage] = useState(0);\n const [consecutiveValid, setConsecutiveValid] = useState(0);\n const [isModelLoaded, setIsModelLoaded] = useState(false);\n const [isScanningActive, setIsScanningActive] = useState(false);\n \n const utteranceRef = useRef<SpeechSynthesisUtterance | null>(null);\n const scanStageRef = useRef(scanStage);\n const consecutiveValidRef = useRef(consecutiveValid);\n const validityBufferRef = useRef<boolean[]>([]);\n const lastSpokenMessageRef = useRef(\"\");\n const lastSpokenTimeRef = useRef(0);\n const voiceEnabledRef = useRef(false);\n \n const VALIDITY_BUFFER_LENGTH = 10;\n const CONSECUTIVE_VALID_LENGTH = 2;\n const TOTAL_STAGES = 4;\n const POSE_TRACKING_INTERVAL = 4000;\n const lastPoseTrackingTimeRef = useRef(0);\n\n // Safe iOS detection (SSR safe)\n const isIOS = typeof navigator !== 'undefined' && (\n /iPad|iPhone|iPod/.test(navigator.userAgent) ||\n (navigator.platform === \"MacIntel\" && navigator.maxTouchPoints > 1)\n );\n\n const directionMessages = useMemo(\n () => [\n \"Face front\",\n \"Turn head fully left\",\n \"Turn head fully right\",\n \"Smile facing front\",\n ],\n []\n );\n\n const debouncedTrackPoseDetection = useMemo(\n () =>\n debounce((payload) => {\n posthog.capture(`${shopDomain}/face_scan_pose_detection`, payload);\n }, 4000),\n [shopDomain]\n );\n\n const trackPoseDetection = useCallback(\n (\n faceKeypoints: Point[],\n bodyKeypoints: Point[],\n stage: number,\n headValid: boolean,\n bodyInvalid: boolean,\n consecutiveValidCount: number\n ) => {\n const now = Date.now();\n if (now - lastPoseTrackingTimeRef.current < POSE_TRACKING_INTERVAL) {\n return;\n }\n\n lastPoseTrackingTimeRef.current = now;\n\n try {\n const nose = faceKeypoints[0] || [0, 0, 0];\n const leftEye = faceKeypoints[2] || [0, 0, 0];\n const rightEye = faceKeypoints[5] || [0, 0, 0];\n const leftMouth = faceKeypoints[9] || [0, 0, 0];\n const rightMouth = faceKeypoints[10] || [0, 0, 0];\n\n const faceKeypointScores = faceKeypoints\n .map((kp: Point) => kp[2])\n .filter((score: number) => score > 0);\n const avgFaceScore =\n faceKeypointScores.length > 0\n ? faceKeypointScores.reduce((a, b) => a + b, 0) /\n faceKeypointScores.length\n : 0;\n\n const bodyKeypointScores = bodyKeypoints\n .map((kp: Point) => kp[2])\n .filter((score: number) => score > 0);\n const avgBodyScore =\n bodyKeypointScores.length > 0\n ? bodyKeypointScores.reduce((a: number, b: number) => a + b, 0) /\n bodyKeypointScores.length\n : 0;\n\n const eyeDistance = Math.abs(leftEye[0] - rightEye[0]);\n const noseToRight = nose[0] - rightEye[0];\n const noseToLeft = leftEye[0] - nose[0];\n\n let expectedPosition = \"frontal\";\n if (stage === 1) expectedPosition = \"left\";\n else if (stage === 2) expectedPosition = \"right\";\n else if (stage === 3) expectedPosition = \"frontal_smile\";\n\n const payload = {\n faceScanId,\n stage,\n timestamp: new Date().toISOString(),\n data: JSON.stringify({\n headValid,\n bodyInvalid,\n consecutiveValid: consecutiveValidCount,\n avgFaceScore: parseFloat(avgFaceScore.toFixed(3)),\n avgBodyScore: parseFloat(avgBodyScore.toFixed(3)),\n faceKeypointsDetected: faceKeypointScores.length,\n bodyKeypointsDetected: bodyKeypointScores.length,\n eyeDistance: parseFloat(eyeDistance.toFixed(2)),\n noseToRightRatio: parseFloat((noseToRight / eyeDistance).toFixed(3)),\n noseToLeftRatio: parseFloat((noseToLeft / eyeDistance).toFixed(3)),\n noseScore: parseFloat(nose[2].toFixed(3)),\n leftEyeScore: parseFloat(leftEye[2].toFixed(3)),\n rightEyeScore: parseFloat(rightEye[2].toFixed(3)),\n leftMouthScore: parseFloat(leftMouth[2].toFixed(3)),\n rightMouthScore: parseFloat(rightMouth[2].toFixed(3)),\n expectedPosition,\n }),\n };\n debouncedTrackPoseDetection(payload);\n } catch (error) {\n console.warn(\"Failed to track pose detection:\", error);\n }\n },\n [faceScanId, debouncedTrackPoseDetection]\n );\n\n const enableVoiceCommands = () => {\n voiceEnabledRef.current = true;\n if (typeof window !== 'undefined' && isIOS && \"speechSynthesis\" in window) {\n const silentUtterance = new SpeechSynthesisUtterance(\"\");\n silentUtterance.volume = 0;\n speechSynthesis.speak(silentUtterance);\n }\n };\n\n const startScanSequence = () => {\n posthog.capture(`${shopDomain}/face_scan_detection_started`, {\n faceScanId,\n stage: scanStageRef.current,\n timestamp: new Date().toISOString(),\n stageName: directionMessages[scanStageRef.current],\n });\n\n setTimeout(() => {\n setIsScanningActive(true);\n }, 1500);\n };\n\n const isHeadInPosition = (stage: number, keypoints: Point[]) => {\n // ... logic unchanged ...\n const nose = 0;\n const leftEye = 2;\n const rightEye = 5;\n const leftMouth = 9;\n const rightMouth = 10;\n\n const minScore = 0.2;\n if (\n !keypoints[nose] || !keypoints[leftEye] || !keypoints[rightEye] ||\n keypoints[nose][2] < minScore ||\n keypoints[leftEye][2] < minScore ||\n keypoints[rightEye][2] < minScore\n ) {\n return false;\n }\n\n const eyesToMouthVerticalDistance =\n 0.5 * (keypoints[rightMouth][1] + keypoints[leftMouth][1]) -\n 0.5 * (keypoints[rightEye][1] + keypoints[leftEye][1]);\n\n const faceTiltedness =\n (0.5 * (keypoints[rightMouth][1] + keypoints[leftMouth][1]) -\n keypoints[nose][1]) /\n eyesToMouthVerticalDistance;\n\n const faceIsTilted = faceTiltedness > 0.7 || faceTiltedness < 0.3;\n\n const faceIsVertical =\n Math.abs(keypoints[leftEye][1] - keypoints[rightEye][1]) <\n 0.5 * eyesToMouthVerticalDistance &&\n Math.abs(keypoints[leftMouth][1] - keypoints[rightMouth][1]) <\n 0.5 * eyesToMouthVerticalDistance;\n\n const eyeDistance = keypoints[leftEye][0] - keypoints[rightEye][0];\n const noseToRight = keypoints[nose][0] - keypoints[rightEye][0];\n const noseToLeft = keypoints[leftEye][0] - keypoints[nose][0];\n\n const frontalFace =\n noseToRight > 0.4 * eyeDistance && noseToRight < 0.6 * eyeDistance;\n const fullLeft = noseToRight > 0.85 * eyeDistance;\n const fullRight = noseToLeft > 0.85 * eyeDistance;\n\n switch (stage) {\n case 0:\n return !faceIsTilted && faceIsVertical && frontalFace;\n case 1:\n return !faceIsTilted && faceIsVertical && fullLeft;\n case 2:\n return !faceIsTilted && faceIsVertical && fullRight;\n case 3:\n return !faceIsTilted && faceIsVertical && frontalFace;\n default:\n return false;\n }\n };\n\n const faceScanDetector = async (\n webcamRef: RefObject<HTMLVideoElement | null>,\n canvasRef: RefObject<HTMLCanvasElement | null>\n ) => {\n if (\n !detectorRef.current ||\n !webcamRef.current ||\n !isModelLoaded ||\n !isScanningActive\n )\n return;\n\n // Safety: ensure video is actually playing with data\n if (webcamRef.current.readyState < 2) return;\n\n try {\n const poses = await detectorRef.current.estimatePoses(webcamRef.current);\n if (poses.length > 0) {\n const faceKeypoints: Point[] = [];\n const bodyKeypoints: Point[] = [];\n poses[0].keypoints.forEach((landmark, idx) => {\n const score = landmark.score ?? 0;\n const point: Point = [landmark.x ?? -1, landmark.y ?? -1, score];\n if (idx <= 10) faceKeypoints.push(point);\n else bodyKeypoints.push(point);\n });\n\n // Optional drawing\n if (canvasRef?.current) {\n const ctx = canvasRef.current.getContext(\"2d\");\n const { width, height } = canvasRef.current;\n if (ctx) {\n ctx.clearRect(0, 0, width, height);\n\n const progress = Math.min(\n consecutiveValidRef.current / CONSECUTIVE_VALID_LENGTH,\n 1\n );\n const radius = height * 0.35;\n ctx.beginPath();\n ctx.strokeStyle = \"#00ff00\";\n ctx.lineWidth = 6;\n ctx.arc(\n width / 2,\n height / 2,\n radius + 10,\n -Math.PI / 2,\n -Math.PI / 2 + progress * 2 * Math.PI\n );\n ctx.stroke();\n }\n }\n\n const headValid = isHeadInPosition(scanStageRef.current, faceKeypoints);\n const bodyInvalid =\n bodyKeypoints.slice(2).filter((p) => p[2] > 0.7).length <= 2;\n\n validityBufferRef.current.push(headValid && bodyInvalid);\n if (validityBufferRef.current.length > VALIDITY_BUFFER_LENGTH) {\n validityBufferRef.current.shift();\n }\n\n const validCount = validityBufferRef.current.filter(Boolean).length;\n if (validCount >= 2) {\n const nextValid = consecutiveValidRef.current + 1;\n setConsecutiveValid(nextValid);\n } else {\n setConsecutiveValid(Math.max(0, consecutiveValidRef.current - 1));\n }\n\n if (consecutiveValidRef.current >= CONSECUTIVE_VALID_LENGTH) {\n onValidPose?.();\n\n const nextStage = scanStageRef.current + 1;\n setConsecutiveValid(0);\n validityBufferRef.current = [];\n setScanStage(nextStage);\n setIsScanningActive(false);\n\n (async () => {\n await speechService.playAudio(\n `${voiceOverAssetsPath}face-scan-vos/Sound-effect.mp3`\n );\n posthog.capture(\n `${shopDomain}/face_scan_detection_stage_completed`,\n {\n faceScanId,\n completedStage: scanStageRef.current,\n nextStage,\n timestamp: new Date().toISOString(),\n totalStages: TOTAL_STAGES,\n stageName: directionMessages[scanStageRef.current],\n }\n );\n \n if (nextStage === 1 && onStartRecording) {\n onStartRecording();\n }\n if (nextStage >= TOTAL_STAGES) {\n onScanComplete?.();\n } else {\n let nextSound = null;\n switch (nextStage) {\n case 1: nextSound = \"Left.mp3\"; break;\n case 2: nextSound = \"Right.mp3\"; break;\n case 3: nextSound = \"Smile.mp3\"; break;\n }\n if (nextSound) {\n await speechService.playAudio(\n `${voiceOverAssetsPath}face-scan-vos/${nextSound}`\n );\n }\n startScanSequence();\n }\n })();\n }\n\n trackPoseDetection(\n faceKeypoints,\n bodyKeypoints,\n scanStageRef.current,\n headValid,\n bodyInvalid,\n consecutiveValidRef.current\n );\n }\n } catch (err) {\n console.error(\"Pose detection error:\", err);\n }\n };\n\n const initializeModels = async () => {\n // 3. Browser Guard\n if (typeof window === 'undefined') return;\n\n try {\n console.log(\"Initializing Face Scan...\");\n // Calls our safe, dynamic preloader\n const detector = await preloadDetector();\n \n if (detector) {\n detectorRef.current = detector;\n console.log(\"Face scan model loaded successfully\");\n setIsModelLoaded(true);\n onModelReady?.();\n }\n } catch (err) {\n console.error(\"Error initializing face scan:\", err);\n }\n };\n\n const disposeModelAndTf = async () => {\n // Optional: We might NOT want to dispose if we want to keep the cache \n // for re-mounting. But if you must dispose:\n try {\n setIsModelLoaded(false);\n // NOTE: We generally don't dispose the detector if we want to reuse it \n // via the 'preloadedDetector' global variable.\n // If you dispose here, make sure to set 'preloadedDetector = null' too.\n /* if (detectorRef.current) {\n await detectorRef.current.dispose();\n detectorRef.current = null;\n preloadedDetector = null; // Clear global cache if disposing\n }\n */\n } catch (err) {\n console.error(\"Error disposing resources:\", err);\n } finally {\n // Reset state\n setScanStage(0);\n setConsecutiveValid(0);\n scanStageRef.current = 0;\n consecutiveValidRef.current = 0;\n validityBufferRef.current = [];\n }\n };\n\n const resetScan = () => {\n posthog.capture(`${shopDomain}/face_scan_reset`, {\n faceScanId,\n stage: scanStageRef.current,\n timestamp: new Date().toISOString(),\n stageName: directionMessages[scanStageRef.current],\n consecutiveValid: consecutiveValidRef.current,\n });\n\n setScanStage(0);\n setConsecutiveValid(0);\n setIsScanningActive(false);\n scanStageRef.current = 0;\n consecutiveValidRef.current = 0;\n validityBufferRef.current = [];\n lastSpokenMessageRef.current = \"\";\n lastSpokenTimeRef.current = 0;\n voiceEnabledRef.current = false;\n if (utteranceRef.current) {\n speechSynthesis.cancel();\n utteranceRef.current = null;\n }\n };\n\n useEffect(() => {\n initializeModels();\n return () => {\n disposeModelAndTf();\n };\n }, []);\n\n useEffect(() => {\n scanStageRef.current = scanStage;\n }, [scanStage]);\n\n useEffect(() => {\n consecutiveValidRef.current = consecutiveValid;\n }, [consecutiveValid]);\n\n return {\n faceScanDetector,\n scanStage,\n setScanStage,\n consecutiveValid,\n isModelLoaded,\n resetScan,\n enableVoiceCommands,\n startScanSequence,\n isScanningActive,\n };\n}\n\nexport default useFaceScan;","import { ArrowRight } from \"lucide-react\";\nimport Header from \"../Header\";\nimport SpecificButton from \"../../atoms/specificButton/SpecificButton\";\nimport { useContext } from \"react\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { Config } from \"../../types/interfaces\";\n\nfunction FaceScanErrorScreen({ resetScanState, loading, config }: { resetScanState?: () => void; loading?: boolean; config?: Config }) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\n\treturn (\n\t\t<div className=\"fixed top-[0] left-[0] z-[999] flex justify-center items-center w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t<div className=\"flex flex-col w-full items-center p-[1rem] rounded-lg \">\n\t\t\t\t<Header noTitle resolvedConfig={config} />\n\t\t\t\t<div className=\"flex flex-col items-center w-full\">\n\t\t\t\t\t<h2\n\t\t\t\t\t\tclassName=\"text-xl font-semibold text-gray-800 \"\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.scanFailed)}\n\t\t\t\t\t</h2>\n\t\t\t\t\t<p\n\t\t\t\t\t\tclassName=\"mb-[1.5rem]\"\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\tcolor: config?.style?.subheading?.subheadingColor || \"#4b5563\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.clickToResetScan)}\n\t\t\t\t\t</p>\n\t\t\t\t\t<SpecificButton\n\t\t\t\t\t\tdisabled={loading}\n\t\t\t\t\t\tclassName=\"w-full h-[45px] shadow-[0px_1px_2px_0px_#00000040] text-[16px]\"\n\t\t\t\t\t\tbuttonText={translate?.(LanguageKeys.resetScan)}\n\t\t\t\t\t\tpostfixIcon={<ArrowRight />}\n\t\t\t\t\t\tbuttonFunc={() => resetScanState?.()}\n\t\t\t\t\t\tresolvedConfig={config}\n\t\t\t\t\t/>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t</div>\n\t);\n}\n\nexport default FaceScanErrorScreen;\n","import { FaceScanGuideProps } from \"../../types/interfaces\";\nimport Header from \"../Header\";\nimport { GenderType } from \"../../utils/enums\";\nimport { directionMessages, FACE_SCAN_HEADSHOT, glassesOffVideo, maleGlassesOffVideo } from \"../../utils/constants\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { useContext } from \"react\";\nimport SpecificButton from \"../../atoms/specificButton/SpecificButton\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\n\nfunction FaceScanGuide({ stage, videoLoading, setVideoLoading, videoError, setVideoError, gender, config, btnConfig }: FaceScanGuideProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\n\n\n\tif (stage === -1) {\n\t\treturn (\n\t\t\t<div className=\"text-center p-[16px] w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t\t<div className=\"w-full\">\n\t\t\t\t\t<Header noTitle resolvedConfig={config} />\n\t\t\t\t\t<h2\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.faceVisible)}\n\t\t\t\t\t</h2>\n\t\t\t\t\t{videoLoading && (\n\t\t\t\t\t\t<div\n\t\t\t\t\t\t\tclassName=\"mb-4 flex items-center justify-center\"\n\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\t\tcolor: config?.style?.subheading?.subheadingColor || \"#4b5563\",\n\t\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t>\n\t\t\t\t\t\t\t{translate?.(LanguageKeys.loadingVideo)}\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t\t{videoError && (\n\t\t\t\t\t\t<div\n\t\t\t\t\t\t\tclassName=\"mb-4 text-red-600\"\n\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\t\tcolor: \"#ff0000\",\n\t\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t>\n\t\t\t\t\t\t\t{translate?.(LanguageKeys.videoLoadFailed)}\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t\t{!videoError && (\n\t\t\t\t\t\t<div className={` w-[100px] mx-auto`}>\n\t\t\t\t\t\t\t<video\n\t\t\t\t\t\t\t\tsrc={gender === GenderType.Male ? maleGlassesOffVideo : glassesOffVideo}\n\t\t\t\t\t\t\t\tautoPlay\n\t\t\t\t\t\t\t\tloop\n\t\t\t\t\t\t\t\tcontrols={false}\n\t\t\t\t\t\t\t\tmuted\n\t\t\t\t\t\t\t\tplaysInline\n\t\t\t\t\t\t\t\tclassName=\"h-full w-full object-contain border-none\"\n\t\t\t\t\t\t\t\tonCanPlay={() => setVideoLoading(false)}\n\t\t\t\t\t\t\t\tonLoadStart={() => setVideoLoading(true)}\n\t\t\t\t\t\t\t\tonError={() => setVideoError(true)}\n\t\t\t\t\t\t\t\taria-label=\"Instructional video: Please remove your glasses before starting the scan.\"\n\t\t\t\t\t\t\t/>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t</div>\n\t\t\t\t<div className=\"flex justify-center w-full p-[1rem]\">\n\t\t\t\t\t<SpecificButton\n\t\t\t\t\t\tdisabled={btnConfig.isDisabled}\n\t\t\t\t\t\tclassName=\"!w-[60px] !h-[35px] !py-0 !px-0\"\n\t\t\t\t\t\tbuttonText={translate?.(btnConfig.label)}\n\t\t\t\t\t\tpostfixIcon={btnConfig?.icon}\n\t\t\t\t\t\tbuttonFunc={btnConfig.onClick}\n\t\t\t\t\t\tresolvedConfig={config}\n\t\t\t\t\t/>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t);\n\t}\n\n\treturn (\n\t\t<div className=\"text-center p-[16px] w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t<div className=\"w-full\">\n\t\t\t\t<Header noTitle resolvedConfig={config} />\n\t\t\t\t<h3\n\t\t\t\t\tclassName=\"mb-[0.5rem]\"\n\t\t\t\t\tstyle={{\n\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t{translate?.(directionMessages[stage])}\n\t\t\t\t</h3>\n\t\t\t\t{/* <p>We'll guide you to take 6 selfies </p> */}\n\t\t\t\t<div className=\"max-w-[400px] justify-center gap-1 mx-auto flex items-center\">\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 0 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.forward : FACE_SCAN_HEADSHOT.female.forward} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 1 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.left : FACE_SCAN_HEADSHOT.female.left} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 2 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.right : FACE_SCAN_HEADSHOT.female.right} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 3 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.smile : FACE_SCAN_HEADSHOT.female.smile} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t</div>\n\t);\n}\n\nexport default FaceScanGuide;\n","import React, { useContext, useEffect } from \"react\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\nimport FaceScanErrorScreen from \"./FaceScanErrorScreen\";\nimport LoadingScreen from \"../LoadingScreen\";\nimport { videoConstraints } from \"../../utils/constants\";\nimport { Drawer } from \"@mui/material\";\nimport FaceScanGuide from \"./FaceScanGuide\";\nimport { ArrowRight } from \"lucide-react\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { useLocalConfig } from \"../../config/useLocalConfig\";\ninterface FaceScanStepProps {\n\twebcamRef: React.RefObject<HTMLVideoElement | null>;\n\tcanvasRef: React.RefObject<HTMLCanvasElement | null>;\n}\n\nconst FaceScanStep: React.FC<FaceScanStepProps> = ({ webcamRef, canvasRef }) => {\n\tconst {\n\t\tisError,\n\t\tisSuccess,\n\t\tshowLoader,\n\t\thasError,\n\t\tresetScanState,\n\t\tshowGuideCard,\n\t\tscanStage,\n\t\tvideoLoading,\n\t\tsetVideoLoading,\n\t\tvideoError,\n\t\tsetVideoError,\n\t\tgender,\n\t\tgetButtonText,\n\t\tisScanning,\n\t\tstartScan,\n\t\tisModelLoaded,\n\t\tconfig,\n\t\tonRetry\n\t} = useContext(ParamsContext);\n\tconst resolvedConfig = useLocalConfig(config);\n\n\tconst { setPreferredLanguage } = useContext(LanguageContext) || {};\n\tuseEffect(() => {\n\t\tsetPreferredLanguage?.(resolvedConfig?.language);\n\t}, [resolvedConfig]);\n\tif (isError) {\n\t\treturn <FaceScanErrorScreen resetScanState={onRetry} config={resolvedConfig} />;\n\t}\n\tif (isSuccess) {\n\t\treturn (\n\t\t\t<div className=\"fixed z-[9] w-full h-full\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t<LoadingScreen url={resolvedConfig?.loader} loaderType=\"black\" />\n\t\t\t</div>\n\t\t);\n\t}\n\n\treturn (\n\t\t<>\n\t\t\t<audio id=\"audioElement\" crossOrigin=\"anonymous\" preload=\"auto\" style={{ position: \"absolute\", zIndex: -99999 }} src={undefined} />\n\n\t\t\t{/* Error overlay */}\n\t\t\t{showLoader && !hasError && (\n\t\t\t\t<div className=\"fixed z-[9] w-full h-full\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t\t{/* <Asset genderType={gender} /> */}\n\t\t\t\t\t<LoadingScreen url={resolvedConfig?.loader} loaderType=\"black\" />\n\t\t\t\t</div>\n\t\t\t)}\n\n\t\t\t{/* Always show camera view */}\n\t\t\t<div className=\"h-full flex-col relative w-full flex justify-center items-center text-center rounded-t-[20px]\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t<div className=\"flex-1 w-full max-w-md overflow-hidden\">\n\t\t\t\t\t<div className=\"w-full h-full\">\n\t\t\t\t\t\t<video\n\t\t\t\t\t\t\tref={webcamRef}\n\t\t\t\t\t\t\tautoPlay\n\t\t\t\t\t\t\tplaysInline\n\t\t\t\t\t\t\tmuted\n\t\t\t\t\t\t\twidth={videoConstraints.width.ideal}\n\t\t\t\t\t\t\theight={videoConstraints.height.ideal}\n\t\t\t\t\t\t\tclassName=\"w-full h-full object-cover fixed left-0 top-0 z-0\"\n\t\t\t\t\t\t\tstyle={{ transform: \"scaleX(-1)\" }}\n\t\t\t\t\t\t/>\n\t\t\t\t\t\t<canvas ref={canvasRef} width={videoConstraints.width.ideal} height={videoConstraints.height.ideal} style={{ transform: \"scaleX(-1)\", opacity: \"0\" }} />\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\n\t\t\t{/* Error overlay */}\n\t\t\t{!showLoader && hasError && <FaceScanErrorScreen loading={showLoader} resetScanState={resetScanState} config={resolvedConfig} />}\n\n\t\t\t{/* Scan guide drawer - only show when scanning */}\n\t\t\t{showGuideCard && !hasError && !showLoader && (\n\t\t\t\t<Drawer\n\t\t\t\t\topen\n\t\t\t\t\tclassName=\"face-scan-small camera-draw\"\n\t\t\t\t\tanchor=\"bottom\"\n\t\t\t\t\tonClose={(event, reason) => {\n\t\t\t\t\t\tif (reason === \"backdropClick\") {\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t}\n\t\t\t\t\t}}\n\t\t\t\t\thideBackdrop\n\t\t\t\t>\n\t\t\t\t\t<FaceScanGuide\n\t\t\t\t\t\tstage={scanStage}\n\t\t\t\t\t\tvideoLoading={videoLoading}\n\t\t\t\t\t\tsetVideoLoading={setVideoLoading}\n\t\t\t\t\t\tvideoError={videoError}\n\t\t\t\t\t\tsetVideoError={setVideoError}\n\t\t\t\t\t\tgender={gender}\n\t\t\t\t\t\tconfig={resolvedConfig}\n\t\t\t\t\t\tbtnConfig={{\n\t\t\t\t\t\t\tlabel: getButtonText(),\n\t\t\t\t\t\t\tonClick: isScanning ? resetScanState : startScan,\n\t\t\t\t\t\t\tisDisabled: showLoader || !isModelLoaded,\n\t\t\t\t\t\t\ticon: isScanning ? <ArrowRight /> : <></>,\n\t\t\t\t\t\t}}\n\t\t\t\t\t/>\n\t\t\t\t</Drawer>\n\t\t\t)}\n\t\t</>\n\t);\n};\n\nexport default FaceScanStep;\n","\"use client\";\nimport { useEffect, useRef, useState, useCallback, useMemo } from \"react\";\nimport { FaceScanMetaData, FaceScanProps } from \"../../types/interfaces\";\nimport {\n\tgenerateUuid,\n\tgetPreferredMediaRecorderTypes,\n\tgetRecordingExtension,\n\tgetRecordingMimeType,\n\tgetSupportedMediaRecorderTypes,\n\tgetVideoFPS,\n\thandleScanTimeCapture,\n\thandleWebSocketCapture,\n} from \"../../utils/utils\";\nimport { videoConstraintsExact, videoConstraintsFallback, videoTypes, voiceOverAssetsPath } from \"../../utils/constants\";\nimport { getSwanService } from \"../../utils/service/swanService\";\nimport speechService from \"../../utils/service/speechService\";\nimport useFaceScan from \"../../customHooks/useFaceScan\";\nimport usePosthogPageview from \"../../customHooks/usePosthogPageview\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport LanguageContextProvider from \"../../utils/context/languageContext\";\nimport FaceScanStep from \"./FaceScanStep\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\n\nexport const FaceScan: React.FC<FaceScanProps> = (props) => {\n\tconst {\n\t\tonScanSuccess,\n\t\tonScanError,\n\t\tonRetry,\n\t\tonScanStart,\n\t\tonUploadingStart,\n\t\tonUploadingEnd,\n\t\tonMeasurementSocketStart,\n\t\tonMeasurementSocketClose,\n\t\tconfig,\n\t\tisError,\n\t\tisSuccess,\n\t} = props;\n\tconst userDetails: FaceScanMetaData = props.userDetails ?? {\n\t\temail: \"\",\n\t\tgender: \"male\",\n\t\tdeviceFocalLength: 0,\n\t\tshopDomain: \"\",\n\t};\n\tconst token = props.token ?? \"\";\n\tconst webcamRef = useRef<HTMLVideoElement | null>(null);\n\tconst canvasRef = useRef<HTMLCanvasElement | null>(null);\n\tconst mediaRecorderRef = useRef<MediaRecorder | null>(null);\n\tconst recordedBlobsRef = useRef<Blob[] | null>([]);\n\tconst streamRef = useRef<MediaStream | null>(null);\n\tconst recordedFpsRef = useRef<number | null>(null);\n\tconst [showLoader, setShowLoader] = useState(false);\n\tconst [modelReady, setModelReady] = useState(false);\n\tconst [isScanning, setIsScanning] = useState(false);\n\tconst [showGuideCard, setShowGuideCard] = useState(true);\n\tconst [videoLoading, setVideoLoading] = useState(false);\n\tconst [videoError, setVideoError] = useState(false);\n\tconst [hasError, setHasError] = useState(false);\n\tconst [faceScanId, setFaceScanId] = useState(generateUuid());\n\tconst swan = useMemo(() => getSwanService(token), [token]);\n\tconst { email, gender, deviceFocalLength, shopDomain, callbackUrl } = userDetails;\n\tconst [supportedTypes, setSupportedTypes] = useState<string[]>([]);\n\tconst [currentVideoConstraints, setCurrentVideoConstraints] = useState<MediaTrackConstraints>(videoConstraintsExact);\n\tconst preferredVideoTypes = useMemo(() => getPreferredMediaRecorderTypes(), []);\n\tconst recordingMimeType = getRecordingMimeType(supportedTypes);\n\tconst recordingExtension = getRecordingExtension(recordingMimeType);\n\tusePosthogPageview();\n\n\tconst stopRecording = useCallback(() => {\n\t\tconsole.log(\"Stopping recording...\");\n\t\tif (mediaRecorderRef.current && mediaRecorderRef.current.state === \"recording\") {\n\t\t\tmediaRecorderRef.current.stop();\n\t\t}\n\t\tmediaRecorderRef.current = null;\n\t}, []);\n\n\tconst uploadFinalVideo = async () => {\n\t\tif (recordedBlobsRef.current) {\n\t\t\tif (recordedBlobsRef.current.length === 0) {\n\t\t\t\tconsole.error(\"No video data recorded\");\n\t\t\t\tsetHasError(true);\n\t\t\t\tsetShowLoader(false);\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tonUploadingStart?.();\n\t\t\tsetShowLoader(true);\n\t\t\tconst videoFile = new File(recordedBlobsRef.current, `${faceScanId}.${recordingExtension}`, {\n\t\t\t\ttype: recordingMimeType,\n\t\t\t});\n\t\t\t// Use FPS captured at recording time, fallback to calculation if not available\n\t\t\tconst videoFps = recordedFpsRef.current ?? (await getVideoFPS(videoFile));\n\n\t\t\tconst fileSize = videoFile.size;\n\t\t\tconst fileSizeMB = (fileSize / (1024 * 1024)).toFixed(2);\n\t\t\tconst estimatedDuration = Math.round(fileSize / 10000);\n\t\t\tconst videoData = {\n\t\t\t\tvideo_size_mb: parseFloat(fileSizeMB),\n\t\t\t\tvideo_size_bytes: fileSize,\n\t\t\t\tvideo_fps: videoFps || 0,\n\t\t\t\tblob_count: recordedBlobsRef.current.length,\n\t\t\t\testimated_duration_seconds: estimatedDuration,\n\t\t\t};\n\n\t\t\tconst metaData: any[] = [\n\t\t\t\t{ gender: gender },\n\t\t\t\t{ face_scan_id: faceScanId },\n\t\t\t\t{\n\t\t\t\t\tfocal_length: `${deviceFocalLength}`,\n\t\t\t\t},\n\t\t\t\t{ customer_store_url: shopDomain },\n\t\t\t\t{ scan_type: \"face_scan\" },\n\t\t\t];\n\t\t\tif (callbackUrl) {\n\t\t\t\tmetaData.push({ callback_url: callbackUrl });\n\t\t\t}\n\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/face_scan_meta_data`,\n\t\t\t\tfaceScanId,\n\t\t\t\temail,\n\t\t\t\tdata: JSON.stringify({ metaData, videoData }),\n\t\t\t});\n\n\t\t\tconst uploadStartTime = Date.now();\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/face_scan_upload_start`,\n\t\t\t\tfaceScanId,\n\t\t\t\temail,\n\t\t\t\tstartTime: uploadStartTime,\n\t\t\t});\n\n\t\t\ttry {\n\t\t\t\tconst res = await swan.fileUpload.faceScanFileUploader({\n\t\t\t\t\tfile: videoFile,\n\t\t\t\t\tarrayMetaData: metaData,\n\t\t\t\t\tobjectKey: faceScanId,\n\t\t\t\t\temail,\n\t\t\t\t\tcontentType: videoFile.type,\n\t\t\t\t});\n\n\t\t\t\tconst uploadEndTime = Date.now();\n\t\t\t\tconst uploadDuration = uploadEndTime - uploadStartTime;\n\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: `${shopDomain}/face_scan_upload_complete`,\n\t\t\t\t\tfaceScanId,\n\t\t\t\t\temail,\n\t\t\t\t\tcompletionTime: uploadEndTime,\n\t\t\t\t\tuploadDuration,\n\t\t\t\t});\n\t\t\t\tonUploadingEnd?.();\n\t\t\t\tconsole.log(\"✅ Upload successful\", res);\n\t\t\t\tswan.measurement.handlFaceScaneSocket({\n\t\t\t\t\tonPreopen: () => {\n\t\t\t\t\t\tonMeasurementSocketStart?.();\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"pre_open\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t},\n\t\t\t\t\tfaceScanId,\n\t\t\t\t\tonOpen: () => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"open\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tconsole.log(\"websocket connect open\");\n\t\t\t\t\t},\n\t\t\t\t\tonError: (err) => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"error\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tonError(err);\n\t\t\t\t\t},\n\t\t\t\t\tonSuccess: (data) => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"success\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tonSuccess(data);\n\t\t\t\t\t},\n\t\t\t\t\tonClose: () => {\n\t\t\t\t\t\tonMeasurementSocketClose?.();\n\t\t\t\t\t\tconsole.log(\"websocket connect close\");\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"close\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t},\n\t\t\t\t});\n\t\t\t} catch (error) {\n\t\t\t\tonUploadingEnd?.();\n\t\t\t\tonError(error);\n\t\t\t}\n\t\t}\n\t};\n\n\tconst startRecording = useCallback(() => {\n\t\tconst stream = webcamRef.current?.srcObject as MediaStream | null;\n\t\tif (!stream) return;\n\n\t\t// Create a canvas to normalize orientation\n\t\tconst videoTrack = stream.getVideoTracks()[0];\n\t\tconst settings = videoTrack.getSettings();\n\t\tconst defaultWidth = typeof currentVideoConstraints.width === \"number\" ? currentVideoConstraints.width : 720;\n\t\tconst defaultHeight = typeof currentVideoConstraints.height === \"number\" ? currentVideoConstraints.height : 1280;\n\t\tconst { width = defaultWidth, height = defaultHeight } = settings;\n\n\t\tconst canvas = document.createElement(\"canvas\");\n\t\tconst ctx = canvas.getContext(\"2d\");\n\n\t\t// Always force portrait (swap width/height if landscape)\n\t\tif (width > height) {\n\t\t\tcanvas.width = height;\n\t\t\tcanvas.height = width;\n\t\t} else {\n\t\t\tcanvas.width = width;\n\t\t\tcanvas.height = height;\n\t\t}\n\n\t\tconst videoEl = document.createElement(\"video\");\n\t\tvideoEl.srcObject = stream;\n\t\tvideoEl.muted = true;\n\t\tvideoEl.playsInline = true;\n\t\tvideoEl.play();\n\n\t\t// Draw video frames onto canvas\n\t\tconst drawFrame = () => {\n\t\t\t// Rotate if camera gives landscape frames\n\t\t\tif (width > height) {\n\t\t\t\tctx?.save();\n\t\t\t\tctx?.translate(canvas.width, 0);\n\t\t\t\tctx?.rotate(Math.PI / 2);\n\t\t\t\tctx?.drawImage(videoEl, 0, 0, height, width);\n\t\t\t\tctx?.restore();\n\t\t\t} else {\n\t\t\t\tctx?.drawImage(videoEl, 0, 0, width, height);\n\t\t\t}\n\t\t\trequestAnimationFrame(drawFrame);\n\t\t};\n\t\tdrawFrame();\n\n\t\tconst recordingFps = 30; // Canvas capture stream FPS\n\t\trecordedFpsRef.current = recordingFps; // Store FPS at recording time\n\t\tconst canvasStream = canvas.captureStream(recordingFps);\n\t\tconst mediaRecorderOptions: MediaRecorderOptions = {\n\t\t\tvideoBitsPerSecond: 1_500_000,\n\t\t};\n\t\tif (supportedTypes.length > 0) {\n\t\t\tmediaRecorderOptions.mimeType = supportedTypes[0];\n\t\t}\n\t\tlet mediaRecorder;\n\n\t\ttry {\n\t\t\tmediaRecorder = new MediaRecorder(canvasStream, mediaRecorderOptions);\n\t\t} catch (e) {\n\t\t\tconsole.error(\"MediaRecorder init failed:\", e);\n\t\t\tsetHasError(true);\n\t\t\tsetShowLoader(false);\n\t\t\treturn;\n\t\t}\n\n\t\trecordedBlobsRef.current = [];\n\t\tmediaRecorder.ondataavailable = (event) => {\n\t\t\tif (event?.data && event.data.size > 0 && recordedBlobsRef.current) {\n\t\t\t\trecordedBlobsRef.current.push(event.data);\n\t\t\t}\n\t\t};\n\t\tmediaRecorder.onstop = () => {\n\t\t\tconsole.log(\"Recording stopped, total blobs:\", recordedBlobsRef?.current?.length);\n\t\t\tuploadFinalVideo();\n\t\t};\n\n\t\tmediaRecorder.start(100); // 100ms chunks\n\t\tmediaRecorderRef.current = mediaRecorder;\n\t\tconsole.log(\"Recording started successfully (portrait normalized)\");\n\t}, [supportedTypes, uploadFinalVideo]);\n\n\tconst { faceScanDetector, scanStage, setScanStage, resetScan, isModelLoaded, startScanSequence } = useFaceScan({\n\t\tfaceScanId,\n\t\tshopDomain,\n\t\tonScanComplete: () => {\n\t\t\tstopRecording();\n\t\t},\n\t\tonModelReady: () => {\n\t\t\tsetModelReady(true);\n\t\t},\n\t\tonStartRecording: () => {\n\t\t\tconsole.log(\"Stage 0 completed - starting recording for stages 1-3\");\n\t\t\tstartRecording();\n\t\t},\n\t});\n\n\tconst startScan = useCallback(() => {\n\t\tif (!isModelLoaded) return;\n\t\tonScanStart?.();\n\t\tsetScanStage(0);\n\t\tspeechService.playAudio(`${voiceOverAssetsPath}face-scan-vos/Face-forward.mp3`);\n\t\tsetTimeout(() => {\n\t\t\tsetIsScanning(true);\n\t\t\tsetTimeout(() => {\n\t\t\t\tstartScanSequence();\n\t\t\t}, 200);\n\t\t}, 200);\n\t}, [setScanStage, setIsScanning, startScanSequence, isModelLoaded, onScanStart]);\n\n\tconst resetScanState = useCallback(() => {\n\t\tsetIsScanning(false);\n\t\tsetShowGuideCard(true);\n\t\tstopRecording();\n\t\tresetScan();\n\t\tonRetry?.();\n\t\trecordedBlobsRef.current = [];\n\t\trecordedFpsRef.current = null;\n\t\tif (mediaRecorderRef.current) {\n\t\t\tmediaRecorderRef.current = null;\n\t\t}\n\t\tsetHasError(false);\n\t\tsetScanStage(-1);\n\t\tsetFaceScanId(generateUuid());\n\t\tif (webcamRef.current && streamRef.current) {\n\t\t\twebcamRef.current.srcObject = streamRef.current;\n\t\t\tconsole.log(\"Camera stream restored after reset\");\n\t\t}\n\t}, [setScanStage, setIsScanning, setShowGuideCard, stopRecording, resetScan, setFaceScanId]);\n\n\tconst onError = (data: any) => {\n\t\tconsole.log(data, \"ws error\");\n\t\tonScanError?.(data);\n\t\tsetHasError(true);\n\t\tsetShowLoader(false);\n\t\tsetShowGuideCard(false);\n\t\thandleScanTimeCapture({\n\t\t\teventName: `${shopDomain}/faceScan`,\n\t\t\tfaceScanId,\n\t\t\tstatus: \"failed\",\n\t\t\temail,\n\t\t\tdata: JSON.stringify(data),\n\t\t});\n\t};\n\n\tconst onSuccess = (data: any) => {\n\t\tconsole.log(data, \"ws success\");\n\t\tif (data && data?.resultType === \"intermediate\") {\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/faceScan_success/intermediate`,\n\t\t\t\tfaceScanId,\n\t\t\t\tstatus: \"success\",\n\t\t\t\temail,\n\t\t\t\tdata: JSON.stringify(data),\n\t\t\t});\n\t\t}\n\t\tonScanSuccess?.(data);\n\t};\n\n\tuseEffect(() => {\n\t\tif (isError || isSuccess) return;\n\t\tif (navigator.mediaDevices.getUserMedia) {\n\t\t\tnavigator.mediaDevices\n\t\t\t\t.getUserMedia({ video: currentVideoConstraints })\n\t\t\t\t.then((stream) => {\n\t\t\t\t\tstreamRef.current = stream;\n\t\t\t\t\tif (webcamRef.current) {\n\t\t\t\t\t\twebcamRef.current.srcObject = stream;\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t\t\t.catch((err) => {\n\t\t\t\t\tconsole.error(\"Error accessing webcam:\", err);\n\t\t\t\t\tif (currentVideoConstraints === videoConstraintsExact) {\n\t\t\t\t\t\tsetCurrentVideoConstraints(videoConstraintsFallback);\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\t\t\t\t\tsetHasError(true);\n\t\t\t\t});\n\t\t}\n\t\treturn () => {\n\t\t\tif (streamRef.current) {\n\t\t\t\tstreamRef.current.getTracks().forEach((track) => track.stop());\n\t\t\t}\n\t\t};\n\t}, [isError, isSuccess, currentVideoConstraints]);\n\n\t// Face detection interval - only run when scanning is active\n\tuseEffect(() => {\n\t\tif (!modelReady || !isScanning) return;\n\t\tconst intervalId = setInterval(() => {\n\t\t\tfaceScanDetector(webcamRef, canvasRef);\n\t\t}, 500);\n\n\t\t// eslint-disable-next-line consistent-return\n\t\treturn () => clearInterval(intervalId);\n\t}, [faceScanDetector, modelReady, isScanning]);\n\n\tconst getButtonText = () => {\n\t\tif (isScanning) return LanguageKeys.reset;\n\t\tif (!isModelLoaded) return LanguageKeys.loading;\n\t\treturn LanguageKeys.start;\n\t};\n\n\tconsole.log(\"Model ready:\", modelReady, \"Has error:\", hasError, \"Is scanning:\", isScanning, \"showLoader\", showLoader);\n\tuseEffect(() => {\n\t\tconst supported = getSupportedMediaRecorderTypes(preferredVideoTypes, videoTypes);\n\t\tsetSupportedTypes(supported);\n\t}, [preferredVideoTypes]);\n\tuseEffect(() => {\n\t\tsetScanStage(-1);\n\t}, []);\n\n\treturn (\n\t\t<LanguageContextProvider>\n\t\t\t<ParamsContext.Provider\n\t\t\t\tvalue={{\n\t\t\t\t\tisError,\n\t\t\t\t\tisSuccess,\n\t\t\t\t\tshowLoader,\n\t\t\t\t\thasError,\n\t\t\t\t\tresetScanState,\n\t\t\t\t\tshowGuideCard,\n\t\t\t\t\tscanStage,\n\t\t\t\t\tvideoLoading,\n\t\t\t\t\tsetVideoLoading,\n\t\t\t\t\tvideoError,\n\t\t\t\t\tsetVideoError,\n\t\t\t\t\tgender,\n\t\t\t\t\tgetButtonText,\n\t\t\t\t\tisScanning,\n\t\t\t\t\tstartScan,\n\t\t\t\t\tisModelLoaded,\n\t\t\t\t\tconfig,\n\t\t\t\t\tonRetry\n\t\t\t\t}}\n\t\t\t>\n\t\t\t\t<FaceScanStep webcamRef={webcamRef} canvasRef={canvasRef} />\n\t\t\t</ParamsContext.Provider>\n\t\t</LanguageContextProvider>\n\t);\n};\n"],"names":["preloadedDetector","isPreloading","useFaceScan","faceScanId","onValidPose","onScanComplete","onModelReady","onStartRecording","shopDomain","detectorRef","useRef","scanStage","setScanStage","useState","consecutiveValid","setConsecutiveValid","isModelLoaded","setIsModelLoaded","isScanningActive","setIsScanningActive","utteranceRef","scanStageRef","consecutiveValidRef","validityBufferRef","lastSpokenMessageRef","lastSpokenTimeRef","voiceEnabledRef","lastPoseTrackingTimeRef","isIOS","navigator","test","userAgent","platform","maxTouchPoints","directionMessages","useMemo","debouncedTrackPoseDetection","fn","delay","timer","args","clearTimeout","setTimeout","debounce","payload","posthog","capture","trackPoseDetection","useCallback","faceKeypoints","bodyKeypoints","stage","headValid","bodyInvalid","consecutiveValidCount","now","Date","current","nose","leftEye","rightEye","leftMouth","rightMouth","faceKeypointScores","map","kp","filter","score","avgFaceScore","length","reduce","a","b","bodyKeypointScores","avgBodyScore","eyeDistance","Math","abs","noseToRight","noseToLeft","expectedPosition","timestamp","toISOString","data","JSON","stringify","parseFloat","toFixed","faceKeypointsDetected","bodyKeypointsDetected","noseToRightRatio","noseToLeftRatio","noseScore","leftEyeScore","rightEyeScore","leftMouthScore","rightMouthScore","error","console","warn","startScanSequence","stageName","initializeModels","async","window","log","detector","Promise","r","tf","poseDetection","all","import","setBackend","ready","createDetector","SupportedModels","BlazePose","runtime","modelType","dummyCanvas","document","createElement","width","videoConstraints","height","ctx","getContext","fillStyle","fillRect","estimatePoses","e","err","preloadDetector","useEffect","disposeModelAndTf","faceScanDetector","webcamRef","canvasRef","readyState","poses","keypoints","forEach","landmark","idx","point","x","y","push","clearRect","progress","min","radius","beginPath","strokeStyle","lineWidth","arc","PI","stroke","eyesToMouthVerticalDistance","faceTiltedness","faceIsTilted","faceIsVertical","frontalFace","fullLeft","fullRight","isHeadInPosition","slice","p","shift","Boolean","nextValid","max","nextStage","speechService","playAudio","voiceOverAssetsPath","completedStage","totalStages","nextSound","resetScan","speechSynthesis","cancel","enableVoiceCommands","silentUtterance","SpeechSynthesisUtterance","volume","speak","FaceScanErrorScreen","resetScanState","loading","config","translate","useContext","LanguageContext","_jsx","className","style","background","base","backgroundColor","children","_jsxs","Header","noTitle","resolvedConfig","fontFamily","heading","headingFontFamily","fontSize","headingFontSize","color","headingColor","fontWeight","headingFontWeight","LanguageKeys","scanFailed","subheading","subheadingFontFamily","subheadingFontSize","subheadingColor","subheadingFontWeight","clickToResetScan","SpecificButton","disabled","buttonText","postfixIcon","ArrowRight","buttonFunc","FaceScanGuide","videoLoading","setVideoLoading","videoError","setVideoError","gender","btnConfig","faceVisible","loadingVideo","videoLoadFailed","src","GenderType","Male","maleGlassesOffVideo","glassesOffVideo","autoPlay","loop","controls","muted","playsInline","onCanPlay","onLoadStart","onError","isDisabled","label","icon","onClick","FACE_SCAN_HEADSHOT","male","forward","female","type","left","right","smile","FaceScanStep","isError","isSuccess","showLoader","hasError","showGuideCard","getButtonText","isScanning","startScan","onRetry","ParamsContext","useLocalConfig","setPreferredLanguage","language","LoadingScreen","url","loader","loaderType","_Fragment","id","crossOrigin","preload","position","zIndex","undefined","ref","ideal","transform","opacity","Drawer","open","anchor","onClose","event","reason","hideBackdrop","props","onScanSuccess","onScanError","onScanStart","onUploadingStart","onUploadingEnd","onMeasurementSocketStart","onMeasurementSocketClose","userDetails","email","deviceFocalLength","token","mediaRecorderRef","recordedBlobsRef","streamRef","recordedFpsRef","setShowLoader","modelReady","setModelReady","setIsScanning","setShowGuideCard","setHasError","setFaceScanId","generateUuid","swan","getSwanService","callbackUrl","supportedTypes","setSupportedTypes","currentVideoConstraints","setCurrentVideoConstraints","videoConstraintsExact","preferredVideoTypes","getPreferredMediaRecorderTypes","recordingMimeType","getRecordingMimeType","recordingExtension","getRecordingExtension","usePosthogPageview","stopRecording","state","stop","uploadFinalVideo","videoFile","File","videoFps","getVideoFPS","fileSize","size","fileSizeMB","estimatedDuration","round","videoData","video_size_mb","video_size_bytes","video_fps","blob_count","estimated_duration_seconds","metaData","face_scan_id","focal_length","customer_store_url","scan_type","callback_url","handleScanTimeCapture","eventName","uploadStartTime","startTime","res","fileUpload","faceScanFileUploader","file","arrayMetaData","objectKey","contentType","uploadEndTime","uploadDuration","completionTime","measurement","handlFaceScaneSocket","onPreopen","handleWebSocketCapture","faceScanID","connection","onOpen","onSuccess","startRecording","stream","srcObject","settings","getVideoTracks","getSettings","defaultWidth","defaultHeight","canvas","videoEl","play","drawFrame","save","rotate","drawImage","restore","requestAnimationFrame","canvasStream","captureStream","mediaRecorderOptions","videoBitsPerSecond","mediaRecorder","mimeType","MediaRecorder","ondataavailable","onstop","start","status","resultType","mediaDevices","getUserMedia","video","then","catch","videoConstraintsFallback","getTracks","track","intervalId","setInterval","clearInterval","supported","getSupportedMediaRecorderTypes","videoTypes","LanguageContextProvider","Provider","value","reset"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import{jsx as e,jsxs as t,Fragment as a}from"react/jsx-runtime";import{useRef as n,useState as o,useMemo as r,useCallback as c,useEffect as i,useContext as s}from"react";import l,{posthog as d}from"posthog-js";import{s as u,v as f,a as h,L as g,H as m,b as p,S as y,m as S,g as b,G as w,d as v,F as x,P as F,u as N,c as _,e as I,f as k,h as C,i as M,j as D,k as P,l as T,n as E,p as $,o as z,q as j,r as L,t as R,w as V,x as O}from"./LoadingScreen-B27DzMIt.js";import{ArrowRight as W}from"lucide-react";import{Drawer as B}from"@mui/material";let U=null,A=!1;function q({faceScanId:e,onValidPose:t,onScanComplete:a,onModelReady:s,onStartRecording:l,shopDomain:g}){const m=n(null),[p,y]=o(0),[S,b]=o(0),[w,v]=o(!1),[x,F]=o(!1),N=n(null),_=n(p),I=n(S),k=n([]),C=n(""),M=n(0),D=n(!1),P=n(0),T="undefined"!=typeof navigator&&(/iPad|iPhone|iPod/.test(navigator.userAgent)||"MacIntel"===navigator.platform&&navigator.maxTouchPoints>1),E=r(()=>["Face front","Turn head fully left","Turn head fully right","Smile facing front"],[]),$=r(()=>function(e,t){let a;return(...n)=>{a&&clearTimeout(a),a=setTimeout(()=>e(...n),t)}}(e=>{d.capture(`${g}/face_scan_pose_detection`,e)},4e3),[g]),z=c((t,a,n,o,r,c)=>{const i=Date.now();if(!(i-P.current<4e3)){P.current=i;try{const i=t[0]||[0,0,0],s=t[2]||[0,0,0],l=t[5]||[0,0,0],d=t[9]||[0,0,0],u=t[10]||[0,0,0],f=t.map(e=>e[2]).filter(e=>e>0),h=f.length>0?f.reduce((e,t)=>e+t,0)/f.length:0,g=a.map(e=>e[2]).filter(e=>e>0),m=g.length>0?g.reduce((e,t)=>e+t,0)/g.length:0,p=Math.abs(s[0]-l[0]),y=i[0]-l[0],S=s[0]-i[0];let b="frontal";1===n?b="left":2===n?b="right":3===n&&(b="frontal_smile");const w={faceScanId:e,stage:n,timestamp:(new Date).toISOString(),data:JSON.stringify({headValid:o,bodyInvalid:r,consecutiveValid:c,avgFaceScore:parseFloat(h.toFixed(3)),avgBodyScore:parseFloat(m.toFixed(3)),faceKeypointsDetected:f.length,bodyKeypointsDetected:g.length,eyeDistance:parseFloat(p.toFixed(2)),noseToRightRatio:parseFloat((y/p).toFixed(3)),noseToLeftRatio:parseFloat((S/p).toFixed(3)),noseScore:parseFloat(i[2].toFixed(3)),leftEyeScore:parseFloat(s[2].toFixed(3)),rightEyeScore:parseFloat(l[2].toFixed(3)),leftMouthScore:parseFloat(d[2].toFixed(3)),rightMouthScore:parseFloat(u[2].toFixed(3)),expectedPosition:b})};$(w)}catch(e){console.warn("Failed to track pose detection:",e)}}},[e,$]),j=()=>{d.capture(`${g}/face_scan_detection_started`,{faceScanId:e,stage:_.current,timestamp:(new Date).toISOString(),stageName:E[_.current]}),setTimeout(()=>{F(!0)},1500)},L=async()=>{if("undefined"!=typeof window)try{console.log("Initializing Face Scan...");const e=await(async()=>{if("undefined"==typeof window)return null;if(U)return U;if(A)for(;A;)if(await new Promise(e=>setTimeout(e,100)),U)return U;A=!0;try{const[e,t]=await Promise.all([import("@tensorflow/tfjs"),import("./pose-detection.esm-qPaFsqLN.js")]);await e.setBackend("webgl"),await e.ready();const a=await t.createDetector(t.SupportedModels.BlazePose,{runtime:"tfjs",modelType:"full"}),n=document.createElement("canvas");n.width=h.width,n.height=h.height;const o=n.getContext("2d");if(o){o.fillStyle="black",o.fillRect(0,0,n.width,n.height);try{await a.estimatePoses(n)}catch(e){console.warn("Warmup frame failed (non-fatal):",e)}}return U=a,a}catch(e){return console.error("Failed to preload detector:",e),null}finally{A=!1}})();e&&(m.current=e,console.log("Face scan model loaded successfully"),v(!0),s?.())}catch(e){console.error("Error initializing face scan:",e)}};return i(()=>(L(),()=>{(async()=>{try{v(!1)}catch(e){console.error("Error disposing resources:",e)}finally{y(0),b(0),_.current=0,I.current=0,k.current=[]}})()}),[]),i(()=>{_.current=p},[p]),i(()=>{I.current=S},[S]),{faceScanDetector:async(n,o)=>{if(m.current&&n.current&&w&&x&&!(n.current.readyState<2))try{const r=await m.current.estimatePoses(n.current);if(r.length>0){const n=[],c=[];if(r[0].keypoints.forEach((e,t)=>{const a=e.score??0,o=[e.x??-1,e.y??-1,a];t<=10?n.push(o):c.push(o)}),o?.current){const e=o.current.getContext("2d"),{width:t,height:a}=o.current;if(e){e.clearRect(0,0,t,a);const n=Math.min(I.current/2,1),o=.35*a;e.beginPath(),e.strokeStyle="#00ff00",e.lineWidth=6,e.arc(t/2,a/2,o+10,-Math.PI/2,-Math.PI/2+2*n*Math.PI),e.stroke()}}const i=((e,t)=>{if(!t[0]||!t[2]||!t[5]||t[0][2]<.2||t[2][2]<.2||t[5][2]<.2)return!1;const a=.5*(t[10][1]+t[9][1])-.5*(t[5][1]+t[2][1]),n=(.5*(t[10][1]+t[9][1])-t[0][1])/a,o=n>.7||n<.3,r=Math.abs(t[2][1]-t[5][1])<.5*a&&Math.abs(t[9][1]-t[10][1])<.5*a,c=t[2][0]-t[5][0],i=t[0][0]-t[5][0],s=i>.4*c&&i<.6*c,l=i>.85*c,d=t[2][0]-t[0][0]>.85*c;switch(e){case 0:case 3:return!o&&r&&s;case 1:return!o&&r&&l;case 2:return!o&&r&&d;default:return!1}})(_.current,n),s=c.slice(2).filter(e=>e[2]>.7).length<=2;k.current.push(i&&s),k.current.length>10&&k.current.shift();if(k.current.filter(Boolean).length>=2){const e=I.current+1;b(e)}else b(Math.max(0,I.current-1));if(I.current>=2){t?.();const n=_.current+1;b(0),k.current=[],y(n),F(!1),(async()=>{if(await u.playAudio(`${f}face-scan-vos/Sound-effect.mp3`),d.capture(`${g}/face_scan_detection_stage_completed`,{faceScanId:e,completedStage:_.current,nextStage:n,timestamp:(new Date).toISOString(),totalStages:4,stageName:E[_.current]}),1===n&&l&&l(),n>=4)a?.();else{let e=null;switch(n){case 1:e="Left.mp3";break;case 2:e="Right.mp3";break;case 3:e="Smile.mp3"}e&&await u.playAudio(`${f}face-scan-vos/${e}`),j()}})()}z(n,c,_.current,i,s,I.current)}}catch(e){console.error("Pose detection error:",e)}},scanStage:p,setScanStage:y,consecutiveValid:S,isModelLoaded:w,resetScan:()=>{d.capture(`${g}/face_scan_reset`,{faceScanId:e,stage:_.current,timestamp:(new Date).toISOString(),stageName:E[_.current],consecutiveValid:I.current}),y(0),b(0),F(!1),_.current=0,I.current=0,k.current=[],C.current="",M.current=0,D.current=!1,N.current&&(speechSynthesis.cancel(),N.current=null)},enableVoiceCommands:()=>{if(D.current=!0,"undefined"!=typeof window&&T&&"speechSynthesis"in window){const e=new SpeechSynthesisUtterance("");e.volume=0,speechSynthesis.speak(e)}},startScanSequence:j,isScanningActive:x}}function J({resetScanState:a,loading:n,config:o}){const{translate:r}=s(g)||{};return e("div",{className:"fixed top-[0] left-[0] z-[999] flex justify-center items-center w-full h-full",style:{background:o?.style?.base?.backgroundColor},children:t("div",{className:"flex flex-col w-full items-center p-[1rem] rounded-lg ",children:[e(m,{noTitle:!0,resolvedConfig:o}),t("div",{className:"flex flex-col items-center w-full",children:[e("h2",{className:"text-xl font-semibold text-gray-800 ",style:{fontFamily:o?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:o?.style?.heading?.headingFontSize||"32px",color:o?.style?.heading?.headingColor||"#000",fontWeight:o?.style?.heading?.headingFontWeight||"normal"},children:r?.(p.scanFailed)}),e("p",{className:"mb-[1.5rem]",style:{fontFamily:o?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:o?.style?.subheading?.subheadingFontSize||"14px",color:o?.style?.subheading?.subheadingColor||"#4b5563",fontWeight:o?.style?.subheading?.subheadingFontWeight||"normal"},children:r?.(p.clickToResetScan)}),e(y,{disabled:n,className:"w-full h-[45px] shadow-[0px_1px_2px_0px_#00000040] text-[16px]",buttonText:r?.(p.resetScan),postfixIcon:e(W,{}),buttonFunc:()=>a?.(),resolvedConfig:o})]})]})})}function G({stage:a,videoLoading:n,setVideoLoading:o,videoError:r,setVideoError:c,gender:i,config:l,btnConfig:d}){const{translate:u}=s(g)||{};return-1===a?t("div",{className:"text-center p-[16px] w-full h-full",style:{background:l?.style?.base?.backgroundColor},children:[t("div",{className:"w-full",children:[e(m,{noTitle:!0,resolvedConfig:l}),e("h2",{style:{fontFamily:l?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:l?.style?.heading?.headingFontSize||"32px",color:l?.style?.heading?.headingColor||"#000",fontWeight:l?.style?.heading?.headingFontWeight||"normal"},children:u?.(p.faceVisible)}),n&&e("div",{className:"mb-4 flex items-center justify-center",style:{fontFamily:l?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:l?.style?.subheading?.subheadingFontSize||"14px",color:l?.style?.subheading?.subheadingColor||"#4b5563",fontWeight:l?.style?.subheading?.subheadingFontWeight||"normal"},children:u?.(p.loadingVideo)}),r&&e("div",{className:"mb-4 text-red-600",style:{fontFamily:l?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:l?.style?.subheading?.subheadingFontSize||"14px",color:"#ff0000",fontWeight:l?.style?.subheading?.subheadingFontWeight||"normal"},children:u?.(p.videoLoadFailed)}),!r&&e("div",{className:" w-[100px] mx-auto",children:e("video",{src:i===w.Male?S:b,autoPlay:!0,loop:!0,controls:!1,muted:!0,playsInline:!0,className:"h-full w-full object-contain border-none",onCanPlay:()=>o(!1),onLoadStart:()=>o(!0),onError:()=>c(!0),"aria-label":"Instructional video: Please remove your glasses before starting the scan."})})]}),e("div",{className:"flex justify-center w-full p-[1rem]",children:e(y,{disabled:d.isDisabled,className:"!w-[60px] !h-[35px] !py-0 !px-0",buttonText:u?.(d.label),postfixIcon:d?.icon,buttonFunc:d.onClick,resolvedConfig:l})})]}):e("div",{className:"text-center p-[16px] w-full h-full",style:{background:l?.style?.base?.backgroundColor},children:t("div",{className:"w-full",children:[e(m,{noTitle:!0,resolvedConfig:l}),e("h3",{className:"mb-[0.5rem]",style:{fontFamily:l?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:l?.style?.heading?.headingFontSize||"32px",color:l?.style?.heading?.headingColor||"#000",fontWeight:l?.style?.heading?.headingFontWeight||"normal"},children:u?.(v[a])}),t("div",{className:"max-w-[400px] justify-center gap-1 mx-auto flex items-center",children:[e("div",{className:` w-[120px] overflow-hidden ${0===a?"opacity-100":"opacity-0 hidden"} `,children:e("video",{className:"h-full w-full object-contain border-none",muted:!0,loop:!0,autoPlay:!0,playsInline:!0,children:e("source",{src:i===w.Male?x.male.forward:x.female.forward,type:"video/mp4"})})}),e("div",{className:` w-[120px] overflow-hidden ${1===a?"opacity-100":"opacity-0 hidden"} `,children:e("video",{className:"h-full w-full object-contain border-none",muted:!0,loop:!0,autoPlay:!0,playsInline:!0,children:e("source",{src:i===w.Male?x.male.left:x.female.left,type:"video/mp4"})})}),e("div",{className:` w-[120px] overflow-hidden ${2===a?"opacity-100":"opacity-0 hidden"} `,children:e("video",{className:"h-full w-full object-contain border-none",muted:!0,loop:!0,autoPlay:!0,playsInline:!0,children:e("source",{src:i===w.Male?x.male.right:x.female.right,type:"video/mp4"})})}),e("div",{className:` w-[120px] overflow-hidden ${3===a?"opacity-100":"opacity-0 hidden"} `,children:e("video",{className:"h-full w-full object-contain border-none",muted:!0,loop:!0,autoPlay:!0,playsInline:!0,children:e("source",{src:i===w.Male?x.male.smile:x.female.smile,type:"video/mp4"})})})]})]})})}const K=({webcamRef:n,canvasRef:o})=>{const{isError:r,isSuccess:c,showLoader:l,hasError:d,resetScanState:u,showGuideCard:f,scanStage:m,videoLoading:p,setVideoLoading:y,videoError:S,setVideoError:b,gender:w,getButtonText:v,isScanning:x,startScan:I,isModelLoaded:k,config:C}=s(F),M=N(C),{setPreferredLanguage:D}=s(g)||{};return i(()=>{D?.(M?.language)},[M]),r?e(J,{config:M}):c?e("div",{className:"fixed z-[9] w-full h-full",style:{background:M?.style?.base?.backgroundColor},children:e(_,{url:M?.loader,loaderType:"black"})}):t(a,{children:[e("audio",{id:"audioElement",crossOrigin:"anonymous",preload:"auto",style:{position:"absolute",zIndex:-99999},src:void 0}),l&&!d&&e("div",{className:"fixed z-[9] w-full h-full",style:{background:M?.style?.base?.backgroundColor},children:e(_,{url:M?.loader,loaderType:"black"})}),e("div",{className:"h-full flex-col relative w-full flex justify-center items-center text-center rounded-t-[20px]",style:{background:M?.style?.base?.backgroundColor},children:e("div",{className:"flex-1 w-full max-w-md overflow-hidden",children:t("div",{className:"w-full h-full",children:[e("video",{ref:n,autoPlay:!0,playsInline:!0,muted:!0,width:h.width.ideal,height:h.height.ideal,className:"w-full h-full object-cover fixed left-0 top-0 z-0",style:{transform:"scaleX(-1)"}}),e("canvas",{ref:o,width:h.width.ideal,height:h.height.ideal,style:{transform:"scaleX(-1)",opacity:"0"}})]})})}),!l&&d&&e(J,{loading:l,resetScanState:u,config:M}),f&&!d&&!l&&e(B,{open:!0,className:"face-scan-small camera-draw",anchor:"bottom",onClose:(e,t)=>{},hideBackdrop:!0,children:e(G,{stage:m,videoLoading:p,setVideoLoading:y,videoError:S,setVideoError:b,gender:w,config:M,btnConfig:{label:v(),onClick:x?u:I,isDisabled:l||!k,icon:e(x?W:a,{})}})})]})},H=({userDetails:t,onScanSuccess:a,onScanError:s,onRetry:d,config:h,isError:g,isSuccess:m,onUpload:y})=>{const S=n(null),b=n(null),w=n(null),v=n([]),x=n(null),N=n(null),[_,W]=o(!1),[B,U]=o(!1),[A,J]=o(!1),[G,H]=o(!0),[X,Q]=o(!1),[Y,Z]=o(!1),[ee,te]=o(!1),[ae,ne]=o(I()),{email:oe,gender:re,deviceFocalLength:ce,shopDomain:ie,callbackUrl:se}=t,[le,de]=o([]),[ue,fe]=o(k),he=r(()=>C(),[]),ge=M(le),me=D(ge),pe=c(()=>{console.log("Stopping recording..."),w.current&&"recording"===w.current.state&&w.current.stop(),w.current=null},[]),ye=async()=>{if(v.current){if(0===v.current.length)return console.error("No video data recorded"),te(!0),void W(!1);y?.(),W(!0);const e=new File(v.current,`${ae}.${me}`,{type:ge}),t=N.current??await L(e),a=e.size,n=(a/1048576).toFixed(2),o=Math.round(a/1e4),r={video_size_mb:parseFloat(n),video_size_bytes:a,video_fps:t||0,blob_count:v.current.length,estimated_duration_seconds:o},c=[{gender:re},{face_scan_id:ae},{focal_length:`${ce}`},{customer_store_url:ie},{scan_type:"face_scan"}];se&&c.push({callback_url:se}),R({eventName:`${ie}/face_scan_meta_data`,faceScanId:ae,email:oe,data:JSON.stringify({metaData:c,videoData:r})});const i=Date.now();R({eventName:`${ie}/face_scan_upload_start`,faceScanId:ae,email:oe,startTime:i});try{const t=await V.fileUpload.faceScanFileUploader({file:e,arrayMetaData:c,objectKey:ae,email:oe,contentType:e.type}),a=Date.now();R({eventName:`${ie}/face_scan_upload_complete`,faceScanId:ae,email:oe,completionTime:a,uploadDuration:a-i}),console.log("✅ Upload successful",t),V.measurement.handlFaceScaneSocket({onPreopen:()=>{O({eventName:`${ie}/webSocket`,faceScanID:ae,connection:"pre_open",type:"faceScan_recommendation",email:oe})},faceScanId:ae,onOpen:()=>{O({eventName:`${ie}/webSocket`,faceScanID:ae,connection:"open",type:"faceScan_recommendation",email:oe}),console.log("websocket connect open")},onError:e=>{O({eventName:`${ie}/webSocket`,faceScanID:ae,connection:"error",type:"faceScan_recommendation",email:oe}),ke(e)},onSuccess:e=>{O({eventName:`${ie}/webSocket`,faceScanID:ae,connection:"success",type:"faceScan_recommendation",email:oe}),Ce(e)},onClose:()=>{console.log("websocket connect close"),O({eventName:`${ie}/webSocket`,faceScanID:ae,connection:"close",type:"faceScan_recommendation",email:oe})}})}catch(e){ke(e)}}},Se=c(()=>{const e=S.current?.srcObject;if(!e)return;const t=e.getVideoTracks()[0].getSettings(),a="number"==typeof ue.width?ue.width:1280,n="number"==typeof ue.height?ue.height:720,{width:o=a,height:r=n}=t,c=document.createElement("canvas"),i=c.getContext("2d");o>r?(c.width=r,c.height=o):(c.width=o,c.height=r);const s=document.createElement("video");s.srcObject=e,s.muted=!0,s.playsInline=!0,s.play();const l=()=>{o>r?(i?.save(),i?.translate(c.width,0),i?.rotate(Math.PI/2),i?.drawImage(s,0,0,r,o),i?.restore()):i?.drawImage(s,0,0,o,r),requestAnimationFrame(l)};l();N.current=30;const d=c.captureStream(30),u={videoBitsPerSecond:15e5};let f;le.length>0&&(u.mimeType=le[0]);try{f=new MediaRecorder(d,u)}catch(e){return console.error("MediaRecorder init failed:",e),te(!0),void W(!1)}v.current=[],f.ondataavailable=e=>{e?.data&&e.data.size>0&&v.current&&v.current.push(e.data)},f.onstop=()=>{console.log("Recording stopped, total blobs:",v?.current?.length),ye()},f.start(100),w.current=f,console.log("Recording started successfully (portrait normalized)")},[le,ye]),{faceScanDetector:be,scanStage:we,setScanStage:ve,resetScan:xe,isModelLoaded:Fe,startScanSequence:Ne}=q({faceScanId:ae,shopDomain:ie,onScanComplete:()=>{pe()},onModelReady:()=>{U(!0)},onStartRecording:()=>{console.log("Stage 0 completed - starting recording for stages 1-3"),Se()}}),_e=c(()=>{Fe&&(ve(0),u.playAudio(`${f}face-scan-vos/Face-forward.mp3`),setTimeout(()=>{J(!0),setTimeout(()=>{Ne()},200)},200))},[ve,J,Ne,Fe]),Ie=c(()=>{J(!1),H(!0),pe(),xe(),d?.(),v.current=[],N.current=null,w.current&&(w.current=null),te(!1),ve(-1),ne(I()),S.current&&x.current&&(S.current.srcObject=x.current,console.log("Camera stream restored after reset"))},[ve,J,H,pe,xe,ne]),ke=e=>{console.log(e,"ws error"),s?.(e),te(!0),W(!1),H(!1),R({eventName:`${ie}/faceScan`,faceScanId:ae,status:"failed",email:oe,data:JSON.stringify(e)})},Ce=e=>{console.log(e,"ws success"),e&&"intermediate"===e?.resultType&&R({eventName:`${ie}/faceScan_success/intermediate`,faceScanId:ae,status:"success",email:oe,data:JSON.stringify(e)}),a?.(e)};i(()=>{if(!g&&!m)return navigator.mediaDevices.getUserMedia&&navigator.mediaDevices.getUserMedia({video:ue}).then(e=>{x.current=e,S.current&&(S.current.srcObject=e)}).catch(e=>{console.error("Error accessing webcam:",e),ue!==k?te(!0):fe(P)}),()=>{x.current&&x.current.getTracks().forEach(e=>e.stop())}},[g,m,ue]),i(()=>{if(!B||!A)return;const e=setInterval(()=>{be(S,b)},500);return()=>clearInterval(e)},[be,B,A]);return console.log("Model ready:",B,"Has error:",ee,"Is scanning:",A,"showLoader",_),i(()=>{const e=T(he,E);de(e)},[he]),i(()=>{ve(-1)},[g,m]),i(()=>{l.init($,{api_host:z}),l.capture("$pageview")},[]),e(j,{children:e(F.Provider,{value:{isError:g,isSuccess:m,showLoader:_,hasError:ee,resetScanState:Ie,showGuideCard:G,scanStage:we,videoLoading:X,setVideoLoading:Q,videoError:Y,setVideoError:Z,gender:re,getButtonText:()=>A?p.reset:Fe?p.start:p.loading,isScanning:A,startScan:_e,isModelLoaded:Fe,config:h},children:e(K,{webcamRef:S,canvasRef:b})})})};export{H as F};
2
+ //# sourceMappingURL=FaceScan-DL21ykY_.js.map