From 26beec90d081d5925055178790316efe2026fd89 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?David=20G=C3=B6ransson?= Date: Fri, 20 Dec 2024 10:25:17 +0100 Subject: [PATCH] Rework UI state --- .../FilterUiStatePreviewParameterProvider.kt | 5 +- .../mullvadvpn/compose/screen/FilterScreen.kt | 28 ++++--- .../state/FilterConstrainExtensions.kt | 8 +- .../compose/state/RelayFilterUiState.kt | 44 ++++++----- .../relaylist/RelayItemExtensions.kt | 2 +- .../mullvadvpn/usecase/FilterChipUseCase.kt | 2 +- .../mullvadvpn/viewmodel/FilterViewModel.kt | 75 ++++++++++++------- .../RelayListFilterRepositoryTest.kt | 7 +- .../usecase/FilterChipUseCaseTest.kt | 4 +- .../viewmodel/FilterViewModelTest.kt | 17 ++--- .../lib/daemon/grpc/mapper/FromDomain.kt | 2 +- .../lib/daemon/grpc/mapper/ToDomain.kt | 2 +- .../mullvad/mullvadvpn/lib/model/Providers.kt | 7 +- 13 files changed, 109 insertions(+), 94 deletions(-) diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/preview/FilterUiStatePreviewParameterProvider.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/preview/FilterUiStatePreviewParameterProvider.kt index 0aa3be815831..319f4832151b 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/preview/FilterUiStatePreviewParameterProvider.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/preview/FilterUiStatePreviewParameterProvider.kt @@ -2,6 +2,7 @@ package net.mullvad.mullvadvpn.compose.preview import androidx.compose.ui.tooling.preview.PreviewParameterProvider import net.mullvad.mullvadvpn.compose.state.RelayFilterUiState +import net.mullvad.mullvadvpn.lib.model.Constraint import net.mullvad.mullvadvpn.lib.model.Ownership import net.mullvad.mullvadvpn.lib.model.ProviderId @@ -12,8 +13,8 @@ class FilterUiStatePreviewParameterProvider : PreviewParameterProvider Unit, onApplyClick: () -> Unit, - onSelectedOwnership: (ownership: Ownership?) -> Unit, + onSelectedOwnership: (ownership: Constraint) -> Unit, onAllProviderCheckChange: (isChecked: Boolean) -> Unit, onSelectedProvider: (checked: Boolean, provider: ProviderId) -> Unit, ) { @@ -121,14 +122,14 @@ fun FilterScreen( } if (ownershipExpanded) { item(key = Keys.OWNERSHIP_ALL, contentType = ContentType.ITEM) { - AnyOwnership(state, onSelectedOwnership) + AnyOwnership(state, { onSelectedOwnership(Constraint.Any) }) } itemsWithDivider( key = { it.name }, contentType = { ContentType.ITEM }, items = state.selectableOwnerships, ) { ownership -> - Ownership(ownership, state, onSelectedOwnership) + Ownership(ownership, state, { onSelectedOwnership(Constraint.Only(it)) }) } } itemWithDivider(key = Keys.PROVIDERS_TITLE, contentType = ContentType.HEADER) { @@ -163,14 +164,11 @@ private fun LazyItemScope.OwnershipHeader(expanded: Boolean, onToggleExpanded: ( } @Composable -private fun LazyItemScope.AnyOwnership( - state: RelayFilterUiState, - onSelectedOwnership: (ownership: Ownership?) -> Unit, -) { +private fun LazyItemScope.AnyOwnership(state: RelayFilterUiState, onSelectedOwnership: () -> Unit) { SelectableCell( title = stringResource(id = R.string.any), - isSelected = state.selectedOwnership == null, - onCellClicked = { onSelectedOwnership(null) }, + isSelected = state.selectedOwnership is Constraint.Any, + onCellClicked = { onSelectedOwnership() }, modifier = Modifier.animateItem(), ) } @@ -179,11 +177,11 @@ private fun LazyItemScope.AnyOwnership( private fun LazyItemScope.Ownership( ownership: Ownership, state: RelayFilterUiState, - onSelectedOwnership: (ownership: Ownership?) -> Unit, + onSelectedOwnership: (ownership: Ownership) -> Unit, ) { SelectableCell( title = stringResource(id = ownership.stringResource()), - isSelected = ownership == state.selectedOwnership, + isSelected = ownership == state.selectedOwnership.getOrNull(), onCellClicked = { onSelectedOwnership(ownership) }, modifier = Modifier.animateItem(), ) @@ -222,12 +220,18 @@ private fun LazyItemScope.Provider( ) { CheckboxCell( title = providerId.value, - checked = providerId in state.selectedProviders, + checked = providerId.isChecked(state.selectedProviders), onCheckedChange = { checked -> onSelectedProvider(checked, providerId) }, modifier = Modifier.animateItem(), ) } +private fun ProviderId.isChecked(constraint: Constraint>) = + when (constraint) { + Constraint.Any -> true + is Constraint.Only -> this in constraint.value + } + @Composable private fun TopBar(onBackClick: () -> Unit) { Row(Modifier.fillMaxWidth(), verticalAlignment = Alignment.CenterVertically) { diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/state/FilterConstrainExtensions.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/state/FilterConstrainExtensions.kt index 2ddf35fad52c..4106c9c45105 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/state/FilterConstrainExtensions.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/state/FilterConstrainExtensions.kt @@ -11,15 +11,9 @@ fun Ownership?.toOwnershipConstraint(): Constraint = else -> Constraint.Only(this) } -fun Constraint.toSelectedProviders(allProviders: List): List = - when (this) { - Constraint.Any -> allProviders - is Constraint.Only -> value.providers.toList() - } - fun List.toConstraintProviders(allProviders: List): Constraint = if (size == allProviders.size) { Constraint.Any } else { - Constraint.Only(Providers(toHashSet())) + Constraint.Only(this) } diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/state/RelayFilterUiState.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/state/RelayFilterUiState.kt index 959e9701b30d..db7d841d456d 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/state/RelayFilterUiState.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/state/RelayFilterUiState.kt @@ -1,35 +1,43 @@ package net.mullvad.mullvadvpn.compose.state +import net.mullvad.mullvadvpn.lib.model.Constraint import net.mullvad.mullvadvpn.lib.model.Ownership import net.mullvad.mullvadvpn.lib.model.ProviderId data class RelayFilterUiState( private val providerToOwnerships: Map> = emptyMap(), - val selectedOwnership: Ownership? = null, - val selectedProviders: List = emptyList(), + val selectedOwnership: Constraint = Constraint.Any, + val selectedProviders: Constraint> = Constraint.Any, ) { val allProviders: List = providerToOwnerships.keys.toList().sorted() val selectableOwnerships: List = - if (selectedProviders.isEmpty()) { - Ownership.entries - } else { - providerToOwnerships - .filterKeys { it in selectedProviders } - .values - .flatten() - .distinct() - } - .sorted() + when (selectedProviders) { + Constraint.Any -> Ownership.entries + is Constraint.Only -> + if (selectedProviders.value.isEmpty()) { + Ownership.entries + } else { + providerToOwnerships + .filterKeys { it in selectedProviders.value } + .values + .flatten() + .distinct() + } + }.sorted() val selectableProviders: List = - if (selectedOwnership != null) { - providerToOwnerships.filterValues { selectedOwnership in it }.keys.toList().sorted() - } else { - allProviders + when (selectedOwnership) { + Constraint.Any -> allProviders + is Constraint.Only -> + providerToOwnerships + .filterValues { selectedOwnership.value in it } + .keys + .toList() + .sorted() } - val isApplyButtonEnabled = selectedProviders.isNotEmpty() + val isApplyButtonEnabled = selectedProviders.getOrNull()?.isNotEmpty() != false - val isAllProvidersChecked = allProviders.size == selectedProviders.size + val isAllProvidersChecked = selectedProviders is Constraint.Any } diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/relaylist/RelayItemExtensions.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/relaylist/RelayItemExtensions.kt index 21d3294de408..d58da5bc9a58 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/relaylist/RelayItemExtensions.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/relaylist/RelayItemExtensions.kt @@ -46,7 +46,7 @@ private fun RelayItem.Location.hasProvider(providersConstraint: Constraint cities.any { it.hasProvider(providersConstraint) } is RelayItem.Location.City -> relays.any { it.hasProvider(providersConstraint) } - is RelayItem.Location.Relay -> provider in providersConstraint.value.providers + is RelayItem.Location.Relay -> provider in providersConstraint.value } } else { true diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/usecase/FilterChipUseCase.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/usecase/FilterChipUseCase.kt index 0b6f25adf860..e7cc4e3a9b0b 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/usecase/FilterChipUseCase.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/usecase/FilterChipUseCase.kt @@ -57,7 +57,7 @@ class FilterChipUseCase( when (selectedConstraintProviders) { is Constraint.Any -> null is Constraint.Only -> - selectedConstraintProviders.value.providers + selectedConstraintProviders.value .filter { providerId -> if (ownershipFilter == null) { true diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/FilterViewModel.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/FilterViewModel.kt index c05515126e31..8232ff332912 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/FilterViewModel.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/FilterViewModel.kt @@ -2,6 +2,7 @@ package net.mullvad.mullvadvpn.viewmodel import androidx.lifecycle.ViewModel import androidx.lifecycle.viewModelScope +import kotlin.collections.plus import kotlinx.coroutines.channels.Channel import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.SharingStarted @@ -10,13 +11,13 @@ import kotlinx.coroutines.flow.combine import kotlinx.coroutines.flow.first import kotlinx.coroutines.flow.receiveAsFlow import kotlinx.coroutines.flow.stateIn +import kotlinx.coroutines.flow.update import kotlinx.coroutines.launch import net.mullvad.mullvadvpn.compose.state.RelayFilterUiState -import net.mullvad.mullvadvpn.compose.state.toConstraintProviders -import net.mullvad.mullvadvpn.compose.state.toOwnershipConstraint -import net.mullvad.mullvadvpn.compose.state.toSelectedProviders +import net.mullvad.mullvadvpn.lib.model.Constraint import net.mullvad.mullvadvpn.lib.model.Ownership import net.mullvad.mullvadvpn.lib.model.ProviderId +import net.mullvad.mullvadvpn.lib.model.Providers import net.mullvad.mullvadvpn.repository.RelayListFilterRepository import net.mullvad.mullvadvpn.usecase.ProviderToOwnershipsUseCase @@ -27,24 +28,13 @@ class FilterViewModel( private val _uiSideEffect = Channel() val uiSideEffect = _uiSideEffect.receiveAsFlow() - private val selectedOwnership = MutableStateFlow(null) - private val selectedProviders = MutableStateFlow>(emptyList()) + private val selectedOwnership = MutableStateFlow>(Constraint.Any) + private val selectedProviders = MutableStateFlow>(Constraint.Any) init { viewModelScope.launch { - selectedProviders.value = - combine( - providerToOwnershipsUseCase(), - relayListFilterRepository.selectedProviders, - ) { providerToOwnerships, selectedConstraintProviders -> - selectedConstraintProviders.toSelectedProviders( - providerToOwnerships.keys.toList() - ) - } - .first() - - val ownershipConstraint = relayListFilterRepository.selectedOwnership.first() - selectedOwnership.value = ownershipConstraint.getOrNull() + selectedProviders.value = relayListFilterRepository.selectedProviders.first() + selectedOwnership.value = relayListFilterRepository.selectedOwnership.first() } } @@ -54,8 +44,8 @@ class FilterViewModel( private fun createState( providerToOwnerships: Map>, - selectedOwnership: Ownership?, - selectedProviders: List, + selectedOwnership: Constraint, + selectedProviders: Constraint, ): RelayFilterUiState = RelayFilterUiState( providerToOwnerships = providerToOwnerships, @@ -63,34 +53,61 @@ class FilterViewModel( selectedProviders = selectedProviders, ) - fun setSelectedOwnership(ownership: Ownership?) { + fun setSelectedOwnership(ownership: Constraint) { selectedOwnership.value = ownership } fun setSelectedProvider(checked: Boolean, provider: ProviderId) { - selectedProviders.value = + selectedProviders.update { if (checked) { - selectedProviders.value + provider + it.check(provider, uiState.value.allProviders) } else { - selectedProviders.value - provider + it.uncheck(provider, uiState.value.allProviders) + } + } + } + + private fun Constraint.check( + provider: ProviderId, + allProviders: Providers, + ): Constraint { + return when (this) { + is Constraint.Any -> Constraint.Any + is Constraint.Only -> { + val newProviderList = value + provider + if (allProviders.size == newProviderList.size) { + Constraint.Any + } else { + Constraint.Only(newProviderList) + } } + } + } + + private fun Constraint.uncheck( + provider: ProviderId, + allProviders: Providers, + ): Constraint { + return when (this) { + is Constraint.Any -> Constraint.Only(allProviders - provider) + is Constraint.Only -> Constraint.Only(value - provider) + } } fun setAllProviders(isChecked: Boolean) { viewModelScope.launch { selectedProviders.value = if (isChecked) { - providerToOwnershipsUseCase().first().keys.toList() + Constraint.Any } else { - emptyList() + Constraint.Only(emptyList()) } } } fun onApplyButtonClicked() { - val newSelectedOwnership = selectedOwnership.value.toOwnershipConstraint() - val newSelectedProviders = - selectedProviders.value.toConstraintProviders(uiState.value.allProviders) + val newSelectedOwnership = selectedOwnership.value + val newSelectedProviders = selectedProviders.value viewModelScope.launch { relayListFilterRepository.updateSelectedOwnershipAndProviderFilter( diff --git a/android/app/src/test/kotlin/net/mullvad/mullvadvpn/repository/RelayListFilterRepositoryTest.kt b/android/app/src/test/kotlin/net/mullvad/mullvadvpn/repository/RelayListFilterRepositoryTest.kt index 6ec1ae1feab2..3482f718bcac 100644 --- a/android/app/src/test/kotlin/net/mullvad/mullvadvpn/repository/RelayListFilterRepositoryTest.kt +++ b/android/app/src/test/kotlin/net/mullvad/mullvadvpn/repository/RelayListFilterRepositoryTest.kt @@ -59,8 +59,7 @@ class RelayListFilterRepositoryTest { fun `when settings is updated selected providers should update`() = runTest { // Arrange val mockSettings: Settings = mockk() - val selectedProviders: Constraint = - Constraint.Only(Providers(setOf(ProviderId("Prove")))) + val selectedProviders: Constraint = Constraint.Only(listOf(ProviderId("Prove"))) every { mockSettings.relaySettings.relayConstraints.providers } returns selectedProviders // Act, Assert @@ -147,7 +146,7 @@ class RelayListFilterRepositoryTest { @Test fun `when successfully updating selected providers should return successful`() = runTest { // Arrange - val providers = Constraint.Only(Providers(setOf(ProviderId("Mopp")))) + val providers = Constraint.Only(listOf(ProviderId("Mopp"))) coEvery { mockManagementService.setProviders(providers) } returns Unit.right() // Act @@ -162,7 +161,7 @@ class RelayListFilterRepositoryTest { fun `when failing to update selected providers should return SetWireguardConstraintsError`() = runTest { // Arrange - val providers = Constraint.Only(Providers(setOf(ProviderId("Mopp")))) + val providers = Constraint.Only(listOf(ProviderId("Mopp"))) val error = SetWireguardConstraintsError.Unknown(mockk()) coEvery { mockManagementService.setProviders(providers) } returns error.left() diff --git a/android/app/src/test/kotlin/net/mullvad/mullvadvpn/usecase/FilterChipUseCaseTest.kt b/android/app/src/test/kotlin/net/mullvad/mullvadvpn/usecase/FilterChipUseCaseTest.kt index c1d056b77792..4328ca407735 100644 --- a/android/app/src/test/kotlin/net/mullvad/mullvadvpn/usecase/FilterChipUseCaseTest.kt +++ b/android/app/src/test/kotlin/net/mullvad/mullvadvpn/usecase/FilterChipUseCaseTest.kt @@ -71,7 +71,7 @@ class FilterChipUseCaseTest { @Test fun `when provider filter is applied should return correct number of providers`() = runTest { // Arrange - val expectedProviders = Providers(providers = setOf(ProviderId("1"), ProviderId("2"))) + val expectedProviders = listOf(ProviderId("1"), ProviderId("2")) selectedProviders.value = Constraint.Only(expectedProviders) providerToOwnerships.value = mapOf( @@ -88,7 +88,7 @@ class FilterChipUseCaseTest { fun `when provider and ownership filter is applied should return correct filter chips`() = runTest { // Arrange - val expectedProviders = Providers(providers = setOf(ProviderId("1"))) + val expectedProviders = listOf(ProviderId("1")) val expectedOwnership = Ownership.MullvadOwned selectedProviders.value = Constraint.Only(expectedProviders) selectedOwnership.value = Constraint.Only(expectedOwnership) diff --git a/android/app/src/test/kotlin/net/mullvad/mullvadvpn/viewmodel/FilterViewModelTest.kt b/android/app/src/test/kotlin/net/mullvad/mullvadvpn/viewmodel/FilterViewModelTest.kt index 49b9b4baffa8..98fe6becfffe 100644 --- a/android/app/src/test/kotlin/net/mullvad/mullvadvpn/viewmodel/FilterViewModelTest.kt +++ b/android/app/src/test/kotlin/net/mullvad/mullvadvpn/viewmodel/FilterViewModelTest.kt @@ -20,7 +20,6 @@ import net.mullvad.mullvadvpn.lib.common.test.assertLists import net.mullvad.mullvadvpn.lib.model.Constraint import net.mullvad.mullvadvpn.lib.model.Ownership import net.mullvad.mullvadvpn.lib.model.ProviderId -import net.mullvad.mullvadvpn.lib.model.Providers import net.mullvad.mullvadvpn.repository.RelayListFilterRepository import net.mullvad.mullvadvpn.usecase.ProviderToOwnershipsUseCase import org.junit.jupiter.api.AfterEach @@ -63,7 +62,7 @@ class FilterViewModelTest { every { mockRelayListFilterRepository.selectedOwnership } returns selectedOwnership every { mockProvidersOwnershipUseCase() } returns flowOf(dummyListOfAllProviders) every { mockRelayListFilterRepository.selectedProviders } returns - MutableStateFlow(Constraint.Only(Providers(mockSelectedProviders.toHashSet()))) + MutableStateFlow(Constraint.Only(mockSelectedProviders)) viewModel = FilterViewModel( providerToOwnershipsUseCase = mockProvidersOwnershipUseCase, @@ -84,9 +83,9 @@ class FilterViewModelTest { val mockOwnership = Ownership.Rented // Assert viewModel.uiState.test { - assertEquals(awaitItem().selectedOwnership, Ownership.MullvadOwned) - viewModel.setSelectedOwnership(mockOwnership) - assertEquals(mockOwnership, awaitItem().selectedOwnership) + assertEquals(Constraint.Only(Ownership.MullvadOwned), awaitItem().selectedOwnership) + viewModel.setSelectedOwnership(Constraint.Only(mockOwnership)) + assertEquals(Constraint.Only(mockOwnership), awaitItem().selectedOwnership) } } @@ -97,11 +96,11 @@ class FilterViewModelTest { val mockSelectedProvidersList = ProviderId("ptisp") // Assert viewModel.uiState.test { - assertLists(awaitItem().selectedProviders, mockSelectedProviders) + assertLists(mockSelectedProviders, awaitItem().selectedProviders.getOrNull()!!) viewModel.setSelectedProvider(true, mockSelectedProvidersList) assertLists( listOf(mockSelectedProvidersList) + mockSelectedProviders, - awaitItem().selectedProviders, + awaitItem().selectedProviders.getOrNull()!!, ) } } @@ -109,14 +108,12 @@ class FilterViewModelTest { @Test fun `setAllProvider with true should emit uiState with selectedProviders includes all providers`() = runTest { - // Arrange - val mockProvidersList = dummyListOfAllProviders.keys.toList() // Act viewModel.setAllProviders(true) // Assert viewModel.uiState.test { val state = awaitItem() - assertEquals(mockProvidersList, state.selectedProviders) + assertEquals(Constraint.Any, state.selectedProviders) } } diff --git a/android/lib/daemon-grpc/src/main/kotlin/net/mullvad/mullvadvpn/lib/daemon/grpc/mapper/FromDomain.kt b/android/lib/daemon-grpc/src/main/kotlin/net/mullvad/mullvadvpn/lib/daemon/grpc/mapper/FromDomain.kt index f62124a1712b..e4d3e84a4ff1 100644 --- a/android/lib/daemon-grpc/src/main/kotlin/net/mullvad/mullvadvpn/lib/daemon/grpc/mapper/FromDomain.kt +++ b/android/lib/daemon-grpc/src/main/kotlin/net/mullvad/mullvadvpn/lib/daemon/grpc/mapper/FromDomain.kt @@ -46,7 +46,7 @@ internal fun Constraint.fromDomain(): ManagementInterface.LocationC internal fun Constraint.fromDomain(): List = when (this) { is Constraint.Any -> emptyList() - is Constraint.Only -> value.providers.map { it.value } + is Constraint.Only -> value.map { it.value } } internal fun DnsOptions.fromDomain(): ManagementInterface.DnsOptions = diff --git a/android/lib/daemon-grpc/src/main/kotlin/net/mullvad/mullvadvpn/lib/daemon/grpc/mapper/ToDomain.kt b/android/lib/daemon-grpc/src/main/kotlin/net/mullvad/mullvadvpn/lib/daemon/grpc/mapper/ToDomain.kt index 91399b48b614..4c2e55455cfb 100644 --- a/android/lib/daemon-grpc/src/main/kotlin/net/mullvad/mullvadvpn/lib/daemon/grpc/mapper/ToDomain.kt +++ b/android/lib/daemon-grpc/src/main/kotlin/net/mullvad/mullvadvpn/lib/daemon/grpc/mapper/ToDomain.kt @@ -365,7 +365,7 @@ internal fun ManagementInterface.GeographicLocationConstraint.toDomain(): GeoLoc } internal fun List.toDomain(): Constraint = - if (isEmpty()) Constraint.Any else Constraint.Only(Providers(map { ProviderId(it) }.toSet())) + if (isEmpty()) Constraint.Any else Constraint.Only(map { ProviderId(it) }) internal fun ManagementInterface.WireguardConstraints.toDomain(): WireguardConstraints = WireguardConstraints( diff --git a/android/lib/model/src/main/kotlin/net/mullvad/mullvadvpn/lib/model/Providers.kt b/android/lib/model/src/main/kotlin/net/mullvad/mullvadvpn/lib/model/Providers.kt index 73cf9facdbf7..17a75640e7ff 100644 --- a/android/lib/model/src/main/kotlin/net/mullvad/mullvadvpn/lib/model/Providers.kt +++ b/android/lib/model/src/main/kotlin/net/mullvad/mullvadvpn/lib/model/Providers.kt @@ -1,8 +1,3 @@ package net.mullvad.mullvadvpn.lib.model -import arrow.optics.optics - -@optics -data class Providers(val providers: Set) { - companion object -} +typealias Providers = List