mirror of
https://github.com/SimpleMobileTools/Simple-Camera.git
synced 2025-04-03 18:21:06 +02:00
785 lines
26 KiB
Java
785 lines
26 KiB
Java
package com.simplemobiletools.camera;
|
|
|
|
import android.content.Context;
|
|
import android.graphics.Point;
|
|
import android.graphics.Rect;
|
|
import android.hardware.Camera;
|
|
import android.media.AudioManager;
|
|
import android.media.CamcorderProfile;
|
|
import android.media.MediaActionSound;
|
|
import android.media.MediaRecorder;
|
|
import android.media.MediaScannerConnection;
|
|
import android.net.Uri;
|
|
import android.os.Handler;
|
|
import android.util.Log;
|
|
import android.view.MotionEvent;
|
|
import android.view.ScaleGestureDetector;
|
|
import android.view.Surface;
|
|
import android.view.SurfaceHolder;
|
|
import android.view.SurfaceView;
|
|
import android.view.View;
|
|
import android.view.ViewGroup;
|
|
|
|
import com.simplemobiletools.camera.activities.MainActivity;
|
|
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.io.Serializable;
|
|
import java.util.ArrayList;
|
|
import java.util.Collections;
|
|
import java.util.Comparator;
|
|
import java.util.List;
|
|
|
|
public class Preview extends ViewGroup
|
|
implements SurfaceHolder.Callback, View.OnTouchListener, View.OnClickListener, MediaScannerConnection.OnScanCompletedListener {
|
|
public static final int PHOTO_PREVIEW_LENGTH = 1000;
|
|
private static final String TAG = Preview.class.getSimpleName();
|
|
private static final int FOCUS_AREA_SIZE = 100;
|
|
private static final float RATIO_TOLERANCE = 0.1f;
|
|
|
|
private static SurfaceHolder mSurfaceHolder;
|
|
private static Camera mCamera;
|
|
private static List<Camera.Size> mSupportedPreviewSizes;
|
|
private static SurfaceView mSurfaceView;
|
|
private static Camera.Size mPreviewSize;
|
|
private static MainActivity mActivity;
|
|
private static Camera.Parameters mParameters;
|
|
private static PreviewListener mCallback;
|
|
private static MediaRecorder mRecorder;
|
|
private static String mCurVideoPath;
|
|
private static Point mScreenSize;
|
|
private static Uri mTargetUri;
|
|
private static Context mContext;
|
|
private static ScaleGestureDetector mScaleGestureDetector;
|
|
private static List<Integer> mZoomRatios;
|
|
|
|
private static boolean mCanTakePicture;
|
|
private static boolean mIsFlashEnabled;
|
|
private static boolean mIsRecording;
|
|
private static boolean mIsVideoMode;
|
|
private static boolean mIsSurfaceCreated;
|
|
private static boolean mSwitchToVideoAsap;
|
|
private static boolean mSetupPreviewAfterMeasure;
|
|
private static boolean mForceAspectRatio;
|
|
private static boolean mWasZooming;
|
|
private static int mLastClickX;
|
|
private static int mLastClickY;
|
|
private static int mInitVideoRotation;
|
|
private static int mCurrCameraId;
|
|
private static int mMaxZoom;
|
|
|
|
public Preview(Context context) {
|
|
super(context);
|
|
}
|
|
|
|
public Preview(MainActivity activity, SurfaceView surfaceView, PreviewListener previewListener) {
|
|
super(activity);
|
|
|
|
mActivity = activity;
|
|
mCallback = previewListener;
|
|
mSurfaceView = surfaceView;
|
|
mSurfaceHolder = mSurfaceView.getHolder();
|
|
mSurfaceHolder.addCallback(this);
|
|
mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
|
|
mCanTakePicture = false;
|
|
mSurfaceView.setOnTouchListener(this);
|
|
mSurfaceView.setOnClickListener(this);
|
|
mIsFlashEnabled = false;
|
|
mIsVideoMode = false;
|
|
mIsSurfaceCreated = false;
|
|
mSetupPreviewAfterMeasure = false;
|
|
mCurVideoPath = "";
|
|
mScreenSize = Utils.getScreenSize(mActivity);
|
|
mContext = getContext();
|
|
initGestureDetector();
|
|
}
|
|
|
|
public void trySwitchToVideo() {
|
|
if (mIsSurfaceCreated) {
|
|
initRecorder();
|
|
} else {
|
|
mSwitchToVideoAsap = true;
|
|
}
|
|
}
|
|
|
|
public boolean setCamera(int cameraId) {
|
|
mCurrCameraId = cameraId;
|
|
Camera newCamera;
|
|
try {
|
|
newCamera = Camera.open(cameraId);
|
|
mCallback.setIsCameraAvailable(true);
|
|
} catch (Exception e) {
|
|
Utils.showToast(mContext, R.string.camera_open_error);
|
|
Log.e(TAG, "setCamera open " + e.getMessage());
|
|
mCallback.setIsCameraAvailable(false);
|
|
return false;
|
|
}
|
|
|
|
if (mCamera == newCamera) {
|
|
return false;
|
|
}
|
|
|
|
releaseCamera();
|
|
mCamera = newCamera;
|
|
if (mCamera != null) {
|
|
mParameters = mCamera.getParameters();
|
|
mMaxZoom = mParameters.getMaxZoom();
|
|
mZoomRatios = mParameters.getZoomRatios();
|
|
mSupportedPreviewSizes = mParameters.getSupportedPreviewSizes();
|
|
Collections.sort(mSupportedPreviewSizes, new SizesComparator());
|
|
requestLayout();
|
|
invalidate();
|
|
mSetupPreviewAfterMeasure = true;
|
|
|
|
final List<String> focusModes = mParameters.getSupportedFocusModes();
|
|
if (focusModes.contains(Camera.Parameters.FOCUS_MODE_CONTINUOUS_PICTURE))
|
|
mParameters.setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_PICTURE);
|
|
|
|
final int rotation = getPreviewRotation(cameraId);
|
|
mCamera.setDisplayOrientation(rotation);
|
|
mCamera.setParameters(mParameters);
|
|
|
|
if (mCanTakePicture) {
|
|
try {
|
|
mCamera.setPreviewDisplay(mSurfaceHolder);
|
|
} catch (IOException e) {
|
|
Log.e(TAG, "setCamera setPreviewDisplay " + e.getMessage());
|
|
return false;
|
|
}
|
|
}
|
|
|
|
mCallback.setFlashAvailable(Utils.hasFlash(mCamera));
|
|
}
|
|
|
|
if (mIsVideoMode) {
|
|
initRecorder();
|
|
}
|
|
|
|
final Config config = Config.newInstance(mContext);
|
|
mForceAspectRatio = config.getForceRatioEnabled();
|
|
|
|
return true;
|
|
}
|
|
|
|
public void setTargetUri(Uri uri) {
|
|
mTargetUri = uri;
|
|
}
|
|
|
|
private void initGestureDetector() {
|
|
mScaleGestureDetector = new ScaleGestureDetector(mContext, new ScaleGestureDetector.SimpleOnScaleGestureListener() {
|
|
@Override
|
|
public boolean onScale(ScaleGestureDetector detector) {
|
|
int zoomFactor = mParameters.getZoom();
|
|
float zoomRatio = mZoomRatios.get(zoomFactor) / 100.f;
|
|
zoomRatio *= detector.getScaleFactor();
|
|
|
|
int newZoomFactor = zoomFactor;
|
|
if (zoomRatio <= 1.f) {
|
|
newZoomFactor = 0;
|
|
} else if (zoomRatio >= mZoomRatios.get(mMaxZoom) / 100.f) {
|
|
newZoomFactor = mMaxZoom;
|
|
} else {
|
|
if (detector.getScaleFactor() > 1.f) {
|
|
for (int i = zoomFactor; i < mZoomRatios.size(); i++) {
|
|
if (mZoomRatios.get(i) / 100.0f >= zoomRatio) {
|
|
newZoomFactor = i;
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
for (int i = zoomFactor; i >= 0; i--) {
|
|
if (mZoomRatios.get(i) / 100.0f <= zoomRatio) {
|
|
newZoomFactor = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
newZoomFactor = Math.max(newZoomFactor, 0);
|
|
newZoomFactor = Math.min(mMaxZoom, newZoomFactor);
|
|
|
|
mParameters.setZoom(newZoomFactor);
|
|
if (mCamera != null)
|
|
mCamera.setParameters(mParameters);
|
|
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public void onScaleEnd(ScaleGestureDetector detector) {
|
|
super.onScaleEnd(detector);
|
|
mWasZooming = true;
|
|
mSurfaceView.setSoundEffectsEnabled(false);
|
|
mParameters.setFocusAreas(null);
|
|
}
|
|
});
|
|
}
|
|
|
|
private static int getPreviewRotation(int cameraId) {
|
|
final Camera.CameraInfo info = Utils.getCameraInfo(cameraId);
|
|
final int degrees = getRotationDegrees();
|
|
|
|
int result;
|
|
if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
|
|
result = (info.orientation + degrees) % 360;
|
|
result = 360 - result;
|
|
} else {
|
|
result = info.orientation - degrees + 360;
|
|
}
|
|
|
|
return result % 360;
|
|
}
|
|
|
|
private static int getMediaRotation(int cameraId) {
|
|
final int degrees = getRotationDegrees();
|
|
final Camera.CameraInfo info = Utils.getCameraInfo(cameraId);
|
|
if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
|
|
return (360 + info.orientation + degrees) % 360;
|
|
}
|
|
|
|
return (360 + info.orientation - degrees) % 360;
|
|
}
|
|
|
|
private static int getRotationDegrees() {
|
|
int rotation = mActivity.getWindowManager().getDefaultDisplay().getRotation();
|
|
switch (rotation) {
|
|
case Surface.ROTATION_0:
|
|
return 0;
|
|
case Surface.ROTATION_90:
|
|
return 90;
|
|
case Surface.ROTATION_180:
|
|
return 180;
|
|
case Surface.ROTATION_270:
|
|
return 270;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
public void takePicture() {
|
|
if (mCanTakePicture) {
|
|
if (mIsFlashEnabled) {
|
|
mParameters.setFlashMode(Camera.Parameters.FLASH_MODE_TORCH);
|
|
} else {
|
|
mParameters.setFlashMode(Camera.Parameters.FLASH_MODE_OFF);
|
|
}
|
|
|
|
int rotation = getMediaRotation(mCurrCameraId);
|
|
rotation += compensateDeviceRotation();
|
|
|
|
final Camera.Size maxSize = getOptimalPictureSize();
|
|
mParameters.setPictureSize(maxSize.width, maxSize.height);
|
|
mParameters.setRotation(rotation % 360);
|
|
|
|
if (Config.newInstance(mContext).getIsSoundEnabled()) {
|
|
new MediaActionSound().play(MediaActionSound.SHUTTER_CLICK);
|
|
}
|
|
|
|
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR1) {
|
|
mCamera.enableShutterSound(false);
|
|
}
|
|
|
|
mCamera.setParameters(mParameters);
|
|
mCamera.takePicture(null, null, takePictureCallback);
|
|
}
|
|
mCanTakePicture = false;
|
|
}
|
|
|
|
private Camera.PictureCallback takePictureCallback = new Camera.PictureCallback() {
|
|
@Override
|
|
public void onPictureTaken(byte[] data, Camera cam) {
|
|
new Handler().postDelayed(new Runnable() {
|
|
@Override
|
|
public void run() {
|
|
if (mCamera != null) {
|
|
mCamera.startPreview();
|
|
}
|
|
|
|
mCanTakePicture = true;
|
|
|
|
if (mIsFlashEnabled) {
|
|
mParameters.setFlashMode(Camera.Parameters.FLASH_MODE_OFF);
|
|
mCamera.setParameters(mParameters);
|
|
}
|
|
}
|
|
}, PHOTO_PREVIEW_LENGTH);
|
|
|
|
new PhotoProcessor(mActivity, mTargetUri).execute(data);
|
|
}
|
|
};
|
|
|
|
private Camera.Size getOptimalPictureSize() {
|
|
final int maxResolution = getMaxPhotoResolution();
|
|
final List<Camera.Size> sizes = mParameters.getSupportedPictureSizes();
|
|
Collections.sort(sizes, new SizesComparator());
|
|
Camera.Size maxSize = sizes.get(0);
|
|
for (Camera.Size size : sizes) {
|
|
final boolean isProperRatio = isProperRatio(size);
|
|
final boolean isProperResolution = isProperResolution(size, maxResolution);
|
|
if (isProperResolution && isProperRatio) {
|
|
maxSize = size;
|
|
break;
|
|
}
|
|
}
|
|
return maxSize;
|
|
}
|
|
|
|
private int getMaxPhotoResolution() {
|
|
final int maxRes = Config.newInstance(mContext).getMaxPhotoResolution();
|
|
switch (maxRes) {
|
|
case 0:
|
|
return 6000000;
|
|
case 1:
|
|
return 9000000;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
private boolean isProperResolution(Camera.Size size, int maxRes) {
|
|
return maxRes == 0 || size.width * size.height < maxRes;
|
|
}
|
|
|
|
private int getMaxVideoResolution() {
|
|
final int maxRes = Config.newInstance(mContext).getMaxVideoResolution();
|
|
switch (maxRes) {
|
|
case 0:
|
|
return 400000;
|
|
case 1:
|
|
return 1000000;
|
|
case 2:
|
|
return 2100000;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
private boolean isProperRatio(Camera.Size size) {
|
|
final float currRatio = (float) size.height / size.width;
|
|
float wantedRatio = (float) 3 / 4;
|
|
if (mForceAspectRatio || mIsVideoMode)
|
|
wantedRatio = (float) 9 / 16;
|
|
|
|
final float diff = Math.abs(currRatio - wantedRatio);
|
|
return diff < RATIO_TOLERANCE;
|
|
}
|
|
|
|
private Camera.Size getOptimalVideoSize() {
|
|
final int maxResolution = getMaxVideoResolution();
|
|
final List<Camera.Size> sizes = getSupportedVideoSizes();
|
|
Collections.sort(sizes, new SizesComparator());
|
|
Camera.Size maxSize = sizes.get(0);
|
|
final int cnt = sizes.size();
|
|
for (int i = 0; i < cnt; i++) {
|
|
Camera.Size size = sizes.get(i);
|
|
final boolean isProperRatio = !mForceAspectRatio || isProperRatio(size);
|
|
final boolean isProperResolution = isProperResolution(size, maxResolution);
|
|
if (isProperResolution && isProperRatio) {
|
|
maxSize = size;
|
|
break;
|
|
}
|
|
|
|
if (i == cnt - 1) {
|
|
Utils.showToast(mContext, R.string.no_valid_resolution_found);
|
|
}
|
|
}
|
|
return maxSize;
|
|
}
|
|
|
|
public List<Camera.Size> getSupportedVideoSizes() {
|
|
if (mParameters.getSupportedVideoSizes() != null) {
|
|
return mParameters.getSupportedVideoSizes();
|
|
} else {
|
|
return mParameters.getSupportedPreviewSizes();
|
|
}
|
|
}
|
|
|
|
private int compensateDeviceRotation() {
|
|
int degrees = 0;
|
|
boolean isFrontCamera = (mCurrCameraId == Camera.CameraInfo.CAMERA_FACING_FRONT);
|
|
int deviceOrientation = mCallback.getCurrentOrientation();
|
|
if (deviceOrientation == Constants.ORIENT_LANDSCAPE_LEFT) {
|
|
degrees += isFrontCamera ? 90 : 270;
|
|
} else if (deviceOrientation == Constants.ORIENT_LANDSCAPE_RIGHT) {
|
|
degrees += isFrontCamera ? 270 : 90;
|
|
}
|
|
return degrees;
|
|
}
|
|
|
|
private int getFinalRotation() {
|
|
int rotation = getMediaRotation(mCurrCameraId);
|
|
rotation += compensateDeviceRotation();
|
|
return rotation % 360;
|
|
}
|
|
|
|
private void focusArea(final boolean takePictureAfter) {
|
|
if (mCamera == null)
|
|
return;
|
|
|
|
mCamera.cancelAutoFocus();
|
|
final Rect focusRect = calculateFocusArea(mLastClickX, mLastClickY);
|
|
if (mParameters.getMaxNumFocusAreas() > 0) {
|
|
final List<Camera.Area> focusAreas = new ArrayList<>(1);
|
|
focusAreas.add(new Camera.Area(focusRect, 1000));
|
|
mParameters.setFocusAreas(focusAreas);
|
|
mCallback.drawFocusRect(mLastClickX, mLastClickY);
|
|
}
|
|
|
|
mCamera.setParameters(mParameters);
|
|
mCamera.autoFocus(new Camera.AutoFocusCallback() {
|
|
@Override
|
|
public void onAutoFocus(boolean success, Camera camera) {
|
|
camera.cancelAutoFocus();
|
|
final List<String> focusModes = mParameters.getSupportedFocusModes();
|
|
if (focusModes.contains(Camera.Parameters.FOCUS_MODE_CONTINUOUS_PICTURE))
|
|
mParameters.setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_PICTURE);
|
|
|
|
camera.setParameters(mParameters);
|
|
if (takePictureAfter) {
|
|
takePicture();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
private Rect calculateFocusArea(float x, float y) {
|
|
int left = Float.valueOf((x / mSurfaceView.getWidth()) * 2000 - 1000).intValue();
|
|
int top = Float.valueOf((y / mSurfaceView.getHeight()) * 2000 - 1000).intValue();
|
|
|
|
int tmp = left;
|
|
left = top;
|
|
top = -tmp;
|
|
|
|
final int rectLeft = Math.max(left - FOCUS_AREA_SIZE / 2, -1000);
|
|
final int rectTop = Math.max(top - FOCUS_AREA_SIZE / 2, -1000);
|
|
final int rectRight = Math.min(left + FOCUS_AREA_SIZE / 2, 1000);
|
|
final int rectBottom = Math.min(top + FOCUS_AREA_SIZE / 2, 1000);
|
|
return new Rect(rectLeft, rectTop, rectRight, rectBottom);
|
|
}
|
|
|
|
public void releaseCamera() {
|
|
stopRecording();
|
|
|
|
if (mCamera != null) {
|
|
mCamera.stopPreview();
|
|
mCamera.release();
|
|
mCamera = null;
|
|
}
|
|
|
|
cleanupRecorder();
|
|
}
|
|
|
|
@Override
|
|
public void surfaceCreated(SurfaceHolder holder) {
|
|
mIsSurfaceCreated = true;
|
|
try {
|
|
if (mCamera != null) {
|
|
mCamera.setPreviewDisplay(mSurfaceHolder);
|
|
}
|
|
|
|
if (mSwitchToVideoAsap)
|
|
initRecorder();
|
|
} catch (IOException e) {
|
|
Log.e(TAG, "surfaceCreated IOException " + e.getMessage());
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
|
|
mIsSurfaceCreated = true;
|
|
|
|
if (mIsVideoMode) {
|
|
initRecorder();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void surfaceDestroyed(SurfaceHolder holder) {
|
|
mIsSurfaceCreated = false;
|
|
if (mCamera != null) {
|
|
mCamera.stopPreview();
|
|
}
|
|
|
|
cleanupRecorder();
|
|
}
|
|
|
|
private void setupPreview() {
|
|
mCanTakePicture = true;
|
|
if (mCamera != null && mPreviewSize != null) {
|
|
mParameters.setPreviewSize(mPreviewSize.width, mPreviewSize.height);
|
|
mCamera.setParameters(mParameters);
|
|
mCamera.startPreview();
|
|
}
|
|
}
|
|
|
|
private void cleanupRecorder() {
|
|
if (mRecorder != null) {
|
|
if (mIsRecording) {
|
|
stopRecording();
|
|
}
|
|
|
|
mRecorder.release();
|
|
mRecorder = null;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onLayout(boolean changed, int l, int t, int r, int b) {
|
|
|
|
}
|
|
|
|
private Camera.Size getOptimalPreviewSize(List<Camera.Size> sizes, int width, int height) {
|
|
Camera.Size result = null;
|
|
for (Camera.Size size : sizes) {
|
|
if (size.width <= width && size.height <= height) {
|
|
if (result == null) {
|
|
result = size;
|
|
} else {
|
|
int resultArea = result.width * result.height;
|
|
int newArea = size.width * size.height;
|
|
|
|
if (newArea > resultArea) {
|
|
result = size;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
@Override
|
|
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
|
|
setMeasuredDimension(mScreenSize.x, mScreenSize.y);
|
|
|
|
if (mSupportedPreviewSizes != null) {
|
|
// for simplicity lets assume that most displays are 16:9 and the remaining ones are 4:3
|
|
// always set 16:9 for videos as many devices support 4:3 only in low quality
|
|
if (mForceAspectRatio || mIsVideoMode) {
|
|
mPreviewSize = getOptimalPreviewSize(mSupportedPreviewSizes, mScreenSize.y, mScreenSize.x);
|
|
} else {
|
|
final int newRatioHeight = (int) (mScreenSize.x * ((double) 4 / 3));
|
|
setMeasuredDimension(mScreenSize.x, newRatioHeight);
|
|
mPreviewSize = getOptimalPreviewSize(mSupportedPreviewSizes, newRatioHeight, mScreenSize.x);
|
|
}
|
|
final LayoutParams lp = mSurfaceView.getLayoutParams();
|
|
|
|
// make sure to occupy whole width in every case
|
|
if (mScreenSize.x > mPreviewSize.height) {
|
|
final float ratio = (float) mScreenSize.x / mPreviewSize.height;
|
|
lp.width = (int) (mPreviewSize.height * ratio);
|
|
if (mForceAspectRatio || mIsVideoMode) {
|
|
lp.height = mScreenSize.y;
|
|
} else {
|
|
lp.height = (int) (mPreviewSize.width * ratio);
|
|
}
|
|
} else {
|
|
lp.width = mPreviewSize.height;
|
|
lp.height = mPreviewSize.width;
|
|
}
|
|
|
|
if (mSetupPreviewAfterMeasure) {
|
|
mSetupPreviewAfterMeasure = false;
|
|
if (mCamera != null)
|
|
mCamera.stopPreview();
|
|
|
|
setupPreview();
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean onTouch(View v, MotionEvent event) {
|
|
mLastClickX = (int) event.getX();
|
|
mLastClickY = (int) event.getY();
|
|
|
|
if (mMaxZoom > 0)
|
|
mScaleGestureDetector.onTouchEvent(event);
|
|
return false;
|
|
}
|
|
|
|
public void enableFlash() {
|
|
if (mIsVideoMode) {
|
|
mParameters.setFlashMode(Camera.Parameters.FLASH_MODE_TORCH);
|
|
mCamera.setParameters(mParameters);
|
|
}
|
|
|
|
mIsFlashEnabled = true;
|
|
}
|
|
|
|
public void disableFlash() {
|
|
mIsFlashEnabled = false;
|
|
mParameters.setFlashMode(Camera.Parameters.FLASH_MODE_OFF);
|
|
mCamera.setParameters(mParameters);
|
|
}
|
|
|
|
public void initPhotoMode() {
|
|
stopRecording();
|
|
cleanupRecorder();
|
|
mIsRecording = false;
|
|
mIsVideoMode = false;
|
|
recheckAspectRatio();
|
|
}
|
|
|
|
private void recheckAspectRatio() {
|
|
if (!mForceAspectRatio) {
|
|
mSetupPreviewAfterMeasure = true;
|
|
invalidate();
|
|
requestLayout();
|
|
}
|
|
}
|
|
|
|
// VIDEO RECORDING
|
|
public boolean initRecorder() {
|
|
if (mCamera == null || mRecorder != null || !mIsSurfaceCreated)
|
|
return false;
|
|
|
|
mSwitchToVideoAsap = false;
|
|
Camera.Size preferred = mParameters.getPreferredPreviewSizeForVideo();
|
|
|
|
if (preferred == null) {
|
|
final List<Camera.Size> previewSizes = mParameters.getSupportedPreviewSizes();
|
|
Collections.sort(previewSizes, new SizesComparator());
|
|
preferred = previewSizes.get(0);
|
|
}
|
|
|
|
mParameters.setPreviewSize(preferred.width, preferred.height);
|
|
mCamera.setParameters(mParameters);
|
|
|
|
mIsRecording = false;
|
|
mIsVideoMode = true;
|
|
recheckAspectRatio();
|
|
mRecorder = new MediaRecorder();
|
|
mRecorder.setCamera(mCamera);
|
|
mRecorder.setVideoSource(MediaRecorder.VideoSource.DEFAULT);
|
|
mRecorder.setAudioSource(MediaRecorder.AudioSource.DEFAULT);
|
|
|
|
mCurVideoPath = Utils.getOutputMediaFile(mContext, false);
|
|
if (mCurVideoPath.isEmpty()) {
|
|
Utils.showToast(mContext, R.string.video_creating_error);
|
|
return false;
|
|
}
|
|
|
|
final Camera.Size videoSize = getOptimalVideoSize();
|
|
final CamcorderProfile cpHigh = CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH);
|
|
cpHigh.videoFrameWidth = videoSize.width;
|
|
cpHigh.videoFrameHeight = videoSize.height;
|
|
mRecorder.setProfile(cpHigh);
|
|
mRecorder.setOutputFile(mCurVideoPath);
|
|
mRecorder.setPreviewDisplay(mSurfaceHolder.getSurface());
|
|
|
|
int rotation = getFinalRotation();
|
|
mInitVideoRotation = rotation;
|
|
mRecorder.setOrientationHint(rotation);
|
|
|
|
try {
|
|
mRecorder.prepare();
|
|
} catch (Exception e) {
|
|
Utils.showToast(mContext, R.string.video_setup_error);
|
|
Log.e(TAG, "initRecorder " + e.getMessage());
|
|
releaseCamera();
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
public boolean toggleRecording() {
|
|
if (mIsRecording) {
|
|
stopRecording();
|
|
initRecorder();
|
|
} else {
|
|
startRecording();
|
|
}
|
|
return mIsRecording;
|
|
}
|
|
|
|
private void startRecording() {
|
|
if (mInitVideoRotation != getFinalRotation()) {
|
|
cleanupRecorder();
|
|
initRecorder();
|
|
}
|
|
|
|
try {
|
|
mCamera.unlock();
|
|
toggleShutterSound(true);
|
|
mRecorder.start();
|
|
toggleShutterSound(false);
|
|
mIsRecording = true;
|
|
} catch (Exception e) {
|
|
Utils.showToast(mContext, R.string.video_setup_error);
|
|
Log.e(TAG, "toggleRecording " + e.getMessage());
|
|
releaseCamera();
|
|
}
|
|
}
|
|
|
|
private void stopRecording() {
|
|
if (mRecorder != null && mIsRecording) {
|
|
try {
|
|
toggleShutterSound(true);
|
|
mRecorder.stop();
|
|
final String[] paths = {mCurVideoPath};
|
|
MediaScannerConnection.scanFile(mContext, paths, null, this);
|
|
} catch (RuntimeException e) {
|
|
toggleShutterSound(false);
|
|
new File(mCurVideoPath).delete();
|
|
Utils.showToast(mContext, R.string.video_saving_error);
|
|
Log.e(TAG, "stopRecording " + e.getMessage());
|
|
mRecorder = null;
|
|
mIsRecording = false;
|
|
releaseCamera();
|
|
}
|
|
}
|
|
|
|
mRecorder = null;
|
|
mIsRecording = false;
|
|
|
|
final File file = new File(mCurVideoPath);
|
|
if (file.exists() && file.length() == 0) {
|
|
file.delete();
|
|
}
|
|
}
|
|
|
|
private void toggleShutterSound(Boolean mute) {
|
|
if (!Config.newInstance(mContext).getIsSoundEnabled()) {
|
|
((AudioManager) mContext.getSystemService(Context.AUDIO_SERVICE)).setStreamMute(AudioManager.STREAM_SYSTEM, mute);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onClick(View v) {
|
|
if (!mWasZooming)
|
|
focusArea(false);
|
|
|
|
mWasZooming = false;
|
|
mSurfaceView.setSoundEffectsEnabled(true);
|
|
}
|
|
|
|
@Override
|
|
public void onScanCompleted(String path, Uri uri) {
|
|
mCallback.videoSaved(uri);
|
|
toggleShutterSound(false);
|
|
}
|
|
|
|
private static class SizesComparator implements Comparator<Camera.Size>, Serializable {
|
|
private static final long serialVersionUID = 5431278455314658485L;
|
|
|
|
@Override
|
|
public int compare(final Camera.Size a, final Camera.Size b) {
|
|
return b.width * b.height - a.width * a.height;
|
|
}
|
|
}
|
|
|
|
public interface PreviewListener {
|
|
void setFlashAvailable(boolean available);
|
|
|
|
void setIsCameraAvailable(boolean available);
|
|
|
|
int getCurrentOrientation();
|
|
|
|
void videoSaved(Uri uri);
|
|
|
|
void drawFocusRect(int x, int y);
|
|
}
|
|
}
|