/* * 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; /** * 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 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 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 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 sizeListToStringList(List sizes) { ArrayList list = new ArrayList(); for (Size size : sizes) { list.add(String.format("%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 getSupportedVideoQuality() { ArrayList supported = new ArrayList(); // 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 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 supported = new ArrayList(); 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); } }