fluffychat/lib/pangea/analytics_downloads/analytics_dowload_dialog.dart
ggurdin e8428783e6
Fluffychat merge 2 (#5590)
* build: Reenable shrink resources and minify in gradle

* build: (deps): bump image from 4.6.0 to 4.7.1

Bumps [image](https://github.com/brendan-duncan/image) from 4.6.0 to 4.7.1.
- [Changelog](https://github.com/brendan-duncan/image/blob/main/CHANGELOG.md)
- [Commits](https://github.com/brendan-duncan/image/commits)

---
updated-dependencies:
- dependency-name: image
  dependency-version: 4.7.1
  dependency-type: direct:production
  update-type: version-update:semver-minor
...

Signed-off-by: dependabot[bot] <support@github.com>

* build: (deps): bump file_picker from 10.3.7 to 10.3.8

Bumps [file_picker](https://github.com/miguelpruivo/flutter_file_picker) from 10.3.7 to 10.3.8.
- [Release notes](https://github.com/miguelpruivo/flutter_file_picker/releases)
- [Changelog](https://github.com/miguelpruivo/flutter_file_picker/blob/master/CHANGELOG.md)
- [Commits](https://github.com/miguelpruivo/flutter_file_picker/compare/v10.3.7...v10.3.8)

---
updated-dependencies:
- dependency-name: file_picker
  dependency-version: 10.3.8
  dependency-type: direct:production
  update-type: version-update:semver-patch
...

Signed-off-by: dependabot[bot] <support@github.com>

* feat: Improved search

* build: Use matrix sdk vom pub.dev again

* chore: Follow up better search

* build: (deps): bump image from 4.7.1 to 4.7.2

Bumps [image](https://github.com/brendan-duncan/image) from 4.7.1 to 4.7.2.
- [Changelog](https://github.com/brendan-duncan/image/blob/main/CHANGELOG.md)
- [Commits](https://github.com/brendan-duncan/image/commits)

---
updated-dependencies:
- dependency-name: image
  dependency-version: 4.7.2
  dependency-type: direct:production
  update-type: version-update:semver-patch
...

Signed-off-by: dependabot[bot] <support@github.com>

* chore: Make cross signing self sign mandatory for bootstrap

* chore: Update user device keys before creating bootstrap

* fix: Better wait for secrets after verification bootstrap

* refactor: Remove native imaging and enable web worker

* refactor: Remove unused html onfocus streams

* build: (deps): bump flutter_foreground_task from 9.1.0 to 9.2.0

Bumps [flutter_foreground_task](https://github.com/Dev-hwang/flutter_foreground_task) from 9.1.0 to 9.2.0.
- [Changelog](https://github.com/Dev-hwang/flutter_foreground_task/blob/master/CHANGELOG.md)
- [Commits](https://github.com/Dev-hwang/flutter_foreground_task/commits)

---
updated-dependencies:
- dependency-name: flutter_foreground_task
  dependency-version: 9.2.0
  dependency-type: direct:production
  update-type: version-update:semver-minor
...

Signed-off-by: dependabot[bot] <support@github.com>

* chore(translations): Translated using Weblate (Uzbek)

Currently translated at 99.7% (823 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uz/

* chore(translations): Translated using Weblate (Russian)

Currently translated at 99.8% (824 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/

* chore(translations): Translated using Weblate (Norwegian Bokmål)

Currently translated at 90.9% (750 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nb_NO/

* chore(translations): Translated using Weblate (Galician)

Currently translated at 100.0% (825 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/gl/

* chore(translations): Translated using Weblate (Basque)

Currently translated at 99.7% (823 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/

* chore(translations): Translated using Weblate (Ukrainian)

Currently translated at 100.0% (825 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/

* chore(translations): Translated using Weblate (Estonian)

Currently translated at 100.0% (825 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/

* chore(translations): Translated using Weblate (Dutch)

Currently translated at 100.0% (825 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/

* chore(translations): Translated using Weblate (Russian)

Currently translated at 100.0% (825 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/

* chore(translations): Translated using Weblate (Spanish)

Currently translated at 95.2% (788 of 827 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/

* chore(translations): Translated using Weblate (Spanish)

Currently translated at 96.3% (797 of 827 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/

* chore(translations): Translated using Weblate (Russian)

Currently translated at 100.0% (825 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/

* chore(translations): Translated using Weblate (Russian)

Currently translated at 100.0% (825 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/

* fix: Broken ruzzian plurals

* chore(translations): Translated using Weblate (Norwegian Bokmål)

Currently translated at 91.2% (753 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nb_NO/

* chore(translations): Translated using Weblate (Bengali)

Currently translated at 4.5% (38 of 827 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/bn/

* chore(translations): Translated using Weblate (French)

Currently translated at 82.3% (679 of 825 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/fr/

* build: (deps): bump translations_cleaner from 0.0.5 to 0.1.0

Bumps [translations_cleaner](https://github.com/Chinmay-KB/translations_cleaner) from 0.0.5 to 0.1.0.
- [Changelog](https://github.com/Chinmay-KB/translations_cleaner/blob/main/CHANGELOG.md)
- [Commits](https://github.com/Chinmay-KB/translations_cleaner/commits)

---
updated-dependencies:
- dependency-name: translations_cleaner
  dependency-version: 0.1.0
  dependency-type: direct:production
  update-type: version-update:semver-minor
...

Signed-off-by: dependabot[bot] <support@github.com>

* chore(translations): Translated using Weblate (German)

Currently translated at 99.2% (821 of 827 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/

* chore(translations): Translated using Weblate (Estonian)

Currently translated at 100.0% (827 of 827 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/

* build: Bump version to 2.4.0

* build: (deps): bump sqflite_common_ffi from 2.3.6 to 2.3.7+1

Bumps [sqflite_common_ffi](https://github.com/tekartik/sqflite) from 2.3.6 to 2.3.7+1.
- [Commits](https://github.com/tekartik/sqflite/compare/sqflite_common_ffi_v2.3.6...sqflite_common_ffi/v2.3.7)

---
updated-dependencies:
- dependency-name: sqflite_common_ffi
  dependency-version: 2.3.7+1
  dependency-type: direct:production
  update-type: version-update:semver-patch
...

Signed-off-by: dependabot[bot] <support@github.com>

* chore(translations): Translated using Weblate (Czech)

Currently translated at 66.1% (547 of 827 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/cs/

* chore(translations): Translated using Weblate (Czech)

Currently translated at 72.7% (602 of 827 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/cs/

* chore(translations): Translated using Weblate (German)

Currently translated at 99.8% (826 of 827 strings)

Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/

* chore: Add security.md file

* fix: Locale unlocalized strings

* build: (deps): bump matrix from 4.1.0 to 5.0.0

Bumps [matrix](https://github.com/famedly/matrix-dart-sdk) from 4.1.0 to 5.0.0.
- [Release notes](https://github.com/famedly/matrix-dart-sdk/releases)
- [Changelog](https://github.com/famedly/matrix-dart-sdk/blob/main/CHANGELOG.md)
- [Commits](https://github.com/famedly/matrix-dart-sdk/compare/v4.1.0...v5.0.0)

---
updated-dependencies:
- dependency-name: matrix
  dependency-version: 5.0.0
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>

* fix: Notifications on web correctly managed when tab not focused

* chore: Add changelog for android

* chore: Remove duplicated localization

* fix: Sign in label

* chore: Versionize fcm shared isolate

* build: Remove unused packag

* build: (deps): bump package_info_plus from 8.3.1 to 9.0.0

Bumps [package_info_plus](https://github.com/fluttercommunity/plus_plugins/tree/main/packages/package_info_plus) from 8.3.1 to 9.0.0.
- [Release notes](https://github.com/fluttercommunity/plus_plugins/releases)
- [Commits](https://github.com/fluttercommunity/plus_plugins/commits/package_info_plus-v9.0.0/packages/package_info_plus)

---
updated-dependencies:
- dependency-name: package_info_plus
  dependency-version: 9.0.0
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>

* feat: Display particle animation on login page

* chore: Use fixed version of fcm shared isolate

* fix: apk crash on some platforms due new flutter version

* chore: Correct kotlin format

* fix iOS notifications

* fluffychat merge

* fluffychat merge

* fluffychat merge

* fluffychat merge

* fluffychat merge

* fluffychat merge

* add missing type annotations

* update matrix version

* fluffychat merge

* fluffychat merge

* fix notification on click actions

---------

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: Christian Kußowski <c.kussowski@famedly.com>
Co-authored-by: Krille-chan <christian-kussowski@posteo.de>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: BeMeritus <bemerituss@gmail.com>
Co-authored-by: Frank Paul Silye <frankps@gmail.com>
Co-authored-by: josé m. <correoxm@disroot.org>
Co-authored-by: xabirequejo <xabi.rn@gmail.com>
Co-authored-by: Ihor Hordiichuk <igor_ck@outlook.com>
Co-authored-by: Priit Jõerüüt <jrthwlate@users.noreply.hosted.weblate.org>
Co-authored-by: Jelv <post@jelv.nl>
Co-authored-by: Дмитрий Михирев <bizdelnick@gmail.com>
Co-authored-by: Kimby <kimbyqs@gmail.com>
Co-authored-by: Christian <christian-pauly@posteo.de>
Co-authored-by: Kom nake <kominak310@svcache.com>
Co-authored-by: hugues de keyzer <komputilisto@hugues.info>
Co-authored-by: nautilusx <translate@disroot.org>
Co-authored-by: Šebestová <ka.sebestova.cz@gmail.com>
2026-02-10 08:01:12 -05:00

483 lines
15 KiB
Dart

import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:csv/csv.dart';
import 'package:excel/excel.dart';
import 'package:intl/intl.dart';
import 'package:matrix/matrix.dart';
import 'package:fluffychat/config/app_config.dart';
import 'package:fluffychat/config/setting_keys.dart';
import 'package:fluffychat/config/themes.dart';
import 'package:fluffychat/l10n/l10n.dart';
import 'package:fluffychat/pangea/analytics_downloads/analytics_summary_enum.dart';
import 'package:fluffychat/pangea/analytics_downloads/analytics_summary_model.dart';
import 'package:fluffychat/pangea/analytics_misc/construct_type_enum.dart';
import 'package:fluffychat/pangea/analytics_misc/construct_use_model.dart';
import 'package:fluffychat/pangea/analytics_misc/construct_use_type_enum.dart';
import 'package:fluffychat/pangea/analytics_misc/constructs_model.dart';
import 'package:fluffychat/pangea/analytics_misc/learning_skills_enum.dart';
import 'package:fluffychat/pangea/common/utils/error_handler.dart';
import 'package:fluffychat/pangea/common/widgets/error_indicator.dart';
import 'package:fluffychat/pangea/constructs/construct_identifier.dart';
import 'package:fluffychat/pangea/download/download_file_util.dart';
import 'package:fluffychat/pangea/download/download_type_enum.dart';
import 'package:fluffychat/pangea/events/event_wrappers/pangea_message_event.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_repo.dart';
import 'package:fluffychat/widgets/matrix.dart';
class AnalyticsDownloadDialog extends StatefulWidget {
const AnalyticsDownloadDialog({super.key});
@override
AnalyticsDownloadDialogState createState() => AnalyticsDownloadDialogState();
}
class AnalyticsDownloadDialogState extends State<AnalyticsDownloadDialog> {
DownloadType _downloadType = DownloadType.csv;
bool _downloading = false;
bool _downloaded = false;
String? _error;
String? get _statusText {
if (_downloading) return L10n.of(context).downloading;
if (_downloaded) return L10n.of(context).downloadInitiated;
return null;
}
void _setDownloadType(DownloadType type) {
if (mounted) {
setState(() {
_downloadType = type;
_downloaded = false;
_error = null;
});
}
}
Future<void> _downloadAnalytics() async {
List<AnalyticsSummaryModel> vocabSummary;
List<AnalyticsSummaryModel> morphSummary;
try {
setState(() {
_downloading = true;
_downloaded = false;
_error = null;
});
vocabSummary = await _getVocabAnalytics();
morphSummary = await _getMorphAnalytics();
} catch (e, s) {
ErrorHandler.logError(e: e, s: s, data: {"downloadType": _downloadType});
if (mounted) {
setState(() {
_downloading = false;
_error = L10n.of(context).errorProcessAnalytics;
});
}
return;
}
final client = Matrix.of(context).client;
try {
if (_downloadType == DownloadType.csv) {
final vocabContent = _getCSVFileContent(
ConstructTypeEnum.vocab,
vocabSummary,
);
final morphContent = _getCSVFileContent(
ConstructTypeEnum.morph,
morphSummary,
);
final vocabFileName =
"analytics_vocab_${client.userID?.localpart}_${DateFormat('yyyy-MM-dd-hh:mm:ss').format(DateTime.now())}.csv";
final morphFileName =
"analytics_morph_${client.userID?.localpart}_${DateFormat('yyyy-MM-dd-hh:mm:ss').format(DateTime.now())}.csv";
final futures = [
DownloadUtil.downloadFile(vocabContent, vocabFileName, _downloadType),
DownloadUtil.downloadFile(morphContent, morphFileName, _downloadType),
];
await Future.wait(futures);
} else {
final content = _getExcelFileContent({
ConstructTypeEnum.vocab: vocabSummary,
ConstructTypeEnum.morph: morphSummary,
});
final fileName =
"analytics_${client.userID?.localpart}_${DateFormat('yyyy-MM-dd-hh:mm:ss').format(DateTime.now())}.xlsx";
await DownloadUtil.downloadFile(content, fileName, _downloadType);
}
_downloaded = true;
} catch (e, s) {
ErrorHandler.logError(e: e, s: s, data: {"downloadType": _downloadType});
_error = L10n.of(context).errorDownloading;
} finally {
if (mounted) setState(() => _downloading = false);
}
}
Future<List<AnalyticsSummaryModel>> _getVocabAnalytics() async {
final analyticsService = Matrix.of(context).analyticsDataService;
final aggregatedVocab = await analyticsService.getAggregatedConstructs(
ConstructTypeEnum.vocab,
);
final uses = aggregatedVocab.values.toList();
final Map<String, List<ConstructUses>> lemmasToUses = {};
for (final use in uses) {
lemmasToUses[use.lemma] ??= [];
lemmasToUses[use.lemma]!.add(use);
}
final List<AnalyticsSummaryModel> summaries = [];
for (final entry in lemmasToUses.entries) {
final lemma = entry.key;
final uses = entry.value;
final xp = uses.map((e) => e.points).reduce((a, total) => a + total);
final exampleMessages = await _getExampleMessages(uses);
final allUses = uses
.map((u) => u.cappedUses)
.expand((element) => element);
int independantUseOccurrences = 0;
int assistedUseOccurrences = 0;
for (final use in allUses) {
use.useType == ConstructUseTypeEnum.wa
? independantUseOccurrences++
: assistedUseOccurrences++;
}
final forms = allUses
.map((e) => e.form?.toLowerCase())
.toSet()
.whereType<String>()
.toList();
final summary = AnalyticsSummaryModel(
lemma: lemma,
xp: xp,
forms: forms,
exampleMessages: exampleMessages,
independantUseOccurrences: independantUseOccurrences,
assistedUseOccurrences: assistedUseOccurrences,
);
summaries.add(summary);
}
return summaries;
}
Future<List<AnalyticsSummaryModel>> _getMorphAnalytics() async {
final analyticsService = Matrix.of(context).analyticsDataService;
final morphs = await MorphsRepo.get();
final List<AnalyticsSummaryModel> summaries = [];
for (final feature in morphs.displayFeatures) {
final allTags = morphs
.getDisplayTags(feature.feature)
.map((tag) => tag.toLowerCase())
.toSet();
for (final morphTag in allTags) {
final id = ConstructIdentifier(
lemma: morphTag,
type: ConstructTypeEnum.morph,
category: feature.feature,
);
final uses = await analyticsService.getConstructUse(id);
final xp = uses.points;
final exampleMessages = await _getExampleMessages([uses]);
final allUses = uses.cappedUses;
int independantUseOccurrences = 0;
int assistedUseOccurrences = 0;
for (final use in allUses) {
use.useType == ConstructUseTypeEnum.wa
? independantUseOccurrences++
: assistedUseOccurrences++;
}
final forms = allUses
.map((e) => e.form?.toLowerCase())
.toSet()
.whereType<String>()
.toList();
final tagCopy = getGrammarCopy(
category: feature.feature,
lemma: morphTag,
context: context,
);
final summary = AnalyticsSummaryModel(
morphFeature: MorphFeaturesEnumExtension.fromString(
feature.feature,
).getDisplayCopy(context),
morphTag: tagCopy,
xp: xp,
forms: forms,
exampleMessages: exampleMessages,
independantUseOccurrences: independantUseOccurrences,
assistedUseOccurrences: assistedUseOccurrences,
);
summaries.add(summary);
}
}
return summaries;
}
Future<List<String>> _getExampleMessages(
List<ConstructUses> constructUses,
) async {
final allUses = constructUses
.map((e) => e.cappedUses)
.expand((e) => e)
.toList();
final List<PangeaMessageEvent> examples = [];
for (final OneConstructUse use in allUses) {
if (use.metadata.roomId == null) continue;
final client = Matrix.of(context).client;
final Room? room = client.getRoomById(use.metadata.roomId!);
if (room == null) continue;
if (use.useType.skillsEnumType != LearningSkillsEnum.writing ||
use.metadata.eventId == null ||
use.form == null ||
use.xp <= 0) {
continue;
}
final exampleIndex = examples.indexWhere(
(example) => example.eventId == use.metadata.eventId!,
);
if (exampleIndex != -1) continue;
if (use.metadata.roomId == null) continue;
Timeline? timeline = room.timeline;
if (room.timeline == null) {
timeline = await room.getTimeline();
}
final Event? event = await room.getEventById(use.metadata.eventId!);
if (event == null || event.senderId != client.userID) continue;
final PangeaMessageEvent pangeaMessageEvent = PangeaMessageEvent(
event: event,
timeline: timeline!,
ownMessage: event.senderId == client.userID,
);
examples.add(pangeaMessageEvent);
if (examples.length >= 5) break;
}
return examples.map((m) => m.messageDisplayText).toSet().toList();
}
List<int> _getExcelFileContent(
Map<ConstructTypeEnum, List<AnalyticsSummaryModel>> summaries,
) {
final excel = Excel.createExcel();
for (final entry in summaries.entries) {
final sheet = excel[entry.key.sheetname(context)];
final values = entry.key == ConstructTypeEnum.vocab
? AnalyticsSummaryEnum.vocabValues
: AnalyticsSummaryEnum.morphValues;
for (final key in values) {
sheet
.cell(
CellIndex.indexByColumnRow(
rowIndex: 0,
columnIndex: values.indexOf(key),
),
)
.value = TextCellValue(
key.header(context),
);
}
final rows = entry.value
.map((summary) => _formatExcelRow(summary, entry.key))
.toList();
for (int i = 0; i < rows.length; i++) {
final row = rows[i];
for (int j = 0; j < row.length; j++) {
final cell = row[j];
sheet
.cell(
CellIndex.indexByColumnRow(rowIndex: i + 2, columnIndex: j),
)
.value =
cell;
}
}
}
excel.setDefaultSheet(ConstructTypeEnum.vocab.sheetname(context));
excel.delete('Sheet1');
return excel.encode() ?? [];
}
String _getCSVFileContent(
ConstructTypeEnum type,
List<AnalyticsSummaryModel> summaries,
) {
final values = type == ConstructTypeEnum.vocab
? AnalyticsSummaryEnum.vocabValues
: AnalyticsSummaryEnum.morphValues;
final List<List<dynamic>> rows = [];
final headerRow = [];
for (final key in values) {
headerRow.add(key.header(context));
}
rows.add(headerRow);
for (final summary in summaries) {
final List<dynamic> row = [];
for (int i = 0; i < values.length; i++) {
final key = values[i];
final value = summary.getValue(key);
row.add(
value is List<String> ? value.map((v) => "\"$v\"").join(", ") : value,
);
}
rows.add(row);
}
final String fileString = const ListToCsvConverter().convert(rows);
return fileString;
}
List<CellValue> _formatExcelRow(
AnalyticsSummaryModel summary,
ConstructTypeEnum type,
) {
final List<CellValue> row = [];
final values = type == ConstructTypeEnum.vocab
? AnalyticsSummaryEnum.vocabValues
: AnalyticsSummaryEnum.morphValues;
for (int i = 0; i < values.length; i++) {
final key = values[i];
final value = summary.getValue(key);
if (value is int) {
row.add(IntCellValue(value));
} else if (value is String) {
row.add(TextCellValue(value));
} else if (value is List<String>) {
row.add(TextCellValue(value.map((v) => "\"$v\"").join(", ")));
}
}
return row;
}
@override
Widget build(BuildContext context) {
return Dialog(
child: Container(
constraints: const BoxConstraints(maxWidth: 400),
padding: const EdgeInsets.symmetric(vertical: 20),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
Text(
L10n.of(context).fileType,
style: TextStyle(
fontSize:
AppSettings.fontSizeFactor.value *
AppConfig.messageFontSize,
),
),
Padding(
padding: const EdgeInsets.all(8.0),
child: SegmentedButton<DownloadType>(
selected: {_downloadType},
onSelectionChanged: _downloading
? null
: (c) => _setDownloadType(c.first),
segments: [
ButtonSegment(
value: DownloadType.csv,
label: Text(L10n.of(context).commaSeparatedFile),
),
ButtonSegment(
value: DownloadType.xlsx,
label: Text(L10n.of(context).excelFile),
),
],
),
),
if (!_downloaded)
Padding(
padding: const EdgeInsets.fromLTRB(8.0, 16.0, 8.0, 8.0),
child: OutlinedButton(
onPressed: _downloading ? null : _downloadAnalytics,
child: _downloading
? const SizedBox(
height: 10,
width: 100,
child: LinearProgressIndicator(),
)
: Text(L10n.of(context).download),
),
),
AnimatedSize(
duration: FluffyThemes.animationDuration,
child: _statusText != null
? Padding(
padding: const EdgeInsets.all(8.0),
child: Text(_statusText!),
)
: const SizedBox(),
),
AnimatedSize(
duration: FluffyThemes.animationDuration,
child: kIsWeb && _downloaded
? Padding(
padding: const EdgeInsets.symmetric(horizontal: 16.0),
child: Text(
L10n.of(context).webDownloadPermissionMessage,
textAlign: TextAlign.center,
style: TextStyle(
color: Theme.of(context).disabledColor,
),
),
)
: const SizedBox(),
),
AnimatedSize(
duration: FluffyThemes.animationDuration,
child: _error != null
? Padding(
padding: const EdgeInsets.all(8.0),
child: ErrorIndicator(message: _error!),
)
: const SizedBox(),
),
],
),
),
);
}
}