{"version":3,"file":"FaceScan-uo2FlNED.mjs","sources":["../src/customHooks/useFaceScan.ts","../src/components/faceScan/FaceScanErrorScreen.tsx","../src/components/faceScan/FaceScanGuide.tsx","../src/components/faceScan/FaceScanStep.tsx","../src/components/faceScan/FaceScan.tsx"],"sourcesContent":["import {\n  useState,\n  useRef,\n  useEffect,\n  useCallback,\n  useMemo,\n  RefObject,\n} from \"react\";\n// 1. Use TYPE-ONLY imports. These are erased at build time and won't crash SSR.\nimport type { PoseDetector } from \"@tensorflow-models/pose-detection\";\nimport posthog from \"../utils/posthog\";\nimport speechService from \"../utils/service/speechService\";\nimport { videoConstraints, voiceOverAssetsPath } from \"../utils/constants\";\n\n/**\n * useFaceScan Hook with PostHog Analytics\n * Refactored for Zero-Config SSR/CSR Compatibility\n */\n\n// Debounce utility\nfunction debounce(fn: (...args: any) => void, delay: number) {\n  let timer: number | NodeJS.Timeout;\n  return (...args: any) => {\n    if (timer) clearTimeout(timer);\n    timer = setTimeout(() => fn(...args), delay);\n  };\n}\n\n// Global Singleton Cache (Preserves state across re-renders)\nlet preloadedDetector: PoseDetector | null = null;\nlet isPreloading = false;\n\n// 2. Async Preloader with Dynamic Imports\nexport const preloadDetector = async (): Promise<PoseDetector | null> => {\n  // Guard: Never run on server\n  if (typeof window === \"undefined\") return null;\n  if (preloadedDetector) return preloadedDetector;\n  if (isPreloading) {\n    // Simple wait mechanism if already loading\n    while (isPreloading) {\n      await new Promise(r => setTimeout(r, 100));\n      if (preloadedDetector) return preloadedDetector;\n    }\n  }\n\n  isPreloading = true;\n\n  try {\n    // DYNAMIC IMPORTS: Only fetch these heavy bundles in the browser\n    const [tf, poseDetection] = await Promise.all([\n      import(\"@tensorflow/tfjs\"),\n      import(\"@tensorflow-models/pose-detection\")\n    ]);\n\n    // Initialize Backend\n    await tf.setBackend(\"webgl\");\n    await tf.ready();\n\n    const detector = await poseDetection.createDetector(\n      poseDetection.SupportedModels.BlazePose,\n      {\n        runtime: \"tfjs\",\n        modelType: \"full\",\n      }\n    );\n\n    // Dummy video warmup (Performance Optimization)\n    // This runs the model once on a blank canvas to \"wake up\" the GPU\n    // so the first user frame detects instantly.\n    const dummyCanvas = document.createElement(\"canvas\");\n    dummyCanvas.width = typeof videoConstraints.width === \"number\" ? videoConstraints.width : videoConstraints.width.ideal ?? videoConstraints.width.max ?? 1280;\n    dummyCanvas.height = typeof videoConstraints.height === \"number\" ? videoConstraints.height : videoConstraints.height.ideal ?? videoConstraints.height.max ?? 720;\n    const ctx = dummyCanvas.getContext(\"2d\");\n    if (ctx) {\n      ctx.fillStyle = \"black\";\n      ctx.fillRect(0, 0, dummyCanvas.width, dummyCanvas.height);\n      try {\n        await detector.estimatePoses(dummyCanvas);\n      } catch (e) {\n        console.warn(\"Warmup frame failed (non-fatal):\", e);\n      }\n    }\n\n    preloadedDetector = detector;\n    return detector;\n  } catch (err) {\n    console.error(\"Failed to preload detector:\", err);\n    return null;\n  } finally {\n    isPreloading = false;\n  }\n};\n\ntype Point = [number, number, number];\n\nfunction useFaceScan({\n  faceScanId,\n  onValidPose,\n  onScanComplete,\n  onModelReady,\n  onStartRecording,\n  shopDomain,\n  isError,\n  isSuccess,\n}: {\n  faceScanId: string;\n  onValidPose?: () => void;\n  onScanComplete?: () => void;\n  onModelReady?: () => void;\n  onStartRecording: () => void;\n  shopDomain: string;\n  isError?: boolean;\n  isSuccess?: boolean;\n}) {\n  const detectorRef = useRef<PoseDetector | null>(null);\n  const [scanStage, setScanStage] = useState(0);\n  const [consecutiveValid, setConsecutiveValid] = useState(0);\n  const [isModelLoaded, setIsModelLoaded] = useState(false);\n  const [isScanningActive, setIsScanningActive] = useState(false);\n  const isInferencingRef = useRef(false);\n  // Persistent offscreen canvas for inference input.\n  // tf.browser.fromPixels(videoElement) calls gl.texImage2D with the video,\n  // which returns black frames on SwiftShader (headless CI) because video decoded\n  // frames aren't in SW-composited memory. Drawing to a canvas first makes the\n  // pixels CPU-accessible so SwiftShader can read them correctly.\n  const inferenceCanvasRef = useRef<HTMLCanvasElement | null>(null);\n  \n  const utteranceRef = useRef<SpeechSynthesisUtterance | null>(null);\n  const scanStageRef = useRef(scanStage);\n  const consecutiveValidRef = useRef(consecutiveValid);\n  const validityBufferRef = useRef<boolean[]>([]);\n  const lastSpokenMessageRef = useRef(\"\");\n  const lastSpokenTimeRef = useRef(0);\n  const voiceEnabledRef = useRef(false);\n  \n  const VALIDITY_BUFFER_LENGTH = 10;\n  const CONSECUTIVE_VALID_LENGTH = 2;\n  const TOTAL_STAGES = 4;\n  const POSE_TRACKING_INTERVAL = 4000;\n  const lastPoseTrackingTimeRef = useRef(0);\n\n  // Safe iOS detection (SSR safe)\n  const isIOS = typeof navigator !== 'undefined' && (\n    /iPad|iPhone|iPod/.test(navigator.userAgent) ||\n    (navigator.platform === \"MacIntel\" && navigator.maxTouchPoints > 1)\n  );\n\n  const directionMessages = useMemo(\n    () => [\n      \"Face front\",\n      \"Turn head fully left\",\n      \"Turn head fully right\",\n      \"Smile facing front\",\n    ],\n    []\n  );\n\n  const debouncedTrackPoseDetection = useMemo(\n    () =>\n      debounce((payload) => {\n        posthog.capture(`${shopDomain}/face_scan_pose_detection`, payload);\n      }, 4000),\n    [shopDomain]\n  );\n\n  const trackPoseDetection = useCallback(\n    (\n      faceKeypoints: Point[],\n      bodyKeypoints: Point[],\n      stage: number,\n      headValid: boolean,\n      bodyInvalid: boolean,\n      consecutiveValidCount: number\n    ) => {\n      const now = Date.now();\n      if (now - lastPoseTrackingTimeRef.current < POSE_TRACKING_INTERVAL) {\n        return;\n      }\n\n      lastPoseTrackingTimeRef.current = now;\n\n      try {\n        const nose = faceKeypoints[0] || [0, 0, 0];\n        const leftEye = faceKeypoints[2] || [0, 0, 0];\n        const rightEye = faceKeypoints[5] || [0, 0, 0];\n        const leftMouth = faceKeypoints[9] || [0, 0, 0];\n        const rightMouth = faceKeypoints[10] || [0, 0, 0];\n\n        const faceKeypointScores = faceKeypoints\n          .map((kp: Point) => kp[2])\n          .filter((score: number) => score > 0);\n        const avgFaceScore =\n          faceKeypointScores.length > 0\n            ? faceKeypointScores.reduce((a, b) => a + b, 0) /\n              faceKeypointScores.length\n            : 0;\n\n        const bodyKeypointScores = bodyKeypoints\n          .map((kp: Point) => kp[2])\n          .filter((score: number) => score > 0);\n        const avgBodyScore =\n          bodyKeypointScores.length > 0\n            ? bodyKeypointScores.reduce((a: number, b: number) => a + b, 0) /\n              bodyKeypointScores.length\n            : 0;\n\n        const eyeDistance = Math.abs(leftEye[0] - rightEye[0]);\n        const noseToRight = nose[0] - rightEye[0];\n        const noseToLeft = leftEye[0] - nose[0];\n\n        let expectedPosition = \"frontal\";\n        if (stage === 1) expectedPosition = \"left\";\n        else if (stage === 2) expectedPosition = \"right\";\n        else if (stage === 3) expectedPosition = \"frontal_smile\";\n\n        const payload = {\n          faceScanId,\n          stage,\n          timestamp: new Date().toISOString(),\n          data: JSON.stringify({\n            headValid,\n            bodyInvalid,\n            consecutiveValid: consecutiveValidCount,\n            avgFaceScore: parseFloat(avgFaceScore.toFixed(3)),\n            avgBodyScore: parseFloat(avgBodyScore.toFixed(3)),\n            faceKeypointsDetected: faceKeypointScores.length,\n            bodyKeypointsDetected: bodyKeypointScores.length,\n            eyeDistance: parseFloat(eyeDistance.toFixed(2)),\n            noseToRightRatio: parseFloat((noseToRight / eyeDistance).toFixed(3)),\n            noseToLeftRatio: parseFloat((noseToLeft / eyeDistance).toFixed(3)),\n            noseScore: parseFloat(nose[2].toFixed(3)),\n            leftEyeScore: parseFloat(leftEye[2].toFixed(3)),\n            rightEyeScore: parseFloat(rightEye[2].toFixed(3)),\n            leftMouthScore: parseFloat(leftMouth[2].toFixed(3)),\n            rightMouthScore: parseFloat(rightMouth[2].toFixed(3)),\n            expectedPosition,\n          }),\n        };\n        debouncedTrackPoseDetection(payload);\n      } catch (error) {\n        console.warn(\"Failed to track pose detection:\", error);\n      }\n    },\n    [faceScanId, debouncedTrackPoseDetection]\n  );\n\n  const enableVoiceCommands = () => {\n    voiceEnabledRef.current = true;\n    if (typeof window !== 'undefined' && isIOS && \"speechSynthesis\" in window) {\n      const silentUtterance = new SpeechSynthesisUtterance(\"\");\n      silentUtterance.volume = 0;\n      speechSynthesis.speak(silentUtterance);\n    }\n  };\n\n  const startScanSequence = () => {\n    posthog.capture(`${shopDomain}/face_scan_detection_started`, {\n      faceScanId,\n      stage: scanStageRef.current,\n      timestamp: new Date().toISOString(),\n      stageName: directionMessages[scanStageRef.current],\n    });\n\n    setTimeout(() => {\n      setIsScanningActive(true);\n    }, 1500);\n  };\n\n  const isHeadInPosition = (stage: number, keypoints: Point[]) => {\n    // ... logic unchanged ...\n    const nose = 0;\n    const leftEye = 2;\n    const rightEye = 5;\n    const leftMouth = 9;\n    const rightMouth = 10;\n\n    const minScore = 0.2;\n    if (\n      !keypoints[nose] || !keypoints[leftEye] || !keypoints[rightEye] ||\n      keypoints[nose][2] < minScore ||\n      keypoints[leftEye][2] < minScore ||\n      keypoints[rightEye][2] < minScore\n    ) {\n      return false;\n    }\n\n    const eyesToMouthVerticalDistance =\n      0.5 * (keypoints[rightMouth][1] + keypoints[leftMouth][1]) -\n      0.5 * (keypoints[rightEye][1] + keypoints[leftEye][1]);\n\n    const faceTiltedness =\n      (0.5 * (keypoints[rightMouth][1] + keypoints[leftMouth][1]) -\n        keypoints[nose][1]) /\n      eyesToMouthVerticalDistance;\n\n    const faceIsTilted = faceTiltedness > 0.7 || faceTiltedness < 0.3;\n\n    const faceIsVertical =\n      Math.abs(keypoints[leftEye][1] - keypoints[rightEye][1]) <\n        0.5 * eyesToMouthVerticalDistance &&\n      Math.abs(keypoints[leftMouth][1] - keypoints[rightMouth][1]) <\n        0.5 * eyesToMouthVerticalDistance;\n\n    const eyeDistance = keypoints[leftEye][0] - keypoints[rightEye][0];\n    const noseToRight = keypoints[nose][0] - keypoints[rightEye][0];\n    const noseToLeft = keypoints[leftEye][0] - keypoints[nose][0];\n\n    const frontalFace =\n      noseToRight > 0.4 * eyeDistance && noseToRight < 0.6 * eyeDistance;\n    const fullLeft = noseToRight > 0.85 * eyeDistance;\n    const fullRight = noseToLeft > 0.85 * eyeDistance;\n\n    switch (stage) {\n      case 0:\n        return !faceIsTilted && faceIsVertical && frontalFace;\n      case 1:\n        return !faceIsTilted && faceIsVertical && fullLeft;\n      case 2:\n        return !faceIsTilted && faceIsVertical && fullRight;\n      case 3:\n        return !faceIsTilted && faceIsVertical && frontalFace;\n      default:\n        return false;\n    }\n  };\n\n  const faceScanDetector = async (\n    webcamRef: RefObject<HTMLVideoElement | null>,\n    canvasRef: RefObject<HTMLCanvasElement | null>\n  ) => {\n    if (\n      !detectorRef.current ||\n      !webcamRef.current ||\n      !isModelLoaded ||\n      !isScanningActive\n    )\n      return;\n\n    // Safety: ensure video is actually playing with data\n    if (webcamRef.current.readyState < 2) return;\n\n    // Prevent concurrent inference calls — important on slow GPUs (SwiftShader/CI)\n    // where a single estimatePoses() can take longer than the 500ms interval.\n    if (isInferencingRef.current) return;\n    isInferencingRef.current = true;\n\n    try {\n      const vw = webcamRef.current.videoWidth || 720;\n      const vh = webcamRef.current.videoHeight || 1280;\n      if (!inferenceCanvasRef.current) {\n        inferenceCanvasRef.current = document.createElement(\"canvas\");\n        inferenceCanvasRef.current.width = vw;\n        inferenceCanvasRef.current.height = vh;\n      }\n      const ictx = inferenceCanvasRef.current.getContext(\"2d\");\n      if (ictx) ictx.drawImage(webcamRef.current, 0, 0, vw, vh);\n      const poses = await detectorRef.current.estimatePoses(\n        ictx ? inferenceCanvasRef.current : webcamRef.current\n      );\n      if (poses.length > 0) {\n        const faceKeypoints: Point[] = [];\n        const bodyKeypoints: Point[] = [];\n        poses[0].keypoints.forEach((landmark, idx) => {\n          const score = landmark.score ?? 0;\n          const point: Point = [landmark.x ?? -1, landmark.y ?? -1, score];\n          if (idx <= 10) faceKeypoints.push(point);\n          else bodyKeypoints.push(point);\n        });\n\n        // Optional drawing\n        if (canvasRef?.current) {\n          const ctx = canvasRef.current.getContext(\"2d\");\n          const { width, height } = canvasRef.current;\n          if (ctx) {\n            ctx.clearRect(0, 0, width, height);\n\n            const progress = Math.min(\n              consecutiveValidRef.current / CONSECUTIVE_VALID_LENGTH,\n              1\n            );\n            const radius = height * 0.35;\n            ctx.beginPath();\n            ctx.strokeStyle = \"#00ff00\";\n            ctx.lineWidth = 6;\n            ctx.arc(\n              width / 2,\n              height / 2,\n              radius + 10,\n              -Math.PI / 2,\n              -Math.PI / 2 + progress * 2 * Math.PI\n            );\n            ctx.stroke();\n          }\n        }\n\n        const headValid = isHeadInPosition(scanStageRef.current, faceKeypoints);\n        const bodyInvalid =\n          bodyKeypoints.slice(2).filter((p) => p[2] > 0.7).length <= 2;\n\n        validityBufferRef.current.push(headValid && bodyInvalid);\n        if (validityBufferRef.current.length > VALIDITY_BUFFER_LENGTH) {\n          validityBufferRef.current.shift();\n        }\n\n        const validCount = validityBufferRef.current.filter(Boolean).length;\n        if (validCount >= 2) {\n          const nextValid = consecutiveValidRef.current + 1;\n          setConsecutiveValid(nextValid);\n        } else {\n          setConsecutiveValid(Math.max(0, consecutiveValidRef.current - 1));\n        }\n\n        if (consecutiveValidRef.current >= CONSECUTIVE_VALID_LENGTH) {\n          onValidPose?.();\n\n          const nextStage = scanStageRef.current + 1;\n          setConsecutiveValid(0);\n          validityBufferRef.current = [];\n          setScanStage(nextStage);\n          setIsScanningActive(false);\n\n          (async () => {\n            await speechService.playAudio(\n              `${voiceOverAssetsPath}face-scan-vos/Sound-effect.mp3`\n            );\n            posthog.capture(\n              `${shopDomain}/face_scan_detection_stage_completed`,\n              {\n                faceScanId,\n                completedStage: scanStageRef.current,\n                nextStage,\n                timestamp: new Date().toISOString(),\n                totalStages: TOTAL_STAGES,\n                stageName: directionMessages[scanStageRef.current],\n              }\n            );\n            \n            if (nextStage === 1 && onStartRecording) {\n              onStartRecording();\n            }\n            if (nextStage >= TOTAL_STAGES) {\n              onScanComplete?.();\n            } else {\n              let nextSound = null;\n              switch (nextStage) {\n                case 1: nextSound = \"Left.mp3\"; break;\n                case 2: nextSound = \"Right.mp3\"; break;\n                case 3: nextSound = \"Smile.mp3\"; break;\n              }\n              if (nextSound) {\n                await speechService.playAudio(\n                  `${voiceOverAssetsPath}face-scan-vos/${nextSound}`\n                );\n              }\n              startScanSequence();\n            }\n          })();\n        }\n\n        trackPoseDetection(\n          faceKeypoints,\n          bodyKeypoints,\n          scanStageRef.current,\n          headValid,\n          bodyInvalid,\n          consecutiveValidRef.current\n        );\n      }\n    } catch (err) {\n      console.error(\"Pose detection error:\", err);\n    } finally {\n      isInferencingRef.current = false;\n    }\n  };\n\n  const initializeModels = async () => {\n    // 3. Browser Guard\n    if (typeof window === 'undefined') return;\n\n    try {\n      console.log(\"Initializing Face Scan...\");\n      // Calls our safe, dynamic preloader\n      const detector = await preloadDetector();\n      \n      if (detector) {\n        detectorRef.current = detector;\n        console.log(\"Face scan model loaded successfully\");\n        setIsModelLoaded(true);\n        onModelReady?.();\n      }\n    } catch (err) {\n      console.error(\"Error initializing face scan:\", err);\n    }\n  };\n\n  const disposeModelAndTf = async () => {\n    // Optional: We might NOT want to dispose if we want to keep the cache \n    // for re-mounting. But if you must dispose:\n    try {\n      setIsModelLoaded(false);\n      // NOTE: We generally don't dispose the detector if we want to reuse it \n      // via the 'preloadedDetector' global variable.\n      // If you dispose here, make sure to set 'preloadedDetector = null' too.\n      /* if (detectorRef.current) {\n        await detectorRef.current.dispose();\n        detectorRef.current = null;\n        preloadedDetector = null; // Clear global cache if disposing\n      }\n      */\n    } catch (err) {\n      console.error(\"Error disposing resources:\", err);\n    } finally {\n      // Reset state\n      setScanStage(0);\n      setConsecutiveValid(0);\n      scanStageRef.current = 0;\n      consecutiveValidRef.current = 0;\n      validityBufferRef.current = [];\n    }\n  };\n\n  const resetScan = () => {\n    posthog.capture(`${shopDomain}/face_scan_reset`, {\n      faceScanId,\n      stage: scanStageRef.current,\n      timestamp: new Date().toISOString(),\n      stageName: directionMessages[scanStageRef.current],\n      consecutiveValid: consecutiveValidRef.current,\n    });\n\n    setScanStage(0);\n    setConsecutiveValid(0);\n    setIsScanningActive(false);\n    scanStageRef.current = 0;\n    consecutiveValidRef.current = 0;\n    validityBufferRef.current = [];\n    lastSpokenMessageRef.current = \"\";\n    lastSpokenTimeRef.current = 0;\n    voiceEnabledRef.current = false;\n    if (utteranceRef.current) {\n      speechSynthesis.cancel();\n      utteranceRef.current = null;\n    }\n  };\n\n  useEffect(() => {\n    if (isError || isSuccess) return;\n    initializeModels();\n    return () => {\n      disposeModelAndTf();\n    };\n  }, [isError, isSuccess]);\n\n  useEffect(() => {\n    scanStageRef.current = scanStage;\n  }, [scanStage]);\n\n  useEffect(() => {\n    consecutiveValidRef.current = consecutiveValid;\n  }, [consecutiveValid]);\n\n  return {\n    faceScanDetector,\n    scanStage,\n    setScanStage,\n    consecutiveValid,\n    isModelLoaded,\n    resetScan,\n    enableVoiceCommands,\n    startScanSequence,\n    isScanningActive,\n  };\n}\n\nexport default useFaceScan;","import { ArrowRight } from \"lucide-react\";\nimport Header from \"../Header\";\nimport SpecificButton from \"../../atoms/specificButton/SpecificButton\";\nimport { useContext } from \"react\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { useConfig } from \"../../utils/context/configContext\";\n\nfunction FaceScanErrorScreen({ onRetry, loading }: { onRetry?: () => void; loading?: boolean }) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\tconst config = useConfig();\n\n\treturn (\n\t\t<div data-testid=\"face-scan-error-screen\" className=\"fixed common-ui-main  top-[0] left-[0] z-[999]  flex justify-center items-center w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t<div className=\"flex flex-col w-full items-center p-[1rem] rounded-lg  \">\n\t\t\t\t<Header noTitle />\n\t\t\t\t<div className=\"flex flex-col items-center w-full\">\n\t\t\t\t\t<h2\n\t\t\t\t\t\tclassName=\"text-xl font-semibold text-gray-800 \"\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.scanFailed)}\n\t\t\t\t\t</h2>\n\t\t\t\t\t<p\n\t\t\t\t\t\tclassName=\"mb-[1.5rem]\"\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\tcolor: config?.style?.subheading?.subheadingColor || \"#4b5563\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.clickToResetScan)}\n\t\t\t\t\t</p>\n\t\t\t\t\t<SpecificButton\n\t\t\t\t\t\tdata-testid=\"face-scan-retry-btn\"\n\t\t\t\t\t\tdisabled={loading}\n\t\t\t\t\t\tclassName=\"w-full h-[45px] shadow-[0px_1px_2px_0px_#00000040] text-[16px]\"\n\t\t\t\t\t\tbuttonText={translate?.(LanguageKeys.resetScan)}\n\t\t\t\t\t\tpostfixIcon={<ArrowRight />}\n\t\t\t\t\t\tbuttonFunc={() => onRetry?.()}\n\t\t\t\t\t/>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t</div>\n\t);\n}\n\nexport default FaceScanErrorScreen;\n","import Header from \"../Header\";\nimport { GenderType } from \"../../utils/enums\";\nimport { directionMessages, FACE_SCAN_HEADSHOT, glassesOffVideo, maleGlassesOffVideo } from \"../../utils/constants\";\nimport { LanguageContext } from \"../../utils/context/languageContext\";\nimport { useContext } from \"react\";\nimport SpecificButton from \"../../atoms/specificButton/SpecificButton\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport { useConfig } from \"../../utils/context/configContext\";\nimport { Dispatch, ReactNode, SetStateAction } from \"react\";\nimport { Gender } from \"../../types/interfaces\";\n\ninterface IButtonConfig {\n\tisDisabled: boolean;\n\ticon?: ReactNode;\n\tonClick: () => void;\n\tlabel: string;\n}\n\ninterface FaceScanGuideProps {\n\tstage: number;\n\tvideoLoading: boolean;\n\tsetVideoLoading: Dispatch<SetStateAction<boolean>>;\n\tvideoError: boolean;\n\tsetVideoError: Dispatch<SetStateAction<boolean>>;\n\tgender: Gender;\n\tbtnConfig: IButtonConfig;\n}\n\nfunction FaceScanGuide({ stage, videoLoading, setVideoLoading, videoError, setVideoError, gender, btnConfig }: FaceScanGuideProps) {\n\tconst { translate } = useContext(LanguageContext) || {};\n\tconst config = useConfig();\n\n\tif (stage === -1) {\n\t\treturn (\n\t\t\t<div data-testid=\"face-scan-guide\" className=\"text-center common-ui-main  p-[16px] w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t\t<div className=\"w-full\">\n\t\t\t\t\t<Header noTitle />\n\t\t\t\t\t<h2\n\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t\t}}\n\t\t\t\t\t>\n\t\t\t\t\t\t{translate?.(LanguageKeys.faceVisible)}\n\t\t\t\t\t</h2>\n\t\t\t\t\t{videoLoading && (\n\t\t\t\t\t\t<div\n\t\t\t\t\t\t\tclassName=\"mb-4 flex items-center justify-center\"\n\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\t\tcolor: config?.style?.subheading?.subheadingColor || \"#4b5563\",\n\t\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t>\n\t\t\t\t\t\t\t{translate?.(LanguageKeys.loadingVideo)}\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t\t{videoError && (\n\t\t\t\t\t\t<div\n\t\t\t\t\t\t\tclassName=\"mb-4 text-red-600\"\n\t\t\t\t\t\t\tstyle={{\n\t\t\t\t\t\t\t\tfontFamily: config?.style?.subheading?.subheadingFontFamily || \"'Inter', sans-serif\",\n\t\t\t\t\t\t\t\tfontSize: config?.style?.subheading?.subheadingFontSize || \"14px\",\n\t\t\t\t\t\t\t\tcolor: \"#ff0000\",\n\t\t\t\t\t\t\t\tfontWeight: config?.style?.subheading?.subheadingFontWeight || \"normal\",\n\t\t\t\t\t\t\t}}\n\t\t\t\t\t\t>\n\t\t\t\t\t\t\t{translate?.(LanguageKeys.videoLoadFailed)}\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t\t{!videoError && (\n\t\t\t\t\t\t<div className={` w-[100px] mx-auto`}>\n\t\t\t\t\t\t\t<video\n\t\t\t\t\t\t\t\tsrc={gender === GenderType.Male ? maleGlassesOffVideo : glassesOffVideo}\n\t\t\t\t\t\t\t\tautoPlay\n\t\t\t\t\t\t\t\tloop\n\t\t\t\t\t\t\t\tcontrols={false}\n\t\t\t\t\t\t\t\tmuted\n\t\t\t\t\t\t\t\tplaysInline\n\t\t\t\t\t\t\t\tclassName=\"h-full w-full object-contain border-none\"\n\t\t\t\t\t\t\t\tonCanPlay={() => setVideoLoading(false)}\n\t\t\t\t\t\t\t\tonLoadStart={() => setVideoLoading(true)}\n\t\t\t\t\t\t\t\tonError={() => setVideoError(true)}\n\t\t\t\t\t\t\t\taria-label=\"Instructional video: Please remove your glasses before starting the scan.\"\n\t\t\t\t\t\t\t/>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t)}\n\t\t\t\t</div>\n\t\t\t\t<div className=\"flex justify-center w-full  p-[1rem]\">\n\t\t\t\t\t<SpecificButton\n\t\t\t\t\t\tdata-testid=\"face-scan-start-btn\"\n\t\t\t\t\t\tdisabled={btnConfig.isDisabled}\n\t\t\t\t\t\tclassName=\"!w-[60px] !h-[35px] !py-0 !px-0\"\n\t\t\t\t\t\tbuttonText={translate?.(btnConfig.label)}\n\t\t\t\t\t\tpostfixIcon={btnConfig?.icon}\n\t\t\t\t\t\tbuttonFunc={btnConfig.onClick}\n\t\t\t\t\t/>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t);\n\t}\n\n\treturn (\n\t\t<div className=\"text-center common-ui-main  p-[16px] w-full h-full\" style={{ background: config?.style?.base?.backgroundColor }}>\n\t\t\t<div className=\"w-full\">\n\t\t\t\t<Header noTitle />\n\t\t\t\t<h3\n\t\t\t\t\tclassName=\"mb-[0.5rem]\"\n\t\t\t\t\tstyle={{\n\t\t\t\t\t\tfontFamily: config?.style?.heading?.headingFontFamily || \"SeriouslyNostalgic Fn\",\n\t\t\t\t\t\tfontSize: config?.style?.heading?.headingFontSize || \"32px\",\n\t\t\t\t\t\tcolor: config?.style?.heading?.headingColor || \"#000\",\n\t\t\t\t\t\tfontWeight: config?.style?.heading?.headingFontWeight || \"normal\",\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t{translate?.(directionMessages[stage])}\n\t\t\t\t</h3>\n\t\t\t\t<div className=\"max-w-[400px] justify-center gap-1 mx-auto flex items-center\">\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 0 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.forward : FACE_SCAN_HEADSHOT.female.forward} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 1 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.left : FACE_SCAN_HEADSHOT.female.left} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 2 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.right : FACE_SCAN_HEADSHOT.female.right} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div className={` w-[120px] overflow-hidden ${stage === 3 ? \"opacity-100\" : \"opacity-0 hidden\"} `}>\n\t\t\t\t\t\t<video className=\"h-full w-full object-contain border-none\" muted loop autoPlay playsInline>\n\t\t\t\t\t\t\t<source src={gender === GenderType.Male ? FACE_SCAN_HEADSHOT.male.smile : FACE_SCAN_HEADSHOT.female.smile} type=\"video/mp4\" />\n\t\t\t\t\t\t</video>\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\t\t</div>\n\t);\n}\n\nexport default FaceScanGuide;\n","import React, { useContext } from \"react\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\nimport FaceScanErrorScreen from \"./FaceScanErrorScreen\";\nimport LoadingScreen from \"../LoadingScreen\";\nimport { videoConstraints } from \"../../utils/constants\";\nimport { Drawer } from \"@mui/material\";\nimport FaceScanGuide from \"./FaceScanGuide\";\nimport Modal from \"../Modal\";\nimport { FaceScanErrorType } from \"../../utils/enums\";\nimport { ArrowRight } from \"lucide-react\";\nimport { useConfig } from \"../../utils/context/configContext\";\ninterface FaceScanStepProps {\n\twebcamRef: React.RefObject<HTMLVideoElement | null>;\n\tcanvasRef: React.RefObject<HTMLCanvasElement | null>;\n}\n\nconst FaceScanStep: React.FC<FaceScanStepProps> = ({ webcamRef, canvasRef }) => {\n\tconst {\n\t\tisError,\n\t\tisSuccess,\n\t\tshowLoader,\n\t\thasError,\n\t\terrorType,\n\t\tresetScanState,\n\t\tshowGuideCard,\n\t\tscanStage,\n\t\tvideoLoading,\n\t\tsetVideoLoading,\n\t\tvideoError,\n\t\tsetVideoError,\n\t\tgender,\n\t\tgetButtonText,\n\t\tisScanning,\n\t\tstartScan,\n\t\tisModelLoaded,\n\t\tonRetry,\n\t} = useContext(ParamsContext);\n\tconst resolvedConfig = useConfig();\n\tif (isError) {\n\t\treturn <FaceScanErrorScreen onRetry={onRetry} />;\n\t}\n\tif (isSuccess) {\n\t\treturn (\n\t\t\t<div className=\"fixed z-[9] common-ui-main  w-full h-full\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t<LoadingScreen url={resolvedConfig?.loader} loaderType=\"black\" />\n\t\t\t</div>\n\t\t);\n\t}\n\n\treturn (\n\t\t<>\n\t\t\t<audio id=\"audioElement\" crossOrigin=\"anonymous\" preload=\"auto\" style={{ position: \"absolute\", zIndex: -99999 }} src={undefined} />\n\n\t\t\t{/* Error overlay */}\n\t\t\t{showLoader && !hasError && (\n\t\t\t\t<div className=\"fixed z-[9] common-ui-main  w-full h-full\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t\t{/* <Asset genderType={gender} /> */}\n\t\t\t\t\t<LoadingScreen url={resolvedConfig?.loader} loaderType=\"black\" />\n\t\t\t\t</div>\n\t\t\t)}\n\n\t\t\t{/* Always show camera view */}\n\t\t\t<div className=\"h-full flex-col common-ui-main relative w-full flex justify-center items-center text-center  rounded-t-[20px]\" style={{ background: resolvedConfig?.style?.base?.backgroundColor }}>\n\t\t\t\t<div className=\"flex-1 w-full max-w-md overflow-hidden\">\n\t\t\t\t\t<div className=\"w-full h-full\">\n\t\t\t\t\t\t<video\n\t\t\t\t\t\t\tref={webcamRef}\n\t\t\t\t\t\t\tautoPlay\n\t\t\t\t\t\t\tplaysInline\n\t\t\t\t\t\t\tmuted\n\t\t\t\t\t\t\twidth={videoConstraints.width.ideal}\n\t\t\t\t\t\t\theight={videoConstraints.height.ideal}\n\t\t\t\t\t\t\tclassName=\"w-full h-full object-cover fixed left-0 top-0 z-0\"\n\t\t\t\t\t\t\tstyle={{ transform: \"scaleX(-1)\" }}\n\t\t\t\t\t\t/>\n\t\t\t\t\t\t<canvas ref={canvasRef} width={videoConstraints.width.ideal} height={videoConstraints.height.ideal} style={{ transform: \"scaleX(-1)\", opacity: \"0\" }} />\n\t\t\t\t\t</div>\n\t\t\t\t</div>\n\t\t\t</div>\n\n\t\t\t{/* Error overlay */}\n\t\t\t{!showLoader && hasError && errorType === FaceScanErrorType.Camera && (\n\t\t\t\t<Modal testId=\"face-scan-camera-error-screen\" config={resolvedConfig} />\n\t\t\t)}\n\t\t\t{!showLoader && hasError && errorType !== FaceScanErrorType.Camera && (\n\t\t\t\t<FaceScanErrorScreen loading={showLoader} onRetry={resetScanState} />\n\t\t\t)}\n\n\t\t\t{/* Scan guide drawer - only show when scanning */}\n\t\t\t{showGuideCard && !hasError && !showLoader && (\n\t\t\t\t<Drawer\n\t\t\t\t\topen\n\t\t\t\t\tclassName=\"face-scan-small camera-draw\"\n\t\t\t\t\tanchor=\"bottom\"\n\t\t\t\t\tonClose={(event, reason) => {\n\t\t\t\t\t\tif (reason === \"backdropClick\") {\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t}\n\t\t\t\t\t}}\n\t\t\t\t\thideBackdrop\n\t\t\t\t>\n\t\t\t\t\t<FaceScanGuide\n\t\t\t\t\t\tstage={scanStage}\n\t\t\t\t\t\tvideoLoading={videoLoading}\n\t\t\t\t\t\tsetVideoLoading={setVideoLoading}\n\t\t\t\t\t\tvideoError={videoError}\n\t\t\t\t\t\tsetVideoError={setVideoError}\n\t\t\t\t\t\tgender={gender}\n\t\t\t\t\t\tbtnConfig={{\n\t\t\t\t\t\t\tlabel: getButtonText(),\n\t\t\t\t\t\t\tonClick: isScanning ? resetScanState : startScan,\n\t\t\t\t\t\t\tisDisabled: showLoader || !isModelLoaded,\n\t\t\t\t\t\t\ticon: isScanning ? <ArrowRight /> : <></>,\n\t\t\t\t\t\t}}\n\t\t\t\t\t/>\n\t\t\t\t</Drawer>\n\t\t\t)}\n\t\t</>\n\t);\n};\n\nexport default FaceScanStep;\n","\"use client\";\nimport { useEffect, useRef, useState, useCallback, useMemo } from \"react\";\nimport { FaceScanMetaData, FaceScanProps } from \"../../types/interfaces\";\nimport {\n\tgenerateUuid,\n\tgetPreferredMediaRecorderTypes,\n\tgetRecordingExtension,\n\tgetRecordingMimeType,\n\tgetSupportedMediaRecorderTypes,\n\tgetVideoFPS,\n\thandleScanTimeCapture,\n\thandleWebSocketCapture,\n} from \"../../utils/utils\";\nimport { videoConstraintsExact, videoConstraintsFallback, videoTypes, voiceOverAssetsPath } from \"../../utils/constants\";\nimport { getSwanService } from \"../../utils/service/swanService\";\nimport speechService from \"../../utils/service/speechService\";\nimport useFaceScan from \"../../customHooks/useFaceScan\";\nimport usePosthogPageview from \"../../customHooks/usePosthogPageview\";\nimport { LanguageKeys } from \"../../utils/languageKeys\";\nimport { FaceScanErrorType } from \"../../utils/enums\";\nimport LanguageContextProvider from \"../../utils/context/languageContext\";\nimport { ConfigProvider } from \"../../utils/context/configContext\";\nimport FaceScanStep from \"./FaceScanStep\";\nimport ParamsContext from \"../../utils/context/paramsContext\";\n\nexport const FaceScan: React.FC<FaceScanProps> = (props) => {\n\tconst { onScanSuccess, onScanError, onRetry, onScanStart, onCaptureComplete, onUploadStart, onUploadEnd, onMeasurementSocketStart, onMeasurementSocketClose, config, isError, isSuccess } = props;\n\tconst userDetails: FaceScanMetaData = props.userDetails ?? {\n\t\temail: \"\",\n\t\tgender: \"male\",\n\t\tdeviceFocalLength: 0,\n\t\tshopDomain: \"\",\n\t\tbodyScanId: \"\",\n\t\tcallbackUrl: null,\n\t};\n\tconst token = props.token ?? \"\";\n\tconst webcamRef = useRef<HTMLVideoElement | null>(null);\n\tconst canvasRef = useRef<HTMLCanvasElement | null>(null);\n\tconst mediaRecorderRef = useRef<MediaRecorder | null>(null);\n\tconst recordedBlobsRef = useRef<Blob[] | null>([]);\n\tconst streamRef = useRef<MediaStream | null>(null);\n\tconst recordedFpsRef = useRef<number | null>(null);\n\tconst drawFrameRafRef = useRef<number>(0);\n\tconst recordingVideoElRef = useRef<HTMLVideoElement | null>(null);\n\tconst [showLoader, setShowLoader] = useState(false);\n\tconst [modelReady, setModelReady] = useState(false);\n\tconst [isScanning, setIsScanning] = useState(false);\n\tconst [showGuideCard, setShowGuideCard] = useState(true);\n\tconst [videoLoading, setVideoLoading] = useState(false);\n\tconst [videoError, setVideoError] = useState(false);\n\tconst [hasError, setHasError] = useState(false);\n\tconst [errorType, setErrorType] = useState<FaceScanErrorType | null>(null);\n\tconst [faceScanId, setFaceScanId] = useState(generateUuid());\n\tconst swan = useMemo(() => getSwanService(token), [token]);\n\tconst { email, gender, deviceFocalLength, shopDomain, callbackUrl, bodyScanId } = userDetails;\n\tconst [supportedTypes, setSupportedTypes] = useState<string[]>([]);\n\tconst [currentVideoConstraints, setCurrentVideoConstraints] = useState<MediaTrackConstraints>(videoConstraintsExact);\n\tconst preferredVideoTypes = useMemo(() => getPreferredMediaRecorderTypes(), []);\n\tconst recordingMimeType = getRecordingMimeType(supportedTypes);\n\tconst recordingExtension = getRecordingExtension(recordingMimeType);\n\tusePosthogPageview();\n\n\tconst stopRecording = useCallback(() => {\n\t\tconsole.log(\"Stopping recording...\");\n\t\tcancelAnimationFrame(drawFrameRafRef.current);\n\t\tif (recordingVideoElRef.current) {\n\t\t\trecordingVideoElRef.current.pause();\n\t\t\trecordingVideoElRef.current.srcObject = null;\n\t\t\trecordingVideoElRef.current = null;\n\t\t}\n\t\tif (mediaRecorderRef.current && mediaRecorderRef.current.state === \"recording\") {\n\t\t\tmediaRecorderRef.current.stop();\n\t\t}\n\t\tmediaRecorderRef.current = null;\n\t}, []);\n\n\tconst uploadFinalVideo = async () => {\n\t\tonCaptureComplete?.();\n\t\tif (recordedBlobsRef.current) {\n\t\t\tif (recordedBlobsRef.current.length === 0) {\n\t\t\t\tconsole.error(\"No video data recorded\");\n\t\t\t\tsetErrorType(FaceScanErrorType.Scan);\n\t\t\t\tsetHasError(true);\n\t\t\t\tsetShowLoader(false);\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tonUploadStart?.();\n\t\t\tsetShowLoader(true);\n\t\t\tconst videoFile = new File(recordedBlobsRef.current, `${faceScanId}.${recordingExtension}`, {\n\t\t\t\ttype: recordingMimeType,\n\t\t\t});\n\t\t\t// Use FPS captured at recording time, fallback to calculation if not available\n\t\t\tconst videoFps = recordedFpsRef.current ?? (await getVideoFPS(videoFile));\n\n\t\t\tconst fileSize = videoFile.size;\n\t\t\tconst fileSizeMB = (fileSize / (1024 * 1024)).toFixed(2);\n\t\t\tconst estimatedDuration = Math.round(fileSize / 10000);\n\t\t\tconst videoData = {\n\t\t\t\tvideo_size_mb: parseFloat(fileSizeMB),\n\t\t\t\tvideo_size_bytes: fileSize,\n\t\t\t\tvideo_fps: videoFps || 0,\n\t\t\t\tblob_count: recordedBlobsRef.current.length,\n\t\t\t\testimated_duration_seconds: estimatedDuration,\n\t\t\t};\n\n\t\t\tconst metaData: any[] = [\n\t\t\t\t{ gender: gender },\n\t\t\t\t{ face_scan_id: faceScanId },\n\t\t\t\t{focal_length: `${deviceFocalLength}`},\n\t\t\t\t{ customer_store_url: shopDomain },\n\t\t\t\t{ scan_type: \"face_scan\" },\n\t\t\t\t{ body_scan_id: bodyScanId }\n\t\t\t];\n\t\t\tif (callbackUrl) {\n\t\t\t\tmetaData.push({ callback_url: callbackUrl });\n\t\t\t}\n\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/face_scan_meta_data`,\n\t\t\t\tfaceScanId,\n\t\t\t\temail,\n\t\t\t\tdata: JSON.stringify({ metaData, videoData }),\n\t\t\t});\n\n\t\t\tconst uploadStartTime = Date.now();\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/face_scan_upload_start`,\n\t\t\t\tfaceScanId,\n\t\t\t\temail,\n\t\t\t\tstartTime: uploadStartTime,\n\t\t\t});\n\n\t\t\ttry {\n\t\t\t\tconst res = await swan.fileUpload.faceScanFileUploader({\n\t\t\t\t\tfile: videoFile,\n\t\t\t\t\tarrayMetaData: metaData,\n\t\t\t\t\tobjectKey: faceScanId,\n\t\t\t\t\temail,\n\t\t\t\t\tcontentType: videoFile.type,\n\t\t\t\t});\n\n\t\t\t\tconst uploadEndTime = Date.now();\n\t\t\t\tconst uploadDuration = uploadEndTime - uploadStartTime;\n\n\t\t\t\thandleScanTimeCapture({\n\t\t\t\t\teventName: `${shopDomain}/face_scan_upload_complete`,\n\t\t\t\t\tfaceScanId,\n\t\t\t\t\temail,\n\t\t\t\t\tcompletionTime: uploadEndTime,\n\t\t\t\t\tuploadDuration,\n\t\t\t\t});\n\t\t\t\tonUploadEnd?.();\n\t\t\t\tconsole.log(\"✅ Upload successful\", res);\n\t\t\t\tswan.measurement.handlFaceScaneSocket({\n\t\t\t\t\tonPreopen: () => {\n\t\t\t\t\t\tonMeasurementSocketStart?.();\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"pre_open\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t},\n\t\t\t\t\tfaceScanId,\n\t\t\t\t\tonOpen: () => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"open\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tconsole.log(\"websocket connect open\");\n\t\t\t\t\t},\n\t\t\t\t\tonError: (err) => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"error\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tonError(err);\n\t\t\t\t\t},\n\t\t\t\t\tonSuccess: (data) => {\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"success\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tonSuccess(data);\n\t\t\t\t\t},\n\t\t\t\t\tonClose: () => {\n\t\t\t\t\t\tonMeasurementSocketClose?.();\n\t\t\t\t\t\tconsole.log(\"websocket connect close\");\n\t\t\t\t\t\thandleWebSocketCapture({\n\t\t\t\t\t\t\teventName: `${shopDomain}/webSocket`,\n\t\t\t\t\t\t\tfaceScanID: faceScanId,\n\t\t\t\t\t\t\tconnection: \"close\",\n\t\t\t\t\t\t\ttype: \"faceScan_recommendation\",\n\t\t\t\t\t\t\temail,\n\t\t\t\t\t\t});\n\t\t\t\t\t},\n\t\t\t\t});\n\t\t\t} catch (error) {\n\t\t\t\tonUploadEnd?.();\n\t\t\t\tonError(error);\n\t\t\t}\n\t\t}\n\t};\n\n\tconst startRecording = useCallback(() => {\n\t\tconst stream = webcamRef.current?.srcObject as MediaStream | null;\n\t\tif (!stream) return;\n\n\t\t// Create a canvas to normalize orientation\n\t\tconst videoTrack = stream.getVideoTracks()[0];\n\t\tconst settings = videoTrack.getSettings();\n\t\t// Fallback to landscape defaults (1280×720); portrait swap below handles orientation\n\t\tconst { width = 1280, height = 720 } = settings;\n\n\t\tconst canvas = document.createElement(\"canvas\");\n\t\tconst ctx = canvas.getContext(\"2d\");\n\n\t\t// Always force portrait (swap width/height if landscape)\n\t\tif (width > height) {\n\t\t\tcanvas.width = height;\n\t\t\tcanvas.height = width;\n\t\t} else {\n\t\t\tcanvas.width = width;\n\t\t\tcanvas.height = height;\n\t\t}\n\n\t\tconst videoEl = document.createElement(\"video\");\n\t\tvideoEl.srcObject = stream;\n\t\tvideoEl.muted = true;\n\t\tvideoEl.playsInline = true;\n\t\tvideoEl.play();\n\t\trecordingVideoElRef.current = videoEl;\n\n\t\t// Draw video frames onto canvas\n\t\tconst drawFrame = () => {\n\t\t\t// Rotate if camera gives landscape frames\n\t\t\tif (width > height) {\n\t\t\t\tctx?.save();\n\t\t\t\tctx?.translate(canvas.width, 0);\n\t\t\t\tctx?.rotate(Math.PI / 2);\n\t\t\t\tctx?.drawImage(videoEl, 0, 0, height, width);\n\t\t\t\tctx?.restore();\n\t\t\t} else {\n\t\t\t\tctx?.drawImage(videoEl, 0, 0, width, height);\n\t\t\t}\n\t\t\tdrawFrameRafRef.current = requestAnimationFrame(drawFrame);\n\t\t};\n\t\tdrawFrameRafRef.current = requestAnimationFrame(drawFrame);\n\n\t\t// Match recording FPS to actual camera FPS to avoid duplicate frames in the encoded video\n\t\tconst recordingFps = settings.frameRate ?? 10;\n\t\trecordedFpsRef.current = recordingFps; // Store FPS at recording time\n\t\tconst canvasStream = canvas.captureStream(recordingFps);\n\t\tconst mediaRecorderOptions: MediaRecorderOptions = {\n\t\t\tvideoBitsPerSecond: 1_500_000,\n\t\t};\n\t\tif (supportedTypes.length > 0) {\n\t\t\tmediaRecorderOptions.mimeType = supportedTypes[0];\n\t\t}\n\t\tlet mediaRecorder;\n\n\t\ttry {\n\t\t\tmediaRecorder = new MediaRecorder(canvasStream, mediaRecorderOptions);\n\t\t} catch (e) {\n\t\t\tconsole.error(\"MediaRecorder init failed:\", e);\n\t\t\tsetErrorType(FaceScanErrorType.Scan);\n\t\t\tsetHasError(true);\n\t\t\tsetShowLoader(false);\n\t\t\treturn;\n\t\t}\n\n\t\trecordedBlobsRef.current = [];\n\t\tmediaRecorder.ondataavailable = (event) => {\n\t\t\tif (event?.data && event.data.size > 0 && recordedBlobsRef.current) {\n\t\t\t\trecordedBlobsRef.current.push(event.data);\n\t\t\t}\n\t\t};\n\t\tmediaRecorder.onstop = () => {\n\t\t\tconsole.log(\"Recording stopped, total blobs:\", recordedBlobsRef?.current?.length);\n\t\t\tuploadFinalVideo();\n\t\t};\n\n\t\tmediaRecorder.start(100); // 100ms chunks\n\t\tmediaRecorderRef.current = mediaRecorder;\n\t\tconsole.log(\"Recording started successfully (portrait normalized)\");\n\t}, [supportedTypes, uploadFinalVideo]);\n\n\tconst { faceScanDetector, scanStage, setScanStage, resetScan, isModelLoaded, startScanSequence } = useFaceScan({\n\t\tfaceScanId,\n\t\tshopDomain,\n\t\tisError,\n\t\tisSuccess,\n\t\tonScanComplete: () => {\n\t\t\tstopRecording();\n\t\t},\n\t\tonModelReady: () => {\n\t\t\tsetModelReady(true);\n\t\t},\n\t\tonStartRecording: () => {\n\t\t\tconsole.log(\"Stage 0 completed - starting recording for stages 1-3\");\n\t\t\tstartRecording();\n\t\t},\n\t});\n\n\tconst startScan = useCallback(() => {\n\t\tif (!isModelLoaded) return;\n\t\tonScanStart?.();\n\t\tsetScanStage(0);\n\t\tspeechService.playAudio(`${voiceOverAssetsPath}face-scan-vos/Face-forward.mp3`);\n\t\tsetTimeout(() => {\n\t\t\tsetIsScanning(true);\n\t\t\tsetTimeout(() => {\n\t\t\t\tstartScanSequence();\n\t\t\t}, 200);\n\t\t}, 200);\n\t}, [setScanStage, setIsScanning, startScanSequence, isModelLoaded, onScanStart]);\n\n\tconst resetScanState = useCallback(() => {\n\t\tsetIsScanning(false);\n\t\tsetShowGuideCard(true);\n\t\tstopRecording();\n\t\tresetScan();\n\t\tonRetry?.();\n\t\trecordedBlobsRef.current = [];\n\t\trecordedFpsRef.current = null;\n\t\tif (mediaRecorderRef.current) {\n\t\t\tmediaRecorderRef.current = null;\n\t\t}\n\t\tsetHasError(false);\n\t\tsetErrorType(null);\n\t\tsetScanStage(-1);\n\t\tsetFaceScanId(generateUuid());\n\t\tif (webcamRef.current && streamRef.current) {\n\t\t\twebcamRef.current.srcObject = streamRef.current;\n\t\t\tconsole.log(\"Camera stream restored after reset\");\n\t\t}\n\t}, [setScanStage, setIsScanning, setShowGuideCard, stopRecording, resetScan, setFaceScanId]);\n\n\tconst onError = (data: any) => {\n\t\tconsole.log(data, \"ws error\");\n\t\tonScanError?.(data);\n\t\tsetErrorType(FaceScanErrorType.Scan);\n\t\tsetHasError(true);\n\t\tsetShowLoader(false);\n\t\tsetShowGuideCard(false);\n\t\thandleScanTimeCapture({\n\t\t\teventName: `${shopDomain}/faceScan`,\n\t\t\tfaceScanId,\n\t\t\tstatus: \"failed\",\n\t\t\temail,\n\t\t\tdata: JSON.stringify(data),\n\t\t});\n\t};\n\n\tconst onSuccess = (data: any) => {\n\t\tconsole.log(data, \"ws success\");\n\t\tif (data && data?.resultType === \"intermediate\") {\n\t\t\thandleScanTimeCapture({\n\t\t\t\teventName: `${shopDomain}/faceScan_success/intermediate`,\n\t\t\t\tfaceScanId,\n\t\t\t\tstatus: \"success\",\n\t\t\t\temail,\n\t\t\t\tdata: JSON.stringify(data),\n\t\t\t});\n\t\t}\n\t\tonScanSuccess?.(data);\n\t};\n\n\tuseEffect(() => {\n\t\tif (isError || isSuccess) return;\n\t\tif (navigator.mediaDevices.getUserMedia) {\n\t\t\tnavigator.mediaDevices\n\t\t\t\t.getUserMedia({ video: currentVideoConstraints })\n\t\t\t\t.then((stream) => {\n\t\t\t\t\tstreamRef.current = stream;\n\t\t\t\t\tif (webcamRef.current) {\n\t\t\t\t\t\twebcamRef.current.srcObject = stream;\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t\t\t.catch((err) => {\n\t\t\t\t\tconsole.error(\"Error accessing webcam:\", err);\n\t\t\t\t\tif (currentVideoConstraints === videoConstraintsExact) {\n\t\t\t\t\t\tsetCurrentVideoConstraints(videoConstraintsFallback);\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\t\t\t\t\tconst isCameraAccessError = [\"NotAllowedError\", \"PermissionDeniedError\", \"NotFoundError\", \"DevicesNotFoundError\", \"NotReadableError\", \"TrackStartError\"].includes(err?.name);\n\t\t\t\t\tsetErrorType(isCameraAccessError ? FaceScanErrorType.Camera : FaceScanErrorType.Scan);\n\t\t\t\t\tsetHasError(true);\n\t\t\t\t});\n\t\t}\n\t\treturn () => {\n\t\t\tif (streamRef.current) {\n\t\t\t\tstreamRef.current.getTracks().forEach((track) => track.stop());\n\t\t\t}\n\t\t};\n\t}, [isError, isSuccess, currentVideoConstraints]);\n\n\t// Face detection interval - only run when scanning is active\n\tuseEffect(() => {\n\t\tif (!modelReady || !isScanning) return;\n\t\tconst intervalId = setInterval(() => {\n\t\t\tfaceScanDetector(webcamRef, canvasRef);\n\t\t}, 500);\n\n\t\t// eslint-disable-next-line consistent-return\n\t\treturn () => clearInterval(intervalId);\n\t}, [faceScanDetector, modelReady, isScanning]);\n\n\tconst getButtonText = () => {\n\t\tif (isScanning) return LanguageKeys.reset;\n\t\tif (!isModelLoaded) return LanguageKeys.loading;\n\t\treturn LanguageKeys.start;\n\t};\n\n\tconsole.log(\"Model ready:\", modelReady, \"Has error:\", hasError, \"Is scanning:\", isScanning, \"showLoader\", showLoader);\n\tuseEffect(() => {\n\t\tconst supported = getSupportedMediaRecorderTypes(preferredVideoTypes, videoTypes);\n\t\tsetSupportedTypes(supported);\n\t}, [preferredVideoTypes]);\n\tuseEffect(() => {\n\t\tsetScanStage(-1);\n\t}, []);\n\n\treturn (\n\t\t<div data-face-scan-stage={scanStage}>\n\t\t<LanguageContextProvider>\n\t\t\t<ConfigProvider config={config}>\n\t\t\t\t<ParamsContext.Provider\n\t\t\t\t\tvalue={{\n\t\t\t\t\t\tisError,\n\t\t\t\t\t\tisSuccess,\n\t\t\t\t\t\tshowLoader,\n\t\t\t\t\t\thasError,\n\t\t\t\t\t\terrorType,\n\t\t\t\t\t\tresetScanState,\n\t\t\t\t\t\tshowGuideCard,\n\t\t\t\t\t\tscanStage,\n\t\t\t\t\t\tvideoLoading,\n\t\t\t\t\t\tsetVideoLoading,\n\t\t\t\t\t\tvideoError,\n\t\t\t\t\t\tsetVideoError,\n\t\t\t\t\t\tgender,\n\t\t\t\t\t\tgetButtonText,\n\t\t\t\t\t\tisScanning,\n\t\t\t\t\t\tstartScan,\n\t\t\t\t\t\tisModelLoaded,\n\t\t\t\t\t\tonRetry,\n\t\t\t\t\t}}\n\t\t\t\t>\n\t\t\t\t\t<FaceScanStep webcamRef={webcamRef} canvasRef={canvasRef} />\n\t\t\t\t</ParamsContext.Provider>\n\t\t\t</ConfigProvider>\n\t\t</LanguageContextProvider>\n\t\t</div>\n\t);\n};\n"],"names":["preloadedDetector","isPreloading","preloadDetector","async","window","Promise","r","setTimeout","tf","poseDetection","all","import","setBackend","ready","detector","createDetector","SupportedModels","BlazePose","runtime","modelType","dummyCanvas","document","createElement","width","videoConstraints","ideal","max","height","ctx","getContext","fillStyle","fillRect","estimatePoses","e","console","warn","err","error","useFaceScan","faceScanId","onValidPose","onScanComplete","onModelReady","onStartRecording","shopDomain","isError","isSuccess","detectorRef","useRef","scanStage","setScanStage","useState","consecutiveValid","setConsecutiveValid","isModelLoaded","setIsModelLoaded","isScanningActive","setIsScanningActive","isInferencingRef","inferenceCanvasRef","utteranceRef","scanStageRef","consecutiveValidRef","validityBufferRef","lastSpokenMessageRef","lastSpokenTimeRef","voiceEnabledRef","lastPoseTrackingTimeRef","isIOS","navigator","test","userAgent","platform","maxTouchPoints","directionMessages","useMemo","debouncedTrackPoseDetection","fn","delay","timer","args","clearTimeout","debounce","payload","posthog","capture","trackPoseDetection","useCallback","faceKeypoints","bodyKeypoints","stage","headValid","bodyInvalid","consecutiveValidCount","now","Date","current","nose","leftEye","rightEye","leftMouth","rightMouth","faceKeypointScores","map","kp","filter","score","avgFaceScore","length","reduce","a","b","bodyKeypointScores","avgBodyScore","eyeDistance","Math","abs","noseToRight","noseToLeft","expectedPosition","timestamp","toISOString","data","JSON","stringify","parseFloat","toFixed","faceKeypointsDetected","bodyKeypointsDetected","noseToRightRatio","noseToLeftRatio","noseScore","leftEyeScore","rightEyeScore","leftMouthScore","rightMouthScore","startScanSequence","stageName","useEffect","log","initializeModels","disposeModelAndTf","faceScanDetector","webcamRef","canvasRef","readyState","vw","videoWidth","vh","videoHeight","ictx","drawImage","poses","keypoints","forEach","landmark","idx","point","x","y","push","clearRect","progress","min","radius","beginPath","strokeStyle","lineWidth","arc","PI","stroke","eyesToMouthVerticalDistance","faceTiltedness","faceIsTilted","faceIsVertical","frontalFace","fullLeft","fullRight","isHeadInPosition","slice","p","shift","Boolean","nextValid","nextStage","speechService","playAudio","voiceOverAssetsPath","completedStage","totalStages","nextSound","resetScan","speechSynthesis","cancel","enableVoiceCommands","silentUtterance","SpeechSynthesisUtterance","volume","speak","FaceScanErrorScreen","onRetry","loading","translate","useContext","LanguageContext","config","useConfig","_jsx","className","style","background","base","backgroundColor","children","_jsxs","Header","noTitle","fontFamily","heading","headingFontFamily","fontSize","headingFontSize","color","headingColor","fontWeight","headingFontWeight","LanguageKeys","scanFailed","subheading","subheadingFontFamily","subheadingFontSize","subheadingColor","subheadingFontWeight","clickToResetScan","SpecificButton","disabled","buttonText","postfixIcon","ArrowRight","buttonFunc","FaceScanGuide","videoLoading","setVideoLoading","videoError","setVideoError","gender","btnConfig","faceVisible","loadingVideo","videoLoadFailed","src","GenderType","Male","maleGlassesOffVideo","glassesOffVideo","autoPlay","loop","controls","muted","playsInline","onCanPlay","onLoadStart","onError","isDisabled","label","icon","onClick","FACE_SCAN_HEADSHOT","male","forward","female","type","left","right","smile","FaceScanStep","showLoader","hasError","errorType","resetScanState","showGuideCard","getButtonText","isScanning","startScan","ParamsContext","resolvedConfig","LoadingScreen","url","loader","loaderType","_Fragment","id","crossOrigin","preload","position","zIndex","undefined","ref","transform","opacity","FaceScanErrorType","Camera","Modal","testId","Drawer","open","anchor","onClose","event","reason","hideBackdrop","FaceScan","props","onScanSuccess","onScanError","onScanStart","onCaptureComplete","onUploadStart","onUploadEnd","onMeasurementSocketStart","onMeasurementSocketClose","userDetails","email","deviceFocalLength","bodyScanId","callbackUrl","token","mediaRecorderRef","recordedBlobsRef","streamRef","recordedFpsRef","drawFrameRafRef","recordingVideoElRef","setShowLoader","modelReady","setModelReady","setIsScanning","setShowGuideCard","setHasError","setErrorType","setFaceScanId","generateUuid","swan","getSwanService","supportedTypes","setSupportedTypes","currentVideoConstraints","setCurrentVideoConstraints","videoConstraintsExact","preferredVideoTypes","getPreferredMediaRecorderTypes","recordingMimeType","getRecordingMimeType","recordingExtension","getRecordingExtension","usePosthogPageview","stopRecording","cancelAnimationFrame","pause","srcObject","state","stop","uploadFinalVideo","Scan","videoFile","File","videoFps","getVideoFPS","fileSize","size","fileSizeMB","estimatedDuration","round","videoData","video_size_mb","video_size_bytes","video_fps","blob_count","estimated_duration_seconds","metaData","face_scan_id","focal_length","customer_store_url","scan_type","body_scan_id","callback_url","handleScanTimeCapture","eventName","uploadStartTime","startTime","res","fileUpload","faceScanFileUploader","file","arrayMetaData","objectKey","contentType","uploadEndTime","completionTime","uploadDuration","measurement","handlFaceScaneSocket","onPreopen","handleWebSocketCapture","faceScanID","connection","onOpen","onSuccess","startRecording","stream","settings","getVideoTracks","getSettings","canvas","videoEl","play","drawFrame","save","rotate","restore","requestAnimationFrame","recordingFps","frameRate","canvasStream","captureStream","mediaRecorderOptions","videoBitsPerSecond","mediaRecorder","mimeType","MediaRecorder","ondataavailable","onstop","start","status","resultType","mediaDevices","getUserMedia","video","then","catch","videoConstraintsFallback","isCameraAccessError","includes","name","getTracks","track","intervalId","setInterval","clearInterval","supported","getSupportedMediaRecorderTypes","videoTypes","LanguageContextProvider","ConfigProvider","Provider","value","reset"],"mappings":"ogBA6BA,IAAIA,EAAyC,KACzCC,GAAe,EAGZ,MAAMC,EAAkBC,UAE7B,GAAsB,oBAAXC,OAAwB,OAAO,KAC1C,GAAIJ,EAAmB,OAAOA,EAC9B,GAAIC,EAEF,KAAOA,GAEL,SADM,IAAII,QAAQC,GAAKC,WAAWD,EAAG,MACjCN,EAAmB,OAAOA,EAIlCC,GAAe,EAEf,IAEE,MAAOO,EAAIC,SAAuBJ,QAAQK,IAAI,CAC5CC,OAAO,oBACPA,OAAO,6CAIHH,EAAGI,WAAW,eACdJ,EAAGK,QAET,MAAMC,QAAiBL,EAAcM,eACnCN,EAAcO,gBAAgBC,UAC9B,CACEC,QAAS,OACTC,UAAW,SAOTC,EAAcC,SAASC,cAAc,UAC3CF,EAAYG,MAA0C,iBAA3BC,EAAiBD,MAAqBC,EAAiBD,MAAQC,EAAiBD,MAAME,OAASD,EAAiBD,MAAMG,KAAO,KACxJN,EAAYO,OAA4C,iBAA5BH,EAAiBG,OAAsBH,EAAiBG,OAASH,EAAiBG,OAAOF,OAASD,EAAiBG,OAAOD,KAAO,IAC7J,MAAME,EAAMR,EAAYS,WAAW,MACnC,GAAID,EAAK,CACPA,EAAIE,UAAY,QAChBF,EAAIG,SAAS,EAAG,EAAGX,EAAYG,MAAOH,EAAYO,QAClD,UACQb,EAASkB,cAAcZ,EAC9B,CAAC,MAAOa,GACPC,QAAQC,KAAK,mCAAoCF,EAClD,CACF,CAGD,OADAjC,EAAoBc,EACbA,CACR,CAAC,MAAOsB,GAEP,OADAF,QAAQG,MAAM,8BAA+BD,GACtC,IACR,CAAS,QACRnC,GAAe,CAChB,GAKH,SAASqC,GAAYC,WACnBA,EAAUC,YACVA,EAAWC,eACXA,EAAcC,aACdA,EAAYC,iBACZA,EAAgBC,WAChBA,EAAUC,QACVA,EAAOC,UACPA,IAWA,MAAMC,EAAcC,EAA4B,OACzCC,EAAWC,GAAgBC,EAAS,IACpCC,EAAkBC,GAAuBF,EAAS,IAClDG,EAAeC,GAAoBJ,GAAS,IAC5CK,EAAkBC,GAAuBN,GAAS,GACnDO,EAAmBV,GAAO,GAM1BW,EAAqBX,EAAiC,MAEtDY,EAAeZ,EAAwC,MACvDa,EAAeb,EAAOC,GACtBa,EAAsBd,EAAOI,GAC7BW,EAAoBf,EAAkB,IACtCgB,EAAuBhB,EAAO,IAC9BiB,EAAoBjB,EAAO,GAC3BkB,EAAkBlB,GAAO,GAMzBmB,EAA0BnB,EAAO,GAGjCoB,EAA6B,oBAAdC,YACnB,mBAAmBC,KAAKD,UAAUE,YACV,aAAvBF,UAAUG,UAA2BH,UAAUI,eAAiB,GAG7DC,EAAoBC,EACxB,IAAM,CACJ,aACA,uBACA,wBACA,sBAEF,IAGIC,EAA8BD,EAClC,IA1IJ,SAAkBE,EAA4BC,GAC5C,IAAIC,EACJ,MAAO,IAAIC,KACLD,GAAOE,aAAaF,GACxBA,EAAQxE,WAAW,IAAMsE,KAAMG,GAAOF,GAE1C,CAqIMI,CAAUC,IACRC,EAAQC,QAAQ,GAAGzC,6BAAuCuC,IACzD,KACL,CAACvC,IAGG0C,EAAqBC,EACzB,CACEC,EACAC,EACAC,EACAC,EACAC,EACAC,KAEA,MAAMC,EAAMC,KAAKD,MACjB,KAAIA,EAAM3B,EAAwB6B,QArCP,KAqC3B,CAIA7B,EAAwB6B,QAAUF,EAElC,IACE,MAAMG,EAAOT,EAAc,IAAM,CAAC,EAAG,EAAG,GAClCU,EAAUV,EAAc,IAAM,CAAC,EAAG,EAAG,GACrCW,EAAWX,EAAc,IAAM,CAAC,EAAG,EAAG,GACtCY,EAAYZ,EAAc,IAAM,CAAC,EAAG,EAAG,GACvCa,EAAab,EAAc,KAAO,CAAC,EAAG,EAAG,GAEzCc,EAAqBd,EACxBe,IAAKC,GAAcA,EAAG,IACtBC,OAAQC,GAAkBA,EAAQ,GAC/BC,EACJL,EAAmBM,OAAS,EACxBN,EAAmBO,OAAO,CAACC,EAAGC,IAAMD,EAAIC,EAAG,GAC3CT,EAAmBM,OACnB,EAEAI,EAAqBvB,EACxBc,IAAKC,GAAcA,EAAG,IACtBC,OAAQC,GAAkBA,EAAQ,GAC/BO,EACJD,EAAmBJ,OAAS,EACxBI,EAAmBH,OAAO,CAACC,EAAWC,IAAcD,EAAIC,EAAG,GAC3DC,EAAmBJ,OACnB,EAEAM,EAAcC,KAAKC,IAAIlB,EAAQ,GAAKC,EAAS,IAC7CkB,EAAcpB,EAAK,GAAKE,EAAS,GACjCmB,EAAapB,EAAQ,GAAKD,EAAK,GAErC,IAAIsB,EAAmB,UACT,IAAV7B,EAAa6B,EAAmB,OACjB,IAAV7B,EAAa6B,EAAmB,QACtB,IAAV7B,IAAa6B,EAAmB,iBAEzC,MAAMpC,EAAU,CACd5C,aACAmD,QACA8B,WAAW,IAAIzB,MAAO0B,cACtBC,KAAMC,KAAKC,UAAU,CACnBjC,YACAC,cACAxC,iBAAkByC,EAClBc,aAAckB,WAAWlB,EAAamB,QAAQ,IAC9Cb,aAAcY,WAAWZ,EAAaa,QAAQ,IAC9CC,sBAAuBzB,EAAmBM,OAC1CoB,sBAAuBhB,EAAmBJ,OAC1CM,YAAaW,WAAWX,EAAYY,QAAQ,IAC5CG,iBAAkBJ,YAAYR,EAAcH,GAAaY,QAAQ,IACjEI,gBAAiBL,YAAYP,EAAaJ,GAAaY,QAAQ,IAC/DK,UAAWN,WAAW5B,EAAK,GAAG6B,QAAQ,IACtCM,aAAcP,WAAW3B,EAAQ,GAAG4B,QAAQ,IAC5CO,cAAeR,WAAW1B,EAAS,GAAG2B,QAAQ,IAC9CQ,eAAgBT,WAAWzB,EAAU,GAAG0B,QAAQ,IAChDS,gBAAiBV,WAAWxB,EAAW,GAAGyB,QAAQ,IAClDP,sBAGJ3C,EAA4BO,EAC7B,CAAC,MAAO9C,GACPH,QAAQC,KAAK,kCAAmCE,EACjD,CAhEA,GAkEH,CAACE,EAAYqC,IAYT4D,EAAoB,KACxBpD,EAAQC,QAAQ,GAAGzC,gCAA0C,CAC3DL,aACAmD,MAAO7B,EAAamC,QACpBwB,WAAW,IAAIzB,MAAO0B,cACtBgB,UAAW/D,EAAkBb,EAAamC,WAG5CzF,WAAW,KACTkD,GAAoB,IACnB,OAwSL,OAhBAiF,EAAU,KACR,IAAI7F,IAAWC,EAEf,MAzEuB3C,WAEvB,GAAsB,oBAAXC,OAEX,IACE8B,QAAQyG,IAAI,6BAEZ,MAAM7H,QAAiBZ,IAEnBY,IACFiC,EAAYiD,QAAUlF,EACtBoB,QAAQyG,IAAI,uCACZpF,GAAiB,GACjBb,MAEH,CAAC,MAAON,GACPF,QAAQG,MAAM,gCAAiCD,EAChD,GAuDDwG,GACO,KArDiBzI,WAGxB,IACEoD,GAAiB,EAUlB,CAAC,MAAOnB,GACPF,QAAQG,MAAM,6BAA8BD,EAC7C,CAAS,QAERc,EAAa,GACbG,EAAoB,GACpBQ,EAAamC,QAAU,EACvBlC,EAAoBkC,QAAU,EAC9BjC,EAAkBiC,QAAU,EAC7B,GA+BC6C,KAED,CAAChG,EAASC,IAEb4F,EAAU,KACR7E,EAAamC,QAAU/C,GACtB,CAACA,IAEJyF,EAAU,KACR5E,EAAoBkC,QAAU5C,GAC7B,CAACA,IAEG,CACL0F,iBA5OuB3I,MACvB4I,EACAC,KAEA,GACGjG,EAAYiD,SACZ+C,EAAU/C,SACV1C,GACAE,KAKCuF,EAAU/C,QAAQiD,WAAa,GAI/BvF,EAAiBsC,SAArB,CACAtC,EAAiBsC,SAAU,EAE3B,IACE,MAAMkD,EAAKH,EAAU/C,QAAQmD,YAAc,IACrCC,EAAKL,EAAU/C,QAAQqD,aAAe,KACvC1F,EAAmBqC,UACtBrC,EAAmBqC,QAAU3E,SAASC,cAAc,UACpDqC,EAAmBqC,QAAQzE,MAAQ2H,EACnCvF,EAAmBqC,QAAQrE,OAASyH,GAEtC,MAAME,EAAO3F,EAAmBqC,QAAQnE,WAAW,MAC/CyH,GAAMA,EAAKC,UAAUR,EAAU/C,QAAS,EAAG,EAAGkD,EAAIE,GACtD,MAAMI,QAAczG,EAAYiD,QAAQhE,cACtCsH,EAAO3F,EAAmBqC,QAAU+C,EAAU/C,SAEhD,GAAIwD,EAAM5C,OAAS,EAAG,CACpB,MAAMpB,EAAyB,GACzBC,EAAyB,GAS/B,GARA+D,EAAM,GAAGC,UAAUC,QAAQ,CAACC,EAAUC,KACpC,MAAMlD,EAAQiD,EAASjD,OAAS,EAC1BmD,EAAe,CAACF,EAASG,IAAM,EAAGH,EAASI,IAAM,EAAGrD,GACtDkD,GAAO,GAAIpE,EAAcwE,KAAKH,GAC7BpE,EAAcuE,KAAKH,KAItBb,GAAWhD,QAAS,CACtB,MAAMpE,EAAMoH,EAAUhD,QAAQnE,WAAW,OACnCN,MAAEA,EAAKI,OAAEA,GAAWqH,EAAUhD,QACpC,GAAIpE,EAAK,CACPA,EAAIqI,UAAU,EAAG,EAAG1I,EAAOI,GAE3B,MAAMuI,EAAW/C,KAAKgD,IACpBrG,EAAoBkC,QAjPC,EAkPrB,GAEIoE,EAAkB,IAATzI,EACfC,EAAIyI,YACJzI,EAAI0I,YAAc,UAClB1I,EAAI2I,UAAY,EAChB3I,EAAI4I,IACFjJ,EAAQ,EACRI,EAAS,EACTyI,EAAS,IACRjD,KAAKsD,GAAK,GACVtD,KAAKsD,GAAK,EAAe,EAAXP,EAAe/C,KAAKsD,IAErC7I,EAAI8I,QACL,CACF,CAED,MAAM/E,EA/Ha,EAACD,EAAe+D,KASvC,IACGA,EARU,KAQUA,EAPP,KAO8BA,EAN7B,IAOfA,EATW,GASK,GAHD,IAIfA,EATc,GASK,GAJJ,IAKfA,EATe,GASK,GALL,GAOf,OAAO,EAGT,MAAMkB,EACJ,IAAOlB,EAbU,IAaY,GAAKA,EAdlB,GAcuC,IACvD,IAAOA,EAhBQ,GAgBY,GAAKA,EAjBlB,GAiBqC,IAE/CmB,GACH,IAAOnB,EAjBS,IAiBa,GAAKA,EAlBnB,GAkBwC,IACtDA,EAtBS,GAsBO,IAClBkB,EAEIE,EAAeD,EAAiB,IAAOA,EAAiB,GAExDE,EACJ3D,KAAKC,IAAIqC,EA3BK,GA2Bc,GAAKA,EA1BlB,GA0BsC,IACnD,GAAMkB,GACRxD,KAAKC,IAAIqC,EA3BO,GA2Bc,GAAKA,EA1BlB,IA0BwC,IACvD,GAAMkB,EAEJzD,EAAcuC,EAhCJ,GAgCuB,GAAKA,EA/B3B,GA+B+C,GAC1DpC,EAAcoC,EAlCP,GAkCuB,GAAKA,EAhCxB,GAgC4C,GAGvDsB,EACJ1D,EAAc,GAAMH,GAAeG,EAAc,GAAMH,EACnD8D,EAAW3D,EAAc,IAAOH,EAChC+D,EALaxB,EAlCH,GAkCsB,GAAKA,EAnC9B,GAmC8C,GAK5B,IAAOvC,EAEtC,OAAQxB,GACN,KAAK,EAML,KAAK,EACH,OAAQmF,GAAgBC,GAAkBC,EAL5C,KAAK,EACH,OAAQF,GAAgBC,GAAkBE,EAC5C,KAAK,EACH,OAAQH,GAAgBC,GAAkBG,EAG5C,QACE,OAAO,IAyEWC,CAAiBrH,EAAamC,QAASR,GACnDI,EACJH,EAAc0F,MAAM,GAAG1E,OAAQ2E,GAAMA,EAAE,GAAK,IAAKxE,QAAU,EAE7D7C,EAAkBiC,QAAQgE,KAAKrE,GAAaC,GACxC7B,EAAkBiC,QAAQY,OAzQL,IA0QvB7C,EAAkBiC,QAAQqF,QAI5B,GADmBtH,EAAkBiC,QAAQS,OAAO6E,SAAS1E,QAC3C,EAAG,CACnB,MAAM2E,EAAYzH,EAAoBkC,QAAU,EAChD3C,EAAoBkI,EACrB,MACClI,EAAoB8D,KAAKzF,IAAI,EAAGoC,EAAoBkC,QAAU,IAGhE,GAAIlC,EAAoBkC,SApRG,EAoRkC,CAC3DxD,MAEA,MAAMgJ,EAAY3H,EAAamC,QAAU,EACzC3C,EAAoB,GACpBU,EAAkBiC,QAAU,GAC5B9C,EAAasI,GACb/H,GAAoB,GAEpB,WAmBE,SAlBMgI,EAAcC,UAClB,GAAGC,mCAELvG,EAAQC,QACN,GAAGzC,wCACH,CACEL,aACAqJ,eAAgB/H,EAAamC,QAC7BwF,YACAhE,WAAW,IAAIzB,MAAO0B,cACtBoE,YAvSO,EAwSPpD,UAAW/D,EAAkBb,EAAamC,WAI5B,IAAdwF,GAAmB7I,GACrBA,IAEE6I,GA/SO,EAgTT/I,UACK,CACL,IAAIqJ,EAAY,KAChB,OAAQN,GACN,KAAK,EAAGM,EAAY,WAAY,MAChC,KAAK,EAAGA,EAAY,YAAa,MACjC,KAAK,EAAGA,EAAY,YAElBA,SACIL,EAAcC,UAClB,GAAGC,kBAAoCG,KAG3CtD,GACD,CACF,EAnCD,EAoCD,CAEDlD,EACEE,EACAC,EACA5B,EAAamC,QACbL,EACAC,EACA9B,EAAoBkC,QAEvB,CACF,CAAC,MAAO5D,GACPF,QAAQG,MAAM,wBAAyBD,EACxC,CAAS,QACRsB,EAAiBsC,SAAU,CAC5B,CAjI6B,GA4N9B/C,YACAC,eACAE,mBACAE,gBACAyI,UA9CgB,KAChB3G,EAAQC,QAAQ,GAAGzC,oBAA8B,CAC/CL,aACAmD,MAAO7B,EAAamC,QACpBwB,WAAW,IAAIzB,MAAO0B,cACtBgB,UAAW/D,EAAkBb,EAAamC,SAC1C5C,iBAAkBU,EAAoBkC,UAGxC9C,EAAa,GACbG,EAAoB,GACpBI,GAAoB,GACpBI,EAAamC,QAAU,EACvBlC,EAAoBkC,QAAU,EAC9BjC,EAAkBiC,QAAU,GAC5BhC,EAAqBgC,QAAU,GAC/B/B,EAAkB+B,QAAU,EAC5B9B,EAAgB8B,SAAU,EACtBpC,EAAaoC,UACfgG,gBAAgBC,SAChBrI,EAAaoC,QAAU,OA2BzBkG,oBAlU0B,KAE1B,GADAhI,EAAgB8B,SAAU,EACJ,oBAAX5F,QAA0BgE,GAAS,oBAAqBhE,OAAQ,CACzE,MAAM+L,EAAkB,IAAIC,yBAAyB,IACrDD,EAAgBE,OAAS,EACzBL,gBAAgBM,MAAMH,EACvB,GA6TD3D,oBACAhF,mBAEJ,CCpjBA,SAAS+I,GAAoBC,QAAEA,EAAOC,QAAEA,IACvC,MAAMC,UAAEA,GAAcC,EAAWC,IAAoB,CAAA,EAC/CC,EAASC,IAEf,OACCC,uBAAiB,yBAAyBC,UAAU,iGAAiGC,MAAO,CAAEC,WAAYL,GAAQI,OAAOE,MAAMC,iBAAiBC,SAC/MC,SAAKN,UAAU,0DAAyDK,SAAA,CACvEN,EAACQ,EAAM,CAACC,aACRF,EAAA,MAAA,CAAKN,UAAU,8CACdD,EAAA,KAAA,CACCC,UAAU,uCACVC,MAAO,CACNQ,WAAYZ,GAAQI,OAAOS,SAASC,mBAAqB,wBACzDC,SAAUf,GAAQI,OAAOS,SAASG,iBAAmB,OACrDC,MAAOjB,GAAQI,OAAOS,SAASK,cAAgB,OAC/CC,WAAYnB,GAAQI,OAAOS,SAASO,mBAAqB,UACzDZ,SAEAX,IAAYwB,EAAaC,cAE3BpB,OACCC,UAAU,cACVC,MAAO,CACNQ,WAAYZ,GAAQI,OAAOmB,YAAYC,sBAAwB,sBAC/DT,SAAUf,GAAQI,OAAOmB,YAAYE,oBAAsB,OAC3DR,MAAOjB,GAAQI,OAAOmB,YAAYG,iBAAmB,UACrDP,WAAYnB,GAAQI,OAAOmB,YAAYI,sBAAwB,UAC/DnB,SAEAX,IAAYwB,EAAaO,oBAE3B1B,EAAC2B,iBACY,sBACZC,SAAUlC,EACVO,UAAU,iEACV4B,WAAYlC,IAAYwB,EAAanC,WACrC8C,YAAa9B,EAAC+B,EAAU,CAAA,GACxBC,WAAY,IAAMvC,eAMxB,CCvBA,SAASwC,GAActJ,MAAEA,EAAKuJ,aAAEA,EAAYC,gBAAEA,EAAeC,WAAEA,EAAUC,cAAEA,EAAaC,OAAEA,EAAMC,UAAEA,IACjG,MAAM5C,UAAEA,GAAcC,EAAWC,IAAoB,CAAA,EAC/CC,EAASC,IAEf,OAAc,IAAVpH,EAEF4H,EAAA,MAAA,CAAA,cAAiB,kBAAkBN,UAAU,qDAAqDC,MAAO,CAAEC,WAAYL,GAAQI,OAAOE,MAAMC,iBAAiBC,SAAA,CAC5JC,EAAA,MAAA,CAAKN,UAAU,SAAQK,SAAA,CACtBN,EAACQ,GAAOC,SAAO,IACfT,EAAA,KAAA,CACCE,MAAO,CACNQ,WAAYZ,GAAQI,OAAOS,SAASC,mBAAqB,wBACzDC,SAAUf,GAAQI,OAAOS,SAASG,iBAAmB,OACrDC,MAAOjB,GAAQI,OAAOS,SAASK,cAAgB,OAC/CC,WAAYnB,GAAQI,OAAOS,SAASO,mBAAqB,UACzDZ,SAEAX,IAAYwB,EAAaqB,eAE1BN,GACAlC,EAAA,MAAA,CACCC,UAAU,wCACVC,MAAO,CACNQ,WAAYZ,GAAQI,OAAOmB,YAAYC,sBAAwB,sBAC/DT,SAAUf,GAAQI,OAAOmB,YAAYE,oBAAsB,OAC3DR,MAAOjB,GAAQI,OAAOmB,YAAYG,iBAAmB,UACrDP,WAAYnB,GAAQI,OAAOmB,YAAYI,sBAAwB,UAC/DnB,SAEAX,IAAYwB,EAAasB,gBAG3BL,GACApC,EAAA,MAAA,CACCC,UAAU,oBACVC,MAAO,CACNQ,WAAYZ,GAAQI,OAAOmB,YAAYC,sBAAwB,sBAC/DT,SAAUf,GAAQI,OAAOmB,YAAYE,oBAAsB,OAC3DR,MAAO,UACPE,WAAYnB,GAAQI,OAAOmB,YAAYI,sBAAwB,UAC/DnB,SAEAX,IAAYwB,EAAauB,oBAG1BN,GACDpC,EAAA,MAAA,CAAKC,UAAW,qBAAoBK,SACnCN,EAAA,QAAA,CACC2C,IAAKL,IAAWM,EAAWC,KAAOC,EAAsBC,EACxDC,UAAQ,EACRC,MAAI,EACJC,UAAU,EACVC,OAAK,EACLC,aAAW,EACXnD,UAAU,2CACVoD,UAAW,IAAMlB,GAAgB,GACjCmB,YAAa,IAAMnB,GAAgB,GACnCoB,QAAS,IAAMlB,GAAc,GAAK,aACvB,mFAKfrC,EAAA,MAAA,CAAKC,UAAU,uCAAsCK,SACpDN,EAAC2B,EAAc,CAAA,cACF,sBACZC,SAAUW,EAAUiB,WACpBvD,UAAU,kCACV4B,WAAYlC,IAAY4C,EAAUkB,OAClC3B,YAAaS,GAAWmB,KACxB1B,WAAYO,EAAUoB,eAQ1B3D,EAAA,MAAA,CAAKC,UAAU,qDAAqDC,MAAO,CAAEC,WAAYL,GAAQI,OAAOE,MAAMC,iBAAiBC,SAC9HC,SAAKN,UAAU,SAAQK,SAAA,CACtBN,EAACQ,EAAM,CAACC,SAAO,IACfT,EAAA,KAAA,CACCC,UAAU,cACVC,MAAO,CACNQ,WAAYZ,GAAQI,OAAOS,SAASC,mBAAqB,wBACzDC,SAAUf,GAAQI,OAAOS,SAASG,iBAAmB,OACrDC,MAAOjB,GAAQI,OAAOS,SAASK,cAAgB,OAC/CC,WAAYnB,GAAQI,OAAOS,SAASO,mBAAqB,UACzDZ,SAEAX,IAAYhI,EAAkBgB,MAEhC4H,EAAA,MAAA,CAAKN,UAAU,+DAA8DK,SAAA,CAC5EN,EAAA,MAAA,CAAKC,UAAW,8BAAwC,IAAVtH,EAAc,cAAgB,sBAAqB2H,SAChGN,EAAA,QAAA,CAAOC,UAAU,2CAA2CkD,OAAK,EAACF,MAAI,EAACD,UAAQ,EAACI,aAAW,EAAA9C,SAC1FN,YAAQ2C,IAAKL,IAAWM,EAAWC,KAAOe,EAAmBC,KAAKC,QAAUF,EAAmBG,OAAOD,QAASE,KAAK,kBAGtHhE,EAAA,MAAA,CAAKC,UAAW,8BAAwC,IAAVtH,EAAc,cAAgB,sBAAqB2H,SAChGN,EAAA,QAAA,CAAOC,UAAU,2CAA2CkD,OAAK,EAACF,MAAI,EAACD,UAAQ,EAACI,aAAW,EAAA9C,SAC1FN,EAAA,SAAA,CAAQ2C,IAAKL,IAAWM,EAAWC,KAAOe,EAAmBC,KAAKI,KAAOL,EAAmBG,OAAOE,KAAMD,KAAK,kBAGhHhE,EAAA,MAAA,CAAKC,UAAW,8BAAwC,IAAVtH,EAAc,cAAgB,sBAAqB2H,SAChGN,EAAA,QAAA,CAAOC,UAAU,2CAA2CkD,OAAK,EAACF,MAAI,EAACD,UAAQ,EAACI,aAAW,EAAA9C,SAC1FN,EAAA,SAAA,CAAQ2C,IAAKL,IAAWM,EAAWC,KAAOe,EAAmBC,KAAKK,MAAQN,EAAmBG,OAAOG,MAAOF,KAAK,kBAGlHhE,EAAA,MAAA,CAAKC,UAAW,8BAAwC,IAAVtH,EAAc,cAAgB,sBAAqB2H,SAChGN,EAAA,QAAA,CAAOC,UAAU,2CAA2CkD,OAAK,EAACF,MAAI,EAACD,UAAQ,EAACI,aAAW,EAAA9C,SAC1FN,YAAQ2C,IAAKL,IAAWM,EAAWC,KAAOe,EAAmBC,KAAKM,MAAQP,EAAmBG,OAAOI,MAAOH,KAAK,yBAOvH,CCjIA,MAAMI,EAA4C,EAAGpI,YAAWC,gBAC/D,MAAMnG,QACLA,EAAOC,UACPA,EAASsO,WACTA,EAAUC,SACVA,EAAQC,UACRA,EAASC,eACTA,EAAcC,cACdA,EAAavO,UACbA,EAASgM,aACTA,EAAYC,gBACZA,EAAeC,WACfA,EAAUC,cACVA,EAAaC,OACbA,EAAMoC,cACNA,EAAaC,WACbA,EAAUC,UACVA,EAASrO,cACTA,EAAakJ,QACbA,GACGG,EAAWiF,GACTC,EAAiB/E,IACvB,OAAIjK,EACIkK,EAACR,EAAmB,CAACC,QAASA,IAElC1J,EAEFiK,EAAA,MAAA,CAAKC,UAAU,4CAA4CC,MAAO,CAAEC,WAAY2E,GAAgB5E,OAAOE,MAAMC,0BAC5GL,EAAC+E,EAAa,CAACC,IAAKF,GAAgBG,OAAQC,WAAW,YAMzD3E,EAAA4E,EAAA,CAAA7E,SAAA,CACCN,EAAA,QAAA,CAAOoF,GAAG,eAAeC,YAAY,YAAYC,QAAQ,OAAOpF,MAAO,CAAEqF,SAAU,WAAYC,QAAQ,OAAU7C,SAAK8C,IAGrHpB,IAAeC,GACftE,EAAA,MAAA,CAAKC,UAAU,4CAA4CC,MAAO,CAAEC,WAAY2E,GAAgB5E,OAAOE,MAAMC,iBAAiBC,SAE7HN,EAAC+E,GAAcC,IAAKF,GAAgBG,OAAQC,WAAW,YAKzDlF,EAAA,MAAA,CAAKC,UAAU,gHAAgHC,MAAO,CAAEC,WAAY2E,GAAgB5E,OAAOE,MAAMC,iBAAiBC,SACjMN,SAAKC,UAAU,yCAAwCK,SACtDC,EAAA,MAAA,CAAKN,UAAU,gBAAeK,SAAA,CAC7BN,EAAA,QAAA,CACC0F,IAAK1J,EACLgH,UAAQ,EACRI,aAAW,EACXD,OAAK,EACL3O,MAAOC,EAAiBD,MAAME,MAC9BE,OAAQH,EAAiBG,OAAOF,MAChCuL,UAAU,oDACVC,MAAO,CAAEyF,UAAW,gBAErB3F,EAAA,SAAA,CAAQ0F,IAAKzJ,EAAWzH,MAAOC,EAAiBD,MAAME,MAAOE,OAAQH,EAAiBG,OAAOF,MAAOwL,MAAO,CAAEyF,UAAW,aAAcC,QAAS,eAMhJvB,GAAcC,GAAYC,IAAcsB,EAAkBC,QAC3D9F,EAAC+F,EAAK,CAACC,OAAO,gCAAgClG,OAAQgF,KAErDT,GAAcC,GAAYC,IAAcsB,EAAkBC,QAC3D9F,EAACR,EAAmB,CAACE,QAAS2E,EAAY5E,QAAS+E,IAInDC,IAAkBH,IAAaD,GAC/BrE,EAACiG,EAAM,CACNC,MAAI,EACJjG,UAAU,8BACVkG,OAAO,SACPC,QAAS,CAACC,EAAOC,OAKjBC,cAAY,EAAAjG,SAEZN,EAACiC,EAAa,CACbtJ,MAAOzC,EACPgM,aAAcA,EACdC,gBAAiBA,EACjBC,WAAYA,EACZC,cAAeA,EACfC,OAAQA,EACRC,UAAW,CACVkB,MAAOiB,IACPf,QAASgB,EAAaH,EAAiBI,EACvCpB,WAAYa,IAAe9N,EAC3BmN,KAAmB1D,EAAb2E,EAAc5C,EAAgBoD,EAAN,CAAA,YCvFxBqB,EAAqCC,IACjD,MAAMC,cAAEA,EAAaC,YAAEA,EAAWlH,QAAEA,EAAOmH,YAAEA,EAAWC,kBAAEA,EAAiBC,cAAEA,EAAaC,YAAEA,EAAWC,yBAAEA,EAAwBC,yBAAEA,EAAwBnH,OAAEA,EAAMhK,QAAEA,EAAOC,UAAEA,GAAc0Q,EACtLS,EAAgCT,EAAMS,aAAe,CAC1DC,MAAO,GACP7E,OAAQ,OACR8E,kBAAmB,EACnBvR,WAAY,GACZwR,WAAY,GACZC,YAAa,MAERC,EAAQd,EAAMc,OAAS,GACvBvL,EAAY/F,EAAgC,MAC5CgG,EAAYhG,EAAiC,MAC7CuR,EAAmBvR,EAA6B,MAChDwR,EAAmBxR,EAAsB,IACzCyR,EAAYzR,EAA2B,MACvC0R,EAAiB1R,EAAsB,MACvC2R,EAAkB3R,EAAe,GACjC4R,EAAsB5R,EAAgC,OACrDoO,EAAYyD,GAAiB1R,GAAS,IACtC2R,EAAYC,IAAiB5R,GAAS,IACtCuO,GAAYsD,IAAiB7R,GAAS,IACtCqO,GAAeyD,IAAoB9R,GAAS,IAC5C8L,GAAcC,IAAmB/L,GAAS,IAC1CgM,GAAYC,IAAiBjM,GAAS,IACtCkO,GAAU6D,IAAe/R,GAAS,IAClCmO,GAAW6D,IAAgBhS,EAAmC,OAC9DZ,GAAY6S,IAAiBjS,EAASkS,KACvCC,GAAO3Q,EAAQ,IAAM4Q,EAAejB,GAAQ,CAACA,KAC7CJ,MAAEA,GAAK7E,OAAEA,GAAM8E,kBAAEA,GAAiBvR,WAAEA,GAAUyR,YAAEA,GAAWD,WAAEA,IAAeH,GAC3EuB,GAAgBC,IAAqBtS,EAAmB,KACxDuS,GAAyBC,IAA8BxS,EAAgCyS,GACxFC,GAAsBlR,EAAQ,IAAMmR,IAAkC,IACtEC,GAAoBC,EAAqBR,IACzCS,GAAqBC,EAAsBH,IACjDI,IAEA,MAAMC,GAAgB7Q,EAAY,KACjCrD,QAAQyG,IAAI,yBACZ0N,qBAAqB1B,EAAgB3O,SACjC4O,EAAoB5O,UACvB4O,EAAoB5O,QAAQsQ,QAC5B1B,EAAoB5O,QAAQuQ,UAAY,KACxC3B,EAAoB5O,QAAU,MAE3BuO,EAAiBvO,SAA8C,cAAnCuO,EAAiBvO,QAAQwQ,OACxDjC,EAAiBvO,QAAQyQ,OAE1BlC,EAAiBvO,QAAU,MACzB,IAEG0Q,GAAmBvW,UAExB,GADAyT,MACIY,EAAiBxO,QAAS,CAC7B,GAAwC,IAApCwO,EAAiBxO,QAAQY,OAK5B,OAJA1E,QAAQG,MAAM,0BACd8S,GAAavC,EAAkB+D,MAC/BzB,IAAY,QACZL,GAAc,GAIfhB,MACAgB,GAAc,GACd,MAAM+B,EAAY,IAAIC,KAAKrC,EAAiBxO,QAAS,GAAGzD,MAAc0T,KAAsB,CAC3FlF,KAAMgF,KAGDe,EAAWpC,EAAe1O,eAAkB+Q,EAAYH,GAExDI,EAAWJ,EAAUK,KACrBC,GAAcF,EAAQ,SAAkBlP,QAAQ,GAChDqP,EAAoBhQ,KAAKiQ,MAAMJ,EAAW,KAC1CK,EAAY,CACjBC,cAAezP,WAAWqP,GAC1BK,iBAAkBP,EAClBQ,UAAWV,GAAY,EACvBW,WAAYjD,EAAiBxO,QAAQY,OACrC8Q,2BAA4BP,GAGvBQ,EAAkB,CACvB,CAAEtI,OAAQA,IACV,CAAEuI,aAAcrV,IAChB,CAACsV,aAAc,GAAG1D,MAClB,CAAE2D,mBAAoBlV,IACtB,CAAEmV,UAAW,aACb,CAAEC,aAAc5D,KAEbC,IACHsD,EAAS3N,KAAK,CAAEiO,aAAc5D,KAG/B6D,EAAsB,CACrBC,UAAW,GAAGvV,yBACdL,cACA2R,SACAxM,KAAMC,KAAKC,UAAU,CAAE+P,WAAUN,gBAGlC,MAAMe,EAAkBrS,KAAKD,MAC7BoS,EAAsB,CACrBC,UAAW,GAAGvV,4BACdL,cACA2R,SACAmE,UAAWD,IAGZ,IACC,MAAME,QAAYhD,GAAKiD,WAAWC,qBAAqB,CACtDC,KAAM7B,EACN8B,cAAef,EACfgB,UAAWpW,GACX2R,SACA0E,YAAahC,EAAU7F,OAGlB8H,EAAgB9S,KAAKD,MAG3BoS,EAAsB,CACrBC,UAAW,GAAGvV,+BACdL,cACA2R,SACA4E,eAAgBD,EAChBE,eAPsBF,EAAgBT,IASvCtE,MACA5R,QAAQyG,IAAI,sBAAuB2P,GACnChD,GAAK0D,YAAYC,qBAAqB,CACrCC,UAAW,KACVnF,MACAoF,EAAuB,CACtBhB,UAAW,GAAGvV,eACdwW,WAAY7W,GACZ8W,WAAY,WACZtI,KAAM,0BACNmD,YAGF3R,cACA+W,OAAQ,KACPH,EAAuB,CACtBhB,UAAW,GAAGvV,eACdwW,WAAY7W,GACZ8W,WAAY,OACZtI,KAAM,0BACNmD,WAEDhS,QAAQyG,IAAI,2BAEb2H,QAAUlO,IACT+W,EAAuB,CACtBhB,UAAW,GAAGvV,eACdwW,WAAY7W,GACZ8W,WAAY,QACZtI,KAAM,0BACNmD,WAED5D,GAAQlO,IAETmX,UAAY7R,IACXyR,EAAuB,CACtBhB,UAAW,GAAGvV,eACdwW,WAAY7W,GACZ8W,WAAY,UACZtI,KAAM,0BACNmD,WAEDqF,GAAU7R,IAEXyL,QAAS,KACRa,MACA9R,QAAQyG,IAAI,2BACZwQ,EAAuB,CACtBhB,UAAW,GAAGvV,eACdwW,WAAY7W,GACZ8W,WAAY,QACZtI,KAAM,0BACNmD,aAIH,CAAC,MAAO7R,GACRyR,MACAxD,GAAQjO,EACR,CACD,GAGImX,GAAiBjU,EAAY,KAClC,MAAMkU,EAAS1Q,EAAU/C,SAASuQ,UAClC,IAAKkD,EAAQ,OAGb,MACMC,EADaD,EAAOE,iBAAiB,GACfC,eAEtBrY,MAAEA,EAAQ,KAAII,OAAEA,EAAS,KAAQ+X,EAEjCG,EAASxY,SAASC,cAAc,UAChCM,EAAMiY,EAAOhY,WAAW,MAG1BN,EAAQI,GACXkY,EAAOtY,MAAQI,EACfkY,EAAOlY,OAASJ,IAEhBsY,EAAOtY,MAAQA,EACfsY,EAAOlY,OAASA,GAGjB,MAAMmY,EAAUzY,SAASC,cAAc,SACvCwY,EAAQvD,UAAYkD,EACpBK,EAAQ5J,OAAQ,EAChB4J,EAAQ3J,aAAc,EACtB2J,EAAQC,OACRnF,EAAoB5O,QAAU8T,EAG9B,MAAME,EAAY,KAEbzY,EAAQI,GACXC,GAAKqY,OACLrY,GAAK8K,UAAUmN,EAAOtY,MAAO,GAC7BK,GAAKsY,OAAO/S,KAAKsD,GAAK,GACtB7I,GAAK2H,UAAUuQ,EAAS,EAAG,EAAGnY,EAAQJ,GACtCK,GAAKuY,WAELvY,GAAK2H,UAAUuQ,EAAS,EAAG,EAAGvY,EAAOI,GAEtCgT,EAAgB3O,QAAUoU,sBAAsBJ,IAEjDrF,EAAgB3O,QAAUoU,sBAAsBJ,GAGhD,MAAMK,EAAeX,EAASY,WAAa,GAC3C5F,EAAe1O,QAAUqU,EACzB,MAAME,EAAeV,EAAOW,cAAcH,GACpCI,EAA6C,CAClDC,mBAAoB,MAKrB,IAAIC,EAHAnF,GAAe5O,OAAS,IAC3B6T,EAAqBG,SAAWpF,GAAe,IAIhD,IACCmF,EAAgB,IAAIE,cAAcN,EAAcE,EAChD,CAAC,MAAOxY,GAKR,OAJAC,QAAQG,MAAM,6BAA8BJ,GAC5CkT,GAAavC,EAAkB+D,MAC/BzB,IAAY,QACZL,GAAc,EAEd,CAEDL,EAAiBxO,QAAU,GAC3B2U,EAAcG,gBAAmB1H,IAC5BA,GAAO1L,MAAQ0L,EAAM1L,KAAKuP,KAAO,GAAKzC,EAAiBxO,SAC1DwO,EAAiBxO,QAAQgE,KAAKoJ,EAAM1L,OAGtCiT,EAAcI,OAAS,KACtB7Y,QAAQyG,IAAI,kCAAmC6L,GAAkBxO,SAASY,QAC1E8P,MAGDiE,EAAcK,MAAM,KACpBzG,EAAiBvO,QAAU2U,EAC3BzY,QAAQyG,IAAI,yDACV,CAAC6M,GAAgBkB,MAEd5N,iBAAEA,GAAgB7F,UAAEA,GAASC,aAAEA,GAAY6I,UAAEA,GAASzI,cAAEA,GAAakF,kBAAEA,IAAsBlG,EAAY,CAC9GC,cACAK,cACAC,UACAC,YACAL,eAAgB,KACf2T,MAED1T,aAAc,KACbqS,IAAc,IAEfpS,iBAAkB,KACjBT,QAAQyG,IAAI,yDACZ6Q,QAII7H,GAAYpM,EAAY,KACxBjC,KACLqQ,MACAzQ,GAAa,GACbuI,EAAcC,UAAU,GAAGC,mCAC3BpL,WAAW,KACVyU,IAAc,GACdzU,WAAW,KACViI,MACE,MACD,OACD,CAACtF,GAAc8R,GAAexM,GAAmBlF,GAAeqQ,IAE7DpC,GAAiBhM,EAAY,KAClCyP,IAAc,GACdC,IAAiB,GACjBmB,KACArK,KACAS,MACAgI,EAAiBxO,QAAU,GAC3B0O,EAAe1O,QAAU,KACrBuO,EAAiBvO,UACpBuO,EAAiBvO,QAAU,MAE5BkP,IAAY,GACZC,GAAa,MACbjS,IAAa,GACbkS,GAAcC,KACVtM,EAAU/C,SAAWyO,EAAUzO,UAClC+C,EAAU/C,QAAQuQ,UAAY9B,EAAUzO,QACxC9D,QAAQyG,IAAI,wCAEX,CAACzF,GAAc8R,GAAeC,GAAkBmB,GAAerK,GAAWqJ,KAEvE9E,GAAW5I,IAChBxF,QAAQyG,IAAIjB,EAAM,YAClBgM,IAAchM,GACdyN,GAAavC,EAAkB+D,MAC/BzB,IAAY,GACZL,GAAc,GACdI,IAAiB,GACjBiD,EAAsB,CACrBC,UAAW,GAAGvV,cACdL,cACA0Y,OAAQ,SACR/G,SACAxM,KAAMC,KAAKC,UAAUF,MAIjB6R,GAAa7R,IAClBxF,QAAQyG,IAAIjB,EAAM,cACdA,GAA6B,iBAArBA,GAAMwT,YACjBhD,EAAsB,CACrBC,UAAW,GAAGvV,mCACdL,cACA0Y,OAAQ,UACR/G,SACAxM,KAAMC,KAAKC,UAAUF,KAGvB+L,IAAgB/L,IAGjBgB,EAAU,KACT,IAAI7F,IAAWC,EAqBf,OApBIuB,UAAU8W,aAAaC,cAC1B/W,UAAU8W,aACRC,aAAa,CAAEC,MAAO3F,KACtB4F,KAAM7B,IACNhF,EAAUzO,QAAUyT,EAChB1Q,EAAU/C,UACb+C,EAAU/C,QAAQuQ,UAAYkD,KAG/B8B,MAAOnZ,IAEP,GADAF,QAAQG,MAAM,0BAA2BD,GACrCsT,KAA4BE,EAE/B,YADAD,GAA2B6F,GAG5B,MAAMC,EAAsB,CAAC,kBAAmB,wBAAyB,gBAAiB,uBAAwB,mBAAoB,mBAAmBC,SAAStZ,GAAKuZ,MACvKxG,GAAasG,EAAsB7I,EAAkBC,OAASD,EAAkB+D,MAChFzB,IAAY,KAGR,KACFT,EAAUzO,SACbyO,EAAUzO,QAAQ4V,YAAYlS,QAASmS,GAAUA,EAAMpF,UAGvD,CAAC5T,EAASC,EAAW4S,KAGxBhN,EAAU,KACT,IAAKoM,IAAepD,GAAY,OAChC,MAAMoK,EAAaC,YAAY,KAC9BjT,GAAiBC,EAAWC,IAC1B,KAGH,MAAO,IAAMgT,cAAcF,IACzB,CAAChT,GAAkBgM,EAAYpD,KAiBlC,OATAxP,QAAQyG,IAAI,eAAgBmM,EAAY,aAAczD,GAAU,eAAgBK,GAAY,aAAcN,GAC1G1I,EAAU,KACT,MAAMuT,EAAYC,EAA+BrG,GAAqBsG,GACtE1G,GAAkBwG,IAChB,CAACpG,KACJnN,EAAU,KACTxF,IAAa,IACX,IAGF6J,gCAA2B9J,GAASoK,SACpCN,EAACqP,EAAuB,CAAA/O,SACvBN,EAACsP,EAAc,CAACxP,OAAQA,WACvBE,EAAC6E,EAAc0K,SAAQ,CACtBC,MAAO,CACN1Z,UACAC,YACAsO,aACAC,YACAC,aACAC,kBACAC,iBACAvO,aACAgM,gBACAC,mBACAC,cACAC,iBACAC,UACAoC,cAlCiB,IACjBC,GAAmBxD,EAAasO,MAC/BlZ,GACE4K,EAAa8M,MADO9M,EAAazB,QAiCpCiF,cACAC,aACArO,iBACAkJ,WACAa,SAEDN,EAACoE,EAAY,CAACpI,UAAWA,EAAWC,UAAWA"}