mirror of
https://github.com/oxen-io/session-android.git
synced 2025-10-24 08:08:43 +00:00
Fix "twitchy fingers" bug in CameraView
better diagnostic information, too. Fixes #4422 Closes #4427 // FREEBIE
This commit is contained in:

committed by
Moxie Marlinspike

parent
56a3c99289
commit
5fd5b1e1ed
@@ -1,10 +1,7 @@
|
||||
package org.thoughtcrime.securesms.components.camera;
|
||||
|
||||
import android.annotation.TargetApi;
|
||||
import android.hardware.Camera;
|
||||
import android.hardware.Camera.Parameters;
|
||||
import android.hardware.Camera.Size;
|
||||
import android.os.Build.VERSION;
|
||||
import android.support.annotation.NonNull;
|
||||
import android.support.annotation.Nullable;
|
||||
|
||||
@@ -14,24 +11,13 @@ import java.util.List;
|
||||
|
||||
@SuppressWarnings("deprecation")
|
||||
public class CameraUtils {
|
||||
@TargetApi(11)
|
||||
public static @Nullable Size getPreferredPreviewSize(int orientation, int width, int height, @NonNull Camera camera) {
|
||||
final Parameters parameters = camera.getParameters();
|
||||
final Size preferredSize = VERSION.SDK_INT > 11
|
||||
? parameters.getPreferredPreviewSizeForVideo()
|
||||
: null;
|
||||
|
||||
return preferredSize == null ? getBestAspectPreviewSize(orientation, width, height, parameters)
|
||||
: preferredSize;
|
||||
}
|
||||
|
||||
/*
|
||||
* modified from: https://github.com/commonsguy/cwac-camera/blob/master/camera/src/com/commonsware/cwac/camera/CameraUtils.java
|
||||
*/
|
||||
public static @Nullable Size getBestAspectPreviewSize(int displayOrientation,
|
||||
int width,
|
||||
int height,
|
||||
Parameters parameters) {
|
||||
public static @Nullable Size getPreferredPreviewSize(int displayOrientation,
|
||||
int width,
|
||||
int height,
|
||||
@NonNull Camera camera) {
|
||||
double targetRatio = (double)width / height;
|
||||
Size optimalSize = null;
|
||||
double minDiff = Double.MAX_VALUE;
|
||||
@@ -40,7 +26,7 @@ public class CameraUtils {
|
||||
targetRatio = (double)height / width;
|
||||
}
|
||||
|
||||
List<Size> sizes = parameters.getSupportedPreviewSizes();
|
||||
List<Size> sizes = camera.getParameters().getSupportedPreviewSizes();
|
||||
|
||||
Collections.sort(sizes, Collections.reverseOrder(new SizeComparator()));
|
||||
|
||||
|
@@ -90,12 +90,9 @@ public class CameraView extends FrameLayout {
|
||||
@Override
|
||||
protected @Nullable Camera onRunBackground() {
|
||||
try {
|
||||
if (cameraId >= 0) {
|
||||
return Camera.open(cameraId);
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
return Camera.open(cameraId);
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, e);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
@@ -103,6 +100,7 @@ public class CameraView extends FrameLayout {
|
||||
@Override
|
||||
protected void onPostMain(@Nullable Camera camera) {
|
||||
if (camera == null) {
|
||||
Log.w(TAG, "tried to open camera but got null");
|
||||
if (listener != null) listener.onCameraFail();
|
||||
return;
|
||||
}
|
||||
@@ -132,17 +130,23 @@ public class CameraView extends FrameLayout {
|
||||
if (!started) return;
|
||||
started = false;
|
||||
Log.w(TAG, "onPause() queued");
|
||||
final Optional<Camera> cameraToDestroy = camera;
|
||||
|
||||
enqueueTask(new SerialAsyncTask<Void>() {
|
||||
private Optional<Camera> cameraToDestroy;
|
||||
@Override protected void onPreMain() {
|
||||
cameraToDestroy = camera;
|
||||
camera = Optional.absent();
|
||||
}
|
||||
|
||||
@Override protected Void onRunBackground() {
|
||||
if (cameraToDestroy.isPresent()) {
|
||||
stopPreview();
|
||||
cameraToDestroy.get().release();
|
||||
try {
|
||||
stopPreview();
|
||||
cameraToDestroy.get().release();
|
||||
Log.w(TAG, "released old camera instance");
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, e);
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
@@ -171,6 +175,7 @@ public class CameraView extends FrameLayout {
|
||||
camera.get());
|
||||
final Parameters parameters = camera.get().getParameters();
|
||||
if (preferredPreviewSize != null && !parameters.getPreviewSize().equals(preferredPreviewSize)) {
|
||||
Log.w(TAG, "setting preview size to " + preferredPreviewSize.width + "x" + preferredPreviewSize.height);
|
||||
stopPreview();
|
||||
parameters.setPreviewSize(preferredPreviewSize.width, preferredPreviewSize.height);
|
||||
camera.get().setParameters(parameters);
|
||||
@@ -239,7 +244,6 @@ public class CameraView extends FrameLayout {
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@TargetApi(14)
|
||||
private void onCameraReady() {
|
||||
if (!camera.isPresent()) return;
|
||||
@@ -262,8 +266,8 @@ public class CameraView extends FrameLayout {
|
||||
if (camera.isPresent()) {
|
||||
try {
|
||||
camera.get().setPreviewDisplay(surface.getHolder());
|
||||
startPreview();
|
||||
} catch (IOException e) {
|
||||
requestLayout();
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, e);
|
||||
}
|
||||
}
|
||||
@@ -273,13 +277,21 @@ public class CameraView extends FrameLayout {
|
||||
|
||||
private void startPreview() {
|
||||
if (camera.isPresent()) {
|
||||
camera.get().startPreview();
|
||||
try {
|
||||
camera.get().startPreview();
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void stopPreview() {
|
||||
if (camera.isPresent()) {
|
||||
camera.get().stopPreview();
|
||||
try {
|
||||
camera.get().stopPreview();
|
||||
} catch (Exception e) {
|
||||
Log.w(TAG, e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -304,8 +316,7 @@ public class CameraView extends FrameLayout {
|
||||
if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
|
||||
displayOrientation = (info.orientation + degrees ) % 360;
|
||||
displayOrientation = (360 - displayOrientation) % 360;
|
||||
}
|
||||
else {
|
||||
} else {
|
||||
displayOrientation = (info.orientation - degrees + 360) % 360;
|
||||
}
|
||||
|
||||
@@ -401,7 +412,7 @@ public class CameraView extends FrameLayout {
|
||||
Log.w(TAG, "previewFormat: " + camera.getParameters().getPreviewFormat());
|
||||
Log.w(TAG, "croppingRect: " + croppingRect.toString());
|
||||
Log.w(TAG, "rotation: " + rotation);
|
||||
new RotatePreviewAsyncTask(previewSize, rotation, croppingRect).execute(data);
|
||||
new CaptureTask(previewSize, rotation, croppingRect).execute(data);
|
||||
}
|
||||
});
|
||||
}
|
||||
@@ -430,7 +441,6 @@ public class CameraView extends FrameLayout {
|
||||
return visibleRect;
|
||||
}
|
||||
|
||||
|
||||
@SuppressWarnings("SuspiciousNameCombination")
|
||||
private void rotateRect(Rect rect) {
|
||||
rect.set(rect.top, rect.left, rect.bottom, rect.right);
|
||||
@@ -495,12 +505,12 @@ public class CameraView extends FrameLayout {
|
||||
}
|
||||
}
|
||||
|
||||
private class RotatePreviewAsyncTask extends AsyncTask<byte[], Void, byte[]> {
|
||||
private class CaptureTask extends AsyncTask<byte[], Void, byte[]> {
|
||||
private final Size previewSize;
|
||||
private final int rotation;
|
||||
private final Rect croppingRect;
|
||||
|
||||
public RotatePreviewAsyncTask(Size previewSize, int rotation, Rect croppingRect) {
|
||||
public CaptureTask(Size previewSize, int rotation, Rect croppingRect) {
|
||||
this.previewSize = previewSize;
|
||||
this.rotation = rotation;
|
||||
this.croppingRect = croppingRect;
|
||||
@@ -516,6 +526,7 @@ public class CameraView extends FrameLayout {
|
||||
rotation,
|
||||
croppingRect);
|
||||
} catch (IOException e) {
|
||||
Log.w(TAG, e);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user