@@ -5,8 +5,11 @@ import android.net.Uri
5
5
import androidx.compose.animation.animateContentSize
6
6
import androidx.compose.animation.core.animateFloatAsState
7
7
import androidx.compose.animation.core.tween
8
+ import androidx.compose.foundation.ScrollState
8
9
import androidx.compose.foundation.layout.Arrangement
9
10
import androidx.compose.foundation.layout.Column
11
+ import androidx.compose.foundation.layout.ColumnScope
12
+ import androidx.compose.foundation.layout.PaddingValues
10
13
import androidx.compose.foundation.layout.Spacer
11
14
import androidx.compose.foundation.layout.defaultMinSize
12
15
import androidx.compose.foundation.layout.fillMaxHeight
@@ -19,7 +22,6 @@ import androidx.compose.material3.MaterialTheme
19
22
import androidx.compose.material3.Text
20
23
import androidx.compose.runtime.Composable
21
24
import androidx.compose.runtime.LaunchedEffect
22
- import androidx.compose.runtime.collectAsState
23
25
import androidx.compose.runtime.getValue
24
26
import androidx.compose.runtime.mutableFloatStateOf
25
27
import androidx.compose.runtime.mutableLongStateOf
@@ -37,6 +39,7 @@ import androidx.compose.ui.platform.LocalContext
37
39
import androidx.compose.ui.platform.testTag
38
40
import androidx.compose.ui.res.stringResource
39
41
import androidx.compose.ui.tooling.preview.Preview
42
+ import androidx.lifecycle.compose.collectAsStateWithLifecycle
40
43
import com.ramcosta.composedestinations.annotation.Destination
41
44
import com.ramcosta.composedestinations.navigation.DestinationsNavigator
42
45
import com.ramcosta.composedestinations.navigation.popUpTo
@@ -95,7 +98,7 @@ private fun PreviewConnectScreen() {
95
98
val state = ConnectUiState .INITIAL
96
99
AppTheme {
97
100
ConnectScreen (
98
- uiState = state,
101
+ state = state,
99
102
)
100
103
}
101
104
}
@@ -105,7 +108,7 @@ private fun PreviewConnectScreen() {
105
108
fun Connect (navigator : DestinationsNavigator ) {
106
109
val connectViewModel: ConnectViewModel = koinViewModel()
107
110
108
- val state = connectViewModel.uiState.collectAsState().value
111
+ val state by connectViewModel.uiState.collectAsStateWithLifecycle()
109
112
110
113
val context = LocalContext .current
111
114
LaunchedEffect (key1 = Unit ) {
@@ -130,7 +133,7 @@ fun Connect(navigator: DestinationsNavigator) {
130
133
}
131
134
}
132
135
ConnectScreen (
133
- uiState = state,
136
+ state = state,
134
137
onDisconnectClick = connectViewModel::onDisconnectClick,
135
138
onReconnectClick = connectViewModel::onReconnectClick,
136
139
onConnectClick = connectViewModel::onConnectClick,
@@ -160,7 +163,7 @@ fun Connect(navigator: DestinationsNavigator) {
160
163
161
164
@Composable
162
165
fun ConnectScreen (
163
- uiState : ConnectUiState ,
166
+ state : ConnectUiState ,
164
167
onDisconnectClick : () -> Unit = {},
165
168
onReconnectClick : () -> Unit = {},
166
169
onConnectClick : () -> Unit = {},
@@ -174,65 +177,22 @@ fun ConnectScreen(
174
177
) {
175
178
176
179
val scrollState = rememberScrollState()
177
- var lastConnectionActionTimestamp by remember { mutableLongStateOf(0L ) }
178
-
179
- fun handleThrottledAction (action : () -> Unit ) {
180
- val currentTime = System .currentTimeMillis()
181
- if ((currentTime - lastConnectionActionTimestamp) > CONNECT_BUTTON_THROTTLE_MILLIS ) {
182
- lastConnectionActionTimestamp = currentTime
183
- action.invoke()
184
- }
185
- }
186
180
187
181
ScaffoldWithTopBarAndDeviceName (
188
- topBarColor = uiState .tunnelUiState.topBarColor(),
189
- iconTintColor = uiState .tunnelUiState.iconTintColor(),
182
+ topBarColor = state .tunnelUiState.topBarColor(),
183
+ iconTintColor = state .tunnelUiState.iconTintColor(),
190
184
onSettingsClicked = onSettingsClick,
191
185
onAccountClicked = onAccountClick,
192
- deviceName = uiState .deviceName,
193
- timeLeft = uiState .daysLeftUntilExpiry
186
+ deviceName = state .deviceName,
187
+ timeLeft = state .daysLeftUntilExpiry
194
188
) {
195
189
var progressIndicatorBias by remember { mutableFloatStateOf(0f ) }
196
190
197
- // Distance to marker when secure/unsecure
198
- val baseZoom =
199
- animateFloatAsState(
200
- targetValue =
201
- if (uiState.tunnelRealState is TunnelState .Connected ) SECURE_ZOOM
202
- else UNSECURE_ZOOM ,
203
- animationSpec = tween(SECURE_ZOOM_ANIMATION_MILLIS ),
204
- label = " baseZoom"
205
- )
206
-
207
- val markers =
208
- uiState.tunnelRealState.toMarker(uiState.location)?.let { listOf (it) } ? : emptyList()
209
-
210
- AnimatedMap (
211
- modifier = Modifier .padding(top = it.calculateTopPadding()),
212
- cameraLocation = uiState.location?.toLatLong() ? : fallbackLatLong,
213
- cameraBaseZoom = baseZoom.value,
214
- cameraVerticalBias = progressIndicatorBias,
215
- markers = markers,
216
- globeColors =
217
- GlobeColors (
218
- landColor = MaterialTheme .colorScheme.primary,
219
- oceanColor = MaterialTheme .colorScheme.secondary,
220
- )
221
- )
222
-
223
- Column (
224
- verticalArrangement = Arrangement .Bottom ,
225
- horizontalAlignment = Alignment .Start ,
226
- modifier =
227
- Modifier .animateContentSize()
228
- .padding(top = it.calculateTopPadding())
229
- .fillMaxHeight()
230
- .drawVerticalScrollbar(
231
- scrollState,
232
- color = MaterialTheme .colorScheme.onPrimary.copy(alpha = AlphaScrollbar )
233
- )
234
- .verticalScroll(scrollState)
235
- .testTag(SCROLLABLE_COLUMN_TEST_TAG )
191
+ MapColumn (
192
+ state,
193
+ it,
194
+ progressIndicatorBias,
195
+ scrollState,
236
196
) {
237
197
Spacer (modifier = Modifier .defaultMinSize(minHeight = Dimens .mediumPadding).weight(1f ))
238
198
MullvadCircularProgressIndicatorLarge (
@@ -244,7 +204,7 @@ fun ConnectScreen(
244
204
end = Dimens .sideMargin,
245
205
top = Dimens .mediumPadding
246
206
)
247
- .alpha(if (uiState .showLoading) AlphaVisible else AlphaInvisible )
207
+ .alpha(if (state .showLoading) AlphaVisible else AlphaInvisible )
248
208
.align(Alignment .CenterHorizontally )
249
209
.testTag(CIRCULAR_PROGRESS_INDICATOR )
250
210
.onGloballyPositioned {
@@ -259,79 +219,167 @@ fun ConnectScreen(
259
219
}
260
220
)
261
221
Spacer (modifier = Modifier .defaultMinSize(minHeight = Dimens .mediumPadding).weight(1f ))
262
- ConnectionStatusText (
263
- state = uiState.tunnelRealState,
264
- modifier = Modifier .padding(horizontal = Dimens .sideMargin)
265
- )
266
- Text (
267
- text = uiState.location?.country ? : " " ,
268
- style = MaterialTheme .typography.headlineLarge,
269
- color = MaterialTheme .colorScheme.onPrimary,
270
- modifier = Modifier .padding(horizontal = Dimens .sideMargin)
271
- )
272
- Text (
273
- text = uiState.location?.city ? : " " ,
274
- style = MaterialTheme .typography.headlineLarge,
275
- color = MaterialTheme .colorScheme.onPrimary,
276
- modifier = Modifier .padding(horizontal = Dimens .sideMargin)
277
- )
278
- var expanded by rememberSaveable { mutableStateOf(false ) }
279
- LocationInfo (
280
- onToggleTunnelInfo = { expanded = ! expanded },
281
- isVisible = uiState.showLocationInfo,
282
- isExpanded = expanded,
283
- location = uiState.location,
284
- inAddress = uiState.inAddress,
285
- outAddress = uiState.outAddress,
286
- modifier =
287
- Modifier .fillMaxWidth()
288
- .padding(horizontal = Dimens .sideMargin)
289
- .testTag(LOCATION_INFO_TEST_TAG )
290
- )
291
- Spacer (modifier = Modifier .height(Dimens .buttonSpacing))
292
- SwitchLocationButton (
293
- modifier =
294
- Modifier .fillMaxWidth()
295
- .padding(horizontal = Dimens .sideMargin)
296
- .testTag(SELECT_LOCATION_BUTTON_TEST_TAG ),
297
- onClick = onSwitchLocationClick,
298
- showChevron = uiState.showLocation,
299
- text =
300
- if (uiState.showLocation && uiState.selectedRelayItem != null ) {
301
- uiState.selectedRelayItem.locationName
302
- } else {
303
- stringResource(id = R .string.switch_location)
304
- }
305
- )
222
+
223
+ ConnectionInfo (state = state)
224
+
306
225
Spacer (modifier = Modifier .height(Dimens .buttonSpacing))
307
- ConnectionButton (
308
- state = uiState.tunnelUiState,
309
- modifier =
310
- Modifier .padding(horizontal = Dimens .sideMargin)
311
- .padding(bottom = Dimens .screenVerticalMargin)
312
- .testTag(CONNECT_BUTTON_TEST_TAG ),
313
- disconnectClick = onDisconnectClick,
314
- reconnectClick = { handleThrottledAction(onReconnectClick) },
315
- cancelClick = onCancelClick,
316
- connectClick = { handleThrottledAction(onConnectClick) },
317
- reconnectButtonTestTag = RECONNECT_BUTTON_TEST_TAG
226
+
227
+ ButtonPanel (
228
+ state,
229
+ onSwitchLocationClick,
230
+ onDisconnectClick,
231
+ onReconnectClick,
232
+ onCancelClick,
233
+ onConnectClick,
318
234
)
319
- // We need to manually add this padding so we align size with the map
320
- // component and marker with the progress indicator.
321
- Spacer (modifier = Modifier .height(it.calculateBottomPadding()))
322
235
}
323
236
324
237
NotificationBanner (
325
238
modifier = Modifier .padding(top = it.calculateTopPadding()),
326
- notification = uiState .inAppNotification,
327
- isPlayBuild = uiState .isPlayBuild,
239
+ notification = state .inAppNotification,
240
+ isPlayBuild = state .isPlayBuild,
328
241
onClickUpdateVersion = onUpdateVersionClick,
329
242
onClickShowAccount = onManageAccountClick,
330
243
onClickDismissNewDevice = onDismissNewDeviceClick,
331
244
)
332
245
}
333
246
}
334
247
248
+ @Composable
249
+ private fun MapColumn (
250
+ state : ConnectUiState ,
251
+ it : PaddingValues ,
252
+ progressIndicatorBias : Float ,
253
+ scrollState : ScrollState ,
254
+ content : @Composable ColumnScope .() -> Unit
255
+ ) {
256
+
257
+ // Distance to marker when secure/unsecure
258
+ val baseZoom =
259
+ animateFloatAsState(
260
+ targetValue =
261
+ if (state.tunnelRealState is TunnelState .Connected ) SECURE_ZOOM else UNSECURE_ZOOM ,
262
+ animationSpec = tween(SECURE_ZOOM_ANIMATION_MILLIS ),
263
+ label = " baseZoom"
264
+ )
265
+
266
+ val markers = state.tunnelRealState.toMarker(state.location)?.let { listOf (it) } ? : emptyList()
267
+
268
+ AnimatedMap (
269
+ modifier = Modifier .padding(top = it.calculateTopPadding()),
270
+ cameraLocation = state.location?.toLatLong() ? : fallbackLatLong,
271
+ cameraBaseZoom = baseZoom.value,
272
+ cameraVerticalBias = progressIndicatorBias,
273
+ markers = markers,
274
+ globeColors =
275
+ GlobeColors (
276
+ landColor = MaterialTheme .colorScheme.primary,
277
+ oceanColor = MaterialTheme .colorScheme.secondary,
278
+ )
279
+ )
280
+
281
+ Column (
282
+ verticalArrangement = Arrangement .Bottom ,
283
+ horizontalAlignment = Alignment .Start ,
284
+ modifier =
285
+ Modifier .animateContentSize()
286
+ .padding(top = it.calculateTopPadding())
287
+ .fillMaxHeight()
288
+ .drawVerticalScrollbar(
289
+ scrollState,
290
+ color = MaterialTheme .colorScheme.onPrimary.copy(alpha = AlphaScrollbar )
291
+ )
292
+ .verticalScroll(scrollState)
293
+ .testTag(SCROLLABLE_COLUMN_TEST_TAG )
294
+ ) {
295
+ content()
296
+ // We need to manually add this padding so we align size with the map
297
+ // component and marker with the progress indicator.
298
+ Spacer (modifier = Modifier .height(it.calculateBottomPadding()))
299
+ }
300
+ }
301
+
302
+ @Composable
303
+ private fun ConnectionInfo (state : ConnectUiState ) {
304
+ ConnectionStatusText (
305
+ state = state.tunnelRealState,
306
+ modifier = Modifier .padding(horizontal = Dimens .sideMargin)
307
+ )
308
+ Text (
309
+ text = state.location?.country ? : " " ,
310
+ style = MaterialTheme .typography.headlineLarge,
311
+ color = MaterialTheme .colorScheme.onPrimary,
312
+ modifier = Modifier .padding(horizontal = Dimens .sideMargin)
313
+ )
314
+ Text (
315
+ text = state.location?.city ? : " " ,
316
+ style = MaterialTheme .typography.headlineLarge,
317
+ color = MaterialTheme .colorScheme.onPrimary,
318
+ modifier = Modifier .padding(horizontal = Dimens .sideMargin)
319
+ )
320
+ var expanded by rememberSaveable { mutableStateOf(false ) }
321
+ LocationInfo (
322
+ onToggleTunnelInfo = { expanded = ! expanded },
323
+ isVisible = state.showLocationInfo,
324
+ isExpanded = expanded,
325
+ location = state.location,
326
+ inAddress = state.inAddress,
327
+ outAddress = state.outAddress,
328
+ modifier =
329
+ Modifier .fillMaxWidth()
330
+ .padding(horizontal = Dimens .sideMargin)
331
+ .testTag(LOCATION_INFO_TEST_TAG )
332
+ )
333
+ }
334
+
335
+ @Composable
336
+ private fun ButtonPanel (
337
+ state : ConnectUiState ,
338
+ onSwitchLocationClick : () -> Unit ,
339
+ onDisconnectClick : () -> Unit ,
340
+ onReconnectClick : () -> Unit ,
341
+ onCancelClick : () -> Unit ,
342
+ onConnectClick : () -> Unit ,
343
+ ) {
344
+ var lastConnectionActionTimestamp by remember { mutableLongStateOf(0L ) }
345
+
346
+ fun handleThrottledAction (action : () -> Unit ) {
347
+ val currentTime = System .currentTimeMillis()
348
+ if ((currentTime - lastConnectionActionTimestamp) > CONNECT_BUTTON_THROTTLE_MILLIS ) {
349
+ lastConnectionActionTimestamp = currentTime
350
+ action.invoke()
351
+ }
352
+ }
353
+
354
+ SwitchLocationButton (
355
+ modifier =
356
+ Modifier .fillMaxWidth()
357
+ .padding(horizontal = Dimens .sideMargin)
358
+ .testTag(SELECT_LOCATION_BUTTON_TEST_TAG ),
359
+ onClick = onSwitchLocationClick,
360
+ showChevron = state.showLocation,
361
+ text =
362
+ if (state.showLocation && state.selectedRelayItem != null ) {
363
+ state.selectedRelayItem.locationName
364
+ } else {
365
+ stringResource(id = R .string.switch_location)
366
+ }
367
+ )
368
+ Spacer (modifier = Modifier .height(Dimens .buttonSpacing))
369
+ ConnectionButton (
370
+ state = state.tunnelUiState,
371
+ modifier =
372
+ Modifier .padding(horizontal = Dimens .sideMargin)
373
+ .padding(bottom = Dimens .screenVerticalMargin)
374
+ .testTag(CONNECT_BUTTON_TEST_TAG ),
375
+ disconnectClick = onDisconnectClick,
376
+ reconnectClick = { handleThrottledAction(onReconnectClick) },
377
+ cancelClick = onCancelClick,
378
+ connectClick = { handleThrottledAction(onConnectClick) },
379
+ reconnectButtonTestTag = RECONNECT_BUTTON_TEST_TAG
380
+ )
381
+ }
382
+
335
383
@Composable
336
384
fun TunnelState.toMarker (location : GeoIpLocation ? ): Marker ? {
337
385
if (location == null ) return null
0 commit comments