/* The contents of this file are subject to the Mozilla Public License Version 1.1 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.mozilla.org/MPL/ Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for the specific language governing rights and limitations under the License. The Original Code is collection of files collectively known as Open Camera. The Initial Developer of the Original Code is Almalence Inc. Portions created by Initial Developer are Copyright (C) 2013 by Almalence Inc. All Rights Reserved. */ package com.almalence.plugins.capture.burst; import java.util.Arrays; import java.util.LinkedHashMap; import java.util.Map.Entry; import android.annotation.TargetApi; import android.app.AlertDialog; import android.content.DialogInterface; import android.content.SharedPreferences; import android.content.SharedPreferences.Editor; import android.hardware.camera2.CaptureResult; import android.preference.PreferenceManager; import android.util.Log; import android.view.Gravity; import android.widget.LinearLayout; import android.widget.Toast; /* <!-- +++ import com.almalence.opencam_plus.cameracontroller.CameraController; import com.almalence.opencam_plus.ui.GUI.CameraParameter; import com.almalence.opencam_plus.CameraParameters; import com.almalence.opencam_plus.ApplicationScreen; import com.almalence.opencam_plus.ApplicationInterface; import com.almalence.opencam_plus.PluginCapture; import com.almalence.opencam_plus.PluginManager; import com.almalence.opencam_plus.R; +++ --> */ // <!-- -+- import com.almalence.opencam.cameracontroller.CameraController; import com.almalence.opencam.ui.GUI.CameraParameter; import com.almalence.opencam.ApplicationInterface; import com.almalence.opencam.CameraParameters; import com.almalence.opencam.ApplicationScreen; import com.almalence.opencam.PluginCapture; import com.almalence.opencam.PluginManager; import com.almalence.opencam.R; //-+- --> import com.almalence.util.HeapUtil; /*** * Implements burst capture plugin - captures predefined number of images ***/ public class BurstCapturePlugin extends PluginCapture { // defaul val. value should come from config private int imageAmount = 3; private int pauseBetweenShots = 0; private int preferenceFlashMode; private static String sImagesAmountPref; private static String sPauseBetweenShotsPref; private static Toast capturingDialog; //That map helps to find suitable amount of RAW frames to be captured in case of low memory protected static final LinkedHashMap<Integer, Integer> IMAGE_AMOUNT_VALUES = new LinkedHashMap<Integer, Integer>() { { put(0, 20); put(1, 15); put(2, 10); put(3, 5); put(4, 3); } }; public BurstCapturePlugin() { super("com.almalence.plugins.burstcapture", R.xml.preferences_capture_burst, 0, R.drawable.gui_almalence_mode_burst, "Burst images"); } @Override public void onCreate() { sImagesAmountPref = ApplicationScreen.getAppResources().getString(R.string.Preference_BurstImagesAmount); sPauseBetweenShotsPref = ApplicationScreen.getAppResources().getString(R.string.Preference_BurstPauseBetweenShots); } @Override public void onStart() { refreshPreferences(); } @Override public void onResume() { imagesTaken = 0; imagesTakenRAW = 0; inCapture = false; aboutToTakePicture = false; isAllImagesTaken = false; isAllCaptureResultsCompleted = true; if (CameraController.isUseCamera2() && CameraController.isNexus5or6) { SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(ApplicationScreen.getMainContext()); preferenceFlashMode = prefs.getInt(ApplicationScreen.sFlashModePref, ApplicationScreen.sDefaultFlashValue); SharedPreferences.Editor editor = prefs.edit(); editor.putInt(ApplicationScreen.sFlashModePref, CameraParameters.FLASH_MODE_OFF); editor.commit(); } // refreshPreferences(); if(captureRAW && CameraController.isRAWCaptureSupported()) ApplicationScreen.setCaptureFormat(CameraController.RAW); else { captureRAW = false; ApplicationScreen.setCaptureFormat(CameraController.JPEG); } } @Override public void onGUICreate() { if (CameraController.isUseCamera2() && CameraController.isNexus5or6) ApplicationScreen.instance.disableCameraParameter(CameraParameter.CAMERA_PARAMETER_FLASH, true, false, true); } @Override public void setupCameraParameters() { //Warn user if current free memory level is not enough to capture all RAW frames //in case of RAW capturing enabled, for other capture formats amount of memory is enough by default if(!checkFreeMemory(imageAmount)) { LinearLayout bottom_layout = (LinearLayout) ApplicationScreen.instance.findViewById(R.id.mainButtons); capturingDialog = Toast.makeText(ApplicationScreen.instance, R.string.not_enough_memory_for_capture, Toast.LENGTH_LONG); capturingDialog.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, bottom_layout.getHeight()); capturingDialog.show(); } try { int[] flashModes = CameraController.getSupportedFlashModes(); if (flashModes != null && flashModes.length > 0 && CameraController.isUseCamera2() && CameraController.isNexus5or6) { CameraController.setCameraFlashMode(CameraParameters.FLASH_MODE_OFF); SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(ApplicationScreen.getMainContext()); SharedPreferences.Editor editor = prefs.edit(); editor.putInt(ApplicationScreen.sFlashModePref, CameraParameters.FLASH_MODE_OFF); editor.commit(); } } catch (RuntimeException e) { Log.e("Burst capture", "ApplicationScreen.setupCamera unable to setFlashMode"); } } @Override public void onPause() { if (CameraController.isUseCamera2() && CameraController.isNexus5or6) { SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(ApplicationScreen.getMainContext()); prefs.edit().putInt(ApplicationScreen.sFlashModePref, preferenceFlashMode).commit(); CameraController.setCameraFlashMode(preferenceFlashMode); } } private void refreshPreferences() { try { SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(ApplicationScreen.getMainContext()); imageAmount = Integer.parseInt(prefs.getString(sImagesAmountPref, "3")); pauseBetweenShots = Integer.parseInt(prefs.getString(sPauseBetweenShotsPref, "0")); captureRAW = prefs.getBoolean(ApplicationScreen.sCaptureRAWPref, false); } catch (Exception e) { Log.e("Burst capture", "Cought exception " + e.getMessage()); } switch (imageAmount) { case 3: quickControlIconID = R.drawable.gui_almalence_mode_burst3; break; case 5: quickControlIconID = R.drawable.gui_almalence_mode_burst5; break; case 10: quickControlIconID = R.drawable.gui_almalence_mode_burst10; break; case 15: quickControlIconID = R.drawable.gui_almalence_mode_burst15; break; case 20: quickControlIconID = R.drawable.gui_almalence_mode_burst20; break; default: break; } } @Override public void onQuickControlClick() { SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(ApplicationScreen.getMainContext()); int val = Integer.parseInt(prefs.getString(sImagesAmountPref, "1")); int selected = 0; switch (val) { case 3: selected = 0; break; case 5: selected = 1; break; case 10: selected = 2; break; case 15: selected = 3; break; case 20: selected = 4; break; default: break; } selected = (selected + 1) % 5; Editor editor = prefs.edit(); switch (selected) { case 0: quickControlIconID = R.drawable.gui_almalence_mode_burst3; editor.putString("burstImagesAmount", "3"); imageAmount = 3; break; case 1: quickControlIconID = R.drawable.gui_almalence_mode_burst5; editor.putString("burstImagesAmount", "5"); imageAmount = 5; break; case 2: quickControlIconID = R.drawable.gui_almalence_mode_burst10; editor.putString("burstImagesAmount", "10"); imageAmount = 10; break; case 3: quickControlIconID = R.drawable.gui_almalence_mode_burst15; editor.putString("burstImagesAmount", "15"); imageAmount = 15; break; case 4: quickControlIconID = R.drawable.gui_almalence_mode_burst20; editor.putString("burstImagesAmount", "20"); imageAmount = 20; break; default: break; } editor.commit(); //Warn user if current free memory level is not enough to capture all RAW frames //in case of RAW capturing enabled, for other capture formats amount of memory is enough by default if(!checkFreeMemory(imageAmount)) { LinearLayout bottom_layout = (LinearLayout) ApplicationScreen.instance.findViewById(R.id.mainButtons); capturingDialog = Toast.makeText(ApplicationScreen.instance, R.string.not_enough_memory_for_capture, Toast.LENGTH_LONG); capturingDialog.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, bottom_layout.getHeight()); capturingDialog.show(); } } public boolean delayedCaptureSupported() { return true; } public void takePicture() { refreshPreferences(); inCapture = true; resultCompleted = 0; if (captureRAW) { //Some device (such as LG G Flex 2 has a bad memory management.) //As a result on such devices is impossible to capture all set of amount of RAW frames. //To prevent crash we used 'reduced amount of frames' logic: //Checking one by one RAW frames amount starting from initial imageAmount to suit current free memory //If current RAW amount can't be captured we try to capture less images. //If we can't capture even 3 RAW picture we capture only JPEG frames but initial amount. //Find index of current imageAmount in helper's map int idx = IMAGE_AMOUNT_VALUES.size(); for (Entry<Integer, Integer> entry : IMAGE_AMOUNT_VALUES.entrySet()) { if (imageAmount == entry.getValue()) { idx = entry.getKey(); break; } } //Iterate trough image amount map. Values is reduced on each step for(int i = idx; i < IMAGE_AMOUNT_VALUES.size(); i++) { int imageAmountChecked = IMAGE_AMOUNT_VALUES.get(i); if(checkFreeMemory(imageAmountChecked)) { //Checked image amount is suitable for current memory level //If checked image amount is less than initial then warn user about that and change initial image amount to suitable amount if(imageAmount > imageAmountChecked) { LinearLayout bottom_layout = (LinearLayout) ApplicationScreen.instance.findViewById(R.id.mainButtons); capturingDialog = Toast.makeText(ApplicationScreen.instance, R.string.capture_less_raw, Toast.LENGTH_LONG); capturingDialog.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, bottom_layout.getHeight()); capturingDialog.show(); imageAmount = imageAmountChecked; } final int[] pause = new int[imageAmount]; Arrays.fill(pause, pauseBetweenShots); createRequestIDList(imageAmount * 2); CameraController.captureImagesWithParams(imageAmount, CameraController.RAW, pause, null, null, null, false, true, true); return; } } //If no one RAW frame can be captured, capture JPEG frames. LinearLayout bottom_layout = (LinearLayout) ApplicationScreen.instance.findViewById(R.id.mainButtons); capturingDialog = Toast.makeText(ApplicationScreen.instance, R.string.capture_only_jpeg, Toast.LENGTH_LONG); capturingDialog.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, bottom_layout.getHeight()); capturingDialog.show(); } final int[] pause = new int[imageAmount]; Arrays.fill(pause, pauseBetweenShots); createRequestIDList(imageAmount); CameraController.captureImagesWithParams(imageAmount, CameraController.JPEG, pause, null, null, null, false, true, true); } @Override public void onImageTaken(int frame, byte[] frameData, int frame_len, int format) { if (frame == 0) { Log.d("Burst", "Load to heap failed"); imagesTaken = 0; imagesTakenRAW = 0; resultCompleted = 0; isAllImagesTaken = false; inCapture = false; ApplicationScreen.instance.muteShutter(false); PluginManager.getInstance().sendMessage(ApplicationInterface.MSG_CAPTURE_FINISHED_NORESULT, String.valueOf(SessionID)); return; } boolean isRAW = (format == CameraController.RAW); if (isRAW) imagesTakenRAW++; imagesTaken++; PluginManager.getInstance().addToSharedMem("frame" + imagesTaken + SessionID, String.valueOf(frame)); PluginManager.getInstance().addToSharedMem("framelen" + imagesTaken + SessionID, String.valueOf(frame_len)); PluginManager.getInstance().addToSharedMem("frameisraw" + imagesTaken + SessionID, String.valueOf(isRAW)); PluginManager.getInstance().addToSharedMem("frameorientation" + imagesTaken + SessionID, String.valueOf(ApplicationScreen.getGUIManager().getImageDataOrientation())); PluginManager.getInstance().addToSharedMem("framemirrored" + imagesTaken + SessionID, String.valueOf(CameraController.isFrontCamera())); try { CameraController.startCameraPreview(); } catch (RuntimeException e) { Log.e("Burst", "StartPreview fail"); PluginManager.getInstance().sendMessage(ApplicationInterface.MSG_CAPTURE_FINISHED, String.valueOf(SessionID)); imagesTaken = 0; imagesTakenRAW = 0; resultCompleted = 0; ApplicationScreen.instance.muteShutter(false); return; } if ((captureRAW && imagesTaken >= (imageAmount * 2)) || (!captureRAW && imagesTaken >= imageAmount)) { if(isAllCaptureResultsCompleted) { PluginManager.getInstance().addToSharedMem("amountofcapturedframes" + SessionID, String.valueOf(imagesTaken)); PluginManager.getInstance().addToSharedMem("amountofcapturedrawframes" + SessionID, String.valueOf(imagesTakenRAW)); PluginManager.getInstance().sendMessage(ApplicationInterface.MSG_CAPTURE_FINISHED, String.valueOf(SessionID)); imagesTaken = 0; imagesTakenRAW = 0; resultCompleted = 0; inCapture = false; isAllImagesTaken = false; } else isAllImagesTaken = true; } } @TargetApi(21) @Override public void onCaptureCompleted(CaptureResult result) { isAllCaptureResultsCompleted = false; int requestID = requestIDArray[resultCompleted]; resultCompleted++; if (result.getSequenceId() == requestID) { PluginManager.getInstance().addToSharedMemExifTagsFromCaptureResult(result, SessionID, resultCompleted); } if (captureRAW) PluginManager.getInstance().addRAWCaptureResultToSharedMem("captureResult" + resultCompleted + SessionID, result); if ((captureRAW && resultCompleted >= (imageAmount * 2)) || (!captureRAW && resultCompleted >= imageAmount)) { isAllCaptureResultsCompleted = true; if(isAllImagesTaken) { PluginManager.getInstance().addToSharedMem("amountofcapturedframes" + SessionID, String.valueOf(imagesTaken)); PluginManager.getInstance().addToSharedMem("amountofcapturedrawframes" + SessionID, String.valueOf(imagesTakenRAW)); PluginManager.getInstance().sendMessage(ApplicationInterface.MSG_CAPTURE_FINISHED, String.valueOf(SessionID)); inCapture = false; resultCompleted = 0; imagesTaken = 0; isAllImagesTaken = false; } } } @Override public void onPreviewFrame(byte[] data) { } //On some devices which supports RAW capturing may be not enough free memory to capture several RAW //So we need to check available size of memory and compare it to approximate size of all RAWs to be captured private boolean checkFreeMemory(int imgAmount) { if(captureRAW) { CameraController.Size imageSize = CameraController.getCameraImageSize(); int imageWidth = imageSize.getWidth(); int imageHeight = imageSize.getHeight(); final int freeMemoryAprox = (int) (HeapUtil.getAmountOfMemoryToFitFrames() - (HeapUtil.getRAWFrameSizeInBytes(imageWidth, imageHeight)*imgAmount)); if(freeMemoryAprox < 20000000.f) //Left 20 Mb for safety reason - to prevent unexpected system's behavior return false; } return true; } }