fluffychat/lib/pangea/analytics_misc/example_message_util.dart
ggurdin 8a8ca1026a
Vocab practice updates (#5180)
* reorganization, reload on language change

* make choice card widget

* make completed activity view stateless

* use analytics updater mixin to display points gained animation

* simplify animation in game card

* better encapsulate practice session data

* reset session loader instead of dispose

* simplify practice session model

* queue activities

* visually remove duplicate answers without editing activity content

* review updates

* don't shuffle filtered choices
2026-01-14 12:54:27 -05:00

80 lines
2.4 KiB
Dart

import 'package:flutter/material.dart';
import 'package:collection/collection.dart';
import 'package:matrix/matrix.dart';
import 'package:fluffychat/pangea/analytics_misc/client_analytics_extension.dart';
import 'package:fluffychat/pangea/analytics_misc/construct_use_model.dart';
import 'package:fluffychat/pangea/events/event_wrappers/pangea_message_event.dart';
class ExampleMessageUtil {
static Future<List<InlineSpan>?> getExampleMessage(
ConstructUses construct,
Client client,
) async {
for (final use in construct.cappedUses) {
final event = await client.getEventByConstructUse(use);
if (event == null) continue;
final spans = _buildExampleMessage(use.form, event);
if (spans != null) return spans;
}
return null;
}
static Future<List<List<InlineSpan>>> getExampleMessages(
ConstructUses construct,
Client client,
int maxMessages,
) async {
final List<List<InlineSpan>> allSpans = [];
for (final use in construct.cappedUses) {
if (allSpans.length >= maxMessages) break;
final event = await client.getEventByConstructUse(use);
if (event == null) continue;
final spans = _buildExampleMessage(use.form, event);
if (spans != null) {
allSpans.add(spans);
}
}
return allSpans;
}
static List<InlineSpan>? _buildExampleMessage(
String? form,
PangeaMessageEvent messageEvent,
) {
final tokens = messageEvent.messageDisplayRepresentation?.tokens;
if (tokens == null || tokens.isEmpty) return null;
final token = tokens.firstWhereOrNull(
(token) => token.text.content == form,
);
if (token == null) return null;
final text = messageEvent.messageDisplayText;
final tokenText = token.text.content;
int tokenIndex = text.indexOf(tokenText);
if (tokenIndex == -1) return null;
final beforeSubstring = text.substring(0, tokenIndex);
if (beforeSubstring.length != beforeSubstring.characters.length) {
tokenIndex = beforeSubstring.characters.length;
}
final int tokenLength = tokenText.characters.length;
final before = text.characters.take(tokenIndex).toString();
final after = text.characters.skip(tokenIndex + tokenLength).toString();
return [
TextSpan(text: before),
TextSpan(
text: tokenText,
style: const TextStyle(
fontWeight: FontWeight.bold,
),
),
TextSpan(text: after),
];
}
}