Skip to content

Commit c32ea80

Browse files
Support dropping avatar images
1 parent c62eb6c commit c32ea80

File tree

2 files changed

+66
-6
lines changed

2 files changed

+66
-6
lines changed

app/src/main/kotlin/app/fyreplace/fyreplace/ImageSelectorActivity.kt

+45-2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,12 @@ import android.net.Uri
44
import android.os.Bundle
55
import androidx.activity.result.ActivityResultLauncher
66
import androidx.activity.result.contract.ActivityResultContracts
7+
import androidx.compose.runtime.getValue
8+
import androidx.compose.runtime.mutableStateOf
9+
import androidx.compose.runtime.setValue
10+
import androidx.compose.ui.draganddrop.DragAndDropEvent
11+
import androidx.compose.ui.draganddrop.DragAndDropTarget
12+
import androidx.compose.ui.draganddrop.toAndroidDragEvent
713
import androidx.lifecycle.lifecycleScope
814
import app.fyreplace.fyreplace.events.EventBus
915
import kotlinx.coroutines.Dispatchers
@@ -34,13 +40,50 @@ abstract class ImageSelectorActivity : SecureActivity() {
3440
}
3541

3642

37-
private suspend fun makeFileFromUri(uri: Uri) = withContext(Dispatchers.IO) {
38-
val file = File.createTempFile("image", ".tmp")
43+
suspend fun makeFileFromUri(uri: Uri): File = withContext(Dispatchers.IO) {
44+
val file = File.createTempFile("image", null)
45+
file.deleteOnExit()
3946

4047
contentResolver.openInputStream(uri)?.use { stream ->
4148
file.outputStream().use(stream::copyTo)
4249
}
4350

4451
return@withContext file
4552
}
53+
54+
fun makeFileDropTarget(onFile: suspend (File) -> Unit) = FileDropTarget(onFile)
55+
56+
inner class FileDropTarget(private val onFile: suspend (File) -> Unit) : DragAndDropTarget {
57+
var isReady by mutableStateOf(false)
58+
private set
59+
60+
override fun onDrop(event: DragAndDropEvent): Boolean {
61+
val dragEvent = event.toAndroidDragEvent()
62+
val uri = dragEvent.clipData?.getItemAt(0)?.uri ?: return false
63+
64+
lifecycleScope.launch {
65+
val permissions = requestDragAndDropPermissions(dragEvent)
66+
67+
try {
68+
onFile(makeFileFromUri(uri))
69+
} finally {
70+
permissions.release()
71+
}
72+
}
73+
74+
return true
75+
}
76+
77+
override fun onEnded(event: DragAndDropEvent) {
78+
isReady = false
79+
}
80+
81+
override fun onEntered(event: DragAndDropEvent) {
82+
isReady = true
83+
}
84+
85+
override fun onExited(event: DragAndDropEvent) {
86+
isReady = false
87+
}
88+
}
4689
}

app/src/main/kotlin/app/fyreplace/fyreplace/ui/screens/SettingsScreen.kt

+21-4
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,10 @@ import androidx.compose.animation.AnimatedVisibility
44
import androidx.compose.animation.core.animateDpAsState
55
import androidx.compose.animation.fadeIn
66
import androidx.compose.animation.fadeOut
7+
import androidx.compose.foundation.ExperimentalFoundationApi
78
import androidx.compose.foundation.background
89
import androidx.compose.foundation.clickable
10+
import androidx.compose.foundation.draganddrop.dragAndDropTarget
911
import androidx.compose.foundation.hoverable
1012
import androidx.compose.foundation.interaction.MutableInteractionSource
1113
import androidx.compose.foundation.interaction.collectIsHoveredAsState
@@ -19,7 +21,7 @@ import androidx.compose.foundation.rememberScrollState
1921
import androidx.compose.foundation.shape.CircleShape
2022
import androidx.compose.foundation.verticalScroll
2123
import androidx.compose.material.icons.Icons
22-
import androidx.compose.material.icons.twotone.Edit
24+
import androidx.compose.material.icons.filled.Upload
2325
import androidx.compose.material3.Button
2426
import androidx.compose.material3.Icon
2527
import androidx.compose.material3.MaterialTheme
@@ -29,6 +31,7 @@ import androidx.compose.runtime.getValue
2931
import androidx.compose.runtime.remember
3032
import androidx.compose.ui.Alignment
3133
import androidx.compose.ui.Modifier
34+
import androidx.compose.ui.draganddrop.mimeTypes
3235
import androidx.compose.ui.draw.blur
3336
import androidx.compose.ui.draw.clip
3437
import androidx.compose.ui.graphics.Color
@@ -72,6 +75,7 @@ fun SettingsScreen(viewModel: SettingsViewModel = hiltViewModel()) {
7275
}
7376
}
7477

78+
@OptIn(ExperimentalFoundationApi::class)
7579
@Composable
7680
private fun UserInfo(user: User?, onAvatarFile: (File) -> Unit) {
7781
val activity = activity
@@ -80,7 +84,12 @@ private fun UserInfo(user: User?, onAvatarFile: (File) -> Unit) {
8084
val avatarSize = 128.dp
8185
val avatarInteraction = remember { MutableInteractionSource() }
8286
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+
)
8493

8594
Box(contentAlignment = Alignment.Center) {
8695
Avatar(
@@ -91,15 +100,23 @@ private fun UserInfo(user: User?, onAvatarFile: (File) -> Unit) {
91100
.blur(avatarBlur)
92101
.hoverable(avatarInteraction)
93102
.clickable { activity?.selectImage(onAvatarFile) }
103+
.dragAndDropTarget(
104+
shouldStartDragAndDrop = { dropEvent ->
105+
dropEvent
106+
.mimeTypes()
107+
.any { it.startsWith("image/") }
108+
},
109+
target = avatarDropTarget
110+
)
94111
)
95112

96113
AnimatedVisibility(
97-
visible = isAvatarHovered,
114+
visible = isAvatarUpdatable,
98115
enter = fadeIn(),
99116
exit = fadeOut()
100117
) {
101118
Icon(
102-
imageVector = Icons.TwoTone.Edit,
119+
imageVector = Icons.Default.Upload,
103120
contentDescription = null,
104121
tint = Color.White,
105122
modifier = Modifier

0 commit comments

Comments
 (0)