Skip to content

Commit 83c3e79

Browse files
committed
Fix long method lints
1 parent ebb52fd commit 83c3e79

File tree

6 files changed

+607
-486
lines changed

6 files changed

+607
-486
lines changed

android/app/src/main/kotlin/net/mullvad/mullvadvpn/compose/screen/ConnectScreen.kt

+166-118
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,11 @@ import android.net.Uri
55
import androidx.compose.animation.animateContentSize
66
import androidx.compose.animation.core.animateFloatAsState
77
import androidx.compose.animation.core.tween
8+
import androidx.compose.foundation.ScrollState
89
import androidx.compose.foundation.layout.Arrangement
910
import androidx.compose.foundation.layout.Column
11+
import androidx.compose.foundation.layout.ColumnScope
12+
import androidx.compose.foundation.layout.PaddingValues
1013
import androidx.compose.foundation.layout.Spacer
1114
import androidx.compose.foundation.layout.defaultMinSize
1215
import androidx.compose.foundation.layout.fillMaxHeight
@@ -19,7 +22,6 @@ import androidx.compose.material3.MaterialTheme
1922
import androidx.compose.material3.Text
2023
import androidx.compose.runtime.Composable
2124
import androidx.compose.runtime.LaunchedEffect
22-
import androidx.compose.runtime.collectAsState
2325
import androidx.compose.runtime.getValue
2426
import androidx.compose.runtime.mutableFloatStateOf
2527
import androidx.compose.runtime.mutableLongStateOf
@@ -37,6 +39,7 @@ import androidx.compose.ui.platform.LocalContext
3739
import androidx.compose.ui.platform.testTag
3840
import androidx.compose.ui.res.stringResource
3941
import androidx.compose.ui.tooling.preview.Preview
42+
import androidx.lifecycle.compose.collectAsStateWithLifecycle
4043
import com.ramcosta.composedestinations.annotation.Destination
4144
import com.ramcosta.composedestinations.navigation.DestinationsNavigator
4245
import com.ramcosta.composedestinations.navigation.popUpTo
@@ -95,7 +98,7 @@ private fun PreviewConnectScreen() {
9598
val state = ConnectUiState.INITIAL
9699
AppTheme {
97100
ConnectScreen(
98-
uiState = state,
101+
state = state,
99102
)
100103
}
101104
}
@@ -105,7 +108,7 @@ private fun PreviewConnectScreen() {
105108
fun Connect(navigator: DestinationsNavigator) {
106109
val connectViewModel: ConnectViewModel = koinViewModel()
107110

108-
val state = connectViewModel.uiState.collectAsState().value
111+
val state by connectViewModel.uiState.collectAsStateWithLifecycle()
109112

110113
val context = LocalContext.current
111114
LaunchedEffect(key1 = Unit) {
@@ -130,7 +133,7 @@ fun Connect(navigator: DestinationsNavigator) {
130133
}
131134
}
132135
ConnectScreen(
133-
uiState = state,
136+
state = state,
134137
onDisconnectClick = connectViewModel::onDisconnectClick,
135138
onReconnectClick = connectViewModel::onReconnectClick,
136139
onConnectClick = connectViewModel::onConnectClick,
@@ -160,7 +163,7 @@ fun Connect(navigator: DestinationsNavigator) {
160163

161164
@Composable
162165
fun ConnectScreen(
163-
uiState: ConnectUiState,
166+
state: ConnectUiState,
164167
onDisconnectClick: () -> Unit = {},
165168
onReconnectClick: () -> Unit = {},
166169
onConnectClick: () -> Unit = {},
@@ -174,65 +177,22 @@ fun ConnectScreen(
174177
) {
175178

176179
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-
}
186180

187181
ScaffoldWithTopBarAndDeviceName(
188-
topBarColor = uiState.tunnelUiState.topBarColor(),
189-
iconTintColor = uiState.tunnelUiState.iconTintColor(),
182+
topBarColor = state.tunnelUiState.topBarColor(),
183+
iconTintColor = state.tunnelUiState.iconTintColor(),
190184
onSettingsClicked = onSettingsClick,
191185
onAccountClicked = onAccountClick,
192-
deviceName = uiState.deviceName,
193-
timeLeft = uiState.daysLeftUntilExpiry
186+
deviceName = state.deviceName,
187+
timeLeft = state.daysLeftUntilExpiry
194188
) {
195189
var progressIndicatorBias by remember { mutableFloatStateOf(0f) }
196190

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,
236196
) {
237197
Spacer(modifier = Modifier.defaultMinSize(minHeight = Dimens.mediumPadding).weight(1f))
238198
MullvadCircularProgressIndicatorLarge(
@@ -244,7 +204,7 @@ fun ConnectScreen(
244204
end = Dimens.sideMargin,
245205
top = Dimens.mediumPadding
246206
)
247-
.alpha(if (uiState.showLoading) AlphaVisible else AlphaInvisible)
207+
.alpha(if (state.showLoading) AlphaVisible else AlphaInvisible)
248208
.align(Alignment.CenterHorizontally)
249209
.testTag(CIRCULAR_PROGRESS_INDICATOR)
250210
.onGloballyPositioned {
@@ -259,79 +219,167 @@ fun ConnectScreen(
259219
}
260220
)
261221
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+
306225
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,
318234
)
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()))
322235
}
323236

324237
NotificationBanner(
325238
modifier = Modifier.padding(top = it.calculateTopPadding()),
326-
notification = uiState.inAppNotification,
327-
isPlayBuild = uiState.isPlayBuild,
239+
notification = state.inAppNotification,
240+
isPlayBuild = state.isPlayBuild,
328241
onClickUpdateVersion = onUpdateVersionClick,
329242
onClickShowAccount = onManageAccountClick,
330243
onClickDismissNewDevice = onDismissNewDeviceClick,
331244
)
332245
}
333246
}
334247

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+
335383
@Composable
336384
fun TunnelState.toMarker(location: GeoIpLocation?): Marker? {
337385
if (location == null) return null

0 commit comments

Comments
 (0)