@moveris/shared 3.2.0 → 3.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -550,6 +550,7 @@ const status = getStatusMessage('capturing', DEFAULT_LOCALE);
550
550
  | `camera_angle_low` | "Raise camera to eye level" | Camera below face (Y < 0.3) |
551
551
  | `camera_angle_high` | "Lower camera to eye level" | Camera above face (Y > 0.7) |
552
552
  | `camera_tilted` | "Hold camera level" | Device tilt > 15° (via transformation matrix, falls back to eye-corner landmarks) |
553
+ | `flat_lighting` | "Find better lighting" | Face-region dynamic range < 180 L-units — soft warning, capture continues |
553
554
 
554
555
  ---
555
556
 
@@ -677,6 +678,13 @@ if (facialTransformationMatrix) {
677
678
  const angleResult = detectCameraAngle(landmarks); // requires ≥153 landmarks
678
679
  // { ratio: number, cameraAbove: boolean, cameraBelow: boolean }
679
680
  // ratio ~1.0 at eye level; >1.35 = camera above user; <0.75 = camera below user
681
+
682
+ // Analyse face-region dynamic range (flat lighting detection, MOV-1280)
683
+ // Pass pre-sampled face-region pixels — use sampleFaceRegionPixels() in @moveris/react
684
+ import { analyzeDynamicRange } from '@moveris/shared';
685
+ const drResult = analyzeDynamicRange(rgbaPixels);
686
+ // { range: number (max L* − min L*), isLow: boolean (< 180) }
687
+ // Soft warning — do not use as a hard capture gate
680
688
  ```
681
689
 
682
690
  #### Crop Constants (aligned with cognito-check)
package/dist/index.d.mts CHANGED
@@ -341,6 +341,14 @@ declare const MAX_FACE_RATIO = 0.7;
341
341
  declare const FACE_CROP_OUTPUT_SIZE = 224;
342
342
  declare const MAX_FACE_PERCENTAGE_IN_CROP = 0.45;
343
343
  declare const TARGET_FACE_PERCENTAGE_IN_CROP = 0.33;
344
+ declare const DYNAMIC_RANGE_WARNING_THRESHOLD = 180;
345
+ interface DynamicRangeAnalysis {
346
+ range: number;
347
+ isLow: boolean;
348
+ }
349
+ declare function srgbToLinear(c: number): number;
350
+ declare function linearRgbToLabL(rLin: number, gLin: number, bLin: number): number;
351
+ declare function analyzeDynamicRange(rgbaPixels: Uint8ClampedArray | number[]): DynamicRangeAnalysis;
344
352
  declare function analyzeBlur(grayscalePixels: number[], width: number, height: number, threshold?: number): BlurAnalysis;
345
353
  declare function rgbaToGrayscale(rgbaPixels: Uint8ClampedArray | number[]): number[];
346
354
  declare function calculateBrightness(rgbaPixels: Uint8ClampedArray | number[]): number;
@@ -750,6 +758,7 @@ declare const FEEDBACK_MESSAGES: {
750
758
  readonly poor_lighting: "Improve lighting";
751
759
  readonly too_dark: "Low lighting - move to a brighter area";
752
760
  readonly backlit: "Backlit - try facing the light source";
761
+ readonly flat_lighting: "Find better lighting";
753
762
  readonly camera_angle_low: "Raise camera to eye level";
754
763
  readonly camera_angle_high: "Lower camera to eye level";
755
764
  readonly camera_tilted: "Hold camera level";
@@ -789,6 +798,7 @@ interface CaptureQualityState {
789
798
  cameraAngleHigh?: boolean;
790
799
  cameraTilted?: boolean;
791
800
  faceAreaLow?: boolean;
801
+ dynamicRangeLow?: boolean;
792
802
  }
793
803
  declare function getCaptureQualityFeedback(state: CaptureQualityState): string;
794
804
  declare function canCaptureFrame(state: Omit<CaptureQualityState, 'framesCaptured' | 'targetFrames' | 'isCapturing'>): boolean;
@@ -839,4 +849,4 @@ declare function analyzeEyeRegionContrast(pixels: Uint8Array | Uint8ClampedArray
839
849
  declare function detectSpecularHighlights(pixels: Uint8Array | Uint8ClampedArray, relativeFactor?: number, meanBrightness?: number): number;
840
850
  declare function checkEyeRegionQuality(pixels: Uint8Array | Uint8ClampedArray, thresholds?: EyeQualityThresholds): EyeRegionQuality;
841
851
 
842
- export { ALIGNMENT_THRESHOLD_CAPTURE, ALIGNMENT_THRESHOLD_GOOD, ALIGNMENT_THRESHOLD_PERFECT, ALIGNMENT_THRESHOLD_POOR, API_ENDPOINTS, API_ERROR_CODES, API_PATHS, AUTH_CONFIG, type ApiErrorCode, BACKLIT_RATIO_THRESHOLD, BLUR_THRESHOLD_MOBILE, BaseFrameCollector, type BlurAnalysis, CAMERA_ANGLE_HIGH_RATIO, CAMERA_ANGLE_LOW_RATIO, type CameraAngleResult, type CaptureQualityState, type CapturedFrame, type CropData, DEFAULT_BLUR_THRESHOLD, DEFAULT_ENDPOINT, DEFAULT_FACE_DETECTION_TIERS, DEFAULT_GAZE_THRESHOLDS, DEFAULT_HAND_OCCLUSION_CONFIG, DEFAULT_LIVENESS_CONFIG, DEFAULT_LOCALE, DEFAULT_OVAL_REGION, DEFAULT_STABILIZER_CONFIG, DEFAULT_STATUS_MESSAGES, type DeprecationInfo, type DetectionResult, type DetectionSummary, type DetectorConfig, ERROR_MESSAGES, ERROR_MESSAGES_ES, ES_LOCALE, EYE_LANDMARK_INDICES, EYE_QUALITY_THRESHOLDS, type ErrorResponse, type EyeQualityThresholds, type EyeRegionBounds, type EyeRegionQuality, type EyeRegionsBounds, FACE_CENTER_VERTICAL_OFFSET, FACE_CROP_OUTPUT_SIZE, FEEDBACK_MESSAGES, FRAME_BUFFER_CONFIG, FRAME_CONFIG, type FaceAlignmentResult, type FaceBoundingBox, type FaceDetectionTiers, type FaceInOvalResult, type FaceLandmarkPoint, type FaceRollResult, type FaceVisibilityResult, type FastCheckCropsRequest, type FastCheckModel, type FastCheckRequest, type FastCheckResponse, type FastCheckStreamRequest, type FastCheckStreamResponse, type FeedbackLocale, type FeedbackMessageKey, type Frame, FrameBuffer, type FrameData, type FrameQualityResult, FrameQueue, type FrameSource, GOOD_ALIGNMENT, type GazeThresholds, HIGH_ALIGNMENT, HYBRID_MODEL_CONFIGS, type HandOcclusionConfig, type HeadPose, type HealthResponse, type Hybrid150CheckRequest, type Hybrid50CheckRequest, type HybridCheckRequest, type HybridCheckResponse, type HybridFrameData, type HybridModelConfig, IDEAL_CROP_MULTIPLIER, type JobStatus, type JobStatusResponse, LANDMARK_INDEX, LANDMARK_MAX_BOUND, LANDMARK_MIN_BOUND, LOW_LIGHT_THRESHOLD, type LandmarkValidationResult, type LightingAnalysis, LivenessApiError, type LivenessCallbacks, LivenessClient, type LivenessClientConfig, type LivenessConfig, type LivenessResult, type LivenessState, MAX_CROP_MULTIPLIER, MAX_FACE_PERCENTAGE_IN_CROP, MAX_FACE_RATIO, MAX_FACE_ROLL_DEGREES, MAX_IDEAL_FACE_RATIO, MIN_CAPTURE_ALIGNMENT, MIN_CROP_MULTIPLIER, MIN_FACE_AREA_RATIO, MIN_FACE_BOTTOM_MARGIN, MIN_FACE_RATIO, MIN_FACE_SIDE_MARGIN, MIN_FACE_TOP_MARGIN, MIN_IDEAL_FACE_RATIO, MIN_LANDMARK_COUNT, MODEL_CONFIGS, type ModelConfig, type ModelEntry, type ModelType, type ModelVersion, type ModelsResponse, OVAL_GUIDE_COLORS, OVAL_GUIDE_STYLES, OVAL_REGION_DESKTOP, OVAL_REGION_MOBILE, type OnErrorCallback, type OnFrameCapturedCallback, type OnProgressCallback, type OnResultCallback, type OnStateChangeCallback, type OvalGuideState, type OvalRegion, type QueueStatsResponse, RETRY_CONFIG, type RetryOptions, type StabilizationProgress, type StabilizationResult, type StabilizerConfig, type StatusMessageKey, type StreamingStatus, TARGET_FACE_PERCENTAGE_IN_CROP, VALID_FRAME_COUNTS, type Verdict, type VerifyRequest, type VerifyResponse, type VideoFrameMetadata, analyzeBlur, analyzeEyeRegionBrightness, analyzeEyeRegionContrast, analyzeLighting, calculateAdaptiveCropMultiplier, calculateBrightness, calculateFaceAlignment, calculateFaceCropRegion, canCaptureFrame, checkEyeRegionQuality, checkFrameQuality, decodeBase64, detectCameraAngle, detectFaceRoll, detectFaceRollFromMatrix, detectSpecularHighlights, encodeBase64, generateSessionId, getActiveModels, getApiErrorMessage, getCaptureQualityFeedback, getEyeRegionBounds, getFeedbackMessage, getMinFramesForModel, getOvalGuideState, getStatusMessage, hasEnoughFrames, isDeprecatedModel, isFaceCropFullyInFrame, isFaceFullyVisible, isFaceInOval, isRetryableError, retryWithBackoff, rgbaToGrayscale, sleep, toFrameData, toHybridFrameData, toLivenessResult, toLivenessResultFromStream, validateApiKey, validateFaceLandmarks, validateFrameCount, validateFrameData, validateFrameIndex, validateTimestamp, validateUUID, validateUrl };
852
+ export { ALIGNMENT_THRESHOLD_CAPTURE, ALIGNMENT_THRESHOLD_GOOD, ALIGNMENT_THRESHOLD_PERFECT, ALIGNMENT_THRESHOLD_POOR, API_ENDPOINTS, API_ERROR_CODES, API_PATHS, AUTH_CONFIG, type ApiErrorCode, BACKLIT_RATIO_THRESHOLD, BLUR_THRESHOLD_MOBILE, BaseFrameCollector, type BlurAnalysis, CAMERA_ANGLE_HIGH_RATIO, CAMERA_ANGLE_LOW_RATIO, type CameraAngleResult, type CaptureQualityState, type CapturedFrame, type CropData, DEFAULT_BLUR_THRESHOLD, DEFAULT_ENDPOINT, DEFAULT_FACE_DETECTION_TIERS, DEFAULT_GAZE_THRESHOLDS, DEFAULT_HAND_OCCLUSION_CONFIG, DEFAULT_LIVENESS_CONFIG, DEFAULT_LOCALE, DEFAULT_OVAL_REGION, DEFAULT_STABILIZER_CONFIG, DEFAULT_STATUS_MESSAGES, DYNAMIC_RANGE_WARNING_THRESHOLD, type DeprecationInfo, type DetectionResult, type DetectionSummary, type DetectorConfig, type DynamicRangeAnalysis, ERROR_MESSAGES, ERROR_MESSAGES_ES, ES_LOCALE, EYE_LANDMARK_INDICES, EYE_QUALITY_THRESHOLDS, type ErrorResponse, type EyeQualityThresholds, type EyeRegionBounds, type EyeRegionQuality, type EyeRegionsBounds, FACE_CENTER_VERTICAL_OFFSET, FACE_CROP_OUTPUT_SIZE, FEEDBACK_MESSAGES, FRAME_BUFFER_CONFIG, FRAME_CONFIG, type FaceAlignmentResult, type FaceBoundingBox, type FaceDetectionTiers, type FaceInOvalResult, type FaceLandmarkPoint, type FaceRollResult, type FaceVisibilityResult, type FastCheckCropsRequest, type FastCheckModel, type FastCheckRequest, type FastCheckResponse, type FastCheckStreamRequest, type FastCheckStreamResponse, type FeedbackLocale, type FeedbackMessageKey, type Frame, FrameBuffer, type FrameData, type FrameQualityResult, FrameQueue, type FrameSource, GOOD_ALIGNMENT, type GazeThresholds, HIGH_ALIGNMENT, HYBRID_MODEL_CONFIGS, type HandOcclusionConfig, type HeadPose, type HealthResponse, type Hybrid150CheckRequest, type Hybrid50CheckRequest, type HybridCheckRequest, type HybridCheckResponse, type HybridFrameData, type HybridModelConfig, IDEAL_CROP_MULTIPLIER, type JobStatus, type JobStatusResponse, LANDMARK_INDEX, LANDMARK_MAX_BOUND, LANDMARK_MIN_BOUND, LOW_LIGHT_THRESHOLD, type LandmarkValidationResult, type LightingAnalysis, LivenessApiError, type LivenessCallbacks, LivenessClient, type LivenessClientConfig, type LivenessConfig, type LivenessResult, type LivenessState, MAX_CROP_MULTIPLIER, MAX_FACE_PERCENTAGE_IN_CROP, MAX_FACE_RATIO, MAX_FACE_ROLL_DEGREES, MAX_IDEAL_FACE_RATIO, MIN_CAPTURE_ALIGNMENT, MIN_CROP_MULTIPLIER, MIN_FACE_AREA_RATIO, MIN_FACE_BOTTOM_MARGIN, MIN_FACE_RATIO, MIN_FACE_SIDE_MARGIN, MIN_FACE_TOP_MARGIN, MIN_IDEAL_FACE_RATIO, MIN_LANDMARK_COUNT, MODEL_CONFIGS, type ModelConfig, type ModelEntry, type ModelType, type ModelVersion, type ModelsResponse, OVAL_GUIDE_COLORS, OVAL_GUIDE_STYLES, OVAL_REGION_DESKTOP, OVAL_REGION_MOBILE, type OnErrorCallback, type OnFrameCapturedCallback, type OnProgressCallback, type OnResultCallback, type OnStateChangeCallback, type OvalGuideState, type OvalRegion, type QueueStatsResponse, RETRY_CONFIG, type RetryOptions, type StabilizationProgress, type StabilizationResult, type StabilizerConfig, type StatusMessageKey, type StreamingStatus, TARGET_FACE_PERCENTAGE_IN_CROP, VALID_FRAME_COUNTS, type Verdict, type VerifyRequest, type VerifyResponse, type VideoFrameMetadata, analyzeBlur, analyzeDynamicRange, analyzeEyeRegionBrightness, analyzeEyeRegionContrast, analyzeLighting, calculateAdaptiveCropMultiplier, calculateBrightness, calculateFaceAlignment, calculateFaceCropRegion, canCaptureFrame, checkEyeRegionQuality, checkFrameQuality, decodeBase64, detectCameraAngle, detectFaceRoll, detectFaceRollFromMatrix, detectSpecularHighlights, encodeBase64, generateSessionId, getActiveModels, getApiErrorMessage, getCaptureQualityFeedback, getEyeRegionBounds, getFeedbackMessage, getMinFramesForModel, getOvalGuideState, getStatusMessage, hasEnoughFrames, isDeprecatedModel, isFaceCropFullyInFrame, isFaceFullyVisible, isFaceInOval, isRetryableError, linearRgbToLabL, retryWithBackoff, rgbaToGrayscale, sleep, srgbToLinear, toFrameData, toHybridFrameData, toLivenessResult, toLivenessResultFromStream, validateApiKey, validateFaceLandmarks, validateFrameCount, validateFrameData, validateFrameIndex, validateTimestamp, validateUUID, validateUrl };
package/dist/index.d.ts CHANGED
@@ -341,6 +341,14 @@ declare const MAX_FACE_RATIO = 0.7;
341
341
  declare const FACE_CROP_OUTPUT_SIZE = 224;
342
342
  declare const MAX_FACE_PERCENTAGE_IN_CROP = 0.45;
343
343
  declare const TARGET_FACE_PERCENTAGE_IN_CROP = 0.33;
344
+ declare const DYNAMIC_RANGE_WARNING_THRESHOLD = 180;
345
+ interface DynamicRangeAnalysis {
346
+ range: number;
347
+ isLow: boolean;
348
+ }
349
+ declare function srgbToLinear(c: number): number;
350
+ declare function linearRgbToLabL(rLin: number, gLin: number, bLin: number): number;
351
+ declare function analyzeDynamicRange(rgbaPixels: Uint8ClampedArray | number[]): DynamicRangeAnalysis;
344
352
  declare function analyzeBlur(grayscalePixels: number[], width: number, height: number, threshold?: number): BlurAnalysis;
345
353
  declare function rgbaToGrayscale(rgbaPixels: Uint8ClampedArray | number[]): number[];
346
354
  declare function calculateBrightness(rgbaPixels: Uint8ClampedArray | number[]): number;
@@ -750,6 +758,7 @@ declare const FEEDBACK_MESSAGES: {
750
758
  readonly poor_lighting: "Improve lighting";
751
759
  readonly too_dark: "Low lighting - move to a brighter area";
752
760
  readonly backlit: "Backlit - try facing the light source";
761
+ readonly flat_lighting: "Find better lighting";
753
762
  readonly camera_angle_low: "Raise camera to eye level";
754
763
  readonly camera_angle_high: "Lower camera to eye level";
755
764
  readonly camera_tilted: "Hold camera level";
@@ -789,6 +798,7 @@ interface CaptureQualityState {
789
798
  cameraAngleHigh?: boolean;
790
799
  cameraTilted?: boolean;
791
800
  faceAreaLow?: boolean;
801
+ dynamicRangeLow?: boolean;
792
802
  }
793
803
  declare function getCaptureQualityFeedback(state: CaptureQualityState): string;
794
804
  declare function canCaptureFrame(state: Omit<CaptureQualityState, 'framesCaptured' | 'targetFrames' | 'isCapturing'>): boolean;
@@ -839,4 +849,4 @@ declare function analyzeEyeRegionContrast(pixels: Uint8Array | Uint8ClampedArray
839
849
  declare function detectSpecularHighlights(pixels: Uint8Array | Uint8ClampedArray, relativeFactor?: number, meanBrightness?: number): number;
840
850
  declare function checkEyeRegionQuality(pixels: Uint8Array | Uint8ClampedArray, thresholds?: EyeQualityThresholds): EyeRegionQuality;
841
851
 
842
- export { ALIGNMENT_THRESHOLD_CAPTURE, ALIGNMENT_THRESHOLD_GOOD, ALIGNMENT_THRESHOLD_PERFECT, ALIGNMENT_THRESHOLD_POOR, API_ENDPOINTS, API_ERROR_CODES, API_PATHS, AUTH_CONFIG, type ApiErrorCode, BACKLIT_RATIO_THRESHOLD, BLUR_THRESHOLD_MOBILE, BaseFrameCollector, type BlurAnalysis, CAMERA_ANGLE_HIGH_RATIO, CAMERA_ANGLE_LOW_RATIO, type CameraAngleResult, type CaptureQualityState, type CapturedFrame, type CropData, DEFAULT_BLUR_THRESHOLD, DEFAULT_ENDPOINT, DEFAULT_FACE_DETECTION_TIERS, DEFAULT_GAZE_THRESHOLDS, DEFAULT_HAND_OCCLUSION_CONFIG, DEFAULT_LIVENESS_CONFIG, DEFAULT_LOCALE, DEFAULT_OVAL_REGION, DEFAULT_STABILIZER_CONFIG, DEFAULT_STATUS_MESSAGES, type DeprecationInfo, type DetectionResult, type DetectionSummary, type DetectorConfig, ERROR_MESSAGES, ERROR_MESSAGES_ES, ES_LOCALE, EYE_LANDMARK_INDICES, EYE_QUALITY_THRESHOLDS, type ErrorResponse, type EyeQualityThresholds, type EyeRegionBounds, type EyeRegionQuality, type EyeRegionsBounds, FACE_CENTER_VERTICAL_OFFSET, FACE_CROP_OUTPUT_SIZE, FEEDBACK_MESSAGES, FRAME_BUFFER_CONFIG, FRAME_CONFIG, type FaceAlignmentResult, type FaceBoundingBox, type FaceDetectionTiers, type FaceInOvalResult, type FaceLandmarkPoint, type FaceRollResult, type FaceVisibilityResult, type FastCheckCropsRequest, type FastCheckModel, type FastCheckRequest, type FastCheckResponse, type FastCheckStreamRequest, type FastCheckStreamResponse, type FeedbackLocale, type FeedbackMessageKey, type Frame, FrameBuffer, type FrameData, type FrameQualityResult, FrameQueue, type FrameSource, GOOD_ALIGNMENT, type GazeThresholds, HIGH_ALIGNMENT, HYBRID_MODEL_CONFIGS, type HandOcclusionConfig, type HeadPose, type HealthResponse, type Hybrid150CheckRequest, type Hybrid50CheckRequest, type HybridCheckRequest, type HybridCheckResponse, type HybridFrameData, type HybridModelConfig, IDEAL_CROP_MULTIPLIER, type JobStatus, type JobStatusResponse, LANDMARK_INDEX, LANDMARK_MAX_BOUND, LANDMARK_MIN_BOUND, LOW_LIGHT_THRESHOLD, type LandmarkValidationResult, type LightingAnalysis, LivenessApiError, type LivenessCallbacks, LivenessClient, type LivenessClientConfig, type LivenessConfig, type LivenessResult, type LivenessState, MAX_CROP_MULTIPLIER, MAX_FACE_PERCENTAGE_IN_CROP, MAX_FACE_RATIO, MAX_FACE_ROLL_DEGREES, MAX_IDEAL_FACE_RATIO, MIN_CAPTURE_ALIGNMENT, MIN_CROP_MULTIPLIER, MIN_FACE_AREA_RATIO, MIN_FACE_BOTTOM_MARGIN, MIN_FACE_RATIO, MIN_FACE_SIDE_MARGIN, MIN_FACE_TOP_MARGIN, MIN_IDEAL_FACE_RATIO, MIN_LANDMARK_COUNT, MODEL_CONFIGS, type ModelConfig, type ModelEntry, type ModelType, type ModelVersion, type ModelsResponse, OVAL_GUIDE_COLORS, OVAL_GUIDE_STYLES, OVAL_REGION_DESKTOP, OVAL_REGION_MOBILE, type OnErrorCallback, type OnFrameCapturedCallback, type OnProgressCallback, type OnResultCallback, type OnStateChangeCallback, type OvalGuideState, type OvalRegion, type QueueStatsResponse, RETRY_CONFIG, type RetryOptions, type StabilizationProgress, type StabilizationResult, type StabilizerConfig, type StatusMessageKey, type StreamingStatus, TARGET_FACE_PERCENTAGE_IN_CROP, VALID_FRAME_COUNTS, type Verdict, type VerifyRequest, type VerifyResponse, type VideoFrameMetadata, analyzeBlur, analyzeEyeRegionBrightness, analyzeEyeRegionContrast, analyzeLighting, calculateAdaptiveCropMultiplier, calculateBrightness, calculateFaceAlignment, calculateFaceCropRegion, canCaptureFrame, checkEyeRegionQuality, checkFrameQuality, decodeBase64, detectCameraAngle, detectFaceRoll, detectFaceRollFromMatrix, detectSpecularHighlights, encodeBase64, generateSessionId, getActiveModels, getApiErrorMessage, getCaptureQualityFeedback, getEyeRegionBounds, getFeedbackMessage, getMinFramesForModel, getOvalGuideState, getStatusMessage, hasEnoughFrames, isDeprecatedModel, isFaceCropFullyInFrame, isFaceFullyVisible, isFaceInOval, isRetryableError, retryWithBackoff, rgbaToGrayscale, sleep, toFrameData, toHybridFrameData, toLivenessResult, toLivenessResultFromStream, validateApiKey, validateFaceLandmarks, validateFrameCount, validateFrameData, validateFrameIndex, validateTimestamp, validateUUID, validateUrl };
852
+ export { ALIGNMENT_THRESHOLD_CAPTURE, ALIGNMENT_THRESHOLD_GOOD, ALIGNMENT_THRESHOLD_PERFECT, ALIGNMENT_THRESHOLD_POOR, API_ENDPOINTS, API_ERROR_CODES, API_PATHS, AUTH_CONFIG, type ApiErrorCode, BACKLIT_RATIO_THRESHOLD, BLUR_THRESHOLD_MOBILE, BaseFrameCollector, type BlurAnalysis, CAMERA_ANGLE_HIGH_RATIO, CAMERA_ANGLE_LOW_RATIO, type CameraAngleResult, type CaptureQualityState, type CapturedFrame, type CropData, DEFAULT_BLUR_THRESHOLD, DEFAULT_ENDPOINT, DEFAULT_FACE_DETECTION_TIERS, DEFAULT_GAZE_THRESHOLDS, DEFAULT_HAND_OCCLUSION_CONFIG, DEFAULT_LIVENESS_CONFIG, DEFAULT_LOCALE, DEFAULT_OVAL_REGION, DEFAULT_STABILIZER_CONFIG, DEFAULT_STATUS_MESSAGES, DYNAMIC_RANGE_WARNING_THRESHOLD, type DeprecationInfo, type DetectionResult, type DetectionSummary, type DetectorConfig, type DynamicRangeAnalysis, ERROR_MESSAGES, ERROR_MESSAGES_ES, ES_LOCALE, EYE_LANDMARK_INDICES, EYE_QUALITY_THRESHOLDS, type ErrorResponse, type EyeQualityThresholds, type EyeRegionBounds, type EyeRegionQuality, type EyeRegionsBounds, FACE_CENTER_VERTICAL_OFFSET, FACE_CROP_OUTPUT_SIZE, FEEDBACK_MESSAGES, FRAME_BUFFER_CONFIG, FRAME_CONFIG, type FaceAlignmentResult, type FaceBoundingBox, type FaceDetectionTiers, type FaceInOvalResult, type FaceLandmarkPoint, type FaceRollResult, type FaceVisibilityResult, type FastCheckCropsRequest, type FastCheckModel, type FastCheckRequest, type FastCheckResponse, type FastCheckStreamRequest, type FastCheckStreamResponse, type FeedbackLocale, type FeedbackMessageKey, type Frame, FrameBuffer, type FrameData, type FrameQualityResult, FrameQueue, type FrameSource, GOOD_ALIGNMENT, type GazeThresholds, HIGH_ALIGNMENT, HYBRID_MODEL_CONFIGS, type HandOcclusionConfig, type HeadPose, type HealthResponse, type Hybrid150CheckRequest, type Hybrid50CheckRequest, type HybridCheckRequest, type HybridCheckResponse, type HybridFrameData, type HybridModelConfig, IDEAL_CROP_MULTIPLIER, type JobStatus, type JobStatusResponse, LANDMARK_INDEX, LANDMARK_MAX_BOUND, LANDMARK_MIN_BOUND, LOW_LIGHT_THRESHOLD, type LandmarkValidationResult, type LightingAnalysis, LivenessApiError, type LivenessCallbacks, LivenessClient, type LivenessClientConfig, type LivenessConfig, type LivenessResult, type LivenessState, MAX_CROP_MULTIPLIER, MAX_FACE_PERCENTAGE_IN_CROP, MAX_FACE_RATIO, MAX_FACE_ROLL_DEGREES, MAX_IDEAL_FACE_RATIO, MIN_CAPTURE_ALIGNMENT, MIN_CROP_MULTIPLIER, MIN_FACE_AREA_RATIO, MIN_FACE_BOTTOM_MARGIN, MIN_FACE_RATIO, MIN_FACE_SIDE_MARGIN, MIN_FACE_TOP_MARGIN, MIN_IDEAL_FACE_RATIO, MIN_LANDMARK_COUNT, MODEL_CONFIGS, type ModelConfig, type ModelEntry, type ModelType, type ModelVersion, type ModelsResponse, OVAL_GUIDE_COLORS, OVAL_GUIDE_STYLES, OVAL_REGION_DESKTOP, OVAL_REGION_MOBILE, type OnErrorCallback, type OnFrameCapturedCallback, type OnProgressCallback, type OnResultCallback, type OnStateChangeCallback, type OvalGuideState, type OvalRegion, type QueueStatsResponse, RETRY_CONFIG, type RetryOptions, type StabilizationProgress, type StabilizationResult, type StabilizerConfig, type StatusMessageKey, type StreamingStatus, TARGET_FACE_PERCENTAGE_IN_CROP, VALID_FRAME_COUNTS, type Verdict, type VerifyRequest, type VerifyResponse, type VideoFrameMetadata, analyzeBlur, analyzeDynamicRange, analyzeEyeRegionBrightness, analyzeEyeRegionContrast, analyzeLighting, calculateAdaptiveCropMultiplier, calculateBrightness, calculateFaceAlignment, calculateFaceCropRegion, canCaptureFrame, checkEyeRegionQuality, checkFrameQuality, decodeBase64, detectCameraAngle, detectFaceRoll, detectFaceRollFromMatrix, detectSpecularHighlights, encodeBase64, generateSessionId, getActiveModels, getApiErrorMessage, getCaptureQualityFeedback, getEyeRegionBounds, getFeedbackMessage, getMinFramesForModel, getOvalGuideState, getStatusMessage, hasEnoughFrames, isDeprecatedModel, isFaceCropFullyInFrame, isFaceFullyVisible, isFaceInOval, isRetryableError, linearRgbToLabL, retryWithBackoff, rgbaToGrayscale, sleep, srgbToLinear, toFrameData, toHybridFrameData, toLivenessResult, toLivenessResultFromStream, validateApiKey, validateFaceLandmarks, validateFrameCount, validateFrameData, validateFrameIndex, validateTimestamp, validateUUID, validateUrl };
package/dist/index.js CHANGED
@@ -43,6 +43,7 @@ __export(index_exports, {
43
43
  DEFAULT_OVAL_REGION: () => DEFAULT_OVAL_REGION,
44
44
  DEFAULT_STABILIZER_CONFIG: () => DEFAULT_STABILIZER_CONFIG,
45
45
  DEFAULT_STATUS_MESSAGES: () => DEFAULT_STATUS_MESSAGES,
46
+ DYNAMIC_RANGE_WARNING_THRESHOLD: () => DYNAMIC_RANGE_WARNING_THRESHOLD,
46
47
  ERROR_MESSAGES: () => ERROR_MESSAGES,
47
48
  ERROR_MESSAGES_ES: () => ERROR_MESSAGES_ES,
48
49
  ES_LOCALE: () => ES_LOCALE,
@@ -88,6 +89,7 @@ __export(index_exports, {
88
89
  TARGET_FACE_PERCENTAGE_IN_CROP: () => TARGET_FACE_PERCENTAGE_IN_CROP,
89
90
  VALID_FRAME_COUNTS: () => VALID_FRAME_COUNTS,
90
91
  analyzeBlur: () => analyzeBlur,
92
+ analyzeDynamicRange: () => analyzeDynamicRange,
91
93
  analyzeEyeRegionBrightness: () => analyzeEyeRegionBrightness,
92
94
  analyzeEyeRegionContrast: () => analyzeEyeRegionContrast,
93
95
  analyzeLighting: () => analyzeLighting,
@@ -119,9 +121,11 @@ __export(index_exports, {
119
121
  isFaceFullyVisible: () => isFaceFullyVisible,
120
122
  isFaceInOval: () => isFaceInOval,
121
123
  isRetryableError: () => isRetryableError,
124
+ linearRgbToLabL: () => linearRgbToLabL,
122
125
  retryWithBackoff: () => retryWithBackoff,
123
126
  rgbaToGrayscale: () => rgbaToGrayscale,
124
127
  sleep: () => sleep,
128
+ srgbToLinear: () => srgbToLinear,
125
129
  toFrameData: () => toFrameData,
126
130
  toHybridFrameData: () => toHybridFrameData,
127
131
  toLivenessResult: () => toLivenessResult,
@@ -1377,6 +1381,7 @@ var FEEDBACK_MESSAGES = {
1377
1381
  poor_lighting: "Improve lighting",
1378
1382
  too_dark: "Low lighting - move to a brighter area",
1379
1383
  backlit: "Backlit - try facing the light source",
1384
+ flat_lighting: "Find better lighting",
1380
1385
  // Camera angle (platform-agnostic)
1381
1386
  camera_angle_low: "Raise camera to eye level",
1382
1387
  camera_angle_high: "Lower camera to eye level",
@@ -1440,6 +1445,7 @@ var ES_LOCALE = {
1440
1445
  poor_lighting: "Mejora la iluminaci\xF3n",
1441
1446
  too_dark: "Poca luz - mu\xE9vete a un \xE1rea m\xE1s iluminada",
1442
1447
  backlit: "Contraluz - intenta mirar hacia la fuente de luz",
1448
+ flat_lighting: "Busca mejor iluminaci\xF3n",
1443
1449
  // Camera angle (platform-agnostic)
1444
1450
  camera_angle_low: "Levanta la c\xE1mara a la altura de los ojos",
1445
1451
  camera_angle_high: "Baja la c\xE1mara a la altura de los ojos",
@@ -1482,7 +1488,8 @@ function getCaptureQualityFeedback(state) {
1482
1488
  cameraAngleLow,
1483
1489
  cameraAngleHigh,
1484
1490
  cameraTilted,
1485
- faceAreaLow
1491
+ faceAreaLow,
1492
+ dynamicRangeLow
1486
1493
  } = state;
1487
1494
  if (!hasFace) {
1488
1495
  return FEEDBACK_MESSAGES.no_face;
@@ -1508,6 +1515,9 @@ function getCaptureQualityFeedback(state) {
1508
1515
  if (isBlurry) {
1509
1516
  return FEEDBACK_MESSAGES.blurry;
1510
1517
  }
1518
+ if (dynamicRangeLow) {
1519
+ return FEEDBACK_MESSAGES.flat_lighting;
1520
+ }
1511
1521
  if (tooFarFromIdeal) {
1512
1522
  return FEEDBACK_MESSAGES.move_closer_ideal;
1513
1523
  }
@@ -1632,6 +1642,33 @@ var MAX_FACE_RATIO = 0.7;
1632
1642
  var FACE_CROP_OUTPUT_SIZE = 224;
1633
1643
  var MAX_FACE_PERCENTAGE_IN_CROP = 0.45;
1634
1644
  var TARGET_FACE_PERCENTAGE_IN_CROP = 0.33;
1645
+ var DYNAMIC_RANGE_WARNING_THRESHOLD = 180;
1646
+ function srgbToLinear(c) {
1647
+ const n = c / 255;
1648
+ return n <= 0.04045 ? n / 12.92 : Math.pow((n + 0.055) / 1.055, 2.4);
1649
+ }
1650
+ function linearRgbToLabL(rLin, gLin, bLin) {
1651
+ const Y = 0.2126 * rLin + 0.7152 * gLin + 0.0722 * bLin;
1652
+ const f = Y > 8856e-6 ? Math.cbrt(Y) : 7.787 * Y + 16 / 116;
1653
+ return 116 * f - 16;
1654
+ }
1655
+ function analyzeDynamicRange(rgbaPixels) {
1656
+ let minL = Infinity;
1657
+ let maxL = -Infinity;
1658
+ for (let i = 0; i < rgbaPixels.length; i += 4) {
1659
+ const r = rgbaPixels[i] ?? 0;
1660
+ const g = rgbaPixels[i + 1] ?? 0;
1661
+ const b = rgbaPixels[i + 2] ?? 0;
1662
+ const L = linearRgbToLabL(srgbToLinear(r), srgbToLinear(g), srgbToLinear(b));
1663
+ if (L < minL) minL = L;
1664
+ if (L > maxL) maxL = L;
1665
+ }
1666
+ if (!isFinite(minL) || !isFinite(maxL)) {
1667
+ return { range: 0, isLow: false };
1668
+ }
1669
+ const range = maxL - minL;
1670
+ return { range, isLow: range < DYNAMIC_RANGE_WARNING_THRESHOLD };
1671
+ }
1635
1672
  function analyzeBlur(grayscalePixels, width, height, threshold = DEFAULT_BLUR_THRESHOLD) {
1636
1673
  const laplacian = [];
1637
1674
  for (let y = 1; y < height - 1; y++) {
@@ -2180,6 +2217,7 @@ function checkEyeRegionQuality(pixels, thresholds = EYE_QUALITY_THRESHOLDS) {
2180
2217
  DEFAULT_OVAL_REGION,
2181
2218
  DEFAULT_STABILIZER_CONFIG,
2182
2219
  DEFAULT_STATUS_MESSAGES,
2220
+ DYNAMIC_RANGE_WARNING_THRESHOLD,
2183
2221
  ERROR_MESSAGES,
2184
2222
  ERROR_MESSAGES_ES,
2185
2223
  ES_LOCALE,
@@ -2225,6 +2263,7 @@ function checkEyeRegionQuality(pixels, thresholds = EYE_QUALITY_THRESHOLDS) {
2225
2263
  TARGET_FACE_PERCENTAGE_IN_CROP,
2226
2264
  VALID_FRAME_COUNTS,
2227
2265
  analyzeBlur,
2266
+ analyzeDynamicRange,
2228
2267
  analyzeEyeRegionBrightness,
2229
2268
  analyzeEyeRegionContrast,
2230
2269
  analyzeLighting,
@@ -2256,9 +2295,11 @@ function checkEyeRegionQuality(pixels, thresholds = EYE_QUALITY_THRESHOLDS) {
2256
2295
  isFaceFullyVisible,
2257
2296
  isFaceInOval,
2258
2297
  isRetryableError,
2298
+ linearRgbToLabL,
2259
2299
  retryWithBackoff,
2260
2300
  rgbaToGrayscale,
2261
2301
  sleep,
2302
+ srgbToLinear,
2262
2303
  toFrameData,
2263
2304
  toHybridFrameData,
2264
2305
  toLivenessResult,
package/dist/index.mjs CHANGED
@@ -1238,6 +1238,7 @@ var FEEDBACK_MESSAGES = {
1238
1238
  poor_lighting: "Improve lighting",
1239
1239
  too_dark: "Low lighting - move to a brighter area",
1240
1240
  backlit: "Backlit - try facing the light source",
1241
+ flat_lighting: "Find better lighting",
1241
1242
  // Camera angle (platform-agnostic)
1242
1243
  camera_angle_low: "Raise camera to eye level",
1243
1244
  camera_angle_high: "Lower camera to eye level",
@@ -1301,6 +1302,7 @@ var ES_LOCALE = {
1301
1302
  poor_lighting: "Mejora la iluminaci\xF3n",
1302
1303
  too_dark: "Poca luz - mu\xE9vete a un \xE1rea m\xE1s iluminada",
1303
1304
  backlit: "Contraluz - intenta mirar hacia la fuente de luz",
1305
+ flat_lighting: "Busca mejor iluminaci\xF3n",
1304
1306
  // Camera angle (platform-agnostic)
1305
1307
  camera_angle_low: "Levanta la c\xE1mara a la altura de los ojos",
1306
1308
  camera_angle_high: "Baja la c\xE1mara a la altura de los ojos",
@@ -1343,7 +1345,8 @@ function getCaptureQualityFeedback(state) {
1343
1345
  cameraAngleLow,
1344
1346
  cameraAngleHigh,
1345
1347
  cameraTilted,
1346
- faceAreaLow
1348
+ faceAreaLow,
1349
+ dynamicRangeLow
1347
1350
  } = state;
1348
1351
  if (!hasFace) {
1349
1352
  return FEEDBACK_MESSAGES.no_face;
@@ -1369,6 +1372,9 @@ function getCaptureQualityFeedback(state) {
1369
1372
  if (isBlurry) {
1370
1373
  return FEEDBACK_MESSAGES.blurry;
1371
1374
  }
1375
+ if (dynamicRangeLow) {
1376
+ return FEEDBACK_MESSAGES.flat_lighting;
1377
+ }
1372
1378
  if (tooFarFromIdeal) {
1373
1379
  return FEEDBACK_MESSAGES.move_closer_ideal;
1374
1380
  }
@@ -1493,6 +1499,33 @@ var MAX_FACE_RATIO = 0.7;
1493
1499
  var FACE_CROP_OUTPUT_SIZE = 224;
1494
1500
  var MAX_FACE_PERCENTAGE_IN_CROP = 0.45;
1495
1501
  var TARGET_FACE_PERCENTAGE_IN_CROP = 0.33;
1502
+ var DYNAMIC_RANGE_WARNING_THRESHOLD = 180;
1503
+ function srgbToLinear(c) {
1504
+ const n = c / 255;
1505
+ return n <= 0.04045 ? n / 12.92 : Math.pow((n + 0.055) / 1.055, 2.4);
1506
+ }
1507
+ function linearRgbToLabL(rLin, gLin, bLin) {
1508
+ const Y = 0.2126 * rLin + 0.7152 * gLin + 0.0722 * bLin;
1509
+ const f = Y > 8856e-6 ? Math.cbrt(Y) : 7.787 * Y + 16 / 116;
1510
+ return 116 * f - 16;
1511
+ }
1512
+ function analyzeDynamicRange(rgbaPixels) {
1513
+ let minL = Infinity;
1514
+ let maxL = -Infinity;
1515
+ for (let i = 0; i < rgbaPixels.length; i += 4) {
1516
+ const r = rgbaPixels[i] ?? 0;
1517
+ const g = rgbaPixels[i + 1] ?? 0;
1518
+ const b = rgbaPixels[i + 2] ?? 0;
1519
+ const L = linearRgbToLabL(srgbToLinear(r), srgbToLinear(g), srgbToLinear(b));
1520
+ if (L < minL) minL = L;
1521
+ if (L > maxL) maxL = L;
1522
+ }
1523
+ if (!isFinite(minL) || !isFinite(maxL)) {
1524
+ return { range: 0, isLow: false };
1525
+ }
1526
+ const range = maxL - minL;
1527
+ return { range, isLow: range < DYNAMIC_RANGE_WARNING_THRESHOLD };
1528
+ }
1496
1529
  function analyzeBlur(grayscalePixels, width, height, threshold = DEFAULT_BLUR_THRESHOLD) {
1497
1530
  const laplacian = [];
1498
1531
  for (let y = 1; y < height - 1; y++) {
@@ -2040,6 +2073,7 @@ export {
2040
2073
  DEFAULT_OVAL_REGION,
2041
2074
  DEFAULT_STABILIZER_CONFIG,
2042
2075
  DEFAULT_STATUS_MESSAGES,
2076
+ DYNAMIC_RANGE_WARNING_THRESHOLD,
2043
2077
  ERROR_MESSAGES,
2044
2078
  ERROR_MESSAGES_ES,
2045
2079
  ES_LOCALE,
@@ -2085,6 +2119,7 @@ export {
2085
2119
  TARGET_FACE_PERCENTAGE_IN_CROP,
2086
2120
  VALID_FRAME_COUNTS,
2087
2121
  analyzeBlur,
2122
+ analyzeDynamicRange,
2088
2123
  analyzeEyeRegionBrightness,
2089
2124
  analyzeEyeRegionContrast,
2090
2125
  analyzeLighting,
@@ -2116,9 +2151,11 @@ export {
2116
2151
  isFaceFullyVisible,
2117
2152
  isFaceInOval,
2118
2153
  isRetryableError,
2154
+ linearRgbToLabL,
2119
2155
  retryWithBackoff,
2120
2156
  rgbaToGrayscale,
2121
2157
  sleep,
2158
+ srgbToLinear,
2122
2159
  toFrameData,
2123
2160
  toHybridFrameData,
2124
2161
  toLivenessResult,
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@moveris/shared",
3
- "version": "3.2.0",
3
+ "version": "3.3.0",
4
4
  "description": "Core business logic for Moveris Live SDK",
5
5
  "main": "./dist/index.js",
6
6
  "module": "./dist/index.mjs",