如何在颤动中创建一个可移动的小部件,使其停留在被拖动到的位置
How to create a movable widget in flutter such that is stays at the position it is dragged to
我如何在 flutter 中创建一个 movable/draggable 小部件,使其保持在被拖动到的位置。我尝试使用可拖动小部件,但被包裹在可拖动 returns 中的小部件回到原来的位置释放拖动后的位置。
您需要使用某种状态管理来管理拖动项的位置。在下面的代码示例中,我使用了 Flutter Hooks useState
.
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
const imgData =
'';
void main() {
runApp(
MaterialApp(
title: 'Flutter Demo',
home: Scaffold(
body: DragArea(
child: Image.network(imgData),
),
),
),
);
}
class DragArea extends HookWidget {
final Widget child;
const DragArea({Key key, this.child}) : super(key: key);
@override
Widget build(BuildContext context) {
final position = useState(Offset(100, 100));
return Stack(
children: [
Positioned(
left: position.value.dx,
top: position.value.dy,
child: Draggable(
feedback: child,
childWhenDragging: Opacity(
opacity: .3,
child: child,
),
onDragEnd: (details) => position.value = details.offset,
child: child,
),
)
],
);
}
}
有状态版本
根据 anikait 的要求
class StatefulDragArea extends StatefulWidget {
final Widget child;
const StatefulDragArea({Key key, this.child}) : super(key: key);
@override
_DragAreaStateStateful createState() => _DragAreaStateStateful();
}
class _DragAreaStateStateful extends State<StatefulDragArea> {
Offset position = Offset(100, 100);
@override
Widget build(BuildContext context) {
return Stack(
children: [
Positioned(
left: position.dx,
top: position.dy,
child: Draggable(
feedback: widget.child,
childWhenDragging: Opacity(
opacity: .3,
child: widget.child,
),
onDragEnd: (details) => setState(() => position = details.offset),
child: widget.child,
),
)
],
);
}
}
除了四处拖动对象之外,您还可以借助 GestureDetector
使其可缩放。我将 GestureDetector
应用于主堆栈,这样您就可以在屏幕上的任何位置捏合缩放 in/out。它使您更容易看到自己在做什么。
HookWidget 版本
class DragArea extends HookWidget {
final Widget child;
const DragArea({Key key, this.child}) : super(key: key);
@override
Widget build(BuildContext context) {
final position = useState(Offset(100, 100));
final prevScale = useState(1.0);
final scale = useState(1.0);
return GestureDetector(
onScaleUpdate: (details) => scale.value = prevScale.value * details.scale,
onScaleEnd: (_) => prevScale.value = scale.value,
child: Stack(
children: [
Positioned.fill(
child: Container(color: Colors.amber.withOpacity(.4))),
Positioned(
left: position.value.dx,
top: position.value.dy,
child: Draggable(
maxSimultaneousDrags: 1,
feedback: Transform.scale(
scale: scale.value,
child: child,
),
childWhenDragging: Opacity(
opacity: .3,
child: Transform.scale(
scale: scale.value,
child: child,
),
),
onDragEnd: (details) => position.value = details.offset,
child: Transform.scale(
scale: scale.value,
child: child,
),
),
)
],
),
);
}
}
StatefulWidget 版本
class StatefulDragArea extends StatefulWidget {
final Widget child;
const StatefulDragArea({Key key, this.child}) : super(key: key);
@override
_DragAreaStateStateful createState() => _DragAreaStateStateful();
}
class _DragAreaStateStateful extends State<StatefulDragArea> {
Offset position = Offset(100, 100);
double prevScale = 1;
double scale = 1;
void updateScale(double zoom) => setState(() => scale = prevScale * zoom);
void commitScale() => setState(() => prevScale = scale);
void updatePosition(Offset newPosition) =>
setState(() => position = newPosition);
@override
Widget build(BuildContext context) {
return GestureDetector(
onScaleUpdate: (details) => updateScale(details.scale),
onScaleEnd: (_) => commitScale(),
child: Stack(
children: [
Positioned.fill(
child: Container(color: Colors.amber.withOpacity(.4))),
Positioned(
left: position.dx,
top: position.dy,
child: Draggable(
maxSimultaneousDrags: 1,
feedback: widget.child,
childWhenDragging: Opacity(
opacity: .3,
child: widget.child,
),
onDragEnd: (details) => updatePosition(details.offset),
child: Transform.scale(
scale: scale,
child: widget.child,
),
),
),
],
),
);
}
}
我如何在 flutter 中创建一个 movable/draggable 小部件,使其保持在被拖动到的位置。我尝试使用可拖动小部件,但被包裹在可拖动 returns 中的小部件回到原来的位置释放拖动后的位置。
您需要使用某种状态管理来管理拖动项的位置。在下面的代码示例中,我使用了 Flutter Hooks useState
.
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
const imgData =
'';
void main() {
runApp(
MaterialApp(
title: 'Flutter Demo',
home: Scaffold(
body: DragArea(
child: Image.network(imgData),
),
),
),
);
}
class DragArea extends HookWidget {
final Widget child;
const DragArea({Key key, this.child}) : super(key: key);
@override
Widget build(BuildContext context) {
final position = useState(Offset(100, 100));
return Stack(
children: [
Positioned(
left: position.value.dx,
top: position.value.dy,
child: Draggable(
feedback: child,
childWhenDragging: Opacity(
opacity: .3,
child: child,
),
onDragEnd: (details) => position.value = details.offset,
child: child,
),
)
],
);
}
}
有状态版本
根据 anikait 的要求
class StatefulDragArea extends StatefulWidget {
final Widget child;
const StatefulDragArea({Key key, this.child}) : super(key: key);
@override
_DragAreaStateStateful createState() => _DragAreaStateStateful();
}
class _DragAreaStateStateful extends State<StatefulDragArea> {
Offset position = Offset(100, 100);
@override
Widget build(BuildContext context) {
return Stack(
children: [
Positioned(
left: position.dx,
top: position.dy,
child: Draggable(
feedback: widget.child,
childWhenDragging: Opacity(
opacity: .3,
child: widget.child,
),
onDragEnd: (details) => setState(() => position = details.offset),
child: widget.child,
),
)
],
);
}
}
除了四处拖动对象之外,您还可以借助 GestureDetector
使其可缩放。我将 GestureDetector
应用于主堆栈,这样您就可以在屏幕上的任何位置捏合缩放 in/out。它使您更容易看到自己在做什么。
HookWidget 版本
class DragArea extends HookWidget {
final Widget child;
const DragArea({Key key, this.child}) : super(key: key);
@override
Widget build(BuildContext context) {
final position = useState(Offset(100, 100));
final prevScale = useState(1.0);
final scale = useState(1.0);
return GestureDetector(
onScaleUpdate: (details) => scale.value = prevScale.value * details.scale,
onScaleEnd: (_) => prevScale.value = scale.value,
child: Stack(
children: [
Positioned.fill(
child: Container(color: Colors.amber.withOpacity(.4))),
Positioned(
left: position.value.dx,
top: position.value.dy,
child: Draggable(
maxSimultaneousDrags: 1,
feedback: Transform.scale(
scale: scale.value,
child: child,
),
childWhenDragging: Opacity(
opacity: .3,
child: Transform.scale(
scale: scale.value,
child: child,
),
),
onDragEnd: (details) => position.value = details.offset,
child: Transform.scale(
scale: scale.value,
child: child,
),
),
)
],
),
);
}
}
StatefulWidget 版本
class StatefulDragArea extends StatefulWidget {
final Widget child;
const StatefulDragArea({Key key, this.child}) : super(key: key);
@override
_DragAreaStateStateful createState() => _DragAreaStateStateful();
}
class _DragAreaStateStateful extends State<StatefulDragArea> {
Offset position = Offset(100, 100);
double prevScale = 1;
double scale = 1;
void updateScale(double zoom) => setState(() => scale = prevScale * zoom);
void commitScale() => setState(() => prevScale = scale);
void updatePosition(Offset newPosition) =>
setState(() => position = newPosition);
@override
Widget build(BuildContext context) {
return GestureDetector(
onScaleUpdate: (details) => updateScale(details.scale),
onScaleEnd: (_) => commitScale(),
child: Stack(
children: [
Positioned.fill(
child: Container(color: Colors.amber.withOpacity(.4))),
Positioned(
left: position.dx,
top: position.dy,
child: Draggable(
maxSimultaneousDrags: 1,
feedback: widget.child,
childWhenDragging: Opacity(
opacity: .3,
child: widget.child,
),
onDragEnd: (details) => updatePosition(details.offset),
child: Transform.scale(
scale: scale,
child: widget.child,
),
),
),
],
),
);
}
}