@swan-admin/swan-web-component 1.0.117 → 1.0.119
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/BodyScan--QQkmBvw.js +2 -0
- package/dist/BodyScan--QQkmBvw.js.map +1 -0
- package/dist/BodyScan-1GnGflqx.js +2 -0
- package/dist/BodyScan-1GnGflqx.js.map +1 -0
- package/dist/BodyScan-B0MKE1vY.js +2 -0
- package/dist/BodyScan-B0MKE1vY.js.map +1 -0
- package/dist/BodyScan-B85rcTFy.js +2 -0
- package/dist/BodyScan-B85rcTFy.js.map +1 -0
- package/dist/BodyScan-B90ht9_5.js +2 -0
- package/dist/BodyScan-B90ht9_5.js.map +1 -0
- package/dist/BodyScan-BVAqja5k.js +2 -0
- package/dist/BodyScan-BVAqja5k.js.map +1 -0
- package/dist/BodyScan-BVrc-R6j.js +2 -0
- package/dist/BodyScan-BVrc-R6j.js.map +1 -0
- package/dist/BodyScan-BYbhTHgG.js +2 -0
- package/dist/BodyScan-BYbhTHgG.js.map +1 -0
- package/dist/BodyScan-Be0NcDVq.js +2 -0
- package/dist/BodyScan-Be0NcDVq.js.map +1 -0
- package/dist/BodyScan-BlT-y35F.js +2 -0
- package/dist/BodyScan-BlT-y35F.js.map +1 -0
- package/dist/BodyScan-C7dkFVxY.js +2 -0
- package/dist/BodyScan-C7dkFVxY.js.map +1 -0
- package/dist/BodyScan-CAIYPUJI.js +2 -0
- package/dist/BodyScan-CAIYPUJI.js.map +1 -0
- package/dist/BodyScan-CRfmNpg7.js +2 -0
- package/dist/BodyScan-CRfmNpg7.js.map +1 -0
- package/dist/BodyScan-Cgut5BZ2.js +2 -0
- package/dist/BodyScan-Cgut5BZ2.js.map +1 -0
- package/dist/BodyScan-CrmHVOBq.js +2 -0
- package/dist/BodyScan-CrmHVOBq.js.map +1 -0
- package/dist/BodyScan-CvT6nbKG.js +2 -0
- package/dist/BodyScan-CvT6nbKG.js.map +1 -0
- package/dist/BodyScan-DChxaIeM.js +2 -0
- package/dist/BodyScan-DChxaIeM.js.map +1 -0
- package/dist/BodyScan-De-sb6e1.js +2 -0
- package/dist/BodyScan-De-sb6e1.js.map +1 -0
- package/dist/BodyScan-DogOXXY0.js +2 -0
- package/dist/BodyScan-DogOXXY0.js.map +1 -0
- package/dist/BodyScan-b7bRBOYK.js +2 -0
- package/dist/BodyScan-b7bRBOYK.js.map +1 -0
- package/dist/BodyScan-kpbgefyc.js +2 -0
- package/dist/BodyScan-kpbgefyc.js.map +1 -0
- package/dist/BodyScan-lCqgPd6z.js +2 -0
- package/dist/BodyScan-lCqgPd6z.js.map +1 -0
- package/dist/BodyScan-lc189qhY.js +2 -0
- package/dist/BodyScan-lc189qhY.js.map +1 -0
- package/dist/BodyScan-liyrkU6l.js +2 -0
- package/dist/BodyScan-liyrkU6l.js.map +1 -0
- package/dist/BodyScan-xFn2tlOj.js +2 -0
- package/dist/BodyScan-xFn2tlOj.js.map +1 -0
- package/dist/BodyScan-ym1Rn7E9.js +2 -0
- package/dist/BodyScan-ym1Rn7E9.js.map +1 -0
- package/dist/FaceScan-1Fwy5JDo.js +2 -0
- package/dist/FaceScan-1Fwy5JDo.js.map +1 -0
- package/dist/FaceScan-D1B_6J78.js +2 -0
- package/dist/FaceScan-D1B_6J78.js.map +1 -0
- package/dist/LoadingScreen-C_pA2xaB.js +2 -0
- package/dist/LoadingScreen-C_pA2xaB.js.map +1 -0
- package/dist/LoadingScreen-HQLN5ZOg.js +2 -0
- package/dist/LoadingScreen-HQLN5ZOg.js.map +1 -0
- package/dist/bodyScan.js +1 -1
- package/dist/bodyScan.mjs +1 -1
- package/dist/faceScan.js +1 -1
- package/dist/faceScan.mjs +1 -1
- package/dist/index.js +1 -1
- package/dist/index.mjs +1 -1
- package/dist/index.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BodyScan-ym1Rn7E9.js","sources":["../src/utils/context/mediaContext.tsx","../src/Icons/SwitchIcon.tsx","../src/components/bodyScan/LevelScreen.tsx","../src/components/bodyScan/CameraScanChild.tsx","../src/customHooks/useTensorFlow.ts","../src/components/bodyScan/ScanningComponent.tsx","../src/components/bodyScan/AngleDetector.tsx","../src/components/Modal.tsx","../src/components/bodyScan/CameraPermission.tsx","../src/components/bodyScan/VideoPlayer.tsx","../src/components/bodyScan/ScanErrorMessage.tsx","../src/components/Signup.tsx","../src/components/bodyScan/BodyScanSteps.tsx","../src/components/bodyScan/BodyScan.tsx","../src/customHooks/useGyroSensor.ts"],"sourcesContent":["\"use client\"\nimport React, {\n createContext,\n Dispatch,\n ReactNode,\n SetStateAction,\n useEffect,\n useLayoutEffect,\n useMemo,\n useState,\n} from \"react\";\n\nexport const MEDIA_TYPES = {\n DESKTOP: \"DESKTOP\",\n TAB: \"TAB\",\n MOBILE: \"MOBILE\",\n};\n\ninterface MediaContextType {\n size: number[];\n setSize: Dispatch<SetStateAction<number[]>>;\n clearInputs: boolean;\n setClearInputs: Dispatch<SetStateAction<boolean>>;\n media: string;\n}\n\nexport const MediaContext = createContext<MediaContextType | undefined>(\n undefined\n);\n\nfunction MediaContextProvider({ children }: { children: ReactNode }) {\n const [size, setSize] = useState([window?.innerWidth, window?.innerHeight]);\n const [clearInputs, setClearInputs] = useState(false);\n\n const updateSize = () => {\n setSize([window?.innerWidth, window?.innerHeight]);\n };\n useEffect(() => {\n updateSize();\n }, []);\n useLayoutEffect(() => {\n window.addEventListener(\"resize\", updateSize);\n return () => window.removeEventListener(\"resize\", updateSize);\n }, [updateSize]);\n let media = MEDIA_TYPES.DESKTOP;\n if (size[0] > 768 && size[0] < 1024) {\n media = MEDIA_TYPES.TAB;\n }\n if (size[0] < 768) {\n media = MEDIA_TYPES.MOBILE;\n }\n\n const mediaDetails = useMemo(\n () => ({\n size,\n setSize,\n clearInputs,\n setClearInputs,\n media,\n }),\n [size, clearInputs, media]\n );\n\n return (\n <MediaContext.Provider value={mediaDetails}>\n {children}\n </MediaContext.Provider>\n );\n}\n\nexport default MediaContextProvider;\n","import React from \"react\";\n\nfunction SwitchIcon({ size = 16 }: { size?: number }) {\n return (\n <svg\n width={size}\n height={size}\n viewBox=\"0 0 25 25\"\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n >\n <path\n d=\"M22.6968 14.6968C22.6968 16.8185 21.8539 18.8533 20.3536 20.3536C18.8533 21.8539 16.8185 22.6968 14.6968 22.6968\"\n stroke=\"white\"\n strokeWidth=\"2\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n <path\n d=\"M18.6968 11.6968V10.6968C18.6968 10.1663 18.4861 9.65764 18.111 9.28256C17.7359 8.90749 17.2272 8.69678 16.6968 8.69678C16.1663 8.69678 15.6576 8.90749 15.2826 9.28256C14.9075 9.65764 14.6968 10.1663 14.6968 10.6968\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n <path\n d=\"M14.6968 10.6968V9.69678C14.6968 9.16634 14.4861 8.65764 14.111 8.28256C13.7359 7.90749 13.2272 7.69678 12.6968 7.69678C12.1663 7.69678 11.6576 7.90749 11.2826 8.28256C10.9075 8.65764 10.6968 9.16634 10.6968 9.69678V10.6968\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n <path\n d=\"M10.6968 10.1968V4.69678C10.6968 4.16634 10.4861 3.65764 10.111 3.28256C9.73592 2.90749 9.22721 2.69678 8.69678 2.69678C8.16634 2.69678 7.65764 2.90749 7.28256 3.28256C6.90749 3.65764 6.69678 4.16634 6.69678 4.69678V14.6968\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n <path\n d=\"M18.6969 11.6968C18.6969 11.1663 18.9076 10.6576 19.2827 10.2826C19.6577 9.90749 20.1664 9.69678 20.6969 9.69678C21.2273 9.69678 21.736 9.90749 22.1111 10.2826C22.4862 10.6576 22.6969 11.1663 22.6969 11.6968V14.6968C22.6969 16.8185 21.854 18.8533 20.3537 20.3536C18.8534 21.8539 16.8186 22.6968 14.6969 22.6968H12.6969C9.89688 22.6968 8.19688 21.8368 6.70688 20.3568L3.10688 16.7568C2.76282 16.3757 2.57847 15.8769 2.592 15.3637C2.60554 14.8505 2.81593 14.3621 3.1796 13.9997C3.54327 13.6373 4.03238 13.4287 4.54565 13.417C5.05892 13.4053 5.55704 13.5914 5.93688 13.9368L7.69688 15.6968\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n />\n </svg>\n );\n}\n\nexport default React.memo(SwitchIcon);\n","/* eslint-disable no-nested-ternary */\n\nimport React, { useCallback, useContext, useMemo } from \"react\";\nimport Header from \"../Header\";\nimport cn from \"clsx\";\nimport { LevelScreenProps } from \"../../types/interfaces\";\nimport SwitchIcon from \"../../Icons/SwitchIcon\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport { useConfig } from \"../../utils/context/configContext\";\n\nfunction LevelScreen({ angle, countdown, isScanning, isInTargetRange, stabilityScore, children }: Omit<LevelScreenProps, \"config\">) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\tconst config = useConfig();\n\n\tconst getProximityToTarget = useCallback(() => {\n\t\tif (angle < 80) return Math.max(0, Math.min(100, (angle - 60) * 10));\n\t\tif (angle > 95) return Math.max(0, Math.min(100, (105 - angle) * 10));\n\t\treturn 100;\n\t}, [angle]);\n\n\tconst backgroundColor = useMemo(() => {\n\t\tif (isScanning) return config?.style?.angleDetector?.successAngleBackground || \"#4f46e5\";\n\n\t\tconst proximity = getProximityToTarget();\n\t\tif (proximity === 0) return config?.style?.angleDetector?.successAngleLowBackground || \"#ffffff\";\n\t\tif (proximity === 100) return config?.style?.angleDetector?.successAngleBackground || \"#4f46e5\";\n\n const r = Math.round(255 - (255 - 139) * (proximity / 100));\n const g = Math.round(255 - (255 - 92) * (proximity / 100));\n const b = Math.round(255 - (255 - 246) * (proximity / 100));\n return `rgb(${r}, ${g}, ${b})`;\n }, [isScanning, getProximityToTarget]);\n\n\tconst handelTextColour = useCallback((isLight: boolean, proximity: number) => {\n\t\tif (proximity > 70) {\n\t\t\treturn isLight ? `text-[${config?.style?.angleDetector?.successAngleTextLightColor}]/70` : `text-[${config?.style?.angleDetector?.successAngleTextLightColor}]`;\n\t\t}\n\t\treturn isLight ? `text-[${config?.style?.angleDetector?.successAngleTextLightColor}]/40` : `text-[${config?.style?.angleDetector?.successAngleTextDarkColor}]/70`;\n\t}, []);\n\n\tconst getTextColor = useCallback(\n\t\t(isLight = false) => {\n\t\t\tconst proximity = getProximityToTarget();\n\t\t\tif (isScanning) return `text-[${config?.style?.angleDetector?.successAngleTextLightColor}]`;\n\t\t\treturn handelTextColour(isLight, proximity);\n\t\t},\n\t\t[isScanning, getProximityToTarget],\n\t);\n\n\tconst handelColourCode = useCallback((isLight: boolean, proximity: number) => {\n\t\tif (proximity > 70) {\n\t\t\treturn isLight ? config?.style?.angleDetector?.successAngleTextLightColor : config?.style?.angleDetector?.successAngleTextLightColor;\n\t\t}\n\t\treturn isLight ? config?.style?.angleDetector?.successAngleTextLightColor : config?.style?.angleDetector?.successAngleTextDarkColor;\n\t}, []);\n\tconst getColorCode = useCallback(\n\t\t(isLight = false) => {\n\t\t\tconst proximity = getProximityToTarget();\n\t\t\tif (isScanning) return config?.style?.angleDetector?.successAngleTextLightColor;\n\t\t\treturn handelColourCode(isLight, proximity);\n\t\t},\n\t\t[isScanning, getProximityToTarget],\n\t);\n\tconst cd = countdown;\n\tfunction fixVh() {\n\t\tdocument.documentElement.style.setProperty(\"--real-vh\", window.innerHeight + \"px\");\n\t}\n\tfixVh();\n\twindow.addEventListener(\"resize\", fixVh);\n\treturn (\n\t\t<div\n\t\t\tclassName=\"flex common-ui-main w-screen flex-col items-center h-[var(--real-vh)] overflow-hidden touch-none justify-center transition-all duration-300 max-w-[28rem] mx-auto\"\n\t\t\tstyle={{ backgroundColor }}\n\t\t\t// onDoubleClick={onDoubleClick}\n\t\t>\n\t\t\t<div className=\"flex justify-start fixed top-[.5rem] max-w-[28rem] mx-auto w-full px-[1rem]\">\n\t\t\t\t<div className=\"flex justify-start \">\n\t\t\t\t\t<Header noTitle />\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t\t{cd !== null ? (\n\t\t\t\t<div\n\t\t\t\t\tclassName=\"relative flex h-[6rem] w-[6rem] items-center justify-center rounded-[9999px] bg-[#fff]/20 transition-all duration-300\"\n\t\t\t\t\tstyle={{\n\t\t\t\t\t\tborder: `2px solid ${getColorCode()}`,\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t<div\n\t\t\t\t\t\tclassName={`text-[3rem] font-bold text-[${getColorCode()}]`}\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.base?.baseFontFamily || \"Inter, sans-serif\",\n color: getColorCode(),\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{cd}\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t) : isScanning ? (\n\t\t\t\t<div className=\"relative flex flex-col items-center justify-center\">\n\t\t\t\t\t<div className=\"relative flex h-16 w-16 items-center justify-center rounded-[9999px] border-2 bg-[#fff]/30 border-[#fff]\">\n\t\t\t\t\t\t<div className=\"h-4 w-4 rounded-[9999px] animate-pulse bg-[#fff]/80\" />\n\t\t\t\t\t</div>\n\t\t\t\t\t{children}\n\t\t\t\t</div>\n\t\t\t) : (\n\t\t\t\t<div\n\t\t\t\t\tclassName={cn(\"relative flex h-[4rem] w-[4rem] items-center justify-center rounded-[9999px] \", isInTargetRange ? \"bg-[#fff]/20 border-[#fff]\" : \"bg-[#fff]/10 border-[#fff]\")}\n\t\t\t\t\tstyle={{\n\t\t\t\t\t\ttransform: `translateY(${(90 - angle) * 3}px)`,\n\t\t\t\t\t\ttransition: \"transform 0.2s ease-out\",\n\t\t\t\t\t\tborder: `2px solid ${getColorCode()}`,\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t<div className={`h-[1rem] w-[1rem] rounded-[9999px] bg-[${getColorCode()}]/80`}\n style={{\n\t\t\t\t\t\tbackgroundColor: `${getColorCode()}B3`,\n\t\t\t\t\t}}\n />\n\t\t\t\t\t<div\n\t\t\t\t\t\t// className=\" text-[#fff]\"\n\t\t\t\t\t\tclassName={cn(\"mt-[.5rem] text-center w-[180px] flex-col flex items-center absolute top-[60px]\", getTextColor())}\n style={{\n\t\t\t\t\t\tcolor: getColorCode(),\n\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t<SwitchIcon size={30} />\n\t\t\t\t\t\t<p\n\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\tfontFamily: config?.style?.base?.baseFontFamily || \"Inter, sans-serif\",\n\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t>\n\t\t\t\t\t\t\t{\" \"}\n\t\t\t\t\t\t\t{translate?.(LanguageKeys.startLevelCheck)}\n\t\t\t\t\t\t\t<br />\n\t\t\t\t\t\t</p>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t)}\n\n\t\t\t{cd !== null && (\n\t\t\t\t<div className=\"absolute bottom-[8rem] text-center\">\n\t\t\t\t\t<div\n\t\t\t\t\t\tclassName={cn(\"text-sm font-medium px-4 py-1.5 bg-black/20 rounded-[9999px] mt-8\", getTextColor())}\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.base?.baseFontFamily || \"Inter, sans-serif\",\n color: getColorCode(),\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.leavePhone)}\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t)}\n\t\t\t{isInTargetRange && cd === null && !isScanning && (\n\t\t\t\t<div className=\"absolute bottom-[8rem] w-[12rem]\">\n\t\t\t\t\t<div\n\t\t\t\t\t\tclassName={`h-[.375rem] w-full bg-[${getColorCode()}]/20 rounded-[9999px] overflow-hidden`}\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tbackgroundColor: `${getColorCode()}33`, // 20% opacity\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t<div\n\t\t\t\t\t\t\tclassName={`h-full bg-[${getColorCode()}]/70 transition-all duration-300 rounded-[9999px]`}\n\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\twidth: `${angle}%`,\n\t\t\t\t\t\t\t\tbackgroundColor: `${getColorCode()}B3`,\n\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t/>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t)}\n\n {cd === null && !isScanning && (\n <div className=\"absolute bottom-[5rem] text-center\">\n <div className={cn(\"text-[1.5rem] font-light\", getTextColor())}\n style={{\n\t\t\t\t\t\tcolor: getColorCode(),\n\t\t\t\t\t}}\n >\n {Math.round(angle)}°\n </div>\n </div>\n )}\n\n\t\t\t{cd === null && !isScanning && (\n\t\t\t\t<div className=\"absolute bottom-[2rem] text-center max-w-[20rem] mx-auto\">\n\t\t\t\t\t<div\n\t\t\t\t\t\tclassName={cn(\"text-[.75rem] opacity-50\", getTextColor())}\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.base?.baseFontFamily || \"Inter, sans-serif\",\n color: getColorCode(),\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.placePhoneUpright)}\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t)}\n\t\t</div>\n\t);\n}\nexport default React.memo(LevelScreen);\n","import { useContext } from \"react\";\nimport Webcam from \"react-webcam\";\nimport { CameraScanChildProps } from \"../../types/interfaces\";\nimport { MEDIA_TYPES, MediaContext } from \"../../utils/context/mediaContext\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { pauseIcon, voiceOverAssetsPath } from \"../../utils/constants\";\nimport SpecificButton from \"../../atoms/specificButton/SpecificButton\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport { X } from \"lucide-react\";\n\n\n\nfunction CameraScanChild({\n resetScan,\n loadingCam,\n onUserMedia,\n onPause,\n pause,\n recordingStarted,\n startSendingVideoFrames,\n showPause,\n webcamRef,\n onUserMediaError,\n resetDetector,\n videoConstraints,\n webcamKey,\n}: Omit<CameraScanChildProps, \"config\">) {\n const { media } = useContext(MediaContext) || {};\n const { translate } = useContext(LanguageContext) || {};\n\n return (\n <div className=\"App w-screen h-[100vh] relative common-ui-main\">\n <span\n onClick={() => {\n resetScan();\n resetDetector();\n }}\n className=\"fixed right-[20px] top-[20px] z-[999]\"\n >\n <X className=\"text-[#fff]\" />\n </span>\n\n <div className=\"w-full h-full overflow-hidden \">\n {media === MEDIA_TYPES.MOBILE && (\n <Webcam\n key={webcamKey}\n audio={false}\n ref={webcamRef}\n screenshotQuality={1}\n videoConstraints={videoConstraints}\n mirrored\n screenshotFormat=\"image/jpeg\"\n onUserMedia={onUserMedia}\n onUserMediaError={onUserMediaError}\n style={{\n position: \"fixed\",\n top: 0,\n bottom: 0,\n zIndex: 0,\n width: \"100%\",\n height: \"100%\",\n objectFit: \"cover\",\n }}\n />\n )}\n </div>\n\n <div className=\"fixed bottom-[30px] w-full z-[999] flex flex-col gap-4 items-center justify-center\">\n {showPause && (\n <SpecificButton\n className=\"!w-[180px] !h-[40px] !py-[0] mx-auto \"\n prefix={pauseIcon}\n buttonText={translate?.(LanguageKeys.pause)}\n buttonFunc={onPause}\n btnSecondary\n />\n )}\n\n {pause ? (\n <div>\n <SpecificButton\n className=\"!w-[180px] !h-[40px] !py-[0] mx-auto\"\n buttonText={translate?.(LanguageKeys.restart)}\n buttonFunc={resetScan}\n btnSecondary\n />\n </div>\n ) : !recordingStarted ? (\n <SpecificButton\n className={`!w-[180px] !h-[40px] !py-[0] mx-auto !bg-[#ffffff] !text-[#000000] ${\n loadingCam ? \"!opacity-50\" : \"\"\n }`}\n buttonText={translate?.(LanguageKeys.startScan)}\n buttonFunc={startSendingVideoFrames}\n disabled={loadingCam}\n />\n ) : null}\n </div>\n\n <audio\n id=\"audioElement\"\n crossOrigin=\"anonymous\"\n preload=\"auto\"\n style={{\n position: \"absolute\",\n zIndex: -99999,\n }}\n src={`${voiceOverAssetsPath}scanAudioInstructions/silence.mp3`}\n />\n </div>\n );\n}\n\nexport default CameraScanChild;\n","import { useState, useRef, useEffect } from \"react\";\n// We keep this, but we will make it optional in the logic below\n\ntype KeypointTuple = [number, number, number];\n\n// GLOBAL SINGLETONS (Persist across component re-mounts)\n// This prevents reloading the heavy model if the user navigates away and back.\nlet globalLoadingPromise: Promise<boolean> | null = null;\nlet globalDetector: any = null;\nlet globalPoseLib: any = null;\n\nexport default function useTensorFlow() {\n const [consecutiveFronts, setConsecutiveFronts] = useState(0);\n const [spinPhase, setSpinPhase] = useState(0);\n const [isLoaded, setIsLoaded] = useState(false);\n\n const spinPhaseRef = useRef(0);\n const consecutiveFrontsRef = useRef(0);\n const mounted = useRef(true);\n\n useEffect(() => {\n mounted.current = true; // Reset on mount\n\n // 1. If already loaded globally, just use it.\n if (globalDetector) {\n setIsLoaded(true);\n return;\n }\n\n // 2. If not loading, start the process.\n if (!globalLoadingPromise) {\n globalLoadingPromise = loadDependenciesGlobal();\n }\n\n // 3. Wait for the global promise to resolve.\n globalLoadingPromise.then((success) => {\n if (success && mounted.current) {\n setIsLoaded(true);\n }\n });\n\n return () => {\n mounted.current = false;\n };\n }, []);\n\n /**\n * Loads TensorFlow/MediaPipe dynamically.\n * This logic is ISOLATED from the server.\n */\n async function loadDependenciesGlobal(): Promise<boolean> {\n // GUARDRAIL 1: Strict Environment Check\n // Ensure we are in a browser environment with media capabilities\n if (\n typeof window === 'undefined' || \n typeof navigator === 'undefined'\n ) {\n return false;\n }\n \n try {\n console.log(\"Starting TensorFlow preload...\");\n \n // GUARDRAIL 2: Dynamic Imports\n // These heavy modules are only fetched by the browser, never the server.\n const [poseLib, tfjsCore, tfjsBackend] = await Promise.all([\n import(\"@tensorflow-models/pose-detection\"),\n import(\"@tensorflow/tfjs-core\"),\n import(\"@tensorflow/tfjs-backend-webgl\") // Triggers backend registration side-effect\n ]);\n \n globalPoseLib = poseLib;\n \n // Explicitly set the backend to WebGL for performance\n await tfjsCore.setBackend('webgl');\n await tfjsCore.ready();\n console.log(\"TensorFlow backend ready (WebGL)\");\n\n // GUARDRAIL 3: Zero-Config Asset Loading\n // If TENSORFLOW_SOLUTION_PATH fails or is missing, we must fallback \n // to a CDN so the user doesn't have to copy files manually.\n const modelConfig = {\n runtime: \"mediapipe\",\n modelType: \"full\",\n solutionPath:\"https://cdn.jsdelivr.net/npm/@mediapipe/pose\", // Fallback for Zero Config\n } as const;\n\n try {\n globalDetector = await globalPoseLib.createDetector(\n globalPoseLib.SupportedModels.BlazePose,\n modelConfig\n );\n console.log(\"MediaPipe detector created successfully\");\n } catch (mediapipeError) {\n console.warn(\"MediaPipe failed, falling back to TFJS runtime:\", mediapipeError);\n \n // Fallback to TFJS runtime (slower but works without external assets)\n globalDetector = await globalPoseLib.createDetector(\n globalPoseLib.SupportedModels.BlazePose,\n {\n runtime: \"tfjs\",\n modelType: \"full\",\n }\n );\n console.log(\"TFJS detector created successfully\");\n }\n \n return true;\n } catch (error) {\n console.error(\"Failed to load TensorFlow dependencies:\", error);\n globalLoadingPromise = null; // Reset so retry is possible\n return false;\n }\n }\n\n function isFrontFrame(body: KeypointTuple[]) {\n // Guardrail: Ensure points exist before filtering\n if (!body || body.length === 0) return false;\n return body.filter((p) => p[2] > 0.7).length === 22;\n }\n\n const poseDetector = async (callback: () => void, webcamRef: any) => {\n // GUARDRAIL 4: Runtime Safety\n // Check if everything is truly ready before calculating\n if (!globalDetector || !mounted.current || !webcamRef?.current?.video) {\n return;\n }\n \n // Check if video is actually playing and has data\n const video = webcamRef.current.video;\n if (video.readyState < 2) return; \n\n try {\n const poses = await globalDetector.estimatePoses(\n video,\n { flipHorizontal: false }\n );\n\n if (!poses || !poses.length) return;\n\n const keypoints = poses[0].keypoints;\n \n // Standardize keypoints\n const body: KeypointTuple[] = keypoints\n .slice(11) // Ignore face keypoints (0-10)\n .map((kp: any) => [\n kp.x > 0 && kp.x < 720 ? kp.x : -1,\n kp.y > 0 && kp.y < 1280 ? kp.y : -1,\n kp.score ?? 0,\n ]);\n\n if (spinPhaseRef.current === 0 && !isFrontFrame(body)) {\n setConsecutiveFronts((prev) => prev + 1);\n }\n\n if (spinPhaseRef.current === 1 && isFrontFrame(body)) {\n setConsecutiveFronts((prev) => prev + 1);\n }\n\n if (spinPhaseRef.current === 2) callback();\n } catch (err) {\n console.error(\"Pose detection error:\", err);\n }\n };\n\n useEffect(() => {\n consecutiveFrontsRef.current = consecutiveFronts;\n if (consecutiveFronts > 6 && spinPhaseRef.current < 2) {\n setSpinPhase((s) => s + 1);\n setConsecutiveFronts(0);\n }\n }, [consecutiveFronts]);\n\n useEffect(() => {\n spinPhaseRef.current = spinPhase;\n }, [spinPhase]);\n\n const resetDetector = () => {\n spinPhaseRef.current = 0;\n consecutiveFrontsRef.current = 0;\n setConsecutiveFronts(0);\n setSpinPhase(0);\n };\n\n const retryLoading = async () => {\n if (!globalDetector && !globalLoadingPromise) {\n globalLoadingPromise = loadDependenciesGlobal();\n const success = await globalLoadingPromise;\n if (success && mounted.current) {\n setIsLoaded(true);\n }\n }\n };\n\n return { \n poseDetector, \n isLoaded,\n spinPhase,\n resetDetector,\n retryLoading\n };\n}","/* eslint-disable no-use-before-define */\nimport React, { useCallback, useContext, useEffect, useMemo, useRef, useState } from \"react\";\nimport CameraScanChild from \"./CameraScanChild\";\nimport posthog from \"../../utils/posthog\";\nimport Webcam from \"react-webcam\";\nimport { ScanningComponentProps } from \"../../types/interfaces\";\nimport useTensorFlow from \"../../customHooks/useTensorFlow\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\nimport {\n\tgenerateUuid,\n\tgetCurrentTimeInSeconds,\n\tgetPreferredMediaRecorderTypes,\n\tgetRecordingExtension,\n\tgetRecordingMimeType,\n\tgetSupportedMediaRecorderTypes,\n\thandleScanTimeCapture,\n\trescanSupportCaptureEvent,\n} from \"../../utils/utils\";\nimport speechService from \"../../utils/service/speechService\";\nimport { videoConstraintsExact, videoConstraintsFallback, videoTypes, voiceOverAssetsPath } from \"../../utils/constants\";\n\nlet id: string | null = null;\nlet audioTimeoutId: ReturnType<typeof setTimeout> | undefined;\nlet lastVideoPlayed: {\n\tskipCount: number;\n\tno_of_times_skipped: number;\n\taudioName: string;\n} | null = null;\n\nfunction ScanningComponent({ setIsScanLocked, resetDetector, scanID, setIsVideoUploaded, setScanFailsError, setScanStartTime, setScanUniqueKey, userDetails }: Omit<ScanningComponentProps, \"config\">) {\n\tconst { gender, heightInCm, email, shopDomain } = userDetails;\n\tconst webcamRef = useRef<Webcam | null>(null);\n\tconst mediaRecorderRef = useRef<MediaRecorder | null>(null);\n\tconst recordedFpsRef = useRef<number | null>(null);\n\tconst [recordedChunks, setRecordedChunks] = useState<Blob[]>([]);\n\tconst [loadingCam, setLoadingCam] = useState(true);\n\tconst [recordingStarted, setRecordingStarted] = useState(false);\n\tconst [faceDone, setFaceDone] = useState(false);\n\tconst [mediaRecorderStopped, setMediaRecorderStopped] = useState(true);\n\tconst [isScanning, setIsScanning] = useState(false);\n\tconst captureVideoTimeOutHandle = useRef<ReturnType<typeof setTimeout> | null>(null);\n\tconst [audioSource, setAudioSource] = useState(\"\");\n\tconst [audioSourceList, setAudioSourceList] = useState<string[]>([]);\n\tconst [emptyAudioSource, setEmptyAudioSource] = useState(\"\");\n\tconst [startAgain, setStartAgain] = useState(false);\n\tconst [showRestart, setRestart] = useState(false);\n\tconst [pause, setPause] = useState(false);\n\tconst [events, setEvents] = useState<any>([]);\n\tconst [showPause, setShowPause] = useState(false);\n\tconst allowAudioToPlay = useRef(true);\n\tconst { poseDetector } = useTensorFlow();\n\tconst firstScan = useRef(true);\n\tconst poseStoppedRef = useRef(false);\n\tconst counterRef = useRef(0);\n\tconst consecutiveFalseRef = useRef(0);\n\tconst poseListenerRegistered = useRef(false);\n\tconst brilliantPlayedRef = useRef(false);\n\tconst postPoseProcessRef = useRef<() => void>(() => {});\n\tconst [supportedTypes, setSupportedTypes] = useState<string[]>([]);\n\tconst [webcamKey, setWebcamKey] = useState(0);\n\tconst [currentVideoConstraints, setCurrentVideoConstraints] = useState<MediaTrackConstraints>(videoConstraintsExact);\n\n\tconst { setStartGyro, uploadScanFile, setUploadLoading, swan, onScanStart, onCaptureComplete } = useContext(ParamsContext);\n\n\t// Refs to avoid stale closures in callbacks after pause/restart\n\tconst scanIDRef = useRef(scanID);\n\tuseEffect(() => { scanIDRef.current = scanID; }, [scanID]);\n\n\tconst eventsRef = useRef(events);\n\tuseEffect(() => { eventsRef.current = events; }, [events]);\n\n\tconst startSendingVideoFramesRef = useRef<() => void>(() => {});\n\n\tconst resetScan = () => {\n\t\tposeStoppedRef.current = false;\n\t\tclearTimeout(audioTimeoutId);\n\t\tif (captureVideoTimeOutHandle.current) clearTimeout(captureVideoTimeOutHandle.current);\n\t\tsetScanUniqueKey(generateUuid());\n\t\tsetScanFailsError(\"\");\n\t\tsetUploadLoading?.(false);\n\t\tsetRecordedChunks([]);\n\t\trecordedFpsRef.current = null;\n\t\tsetLoadingCam(true);\n\t\tsetRecordingStarted(false);\n\t\tsetFaceDone(false);\n\t\tsetMediaRecorderStopped(true);\n\t\tsetIsScanning(false);\n\t\tcaptureVideoTimeOutHandle.current = null;\n\t\tsetAudioSource(\"\");\n\t\tsetAudioSourceList([]);\n\t\tsetEmptyAudioSource(\"\");\n\t\tsetStartAgain(!startAgain);\n\t\tsetStartGyro(false);\n\t\tspeechService.stopAudio();\n\t\tsetRestart(false);\n\t\tsetPause(false);\n\t\tcounterRef.current = 0;\n\t\tconsecutiveFalseRef.current = 0;\n\t\tposeListenerRegistered.current = false;\n\t\tbrilliantPlayedRef.current = false;\n\t\tif (mediaRecorderRef.current !== null) {\n\t\t\tmediaRecorderRef.current.stop();\n\t\t}\n\n\t\tevents.forEach((el: any) => {\n\t\t\tif (mediaRecorderRef.current) mediaRecorderRef.current.removeEventListener(\"dataavailable\", el);\n\t\t});\n\n\t\tsetEvents([]);\n\t\tallowAudioToPlay.current = true;\n\t\tsetShowPause(false);\n\t\tsetIsVideoUploaded(false);\n\t\tif (webcamRef.current?.stream) {\n\t\t\tsetLoadingCam(false);\n\t\t}\n\t};\n\n\tconst handleUserMedia = useCallback(() => {\n\t\tsetTimeout(() => {\n\t\t\tsetLoadingCam(false);\n\t\t}, 1000);\n\t}, []);\n\n\tconst handleUserMediaError = useCallback(\n\t\t(error: string | DOMException) => {\n\t\t\tconsole.log(\"camera error\", error);\n\t\t\t// Retry once with relaxed constraints for broader device compatibility.\n\t\t\tif (currentVideoConstraints === videoConstraintsExact) {\n\t\t\t\tsetCurrentVideoConstraints(videoConstraintsFallback);\n\t\t\t\tsetWebcamKey((prev) => prev + 1);\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tsetLoadingCam(false);\n\t\t},\n\t\t[currentVideoConstraints],\n\t);\n\n\tconst handlePause = useCallback(() => {\n\t\tallowAudioToPlay.current = false;\n\t\tsetPause(true);\n\t\tsetShowPause(false);\n\t\tif (captureVideoTimeOutHandle.current) clearTimeout(captureVideoTimeOutHandle.current);\n\t\tif (mediaRecorderRef.current) {\n\t\t\tmediaRecorderRef.current.pause();\n\t\t}\n\t\tspeechService.stopAudio();\n\t\tswan.poseDetection.disconnect();\n\t\tcounterRef.current = 0;\n\t\tconsecutiveFalseRef.current = 0;\n\t\tposeListenerRegistered.current = false;\n\t\tbrilliantPlayedRef.current = false;\n\t\tif (mediaRecorderRef.current) {\n\t\t\tmediaRecorderRef.current.stop();\n\t\t}\n\t\tevents.forEach((el: any) => {\n\t\t\tif (mediaRecorderRef.current) mediaRecorderRef.current.removeEventListener(\"dataavailable\", el);\n\t\t});\n\t\tsetEvents([]);\n\t\tclearTimeout(audioTimeoutId);\n\t}, [mediaRecorderRef, events, speechService, allowAudioToPlay]);\n\n\t// const supportedTypes = videoTypes.filter((type) => MediaRecorder.isTypeSupported(type));\n\tconst preferredVideoTypes = useMemo(() => getPreferredMediaRecorderTypes(), []);\n\tconst recordingMimeType = getRecordingMimeType(supportedTypes);\n\tconst recordingExtension = getRecordingExtension(recordingMimeType);\n\n\tconst stopRecording = useCallback(async () => {\n\t\tallowAudioToPlay.current = true;\n\t\tawait speechService.playAudio(`${voiceOverAssetsPath}SpotOn.mp3`);\n\t\tsetPause(false);\n\t\tsetShowPause(false);\n\t\tif (captureVideoTimeOutHandle.current) clearTimeout(captureVideoTimeOutHandle.current);\n\t\tsetRecordingStarted(false);\n\t}, [captureVideoTimeOutHandle, speechService]);\n\n\tconst handleSocket = useCallback(async () => {\n\t\ttry {\n\t\t\tid = await swan.poseDetection.connect();\n console.log(id, \"websocket connected\");\n\n\t\t\tposthog.capture(`${shopDomain}/pose_detection_connected`, {\n\t\t\t\tscanID: scanIDRef.current,\n\t\t\t\temail,\n\t\t\t\tid,\n\t\t\t});\n\t\t} catch (error) {\n\t\t\tconsole.log(error, \"while connecting websocket\");\n\t\t}\n\t}, [shopDomain, email, swan]);\n\n\tconst handleReScan = useCallback(async () => {\n\t\trescanSupportCaptureEvent({\n\t\t\teventName: `${shopDomain}/rescan`,\n\t\t\temail,\n\t\t\tscanID: scanIDRef.current,\n\t\t\theight: heightInCm,\n\t\t\tgender,\n\t\t\tstatus: false,\n\t\t});\n\t\tallowAudioToPlay.current = false;\n\t\tswan.poseDetection.disconnect();\n\t\tresetScan();\n\t\tawait handleSocket();\n\t\tstartSendingVideoFramesRef.current();\n\t}, [handleSocket, email, shopDomain, heightInCm, gender]);\n\n\tconst handleSpinDataAvailable = useCallback(\n\t\t({ data }: BlobEvent) => {\n\t\t\tif (data && data.size > 0 && allowAudioToPlay.current) {\n\t\t\t\tsetRecordedChunks((prev) => prev.concat(data));\n\t\t\t\tif (!poseStoppedRef.current && webcamRef.current) {\n\t\t\t\t\tposeStoppedRef.current = true;\n\t\t\t\t\tposeDetector(() => {\n\t\t\t\t\t\tstopRecording();\n\t\t\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/tensorFlow`,\n\t\t\t\t\t\t\tscanID: scanIDRef.current,\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t\tmessage: \"recording stopped by tensorflow \",\n\t\t\t\t\t\t});\n\t\t\t\t\t}, webcamRef);\n\t\t\t\t}\n\t\t\t}\n\t\t},\n\t\t[stopRecording, handleScanTimeCapture, shopDomain, email, webcamRef],\n\t);\n\n\tconst playAudio = useCallback(async () => {\n\t\tif (audioSourceList.length > 0 && allowAudioToPlay.current) {\n\t\t\tawait speechService.playAudio(voiceOverAssetsPath + audioSourceList[audioSourceList.length - 1]);\n\t\t\tif (allowAudioToPlay.current) {\n\t\t\t\taudioTimeoutId = setTimeout(playAudio, 2000);\n\t\t\t}\n\t\t}\n\t}, [audioSourceList, allowAudioToPlay]);\n\n\t\tconst handleStartCaptureClick = useCallback(() => {\n\t\tif (mediaRecorderRef && mediaRecorderRef.current) {\n\t\t\tmediaRecorderRef.current.stop();\n\t\t}\n\t\tsetRecordingStarted(true);\n\t\ttry {\n\t\t\tif (webcamRef && webcamRef.current && webcamRef.current.stream) {\n\t\t\t\t// Capture FPS from stream settings\n\t\t\t\tconst videoTrack = webcamRef.current.stream.getVideoTracks()[0];\n\t\t\t\tif (videoTrack) {\n\t\t\t\t\tconst settings = videoTrack.getSettings();\n\t\t\t\t\trecordedFpsRef.current = settings.frameRate || null;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\tconst mediaRecorderOptions: MediaRecorderOptions = {\n\t\t\t\t\tvideoBitsPerSecond: 1_500_000,\n\t\t\t\t};\n\t\t\t\tif (supportedTypes[0]) {\n\t\t\t\t\tmediaRecorderOptions.mimeType = supportedTypes[0];\n\t\t\t\t}\n\t\t\t\tmediaRecorderRef.current = new MediaRecorder(webcamRef.current.stream, mediaRecorderOptions);\n\t\t\t\tmediaRecorderRef.current.addEventListener(\"dataavailable\", handleSpinDataAvailable);\n\n\t\t\t\tsetEvents([...events, handleSpinDataAvailable]);\n\t\t\t\tmediaRecorderRef.current.start(1000);\n\t\t\t\tsetMediaRecorderStopped(false);\n\t\t\t}\n\t\t} catch (e) {\n\t\t\tconsole.log(\"error while using media recorder\", e);\n\t\t}\n\t}, [webcamRef, supportedTypes, handleSpinDataAvailable, events]);\n\n\tconst postPoseProcess = useCallback(async () => {\n\t\tif (captureVideoTimeOutHandle.current) {\n\t\t\tclearTimeout(captureVideoTimeOutHandle.current);\n\t\t}\n\t\thandleStartCaptureClick();\n\t\tif (allowAudioToPlay.current) {\n\t\t\tcaptureVideoTimeOutHandle.current = setTimeout(async () => {\n\t\t\t\tif (allowAudioToPlay.current) {\n\t\t\t\t\tawait speechService.playAudio(`${voiceOverAssetsPath}SpotOn.mp3`);\n\t\t\t\t\tsetRecordingStarted(false);\n\t\t\t\t}\n\t\t\t}, 15000);\n\t\t}\n\t\tsetFaceDone(true);\n\t\tif (allowAudioToPlay.current) {\n\t\t\tawait speechService.playAudio(`${voiceOverAssetsPath}Spin.mp3`);\n\t\t}\n\t}, [handleStartCaptureClick, allowAudioToPlay, speechService]);\n\n\tuseEffect(() => {\n\t\tpostPoseProcessRef.current = postPoseProcess;\n\t}, [postPoseProcess]);\n\n\tconst handleDataAvailable = useCallback(\n\t\t({ data }: BlobEvent) => {\n\t\t\tif (data.size > 0 && swan.poseDetection.connected()) {\n\t\t\t\tif (!poseListenerRegistered.current) {\n\t\t\t\t\tposeListenerRegistered.current = true;\n\t\t\t\t\tswan.poseDetection.poseStatus(async (data: any) => {\n\t\t\t\t\t\tconsole.log(\"poseStatus data:\", data); \n\t\t\t\t\t\tif (data && data.audio && data.audio.length > 0) {\n\t\t\t\t\t\t\tposthog.capture(`${shopDomain}/pose_instruction_received`, {\n\t\t\t\t\t\t\t\tscanID: scanIDRef.current,\n\t\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t\t\taudio: data.audio,\n\t\t\t\t\t\t\t\tstatus: data.status,\n\t\t\t\t\t\t\t\tsid: data.sid,\n\t\t\t\t\t\t\t\tcounter: counterRef.current,\n\t\t\t\t\t\t\t\tconsecutiveFalse: consecutiveFalseRef.current,\n\t\t\t\t\t\t\t\ttype:\"body_scan_instruction\",\n\t\t\t\t\t\t\t});\n\t\t\t\t\t\t\tconst audio = document.querySelector(\"#audioElement\") as HTMLAudioElement | null;\n\t\t\t\t\t\t\tif (data.status === true && data.sid === id) {\n\t\t\t\t\t\t\t\tconsecutiveFalseRef.current = 0;\n\t\t\t\t\t\t\t\tif (counterRef.current < 2) {\n\t\t\t\t\t\t\t\t\tcounterRef.current += 1;\n\t\t\t\t\t\t\t\t\tif (counterRef.current === 1 && !brilliantPlayedRef.current) {\n\t\t\t\t\t\t\t\t\t\tbrilliantPlayedRef.current = true;\n\t\t\t\t\t\t\t\t\t\tclearTimeout(audioTimeoutId);\n\t\t\t\t\t\t\t\t\t\tspeechService.stopAudio();\n\t\t\t\t\t\t\t\t\t\tawait speechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t\t\t} else if (counterRef.current === 2 && audio?.paused) {\n\t\t\t\t\t\t\t\t\t\tclearTimeout(audioTimeoutId);\n\t\t\t\t\t\t\t\t\t\tspeechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\tsetEmptyAudioSource(data.audio);\n\t\t\t\t\t\t\t\t\tclearTimeout(audioTimeoutId);\n\t\t\t\t\t\t\t\t\tswan.poseDetection.disconnect();\n\t\t\t\t\t\t\t\t\tposeListenerRegistered.current = false;\n\t\t\t\t\t\t\t\t\tsetTimeout(() => postPoseProcessRef.current(), 1000);\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tconsecutiveFalseRef.current += 1;\n\t\t\t\t\t\t\t\tif (consecutiveFalseRef.current >= 3) {\n\t\t\t\t\t\t\t\t\tcounterRef.current = 0;\n\t\t\t\t\t\t\t\t\tbrilliantPlayedRef.current = false;\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\tif (audio?.paused && (!lastVideoPlayed || lastVideoPlayed?.audioName !== data.audio)) {\n\t\t\t\t\t\t\t\t\tlastVideoPlayed = {\n\t\t\t\t\t\t\t\t\t\tskipCount: 2,\n\t\t\t\t\t\t\t\t\t\tno_of_times_skipped: 0,\n\t\t\t\t\t\t\t\t\t\taudioName: data.audio,\n\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t\tspeechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t\t} else if (lastVideoPlayed?.audioName === data.audio && audio?.paused) {\n\t\t\t\t\t\t\t\t\tif (lastVideoPlayed && lastVideoPlayed.no_of_times_skipped >= lastVideoPlayed.skipCount) {\n\t\t\t\t\t\t\t\t\t\tlastVideoPlayed.no_of_times_skipped = 0;\n\t\t\t\t\t\t\t\t\t\tspeechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\t\t\tif (lastVideoPlayed) lastVideoPlayed.no_of_times_skipped += 1;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t\tif (webcamRef?.current && webcamRef.current.getScreenshot() !== null) {\n\t\t\t\t\tswan.poseDetection.videoEmit({\n\t\t\t\t\t\timage: webcamRef.current.getScreenshot() || \"\",\n\t\t\t\t\t\tscanId: scanIDRef.current,\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t\t},\n\t\t[webcamRef, swan, speechService, shopDomain, email],\n\t);\n\n\tconst startSendingVideoFrames = useCallback(async () => {\n\t\tsetIsScanning(true);\n\t\tsetStartGyro(true);\n\t\tonScanStart?.();\n\t\tif (firstScan.current) {\n\t\t\tfirstScan.current = false;\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: \"scan started\",\n\t\t\t\tscanID: scanIDRef.current,\n\t\t\t\tstatus: \"success\",\n\t\t\t\temail,\n\t\t\t});\n\t\t}\n\t\tsetScanStartTime(getCurrentTimeInSeconds());\n\t\tsetLoadingCam(true);\n\t\tif (allowAudioToPlay.current) {\n\t\t\tawait speechService.playAudio(`${voiceOverAssetsPath}StartScan.mp3`);\n\t\t}\n\t\tif (allowAudioToPlay.current) {\n\t\t\tawait speechService.playAudio(`${voiceOverAssetsPath}LiftArmsAndHoldAtHip.mp3`);\n\t\t}\n\t\tif (allowAudioToPlay.current) {\n\t\t\tsetIsScanning(false);\n\t\t\tsetRecordingStarted(true);\n\t\t\tsetRestart(true);\n\t\t\tsetShowPause(true);\n\t\t\thandleUserMedia();\n\t\t}\n\t\ttry {\n\t\t\tif (webcamRef && webcamRef.current && webcamRef.current.stream && allowAudioToPlay.current) {\n\t\t\t\t// Capture FPS from stream settings\n\t\t\t\tconst videoTrack = webcamRef.current.stream.getVideoTracks()[0];\n\t\t\t\tif (videoTrack) {\n\t\t\t\t\tconst settings = videoTrack.getSettings();\n\t\t\t\t\trecordedFpsRef.current = settings.frameRate || null;\n\t\t\t\t}\n\t\t\t\t\n\t\t\t\tconst mediaRecorderOptions: MediaRecorderOptions = {\n\t\t\t\t\tvideoBitsPerSecond: 1_500_000,\n\t\t\t\t};\n\t\t\t\tif (supportedTypes[0]) {\n\t\t\t\t\tmediaRecorderOptions.mimeType = supportedTypes[0];\n\t\t\t\t}\n\t\t\t\tif (allowAudioToPlay.current) {\n\t\t\t\t\tmediaRecorderRef.current = new MediaRecorder(webcamRef.current.stream, mediaRecorderOptions);\n\t\t\t\t}\n\t\t\t\tif (allowAudioToPlay.current && mediaRecorderRef.current) {\n\t\t\t\t\tmediaRecorderRef.current.addEventListener(\"dataavailable\", handleDataAvailable);\n\t\t\t\t}\n\n\t\t\t\tsetEvents([...eventsRef.current, handleDataAvailable]);\n\t\t\t\tif (mediaRecorderRef.current) mediaRecorderRef.current.start(1000);\n\t\t\t\tsetMediaRecorderStopped(false);\n\t\t\t\tif (allowAudioToPlay.current) {\n\t\t\t\t\taudioTimeoutId = setTimeout(playAudio, 2000);\n\t\t\t\t}\n\t\t\t}\n\t\t} catch (err) {\n\t\t\tconsole.log(\"error ----------\", err);\n\t\t}\n\t}, [webcamRef, supportedTypes, handleDataAvailable, playAudio, allowAudioToPlay, onScanStart]);\n\n\tuseEffect(() => { startSendingVideoFramesRef.current = startSendingVideoFrames; }, [startSendingVideoFrames]);\n\n\tuseEffect(() => {\n\t\tif (shopDomain) {\n\t\t\tsetIsScanLocked(true);\n\t\t\thandleSocket();\n\t\t}\n\t\treturn () => {\n\t\t\tif (id) {\n\t\t\t\tswan.poseDetection.disconnect();\n\t\t\t\tposthog.capture(`${shopDomain}/pose_detection_disconnected`, {\n\t\t\t\t\tscanID: scanIDRef.current,\n\t\t\t\t\temail,\n\t\t\t\t\tid,\n\t\t\t\t});\n\t\t\t}\n\t\t\tevents.forEach((el: any) => {\n\t\t\t\tmediaRecorderRef?.current?.removeEventListener(\"dataavailable\", el);\n\t\t\t});\n\t\t};\n\t}, [shopDomain]);\n\tuseEffect(() => {\n\t\taudioSourceList.push(audioSource);\n\t}, [audioSource]);\n\n\tuseEffect(() => {\n\t\tsetAudioSourceList([]);\n\t}, [emptyAudioSource]);\n\tuseEffect(() => {\n\t\tconst bypassChecksToBackup = recordedChunks.length && recordedChunks.length > 0;\n\t\tif (!mediaRecorderStopped && bypassChecksToBackup && !recordingStarted && allowAudioToPlay.current && !pause) {\n\t\t\tif (mediaRecorderRef && mediaRecorderRef.current) {\n\t\t\t\tif (!recordingStarted) {\n\t\t\t\t\tmediaRecorderRef.current.stop();\n\t\t\t\t\tsetMediaRecorderStopped(true);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}, [mediaRecorderStopped, recordingStarted, recordedChunks, pause]);\n\n\tuseEffect(() => {\n\t\tconst bypassChecksToBackup = recordedChunks.length && recordedChunks.length > 0;\n\t\tif (!mediaRecorderStopped && bypassChecksToBackup && !recordingStarted) {\n\t\t\tif (mediaRecorderRef && mediaRecorderRef.current && allowAudioToPlay.current && !pause) {\n\t\t\t\tif (!recordingStarted) {\n\t\t\t\t\tconst videoFile = new File(recordedChunks, `${scanID}.${recordingExtension}`, {\n\t\t\t\t\t\ttype: recordingMimeType,\n\t\t\t\t\t});\n\t\t\t\t\tonCaptureComplete?.();\n\t\t\t\t\tsetUploadLoading?.(true);\n\t\t\t\t\tsetScanFailsError(\"\");\n\t\t\t\t\t// Pass FPS captured at recording time\n\t\t\t\t\tuploadScanFile?.(videoFile, recordedFpsRef.current);\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\tconsole.log(\"No video found to upload.\");\n\t\t}\n\t}, [mediaRecorderStopped, recordingStarted, recordedChunks, pause, mediaRecorderRef, allowAudioToPlay]);\n\tuseEffect(() => {\n\t\tconst supported = getSupportedMediaRecorderTypes(preferredVideoTypes, videoTypes);\n\t\tsetSupportedTypes(supported);\n\t}, [preferredVideoTypes]);\n\tuseEffect(() => {\n\t\tresetScan();\n\t}, []);\n\n\treturn (\n\t\t<CameraScanChild\n\t\t\tresetDetector={resetDetector}\n\t\t\tresetScan={resetScan}\n\t\t\tloadingCam={loadingCam}\n\t\t\tonPause={handlePause}\n\t\t\tshowRestart={showRestart}\n\t\t\tpause={pause}\n\t\t\tonReScan={handleReScan}\n\t\t\trecordingStarted={recordingStarted}\n\t\t\tisScanning={isScanning}\n\t\t\tstartSendingVideoFrames={startSendingVideoFrames}\n\t\t\tfaceDone={faceDone}\n\t\t\tstopRecording={stopRecording}\n\t\t\tshowPause={showPause}\n\t\t\twebcamRef={webcamRef}\n\t\t\tonUserMedia={handleUserMedia}\n\t\t\tonUserMediaError={handleUserMediaError}\n\t\t\tvideoConstraints={currentVideoConstraints}\n\t\t\twebcamKey={webcamKey}\n\t\t\t/>\n\t);\n}\n\nexport default React.memo(ScanningComponent);\n","/* eslint-disable no-use-before-define */\n\nimport { useEffect, useState, useRef } from \"react\";\nimport { IOSDeviceOrientationEvent } from \"../../customHooks/useGyroSensor\";\nimport LevelScreen from \"./LevelScreen\";\nimport ScanningComponent from \"./ScanningComponent\";\nimport { AngleDetectorProps } from \"../../types/interfaces\";\n\nexport default function AngleDetector({\n scanID,\n userDetails,\n setIsVideoUploaded,\n setScanFailsError,\n setScanStartTime,\n setScanUniqueKey,\n}: Omit<AngleDetectorProps, \"config\">) {\n const [angle, setAngle] = useState(90);\n const [calibrationOffset, setCalibrationOffset] = useState(0);\n const [countdown, setCountdown] = useState<number | null>(null);\n const [isScanning, setIsScanning] = useState(false);\n const [stabilityScore, setStabilityScore] = useState(0);\n const lastAnglesRef = useRef<number[]>([]);\n const [isScanLocked, setIsScanLocked] = useState(false);\n const calibratedAngle = angle - calibrationOffset;\n const isInTargetRange = calibratedAngle >= 80 && calibratedAngle <= 95;\n\n const resetDetector = () => {\n setAngle(90);\n setCalibrationOffset(0);\n setCountdown(null);\n setIsScanning(false);\n setStabilityScore(0);\n setIsScanLocked(false);\n lastAnglesRef.current = [];\n };\n\n useEffect(() => {\n if (typeof window !== \"undefined\" && window.DeviceOrientationEvent) {\n const handleOrientation = (event: DeviceOrientationEvent) => {\n if (event.beta !== null) {\n let newAngle = Math.abs(event.beta);\n if (event.gamma !== null) {\n newAngle *= Math.cos((event.gamma * Math.PI) / 180);\n }\n newAngle = Math.max(0, Math.min(180, newAngle));\n setAngle(newAngle);\n }\n };\n\n const requestPermission = async () => {\n const DeviceOrientation =\n DeviceOrientationEvent as unknown as IOSDeviceOrientationEvent;\n\n if (\n DeviceOrientation &&\n typeof DeviceOrientation.requestPermission === \"function\"\n ) {\n try {\n const permission = await DeviceOrientation.requestPermission();\n if (permission === \"granted\") {\n // wrappedHandler = timeout(handleOrientation, 2000);\n window.addEventListener(\"deviceorientation\", handleOrientation);\n }\n } catch (e) {\n console.error(\"Permission error\", e);\n }\n } else {\n // wrappedHandler = handleOrientation;\n window.addEventListener(\"deviceorientation\", handleOrientation);\n }\n\n return () => {\n if (handleOrientation) {\n window.removeEventListener(\"deviceorientation\", handleOrientation);\n }\n };\n };\n\n document.addEventListener(\"click\", requestPermission, { once: true });\n }\n }, []);\n\n useEffect(() => {\n lastAnglesRef.current = [\n ...lastAnglesRef.current.slice(-4),\n calibratedAngle,\n ];\n\n if (lastAnglesRef.current.length >= 5) {\n const variation =\n Math.max(...lastAnglesRef.current) - Math.min(...lastAnglesRef.current);\n if (isInTargetRange && variation < 2) {\n setStabilityScore((prev) => Math.min(100, prev + 10));\n } else {\n setStabilityScore((prev) => Math.max(0, prev - 20));\n }\n }\n\n if (!isInTargetRange && (countdown !== null || isScanning)) {\n resetCountdown();\n }\n }, [calibratedAngle, isInTargetRange, countdown, isScanning]);\n\n\tuseEffect(() => {\n\t\tif (stabilityScore >= 100 && countdown === null && !isScanning ) {\n\t\t\tstartCountdown();\n\t\t}\n\t\tif (stabilityScore < 50 && countdown !== null) {\n\t\t\tresetCountdown();\n\t\t}\n\t}, [stabilityScore, countdown, isScanning]);\n\n const startCountdown = () => {\n setCountdown(3);\n const timer = setInterval(() => {\n setCountdown((prev) => {\n if (prev === null || prev <= 1) {\n clearInterval(timer);\n setIsScanning(true);\n return null;\n }\n return prev - 1;\n });\n }, 1000);\n };\n\n const resetCountdown = () => {\n setCountdown(null);\n if (!isScanLocked) {\n setIsScanning(false);\n }\n };\n\n\n return (\n <LevelScreen\n angle={calibratedAngle}\n countdown={countdown}\n isScanning={isScanning}\n isInTargetRange={isInTargetRange}\n stabilityScore={stabilityScore}\n >\n {true && (\n <ScanningComponent\n setIsScanLocked={setIsScanLocked}\n resetDetector={resetDetector}\n scanID={scanID}\n userDetails={userDetails}\n setIsVideoUploaded={setIsVideoUploaded}\n setScanFailsError={setScanFailsError}\n setScanStartTime={setScanStartTime}\n setScanUniqueKey={setScanUniqueKey}\n />\n )}\n </LevelScreen>\n );\n}\n","import { useContext } from \"react\";\nimport {Dialog} from \"@mui/material\";\nimport { getBrowserName } from \"../utils/utils\";\nimport { LanguageContext } from \"../utils/context/languageContext\";\nimport { LanguageKeys } from \"../utils/languageKeys\";\nimport { Config } from \"../types/interfaces\";\n\nfunction Modal({ message, config }: { message?: string; config?: Config }) {\n const { translate } = useContext(LanguageContext) || {};\n\n\n\n return (\n <Dialog open className=\"confirm-modal common-ui-main\">\n <div className=\"modal-main\">\n <div className=\"text-center\">\n {message ? (\n <>\n <h2\n style={{\n fontFamily:\n config?.style?.heading?.headingFontFamily ||\n \"SeriouslyNostalgic Fn\",\n fontSize: config?.style?.heading?.headingFontSize || \"32px\",\n color: config?.style?.heading?.headingColor || \"#000\",\n fontWeight:\n config?.style?.heading?.headingFontWeight || \"normal\",\n }}\n >\n {translate?.(LanguageKeys.cameraAlreadyInUse)}\n </h2>\n <p\n className=\"mt-[0.5rem] text-sm\"\n style={{\n fontFamily:\n config?.style?.base?.baseFontFamily || \"Inter, sans-serif\",\n fontSize: config?.style?.base?.baseFontSize || \"16px\",\n color: config?.style?.base?.baseTextColor || \"#000\",\n }}\n >\n {translate?.(LanguageKeys.tryClosingBrowser)}\n </p>\n </>\n ) : (\n <>\n <h2\n style={{\n fontFamily:\n config?.style?.heading?.headingFontFamily ||\n \"SeriouslyNostalgic Fn\",\n fontSize: config?.style?.heading?.headingFontSize || \"32px\",\n color: config?.style?.heading?.headingColor || \"#000\",\n fontWeight:\n config?.style?.heading?.headingFontWeight || \"normal\",\n }}\n >\n {translate?.(LanguageKeys.checkCameraSettings)}\n </h2>\n <p\n className=\"mt-[0.5rem] text-sm\"\n style={{\n fontFamily:\n config?.style?.base?.baseFontFamily || \"Inter, sans-serif\",\n fontSize: config?.style?.base?.baseFontSize || \"16px\",\n color: config?.style?.base?.baseTextColor || \"#000\",\n }}\n >{`${translate?.(\n LanguageKeys.setting\n )} > ${getBrowserName()} > ${translate?.(\n LanguageKeys.enableCameraPermissions\n )}`}</p>\n </>\n )}\n </div>\n </div>\n </Dialog>\n );\n}\n\nexport default Modal;\n","import { useEffect, useState, useRef, useCallback, Dispatch, SetStateAction } from \"react\";\nimport Webcam from \"react-webcam\";\nimport Modal from \"../Modal\";\nimport LoadingScreen from \"../LoadingScreen\";\nimport { checkCameraPermission } from \"../../utils/utils\";\nimport { videoConstraints } from \"../../utils/constants\";\n\n\nfunction CameraPermission({ setShowDrawer, config,loader }: { setShowDrawer?: Dispatch<SetStateAction<boolean>>; config?: any,loader?:string }) {\n\tconst [showDeniedModal, setShowDeniedModal] = useState({\n\t\tdisabled: false,\n\t\tmessage: \"\",\n\t});\n\tconst [loading, setLoading] = useState(true);\n\tconst webcamRef = useRef(null);\n\n\tconst handleCameraPermission = useCallback(async () => {\n\t\tconst permission = await checkCameraPermission();\n\t\tsetShowDeniedModal(permission);\n\t\tsetLoading(false);\n\t}, []);\n\n\tuseEffect(() => {\n\t\thandleCameraPermission();\n\t}, []);\n\n\tif (loading) {\n\t\treturn <LoadingScreen url={loader} loaderType=\"black\" />\n\t}\n\n\tif (showDeniedModal?.disabled) {\n\t\treturn <Modal config={config} message={showDeniedModal?.message} />;\n\t}\n\treturn (\n\t\t<Webcam\n\t\t\taudio={false}\n\t\t\tref={webcamRef}\n\t\t\tscreenshotQuality={1}\n\t\t\tvideoConstraints={videoConstraints}\n\t\t\tmirrored\n\t\t\tonUserMedia={() => setShowDrawer?.(true)}\n\t\t\tscreenshotFormat=\"image/jpeg\"\n\t\t\tstyle={{\n\t\t\t\tposition: \"relative\",\n\t\t\t\ttop: 0,\n\t\t\t\tbottom: 0,\n\t\t\t\tzIndex: 0,\n\t\t\t\twidth: \"100%\",\n\t\t\t\theight: \"100%\",\n\t\t\t\tobjectFit: \"cover\",\n\t\t\t}}\n\t\t/>\n\t);\n}\n\nexport default CameraPermission;\n","\nimport React, { useEffect, useRef } from \"react\";\nimport videojs from \"video.js\";\nimport type Player from \"video.js/dist/types/player\";\nimport { videoPoster } from \"../../utils/constants\";\n\nfunction VideoPlayer({ link, onReady, wrapperClassName = \"[&_video]:rounded-t-[20px] w-full h-full\" }: { link?: string; wrapperClassName?: string; onReady?: (args: Player) => void }) {\n\tconst videoReference = useRef<HTMLVideoElement | null>(null);\n\tconst playerReference = useRef<Player | null>(null);\n\n\tlet videoJsOptions: any = {\n\t\tautoplay: true,\n\t\tcontrols: false,\n\t\tresponsive: true,\n\t\tfluid: true,\n\t\tmuted: true,\n\t\tnavigationUI: \"hide\",\n\t\tpreload: \"metadata\",\n\t\tposter: videoPoster,\n\t};\n\n\tuseEffect(() => {\n\t\tif (!playerReference.current && link && videoReference?.current) {\n\t\t\tconst player = videojs(videoReference.current, {\n\t\t\t\t...videoJsOptions,\n\t\t\t});\n\t\t\tplayer.ready(() => {\n\t\t\t\tplayerReference.current = player;\n\n\t\t\t\tconst updatedOptions = {\n\t\t\t\t\t...videoJsOptions,\n\t\t\t\t\tsources: [{ src: link, type: \"application/x-mpegURL\" }],\n\t\t\t\t};\n\n\t\t\t\tplayer.autoplay(updatedOptions.autoplay);\n\t\t\t\tplayer.src(updatedOptions.sources);\n\n\t\t\t\tonReady?.(player);\n\t\t\t});\n\t\t}\n\t}, [link, videoReference]);\n\n\tuseEffect(() => {\n\t\tconst player = playerReference.current;\n\t\treturn () => {\n\t\t\tif (player && !player.isDisposed()) {\n\t\t\t\tplayer.dispose();\n\t\t\t\tplayerReference.current = null;\n\t\t\t}\n\t\t};\n\t}, [playerReference]);\n\n\treturn (\n\t\t<div className={wrapperClassName}>\n\t\t\t<video ref={videoReference} muted className=\"video-js\" playsInline onDrag={(e) => e.preventDefault()} />\n\t\t</div>\n\t);\n}\nexport default VideoPlayer;\n","import { useContext, useState } from \"react\";\nimport {Dialog} from \"@mui/material\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport Header from \"../Header\";\nimport VideoPlayer from \"./VideoPlayer\";\nimport { ScanErrorMessageProps } from \"../../types/interfaces\";\nimport { generateUuid, handleErrorMessage } from \"../../utils/utils\";\nimport { GENDER, VIDEO_POSTER_GENDER_BASED } from \"../../utils/constants\";\nimport SpecificButton from \"../../atoms/specificButton/SpecificButton\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport { X } from \"lucide-react\";\nimport { useConfig } from \"../../utils/context/configContext\";\n\nfunction ScanErrorMessage({ scanFailsError, onNext, gender, setScanUniqueKey, setIsVideoUploaded }: ScanErrorMessageProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\tconst resolvedConfig = useConfig();\n\tconst [showModal, setShowModal] = useState(false);\n\n\tconst handleBtnClick = () => {\n\t\tif (onNext) {\n\t\t\tonNext?.();\n\t\t} else {\n\t\t\tsetScanUniqueKey?.(generateUuid());\n\t\t}\n\t\tsetIsVideoUploaded?.(false);\n\t};\n\n\tconst handleShowModal = () => {\n\t\tsetShowModal(!showModal);\n\t};\n\n\n\treturn (\n\t\t<>\n\t\t\t<div className=\"flex common-ui-main flex-col h-full max-w-[28rem] mx-auto w-full rounded-t-[20px] overflow-y-auto\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t<div className=\"w-full max-w-[28rem] mx-auto pt-[1rem] px-[1rem]\">\n\t\t\t\t\t<Header noTitle />\n\t\t\t\t</div>\n\t\t\t\t<div className=\"flex-1\">\n\t\t\t\t\t{scanFailsError && (\n\t\t\t\t\t\t<div className=\"px-[1rem]\">\n\t\t\t\t\t\t\t<h2\n\t\t\t\t\t\t\t\tclassName=\"text-center\"\n\t\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\t\tfontFamily: resolvedConfig?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\t\t\t\tfontSize: resolvedConfig?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\t\t\t\tcolor: resolvedConfig?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\t\t\t\tfontWeight: resolvedConfig?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t\t>\n\t\t\t\t\t\t\t\t{translate?.(LanguageKeys.issueWithScan)}\n\t\t\t\t\t\t\t</h2>\n\t\t\t\t\t\t\t<p\n\t\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\t\tfontFamily: resolvedConfig?.style?.base?.baseFontFamily || \"Inter, sans-serif\",\n\t\t\t\t\t\t\t\t\tfontSize: resolvedConfig?.style?.base?.baseFontSize || \"16px\",\n\t\t\t\t\t\t\t\t\tcolor: resolvedConfig?.style?.base?.baseTextColor || \"#1E1E1E\",\n\t\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t\t>\n\t\t\t\t\t\t\t\t{translate?.(LanguageKeys.reason)}\n\t\t\t\t\t\t\t</p>\n\t\t\t\t\t\t\t<p\n\t\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\t\tfontFamily: resolvedConfig?.style?.base?.baseFontFamily || \"Inter, sans-serif\",\n\t\t\t\t\t\t\t\t\tfontSize: resolvedConfig?.style?.base?.baseFontSize || \"16px\",\n\t\t\t\t\t\t\t\t\tcolor: resolvedConfig?.style?.base?.baseTextColor || \"#1E1E1E\",\n\t\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t\t>\n\t\t\t\t\t\t\t\t{handleErrorMessage(scanFailsError)}\n\t\t\t\t\t\t\t</p>\n\n\t\t\t\t\t\t\t<img className=\"my-[0.5rem] aspect-[2/1.4] w-full object-cover\" onClick={handleShowModal} src={VIDEO_POSTER_GENDER_BASED[gender]} alt=\"icon\" />\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{scanFailsError && (\n\t\t\t\t\t<div className=\"p-[1rem] flex gap-[0.5rem]\">\n\t\t\t\t\t\t<SpecificButton disabled={false} buttonText={translate?.(LanguageKeys.scanAgain)} className=\"!shadow-none\" buttonFunc={handleBtnClick} />\n\t\t\t\t\t</div>\n\t\t\t\t)}\n\t\t\t</div>\n\t\t\t{showModal && (\n\t\t\t\t<Dialog className=\"w-screen h-screen video-modal\" onClose={handleShowModal} open={showModal}>\n\t\t\t\t\t<div className=\"flex justifyEnd \">\n\t\t\t\t\t\t<span className=\"closeBtn\" onClick={handleShowModal}>\n\t\t\t\t\t\t\t<X className=\"absolute right-[8px] top-[8px] text-white z-[9]\" />\n\t\t\t\t\t\t</span>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className=\"aspect-video object-cover rounded-[20px] \">\n\t\t\t\t\t\t<VideoPlayer link={gender ? GENDER[gender].PRE_LINK : GENDER.male.PRE_LINK} wrapperClassName=\"w-screen h-screen fixed top-[0] left-[0]\" />\n\t\t\t\t\t</div>\n\t\t\t\t</Dialog>\n\t\t\t)}\n\t\t</>\n\t);\n}\n\nexport default ScanErrorMessage;\n","import { useCallback, useContext, useEffect, useState } from \"react\";\nimport {Box,Drawer} from \"@mui/material\";\nimport posthog from \"../utils/posthog\";\nimport Header from \"./Header\";\nimport CameraPermission from \"./bodyScan/CameraPermission\";\nimport { SignUpProps } from \"../types/interfaces\";\nimport { CLOTHING_BANNER_SCAN, CLOTHING_CUSTOM_FIT_SCAN, CLOTHING_CUSTOM_SCAN, maleMeasurementProgress, measurementProgress } from \"../utils/constants\";\nimport { GenderType } from \"../utils/enums\";\nimport SpecificButton from \"../atoms/specificButton/SpecificButton\";\nimport { LanguageContext } from \"../utils/context/languageContext\";\nimport { LanguageKeys } from \"../utils/languageKeys\";\n\nfunction SignUp({\n\tscanId,\n\tuserDetails,\n\tconfig,\n\tisVideoUploadedCorrect,\n\tisMeasurementAvailable,\n\tonComplete,\n\tisSuccess,\n\tonCustomScanSuccess,\n\tswan,\n}: SignUpProps) {\n\tconst { gender, shopDomain, heightInCm, deviceFocalLength, userName, email, scanType } = userDetails || {};\n\tconst isCustom = [CLOTHING_CUSTOM_SCAN, CLOTHING_BANNER_SCAN, CLOTHING_CUSTOM_FIT_SCAN].includes(scanType ?? \"\");\n\tconst [showDrawer, setShowDrawer] = useState(true);\n\tconst handlePostHogEvent = () => {\n\t\tposthog.capture(shopDomain??\"\", {\n\t\t\tscanID: scanId,\n\t\t\temail: email,\n\t\t\theight: heightInCm,\n\t\t\tfocalLength: deviceFocalLength,\n\t\t\tclothesFit: \"0\",\n\t\t\tgender: gender,\n\t\t});\n\t};\n\n\tconst handleSignUp = useCallback(async () => {\n\t\ttry {\n\t\t\tif (isCustom) {\n\t\t\t\tawait swan?.auth.addUser({\n\t\t\t\t\tscanId,\n\t\t\t\t\temail,\n\t\t\t\t\tname: userName,\n\t\t\t\t\tgender,\n\t\t\t\t\theight: heightInCm,\n\t\t\t\t});\n\t\t\t\tonCustomScanSuccess?.();\n\t\t\t}\n\t\t\thandlePostHogEvent();\n\t\t} catch (error) {\n\t\t\tconsole.log(error);\n\t\t}\n\t}, [isCustom]);\n\n\tconst { translate } = useContext(LanguageContext) || {};\n\n\tuseEffect(() => {\n\t\tif (isVideoUploadedCorrect || isMeasurementAvailable) {\n\t\t\thandleSignUp();\n\t\t}\n\t}, [isVideoUploadedCorrect, isMeasurementAvailable]);\n\n\tconst showNextButton = isSuccess || isVideoUploadedCorrect || isMeasurementAvailable;\n\n\treturn (\n\t\t<Box className=\"flex h-full w-full flex-col common-ui-main\">\n\t\t\t<div className=\"h-full w-full flex-col items-center justify-center flex\">\n\t\t\t\t<CameraPermission loader={config?.loader} setShowDrawer={setShowDrawer} />\n\t\t\t\t<Drawer\n\t\t\t\t\topen={showDrawer}\n\t\t\t\t\tonClose={(_, 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\tclassName=\"camera-drawer\"\n\t\t\t\t\tanchor=\"bottom\"\n\t\t\t\t>\n\t\t\t\t\t<div\n\t\t\t\t\t\tclassName=\"max-w-[28rem] mx-auto w-full h-full flex text-center flex-col justify-between items-center bg-primary rounded-t-[30px] p-[1rem]\"\n\t\t\t\t\t\tstyle={{ background: config?.style?.base?.backgroundColor }}\n\t\t\t\t\t>\n\t\t\t\t\t\t<div className=\"w-full h-full flex flex-col\">\n\t\t\t\t\t\t\t<Header title={translate?.(LanguageKeys.measurementsBeingTaken)} />\n\t\t\t\t\t\t\t<div className=\"flex items-center justify-center flex-1\">\n\t\t\t\t\t\t\t\t<video preload=\"auto\" className=\"max-h-[calc(100vh-450px)] mx-auto w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? maleMeasurementProgress : measurementProgress} type=\"video/mp4\" />\n\t\t\t\t\t\t\t\t</video>\n\t\t\t\t\t\t\t</div>\n\t\t\t\t\t\t\t{showNextButton && <SpecificButton className=\"!w-[180px] mx-auto\" buttonText={translate?.(LanguageKeys.next)} buttonFunc={()=>{onComplete?.()\n\t\t\t\t\t\t\t}} />}\n\t\t\t\t\t\t</div>\n\t\t\t\t\t</div>\n\t\t\t\t</Drawer>\n\t\t\t</div>\n\t\t</Box>\n\t);\n}\nexport default SignUp;\n","\"use client\";\nimport { useContext } from \"react\";\nimport AngleDetector from \"./AngleDetector\";\nimport CameraPermission from \"./CameraPermission\";\nimport Modal from \"../Modal\";\nimport { Drawer } from \"@mui/material\";\nimport ScanErrorMessage from \"./ScanErrorMessage\";\nimport LoadingScreen from \"../LoadingScreen\";\nimport SignUp from \"../Signup\";\nimport { useConfig } from \"../../utils/context/configContext\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\n\nexport const BodyScanSteps = () => {\n\tconst {\n\t\tuserDetails,\n\t\tonRetry,\n\t\tisError,\n\t\tisSuccess,\n\t\tgender,\n\t\tscanUniqueKey,\n\t\tscanFailsError,\n\t\tsetScanUniqueKey,\n\t\tsetIsVideoUploaded,\n\t\tisMeasurementAvailable,\n\t\tisVideoUploadedCorrect,\n\t\tloading,\n\t\tshowDeniedModal,\n\t\tuploadLoading,\n\t\tsetScanFailsError,\n\t\tsetScanStartTime,\n\t\tresetScan,\n\t\tonCustomScanSuccess,\n\t\tswan,\n\t\tonComplete,\n\t} = useContext(ParamsContext);\n\tconst resolvedConfig = useConfig();\n\n\tif (isError) {\n\t\treturn (\n\t\t\t<>\n\t\t\t\t<CameraPermission />\n\t\t\t\t<Drawer\n\t\t\t\t\tanchor=\"bottom\"\n\t\t\t\t\topen={true}\n\t\t\t\t\tclassName=\"camera-drawer\"\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>\n\t\t\t\t\t<ScanErrorMessage scanFailsError onNext={onRetry} gender={gender} />\n\t\t\t\t</Drawer>\n\t\t\t</>\n\t\t);\n\t}\n\n\tif (isSuccess) {\n\t\treturn <SignUp isSuccess onComplete={onComplete} config={resolvedConfig} />;\n\t}\n\n\tif (loading) {\n\t\treturn (\n\t\t\t<div className=\"flex top-0 !mt-0 left-0 z-[999] absolute justify-center items-center 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\tif (showDeniedModal.disabled) {\n\t\treturn <Modal />;\n\t}\n\n\tif (!uploadLoading && !scanFailsError && !isMeasurementAvailable && !isVideoUploadedCorrect) {\n\t\treturn (\n\t\t\t<AngleDetector\n\t\t\t\tscanID={scanUniqueKey}\n\t\t\t\tuserDetails={userDetails}\n\t\t\t\tsetIsVideoUploaded={setIsVideoUploaded}\n\t\t\t\tsetScanFailsError={setScanFailsError}\n\t\t\t\tsetScanStartTime={setScanStartTime}\n\t\t\t\tsetScanUniqueKey={setScanUniqueKey}\n\t\t\t/>\n\t\t);\n\t}\n\tif ((uploadLoading || isMeasurementAvailable || isVideoUploadedCorrect) && !scanFailsError) {\n\t\treturn (\n\t\t\t<SignUp\n\t\t\t\tscanId={scanUniqueKey}\n\t\t\t\tisMeasurementAvailable={isMeasurementAvailable}\n\t\t\t\tuserDetails={userDetails}\n\t\t\t\tisVideoUploadedCorrect={isVideoUploadedCorrect}\n\t\t\t\tonCustomScanSuccess={onCustomScanSuccess}\n\t\t\t\tswan={swan}\n\t\t\t\tonComplete={onComplete}\n\t\t\t\tconfig={resolvedConfig}\n\t\t\t/>\n\t\t);\n\t}\n\treturn (\n\t\t<>\n\t\t\t<CameraPermission />\n\t\t\t<Drawer\n\t\t\t\tanchor=\"bottom\"\n\t\t\t\topen\n\t\t\t\tclassName=\"camera-drawer\"\n\t\t\t\tonClose={(event, reason) => {\n\t\t\t\t\tif (reason === \"backdropClick\") {\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\t\t\t\t}}\n\t\t\t>\n\t\t\t\t<ScanErrorMessage\n\t\t\t\t\tscanFailsError={scanFailsError}\n\t\t\t\t\tonNext={() => {\n\t\t\t\t\t\tonRetry?.();\n\t\t\t\t\t\tresetScan();\n\t\t\t\t\t}}\n\t\t\t\t\tsetScanUniqueKey={setScanUniqueKey}\n\t\t\t\t\tgender={gender}\n\t\t\t\t\tsetIsVideoUploaded={setIsVideoUploaded}\n\t\t\t\t/>\n\t\t\t</Drawer>\n\t\t</>\n\t);\n};\n","\"use client\";\nimport { useCallback, useEffect, useMemo, useState } from \"react\";\nimport useGyroSensor from \"../../customHooks/useGyroSensor\";\nimport usePosthogPageview from \"../../customHooks/usePosthogPageview\";\nimport {\n\tcheckCameraPermission,\n\tcreateObjectMetadataArray,\n\tgenerateUuid,\n\tgetCurrentTimeInSeconds,\n\tgetVideoFPS,\n\thandleErrorMessage,\n\thandleScanTimeCapture,\n\thandleWebSocketCapture,\n} from \"../../utils/utils\";\nimport LanguageContextProvider from \"../../utils/context/languageContext\";\nimport { ConfigProvider } from \"../../utils/context/configContext\";\nimport { getSwanService } from \"../../utils/service/swanService\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\nimport MediaContextProvider from \"../../utils/context/mediaContext\";\nimport { BodyScanProps, UserDetails } from \"../../types/interfaces\";\nimport { BodyScanSteps } from \"./BodyScanSteps\";\n\nconst clothesFit = \"0\";\n\nexport const BodyScan: React.FC<BodyScanProps> = (props) => {\n\tconst {\n\t\tconfig,\n\t\tonRetry,\n\t\tonScanStart,\n\t\tonCaptureComplete,\n\t\tonUploadStart,\n\t\tonUploadEnd,\n\t\tonMeasurementSocketStart,\n\t\tonMeasurementSocketClose,\n\t\tonIntermediateScanSuccess,\n\t\tonScanError,\n\t\tonScanSuccess,\n\t\tonCustomScanSuccess,\n\t\tonComplete,\n\t\tisError,\n\t\tisSuccess,\n\t} = props;\n\tconst userDetails: UserDetails = {\n\t\temail: \"\",\n\t\tshopDomain: \"\",\n\t\tgender: \"male\",\n\t\theightInCm: 0,\n\t\tscanType: \"\",\n\t\tdeviceFocalLength: 0,\n\t\tdeviceModelName: \"\",\n\t\tsourceTag: \"\",\n\t\t...(props.userDetails ?? {}),\n\t};\n\tconst token = props.token ?? \"\";\n\tconst { gender, scanType, shopDomain, heightInCm, email, deviceFocalLength, deviceModelName, callbackUrl, sourceTag } = userDetails;\n\tconst swan = useMemo(() => getSwanService(token), [token]);\n\tconst [uploadLoading, setUploadLoading] = useState(false);\n\tconst [isVideoUploadedCorrect, setIsVideoUploadedCorrect] = useState(false);\n\tconst [isMeasurementAvailable, setIsMeasurementAvailable] = useState(false);\n\tconst [showDeniedModal, setShowDeniedModal] = useState({\n\t\tdisabled: false,\n\t\tmessage: \"\",\n\t});\n\tconst [scanFailsError, setScanFailsError] = useState(\"\");\n\tconst [isVideoUploaded, setIsVideoUploaded] = useState(false);\n\tconst [loading, setLoading] = useState(true);\n\tconst [startGyro, setStartGyro] = useState(false);\n\tconst [scanUniqueKey, setScanUniqueKey] = useState(\"\");\n\tconst [scanStartTime, setScanStartTime] = useState<number | null>(getCurrentTimeInSeconds());\n\tconst { gyroData } = useGyroSensor(startGyro);\n\tusePosthogPageview();\n\n\tconst resetScan = useCallback(() => {\n\t\tsetScanUniqueKey(generateUuid());\n\t\tsetScanFailsError(\"\");\n\t\tsetUploadLoading(false);\n\t\tsetIsVideoUploaded(false);\n\t}, []);\n\n\tconst onError = (data: any) => {\n\t\tonScanError?.({ ...data, message: handleErrorMessage(data) });\n\t\tclearScanStartTimeAndScanId();\n\t\tsetIsMeasurementAvailable(false);\n\t\tsetScanFailsError(data);\n\t\tsetIsVideoUploadedCorrect(false);\n\t\thandleScanTimeCapture({\n\t\t\teventName: `${shopDomain}/measurement_failed/fit-view`,\n\t\t\tscanID: scanUniqueKey,\n\t\t\tstatus: \"failed\",\n\t\t\temail,\n\t\t\tmessage: handleErrorMessage(data),\n\t\t});\n\t\tif (scanStartTime)\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/scan_completion_time`,\n\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\tstatus: \"failed\",\n\t\t\t\tcompletionTime: getCurrentTimeInSeconds() - scanStartTime,\n\t\t\t\temail,\n\t\t\t});\n\t};\n\n\tconst clearScanStartTimeAndScanId = () => {\n\t\tsetScanStartTime(null);\n\t\tsetScanUniqueKey(\"\");\n\t};\n\n\tconst onSuccess = useCallback(\n\t\tasync (data: any) => {\n\t\t\tif (data && data?.scanStatus === \"success\" && data?.resultType === \"intermediate\" && data?.code === 200) {\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: `${shopDomain}/measurement_success/intermediate`,\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tstatus: \"success\",\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t\t\tsetIsVideoUploadedCorrect(true);\n\t\t\t\tonIntermediateScanSuccess?.(data);\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tonScanSuccess?.(data);\n\t\t\tsetIsVideoUploadedCorrect(true);\n\t\t\tclearScanStartTimeAndScanId();\n\t\t\tconst scanIdToSet = scanUniqueKey;\n\t\t\tsetIsMeasurementAvailable(true);\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/measurement_success/fit-view`,\n\t\t\t\tscanID: scanIdToSet,\n\t\t\t\tstatus: \"success\",\n\t\t\t\temail,\n\t\t\t});\n\t\t\tif (scanStartTime)\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: `${shopDomain}/scan_completion_time`,\n\t\t\t\t\tscanID: scanIdToSet,\n\t\t\t\t\tstatus: \"success\",\n\t\t\t\t\tcompletionTime: getCurrentTimeInSeconds() - scanStartTime,\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t},\n\t\t[scanType, shopDomain, scanUniqueKey],\n\t);\n\n\tconst handleMeasurementRecommendation = (scanId?: string) => {\n\t\tsetIsMeasurementAvailable(false);\n\t\tsetScanFailsError(\"\");\n\t\tsetIsVideoUploadedCorrect(false);\n\t\tswan.measurement.handleMeasurementSocket({\n\t\t\tscanId: scanId || scanUniqueKey,\n\t\t\tonPreopen: () => {\n\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tconnection: \"pre_open\",\n\t\t\t\t\ttype: \"measurement_recommendation\",\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t\t},\n\t\t\tonOpen: () => {\n\t\t\t\tonMeasurementSocketStart?.();\n\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tconnection: \"open\",\n\t\t\t\t\ttype: \"measurement_recommendation\",\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t\t},\n\t\t\tonClose: () => {\n\t\t\t\tonMeasurementSocketClose?.();\n\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tconnection: \"close\",\n\t\t\t\t\ttype: \"measurement_recommendation\",\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t\t},\n\t\t\tonError: (err) => {\n\t\t\t\tonError(err);\n\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tconnection: \"error\",\n\t\t\t\t\ttype: \"measurement_recommendation\",\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t\t},\n\t\t\tonSuccess: (data) => {\n\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tconnection: \"success\",\n\t\t\t\t\ttype: \"measurement_recommendation\",\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t\t\tonSuccess(data);\n\t\t\t},\n\t\t});\n\t};\n\n\tconst uploadScanFile = useCallback(\n\t\tasync (file: File, recordedFps?: number | null) => {\n\t\t\tconst fileSizeInMB = Number((file.size / (1024 * 1024)).toFixed(2));\n\t\t\t// Use FPS captured at recording time, fallback to calculation if not available\n\t\t\tconst videoFps = recordedFps !== undefined && recordedFps !== null ? recordedFps : await getVideoFPS(file);\n\t\t\tconst arrayMetaData = createObjectMetadataArray({\n\t\t\t\tgender,\n\t\t\t\tfocal_length: `${deviceFocalLength}`,\n\t\t\t\theight: `${heightInCm}`,\n\t\t\t\tcustomer_store_url: shopDomain,\n\t\t\t\tclothes_fit: clothesFit,\n\t\t\t\tscan_type: scanType,\n\t\t\t\tcallback_url: callbackUrl,\n\t\t\t\tsource_tag: sourceTag,\n\t\t\t});\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/body_scan_meta_data`,\n\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\temail,\n\t\t\t\tdata: JSON.stringify([...arrayMetaData, { fileSizeInMB, video_fps: videoFps || 0 }]),\n\t\t\t});\n\t\t\tonUploadStart?.();\n\t\t\ttry {\n\t\t\t\tawait swan.fileUpload.uploadFileFrontend({\n\t\t\t\t\tfile,\n\t\t\t\t\tarrayMetaData,\n\t\t\t\t\tscanId: scanUniqueKey,\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t\t\tawait swan.fileUpload.setDeviceInfo({\n\t\t\t\t\tmodel: deviceModelName,\n\t\t\t\t\tdetection: \"manual\",\n\t\t\t\t\tgyro: gyroData,\n\t\t\t\t\tscanId: scanUniqueKey,\n\t\t\t\t});\n\t\t\t\tconsole.log(\"video successfully uploaded\");\n\t\t\t\tsetIsVideoUploaded(false);\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: `${shopDomain}/scan_success`,\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tstatus: \"success\",\n\t\t\t\t\temail,\n\t\t\t\t\tdata: JSON.stringify(arrayMetaData),\n\t\t\t\t});\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: \"scan finished\",\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tstatus: \"success\",\n\t\t\t\t\temail,\n\t\t\t\t});\n\t\t\t\tsetScanStartTime(getCurrentTimeInSeconds());\n\t\t\t\tsetTimeout(() => {\n\t\t\t\t\tsetIsVideoUploaded(true);\n\t\t\t\t}, 3000);\n\t\t\t} catch (error:any) {\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: \"scan finished\",\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tstatus: \"failed\",\n\t\t\t\t\temail,\n\t\t\t\t\tmessage: handleErrorMessage(error),\n\t\t\t\t});\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: `${shopDomain}/scan_failed`,\n\t\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\t\tstatus: \"failed\",\n\t\t\t\t\temail,\n\t\t\t\t\tmessage: handleErrorMessage(error),\n\t\t\t\t\tdata: JSON.stringify(arrayMetaData),\n\t\t\t\t});\n\t\t\t\tsetScanFailsError(handleErrorMessage(error));\n\t\t\t\tsetIsVideoUploaded(false);\n\t\t\t\tonScanError?.({...error, message: handleErrorMessage(error) });\n\t\t\t\tclearScanStartTimeAndScanId();\n\t\t\t} finally {\n\t\t\t\tsetStartGyro(false);\n\t\t\t\tonUploadEnd?.();\n\t\t\t}\n\t\t},\n\t\t[scanUniqueKey, gyroData, onUploadStart, onUploadEnd],\n\t);\n\t// const checkMeasurementStatus = useCallback(\n\t// \tasync (scanId: string) => {\n\t// \t\ttry {\n\t// \t\t\tconst res = await swan.measurement.getMeasurementResult(scanId);\n\t// \t\t\tconst isMeasured = res?.data?.isMeasured;\n\t// \t\t\tconst tempScanStartTime = scanStartTime || getCurrentTimeInSeconds();\n\t// \t\t\tconst currentTimeInSeconds = getCurrentTimeInSeconds();\n\t// \t\t\tconst fiveMinutesInSeconds = 5 * 60;\n\t// \t\t\tsetLoading(false);\n\t// \t\t\tif (isMeasured === false) {\n\t// \t\t\t\tclearScanStartTimeAndScanId();\n\t// \t\t\t\tsetScanUniqueKey(generateUuid());\n\t// \t\t\t\treturn;\n\t// \t\t\t}\n\t// \t\t\tsetUploadLoading(true);\n\t// \t\t\tif (isMeasured === true) {\n\t// \t\t\t\tawait onSuccess(null);\n\t// \t\t\t} else if (isMeasured === null && tempScanStartTime > currentTimeInSeconds + fiveMinutesInSeconds) {\n\t// \t\t\t\tonError(res?.data?.error);\n\t// \t\t\t} else {\n\t// \t\t\t\thandleMeasurementRecommendation(scanId);\n\t// \t\t\t}\n\t// \t\t} catch (error) {\n\t// \t\t\tconsole.log(error);\n\t// \t\t\tclearScanStartTimeAndScanId();\n\t// \t\t\tsetScanUniqueKey(generateUuid());\n\t// \t\t\tsetUploadLoading(false);\n\t// \t\t}\n\t// \t},\n\t// \t[onError, onSuccess, setScanUniqueKey],\n\t// );\n\n\tconst handleCameraPermission = useCallback(async () => {\n\t\tif (+heightInCm < 152.4 || +heightInCm > 213.36) {\n\t\t\tonScanError?.({\n\t\t\t\tmessage: \"Height must be between 152.4cm (5ft) and 213.36cm (7ft)\",\n\t\t\t});\n\t\t\treturn;\n\t\t}\n\n\t\tconst permission = await checkCameraPermission();\n\t\tif (permission.disabled) {\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/camera_activation`,\n\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\tstatus: \"failed\",\n\t\t\t\temail,\n\t\t\t});\n\t\t\tsetLoading(false);\n\t\t} else {\n\t\t\t// const scanId = scanUniqueKey;\n\t\t\t// if (scanId) {\n\t\t\t// \tcheckMeasurementStatus(scanId);\n\t\t\t// } else {\n\t\t\t// \tsetLoading(false);\n\t\t\t// }\n\t\t\tsetLoading(false);\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/camera_activation`,\n\t\t\t\tscanID: scanUniqueKey,\n\t\t\t\tstatus: \"success\",\n\t\t\t\temail,\n\t\t\t});\n\t\t}\n\t\tsetShowDeniedModal(permission);\n\t\tsetScanFailsError(\"\");\n\t\tsetIsVideoUploaded(false);\n\t}, [email, scanUniqueKey, setScanUniqueKey, shopDomain]);\n\n\tuseEffect(() => {\n\t\tif (isError || isSuccess) return;\n\t\tif (shopDomain) {\n\t\t\thandleCameraPermission();\n\t\t}\n\t}, [shopDomain, heightInCm, isError, isSuccess]);\n\n\tuseEffect(() => {\n\t\tif (isError || isSuccess) return;\n\t\tif (isVideoUploaded && shopDomain && scanUniqueKey) {\n\t\t\thandleMeasurementRecommendation();\n\t\t}\n\t}, [isVideoUploaded, shopDomain, scanUniqueKey, isError, isSuccess]);\n\treturn (\n\t\t<LanguageContextProvider>\n\t\t\t<ConfigProvider config={config}>\n\t\t\t\t<MediaContextProvider>\n\t\t\t\t\t<ParamsContext.Provider\n\t\t\t\t\t\tvalue={{\n\t\t\t\t\t\t\tuserDetails,\n\t\t\t\t\t\t\tonRetry,\n\t\t\t\t\t\t\tonScanError,\n\t\t\t\t\t\t\tisError,\n\t\t\t\t\t\t\tisSuccess,\n\t\t\t\t\t\t\tonScanSuccess,\n\t\t\t\t\t\t\tgender,\n\t\t\t\t\t\t\tscanUniqueKey,\n\t\t\t\t\t\t\tscanFailsError,\n\t\t\t\t\t\t\tsetScanUniqueKey,\n\t\t\t\t\t\t\tsetIsVideoUploaded,\n\t\t\t\t\t\t\tisMeasurementAvailable,\n\t\t\t\t\t\t\tisVideoUploadedCorrect,\n\t\t\t\t\t\t\tloading,\n\t\t\t\t\t\t\tshowDeniedModal,\n\t\t\t\t\t\t\tuploadLoading,\n\t\t\t\t\t\t\tsetStartGyro,\n\t\t\t\t\t\t\tuploadScanFile,\n\t\t\t\t\t\t\tsetUploadLoading,\n\t\t\t\t\t\t\tresetScan,\n\t\t\t\t\t\t\tsetScanFailsError,\n\t\t\t\t\t\t\tsetScanStartTime,\n\t\t\t\t\t\t\tonScanStart,\n\t\t\t\t\t\t\tonCaptureComplete,\n\t\t\t\t\t\t\tonCustomScanSuccess,\n\t\t\t\t\t\t\tonComplete,\n\t\t\t\t\t\t\tswan,\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t<BodyScanSteps />\n\t\t\t\t\t</ParamsContext.Provider>\n\t\t\t\t</MediaContextProvider>\n\t\t\t</ConfigProvider>\n\t\t</LanguageContextProvider>\n\t);\n};\n","import { useCallback, useEffect, useState } from \"react\";\n\ninterface GyroDataItem {\n alpha?: string;\n beta?: string;\n gamma?: string;\n timestamp?: string;\n}\n\nexport interface IOSDeviceOrientationEvent extends DeviceOrientationEvent {\n requestPermission?: () => Promise<PermissionState>;\n}\n\nfunction useGyroSensor(startGyro: boolean) {\n const [gyroData, setGyroData] = useState<GyroDataItem[]>([]);\n const [permissionGranted, setPermissionGranted] = useState(false);\n\n const handleOrientation = useCallback((event: DeviceOrientationEvent) => {\n try {\n const { alpha, beta, gamma } = event;\n setGyroData((prev: GyroDataItem[]) => [\n ...prev,\n {\n alpha: alpha?.toString() || undefined,\n beta: beta?.toString() || undefined,\n gamma: gamma?.toString() || undefined,\n timestamp: new Date().toISOString(),\n },\n ]);\n } catch (error) {\n console.log(error);\n }\n }, []);\n\n const requestPermission = useCallback(async () => {\n const DeviceOrientation =\n DeviceOrientationEvent as unknown as IOSDeviceOrientationEvent;\n\n if (\n typeof DeviceOrientation !== \"undefined\" &&\n typeof DeviceOrientation.requestPermission === \"function\"\n ) {\n try {\n const response = await DeviceOrientation.requestPermission();\n if (response === \"granted\") {\n setPermissionGranted(true);\n } else {\n console.warn(\"Device orientation permission denied.\");\n }\n } catch (error) {\n console.error(\"Error requesting device orientation permission:\", error);\n }\n } else {\n // Permission not required on non-iOS devices\n setPermissionGranted(true);\n }\n }, []);\n\n useEffect(() => {\n if (startGyro && permissionGranted) {\n window.addEventListener(\"deviceorientation\", handleOrientation);\n } else {\n setGyroData([]);\n }\n return () => {\n window.removeEventListener(\"deviceorientation\", handleOrientation);\n };\n }, [startGyro, permissionGranted, handleOrientation]);\n\n useEffect(() => {\n if (startGyro) {\n requestPermission();\n }\n }, [startGyro, requestPermission]);\n\n return { gyroData };\n}\n\nexport default useGyroSensor;\n"],"names":["MEDIA_TYPES","MediaContext","createContext","undefined","MediaContextProvider","children","size","setSize","useState","window","innerWidth","innerHeight","clearInputs","setClearInputs","updateSize","useEffect","useLayoutEffect","addEventListener","removeEventListener","media","mediaDetails","useMemo","_jsx","Provider","value","SwitchIcon$1","React","memo","_jsxs","width","height","viewBox","fill","xmlns","d","stroke","strokeWidth","strokeLinecap","strokeLinejoin","LevelScreen$1","angle","countdown","isScanning","isInTargetRange","stabilityScore","translate","useContext","LanguageContext","config","useConfig","getProximityToTarget","useCallback","Math","max","min","backgroundColor","style","angleDetector","successAngleBackground","proximity","successAngleLowBackground","round","handelTextColour","isLight","successAngleTextLightColor","successAngleTextDarkColor","getTextColor","handelColourCode","getColorCode","cd","fixVh","document","documentElement","setProperty","className","Header","noTitle","border","fontFamily","base","baseFontFamily","color","cn","transform","transition","SwitchIcon","LanguageKeys","startLevelCheck","leavePhone","placePhoneUpright","CameraScanChild","resetScan","loadingCam","onUserMedia","onPause","pause","recordingStarted","startSendingVideoFrames","showPause","webcamRef","onUserMediaError","resetDetector","videoConstraints","webcamKey","onClick","X","Webcam","audio","ref","screenshotQuality","mirrored","screenshotFormat","position","top","bottom","zIndex","objectFit","SpecificButton","prefix","pauseIcon","buttonText","buttonFunc","btnSecondary","restart","startScan","disabled","id","crossOrigin","preload","src","voiceOverAssetsPath","globalLoadingPromise","globalDetector","globalPoseLib","audioTimeoutId","lastVideoPlayed","ScanningComponent$1","setIsScanLocked","scanID","setIsVideoUploaded","setScanFailsError","setScanStartTime","setScanUniqueKey","userDetails","gender","heightInCm","email","shopDomain","useRef","mediaRecorderRef","recordedFpsRef","recordedChunks","setRecordedChunks","setLoadingCam","setRecordingStarted","faceDone","setFaceDone","mediaRecorderStopped","setMediaRecorderStopped","setIsScanning","captureVideoTimeOutHandle","audioSource","setAudioSource","audioSourceList","setAudioSourceList","emptyAudioSource","setEmptyAudioSource","startAgain","setStartAgain","showRestart","setRestart","setPause","events","setEvents","setShowPause","allowAudioToPlay","poseDetector","consecutiveFronts","setConsecutiveFronts","spinPhase","setSpinPhase","isLoaded","setIsLoaded","spinPhaseRef","consecutiveFrontsRef","mounted","async","loadDependenciesGlobal","navigator","console","log","poseLib","tfjsCore","tfjsBackend","Promise","all","resolve","then","require","import","setBackend","ready","modelConfig","runtime","modelType","solutionPath","createDetector","SupportedModels","BlazePose","mediapipeError","warn","error","isFrontFrame","body","length","filter","p","current","success","s","callback","video","readyState","poses","estimatePoses","flipHorizontal","keypoints","slice","map","kp","x","y","score","prev","err","retryLoading","useTensorFlow","firstScan","poseStoppedRef","counterRef","consecutiveFalseRef","poseListenerRegistered","brilliantPlayedRef","postPoseProcessRef","supportedTypes","setSupportedTypes","setWebcamKey","currentVideoConstraints","setCurrentVideoConstraints","videoConstraintsExact","setStartGyro","uploadScanFile","setUploadLoading","swan","onScanStart","onCaptureComplete","ParamsContext","scanIDRef","eventsRef","startSendingVideoFramesRef","clearTimeout","generateUuid","speechService","stopAudio","stop","forEach","el","stream","handleUserMedia","setTimeout","handleUserMediaError","videoConstraintsFallback","handlePause","poseDetection","disconnect","preferredVideoTypes","getPreferredMediaRecorderTypes","recordingMimeType","getRecordingMimeType","recordingExtension","getRecordingExtension","stopRecording","playAudio","handleSocket","connect","posthog","capture","handleReScan","rescanSupportCaptureEvent","eventName","status","handleSpinDataAvailable","data","concat","handleScanTimeCapture","message","handleStartCaptureClick","videoTrack","getVideoTracks","settings","getSettings","frameRate","mediaRecorderOptions","videoBitsPerSecond","mimeType","MediaRecorder","start","e","postPoseProcess","handleDataAvailable","connected","poseStatus","sid","counter","consecutiveFalse","type","querySelector","paused","audioName","no_of_times_skipped","skipCount","getScreenshot","videoEmit","image","scanId","getCurrentTimeInSeconds","push","bypassChecksToBackup","videoFile","File","supported","getSupportedMediaRecorderTypes","videoTypes","onReScan","AngleDetector","setAngle","calibrationOffset","setCalibrationOffset","setCountdown","setStabilityScore","lastAnglesRef","isScanLocked","calibratedAngle","DeviceOrientationEvent","handleOrientation","event","beta","newAngle","abs","gamma","cos","PI","requestPermission","DeviceOrientation","once","variation","resetCountdown","startCountdown","timer","setInterval","clearInterval","LevelScreen","ScanningComponent","Modal","Dialog","open","_Fragment","heading","headingFontFamily","fontSize","headingFontSize","headingColor","fontWeight","headingFontWeight","cameraAlreadyInUse","baseFontSize","baseTextColor","tryClosingBrowser","checkCameraSettings","setting","getBrowserName","enableCameraPermissions","CameraPermission","setShowDrawer","loader","showDeniedModal","setShowDeniedModal","loading","setLoading","handleCameraPermission","permission","checkCameraPermission","LoadingScreen","url","loaderType","VideoPlayer","link","onReady","wrapperClassName","videoReference","playerReference","videoJsOptions","autoplay","controls","responsive","fluid","muted","navigationUI","poster","videoPoster","player","videojs","updatedOptions","sources","isDisposed","dispose","playsInline","onDrag","preventDefault","ScanErrorMessage","scanFailsError","onNext","resolvedConfig","showModal","setShowModal","handleShowModal","background","issueWithScan","reason","handleErrorMessage","VIDEO_POSTER_GENDER_BASED","alt","scanAgain","onClose","GENDER","PRE_LINK","male","SignUp","isVideoUploadedCorrect","isMeasurementAvailable","onComplete","isSuccess","onCustomScanSuccess","deviceFocalLength","userName","scanType","isCustom","CLOTHING_CUSTOM_SCAN","CLOTHING_BANNER_SCAN","CLOTHING_CUSTOM_FIT_SCAN","includes","showDrawer","handleSignUp","auth","addUser","name","focalLength","clothesFit","showNextButton","Box","Drawer","_","anchor","title","measurementsBeingTaken","loop","autoPlay","GenderType","Male","maleMeasurementProgress","measurementProgress","next","BodyScanSteps","onRetry","isError","scanUniqueKey","uploadLoading","props","onUploadStart","onUploadEnd","onMeasurementSocketStart","onMeasurementSocketClose","onIntermediateScanSuccess","onScanError","onScanSuccess","deviceModelName","sourceTag","token","callbackUrl","getSwanService","setIsVideoUploadedCorrect","setIsMeasurementAvailable","isVideoUploaded","startGyro","scanStartTime","gyroData","setGyroData","permissionGranted","setPermissionGranted","alpha","toString","timestamp","Date","toISOString","useGyroSensor","usePosthogPageview","clearScanStartTimeAndScanId","onSuccess","scanStatus","resultType","code","scanIdToSet","completionTime","handleMeasurementRecommendation","measurement","handleMeasurementSocket","onPreopen","handleWebSocketCapture","connection","onOpen","onError","file","recordedFps","fileSizeInMB","Number","toFixed","videoFps","getVideoFPS","arrayMetaData","createObjectMetadataArray","focal_length","customer_store_url","clothes_fit","scan_type","callback_url","source_tag","JSON","stringify","video_fps","fileUpload","uploadFileFrontend","setDeviceInfo","model","detection","gyro","LanguageContextProvider","ConfigProvider"],"mappings":"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"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
"use strict";var e=require("react/jsx-runtime"),t=require("react"),a=require("./LoadingScreen-HQLN5ZOg.js"),n=require("lucide-react"),o=require("@mui/material");let r=null,s=!1;function c({faceScanId:e,onValidPose:n,onScanComplete:o,onModelReady:c,onStartRecording:i,shopDomain:l,isError:d,isSuccess:u}){const f=t.useRef(null),[g,h]=t.useState(0),[m,p]=t.useState(0),[y,S]=t.useState(!1),[x,v]=t.useState(!1),b=t.useRef(null),w=t.useRef(g),C=t.useRef(m),F=t.useRef([]),j=t.useRef(""),_=t.useRef(0),N=t.useRef(!1),E=t.useRef(0),k="undefined"!=typeof navigator&&(/iPad|iPhone|iPod/.test(navigator.userAgent)||"MacIntel"===navigator.platform&&navigator.maxTouchPoints>1),T=t.useMemo(()=>["Face front","Turn head fully left","Turn head fully right","Smile facing front"],[]),I=t.useMemo(()=>function(e,t){let a;return(...n)=>{a&&clearTimeout(a),a=setTimeout(()=>e(...n),t)}}(e=>{a.posthog.capture(`${l}/face_scan_pose_detection`,e)},4e3),[l]),R=t.useCallback((t,a,n,o,r,s)=>{const c=Date.now();if(!(c-E.current<4e3)){E.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],f=t.map(e=>e[2]).filter(e=>e>0),g=f.length>0?f.reduce((e,t)=>e+t,0)/f.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]),y=c[0]-l[0],S=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(g.toFixed(3)),avgBodyScore:parseFloat(m.toFixed(3)),faceKeypointsDetected:f.length,bodyKeypointsDetected:h.length,eyeDistance:parseFloat(p.toFixed(2)),noseToRightRatio:parseFloat((y/p).toFixed(3)),noseToLeftRatio:parseFloat((S/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})};I(v)}catch(e){console.warn("Failed to track pose detection:",e)}}},[e,I]),M=()=>{a.posthog.capture(`${l}/face_scan_detection_started`,{faceScanId:e,stage:w.current,timestamp:(new Date).toISOString(),stageName:T[w.current]}),setTimeout(()=>{v(!0)},1500)},D=async()=>{if("undefined"!=typeof window)try{console.log("Initializing Face Scan...");const e=await(async()=>{if("undefined"==typeof window)return null;if(r)return r;if(s)for(;s;)if(await new Promise(e=>setTimeout(e,100)),r)return r;s=!0;try{const[e,t]=await Promise.all([import("@tensorflow/tfjs"),Promise.resolve().then(function(){return require("./pose-detection.esm-BFEzAYBI.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="number"==typeof a.videoConstraints.width?a.videoConstraints.width:a.videoConstraints.width.ideal??a.videoConstraints.width.max??1280,o.height="number"==typeof a.videoConstraints.height?a.videoConstraints.height:a.videoConstraints.height.ideal??a.videoConstraints.height.max??720;const s=o.getContext("2d");if(s){s.fillStyle="black",s.fillRect(0,0,o.width,o.height);try{await n.estimatePoses(o)}catch(e){console.warn("Warmup frame failed (non-fatal):",e)}}return r=n,n}catch(e){return console.error("Failed to preload detector:",e),null}finally{s=!1}})();e&&(f.current=e,console.log("Face scan model loaded successfully"),S(!0),c?.())}catch(e){console.error("Error initializing face scan:",e)}};return t.useEffect(()=>{if(!d&&!u)return D(),()=>{(async()=>{try{S(!1)}catch(e){console.error("Error disposing resources:",e)}finally{h(0),p(0),w.current=0,C.current=0,F.current=[]}})()}},[d,u]),t.useEffect(()=>{w.current=g},[g]),t.useEffect(()=>{C.current=m},[m]),{faceScanDetector:async(t,r)=>{if(f.current&&t.current&&y&&x&&!(t.current.readyState<2))try{const s=await f.current.estimatePoses(t.current);if(s.length>0){const t=[],c=[];if(s[0].keypoints.forEach((e,a)=>{const n=e.score??0,o=[e.x??-1,e.y??-1,n];a<=10?t.push(o):c.push(o)}),r?.current){const e=r.current.getContext("2d"),{width:t,height:a}=r.current;if(e){e.clearRect(0,0,t,a);const n=Math.min(C.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 d=((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}})(w.current,t),u=c.slice(2).filter(e=>e[2]>.7).length<=2;F.current.push(d&&u),F.current.length>10&&F.current.shift();if(F.current.filter(Boolean).length>=2){const e=C.current+1;p(e)}else p(Math.max(0,C.current-1));if(C.current>=2){n?.();const t=w.current+1;p(0),F.current=[],h(t),v(!1),(async()=>{if(await a.speechService.playAudio(`${a.voiceOverAssetsPath}face-scan-vos/Sound-effect.mp3`),a.posthog.capture(`${l}/face_scan_detection_stage_completed`,{faceScanId:e,completedStage:w.current,nextStage:t,timestamp:(new Date).toISOString(),totalStages:4,stageName:T[w.current]}),1===t&&i&&i(),t>=4)o?.();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}`),M()}})()}R(t,c,w.current,d,u,C.current)}}catch(e){console.error("Pose detection error:",e)}},scanStage:g,setScanStage:h,consecutiveValid:m,isModelLoaded:y,resetScan:()=>{a.posthog.capture(`${l}/face_scan_reset`,{faceScanId:e,stage:w.current,timestamp:(new Date).toISOString(),stageName:T[w.current],consecutiveValid:C.current}),h(0),p(0),v(!1),w.current=0,C.current=0,F.current=[],j.current="",_.current=0,N.current=!1,b.current&&(speechSynthesis.cancel(),b.current=null)},enableVoiceCommands:()=>{if(N.current=!0,"undefined"!=typeof window&&k&&"speechSynthesis"in window){const e=new SpeechSynthesisUtterance("");e.volume=0,speechSynthesis.speak(e)}},startScanSequence:M,isScanningActive:x}}function i({onRetry:o,loading:r}){const{translate:s}=t.useContext(a.LanguageContext)||{},c=a.useConfig();return e.jsx("div",{className:"fixed common-ui-main top-[0] left-[0] z-[999] flex justify-center items-center w-full h-full",style:{background:c?.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}),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:c?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:c?.style?.heading?.headingFontSize||"32px",color:c?.style?.heading?.headingColor||"#000",fontWeight:c?.style?.heading?.headingFontWeight||"normal"},children:s?.(a.LanguageKeys.scanFailed)}),e.jsx("p",{className:"mb-[1.5rem]",style:{fontFamily:c?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:c?.style?.subheading?.subheadingFontSize||"14px",color:c?.style?.subheading?.subheadingColor||"#4b5563",fontWeight:c?.style?.subheading?.subheadingFontWeight||"normal"},children:s?.(a.LanguageKeys.clickToResetScan)}),e.jsx(a.SpecificButton,{disabled:r,className:"w-full h-[45px] shadow-[0px_1px_2px_0px_#00000040] text-[16px]",buttonText:s?.(a.LanguageKeys.resetScan),postfixIcon:e.jsx(n.ArrowRight,{}),buttonFunc:()=>o?.()})]})]})})}function l({stage:n,videoLoading:o,setVideoLoading:r,videoError:s,setVideoError:c,gender:i,btnConfig:l}){const{translate:d}=t.useContext(a.LanguageContext)||{},u=a.useConfig();return-1===n?e.jsxs("div",{className:"text-center common-ui-main p-[16px] w-full h-full",style:{background:u?.style?.base?.backgroundColor},children:[e.jsxs("div",{className:"w-full",children:[e.jsx(a.Header,{noTitle:!0}),e.jsx("h2",{style:{fontFamily:u?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:u?.style?.heading?.headingFontSize||"32px",color:u?.style?.heading?.headingColor||"#000",fontWeight:u?.style?.heading?.headingFontWeight||"normal"},children:d?.(a.LanguageKeys.faceVisible)}),o&&e.jsx("div",{className:"mb-4 flex items-center justify-center",style:{fontFamily:u?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:u?.style?.subheading?.subheadingFontSize||"14px",color:u?.style?.subheading?.subheadingColor||"#4b5563",fontWeight:u?.style?.subheading?.subheadingFontWeight||"normal"},children:d?.(a.LanguageKeys.loadingVideo)}),s&&e.jsx("div",{className:"mb-4 text-red-600",style:{fontFamily:u?.style?.subheading?.subheadingFontFamily||"'Inter', sans-serif",fontSize:u?.style?.subheading?.subheadingFontSize||"14px",color:"#ff0000",fontWeight:u?.style?.subheading?.subheadingFontWeight||"normal"},children:d?.(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:l.isDisabled,className:"!w-[60px] !h-[35px] !py-0 !px-0",buttonText:d?.(l.label),postfixIcon:l?.icon,buttonFunc:l.onClick})})]}):e.jsx("div",{className:"text-center common-ui-main p-[16px] w-full h-full",style:{background:u?.style?.base?.backgroundColor},children:e.jsxs("div",{className:"w-full",children:[e.jsx(a.Header,{noTitle:!0}),e.jsx("h3",{className:"mb-[0.5rem]",style:{fontFamily:u?.style?.heading?.headingFontFamily||"SeriouslyNostalgic Fn",fontSize:u?.style?.heading?.headingFontSize||"32px",color:u?.style?.heading?.headingColor||"#000",fontWeight:u?.style?.heading?.headingFontWeight||"normal"},children:d?.(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 d=({webcamRef:r,canvasRef:s})=>{const{isError:c,isSuccess:d,showLoader:u,hasError:f,resetScanState:g,showGuideCard:h,scanStage:m,videoLoading:p,setVideoLoading:y,videoError:S,setVideoError:x,gender:v,getButtonText:b,isScanning:w,startScan:C,isModelLoaded:F,onRetry:j}=t.useContext(a.ParamsContext),_=a.useConfig();return c?e.jsx(i,{onRetry:j}):d?e.jsx("div",{className:"fixed z-[9] common-ui-main w-full h-full",style:{background:_?.style?.base?.backgroundColor},children:e.jsx(a.LoadingScreen,{url:_?.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&&!f&&e.jsx("div",{className:"fixed z-[9] common-ui-main w-full h-full",style:{background:_?.style?.base?.backgroundColor},children:e.jsx(a.LoadingScreen,{url:_?.loader,loaderType:"black"})}),e.jsx("div",{className:"h-full flex-col common-ui-main relative w-full flex justify-center items-center text-center rounded-t-[20px]",style:{background:_?.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:r,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&&f&&e.jsx(i,{loading:u,onRetry:g}),h&&!f&&!u&&e.jsx(o.Drawer,{open:!0,className:"face-scan-small camera-draw",anchor:"bottom",onClose:(e,t)=>{},hideBackdrop:!0,children:e.jsx(l,{stage:m,videoLoading:p,setVideoLoading:y,videoError:S,setVideoError:x,gender:v,btnConfig:{label:b(),onClick:w?g:C,isDisabled:u||!F,icon:w?e.jsx(n.ArrowRight,{}):e.jsx(e.Fragment,{})}})})]})};exports.FaceScan=n=>{const{onScanSuccess:o,onScanError:r,onRetry:s,onScanStart:i,onCaptureComplete:l,onUploadStart:u,onUploadEnd:f,onMeasurementSocketStart:g,onMeasurementSocketClose:h,config:m,isError:p,isSuccess:y}=n,S=n.userDetails??{email:"",gender:"male",deviceFocalLength:0,shopDomain:"",bodyScanId:""},x=n.token??"",v=t.useRef(null),b=t.useRef(null),w=t.useRef(null),C=t.useRef([]),F=t.useRef(null),j=t.useRef(null),[_,N]=t.useState(!1),[E,k]=t.useState(!1),[T,I]=t.useState(!1),[R,M]=t.useState(!0),[D,P]=t.useState(!1),[L,A]=t.useState(!1),[O,z]=t.useState(!1),[$,H]=t.useState(a.generateUuid()),V=t.useMemo(()=>a.getSwanService(x),[x]),{email:W,gender:K,deviceFocalLength:U,shopDomain:B,callbackUrl:q,bodyScanId:G}=S,[J,X]=t.useState([]),[Q,Y]=t.useState(a.videoConstraintsExact),Z=t.useMemo(()=>a.getPreferredMediaRecorderTypes(),[]),ee=a.getRecordingMimeType(J),te=a.getRecordingExtension(ee);a.usePosthogPageview();const ae=t.useCallback(()=>{console.log("Stopping recording..."),w.current&&"recording"===w.current.state&&w.current.stop(),w.current=null},[]),ne=async()=>{if(l?.(),C.current){if(0===C.current.length)return console.error("No video data recorded"),z(!0),void N(!1);u?.(),N(!0);const e=new File(C.current,`${$}.${te}`,{type:ee}),t=j.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:C.current.length,estimated_duration_seconds:r},c=[{gender:K},{face_scan_id:$},{focal_length:`${U}`},{customer_store_url:B},{scan_type:"face_scan"},{body_scan_id:G}];q&&c.push({callback_url:q}),a.handleScanTimeCapture({eventName:`${B}/face_scan_meta_data`,faceScanId:$,email:W,data:JSON.stringify({metaData:c,videoData:s})});const i=Date.now();a.handleScanTimeCapture({eventName:`${B}/face_scan_upload_start`,faceScanId:$,email:W,startTime:i});try{const t=await V.fileUpload.faceScanFileUploader({file:e,arrayMetaData:c,objectKey:$,email:W,contentType:e.type}),n=Date.now(),o=n-i;a.handleScanTimeCapture({eventName:`${B}/face_scan_upload_complete`,faceScanId:$,email:W,completionTime:n,uploadDuration:o}),f?.(),console.log("✅ Upload successful",t),V.measurement.handlFaceScaneSocket({onPreopen:()=>{g?.(),a.handleWebSocketCapture({eventName:`${B}/webSocket`,faceScanID:$,connection:"pre_open",type:"faceScan_recommendation",email:W})},faceScanId:$,onOpen:()=>{a.handleWebSocketCapture({eventName:`${B}/webSocket`,faceScanID:$,connection:"open",type:"faceScan_recommendation",email:W}),console.log("websocket connect open")},onError:e=>{a.handleWebSocketCapture({eventName:`${B}/webSocket`,faceScanID:$,connection:"error",type:"faceScan_recommendation",email:W}),ge(e)},onSuccess:e=>{a.handleWebSocketCapture({eventName:`${B}/webSocket`,faceScanID:$,connection:"success",type:"faceScan_recommendation",email:W}),he(e)},onClose:()=>{h?.(),console.log("websocket connect close"),a.handleWebSocketCapture({eventName:`${B}/webSocket`,faceScanID:$,connection:"close",type:"faceScan_recommendation",email:W})}})}catch(e){f?.(),ge(e)}}},oe=t.useCallback(()=>{const e=v.current?.srcObject;if(!e)return;const t=e.getVideoTracks()[0].getSettings(),{width:a=1280,height:n=720}=t,o=document.createElement("canvas"),r=o.getContext("2d");a>n?(o.width=n,o.height=a):(o.width=a,o.height=n);const s=document.createElement("video");s.srcObject=e,s.muted=!0,s.playsInline=!0,s.play();const c=()=>{a>n?(r?.save(),r?.translate(o.width,0),r?.rotate(Math.PI/2),r?.drawImage(s,0,0,n,a),r?.restore()):r?.drawImage(s,0,0,a,n),requestAnimationFrame(c)};c();const i=t.frameRate??10;j.current=i;const l=o.captureStream(i),d={videoBitsPerSecond:15e5};let u;J.length>0&&(d.mimeType=J[0]);try{u=new MediaRecorder(l,d)}catch(e){return console.error("MediaRecorder init failed:",e),z(!0),void N(!1)}C.current=[],u.ondataavailable=e=>{e?.data&&e.data.size>0&&C.current&&C.current.push(e.data)},u.onstop=()=>{console.log("Recording stopped, total blobs:",C?.current?.length),ne()},u.start(100),w.current=u,console.log("Recording started successfully (portrait normalized)")},[J,ne]),{faceScanDetector:re,scanStage:se,setScanStage:ce,resetScan:ie,isModelLoaded:le,startScanSequence:de}=c({faceScanId:$,shopDomain:B,isError:p,isSuccess:y,onScanComplete:()=>{ae()},onModelReady:()=>{k(!0)},onStartRecording:()=>{console.log("Stage 0 completed - starting recording for stages 1-3"),oe()}}),ue=t.useCallback(()=>{le&&(i?.(),ce(0),a.speechService.playAudio(`${a.voiceOverAssetsPath}face-scan-vos/Face-forward.mp3`),setTimeout(()=>{I(!0),setTimeout(()=>{de()},200)},200))},[ce,I,de,le,i]),fe=t.useCallback(()=>{I(!1),M(!0),ae(),ie(),s?.(),C.current=[],j.current=null,w.current&&(w.current=null),z(!1),ce(-1),H(a.generateUuid()),v.current&&F.current&&(v.current.srcObject=F.current,console.log("Camera stream restored after reset"))},[ce,I,M,ae,ie,H]),ge=e=>{console.log(e,"ws error"),r?.(e),z(!0),N(!1),M(!1),a.handleScanTimeCapture({eventName:`${B}/faceScan`,faceScanId:$,status:"failed",email:W,data:JSON.stringify(e)})},he=e=>{console.log(e,"ws success"),e&&"intermediate"===e?.resultType&&a.handleScanTimeCapture({eventName:`${B}/faceScan_success/intermediate`,faceScanId:$,status:"success",email:W,data:JSON.stringify(e)}),o?.(e)};t.useEffect(()=>{if(!p&&!y)return navigator.mediaDevices.getUserMedia&&navigator.mediaDevices.getUserMedia({video:Q}).then(e=>{F.current=e,v.current&&(v.current.srcObject=e)}).catch(e=>{console.error("Error accessing webcam:",e),Q!==a.videoConstraintsExact?z(!0):Y(a.videoConstraintsFallback)}),()=>{F.current&&F.current.getTracks().forEach(e=>e.stop())}},[p,y,Q]),t.useEffect(()=>{if(!E||!T)return;const e=setInterval(()=>{re(v,b)},500);return()=>clearInterval(e)},[re,E,T]);return console.log("Model ready:",E,"Has error:",O,"Is scanning:",T,"showLoader",_),t.useEffect(()=>{const e=a.getSupportedMediaRecorderTypes(Z,a.videoTypes);X(e)},[Z]),t.useEffect(()=>{ce(-1)},[]),e.jsx(a.LanguageContextProvider,{children:e.jsx(a.ConfigProvider,{config:m,children:e.jsx(a.ParamsContext.Provider,{value:{isError:p,isSuccess:y,showLoader:_,hasError:O,resetScanState:fe,showGuideCard:R,scanStage:se,videoLoading:D,setVideoLoading:P,videoError:L,setVideoError:A,gender:K,getButtonText:()=>T?a.LanguageKeys.reset:le?a.LanguageKeys.start:a.LanguageKeys.loading,isScanning:T,startScan:ue,isModelLoaded:le,onRetry:s},children:e.jsx(d,{webcamRef:v,canvasRef:b})})})})};
|
|
2
|
+
//# sourceMappingURL=FaceScan-1Fwy5JDo.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"FaceScan-1Fwy5JDo.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 \"../utils/posthog\";\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 = typeof videoConstraints.width === \"number\" ? videoConstraints.width : videoConstraints.width.ideal ?? videoConstraints.width.max ?? 1280;\n dummyCanvas.height = typeof videoConstraints.height === \"number\" ? videoConstraints.height : videoConstraints.height.ideal ?? videoConstraints.height.max ?? 720;\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 isError,\n isSuccess,\n}: {\n faceScanId: string;\n onValidPose?: () => void;\n onScanComplete?: () => void;\n onModelReady?: () => void;\n onStartRecording: () => void;\n shopDomain: string;\n isError?: boolean;\n isSuccess?: boolean;\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 if (isError || isSuccess) return;\n initializeModels();\n return () => {\n disposeModelAndTf();\n };\n }, [isError, isSuccess]);\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 { useConfig } from \"../../utils/context/configContext\";\n\nfunction FaceScanErrorScreen({ onRetry, loading }: { onRetry?: () => void; loading?: boolean }) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\tconst config = useConfig();\n\n\treturn (\n\t\t<div className=\"fixed common-ui-main 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 />\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={() => onRetry?.()}\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 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\";\nimport { useConfig } from \"../../utils/context/configContext\";\nimport { Dispatch, ReactNode, SetStateAction } from \"react\";\nimport { Gender } from \"../../types/interfaces\";\n\ninterface IButtonConfig {\n\tisDisabled: boolean;\n\ticon?: ReactNode;\n\tonClick: () => void;\n\tlabel: string;\n}\n\ninterface FaceScanGuideProps {\n\tstage: number;\n\tvideoLoading: boolean;\n\tsetVideoLoading: Dispatch<SetStateAction<boolean>>;\n\tvideoError: boolean;\n\tsetVideoError: Dispatch<SetStateAction<boolean>>;\n\tgender: Gender;\n\tbtnConfig: IButtonConfig;\n}\n\nfunction FaceScanGuide({ stage, videoLoading, setVideoLoading, videoError, setVideoError, gender, btnConfig }: FaceScanGuideProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\tconst config = useConfig();\n\n\tif (stage === -1) {\n\t\treturn (\n\t\t\t<div className=\"text-center common-ui-main 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 />\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/>\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 common-ui-main 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 />\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<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 } 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 { useConfig } from \"../../utils/context/configContext\";\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\tonRetry,\n\t} = useContext(ParamsContext);\n\tconst resolvedConfig = useConfig();\n\tif (isError) {\n\t\treturn <FaceScanErrorScreen onRetry={onRetry} />;\n\t}\n\tif (isSuccess) {\n\t\treturn (\n\t\t\t<div className=\"fixed z-[9] common-ui-main 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] common-ui-main 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 common-ui-main 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} onRetry={resetScanState} />}\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\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 { ConfigProvider } from \"../../utils/context/configContext\";\nimport FaceScanStep from \"./FaceScanStep\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\n\nexport const FaceScan: React.FC<FaceScanProps> = (props) => {\n\tconst { onScanSuccess, onScanError, onRetry, onScanStart, onCaptureComplete, onUploadStart, onUploadEnd, onMeasurementSocketStart, onMeasurementSocketClose, config, isError, isSuccess } = props;\n\tconst userDetails: FaceScanMetaData = props.userDetails ?? {\n\t\temail: \"\",\n\t\tgender: \"male\",\n\t\tdeviceFocalLength: 0,\n\t\tshopDomain: \"\",\n\t\tbodyScanId: \"\",\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, bodyScanId } = 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\tonCaptureComplete?.();\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\n\t\t\tonUploadStart?.();\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{focal_length: `${deviceFocalLength}`},\n\t\t\t\t{ customer_store_url: shopDomain },\n\t\t\t\t{ scan_type: \"face_scan\" },\n\t\t\t\t{ body_scan_id: bodyScanId }\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\tonUploadEnd?.();\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\tonUploadEnd?.();\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\t// Fallback to landscape defaults (1280×720); portrait swap below handles orientation\n\t\tconst { width = 1280, height = 720 } = 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\t// Match recording FPS to actual camera FPS to avoid duplicate frames in the encoded video\n\t\tconst recordingFps = settings.frameRate ?? 10;\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\tisError,\n\t\tisSuccess,\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<ConfigProvider config={config}>\n\t\t\t\t<ParamsContext.Provider\n\t\t\t\t\tvalue={{\n\t\t\t\t\t\tisError,\n\t\t\t\t\t\tisSuccess,\n\t\t\t\t\t\tshowLoader,\n\t\t\t\t\t\thasError,\n\t\t\t\t\t\tresetScanState,\n\t\t\t\t\t\tshowGuideCard,\n\t\t\t\t\t\tscanStage,\n\t\t\t\t\t\tvideoLoading,\n\t\t\t\t\t\tsetVideoLoading,\n\t\t\t\t\t\tvideoError,\n\t\t\t\t\t\tsetVideoError,\n\t\t\t\t\t\tgender,\n\t\t\t\t\t\tgetButtonText,\n\t\t\t\t\t\tisScanning,\n\t\t\t\t\t\tstartScan,\n\t\t\t\t\t\tisModelLoaded,\n\t\t\t\t\t\tonRetry,\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t<FaceScanStep webcamRef={webcamRef} canvasRef={canvasRef} />\n\t\t\t\t</ParamsContext.Provider>\n\t\t\t</ConfigProvider>\n\t\t</LanguageContextProvider>\n\t);\n};\n"],"names":["preloadedDetector","isPreloading","useFaceScan","faceScanId","onValidPose","onScanComplete","onModelReady","onStartRecording","shopDomain","isError","isSuccess","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","ideal","max","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","nextStage","speechService","playAudio","voiceOverAssetsPath","completedStage","totalStages","nextSound","resetScan","speechSynthesis","cancel","enableVoiceCommands","silentUtterance","SpeechSynthesisUtterance","volume","speak","FaceScanErrorScreen","onRetry","loading","translate","useContext","LanguageContext","config","useConfig","_jsx","className","style","background","base","backgroundColor","children","_jsxs","Header","noTitle","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","showLoader","hasError","resetScanState","showGuideCard","getButtonText","isScanning","startScan","ParamsContext","resolvedConfig","LoadingScreen","url","loader","loaderType","_Fragment","id","crossOrigin","preload","position","zIndex","undefined","ref","transform","opacity","Drawer","open","anchor","onClose","event","reason","hideBackdrop","props","onScanSuccess","onScanError","onScanStart","onCaptureComplete","onUploadStart","onUploadEnd","onMeasurementSocketStart","onMeasurementSocketClose","userDetails","email","deviceFocalLength","bodyScanId","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","body_scan_id","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","canvas","videoEl","play","drawFrame","save","rotate","drawImage","restore","requestAnimationFrame","recordingFps","frameRate","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","ConfigProvider","Provider","value","reset"],"mappings":"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"}
|