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

Commit b44b609c authored by Michal Brzezinski's avatar Michal Brzezinski
Browse files

Showing dialog even when dismissed externally and timeout didn't pass

Flag: KEYBOARD_BACKLIGHT_INDICATOR
Bug: 281845445
Test: Show dialog -> within 3 s [touch outside of it to dismiss -> change brightness again] -> dialog should appear
Test: KeyboardBacklightDialogCoordinatorTest
Change-Id: I32168b24e24cf795133ae2699746b481cc3caf4f
parent 75c52789
Loading
Loading
Loading
Loading
+29 −15
Original line number Diff line number Diff line
@@ -21,41 +21,44 @@ import android.content.Context
import com.android.systemui.dagger.SysUISingleton
import com.android.systemui.dagger.qualifiers.Application
import com.android.systemui.keyboard.backlight.ui.view.KeyboardBacklightDialog
import com.android.systemui.keyboard.backlight.ui.viewmodel.BacklightDialogContentViewModel
import com.android.systemui.keyboard.backlight.ui.viewmodel.BacklightDialogViewModel
import javax.inject.Inject
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch

private fun defaultCreateDialog(context: Context): (Int, Int) -> KeyboardBacklightDialog {
    return { currentLevel: Int, maxLevel: Int ->
        KeyboardBacklightDialog(context, currentLevel, maxLevel)
    }
}

/**
 * Based on the state produced from [BacklightDialogViewModel] shows or hides keyboard backlight
 * indicator
 */
