@@ -4,8 +4,10 @@ import androidx.compose.animation.AnimatedVisibility
4
4
import androidx.compose.animation.core.animateDpAsState
5
5
import androidx.compose.animation.fadeIn
6
6
import androidx.compose.animation.fadeOut
7
+ import androidx.compose.foundation.ExperimentalFoundationApi
7
8
import androidx.compose.foundation.background
8
9
import androidx.compose.foundation.clickable
10
+ import androidx.compose.foundation.draganddrop.dragAndDropTarget
9
11
import androidx.compose.foundation.hoverable
10
12
import androidx.compose.foundation.interaction.MutableInteractionSource
11
13
import androidx.compose.foundation.interaction.collectIsHoveredAsState
@@ -19,7 +21,7 @@ import androidx.compose.foundation.rememberScrollState
19
21
import androidx.compose.foundation.shape.CircleShape
20
22
import androidx.compose.foundation.verticalScroll
21
23
import androidx.compose.material.icons.Icons
22
- import androidx.compose.material.icons.twotone.Edit
24
+ import androidx.compose.material.icons.filled.Upload
23
25
import androidx.compose.material3.Button
24
26
import androidx.compose.material3.Icon
25
27
import androidx.compose.material3.MaterialTheme
@@ -29,6 +31,7 @@ import androidx.compose.runtime.getValue
29
31
import androidx.compose.runtime.remember
30
32
import androidx.compose.ui.Alignment
31
33
import androidx.compose.ui.Modifier
34
+ import androidx.compose.ui.draganddrop.mimeTypes
32
35
import androidx.compose.ui.draw.blur
33
36
import androidx.compose.ui.draw.clip
34
37
import androidx.compose.ui.graphics.Color
@@ -72,6 +75,7 @@ fun SettingsScreen(viewModel: SettingsViewModel = hiltViewModel()) {
72
75
}
73
76
}
74
77
78
+ @OptIn(ExperimentalFoundationApi ::class )
75
79
@Composable
76
80
private fun UserInfo (user : User ? , onAvatarFile : (File ) -> Unit ) {
77
81
val activity = activity
@@ -80,7 +84,12 @@ private fun UserInfo(user: User?, onAvatarFile: (File) -> Unit) {
80
84
val avatarSize = 128 .dp
81
85
val avatarInteraction = remember { MutableInteractionSource () }
82
86
val isAvatarHovered by avatarInteraction.collectIsHoveredAsState()
83
- val avatarBlur by animateDpAsState(if (isAvatarHovered) 1 .dp else 0 .dp, label = " Avatar blur" )
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
+ )
84
93
85
94
Box (contentAlignment = Alignment .Center ) {
86
95
Avatar (
@@ -91,15 +100,23 @@ private fun UserInfo(user: User?, onAvatarFile: (File) -> Unit) {
91
100
.blur(avatarBlur)
92
101
.hoverable(avatarInteraction)
93
102
.clickable { activity?.selectImage(onAvatarFile) }
103
+ .dragAndDropTarget(
104
+ shouldStartDragAndDrop = { dropEvent ->
105
+ dropEvent
106
+ .mimeTypes()
107
+ .any { it.startsWith(" image/" ) }
108
+ },
109
+ target = avatarDropTarget
110
+ )
94
111
)
95
112
96
113
AnimatedVisibility (
97
- visible = isAvatarHovered ,
114
+ visible = isAvatarUpdatable ,
98
115
enter = fadeIn(),
99
116
exit = fadeOut()
100
117
) {
101
118
Icon (
102
- imageVector = Icons .TwoTone . Edit ,
119
+ imageVector = Icons .Default . Upload ,
103
120
contentDescription = null ,
104
121
tint = Color .White ,
105
122
modifier = Modifier
0 commit comments