|
1 | 1 | package app.fyreplace.fyreplace.ui.screens
|
2 | 2 |
|
3 |
| -import androidx.compose.animation.AnimatedVisibility |
4 |
| -import androidx.compose.animation.core.animateDpAsState |
5 |
| -import androidx.compose.animation.fadeIn |
6 |
| -import androidx.compose.animation.fadeOut |
7 |
| -import androidx.compose.foundation.ExperimentalFoundationApi |
8 |
| -import androidx.compose.foundation.background |
9 |
| -import androidx.compose.foundation.clickable |
10 |
| -import androidx.compose.foundation.draganddrop.dragAndDropTarget |
11 |
| -import androidx.compose.foundation.hoverable |
12 |
| -import androidx.compose.foundation.interaction.MutableInteractionSource |
13 |
| -import androidx.compose.foundation.interaction.collectIsHoveredAsState |
14 | 3 | import androidx.compose.foundation.layout.Arrangement
|
15 |
| -import androidx.compose.foundation.layout.Box |
16 | 4 | import androidx.compose.foundation.layout.Column
|
17 | 5 | import androidx.compose.foundation.layout.fillMaxWidth
|
18 |
| -import androidx.compose.foundation.layout.padding |
19 |
| -import androidx.compose.foundation.layout.size |
20 | 6 | import androidx.compose.foundation.rememberScrollState
|
21 |
| -import androidx.compose.foundation.shape.CircleShape |
22 | 7 | import androidx.compose.foundation.verticalScroll
|
23 | 8 | import androidx.compose.material.icons.Icons
|
24 |
| -import androidx.compose.material.icons.filled.Upload |
25 |
| -import androidx.compose.material3.Button |
| 9 | +import androidx.compose.material.icons.automirrored.filled.Logout |
26 | 10 | import androidx.compose.material3.Icon
|
27 |
| -import androidx.compose.material3.MaterialTheme |
28 |
| -import androidx.compose.material3.Text |
29 | 11 | import androidx.compose.runtime.Composable
|
30 | 12 | import androidx.compose.runtime.getValue
|
31 |
| -import androidx.compose.runtime.remember |
32 |
| -import androidx.compose.ui.Alignment |
33 | 13 | import androidx.compose.ui.Modifier
|
34 |
| -import androidx.compose.ui.draganddrop.mimeTypes |
35 |
| -import androidx.compose.ui.draw.blur |
36 |
| -import androidx.compose.ui.draw.clip |
37 |
| -import androidx.compose.ui.graphics.Color |
38 | 14 | import androidx.compose.ui.res.dimensionResource
|
39 | 15 | import androidx.compose.ui.res.stringResource
|
40 | 16 | import androidx.compose.ui.tooling.preview.Preview
|
41 |
| -import androidx.compose.ui.unit.dp |
42 | 17 | import androidx.hilt.navigation.compose.hiltViewModel
|
43 | 18 | import androidx.lifecycle.SavedStateHandle
|
44 | 19 | import androidx.lifecycle.compose.collectAsStateWithLifecycle
|
45 | 20 | import app.fyreplace.api.data.User
|
46 | 21 | import app.fyreplace.fyreplace.R
|
47 |
| -import app.fyreplace.fyreplace.extensions.activity |
48 | 22 | import app.fyreplace.fyreplace.fakes.FakeApiResolver
|
49 | 23 | import app.fyreplace.fyreplace.fakes.FakeEventBus
|
50 | 24 | import app.fyreplace.fyreplace.fakes.FakeStoreResolver
|
51 | 25 | import app.fyreplace.fyreplace.fakes.placeholder
|
52 | 26 | import app.fyreplace.fyreplace.ui.theme.AppTheme
|
53 |
| -import app.fyreplace.fyreplace.ui.views.Avatar |
| 27 | +import app.fyreplace.fyreplace.ui.views.settings.AvatarPreference |
| 28 | +import app.fyreplace.fyreplace.ui.views.settings.Preference |
| 29 | +import app.fyreplace.fyreplace.ui.views.settings.Section |
54 | 30 | import app.fyreplace.fyreplace.viewmodels.screens.SettingsViewModel
|
55 |
| -import java.io.File |
56 |
| -import java.time.format.DateTimeFormatter |
57 |
| -import java.time.format.FormatStyle |
58 | 31 |
|
59 | 32 | @Composable
|
60 | 33 | fun SettingsScreen(viewModel: SettingsViewModel = hiltViewModel()) {
|
61 | 34 | Column(
|
62 | 35 | verticalArrangement = Arrangement.spacedBy(dimensionResource(R.dimen.spacing_medium)),
|
63 |
| - horizontalAlignment = Alignment.CenterHorizontally, |
64 | 36 | modifier = Modifier
|
65 | 37 | .fillMaxWidth()
|
66 | 38 | .verticalScroll(rememberScrollState())
|
67 |
| - .padding(vertical = dimensionResource(R.dimen.spacing_medium)) |
68 | 39 | ) {
|
69 | 40 | val currentUser by viewModel.currentUser.collectAsStateWithLifecycle()
|
70 | 41 |
|
71 |
| - UserInfo(user = currentUser, onAvatarFile = viewModel::updateAvatar) |
72 |
| - Button(onClick = viewModel::logout) { |
73 |
| - Text(stringResource(R.string.settings_logout)) |
74 |
| - } |
75 |
| - } |
76 |
| -} |
77 |
| - |
78 |
| -@OptIn(ExperimentalFoundationApi::class) |
79 |
| -@Composable |
80 |
| -private fun UserInfo(user: User?, onAvatarFile: (File) -> Unit) { |
81 |
| - val activity = activity |
82 |
| - val dateFormatter = remember { DateTimeFormatter.ofLocalizedDateTime(FormatStyle.SHORT) } |
83 |
| - val loading = stringResource(R.string.loading) |
84 |
| - val avatarSize = 128.dp |
85 |
| - val avatarInteraction = remember { MutableInteractionSource() } |
86 |
| - val isAvatarHovered by avatarInteraction.collectIsHoveredAsState() |
87 |
| - val avatarDropTarget = remember { requireNotNull(activity).makeFileDropTarget(onAvatarFile) } |
88 |
| - val isAvatarUpdatable = isAvatarHovered || avatarDropTarget.isReady |
89 |
| - val avatarBlur by animateDpAsState( |
90 |
| - targetValue = if (isAvatarUpdatable) 1.dp else 0.dp, |
91 |
| - label = "Avatar blur" |
92 |
| - ) |
93 |
| - |
94 |
| - Box(contentAlignment = Alignment.Center) { |
95 |
| - Avatar( |
96 |
| - user = user, |
97 |
| - tinted = true, |
98 |
| - size = avatarSize, |
99 |
| - modifier = Modifier |
100 |
| - .blur(avatarBlur) |
101 |
| - .hoverable(avatarInteraction) |
102 |
| - .clickable { activity?.selectImage(onAvatarFile) } |
103 |
| - .dragAndDropTarget( |
104 |
| - shouldStartDragAndDrop = { dropEvent -> |
105 |
| - dropEvent |
106 |
| - .mimeTypes() |
107 |
| - .any { it.startsWith("image/") } |
108 |
| - }, |
109 |
| - target = avatarDropTarget |
110 |
| - ) |
111 |
| - ) |
| 42 | + Section(stringResource(R.string.settings_header_profile)) { |
| 43 | + AvatarPreference( |
| 44 | + user = currentUser, |
| 45 | + onUpdateAvatar = viewModel::updateAvatar, |
| 46 | + onRemoveAvatar = viewModel::removeAvatar |
| 47 | + ) |
112 | 48 |
|
113 |
| - AnimatedVisibility( |
114 |
| - visible = isAvatarUpdatable, |
115 |
| - enter = fadeIn(), |
116 |
| - exit = fadeOut() |
117 |
| - ) { |
118 |
| - Icon( |
119 |
| - imageVector = Icons.Default.Upload, |
120 |
| - contentDescription = null, |
121 |
| - tint = Color.White, |
122 |
| - modifier = Modifier |
123 |
| - .size(avatarSize) |
124 |
| - .clip(CircleShape) |
125 |
| - .background(Color.Black.copy(alpha = 0.5f)) |
126 |
| - .padding(avatarSize / 4) |
| 49 | + Preference( |
| 50 | + title = stringResource(R.string.settings_logout), |
| 51 | + summary = stringResource(R.string.settings_logout_summary), |
| 52 | + icon = { Icon(Icons.AutoMirrored.Filled.Logout, null) }, |
| 53 | + onClick = viewModel::logout |
127 | 54 | )
|
128 | 55 | }
|
129 | 56 | }
|
130 |
| - |
131 |
| - Column(horizontalAlignment = Alignment.CenterHorizontally) { |
132 |
| - Text( |
133 |
| - text = user?.username ?: loading, |
134 |
| - style = MaterialTheme.typography.headlineMedium |
135 |
| - ) |
136 |
| - Text( |
137 |
| - text = when (user) { |
138 |
| - null -> loading |
139 |
| - else -> stringResource( |
140 |
| - R.string.settings_date_joined, |
141 |
| - dateFormatter.format(user.dateCreated) |
142 |
| - ) |
143 |
| - }, |
144 |
| - style = MaterialTheme.typography.titleMedium |
145 |
| - ) |
146 |
| - } |
147 | 57 | }
|
148 | 58 |
|
149 | 59 | @Preview(showSystemUi = true, showBackground = true)
|
|
0 commit comments