@SysUISingleton
class KeyboardBacklightDialogCoordinator
@Inject
constructor(
    @Application private val applicationScope: CoroutineScope,
    private val context: Context,
    private val viewModel: BacklightDialogViewModel,
    private val createDialog: (Int, Int) -> KeyboardBacklightDialog
) {

    @Inject
    constructor(
        @Application applicationScope: CoroutineScope,
        context: Context,
        viewModel: BacklightDialogViewModel
    ) : this(applicationScope, viewModel, defaultCreateDialog(context))

    var dialog: KeyboardBacklightDialog? = null

    fun startListening() {
        applicationScope.launch {
            viewModel.dialogContent.collect { dialogViewModel ->
                if (dialogViewModel != null) {
                    if (dialog == null) {
                        dialog =
                            KeyboardBacklightDialog(
                                context,
                                initialCurrentLevel = dialogViewModel.currentValue,
                                initialMaxLevel = dialogViewModel.maxValue
                            )
                        dialog?.show()
                    } else {
                        dialog?.updateState(dialogViewModel.currentValue, dialogViewModel.maxValue)
                    }
            viewModel.dialogContent.collect { contentModel ->
                if (contentModel != null) {
                    showDialog(contentModel)
                } else {
                    dialog?.dismiss()
                    dialog = null
@@ -63,4 +66,15 @@ constructor(
            }
        }
    }

    private fun showDialog(model: BacklightDialogContentViewModel) {
        if (dialog == null) {
            dialog = createDialog(model.currentValue, model.maxValue)
        } else {
            dialog?.updateState(model.currentValue, model.maxValue)
        }
        // let's always show dialog - even if we're just updating it, it might have been dismissed
        // externally by tapping finger outside of it
        dialog?.show()
    }
}
+158 −0
Original line number Diff line number Diff line
@@ -15,88 +15,144 @@
 *
 */

package com.android.systemui.keyboard.backlight.ui.viewmodel
package com.android.systemui.keyboard.backlight.ui

import androidx.test.filters.SmallTest
import com.android.systemui.SysuiTestCase
import com.android.systemui.coroutines.collectLastValue
import com.android.systemui.keyboard.backlight.domain.interactor.KeyboardBacklightInteractor
import com.android.systemui.keyboard.backlight.ui.view.KeyboardBacklightDialog
import com.android.systemui.keyboard.backlight.ui.viewmodel.BacklightDialogViewModel
import com.android.systemui.keyboard.data.repository.FakeKeyboardRepository
import com.android.systemui.keyboard.shared.model.BacklightModel
import com.android.systemui.statusbar.policy.AccessibilityManagerWrapper
import com.android.systemui.util.mockito.any
import com.android.systemui.util.mockito.whenever
import com.google.common.truth.Truth.assertThat
import com.google.common.truth.Truth
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.flow.first
import kotlinx.coroutines.test.StandardTestDispatcher
import kotlinx.coroutines.test.TestScope
import kotlinx.coroutines.test.advanceTimeBy
import kotlinx.coroutines.test.runCurrent
import kotlinx.coroutines.test.runTest
import org.junit.Before
import org.junit.Test
import org.junit.runner.RunWith
import org.junit.runners.JUnit4
import org.mockito.Mock
import org.mockito.Mockito.never
import org.mockito.Mockito.times
import org.mockito.Mockito.verify
import org.mockito.MockitoAnnotations

@OptIn(ExperimentalCoroutinesApi::class)
@SmallTest
@RunWith(JUnit4::class)
class BacklightDialogViewModelTest : SysuiTestCase() {
class KeyboardBacklightDialogCoordinatorTest : SysuiTestCase() {

    private val keyboardRepository = FakeKeyboardRepository()
    private lateinit var underTest: BacklightDialogViewModel
    @Mock private lateinit var accessibilityManagerWrapper: AccessibilityManagerWrapper
    @Mock private lateinit var dialog: KeyboardBacklightDialog

    private val keyboardRepository = FakeKeyboardRepository()
    private lateinit var underTest: KeyboardBacklightDialogCoordinator
    private val timeoutMillis = 3000L
    private val testScope = TestScope(StandardTestDispatcher())

    private val createDialog = { value: Int, maxValue: Int ->
        dialogCreationValue = value
        dialogCreationMaxValue = maxValue
        dialog
    }
    private var dialogCreationValue = -1
    private var dialogCreationMaxValue = -1

    @Before
    fun setUp() {
        MockitoAnnotations.initMocks(this)
        whenever(accessibilityManagerWrapper.getRecommendedTimeoutMillis(any(), any()))
            .thenReturn(timeoutMillis.toInt())
        underTest =
        val viewModel =
            BacklightDialogViewModel(
                KeyboardBacklightInteractor(keyboardRepository),
                accessibilityManagerWrapper
            )
        underTest =
            KeyboardBacklightDialogCoordinator(testScope.backgroundScope, viewModel, createDialog)
        underTest.startListening()
        keyboardRepository.setIsAnyKeyboardConnected(true)
    }

    @Test
    fun emitsViewModel_whenBacklightChanged() = runTest {
        keyboardRepository.setBacklight(BacklightModel(1, 5))
    fun showsDialog_afterBacklightChange() =
        testScope.runTest {
            setBacklightValue(1)

        assertThat(underTest.dialogContent.first()).isEqualTo(BacklightDialogContentViewModel(1, 5))
            verify(dialog).show()
        }

    @Test
    fun emitsNull_afterTimeout() = runTest {
        val latest by collectLastValue(underTest.dialogContent)
        keyboardRepository.setBacklight(BacklightModel(1, 5))
    fun updatesDialog_withLatestValues_afterBacklightChange() =
        testScope.runTest {
            setBacklightValue(value = 1, maxValue = 5)
            setBacklightValue(value = 2, maxValue = 5)

        assertThat(latest).isEqualTo(BacklightDialogContentViewModel(1, 5))
        advanceTimeBy(timeoutMillis + 1)
        assertThat(latest).isNull()
            verify(dialog).updateState(2, 5)
        }

    @Test
    fun emitsNull_after5secDelay_fromLastBacklightChange() = runTest {
        val latest by collectLastValue(underTest.dialogContent)
        keyboardRepository.setIsAnyKeyboardConnected(true)
    fun showsDialog_withDataFromBacklightChange() =
        testScope.runTest {
            setBacklightValue(value = 4, maxValue = 5)

            Truth.assertThat(dialogCreationValue).isEqualTo(4)
            Truth.assertThat(dialogCreationMaxValue).isEqualTo(5)
        }

    @Test
    fun dismissesDialog_afterTimeout() =
        testScope.runTest {
            setBacklightValue(1)

        keyboardRepository.setBacklight(BacklightModel(1, 5))
        assertThat(latest).isEqualTo(BacklightDialogContentViewModel(1, 5))
            advanceTimeBy(timeoutMillis + 1)

            verify(dialog).dismiss()
        }

    @Test
    fun dismissesDialog_onlyAfterTimeout_fromLastBacklightChange() =
        testScope.runTest {
            setBacklightValue(1)
            advanceTimeBy(timeoutMillis * 2 / 3)
        // timeout yet to pass, no new emission
        keyboardRepository.setBacklight(BacklightModel(2, 5))
        assertThat(latest).isEqualTo(BacklightDialogContentViewModel(2, 5))
            // majority of timeout passed

            // this should restart timeout
            setBacklightValue(2)
            advanceTimeBy(timeoutMillis * 2 / 3)
        // timeout refreshed because of last `setBacklight`, still content present
        assertThat(latest).isEqualTo(BacklightDialogContentViewModel(2, 5))
            verify(dialog, never()).dismiss()

            advanceTimeBy(timeoutMillis * 2 / 3)
        // finally timeout reached and null emitted
        assertThat(latest).isNull()
            // finally timeout reached and dialog was dismissed
            verify(dialog, times(1)).dismiss()
        }

    @Test
    fun showsDialog_ifItWasAlreadyShownAndDismissedBySomethingElse() =
        testScope.runTest {
            setBacklightValue(1)
            // let's pretend dialog is dismissed e.g. by user tapping on the screen
            whenever(dialog.isShowing).thenReturn(false)

            // no advancing time, we're still in timeout period
            setBacklightValue(2)

            verify(dialog, times(2)).show()
        }

    private fun TestScope.setBacklightValue(value: Int, maxValue: Int = MAX_BACKLIGHT) {
        keyboardRepository.setBacklight(BacklightModel(value, maxValue))
        runCurrent()
    }

    private companion object {
        const val MAX_BACKLIGHT = 5
    }
}