Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 8f4f3711 authored by Mill Chen's avatar Mill Chen
Browse files

[Catalyst] Call volume migration

Bug: 364898621
Test: atest CallVolumePreferenceTest
Flag: com.android.settings.flags.catalyst_sound_screen
Change-Id: I1464bd42fde7e0ae9063e2319c674b5e439b97f9
parent 3f1c7b0f
Loading
Loading
Loading
Loading
+115 −0
Original line number Diff line number Diff line
/*
 * 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.notification

import android.content.Context
import android.media.AudioManager
import android.media.AudioManager.STREAM_BLUETOOTH_SCO
import android.media.AudioManager.STREAM_VOICE_CALL
import android.os.UserHandle
import android.os.UserManager.DISALLOW_ADJUST_VOLUME
import androidx.preference.Preference
import com.android.settings.R
import com.android.settingslib.RestrictedLockUtilsInternal
import com.android.settingslib.datastore.KeyValueStore
import com.android.settingslib.datastore.NoOpKeyedObservable
import com.android.settingslib.metadata.PersistentPreference
import com.android.settingslib.metadata.PreferenceAvailabilityProvider
import com.android.settingslib.metadata.PreferenceIconProvider
import com.android.settingslib.metadata.PreferenceMetadata
import com.android.settingslib.metadata.PreferenceRestrictionProvider
import com.android.settingslib.metadata.RangeValue
import com.android.settingslib.preference.PreferenceBinding

// LINT.IfChange
open class CallVolumePreference :
    PreferenceMetadata,
    PreferenceBinding,
    PersistentPreference<Int>,
    RangeValue,
    PreferenceAvailabilityProvider,
    PreferenceIconProvider,
    PreferenceRestrictionProvider {
    override val key: String
        get() = KEY

    override val title: Int
        get() = R.string.call_volume_option_title

    override fun getIcon(context: Context) = R.drawable.ic_local_phone_24_lib

    override fun order(context: Context) = -170

    override fun isAvailable(context: Context) =
        context.resources.getBoolean(R.bool.config_show_call_volume) &&
                !createAudioHelper(context).isSingleVolume()

    override fun isRestricted(context: Context) =
        RestrictedLockUtilsInternal.hasBaseUserRestriction(
            context,
            DISALLOW_ADJUST_VOLUME,
            UserHandle.myUserId()
        ) || RestrictedLockUtilsInternal.checkIfRestrictionEnforced(
            context,
            DISALLOW_ADJUST_VOLUME,
            UserHandle.myUserId()
        ) != null

    override fun storage(context: Context): KeyValueStore {
        val helper = createAudioHelper(context)
        return object : NoOpKeyedObservable<String>(), KeyValueStore {
            override fun contains(key: String) = key == KEY

            @Suppress("UNCHECKED_CAST")
            override fun <T : Any> getValue(key: String, valueType: Class<T>) =
                helper.getStreamVolume(getAudioStream(context)) as T

            override fun <T : Any> setValue(key: String, valueType: Class<T>, value: T?) {
                helper.setStreamVolume(getAudioStream(context), value as Int)
            }
        }
    }

    override fun getMinValue(context: Context) =
        createAudioHelper(context).getMinVolume(getAudioStream(context))

    override fun getMaxValue(context: Context) =
        createAudioHelper(context).getMaxVolume(getAudioStream(context))

    override fun createWidget(context: Context) = VolumeSeekBarPreference(context)

    override fun bind(preference: Preference, metadata: PreferenceMetadata) {
        super.bind(preference, metadata)
        (preference as VolumeSeekBarPreference).setStream(getAudioStream(preference.context))
    }

    open fun createAudioHelper(context: Context) = AudioHelper(context)

    @Suppress("DEPRECATION")
    fun getAudioStream(context: Context): Int {
        val audioManager = context.getSystemService(AudioManager::class.java)
        return when {
            audioManager.isBluetoothScoOn() -> STREAM_BLUETOOTH_SCO
            else -> STREAM_VOICE_CALL
        }
    }

    companion object {
        const val KEY = "call_volume"
    }
}
// LINT.ThenChange(CallVolumePreferenceController.java)
+2 −1
Original line number Diff line number Diff line
@@ -22,7 +22,7 @@ import android.text.TextUtils;

import com.android.settings.R;


// LINT.IfChange
public class CallVolumePreferenceController extends VolumeSeekBarPreferenceController {

    private AudioManager mAudioManager;
@@ -69,3 +69,4 @@ public class CallVolumePreferenceController extends VolumeSeekBarPreferenceContr
    }

}
// LINT.ThenChange(CallVolumePreference.kt)
+1 −0
Original line number Diff line number Diff line
@@ -49,6 +49,7 @@ class SoundScreen : PreferenceScreenCreator, PreferenceIconProvider {

    override fun getPreferenceHierarchy(context: Context) =
        preferenceHierarchy(this) {
            +CallVolumePreference()
            +DialPadTonePreference()
        }

+2 −0
Original line number Diff line number Diff line
@@ -36,6 +36,7 @@ import org.mockito.MockitoAnnotations;
import org.robolectric.RobolectricTestRunner;
import org.robolectric.RuntimeEnvironment;

// LINT.IfChange
@RunWith(RobolectricTestRunner.class)
public class CallVolumePreferenceControllerTest {
    private static final String TEST_KEY = "Test_Key";
@@ -108,3 +109,4 @@ public class CallVolumePreferenceControllerTest {
        assertThat(mController.isPublicSlice()).isTrue();
    }
}
// LINT.ThenChange(CallVolumePreferenceTest.kt)
+100 −0
Original line number Diff line number Diff line
/*
 * 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.notification

import android.content.Context
import android.content.ContextWrapper
import android.content.res.Resources
import android.media.AudioManager
import android.media.AudioManager.STREAM_BLUETOOTH_SCO
import android.media.AudioManager.STREAM_VOICE_CALL
import androidx.test.core.app.ApplicationProvider
import androidx.test.ext.junit.runners.AndroidJUnit4
import com.google.common.truth.Truth.assertThat
import org.junit.Test
import org.junit.runner.RunWith
import org.mockito.ArgumentMatchers.anyInt
import org.mockito.kotlin.doReturn
import org.mockito.kotlin.mock
import org.mockito.kotlin.spy
import org.mockito.kotlin.stub

// LINT.IfChange
@RunWith(AndroidJUnit4::class)
class CallVolumePreferenceTest {
    private var audioHelper = mock<AudioHelper>()
    private var mockResources = mock<Resources>()

    private var audioManager: AudioManager? = null

    private var callVolumePreference = CallVolumePreference()
    private val context = object : ContextWrapper(ApplicationProvider.getApplicationContext()) {
        override fun getSystemService(name: String): Any? =
            when (name) {
                Context.AUDIO_SERVICE -> audioManager
                else -> super.getSystemService(name)
            }

        override fun getResources(): Resources = mockResources
    }

    @Test
    fun isAvailable_configTrueAndNoSingleVolume_shouldReturnTrue() {
        mockResources.stub { on { getBoolean(anyInt()) } doReturn true }
        audioHelper = mock { on { isSingleVolume } doReturn false }
        callVolumePreference = spy(callVolumePreference).stub {
            onGeneric { createAudioHelper(context) } doReturn audioHelper
        }

        assertThat(callVolumePreference.isAvailable(context)).isTrue()
    }

    @Test
    fun isAvailable_configTrueAndSingleVolume_shouldReturnFalse() {
        mockResources.stub { on { getBoolean(anyInt()) } doReturn true }
        audioHelper = mock { on { isSingleVolume } doReturn true }
        callVolumePreference = spy(callVolumePreference).stub {
            onGeneric { createAudioHelper(context) } doReturn audioHelper
        }

        assertThat(callVolumePreference.isAvailable(context)).isFalse()
    }

    @Test
    fun isAvailable_configFalse_shouldReturnFalse() {
        mockResources.stub { on { getBoolean(anyInt()) } doReturn false }

        assertThat(callVolumePreference.isAvailable(context)).isFalse()
    }

    @Test
    @Suppress("DEPRECATION")
    fun getAudioStream_onBluetoothScoOn_shouldEqualToStreamBluetoothSco() {
        audioManager = mock { on { isBluetoothScoOn } doReturn true }

        assertThat(callVolumePreference.getAudioStream(context)).isEqualTo(STREAM_BLUETOOTH_SCO)
    }

    @Test
    @Suppress("DEPRECATION")
    fun getAudioStream_onBluetoothScoOff_shouldEqualToStreamVoiceCall() {
        audioManager = mock { on { isBluetoothScoOn } doReturn false }

        assertThat(callVolumePreference.getAudioStream(context)).isEqualTo(STREAM_VOICE_CALL)
    }
}
// LINT.ThenChange(CallVolumePreferenceControllerTest.java)