You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
fluffychat/lib/pangea/chat/utils/unlocked_morphs_snackbar.dart

324 lines
12 KiB
Dart

// ignore_for_file: depend_on_referenced_packages, implementation_imports
import 'dart:async';
import 'package:flutter/material.dart';
import 'package:flutter_gen/gen_l10n/l10n.dart';
import 'package:fluffychat/config/app_config.dart';
import 'package:fluffychat/config/themes.dart';
import 'package:fluffychat/pangea/analytics_details_popup/analytics_details_popup.dart';
import 'package:fluffychat/pangea/analytics_misc/construct_type_enum.dart';
import 'package:fluffychat/pangea/analytics_misc/gain_points_animation.dart';
import 'package:fluffychat/pangea/common/utils/overlay.dart';
import 'package:fluffychat/pangea/constructs/construct_identifier.dart';
import 'package:fluffychat/pangea/morphs/get_grammar_copy.dart';
import 'package:fluffychat/pangea/morphs/morph_features_enum.dart';
import 'package:fluffychat/pangea/morphs/morph_icon.dart';
import 'package:fluffychat/widgets/matrix.dart';
class ConstructNotificationUtil {
static Completer? closeCompleter;
static final Set<ConstructIdentifier> unlockedConstructs = {};
static bool showingNotification = false;
static void addUnlockedConstruct(
List<ConstructIdentifier> constructs,
BuildContext context,
) {
unlockedConstructs.addAll(constructs);
if (!showingNotification) {
showUnlockedMorphsSnackbar(context);
}
}
static void onClose(ConstructIdentifier construct) {
final overlayKey = "${construct.string}_snackbar";
MatrixState.pAnyState.closeOverlay(overlayKey);
MatrixState.pAnyState.activeOverlays.remove(overlayKey);
unlockedConstructs.remove(construct);
closeCompleter?.complete();
closeCompleter = null;
}
static Future<void> showUnlockedMorphsSnackbar(BuildContext context) async {
showingNotification = true;
while (unlockedConstructs.isNotEmpty) {
final construct = unlockedConstructs.first;
try {
final copy = getGrammarCopy(
category: construct.category,
lemma: construct.lemma,
context: context,
);
closeCompleter = Completer();
OverlayUtil.showOverlay(
overlayKey: "${construct.string}_snackbar",
context: context,
child: ConstructNotificationOverlay(
construct: construct,
copy: copy,
),
transformTargetId: "",
position: OverlayPositionEnum.top,
backDropToDismiss: false,
closePrevOverlay: false,
canPop: false,
);
MatrixState.pAnyState.activeOverlays
.add("${construct.string}_snackbar");
await closeCompleter!.future;
} catch (e) {
MatrixState.pAnyState.activeOverlays
.remove("${construct.string}_snackbar");
showingNotification = false;
break;
}
}
showingNotification = false;
}
}
class ConstructNotificationOverlay extends StatefulWidget {
final ConstructIdentifier construct;
final String? copy;
const ConstructNotificationOverlay({
super.key,
required this.construct,
this.copy,
});
@override
State<ConstructNotificationOverlay> createState() =>
ConstructNotificationOverlayState();
}
class ConstructNotificationOverlayState
extends State<ConstructNotificationOverlay> with TickerProviderStateMixin {
AnimationController? _controller;
Animation<double>? _animation;
@override
void initState() {
super.initState();
_controller = AnimationController(
duration: FluffyThemes.animationDuration,
vsync: this,
);
_animation = CurvedAnimation(
parent: _controller!,
curve: Curves.easeInOut,
);
_controller!.forward().then((_) {
OverlayUtil.showOverlay(
overlayKey: "${widget.construct.string}_points",
followerAnchor: Alignment.topCenter,
targetAnchor: Alignment.topCenter,
context: context,
child: PointsGainedAnimation(
points: 50,
targetID: "${widget.construct.string}_notification",
invert: true,
),
transformTargetId: "${widget.construct.string}_notification",
closePrevOverlay: false,
backDropToDismiss: false,
ignorePointer: true,
);
Future.delayed(const Duration(seconds: 15), () {
if (mounted) _close();
});
});
}
@override
void dispose() {
_controller?.dispose();
super.dispose();
}
void _close() {
_controller?.reverse().then((_) {
ConstructNotificationUtil.onClose(widget.construct);
});
}
void _showDetails() {
showDialog<AnalyticsPopupWrapper>(
context: context,
builder: (context) => AnalyticsPopupWrapper(
constructZoom: widget.construct,
view: ConstructTypeEnum.morph,
backButtonOverride: IconButton(
icon: const Icon(Icons.close),
onPressed: () => Navigator.of(context).pop(),
),
),
);
}
@override
Widget build(BuildContext context) {
final isColumnMode = FluffyThemes.isColumnMode(context);
return CompositedTransformTarget(
link: MatrixState.pAnyState
.layerLinkAndKey("${widget.construct.string}_notification")
.link,
child: SafeArea(
key: MatrixState.pAnyState
.layerLinkAndKey("${widget.construct.string}_notification")
.key,
child: Material(
type: MaterialType.transparency,
child: SizeTransition(
sizeFactor: _animation!,
axisAlignment: -1.0,
child: LayoutBuilder(
builder: (context, constraints) {
return GestureDetector(
onPanUpdate: (details) {
if (details.delta.dy < -10) _close();
},
onTap: _showDetails,
child: Container(
padding: const EdgeInsets.symmetric(
vertical: 16.0,
horizontal: 4.0,
),
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.surface,
border: Border(
bottom: BorderSide(
color: AppConfig.gold.withAlpha(200),
width: 2.0,
),
),
borderRadius: const BorderRadius.only(
bottomLeft: Radius.circular(AppConfig.borderRadius),
bottomRight: Radius.circular(AppConfig.borderRadius),
),
),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
SizedBox(
width: constraints.maxWidth >= 600 ? 120.0 : 65.0,
),
Expanded(
child: Padding(
padding: EdgeInsets.symmetric(
horizontal: isColumnMode ? 16.0 : 8.0,
),
child: Wrap(
spacing: 16.0,
alignment: WrapAlignment.center,
crossAxisAlignment: WrapCrossAlignment.center,
children: [
Text(
widget.copy ?? widget.construct.lemma,
style: TextStyle(
fontSize: FluffyThemes.isColumnMode(context)
? 32.0
: 16.0,
color: AppConfig.gold,
fontWeight: FontWeight.bold,
),
overflow: TextOverflow.ellipsis,
),
MorphIcon(
size: isColumnMode
? null
: const Size(24.0, 24.0),
morphFeature:
MorphFeaturesEnumExtension.fromString(
widget.construct.category,
),
morphTag: widget.construct.lemma,
),
],
),
),
),
SizedBox(
width: constraints.maxWidth >= 600 ? 120.0 : 65.0,
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Tooltip(
message: L10n.of(context).details,
child: constraints.maxWidth >= 600
? ElevatedButton(
style: IconButton.styleFrom(
padding: const EdgeInsets.symmetric(
vertical: 4.0,
horizontal: 16.0,
),
),
onPressed: _showDetails,
child: Text(
L10n.of(context).details,
),
)
: SizedBox(
width: 32.0,
height: 32.0,
child: Center(
child: IconButton(
icon: const Icon(
Icons.info_outline,
),
style: IconButton.styleFrom(
padding:
const EdgeInsets.all(4.0),
),
onPressed: _showDetails,
constraints: const BoxConstraints(),
),
),
),
),
SizedBox(
width: 32.0,
height: 32.0,
child: Center(
child: Tooltip(
message: L10n.of(context).close,
child: IconButton(
icon: const Icon(
Icons.close,
),
style: IconButton.styleFrom(
padding: const EdgeInsets.all(4.0),
),
onPressed: _close,
constraints: const BoxConstraints(),
),
),
),
),
],
),
),
],
),
),
);
},
),
),
),
),
);
}
}