/* * Copyright (C) 2014 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.settings.fuelgauge; import android.content.BroadcastReceiver; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.res.Resources; import android.database.ContentObserver; import android.net.Uri; import android.os.AsyncTask; import android.os.BatteryManager; import android.os.Bundle; import android.os.Handler; import android.os.PowerManager; import android.provider.Settings.Global; import android.util.Log; import android.widget.Switch; import com.android.internal.logging.MetricsProto.MetricsEvent; import com.android.settings.R; import com.android.settings.SettingsActivity; import com.android.settings.SettingsPreferenceFragment; import com.android.settings.Utils; import com.android.settings.dashboard.conditional.BatterySaverCondition; import com.android.settings.dashboard.conditional.ConditionManager; import com.android.settings.notification.SettingPref; import com.android.settings.widget.SwitchBar; import static android.os.PowerManager.ACTION_POWER_SAVE_MODE_CHANGING; public class BatterySaverSettings extends SettingsPreferenceFragment implements SwitchBar.OnSwitchChangeListener { private static final String TAG = "BatterySaverSettings"; private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG); private static final String KEY_TURN_ON_AUTOMATICALLY = "turn_on_automatically"; private static final long WAIT_FOR_SWITCH_ANIM = 500; private final Handler mHandler = new Handler(); private final SettingsObserver mSettingsObserver = new SettingsObserver(mHandler); private final Receiver mReceiver = new Receiver(); private Context mContext; private boolean mCreated; private SettingPref mTriggerPref; private SwitchBar mSwitchBar; private Switch mSwitch; private boolean mValidListener; private PowerManager mPowerManager; @Override protected int getMetricsCategory() { return MetricsEvent.FUELGAUGE_BATTERY_SAVER; } @Override public void onActivityCreated(Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); if (mCreated) { mSwitchBar.show(); return; } mCreated = true; addPreferencesFromResource(R.xml.battery_saver_settings); mContext = getActivity(); mSwitchBar = ((SettingsActivity) mContext).getSwitchBar(); mSwitch = mSwitchBar.getSwitch(); mSwitchBar.show(); mTriggerPref = new SettingPref(SettingPref.TYPE_GLOBAL, KEY_TURN_ON_AUTOMATICALLY, Global.LOW_POWER_MODE_TRIGGER_LEVEL, 0, /*default*/ getResources().getIntArray(R.array.battery_saver_trigger_values)) { @Override protected String getCaption(Resources res, int value) { if (value > 0 && value < 100) { return res.getString(R.string.battery_saver_turn_on_automatically_pct, Utils.formatPercentage(value)); } return res.getString(R.string.battery_saver_turn_on_automatically_never); } }; mTriggerPref.init(this); mPowerManager = (PowerManager) mContext.getSystemService(Context.POWER_SERVICE); } @Override public void onDestroyView() { super.onDestroyView(); mSwitchBar.hide(); } @Override public void onResume() { super.onResume(); mSettingsObserver.setListening(true); mReceiver.setListening(true); if (!mValidListener) { mSwitchBar.addOnSwitchChangeListener(this); mValidListener = true; } updateSwitch(); } @Override public void onPause() { super.onPause(); mSettingsObserver.setListening(false); mReceiver.setListening(false); if (mValidListener) { mSwitchBar.removeOnSwitchChangeListener(this); mValidListener = false; } } @Override public void onSwitchChanged(Switch switchView, boolean isChecked) { mHandler.removeCallbacks(mStartMode); if (isChecked) { mHandler.postDelayed(mStartMode, WAIT_FOR_SWITCH_ANIM); } else { if (DEBUG) Log.d(TAG, "Stopping low power mode from settings"); trySetPowerSaveMode(false); } } private void trySetPowerSaveMode(boolean mode) { if (!mPowerManager.setPowerSaveMode(mode)) { if (DEBUG) Log.d(TAG, "Setting mode failed, fallback to current value"); mHandler.post(mUpdateSwitch); } // TODO: Remove once broadcast is in place. ConditionManager.get(getContext()).getCondition(BatterySaverCondition.class).refreshState(); } private void updateSwitch() { final boolean mode = mPowerManager.isPowerSaveMode(); if (DEBUG) Log.d(TAG, "updateSwitch: isChecked=" + mSwitch.isChecked() + " mode=" + mode); if (mode == mSwitch.isChecked()) return; // set listener to null so that that code below doesn't trigger onCheckedChanged() if (mValidListener) { mSwitchBar.removeOnSwitchChangeListener(this); } mSwitch.setChecked(mode); if (mValidListener) { mSwitchBar.addOnSwitchChangeListener(this); } } private final Runnable mUpdateSwitch = new Runnable() { @Override public void run() { updateSwitch(); } }; private final Runnable mStartMode = new Runnable() { @Override public void run() { AsyncTask.execute(new Runnable() { @Override public void run() { if (DEBUG) Log.d(TAG, "Starting low power mode from settings"); trySetPowerSaveMode(true); } }); } }; private final class Receiver extends BroadcastReceiver { private boolean mRegistered; @Override public void onReceive(Context context, Intent intent) { if (DEBUG) Log.d(TAG, "Received " + intent.getAction()); String action = intent.getAction(); if (action.equals(ACTION_POWER_SAVE_MODE_CHANGING)) { mHandler.post(mUpdateSwitch); } else if (action.equals(Intent.ACTION_BATTERY_CHANGED)) { final int status = intent.getIntExtra(BatteryManager.EXTRA_STATUS, -1); // disable BSM switch if phone is plugged in or at 100% while plugged in mSwitchBar.setEnabled( !(status == BatteryManager.BATTERY_STATUS_CHARGING || status == BatteryManager.BATTERY_STATUS_FULL)); } } public void setListening(boolean listening) { if (listening && !mRegistered) { final IntentFilter ifilter = new IntentFilter(); ifilter.addAction(ACTION_POWER_SAVE_MODE_CHANGING); ifilter.addAction(Intent.ACTION_BATTERY_CHANGED); mContext.registerReceiver(this, ifilter); mRegistered = true; } else if (!listening && mRegistered) { mContext.unregisterReceiver(this); mRegistered = false; } } } private final class SettingsObserver extends ContentObserver { private final Uri LOW_POWER_MODE_TRIGGER_LEVEL_URI = Global.getUriFor(Global.LOW_POWER_MODE_TRIGGER_LEVEL); public SettingsObserver(Handler handler) { super(handler); } @Override public void onChange(boolean selfChange, Uri uri) { if (LOW_POWER_MODE_TRIGGER_LEVEL_URI.equals(uri)) { mTriggerPref.update(mContext); } } public void setListening(boolean listening) { final ContentResolver cr = getContentResolver(); if (listening) { cr.registerContentObserver(LOW_POWER_MODE_TRIGGER_LEVEL_URI, false, this); } else { cr.unregisterContentObserver(this); } } } }