@swan-admin/swan-dev-web-component 0.1.0-alpha.1771570342 → 0.1.0-alpha.1771587079

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.
@@ -1 +1 @@
1
- {"version":3,"file":"BodyScan-BYiULEJt.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\";\n\nfunction LevelScreen({ angle, countdown, isScanning, isInTargetRange, stabilityScore, children, config }: LevelScreenProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\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 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 resolvedConfig={config} />\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 handleShowStreamCamera,\n loadingCam,\n handleUserMedia,\n handlePause,\n pause,\n recordingStarted,\n startSendingVideoFrames,\n showPause,\n webcamRef,\n handleUserMediaError,\n resetDetector,\n videoConstraints,\n webcamKey,\n config,\n}: CameraScanChildProps) {\n const { media } = useContext(MediaContext) || {};\n const { translate } = useContext(LanguageContext) || {};\n\n return (\n <div className=\"App w-screen h-[100vh] relative \">\n <span\n onClick={() => {\n handleShowStreamCamera();\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={handleUserMedia}\n onUserMediaError={handleUserMediaError}\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={handlePause}\n resolvedConfig={config}\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={handleShowStreamCamera}\n resolvedConfig={config}\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 resolvedConfig={config}\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 \"posthog-js\";\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: number | undefined | NodeJS.Timeout;\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, config }: ScanningComponentProps) {\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<number | null | NodeJS.Timeout>(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 [supportedTypes, setSupportedTypes] = useState<string[]>([]);\n\tconst [webcamKey, setWebcamKey] = useState(0);\n\tconst [currentVideoConstraints, setCurrentVideoConstraints] = useState<MediaTrackConstraints>(videoConstraintsExact);\n\n\tlet counter = 0;\n\n\tconst { setStartGyro, handleFileUpload, setUploadLoading, swan, onScanStart } = useContext(ParamsContext);\n\n\tconst handleShowStreamCamera = () => {\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\tcounter = 0;\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 handleReScan = useCallback(() => {\n\t\trescanSupportCaptureEvent({\n\t\t\teventName: `${shopDomain}/rescan`,\n\t\t\temail,\n\t\t\tscanID: scanID,\n\t\t\theight: heightInCm,\n\t\t\tgender,\n\t\t\tstatus: false,\n\t\t});\n\t\tallowAudioToPlay.current = false;\n\t\thandleShowStreamCamera();\n\t\tstartSendingVideoFrames();\n\t}, [handleShowStreamCamera, scanID, email]);\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\tcounter = 0;\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\n\t\t\tposthog.capture(`${shopDomain}/pose_detection_connected`, {\n\t\t\t\tscanID: scanID,\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}, [scanID, shopDomain, email]);\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: scanID,\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, scanID, 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\tconst handleDataAvailable = useCallback(\n\t\t({ data }: BlobEvent) => {\n\t\t\tif (data.size > 0 && swan.poseDetection.connected()) {\n\t\t\t\tswan.poseDetection.poseStatus(async (data:any) => {\n\t\t\t\t\tif (data && data.audio && data.audio.length > 0) {\n\t\t\t\t\t\tconst audio = document.querySelector(\"#audioElement\") as HTMLAudioElement | null;\n\t\t\t\t\t\tif (data.status === true && data.sid === id) {\n\t\t\t\t\t\t\tif (counter < 2) {\n\t\t\t\t\t\t\t\tif (counter === 0 && audio?.paused) {\n\t\t\t\t\t\t\t\t\tawait speechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\tcounter += 1;\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tsetEmptyAudioSource(data.audio);\n\t\t\t\t\t\t\t\tclearTimeout(audioTimeoutId);\n\t\t\t\t\t\t\t\tswan.poseDetection.disconnect();\n\t\t\t\t\t\t\t\tsetTimeout(postPoseProcess, 1000);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tcounter = 0;\n\t\t\t\t\t\t\tif (audio?.paused && (!lastVideoPlayed || lastVideoPlayed?.audioName !== data.audio)) {\n\t\t\t\t\t\t\t\tlastVideoPlayed = {\n\t\t\t\t\t\t\t\t\tskipCount: 2,\n\t\t\t\t\t\t\t\t\tno_of_times_skipped: 0,\n\t\t\t\t\t\t\t\t\taudioName: data.audio,\n\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\tspeechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t} else if (lastVideoPlayed?.audioName === data.audio && audio?.paused) {\n\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\tlastVideoPlayed.no_of_times_skipped = 0;\n\t\t\t\t\t\t\t\t\tspeechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t\t} else {\n\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}\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: scanID,\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t\t},\n\t\t[webcamRef, scanID, swan, id, counter, postPoseProcess, speechService],\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: scanID,\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([...events, 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, events, playAudio, allowAudioToPlay, onScanStart]);\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: scanID,\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}, [startAgain, 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\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\thandleFileUpload?.(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\thandleShowStreamCamera();\n\t}, []);\n\n\treturn (\n\t\t<CameraScanChild\n\t\t\tresetDetector={resetDetector}\n\t\t\thandleShowStreamCamera={handleShowStreamCamera}\n\t\t\tloadingCam={loadingCam}\n\t\t\thandlePause={handlePause}\n\t\t\tshowRestart={showRestart}\n\t\t\tpause={pause}\n\t\t\thandleReScan={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\thandleUserMedia={handleUserMedia}\n\t\t\thandleUserMediaError={handleUserMediaError}\n\t\t\tvideoConstraints={currentVideoConstraints}\n\t\t\twebcamKey={webcamKey}\n\t\t\tconfig={config}\n\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 config,\n}: AngleDetectorProps) {\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 config={config}\n >\n {isScanning && (\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 config={config}\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\">\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\";\n// import \"video.js/dist/video-js.css\";\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\";\n\nfunction ScanErrorMessage({ scanFailsError, serverAtCapacity = false, onNext, gender, setScanUniqueKey, resolvedConfig, setIsVideoUploaded }: ScanErrorMessageProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\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 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 resolvedConfig={resolvedConfig} />\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\t{serverAtCapacity && (\n\t\t\t\t\t\t<div className=\"p-[1rem] text-center\">\n\t\t\t\t\t\t\t<h3\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.serverAtCapacity)}\n\t\t\t\t\t\t\t</h3>\n\t\t\t\t\t\t\t<p\n\t\t\t\t\t\t\t\tclassName=\"text-base mt-[0.5rem]\"\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.serverAtCapacityDescription)}\n\t\t\t\t\t\t\t</p>\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} resolvedConfig={resolvedConfig} />\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 \"posthog-js\";\nimport Header from \"./Header\";\nimport CameraPermission from \"./bodyScan/CameraPermission\";\nimport { UserDetails } 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\tisFaceScan,\n\tisVideoUploadedCorrect,\n\tisMeasurementAvailable,\n\tonComplete,\n\tisSuccess,\n\tonCustomScanSuccess,\n\tswan,\n}: {\n\tconfig?: any;\n\tisFaceScan: boolean;\n\tscanId: string;\n\tisVideoUploadedCorrect?: boolean;\n\tisMeasurementAvailable?: boolean;\n\tuserDetails: UserDetails;\n\tonComplete?: () => void;\n\tisSuccess?: boolean;\n\tonCustomScanSuccess?: () => void;\n\tswan: any;\n}) {\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 || (isFaceScan ? isMeasurementAvailable && isVideoUploadedCorrect : isVideoUploadedCorrect || isMeasurementAvailable);\n\n\treturn (\n\t\t<Box className=\"flex h-full w-full flex-col \">\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)} resolvedConfig={config} />\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 resolvedConfig={config} 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, useEffect } 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 { LanguageContext } from \"../../utils/context/languageContext\";\nimport { useLocalConfig } from \"../../config/useLocalConfig\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\n\nexport const BodyScanSteps = () => {\n\tconst {\n\t\tuserDetails,\n\t\tconfig,\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\thandleShowStreamCamera,\n\t\tonCustomScanSuccess,\n\t\tswan,\n\t\tonComplete\n\t} = useContext(ParamsContext);\n\tconst { setPreferredLanguage } = useContext(LanguageContext) || {};\n\tconst resolvedConfig = useLocalConfig(config);\n\n\tuseEffect(() => {\n\t\tsetPreferredLanguage?.(resolvedConfig?.language);\n\t}, [resolvedConfig]);\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\n\t\t\t\t\t\tscanFailsError={scanFailsError || isError}\n\t\t\t\t\t\tonNext={() => {\n\t\t\t\t\t\t\tonRetry?.();\n\t\t\t\t\t\t\thandleShowStreamCamera();\n\t\t\t\t\t\t}}\n\t\t\t\t\t\tsetScanUniqueKey={setScanUniqueKey}\n\t\t\t\t\t\tgender={gender}\n\t\t\t\t\t\tresolvedConfig={resolvedConfig}\n\t\t\t\t\t\tsetIsVideoUploaded={setIsVideoUploaded}\n\t\t\t\t\t/>\n\t\t\t\t</Drawer>\n\t\t\t</>\n\t\t);\n\t}\n\n\tif (isSuccess) {\n\t\treturn (\n\t\t\t<SignUp\n\t\t\t\tisFaceScan={false}\n\t\t\t\tscanId={scanUniqueKey}\n\t\t\t\t// isMeasurementAvailable={isMeasurementAvailable}\n\t\t\t\tuserDetails={userDetails}\n\t\t\t\t// onComplete={onComplete}\n\t\t\t\t// isVideoUploadedCorrect={isVideoUploadedCorrect}\n\t\t\t\t// config={resolvedConfig}\n\t\t\t\t// isSuccess={isSuccess}\n\t\t\t\t// onCustomScanSuccess={onCustomScanSuccess}\n\t\t\t\tswan={swan}\n\t\t\t\t// onComplete={onComplete}\n\t\t\t/>\n\t\t);\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\"\n style={{ background: resolvedConfig?.style?.base?.backgroundColor }}\n >\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\tconfig={resolvedConfig}\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\tisFaceScan={false}\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\tconfig={resolvedConfig}\n\t\t\t\tonCustomScanSuccess={onCustomScanSuccess}\n\t\t\t\tswan={swan}\n\t\t\t\tonComplete={onComplete}\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\thandleShowStreamCamera();\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\tresolvedConfig={resolvedConfig}\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 posthog from \"posthog-js\";\nimport useGyroSensor from \"../../customHooks/useGyroSensor\";\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 { getSwanService } from \"../../utils/service/swanService\";\nimport { posthogPublicHost, posthogPublicKey } from \"../../utils/constants\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\nimport MediaContextProvider from \"../../utils/context/mediaContext\";\nimport { BodyScanProps } from \"../../bodyScan\";\nimport { BodyScanSteps } from \"./BodyScanSteps\";\n\nconst clothesFit = \"0\";\n\nexport const BodyScan: React.FC<BodyScanProps> = ({\n\tuserDetails,\n\tconfig,\n\ttoken,\n\tonRetry,\n\tonScanStart,\n\tonUploadingStart,\n\tonUploadingEnd,\n\tonMeasurementSocketStart,\n\tonMeasurementSocketClose,\n\tonScanSuccessIntermediate,\n\tonScanError,\n\tonScanSuccess,\n\tonCustomScanSuccess,\n\tonComplete,\n\tisError,\n\tisSuccess,\n}) => {\n\tconst { gender, scanType, shopDomain, heightInCm, email, deviceFocalLength, deviceModelName, callbackUrl } = 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\n\tconst handleShowStreamCamera = 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\tonScanSuccessIntermediate?.(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 handleFileUpload = 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});\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\tonUploadingStart?.();\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) {\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\tconsole.log(error, \"video upload failed\");\n\t\t\t} finally {\n\t\t\t\tsetStartGyro(false);\n\t\t\t\tonUploadingEnd?.();\n\t\t\t}\n\t\t},\n\t\t[scanUniqueKey, gyroData, onUploadingStart, onUploadingEnd],\n\t);\n\tconst 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\tconst scanId = scanUniqueKey;\n\t\t\tif (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\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}, [checkMeasurementStatus, email, scanUniqueKey, setScanUniqueKey, shopDomain]);\n\n\tuseEffect(() => {\n\t\tposthog.init(posthogPublicKey, { api_host: posthogPublicHost });\n\t\tposthog.capture(\"$pageview\");\n\t}, []);\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<MediaContextProvider>\n\t\t\t\t<ParamsContext.Provider\n\t\t\t\t\tvalue={{\n\t\t\t\t\t\tuserDetails,\n\t\t\t\t\t\tconfig,\n\t\t\t\t\t\tonRetry,\n\t\t\t\t\t\tonScanError,\n\t\t\t\t\t\tisError,\n\t\t\t\t\t\tisSuccess,\n\t\t\t\t\t\tonScanSuccess,\n\t\t\t\t\t\tgender,\n\t\t\t\t\t\tscanUniqueKey,\n\t\t\t\t\t\tscanFailsError,\n\t\t\t\t\t\tsetScanUniqueKey,\n\t\t\t\t\t\tsetIsVideoUploaded,\n\t\t\t\t\t\tisMeasurementAvailable,\n\t\t\t\t\t\tisVideoUploadedCorrect,\n\t\t\t\t\t\tloading,\n\t\t\t\t\t\tshowDeniedModal,\n\t\t\t\t\t\tuploadLoading,\n\t\t\t\t\t\tsetStartGyro,\n\t\t\t\t\t\thandleFileUpload,\n\t\t\t\t\t\tsetUploadLoading,\n\t\t\t\t\t\thandleShowStreamCamera,\n\t\t\t\t\t\tsetScanFailsError,\n\t\t\t\t\t\tsetScanStartTime,\n\t\t\t\t\t\tonScanStart,\n\t\t\t\t\t\tonCustomScanSuccess,\n\t\t\t\t\t\tonComplete,\n\t\t\t\t\t\tswan,\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t<BodyScanSteps />\n\t\t\t\t</ParamsContext.Provider>\n\t\t\t</MediaContextProvider>\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","React","memo","_jsxs","width","height","viewBox","fill","xmlns","d","stroke","strokeWidth","strokeLinecap","strokeLinejoin","LevelScreen","angle","countdown","isScanning","isInTargetRange","stabilityScore","config","translate","useContext","LanguageContext","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","resolvedConfig","border","fontFamily","base","baseFontFamily","color","cn","transform","transition","LanguageKeys","startLevelCheck","leavePhone","placePhoneUpright","CameraScanChild","handleShowStreamCamera","loadingCam","handleUserMedia","handlePause","pause","recordingStarted","startSendingVideoFrames","showPause","webcamRef","handleUserMediaError","resetDetector","videoConstraints","webcamKey","onClick","X","Webcam","audio","ref","screenshotQuality","mirrored","screenshotFormat","onUserMedia","onUserMediaError","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","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","import","setBackend","ready","modelConfig","runtime","modelType","solutionPath","createDetector","SupportedModels","BlazePose","mediapipeError","warn","error","isFrontFrame","body","length","filter","p","current","then","success","s","callback","video","readyState","poses","estimatePoses","flipHorizontal","keypoints","slice","map","kp","x","y","score","prev","err","retryLoading","useTensorFlow","firstScan","poseStoppedRef","supportedTypes","setSupportedTypes","setWebcamKey","currentVideoConstraints","setCurrentVideoConstraints","videoConstraintsExact","counter","setStartGyro","handleFileUpload","setUploadLoading","swan","onScanStart","ParamsContext","clearTimeout","generateUuid","speechService","stopAudio","stop","forEach","el","stream","setTimeout","videoConstraintsFallback","handleReScan","rescanSupportCaptureEvent","eventName","status","poseDetection","disconnect","preferredVideoTypes","getPreferredMediaRecorderTypes","recordingMimeType","getRecordingMimeType","recordingExtension","getRecordingExtension","stopRecording","playAudio","handleSocket","connect","posthog","capture","handleSpinDataAvailable","data","concat","handleScanTimeCapture","message","handleStartCaptureClick","videoTrack","getVideoTracks","settings","getSettings","frameRate","mediaRecorderOptions","videoBitsPerSecond","mimeType","MediaRecorder","start","e","postPoseProcess","handleDataAvailable","connected","poseStatus","querySelector","sid","paused","audioName","no_of_times_skipped","skipCount","getScreenshot","videoEmit","image","scanId","getCurrentTimeInSeconds","push","bypassChecksToBackup","videoFile","File","type","supported","getSupportedMediaRecorderTypes","videoTypes","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","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","serverAtCapacity","onNext","showModal","setShowModal","handleShowModal","background","issueWithScan","reason","handleErrorMessage","VIDEO_POSTER_GENDER_BASED","alt","serverAtCapacityDescription","scanAgain","onClose","GENDER","PRE_LINK","male","SignUp","isFaceScan","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","setPreferredLanguage","useLocalConfig","language","BodyScan","token","onUploadingStart","onUploadingEnd","onMeasurementSocketStart","onMeasurementSocketClose","onScanSuccessIntermediate","onScanError","onScanSuccess","deviceModelName","callbackUrl","getSwanService","setIsVideoUploadedCorrect","setIsMeasurementAvailable","isVideoUploaded","startGyro","scanStartTime","gyroData","setGyroData","permissionGranted","setPermissionGranted","alpha","toString","timestamp","Date","toISOString","useGyroSensor","onError","clearScanStartTimeAndScanId","completionTime","onSuccess","scanStatus","resultType","code","scanIdToSet","handleMeasurementRecommendation","measurement","handleMeasurementSocket","onPreopen","handleWebSocketCapture","connection","onOpen","file","recordedFps","fileSizeInMB","Number","toFixed","videoFps","getVideoFPS","arrayMetaData","createObjectMetadataArray","focal_length","customer_store_url","clothes_fit","scan_type","callback_url","JSON","stringify","video_fps","fileUpload","uploadFileFrontend","setDeviceInfo","model","detection","gyro","checkMeasurementStatus","res","getMeasurementResult","isMeasured","tempScanStartTime","currentTimeInSeconds","fiveMinutesInSeconds","init","posthogPublicKey","api_host","posthogPublicHost","LanguageContextProvider"],"mappings":"6uBAYO,MAAMA,GACF,UADEA,GAEN,MAFMA,GAGH,SAWGC,GAAeC,OAC1BC,GAGF,SAASC,IAAqBC,SAAEA,IAC9B,MAAOC,EAAMC,GAAWC,EAAS,CAACC,QAAQC,WAAYD,QAAQE,eACvDC,EAAaC,GAAkBL,GAAS,GAEzCM,EAAa,KACjBP,EAAQ,CAACE,QAAQC,WAAYD,QAAQE,eAEvCI,EAAU,KACRD,KACC,IACHE,EAAgB,KACdP,OAAOQ,iBAAiB,SAAUH,GAC3B,IAAML,OAAOS,oBAAoB,SAAUJ,IACjD,CAACA,IACJ,IAAIK,EAAQnB,GACRM,EAAK,GAAK,KAAOA,EAAK,GAAK,OAC7Ba,EAAQnB,IAENM,EAAK,GAAK,MACZa,EAAQnB,IAGV,MAAMoB,EAAeC,EACnB,KAAA,CACEf,OACAC,UACAK,cACAC,iBACAM,UAEF,CAACb,EAAMM,EAAaO,IAGtB,OACEG,EAACrB,GAAasB,SAAQ,CAACC,MAAOJ,EAAYf,SACvCA,GAGP,CClBA,IAAAoB,GAAeC,EAAMC,KAhDrB,UAAoBrB,KAAEA,EAAO,KAC3B,OACEsB,EAAA,MAAA,CACEC,MAAOvB,EACPwB,OAAQxB,EACRyB,QAAQ,YACRC,KAAK,OACLC,MAAM,6BAA4B5B,SAAA,CAElCiB,EAAA,OAAA,CACEY,EAAE,mHACFC,OAAO,QACPC,YAAY,IACZC,cAAc,QACdC,eAAe,UAEjBhB,EAAA,OAAA,CACEY,EAAE,0NACFC,OAAO,eACPC,YAAY,IACZC,cAAc,QACdC,eAAe,UAEjBhB,EAAA,OAAA,CACEY,EAAE,kOACFC,OAAO,eACPC,YAAY,IACZC,cAAc,QACdC,eAAe,UAEjBhB,EAAA,OAAA,CACEY,EAAE,kOACFC,OAAO,eACPC,YAAY,IACZC,cAAc,QACdC,eAAe,UAEjBhB,EAAA,OAAA,CACEY,EAAE,6kBACFC,OAAO,eACPC,YAAY,IACZC,cAAc,QACdC,eAAe,YAIvB,GCsJA,IAAAC,GAAeb,EAAMC,KA5LrB,UAAqBa,MAAEA,EAAKC,UAAEA,EAASC,WAAEA,EAAUC,gBAAEA,EAAeC,eAAEA,EAAcvC,SAAEA,EAAQwC,OAAEA,IAC/F,MAAMC,UAAEA,GAAcC,EAAWC,IAAoB,CAAA,EAE/CC,EAAuBC,EAAY,IACpCV,EAAQ,GAAWW,KAAKC,IAAI,EAAGD,KAAKE,IAAI,IAAoB,IAAdb,EAAQ,MACtDA,EAAQ,GAAWW,KAAKC,IAAI,EAAGD,KAAKE,IAAI,IAAqB,IAAf,IAAMb,KACjD,IACL,CAACA,IAEEc,EAAkBjC,EAAQ,KAC/B,GAAIqB,EAAY,OAAOG,GAAQU,OAAOC,eAAeC,wBAA0B,UAE/E,MAAMC,EAAYT,IAClB,GAAkB,IAAdS,EAAiB,OAAOb,GAAQU,OAAOC,eAAeG,2BAA6B,UACvF,GAAkB,MAAdD,EAAmB,OAAOb,GAAQU,OAAOC,eAAeC,wBAA0B,UAKpF,MAAO,OAHGN,KAAKS,MAAM,IAAqBF,EAAY,IAA3B,SACjBP,KAAKS,MAAM,IAAoBF,EAAY,IAA1B,SACjBP,KAAKS,MAAM,IAAqBF,EAAY,IAA3B,OAE1B,CAAChB,EAAYO,IAEXY,EAAmBX,EAAY,CAACY,EAAkBJ,IACnDA,EAAY,GACRI,EAAU,SAASjB,GAAQU,OAAOC,eAAeO,iCAAmC,SAASlB,GAAQU,OAAOC,eAAeO,8BAE5HD,EAAU,SAASjB,GAAQU,OAAOC,eAAeO,iCAAmC,SAASlB,GAAQU,OAAOC,eAAeQ,gCAChI,IAEGC,EAAef,EACpB,CAACY,GAAU,KACV,MAAMJ,EAAYT,IAClB,OAAIP,EAAmB,SAASG,GAAQU,OAAOC,eAAeO,8BACvDF,EAAiBC,EAASJ,IAElC,CAAChB,EAAYO,IAGRiB,EAAmBhB,EAAY,CAACY,EAAkBJ,IACnDA,EAAY,IAGTI,EAFWjB,GAAQU,OAAOC,eAAeO,2BAE4BlB,GAAQU,OAAOC,eAAeQ,0BACxG,IACGG,EAAejB,EACpB,CAACY,GAAU,KACV,MAAMJ,EAAYT,IAClB,OAAIP,EAAmBG,GAAQU,OAAOC,eAAeO,2BAC9CG,EAAiBJ,EAASJ,IAElC,CAAChB,EAAYO,IAERmB,EAAK3B,EACX,SAAS4B,IACRC,SAASC,gBAAgBhB,MAAMiB,YAAY,YAAa/D,OAAOE,YAAc,KAC9E,CAGA,OAFA0D,IACA5D,OAAOQ,iBAAiB,SAAUoD,GAEjCzC,SACC6C,UAAU,sJACVlB,MAAO,CAAED,mBAAiBjD,SAAA,CAG1BiB,EAAA,MAAA,CAAKmD,UAAU,uFACdnD,EAAA,MAAA,CAAKmD,UAAU,wBAAuBpE,SACrCiB,EAACoD,EAAM,CAACC,WAAQC,eAAgB/B,QAG1B,OAAPuB,EACA9C,EAAA,MAAA,CACCmD,UAAU,yHACVlB,MAAO,CACNsB,OAAQ,aAAaV,OACrB9D,SAEDiB,EAAA,MAAA,CACCmD,UAAW,+BAA+BN,OAC1CZ,MAAO,CACNuB,WAAYjC,GAAQU,OAAOwB,MAAMC,gBAAkB,oBAC9BC,MAAOd,KAC5B9D,SAEA+D,MAIHxC,QADGc,GACE+B,UAAU,qDAAoDpE,SAAA,CAClEiB,EAAA,MAAA,CAAKmD,UAAU,2GAA0GpE,SACxHiB,EAAA,MAAA,CAAKmD,UAAU,2DAEfpE,IAGF,CACCoE,UAAWS,GAAG,iFAAkFvC,EAAkB,6BAA+B,8BACjJY,MAAO,CACN4B,UAAW,cAA6B,GAAd,GAAK3C,QAC/B4C,WAAY,0BACZP,OAAQ,aAAaV,OACrB9D,SAAA,CAEDiB,EAAA,MAAA,CAAKmD,UAAW,0CAA0CN,UAC3CZ,MAAO,CACrBD,gBAAiB,GAAGa,WAGrBvC,EAAA,MAAA,CAEC6C,UAAWS,GAAG,mFAAoFjB,KAC/EV,MAAO,CAC1B0B,MAAOd,KACP9D,SAAA,CAEAiB,EAACG,GAAU,CAACnB,KAAM,KAClBsB,EAAA,IAAA,CACC2B,MAAO,CACNuB,WAAYjC,GAAQU,OAAOwB,MAAMC,gBAAkB,qBACnD3E,SAAA,CAEA,IACAyC,IAAYuC,EAAaC,iBAC1BhE,oBAMI,OAAP8C,GACA9C,SAAKmD,UAAU,qCAAoCpE,SAClDiB,SACCmD,UAAWS,GAAG,oEAAqEjB,KACnFV,MAAO,CACNuB,WAAYjC,GAAQU,OAAOwB,MAAMC,gBAAkB,oBAC9BC,MAAOd,KAC5B9D,SAEAyC,IAAYuC,EAAaE,gBAI5B5C,GAA0B,OAAPyB,IAAgB1B,GACnCpB,EAAA,MAAA,CAAKmD,UAAU,mCAAkCpE,SAChDiB,EAAA,MAAA,CACCmD,UAAW,0BAA0BN,2CACrCZ,MAAO,CACND,gBAAiB,GAAGa,SACpB9D,SAEDiB,EAAA,MAAA,CACCmD,UAAW,cAAcN,uDACzBZ,MAAO,CACN1B,MAAO,GAAGW,KACVc,gBAAiB,GAAGa,eAOd,OAAPC,IAAgB1B,GACfpB,EAAA,MAAA,CAAKmD,UAAU,qCAAoCpE,SACjDuB,EAAA,MAAA,CAAK6C,UAAWS,GAAG,2BAA4BjB,KAC9CV,MAAO,CACZ0B,MAAOd,KACP9D,SAAA,CAEO8C,KAAKS,MAAMpB,GAAM,SAKnB,OAAP4B,IAAgB1B,GAChBpB,EAAA,MAAA,CAAKmD,UAAU,2DAA0DpE,SACxEiB,EAAA,MAAA,CACCmD,UAAWS,GAAG,2BAA4BjB,KAC1CV,MAAO,CACNuB,WAAYjC,GAAQU,OAAOwB,MAAMC,gBAAkB,oBAC9BC,MAAOd,KAC5B9D,SAEAyC,IAAYuC,EAAaG,yBAMhC,GCzLA,SAASC,IAAgBC,uBACvBA,EAAsBC,WACtBA,EAAUC,gBACVA,EAAeC,YACfA,EAAWC,MACXA,EAAKC,iBACLA,EAAgBC,wBAChBA,EAAuBC,UACvBA,EAASC,UACTA,EAASC,qBACTA,EAAoBC,cACpBA,EAAaC,iBACbA,EAAgBC,UAChBA,EAASzD,OACTA,IAEA,MAAM1B,MAAEA,GAAU4B,EAAW9C,KAAiB,CAAA,GACxC6C,UAAEA,GAAcC,EAAWC,IAAoB,CAAA,EAErD,OACEpB,EAAA,MAAA,CAAK6C,UAAU,6CACbnD,EAAA,OAAA,CACEiF,QAAS,KACPb,IACAU,KAEF3B,UAAU,wCAAuCpE,SAEjDiB,EAACkF,GAAC,CAAC/B,UAAU,kBAGfnD,EAAA,MAAA,CAAKmD,UAAU,iCAAgCpE,SAC5Cc,IAAUnB,IACTsB,EAACmF,GAAM,CAELC,OAAO,EACPC,IAAKT,EACLU,kBAAmB,EACnBP,iBAAkBA,EAClBQ,UAAQ,EACRC,iBAAiB,aACjBC,YAAanB,EACboB,iBAAkBb,EAClB5C,MAAO,CACL0D,SAAU,QACVC,IAAK,EACLC,OAAQ,EACRC,OAAQ,EACRvF,MAAO,OACPC,OAAQ,OACRuF,UAAW,UAhBRf,KAsBX1E,EAAA,MAAA,CAAK6C,UAAU,+FACZwB,GACC3E,EAACgG,EAAc,CACb7C,UAAU,2CACV8C,OAAQC,EACRC,WAAY3E,IAAYuC,EAAaS,OACrC4B,WAAY7B,EACZjB,eAAgB/B,EAChB8E,cAAY,IAIf7B,EACCxE,EAAA,MAAA,CAAAjB,SACEiB,EAACgG,EAAc,CACb7C,UAAU,wCACVgD,WAAY3E,IAAYuC,EAAauC,SACrCF,WAAYhC,EACZd,eAAgB/B,EAChB8E,cAAY,MAGb5B,EAUD,KATFzE,EAACgG,EAAc,CACb7C,UAAW,wEACTkB,EAAa,cAAgB,IAE/B8B,WAAY3E,IAAYuC,EAAawC,WACrCH,WAAY1B,EACZ8B,SAAUnC,EACVf,eAAgB/B,OAKtBvB,EAAA,QAAA,CACEyG,GAAG,eACHC,YAAY,YACZC,QAAQ,OACR1E,MAAO,CACL0D,SAAU,WACVG,QAAQ,OAEVc,IAAK,GAAGC,yCAIhB,CC5GA,IAAIC,GAAgD,KAChDC,GAAsB,KACtBC,GAAqB,KCYzB,IACIC,GADAR,GAAoB,KAEpBS,GAIO,KAubX,IAAAC,GAAe/G,EAAMC,KArbrB,UAA2B+G,gBAAEA,EAAetC,cAAEA,EAAauC,OAAEA,EAAMC,mBAAEA,EAAkBC,kBAAEA,EAAiBC,iBAAEA,EAAgBC,iBAAEA,EAAgBC,YAAEA,EAAWnG,OAAEA,IAC5J,MAAMoG,OAAEA,EAAMC,WAAEA,EAAUC,MAAEA,EAAKC,WAAEA,GAAeJ,EAC5C9C,EAAYmD,EAAsB,MAClCC,EAAmBD,EAA6B,MAChDE,EAAiBF,EAAsB,OACtCG,EAAgBC,GAAqBjJ,EAAiB,KACtDmF,EAAY+D,GAAiBlJ,GAAS,IACtCuF,EAAkB4D,GAAuBnJ,GAAS,IAClDoJ,EAAUC,GAAerJ,GAAS,IAClCsJ,EAAsBC,GAA2BvJ,GAAS,IAC1DkC,EAAYsH,GAAiBxJ,GAAS,GACvCyJ,EAA4BZ,EAAuC,OAClEa,EAAaC,GAAkB3J,EAAS,KACxC4J,EAAiBC,IAAsB7J,EAAmB,KAC1D8J,GAAkBC,IAAuB/J,EAAS,KAClDgK,GAAYC,IAAiBjK,GAAS,IACtCkK,GAAaC,IAAcnK,GAAS,IACpCsF,GAAO8E,IAAYpK,GAAS,IAC5BqK,GAAQC,IAAatK,EAAc,KACnCyF,GAAW8E,IAAgBvK,GAAS,GACrCwK,GAAmB3B,GAAO,IAC1B4B,aAAEA,IDvCK,WACZ,MAAOC,EAAmBC,GAAwB3K,EAAS,IACpD4K,EAAWC,GAAgB7K,EAAS,IACpC8K,EAAUC,GAAe/K,GAAS,GAEnCgL,EAAenC,EAAO,GACtBoC,EAAuBpC,EAAO,GAC9BqC,EAAUrC,GAAO,GAgCvBsC,eAAeC,IAGb,GACoB,oBAAXnL,QACc,oBAAdoL,UAEP,OAAO,EAGT,IACEC,QAAQC,IAAI,kCAIZ,MAAOC,EAASC,EAAUC,SAAqBC,QAAQC,IAAI,CACzDC,OAAO,oCACPA,OAAO,yBACPA,OAAO,oCAGT/D,GAAgB0D,QAGVC,EAASK,WAAW,eACpBL,EAASM,QACfT,QAAQC,IAAI,oCAKZ,MAAMS,EAAc,CAClBC,QAAS,YACTC,UAAW,OACXC,aAAa,gDAGf,IACEtE,SAAuBC,GAAcsE,eACnCtE,GAAcuE,gBAAgBC,UAC9BN,GAEFV,QAAQC,IAAI,0CACb,CAAC,MAAOgB,GACPjB,QAAQkB,KAAK,kDAAmDD,GAGhE1E,SAAuBC,GAAcsE,eACnCtE,GAAcuE,gBAAgBC,UAC9B,CACEL,QAAS,OACTC,UAAW,SAGfZ,QAAQC,IAAI,qCACb,CAED,OAAO,CACR,CAAC,MAAOkB,GAGP,OAFAnB,QAAQmB,MAAM,0CAA2CA,GACzD7E,GAAuB,MAChB,CACR,CACH,CAEA,SAAS8E,EAAaC,GAEpB,SAAKA,GAAwB,IAAhBA,EAAKC,SAC+B,KAA1CD,EAAKE,OAAQC,GAAMA,EAAE,GAAK,IAAKF,MACxC,CA2EA,OA9KArM,EAAU,KAIR,GAHA2K,EAAQ6B,SAAU,GAGdlF,GAiBJ,OAXKD,KACHA,GAAuBwD,KAIzBxD,GAAqBoF,KAAMC,IACrBA,GAAW/B,EAAQ6B,SACrBhC,GAAY,KAIT,KACLG,EAAQ6B,SAAU,GAjBlBhC,GAAY,IAmBb,IAyHHxK,EAAU,KACR0K,EAAqB8B,QAAUrC,EAC3BA,EAAoB,GAAKM,EAAa+B,QAAU,IAClDlC,EAAcqC,GAAMA,EAAI,GACxBvC,EAAqB,KAEtB,CAACD,IAEJnK,EAAU,KACRyK,EAAa+B,QAAUnC,GACtB,CAACA,IAmBG,CACLH,aA1EmBU,MAAOgC,EAAsBzH,KAGhD,IAAKmC,KAAmBqD,EAAQ6B,UAAYrH,GAAWqH,SAASK,MAC9D,OAIF,MAAMA,EAAQ1H,EAAUqH,QAAQK,MAChC,KAAIA,EAAMC,WAAa,GAEvB,IACE,MAAMC,QAAczF,GAAe0F,cACjCH,EACA,CAAEI,gBAAgB,IAGpB,IAAKF,IAAUA,EAAMV,OAAQ,OAE7B,MAGMD,EAHYW,EAAM,GAAGG,UAIxBC,MAAM,IACNC,IAAKC,GAAY,CAChBA,EAAGC,EAAI,GAAKD,EAAGC,EAAI,IAAMD,EAAGC,GAAK,EACjCD,EAAGE,EAAI,GAAKF,EAAGE,EAAI,KAAOF,EAAGE,GAAK,EAClCF,EAAGG,OAAS,IAGa,IAAzB/C,EAAa+B,SAAkBL,EAAaC,IAC9ChC,EAAsBqD,GAASA,EAAO,GAGX,IAAzBhD,EAAa+B,SAAiBL,EAAaC,IAC7ChC,EAAsBqD,GAASA,EAAO,GAGX,IAAzBhD,EAAa+B,SAAeI,GACjC,CAAC,MAAOc,GACP3C,QAAQmB,MAAM,wBAAyBwB,EACxC,GAkCDnD,WACAF,YACAhF,cArBoB,KACpBoF,EAAa+B,QAAU,EACvB9B,EAAqB8B,QAAU,EAC/BpC,EAAqB,GACrBE,EAAa,IAkBbqD,aAfmB/C,UACdtD,IAAmBD,KACtBA,GAAuBwD,UACDxD,IACPsD,EAAQ6B,SACrBhC,GAAY,KAYpB,CCvJ0BoD,GACnBC,GAAYvF,GAAO,GACnBwF,GAAiBxF,GAAO,IACvByF,GAAgBC,IAAqBvO,EAAmB,KACxD8F,GAAW0I,IAAgBxO,EAAS,IACpCyO,GAAyBC,IAA8B1O,EAAgC2O,GAE9F,IAAIC,GAAU,EAEd,MAAMC,aAAEA,GAAYC,iBAAEA,GAAgBC,iBAAEA,GAAgBC,KAAEA,GAAIC,YAAEA,IAAgB1M,EAAW2M,GAErFhK,GAAyB,KAC9BmJ,GAAetB,SAAU,EACzBoC,aAAapH,IACT0B,EAA0BsD,SAASoC,aAAa1F,EAA0BsD,SAC9ExE,EAAiB6G,KACjB/G,EAAkB,IAClB0G,MAAmB,GACnB9F,EAAkB,IAClBF,EAAegE,QAAU,KACzB7D,GAAc,GACdC,GAAoB,GACpBE,GAAY,GACZE,GAAwB,GACxBC,GAAc,GACdC,EAA0BsD,QAAU,KACpCpD,EAAe,IACfE,GAAmB,IACnBE,GAAoB,IACpBE,IAAeD,IACf6E,IAAa,GACbQ,EAAcC,YACdnF,IAAW,GACXC,IAAS,GACTwE,GAAU,EACuB,OAA7B9F,EAAiBiE,SACpBjE,EAAiBiE,QAAQwC,OAG1BlF,GAAOmF,QAASC,IACX3G,EAAiBiE,SAASjE,EAAiBiE,QAAQrM,oBAAoB,gBAAiB+O,KAG7FnF,GAAU,IACVE,GAAiBuC,SAAU,EAC3BxC,IAAa,GACbnC,GAAmB,GACf1C,EAAUqH,SAAS2C,QACtBxG,GAAc,IAIV9D,GAAkB1C,EAAY,KACnCiN,WAAW,KACVzG,GAAc,IACZ,MACD,IAEGvD,GAAuBjD,EAC3B+J,IAGA,GAFAnB,QAAQC,IAAI,eAAgBkB,GAExBgC,KAA4BE,EAG/B,OAFAD,GAA2BkB,QAC3BpB,GAAcR,GAASA,EAAO,GAG/B9E,GAAc,IAEf,CAACuF,KAGIoB,GAAenN,EAAY,KAChCoN,EAA0B,CACzBC,UAAW,GAAGnH,WACdD,QACAR,OAAQA,EACR7G,OAAQoH,EACRD,SACAuH,QAAQ,IAETxF,GAAiBuC,SAAU,EAC3B7H,KACAM,MACE,CAACN,GAAwBiD,EAAQQ,IAE9BtD,GAAc3C,EAAY,KAC/B8H,GAAiBuC,SAAU,EAC3B3C,IAAS,GACTG,IAAa,GACTd,EAA0BsD,SAASoC,aAAa1F,EAA0BsD,SAC1EjE,EAAiBiE,SACpBjE,EAAiBiE,QAAQzH,QAE1B+J,EAAcC,YACdN,GAAKiB,cAAcC,aACnBtB,GAAU,EACN9F,EAAiBiE,SACpBjE,EAAiBiE,QAAQwC,OAE1BlF,GAAOmF,QAASC,IACX3G,EAAiBiE,SAASjE,EAAiBiE,QAAQrM,oBAAoB,gBAAiB+O,KAE7FnF,GAAU,IACV6E,aAAapH,KACX,CAACe,EAAkBuB,GAAQgF,EAAe7E,KAGvC2F,GAAsBtP,EAAQ,IAAMuP,IAAkC,IACtEC,GAAoBC,EAAqBhC,IACzCiC,GAAqBC,EAAsBH,IAE3CI,GAAgB/N,EAAYyI,UACjCX,GAAiBuC,SAAU,QACrBsC,EAAcqB,UAAU,GAAG/I,eACjCyC,IAAS,GACTG,IAAa,GACTd,EAA0BsD,SAASoC,aAAa1F,EAA0BsD,SAC9E5D,GAAoB,IAClB,CAACM,EAA2B4F,IAEzBsB,GAAejO,EAAYyI,UAChC,IACC5D,SAAWyH,GAAKiB,cAAcW,UAE9BC,EAAQC,QAAQ,GAAGlI,6BAAuC,CACzDT,OAAQA,EACRQ,QACApB,OAED,CAAC,MAAOkF,GACRnB,QAAQC,IAAIkB,EAAO,6BACnB,GACC,CAACtE,EAAQS,EAAYD,IAElBoI,GAA0BrO,EAC/B,EAAGsO,WACEA,GAAQA,EAAKlR,KAAO,GAAK0K,GAAiBuC,UAC7C9D,EAAmB+E,GAASA,EAAKiD,OAAOD,KACnC3C,GAAetB,SAAWrH,EAAUqH,UACxCsB,GAAetB,SAAU,EACzBtC,GAAa,KACZgG,KACAS,EAAsB,CACrBnB,UAAW,GAAGnH,eACdT,OAAQA,EACRQ,QACAwI,QAAS,sCAERzL,MAIN,CAAC+K,GAAeS,EAAuBtI,EAAYT,EAAQQ,EAAOjD,IAG7DgL,GAAYhO,EAAYyI,UACzBvB,EAAgBgD,OAAS,GAAKpC,GAAiBuC,gBAC5CsC,EAAcqB,UAAU/I,EAAsBiC,EAAgBA,EAAgBgD,OAAS,IACzFpC,GAAiBuC,UACpBhF,GAAiB4H,WAAWe,GAAW,QAGvC,CAAC9G,EAAiBY,KAEd4G,GAA0B1O,EAAY,KACxCoG,GAAoBA,EAAiBiE,SACxCjE,EAAiBiE,QAAQwC,OAE1BpG,GAAoB,GACpB,IACC,GAAIzD,GAAaA,EAAUqH,SAAWrH,EAAUqH,QAAQ2C,OAAQ,CAE/D,MAAM2B,EAAa3L,EAAUqH,QAAQ2C,OAAO4B,iBAAiB,GAC7D,GAAID,EAAY,CACf,MAAME,EAAWF,EAAWG,cAC5BzI,EAAegE,QAAUwE,EAASE,WAAa,IAC/C,CAED,MAAMC,EAA6C,CAClDC,mBAAoB,MAEjBrD,GAAe,KAClBoD,EAAqBE,SAAWtD,GAAe,IAEhDxF,EAAiBiE,QAAU,IAAI8E,cAAcnM,EAAUqH,QAAQ2C,OAAQgC,GACvE5I,EAAiBiE,QAAQtM,iBAAiB,gBAAiBsQ,IAE3DzG,GAAU,IAAID,GAAQ0G,KACtBjI,EAAiBiE,QAAQ+E,MAAM,KAC/BvI,GAAwB,EACxB,CACD,CAAC,MAAOwI,GACRzG,QAAQC,IAAI,mCAAoCwG,EAChD,GACC,CAACrM,EAAW4I,GAAgByC,GAAyB1G,KAElD2H,GAAkBtP,EAAYyI,UAC/B1B,EAA0BsD,SAC7BoC,aAAa1F,EAA0BsD,SAExCqE,KACI5G,GAAiBuC,UACpBtD,EAA0BsD,QAAU4C,WAAWxE,UAC1CX,GAAiBuC,gBACdsC,EAAcqB,UAAU,GAAG/I,eACjCwB,GAAoB,KAEnB,OAEJE,GAAY,GACRmB,GAAiBuC,eACdsC,EAAcqB,UAAU,GAAG/I,cAEhC,CAACyJ,GAAyB5G,GAAkB6E,IAEzC4C,GAAsBvP,EAC3B,EAAGsO,WACEA,EAAKlR,KAAO,GAAKkP,GAAKiB,cAAciC,cACvClD,GAAKiB,cAAckC,WAAWhH,MAAO6F,IACpC,GAAIA,GAAQA,EAAK9K,OAAS8K,EAAK9K,MAAM0G,OAAS,EAAG,CAChD,MAAM1G,EAAQpC,SAASsO,cAAc,kBACjB,IAAhBpB,EAAKhB,QAAmBgB,EAAKqB,MAAQ9K,GACpCqH,GAAU,GACG,IAAZA,IAAiB1I,GAAOoM,cACrBjD,EAAcqB,UAAU/I,EAAsBqJ,EAAK9K,OAE1D0I,IAAW,IAEX7E,GAAoBiH,EAAK9K,OACzBiJ,aAAapH,IACbiH,GAAKiB,cAAcC,aACnBP,WAAWqC,GAAiB,OAG7BpD,GAAU,GACN1I,GAAOoM,QAAYtK,IAAmBA,IAAiBuK,YAAcvB,EAAK9K,MAOnE8B,IAAiBuK,YAAcvB,EAAK9K,OAASA,GAAOoM,SAC1DtK,IAAmBA,GAAgBwK,qBAAuBxK,GAAgByK,WAC7EzK,GAAgBwK,oBAAsB,EACtCnD,EAAcqB,UAAU/I,EAAsBqJ,EAAK9K,QAE/C8B,KAAiBA,GAAgBwK,qBAAuB,KAX7DxK,GAAkB,CACjByK,UAAW,EACXD,oBAAqB,EACrBD,UAAWvB,EAAK9K,OAEjBmJ,EAAcqB,UAAU/I,EAAsBqJ,EAAK9K,QAUrD,IAEER,GAAWqH,SAAiD,OAAtCrH,EAAUqH,QAAQ2F,iBAC3C1D,GAAKiB,cAAc0C,UAAU,CAC5BC,MAAOlN,EAAUqH,QAAQ2F,iBAAmB,GAC5CG,OAAQ1K,MAKZ,CAACzC,EAAWyC,EAAQ6G,GAAMzH,GAAIqH,GAASoD,GAAiB3C,IAGnD7J,GAA0B9C,EAAYyI,UAC3C3B,GAAc,GACdqF,IAAa,GACbI,OACIb,GAAUrB,UACbqB,GAAUrB,SAAU,EACpBmE,EAAsB,CACrBnB,UAAW,eACX5H,OAAQA,EACR6H,OAAQ,UACRrH,WAGFL,EAAiBwK,KACjB5J,GAAc,GACVsB,GAAiBuC,eACdsC,EAAcqB,UAAU,GAAG/I,kBAE9B6C,GAAiBuC,eACdsC,EAAcqB,UAAU,GAAG/I,6BAE9B6C,GAAiBuC,UACpBvD,GAAc,GACdL,GAAoB,GACpBgB,IAAW,GACXI,IAAa,GACbnF,MAED,IACC,GAAIM,GAAaA,EAAUqH,SAAWrH,EAAUqH,QAAQ2C,QAAUlF,GAAiBuC,QAAS,CAE3F,MAAMsE,EAAa3L,EAAUqH,QAAQ2C,OAAO4B,iBAAiB,GAC7D,GAAID,EAAY,CACf,MAAME,EAAWF,EAAWG,cAC5BzI,EAAegE,QAAUwE,EAASE,WAAa,IAC/C,CAED,MAAMC,EAA6C,CAClDC,mBAAoB,MAEjBrD,GAAe,KAClBoD,EAAqBE,SAAWtD,GAAe,IAE5C9D,GAAiBuC,UACpBjE,EAAiBiE,QAAU,IAAI8E,cAAcnM,EAAUqH,QAAQ2C,OAAQgC,IAEpElH,GAAiBuC,SAAWjE,EAAiBiE,SAChDjE,EAAiBiE,QAAQtM,iBAAiB,gBAAiBwR,IAG5D3H,GAAU,IAAID,GAAQ4H,KAClBnJ,EAAiBiE,SAASjE,EAAiBiE,QAAQ+E,MAAM,KAC7DvI,GAAwB,GACpBiB,GAAiBuC,UACpBhF,GAAiB4H,WAAWe,GAAW,KAExC,CACD,CAAC,MAAOzC,GACR3C,QAAQC,IAAI,mBAAoB0C,EAChC,GACC,CAACvI,EAAW4I,GAAgB2D,GAAqB5H,GAAQqG,GAAWlG,GAAkByE,KAkEzF,OAhEA1O,EAAU,KACLqI,IACHV,GAAgB,GAChByI,MAEM,KACFpJ,KACHyH,GAAKiB,cAAcC,aACnBW,EAAQC,QAAQ,GAAGlI,gCAA0C,CAC5DT,OAAQA,EACRQ,QACApB,SAGF8C,GAAOmF,QAASC,IACf3G,GAAkBiE,SAASrM,oBAAoB,gBAAiB+O,OAGhE,CAACzF,GAAYpB,IAChBrI,EAAU,KACTqJ,EAAgBmJ,KAAKrJ,IACnB,CAACA,IAEJnJ,EAAU,KACTsJ,GAAmB,KACjB,CAACC,KACJvJ,EAAU,KACT,MAAMyS,EAAuBhK,EAAe4D,QAAU5D,EAAe4D,OAAS,EACzEtD,IAAwB0J,GAAyBzN,IAAoBiF,GAAiBuC,SAAYzH,IAClGwD,GAAoBA,EAAiBiE,UACnCxH,IACJuD,EAAiBiE,QAAQwC,OACzBhG,GAAwB,MAIzB,CAACD,EAAsB/D,EAAkByD,EAAgB1D,KAE5D/E,EAAU,KACT,MAAMyS,EAAuBhK,EAAe4D,QAAU5D,EAAe4D,OAAS,EAC9E,GAAKtD,IAAwB0J,GAAyBzN,EAarD+F,QAAQC,IAAI,kCAZZ,GAAIzC,GAAoBA,EAAiBiE,SAAWvC,GAAiBuC,UAAYzH,KAC3EC,EAAkB,CACtB,MAAM0N,EAAY,IAAIC,KAAKlK,EAAgB,GAAGb,KAAUoI,KAAsB,CAC7E4C,KAAM9C,KAEPtB,MAAmB,GACnB1G,EAAkB,IAElByG,KAAmBmE,EAAWlK,EAAegE,QAC7C,GAKD,CAACzD,EAAsB/D,EAAkByD,EAAgB1D,GAAOwD,EAAkB0B,KACrFjK,EAAU,KACT,MAAM6S,EAAYC,EAA+BlD,GAAqBmD,GACtE/E,GAAkB6E,IAChB,CAACjD,KACJ5P,EAAU,KACT2E,MACE,IAGFpE,EAACmE,GAAe,CACfW,cAAeA,EACfV,uBAAwBA,GACxBC,WAAYA,EACZE,YAAaA,GACb6E,YAAaA,GACb5E,MAAOA,GACPuK,aAAcA,GACdtK,iBAAkBA,EAClBrD,WAAYA,EACZsD,wBAAyBA,GACzB4D,SAAUA,EACVqH,cAAeA,GACfhL,UAAWA,GACXC,UAAWA,EACXN,gBAAiBA,GACjBO,qBAAsBA,GACtBE,iBAAkB4I,GAClB3I,UAAWA,GACXzD,OAAQA,GAGX,GCxcc,SAAUkR,IAAcpL,OACpCA,EAAMK,YACNA,EAAWJ,mBACXA,EAAkBC,kBAClBA,EAAiBC,iBACjBA,EAAgBC,iBAChBA,EAAgBlG,OAChBA,IAEA,MAAOL,EAAOwR,GAAYxT,EAAS,KAC5ByT,EAAmBC,GAAwB1T,EAAS,IACpDiC,EAAW0R,GAAgB3T,EAAwB,OACnDkC,EAAYsH,GAAiBxJ,GAAS,IACtCoC,EAAgBwR,GAAqB5T,EAAS,GAC/C6T,EAAgBhL,EAAiB,KAChCiL,EAAc5L,GAAmBlI,GAAS,GAC3C+T,EAAkB/R,EAAQyR,EAC1BtR,EAAkB4R,GAAmB,IAAMA,GAAmB,GAYpExT,EAAU,KACR,GAAsB,oBAAXN,QAA0BA,OAAO+T,uBAAwB,CAClE,MAAMC,EAAqBC,IACzB,GAAmB,OAAfA,EAAMC,KAAe,CACvB,IAAIC,EAAWzR,KAAK0R,IAAIH,EAAMC,MACV,OAAhBD,EAAMI,QACRF,GAAYzR,KAAK4R,IAAKL,EAAMI,MAAQ3R,KAAK6R,GAAM,MAEjDJ,EAAWzR,KAAKC,IAAI,EAAGD,KAAKE,IAAI,IAAKuR,IACrCZ,EAASY,EACV,GAGGK,EAAoBtJ,UACxB,MAAMuJ,EACJV,uBAEF,GACEU,GAC+C,mBAAxCA,EAAkBD,kBAEzB,IAEqB,kBADMC,EAAkBD,qBAGzCxU,OAAOQ,iBAAiB,oBAAqBwT,EAEhD,CAAC,MAAOlC,GACPzG,QAAQmB,MAAM,mBAAoBsF,EACnC,MAGD9R,OAAOQ,iBAAiB,oBAAqBwT,GAG/C,MAAO,KACDA,GACFhU,OAAOS,oBAAoB,oBAAqBuT,KAKtDnQ,SAASrD,iBAAiB,QAASgU,EAAmB,CAAEE,MAAM,GAC/D,GACA,IAEHpU,EAAU,KAMR,GALAsT,EAAc9G,QAAU,IACnB8G,EAAc9G,QAAQW,UACzBqG,GAGEF,EAAc9G,QAAQH,QAAU,EAAG,CACrC,MAAMgI,EACJjS,KAAKC,OAAOiR,EAAc9G,SAAWpK,KAAKE,OAAOgR,EAAc9G,SAE/D6G,EADEzR,GAAmByS,EAAY,EACd5G,GAASrL,KAAKE,IAAI,IAAKmL,EAAO,IAE9BA,GAASrL,KAAKC,IAAI,EAAGoL,EAAO,IAElD,CAEI7L,GAAkC,OAAdF,IAAsBC,GAC7C2S,KAED,CAACd,EAAiB5R,EAAiBF,EAAWC,IAElD3B,EAAU,KACL6B,GAAkB,KAAqB,OAAdH,IAAuBC,GACnD4S,IAEG1S,EAAiB,IAAoB,OAAdH,GAC1B4S,KAEC,CAACzS,EAAgBH,EAAWC,IAE9B,MAAM4S,EAAiB,KACrBnB,EAAa,GACb,MAAMoB,EAAQC,YAAY,KACxBrB,EAAc3F,GACC,OAATA,GAAiBA,GAAQ,GAC3BiH,cAAcF,GACdvL,GAAc,GACP,MAEFwE,EAAO,IAEf,MAGC6G,EAAiB,KACrBlB,EAAa,MACRG,GACHtK,GAAc,IAKlB,OACE1I,EAACiB,GAAW,CACVC,MAAO+R,EACP9R,UAAWA,EACXC,WAAYA,EACZC,gBAAiBA,EACjBC,eAAgBA,EAChBC,OAAQA,EAAMxC,SAEbqC,GACCpB,EAACmH,IACCC,gBAAiBA,EACjBtC,cAxHc,KACpB4N,EAAS,IACTE,EAAqB,GACrBC,EAAa,MACbnK,GAAc,GACdoK,EAAkB,GAClB1L,GAAgB,GAChB2L,EAAc9G,QAAU,IAkHlB5E,OAAQA,EACRK,YAAaA,EACbJ,mBAAoBA,EACpBC,kBAAmBA,EACnBC,iBAAkBA,EAClBC,iBAAkBA,EAClBlG,OAAQA,KAKlB,CCxJA,SAAS6S,IAAM/D,QAAEA,EAAO9O,OAAEA,IACxB,MAAMC,UAAEA,GAAcC,EAAWC,IAAoB,CAAA,EAIrD,OACE1B,EAACqU,GAAM,CAACC,MAAI,EAACnR,UAAU,gBAAepE,SACpCiB,EAAA,MAAA,CAAKmD,UAAU,aAAYpE,SACzBiB,EAAA,MAAA,CAAKmD,UAAU,cAAapE,SAExBuB,EAAAiU,EADDlE,EACC,CAAAtR,SAAA,CACEiB,EAAA,KAAA,CACEiC,MAAO,CACLuB,WACEjC,GAAQU,OAAOuS,SAASC,mBACxB,wBACFC,SAAUnT,GAAQU,OAAOuS,SAASG,iBAAmB,OACrDhR,MAAOpC,GAAQU,OAAOuS,SAASI,cAAgB,OAC/CC,WACEtT,GAAQU,OAAOuS,SAASM,mBAAqB,UAChD/V,SAEAyC,IAAYuC,EAAagR,sBAE5B/U,OACEmD,UAAU,sBACVlB,MAAO,CACLuB,WACEjC,GAAQU,OAAOwB,MAAMC,gBAAkB,oBACzCgR,SAAUnT,GAAQU,OAAOwB,MAAMuR,cAAgB,OAC/CrR,MAAOpC,GAAQU,OAAOwB,MAAMwR,eAAiB,QAC9ClW,SAEAyC,IAAYuC,EAAamR,uBAI9B,CAAAnW,SAAA,CACEiB,EAAA,KAAA,CACEiC,MAAO,CACLuB,WACEjC,GAAQU,OAAOuS,SAASC,mBACxB,wBACFC,SAAUnT,GAAQU,OAAOuS,SAASG,iBAAmB,OACrDhR,MAAOpC,GAAQU,OAAOuS,SAASI,cAAgB,OAC/CC,WACEtT,GAAQU,OAAOuS,SAASM,mBAAqB,UAChD/V,SAEAyC,IAAYuC,EAAaoR,uBAE5BnV,OACEmD,UAAU,sBACVlB,MAAO,CACLuB,WACEjC,GAAQU,OAAOwB,MAAMC,gBAAkB,oBACzCgR,SAAUnT,GAAQU,OAAOwB,MAAMuR,cAAgB,OAC/CrR,MAAOpC,GAAQU,OAAOwB,MAAMwR,eAAiB,QAC9ClW,SACD,GAAGyC,IACHuC,EAAaqR,cACRC,SAAsB7T,IAC3BuC,EAAauR,qCAQ7B,CCrEA,SAASC,IAAiBC,cAAEA,EAAajU,OAAEA,EAAMkU,OAACA,IACjD,MAAOC,EAAiBC,GAAsBzW,EAAS,CACtDsH,UAAU,EACV6J,QAAS,MAEHuF,EAASC,GAAc3W,GAAS,GACjC0F,EAAYmD,EAAO,MAEnB+N,EAAyBlU,EAAYyI,UAC1C,MAAM0L,QAAmBC,IACzBL,EAAmBI,GACnBF,GAAW,IACT,IAMH,OAJApW,EAAU,KACTqW,KACE,IAECF,EACI5V,EAACiW,EAAa,CAACC,IAAKT,EAAQU,WAAW,UAG3CT,GAAiBlP,SACbxG,EAACoU,GAAK,CAAC7S,OAAQA,EAAQ8O,QAASqF,GAAiBrF,UAGxDrQ,EAACmF,GAAM,CACNC,OAAO,EACPC,IAAKT,EACLU,kBAAmB,EACnBP,iBAAkBA,EAClBQ,YACAE,YAAa,IAAM+P,KAAgB,GACnChQ,iBAAiB,aACjBvD,MAAO,CACN0D,SAAU,WACVC,IAAK,EACLC,OAAQ,EACRC,OAAQ,EACRvF,MAAO,OACPC,OAAQ,OACRuF,UAAW,UAIf,CC9CA,SAASqQ,IAAYC,KAAEA,EAAIC,QAAEA,EAAOC,iBAAEA,EAAmB,8CACxD,MAAMC,EAAiBzO,EAAgC,MACjD0O,EAAkB1O,EAAsB,MAE9C,IAAI2O,EAAsB,CACzBC,UAAU,EACVC,UAAU,EACVC,YAAY,EACZC,OAAO,EACPC,OAAO,EACPC,aAAc,OACdrQ,QAAS,WACTsQ,OAAQC,GAkCT,OA/BAzX,EAAU,KACT,IAAKgX,EAAgBxK,SAAWoK,GAAQG,GAAgBvK,QAAS,CAChE,MAAMkL,EAASC,GAAQZ,EAAevK,QAAS,IAC3CyK,IAEJS,EAAOlM,MAAM,KACZwL,EAAgBxK,QAAUkL,EAE1B,MAAME,EAAiB,IACnBX,EACHY,QAAS,CAAC,CAAE1Q,IAAKyP,EAAMhE,KAAM,2BAG9B8E,EAAOR,SAASU,EAAeV,UAC/BQ,EAAOvQ,IAAIyQ,EAAeC,SAE1BhB,IAAUa,IAEX,GACC,CAACd,EAAMG,IAEV/W,EAAU,KACT,MAAM0X,EAASV,EAAgBxK,QAC/B,MAAO,KACFkL,IAAWA,EAAOI,eACrBJ,EAAOK,UACPf,EAAgBxK,QAAU,QAG1B,CAACwK,IAGHzW,EAAA,MAAA,CAAKmD,UAAWoT,EAAgBxX,SAC/BiB,EAAA,QAAA,CAAOqF,IAAKmR,EAAgBO,OAAK,EAAC5T,UAAU,WAAWsU,aAAW,EAACC,OAASzG,GAAMA,EAAE0G,oBAGvF,CC9CA,SAASC,IAAiBC,eAAEA,EAAcC,iBAAEA,GAAmB,EAAKC,OAAEA,EAAMpQ,OAAEA,EAAMF,iBAAEA,EAAgBnE,eAAEA,EAAcgE,mBAAEA,IACvH,MAAM9F,UAAEA,GAAcC,EAAWC,IAAoB,CAAA,GAC9CsW,EAAWC,GAAgB/Y,GAAS,GAWrCgZ,EAAkB,KACvBD,GAAcD,IAIf,OACC1X,EAAAiU,EAAA,CAAAxV,SAAA,CACCuB,EAAA,MAAA,CAAK6C,UAAU,qFAAqFlB,MAAO,CAAEkW,WAAY7U,GAAgBrB,OAAOwB,MAAMzB,iBAAiBjD,SAAA,CACtKiB,EAAA,MAAA,CAAKmD,UAAU,oDAAmDpE,SACjEiB,EAACoD,GAAOC,SAAO,EAACC,eAAgBA,MAEjChD,EAAA,MAAA,CAAK6C,UAAU,SAAQpE,SAAA,CACrB8Y,GACAvX,EAAA,MAAA,CAAK6C,UAAU,YAAWpE,SAAA,CACzBiB,EAAA,KAAA,CACCmD,UAAU,cACVlB,MAAO,CACNuB,WAAYF,GAAgBrB,OAAOuS,SAASC,mBAAqB,wBACjEC,SAAUpR,GAAgBrB,OAAOuS,SAASG,iBAAmB,OAC7DhR,MAAOL,GAAgBrB,OAAOuS,SAASI,cAAgB,OACvDC,WAAYvR,GAAgBrB,OAAOuS,SAASM,mBAAqB,UACjE/V,SAEAyC,IAAYuC,EAAaqU,iBAE3BpY,EAAA,IAAA,CACCiC,MAAO,CACNuB,WAAYF,GAAgBrB,OAAOwB,MAAMC,gBAAkB,oBAC3DgR,SAAUpR,GAAgBrB,OAAOwB,MAAMuR,cAAgB,OACvDrR,MAAOL,GAAgBrB,OAAOwB,MAAMwR,eAAiB,WACrDlW,SAEAyC,IAAYuC,EAAasU,UAE3BrY,EAAA,IAAA,CACCiC,MAAO,CACNuB,WAAYF,GAAgBrB,OAAOwB,MAAMC,gBAAkB,oBAC3DgR,SAAUpR,GAAgBrB,OAAOwB,MAAMuR,cAAgB,OACvDrR,MAAOL,GAAgBrB,OAAOwB,MAAMwR,eAAiB,WACrDlW,SAEAuZ,EAAmBT,KAGrB7X,EAAA,MAAA,CAAKmD,UAAU,iDAAiD8B,QAASiT,EAAiBtR,IAAK2R,EAA0B5Q,GAAS6Q,IAAI,YAGvIV,GACAxX,SAAK6C,UAAU,uBAAsBpE,SAAA,CACpCiB,EAAA,KAAA,CACCiC,MAAO,CACNuB,WAAYF,GAAgBrB,OAAOuS,SAASC,mBAAqB,wBACjEC,SAAUpR,GAAgBrB,OAAOuS,SAASG,iBAAmB,OAC7DhR,MAAOL,GAAgBrB,OAAOuS,SAASI,cAAgB,OACvDC,WAAYvR,GAAgBrB,OAAOuS,SAASM,mBAAqB,UACjE/V,SAEAyC,IAAYuC,EAAa+T,oBAE3B9X,OACCmD,UAAU,wBACVlB,MAAO,CACNuB,WAAYF,GAAgBrB,OAAOwB,MAAMC,gBAAkB,oBAC3DgR,SAAUpR,GAAgBrB,OAAOwB,MAAMuR,cAAgB,OACvDrR,MAAOL,GAAgBrB,OAAOwB,MAAMwR,eAAiB,WACrDlW,SAEAyC,IAAYuC,EAAa0U,qCAK7BZ,GACA7X,EAAA,MAAA,CAAKmD,UAAU,6BAA4BpE,SAC1CiB,EAACgG,EAAc,CAACQ,UAAU,EAAOL,WAAY3E,IAAYuC,EAAa2U,WAAYvV,UAAU,eAAeiD,WAnFzF,KAClB2R,EACHA,MAEAtQ,EAAiB6G,KAElBhH,GAAmB,IA6EwHhE,eAAgBA,SAIzJ0U,GACA1X,EAAC+T,GAAM,CAAClR,UAAU,gCAAgCwV,QAAST,EAAiB5D,KAAM0D,EAASjZ,SAAA,CAC1FiB,EAAA,MAAA,CAAKmD,UAAU,mBAAkBpE,SAChCiB,EAAA,OAAA,CAAMmD,UAAU,WAAW8B,QAASiT,EAAenZ,SAClDiB,EAACkF,GAAC,CAAC/B,UAAU,wDAGfnD,EAAA,MAAA,CAAKmD,UAAU,4CAA2CpE,SACzDiB,EAACoW,GAAW,CAACC,KAAM1O,EAASiR,EAAOjR,GAAQkR,SAAWD,EAAOE,KAAKD,SAAUtC,iBAAiB,oDAMnG,CCzGA,SAASwC,IAAOhH,OACfA,EAAMrK,YACNA,EAAWnG,OACXA,EAAMyX,WACNA,EAAUC,uBACVA,EAAsBC,uBACtBA,EAAsBC,WACtBA,EAAUC,UACVA,EAASC,oBACTA,EAAmBnL,KACnBA,IAaA,MAAMvG,OAAEA,EAAMG,WAAEA,EAAUF,WAAEA,EAAU0R,kBAAEA,EAAiBC,SAAEA,EAAQ1R,MAAEA,EAAK2R,SAAEA,GAAa9R,EACnF+R,EAAW,CAACC,EAAsBC,EAAsBC,GAA0BC,SAASL,IAC1FM,EAAYtE,GAAiBtW,GAAS,GAYvC6a,EAAenY,EAAYyI,UAChC,IACKoP,UACGvL,EAAK8L,KAAKC,QAAQ,CACvBlI,SACAlK,QACAqS,KAAMX,EACN5R,SACAnH,OAAQoH,IAETyR,OApBFtJ,EAAQC,QAAQlI,EAAY,CAC3BT,OAAQ0K,EACRlK,MAAOA,EACPrH,OAAQoH,EACRuS,YAAab,EACbc,WAAY,IACZzS,OAAQA,GAiBR,CAAC,MAAOgE,GACRnB,QAAQC,IAAIkB,EACZ,GACC,CAAC8N,KAEEjY,UAAEA,GAAcC,EAAWC,IAAoB,CAAA,EAErDjC,EAAU,MACLwZ,GAA0BC,IAC7Ba,KAEC,CAACd,EAAwBC,IAE5B,MAAMmB,EAAiBjB,IAAcJ,EAAaE,GAA0BD,EAAyBA,GAA0BC,GAE/H,OACClZ,EAACsa,IAAInX,UAAU,+BAA8BpE,SAC5CuB,EAAA,MAAA,CAAK6C,UAAU,0DAAyDpE,SAAA,CACvEiB,EAACuV,GAAgB,CAACE,OAAQlU,GAAQkU,OAAQD,cAAeA,IACzDxV,EAACua,GAAM,CACNjG,KAAMwF,EACNnB,QAAS,CAAC6B,EAAGnC,OAKblV,UAAU,gBACVsX,OAAO,SAAQ1b,SAEfiB,EAAA,MAAA,CACCmD,UAAU,mIACVlB,MAAO,CAAEkW,WAAY5W,GAAQU,OAAOwB,MAAMzB,iBAAiBjD,SAE3DuB,EAAA,MAAA,CAAK6C,UAAU,8BAA6BpE,SAAA,CAC3CiB,EAACoD,EAAM,CAACsX,MAAOlZ,IAAYuC,EAAa4W,wBAAyBrX,eAAgB/B,IACjFvB,EAAA,MAAA,CAAKmD,UAAU,0CAAyCpE,SACvDiB,EAAA,QAAA,CAAO2G,QAAQ,OAAOxD,UAAU,sEAAsE4T,OAAK,EAAC6D,MAAI,EAACC,UAAQ,EAACpD,aAAW,EAAA1Y,SACpIiB,EAAA,SAAA,CAAQ4G,IAAKe,IAAWmT,EAAWC,KAAOC,EAA0BC,EAAqB5I,KAAK,kBAG/FgI,GAAkBra,EAACgG,EAAc,CAAC1C,eAAgB/B,EAAQ4B,UAAU,qBAAqBgD,WAAY3E,IAAYuC,EAAamX,MAAO9U,WAAY,KAAK+S,oBAQ9J,CCjGO,MAAMgC,GAAgB,KAC5B,MAAMzT,YACLA,EAAWnG,OACXA,EAAM6Z,QACNA,EAAOC,QACPA,EAAOjC,UACPA,EAASzR,OACTA,EAAM2T,cACNA,EAAazD,eACbA,EAAcpQ,iBACdA,EAAgBH,mBAChBA,EAAkB4R,uBAClBA,EAAsBD,uBACtBA,EAAsBrD,QACtBA,EAAOF,gBACPA,EAAe6F,cACfA,EAAahU,kBACbA,EAAiBC,iBACjBA,EAAgBpD,uBAChBA,EAAsBiV,oBACtBA,EAAmBnL,KACnBA,EAAIiL,WACJA,GACG1X,EAAW2M,IACToN,qBAAEA,GAAyB/Z,EAAWC,IAAoB,CAAA,EAC1D4B,EAAiBmY,EAAela,GAMtC,OAJA9B,EAAU,KACT+b,IAAuBlY,GAAgBoY,WACrC,CAACpY,IAEA+X,EAEF/a,EAAAiU,EAAA,CAAAxV,SAAA,CACCiB,EAACuV,GAAgB,IACjBvV,EAACua,GAAM,CACNE,OAAO,SACPnG,MAAM,EACNnR,UAAU,gBACVwV,QAAS,CAACvF,EAAOiF,OAIhBtZ,SAEDiB,EAAC4X,GAAgB,CAChBC,eAAgBA,GAAkBwD,EAClCtD,OAAQ,KACPqD,MACAhX,KAEDqD,iBAAkBA,EAClBE,OAAQA,EACRrE,eAAgBA,EAChBgE,mBAAoBA,SAOrB8R,EAEFpZ,EAAC+Y,GAAM,CACNC,YAAY,EACZjH,OAAQuJ,EAER5T,YAAaA,EAMbwG,KAAMA,IAML0H,EAEF5V,EAAA,MAAA,CAAKmD,UAAU,sFACNlB,MAAO,CAAEkW,WAAY7U,GAAgBrB,OAAOwB,MAAMzB,0BAE1DhC,EAACiW,EAAa,CAACC,IAAK5S,GAAgBmS,OAAQU,WAAW,YAItDT,EAAgBlP,SACZxG,EAACoU,GAAK,IAGTmH,GAAkB1D,GAAmBqB,GAA2BD,GAahEsC,GAAiBrC,GAA0BD,KAA4BpB,EAE1E7X,EAAC+Y,IACAC,YAAY,EACZjH,OAAQuJ,EACRpC,uBAAwBA,EACxBxR,YAAaA,EACbuR,uBAAwBA,EACxB1X,OAAQ+B,EACR+V,oBAAqBA,EACrBnL,KAAMA,EACNiL,WAAYA,IAKd7Y,EAAAiU,EAAA,CAAAxV,SAAA,CACCiB,EAACuV,GAAgB,IACjBvV,EAACua,GAAM,CACNE,OAAO,SACPnG,MAAI,EACJnR,UAAU,gBACVwV,QAAS,CAACvF,EAAOiF,OAIhBtZ,SAEDiB,EAAC4X,GAAgB,CAChBC,eAAgBA,EAChBE,OAAQ,KACPqD,MACAhX,KAEDqD,iBAAkBA,EAClBE,OAAQA,EACRrE,eAAgBA,EAChBgE,mBAAoBA,SAhDtBtH,EAACyS,GAAa,CACblR,OAAQ+B,EACR+D,OAAQiU,EACR5T,YAAaA,EACbJ,mBAAoBA,EACpBC,kBAAmBA,EACnBC,iBAAkBA,EAClBC,iBAAkBA,KC1FTkU,GAAoC,EAChDjU,cACAnG,SACAqa,QACAR,UACAjN,cACA0N,mBACAC,iBACAC,2BACAC,2BACAC,4BACAC,cACAC,gBACA9C,sBACAF,aACAkC,UACAjC,gBAEA,MAAMzR,OAAEA,EAAM6R,SAAEA,EAAQ1R,WAAEA,EAAUF,WAAEA,EAAUC,MAAEA,EAAKyR,kBAAEA,EAAiB8C,gBAAEA,EAAeC,YAAEA,GAAgB3U,EACvGwG,EAAOnO,EAAQ,IAAMuc,EAAeV,GAAQ,CAACA,KAC5CL,EAAetN,GAAoB/O,GAAS,IAC5C+Z,EAAwBsD,GAA6Brd,GAAS,IAC9Dga,EAAwBsD,GAA6Btd,GAAS,IAC9DwW,EAAiBC,GAAsBzW,EAAS,CACtDsH,UAAU,EACV6J,QAAS,MAEHwH,EAAgBtQ,GAAqBrI,EAAS,KAC9Cud,EAAiBnV,IAAsBpI,GAAS,IAChD0W,GAASC,IAAc3W,GAAS,IAChCwd,GAAW3O,IAAgB7O,GAAS,IACpCoc,GAAe7T,IAAoBvI,EAAS,KAC5Cyd,GAAenV,IAAoBtI,EAAwB8S,MAC5D4K,SAAEA,IC5CT,SAAuBF,GACrB,MAAOE,EAAUC,GAAe3d,EAAyB,KAClD4d,EAAmBC,GAAwB7d,GAAS,GAErDiU,EAAoBvR,EAAawR,IACrC,IACE,MAAM4J,MAAEA,EAAK3J,KAAEA,EAAIG,MAAEA,GAAUJ,EAC/ByJ,EAAa3P,GAAyB,IACjCA,EACH,CACE8P,MAAOA,GAAOC,iBAAcpe,EAC5BwU,KAAMA,GAAM4J,iBAAcpe,EAC1B2U,MAAOA,GAAOyJ,iBAAcpe,EAC5Bqe,WAAW,IAAIC,MAAOC,gBAG3B,CAAC,MAAOzR,GACPnB,QAAQC,IAAIkB,EACb,GACA,IAEGgI,EAAoB/R,EAAYyI,UACpC,MAAMuJ,EACJV,uBAEF,QAC+B,IAAtBU,GACwC,mBAAxCA,EAAkBD,kBAEzB,IAEmB,kBADMC,EAAkBD,oBAEvCoJ,GAAqB,GAErBvS,QAAQkB,KAAK,wCAEhB,CAAC,MAAOC,GACPnB,QAAQmB,MAAM,kDAAmDA,EAClE,MAGDoR,GAAqB,IAEtB,IAmBH,OAjBAtd,EAAU,KACJid,GAAaI,EACf3d,OAAOQ,iBAAiB,oBAAqBwT,GAE7C0J,EAAY,IAEP,KACL1d,OAAOS,oBAAoB,oBAAqBuT,KAEjD,CAACuJ,EAAWI,EAAmB3J,IAElC1T,EAAU,KACJid,GACF/I,KAED,CAAC+I,EAAW/I,IAER,CAAEiJ,WACX,CDnBsBS,CAAcX,IAE7BtY,GAAyBxC,EAAY,KAC1C6F,GAAiB6G,KACjB/G,EAAkB,IAClB0G,GAAiB,GACjB3G,IAAmB,IACjB,IAEGgW,GAAWpN,IAChBgM,EAAY,IAAKhM,EAAMG,QAASiI,EAAmBpI,KACnDqN,KACAf,GAA0B,GAC1BjV,EAAkB2I,GAClBqM,GAA0B,GAC1BnM,EAAsB,CACrBnB,UAAW,GAAGnH,gCACdT,OAAQiU,GACRpM,OAAQ,SACRrH,QACAwI,QAASiI,EAAmBpI,KAEzByM,IACHvM,EAAsB,CACrBnB,UAAW,GAAGnH,yBACdT,OAAQiU,GACRpM,OAAQ,SACRsO,eAAgBxL,IAA4B2K,GAC5C9U,WAIG0V,GAA8B,KACnC/V,GAAiB,MACjBC,GAAiB,KAGZgW,GAAY7b,EACjByI,MAAO6F,IACN,GAAIA,GAA6B,YAArBA,GAAMwN,YAAiD,iBAArBxN,GAAMyN,YAAgD,MAAfzN,GAAM0N,KAS1F,OARAxN,EAAsB,CACrBnB,UAAW,GAAGnH,qCACdT,OAAQiU,GACRpM,OAAQ,UACRrH,UAED0U,GAA0B,QAC1BN,IAA4B/L,GAG7BiM,IAAgBjM,GAChBqM,GAA0B,GAC1BgB,KACA,MAAMM,EAAcvC,GACpBkB,GAA0B,GAC1BpM,EAAsB,CACrBnB,UAAW,GAAGnH,iCACdT,OAAQwW,EACR3O,OAAQ,UACRrH,UAEG8U,IACHvM,EAAsB,CACrBnB,UAAW,GAAGnH,yBACdT,OAAQwW,EACR3O,OAAQ,UACRsO,eAAgBxL,IAA4B2K,GAC5C9U,WAGH,CAAC2R,EAAU1R,EAAYwT,KAGlBwC,GAAmC/L,IACxCyK,GAA0B,GAC1BjV,EAAkB,IAClBgV,GAA0B,GAC1BrO,EAAK6P,YAAYC,wBAAwB,CACxCjM,OAAQA,GAAUuJ,GAClB2C,UAAW,KACVC,GAAuB,CACtBjP,UAAW,GAAGnH,cACdT,OAAQiU,GACR6C,WAAY,WACZ9L,KAAM,6BACNxK,WAGFuW,OAAQ,KACPrC,MACAmC,GAAuB,CACtBjP,UAAW,GAAGnH,cACdT,OAAQiU,GACR6C,WAAY,OACZ9L,KAAM,6BACNxK,WAGF8Q,QAAS,KACRqD,MACAkC,GAAuB,CACtBjP,UAAW,GAAGnH,cACdT,OAAQiU,GACR6C,WAAY,QACZ9L,KAAM,6BACNxK,WAGFyV,QAAUnQ,IACTmQ,GAAQnQ,GACR+Q,GAAuB,CACtBjP,UAAW,GAAGnH,cACdT,OAAQiU,GACR6C,WAAY,QACZ9L,KAAM,6BACNxK,WAGF4V,UAAYvN,IACXgO,GAAuB,CACtBjP,UAAW,GAAGnH,cACdT,OAAQiU,GACR6C,WAAY,UACZ9L,KAAM,6BACNxK,UAED4V,GAAUvN,OAKPlC,GAAmBpM,EACxByI,MAAOgU,EAAYC,KAClB,MAAMC,EAAeC,QAAQH,EAAKrf,cAAsByf,QAAQ,IAE1DC,EAAWJ,QAAoDA,QAAoBK,EAAYN,GAC/FO,EAAgBC,EAA0B,CAC/ClX,SACAmX,aAAc,GAAGxF,IACjB9Y,OAAQ,GAAGoH,IACXmX,mBAAoBjX,EACpBkX,YAhLe,IAiLfC,UAAWzF,EACX0F,aAAc7C,IAEfjM,EAAsB,CACrBnB,UAAW,GAAGnH,wBACdT,OAAQiU,GACRzT,QACAqI,KAAMiP,KAAKC,UAAU,IAAIR,EAAe,CAAEL,eAAcc,UAAWX,GAAY,OAEhF7C,MACA,UACO3N,EAAKoR,WAAWC,mBAAmB,CACxClB,OACAO,gBACA7M,OAAQuJ,GACRzT,gBAEKqG,EAAKoR,WAAWE,cAAc,CACnCC,MAAOrD,EACPsD,UAAW,SACXC,KAAM/C,GACN7K,OAAQuJ,KAET9Q,QAAQC,IAAI,+BACZnD,IAAmB,GACnB8I,EAAsB,CACrBnB,UAAW,GAAGnH,iBACdT,OAAQiU,GACRpM,OAAQ,UACRrH,QACAqI,KAAMiP,KAAKC,UAAUR,KAEtBxO,EAAsB,CACrBnB,UAAW,gBACX5H,OAAQiU,GACRpM,OAAQ,UACRrH,UAEDL,GAAiBwK,KACjBnD,WAAW,KACVvH,IAAmB,IACjB,IACH,CAAC,MAAOqE,GACRyE,EAAsB,CACrBnB,UAAW,gBACX5H,OAAQiU,GACRpM,OAAQ,SACRrH,QACAwI,QAASiI,EAAmB3M,KAE7ByE,EAAsB,CACrBnB,UAAW,GAAGnH,gBACdT,OAAQiU,GACRpM,OAAQ,SACRrH,QACAwI,QAASiI,EAAmB3M,GAC5BuE,KAAMiP,KAAKC,UAAUR,KAEtBrX,EAAkB+Q,EAAmB3M,IACrCrE,IAAmB,GACnBkD,QAAQC,IAAIkB,EAAO,sBACnB,CAAS,QACToC,IAAa,GACb+N,KACA,GAEF,CAACR,GAAesB,GAAUf,EAAkBC,IAEvC8D,GAAyBhe,EAC9ByI,MAAO0H,IACN,IACC,MAAM8N,QAAY3R,EAAK6P,YAAY+B,qBAAqB/N,GAClDgO,EAAaF,GAAK3P,MAAM6P,WACxBC,EAAoBrD,IAAiB3K,IACrCiO,EAAuBjO,IACvBkO,EAAuB,IAE7B,GADArK,IAAW,IACQ,IAAfkK,EAGH,OAFAxC,UACA9V,GAAiB6G,KAGlBL,GAAiB,IACE,IAAf8R,QACGtC,GAAU,MACS,OAAfsC,GAAuBC,EAAoBC,EAAuBC,EAC5E5C,GAAQuC,GAAK3P,MAAMvE,OAEnBmS,GAAgC/L,EAEjC,CAAC,MAAOpG,GACRnB,QAAQC,IAAIkB,GACZ4R,KACA9V,GAAiB6G,KACjBL,GAAiB,EACjB,GAEF,CAACqP,GAASG,GAAWhW,KAGhBqO,GAAyBlU,EAAYyI,UAC1C,IAAKzC,EAAa,QAAUA,EAAa,OAIxC,YAHAsU,EAAY,CACX7L,QAAS,4DAKX,MAAM0F,QAAmBC,IACzB,GAAID,EAAWvP,SACd4J,EAAsB,CACrBnB,UAAW,GAAGnH,sBACdT,OAAQiU,GACRpM,OAAQ,SACRrH,UAEDgO,IAAW,OACL,CACSyF,GAEdsE,GAFctE,IAIdzF,IAAW,GAEZzF,EAAsB,CACrBnB,UAAW,GAAGnH,sBACdT,OAAQiU,GACRpM,OAAQ,UACRrH,SAED,CACD8N,EAAmBI,GACnBxO,EAAkB,IAClBD,IAAmB,IACjB,CAACsY,GAAwB/X,EAAOyT,GAAe7T,GAAkBK,IAoBpE,OAlBArI,EAAU,KACTsQ,EAAQoQ,KAAKC,EAAkB,CAAEC,SAAUC,IAC3CvQ,EAAQC,QAAQ,cACd,IAEHvQ,EAAU,KACL4b,GAAWjC,GACXtR,GACHgO,MAEC,CAAChO,EAAYF,EAAYyT,EAASjC,IAErC3Z,EAAU,KACL4b,GAAWjC,GACXqD,GAAmB3U,GAAcwT,IACpCwC,MAEC,CAACrB,EAAiB3U,EAAYwT,GAAeD,EAASjC,IAExDpZ,EAACugB,EAAuB,CAAAxhB,SACvBiB,EAAClB,GAAoB,CAAAC,SACpBiB,EAACoO,EAAcnO,SAAQ,CACtBC,MAAO,CACNwH,cACAnG,SACA6Z,UACAc,cACAb,UACAjC,YACA+C,gBACAxU,SACA2T,iBACAzD,iBACApQ,oBACAH,sBACA4R,yBACAD,yBACArD,WACAF,kBACA6F,gBACAxN,gBACAC,oBACAC,mBACA7J,0BACAmD,oBACAC,oBACA2G,cACAkL,sBACAF,aACAjL,QACAnP,SAEDiB,EAACmb,GAAa,CAAA"}
1
+ {"version":3,"file":"BodyScan-BYiULEJt.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\";\n\nfunction LevelScreen({ angle, countdown, isScanning, isInTargetRange, stabilityScore, children, config }: LevelScreenProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\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 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 resolvedConfig={config} />\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 handleShowStreamCamera,\n loadingCam,\n handleUserMedia,\n handlePause,\n pause,\n recordingStarted,\n startSendingVideoFrames,\n showPause,\n webcamRef,\n handleUserMediaError,\n resetDetector,\n videoConstraints,\n webcamKey,\n config,\n}: CameraScanChildProps) {\n const { media } = useContext(MediaContext) || {};\n const { translate } = useContext(LanguageContext) || {};\n\n return (\n <div className=\"App w-screen h-[100vh] relative \">\n <span\n onClick={() => {\n handleShowStreamCamera();\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={handleUserMedia}\n onUserMediaError={handleUserMediaError}\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={handlePause}\n resolvedConfig={config}\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={handleShowStreamCamera}\n resolvedConfig={config}\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 resolvedConfig={config}\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 \"posthog-js\";\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: number | undefined | NodeJS.Timeout;\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, config }: ScanningComponentProps) {\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<number | null | NodeJS.Timeout>(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 [supportedTypes, setSupportedTypes] = useState<string[]>([]);\n\tconst [webcamKey, setWebcamKey] = useState(0);\n\tconst [currentVideoConstraints, setCurrentVideoConstraints] = useState<MediaTrackConstraints>(videoConstraintsExact);\n\n\tlet counter = 0;\n\n\tconst { setStartGyro, handleFileUpload, setUploadLoading, swan, onScanStart } = useContext(ParamsContext);\n\n\tconst handleShowStreamCamera = () => {\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\tcounter = 0;\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 handleReScan = useCallback(() => {\n\t\trescanSupportCaptureEvent({\n\t\t\teventName: `${shopDomain}/rescan`,\n\t\t\temail,\n\t\t\tscanID: scanID,\n\t\t\theight: heightInCm,\n\t\t\tgender,\n\t\t\tstatus: false,\n\t\t});\n\t\tallowAudioToPlay.current = false;\n\t\thandleShowStreamCamera();\n\t\tstartSendingVideoFrames();\n\t}, [handleShowStreamCamera, scanID, email]);\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\tcounter = 0;\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\n\t\t\tposthog.capture(`${shopDomain}/pose_detection_connected`, {\n\t\t\t\tscanID: scanID,\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}, [scanID, shopDomain, email]);\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: scanID,\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, scanID, 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\tconst handleDataAvailable = useCallback(\n\t\t({ data }: BlobEvent) => {\n\t\t\tif (data.size > 0 && swan.poseDetection.connected()) {\n\t\t\t\tswan.poseDetection.poseStatus(async (data:any) => {\n\t\t\t\t\tif (data && data.audio && data.audio.length > 0) {\n\t\t\t\t\t\tconst audio = document.querySelector(\"#audioElement\") as HTMLAudioElement | null;\n\t\t\t\t\t\tif (data.status === true && data.sid === id) {\n\t\t\t\t\t\t\tif (counter < 2) {\n\t\t\t\t\t\t\t\tif (counter === 0 && audio?.paused) {\n\t\t\t\t\t\t\t\t\tawait speechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\tcounter += 1;\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tsetEmptyAudioSource(data.audio);\n\t\t\t\t\t\t\t\tclearTimeout(audioTimeoutId);\n\t\t\t\t\t\t\t\tswan.poseDetection.disconnect();\n\t\t\t\t\t\t\t\tsetTimeout(postPoseProcess, 1000);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tcounter = 0;\n\t\t\t\t\t\t\tif (audio?.paused && (!lastVideoPlayed || lastVideoPlayed?.audioName !== data.audio)) {\n\t\t\t\t\t\t\t\tlastVideoPlayed = {\n\t\t\t\t\t\t\t\t\tskipCount: 2,\n\t\t\t\t\t\t\t\t\tno_of_times_skipped: 0,\n\t\t\t\t\t\t\t\t\taudioName: data.audio,\n\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\tspeechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t} else if (lastVideoPlayed?.audioName === data.audio && audio?.paused) {\n\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\tlastVideoPlayed.no_of_times_skipped = 0;\n\t\t\t\t\t\t\t\t\tspeechService.playAudio(voiceOverAssetsPath + data.audio);\n\t\t\t\t\t\t\t\t} else {\n\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}\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: scanID,\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t\t},\n\t\t[webcamRef, scanID, swan, id, counter, postPoseProcess, speechService],\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: scanID,\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([...events, 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, events, playAudio, allowAudioToPlay, onScanStart]);\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: scanID,\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}, [startAgain, 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\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\thandleFileUpload?.(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\thandleShowStreamCamera();\n\t}, []);\n\n\treturn (\n\t\t<CameraScanChild\n\t\t\tresetDetector={resetDetector}\n\t\t\thandleShowStreamCamera={handleShowStreamCamera}\n\t\t\tloadingCam={loadingCam}\n\t\t\thandlePause={handlePause}\n\t\t\tshowRestart={showRestart}\n\t\t\tpause={pause}\n\t\t\thandleReScan={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\thandleUserMedia={handleUserMedia}\n\t\t\thandleUserMediaError={handleUserMediaError}\n\t\t\tvideoConstraints={currentVideoConstraints}\n\t\t\twebcamKey={webcamKey}\n\t\t\tconfig={config}\n\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 config,\n}: AngleDetectorProps) {\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 config={config}\n >\n {isScanning && (\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 config={config}\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\">\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\";\n// import \"video.js/dist/video-js.css\";\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\";\n\nfunction ScanErrorMessage({ scanFailsError, serverAtCapacity = false, onNext, gender, setScanUniqueKey, resolvedConfig, setIsVideoUploaded }: ScanErrorMessageProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\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 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 resolvedConfig={resolvedConfig} />\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\t{serverAtCapacity && (\n\t\t\t\t\t\t<div className=\"p-[1rem] text-center\">\n\t\t\t\t\t\t\t<h3\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.serverAtCapacity)}\n\t\t\t\t\t\t\t</h3>\n\t\t\t\t\t\t\t<p\n\t\t\t\t\t\t\t\tclassName=\"text-base mt-[0.5rem]\"\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.serverAtCapacityDescription)}\n\t\t\t\t\t\t\t</p>\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} resolvedConfig={resolvedConfig} />\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 \"posthog-js\";\nimport Header from \"./Header\";\nimport CameraPermission from \"./bodyScan/CameraPermission\";\nimport { UserDetails } 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\tisFaceScan,\n\tisVideoUploadedCorrect,\n\tisMeasurementAvailable,\n\tonComplete,\n\tisSuccess,\n\tonCustomScanSuccess,\n\tswan,\n}: {\n\tconfig?: any;\n\tisFaceScan: boolean;\n\tscanId: string;\n\tisVideoUploadedCorrect?: boolean;\n\tisMeasurementAvailable?: boolean;\n\tuserDetails: UserDetails;\n\tonComplete?: () => void;\n\tisSuccess?: boolean;\n\tonCustomScanSuccess?: () => void;\n\tswan: any;\n}) {\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 || (isFaceScan ? isMeasurementAvailable && isVideoUploadedCorrect : isVideoUploadedCorrect || isMeasurementAvailable);\n\n\treturn (\n\t\t<Box className=\"flex h-full w-full flex-col \">\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)} resolvedConfig={config} />\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 resolvedConfig={config} 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, useEffect } 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 { LanguageContext } from \"../../utils/context/languageContext\";\nimport { useLocalConfig } from \"../../config/useLocalConfig\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\n\nexport const BodyScanSteps = () => {\n\tconst {\n\t\tuserDetails,\n\t\tconfig,\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\thandleShowStreamCamera,\n\t\tonCustomScanSuccess,\n\t\tswan,\n\t\tonComplete\n\t} = useContext(ParamsContext);\n\tconst { setPreferredLanguage } = useContext(LanguageContext) || {};\n\tconst resolvedConfig = useLocalConfig(config);\n\n\tuseEffect(() => {\n\t\tsetPreferredLanguage?.(resolvedConfig?.language);\n\t}, [resolvedConfig]);\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\n\t\t\t\t\t\tscanFailsError={scanFailsError || isError}\n\t\t\t\t\t\tonNext={() => {\n\t\t\t\t\t\t\tonRetry?.();\n\t\t\t\t\t\t\thandleShowStreamCamera();\n\t\t\t\t\t\t}}\n\t\t\t\t\t\tsetScanUniqueKey={setScanUniqueKey}\n\t\t\t\t\t\tgender={gender}\n\t\t\t\t\t\tresolvedConfig={resolvedConfig}\n\t\t\t\t\t\tsetIsVideoUploaded={setIsVideoUploaded}\n\t\t\t\t\t/>\n\t\t\t\t</Drawer>\n\t\t\t</>\n\t\t);\n\t}\n\n\tif (isSuccess) {\n\t\treturn (\n\t\t\t<SignUp\n\t\t\t\tisFaceScan={false}\n\t\t\t\tscanId={scanUniqueKey}\n\t\t\t\tuserDetails={userDetails}\n\t\t\t\tswan={swan}\n\t\t\t/>\n\t\t);\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\"\n style={{ background: resolvedConfig?.style?.base?.backgroundColor }}\n >\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\tconfig={resolvedConfig}\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\tisFaceScan={false}\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\tconfig={resolvedConfig}\n\t\t\t\tonCustomScanSuccess={onCustomScanSuccess}\n\t\t\t\tswan={swan}\n\t\t\t\tonComplete={onComplete}\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\thandleShowStreamCamera();\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\tresolvedConfig={resolvedConfig}\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 posthog from \"posthog-js\";\nimport useGyroSensor from \"../../customHooks/useGyroSensor\";\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 { getSwanService } from \"../../utils/service/swanService\";\nimport { posthogPublicHost, posthogPublicKey } from \"../../utils/constants\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\nimport MediaContextProvider from \"../../utils/context/mediaContext\";\nimport { BodyScanProps } from \"../../bodyScan\";\nimport { BodyScanSteps } from \"./BodyScanSteps\";\n\nconst clothesFit = \"0\";\n\nexport const BodyScan: React.FC<BodyScanProps> = ({\n\tuserDetails,\n\tconfig,\n\ttoken,\n\tonRetry,\n\tonScanStart,\n\tonUploadingStart,\n\tonUploadingEnd,\n\tonMeasurementSocketStart,\n\tonMeasurementSocketClose,\n\tonScanSuccessIntermediate,\n\tonScanError,\n\tonScanSuccess,\n\tonCustomScanSuccess,\n\tonComplete,\n\tisError,\n\tisSuccess,\n}) => {\n\tconst { gender, scanType, shopDomain, heightInCm, email, deviceFocalLength, deviceModelName, callbackUrl } = 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\n\tconst handleShowStreamCamera = 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\tonScanSuccessIntermediate?.(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 handleFileUpload = 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});\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\tonUploadingStart?.();\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) {\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\tconsole.log(error, \"video upload failed\");\n\t\t\t} finally {\n\t\t\t\tsetStartGyro(false);\n\t\t\t\tonUploadingEnd?.();\n\t\t\t}\n\t\t},\n\t\t[scanUniqueKey, gyroData, onUploadingStart, onUploadingEnd],\n\t);\n\tconst 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\tconst scanId = scanUniqueKey;\n\t\t\tif (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\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}, [checkMeasurementStatus, email, scanUniqueKey, setScanUniqueKey, shopDomain]);\n\n\tuseEffect(() => {\n\t\tposthog.init(posthogPublicKey, { api_host: posthogPublicHost });\n\t\tposthog.capture(\"$pageview\");\n\t}, []);\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<MediaContextProvider>\n\t\t\t\t<ParamsContext.Provider\n\t\t\t\t\tvalue={{\n\t\t\t\t\t\tuserDetails,\n\t\t\t\t\t\tconfig,\n\t\t\t\t\t\tonRetry,\n\t\t\t\t\t\tonScanError,\n\t\t\t\t\t\tisError,\n\t\t\t\t\t\tisSuccess,\n\t\t\t\t\t\tonScanSuccess,\n\t\t\t\t\t\tgender,\n\t\t\t\t\t\tscanUniqueKey,\n\t\t\t\t\t\tscanFailsError,\n\t\t\t\t\t\tsetScanUniqueKey,\n\t\t\t\t\t\tsetIsVideoUploaded,\n\t\t\t\t\t\tisMeasurementAvailable,\n\t\t\t\t\t\tisVideoUploadedCorrect,\n\t\t\t\t\t\tloading,\n\t\t\t\t\t\tshowDeniedModal,\n\t\t\t\t\t\tuploadLoading,\n\t\t\t\t\t\tsetStartGyro,\n\t\t\t\t\t\thandleFileUpload,\n\t\t\t\t\t\tsetUploadLoading,\n\t\t\t\t\t\thandleShowStreamCamera,\n\t\t\t\t\t\tsetScanFailsError,\n\t\t\t\t\t\tsetScanStartTime,\n\t\t\t\t\t\tonScanStart,\n\t\t\t\t\t\tonCustomScanSuccess,\n\t\t\t\t\t\tonComplete,\n\t\t\t\t\t\tswan,\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t<BodyScanSteps />\n\t\t\t\t</ParamsContext.Provider>\n\t\t\t</MediaContextProvider>\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","React","memo","_jsxs","width","height","viewBox","fill","xmlns","d","stroke","strokeWidth","strokeLinecap","strokeLinejoin","LevelScreen","angle","countdown","isScanning","isInTargetRange","stabilityScore","config","translate","useContext","LanguageContext","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","resolvedConfig","border","fontFamily","base","baseFontFamily","color","cn","transform","transition","LanguageKeys","startLevelCheck","leavePhone","placePhoneUpright","CameraScanChild","handleShowStreamCamera","loadingCam","handleUserMedia","handlePause","pause","recordingStarted","startSendingVideoFrames","showPause","webcamRef","handleUserMediaError","resetDetector","videoConstraints","webcamKey","onClick","X","Webcam","audio","ref","screenshotQuality","mirrored","screenshotFormat","onUserMedia","onUserMediaError","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","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","import","setBackend","ready","modelConfig","runtime","modelType","solutionPath","createDetector","SupportedModels","BlazePose","mediapipeError","warn","error","isFrontFrame","body","length","filter","p","current","then","success","s","callback","video","readyState","poses","estimatePoses","flipHorizontal","keypoints","slice","map","kp","x","y","score","prev","err","retryLoading","useTensorFlow","firstScan","poseStoppedRef","supportedTypes","setSupportedTypes","setWebcamKey","currentVideoConstraints","setCurrentVideoConstraints","videoConstraintsExact","counter","setStartGyro","handleFileUpload","setUploadLoading","swan","onScanStart","ParamsContext","clearTimeout","generateUuid","speechService","stopAudio","stop","forEach","el","stream","setTimeout","videoConstraintsFallback","handleReScan","rescanSupportCaptureEvent","eventName","status","poseDetection","disconnect","preferredVideoTypes","getPreferredMediaRecorderTypes","recordingMimeType","getRecordingMimeType","recordingExtension","getRecordingExtension","stopRecording","playAudio","handleSocket","connect","posthog","capture","handleSpinDataAvailable","data","concat","handleScanTimeCapture","message","handleStartCaptureClick","videoTrack","getVideoTracks","settings","getSettings","frameRate","mediaRecorderOptions","videoBitsPerSecond","mimeType","MediaRecorder","start","e","postPoseProcess","handleDataAvailable","connected","poseStatus","querySelector","sid","paused","audioName","no_of_times_skipped","skipCount","getScreenshot","videoEmit","image","scanId","getCurrentTimeInSeconds","push","bypassChecksToBackup","videoFile","File","type","supported","getSupportedMediaRecorderTypes","videoTypes","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","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","serverAtCapacity","onNext","showModal","setShowModal","handleShowModal","background","issueWithScan","reason","handleErrorMessage","VIDEO_POSTER_GENDER_BASED","alt","serverAtCapacityDescription","scanAgain","onClose","GENDER","PRE_LINK","male","SignUp","isFaceScan","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","setPreferredLanguage","useLocalConfig","language","BodyScan","token","onUploadingStart","onUploadingEnd","onMeasurementSocketStart","onMeasurementSocketClose","onScanSuccessIntermediate","onScanError","onScanSuccess","deviceModelName","callbackUrl","getSwanService","setIsVideoUploadedCorrect","setIsMeasurementAvailable","isVideoUploaded","startGyro","scanStartTime","gyroData","setGyroData","permissionGranted","setPermissionGranted","alpha","toString","timestamp","Date","toISOString","useGyroSensor","onError","clearScanStartTimeAndScanId","completionTime","onSuccess","scanStatus","resultType","code","scanIdToSet","handleMeasurementRecommendation","measurement","handleMeasurementSocket","onPreopen","handleWebSocketCapture","connection","onOpen","file","recordedFps","fileSizeInMB","Number","toFixed","videoFps","getVideoFPS","arrayMetaData","createObjectMetadataArray","focal_length","customer_store_url","clothes_fit","scan_type","callback_url","JSON","stringify","video_fps","fileUpload","uploadFileFrontend","setDeviceInfo","model","detection","gyro","checkMeasurementStatus","res","getMeasurementResult","isMeasured","tempScanStartTime","currentTimeInSeconds","fiveMinutesInSeconds","init","posthogPublicKey","api_host","posthogPublicHost","LanguageContextProvider"],"mappings":"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"}