capacitor-camera-view 1.0.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.
Files changed (42) hide show
  1. package/CapacitorCameraView.podspec +17 -0
  2. package/LICENSE +201 -0
  3. package/Package.swift +28 -0
  4. package/README.md +654 -0
  5. package/android/build.gradle +79 -0
  6. package/android/src/main/AndroidManifest.xml +2 -0
  7. package/android/src/main/java/com/michaelwolz/capacitorcameraview/CameraView.kt +555 -0
  8. package/android/src/main/java/com/michaelwolz/capacitorcameraview/CameraViewPlugin.kt +227 -0
  9. package/android/src/main/java/com/michaelwolz/capacitorcameraview/model/BarcodeDetectionResult.kt +11 -0
  10. package/android/src/main/java/com/michaelwolz/capacitorcameraview/model/CameraDevice.kt +14 -0
  11. package/android/src/main/java/com/michaelwolz/capacitorcameraview/model/CameraSessionConfiguration.kt +10 -0
  12. package/android/src/main/java/com/michaelwolz/capacitorcameraview/model/WebBoundingRect.kt +16 -0
  13. package/android/src/main/java/com/michaelwolz/capacitorcameraview/model/ZoomFactors.kt +14 -0
  14. package/android/src/main/java/com/michaelwolz/capacitorcameraview/utils.kt +86 -0
  15. package/android/src/main/res/.gitkeep +0 -0
  16. package/dist/docs.json +968 -0
  17. package/dist/esm/definitions.d.ts +378 -0
  18. package/dist/esm/definitions.js +2 -0
  19. package/dist/esm/definitions.js.map +1 -0
  20. package/dist/esm/index.d.ts +7 -0
  21. package/dist/esm/index.js +10 -0
  22. package/dist/esm/index.js.map +1 -0
  23. package/dist/esm/utils.d.ts +45 -0
  24. package/dist/esm/utils.js +108 -0
  25. package/dist/esm/utils.js.map +1 -0
  26. package/dist/esm/web.d.ts +108 -0
  27. package/dist/esm/web.js +406 -0
  28. package/dist/esm/web.js.map +1 -0
  29. package/dist/plugin.cjs.js +530 -0
  30. package/dist/plugin.cjs.js.map +1 -0
  31. package/dist/plugin.js +533 -0
  32. package/dist/plugin.js.map +1 -0
  33. package/ios/Sources/CameraViewPlugin/CameraError.swift +39 -0
  34. package/ios/Sources/CameraViewPlugin/CameraSessionConfiguration.swift +32 -0
  35. package/ios/Sources/CameraViewPlugin/CameraViewManager+BarcodeScan.swift +91 -0
  36. package/ios/Sources/CameraViewPlugin/CameraViewManager+PhotoCapture.swift +52 -0
  37. package/ios/Sources/CameraViewPlugin/CameraViewManager+VideoDataOutput.swift +78 -0
  38. package/ios/Sources/CameraViewPlugin/CameraViewManager.swift +633 -0
  39. package/ios/Sources/CameraViewPlugin/CameraViewPlugin.swift +295 -0
  40. package/ios/Sources/CameraViewPlugin/Utils.swift +56 -0
  41. package/ios/Tests/CameraViewPluginTests/CameraViewPluginTests.swift +15 -0
  42. package/package.json +94 -0
