import 'package:flutter/material.dart';
class ElectricAnimated extends StatelessWidget {
final int duration; //动画时长
final List path; //路径的点坐标
const ElectricAnimated({
super.key,
this.duration = 5,
required this.path,
});
@override
Widget build(BuildContext context) {
return Stack(
children: List.generate(
path.length - 1,
(index) {
return AnimatedDots(
path: [
path[index],
path[index + 1],
],
dotCount: 3,
duration: Duration(
seconds: duration,
),
);
},
),
);
}
}
class AnimatedDots extends StatefulWidget {
final List path; //路径的点坐标
final int dotCount; //每条路径上的点的个数
final double dotSize; //点的大小
final Duration duration;
final Color dotColor; //点的颜色
const AnimatedDots({
Key? key,
required this.path,
this.dotCount = 5,
this.dotSize = 2,
this.duration = const Duration(seconds: 2),
this.dotColor = Colors.white,
}) : super(key: key);
@override
State createState() => _AnimatedDotsState();
}
class _AnimatedDotsState extends State
with SingleTickerProviderStateMixin {
late AnimationController _animationController;
late Animation _animation;
int _currentIndex = 0;
@override
void initState() {
super.initState();
_animationController = AnimationController(
vsync: this,
duration: widget.duration,
);
_animation = Tween(
begin: 0,
end: 1,
).animate(_animationController);
_animationController.addStatusListener((status) {
if (status == AnimationStatus.completed) {
_currentIndex++;
if (_currentIndex >= widget.path.length - 1) {
_currentIndex = 0;
}
_animationController.reset();
_animationController.forward();
}
});
_animationController.forward();
}
@override
void dispose() {
_animationController.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
return AnimatedBuilder(
animation: _animation,
builder: (context, child) {
final value = _animation.value;
final current = widget.path[_currentIndex];
final next = widget.path[(_currentIndex + 1) % widget.path.length];
final dots = List.generate(widget.dotCount, (index) {
final dotValue = (value + index / widget.dotCount) % 1;
final x = current.dx + (next.dx - current.dx) * dotValue;
final y = current.dy + (next.dy - current.dy) * dotValue;
return Positioned(
left: x - widget.dotSize / 2,
top: y - widget.dotSize / 2,
child: Container(
width: widget.dotSize,
height: widget.dotSize,
decoration: BoxDecoration(
color: widget.dotColor,
shape: BoxShape.circle,
),
),
);
});
return Stack(
children: dots,
);
},
);
}
}
body: Center(
child: Container(
color: Colors.black,
width: 150,
height: 150,
child: const ElectricAnimated(
duration: 3,
path: [
Offset(30, 0),
Offset(30, 67),
Offset(90, 67),
Offset(90, 137),
],
),
),
),
简书地址