1
1
---
2
2
# title: Drag a UI element
3
3
title : 创建一个可拖放的 UI 组件
4
- description : How to implement a draggable UI element.
4
+ # description: How to implement a draggable UI element.
5
+ description : 如何实现一个可拖放的 UI 组件
5
6
js :
6
7
- defer : true
7
8
url : /assets/js/inject_dartpad.js
@@ -19,17 +20,32 @@ where the user long presses on a choice of food,
19
20
and then drags that food to the picture of the customer who
20
21
is paying for it.
21
22
23
+ 拖放是移动应用程序中的一种常见交互。
24
+ 当用户长按(有时称为“触摸并按住”)某个 widget 时,
25
+ 另一个 widget 会出现在用户的手指下方,
26
+ 用户可以将该 widget 拖动到最终位置并释放。
27
+ 在这个示例中,你将构建一个拖放交互,
28
+ 用户可以长按一个食物选项,
29
+ 然后将该食物拖动到正在为其付款的顾客图片上。
30
+
22
31
The following animation shows the app's behavior:
23
32
33
+ 下面的动画展示了应用程序的行为:
34
+
24
35
![ Ordering the food by dragging it to the person] ( /assets/images/docs/cookbook/effects/DragAUIElement.gif ) {:.site-mobile-screenshot}
25
36
26
37
This recipe begins with a prebuilt list of menu items and
27
38
a row of customers.
28
39
The first step is to recognize a long press
29
40
and display a draggable photo of a menu item.
30
41
42
+ 本示例从一个预先构建的菜单项列表和一排顾客开始。
43
+ 第一步是识别长按操作并显示一个可拖放的菜单项图片。
44
+
31
45
## Press and drag
32
46
47
+ ## 按下并拖动
48
+
33
49
Flutter provides a widget called [ ` LongPressDraggable ` ] [ ]
34
50
that provides the exact behavior that you need to begin
35
51
a drag-and-drop interaction. A ` LongPressDraggable `
@@ -39,9 +55,18 @@ As the user drags, the widget follows the user's finger.
39
55
` LongPressDraggable ` gives you full control over the
40
56
widget that the user drags.
41
57
58
+ Flutter 提供了一个名为 [ ` LongPressDraggable ` ] [ ] 的 widget,
59
+ 它提供了开始拖放交互所需的确切行为。
60
+ ` LongPressDraggable ` widget 能够识别长按操作,
61
+ 然后在用户手指附近显示一个新 widget。
62
+ 当用户拖动时,该 widget 会跟随用户的手指移动。
63
+ ` LongPressDraggable ` 让你完全控制用户拖动的 widget。
64
+
42
65
Each menu list item is displayed with a custom
43
66
` MenuListItem ` widget.
44
67
68
+ 每个菜单列表项都通过一个自定义的 ` MenuListItem ` widget 来显示。
69
+
45
70
<? code-excerpt "lib/main.dart (MenuListItem)" replace="/^child: //g;/^\),$/)/g"?>
46
71
``` dart
47
72
MenuListItem(
@@ -53,6 +78,8 @@ MenuListItem(
53
78
54
79
Wrap the ` MenuListItem ` widget with a ` LongPressDraggable ` widget.
55
80
81
+ 用 ` LongPressDraggable ` widget 包裹 ` MenuLIstItem ` widget。
82
+
56
83
<? code-excerpt "lib/main.dart (LongPressDraggable)" replace="/^return //g;/^\),$/)/g"?>
57
84
``` dart
58
85
LongPressDraggable<Item>(
@@ -77,36 +104,65 @@ This `DraggingListItem` displays a photo of the
77
104
selected food item, centered beneath
78
105
the user's finger.
79
106
107
+ 在本例中,当用户长按 ` MenuListItem ` widget 时,
108
+ ` LongPressDraggable ` widget 会显示一个 ` DraggingListItem ` 。
109
+ 这个 ` DraggingListItem ` 会在用户手指下方居中显示所选的食物图片。
110
+
80
111
The ` dragAnchorStrategy ` property is set to
81
112
[ ` pointerDragAnchorStrategy ` ] [ ] .
82
113
This property value instructs ` LongPressDraggable `
83
114
to base the ` DraggableListItem ` 's position on the
84
115
user's finger. As the user moves a finger,
85
116
the ` DraggableListItem ` moves with it.
86
117
118
+ ` dragAnchorStrategy ` 属性设置为 [ ` pointerDragAnchorStrategy ` ] [ ] 。
119
+ 这个属性的值指示 ` LongPressDraggable `
120
+ 将 ` DraggableListItem ` 的位置基于用户的手指来定位。
121
+ 当用户移动手指时,` DraggableListItem ` 也会跟随移动。
122
+
87
123
Dragging and dropping is of little use if no information
88
124
is transmitted when the item is dropped.
89
125
For this reason, ` LongPressDraggable ` takes a ` data ` parameter.
90
126
In this case, the type of ` data ` is ` Item ` ,
91
127
which holds information about the
92
128
food menu item that the user pressed on.
93
129
130
+ 如果拖放操作在释放时没有传递任何信息,那么它几乎没有什么用处。
131
+ 为此,` LongPressDraggable ` 提供了一个 ` data ` 参数。
132
+ 在本例中,` data ` 的类型是 ` Item ` ,
133
+ 它包含了用户按下的菜单项的食物信息。
134
+
94
135
The ` data ` associated with a ` LongPressDraggable `
95
136
is sent to a special widget called ` DragTarget ` ,
96
137
where the user releases the drag gesture.
97
138
You'll implement the drop behavior next.
98
139
140
+ 与 ` LongPressDraggable ` 相关联的 ` data `
141
+ 会被传递到一个名为 ` DragTarget ` 的特殊 widget 上,
142
+ 用户在此释放拖动手势。接下来你将实现拖放行为。
143
+
99
144
## Drop the draggable
100
145
146
+ ## 放下拖动项
147
+
101
148
The user can drop a ` LongPressDraggable ` wherever they choose,
102
149
but dropping the draggable has no effect unless it's dropped
103
150
on top of a ` DragTarget ` . When the user drops a draggable on
104
151
top of a ` DragTarget ` widget, the ` DragTarget ` widget
105
152
can either accept or reject the data from the draggable.
106
153
154
+ 用户可以将 ` LongPressDraggable ` 放在任意位置,
155
+ 但除非将其放在 ` DragTarget ` 之上,
156
+ 否则拖放操作不会有任何效果。
157
+ 当用户将一个可拖放的 widget 放在 ` DragTarget ` widget 上时,
158
+ ` DragTarget ` 可以选择接受或拒绝来自可拖放 widget 的数据。
159
+
107
160
In this recipe, the user should drop a menu item on a
108
161
` CustomerCart ` widget to add the menu item to the user's cart.
109
162
163
+ 在本示例中,用户应将菜单项拖放到 ` CustomerCart ` widget 上,
164
+ 从而将菜单项添加到用户的购物车中。
165
+
110
166
<? code-excerpt "lib/main.dart (CustomerCart)" replace="/^return //g;/^\),$/)/g"?>
111
167
``` dart
112
168
CustomerCart(
@@ -118,6 +174,8 @@ CustomerCart(
118
174
119
175
Wrap the ` CustomerCart ` widget with a ` DragTarget ` widget.
120
176
177
+ 用 ` DragTarget ` widget 包裹 ` CustomCart ` widget。
178
+
121
179
<? code-excerpt "lib/main.dart (DragTarget)" replace="/^child: //g;/^\),$/)/g"?>
122
180
``` dart
123
181
DragTarget<Item>(
@@ -143,6 +201,11 @@ when the user drags a draggable on top of the `DragTarget`.
143
201
The ` DragTarget ` also recognizes when the user drops
144
202
a draggable on top of the ` DragTarget ` widget.
145
203
204
+ ` DragTarget ` 显示你现有的 widget,
205
+ 并与 ` LongPressDraggable ` 协调,
206
+ 识别用户何时将可拖放的 widget 拖动到 ` DragTarget ` 之上。
207
+ 同时,` DragTarget ` 也能识别用户何时在其上方放下一个可拖放的 widget。
208
+
146
209
When the user drags a draggable on the ` DragTarget ` widget,
147
210
` candidateItems ` contains the data items that the user is dragging.
148
211
This draggable allows you to change what your widget looks
@@ -151,30 +214,53 @@ the `Customer` widget turns red whenever any items are dragged above the
151
214
` DragTarget ` widget. The red visual appearance is configured with the
152
215
` highlighted ` property within the ` CustomerCart ` widget.
153
216
217
+ 当用户将可一个可拖放的 widget 拖动到 ` DragTarget ` widget 上时,
218
+ ` candidateItems ` 包含用户正在拖动的数据项。
219
+ 你可以根据用户拖动的情况更改 widget 的外观。
220
+ 在本例中,` Customer ` widget 在有项目拖动到 ` DragTarget ` widget 上方时会变成红色。
221
+ 这个红色的外观是通过 ` CustomerCart ` widget 中的 ` highlighted ` 属性配置的。
222
+
154
223
When the user drops a draggable on the ` DragTarget ` widget,
155
224
the ` onAcceptWithDetails ` callback is invoked. This is when you get
156
225
to decide whether or not to accept the data that was dropped.
157
226
In this case, the item is always accepted and processed.
158
227
You might choose to inspect the incoming item to make a
159
228
different decision.
160
229
230
+ 当用户将一个可拖放的 widget 放在 ` DragTarget ` widget 上时,
231
+ 会调用 ` onAcceptWithDetails ` 回调。这时你可以决定是否接受拖放的数据。
232
+ 在本例中,数据项总是会被接受和处理。
233
+ 你也可以选择检查传入的数据项,以做出不同的决定。
234
+
161
235
Notice that the type of item dropped on ` DragTarget `
162
236
must match the type of the item dragged from ` LongPressDraggable ` .
163
237
If the types are not compatible, then
164
238
the ` onAcceptWithDetails ` method isn't invoked.
165
239
240
+ 注意,拖放到 ` DragTarget ` 上的数据项类型必须与从 ` LongPressDraggable ` 拖出的数据项类型匹配。
241
+ 如果类型不兼容,` onAcceptWithDetails ` 方法将不会被调用。
242
+
166
243
With a ` DragTarget ` widget configured to accept your
167
244
desired data, you can now transmit data from one part
168
245
of your UI to another by dragging and dropping.
169
246
247
+ 通过配置 ` DragTarget ` widget 来接受你所需的数据,
248
+ 现在你可以通过拖放在 UI 的不同部分之间传递数据。
249
+
170
250
In the next step,
171
251
you update the customer's cart with the dropped menu item.
172
252
253
+ 在下一步中,你将更新顾客的购物车,将放下的菜单项添加进去。
254
+
173
255
## Add a menu item to a cart
174
256
257
+ ## 将菜单项添加到购物车
258
+
175
259
Each customer is represented by a ` Customer ` object,
176
260
which maintains a cart of items and a price total.
177
261
262
+ 每位顾客由一个 ` Customer ` 对象表示,该对象维护一个物品购物车和总价格。
263
+
178
264
<? code-excerpt "lib/main.dart (CustomerClass)"?>
179
265
``` dart
180
266
class Customer {
@@ -199,9 +285,13 @@ class Customer {
199
285
The ` CustomerCart ` widget displays the customer's photo,
200
286
name, total, and item count based on a ` Customer ` instance.
201
287
288
+ ` CustomerCart ` widget 根据一个 ` Customer ` 实例显示顾客的照片、姓名、总价和物品数量。
289
+
202
290
To update a customer's cart when a menu item is dropped,
203
291
add the dropped item to the associated ` Customer ` object.
204
292
293
+ 要在菜单项被拖放时更新顾客的购物车,需要将放下的物品添加到关联的 ` Customer ` 对象中。
294
+
205
295
<? code-excerpt "lib/main.dart (AddCart)"?>
206
296
``` dart
207
297
void _itemDroppedOnCustomerCart({
@@ -221,26 +311,51 @@ The `_itemDroppedOnCustomerCart` method is invoked in
221
311
layout update, the UI refreshes with the new customer's
222
312
price total and item count.
223
313
314
+ 当用户将菜单项拖放到 ` CustomerCart ` widget 上时,
315
+ ` onAcceptWithDetails() ` 中会调用 ` _itemDroppedOnCustomerCart ` 方法。
316
+ 通过将放下的物品添加到 ` customer ` 对象中,
317
+ 并调用 ` setState() ` 来触发布局更新,
318
+ UI 将会刷新,显示新的顾客总价和物品数量。
319
+
224
320
Congratulations! You have a drag-and-drop interaction
225
321
that adds food items to a customer's shopping cart.
226
322
323
+ 恭喜!你现在已经实现了一个将食物项添加到顾客购物车的拖放交互。
324
+
227
325
## Interactive example
228
326
327
+ ## 交互示例
328
+
229
329
Run the app:
230
330
331
+ 运行应用程序
332
+
231
333
* Scroll through the food items.
334
+
335
+ 浏览食物项列表。
336
+
232
337
* Press and hold on one with your
233
338
finger or click and hold with the
234
339
mouse.
340
+
341
+ 用手指长按其中一个食物项,或用鼠标点击并按住。
342
+
235
343
* While holding, the food item's image
236
344
will appear above the list.
345
+
346
+ 按住时,食物项的图片将出现在列表上方。
347
+
237
348
* Drag the image and drop it on one of the
238
349
people at the bottom of the screen.
239
350
The text under the image updates to
240
351
reflect the charge for that person.
241
352
You can continue to add food items
242
353
and watch the charges accumulate.
243
354
355
+ 将图片拖动并放到屏幕底部的某个人身上。
356
+ 图片下方的文本会更新,显示该人的费用。
357
+ 你可以继续添加食物项,观察费用的累计情况。
358
+
244
359
<!-- Start DartPad -->
245
360
246
361
<? code-excerpt "lib/main.dart"?>
0 commit comments