Skip to content

Commit

Permalink
Merge branch 'master' of github.com:spoonconsulting/cordova-plugin-si…
Browse files Browse the repository at this point in the history
…mple-camera-preview into video-rrecording
  • Loading branch information
YushraJewon committed May 24, 2024
2 parents ddcce21 + f65d1b1 commit b3a39f0
Show file tree
Hide file tree
Showing 12 changed files with 366 additions and 13 deletions.
7 changes: 7 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,10 @@
## [2.0.18](https://github.com/spoonconsulting/cordova-plugin-simple-camera-preview/compare/v2.0.17...v2.0.18) (2024-05-12)

* **iOS:** Added a method "switchCameraTo" to help switch between ultra-wide camera and default camera. ([#68](https://github.com/spoonconsulting/cordova-plugin-simple-camera-preview/pull/68))
* **Android:** Added a method "switchCameraTo" to help switch between ultra-wide camera and default camera. ([#68](https://github.com/spoonconsulting/cordova-plugin-simple-camera-preview/pull/68))
* **iOS:** Added a method "deviceHasUltraWideCamera" to check if device has ultra-wide camera. ([#68](https://github.com/spoonconsulting/cordova-plugin-simple-camera-preview/pull/68))
* **Android:** Added a method "deviceHasUltraWideCamera" to check if device has ultra-wide camera. ([#68](https://github.com/spoonconsulting/cordova-plugin-simple-camera-preview/pull/68))

## [2.0.17](https://github.com/spoonconsulting/cordova-plugin-simple-camera-preview/compare/v2.0.16...v2.0.17) (2023-07-04)

* **iOS:** Use interfaceOrientation for ios 13+. ([#67](https://github.com/spoonconsulting/cordova-plugin-simple-camera-preview/pull/67))
Expand Down
43 changes: 43 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -99,3 +99,46 @@ SimpleCameraPreview.setSize(size, () => {
console.log("Camera frame size set");
});
```

### deviceHasUltraWideCamera(successCallback, errorCallback)

Check if device has ultra-wide camera
<br>

```javascript

SimpleCameraPreview.deviceHasUltraWideCamera(size, (value: boolean) => {
console.log("Device has ultra-wide camera?: ", value);
});
```

### switchCameraTo(option, successCallback, errorCallback)

Switch camera between ultra-wide or default

The variable captureDevice can take two values:
```javascript
"ultra-wide-angle"

or

"default"
```
<br>

```javascript

const params = {
captureDevice: "ultra-wide-angle",
}

SimpleCameraPreview.switchCameraTo(
params,
(value: unknown) => {
return (typeof value === "boolean" ? value : false);
},
(e: unknown) => {
console.log("cannot switch camera: ", e);
}
);
```
4 changes: 2 additions & 2 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@spoonconsulting/cordova-plugin-simple-camera-preview",
"version": "2.0.17",
"version": "2.0.18",
"description": "Cordova plugin that allows camera interaction from HTML code for showing camera preview below or on top of the HTML.",
"keywords": [
"cordova",
Expand Down
2 changes: 1 addition & 1 deletion plugin.xml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>

<plugin id="@spoonconsulting/cordova-plugin-simple-camera-preview" version="2.0.17" xmlns="http://apache.org/cordova/ns/plugins/1.0" xmlns:android="http://schemas.android.com/apk/res/android">
<plugin id="@spoonconsulting/cordova-plugin-simple-camera-preview" version="2.0.18" xmlns="http://apache.org/cordova/ns/plugins/1.0" xmlns:android="http://schemas.android.com/apk/res/android">

<name>cordova-plugin-simple-camera-preview</name>
<description>Cordova plugin that allows camera interaction from HTML code. Show camera preview popup on top of the HTML.</description>
Expand Down
151 changes: 151 additions & 0 deletions src/android/CameraPreviewFragment.java
Original file line number Diff line number Diff line change
Expand Up @@ -7,10 +7,13 @@
import android.content.pm.PackageManager;
import android.content.res.Configuration;
import android.graphics.Point;
import android.hardware.camera2.CameraCharacteristics;
import android.location.Location;
import android.net.Uri;
import android.os.Bundle;
import android.provider.MediaStore;
import android.os.Handler;
import android.os.Looper;
import android.util.Log;
import android.util.Size;
import android.view.Display;
Expand All @@ -22,7 +25,9 @@

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.camera.camera2.internal.Camera2CameraInfoImpl;
import androidx.camera.core.Camera;
import androidx.camera.core.CameraInfo;
import androidx.camera.core.CameraSelector;
import androidx.camera.core.ImageCapture;
import androidx.camera.core.ImageCaptureException;
Expand Down Expand Up @@ -50,6 +55,9 @@

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.UUID;
import java.util.concurrent.ExecutionException;

Expand All @@ -74,6 +82,14 @@ interface HasFlashCallback {
void onResult(boolean result);
}

interface CameraSwitchedCallback {
void onSwitch(boolean result);
}

interface HasUltraWideCameraCallback {
void onResult(boolean result);
}

public class CameraPreviewFragment extends Fragment {

private PreviewView viewFinder;
Expand All @@ -91,6 +107,7 @@ public class CameraPreviewFragment extends Fragment {

private static float ratio = (4 / (float) 3);
private static final String TAG = "SimpleCameraPreview";
private String captureDevice;

public CameraPreviewFragment() {

Expand All @@ -104,6 +121,11 @@ public CameraPreviewFragment(int cameraDirection, CameraStartedCallback cameraSt
} catch (JSONException e) {
e.printStackTrace();
}
try {
this.captureDevice = options.getString("captureDevice");
} catch (JSONException e) {
e.printStackTrace();
}
startCameraCallback = cameraStartedCallback;
}

Expand Down Expand Up @@ -181,6 +203,7 @@ public void startCamera() {
}

});
setUpCamera(captureDevice,cameraProvider);

preview.setSurfaceProvider(viewFinder.getSurfaceProvider());

Expand All @@ -189,6 +212,44 @@ public void startCamera() {
}
}

@SuppressLint("RestrictedApi")
public void deviceHasUltraWideCamera(HasUltraWideCameraCallback hasUltraWideCameraCallback) {
ListenableFuture<ProcessCameraProvider> cameraProviderFuture = ProcessCameraProvider.getInstance(getActivity());
ProcessCameraProvider cameraProvider = null;

try {
cameraProvider = cameraProviderFuture.get();
} catch (ExecutionException | InterruptedException e) {
Log.e(TAG, "Error occurred while trying to obtain the camera provider: " + e.getMessage());
e.printStackTrace();
hasUltraWideCameraCallback.onResult(false);
return;
}
List<CameraInfo> cameraInfos = cameraProvider.getAvailableCameraInfos();

boolean defaultCamera = false;
boolean ultraWideCamera = false;
List<Camera2CameraInfoImpl> backCameras = new ArrayList<>();
for (CameraInfo cameraInfo : cameraInfos) {
if (cameraInfo instanceof Camera2CameraInfoImpl) {
Camera2CameraInfoImpl camera2CameraInfo = (Camera2CameraInfoImpl) cameraInfo;
if (camera2CameraInfo.getLensFacing() == CameraSelector.LENS_FACING_BACK) {
backCameras.add(camera2CameraInfo);
}
}
}

for (Camera2CameraInfoImpl backCamera : backCameras) {
if (backCamera.getCameraCharacteristicsCompat().get(CameraCharacteristics.LENS_INFO_AVAILABLE_FOCAL_LENGTHS)[0] >= 2.4) {
defaultCamera = true;
} else if( backCamera.getCameraCharacteristicsCompat().get(CameraCharacteristics.LENS_INFO_AVAILABLE_FOCAL_LENGTHS)[0] < 2.4) {
ultraWideCamera = true;
}
}

hasUltraWideCameraCallback.onResult(defaultCamera == true && ultraWideCamera == true);
}

public static Size calculateResolution(Context context, int targetSize) {
Size calculatedSize;
if (getScreenOrientation(context) == Configuration.ORIENTATION_PORTRAIT) {
Expand Down Expand Up @@ -396,4 +457,94 @@ public void setLocation(Location loc) {
this.location = loc;
}
}


public void switchCameraTo(String device, CameraSwitchedCallback cameraSwitchedCallback) {
Handler mainHandler = new Handler(Looper.getMainLooper());
mainHandler.post(() -> {
ListenableFuture<ProcessCameraProvider> cameraProviderFuture = ProcessCameraProvider.getInstance(getActivity());
ProcessCameraProvider cameraProvider = null;
try {
cameraProvider = cameraProviderFuture.get();
} catch (ExecutionException | InterruptedException e) {
Log.e(TAG, "Error occurred while trying to obtain the camera provider: " + e.getMessage());
e.printStackTrace();
cameraSwitchedCallback.onSwitch(false);
return;
}

setUpCamera(device,cameraProvider);

preview.setSurfaceProvider(viewFinder.getSurfaceProvider());
cameraSwitchedCallback.onSwitch(true);
});
}

@SuppressLint("RestrictedApi")
public void setUpCamera(String captureDevice, ProcessCameraProvider cameraProvider) {
CameraSelector cameraSelector;
if (captureDevice.equals("ultra-wide-angle")) {
cameraSelector = new CameraSelector.Builder()
.addCameraFilter(cameraInfos -> {
List<Camera2CameraInfoImpl> backCameras = new ArrayList<>();
for (CameraInfo cameraInfo : cameraInfos) {
if (cameraInfo instanceof Camera2CameraInfoImpl) {
Camera2CameraInfoImpl camera2CameraInfo = (Camera2CameraInfoImpl) cameraInfo;
if (camera2CameraInfo.getLensFacing() == CameraSelector.LENS_FACING_BACK) {
backCameras.add(camera2CameraInfo);
}
}
}

Camera2CameraInfoImpl selectedCamera = Collections.min(backCameras, (o1, o2) -> {
Float focalLength1 = o1.getCameraCharacteristicsCompat().get(CameraCharacteristics.LENS_INFO_AVAILABLE_FOCAL_LENGTHS)[0];
Float focalLength2 = o2.getCameraCharacteristicsCompat().get(CameraCharacteristics.LENS_INFO_AVAILABLE_FOCAL_LENGTHS)[0];
return Float.compare(focalLength1, focalLength2);
});

if (selectedCamera != null) {
return Collections.singletonList(selectedCamera);
} else {
return cameraInfos;
}
})
.build();
} else {
cameraSelector = new CameraSelector.Builder()
.requireLensFacing(direction)
.build();
}

Size targetResolution = null;
if (targetSize > 0) {
targetResolution = CameraPreviewFragment.calculateResolution(getContext(), targetSize);
}

preview = new Preview.Builder().build();
imageCapture = new ImageCapture.Builder()
.setTargetResolution(targetResolution)
.build();

cameraProvider.unbindAll();
try {
camera = cameraProvider.bindToLifecycle(
getActivity(),
cameraSelector,
preview,
imageCapture
);
} catch (IllegalArgumentException e) {
// Error with result in capturing image with default resolution
e.printStackTrace();
imageCapture = new ImageCapture.Builder()
.build();
camera = cameraProvider.bindToLifecycle(
getActivity(),
cameraSelector,
preview,
imageCapture
);
}

}
}
43 changes: 43 additions & 0 deletions src/android/SimpleCameraPreview.java
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,12 @@ public boolean execute(String action, JSONArray args, CallbackContext callbackCo

case "deviceHasFlash":
return deviceHasFlash(callbackContext);

case "deviceHasUltraWideCamera":
return deviceHasUltraWideCamera(callbackContext);

case "switchCameraTo":
return switchCameraTo(args.getString(0), callbackContext);
default:
break;
}
Expand Down Expand Up @@ -149,13 +155,28 @@ private boolean enable(JSONObject options, CallbackContext callbackContext) {
e.printStackTrace();
}

String captureDevice = "default";
try {
if (options.getString("captureDevice") != null && !options.getString("captureDevice").equals("null")) {
captureDevice = options.getString("captureDevice");
}
} catch (JSONException | NumberFormatException e) {
e.printStackTrace();
}

JSONObject cameraPreviewOptions = new JSONObject();
try {
cameraPreviewOptions.put("targetSize", targetSize);
} catch (JSONException e) {
e.printStackTrace();
}

try {
cameraPreviewOptions.put("captureDevice", captureDevice);
} catch (JSONException e) {
e.printStackTrace();
}

fragment = new CameraPreviewFragment(cameraDirection, (err) -> {
if (err != null) {
callbackContext.error(err.getMessage());
Expand Down Expand Up @@ -318,6 +339,14 @@ private boolean deviceHasFlash(CallbackContext callbackContext) {
return true;
}

private boolean deviceHasUltraWideCamera(CallbackContext callbackContext) {
fragment.deviceHasUltraWideCamera((boolean result) -> {
PluginResult pluginResult = new PluginResult(PluginResult.Status.OK, result);
callbackContext.sendPluginResult(pluginResult);
});
return true;
}

private boolean torchSwitch(boolean torchState, CallbackContext callbackContext) {
if (fragment == null) {
callbackContext.error("Camera is closed, cannot switch " + torchState + " torch");
Expand Down Expand Up @@ -369,6 +398,19 @@ public void run() {
}
}

private boolean switchCameraTo(String device, CallbackContext callbackContext) {
if (fragment == null) {
callbackContext.error("Camera is closed, cannot switch camera");
return true;
}

fragment.switchCameraTo(device, (boolean result) -> {
PluginResult pluginResult = new PluginResult(PluginResult.Status.OK, result);
callbackContext.sendPluginResult(pluginResult);
});
return true;
}

public boolean hasAllPermissions() {
for(String p : REQUIRED_PERMISSIONS) {
if(!PermissionHelper.hasPermission(this, p)) {
Expand Down Expand Up @@ -441,6 +483,7 @@ public void onRequestPermissionResult(int requestCode, String[] permissions, int
}
}
}


@Override
public void onDestroy() {
Expand Down
2 changes: 2 additions & 0 deletions src/ios/CameraSessionManager.h
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,8 @@
- (void) setupSession:(NSString *)defaultCamera completion:(void(^)(BOOL started))completion options:(NSDictionary *)options photoSettings:(AVCapturePhotoSettings *)photoSettings;
- (void) setFlashMode:(NSInteger)flashMode photoSettings:(AVCapturePhotoSettings *)photoSettings;
- (void) torchSwitch:(NSInteger)torchState;
- (void) switchCameraTo:(NSString *)cameraMode completion:(void (^)(BOOL success))completion;
- (BOOL) deviceHasUltraWideCamera;
- (void) updateOrientation:(AVCaptureVideoOrientation)orientation;
- (AVCaptureVideoOrientation) getCurrentOrientation:(UIInterfaceOrientation)toInterfaceOrientation;
+ (AVCaptureSessionPreset) calculateResolution:(NSInteger)targetSize;
Expand Down
Loading

0 comments on commit b3a39f0

Please sign in to comment.