86 lines
2.1 KiB
Dart
86 lines
2.1 KiB
Dart
import 'package:collection/collection.dart';
|
|
import 'package:flutter/material.dart';
|
|
|
|
class AnimatedText extends StatefulWidget {
|
|
final String data;
|
|
final TextStyle style, changedStyle;
|
|
final Curve curve;
|
|
final Duration duration;
|
|
|
|
const AnimatedText(
|
|
this.data, {
|
|
super.key,
|
|
required this.style,
|
|
required this.changedStyle,
|
|
this.curve = Curves.linear,
|
|
required this.duration,
|
|
});
|
|
|
|
@override
|
|
State<AnimatedText> createState() => _AnimatedTextState();
|
|
}
|
|
|
|
class _AnimatedTextState extends State<AnimatedText> with SingleTickerProviderStateMixin {
|
|
late final AnimationController _controller;
|
|
late final Animation<TextStyle> _style;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
_controller = AnimationController(
|
|
duration: widget.duration,
|
|
vsync: this,
|
|
)
|
|
..value = 1
|
|
..addListener(() => setState(() {}));
|
|
_style = _ShadowedTextStyleTween(begin: widget.changedStyle, end: widget.style).animate(CurvedAnimation(
|
|
parent: _controller,
|
|
curve: widget.curve,
|
|
));
|
|
}
|
|
|
|
@override
|
|
void didUpdateWidget(AnimatedText oldWidget) {
|
|
super.didUpdateWidget(oldWidget);
|
|
if (oldWidget.data != widget.data) {
|
|
_controller
|
|
..value = 0
|
|
..forward();
|
|
}
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_controller.dispose();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Text(
|
|
widget.data,
|
|
style: _style.value,
|
|
);
|
|
}
|
|
}
|
|
|
|
class _ShadowedTextStyleTween extends Tween<TextStyle> {
|
|
_ShadowedTextStyleTween({super.begin, super.end});
|
|
|
|
@override
|
|
TextStyle lerp(double t) {
|
|
final textStyle = TextStyle.lerp(begin, end, t)!;
|
|
final beginShadows = begin!.shadows;
|
|
final endShadows = end!.shadows;
|
|
if (beginShadows != null && endShadows != null && beginShadows.length == endShadows.length) {
|
|
return textStyle.copyWith(
|
|
shadows: beginShadows.mapIndexed((i, a) {
|
|
final b = endShadows[i];
|
|
return Shadow.lerp(a, b, t)!;
|
|
}).toList(),
|
|
);
|
|
} else {
|
|
return textStyle;
|
|
}
|
|
}
|
|
}
|