/*
 * Copyright (C) 2019 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.development.bluetooth;

import static com.google.common.truth.Truth.assertThat;

import static org.mockito.Mockito.eq;
import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;

import android.bluetooth.BluetoothA2dp;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.bluetooth.BluetoothProfile;
import android.content.Context;

import androidx.lifecycle.LifecycleOwner;
import androidx.preference.PreferenceScreen;
import androidx.preference.SwitchPreference;

import com.android.settings.development.BluetoothA2dpConfigStore;
import com.android.settingslib.core.lifecycle.Lifecycle;

import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import org.robolectric.RobolectricTestRunner;
import org.robolectric.RuntimeEnvironment;
import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowBluetoothDevice;

import java.util.Arrays;
import java.util.Collections;

@RunWith(RobolectricTestRunner.class)
@Config(shadows = {ShadowBluetoothDevice.class})
public class BluetoothHDAudioPreferenceControllerTest {

    private static final String TEST_DEVICE_ADDRESS = "00:A1:A1:A1:A1:A1";

    @Mock
    private BluetoothA2dp mBluetoothA2dp;
    @Mock
    private BluetoothAdapter mBluetoothAdapter;
    @Mock
    private PreferenceScreen mScreen;
    @Mock
    private AbstractBluetoothPreferenceController.Callback mCallback;

    private BluetoothHDAudioPreferenceController mController;
    private SwitchPreference mPreference;
    private BluetoothA2dpConfigStore mBluetoothA2dpConfigStore;
    private BluetoothDevice mActiveDevice;
    private Context mContext;
    private LifecycleOwner mLifecycleOwner;
    private Lifecycle mLifecycle;

    @Before
    public void setup() {
        MockitoAnnotations.initMocks(this);
        mContext = RuntimeEnvironment.application;
        mLifecycleOwner = () -> mLifecycle;
        mLifecycle = new Lifecycle(mLifecycleOwner);
        mBluetoothA2dpConfigStore = spy(new BluetoothA2dpConfigStore());
        mController = new BluetoothHDAudioPreferenceController(mContext, mLifecycle,
                mBluetoothA2dpConfigStore, mCallback);
        mController.mBluetoothAdapter = mBluetoothAdapter;
        mPreference = new SwitchPreference(mContext);
        when(mScreen.findPreference(mController.getPreferenceKey())).thenReturn(mPreference);
        mController.displayPreference(mScreen);
        mActiveDevice = ShadowBluetoothDevice.newInstance(TEST_DEVICE_ADDRESS);
    }

    @Test
    public void updateState_noActiveDevice_setDisable() {
        when(mBluetoothAdapter.getActiveDevices(eq(BluetoothProfile.A2DP)))
                .thenReturn(Collections.emptyList());
        mController.onBluetoothServiceConnected(mBluetoothA2dp);
        mController.updateState(mPreference);

        assertThat(mPreference.isEnabled()).isFalse();
    }

    @Test
    public void updateState_codecSupported_setEnable() {
        when(mBluetoothAdapter.getActiveDevices(eq(BluetoothProfile.A2DP)))
                .thenReturn(Arrays.asList(mActiveDevice));
        when(mBluetoothA2dp.isOptionalCodecsSupported(mActiveDevice)).thenReturn(
                mBluetoothA2dp.OPTIONAL_CODECS_SUPPORTED);
        mController.onBluetoothServiceConnected(mBluetoothA2dp);
        mController.updateState(mPreference);

        assertThat(mPreference.isEnabled()).isTrue();
    }

    @Test
    public void updateState_codecNotSupported_setDisable() {
        when(mBluetoothAdapter.getActiveDevices(eq(BluetoothProfile.A2DP)))
                .thenReturn(Arrays.asList(mActiveDevice));
        when(mBluetoothA2dp.isOptionalCodecsSupported(mActiveDevice)).thenReturn(
                mBluetoothA2dp.OPTIONAL_CODECS_NOT_SUPPORTED);
        mController.onBluetoothServiceConnected(mBluetoothA2dp);
        mController.updateState(mPreference);

        assertThat(mPreference.isEnabled()).isFalse();
    }

    @Test
    public void updateState_codecSupportedAndEnabled_checked() {
        when(mBluetoothAdapter.getActiveDevices(eq(BluetoothProfile.A2DP)))
                .thenReturn(Arrays.asList(mActiveDevice));
        when(mBluetoothA2dp.isOptionalCodecsSupported(mActiveDevice)).thenReturn(
                mBluetoothA2dp.OPTIONAL_CODECS_SUPPORTED);
        when(mBluetoothA2dp.isOptionalCodecsEnabled(mActiveDevice)).thenReturn(
                mBluetoothA2dp.OPTIONAL_CODECS_PREF_ENABLED);
        mController.onBluetoothServiceConnected(mBluetoothA2dp);
        mController.updateState(mPreference);

        assertThat(mPreference.isChecked()).isTrue();
    }

    @Test
    public void updateState_codecSupportedAndDisabled_notChecked() {
        when(mBluetoothAdapter.getActiveDevices(eq(BluetoothProfile.A2DP)))
                .thenReturn(Arrays.asList(mActiveDevice));
        when(mBluetoothA2dp.isOptionalCodecsSupported(mActiveDevice)).thenReturn(
                mBluetoothA2dp.OPTIONAL_CODECS_SUPPORTED);
        when(mBluetoothA2dp.isOptionalCodecsEnabled(mActiveDevice)).thenReturn(
                mBluetoothA2dp.OPTIONAL_CODECS_PREF_DISABLED);
        mController.onBluetoothServiceConnected(mBluetoothA2dp);
        mController.updateState(mPreference);

        assertThat(mPreference.isChecked()).isFalse();
    }

    @Test
    public void onPreferenceChange_disable_verifyFlow() {
        when(mBluetoothAdapter.getActiveDevices(eq(BluetoothProfile.A2DP)))
                .thenReturn(Arrays.asList(mActiveDevice));
        mController.onBluetoothServiceConnected(mBluetoothA2dp);
        final boolean enabled = false;
        mController.onPreferenceChange(mPreference, enabled);

        verify(mBluetoothA2dp).disableOptionalCodecs(mActiveDevice);
        verify(mBluetoothA2dp).setOptionalCodecsEnabled(mActiveDevice,
                BluetoothA2dp.OPTIONAL_CODECS_PREF_DISABLED);
        verify(mCallback).onBluetoothHDAudioEnabled(enabled);
    }

    @Test
    public void onPreferenceChange_enable_verifyFlow() {
        when(mBluetoothAdapter.getActiveDevices(eq(BluetoothProfile.A2DP)))
                .thenReturn(Arrays.asList(mActiveDevice));
        mController.onBluetoothServiceConnected(mBluetoothA2dp);
        final boolean enabled = true;
        mController.onPreferenceChange(mPreference, enabled);

        verify(mBluetoothA2dp).enableOptionalCodecs(mActiveDevice);
        verify(mBluetoothA2dp).setOptionalCodecsEnabled(mActiveDevice,
                BluetoothA2dp.OPTIONAL_CODECS_PREF_ENABLED);
        verify(mCallback).onBluetoothHDAudioEnabled(enabled);
    }
}
