| /* |
| * Copyright (C) 2018 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.ComponentName; |
| import android.content.Context; |
| import android.os.BatteryManager; |
| import android.util.Log; |
| |
| import androidx.annotation.VisibleForTesting; |
| import androidx.preference.Preference; |
| import androidx.preference.PreferenceScreen; |
| |
| import com.android.settings.R; |
| import com.android.settings.core.BasePreferenceController; |
| import com.android.settings.overlay.FeatureFactory; |
| import com.android.settingslib.Utils; |
| import com.android.settingslib.core.lifecycle.LifecycleObserver; |
| import com.android.settingslib.core.lifecycle.events.OnStart; |
| import com.android.settingslib.core.lifecycle.events.OnStop; |
| import com.android.settingslib.utils.ThreadUtils; |
| |
| public class TopLevelBatteryPreferenceController extends BasePreferenceController |
| implements LifecycleObserver, OnStart, OnStop, BatteryPreferenceController { |
| |
| private static final String TAG = "TopLvBatteryPrefControl"; |
| |
| @VisibleForTesting Preference mPreference; |
| @VisibleForTesting protected boolean mIsBatteryPresent = true; |
| |
| private final BatteryBroadcastReceiver mBatteryBroadcastReceiver; |
| |
| private BatteryInfo mBatteryInfo; |
| private BatteryStatusFeatureProvider mBatteryStatusFeatureProvider; |
| private String mBatteryStatusLabel; |
| |
| public TopLevelBatteryPreferenceController(Context context, String preferenceKey) { |
| super(context, preferenceKey); |
| mBatteryBroadcastReceiver = new BatteryBroadcastReceiver(mContext); |
| mBatteryBroadcastReceiver.setBatteryChangedListener( |
| type -> { |
| Log.d(TAG, "onBatteryChanged: type=" + type); |
| if (type == BatteryBroadcastReceiver.BatteryUpdateType.BATTERY_NOT_PRESENT) { |
| mIsBatteryPresent = false; |
| } |
| BatteryInfo.getBatteryInfo( |
| mContext, |
| info -> { |
| Log.d(TAG, "getBatteryInfo: " + info); |
| mBatteryInfo = info; |
| updateState(mPreference); |
| // Update the preference summary text to the latest state. |
| setSummaryAsync(info); |
| }, |
| true /* shortString */); |
| }); |
| |
| mBatteryStatusFeatureProvider = |
| FeatureFactory.getFeatureFactory().getBatteryStatusFeatureProvider(); |
| } |
| |
| @Override |
| public int getAvailabilityStatus() { |
| return mContext.getResources().getBoolean(R.bool.config_show_top_level_battery) |
| ? AVAILABLE |
| : UNSUPPORTED_ON_DEVICE; |
| } |
| |
| @Override |
| public void displayPreference(PreferenceScreen screen) { |
| super.displayPreference(screen); |
| mPreference = screen.findPreference(getPreferenceKey()); |
| } |
| |
| @Override |
| public void onStart() { |
| mBatteryBroadcastReceiver.register(); |
| } |
| |
| @Override |
| public void onStop() { |
| mBatteryBroadcastReceiver.unRegister(); |
| } |
| |
| @Override |
| public CharSequence getSummary() { |
| return getSummary(true /* batteryStatusUpdate */); |
| } |
| |
| private CharSequence getSummary(boolean batteryStatusUpdate) { |
| // Display help message if battery is not present. |
| if (!mIsBatteryPresent) { |
| return mContext.getText(R.string.battery_missing_message); |
| } |
| return getDashboardLabel(mContext, mBatteryInfo, batteryStatusUpdate); |
| } |
| |
| protected CharSequence getDashboardLabel( |
| Context context, BatteryInfo info, boolean batteryStatusUpdate) { |
| if (info == null || context == null) { |
| return null; |
| } |
| Log.d( |
| TAG, |
| "getDashboardLabel: " |
| + mBatteryStatusLabel |
| + " batteryStatusUpdate=" |
| + batteryStatusUpdate); |
| |
| if (batteryStatusUpdate) { |
| setSummaryAsync(info); |
| } |
| return mBatteryStatusLabel == null ? generateLabel(info) : mBatteryStatusLabel; |
| } |
| |
| private void setSummaryAsync(BatteryInfo info) { |
| ThreadUtils.postOnBackgroundThread( |
| () -> { |
| // Return false if built-in status should be used, will use |
| // updateBatteryStatus() |
| // method to inject the customized battery status label. |
| final boolean triggerBatteryStatusUpdate = |
| mBatteryStatusFeatureProvider.triggerBatteryStatusUpdate(this, info); |
| ThreadUtils.postOnMainThread( |
| () -> { |
| if (!triggerBatteryStatusUpdate) { |
| mBatteryStatusLabel = null; // will generateLabel() |
| } |
| mPreference.setSummary( |
| mBatteryStatusLabel == null |
| ? generateLabel(info) |
| : mBatteryStatusLabel); |
| }); |
| }); |
| } |
| |
| private CharSequence generateLabel(BatteryInfo info) { |
| if (Utils.containsIncompatibleChargers(mContext, TAG)) { |
| return mContext.getString( |
| com.android.settingslib.R.string.power_incompatible_charging_settings_home_page, |
| info.batteryPercentString); |
| } |
| if (BatteryUtils.isBatteryDefenderOn(info)) { |
| return mContext.getString( |
| com.android.settingslib.R.string.power_charging_on_hold_settings_home_page, |
| info.batteryPercentString); |
| } |
| if (info.batteryStatus == BatteryManager.BATTERY_STATUS_NOT_CHARGING) { |
| // Present status only if no remaining time or status anomalous |
| return info.statusLabel; |
| } else if (!info.discharging && info.chargeLabel != null) { |
| return info.chargeLabel; |
| } else if (info.remainingLabel == null) { |
| return info.batteryPercentString; |
| } else { |
| return mContext.getString( |
| com.android.settingslib.R.string.power_remaining_settings_home_page, |
| info.batteryPercentString, |
| info.remainingLabel); |
| } |
| } |
| |
| /** Callback which receives text for the label. */ |
| @Override |
| public void updateBatteryStatus(String label, BatteryInfo info) { |
| mBatteryStatusLabel = label; // Null if adaptive charging is not active |
| if (mPreference == null) { |
| return; |
| } |
| // Do not triggerBatteryStatusUpdate() here to cause infinite loop |
| final CharSequence summary = getSummary(false /* batteryStatusUpdate */); |
| if (summary != null) { |
| mPreference.setSummary(summary); |
| } |
| Log.d(TAG, "updateBatteryStatus: " + label + " summary: " + summary); |
| } |
| |
| @VisibleForTesting |
| protected static ComponentName convertClassPathToComponentName(String classPath) { |
| if (classPath == null || classPath.isEmpty()) { |
| return null; |
| } |
| String[] split = classPath.split("\\."); |
| int classNameIndex = split.length - 1; |
| if (classNameIndex < 0) { |
| return null; |
| } |
| int lastPkgIndex = classPath.length() - split[classNameIndex].length() - 1; |
| String pkgName = lastPkgIndex > 0 ? classPath.substring(0, lastPkgIndex) : ""; |
| return new ComponentName(pkgName, split[classNameIndex]); |
| } |
| } |