拖动 UI 元素
拖放是移动应用程序中常见的交互方式。当用户长按(有时也称为触摸并按住)某个组件时,另一个组件会出现在用户手指下方,用户将该组件拖动到最终位置并释放。在这个示例中,您将构建一个拖放交互,用户长按选择的食物,然后将该食物拖动到付款顾客的照片上。
以下动画展示了应用程序的行为
本示例从一个预先构建的菜单项列表和一行顾客开始。第一步是识别长按并显示可拖动的菜单项照片。
按住并拖动
#Flutter 提供了一个名为 LongPressDraggable
的组件,它提供了开始拖放交互所需的确切行为。LongPressDraggable
组件会识别长按事件的发生,然后在用户手指附近显示一个新组件。当用户拖动时,该组件会跟随用户的手指移动。LongPressDraggable
让您可以完全控制用户拖动的组件。
每个菜单列表项都通过自定义的 MenuListItem
组件显示。
MenuListItem(
name: item.name,
price: item.formattedTotalItemPrice,
photoProvider: item.imageProvider,
)
将 MenuListItem
组件包装在 LongPressDraggable
组件中。
LongPressDraggable<Item>(
data: item,
dragAnchorStrategy: pointerDragAnchorStrategy,
feedback: DraggingListItem(
dragKey: _draggableKey,
photoProvider: item.imageProvider,
),
child: MenuListItem(
name: item.name,
price: item.formattedTotalItemPrice,
photoProvider: item.imageProvider,
),
);
在这种情况下,当用户长按 MenuListItem
组件时,LongPressDraggable
组件会显示一个 DraggingListItem
。这个 DraggingListItem
会显示所选食物的照片,并居中在用户手指下方。
dragAnchorStrategy
属性被设置为 pointerDragAnchorStrategy
。此属性值指示 LongPressDraggable
将 DraggableListItem
的位置基于用户手指的位置。当用户移动手指时,DraggableListItem
会随之移动。
如果项目被拖放时没有信息传输,那么拖放的用处不大。因此,LongPressDraggable
接受一个 data
参数。在这种情况下,data
的类型是 Item
,它包含用户长按的食物菜单项的信息。
与 LongPressDraggable
相关联的 data
被发送到一个名为 DragTarget
的特殊组件,用户在该组件上释放拖动手势。接下来您将实现拖放行为。
放下可拖动项
#用户可以选择在任何地方放下 LongPressDraggable
,但除非它被放置在 DragTarget
上方,否则放下可拖动项不会产生任何效果。当用户将可拖动项放置在 DragTarget
组件上方时,DragTarget
组件可以接受或拒绝来自可拖动项的数据。
在本示例中,用户应该将菜单项拖放到 CustomerCart
组件上,以将菜单项添加到用户的购物车中。
CustomerCart(
hasItems: customer.items.isNotEmpty,
highlighted: candidateItems.isNotEmpty,
customer: customer,
);
将 CustomerCart
组件包装在 DragTarget
组件中。
DragTarget<Item>(
builder: (context, candidateItems, rejectedItems) {
return CustomerCart(
hasItems: customer.items.isNotEmpty,
highlighted: candidateItems.isNotEmpty,
customer: customer,
);
},
onAcceptWithDetails: (details) {
_itemDroppedOnCustomerCart(item: details.data, customer: customer);
},
)
DragTarget
显示您现有的组件,并与 LongPressDraggable
协调,以识别用户何时将可拖动项拖到 DragTarget
上方。DragTarget
还识别用户何时将可拖动项放置在 DragTarget
组件上方。
当用户在 DragTarget
组件上拖动可拖动项时,candidateItems
包含用户正在拖动的数据项。此可拖动项允许您更改组件在用户拖动其上方时的外观。在这种情况下,当任何项目被拖动到 DragTarget
组件上方时,Customer
组件都会变为红色。红色的视觉外观通过 CustomerCart
组件中的 highlighted
属性进行配置。
当用户将可拖动项放置在 DragTarget
组件上时,会调用 onAcceptWithDetails
回调。此时您可以决定是否接受放置的数据。在这种情况下,项目总是被接受并处理。您也可以选择检查传入的项目以做出不同的决定。
请注意,放置在 DragTarget
上的项目类型必须与从 LongPressDraggable
拖动的项目类型匹配。如果类型不兼容,则不会调用 onAcceptWithDetails
方法。
配置好 DragTarget
组件以接受您所需的数据后,您现在可以通过拖放将数据从 UI 的一部分传输到另一部分。
在下一步中,您将使用拖放的菜单项更新顾客的购物车。
将菜单项添加到购物车
#每个顾客都由一个 Customer
对象表示,该对象维护着一个商品购物车和总价。
class Customer {
Customer({required this.name, required this.imageProvider, List<Item>? items})
: items = items ?? [];
final String name;
final ImageProvider imageProvider;
final List<Item> items;
String get formattedTotalItemPrice {
final totalPriceCents = items.fold<int>(
0,
(prev, item) => prev + item.totalPriceCents,
);
return '\$${(totalPriceCents / 100.0).toStringAsFixed(2)}';
}
}
CustomerCart
组件根据 Customer
实例显示顾客的照片、姓名、总价和商品数量。
要在菜单项被放下时更新顾客的购物车,请将放置的商品添加到关联的 Customer
对象中。
void _itemDroppedOnCustomerCart({
required Item item,
required Customer customer,
}) {
setState(() {
customer.items.add(item);
});
}
当用户将菜单项放置在 CustomerCart
组件上时,会在 onAcceptWithDetails()
中调用 _itemDroppedOnCustomerCart
方法。通过将放置的商品添加到 customer
对象,并调用 setState()
触发布局更新,UI 会刷新并显示新的顾客总价和商品数量。
恭喜!您已经实现了将食物项添加到顾客购物车的拖放交互。
互动示例
#运行应用
- 滚动浏览食物项。
- 用手指长按或用鼠标点击并按住。
- 按住时,食物项的图片将出现在列表上方。
- 拖动图片并将其放置在屏幕底部的人物之一上。图片下方的文本会更新,以反映该人物的费用。您可以继续添加食物项并查看费用累计。
import 'package:flutter/material.dart';
void main() {
runApp(
const MaterialApp(
home: ExampleDragAndDrop(),
debugShowCheckedModeBanner: false,
),
);
}
const List<Item> _items = [
Item(
name: 'Spinach Pizza',
totalPriceCents: 1299,
uid: '1',
imageProvider: NetworkImage(
'https://docs.flutterdart.cn'
'/cookbook/img-files/effects/split-check/Food1.jpg',
),
),
Item(
name: 'Veggie Delight',
totalPriceCents: 799,
uid: '2',
imageProvider: NetworkImage(
'https://docs.flutterdart.cn'
'/cookbook/img-files/effects/split-check/Food2.jpg',
),
),
Item(
name: 'Chicken Parmesan',
totalPriceCents: 1499,
uid: '3',
imageProvider: NetworkImage(
'https://docs.flutterdart.cn'
'/cookbook/img-files/effects/split-check/Food3.jpg',
),
),
];
@immutable
class ExampleDragAndDrop extends StatefulWidget {
const ExampleDragAndDrop({super.key});
@override
State<ExampleDragAndDrop> createState() => _ExampleDragAndDropState();
}
class _ExampleDragAndDropState extends State<ExampleDragAndDrop>
with TickerProviderStateMixin {
final List<Customer> _people = [
Customer(
name: 'Makayla',
imageProvider: const NetworkImage(
'https://docs.flutterdart.cn'
'/cookbook/img-files/effects/split-check/Avatar1.jpg',
),
),
Customer(
name: 'Nathan',
imageProvider: const NetworkImage(
'https://docs.flutterdart.cn'
'/cookbook/img-files/effects/split-check/Avatar2.jpg',
),
),
Customer(
name: 'Emilio',
imageProvider: const NetworkImage(
'https://docs.flutterdart.cn'
'/cookbook/img-files/effects/split-check/Avatar3.jpg',
),
),
];
final GlobalKey _draggableKey = GlobalKey();
void _itemDroppedOnCustomerCart({
required Item item,
required Customer customer,
}) {
setState(() {
customer.items.add(item);
});
}
@override
Widget build(BuildContext context) {
return Scaffold(
backgroundColor: const Color(0xFFF7F7F7),
appBar: _buildAppBar(),
body: _buildContent(),
);
}
PreferredSizeWidget _buildAppBar() {
return AppBar(
iconTheme: const IconThemeData(color: Color(0xFFF64209)),
title: Text(
'Order Food',
style: Theme.of(context).textTheme.headlineMedium?.copyWith(
fontSize: 36,
color: const Color(0xFFF64209),
fontWeight: FontWeight.bold,
),
),
backgroundColor: const Color(0xFFF7F7F7),
elevation: 0,
);
}
Widget _buildContent() {
return Stack(
children: [
SafeArea(
child: Column(
children: [
Expanded(child: _buildMenuList()),
_buildPeopleRow(),
],
),
),
],
);
}
Widget _buildMenuList() {
return ListView.separated(
padding: const EdgeInsets.all(16),
itemCount: _items.length,
separatorBuilder: (context, index) {
return const SizedBox(height: 12);
},
itemBuilder: (context, index) {
final item = _items[index];
return _buildMenuItem(item: item);
},
);
}
Widget _buildMenuItem({required Item item}) {
return LongPressDraggable<Item>(
data: item,
dragAnchorStrategy: pointerDragAnchorStrategy,
feedback: DraggingListItem(
dragKey: _draggableKey,
photoProvider: item.imageProvider,
),
child: MenuListItem(
name: item.name,
price: item.formattedTotalItemPrice,
photoProvider: item.imageProvider,
),
);
}
Widget _buildPeopleRow() {
return Container(
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 20),
child: Row(children: _people.map(_buildPersonWithDropZone).toList()),
);
}
Widget _buildPersonWithDropZone(Customer customer) {
return Expanded(
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 6),
child: DragTarget<Item>(
builder: (context, candidateItems, rejectedItems) {
return CustomerCart(
hasItems: customer.items.isNotEmpty,
highlighted: candidateItems.isNotEmpty,
customer: customer,
);
},
onAcceptWithDetails: (details) {
_itemDroppedOnCustomerCart(item: details.data, customer: customer);
},
),
),
);
}
}
class CustomerCart extends StatelessWidget {
const CustomerCart({
super.key,
required this.customer,
this.highlighted = false,
this.hasItems = false,
});
final Customer customer;
final bool highlighted;
final bool hasItems;
@override
Widget build(BuildContext context) {
final textColor = highlighted ? Colors.white : Colors.black;
return Transform.scale(
scale: highlighted ? 1.075 : 1.0,
child: Material(
elevation: highlighted ? 8 : 4,
borderRadius: BorderRadius.circular(22),
color: highlighted ? const Color(0xFFF64209) : Colors.white,
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 24),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
ClipOval(
child: SizedBox(
width: 46,
height: 46,
child: Image(
image: customer.imageProvider,
fit: BoxFit.cover,
),
),
),
const SizedBox(height: 8),
Text(
customer.name,
style: Theme.of(context).textTheme.titleMedium?.copyWith(
color: textColor,
fontWeight: hasItems ? FontWeight.normal : FontWeight.bold,
),
),
Visibility(
visible: hasItems,
maintainState: true,
maintainAnimation: true,
maintainSize: true,
child: Column(
children: [
const SizedBox(height: 4),
Text(
customer.formattedTotalItemPrice,
style: Theme.of(context).textTheme.bodySmall!.copyWith(
color: textColor,
fontSize: 16,
fontWeight: FontWeight.bold,
),
),
const SizedBox(height: 4),
Text(
'${customer.items.length} item${customer.items.length != 1 ? 's' : ''}',
style: Theme.of(context).textTheme.titleMedium!.copyWith(
color: textColor,
fontSize: 12,
),
),
],
),
),
],
),
),
),
);
}
}
class MenuListItem extends StatelessWidget {
const MenuListItem({
super.key,
this.name = '',
this.price = '',
required this.photoProvider,
this.isDepressed = false,
});
final String name;
final String price;
final ImageProvider photoProvider;
final bool isDepressed;
@override
Widget build(BuildContext context) {
return Material(
elevation: 12,
borderRadius: BorderRadius.circular(20),
child: Padding(
padding: const EdgeInsets.all(12),
child: Row(
mainAxisSize: MainAxisSize.max,
children: [
ClipRRect(
borderRadius: BorderRadius.circular(12),
child: SizedBox(
width: 120,
height: 120,
child: Center(
child: AnimatedContainer(
duration: const Duration(milliseconds: 100),
curve: Curves.easeInOut,
height: isDepressed ? 115 : 120,
width: isDepressed ? 115 : 120,
child: Image(image: photoProvider, fit: BoxFit.cover),
),
),
),
),
const SizedBox(width: 30),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
name,
style: Theme.of(
context,
).textTheme.titleMedium?.copyWith(fontSize: 18),
),
const SizedBox(height: 10),
Text(
price,
style: Theme.of(context).textTheme.titleMedium?.copyWith(
fontWeight: FontWeight.bold,
fontSize: 18,
),
),
],
),
),
],
),
),
);
}
}
class DraggingListItem extends StatelessWidget {
const DraggingListItem({
super.key,
required this.dragKey,
required this.photoProvider,
});
final GlobalKey dragKey;
final ImageProvider photoProvider;
@override
Widget build(BuildContext context) {
return FractionalTranslation(
translation: const Offset(-0.5, -0.5),
child: ClipRRect(
key: dragKey,
borderRadius: BorderRadius.circular(12),
child: SizedBox(
height: 150,
width: 150,
child: Opacity(
opacity: 0.85,
child: Image(image: photoProvider, fit: BoxFit.cover),
),
),
),
);
}
}
@immutable
class Item {
const Item({
required this.totalPriceCents,
required this.name,
required this.uid,
required this.imageProvider,
});
final int totalPriceCents;
final String name;
final String uid;
final ImageProvider imageProvider;
String get formattedTotalItemPrice =>
'\$${(totalPriceCents / 100.0).toStringAsFixed(2)}';
}
class Customer {
Customer({required this.name, required this.imageProvider, List<Item>? items})
: items = items ?? [];
final String name;
final ImageProvider imageProvider;
final List<Item> items;
String get formattedTotalItemPrice {
final totalPriceCents = items.fold<int>(
0,
(prev, item) => prev + item.totalPriceCents,
);
return '\$${(totalPriceCents / 100.0).toStringAsFixed(2)}';
}
}