summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/dummy/DummyDataSource.kt68
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/dummy/Extensions.kt29
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFeature.kt186
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFragment.kt110
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardViewModel.kt8
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFeature.kt110
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFragment.kt80
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyViewModel.kt40
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/location/FakeLocationFeature.kt152
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/location/FakeLocationFragment.kt176
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/features/location/FakeLocationViewModel.kt (renamed from app/src/main/java/foundation/e/privacycentralapp/features/location/FakeMyLocationFragment.kt)30
-rw-r--r--app/src/main/java/foundation/e/privacycentralapp/main/MainActivity.kt4
-rw-r--r--app/src/main/res/drawable/ic_my_location.xml22
-rw-r--r--app/src/main/res/layout/fragment_dashboard.xml412
-rw-r--r--app/src/main/res/layout/fragment_fake_location.xml1
-rw-r--r--app/src/main/res/layout/fragment_internet_activity_policy.xml4
-rw-r--r--app/src/main/res/values/strings.xml8
-rw-r--r--build.gradle1
-rw-r--r--flow-mvi/src/main/java/foundation/e/flowmvi/feature/BaseFeature.kt20
19 files changed, 1174 insertions, 287 deletions
diff --git a/app/src/main/java/foundation/e/privacycentralapp/dummy/DummyDataSource.kt b/app/src/main/java/foundation/e/privacycentralapp/dummy/DummyDataSource.kt
index 65d072a..3f2dc1e 100644
--- a/app/src/main/java/foundation/e/privacycentralapp/dummy/DummyDataSource.kt
+++ b/app/src/main/java/foundation/e/privacycentralapp/dummy/DummyDataSource.kt
@@ -18,17 +18,21 @@
package foundation.e.privacycentralapp.dummy
import foundation.e.privacycentralapp.R
+import kotlinx.coroutines.flow.MutableStateFlow
+import kotlinx.coroutines.flow.asStateFlow
+import kotlin.random.Random
// ======================================================//
//
-// ============ ==== ==== ============
-// ============ ===== ===== ==== ====
-// ==== ====== ====== ==== ====
-// ==== ======= ======= ============
-// ==== ================ ====
-// ==== ==== ====== ==== ====
-// ============ ==== ==== ==== ====
-// ============ ==== == ==== ====
+// ================ ==== ==== ===============
+// ================ ====== ====== ================
+// ==== ======== ======== ==== ====
+// ==== ========= ========= ==== ====
+// ==== ==================== ================
+// ==== ==== ======== ==== ===============
+// ==== ==== ==== ==== ====
+// ================ ==== == ==== ====
+// ================ ==== ==== ====
//
// ======================================================//
@@ -47,7 +51,30 @@ data class Permission(
val packagesAllowed: List<String> = emptyList()
)
+enum class LocationMode {
+ REAL_LOCATION, RANDOM_LOCATION, CUSTOM_LOCATION
+}
+
+enum class InternetPrivacyMode {
+ REAL_IP, HIDE_IP
+}
+
+data class Location(val mode: LocationMode, val latitude: Double, val longitude: Double)
+
object DummyDataSource {
+ private val _appsUsingLocationPerm = MutableStateFlow<List<String>>(emptyList())
+ val appsUsingLocationPerm = _appsUsingLocationPerm.asStateFlow()
+
+ const val trackersCount = 77
+ private val _activeTrackersCount = MutableStateFlow(10)
+ val activeTrackersCount = _activeTrackersCount.asStateFlow()
+
+ private val _location = MutableStateFlow(Location(LocationMode.REAL_LOCATION, 0.0, 0.0))
+ val location = _location.asStateFlow()
+
+ private val _internetActivityMode = MutableStateFlow(InternetPrivacyMode.REAL_IP)
+ val internetActivityMode = _internetActivityMode.asStateFlow()
+
val permissions = arrayOf("Body Sensor", "Calendar", "Call Logs", "Location")
val icons = arrayOf(
R.drawable.ic_body_monitor,
@@ -138,4 +165,29 @@ object DummyDataSource {
fun getPermission(permissionId: Int): Permission {
return populatedPermission.get(permissionId)
}
+
+ fun setLocationMode(locationMode: LocationMode, location: Location? = null): Boolean {
+ when (locationMode) {
+ LocationMode.REAL_LOCATION ->
+ _location.value =
+ Location(LocationMode.REAL_LOCATION, 24.39, 71.80)
+ LocationMode.RANDOM_LOCATION -> _location.value = randomLocation()
+ LocationMode.CUSTOM_LOCATION -> {
+ requireNotNull(location) { "Custom location should be null" }
+ _location.value = location.copy(mode = LocationMode.CUSTOM_LOCATION)
+ }
+ }
+ return true
+ }
+
+ private fun randomLocation(): Location = Location(
+ LocationMode.RANDOM_LOCATION,
+ Random.nextDouble(-90.0, 90.0),
+ Random.nextDouble(-180.0, 180.0)
+ )
+
+ fun setInternetPrivacyMode(mode: InternetPrivacyMode): Boolean {
+ _internetActivityMode.value = mode
+ return true
+ }
}
diff --git a/app/src/main/java/foundation/e/privacycentralapp/dummy/Extensions.kt b/app/src/main/java/foundation/e/privacycentralapp/dummy/Extensions.kt
new file mode 100644
index 0000000..c872012
--- /dev/null
+++ b/app/src/main/java/foundation/e/privacycentralapp/dummy/Extensions.kt
@@ -0,0 +1,29 @@
+/*
+ * Copyright (C) 2021 E FOUNDATION
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package foundation.e.privacycentralapp.dummy
+
+fun LocationMode.mapToString(): String = when (this) {
+ LocationMode.REAL_LOCATION -> "Real location mode"
+ LocationMode.RANDOM_LOCATION -> "Random location mode"
+ LocationMode.CUSTOM_LOCATION -> "Fake location mode"
+}
+
+fun InternetPrivacyMode.mapToString(): String = when (this) {
+ InternetPrivacyMode.REAL_IP -> "I'm exposing my real IP address"
+ InternetPrivacyMode.HIDE_IP -> "I'm anonymous on the internet"
+}
diff --git a/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFeature.kt b/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFeature.kt
index ecadea1..dd4f0ff 100644
--- a/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFeature.kt
+++ b/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFeature.kt
@@ -17,59 +17,189 @@
package foundation.e.privacycentralapp.features.dashboard
+import android.util.Log
import foundation.e.flowmvi.Actor
import foundation.e.flowmvi.Reducer
+import foundation.e.flowmvi.SingleEventProducer
import foundation.e.flowmvi.feature.BaseFeature
+import foundation.e.privacycentralapp.dummy.DummyDataSource
+import foundation.e.privacycentralapp.dummy.InternetPrivacyMode
+import foundation.e.privacycentralapp.dummy.LocationMode
import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.flow.flow
import kotlinx.coroutines.flow.flowOf
+import kotlinx.coroutines.flow.map
+import kotlinx.coroutines.flow.merge
// Define a state machine for Dashboard Feature
-object DashboardFeature {
+class DashboardFeature(
+ initialState: State,
+ coroutineScope: CoroutineScope,
+ reducer: Reducer<State, Effect>,
+ actor: Actor<State, Action, Effect>,
+ singleEventProducer: SingleEventProducer<State, Action, Effect, SingleEvent>
+) : BaseFeature<DashboardFeature.State,
+ DashboardFeature.Action,
+ DashboardFeature.Effect,
+ DashboardFeature.SingleEvent>(
+ initialState, actor, reducer, coroutineScope, { message -> Log.d("DashboardFeature", message) },
+ singleEventProducer
+) {
sealed class State {
- object DashboardState : State()
+ object InitialState : State()
+ object LoadingDashboardState : State()
+ data class DashboardState(
+ val trackersCount: Int,
+ val activeTrackersCount: Int,
+ val totalApps: Int,
+ val permissionCount: Int,
+ val appsUsingLocationPerm: Int,
+ val locationMode: LocationMode,
+ val internetPrivacyMode: InternetPrivacyMode
+ ) : State()
+
object QuickProtectionState : State()
}
sealed class SingleEvent {
object NavigateToQuickProtectionSingleEvent : SingleEvent()
object NavigateToTrackersSingleEvent : SingleEvent()
- object NavigateToInternetActivityPolicySingleEvent : SingleEvent()
+ object NavigateToInternetActivityPrivacySingleEvent : SingleEvent()
object NavigateToLocationSingleEvent : SingleEvent()
- object NavigateToPermissionManagementSingleEvent : SingleEvent()
+ object NavigateToPermissionsSingleEvent : SingleEvent()
}
sealed class Action {
object ShowQuickPrivacyProtectionInfoAction : Action()
+ object ObserveDashboardAction : Action()
object ShowDashboardAction : Action()
+ object ShowFakeMyLocationAction : Action()
+ object ShowInternetActivityPrivacyAction : Action()
+ object ShowAppsPermissions : Action()
}
sealed class Effect {
object OpenQuickPrivacyProtectionEffect : Effect()
- object OpenDashboardEffect : Effect()
- }
-}
+ data class OpenDashboardEffect(
+ val trackersCount: Int,
+ val activeTrackersCount: Int,
+ val totalApps: Int,
+ val permissionCount: Int,
+ val appsUsingLocationPerm: Int,
+ val locationMode: LocationMode,
+ val internetPrivacyMode: InternetPrivacyMode
+ ) : Effect()
-private val reducer: Reducer<DashboardFeature.State, DashboardFeature.Effect> = { _, effect ->
- when (effect) {
- DashboardFeature.Effect.OpenQuickPrivacyProtectionEffect -> DashboardFeature.State.QuickProtectionState
- DashboardFeature.Effect.OpenDashboardEffect -> DashboardFeature.State.DashboardState
+ object LoadingDashboardEffect : Effect()
+ data class UpdateActiveTrackersCountEffect(val count: Int) : Effect()
+ data class UpdateLocationModeEffect(val mode: LocationMode) : Effect()
+ data class UpdateInternetActivityModeEffect(val mode: InternetPrivacyMode) : Effect()
+ data class UpdateAppsUsingLocationPermEffect(val apps: Int) : Effect()
+ object OpenFakeMyLocationEffect : Effect()
+ object OpenInternetActivityPrivacyEffect : Effect()
+ object OpenAppsPermissionsEffect : Effect()
}
-}
-private val actor: Actor<DashboardFeature.State, DashboardFeature.Action, DashboardFeature.Effect> =
- { _, action ->
- when (action) {
- DashboardFeature.Action.ShowQuickPrivacyProtectionInfoAction -> flowOf(DashboardFeature.Effect.OpenQuickPrivacyProtectionEffect)
- DashboardFeature.Action.ShowDashboardAction -> flowOf(DashboardFeature.Effect.OpenDashboardEffect)
- }
- }
+ companion object {
+ fun create(initialState: State, coroutineScope: CoroutineScope): DashboardFeature =
+ DashboardFeature(
+ initialState,
+ coroutineScope,
+ reducer = { state, effect ->
+ when (effect) {
+ Effect.OpenQuickPrivacyProtectionEffect -> State.QuickProtectionState
+ is Effect.OpenDashboardEffect -> State.DashboardState(
+ effect.trackersCount,
+ effect.activeTrackersCount,
+ effect.totalApps,
+ effect.permissionCount,
+ effect.appsUsingLocationPerm,
+ effect.locationMode,
+ effect.internetPrivacyMode
+ )
+ Effect.LoadingDashboardEffect -> {
+ if (state is State.InitialState) {
+ State.LoadingDashboardState
+ } else state
+ }
+ is Effect.UpdateActiveTrackersCountEffect -> {
+ if (state is State.DashboardState) {
+ state.copy(activeTrackersCount = effect.count)
+ } else state
+ }
+ is Effect.UpdateInternetActivityModeEffect -> {
+ if (state is State.DashboardState) {
+ state.copy(internetPrivacyMode = effect.mode)
+ } else state
+ }
+ is Effect.UpdateLocationModeEffect -> {
+ if (state is State.DashboardState) {
+ state.copy(locationMode = effect.mode)
+ } else state
+ }
+ is Effect.UpdateAppsUsingLocationPermEffect -> if (state is State.DashboardState) {
+ state.copy(appsUsingLocationPerm = effect.apps)
+ } else state
-fun homeFeature(
- initialState: DashboardFeature.State = DashboardFeature.State.DashboardState,
- coroutineScope: CoroutineScope
-) = BaseFeature<DashboardFeature.State, DashboardFeature.Action, DashboardFeature.Effect, DashboardFeature.SingleEvent>(
- initialState,
- actor,
- reducer,
- coroutineScope
-)
+ Effect.OpenFakeMyLocationEffect -> state
+ Effect.OpenAppsPermissionsEffect -> state
+ Effect.OpenInternetActivityPrivacyEffect -> state
+ }
+ },
+ actor = { _: State, action: Action ->
+ Log.d("Feature", "action: $action")
+ when (action) {
+ Action.ObserveDashboardAction -> merge(
+ DummyDataSource.activeTrackersCount.map {
+ Effect.UpdateActiveTrackersCountEffect(it)
+ },
+ DummyDataSource.appsUsingLocationPerm.map {
+ Effect.UpdateAppsUsingLocationPermEffect(it.size)
+ },
+ DummyDataSource.location.map {
+ Effect.UpdateLocationModeEffect(it.mode)
+ },
+ DummyDataSource.internetActivityMode.map {
+ Effect.UpdateInternetActivityModeEffect(it)
+ }
+ )
+ Action.ShowQuickPrivacyProtectionInfoAction -> flowOf(
+ Effect.OpenQuickPrivacyProtectionEffect
+ )
+ Action.ShowDashboardAction -> flow {
+ emit(Effect.LoadingDashboardEffect)
+ kotlinx.coroutines.delay(2000)
+ emit(
+ Effect.OpenDashboardEffect(
+ DummyDataSource.trackersCount,
+ DummyDataSource.activeTrackersCount.value,
+ DummyDataSource.packages.size,
+ DummyDataSource.permissions.size,
+ DummyDataSource.appsUsingLocationPerm.value.size,
+ DummyDataSource.location.value.mode,
+ DummyDataSource.internetActivityMode.value
+ )
+ )
+ }
+ Action.ShowFakeMyLocationAction -> flowOf(Effect.OpenFakeMyLocationEffect)
+ Action.ShowAppsPermissions -> flowOf(Effect.OpenAppsPermissionsEffect)
+ Action.ShowInternetActivityPrivacyAction -> flowOf(
+ Effect.OpenInternetActivityPrivacyEffect
+ )
+ }
+ },
+ singleEventProducer = { state, _, effect ->
+ Log.d("DashboardFeature", "$state, $effect")
+ if (state is State.DashboardState && effect is Effect.OpenFakeMyLocationEffect)
+ SingleEvent.NavigateToLocationSingleEvent
+ else if (state is State.QuickProtectionState && effect is Effect.OpenQuickPrivacyProtectionEffect)
+ SingleEvent.NavigateToQuickProtectionSingleEvent
+ else if (state is State.DashboardState && effect is Effect.OpenInternetActivityPrivacyEffect)
+ SingleEvent.NavigateToInternetActivityPrivacySingleEvent
+ else if (state is State.DashboardState && effect is Effect.OpenAppsPermissionsEffect)
+ SingleEvent.NavigateToPermissionsSingleEvent
+ else null
+ }
+ )
+ }
+}
diff --git a/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFragment.kt b/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFragment.kt
index d6a91b8..b9371be 100644
--- a/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFragment.kt
+++ b/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardFragment.kt
@@ -23,8 +23,11 @@ import android.text.Spannable
import android.text.SpannableString
import android.text.style.ForegroundColorSpan
import android.view.View
+import android.widget.ProgressBar
+import android.widget.RelativeLayout
import android.widget.TextView
import android.widget.Toolbar
+import androidx.core.widget.NestedScrollView
import androidx.fragment.app.Fragment
import androidx.fragment.app.activityViewModels
import androidx.fragment.app.add
@@ -32,7 +35,12 @@ import androidx.fragment.app.commit
import androidx.lifecycle.lifecycleScope
import foundation.e.flowmvi.MVIView
import foundation.e.privacycentralapp.R
+import foundation.e.privacycentralapp.dummy.mapToString
+import foundation.e.privacycentralapp.features.internetprivacy.InternetPrivacyFragment
+import foundation.e.privacycentralapp.features.location.FakeLocationFragment
+import foundation.e.privacycentralapp.features.permissions.PermissionsFragment
import kotlinx.coroutines.flow.Flow
+import kotlinx.coroutines.flow.collect
class DashboardFragment :
Fragment(R.layout.fragment_dashboard),
@@ -43,7 +51,40 @@ class DashboardFragment :
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
lifecycleScope.launchWhenStarted {
- viewModel.homeFeature.takeView(this, this@DashboardFragment)
+ viewModel.dashboardFeature.takeView(this, this@DashboardFragment)
+ }
+ lifecycleScope.launchWhenStarted {
+ viewModel.dashboardFeature.singleEvents.collect { event ->
+ if (event is DashboardFeature.SingleEvent.NavigateToLocationSingleEvent) {
+ requireActivity().supportFragmentManager.commit {
+ add<FakeLocationFragment>(R.id.container)
+ setReorderingAllowed(true)
+ addToBackStack("dashboard")
+ }
+ } else if (event is DashboardFeature.SingleEvent.NavigateToQuickProtectionSingleEvent) {
+ requireActivity().supportFragmentManager.commit {
+ add<QuickProtectionFragment>(R.id.container)
+ setReorderingAllowed(true)
+ addToBackStack("dashboard")
+ }
+ } else if (event is DashboardFeature.SingleEvent.NavigateToInternetActivityPrivacySingleEvent) {
+ requireActivity().supportFragmentManager.commit {
+ add<InternetPrivacyFragment>(R.id.container)
+ setReorderingAllowed(true)
+ addToBackStack("dashboard")
+ }
+ } else if (event is DashboardFeature.SingleEvent.NavigateToPermissionsSingleEvent) {
+ requireActivity().supportFragmentManager.commit {
+ add<PermissionsFragment>(R.id.container)
+ setReorderingAllowed(true)
+ addToBackStack("dashboard")
+ }
+ }
+ }
+ }
+ lifecycleScope.launchWhenStarted {
+ viewModel.submitAction(DashboardFeature.Action.ShowDashboardAction)
+ viewModel.submitAction(DashboardFeature.Action.ObserveDashboardAction)
}
}
@@ -56,6 +97,15 @@ class DashboardFragment :
it.findViewById<TextView>(R.id.tap_to_enable_quick_protection).setOnClickListener {
viewModel.submitAction(DashboardFeature.Action.ShowQuickPrivacyProtectionInfoAction)
}
+ it.findViewById<RelativeLayout>(R.id.my_location).setOnClickListener {
+ viewModel.submitAction(DashboardFeature.Action.ShowFakeMyLocationAction)
+ }
+ it.findViewById<RelativeLayout>(R.id.internet_activity_privacy).setOnClickListener {
+ viewModel.submitAction(DashboardFeature.Action.ShowInternetActivityPrivacyAction)
+ }
+ it.findViewById<RelativeLayout>(R.id.apps_permissions).setOnClickListener {
+ viewModel.submitAction(DashboardFeature.Action.ShowAppsPermissions)
+ }
}
}
@@ -78,15 +128,59 @@ class DashboardFragment :
override fun render(state: DashboardFeature.State) {
when (state) {
- is DashboardFeature.State.QuickProtectionState -> {
- requireActivity().supportFragmentManager.commit {
- add<QuickProtectionFragment>(R.id.container)
- setReorderingAllowed(true)
- addToBackStack("dashboard")
+ is DashboardFeature.State.InitialState, is DashboardFeature.State.LoadingDashboardState -> {
+ view?.let {
+ it.findViewById<ProgressBar>(R.id.loadingSpinner).visibility = View.VISIBLE
+ it.findViewById<NestedScrollView>(R.id.scrollContainer).visibility = View.GONE
}
}
- else -> {
- // TODO: any remaining state must either be handled or needs to be passed down to the UI.
+ is DashboardFeature.State.DashboardState -> {
+ view?.let { view ->
+ view.findViewById<ProgressBar>(R.id.loadingSpinner).visibility = View.GONE
+ view.findViewById<NestedScrollView>(R.id.scrollContainer).visibility =
+ View.VISIBLE
+ view.findViewById<TextView>(R.id.am_i_tracked_subtitle).text = getString(
+ R.string.am_i_tracked_subtitle,
+ state.trackersCount,
+ state.activeTrackersCount
+ )
+ view.findViewById<TextView>(R.id.apps_permissions_subtitle).text = getString(
+ R.string.apps_permissions_subtitle,
+ state.totalApps,
+ state.permissionCount
+ )
+ view.findViewById<TextView>(R.id.my_location_subtitle).let {
+ it.text = getString(
+ R.string.my_location_subtitle,
+ state.appsUsingLocationPerm,
+ )
+ it.append(
+ SpannableString(state.locationMode.mapToString())
+ .also {
+ it.setSpan(
+ ForegroundColorSpan(Color.parseColor("#007fff")),
+ 0,
+ it.length,
+ Spannable.SPAN_EXCLUSIVE_EXCLUSIVE
+ )
+ }
+ )
+ }
+ view.findViewById<TextView>(R.id.internet_activity_privacy_subtitle).let {
+ it.text = getString(R.string.internet_activity_privacy_subtitle)
+ it.append(
+ SpannableString(state.internetPrivacyMode.mapToString())
+ .also {
+ it.setSpan(
+ ForegroundColorSpan(Color.parseColor("#007fff")),
+ 0,
+ it.length,
+ Spannable.SPAN_EXCLUSIVE_EXCLUSIVE
+ )
+ }
+ )
+ }
+ }
}
}
}
diff --git a/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardViewModel.kt b/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardViewModel.kt
index 12696d5..9428f41 100644
--- a/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardViewModel.kt
+++ b/app/src/main/java/foundation/e/privacycentralapp/features/dashboard/DashboardViewModel.kt
@@ -17,9 +17,9 @@
package foundation.e.privacycentralapp.features.dashboard
+import android.util.Log
import androidx.lifecycle.ViewModel
import androidx.lifecycle.viewModelScope
-import foundation.e.flowmvi.feature.BaseFeature
import kotlinx.coroutines.flow.MutableSharedFlow
import kotlinx.coroutines.flow.asSharedFlow
import kotlinx.coroutines.launch
@@ -29,12 +29,12 @@ class DashboardViewModel : ViewModel() {
private val _actions = MutableSharedFlow<DashboardFeature.Action>()
val actions = _actions.asSharedFlow()
- val homeFeature: BaseFeature<DashboardFeature.State, DashboardFeature.Action,
- DashboardFeature.Effect, DashboardFeature.SingleEvent> by lazy {
- homeFeature(coroutineScope = viewModelScope)
+ val dashboardFeature: DashboardFeature by lazy {
+ DashboardFeature.create(DashboardFeature.State.InitialState, coroutineScope = viewModelScope)
}
fun submitAction(action: DashboardFeature.Action) {
+ Log.d("DashboardViewModel", "submitAction() called with: action = $action")
viewModelScope.launch {
_actions.emit(action)
}
diff --git a/app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFeature.kt b/app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFeature.kt
new file mode 100644
index 0000000..66e4add
--- /dev/null
+++ b/app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFeature.kt
@@ -0,0 +1,110 @@
+/*
+ * Copyright (C) 2021 E FOUNDATION
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package foundation.e.privacycentralapp.features.internetprivacy
+
+import android.util.Log
+import foundation.e.flowmvi.Actor
+import foundation.e.flowmvi.Reducer
+import foundation.e.flowmvi.SingleEventProducer
+import foundation.e.flowmvi.feature.BaseFeature
+import foundation.e.privacycentralapp.dummy.DummyDataSource
+import foundation.e.privacycentralapp.dummy.InternetPrivacyMode
+import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.flow.flow
+import kotlinx.coroutines.flow.flowOf
+
+// Define a state machine for Fake location feature
+class InternetPrivacyFeature(
+ initialState: State,
+ coroutineScope: CoroutineScope,
+ reducer: Reducer<State, Effect>,
+ actor: Actor<State, Action, Effect>,
+ singleEventProducer: SingleEventProducer<State, Action, Effect, SingleEvent>
+) : BaseFeature<InternetPrivacyFeature.State, InternetPrivacyFeature.Action, InternetPrivacyFeature.Effect, InternetPrivacyFeature.SingleEvent>(
+ initialState,
+ actor,
+ reducer,
+ coroutineScope,
+ { message -> Log.d("FakeLocationFeature", message) },
+ singleEventProducer
+) {
+ data class State(val mode: InternetPrivacyMode)
+
+ sealed class SingleEvent {
+ object RealIPSelectedEvent : SingleEvent()
+ object HiddenIPSelectedEvent : SingleEvent()
+ data class ErrorEvent(val error: String) : SingleEvent()
+ }
+
+ sealed class Action {
+ object LoadInternetModeAction : Action()
+ object UseRealIPAction : Action()
+ object UseHiddenIPAction : Action()
+ }
+
+ sealed class Effect {
+ data class ModeUpdatedEffect(val mode: InternetPrivacyMode) : Effect()
+ data class ErrorEffect(val message: String) : Effect()
+ }
+
+ companion object {
+ fun create(
+ initialState: State = State(InternetPrivacyMode.REAL_IP),
+ coroutineScope: CoroutineScope
+ ) = InternetPrivacyFeature(
+ initialState, coroutineScope,
+ reducer = { state, effect ->
+ when (effect) {
+ is Effect.ModeUpdatedEffect -> state.copy(mode = effect.mode)
+ is Effect.ErrorEffect -> state
+ }
+ },
+ actor = { _, action ->
+ when (action) {
+ Action.LoadInternetModeAction -> flowOf(Effect.ModeUpdatedEffect(DummyDataSource.internetActivityMode.value))
+ Action.UseHiddenIPAction, Action.UseRealIPAction -> flow {
+ val success =
+ DummyDataSource.setInternetPrivacyMode(if (action is Action.UseHiddenIPAction) InternetPrivacyMode.HIDE_IP else InternetPrivacyMode.REAL_IP)
+ emit(
+ if (success) Effect.ModeUpdatedEffect(DummyDataSource.internetActivityMode.value) else Effect.ErrorEffect(
+ "Couldn't update internet mode"
+ )
+ )
+ }
+ }
+ },
+ singleEventProducer = { _, action, effect ->
+ when (action) {
+ Action.UseRealIPAction, Action.UseHiddenIPAction -> when (effect) {
+ is Effect.ModeUpdatedEffect -> {
+ if (effect.mode == InternetPrivacyMode.REAL_IP) {
+ SingleEvent.RealIPSelectedEvent
+ } else {
+ SingleEvent.HiddenIPSelectedEvent
+ }
+ }
+ is Effect.ErrorEffect -> {
+ SingleEvent.ErrorEvent(effect.message)
+ }
+ }
+ else -> null
+ }
+ }
+ )
+ }
+}
diff --git a/app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFragment.kt b/app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFragment.kt
index ddba807..a8c1671 100644
--- a/app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFragment.kt
+++ b/app/src/main/java/foundation/e/privacycentralapp/features/internetprivacy/InternetPrivacyFragment.kt
@@ -19,15 +19,53 @@ package foundation.e.privacycentralapp.features.internetprivacy
import android.os.Bundle
import android.view.View
+import android.widget.RadioButton
+import android.widget.Toast
import android.widget.Toolbar
import androidx.fragment.app.Fragment
+import androidx.fragment.app.viewModels
+import androidx.lifecycle.lifecycleScope
+import foundation.e.flowmvi.MVIView
import foundation.e.privacycentralapp.R
+import foundation.e.privacycentralapp.dummy.InternetPrivacyMode
+import kotlinx.coroutines.flow.Flow
+import kotlinx.coroutines.flow.collect
+
+class InternetPrivacyFragment :
+ Fragment(R.layout.fragment_internet_activity_policy),
+ MVIView<InternetPrivacyFeature.State, InternetPrivacyFeature.Action> {
+
+ private val viewModel: InternetPrivacyViewModel by viewModels()
+
+ override fun onCreate(savedInstanceState: Bundle?) {
+ super.onCreate(savedInstanceState)
+ lifecycleScope.launchWhenStarted {
+ viewModel.internetPrivacyFeature.takeView(this, this@InternetPrivacyFragment)
+ }