Jetpack Compose поддерживает функцию перетаскивания с двумя модификаторами:
-
dragAndDropSource
: указывает компонуемый объект в качестве начальной точки жеста перетаскивания. -
dragAndDropTarget
: указывает компонуемый объект, который принимает перетаскиваемые данные.
Например, чтобы пользователи могли перетаскивать изображение в вашем приложении, создайте изображение, компонуемое и добавьте модификатор dragAndDropSource
. Чтобы настроить цель перетаскивания, создайте еще одно изображение, компонуемое и добавьте модификатор dragAndDropTarget
.
Модификаторы можно применять к нескольким источникам перетаскивания и нескольким целям перетаскивания.
Модификаторы позволяют приложениям обмениваться данными между двумя или более компонуемыми объектами с помощью ClipData
, который совместим с реализациями View
.
Укажите источник перетаскивания
Чтобы включить события перетаскивания внутри компонента, добавьте модификатор dragAndDropSource
. Он принимает функцию в качестве параметра. Внутри этой функции используйте DragAndDropTransferData
для представления передаваемых данных. Данные могут быть удаленным URI, данными форматированного текста в буфере обмена, локальным файлом или чем-то еще, но все они должны быть обернуты в объект ClipData
. Укажите простой текст, например, следующим образом:
Modifier.dragAndDropSource { _ -> DragAndDropTransferData( ClipData.newPlainText( "image Url", url ) ) }
Чтобы разрешить действию перетаскивания пересекать границы приложения, конструктор DragAndDropTransferData
принимает аргумент flags
. В следующем примере константа DRAG_FLAG_GLOBAL
указывает, что данные можно перетаскивать из одного приложения в другое:
Modifier.dragAndDropSource { _ -> DragAndDropTransferData( ClipData.newPlainText( "image Url", url ), flags = View.DRAG_FLAG_GLOBAL ) }
DragAndDropTransferData
принимает флаги, поддерживаемые системой Android View. Полный список доступных флагов см. в списке констант View .
Получение данных о сбросе
Назначьте модификатор dragAndDropTarget
компонуемому, чтобы компонуемый мог получать события перетаскивания. Модификатор имеет два параметра: первый действует как фильтр и указывает тип данных, которые модификатор может принимать, а второй доставляет данные в обратном вызове.
Обратите внимание, что экземпляр обратного вызова должен быть запомнен . Следующий фрагмент показывает, как запомнить обратный вызов:
val callback = remember { object : DragAndDropTarget { override fun onDrop(event: DragAndDropEvent): Boolean { // Parse received data return true } } }
Чтобы принимать данные из других приложений, используйте requestDragAndDropPermission
, чтобы включить прием, и отпустите его после завершения:
val externalAppCallback = remember { object : DragAndDropTarget { override fun onDrop(event: DragAndDropEvent): Boolean { val permission = activity.requestDragAndDropPermissions(event.toAndroidDragEvent()) // Parse received data permission?.release() return true } } }
Важно помнить, что DragAndDropEvent
, возвращаемый методом обратного вызова Compose, отличается от ожидаемого методом requestDragAndDropPermission
, поэтому необходимо вызвать функцию расширения toAndroidDragEvent()
для параметра, прежде чем передавать его в запрос разрешения.
В следующем фрагменте показано, как обрабатывать перетаскиваемый простой текст:
Modifier.dragAndDropTarget( shouldStartDragAndDrop = { event -> event.mimeTypes().contains(ClipDescription.MIMETYPE_TEXT_PLAIN) }, target = callback // or externalAppCallback )
Функция обратного вызова должна возвращать true
если событие обработано, или false
если событие отклонено и не распространяется на родительский компонент.
Обработка событий перетаскивания
Переопределите обратные вызовы в интерфейсе DragAndDropTarget
, чтобы отслеживать, когда событие перетаскивания начинается, заканчивается или входит в компонент или выходит из него, для точного управления пользовательским интерфейсом и поведением приложения:
object : DragAndDropTarget { override fun onStarted(event: DragAndDropEvent) { // When the drag event starts } override fun onEntered(event: DragAndDropEvent) { // When the dragged object enters the target surface } override fun onEnded(event: DragAndDropEvent) { // When the drag event stops } override fun onExited(event: DragAndDropEvent) { // When the dragged object exits the target surface } override fun onDrop(event: DragAndDropEvent): Boolean = true }
Дополнительные ресурсы
Codelab: Перетаскивание в Compose