capacitor-plugin-camera-forked 3.0.110 → 3.0.111

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
@@ -105,6 +105,7 @@ document.documentElement.style.setProperty('--ion-background-color', 'transparen
105
105
  * [`startCamera()`](#startcamera)
106
106
  * [`stopCamera()`](#stopcamera)
107
107
  * [`takeSnapshot(...)`](#takesnapshot)
108
+ * [`detectBlur(...)`](#detectblur)
108
109
  * [`saveFrame()`](#saveframe)
109
110
  * [`takeSnapshot2(...)`](#takesnapshot2)
110
111
  * [`takePhoto(...)`](#takephoto)
@@ -303,6 +304,23 @@ take a snapshot as base64.
303
304
  --------------------
304
305
 
305
306
 
307
+ ### detectBlur(...)
308
+
309
+ ```typescript
310
+ detectBlur(options: { image: string; }) => Promise<{ isBlur: boolean; blurConfidence: number; sharpConfidence: number; }>
311
+ ```
312
+
313
+ analyze an image for blur detection with detailed confidence scores.
314
+
315
+ | Param | Type |
316
+ | ------------- | ------------------------------- |
317
+ | **`options`** | <code>{ image: string; }</code> |
318
+
319
+ **Returns:** <code>Promise&lt;{ isBlur: boolean; blurConfidence: number; sharpConfidence: number; }&gt;</code>
320
+
321
+ --------------------
322
+
323
+
306
324
  ### saveFrame()
307
325
 
308
326
  ```typescript
@@ -520,9 +538,34 @@ measuredByPercentage: 0 in pixel, 1 in percent
520
538
 
521
539
  ## Blur Detection
522
540
 
523
- The plugin includes blur detection capabilities using the Laplacian variance algorithm, providing consistent results across all platforms.
541
+ The plugin includes blur detection capabilities using TensorFlow Lite models with Laplacian variance fallback, providing consistent results across all platforms.
542
+
543
+ #### Analyze Existing Images
524
544
 
525
- #### Basic Usage
545
+ Use the `detectBlur` method to analyze any base64 image with detailed confidence scores:
546
+
547
+ ```typescript
548
+ // Analyze an existing image (base64 string or data URL)
549
+ const result = await CameraPreview.detectBlur({
550
+ image: "data:image/jpeg;base64,/9j/4AAQSkZJRgABAQEASABIAAD..."
551
+ // or just the base64 string: "/9j/4AAQSkZJRgABAQEASABIAAD..."
552
+ });
553
+
554
+ console.log('Is Blurry:', result.isBlur); // boolean: true/false
555
+ console.log('Blur Confidence:', result.blurConfidence); // 0.0-1.0 (higher = more blurry)
556
+ console.log('Sharp Confidence:', result.sharpConfidence); // 0.0-1.0 (higher = more sharp)
557
+
558
+ // Use confidence scores for advanced logic
559
+ if (result.blurConfidence > 0.7) {
560
+ console.log('High confidence this image is blurry');
561
+ } else if (result.sharpConfidence > 0.8) {
562
+ console.log('High confidence this image is sharp');
563
+ } else {
564
+ console.log('Uncertain blur status - manual review needed');
565
+ }
566
+ ```
567
+
568
+ #### Basic Usage (Capture + Detection)
526
569
 
527
570
  ```typescript
528
571
  // Take a snapshot with blur detection
@@ -562,8 +605,15 @@ const resultWithBlur = await CameraPreview.takeSnapshot({
562
605
  });
563
606
  ```
564
607
 
565
- #### Understanding Blur Scores
608
+ #### Understanding Blur Results
609
+
610
+ **New `detectBlur` Method (Recommended):**
611
+ - Returns standardized **confidence scores** (0.0-1.0 range) across all platforms
612
+ - `blurConfidence`: Higher values indicate more blur (>0.7 = likely blurry)
613
+ - `sharpConfidence`: Higher values indicate more sharpness (>0.8 = likely sharp)
614
+ - `isBlur`: Simple boolean result based on confidence thresholds
566
615
 
616
+ **Legacy `takeSnapshot` Method:**
567
617
  - **Higher values** = Sharper images
568
618
  - **Lower values** = Blurrier images
569
619
  - **Threshold guidelines**:
@@ -571,6 +621,20 @@ const resultWithBlur = await CameraPreview.takeSnapshot({
571
621
  - Android/Web: Consider values below `50-100` as blurry
572
622
  - Adjust thresholds based on your specific quality requirements
573
623
 
624
+ #### When to Use Which Method
625
+
626
+ **Use `detectBlur` for:**
627
+ - Analyzing already captured images
628
+ - Batch processing multiple images
629
+ - Getting detailed confidence scores for advanced decision logic
630
+ - Post-processing workflows
631
+ - When you need consistent confidence values across platforms
632
+
633
+ **Use `takeSnapshot` with `checkBlur: true` for:**
634
+ - Real-time capture with immediate blur feedback
635
+ - Simple blur detection during image capture
636
+ - When you only need a basic blur/sharp indication
637
+
574
638
  #### Performance Impact
575
639
 
576
640
  | Platform | Without Blur Detection | With Blur Detection | Overhead |
@@ -581,8 +645,10 @@ const resultWithBlur = await CameraPreview.takeSnapshot({
581
645
 
582
646
  #### Implementation Notes
583
647
 
584
- - Uses Laplacian variance algorithm across all platforms
585
- - Pixel sampling for performance optimization
648
+ - **TensorFlow Lite models** for advanced blur detection with high accuracy
649
+ - **Laplacian variance fallback** when TFLite models unavailable
650
+ - Pixel sampling for performance optimization
586
651
  - Hardware acceleration on iOS with Core Image
587
652
  - Client-side threshold logic for maximum flexibility
588
653
  - Cross-platform algorithm consistency
654
+ - **Dual API**: `takeSnapshot` for capture+detection, `detectBlur` for analyzing existing images
@@ -293,6 +293,87 @@ public class BlurDetectionHelper {
293
293
  return count > 0 ? variance / count : 0.0;
294
294
  }
295
295
 
296
+ /**
297
+ * Detect blur with detailed confidence scores
298
+ * @param bitmap Input image bitmap
299
+ * @return Map with isBlur, blurConfidence, and sharpConfidence
300
+ */
301
+ public java.util.Map<String, Object> detectBlurWithConfidence(Bitmap bitmap) {
302
+ java.util.Map<String, Object> result = new java.util.HashMap<>();
303
+
304
+ if (!isInitialized || tflite == null) {
305
+ Log.w(TAG, "TFLite model not initialized, falling back to Laplacian");
306
+ double laplacianScore = calculateLaplacianBlurScore(bitmap);
307
+ boolean isBlur = laplacianScore < 150;
308
+ double normalizedScore = Math.max(0.0, Math.min(1.0, laplacianScore / 300.0));
309
+ double sharpConfidence = normalizedScore;
310
+ double blurConfidence = 1.0 - normalizedScore;
311
+
312
+ result.put("isBlur", isBlur);
313
+ result.put("blurConfidence", blurConfidence);
314
+ result.put("sharpConfidence", sharpConfidence);
315
+ return result;
316
+ }
317
+
318
+ try {
319
+ // Use the original bitmap directly (no image enhancement)
320
+ Bitmap processedBitmap = bitmap;
321
+
322
+ // Preprocess image for model (resize and potential enhancement)
323
+ inputImageBuffer.load(processedBitmap);
324
+ inputImageBuffer = imageProcessor.process(inputImageBuffer);
325
+
326
+ // Get tensor buffer
327
+ ByteBuffer tensorBuffer = inputImageBuffer.getBuffer();
328
+
329
+ // Check if we need normalization based on data types
330
+ ByteBuffer inferenceBuffer;
331
+ if (inputImageBuffer.getDataType() == DataType.UINT8 && tflite.getInputTensor(0).dataType() == DataType.FLOAT32) {
332
+ inferenceBuffer = normalizeImageBuffer(tensorBuffer);
333
+ } else if (inputImageBuffer.getDataType() == DataType.FLOAT32) {
334
+ // Check if values are in [0,1] range or [0,255] range
335
+ inferenceBuffer = checkAndNormalizeFloat32Buffer(tensorBuffer);
336
+ } else {
337
+ inferenceBuffer = tensorBuffer;
338
+ }
339
+
340
+ // Run inference
341
+ tflite.run(inferenceBuffer, outputProbabilityBuffer.getBuffer().rewind());
342
+
343
+ // Get output probabilities
344
+ float[] probabilities = outputProbabilityBuffer.getFloatArray();
345
+
346
+ // probabilities[0] = blur probability, probabilities[1] = sharp probability
347
+ double blurConfidence = probabilities.length > 0 ? probabilities[0] : 0.0;
348
+ double sharpConfidence = probabilities.length > 1 ? probabilities[1] : 0.0;
349
+
350
+ // Determine if image is blurry using TFLite confidence
351
+ boolean isBlur = (blurConfidence >= 0.99 || sharpConfidence < 0.1);
352
+
353
+ Log.d(TAG, String.format("TFLite Blur Detection with Confidence - Blur: %.6f, Sharp: %.6f, Label: %s",
354
+ blurConfidence, sharpConfidence, isBlur ? "blur" : "sharp"));
355
+
356
+ result.put("isBlur", isBlur);
357
+ result.put("blurConfidence", blurConfidence);
358
+ result.put("sharpConfidence", sharpConfidence);
359
+ return result;
360
+
361
+ } catch (Exception e) {
362
+ Log.e(TAG, "Error during TFLite inference: " + e.getMessage(), e);
363
+ // Fallback to Laplacian algorithm
364
+ double laplacianScore = calculateLaplacianBlurScore(bitmap);
365
+ boolean isBlur = laplacianScore < 150;
366
+ double normalizedScore = Math.max(0.0, Math.min(1.0, laplacianScore / 300.0));
367
+ double sharpConfidence = normalizedScore;
368
+ double blurConfidence = 1.0 - normalizedScore;
369
+
370
+ result.put("isBlur", isBlur);
371
+ result.put("blurConfidence", blurConfidence);
372
+ result.put("sharpConfidence", sharpConfidence);
373
+ return result;
374
+ }
375
+ }
376
+
296
377
  /**
297
378
  * Check if image is blurry
298
379
  * @param bitmap Input image
@@ -7,6 +7,7 @@ import android.content.pm.PackageManager;
7
7
  import android.content.res.Configuration;
8
8
  import android.content.res.Resources;
9
9
  import android.graphics.Bitmap;
10
+ import android.graphics.BitmapFactory;
10
11
  import android.graphics.Color;
11
12
  import android.graphics.drawable.Drawable;
12
13
  import android.net.Uri;
@@ -1372,6 +1373,60 @@ public class CameraPreviewPlugin extends Plugin {
1372
1373
  }
1373
1374
  }
1374
1375
 
1376
+ @PluginMethod
1377
+ public void detectBlur(PluginCall call) {
1378
+ String imageString = call.getString("image");
1379
+ if (imageString == null) {
1380
+ call.reject("Image parameter is required");
1381
+ return;
1382
+ }
1383
+
1384
+ try {
1385
+ // Convert base64 string to Bitmap
1386
+ String base64String = imageString;
1387
+ if (imageString.startsWith("data:")) {
1388
+ base64String = imageString.substring(imageString.indexOf(",") + 1);
1389
+ }
1390
+
1391
+ byte[] decodedBytes = Base64.decode(base64String, Base64.DEFAULT);
1392
+ Bitmap bitmap = BitmapFactory.decodeByteArray(decodedBytes, 0, decodedBytes.length);
1393
+
1394
+ if (bitmap == null) {
1395
+ call.reject("Invalid image data");
1396
+ return;
1397
+ }
1398
+
1399
+ // Use the new confidence detection method
1400
+ if (blurDetectionHelper != null && blurDetectionHelper.isInitialized()) {
1401
+ java.util.Map<String, Object> result = blurDetectionHelper.detectBlurWithConfidence(bitmap);
1402
+
1403
+ JSObject jsResult = new JSObject();
1404
+ jsResult.put("isBlur", result.get("isBlur"));
1405
+ jsResult.put("blurConfidence", result.get("blurConfidence"));
1406
+ jsResult.put("sharpConfidence", result.get("sharpConfidence"));
1407
+
1408
+ call.resolve(jsResult);
1409
+ } else {
1410
+ // Fallback to Laplacian algorithm with confidence scores
1411
+ double laplacianScore = calculateLaplacianBlurScore(bitmap);
1412
+ boolean isBlur = laplacianScore < 150;
1413
+ double normalizedScore = Math.max(0.0, Math.min(1.0, laplacianScore / 300.0));
1414
+ double sharpConfidence = normalizedScore;
1415
+ double blurConfidence = 1.0 - normalizedScore;
1416
+
1417
+ JSObject result = new JSObject();
1418
+ result.put("isBlur", isBlur);
1419
+ result.put("blurConfidence", blurConfidence);
1420
+ result.put("sharpConfidence", sharpConfidence);
1421
+
1422
+ call.resolve(result);
1423
+ }
1424
+
1425
+ } catch (Exception e) {
1426
+ call.reject("Failed to process image: " + e.getMessage());
1427
+ }
1428
+ }
1429
+
1375
1430
  @PluginMethod
1376
1431
  public void getOrientation(PluginCall call) {
1377
1432
  int orientation = getContext().getResources().getConfiguration().orientation;
package/dist/docs.json CHANGED
@@ -203,6 +203,22 @@
203
203
  "complexTypes": [],
204
204
  "slug": "takesnapshot"
205
205
  },
206
+ {
207
+ "name": "detectBlur",
208
+ "signature": "(options: { image: string; }) => Promise<{ isBlur: boolean; blurConfidence: number; sharpConfidence: number; }>",
209
+ "parameters": [
210
+ {
211
+ "name": "options",
212
+ "docs": "",
213
+ "type": "{ image: string; }"
214
+ }
215
+ ],
216
+ "returns": "Promise<{ isBlur: boolean; blurConfidence: number; sharpConfidence: number; }>",
217
+ "tags": [],
218
+ "docs": "analyze an image for blur detection with detailed confidence scores.",
219
+ "complexTypes": [],
220
+ "slug": "detectblur"
221
+ },
206
222
  {
207
223
  "name": "saveFrame",
208
224
  "signature": "() => Promise<{ success: boolean; }>",
@@ -49,6 +49,16 @@ export interface CameraPreviewPlugin {
49
49
  isBlur?: boolean;
50
50
  }>;
51
51
  /**
52
+ * analyze an image for blur detection with detailed confidence scores.
53
+ */
54
+ detectBlur(options: {
55
+ image: string;
56
+ }): Promise<{
57
+ isBlur: boolean;
58
+ blurConfidence: number;
59
+ sharpConfidence: number;
60
+ }>;
61
+ /**
52
62
  * save a frame internally. Android and iOS only.
53
63
  */
54
64
  saveFrame(): Promise<{
@@ -1 +1 @@
1
- {"version":3,"file":"definitions.js","sourceRoot":"","sources":["../../src/definitions.ts"],"names":[],"mappings":"AAuEA,MAAM,CAAN,IAAY,cAOX;AAPD,WAAY,cAAc;IACxB,yEAAmB,CAAA;IACnB,yEAAmB,CAAA;IACnB,yEAAmB,CAAA;IACnB,2EAAoB,CAAA;IACpB,qEAAiB,CAAA;IACjB,qEAAiB,CAAA;AACnB,CAAC,EAPW,cAAc,KAAd,cAAc,QAOzB","sourcesContent":["import { PluginListenerHandle } from \"@capacitor/core\";\n\nexport interface CameraPreviewPlugin {\n initialize(options?: { quality?: number }): Promise<void>;\n getResolution(): Promise<{resolution: string}>;\n setResolution(options: {resolution: number}): Promise<void>;\n getAllCameras(): Promise<{cameras: string[]}>;\n getSelectedCamera(): Promise<{selectedCamera: string}>;\n selectCamera(options: {cameraID: string; }): Promise<void>;\n setScanRegion(options: {region:ScanRegion}): Promise<void>;\n setZoom(options: {factor: number}): Promise<void>;\n setFocus(options: {x: number, y: number}): Promise<void>;\n /**\n * Web Only\n */\n setDefaultUIElementURL(url:string): Promise<void>;\n /**\n * Web Only\n */\n setElement(ele:HTMLElement): Promise<void>;\n startCamera(): Promise<void>;\n stopCamera(): Promise<void>;\n /**\n * take a snapshot as base64.\n */\n takeSnapshot(options:{quality?:number, checkBlur?:boolean}): Promise<{base64:string, isBlur?: boolean}>;\n /**\n * save a frame internally. Android and iOS only.\n */\n saveFrame(): Promise<{success:boolean}>;\n /**\n * take a snapshot on to a canvas. Web Only\n */\n takeSnapshot2(options:{canvas:HTMLCanvasElement,maxLength?:number}): Promise<{scaleRatio?:number}>;\n takePhoto(options: {pathToSave?:string,includeBase64?: boolean}): Promise<{path?:string,base64?:string,blob?:Blob, isBlur?: boolean}>;\n toggleTorch(options: {on: boolean}): Promise<void>;\n /**\n * get the orientation of the device.\n */\n getOrientation(): Promise<{\"orientation\":\"PORTRAIT\"|\"LANDSCAPE\"}>;\n startRecording(): Promise<void>;\n stopRecording(options:{includeBase64?:boolean}): Promise<{path?:string,base64?:string,blob?:Blob}>;\n setLayout(options: {top: string, left:string, width:string, height:string}): Promise<void>;\n requestCameraPermission(): Promise<void>;\n requestMicroPhonePermission(): Promise<void>;\n isOpen():Promise<{isOpen:boolean}>;\n addListener(\n eventName: 'onPlayed',\n listenerFunc: onPlayedListener,\n ): Promise<PluginListenerHandle>;\n addListener(\n eventName: 'onOrientationChanged',\n listenerFunc: onOrientationChangedListener,\n ): Promise<PluginListenerHandle>;\n removeAllListeners(): Promise<void>;\n}\n\nexport type onPlayedListener = (result:{resolution:string}) => void;\nexport type onOrientationChangedListener = () => void;\n\n/**\n * measuredByPercentage: 0 in pixel, 1 in percent\n */\nexport interface ScanRegion{\n left: number;\n top: number;\n right: number;\n bottom: number;\n measuredByPercentage: number;\n}\n\nexport enum EnumResolution {\n RESOLUTION_AUTO = 0,\n RESOLUTION_480P = 1,\n RESOLUTION_720P = 2,\n RESOLUTION_1080P = 3,\n RESOLUTION_2K = 4,\n RESOLUTION_4K = 5\n}\n"]}
1
+ {"version":3,"file":"definitions.js","sourceRoot":"","sources":["../../src/definitions.ts"],"names":[],"mappings":"AA2EA,MAAM,CAAN,IAAY,cAOX;AAPD,WAAY,cAAc;IACxB,yEAAmB,CAAA;IACnB,yEAAmB,CAAA;IACnB,yEAAmB,CAAA;IACnB,2EAAoB,CAAA;IACpB,qEAAiB,CAAA;IACjB,qEAAiB,CAAA;AACnB,CAAC,EAPW,cAAc,KAAd,cAAc,QAOzB","sourcesContent":["import { PluginListenerHandle } from \"@capacitor/core\";\n\nexport interface CameraPreviewPlugin {\n initialize(options?: { quality?: number }): Promise<void>;\n getResolution(): Promise<{resolution: string}>;\n setResolution(options: {resolution: number}): Promise<void>;\n getAllCameras(): Promise<{cameras: string[]}>;\n getSelectedCamera(): Promise<{selectedCamera: string}>;\n selectCamera(options: {cameraID: string; }): Promise<void>;\n setScanRegion(options: {region:ScanRegion}): Promise<void>;\n setZoom(options: {factor: number}): Promise<void>;\n setFocus(options: {x: number, y: number}): Promise<void>;\n /**\n * Web Only\n */\n setDefaultUIElementURL(url:string): Promise<void>;\n /**\n * Web Only\n */\n setElement(ele:HTMLElement): Promise<void>;\n startCamera(): Promise<void>;\n stopCamera(): Promise<void>;\n /**\n * take a snapshot as base64.\n */\n takeSnapshot(options:{quality?:number, checkBlur?:boolean}): Promise<{base64:string, isBlur?: boolean}>;\n /**\n * analyze an image for blur detection with detailed confidence scores.\n */\n detectBlur(options:{image: string}): Promise<{isBlur: boolean, blurConfidence: number, sharpConfidence: number}>;\n /**\n * save a frame internally. Android and iOS only.\n */\n saveFrame(): Promise<{success:boolean}>;\n /**\n * take a snapshot on to a canvas. Web Only\n */\n takeSnapshot2(options:{canvas:HTMLCanvasElement,maxLength?:number}): Promise<{scaleRatio?:number}>;\n takePhoto(options: {pathToSave?:string,includeBase64?: boolean}): Promise<{path?:string,base64?:string,blob?:Blob, isBlur?: boolean}>;\n toggleTorch(options: {on: boolean}): Promise<void>;\n /**\n * get the orientation of the device.\n */\n getOrientation(): Promise<{\"orientation\":\"PORTRAIT\"|\"LANDSCAPE\"}>;\n startRecording(): Promise<void>;\n stopRecording(options:{includeBase64?:boolean}): Promise<{path?:string,base64?:string,blob?:Blob}>;\n setLayout(options: {top: string, left:string, width:string, height:string}): Promise<void>;\n requestCameraPermission(): Promise<void>;\n requestMicroPhonePermission(): Promise<void>;\n isOpen():Promise<{isOpen:boolean}>;\n addListener(\n eventName: 'onPlayed',\n listenerFunc: onPlayedListener,\n ): Promise<PluginListenerHandle>;\n addListener(\n eventName: 'onOrientationChanged',\n listenerFunc: onOrientationChangedListener,\n ): Promise<PluginListenerHandle>;\n removeAllListeners(): Promise<void>;\n}\n\nexport type onPlayedListener = (result:{resolution:string}) => void;\nexport type onOrientationChangedListener = () => void;\n\n/**\n * measuredByPercentage: 0 in pixel, 1 in percent\n */\nexport interface ScanRegion{\n left: number;\n top: number;\n right: number;\n bottom: number;\n measuredByPercentage: number;\n}\n\nexport enum EnumResolution {\n RESOLUTION_AUTO = 0,\n RESOLUTION_480P = 1,\n RESOLUTION_720P = 2,\n RESOLUTION_1080P = 3,\n RESOLUTION_2K = 4,\n RESOLUTION_4K = 5\n}\n"]}
package/dist/esm/web.d.ts CHANGED
@@ -60,6 +60,13 @@ export declare class CameraPreviewWeb extends WebPlugin implements CameraPreview
60
60
  base64: string;
61
61
  isBlur?: boolean;
62
62
  }>;
63
+ detectBlur(options: {
64
+ image: string;
65
+ }): Promise<{
66
+ isBlur: boolean;
67
+ blurConfidence: number;
68
+ sharpConfidence: number;
69
+ }>;
63
70
  takeSnapshot2(options: {
64
71
  canvas: HTMLCanvasElement;
65
72
  maxLength?: number;
package/dist/esm/web.js CHANGED
@@ -267,6 +267,58 @@ export class CameraPreviewWeb extends WebPlugin {
267
267
  throw new Error('Camera not initialized');
268
268
  }
269
269
  }
270
+ async detectBlur(options) {
271
+ try {
272
+ // Create image element from base64
273
+ const img = new Image();
274
+ return new Promise((resolve, reject) => {
275
+ img.onload = () => {
276
+ try {
277
+ // Create canvas to extract image data
278
+ const canvas = document.createElement('canvas');
279
+ const ctx = canvas.getContext('2d');
280
+ if (!ctx) {
281
+ reject(new Error('Cannot create canvas context'));
282
+ return;
283
+ }
284
+ canvas.width = img.width;
285
+ canvas.height = img.height;
286
+ ctx.drawImage(img, 0, 0);
287
+ // Get image data
288
+ const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);
289
+ const blurResult = this.detectBlurWeb(imageData, canvas.width, canvas.height);
290
+ // Convert blur score to confidence values
291
+ // Higher blurScore means sharper image
292
+ const normalizedScore = Math.max(0, Math.min(1, blurResult.blurScore / 300)); // Normalize to 0-1
293
+ const sharpConfidence = normalizedScore;
294
+ const blurConfidence = 1 - normalizedScore;
295
+ const isBlur = blurResult.blurScore < 150;
296
+ resolve({
297
+ isBlur: isBlur,
298
+ blurConfidence: blurConfidence,
299
+ sharpConfidence: sharpConfidence
300
+ });
301
+ }
302
+ catch (error) {
303
+ reject(error);
304
+ }
305
+ };
306
+ img.onerror = () => {
307
+ reject(new Error('Failed to load image'));
308
+ };
309
+ // Handle both data URLs and base64 strings
310
+ if (options.image.startsWith('data:')) {
311
+ img.src = options.image;
312
+ }
313
+ else {
314
+ img.src = `data:image/jpeg;base64,${options.image}`;
315
+ }
316
+ });
317
+ }
318
+ catch (error) {
319
+ throw new Error(`Failed to detect blur: ${error}`);
320
+ }
321
+ }
270
322
  async takeSnapshot2(options) {
271
323
  if (this.camera) {
272
324
  let canvas = options.canvas;
@@ -1 +1 @@
1
- {"version":3,"file":"web.js","sourceRoot":"","sources":["../../src/web.ts"],"names":[],"mappings":"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AAG,CAAC,CAAC;oBAC7H,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBACzI,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC7H,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC7H,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBACzI,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC7H,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;iBAC1I,CAAC;gBAEF,8DAA8D;gBAC9D,MAAM,SAAS,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC;oBAC5C,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC;oBACvC,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;gBAE7D,QAAQ,IAAI,SAAS,GAAG,SAAS,CAAC;gBAClC,KAAK,EAAE,CAAC;aACT;SACF;QAED,MAAM,SAAS,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,QAAQ,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QAEnD,OAAO,EAAE,SAAS,EAAE,CAAC;IACvB,CAAC;CACF","sourcesContent":["import { WebPlugin } from '@capacitor/core';\nimport { CameraEnhancer, PlayCallbackInfo } from 'dynamsoft-camera-enhancer';\nimport { CameraPreviewPlugin, EnumResolution, ScanRegion } from './definitions';\nimport RecordRTC from 'recordrtc';\nCameraEnhancer.defaultUIElementURL = \"https://cdn.jsdelivr.net/npm/dynamsoft-camera-enhancer@3.3.9/dist/dce.ui.html\";\n\nexport class CameraPreviewWeb extends WebPlugin implements CameraPreviewPlugin {\n private camera:CameraEnhancer | undefined;\n private container:HTMLElement | undefined;\n private region:ScanRegion | undefined;\n private recorder:RecordRTC | null | undefined;\n private hasMicrophone: boolean = false;\n async setDefaultUIElementURL(url: string): Promise<void> {\n CameraEnhancer.defaultUIElementURL = url;\n }\n\n async setElement(ele: HTMLElement): Promise<void> {\n this.container = ele;\n }\n\n saveFrame(): Promise<{ success: boolean; }> {\n throw new Error('Method not implemented.');\n }\n\n async getOrientation(): Promise<{\"orientation\":\"PORTRAIT\"|\"LANDSCAPE\"}> {\n let portrait = window.matchMedia(\"(orientation: portrait)\");\n if (portrait.matches) {\n return {orientation:\"PORTRAIT\"};\n }else{\n return {orientation:\"LANDSCAPE\"};\n }\n }\n\n private desiredJpegQuality: number = 0.95; // Default to high quality (0.0-1.0)\n\n async initialize(options?: { quality?: number }): Promise<void> {\n // Get quality parameter from initialization, default to 95% if not specified\n if (options?.quality !== undefined) {\n this.desiredJpegQuality = Math.max(1, Math.min(100, options.quality)) / 100.0;\n console.log(`Camera initialized with JPEG quality: ${this.desiredJpegQuality}`);\n }\n \n this.camera = await CameraEnhancer.createInstance();\n this.camera.on(\"played\", (playCallBackInfo:PlayCallbackInfo) => {\n this.notifyListeners(\"onPlayed\", {resolution:playCallBackInfo.width+\"x\"+playCallBackInfo.height});\n try {\n let canvas = this.camera!.getUIElement().getElementsByClassName(\"dce-video-container\")[0].getElementsByTagName(\"canvas\")[0];\n if (canvas) {\n canvas.remove();\n }\n } catch (error) {\n console.log(error);\n }\n });\n if (this.container) {\n await this.camera.setUIElement(this.container);\n }else{\n await this.camera.setUIElement(CameraEnhancer.defaultUIElementURL);\n this.camera.getUIElement().getElementsByClassName(\"dce-btn-close\")[0].remove();\n this.camera.getUIElement().getElementsByClassName(\"dce-sel-camera\")[0].remove();\n this.camera.getUIElement().getElementsByClassName(\"dce-sel-resolution\")[0].remove();\n this.camera.getUIElement().getElementsByClassName(\"dce-msg-poweredby\")[0].remove();\n }\n this.camera.setVideoFit(\"cover\");\n let portrait = window.matchMedia(\"(orientation: portrait)\");\n portrait.addEventListener(\"change\", () => {\n this.notifyListeners(\"onOrientationChanged\", null);\n })\n }\n\n async getResolution(): Promise<{ resolution: string; }> {\n if (this.camera) {\n let rsl = this.camera.getResolution();\n let resolution:string = rsl[0] + \"x\" + rsl[1];\n return {resolution: resolution};\n }else{\n throw new Error('Camera not initialized');\n }\n }\n\n async setResolution(options: { resolution: number; }): Promise<void> {\n if (this.camera) {\n let res = options.resolution;\n let width = 1280;\n let height = 720;\n if (res == EnumResolution.RESOLUTION_480P){\n width = 640;\n height = 480;\n } else if (res == EnumResolution.RESOLUTION_720P){\n width = 1280;\n height = 720;\n } else if (res == EnumResolution.RESOLUTION_1080P){\n width = 1920;\n height = 1080;\n } else if (res == EnumResolution.RESOLUTION_2K){\n width = 2560;\n height = 1440;\n } else if (res == EnumResolution.RESOLUTION_4K){\n width = 3840;\n height = 2160;\n }\n await this.camera.setResolution(width,height);\n return;\n } else {\n throw new Error('Camera not initialized');\n }\n }\n\n async getAllCameras(): Promise<{ cameras: string[]; }> {\n if (this.camera) {\n let cameras = await this.camera.getAllCameras();\n let labels:string[] = [];\n cameras.forEach(camera => {\n labels.push(camera.label);\n });\n return {cameras:labels};\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async getSelectedCamera(): Promise<{ selectedCamera: string; }> {\n if (this.camera) {\n let cameraInfo = this.camera.getSelectedCamera();\n return {selectedCamera:cameraInfo.label};\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async selectCamera(options: { cameraID: string; }): Promise<void> {\n if (this.camera) {\n let cameras = await this.camera.getAllCameras()\n for (let index = 0; index < cameras.length; index++) {\n const camera = cameras[index];\n if (camera.label === options.cameraID) {\n await this.camera.selectCamera(camera);\n return;\n }\n }\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async setScanRegion(options: { region: ScanRegion; }): Promise<void> {\n if (this.camera){\n this.region = options.region;\n this.applyScanRegion();\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n applyScanRegion(){\n if (this.camera && this.region){\n this.camera.setScanRegion({\n regionLeft:this.region.left,\n regionTop:this.region.top,\n regionRight:this.region.right,\n regionBottom:this.region.bottom,\n regionMeasuredByPercentage: this.region.measuredByPercentage\n });\n }\n }\n\n async setZoom(options: { factor: number; }): Promise<void> {\n if (this.camera) {\n await this.camera.setZoom(options.factor);\n return;\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async setFocus(options: {x: number, y: number}): Promise<void> {\n if (this.camera) {\n this.camera.setFocus({\n mode:\"manual\",\n area:{\n centerPoint:{x:options.x.toString(),y:options.y.toString()}\n }\n });\n }else{\n throw new Error('Camera not initialized');\n }\n }\n\n async toggleTorch(options: { on: boolean; }): Promise<void> {\n if (this.camera) {\n try{\n if (options[\"on\"]){\n await this.camera.turnOnTorch();\n }else{\n await this.camera.turnOffTorch();\n }\n } catch (e){\n throw new Error(\"Torch unsupported\");\n }\n }\n }\n\n async startCamera(): Promise<void> {\n if (this.camera) {\n await this.camera?.open(true);\n if (this.container && this.isSafari() === true) {\n const resetZoom = async () => {\n await this.camera?.setZoom(1.001);\n await this.camera?.setZoom(1.0);\n }\n setTimeout(resetZoom,500);\n }\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n isSafari():boolean{\n const u = navigator.userAgent.toLowerCase()\n if (u.indexOf('safari') > -1 && u.indexOf('chrome') === -1) {\n return true;\n }else{\n return false;\n }\n }\n\n async stopCamera(): Promise<void> {\n if (this.camera) {\n this.camera.close(true);\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async isOpen(): Promise<{isOpen:boolean}> {\n if (this.camera) {\n return {isOpen:this.camera.isOpen()};\n }else{\n throw new Error('Camera not initialized');\n }\n }\n\n async takeSnapshot(options:{quality?:number, checkBlur?:boolean}): Promise<{ base64: string, isBlur?: boolean }> {\n if (this.camera) {\n let desiredQuality = this.desiredJpegQuality;\n if (options?.quality !== undefined) {\n desiredQuality = Math.max(1, Math.min(100, options.quality)) / 100.0;\n }\n let canvas = this.camera.getFrame().toCanvas();\n let dataURL = canvas.toDataURL('image/jpeg', desiredQuality);\n let base64 = dataURL.replace(\"data:image/jpeg;base64,\",\"\");\n \n // Only perform blur detection if checkBlur option is true\n if (options?.checkBlur === true) {\n const ctx = canvas.getContext('2d');\n if (ctx) {\n const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);\n const blurResult = this.detectBlurWeb(imageData, canvas.width, canvas.height);\n return {\n base64: base64,\n isBlur: blurResult.blurScore < 150\n };\n }\n }\n \n return {base64: base64};\n }else{\n throw new Error('Camera not initialized');\n }\n }\n\n async takeSnapshot2(options:{canvas:HTMLCanvasElement,maxLength?:number}): Promise<{scaleRatio?:number}> {\n if (this.camera) {\n let canvas = options.canvas;\n let scaleRatio = 1.0;\n if (options && options.maxLength) {\n let res = (await this.getResolution()).resolution;\n let width = parseInt(res.split(\"x\")[0]);\n let height = parseInt(res.split(\"x\")[1]);\n let targetWidth = width;\n let targetHeight = height;\n if (width > options.maxLength || height > options.maxLength) {\n if (width > height) {\n targetWidth = options.maxLength;\n targetHeight = options.maxLength/width*height;\n scaleRatio = options.maxLength/width;\n }else{\n targetHeight = options.maxLength;\n targetWidth = options.maxLength/height*width;\n scaleRatio = options.maxLength/height;\n }\n canvas.width = targetWidth;\n canvas.height = targetHeight;\n let video = this.camera.getUIElement().getElementsByTagName(\"video\")[0];\n let ctx = canvas.getContext('2d');\n if (ctx) {\n ctx.drawImage(video, 0, 0, targetWidth, targetHeight);\n }\n }else{\n this.drawFullFrame(canvas);\n }\n }else{\n this.drawFullFrame(canvas);\n }\n return {scaleRatio:scaleRatio};\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n drawFullFrame(canvas:HTMLCanvasElement):HTMLCanvasElement{\n let video = this.camera?.getUIElement().getElementsByTagName(\"video\")[0];\n if (video) {\n canvas.width = video.videoWidth;\n canvas.height = video.videoHeight;\n let ctx = canvas.getContext('2d');\n if (ctx) {\n ctx.drawImage(video, 0, 0, canvas.width, canvas.height);\n }\n }\n return canvas;\n }\n\n async takePhoto(_options:{includeBase64?:boolean}): Promise<{ path?:string, base64?: string, blob?:Blob, blurScore?: number }> {\n if (this.camera) {\n let video = this.camera.getUIElement().getElementsByTagName(\"video\")[0];\n let localStream:MediaStream = video.srcObject as MediaStream;\n if (localStream) {\n if (\"ImageCapture\" in window) {\n try {\n //@ts-ignore \n let ImageCapture:any = window[\"ImageCapture\"];\n console.log(\"ImageCapture supported\");\n const track = localStream.getVideoTracks()[0];\n let imageCapture = new ImageCapture(track);\n let blob = await imageCapture.takePhoto();\n return {blob:blob}; \n } catch (error) {\n console.log(error);\n }\n }else{\n console.log(\"ImageCapture not supported\");\n }\n }\n this.camera.setScanRegion(\n {\n regionLeft: 0,\n regionTop: 0,\n regionBottom: 100,\n regionRight: 100,\n regionMeasuredByPercentage: 1\n }\n )\n let canvas = this.camera.getFrame().toCanvas();\n let base64 = this.removeDataURLHead(canvas.toDataURL(\"image/jpeg\", this.desiredJpegQuality));\n this.applyScanRegion();\n \n // Add blur detection if base64 is included\n if (_options?.includeBase64) {\n const ctx = canvas.getContext('2d');\n if (ctx) {\n const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);\n const blurResult = this.detectBlurWeb(imageData, canvas.width, canvas.height);\n return {\n base64: base64,\n blurScore: blurResult.blurScore\n };\n }\n }\n \n return {base64:base64};\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n removeDataURLHead(dataURL:string){\n return dataURL.substring(dataURL.indexOf(\",\")+1,dataURL.length);\n }\n\n\n async requestCameraPermission(): Promise<void> {\n const constraints = {video: true, audio: false};\n const stream = await navigator.mediaDevices.getUserMedia(constraints);\n const tracks = stream.getTracks();\n for (let i=0;i<tracks.length;i++) {\n const track = tracks[i];\n track.stop(); // stop the opened tracks\n }\n }\n\n async requestMicroPhonePermission(): Promise<void> {\n try {\n const constraints = {video: false, audio: true};\n const stream = await navigator.mediaDevices.getUserMedia(constraints);\n const tracks = stream.getTracks();\n for (let i=0;i<tracks.length;i++) {\n const track = tracks[i];\n track.stop(); // stop the opened tracks\n }\n this.hasMicrophone = true;\n } catch (error) {\n this.hasMicrophone = false;\n throw error;\n }\n }\n\n async startRecording(): Promise<void> {\n if (this.camera) {\n if (this.hasMicrophone) {\n let settings = this.camera.getVideoSettings();\n settings.audio = true;\n await this.camera.updateVideoSettings(settings);\n }\n let video = this.camera.getUIElement().getElementsByTagName(\"video\")[0];\n this.recorder = new RecordRTC(video.srcObject as MediaStream, {\n type: 'video'\n });\n this.recorder.startRecording();\n }else{\n throw new Error(\"camera not initialized\");\n }\n }\n\n async stopRecording(): Promise<{blob:Blob}> {\n return new Promise<{blob:Blob}>((resolve, reject) => {\n if (this.recorder) {\n const stopRecordingCallback = () => {\n let blob = this.recorder!.getBlob();\n this.recorder!.destroy();\n this.recorder = null;\n resolve({blob})\n }\n this.recorder.stopRecording(stopRecordingCallback);\n }else{\n reject(\"recorder not initialized\");\n }\n }) \n }\n\n async setLayout(options: {top: string,left:string,width:string, height:string}): Promise<void> {\n if (this.camera) {\n let ele = this.camera.getUIElement();\n if (options.top) {\n ele.style.top = options.top;\n }\n if (options.left) {\n ele.style.left = options.left;\n }\n if (options.width) {\n ele.style.width = options.width;\n }\n if (options.height) {\n ele.style.height = options.height;\n }\n ele.style.position = \"absolute\";\n }else{\n throw new Error(\"Camera not initialized\");\n }\n }\n\n /**\n * Detect blur using Laplacian variance algorithm (same as iOS/Android)\n * Higher values indicate sharper images\n */\n private detectBlurWeb(imageData: ImageData, width: number, height: number): { blurScore: number } {\n const data = imageData.data;\n let variance = 0;\n let count = 0;\n \n // Sample every 4th pixel for performance (similar to Android implementation)\n const step = 4;\n for (let y = step; y < height - step; y += step) {\n for (let x = step; x < width - step; x += step) {\n const idx = (y * width + x) * 4;\n \n // Convert to grayscale using same formula as Android\n const gray = 0.299 * data[idx] + 0.587 * data[idx + 1] + 0.114 * data[idx + 2];\n \n // Calculate neighbors for 3x3 Laplacian kernel\n const neighbors = [\n 0.299 * data[((y-1) * width + (x-1)) * 4] + 0.587 * data[((y-1) * width + (x-1)) * 4 + 1] + 0.114 * data[((y-1) * width + (x-1)) * 4 + 2],\n 0.299 * data[((y-1) * width + x) * 4] + 0.587 * data[((y-1) * width + x) * 4 + 1] + 0.114 * data[((y-1) * width + x) * 4 + 2],\n 0.299 * data[((y-1) * width + (x+1)) * 4] + 0.587 * data[((y-1) * width + (x+1)) * 4 + 1] + 0.114 * data[((y-1) * width + (x+1)) * 4 + 2],\n 0.299 * data[(y * width + (x-1)) * 4] + 0.587 * data[(y * width + (x-1)) * 4 + 1] + 0.114 * data[(y * width + (x-1)) * 4 + 2],\n 0.299 * data[(y * width + (x+1)) * 4] + 0.587 * data[(y * width + (x+1)) * 4 + 1] + 0.114 * data[(y * width + (x+1)) * 4 + 2],\n 0.299 * data[((y+1) * width + (x-1)) * 4] + 0.587 * data[((y+1) * width + (x-1)) * 4 + 1] + 0.114 * data[((y+1) * width + (x-1)) * 4 + 2],\n 0.299 * data[((y+1) * width + x) * 4] + 0.587 * data[((y+1) * width + x) * 4 + 1] + 0.114 * data[((y+1) * width + x) * 4 + 2],\n 0.299 * data[((y+1) * width + (x+1)) * 4] + 0.587 * data[((y+1) * width + (x+1)) * 4 + 1] + 0.114 * data[((y+1) * width + (x+1)) * 4 + 2]\n ];\n \n // Apply 3x3 Laplacian kernel (matches Android implementation)\n const laplacian = -neighbors[0] - neighbors[1] - neighbors[2] +\n -neighbors[3] + 8 * gray - neighbors[4] +\n -neighbors[5] - neighbors[6] - neighbors[7];\n \n variance += laplacian * laplacian;\n count++;\n }\n }\n \n const blurScore = count > 0 ? variance / count : 0;\n \n return { blurScore };\n }\n}\n"]}
1
+ {"version":3,"file":"web.js","sourceRoot":"","sources":["../../src/web.ts"],"names":[],"mappings":"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AK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBACzI,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC7H,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBACzI,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC7H,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC7H,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBACzI,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBAC7H,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,GAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,GAAC,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;iBAC1I,CAAC;gBAEF,8DAA8D;gBAC9D,MAAM,SAAS,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC;oBAC5C,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC;oBACvC,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;gBAE7D,QAAQ,IAAI,SAAS,GAAG,SAAS,CAAC;gBAClC,KAAK,EAAE,CAAC;aACT;SACF;QAED,MAAM,SAAS,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,QAAQ,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QAEnD,OAAO,EAAE,SAAS,EAAE,CAAC;IACvB,CAAC;CACF","sourcesContent":["import { WebPlugin } from '@capacitor/core';\nimport { CameraEnhancer, PlayCallbackInfo } from 'dynamsoft-camera-enhancer';\nimport { CameraPreviewPlugin, EnumResolution, ScanRegion } from './definitions';\nimport RecordRTC from 'recordrtc';\nCameraEnhancer.defaultUIElementURL = \"https://cdn.jsdelivr.net/npm/dynamsoft-camera-enhancer@3.3.9/dist/dce.ui.html\";\n\nexport class CameraPreviewWeb extends WebPlugin implements CameraPreviewPlugin {\n private camera:CameraEnhancer | undefined;\n private container:HTMLElement | undefined;\n private region:ScanRegion | undefined;\n private recorder:RecordRTC | null | undefined;\n private hasMicrophone: boolean = false;\n async setDefaultUIElementURL(url: string): Promise<void> {\n CameraEnhancer.defaultUIElementURL = url;\n }\n\n async setElement(ele: HTMLElement): Promise<void> {\n this.container = ele;\n }\n\n saveFrame(): Promise<{ success: boolean; }> {\n throw new Error('Method not implemented.');\n }\n\n async getOrientation(): Promise<{\"orientation\":\"PORTRAIT\"|\"LANDSCAPE\"}> {\n let portrait = window.matchMedia(\"(orientation: portrait)\");\n if (portrait.matches) {\n return {orientation:\"PORTRAIT\"};\n }else{\n return {orientation:\"LANDSCAPE\"};\n }\n }\n\n private desiredJpegQuality: number = 0.95; // Default to high quality (0.0-1.0)\n\n async initialize(options?: { quality?: number }): Promise<void> {\n // Get quality parameter from initialization, default to 95% if not specified\n if (options?.quality !== undefined) {\n this.desiredJpegQuality = Math.max(1, Math.min(100, options.quality)) / 100.0;\n console.log(`Camera initialized with JPEG quality: ${this.desiredJpegQuality}`);\n }\n \n this.camera = await CameraEnhancer.createInstance();\n this.camera.on(\"played\", (playCallBackInfo:PlayCallbackInfo) => {\n this.notifyListeners(\"onPlayed\", {resolution:playCallBackInfo.width+\"x\"+playCallBackInfo.height});\n try {\n let canvas = this.camera!.getUIElement().getElementsByClassName(\"dce-video-container\")[0].getElementsByTagName(\"canvas\")[0];\n if (canvas) {\n canvas.remove();\n }\n } catch (error) {\n console.log(error);\n }\n });\n if (this.container) {\n await this.camera.setUIElement(this.container);\n }else{\n await this.camera.setUIElement(CameraEnhancer.defaultUIElementURL);\n this.camera.getUIElement().getElementsByClassName(\"dce-btn-close\")[0].remove();\n this.camera.getUIElement().getElementsByClassName(\"dce-sel-camera\")[0].remove();\n this.camera.getUIElement().getElementsByClassName(\"dce-sel-resolution\")[0].remove();\n this.camera.getUIElement().getElementsByClassName(\"dce-msg-poweredby\")[0].remove();\n }\n this.camera.setVideoFit(\"cover\");\n let portrait = window.matchMedia(\"(orientation: portrait)\");\n portrait.addEventListener(\"change\", () => {\n this.notifyListeners(\"onOrientationChanged\", null);\n })\n }\n\n async getResolution(): Promise<{ resolution: string; }> {\n if (this.camera) {\n let rsl = this.camera.getResolution();\n let resolution:string = rsl[0] + \"x\" + rsl[1];\n return {resolution: resolution};\n }else{\n throw new Error('Camera not initialized');\n }\n }\n\n async setResolution(options: { resolution: number; }): Promise<void> {\n if (this.camera) {\n let res = options.resolution;\n let width = 1280;\n let height = 720;\n if (res == EnumResolution.RESOLUTION_480P){\n width = 640;\n height = 480;\n } else if (res == EnumResolution.RESOLUTION_720P){\n width = 1280;\n height = 720;\n } else if (res == EnumResolution.RESOLUTION_1080P){\n width = 1920;\n height = 1080;\n } else if (res == EnumResolution.RESOLUTION_2K){\n width = 2560;\n height = 1440;\n } else if (res == EnumResolution.RESOLUTION_4K){\n width = 3840;\n height = 2160;\n }\n await this.camera.setResolution(width,height);\n return;\n } else {\n throw new Error('Camera not initialized');\n }\n }\n\n async getAllCameras(): Promise<{ cameras: string[]; }> {\n if (this.camera) {\n let cameras = await this.camera.getAllCameras();\n let labels:string[] = [];\n cameras.forEach(camera => {\n labels.push(camera.label);\n });\n return {cameras:labels};\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async getSelectedCamera(): Promise<{ selectedCamera: string; }> {\n if (this.camera) {\n let cameraInfo = this.camera.getSelectedCamera();\n return {selectedCamera:cameraInfo.label};\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async selectCamera(options: { cameraID: string; }): Promise<void> {\n if (this.camera) {\n let cameras = await this.camera.getAllCameras()\n for (let index = 0; index < cameras.length; index++) {\n const camera = cameras[index];\n if (camera.label === options.cameraID) {\n await this.camera.selectCamera(camera);\n return;\n }\n }\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async setScanRegion(options: { region: ScanRegion; }): Promise<void> {\n if (this.camera){\n this.region = options.region;\n this.applyScanRegion();\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n applyScanRegion(){\n if (this.camera && this.region){\n this.camera.setScanRegion({\n regionLeft:this.region.left,\n regionTop:this.region.top,\n regionRight:this.region.right,\n regionBottom:this.region.bottom,\n regionMeasuredByPercentage: this.region.measuredByPercentage\n });\n }\n }\n\n async setZoom(options: { factor: number; }): Promise<void> {\n if (this.camera) {\n await this.camera.setZoom(options.factor);\n return;\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async setFocus(options: {x: number, y: number}): Promise<void> {\n if (this.camera) {\n this.camera.setFocus({\n mode:\"manual\",\n area:{\n centerPoint:{x:options.x.toString(),y:options.y.toString()}\n }\n });\n }else{\n throw new Error('Camera not initialized');\n }\n }\n\n async toggleTorch(options: { on: boolean; }): Promise<void> {\n if (this.camera) {\n try{\n if (options[\"on\"]){\n await this.camera.turnOnTorch();\n }else{\n await this.camera.turnOffTorch();\n }\n } catch (e){\n throw new Error(\"Torch unsupported\");\n }\n }\n }\n\n async startCamera(): Promise<void> {\n if (this.camera) {\n await this.camera?.open(true);\n if (this.container && this.isSafari() === true) {\n const resetZoom = async () => {\n await this.camera?.setZoom(1.001);\n await this.camera?.setZoom(1.0);\n }\n setTimeout(resetZoom,500);\n }\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n isSafari():boolean{\n const u = navigator.userAgent.toLowerCase()\n if (u.indexOf('safari') > -1 && u.indexOf('chrome') === -1) {\n return true;\n }else{\n return false;\n }\n }\n\n async stopCamera(): Promise<void> {\n if (this.camera) {\n this.camera.close(true);\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n async isOpen(): Promise<{isOpen:boolean}> {\n if (this.camera) {\n return {isOpen:this.camera.isOpen()};\n }else{\n throw new Error('Camera not initialized');\n }\n }\n\n async takeSnapshot(options:{quality?:number, checkBlur?:boolean}): Promise<{ base64: string, isBlur?: boolean }> {\n if (this.camera) {\n let desiredQuality = this.desiredJpegQuality;\n if (options?.quality !== undefined) {\n desiredQuality = Math.max(1, Math.min(100, options.quality)) / 100.0;\n }\n let canvas = this.camera.getFrame().toCanvas();\n let dataURL = canvas.toDataURL('image/jpeg', desiredQuality);\n let base64 = dataURL.replace(\"data:image/jpeg;base64,\",\"\");\n \n // Only perform blur detection if checkBlur option is true\n if (options?.checkBlur === true) {\n const ctx = canvas.getContext('2d');\n if (ctx) {\n const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);\n const blurResult = this.detectBlurWeb(imageData, canvas.width, canvas.height);\n return {\n base64: base64,\n isBlur: blurResult.blurScore < 150\n };\n }\n }\n \n return {base64: base64};\n }else{\n throw new Error('Camera not initialized');\n }\n }\n\n async detectBlur(options: {image: string}): Promise<{isBlur: boolean, blurConfidence: number, sharpConfidence: number}> {\n try {\n // Create image element from base64\n const img = new Image();\n \n return new Promise((resolve, reject) => {\n img.onload = () => {\n try {\n // Create canvas to extract image data\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n \n if (!ctx) {\n reject(new Error('Cannot create canvas context'));\n return;\n }\n \n canvas.width = img.width;\n canvas.height = img.height;\n ctx.drawImage(img, 0, 0);\n \n // Get image data\n const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);\n const blurResult = this.detectBlurWeb(imageData, canvas.width, canvas.height);\n \n // Convert blur score to confidence values\n // Higher blurScore means sharper image\n const normalizedScore = Math.max(0, Math.min(1, blurResult.blurScore / 300)); // Normalize to 0-1\n const sharpConfidence = normalizedScore;\n const blurConfidence = 1 - normalizedScore;\n const isBlur = blurResult.blurScore < 150;\n \n resolve({\n isBlur: isBlur,\n blurConfidence: blurConfidence,\n sharpConfidence: sharpConfidence\n });\n } catch (error) {\n reject(error);\n }\n };\n \n img.onerror = () => {\n reject(new Error('Failed to load image'));\n };\n \n // Handle both data URLs and base64 strings\n if (options.image.startsWith('data:')) {\n img.src = options.image;\n } else {\n img.src = `data:image/jpeg;base64,${options.image}`;\n }\n });\n } catch (error) {\n throw new Error(`Failed to detect blur: ${error}`);\n }\n }\n\n async takeSnapshot2(options:{canvas:HTMLCanvasElement,maxLength?:number}): Promise<{scaleRatio?:number}> {\n if (this.camera) {\n let canvas = options.canvas;\n let scaleRatio = 1.0;\n if (options && options.maxLength) {\n let res = (await this.getResolution()).resolution;\n let width = parseInt(res.split(\"x\")[0]);\n let height = parseInt(res.split(\"x\")[1]);\n let targetWidth = width;\n let targetHeight = height;\n if (width > options.maxLength || height > options.maxLength) {\n if (width > height) {\n targetWidth = options.maxLength;\n targetHeight = options.maxLength/width*height;\n scaleRatio = options.maxLength/width;\n }else{\n targetHeight = options.maxLength;\n targetWidth = options.maxLength/height*width;\n scaleRatio = options.maxLength/height;\n }\n canvas.width = targetWidth;\n canvas.height = targetHeight;\n let video = this.camera.getUIElement().getElementsByTagName(\"video\")[0];\n let ctx = canvas.getContext('2d');\n if (ctx) {\n ctx.drawImage(video, 0, 0, targetWidth, targetHeight);\n }\n }else{\n this.drawFullFrame(canvas);\n }\n }else{\n this.drawFullFrame(canvas);\n }\n return {scaleRatio:scaleRatio};\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n drawFullFrame(canvas:HTMLCanvasElement):HTMLCanvasElement{\n let video = this.camera?.getUIElement().getElementsByTagName(\"video\")[0];\n if (video) {\n canvas.width = video.videoWidth;\n canvas.height = video.videoHeight;\n let ctx = canvas.getContext('2d');\n if (ctx) {\n ctx.drawImage(video, 0, 0, canvas.width, canvas.height);\n }\n }\n return canvas;\n }\n\n async takePhoto(_options:{includeBase64?:boolean}): Promise<{ path?:string, base64?: string, blob?:Blob, blurScore?: number }> {\n if (this.camera) {\n let video = this.camera.getUIElement().getElementsByTagName(\"video\")[0];\n let localStream:MediaStream = video.srcObject as MediaStream;\n if (localStream) {\n if (\"ImageCapture\" in window) {\n try {\n //@ts-ignore \n let ImageCapture:any = window[\"ImageCapture\"];\n console.log(\"ImageCapture supported\");\n const track = localStream.getVideoTracks()[0];\n let imageCapture = new ImageCapture(track);\n let blob = await imageCapture.takePhoto();\n return {blob:blob}; \n } catch (error) {\n console.log(error);\n }\n }else{\n console.log(\"ImageCapture not supported\");\n }\n }\n this.camera.setScanRegion(\n {\n regionLeft: 0,\n regionTop: 0,\n regionBottom: 100,\n regionRight: 100,\n regionMeasuredByPercentage: 1\n }\n )\n let canvas = this.camera.getFrame().toCanvas();\n let base64 = this.removeDataURLHead(canvas.toDataURL(\"image/jpeg\", this.desiredJpegQuality));\n this.applyScanRegion();\n \n // Add blur detection if base64 is included\n if (_options?.includeBase64) {\n const ctx = canvas.getContext('2d');\n if (ctx) {\n const imageData = ctx.getImageData(0, 0, canvas.width, canvas.height);\n const blurResult = this.detectBlurWeb(imageData, canvas.width, canvas.height);\n return {\n base64: base64,\n blurScore: blurResult.blurScore\n };\n }\n }\n \n return {base64:base64};\n }else {\n throw new Error('Camera not initialized');\n }\n }\n\n removeDataURLHead(dataURL:string){\n return dataURL.substring(dataURL.indexOf(\",\")+1,dataURL.length);\n }\n\n\n async requestCameraPermission(): Promise<void> {\n const constraints = {video: true, audio: false};\n const stream = await navigator.mediaDevices.getUserMedia(constraints);\n const tracks = stream.getTracks();\n for (let i=0;i<tracks.length;i++) {\n const track = tracks[i];\n track.stop(); // stop the opened tracks\n }\n }\n\n async requestMicroPhonePermission(): Promise<void> {\n try {\n const constraints = {video: false, audio: true};\n const stream = await navigator.mediaDevices.getUserMedia(constraints);\n const tracks = stream.getTracks();\n for (let i=0;i<tracks.length;i++) {\n const track = tracks[i];\n track.stop(); // stop the opened tracks\n }\n this.hasMicrophone = true;\n } catch (error) {\n this.hasMicrophone = false;\n throw error;\n }\n }\n\n async startRecording(): Promise<void> {\n if (this.camera) {\n if (this.hasMicrophone) {\n let settings = this.camera.getVideoSettings();\n settings.audio = true;\n await this.camera.updateVideoSettings(settings);\n }\n let video = this.camera.getUIElement().getElementsByTagName(\"video\")[0];\n this.recorder = new RecordRTC(video.srcObject as MediaStream, {\n type: 'video'\n });\n this.recorder.startRecording();\n }else{\n throw new Error(\"camera not initialized\");\n }\n }\n\n async stopRecording(): Promise<{blob:Blob}> {\n return new Promise<{blob:Blob}>((resolve, reject) => {\n if (this.recorder) {\n const stopRecordingCallback = () => {\n let blob = this.recorder!.getBlob();\n this.recorder!.destroy();\n this.recorder = null;\n resolve({blob})\n }\n this.recorder.stopRecording(stopRecordingCallback);\n }else{\n reject(\"recorder not initialized\");\n }\n }) \n }\n\n async setLayout(options: {top: string,left:string,width:string, height:string}): Promise<void> {\n if (this.camera) {\n let ele = this.camera.getUIElement();\n if (options.top) {\n ele.style.top = options.top;\n }\n if (options.left) {\n ele.style.left = options.left;\n }\n if (options.width) {\n ele.style.width = options.width;\n }\n if (options.height) {\n ele.style.height = options.height;\n }\n ele.style.position = \"absolute\";\n }else{\n throw new Error(\"Camera not initialized\");\n }\n }\n\n /**\n * Detect blur using Laplacian variance algorithm (same as iOS/Android)\n * Higher values indicate sharper images\n */\n private detectBlurWeb(imageData: ImageData, width: number, height: number): { blurScore: number } {\n const data = imageData.data;\n let variance = 0;\n let count = 0;\n \n // Sample every 4th pixel for performance (similar to Android implementation)\n const step = 4;\n for (let y = step; y < height - step; y += step) {\n for (let x = step; x < width - step; x += step) {\n const idx = (y * width + x) * 4;\n \n // Convert to grayscale using same formula as Android\n const gray = 0.299 * data[idx] + 0.587 * data[idx + 1] + 0.114 * data[idx + 2];\n \n // Calculate neighbors for 3x3 Laplacian kernel\n const neighbors = [\n 0.299 * data[((y-1) * width + (x-1)) * 4] + 0.587 * data[((y-1) * width + (x-1)) * 4 + 1] + 0.114 * data[((y-1) * width + (x-1)) * 4 + 2],\n 0.299 * data[((y-1) * width + x) * 4] + 0.587 * data[((y-1) * width + x) * 4 + 1] + 0.114 * data[((y-1) * width + x) * 4 + 2],\n 0.299 * data[((y-1) * width + (x+1)) * 4] + 0.587 * data[((y-1) * width + (x+1)) * 4 + 1] + 0.114 * data[((y-1) * width + (x+1)) * 4 + 2],\n 0.299 * data[(y * width + (x-1)) * 4] + 0.587 * data[(y * width + (x-1)) * 4 + 1] + 0.114 * data[(y * width + (x-1)) * 4 + 2],\n 0.299 * data[(y * width + (x+1)) * 4] + 0.587 * data[(y * width + (x+1)) * 4 + 1] + 0.114 * data[(y * width + (x+1)) * 4 + 2],\n 0.299 * data[((y+1) * width + (x-1)) * 4] + 0.587 * data[((y+1) * width + (x-1)) * 4 + 1] + 0.114 * data[((y+1) * width + (x-1)) * 4 + 2],\n 0.299 * data[((y+1) * width + x) * 4] + 0.587 * data[((y+1) * width + x) * 4 + 1] + 0.114 * data[((y+1) * width + x) * 4 + 2],\n 0.299 * data[((y+1) * width + (x+1)) * 4] + 0.587 * data[((y+1) * width + (x+1)) * 4 + 1] + 0.114 * data[((y+1) * width + (x+1)) * 4 + 2]\n ];\n \n // Apply 3x3 Laplacian kernel (matches Android implementation)\n const laplacian = -neighbors[0] - neighbors[1] - neighbors[2] +\n -neighbors[3] + 8 * gray - neighbors[4] +\n -neighbors[5] - neighbors[6] - neighbors[7];\n \n variance += laplacian * laplacian;\n count++;\n }\n }\n \n const blurScore = count > 0 ? variance / count : 0;\n \n return { blurScore };\n }\n}\n"]}