fluffychat/lib/pangea/analytics_misc/message_analytics_feedback.dart

268 lines
7.6 KiB
Dart

import 'package:flutter/material.dart';
import 'package:fluffychat/config/themes.dart';
import 'package:fluffychat/l10n/l10n.dart';
import 'package:fluffychat/pangea/analytics_misc/analytics_navigation_util.dart';
import 'package:fluffychat/pangea/analytics_misc/construct_type_enum.dart';
class MessageAnalyticsFeedback extends StatefulWidget {
final int newGrammarConstructs;
final int newVocabConstructs;
final VoidCallback close;
const MessageAnalyticsFeedback({
required this.newGrammarConstructs,
required this.newVocabConstructs,
required this.close,
super.key,
});
@override
State<MessageAnalyticsFeedback> createState() =>
MessageAnalyticsFeedbackState();
}
class MessageAnalyticsFeedbackState extends State<MessageAnalyticsFeedback>
with TickerProviderStateMixin {
late AnimationController _numbersController;
late AnimationController _bubbleController;
late AnimationController _tickerController;
late Animation<double> _numbersOpacityAnimation;
late Animation<double> _bubbleScaleAnimation;
late Animation<double> _bubbleOpacityAnimation;
Animation<int>? _grammarTickerAnimation;
Animation<int>? _vocabTickerAnimation;
@override
void initState() {
super.initState();
_numbersController = AnimationController(
vsync: this,
duration: FluffyThemes.animationDuration,
);
_numbersOpacityAnimation = Tween<double>(begin: 0.0, end: 1.0).animate(
CurvedAnimation(parent: _numbersController, curve: Curves.easeInOut),
);
_bubbleController = AnimationController(
vsync: this,
duration: FluffyThemes.animationDuration,
);
_bubbleScaleAnimation = Tween<double>(begin: 0.0, end: 1.0).animate(
CurvedAnimation(parent: _bubbleController, curve: Curves.easeInOut),
);
_bubbleOpacityAnimation = Tween<double>(begin: 0.0, end: 0.9).animate(
CurvedAnimation(parent: _bubbleController, curve: Curves.easeInOut),
);
_tickerController = AnimationController(
vsync: this,
duration: FluffyThemes.animationDuration,
);
_numbersOpacityAnimation.addStatusListener((status) {
if (status == AnimationStatus.completed) {
_startTickerAnimations();
}
});
_bubbleController.forward();
Future.delayed(
const Duration(milliseconds: 400),
_numbersController.forward,
);
Future.delayed(const Duration(milliseconds: 4000), () async {
if (mounted) {
await _bubbleController.reverse();
widget.close();
}
});
}
@override
void dispose() {
_numbersController.dispose();
_bubbleController.dispose();
_tickerController.dispose();
super.dispose();
}
void _startTickerAnimations() {
_vocabTickerAnimation = IntTween(
begin: 0,
end: widget.newVocabConstructs,
).animate(
CurvedAnimation(
parent: _tickerController,
curve: Curves.easeOutCubic,
),
);
_grammarTickerAnimation = IntTween(
begin: 0,
end: widget.newGrammarConstructs,
).animate(
CurvedAnimation(
parent: _tickerController,
curve: Curves.easeOutCubic,
),
);
setState(() {});
_tickerController.forward();
}
@override
Widget build(BuildContext context) {
if (widget.newVocabConstructs <= 0 && widget.newGrammarConstructs <= 0) {
return const SizedBox.shrink();
}
return Material(
type: MaterialType.transparency,
child: InkWell(
onTap: () =>
AnalyticsNavigationUtil.navigateToAnalytics(context: context),
child: ScaleTransition(
scale: _bubbleScaleAnimation,
alignment: Alignment.bottomRight,
child: AnimatedBuilder(
animation: _bubbleController,
builder: (context, child) {
return Container(
decoration: BoxDecoration(
color: Theme.of(context)
.colorScheme
.surfaceContainer
.withAlpha((_bubbleOpacityAnimation.value * 255).round()),
borderRadius: const BorderRadius.only(
topLeft: Radius.circular(16.0),
topRight: Radius.circular(16.0),
bottomLeft: Radius.circular(16.0),
bottomRight: Radius.circular(4.0),
),
),
padding: const EdgeInsets.symmetric(
vertical: 8.0,
horizontal: 16.0,
),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
if (widget.newVocabConstructs > 0)
_NewConstructsBadge(
opacityAnimation: _numbersOpacityAnimation,
tickerAnimation: _vocabTickerAnimation,
type: ConstructTypeEnum.vocab,
tooltip: L10n.of(context).newVocab,
),
if (widget.newGrammarConstructs > 0)
_NewConstructsBadge(
opacityAnimation: _numbersOpacityAnimation,
tickerAnimation: _grammarTickerAnimation,
type: ConstructTypeEnum.morph,
tooltip: L10n.of(context).newGrammar,
),
],
),
);
},
),
),
),
);
}
}
class _NewConstructsBadge extends StatelessWidget {
final Animation<double> opacityAnimation;
final Animation<int>? tickerAnimation;
final ConstructTypeEnum type;
final String tooltip;
const _NewConstructsBadge({
required this.opacityAnimation,
required this.tickerAnimation,
required this.type,
required this.tooltip,
});
@override
Widget build(BuildContext context) {
return InkWell(
onTap: () => AnalyticsNavigationUtil.navigateToAnalytics(
context: context,
view: type.indicator,
),
child: Tooltip(
message: tooltip,
child: AnimatedBuilder(
animation: opacityAnimation,
builder: (context, child) {
return Opacity(
opacity: opacityAnimation.value,
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 8.0),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Icon(
type.indicator.icon,
color: type.indicator.color(context),
size: 24,
),
const SizedBox(width: 4.0),
_AnimatedCounter(
key: ValueKey("$type-counter"),
animation: tickerAnimation,
style: TextStyle(
color: type.indicator.color(context),
fontWeight: FontWeight.bold,
),
),
],
),
),
);
},
),
),
);
}
}
class _AnimatedCounter extends StatelessWidget {
final Animation<int>? animation;
final TextStyle? style;
const _AnimatedCounter({
super.key,
required this.animation,
this.style,
});
@override
Widget build(BuildContext context) {
if (animation == null) {
return Text(
"+ 0",
style: style,
);
}
return AnimatedBuilder(
animation: animation!,
builder: (context, child) {
return Text(
"+ ${animation!.value}",
style: style,
);
},
);
}
}