Merge 24Q3 (ab/AP3A.240905.001) to aosp-main-future
Bug: 347831320 Merged-In: I2489a64804f068a957f9be3f87847d6911b24447 Change-Id: I1885ef34f69ac4eb6e76ac903fcacdebe4139ae1
This commit is contained in:
@@ -2832,17 +2832,20 @@
|
|||||||
<!-- Note this must not be exported since it returns the password in the intent -->
|
<!-- Note this must not be exported since it returns the password in the intent -->
|
||||||
<activity android:name=".password.ConfirmLockPattern$InternalActivity"
|
<activity android:name=".password.ConfirmLockPattern$InternalActivity"
|
||||||
android:exported="false"
|
android:exported="false"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:theme="@style/GlifTheme.Light"/>
|
android:theme="@style/GlifTheme.Light"/>
|
||||||
|
|
||||||
<!-- Note this must not be exported since it returns the password in the intent -->
|
<!-- Note this must not be exported since it returns the password in the intent -->
|
||||||
<activity android:name=".password.ConfirmLockPassword$InternalActivity"
|
<activity android:name=".password.ConfirmLockPassword$InternalActivity"
|
||||||
android:exported="false"
|
android:exported="false"
|
||||||
android:windowSoftInputMode="adjustResize"
|
android:windowSoftInputMode="adjustResize"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:theme="@style/GlifTheme.Light"/>
|
android:theme="@style/GlifTheme.Light"/>
|
||||||
|
|
||||||
<activity android:name=".password.SetupChooseLockGeneric"
|
<activity android:name=".password.SetupChooseLockGeneric"
|
||||||
android:theme="@style/GlifTheme.Light"
|
android:theme="@style/GlifTheme.Light"
|
||||||
android:exported="true"
|
android:exported="true"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:label="@string/lock_settings_picker_title">
|
android:label="@string/lock_settings_picker_title">
|
||||||
<intent-filter android:priority="1">
|
<intent-filter android:priority="1">
|
||||||
<action android:name="com.android.settings.SETUP_LOCK_SCREEN" />
|
<action android:name="com.android.settings.SETUP_LOCK_SCREEN" />
|
||||||
@@ -2852,16 +2855,19 @@
|
|||||||
|
|
||||||
<activity android:name=".password.SetupChooseLockGeneric$InternalActivity"
|
<activity android:name=".password.SetupChooseLockGeneric$InternalActivity"
|
||||||
android:exported="false"
|
android:exported="false"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:excludeFromRecents="true" />
|
android:excludeFromRecents="true" />
|
||||||
|
|
||||||
<activity android:name=".password.ChooseLockGeneric"
|
<activity android:name=".password.ChooseLockGeneric"
|
||||||
android:label="@string/lockpassword_choose_lock_generic_header"
|
android:label="@string/lockpassword_choose_lock_generic_header"
|
||||||
android:excludeFromRecents="true"
|
android:excludeFromRecents="true"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:exported="false" />
|
android:exported="false" />
|
||||||
|
|
||||||
<activity android:name=".password.SetNewPasswordActivity"
|
<activity android:name=".password.SetNewPasswordActivity"
|
||||||
android:theme="@android:style/Theme.NoDisplay"
|
android:theme="@android:style/Theme.NoDisplay"
|
||||||
android:exported="true"
|
android:exported="true"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:excludeFromRecents="true" >
|
android:excludeFromRecents="true" >
|
||||||
<intent-filter android:priority="1">
|
<intent-filter android:priority="1">
|
||||||
<action android:name="android.app.action.SET_NEW_PASSWORD" />
|
<action android:name="android.app.action.SET_NEW_PASSWORD" />
|
||||||
@@ -2907,24 +2913,29 @@
|
|||||||
<activity android:name=".password.ChooseLockGeneric$InternalActivity"
|
<activity android:name=".password.ChooseLockGeneric$InternalActivity"
|
||||||
android:exported="false"
|
android:exported="false"
|
||||||
android:label="@string/lockpassword_choose_lock_generic_header"
|
android:label="@string/lockpassword_choose_lock_generic_header"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:excludeFromRecents="true" />
|
android:excludeFromRecents="true" />
|
||||||
|
|
||||||
<activity android:name=".password.SetupChooseLockPattern"
|
<activity android:name=".password.SetupChooseLockPattern"
|
||||||
android:exported="false"
|
android:exported="false"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:theme="@style/GlifTheme.Light" />
|
android:theme="@style/GlifTheme.Light" />
|
||||||
|
|
||||||
<activity android:name=".password.ChooseLockPattern"
|
<activity android:name=".password.ChooseLockPattern"
|
||||||
android:exported="false"
|
android:exported="false"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:theme="@style/GlifTheme.Light" />
|
android:theme="@style/GlifTheme.Light" />
|
||||||
|
|
||||||
<activity android:name=".password.SetupChooseLockPassword"
|
<activity android:name=".password.SetupChooseLockPassword"
|
||||||
android:exported="false"
|
android:exported="false"
|
||||||
android:theme="@style/GlifTheme.Light"
|
android:theme="@style/GlifTheme.Light"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:windowSoftInputMode="stateVisible|adjustResize" />
|
android:windowSoftInputMode="stateVisible|adjustResize" />
|
||||||
|
|
||||||
<activity android:name=".password.ChooseLockPassword"
|
<activity android:name=".password.ChooseLockPassword"
|
||||||
android:exported="false"
|
android:exported="false"
|
||||||
android:theme="@style/GlifTheme.Light"
|
android:theme="@style/GlifTheme.Light"
|
||||||
|
android:enableOnBackInvokedCallback="false"
|
||||||
android:windowSoftInputMode="stateVisible|adjustResize"/>
|
android:windowSoftInputMode="stateVisible|adjustResize"/>
|
||||||
|
|
||||||
<activity
|
<activity
|
||||||
|
@@ -16,6 +16,7 @@
|
|||||||
<LinearLayout
|
<LinearLayout
|
||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
|
android:fitsSystemWindows="true"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:orientation="horizontal"
|
android:orientation="horizontal"
|
||||||
|
@@ -17,6 +17,7 @@
|
|||||||
<LinearLayout
|
<LinearLayout
|
||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
|
android:fitsSystemWindows="true"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:orientation="vertical"
|
android:orientation="vertical"
|
||||||
|
@@ -18,7 +18,8 @@
|
|||||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="match_parent"
|
||||||
|
android:fitsSystemWindows="true"
|
||||||
android:clipChildren="true">
|
android:clipChildren="true">
|
||||||
|
|
||||||
<RelativeLayout
|
<RelativeLayout
|
||||||
|
@@ -16,5 +16,5 @@
|
|||||||
|
|
||||||
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||||||
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Ne možete da upućujete hitne pozive pomoću pozivanja preko Wi-Fi-ja. Ako probate da uputite hitan poziv, uređaj će automatski koristiti mobilnu mrežu. Hitni pozivi mogu da se upućuju samo u oblastima pokrivenim mobilnom mrežom."</string>
|
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Ne možete da upućujete hitne pozive pomoću pozivanja preko WiFi-a. Ako probate da uputite hitan poziv, uređaj će automatski koristiti mobilnu mrežu. Hitni pozivi mogu da se upućuju samo u oblastima pokrivenim mobilnom mrežom."</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -16,5 +16,5 @@
|
|||||||
|
|
||||||
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||||||
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Та Wi-Fi дуудлагаар яаралтай дуудлага хийх боломжгүй. Хэрэв та яаралтай дуудлага хийвэл таны төхөөрөмж мобайл сүлжээг автоматаар ашиглана. Яаралтай дуудлагыг зөвхөн мобайл сүлжээтэй газар хийх боломжтой."</string>
|
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Та Wi-Fi дуудлагаар яаралтай дуудлага хийх боломжгүй. Хэрэв та яаралтай дуудлага хийвэл таны төхөөрөмж хөдөлгөөнт холбооны сүлжээг автоматаар ашиглана. Яаралтай дуудлагыг зөвхөн хөдөлгөөнт холбооны сүлжээтэй газар хийх боломжтой."</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -16,5 +16,5 @@
|
|||||||
|
|
||||||
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||||||
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Du kan ikke ringe nødnumre via Wi-Fi-anrop. Hvis du prøver å ringe et nødnummer, bruker enheten automatisk mobilnettverket. Du kan bare ringe nødnumre i områder med mobildekning."</string>
|
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Du kan ikke ringe nødnumre via wifi-anrop. Hvis du prøver å ringe et nødnummer, bruker enheten automatisk mobilnettverket. Du kan bare ringe nødnumre i områder med mobildekning."</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -16,5 +16,5 @@
|
|||||||
|
|
||||||
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||||||
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"तपाईं Wi-Fi को कल गर्ने सुविधामार्फत आपतकालीन कलहरू गर्न सक्नुहुन्न। तपाईंले आपतकालीन कल गर्न खोज्नुभयो भने तपाईंको यन्त्रले स्वतः मोबाइल नेटवर्कको प्रयोग गर्ने छ। मोबाइल नेटवर्क उपलब्ध भएका क्षेत्रहरूबाट मात्र आपतकालीन कलहरू गर्न सकिन्छ।"</string>
|
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"तपाईं Wi-Fi को कल गर्ने सुविधामार्फत आपत्कालीन कलहरू गर्न सक्नुहुन्न। तपाईंले आपत्कालीन कल गर्न खोज्नुभयो भने तपाईंको यन्त्रले स्वतः मोबाइल नेटवर्कको प्रयोग गर्ने छ। मोबाइल नेटवर्क उपलब्ध भएका क्षेत्रहरूबाट मात्र आपत्कालीन कलहरू गर्न सकिन्छ।"</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -16,5 +16,5 @@
|
|||||||
|
|
||||||
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||||||
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Не можете да упућујете хитне позиве помоћу позивања преко Wi-Fi-ја. Ако пробате да упутите хитан позив, уређај ће аутоматски користити мобилну мрежу. Хитни позиви могу да се упућују само у областима покривеним мобилном мрежом."</string>
|
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Не можете да упућујете хитне позиве помоћу позивања преко WiFi-а. Ако пробате да упутите хитан позив, уређај ће аутоматски користити мобилну мрежу. Хитни позиви могу да се упућују само у областима покривеним мобилном мрежом."</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -16,5 +16,5 @@
|
|||||||
|
|
||||||
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||||||
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Det går inte att ringa nödsamtal via Wi-Fi. Om du ringer ett nödnummer på enheten används mobilnätverket automatiskt. Det går bara att ringa nödsamtal där mobilnätet har täckning."</string>
|
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"Det går inte att ringa nödsamtal via wifi. Om du ringer ett nödnummer på enheten används mobilnätverket automatiskt. Det går bara att ringa nödsamtal där mobilnätet har täckning."</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -16,5 +16,5 @@
|
|||||||
|
|
||||||
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||||||
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"మీరు Wi-Fi కాలింగ్ ద్వారా అత్యవసర కాల్లను చేయలేరు. అత్యవసర కాల్ చేయడానికి ప్రయత్నిస్తే, మీ పరికరం స్వయంచాలకంగా మొబైల్ నెట్వర్క్ని ఉపయోగిస్తుంది. మొబైల్ నెట్వర్క్ కవరేజ్ ఉన్న ప్రాంతాల్లో మాత్రమే అత్యవసర కాల్లు చేయబడతాయి."</string>
|
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"మీరు Wi-Fi కాలింగ్ ద్వారా అత్యవసర కాల్స్ను చేయలేరు. అత్యవసర కాల్ చేయడానికి ప్రయత్నిస్తే, మీ పరికరం ఆటోమేటిక్గా మొబైల్ నెట్వర్క్ని ఉపయోగిస్తుంది. మొబైల్ నెట్వర్క్ కవరేజ్ ఉన్న ప్రాంతాల్లో మాత్రమే అత్యవసర కాల్స్ చేయబడతాయి."</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -16,5 +16,5 @@
|
|||||||
|
|
||||||
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
<resources xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||||||
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"您不可透過 Wi-Fi 通話撥打緊急電話。如果您嘗試撥打緊急電話,裝置會自動使用流動網絡。您只可在有流動網絡覆蓋的地方撥打緊急電話。"</string>
|
<string name="wifi_calling_off_explanation_2" msgid="5645212744935818562">\n\n"你不可透過 Wi-Fi 通話撥打緊急電話。如果你嘗試撥打緊急電話,裝置會自動使用流動網絡。你只可在有流動網絡覆蓋的地方撥打緊急電話。"</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -1552,4 +1552,8 @@
|
|||||||
<item>Theater</item>
|
<item>Theater</item>
|
||||||
<item>Flower</item>
|
<item>Flower</item>
|
||||||
</string-array>
|
</string-array>
|
||||||
|
|
||||||
|
<!-- Packages that will not show Display over other apps permission -->
|
||||||
|
<string-array name="display_over_apps_permission_change_exempt">
|
||||||
|
</string-array>
|
||||||
</resources>
|
</resources>
|
||||||
|
@@ -356,6 +356,11 @@ public class FingerprintSettings extends SubSettings {
|
|||||||
*/
|
*/
|
||||||
protected void handleError(int errMsgId, CharSequence msg) {
|
protected void handleError(int errMsgId, CharSequence msg) {
|
||||||
switch (errMsgId) {
|
switch (errMsgId) {
|
||||||
|
case FingerprintManager.FINGERPRINT_ERROR_CANCELED:
|
||||||
|
case FingerprintManager.FINGERPRINT_ERROR_USER_CANCELED:
|
||||||
|
// Only happens if we get preempted by another activity, or canceled by the
|
||||||
|
// user (e.g. swipe up to home). Ignored.
|
||||||
|
return;
|
||||||
case FingerprintManager.FINGERPRINT_ERROR_LOCKOUT:
|
case FingerprintManager.FINGERPRINT_ERROR_LOCKOUT:
|
||||||
mInFingerprintLockout = true;
|
mInFingerprintLockout = true;
|
||||||
// We've been locked out. Reset after 30s.
|
// We've been locked out. Reset after 30s.
|
||||||
|
@@ -19,13 +19,16 @@ package com.android.settings.bluetooth;
|
|||||||
import static android.media.Spatializer.SPATIALIZER_IMMERSIVE_LEVEL_NONE;
|
import static android.media.Spatializer.SPATIALIZER_IMMERSIVE_LEVEL_NONE;
|
||||||
|
|
||||||
import android.app.settings.SettingsEnums;
|
import android.app.settings.SettingsEnums;
|
||||||
|
import android.bluetooth.BluetoothProfile;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.media.AudioDeviceAttributes;
|
import android.media.AudioDeviceAttributes;
|
||||||
import android.media.AudioDeviceInfo;
|
import android.media.AudioDeviceInfo;
|
||||||
|
import android.media.AudioManager;
|
||||||
import android.media.Spatializer;
|
import android.media.Spatializer;
|
||||||
import android.text.TextUtils;
|
import android.text.TextUtils;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
import androidx.annotation.VisibleForTesting;
|
import androidx.annotation.VisibleForTesting;
|
||||||
import androidx.preference.Preference;
|
import androidx.preference.Preference;
|
||||||
import androidx.preference.PreferenceCategory;
|
import androidx.preference.PreferenceCategory;
|
||||||
@@ -37,9 +40,14 @@ import androidx.preference.TwoStatePreference;
|
|||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
import com.android.settings.overlay.FeatureFactory;
|
import com.android.settings.overlay.FeatureFactory;
|
||||||
import com.android.settingslib.bluetooth.CachedBluetoothDevice;
|
import com.android.settingslib.bluetooth.CachedBluetoothDevice;
|
||||||
|
import com.android.settingslib.bluetooth.LocalBluetoothProfile;
|
||||||
import com.android.settingslib.core.lifecycle.Lifecycle;
|
import com.android.settingslib.core.lifecycle.Lifecycle;
|
||||||
|
import com.android.settingslib.flags.Flags;
|
||||||
import com.android.settingslib.utils.ThreadUtils;
|
import com.android.settingslib.utils.ThreadUtils;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -53,22 +61,27 @@ public class BluetoothDetailsSpatialAudioController extends BluetoothDetailsCont
|
|||||||
private static final String KEY_SPATIAL_AUDIO = "spatial_audio";
|
private static final String KEY_SPATIAL_AUDIO = "spatial_audio";
|
||||||
private static final String KEY_HEAD_TRACKING = "head_tracking";
|
private static final String KEY_HEAD_TRACKING = "head_tracking";
|
||||||
|
|
||||||
|
private final AudioManager mAudioManager;
|
||||||
private final Spatializer mSpatializer;
|
private final Spatializer mSpatializer;
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
PreferenceCategory mProfilesContainer;
|
PreferenceCategory mProfilesContainer;
|
||||||
@VisibleForTesting
|
@VisibleForTesting @Nullable AudioDeviceAttributes mAudioDevice = null;
|
||||||
AudioDeviceAttributes mAudioDevice = null;
|
|
||||||
|
|
||||||
AtomicBoolean mHasHeadTracker = new AtomicBoolean(false);
|
AtomicBoolean mHasHeadTracker = new AtomicBoolean(false);
|
||||||
AtomicBoolean mInitialRefresh = new AtomicBoolean(true);
|
AtomicBoolean mInitialRefresh = new AtomicBoolean(true);
|
||||||
|
|
||||||
|
public static final Set<Integer> SA_PROFILES =
|
||||||
|
ImmutableSet.of(
|
||||||
|
BluetoothProfile.A2DP, BluetoothProfile.LE_AUDIO, BluetoothProfile.HEARING_AID);
|
||||||
|
|
||||||
public BluetoothDetailsSpatialAudioController(
|
public BluetoothDetailsSpatialAudioController(
|
||||||
Context context,
|
Context context,
|
||||||
PreferenceFragmentCompat fragment,
|
PreferenceFragmentCompat fragment,
|
||||||
CachedBluetoothDevice device,
|
CachedBluetoothDevice device,
|
||||||
Lifecycle lifecycle) {
|
Lifecycle lifecycle) {
|
||||||
super(context, fragment, device, lifecycle);
|
super(context, fragment, device, lifecycle);
|
||||||
|
mAudioManager = context.getSystemService(AudioManager.class);
|
||||||
mSpatializer = FeatureFactory.getFeatureFactory().getBluetoothFeatureProvider()
|
mSpatializer = FeatureFactory.getFeatureFactory().getBluetoothFeatureProvider()
|
||||||
.getSpatializer(context);
|
.getSpatializer(context);
|
||||||
}
|
}
|
||||||
@@ -142,9 +155,13 @@ public class BluetoothDetailsSpatialAudioController extends BluetoothDetailsCont
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void refresh() {
|
protected void refresh() {
|
||||||
|
if (Flags.enableDeterminingSpatialAudioAttributesByProfile()) {
|
||||||
|
getAvailableDeviceByProfileState();
|
||||||
|
} else {
|
||||||
if (mAudioDevice == null) {
|
if (mAudioDevice == null) {
|
||||||
getAvailableDevice();
|
getAvailableDevice();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
ThreadUtils.postOnBackgroundThread(
|
ThreadUtils.postOnBackgroundThread(
|
||||||
() -> {
|
() -> {
|
||||||
mHasHeadTracker.set(
|
mHasHeadTracker.set(
|
||||||
@@ -274,6 +291,77 @@ public class BluetoothDetailsSpatialAudioController extends BluetoothDetailsCont
|
|||||||
+ ", type : " + (mAudioDevice == null ? "no type" : mAudioDevice.getType()));
|
+ ", type : " + (mAudioDevice == null ? "no type" : mAudioDevice.getType()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void getAvailableDeviceByProfileState() {
|
||||||
|
Log.i(
|
||||||
|
TAG,
|
||||||
|
"getAvailableDevice() mCachedDevice: "
|
||||||
|
+ mCachedDevice
|
||||||
|
+ " profiles: "
|
||||||
|
+ mCachedDevice.getProfiles());
|
||||||
|
|
||||||
|
AudioDeviceAttributes saDevice = null;
|
||||||
|
for (LocalBluetoothProfile profile : mCachedDevice.getProfiles()) {
|
||||||
|
// pick first enabled profile that is compatible with spatial audio
|
||||||
|
if (SA_PROFILES.contains(profile.getProfileId())
|
||||||
|
&& profile.isEnabled(mCachedDevice.getDevice())) {
|
||||||
|
switch (profile.getProfileId()) {
|
||||||
|
case BluetoothProfile.A2DP:
|
||||||
|
saDevice =
|
||||||
|
new AudioDeviceAttributes(
|
||||||
|
AudioDeviceAttributes.ROLE_OUTPUT,
|
||||||
|
AudioDeviceInfo.TYPE_BLUETOOTH_A2DP,
|
||||||
|
mCachedDevice.getAddress());
|
||||||
|
break;
|
||||||
|
case BluetoothProfile.LE_AUDIO:
|
||||||
|
if (mAudioManager.getBluetoothAudioDeviceCategory(
|
||||||
|
mCachedDevice.getAddress())
|
||||||
|
== AudioManager.AUDIO_DEVICE_CATEGORY_SPEAKER) {
|
||||||
|
saDevice =
|
||||||
|
new AudioDeviceAttributes(
|
||||||
|
AudioDeviceAttributes.ROLE_OUTPUT,
|
||||||
|
AudioDeviceInfo.TYPE_BLE_SPEAKER,
|
||||||
|
mCachedDevice.getAddress());
|
||||||
|
} else {
|
||||||
|
saDevice =
|
||||||
|
new AudioDeviceAttributes(
|
||||||
|
AudioDeviceAttributes.ROLE_OUTPUT,
|
||||||
|
AudioDeviceInfo.TYPE_BLE_HEADSET,
|
||||||
|
mCachedDevice.getAddress());
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
case BluetoothProfile.HEARING_AID:
|
||||||
|
saDevice =
|
||||||
|
new AudioDeviceAttributes(
|
||||||
|
AudioDeviceAttributes.ROLE_OUTPUT,
|
||||||
|
AudioDeviceInfo.TYPE_HEARING_AID,
|
||||||
|
mCachedDevice.getAddress());
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
Log.i(
|
||||||
|
TAG,
|
||||||
|
"unrecognized profile for spatial audio: "
|
||||||
|
+ profile.getProfileId());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mAudioDevice = null;
|
||||||
|
if (saDevice != null && mSpatializer.isAvailableForDevice(saDevice)) {
|
||||||
|
mAudioDevice = saDevice;
|
||||||
|
}
|
||||||
|
|
||||||
|
Log.d(
|
||||||
|
TAG,
|
||||||
|
"getAvailableDevice() device : "
|
||||||
|
+ mCachedDevice.getDevice().getAnonymizedAddress()
|
||||||
|
+ ", is available : "
|
||||||
|
+ (mAudioDevice != null)
|
||||||
|
+ ", type : "
|
||||||
|
+ (mAudioDevice == null ? "no type" : mAudioDevice.getType()));
|
||||||
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
void setAvailableDevice(AudioDeviceAttributes audioDevice) {
|
void setAvailableDevice(AudioDeviceAttributes audioDevice) {
|
||||||
mAudioDevice = audioDevice;
|
mAudioDevice = audioDevice;
|
||||||
|
@@ -23,11 +23,16 @@ import android.os.Bundle;
|
|||||||
import android.util.ArrayMap;
|
import android.util.ArrayMap;
|
||||||
import android.util.SparseIntArray;
|
import android.util.SparseIntArray;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.BatteryDiffData;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.BatteryEvent;
|
||||||
import com.android.settings.fuelgauge.batteryusage.DetectRequestSourceType;
|
import com.android.settings.fuelgauge.batteryusage.DetectRequestSourceType;
|
||||||
import com.android.settings.fuelgauge.batteryusage.PowerAnomalyEventList;
|
import com.android.settings.fuelgauge.batteryusage.PowerAnomalyEventList;
|
||||||
import com.android.settingslib.fuelgauge.Estimate;
|
import com.android.settingslib.fuelgauge.Estimate;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
/** Feature Provider used in power usage */
|
/** Feature Provider used in power usage */
|
||||||
@@ -154,4 +159,14 @@ public interface PowerUsageFeatureProvider {
|
|||||||
|
|
||||||
/** Whether the device is under the battery defender mode */
|
/** Whether the device is under the battery defender mode */
|
||||||
boolean isBatteryDefend(BatteryInfo info);
|
boolean isBatteryDefend(BatteryInfo info);
|
||||||
|
|
||||||
|
/** Whether the battery usage reattribute is eabled or not. */
|
||||||
|
boolean isBatteryUsageReattributeEnabled();
|
||||||
|
|
||||||
|
/** Collect and process battery reattribute data if needed. */
|
||||||
|
boolean processBatteryReattributeData(
|
||||||
|
@NonNull Context context,
|
||||||
|
@NonNull Map<Long, BatteryDiffData> batteryDiffDataMap,
|
||||||
|
@NonNull List<BatteryEvent> batteryEventList,
|
||||||
|
final boolean isFromPeriodJob);
|
||||||
}
|
}
|
||||||
|
@@ -18,7 +18,6 @@ package com.android.settings.fuelgauge;
|
|||||||
|
|
||||||
import static com.android.settings.Utils.SYSTEMUI_PACKAGE_NAME;
|
import static com.android.settings.Utils.SYSTEMUI_PACKAGE_NAME;
|
||||||
|
|
||||||
import android.annotation.Nullable;
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.pm.PackageManager;
|
import android.content.pm.PackageManager;
|
||||||
@@ -27,13 +26,19 @@ import android.util.ArrayMap;
|
|||||||
import android.util.ArraySet;
|
import android.util.ArraySet;
|
||||||
import android.util.SparseIntArray;
|
import android.util.SparseIntArray;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import com.android.internal.util.ArrayUtils;
|
import com.android.internal.util.ArrayUtils;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.BatteryDiffData;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.BatteryEvent;
|
||||||
import com.android.settings.fuelgauge.batteryusage.DetectRequestSourceType;
|
import com.android.settings.fuelgauge.batteryusage.DetectRequestSourceType;
|
||||||
import com.android.settings.fuelgauge.batteryusage.PowerAnomalyEventList;
|
import com.android.settings.fuelgauge.batteryusage.PowerAnomalyEventList;
|
||||||
import com.android.settingslib.fuelgauge.Estimate;
|
import com.android.settingslib.fuelgauge.Estimate;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
/** Implementation of {@code PowerUsageFeatureProvider} */
|
/** Implementation of {@code PowerUsageFeatureProvider} */
|
||||||
@@ -240,4 +245,18 @@ public class PowerUsageFeatureProviderImpl implements PowerUsageFeatureProvider
|
|||||||
public boolean isBatteryDefend(BatteryInfo info) {
|
public boolean isBatteryDefend(BatteryInfo info) {
|
||||||
return info.isBatteryDefender && !isExtraDefend();
|
return info.isBatteryDefender && !isExtraDefend();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isBatteryUsageReattributeEnabled() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean processBatteryReattributeData(
|
||||||
|
@NonNull Context context,
|
||||||
|
@NonNull Map<Long, BatteryDiffData> batteryDiffDataMap,
|
||||||
|
@NonNull List<BatteryEvent> batteryEventList,
|
||||||
|
final boolean isFromPeriodJob) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -77,11 +77,13 @@ public class BatteryDiffData {
|
|||||||
processAndSortEntries(mSystemEntries);
|
processAndSortEntries(mSystemEntries);
|
||||||
}
|
}
|
||||||
|
|
||||||
long getStartTimestamp() {
|
/** Gets the start timestamp. */
|
||||||
|
public long getStartTimestamp() {
|
||||||
return mStartTimestamp;
|
return mStartTimestamp;
|
||||||
}
|
}
|
||||||
|
|
||||||
long getEndTimestamp() {
|
/** Gets the end timestamp. */
|
||||||
|
public long getEndTimestamp() {
|
||||||
return mEndTimestamp;
|
return mEndTimestamp;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -97,7 +99,8 @@ public class BatteryDiffData {
|
|||||||
return mScreenOnTime;
|
return mScreenOnTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
List<BatteryDiffEntry> getAppDiffEntryList() {
|
/** Gets the {@link BatteryDiffEntry} list for apps. */
|
||||||
|
public List<BatteryDiffEntry> getAppDiffEntryList() {
|
||||||
return mAppEntries;
|
return mAppEntries;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -296,8 +299,7 @@ public class BatteryDiffData {
|
|||||||
* Sets total consume power, and adjusts the percentages to ensure the total round percentage
|
* Sets total consume power, and adjusts the percentages to ensure the total round percentage
|
||||||
* could be 100%, and then sorts entries based on the sorting key.
|
* could be 100%, and then sorts entries based on the sorting key.
|
||||||
*/
|
*/
|
||||||
@VisibleForTesting
|
public static void processAndSortEntries(final List<BatteryDiffEntry> batteryDiffEntries) {
|
||||||
static void processAndSortEntries(final List<BatteryDiffEntry> batteryDiffEntries) {
|
|
||||||
if (batteryDiffEntries.isEmpty()) {
|
if (batteryDiffEntries.isEmpty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@@ -201,6 +201,20 @@ public final class ConvertUtils {
|
|||||||
return defaultInstance;
|
return defaultInstance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Gets the encoded string from {@link BatteryReattribute} instance. */
|
||||||
|
@NonNull
|
||||||
|
public static String encodeBatteryReattribute(
|
||||||
|
@NonNull BatteryReattribute batteryReattribute) {
|
||||||
|
return Base64.encodeToString(batteryReattribute.toByteArray(), Base64.DEFAULT);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Gets the decoded {@link BatteryReattribute} instance from string. */
|
||||||
|
@NonNull
|
||||||
|
public static BatteryReattribute decodeBatteryReattribute(@NonNull String content) {
|
||||||
|
return BatteryUtils.parseProtoFromString(
|
||||||
|
content, BatteryReattribute.getDefaultInstance());
|
||||||
|
}
|
||||||
|
|
||||||
/** Converts to {@link BatteryHistEntry} */
|
/** Converts to {@link BatteryHistEntry} */
|
||||||
public static BatteryHistEntry convertToBatteryHistEntry(
|
public static BatteryHistEntry convertToBatteryHistEntry(
|
||||||
BatteryEntry entry, BatteryUsageStats batteryUsageStats) {
|
BatteryEntry entry, BatteryUsageStats batteryUsageStats) {
|
||||||
|
@@ -28,6 +28,8 @@ import androidx.annotation.NonNull;
|
|||||||
import androidx.annotation.Nullable;
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import com.android.internal.annotations.VisibleForTesting;
|
import com.android.internal.annotations.VisibleForTesting;
|
||||||
|
import com.android.settings.fuelgauge.PowerUsageFeatureProvider;
|
||||||
|
import com.android.settings.overlay.FeatureFactory;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Calendar;
|
import java.util.Calendar;
|
||||||
@@ -78,6 +80,7 @@ public class DataProcessManager {
|
|||||||
// Raw start timestamp with round to the nearest hour.
|
// Raw start timestamp with round to the nearest hour.
|
||||||
private final long mRawStartTimestamp;
|
private final long mRawStartTimestamp;
|
||||||
private final long mLastFullChargeTimestamp;
|
private final long mLastFullChargeTimestamp;
|
||||||
|
private final boolean mIsFromPeriodJob;
|
||||||
private final Context mContext;
|
private final Context mContext;
|
||||||
private final Handler mHandler;
|
private final Handler mHandler;
|
||||||
private final UserIdsSeries mUserIdsSeries;
|
private final UserIdsSeries mUserIdsSeries;
|
||||||
@@ -122,6 +125,7 @@ public class DataProcessManager {
|
|||||||
Context context,
|
Context context,
|
||||||
Handler handler,
|
Handler handler,
|
||||||
final UserIdsSeries userIdsSeries,
|
final UserIdsSeries userIdsSeries,
|
||||||
|
final boolean isFromPeriodJob,
|
||||||
final long rawStartTimestamp,
|
final long rawStartTimestamp,
|
||||||
final long lastFullChargeTimestamp,
|
final long lastFullChargeTimestamp,
|
||||||
@NonNull final OnBatteryDiffDataMapLoadedListener callbackFunction,
|
@NonNull final OnBatteryDiffDataMapLoadedListener callbackFunction,
|
||||||
@@ -130,6 +134,7 @@ public class DataProcessManager {
|
|||||||
mContext = context.getApplicationContext();
|
mContext = context.getApplicationContext();
|
||||||
mHandler = handler;
|
mHandler = handler;
|
||||||
mUserIdsSeries = userIdsSeries;
|
mUserIdsSeries = userIdsSeries;
|
||||||
|
mIsFromPeriodJob = isFromPeriodJob;
|
||||||
mRawStartTimestamp = rawStartTimestamp;
|
mRawStartTimestamp = rawStartTimestamp;
|
||||||
mLastFullChargeTimestamp = lastFullChargeTimestamp;
|
mLastFullChargeTimestamp = lastFullChargeTimestamp;
|
||||||
mCallbackFunction = callbackFunction;
|
mCallbackFunction = callbackFunction;
|
||||||
@@ -147,6 +152,7 @@ public class DataProcessManager {
|
|||||||
mHandler = handler;
|
mHandler = handler;
|
||||||
mUserIdsSeries = userIdsSeries;
|
mUserIdsSeries = userIdsSeries;
|
||||||
mCallbackFunction = callbackFunction;
|
mCallbackFunction = callbackFunction;
|
||||||
|
mIsFromPeriodJob = false;
|
||||||
mRawStartTimestamp = 0L;
|
mRawStartTimestamp = 0L;
|
||||||
mLastFullChargeTimestamp = 0L;
|
mLastFullChargeTimestamp = 0L;
|
||||||
mHourlyBatteryLevelsPerDay = null;
|
mHourlyBatteryLevelsPerDay = null;
|
||||||
@@ -158,14 +164,9 @@ public class DataProcessManager {
|
|||||||
|
|
||||||
/** Starts the async tasks to load battery history data and app usage data. */
|
/** Starts the async tasks to load battery history data and app usage data. */
|
||||||
public void start() {
|
public void start() {
|
||||||
start(/* isFromPeriodJob= */ false);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Starts the async tasks to load battery history data and app usage data. */
|
|
||||||
public void start(boolean isFromPeriodJob) {
|
|
||||||
// If we have battery level data, load the battery history map and app usage simultaneously.
|
// If we have battery level data, load the battery history map and app usage simultaneously.
|
||||||
if (mHourlyBatteryLevelsPerDay != null) {
|
if (mHourlyBatteryLevelsPerDay != null) {
|
||||||
if (isFromPeriodJob) {
|
if (mIsFromPeriodJob) {
|
||||||
mIsCurrentBatteryHistoryLoaded = true;
|
mIsCurrentBatteryHistoryLoaded = true;
|
||||||
mIsCurrentAppUsageLoaded = true;
|
mIsCurrentAppUsageLoaded = true;
|
||||||
mIsBatteryUsageSlotLoaded = true;
|
mIsBatteryUsageSlotLoaded = true;
|
||||||
@@ -514,6 +515,14 @@ public class DataProcessManager {
|
|||||||
mAppUsagePeriodMap,
|
mAppUsagePeriodMap,
|
||||||
getSystemAppsPackageNames(),
|
getSystemAppsPackageNames(),
|
||||||
getSystemAppsUids()));
|
getSystemAppsUids()));
|
||||||
|
// Process the reattributate data for the following two cases:
|
||||||
|
// 1) the latest slot for the timestamp "until now"
|
||||||
|
// 2) walkthrough all BatteryDiffData again to handle "re-compute" case
|
||||||
|
final PowerUsageFeatureProvider featureProvider =
|
||||||
|
FeatureFactory.getFeatureFactory()
|
||||||
|
.getPowerUsageFeatureProvider();
|
||||||
|
featureProvider.processBatteryReattributeData(
|
||||||
|
mContext, batteryDiffDataMap, mBatteryEventList, mIsFromPeriodJob);
|
||||||
|
|
||||||
Log.d(
|
Log.d(
|
||||||
TAG,
|
TAG,
|
||||||
@@ -683,12 +692,13 @@ public class DataProcessManager {
|
|||||||
context,
|
context,
|
||||||
handler,
|
handler,
|
||||||
userIdsSeries,
|
userIdsSeries,
|
||||||
|
isFromPeriodJob,
|
||||||
startTimestamp,
|
startTimestamp,
|
||||||
lastFullChargeTime,
|
lastFullChargeTime,
|
||||||
onBatteryDiffDataMapLoadedListener,
|
onBatteryDiffDataMapLoadedListener,
|
||||||
batteryLevelData.getHourlyBatteryLevelsPerDay(),
|
batteryLevelData.getHourlyBatteryLevelsPerDay(),
|
||||||
processedBatteryHistoryMap)
|
processedBatteryHistoryMap)
|
||||||
.start(isFromPeriodJob);
|
.start();
|
||||||
|
|
||||||
return batteryLevelData;
|
return batteryLevelData;
|
||||||
}
|
}
|
||||||
|
@@ -429,6 +429,7 @@ public final class DatabaseUtils {
|
|||||||
database.batteryEventDao().clearAll();
|
database.batteryEventDao().clearAll();
|
||||||
database.batteryStateDao().clearAll();
|
database.batteryStateDao().clearAll();
|
||||||
database.batteryUsageSlotDao().clearAll();
|
database.batteryUsageSlotDao().clearAll();
|
||||||
|
database.batteryReattributeDao().clearAll();
|
||||||
} catch (RuntimeException e) {
|
} catch (RuntimeException e) {
|
||||||
Log.e(TAG, "clearAll() failed", e);
|
Log.e(TAG, "clearAll() failed", e);
|
||||||
}
|
}
|
||||||
@@ -466,6 +467,7 @@ public final class DatabaseUtils {
|
|||||||
database.batteryEventDao().clearAllBefore(earliestTimestamp);
|
database.batteryEventDao().clearAllBefore(earliestTimestamp);
|
||||||
database.batteryStateDao().clearAllBefore(earliestTimestamp);
|
database.batteryStateDao().clearAllBefore(earliestTimestamp);
|
||||||
database.batteryUsageSlotDao().clearAllBefore(earliestTimestamp);
|
database.batteryUsageSlotDao().clearAllBefore(earliestTimestamp);
|
||||||
|
database.batteryReattributeDao().clearAllBefore(earliestTimestamp);
|
||||||
} catch (RuntimeException e) {
|
} catch (RuntimeException e) {
|
||||||
Log.e(TAG, "clearAllBefore() failed", e);
|
Log.e(TAG, "clearAllBefore() failed", e);
|
||||||
}
|
}
|
||||||
|
@@ -55,6 +55,7 @@ public final class BugReportContentProvider extends ContentProvider {
|
|||||||
}
|
}
|
||||||
writer.println("dump BatteryUsage and AppUsage states:");
|
writer.println("dump BatteryUsage and AppUsage states:");
|
||||||
LogUtils.dumpAppOptimizationModeEventHist(context, writer);
|
LogUtils.dumpAppOptimizationModeEventHist(context, writer);
|
||||||
|
LogUtils.dumpBatteryReattributeDatabaseHist(context, writer);
|
||||||
LogUtils.dumpBatteryUsageDatabaseHist(context, writer);
|
LogUtils.dumpBatteryUsageDatabaseHist(context, writer);
|
||||||
LogUtils.dumpAppUsageDatabaseHist(context, writer);
|
LogUtils.dumpAppUsageDatabaseHist(context, writer);
|
||||||
LogUtils.dumpBatteryUsageSlotDatabaseHist(context, writer);
|
LogUtils.dumpBatteryUsageSlotDatabaseHist(context, writer);
|
||||||
|
@@ -19,6 +19,8 @@ package com.android.settings.fuelgauge.batteryusage.bugreport;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
|
import androidx.annotation.VisibleForTesting;
|
||||||
|
|
||||||
import com.android.settings.fuelgauge.BatteryUtils;
|
import com.android.settings.fuelgauge.BatteryUtils;
|
||||||
import com.android.settings.fuelgauge.batteryusage.AppOptModeSharedPreferencesUtils;
|
import com.android.settings.fuelgauge.batteryusage.AppOptModeSharedPreferencesUtils;
|
||||||
import com.android.settings.fuelgauge.batteryusage.AppOptimizationModeEvent;
|
import com.android.settings.fuelgauge.batteryusage.AppOptimizationModeEvent;
|
||||||
@@ -29,11 +31,14 @@ import com.android.settings.fuelgauge.batteryusage.db.AppUsageEventDao;
|
|||||||
import com.android.settings.fuelgauge.batteryusage.db.AppUsageEventEntity;
|
import com.android.settings.fuelgauge.batteryusage.db.AppUsageEventEntity;
|
||||||
import com.android.settings.fuelgauge.batteryusage.db.BatteryEventDao;
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryEventDao;
|
||||||
import com.android.settings.fuelgauge.batteryusage.db.BatteryEventEntity;
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryEventEntity;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryReattributeDao;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryReattributeEntity;
|
||||||
import com.android.settings.fuelgauge.batteryusage.db.BatteryState;
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryState;
|
||||||
import com.android.settings.fuelgauge.batteryusage.db.BatteryStateDao;
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryStateDao;
|
||||||
import com.android.settings.fuelgauge.batteryusage.db.BatteryStateDatabase;
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryStateDatabase;
|
||||||
import com.android.settings.fuelgauge.batteryusage.db.BatteryUsageSlotDao;
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryUsageSlotDao;
|
||||||
import com.android.settings.fuelgauge.batteryusage.db.BatteryUsageSlotEntity;
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryUsageSlotEntity;
|
||||||
|
import com.android.settings.overlay.FeatureFactory;
|
||||||
|
|
||||||
import java.io.PrintWriter;
|
import java.io.PrintWriter;
|
||||||
import java.time.Clock;
|
import java.time.Clock;
|
||||||
@@ -127,6 +132,33 @@ public final class LogUtils {
|
|||||||
dumpListItems(writer, entities, entity -> entity);
|
dumpListItems(writer, entities, entity -> entity);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void dumpBatteryReattributeDatabaseHist(Context context, PrintWriter writer) {
|
||||||
|
try {
|
||||||
|
dumpBatteryReattributeDatabaseHist(
|
||||||
|
BatteryStateDatabase.getInstance(context).batteryReattributeDao(),
|
||||||
|
writer);
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e(TAG, "failed to run dumpBatteryReattributeDatabaseHist()", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
static void dumpBatteryReattributeDatabaseHist(
|
||||||
|
BatteryReattributeDao batteryReattributeDao, PrintWriter writer) {
|
||||||
|
if (!FeatureFactory.getFeatureFactory().getPowerUsageFeatureProvider()
|
||||||
|
.isBatteryUsageReattributeEnabled()) {
|
||||||
|
writer.println("\n\tBatteryReattribute is disabled!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
writer.println("\n\tBatteryReattribute DatabaseHistory:");
|
||||||
|
final List<BatteryReattributeEntity> entities =
|
||||||
|
batteryReattributeDao.getAllAfter(
|
||||||
|
Clock.systemUTC().millis() - DUMP_TIME_OFFSET.toMillis());
|
||||||
|
if (entities != null && !entities.isEmpty()) {
|
||||||
|
dumpListItems(writer, entities, entity -> entity);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private static <T, S> void dumpListItems(
|
private static <T, S> void dumpListItems(
|
||||||
PrintWriter writer, List<T> itemList, Function<T, S> itemConverter) {
|
PrintWriter writer, List<T> itemList, Function<T, S> itemConverter) {
|
||||||
final AtomicInteger counter = new AtomicInteger(0);
|
final AtomicInteger counter = new AtomicInteger(0);
|
||||||
|
@@ -0,0 +1,50 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2024 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.batteryusage.db;
|
||||||
|
|
||||||
|
import androidx.room.Dao;
|
||||||
|
import androidx.room.Insert;
|
||||||
|
import androidx.room.OnConflictStrategy;
|
||||||
|
import androidx.room.Query;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/** DAO for accessing {@link BatteryReattributeEntity} in the database. */
|
||||||
|
@Dao
|
||||||
|
public interface BatteryReattributeDao {
|
||||||
|
|
||||||
|
/** Inserts a {@link BatteryReattributeEntity} data into the database. */
|
||||||
|
@Insert(onConflict = OnConflictStrategy.REPLACE)
|
||||||
|
fun insert(event: BatteryReattributeEntity)
|
||||||
|
|
||||||
|
/** Gets all recorded data after a specific timestamp. */
|
||||||
|
@Query(
|
||||||
|
"SELECT * FROM BatteryReattributeEntity WHERE "
|
||||||
|
+ "timestampStart >= :timestampStart ORDER BY timestampStart DESC")
|
||||||
|
fun getAllAfter(timestampStart: Long): List<BatteryReattributeEntity>
|
||||||
|
|
||||||
|
/** Deletes all recorded data before a specific timestamp. */
|
||||||
|
@Query("DELETE FROM BatteryReattributeEntity WHERE timestampStart <= :timestampStart")
|
||||||
|
fun clearAllBefore(timestampStart: Long)
|
||||||
|
|
||||||
|
/** Deletes all recorded data after a specific timestamp. */
|
||||||
|
@Query("DELETE FROM BatteryReattributeEntity WHERE timestampStart >= :timestampStart")
|
||||||
|
fun clearAllAfter(timestampStart: Long)
|
||||||
|
|
||||||
|
/** Clears all recorded data in the database. */
|
||||||
|
@Query("DELETE FROM BatteryReattributeEntity") fun clearAll()
|
||||||
|
}
|
@@ -0,0 +1,73 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2024 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.batteryusage.db;
|
||||||
|
|
||||||
|
import static com.android.settings.fuelgauge.batteryusage.ConvertUtils.utcToLocalTimeForLogging;
|
||||||
|
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.BatteryReattribute;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.ConvertUtils;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.VisibleForTesting;
|
||||||
|
import androidx.room.Entity;
|
||||||
|
import androidx.room.PrimaryKey;
|
||||||
|
|
||||||
|
/** A {@link Entity} for battery usage reattribution data in the database. */
|
||||||
|
@Entity
|
||||||
|
public class BatteryReattributeEntity {
|
||||||
|
|
||||||
|
/** The start timestamp of this record data. */
|
||||||
|
@PrimaryKey
|
||||||
|
public final long timestampStart;
|
||||||
|
|
||||||
|
/** The end timestamp of this record data. */
|
||||||
|
public final long timestampEnd;
|
||||||
|
|
||||||
|
/** The battery usage reattribution data for corresponding uids. */
|
||||||
|
public final String reattributeData;
|
||||||
|
|
||||||
|
public BatteryReattributeEntity(@NonNull BatteryReattribute batteryReattribute) {
|
||||||
|
this(
|
||||||
|
batteryReattribute.getTimestampStart(),
|
||||||
|
batteryReattribute.getTimestampEnd(),
|
||||||
|
ConvertUtils.encodeBatteryReattribute(batteryReattribute));
|
||||||
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
BatteryReattributeEntity(
|
||||||
|
long timestampStart, long timestampEnd, @NonNull String reattributeData) {
|
||||||
|
this.timestampStart = timestampStart;
|
||||||
|
this.timestampEnd = timestampEnd;
|
||||||
|
this.reattributeData = reattributeData;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
final BatteryReattribute batteryReattribute =
|
||||||
|
ConvertUtils.decodeBatteryReattribute(reattributeData);
|
||||||
|
final StringBuilder builder = new StringBuilder()
|
||||||
|
.append("\nBatteryReattributeEntity{")
|
||||||
|
.append("\n\t" + utcToLocalTimeForLogging(timestampStart))
|
||||||
|
.append("\n\t" + utcToLocalTimeForLogging(timestampEnd))
|
||||||
|
.append("\n\t" + batteryReattribute);
|
||||||
|
if (batteryReattribute != null) {
|
||||||
|
builder.append("\n\t" + batteryReattribute.getReattributeDataMap());
|
||||||
|
}
|
||||||
|
return builder.append("\n}").toString();
|
||||||
|
}
|
||||||
|
}
|
@@ -19,6 +19,7 @@ package com.android.settings.fuelgauge.batteryusage.db;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
import androidx.room.Database;
|
import androidx.room.Database;
|
||||||
import androidx.room.Room;
|
import androidx.room.Room;
|
||||||
import androidx.room.RoomDatabase;
|
import androidx.room.RoomDatabase;
|
||||||
@@ -29,11 +30,13 @@ import androidx.room.RoomDatabase;
|
|||||||
AppUsageEventEntity.class,
|
AppUsageEventEntity.class,
|
||||||
BatteryEventEntity.class,
|
BatteryEventEntity.class,
|
||||||
BatteryState.class,
|
BatteryState.class,
|
||||||
BatteryUsageSlotEntity.class
|
BatteryUsageSlotEntity.class,
|
||||||
|
BatteryReattributeEntity.class
|
||||||
},
|
},
|
||||||
version = 1)
|
version = 2)
|
||||||
public abstract class BatteryStateDatabase extends RoomDatabase {
|
public abstract class BatteryStateDatabase extends RoomDatabase {
|
||||||
private static final String TAG = "BatteryStateDatabase";
|
private static final String TAG = "BatteryStateDatabase";
|
||||||
|
private static final String DB_FILE_NAME = "battery-usage-db-v10";
|
||||||
|
|
||||||
private static BatteryStateDatabase sBatteryStateDatabase;
|
private static BatteryStateDatabase sBatteryStateDatabase;
|
||||||
|
|
||||||
@@ -49,11 +52,15 @@ public abstract class BatteryStateDatabase extends RoomDatabase {
|
|||||||
/** Provides DAO for battery usage slot table. */
|
/** Provides DAO for battery usage slot table. */
|
||||||
public abstract BatteryUsageSlotDao batteryUsageSlotDao();
|
public abstract BatteryUsageSlotDao batteryUsageSlotDao();
|
||||||
|
|
||||||
|
/** Provides DAO for battery reattribution table. */
|
||||||
|
@NonNull
|
||||||
|
public abstract BatteryReattributeDao batteryReattributeDao();
|
||||||
|
|
||||||
/** Gets or creates an instance of {@link RoomDatabase}. */
|
/** Gets or creates an instance of {@link RoomDatabase}. */
|
||||||
public static BatteryStateDatabase getInstance(Context context) {
|
public static BatteryStateDatabase getInstance(Context context) {
|
||||||
if (sBatteryStateDatabase == null) {
|
if (sBatteryStateDatabase == null) {
|
||||||
sBatteryStateDatabase =
|
sBatteryStateDatabase =
|
||||||
Room.databaseBuilder(context, BatteryStateDatabase.class, "battery-usage-db-v9")
|
Room.databaseBuilder(context, BatteryStateDatabase.class, DB_FILE_NAME)
|
||||||
// Allows accessing data in the main thread for dumping bugreport.
|
// Allows accessing data in the main thread for dumping bugreport.
|
||||||
.allowMainThreadQueries()
|
.allowMainThreadQueries()
|
||||||
.fallbackToDestructiveMigration()
|
.fallbackToDestructiveMigration()
|
||||||
|
@@ -0,0 +1,13 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option java_multiple_files = true;
|
||||||
|
option java_package = "com.android.settings.fuelgauge.batteryusage";
|
||||||
|
option java_outer_classname = "BatteryReaatributeProto";
|
||||||
|
|
||||||
|
// Battery usage reattribute data for a specific timestamp slot.
|
||||||
|
message BatteryReattribute {
|
||||||
|
optional int64 timestamp_start = 1;
|
||||||
|
optional int64 timestamp_end = 2;
|
||||||
|
// Battery reattribute data for uid and its corresponding ratio.
|
||||||
|
map<int32, float> reattribute_data = 3;
|
||||||
|
}
|
@@ -44,6 +44,16 @@ class DisplayOverOtherAppsListModel(context: Context) : AppOpPermissionListModel
|
|||||||
logPermissionChange(newAllowed)
|
logPermissionChange(newAllowed)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO (b/349195999)
|
||||||
|
override fun isChangeable(record: AppOpPermissionRecord): Boolean {
|
||||||
|
if (record.app.packageName in
|
||||||
|
context.resources.getStringArray(R.array.display_over_apps_permission_change_exempt)
|
||||||
|
&& record.app.isSystemApp()) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return super.isChangeable(record)
|
||||||
|
}
|
||||||
|
|
||||||
private fun logPermissionChange(newAllowed: Boolean) {
|
private fun logPermissionChange(newAllowed: Boolean) {
|
||||||
val category = when {
|
val category = when {
|
||||||
newAllowed -> SettingsEnums.APP_SPECIAL_PERMISSION_APPDRAW_ALLOW
|
newAllowed -> SettingsEnums.APP_SPECIAL_PERMISSION_APPDRAW_ALLOW
|
||||||
|
@@ -26,7 +26,7 @@ import com.android.settings.R
|
|||||||
import com.android.settings.core.BasePreferenceController
|
import com.android.settings.core.BasePreferenceController
|
||||||
import com.android.settings.flags.Flags
|
import com.android.settings.flags.Flags
|
||||||
|
|
||||||
class DeviceDiagnosticsPreferenceController(context: Context, preferenceKey: String) :
|
open class DeviceDiagnosticsPreferenceController(context: Context, preferenceKey: String) :
|
||||||
BasePreferenceController(context, preferenceKey) {
|
BasePreferenceController(context, preferenceKey) {
|
||||||
|
|
||||||
override fun getAvailabilityStatus(): Int {
|
override fun getAvailabilityStatus(): Int {
|
||||||
@@ -59,6 +59,12 @@ class DeviceDiagnosticsPreferenceController(context: Context, preferenceKey: Str
|
|||||||
val packageName = mContext.getResources().getString(
|
val packageName = mContext.getResources().getString(
|
||||||
R.string.config_device_diagnostics_package_name)
|
R.string.config_device_diagnostics_package_name)
|
||||||
intent.setPackage(packageName)
|
intent.setPackage(packageName)
|
||||||
|
|
||||||
|
val info = mContext.getPackageManager().resolveActivity(intent, 0)
|
||||||
|
if (info == null) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
|
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
|
||||||
return intent
|
return intent
|
||||||
}
|
}
|
||||||
|
@@ -17,7 +17,6 @@
|
|||||||
package com.android.settings.biometrics.fingerprint;
|
package com.android.settings.biometrics.fingerprint;
|
||||||
|
|
||||||
import static android.hardware.fingerprint.FingerprintSensorProperties.TYPE_POWER_BUTTON;
|
import static android.hardware.fingerprint.FingerprintSensorProperties.TYPE_POWER_BUTTON;
|
||||||
import static android.hardware.fingerprint.FingerprintSensorProperties.TYPE_REAR;
|
|
||||||
import static android.hardware.fingerprint.FingerprintSensorProperties.TYPE_UDFPS_OPTICAL;
|
import static android.hardware.fingerprint.FingerprintSensorProperties.TYPE_UDFPS_OPTICAL;
|
||||||
|
|
||||||
import static com.android.settings.biometrics.fingerprint.FingerprintSettings.FingerprintSettingsFragment;
|
import static com.android.settings.biometrics.fingerprint.FingerprintSettings.FingerprintSettingsFragment;
|
||||||
@@ -34,16 +33,13 @@ import static org.mockito.Mockito.doNothing;
|
|||||||
import static org.mockito.Mockito.doReturn;
|
import static org.mockito.Mockito.doReturn;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.spy;
|
import static org.mockito.Mockito.spy;
|
||||||
import static org.mockito.Mockito.times;
|
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
import static org.mockito.Mockito.when;
|
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.pm.UserInfo;
|
import android.content.pm.UserInfo;
|
||||||
import android.hardware.biometrics.ComponentInfoInternal;
|
import android.hardware.biometrics.ComponentInfoInternal;
|
||||||
import android.hardware.biometrics.SensorProperties;
|
import android.hardware.biometrics.SensorProperties;
|
||||||
import android.hardware.fingerprint.Fingerprint;
|
|
||||||
import android.hardware.fingerprint.FingerprintManager;
|
import android.hardware.fingerprint.FingerprintManager;
|
||||||
import android.hardware.fingerprint.FingerprintSensorProperties;
|
import android.hardware.fingerprint.FingerprintSensorProperties;
|
||||||
import android.hardware.fingerprint.FingerprintSensorPropertiesInternal;
|
import android.hardware.fingerprint.FingerprintSensorPropertiesInternal;
|
||||||
@@ -84,7 +80,6 @@ import org.robolectric.RobolectricTestRunner;
|
|||||||
import org.robolectric.annotation.Config;
|
import org.robolectric.annotation.Config;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
@RunWith(RobolectricTestRunner.class)
|
@RunWith(RobolectricTestRunner.class)
|
||||||
@Config(shadows = {ShadowSettingsPreferenceFragment.class, ShadowUtils.class, ShadowFragment.class,
|
@Config(shadows = {ShadowSettingsPreferenceFragment.class, ShadowUtils.class, ShadowFragment.class,
|
||||||
@@ -152,6 +147,7 @@ public class FingerprintSettingsFragmentTest {
|
|||||||
public void testCancellationSignalLifeCycle() {
|
public void testCancellationSignalLifeCycle() {
|
||||||
setUpFragment(false);
|
setUpFragment(false);
|
||||||
|
|
||||||
|
mFingerprintAuthenticateSidecar.setFingerprintManager(mFingerprintManager);
|
||||||
|
|
||||||
doNothing().when(mFingerprintManager).authenticate(any(),
|
doNothing().when(mFingerprintManager).authenticate(any(),
|
||||||
mCancellationSignalArgumentCaptor.capture(),
|
mCancellationSignalArgumentCaptor.capture(),
|
||||||
@@ -217,7 +213,6 @@ public class FingerprintSettingsFragmentTest {
|
|||||||
doReturn(fragmentManager).when(mActivity).getSupportFragmentManager();
|
doReturn(fragmentManager).when(mActivity).getSupportFragmentManager();
|
||||||
|
|
||||||
mFingerprintAuthenticateSidecar = new FingerprintAuthenticateSidecar();
|
mFingerprintAuthenticateSidecar = new FingerprintAuthenticateSidecar();
|
||||||
mFingerprintAuthenticateSidecar.setFingerprintManager(mFingerprintManager);
|
|
||||||
doReturn(mFingerprintAuthenticateSidecar).when(fragmentManager).findFragmentByTag(
|
doReturn(mFingerprintAuthenticateSidecar).when(fragmentManager).findFragmentByTag(
|
||||||
"authenticate_sidecar");
|
"authenticate_sidecar");
|
||||||
|
|
||||||
@@ -251,27 +246,4 @@ public class FingerprintSettingsFragmentTest {
|
|||||||
true /* resetLockoutRequiresHardwareAuthToken */));
|
true /* resetLockoutRequiresHardwareAuthToken */));
|
||||||
doReturn(props).when(mFingerprintManager).getSensorPropertiesInternal();
|
doReturn(props).when(mFingerprintManager).getSensorPropertiesInternal();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testAuthOnFragmentSetup() {
|
|
||||||
doReturn(List.of(new Fingerprint("Finger 1", 1, 2, 3)))
|
|
||||||
.when(mFingerprintManager).getEnrolledFingerprints(anyInt());
|
|
||||||
setUpFragment(false, 1, TYPE_REAR);
|
|
||||||
|
|
||||||
verify(mFingerprintManager).authenticate(any(), any(),
|
|
||||||
any(), any(), anyInt());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testErrorCancelledRestartsAuth() {
|
|
||||||
doReturn(List.of(new Fingerprint("Finger 1", 1, 2, 3)))
|
|
||||||
.when(mFingerprintManager).getEnrolledFingerprints(anyInt());
|
|
||||||
setUpFragment(false, 1, TYPE_REAR);
|
|
||||||
|
|
||||||
// When we receive a cancel, we should restart auth.
|
|
||||||
mFragment.handleError(FingerprintManager.FINGERPRINT_ERROR_CANCELED, "blah");
|
|
||||||
|
|
||||||
verify(mFingerprintManager, times(2)).authenticate(any(), any(),
|
|
||||||
any(), any(), anyInt());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@@ -21,26 +21,35 @@ import static android.media.Spatializer.SPATIALIZER_IMMERSIVE_LEVEL_NONE;
|
|||||||
|
|
||||||
import static com.google.common.truth.Truth.assertThat;
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
import static org.mockito.Mockito.spy;
|
import static org.mockito.Mockito.spy;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
import android.app.settings.SettingsEnums;
|
import android.app.settings.SettingsEnums;
|
||||||
import android.bluetooth.BluetoothDevice;
|
import android.bluetooth.BluetoothDevice;
|
||||||
|
import android.bluetooth.BluetoothProfile;
|
||||||
import android.media.AudioDeviceAttributes;
|
import android.media.AudioDeviceAttributes;
|
||||||
import android.media.AudioDeviceInfo;
|
import android.media.AudioDeviceInfo;
|
||||||
import android.media.AudioManager;
|
import android.media.AudioManager;
|
||||||
import android.media.Spatializer;
|
import android.media.Spatializer;
|
||||||
|
import android.platform.test.annotations.EnableFlags;
|
||||||
|
import android.platform.test.flag.junit.SetFlagsRule;
|
||||||
|
|
||||||
import androidx.preference.PreferenceCategory;
|
import androidx.preference.PreferenceCategory;
|
||||||
import androidx.preference.TwoStatePreference;
|
import androidx.preference.TwoStatePreference;
|
||||||
|
|
||||||
import com.android.settings.testutils.FakeFeatureFactory;
|
import com.android.settings.testutils.FakeFeatureFactory;
|
||||||
|
import com.android.settingslib.bluetooth.A2dpProfile;
|
||||||
|
import com.android.settingslib.bluetooth.HearingAidProfile;
|
||||||
|
import com.android.settingslib.bluetooth.LeAudioProfile;
|
||||||
import com.android.settingslib.core.lifecycle.Lifecycle;
|
import com.android.settingslib.core.lifecycle.Lifecycle;
|
||||||
|
import com.android.settingslib.flags.Flags;
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
|
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
|
import org.junit.Rule;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.junit.runner.RunWith;
|
import org.junit.runner.RunWith;
|
||||||
import org.mockito.Mock;
|
import org.mockito.Mock;
|
||||||
@@ -54,7 +63,8 @@ import java.util.List;
|
|||||||
|
|
||||||
@RunWith(RobolectricTestRunner.class)
|
@RunWith(RobolectricTestRunner.class)
|
||||||
public class BluetoothDetailsSpatialAudioControllerTest extends BluetoothDetailsControllerTestBase {
|
public class BluetoothDetailsSpatialAudioControllerTest extends BluetoothDetailsControllerTestBase {
|
||||||
|
@Rule
|
||||||
|
public final SetFlagsRule mSetFlagsRule = new SetFlagsRule();
|
||||||
private static final String MAC_ADDRESS = "04:52:C7:0B:D8:3C";
|
private static final String MAC_ADDRESS = "04:52:C7:0B:D8:3C";
|
||||||
private static final String KEY_SPATIAL_AUDIO = "spatial_audio";
|
private static final String KEY_SPATIAL_AUDIO = "spatial_audio";
|
||||||
private static final String KEY_HEAD_TRACKING = "head_tracking";
|
private static final String KEY_HEAD_TRACKING = "head_tracking";
|
||||||
@@ -64,6 +74,9 @@ public class BluetoothDetailsSpatialAudioControllerTest extends BluetoothDetails
|
|||||||
@Mock private Lifecycle mSpatialAudioLifecycle;
|
@Mock private Lifecycle mSpatialAudioLifecycle;
|
||||||
@Mock private PreferenceCategory mProfilesContainer;
|
@Mock private PreferenceCategory mProfilesContainer;
|
||||||
@Mock private BluetoothDevice mBluetoothDevice;
|
@Mock private BluetoothDevice mBluetoothDevice;
|
||||||
|
@Mock private A2dpProfile mA2dpProfile;
|
||||||
|
@Mock private LeAudioProfile mLeAudioProfile;
|
||||||
|
@Mock private HearingAidProfile mHearingAidProfile;
|
||||||
|
|
||||||
private AudioDeviceAttributes mAvailableDevice;
|
private AudioDeviceAttributes mAvailableDevice;
|
||||||
|
|
||||||
@@ -83,6 +96,12 @@ public class BluetoothDetailsSpatialAudioControllerTest extends BluetoothDetails
|
|||||||
when(mAudioManager.getSpatializer()).thenReturn(mSpatializer);
|
when(mAudioManager.getSpatializer()).thenReturn(mSpatializer);
|
||||||
when(mCachedDevice.getAddress()).thenReturn(MAC_ADDRESS);
|
when(mCachedDevice.getAddress()).thenReturn(MAC_ADDRESS);
|
||||||
when(mCachedDevice.getDevice()).thenReturn(mBluetoothDevice);
|
when(mCachedDevice.getDevice()).thenReturn(mBluetoothDevice);
|
||||||
|
when(mCachedDevice.getProfiles())
|
||||||
|
.thenReturn(List.of(mA2dpProfile, mLeAudioProfile, mHearingAidProfile));
|
||||||
|
when(mA2dpProfile.isEnabled(mBluetoothDevice)).thenReturn(true);
|
||||||
|
when(mA2dpProfile.getProfileId()).thenReturn(BluetoothProfile.A2DP);
|
||||||
|
when(mLeAudioProfile.getProfileId()).thenReturn(BluetoothProfile.LE_AUDIO);
|
||||||
|
when(mHearingAidProfile.getProfileId()).thenReturn(BluetoothProfile.HEARING_AID);
|
||||||
when(mBluetoothDevice.getAnonymizedAddress()).thenReturn(MAC_ADDRESS);
|
when(mBluetoothDevice.getAnonymizedAddress()).thenReturn(MAC_ADDRESS);
|
||||||
when(mFeatureFactory.getBluetoothFeatureProvider().getSpatializer(mContext))
|
when(mFeatureFactory.getBluetoothFeatureProvider().getSpatializer(mContext))
|
||||||
.thenReturn(mSpatializer);
|
.thenReturn(mSpatializer);
|
||||||
@@ -272,6 +291,52 @@ public class BluetoothDetailsSpatialAudioControllerTest extends BluetoothDetails
|
|||||||
false);
|
false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@EnableFlags(Flags.FLAG_ENABLE_DETERMINING_SPATIAL_AUDIO_ATTRIBUTES_BY_PROFILE)
|
||||||
|
public void refresh_leAudioProfileEnabledForHeadset_useLeAudioHeadsetAttributes() {
|
||||||
|
when(mLeAudioProfile.isEnabled(mBluetoothDevice)).thenReturn(true);
|
||||||
|
when(mA2dpProfile.isEnabled(mBluetoothDevice)).thenReturn(false);
|
||||||
|
when(mHearingAidProfile.isEnabled(mBluetoothDevice)).thenReturn(false);
|
||||||
|
when(mAudioManager.getBluetoothAudioDeviceCategory(MAC_ADDRESS))
|
||||||
|
.thenReturn(AudioManager.AUDIO_DEVICE_CATEGORY_HEADPHONES);
|
||||||
|
when(mSpatializer.isAvailableForDevice(any())).thenReturn(true);
|
||||||
|
|
||||||
|
mController.refresh();
|
||||||
|
ShadowLooper.idleMainLooper();
|
||||||
|
|
||||||
|
assertThat(mController.mAudioDevice.getType()).isEqualTo(AudioDeviceInfo.TYPE_BLE_HEADSET);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@EnableFlags(Flags.FLAG_ENABLE_DETERMINING_SPATIAL_AUDIO_ATTRIBUTES_BY_PROFILE)
|
||||||
|
public void refresh_leAudioProfileEnabledForSpeaker_useLeAudioSpeakerAttributes() {
|
||||||
|
when(mLeAudioProfile.isEnabled(mBluetoothDevice)).thenReturn(true);
|
||||||
|
when(mA2dpProfile.isEnabled(mBluetoothDevice)).thenReturn(false);
|
||||||
|
when(mHearingAidProfile.isEnabled(mBluetoothDevice)).thenReturn(false);
|
||||||
|
when(mAudioManager.getBluetoothAudioDeviceCategory(MAC_ADDRESS))
|
||||||
|
.thenReturn(AudioManager.AUDIO_DEVICE_CATEGORY_SPEAKER);
|
||||||
|
when(mSpatializer.isAvailableForDevice(any())).thenReturn(true);
|
||||||
|
|
||||||
|
mController.refresh();
|
||||||
|
ShadowLooper.idleMainLooper();
|
||||||
|
|
||||||
|
assertThat(mController.mAudioDevice.getType()).isEqualTo(AudioDeviceInfo.TYPE_BLE_SPEAKER);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@EnableFlags(Flags.FLAG_ENABLE_DETERMINING_SPATIAL_AUDIO_ATTRIBUTES_BY_PROFILE)
|
||||||
|
public void refresh_hearingAidProfileEnabled_useHearingAidAttributes() {
|
||||||
|
when(mLeAudioProfile.isEnabled(mBluetoothDevice)).thenReturn(false);
|
||||||
|
when(mA2dpProfile.isEnabled(mBluetoothDevice)).thenReturn(false);
|
||||||
|
when(mHearingAidProfile.isEnabled(mBluetoothDevice)).thenReturn(true);
|
||||||
|
when(mSpatializer.isAvailableForDevice(any())).thenReturn(true);
|
||||||
|
|
||||||
|
mController.refresh();
|
||||||
|
ShadowLooper.idleMainLooper();
|
||||||
|
|
||||||
|
assertThat(mController.mAudioDevice.getType()).isEqualTo(AudioDeviceInfo.TYPE_HEARING_AID);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void turnedOnSpatialAudio_invokesAddCompatibleAudioDevice() {
|
public void turnedOnSpatialAudio_invokesAddCompatibleAudioDevice() {
|
||||||
mController.setAvailableDevice(mAvailableDevice);
|
mController.setAvailableDevice(mAvailableDevice);
|
||||||
|
@@ -703,4 +703,26 @@ public final class ConvertUtilsTest {
|
|||||||
/* taskRootPackageName= */ ""))
|
/* taskRootPackageName= */ ""))
|
||||||
.isEqualTo(packageName);
|
.isEqualTo(packageName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void decodeBatteryReattribute_returnExpectedResult() {
|
||||||
|
final BatteryReattribute batteryReattribute =
|
||||||
|
BatteryReattribute.newBuilder()
|
||||||
|
.setTimestampStart(100L)
|
||||||
|
.setTimestampEnd(200L)
|
||||||
|
.putReattributeData(1001, 0.2f)
|
||||||
|
.putReattributeData(2001, 0.8f)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
final BatteryReattribute decodeResult = ConvertUtils.decodeBatteryReattribute(
|
||||||
|
ConvertUtils.encodeBatteryReattribute(batteryReattribute));
|
||||||
|
|
||||||
|
assertThat(decodeResult.getTimestampStart()).isEqualTo(100L);
|
||||||
|
assertThat(decodeResult.getTimestampEnd()).isEqualTo(200L);
|
||||||
|
final Map<Integer, Float> reattributeDataMap = decodeResult.getReattributeDataMap();
|
||||||
|
// Verify the reattribute data in the map.
|
||||||
|
assertThat(reattributeDataMap).hasSize(2);
|
||||||
|
assertThat(reattributeDataMap.get(1001)).isEqualTo(0.2f);
|
||||||
|
assertThat(reattributeDataMap.get(2001)).isEqualTo(0.8f);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -112,6 +112,7 @@ public final class DataProcessManagerTest {
|
|||||||
mContext,
|
mContext,
|
||||||
/* handler= */ null,
|
/* handler= */ null,
|
||||||
mUserIdsSeries,
|
mUserIdsSeries,
|
||||||
|
/* isFromPeriodJob= */ false,
|
||||||
/* rawStartTimestamp= */ 0L,
|
/* rawStartTimestamp= */ 0L,
|
||||||
/* lastFullChargeTimestamp= */ 0L,
|
/* lastFullChargeTimestamp= */ 0L,
|
||||||
/* callbackFunction= */ null,
|
/* callbackFunction= */ null,
|
||||||
@@ -258,6 +259,7 @@ public final class DataProcessManagerTest {
|
|||||||
mContext,
|
mContext,
|
||||||
/* handler= */ null,
|
/* handler= */ null,
|
||||||
mUserIdsSeries,
|
mUserIdsSeries,
|
||||||
|
/* isFromPeriodJob= */ false,
|
||||||
/* rawStartTimestamp= */ 2L,
|
/* rawStartTimestamp= */ 2L,
|
||||||
/* lastFullChargeTimestamp= */ 1L,
|
/* lastFullChargeTimestamp= */ 1L,
|
||||||
/* callbackFunction= */ null,
|
/* callbackFunction= */ null,
|
||||||
|
@@ -0,0 +1,122 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2024 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.batteryusage.bugreport;
|
||||||
|
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import androidx.test.core.app.ApplicationProvider;
|
||||||
|
|
||||||
|
import com.android.settings.fuelgauge.PowerUsageFeatureProvider;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.BatteryReattribute;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryReattributeDao;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryReattributeEntity;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.db.BatteryStateDatabase;
|
||||||
|
import com.android.settings.testutils.BatteryTestUtils;
|
||||||
|
import com.android.settings.testutils.FakeFeatureFactory;
|
||||||
|
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.FixMethodOrder;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.junit.runners.MethodSorters;
|
||||||
|
import org.robolectric.RobolectricTestRunner;
|
||||||
|
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
import java.io.StringWriter;
|
||||||
|
|
||||||
|
@FixMethodOrder(MethodSorters.NAME_ASCENDING)
|
||||||
|
@RunWith(RobolectricTestRunner.class)
|
||||||
|
public final class LogUtilsTest {
|
||||||
|
|
||||||
|
private StringWriter mTestStringWriter;
|
||||||
|
private PrintWriter mTestPrintWriter;
|
||||||
|
private Context mContext;
|
||||||
|
private BatteryStateDatabase mDatabase;
|
||||||
|
private BatteryReattributeDao mBatteryReattributeDao;
|
||||||
|
private PowerUsageFeatureProvider mPowerUsageFeatureProvider;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
mContext = ApplicationProvider.getApplicationContext();
|
||||||
|
mTestStringWriter = new StringWriter();
|
||||||
|
mTestPrintWriter = new PrintWriter(mTestStringWriter);
|
||||||
|
mDatabase = BatteryTestUtils.setUpBatteryStateDatabase(mContext);
|
||||||
|
mBatteryReattributeDao = mDatabase.batteryReattributeDao();
|
||||||
|
mPowerUsageFeatureProvider = FakeFeatureFactory.setupForTest().powerUsageFeatureProvider;
|
||||||
|
when(mPowerUsageFeatureProvider.isBatteryUsageReattributeEnabled()).thenReturn(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void cleanUp() {
|
||||||
|
mBatteryReattributeDao.clearAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void dumpBatteryReattributeDatabaseHist_noData_printExpectedResult() {
|
||||||
|
LogUtils.dumpBatteryReattributeDatabaseHist(mBatteryReattributeDao, mTestPrintWriter);
|
||||||
|
|
||||||
|
assertThat(mTestStringWriter.toString())
|
||||||
|
.contains("BatteryReattribute DatabaseHistory:");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void dumpBatteryReattributeDatabaseHist_printExpectedResult() {
|
||||||
|
final long currentTimeMillis = System.currentTimeMillis();
|
||||||
|
// Insert the first testing data.
|
||||||
|
final BatteryReattribute batteryReattribute1 =
|
||||||
|
BatteryReattribute.newBuilder()
|
||||||
|
.setTimestampStart(currentTimeMillis - 20000)
|
||||||
|
.setTimestampEnd(currentTimeMillis - 10000)
|
||||||
|
.putReattributeData(1001, 0.1f)
|
||||||
|
.putReattributeData(1002, 0.99f)
|
||||||
|
.build();
|
||||||
|
mBatteryReattributeDao.insert(new BatteryReattributeEntity(batteryReattribute1));
|
||||||
|
// Insert the second testing data.
|
||||||
|
final BatteryReattribute batteryReattribute2 =
|
||||||
|
BatteryReattribute.newBuilder()
|
||||||
|
.setTimestampStart(currentTimeMillis - 40000)
|
||||||
|
.setTimestampEnd(currentTimeMillis - 20000)
|
||||||
|
.putReattributeData(1003, 1f)
|
||||||
|
.build();
|
||||||
|
mBatteryReattributeDao.insert(new BatteryReattributeEntity(batteryReattribute2));
|
||||||
|
|
||||||
|
LogUtils.dumpBatteryReattributeDatabaseHist(mBatteryReattributeDao, mTestPrintWriter);
|
||||||
|
|
||||||
|
final String result = mTestStringWriter.toString();
|
||||||
|
assertThat(result).contains("BatteryReattribute DatabaseHistory:");
|
||||||
|
assertThat(result).contains(batteryReattribute1.toString());
|
||||||
|
assertThat(result).contains(batteryReattribute2.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void dumpBatteryReattributeDatabaseHist_featureDisable_notPrintData() {
|
||||||
|
mBatteryReattributeDao.insert(new BatteryReattributeEntity(
|
||||||
|
BatteryReattribute.getDefaultInstance()));
|
||||||
|
when(mPowerUsageFeatureProvider.isBatteryUsageReattributeEnabled()).thenReturn(false);
|
||||||
|
|
||||||
|
LogUtils.dumpBatteryReattributeDatabaseHist(mBatteryReattributeDao, mTestPrintWriter);
|
||||||
|
|
||||||
|
final String result = mTestStringWriter.toString();
|
||||||
|
assertThat(result).contains("BatteryReattribute is disabled!");
|
||||||
|
assertThat(result.contains("BatteryReattribute DatabaseHistory:")).isFalse();
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,118 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2024 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.batteryusage.db;
|
||||||
|
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import androidx.test.core.app.ApplicationProvider;
|
||||||
|
|
||||||
|
import com.android.settings.testutils.BatteryTestUtils;
|
||||||
|
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.robolectric.RobolectricTestRunner;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/** Tests for {@link BatteryReattributeDao}. */
|
||||||
|
@RunWith(RobolectricTestRunner.class)
|
||||||
|
public final class BatteryReattributeDaoTest {
|
||||||
|
|
||||||
|
private Context mContext;
|
||||||
|
private BatteryStateDatabase mDatabase;
|
||||||
|
private BatteryReattributeDao mBatteryReattributeDao;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
mContext = ApplicationProvider.getApplicationContext();
|
||||||
|
mDatabase = BatteryTestUtils.setUpBatteryStateDatabase(mContext);
|
||||||
|
mBatteryReattributeDao = mDatabase.batteryReattributeDao();
|
||||||
|
insert(100L, 200L, "reattributeData1");
|
||||||
|
insert(300L, 400L, "reattributeData3");
|
||||||
|
insert(200L, 300L, "reattributeData2");
|
||||||
|
insert(400L, 500L, "reattributeData4");
|
||||||
|
// Ensure there was data inserted into the database.
|
||||||
|
assertThat(getAllEntityData()).isNotEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getAllAfter_returnExpectedEntityData() {
|
||||||
|
final List<BatteryReattributeEntity> entityDataList =
|
||||||
|
mBatteryReattributeDao.getAllAfter(/* timestampStart= */ 300L);
|
||||||
|
|
||||||
|
assertThat(entityDataList).hasSize(2);
|
||||||
|
assertEntity(entityDataList.get(0), 400L, 500L, "reattributeData4");
|
||||||
|
assertEntity(entityDataList.get(1), 300L, 400L, "reattributeData3");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void clearAll_clearAllData() {
|
||||||
|
mBatteryReattributeDao.clearAll();
|
||||||
|
|
||||||
|
assertThat(getAllEntityData()).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void clearAllBefore_clearAllExpectedData() {
|
||||||
|
mBatteryReattributeDao.clearAllBefore(/* timestampStart= */ 300L);
|
||||||
|
|
||||||
|
final List<BatteryReattributeEntity> entityDataList = getAllEntityData();
|
||||||
|
assertThat(entityDataList).hasSize(1);
|
||||||
|
assertEntity(entityDataList.get(0), 400L, 500L, "reattributeData4");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void clearAllAfter_clearAllExpectedData() {
|
||||||
|
mBatteryReattributeDao.clearAllAfter(/* timestampStart= */ 300L);
|
||||||
|
|
||||||
|
final List<BatteryReattributeEntity> entityDataList = getAllEntityData();
|
||||||
|
assertThat(entityDataList).hasSize(2);
|
||||||
|
assertEntity(entityDataList.get(0), 200L, 300L, "reattributeData2");
|
||||||
|
assertEntity(entityDataList.get(1), 100L, 200L, "reattributeData1");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void insert_samePrimaryKeyEntityData_replaceIntoNewEntityData() {
|
||||||
|
// Verify the original data before update.
|
||||||
|
assertEntity(getAllEntityData().get(0), 400L, 500L, "reattributeData4");
|
||||||
|
|
||||||
|
insert(400L, 600L, "reattribute4Update");
|
||||||
|
|
||||||
|
// Verify the new update entity data.
|
||||||
|
assertEntity(getAllEntityData().get(0), 400L, 600L, "reattribute4Update");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void insert(long timestampStart, long timestampEnd, String reattributeData) {
|
||||||
|
mBatteryReattributeDao.insert(
|
||||||
|
new BatteryReattributeEntity(
|
||||||
|
timestampStart, timestampEnd, reattributeData));
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<BatteryReattributeEntity> getAllEntityData() {
|
||||||
|
return mBatteryReattributeDao.getAllAfter(/* timestampStart= */ 0L);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void assertEntity(BatteryReattributeEntity entity, long timestampStart,
|
||||||
|
long timestampEnd, String reattributeData) {
|
||||||
|
assertThat(entity.timestampStart).isEqualTo(timestampStart);
|
||||||
|
assertThat(entity.timestampEnd).isEqualTo(timestampEnd);
|
||||||
|
assertThat(entity.reattributeData).isEqualTo(reattributeData);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,54 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2024 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.batteryusage.db;
|
||||||
|
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.BatteryReattribute;
|
||||||
|
import com.android.settings.fuelgauge.batteryusage.ConvertUtils;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.robolectric.RobolectricTestRunner;
|
||||||
|
|
||||||
|
/** Tests for {@link BatteryReattributeEntity}. */
|
||||||
|
@RunWith(RobolectricTestRunner.class)
|
||||||
|
public final class BatteryReattributeEntityTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void constructor_createExpectedData() {
|
||||||
|
final BatteryReattribute batteryReattribute =
|
||||||
|
BatteryReattribute.newBuilder()
|
||||||
|
.setTimestampStart(100L)
|
||||||
|
.setTimestampEnd(200L)
|
||||||
|
.putReattributeData(1001, 0.2f)
|
||||||
|
.putReattributeData(2001, 0.8f)
|
||||||
|
.build();
|
||||||
|
|
||||||
|
final BatteryReattributeEntity batteryReattributeEntity =
|
||||||
|
new BatteryReattributeEntity(batteryReattribute);
|
||||||
|
|
||||||
|
assertThat(batteryReattributeEntity.timestampStart)
|
||||||
|
.isEqualTo(batteryReattribute.getTimestampStart());
|
||||||
|
assertThat(batteryReattributeEntity.timestampEnd)
|
||||||
|
.isEqualTo(batteryReattribute.getTimestampEnd());
|
||||||
|
// Verify the BatteryReattribute data.
|
||||||
|
final BatteryReattribute decodeResult =
|
||||||
|
ConvertUtils.decodeBatteryReattribute(batteryReattributeEntity.reattributeData);
|
||||||
|
assertThat(decodeResult).isEqualTo(batteryReattribute);
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user