fluffychat/lib/pangea/activity_sessions/activity_user_summaries_widget.dart

333 lines
11 KiB
Dart

import 'package:flutter/material.dart';
import 'package:collection/collection.dart';
import 'package:material_symbols_icons/symbols.dart';
import 'package:matrix/matrix.dart';
import 'package:fluffychat/config/app_config.dart';
import 'package:fluffychat/config/themes.dart';
import 'package:fluffychat/l10n/l10n.dart';
import 'package:fluffychat/pages/chat/chat.dart';
import 'package:fluffychat/pangea/activity_sessions/activity_participant_indicator.dart';
import 'package:fluffychat/pangea/activity_sessions/activity_role_model.dart';
import 'package:fluffychat/pangea/activity_sessions/activity_room_extension.dart';
import 'package:fluffychat/pangea/activity_summary/activity_summary_response_model.dart';
import 'package:fluffychat/widgets/avatar.dart';
class ActivityUserSummaries extends StatelessWidget {
final ChatController controller;
const ActivityUserSummaries({
super.key,
required this.controller,
});
Room get room => controller.room;
@override
Widget build(BuildContext context) {
final summary = room.activitySummary?.summary;
if (summary == null) return const SizedBox();
return Padding(
padding: const EdgeInsets.all(12.0),
child: Column(
spacing: 4.0,
mainAxisSize: MainAxisSize.min,
children: [
Padding(
padding:
const EdgeInsets.symmetric(horizontal: 12.0, vertical: 4.0),
child: Center(
child: Material(
color: Theme.of(context).colorScheme.surface.withAlpha(128),
borderRadius: BorderRadius.circular(AppConfig.borderRadius / 3),
child: Padding(
padding: const EdgeInsets.symmetric(
horizontal: 8.0,
vertical: 4.0,
),
child: Column(
spacing: 4.0,
mainAxisSize: MainAxisSize.min,
children: [
Text(
L10n.of(context).activityFinishedMessage,
),
Text(
summary.summary,
textAlign: TextAlign.center,
),
],
),
),
),
),
),
const Padding(
padding: EdgeInsets.symmetric(
vertical: 8.0,
),
),
ButtonControlledCarouselView(
summary: summary,
controller: controller,
),
],
),
);
}
}
class ButtonControlledCarouselView extends StatelessWidget {
final ActivitySummaryResponseModel summary;
final ChatController controller;
const ButtonControlledCarouselView({
super.key,
required this.summary,
required this.controller,
});
void _scrollToUser(
ActivityRoleModel role,
int index,
double cardWidth,
) {
controller.activityController.highlightRole(role);
final scrollController = controller.activityController.carouselController;
if (!scrollController.hasClients) return;
const spacing = 5.0;
final itemExtent = cardWidth + spacing;
final viewportWidth = scrollController.position.viewportDimension;
final itemCenter = (index * itemExtent) + (cardWidth / 2);
final targetOffset = itemCenter - (viewportWidth / 2);
final clampedOffset = targetOffset.clamp(
scrollController.position.minScrollExtent,
scrollController.position.maxScrollExtent,
);
scrollController.animateTo(
clampedOffset,
duration: const Duration(milliseconds: 300),
curve: Curves.easeOutCubic,
);
}
@override
Widget build(BuildContext context) {
final room = controller.room;
final superlatives =
room.activitySummary?.analytics?.generateSuperlatives();
final availableRoles = room.activityPlan!.roles;
final assignedRoles = room.assignedRoles ?? {};
final userSummaries = summary.participants
.where(
(p) => assignedRoles.values.any(
(role) => role.userId == p.participantId,
),
)
.toList();
final isColumnMode = FluffyThemes.isColumnMode(context);
if (userSummaries.isEmpty) {
return const SizedBox();
}
final cardWidth = isColumnMode ? 400.0 : 350.0;
return Column(
children: [
SizedBox(
height: 270.0,
child: ListView.builder(
key: PageStorageKey('summaries-carousel-${room.id}'),
shrinkWrap: true,
controller: controller.activityController.carouselController,
scrollDirection: Axis.horizontal,
itemCount: userSummaries.length,
itemBuilder: (context, i) {
final p = userSummaries[i];
final user = room.getParticipants().firstWhereOrNull(
(u) => u.id == p.participantId,
);
final userRole = assignedRoles.values.firstWhere(
(role) => role.userId == p.participantId,
);
return Container(
width: cardWidth,
margin: const EdgeInsets.only(right: 5.0),
padding: const EdgeInsets.all(12.0),
decoration: ShapeDecoration(
color: Color.alphaBlend(
Theme.of(context).colorScheme.surface.withAlpha(70),
AppConfig.gold,
),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(12),
),
),
child: Column(
spacing: 4.0,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
spacing: 10.0,
mainAxisSize: MainAxisSize.min,
children: [
Avatar(
name: p.participantId.localpart,
mxContent: user?.avatarUrl,
size: 40,
),
Flexible(
child: Text(
"${userRole.role ?? L10n.of(context).participant} | ${user?.calcDisplayname() ?? p.participantId.localpart}",
style: const TextStyle(
fontSize: 14.0,
),
overflow: TextOverflow.ellipsis,
),
),
],
),
Flexible(
child: SingleChildScrollView(
child: Text(
p.displayFeedback(
user?.calcDisplayname() ??
p.participantId.localpart ??
p.participantId,
),
style: const TextStyle(fontSize: 14.0),
),
),
),
Padding(
padding: const EdgeInsets.all(8.0),
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Wrap(
alignment: WrapAlignment.center,
spacing: 12,
runSpacing: 8,
children: [
Text(
p.cefrLevel,
style: const TextStyle(
fontSize: 14.0,
),
),
//const SizedBox(width: 8),
if (superlatives != null &&
(superlatives['vocab']!.contains(
p.participantId,
))) ...[
const SuperlativeTile(
icon: Symbols.dictionary,
),
],
if (superlatives != null &&
(superlatives['grammar']!.contains(
p.participantId,
))) ...[
const SuperlativeTile(
icon: Symbols.toys_and_games,
),
],
if (superlatives != null &&
(superlatives['xp']!.contains(
p.participantId,
))) ...[
const SuperlativeTile(
icon: Icons.star,
),
],
if (p.superlatives.isNotEmpty) ...[
Text(
p.superlatives.first,
style: const TextStyle(fontSize: 14.0),
),
],
],
),
],
),
),
],
),
);
},
),
),
const SizedBox(height: 12),
SizedBox(
height: 125.0,
child: ValueListenableBuilder(
valueListenable: controller.activityController.highlightedRole,
builder: (context, highlightedRole, __) {
return ListView.builder(
shrinkWrap: true,
scrollDirection: Axis.horizontal,
itemCount: userSummaries.length,
itemBuilder: (context, index) {
final p = userSummaries[index];
final user = room.getParticipants().firstWhereOrNull(
(u) => u.id == p.participantId,
);
final userRole = assignedRoles.values.firstWhere(
(role) => role.userId == p.participantId,
);
final userRoleInfo = availableRoles[userRole.id]!;
return ActivityParticipantIndicator(
name: userRoleInfo.name,
userId: p.participantId,
user: user,
borderRadius: BorderRadius.circular(4),
selected: highlightedRole?.id == userRole.id,
onTap: () => _scrollToUser(userRole, index, cardWidth),
);
},
);
},
),
),
],
);
}
}
class SuperlativeTile extends StatelessWidget {
final IconData icon;
const SuperlativeTile({
super.key,
required this.icon,
});
@override
Widget build(BuildContext context) {
return Row(
mainAxisSize: MainAxisSize.min,
children: [
Icon(icon, size: 14, color: Theme.of(context).colorScheme.onSurface),
const SizedBox(width: 2),
const Text(
"1st",
style: TextStyle(
fontSize: 14.0,
),
),
],
);
}
}