| /* |
| * Copyright (C) 2009 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.android.camera; |
| |
| import android.annotation.TargetApi; |
| import android.app.Activity; |
| import android.content.Context; |
| import android.content.SharedPreferences; |
| import android.content.SharedPreferences.Editor; |
| import android.content.res.Resources; |
| import android.content.res.TypedArray; |
| import android.hardware.Camera.CameraInfo; |
| import android.hardware.Camera.Parameters; |
| import android.hardware.Camera.Size; |
| import android.media.CamcorderProfile; |
| import android.util.FloatMath; |
| import android.util.Log; |
| |
| import com.android.gallery3d.common.ApiHelper; |
| |
| import java.util.ArrayList; |
| import java.util.List; |
| import java.util.Locale; |
| |
| /** |
| * Provides utilities and keys for Camera settings. |
| */ |
| public class CameraSettings { |
| private static final int NOT_FOUND = -1; |
| |
| public static final String KEY_VERSION = "pref_version_key"; |
| public static final String KEY_LOCAL_VERSION = "pref_local_version_key"; |
| public static final String KEY_RECORD_LOCATION = RecordLocationPreference.KEY; |
| public static final String KEY_VIDEO_QUALITY = "pref_video_quality_key"; |
| public static final String KEY_VIDEO_TIME_LAPSE_FRAME_INTERVAL = "pref_video_time_lapse_frame_interval_key"; |
| public static final String KEY_PICTURE_SIZE = "pref_camera_picturesize_key"; |
| public static final String KEY_JPEG_QUALITY = "pref_camera_jpegquality_key"; |
| public static final String KEY_FOCUS_MODE = "pref_camera_focusmode_key"; |
| public static final String KEY_FLASH_MODE = "pref_camera_flashmode_key"; |
| public static final String KEY_VIDEOCAMERA_FLASH_MODE = "pref_camera_video_flashmode_key"; |
| public static final String KEY_WHITE_BALANCE = "pref_camera_whitebalance_key"; |
| public static final String KEY_SCENE_MODE = "pref_camera_scenemode_key"; |
| public static final String KEY_EXPOSURE = "pref_camera_exposure_key"; |
| public static final String KEY_VIDEO_EFFECT = "pref_video_effect_key"; |
| public static final String KEY_CAMERA_ID = "pref_camera_id_key"; |
| public static final String KEY_CAMERA_HDR = "pref_camera_hdr_key"; |
| public static final String KEY_CAMERA_FIRST_USE_HINT_SHOWN = "pref_camera_first_use_hint_shown_key"; |
| public static final String KEY_VIDEO_FIRST_USE_HINT_SHOWN = "pref_video_first_use_hint_shown_key"; |
| |
| public static final String EXPOSURE_DEFAULT_VALUE = "0"; |
| |
| public static final int CURRENT_VERSION = 5; |
| public static final int CURRENT_LOCAL_VERSION = 2; |
| |
| private static final String TAG = "CameraSettings"; |
| |
| private final Context mContext; |
| private final Parameters mParameters; |
| private final CameraInfo[] mCameraInfo; |
| private final int mCameraId; |
| |
| public CameraSettings(Activity activity, Parameters parameters, |
| int cameraId, CameraInfo[] cameraInfo) { |
| mContext = activity; |
| mParameters = parameters; |
| mCameraId = cameraId; |
| mCameraInfo = cameraInfo; |
| } |
| |
| public PreferenceGroup getPreferenceGroup(int preferenceRes) { |
| PreferenceInflater inflater = new PreferenceInflater(mContext); |
| PreferenceGroup group = |
| (PreferenceGroup) inflater.inflate(preferenceRes); |
| if (mParameters != null) initPreference(group); |
| return group; |
| } |
| |
| @TargetApi(ApiHelper.VERSION_CODES.HONEYCOMB) |
| public static String getDefaultVideoQuality(int cameraId, |
| String defaultQuality) { |
| if (ApiHelper.HAS_FINE_RESOLUTION_QUALITY_LEVELS) { |
| if (CamcorderProfile.hasProfile( |
| cameraId, Integer.valueOf(defaultQuality))) { |
| return defaultQuality; |
| } |
| } |
| return Integer.toString(CamcorderProfile.QUALITY_HIGH); |
| } |
| |
| public static void initialCameraPictureSize( |
| Context context, Parameters parameters) { |
| // When launching the camera app first time, we will set the picture |
| // size to the first one in the list defined in "arrays.xml" and is also |
| // supported by the driver. |
| List<Size> supported = parameters.getSupportedPictureSizes(); |
| if (supported == null) return; |
| for (String candidate : context.getResources().getStringArray( |
| R.array.pref_camera_picturesize_entryvalues)) { |
| if (setCameraPictureSize(candidate, supported, parameters)) { |
| SharedPreferences.Editor editor = ComboPreferences |
| .get(context).edit(); |
| editor.putString(KEY_PICTURE_SIZE, candidate); |
| editor.apply(); |
| return; |
| } |
| } |
| Log.e(TAG, "No supported picture size found"); |
| } |
| |
| public static void removePreferenceFromScreen( |
| PreferenceGroup group, String key) { |
| removePreference(group, key); |
| } |
| |
| public static boolean setCameraPictureSize( |
| String candidate, List<Size> supported, Parameters parameters) { |
| int index = candidate.indexOf('x'); |
| if (index == NOT_FOUND) return false; |
| int width = Integer.parseInt(candidate.substring(0, index)); |
| int height = Integer.parseInt(candidate.substring(index + 1)); |
| for (Size size : supported) { |
| if (size.width == width && size.height == height) { |
| parameters.setPictureSize(width, height); |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| public static int getMaxVideoDuration(Context context) { |
| int duration = 0; // in milliseconds, 0 means unlimited. |
| try { |
| duration = context.getResources().getInteger(R.integer.max_video_recording_length); |
| } catch (Resources.NotFoundException ex) { |
| } |
| return duration; |
| } |
| |
| private void initPreference(PreferenceGroup group) { |
| ListPreference videoQuality = group.findPreference(KEY_VIDEO_QUALITY); |
| ListPreference timeLapseInterval = group.findPreference(KEY_VIDEO_TIME_LAPSE_FRAME_INTERVAL); |
| ListPreference pictureSize = group.findPreference(KEY_PICTURE_SIZE); |
| ListPreference whiteBalance = group.findPreference(KEY_WHITE_BALANCE); |
| ListPreference sceneMode = group.findPreference(KEY_SCENE_MODE); |
| ListPreference flashMode = group.findPreference(KEY_FLASH_MODE); |
| ListPreference focusMode = group.findPreference(KEY_FOCUS_MODE); |
| IconListPreference exposure = |
| (IconListPreference) group.findPreference(KEY_EXPOSURE); |
| IconListPreference cameraIdPref = |
| (IconListPreference) group.findPreference(KEY_CAMERA_ID); |
| ListPreference videoFlashMode = |
| group.findPreference(KEY_VIDEOCAMERA_FLASH_MODE); |
| ListPreference videoEffect = group.findPreference(KEY_VIDEO_EFFECT); |
| ListPreference cameraHdr = group.findPreference(KEY_CAMERA_HDR); |
| |
| // Since the screen could be loaded from different resources, we need |
| // to check if the preference is available here |
| if (videoQuality != null) { |
| filterUnsupportedOptions(group, videoQuality, getSupportedVideoQuality()); |
| } |
| |
| if (pictureSize != null) { |
| filterUnsupportedOptions(group, pictureSize, sizeListToStringList( |
| mParameters.getSupportedPictureSizes())); |
| filterSimilarPictureSize(group, pictureSize); |
| } |
| if (whiteBalance != null) { |
| filterUnsupportedOptions(group, |
| whiteBalance, mParameters.getSupportedWhiteBalance()); |
| } |
| if (sceneMode != null) { |
| filterUnsupportedOptions(group, |
| sceneMode, mParameters.getSupportedSceneModes()); |
| } |
| if (flashMode != null) { |
| filterUnsupportedOptions(group, |
| flashMode, mParameters.getSupportedFlashModes()); |
| } |
| if (focusMode != null) { |
| if (!Util.isFocusAreaSupported(mParameters)) { |
| filterUnsupportedOptions(group, |
| focusMode, mParameters.getSupportedFocusModes()); |
| } else { |
| // Remove the focus mode if we can use tap-to-focus. |
| removePreference(group, focusMode.getKey()); |
| } |
| } |
| if (videoFlashMode != null) { |
| filterUnsupportedOptions(group, |
| videoFlashMode, mParameters.getSupportedFlashModes()); |
| } |
| if (exposure != null) buildExposureCompensation(group, exposure); |
| if (cameraIdPref != null) buildCameraId(group, cameraIdPref); |
| |
| if (timeLapseInterval != null) { |
| if (ApiHelper.HAS_TIME_LAPSE_RECORDING) { |
| resetIfInvalid(timeLapseInterval); |
| } else { |
| removePreference(group, timeLapseInterval.getKey()); |
| } |
| } |
| if (videoEffect != null) { |
| if (ApiHelper.HAS_EFFECTS_RECORDING) { |
| initVideoEffect(group, videoEffect); |
| resetIfInvalid(videoEffect); |
| } else { |
| filterUnsupportedOptions(group, videoEffect, null); |
| } |
| } |
| if (cameraHdr != null && (!ApiHelper.HAS_CAMERA_HDR |
| || !Util.isCameraHdrSupported(mParameters))) { |
| removePreference(group, cameraHdr.getKey()); |
| } |
| } |
| |
| private void buildExposureCompensation( |
| PreferenceGroup group, IconListPreference exposure) { |
| int max = mParameters.getMaxExposureCompensation(); |
| int min = mParameters.getMinExposureCompensation(); |
| if (max == 0 && min == 0) { |
| removePreference(group, exposure.getKey()); |
| return; |
| } |
| float step = mParameters.getExposureCompensationStep(); |
| |
| // show only integer values for exposure compensation |
| int maxValue = (int) FloatMath.floor(max * step); |
| int minValue = (int) FloatMath.ceil(min * step); |
| CharSequence entries[] = new CharSequence[maxValue - minValue + 1]; |
| CharSequence entryValues[] = new CharSequence[maxValue - minValue + 1]; |
| int[] icons = new int[maxValue - minValue + 1]; |
| TypedArray iconIds = mContext.getResources().obtainTypedArray( |
| R.array.pref_camera_exposure_icons); |
| for (int i = minValue; i <= maxValue; ++i) { |
| entryValues[maxValue - i] = Integer.toString(Math.round(i / step)); |
| StringBuilder builder = new StringBuilder(); |
| if (i > 0) builder.append('+'); |
| entries[maxValue - i] = builder.append(i).toString(); |
| icons[maxValue - i] = iconIds.getResourceId(3 + i, 0); |
| } |
| exposure.setUseSingleIcon(true); |
| exposure.setEntries(entries); |
| exposure.setEntryValues(entryValues); |
| exposure.setLargeIconIds(icons); |
| } |
| |
| private void buildCameraId( |
| PreferenceGroup group, IconListPreference preference) { |
| int numOfCameras = mCameraInfo.length; |
| if (numOfCameras < 2) { |
| removePreference(group, preference.getKey()); |
| return; |
| } |
| |
| CharSequence[] entryValues = new CharSequence[numOfCameras]; |
| for (int i = 0; i < numOfCameras; ++i) { |
| entryValues[i] = "" + i; |
| } |
| preference.setEntryValues(entryValues); |
| } |
| |
| private static boolean removePreference(PreferenceGroup group, String key) { |
| for (int i = 0, n = group.size(); i < n; i++) { |
| CameraPreference child = group.get(i); |
| if (child instanceof PreferenceGroup) { |
| if (removePreference((PreferenceGroup) child, key)) { |
| return true; |
| } |
| } |
| if (child instanceof ListPreference && |
| ((ListPreference) child).getKey().equals(key)) { |
| group.removePreference(i); |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| private void filterUnsupportedOptions(PreferenceGroup group, |
| ListPreference pref, List<String> supported) { |
| |
| // Remove the preference if the parameter is not supported or there is |
| // only one options for the settings. |
| if (supported == null || supported.size() <= 1) { |
| removePreference(group, pref.getKey()); |
| return; |
| } |
| |
| pref.filterUnsupported(supported); |
| if (pref.getEntries().length <= 1) { |
| removePreference(group, pref.getKey()); |
| return; |
| } |
| |
| resetIfInvalid(pref); |
| } |
| |
| private void filterSimilarPictureSize(PreferenceGroup group, |
| ListPreference pref) { |
| pref.filterDuplicated(); |
| if (pref.getEntries().length <= 1) { |
| removePreference(group, pref.getKey()); |
| return; |
| } |
| resetIfInvalid(pref); |
| } |
| |
| private void resetIfInvalid(ListPreference pref) { |
| // Set the value to the first entry if it is invalid. |
| String value = pref.getValue(); |
| if (pref.findIndexOfValue(value) == NOT_FOUND) { |
| pref.setValueIndex(0); |
| } |
| } |
| |
| private static List<String> sizeListToStringList(List<Size> sizes) { |
| ArrayList<String> list = new ArrayList<String>(); |
| for (Size size : sizes) { |
| list.add(String.format(Locale.ENGLISH, "%dx%d", size.width, size.height)); |
| } |
| return list; |
| } |
| |
| public static void upgradeLocalPreferences(SharedPreferences pref) { |
| int version; |
| try { |
| version = pref.getInt(KEY_LOCAL_VERSION, 0); |
| } catch (Exception ex) { |
| version = 0; |
| } |
| if (version == CURRENT_LOCAL_VERSION) return; |
| |
| SharedPreferences.Editor editor = pref.edit(); |
| if (version == 1) { |
| // We use numbers to represent the quality now. The quality definition is identical to |
| // that of CamcorderProfile.java. |
| editor.remove("pref_video_quality_key"); |
| } |
| editor.putInt(KEY_LOCAL_VERSION, CURRENT_LOCAL_VERSION); |
| editor.apply(); |
| } |
| |
| public static void upgradeGlobalPreferences(SharedPreferences pref) { |
| upgradeOldVersion(pref); |
| upgradeCameraId(pref); |
| } |
| |
| private static void upgradeOldVersion(SharedPreferences pref) { |
| int version; |
| try { |
| version = pref.getInt(KEY_VERSION, 0); |
| } catch (Exception ex) { |
| version = 0; |
| } |
| if (version == CURRENT_VERSION) return; |
| |
| SharedPreferences.Editor editor = pref.edit(); |
| if (version == 0) { |
| // We won't use the preference which change in version 1. |
| // So, just upgrade to version 1 directly |
| version = 1; |
| } |
| if (version == 1) { |
| // Change jpeg quality {65,75,85} to {normal,fine,superfine} |
| String quality = pref.getString(KEY_JPEG_QUALITY, "85"); |
| if (quality.equals("65")) { |
| quality = "normal"; |
| } else if (quality.equals("75")) { |
| quality = "fine"; |
| } else { |
| quality = "superfine"; |
| } |
| editor.putString(KEY_JPEG_QUALITY, quality); |
| version = 2; |
| } |
| if (version == 2) { |
| editor.putString(KEY_RECORD_LOCATION, |
| pref.getBoolean(KEY_RECORD_LOCATION, false) |
| ? RecordLocationPreference.VALUE_ON |
| : RecordLocationPreference.VALUE_NONE); |
| version = 3; |
| } |
| if (version == 3) { |
| // Just use video quality to replace it and |
| // ignore the current settings. |
| editor.remove("pref_camera_videoquality_key"); |
| editor.remove("pref_camera_video_duration_key"); |
| } |
| |
| editor.putInt(KEY_VERSION, CURRENT_VERSION); |
| editor.apply(); |
| } |
| |
| private static void upgradeCameraId(SharedPreferences pref) { |
| // The id stored in the preference may be out of range if we are running |
| // inside the emulator and a webcam is removed. |
| // Note: This method accesses the global preferences directly, not the |
| // combo preferences. |
| int cameraId = readPreferredCameraId(pref); |
| if (cameraId == 0) return; // fast path |
| |
| int n = CameraHolder.instance().getNumberOfCameras(); |
| if (cameraId < 0 || cameraId >= n) { |
| writePreferredCameraId(pref, 0); |
| } |
| } |
| |
| public static int readPreferredCameraId(SharedPreferences pref) { |
| return Integer.parseInt(pref.getString(KEY_CAMERA_ID, "0")); |
| } |
| |
| public static void writePreferredCameraId(SharedPreferences pref, |
| int cameraId) { |
| Editor editor = pref.edit(); |
| editor.putString(KEY_CAMERA_ID, Integer.toString(cameraId)); |
| editor.apply(); |
| } |
| |
| public static int readExposure(ComboPreferences preferences) { |
| String exposure = preferences.getString( |
| CameraSettings.KEY_EXPOSURE, |
| EXPOSURE_DEFAULT_VALUE); |
| try { |
| return Integer.parseInt(exposure); |
| } catch (Exception ex) { |
| Log.e(TAG, "Invalid exposure: " + exposure); |
| } |
| return 0; |
| } |
| |
| public static int readEffectType(SharedPreferences pref) { |
| String effectSelection = pref.getString(KEY_VIDEO_EFFECT, "none"); |
| if (effectSelection.equals("none")) { |
| return EffectsRecorder.EFFECT_NONE; |
| } else if (effectSelection.startsWith("goofy_face")) { |
| return EffectsRecorder.EFFECT_GOOFY_FACE; |
| } else if (effectSelection.startsWith("backdropper")) { |
| return EffectsRecorder.EFFECT_BACKDROPPER; |
| } |
| Log.e(TAG, "Invalid effect selection: " + effectSelection); |
| return EffectsRecorder.EFFECT_NONE; |
| } |
| |
| public static Object readEffectParameter(SharedPreferences pref) { |
| String effectSelection = pref.getString(KEY_VIDEO_EFFECT, "none"); |
| if (effectSelection.equals("none")) { |
| return null; |
| } |
| int separatorIndex = effectSelection.indexOf('/'); |
| String effectParameter = |
| effectSelection.substring(separatorIndex + 1); |
| if (effectSelection.startsWith("goofy_face")) { |
| if (effectParameter.equals("squeeze")) { |
| return EffectsRecorder.EFFECT_GF_SQUEEZE; |
| } else if (effectParameter.equals("big_eyes")) { |
| return EffectsRecorder.EFFECT_GF_BIG_EYES; |
| } else if (effectParameter.equals("big_mouth")) { |
| return EffectsRecorder.EFFECT_GF_BIG_MOUTH; |
| } else if (effectParameter.equals("small_mouth")) { |
| return EffectsRecorder.EFFECT_GF_SMALL_MOUTH; |
| } else if (effectParameter.equals("big_nose")) { |
| return EffectsRecorder.EFFECT_GF_BIG_NOSE; |
| } else if (effectParameter.equals("small_eyes")) { |
| return EffectsRecorder.EFFECT_GF_SMALL_EYES; |
| } |
| } else if (effectSelection.startsWith("backdropper")) { |
| // Parameter is a string that either encodes the URI to use, |
| // or specifies 'gallery'. |
| return effectParameter; |
| } |
| |
| Log.e(TAG, "Invalid effect selection: " + effectSelection); |
| return null; |
| } |
| |
| public static void restorePreferences(Context context, |
| ComboPreferences preferences, Parameters parameters) { |
| int currentCameraId = readPreferredCameraId(preferences); |
| |
| // Clear the preferences of both cameras. |
| int backCameraId = CameraHolder.instance().getBackCameraId(); |
| if (backCameraId != -1) { |
| preferences.setLocalId(context, backCameraId); |
| Editor editor = preferences.edit(); |
| editor.clear(); |
| editor.apply(); |
| } |
| int frontCameraId = CameraHolder.instance().getFrontCameraId(); |
| if (frontCameraId != -1) { |
| preferences.setLocalId(context, frontCameraId); |
| Editor editor = preferences.edit(); |
| editor.clear(); |
| editor.apply(); |
| } |
| |
| // Switch back to the preferences of the current camera. Otherwise, |
| // we may write the preference to wrong camera later. |
| preferences.setLocalId(context, currentCameraId); |
| |
| upgradeGlobalPreferences(preferences.getGlobal()); |
| upgradeLocalPreferences(preferences.getLocal()); |
| |
| // Write back the current camera id because parameters are related to |
| // the camera. Otherwise, we may switch to the front camera but the |
| // initial picture size is that of the back camera. |
| initialCameraPictureSize(context, parameters); |
| writePreferredCameraId(preferences, currentCameraId); |
| } |
| |
| private ArrayList<String> getSupportedVideoQuality() { |
| ArrayList<String> supported = new ArrayList<String>(); |
| // Check for supported quality |
| if (ApiHelper.HAS_FINE_RESOLUTION_QUALITY_LEVELS) { |
| getFineResolutionQuality(supported); |
| } else { |
| supported.add(Integer.toString(CamcorderProfile.QUALITY_HIGH)); |
| CamcorderProfile high = CamcorderProfile.get( |
| mCameraId, CamcorderProfile.QUALITY_HIGH); |
| CamcorderProfile low = CamcorderProfile.get( |
| mCameraId, CamcorderProfile.QUALITY_LOW); |
| if (high.videoFrameHeight * high.videoFrameWidth > |
| low.videoFrameHeight * low.videoFrameWidth) { |
| supported.add(Integer.toString(CamcorderProfile.QUALITY_LOW)); |
| } |
| } |
| |
| return supported; |
| } |
| |
| @TargetApi(ApiHelper.VERSION_CODES.HONEYCOMB) |
| private void getFineResolutionQuality(ArrayList<String> supported) { |
| if (CamcorderProfile.hasProfile(mCameraId, CamcorderProfile.QUALITY_1080P)) { |
| supported.add(Integer.toString(CamcorderProfile.QUALITY_1080P)); |
| } |
| if (CamcorderProfile.hasProfile(mCameraId, CamcorderProfile.QUALITY_720P)) { |
| supported.add(Integer.toString(CamcorderProfile.QUALITY_720P)); |
| } |
| if (CamcorderProfile.hasProfile(mCameraId, CamcorderProfile.QUALITY_480P)) { |
| supported.add(Integer.toString(CamcorderProfile.QUALITY_480P)); |
| } |
| } |
| |
| private void initVideoEffect(PreferenceGroup group, ListPreference videoEffect) { |
| CharSequence[] values = videoEffect.getEntryValues(); |
| |
| boolean goofyFaceSupported = |
| EffectsRecorder.isEffectSupported(EffectsRecorder.EFFECT_GOOFY_FACE); |
| boolean backdropperSupported = |
| EffectsRecorder.isEffectSupported(EffectsRecorder.EFFECT_BACKDROPPER) && |
| Util.isAutoExposureLockSupported(mParameters) && |
| Util.isAutoWhiteBalanceLockSupported(mParameters); |
| |
| ArrayList<String> supported = new ArrayList<String>(); |
| for (CharSequence value : values) { |
| String effectSelection = value.toString(); |
| if (!goofyFaceSupported && effectSelection.startsWith("goofy_face")) continue; |
| if (!backdropperSupported && effectSelection.startsWith("backdropper")) continue; |
| supported.add(effectSelection); |
| } |
| |
| filterUnsupportedOptions(group, videoEffect, supported); |
| } |
| } |