@@ -0,0 +1,530 @@
1
+ 'use strict';
2
+
3
+ var core = require('@capacitor/core');
4
+
5
+ /**
6
+ * The main Capacitor Camera View plugin instance.
7
+ */
8
+ const CameraView = core.registerPlugin('CameraView', {
9
+ web: () => Promise.resolve().then(function () { return web; }).then((m) => new m.CameraViewWeb()),
10
+ });
11
+
12
+ /**
13
+ * Converts canvas to base64 string
14
+ */
15
+ function canvasToBase64(canvas, quality) {
16
+ const dataUrl = canvas.toDataURL('image/jpeg', quality);
17
+ return dataUrl.split(',')[1];
18
+ }
19
+ /**
20
+ * Calculates the visible area of the video based on object-fit: cover
21
+ */
22
+ function calculateVisibleArea(video) {
23
+ // Get the displayed dimensions of the video element
24
+ const videoRect = video.getBoundingClientRect();
25
+ const displayWidth = videoRect.width;
26
+ const displayHeight = videoRect.height;
27
+ // Get the intrinsic dimensions of the video
28
+ const videoWidth = video.videoWidth;
29
+ const videoHeight = video.videoHeight;
30
+ // Calculate which portion of the video is visible (for object-fit: cover)
31
+ const videoAspect = videoWidth / videoHeight;
32
+ const displayAspect = displayWidth / displayHeight;
33
+ let sourceX = 0;
34
+ let sourceY = 0;
35
+ let sourceWidth = videoWidth;
36
+ let sourceHeight = videoHeight;
37
+ // If video aspect ratio is greater than display aspect ratio,
38
+ // the video is cropped on the sides
39
+ if (videoAspect > displayAspect) {
40
+ sourceWidth = videoHeight * displayAspect;
41
+ sourceX = (videoWidth - sourceWidth) / 2;
42
+ }
43
+ // Otherwise the video is cropped on the top and bottom
44
+ else {
45
+ sourceHeight = videoWidth / displayAspect;
46
+ sourceY = (videoHeight - sourceHeight) / 2;
47
+ }
48
+ return {
49
+ sourceX,
50
+ sourceY,
51
+ sourceWidth,
52
+ sourceHeight,
53
+ displayWidth,
54
+ displayHeight,
55
+ };
56
+ }
57
+ /**
58
+ * Draws the visible area of the video to the canvas
59
+ */
60
+ function drawVisibleAreaToCanvas(canvas, videoElement, area) {
61
+ const { sourceX, sourceY, sourceWidth, sourceHeight, displayWidth, displayHeight } = area;
62
+ // Set canvas size to match the displayed dimensions
63
+ canvas.width = displayWidth;
64
+ canvas.height = displayHeight;
65
+ const ctx = canvas.getContext('2d', { alpha: false });
66
+ if (!ctx) {
67
+ throw new Error('Could not get canvas context');
68
+ }
69
+ // Draw only the visible portion of the video to match what the user sees
70
+ ctx.drawImage(videoElement, sourceX, sourceY, sourceWidth, sourceHeight, 0, 0, displayWidth, displayHeight);
71
+ }
72
+ /**
73
+ * Transforms barcode coordinates from the video source space to display space
74
+ * accounting for object-fit: cover scaling and cropping.
75
+ *
76
+ * @param barcodeBoundingBox The original barcode bounding box from the detector
77
+ * @param videoElement The video element with the camera stream
78
+ * @returns The transformed bounding box coordinates in display space
79
+ */
80
+ function transformBarcodeBoundingBox(barcodeBoundingBox, videoElement) {
81
+ // Get the video element's displayed dimensions
82
+ const videoRect = videoElement.getBoundingClientRect();
83
+ const displayWidth = videoRect.width;
84
+ const displayHeight = videoRect.height;
85
+ // Get original video dimensions
86
+ const videoWidth = videoElement.videoWidth;
87
+ const videoHeight = videoElement.videoHeight;
88
+ // Calculate scaling and positioning for object-fit: cover
89
+ const videoAspect = videoWidth / videoHeight;
90
+ const displayAspect = displayWidth / displayHeight;
91
+ let scaledX, scaledY, scaledWidth, scaledHeight;
92
+ if (videoAspect > displayAspect) {
93
+ // Video is wider than display area - height matches, width is centered and cropped
94
+ const scale = displayHeight / videoHeight;
95
+ const scaledVideoWidth = videoWidth * scale;
96
+ const cropX = (scaledVideoWidth - displayWidth) / 2;
97
+ scaledWidth = barcodeBoundingBox.width * scale;
98
+ scaledHeight = barcodeBoundingBox.height * scale;
99
+ scaledX = barcodeBoundingBox.x * scale - cropX;
100
+ scaledY = barcodeBoundingBox.y * scale;
101
+ }
102
+ else {
103
+ // Video is taller than display area - width matches, height is centered and cropped
104
+ const scale = displayWidth / videoWidth;
105
+ const scaledVideoHeight = videoHeight * scale;
106
+ const cropY = (scaledVideoHeight - displayHeight) / 2;
107
+ scaledWidth = barcodeBoundingBox.width * scale;
108
+ scaledHeight = barcodeBoundingBox.height * scale;
109
+ scaledX = barcodeBoundingBox.x * scale;
110
+ scaledY = barcodeBoundingBox.y * scale - cropY;
111
+ }
112
+ return {
113
+ x: scaledX,
114
+ y: scaledY,
115
+ width: scaledWidth,
116
+ height: scaledHeight,
117
+ };
118
+ }
119
+
120
+ var __classPrivateFieldGet = (undefined && undefined.__classPrivateFieldGet) || function (receiver, privateMap) {
121
+ if (!privateMap.has(receiver)) {
122
+ throw new TypeError("attempted to get private field on non-instance");
123
+ }
124
+ return privateMap.get(receiver);
125
+ };
126
+ var __classPrivateFieldSet = (undefined && undefined.__classPrivateFieldSet) || function (receiver, privateMap, value) {
127
+ if (!privateMap.has(receiver)) {
128
+ throw new TypeError("attempted to set private field on non-instance");
129
+ }
130
+ privateMap.set(receiver, value);
131
+ return value;
132
+ };
133
+ var _isRunning;
134
+ /**
135
+ * Web implementation of the CameraViewPlugin.
136
+ * Optimized for performance and battery efficiency.
137
+ */
138
+ class CameraViewWeb extends core.WebPlugin {
139
+ constructor() {
140
+ super();
141
+ // DOM elements
142
+ this.videoElement = null;
143
+ this.canvasElement = null;
144
+ // Stream state
145
+ this.stream = null;
146
+ _isRunning.set(this, false);
147
+ // Configuration state
148
+ this.currentCamera = 'environment'; // Default to back camera
149
+ this.currentZoom = 1.0;
150
+ this.currentFlashMode = 'off';
151
+ // Barcode detection support
152
+ this.barcodeDetectionSupported = false;
153
+ this.barcodeDetector = null;
154
+ this.checkBarcodeDetectionSupport();
155
+ }
156
+ /**
157
+ * Start the camera with the given configuration
158
+ */
159
+ async start(options) {
160
+ if (__classPrivateFieldGet(this, _isRunning)) {
161
+ return;
162
+ }
163
+ const permissionStatus = await this.requestPermissions();
164
+ if (permissionStatus.camera !== 'granted') {
165
+ throw new Error('Camera permission was not granted');
166
+ }
167
+ try {
168
+ // Set up video element if it doesn't exist
169
+ if (!this.videoElement) {
170
+ await this.setupVideoElement(options === null || options === void 0 ? void 0 : options.containerElementId);
171
+ }
172
+ // Set up video constraints based on options
173
+ const videoConstraints = {};
174
+ // Prefer deviceId if specified
175
+ if (options === null || options === void 0 ? void 0 : options.deviceId) {
176
+ videoConstraints.deviceId = { exact: options.deviceId };
177
+ // Remember the current camera mode (though we're using a specific device)
178
+ this.currentCamera = (options === null || options === void 0 ? void 0 : options.position) === 'front' ? 'user' : 'environment';
179
+ }
180
+ else {
181
+ // Fall back to facing mode
182
+ const facingMode = (options === null || options === void 0 ? void 0 : options.position) === 'front' ? 'user' : 'environment';
183
+ this.currentCamera = facingMode;
184
+ videoConstraints.facingMode = facingMode;
185
+ }
186
+ const constraints = {
187
+ video: videoConstraints,
188
+ audio: false,
189
+ };
190
+ this.stream = await navigator.mediaDevices.getUserMedia(constraints);
191
+ if (this.videoElement) {
192
+ this.videoElement.srcObject = this.stream;
193
+ this.videoElement.play();
194
+ __classPrivateFieldSet(this, _isRunning, true);
195
+ // If barcode detection is enabled and supported, start detection
196
+ if ((options === null || options === void 0 ? void 0 : options.enableBarcodeDetection) && this.barcodeDetectionSupported) {
197
+ this.startBarcodeDetection();
198
+ }
199
+ }
200
+ }
201
+ catch (err) {
202
+ throw new Error(`Failed to start camera: ${this.formatError(err)}`);
203
+ }
204
+ }
205
+ /**
206
+ * Stop the camera and release resources
207
+ */
208
+ async stop() {
209
+ if (!__classPrivateFieldGet(this, _isRunning)) {
210
+ return;
211
+ }
212
+ try {
213
+ // Stop all tracks in the stream
214
+ if (this.stream) {
215
+ this.stream.getTracks().forEach((track) => track.stop());
216
+ this.stream = null;
217
+ }
218
+ // Clear video source
219
+ if (this.videoElement) {
220
+ this.videoElement = null;
221
+ }
222
+ __classPrivateFieldSet(this, _isRunning, false);
223
+ }
224
+ catch (err) {
225
+ throw new Error(`Failed to stop camera: ${this.formatError(err)}`);
226
+ }
227
+ }
228
+ /**
229
+ * Check if the camera is currently running
230
+ */
231
+ async isRunning() {
232
+ return { isRunning: __classPrivateFieldGet(this, _isRunning) };
233
+ }
234
+ /**
235
+ * Capture a photo using the camera and return it as a base64-encoded JPEG image.
236
+ * Preserves what the user actually sees in the UI, including cropping from object-fit: cover.
237
+ */
238
+ async capture(options) {
239
+ const videoElement = this.videoElement;
240
+ if (!__classPrivateFieldGet(this, _isRunning) || !videoElement) {
241
+ throw new Error('Camera is not running');
242
+ }
243
+ try {
244
+ const canvas = this.getCanvasElement();
245
+ const visibleArea = calculateVisibleArea(videoElement);
246
+ drawVisibleAreaToCanvas(canvas, videoElement, visibleArea);
247
+ const quality = Math.min(1.0, Math.max(0.1, options.quality / 100));
248
+ const base64Data = canvasToBase64(canvas, quality);
249
+ return { photo: base64Data };
250
+ }
251
+ catch (err) {
252
+ throw new Error(`Failed to capture photo: ${this.formatError(err)}`);
253
+ }
254
+ }
255
+ /**
256
+ * Web implementation already uses images from the video stream, so this is the same as `capture()`
257
+ */
258
+ async captureSample(options) {
259
+ return this.capture(options);
260
+ }
261
+ /**
262
+ * Flip between front and back camera
263
+ */
264
+ async flipCamera() {
265
+ if (!__classPrivateFieldGet(this, _isRunning)) {
266
+ throw new Error('Camera is not running');
267
+ }
268
+ try {
269
+ // Switch current camera
270
+ this.currentCamera = this.currentCamera === 'user' ? 'environment' : 'user';
271
+ // Stop current stream
272
+ if (this.stream) {
273
+ this.stream.getTracks().forEach((track) => track.stop());
274
+ }
275
+ // Restart with new facing mode
276
+ const constraints = {
277
+ video: {
278
+ facingMode: this.currentCamera,
279
+ },
280
+ audio: false,
281
+ };
282
+ this.stream = await navigator.mediaDevices.getUserMedia(constraints);
283
+ if (this.videoElement) {
284
+ this.videoElement.srcObject = this.stream;
285
+ }
286
+ }
287
+ catch (err) {
288
+ throw new Error(`Failed to flip camera: ${this.formatError(err)}`);
289
+ }
290
+ }
291
+ /**
292
+ * Get available camera devices
293
+ */
294
+ async getAvailableDevices() {
295
+ try {
296
+ const devices = await navigator.mediaDevices.enumerateDevices();
297
+ const videoDevices = devices.filter((device) => device.kind === 'videoinput');
298
+ return {
299
+ devices: videoDevices.map((device) => ({
300
+ id: device.deviceId,
301
+ name: device.label || `Camera ${device.deviceId.substring(0, 5)}`,
302
+ position: device.label.toLowerCase().includes('front') ? 'front' : 'back',
303
+ })),
304
+ };
305
+ }
306
+ catch (err) {
307
+ console.error('Failed to get available devices', err);
308
+ return { devices: [] };
309
+ }
310
+ }
311
+ /**
312
+ * Get current zoom information (web has limited zoom support)
313
+ */
314
+ async getZoom() {
315
+ // Web has limited zoom capabilities in most browsers,
316
+ // we fake zoomin by scaling the video element
317
+ return {
318
+ min: 1.0,
319
+ max: 3.0,
320
+ current: this.currentZoom,
321
+ };
322
+ }
323
+ /**
324
+ * Set zoom level (limited support in web)
325
+ */
326
+ async setZoom(options) {
327
+ // Store the requested zoom level
328
+ this.currentZoom = options.level;
329
+ // Apply visual zoom using CSS transform when native zoom isn't supported
330
+ if (this.videoElement) {
331
+ this.videoElement.style.transition = options.ramp ? 'transform 0.2s ease-in-out' : 'none';
332
+ const scale = Math.max(1.0, Math.min(options.level, 3.0)); // Limit scale to reasonable bounds
333
+ this.videoElement.style.transform = `scale(${scale})`;
334
+ this.videoElement.style.transformOrigin = 'center';
335
+ }
336
+ }
337
+ /**
338
+ * Get current flash mode
339
+ */
340
+ async getFlashMode() {
341
+ return { flashMode: this.currentFlashMode };
342
+ }
343
+ /**
344
+ * Get supported flash modes
345
+ */
346
+ async getSupportedFlashModes() {
347
+ // Web has limited flash control
348
+ return { flashModes: ['off'] };
349
+ }
350
+ /**
351
+ * Set flash mode (limited support in web)
352
+ */
353
+ async setFlashMode(options) {
354
+ this.currentFlashMode = options.mode;
355
+ console.warn('Flash mode control is not fully supported in the web implementation');
356
+ }
357
+ /**
358
+ * Check camera permission without requesting
359
+ */
360
+ async checkPermissions() {
361
+ try {
362
+ // Use Permissions API if available
363
+ if (navigator.permissions) {
364
+ const result = await navigator.permissions.query({ name: 'camera' });
365
+ return {
366
+ camera: result.state === 'granted' ? 'granted' : result.state === 'denied' ? 'denied' : 'prompt',
367
+ };
368
+ }
369
+ // If Permissions API is not available, check if we have an active stream
370
+ return {
371
+ camera: this.stream ? 'granted' : 'prompt',
372
+ };
373
+ }
374
+ catch (err) {
375
+ // If permissions API is not supported or fails
376
+ return {
377
+ camera: 'prompt',
378
+ };
379
+ }
380
+ }
381
+ /**
382
+ * Request camera permission from the user
383
+ */
384
+ async requestPermissions() {
385
+ try {
386
+ // Try to access the camera to trigger the permission prompt
387
+ const stream = await navigator.mediaDevices.getUserMedia({ video: true });
388
+ // If we get here, permission was granted
389
+ // Clean up the test stream
390
+ stream.getTracks().forEach((track) => track.stop());
391
+ return { camera: 'granted' };
392
+ }
393
+ catch (err) {
394
+ // Permission denied or other error
395
+ return { camera: 'denied' };
396
+ }
397
+ }
398
+ /**
399
+ * Start barcode detection if supported
400
+ */
401
+ async startBarcodeDetection() {
402
+ const barcodeDetector = this.barcodeDetector;
403
+ const videoElement = this.videoElement;
404
+ if (!this.barcodeDetectionSupported || !barcodeDetector || !videoElement) {
405
+ return;
406
+ }
407
+ // Make sure video is fully loaded before starting detection
408
+ if (videoElement.readyState < 2) {
409
+ await new Promise((resolve) => {
410
+ const loadHandler = () => {
411
+ videoElement.removeEventListener('loadeddata', loadHandler);
412
+ resolve();
413
+ };
414
+ videoElement.addEventListener('loadeddata', loadHandler);
415
+ });
416
+ }
417
+ // Add throttling to reduce CPU usage
418
+ let lastDetectionTime = 0;
419
+ const minTimeBetweenDetections = 100; // ms
420
+ // Set up periodic frame analysis for barcode detection
421
+ const detectFrame = async () => {
422
+ if (!__classPrivateFieldGet(this, _isRunning) || !videoElement || !barcodeDetector) {
423
+ return;
424
+ }
425
+ const now = Date.now();
426
+ if (now - lastDetectionTime >= minTimeBetweenDetections) {
427
+ try {
428
+ const barcodes = await barcodeDetector.detect(videoElement);
429
+ lastDetectionTime = now;
430
+ if (barcodes.length > 0) {
431
+ const barcode = barcodes[0];
432
+ // Transform barcode coordinates using the utility function
433
+ const boundingRect = transformBarcodeBoundingBox(barcode.boundingBox, videoElement);
434
+ this.notifyListeners('barcodeDetected', {
435
+ value: barcode.rawValue,
436
+ type: barcode.format.toLowerCase(),
437
+ boundingRect,
438
+ });
439
+ }
440
+ }
441
+ catch (err) {
442
+ console.error('Barcode detection error', err);
443
+ }
444
+ }
445
+ if (__classPrivateFieldGet(this, _isRunning)) {
446
+ requestAnimationFrame(detectFrame);
447
+ }
448
+ };
449
+ requestAnimationFrame(detectFrame);
450
+ }
451
+ /**
452
+ * Clean up resources when the plugin is disposed
453
+ */
454
+ async handleOnDestroy() {
455
+ var _a;
456
+ await this.stop();
457
+ // Remove elements from DOM
458
+ if ((_a = this.videoElement) === null || _a === void 0 ? void 0 : _a.parentNode) {
459
+ this.videoElement.parentNode.removeChild(this.videoElement);
460
+ this.videoElement = null;
461
+ }
462
+ if (this.canvasElement) {
463
+ this.canvasElement = null;
464
+ }
465
+ this.barcodeDetector = null;
466
+ }
467
+ /**
468
+ * Check if barcode detection is supported in this browser
469
+ */
470
+ async checkBarcodeDetectionSupport() {
471
+ if ('BarcodeDetector' in window) {
472
+ try {
473
+ this.barcodeDetector = new BarcodeDetector();
474
+ this.barcodeDetectionSupported = true;
475
+ }
476
+ catch (e) {
477
+ console.warn('BarcodeDetector is not supported by this browser.');
478
+ this.barcodeDetectionSupported = false;
479
+ }
480
+ }
481
+ }
482
+ /**
483
+ * Set up the video element for the camera view
484
+ */
485
+ async setupVideoElement(containerElementId) {
486
+ this.videoElement = document.createElement('video');
487
+ this.videoElement.playsInline = true;
488
+ this.videoElement.autoplay = true;
489
+ this.videoElement.muted = true;
490
+ this.videoElement.style.width = '100%';
491
+ this.videoElement.style.height = '100%';
492
+ this.videoElement.style.objectFit = 'cover';
493
+ // If a container ID is provided, find that element and append the video to it
494
+ if (containerElementId) {
495
+ const container = document.getElementById(containerElementId);
496
+ if (!container) {
497
+ throw new Error(`Container element with ID ${containerElementId} not found`);
498
+ }
499
+ container.appendChild(this.videoElement);
500
+ }
501
+ else {
502
+ // Otherwise, append to body as fallback
503
+ document.body.appendChild(this.videoElement);
504
+ }
505
+ }
506
+ /**
507
+ * Ensures canvas element exists and returns it
508
+ */
509
+ getCanvasElement() {
510
+ if (!this.canvasElement) {
511
+ this.canvasElement = document.createElement('canvas');
512
+ }
513
+ return this.canvasElement;
514
+ }
515
+ /**
516
+ * Format error message
517
+ */
518
+ formatError(err) {
519
+ return err instanceof Error ? err.message : String(err);
520
+ }
521
+ }
522
+ _isRunning = new WeakMap();
523
+
524
+ var web = /*#__PURE__*/Object.freeze({
525
+ __proto__: null,
526
+ CameraViewWeb: CameraViewWeb
527
+ });
528
+
529
+ exports.CameraView = CameraView;
530
+ //# sourceMappingURL=plugin.cjs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plugin.cjs.js","sources":["esm/index.js","esm/utils.js","esm/web.js"],"sourcesContent":["import { registerPlugin } from '@capacitor/core';\n/**\n * The main Capacitor Camera View plugin instance.\n */\nconst CameraView = registerPlugin('CameraView', {\n web: () => import('./web').then((m) => new m.CameraViewWeb()),\n});\nexport * from './definitions';\nexport { CameraView };\n//# sourceMappingURL=index.js.map","/**\n * Converts canvas to base64 string\n */\nexport function canvasToBase64(canvas, quality) {\n const dataUrl = canvas.toDataURL('image/jpeg', quality);\n return dataUrl.split(',')[1];\n}\n/**\n * Calculates the visible area of the video based on object-fit: cover\n */\nexport function calculateVisibleArea(video) {\n // Get the displayed dimensions of the video element\n const videoRect = video.getBoundingClientRect();\n const displayWidth = videoRect.width;\n const displayHeight = videoRect.height;\n // Get the intrinsic dimensions of the video\n const videoWidth = video.videoWidth;\n const videoHeight = video.videoHeight;\n // Calculate which portion of the video is visible (for object-fit: cover)\n const videoAspect = videoWidth / videoHeight;\n const displayAspect = displayWidth / displayHeight;\n let sourceX = 0;\n let sourceY = 0;\n let sourceWidth = videoWidth;\n let sourceHeight = videoHeight;\n // If video aspect ratio is greater than display aspect ratio,\n // the video is cropped on the sides\n if (videoAspect > displayAspect) {\n sourceWidth = videoHeight * displayAspect;\n sourceX = (videoWidth - sourceWidth) / 2;\n }\n // Otherwise the video is cropped on the top and bottom\n else {\n sourceHeight = videoWidth / displayAspect;\n sourceY = (videoHeight - sourceHeight) / 2;\n }\n return {\n sourceX,\n sourceY,\n sourceWidth,\n sourceHeight,\n displayWidth,\n displayHeight,\n };\n}\n/**\n * Draws the visible area of the video to the canvas\n */\nexport function drawVisibleAreaToCanvas(canvas, videoElement, area) {\n const { sourceX, sourceY, sourceWidth, sourceHeight, displayWidth, displayHeight } = area;\n // Set canvas size to match the displayed dimensions\n canvas.width = displayWidth;\n canvas.height = displayHeight;\n const ctx = canvas.getContext('2d', { alpha: false });\n if (!ctx) {\n throw new Error('Could not get canvas context');\n }\n // Draw only the visible portion of the video to match what the user sees\n ctx.drawImage(videoElement, sourceX, sourceY, sourceWidth, sourceHeight, 0, 0, displayWidth, displayHeight);\n}\n/**\n * Transforms barcode coordinates from the video source space to display space\n * accounting for object-fit: cover scaling and cropping.\n *\n * @param barcodeBoundingBox The original barcode bounding box from the detector\n * @param videoElement The video element with the camera stream\n * @returns The transformed bounding box coordinates in display space\n */\nexport function transformBarcodeBoundingBox(barcodeBoundingBox, videoElement) {\n // Get the video element's displayed dimensions\n const videoRect = videoElement.getBoundingClientRect();\n const displayWidth = videoRect.width;\n const displayHeight = videoRect.height;\n // Get original video dimensions\n const videoWidth = videoElement.videoWidth;\n const videoHeight = videoElement.videoHeight;\n // Calculate scaling and positioning for object-fit: cover\n const videoAspect = videoWidth / videoHeight;\n const displayAspect = displayWidth / displayHeight;\n let scaledX, scaledY, scaledWidth, scaledHeight;\n if (videoAspect > displayAspect) {\n // Video is wider than display area - height matches, width is centered and cropped\n const scale = displayHeight / videoHeight;\n const scaledVideoWidth = videoWidth * scale;\n const cropX = (scaledVideoWidth - displayWidth) / 2;\n scaledWidth = barcodeBoundingBox.width * scale;\n scaledHeight = barcodeBoundingBox.height * scale;\n scaledX = barcodeBoundingBox.x * scale - cropX;\n scaledY = barcodeBoundingBox.y * scale;\n }\n else {\n // Video is taller than display area - width matches, height is centered and cropped\n const scale = displayWidth / videoWidth;\n const scaledVideoHeight = videoHeight * scale;\n const cropY = (scaledVideoHeight - displayHeight) / 2;\n scaledWidth = barcodeBoundingBox.width * scale;\n scaledHeight = barcodeBoundingBox.height * scale;\n scaledX = barcodeBoundingBox.x * scale;\n scaledY = barcodeBoundingBox.y * scale - cropY;\n }\n return {\n x: scaledX,\n y: scaledY,\n width: scaledWidth,\n height: scaledHeight,\n };\n}\n//# sourceMappingURL=utils.js.map","var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, privateMap) {\n if (!privateMap.has(receiver)) {\n throw new TypeError(\"attempted to get private field on non-instance\");\n }\n return privateMap.get(receiver);\n};\nvar __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, privateMap, value) {\n if (!privateMap.has(receiver)) {\n throw new TypeError(\"attempted to set private field on non-instance\");\n }\n privateMap.set(receiver, value);\n return value;\n};\nvar _isRunning;\nimport { WebPlugin } from '@capacitor/core';\nimport { calculateVisibleArea, canvasToBase64, drawVisibleAreaToCanvas, transformBarcodeBoundingBox } from './utils';\n/**\n * Web implementation of the CameraViewPlugin.\n * Optimized for performance and battery efficiency.\n */\nexport class CameraViewWeb extends WebPlugin {\n constructor() {\n super();\n // DOM elements\n this.videoElement = null;\n this.canvasElement = null;\n // Stream state\n this.stream = null;\n _isRunning.set(this, false);\n // Configuration state\n this.currentCamera = 'environment'; // Default to back camera\n this.currentZoom = 1.0;\n this.currentFlashMode = 'off';\n // Barcode detection support\n this.barcodeDetectionSupported = false;\n this.barcodeDetector = null;\n this.checkBarcodeDetectionSupport();\n }\n /**\n * Start the camera with the given configuration\n */\n async start(options) {\n if (__classPrivateFieldGet(this, _isRunning)) {\n return;\n }\n const permissionStatus = await this.requestPermissions();\n if (permissionStatus.camera !== 'granted') {\n throw new Error('Camera permission was not granted');\n }\n try {\n // Set up video element if it doesn't exist\n if (!this.videoElement) {\n await this.setupVideoElement(options === null || options === void 0 ? void 0 : options.containerElementId);\n }\n // Set up video constraints based on options\n const videoConstraints = {};\n // Prefer deviceId if specified\n if (options === null || options === void 0 ? void 0 : options.deviceId) {\n videoConstraints.deviceId = { exact: options.deviceId };\n // Remember the current camera mode (though we're using a specific device)\n this.currentCamera = (options === null || options === void 0 ? void 0 : options.position) === 'front' ? 'user' : 'environment';\n }\n else {\n // Fall back to facing mode\n const facingMode = (options === null || options === void 0 ? void 0 : options.position) === 'front' ? 'user' : 'environment';\n this.currentCamera = facingMode;\n videoConstraints.facingMode = facingMode;\n }\n const constraints = {\n video: videoConstraints,\n audio: false,\n };\n this.stream = await navigator.mediaDevices.getUserMedia(constraints);\n if (this.videoElement) {\n this.videoElement.srcObject = this.stream;\n this.videoElement.play();\n __classPrivateFieldSet(this, _isRunning, true);\n // If barcode detection is enabled and supported, start detection\n if ((options === null || options === void 0 ? void 0 : options.enableBarcodeDetection) && this.barcodeDetectionSupported) {\n this.startBarcodeDetection();\n }\n }\n }\n catch (err) {\n throw new Error(`Failed to start camera: ${this.formatError(err)}`);\n }\n }\n /**\n * Stop the camera and release resources\n */\n async stop() {\n if (!__classPrivateFieldGet(this, _isRunning)) {\n return;\n }\n try {\n // Stop all tracks in the stream\n if (this.stream) {\n this.stream.getTracks().forEach((track) => track.stop());\n this.stream = null;\n }\n // Clear video source\n if (this.videoElement) {\n this.videoElement = null;\n }\n __classPrivateFieldSet(this, _isRunning, false);\n }\n catch (err) {\n throw new Error(`Failed to stop camera: ${this.formatError(err)}`);\n }\n }\n /**\n * Check if the camera is currently running\n */\n async isRunning() {\n return { isRunning: __classPrivateFieldGet(this, _isRunning) };\n }\n /**\n * Capture a photo using the camera and return it as a base64-encoded JPEG image.\n * Preserves what the user actually sees in the UI, including cropping from object-fit: cover.\n */\n async capture(options) {\n const videoElement = this.videoElement;\n if (!__classPrivateFieldGet(this, _isRunning) || !videoElement) {\n throw new Error('Camera is not running');\n }\n try {\n const canvas = this.getCanvasElement();\n const visibleArea = calculateVisibleArea(videoElement);\n drawVisibleAreaToCanvas(canvas, videoElement, visibleArea);\n const quality = Math.min(1.0, Math.max(0.1, options.quality / 100));\n const base64Data = canvasToBase64(canvas, quality);\n return { photo: base64Data };\n }\n catch (err) {\n throw new Error(`Failed to capture photo: ${this.formatError(err)}`);\n }\n }\n /**\n * Web implementation already uses images from the video stream, so this is the same as `capture()`\n */\n async captureSample(options) {\n return this.capture(options);\n }\n /**\n * Flip between front and back camera\n */\n async flipCamera() {\n if (!__classPrivateFieldGet(this, _isRunning)) {\n throw new Error('Camera is not running');\n }\n try {\n // Switch current camera\n this.currentCamera = this.currentCamera === 'user' ? 'environment' : 'user';\n // Stop current stream\n if (this.stream) {\n this.stream.getTracks().forEach((track) => track.stop());\n }\n // Restart with new facing mode\n const constraints = {\n video: {\n facingMode: this.currentCamera,\n },\n audio: false,\n };\n this.stream = await navigator.mediaDevices.getUserMedia(constraints);\n if (this.videoElement) {\n this.videoElement.srcObject = this.stream;\n }\n }\n catch (err) {\n throw new Error(`Failed to flip camera: ${this.formatError(err)}`);\n }\n }\n /**\n * Get available camera devices\n */\n async getAvailableDevices() {\n try {\n const devices = await navigator.mediaDevices.enumerateDevices();\n const videoDevices = devices.filter((device) => device.kind === 'videoinput');\n return {\n devices: videoDevices.map((device) => ({\n id: device.deviceId,\n name: device.label || `Camera ${device.deviceId.substring(0, 5)}`,\n position: device.label.toLowerCase().includes('front') ? 'front' : 'back',\n })),\n };\n }\n catch (err) {\n console.error('Failed to get available devices', err);\n return { devices: [] };\n }\n }\n /**\n * Get current zoom information (web has limited zoom support)\n */\n async getZoom() {\n // Web has limited zoom capabilities in most browsers,\n // we fake zoomin by scaling the video element\n return {\n min: 1.0,\n max: 3.0,\n current: this.currentZoom,\n };\n }\n /**\n * Set zoom level (limited support in web)\n */\n async setZoom(options) {\n // Store the requested zoom level\n this.currentZoom = options.level;\n // Apply visual zoom using CSS transform when native zoom isn't supported\n if (this.videoElement) {\n this.videoElement.style.transition = options.ramp ? 'transform 0.2s ease-in-out' : 'none';\n const scale = Math.max(1.0, Math.min(options.level, 3.0)); // Limit scale to reasonable bounds\n this.videoElement.style.transform = `scale(${scale})`;\n this.videoElement.style.transformOrigin = 'center';\n }\n }\n /**\n * Get current flash mode\n */\n async getFlashMode() {\n return { flashMode: this.currentFlashMode };\n }\n /**\n * Get supported flash modes\n */\n async getSupportedFlashModes() {\n // Web has limited flash control\n return { flashModes: ['off'] };\n }\n /**\n * Set flash mode (limited support in web)\n */\n async setFlashMode(options) {\n this.currentFlashMode = options.mode;\n console.warn('Flash mode control is not fully supported in the web implementation');\n }\n /**\n * Check camera permission without requesting\n */\n async checkPermissions() {\n try {\n // Use Permissions API if available\n if (navigator.permissions) {\n const result = await navigator.permissions.query({ name: 'camera' });\n return {\n camera: result.state === 'granted' ? 'granted' : result.state === 'denied' ? 'denied' : 'prompt',\n };\n }\n // If Permissions API is not available, check if we have an active stream\n return {\n camera: this.stream ? 'granted' : 'prompt',\n };\n }\n catch (err) {\n // If permissions API is not supported or fails\n return {\n camera: 'prompt',\n };\n }\n }\n /**\n * Request camera permission from the user\n */\n async requestPermissions() {\n try {\n // Try to access the camera to trigger the permission prompt\n const stream = await navigator.mediaDevices.getUserMedia({ video: true });\n // If we get here, permission was granted\n // Clean up the test stream\n stream.getTracks().forEach((track) => track.stop());\n return { camera: 'granted' };\n }\n catch (err) {\n // Permission denied or other error\n return { camera: 'denied' };\n }\n }\n /**\n * Start barcode detection if supported\n */\n async startBarcodeDetection() {\n const barcodeDetector = this.barcodeDetector;\n const videoElement = this.videoElement;\n if (!this.barcodeDetectionSupported || !barcodeDetector || !videoElement) {\n return;\n }\n // Make sure video is fully loaded before starting detection\n if (videoElement.readyState < 2) {\n await new Promise((resolve) => {\n const loadHandler = () => {\n videoElement.removeEventListener('loadeddata', loadHandler);\n resolve();\n };\n videoElement.addEventListener('loadeddata', loadHandler);\n });\n }\n // Add throttling to reduce CPU usage\n let lastDetectionTime = 0;\n const minTimeBetweenDetections = 100; // ms\n // Set up periodic frame analysis for barcode detection\n const detectFrame = async () => {\n if (!__classPrivateFieldGet(this, _isRunning) || !videoElement || !barcodeDetector) {\n return;\n }\n const now = Date.now();\n if (now - lastDetectionTime >= minTimeBetweenDetections) {\n try {\n const barcodes = await barcodeDetector.detect(videoElement);\n lastDetectionTime = now;\n if (barcodes.length > 0) {\n const barcode = barcodes[0];\n // Transform barcode coordinates using the utility function\n const boundingRect = transformBarcodeBoundingBox(barcode.boundingBox, videoElement);\n this.notifyListeners('barcodeDetected', {\n value: barcode.rawValue,\n type: barcode.format.toLowerCase(),\n boundingRect,\n });\n }\n }\n catch (err) {\n console.error('Barcode detection error', err);\n }\n }\n if (__classPrivateFieldGet(this, _isRunning)) {\n requestAnimationFrame(detectFrame);\n }\n };\n requestAnimationFrame(detectFrame);\n }\n /**\n * Clean up resources when the plugin is disposed\n */\n async handleOnDestroy() {\n var _a;\n await this.stop();\n // Remove elements from DOM\n if ((_a = this.videoElement) === null || _a === void 0 ? void 0 : _a.parentNode) {\n this.videoElement.parentNode.removeChild(this.videoElement);\n this.videoElement = null;\n }\n if (this.canvasElement) {\n this.canvasElement = null;\n }\n this.barcodeDetector = null;\n }\n /**\n * Check if barcode detection is supported in this browser\n */\n async checkBarcodeDetectionSupport() {\n if ('BarcodeDetector' in window) {\n try {\n this.barcodeDetector = new BarcodeDetector();\n this.barcodeDetectionSupported = true;\n }\n catch (e) {\n console.warn('BarcodeDetector is not supported by this browser.');\n this.barcodeDetectionSupported = false;\n }\n }\n }\n /**\n * Set up the video element for the camera view\n */\n async setupVideoElement(containerElementId) {\n this.videoElement = document.createElement('video');\n this.videoElement.playsInline = true;\n this.videoElement.autoplay = true;\n this.videoElement.muted = true;\n this.videoElement.style.width = '100%';\n this.videoElement.style.height = '100%';\n this.videoElement.style.objectFit = 'cover';\n // If a container ID is provided, find that element and append the video to it\n if (containerElementId) {\n const container = document.getElementById(containerElementId);\n if (!container) {\n throw new Error(`Container element with ID ${containerElementId} not found`);\n }\n container.appendChild(this.videoElement);\n }\n else {\n // Otherwise, append to body as fallback\n document.body.appendChild(this.videoElement);\n }\n }\n /**\n * Ensures canvas element exists and returns it\n */\n getCanvasElement() {\n if (!this.canvasElement) {\n this.canvasElement = document.createElement('canvas');\n }\n return this.canvasElement;\n }\n /**\n * Format error message\n */\n formatError(err) {\n return err instanceof Error ? err.message : String(err);\n }\n}\n_isRunning = new WeakMap();\n//# sourceMappingURL=web.js.map"],"names":["registerPlugin","this","WebPlugin"],"mappings":"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"}