* 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>
416 lines
14 KiB
Dart
416 lines
14 KiB
Dart
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:csv/csv.dart';
|
|
import 'package:excel/excel.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/space_analytics_summary_enum.dart';
|
|
import 'package:fluffychat/pangea/analytics_downloads/space_analytics_summary_model.dart';
|
|
import 'package:fluffychat/pangea/analytics_misc/construct_use_model.dart';
|
|
import 'package:fluffychat/pangea/analytics_misc/saved_analytics_extension.dart';
|
|
import 'package:fluffychat/pangea/analytics_settings/analytics_settings_extension.dart';
|
|
import 'package:fluffychat/pangea/common/utils/error_handler.dart';
|
|
import 'package:fluffychat/pangea/common/widgets/error_indicator.dart';
|
|
import 'package:fluffychat/pangea/download/download_file_util.dart';
|
|
import 'package:fluffychat/pangea/download/download_type_enum.dart';
|
|
import 'package:fluffychat/pangea/extensions/pangea_room_extension.dart';
|
|
import 'package:fluffychat/pangea/morphs/get_grammar_copy.dart';
|
|
import 'package:fluffychat/widgets/matrix.dart';
|
|
|
|
class DownloadAnalyticsDialog extends StatefulWidget {
|
|
final Room space;
|
|
final List<Room> analyticsRooms;
|
|
const DownloadAnalyticsDialog({
|
|
required this.space,
|
|
required this.analyticsRooms,
|
|
super.key,
|
|
});
|
|
|
|
@override
|
|
DownloadAnalyticsDialogState createState() => DownloadAnalyticsDialogState();
|
|
}
|
|
|
|
class DownloadAnalyticsDialogState extends State<DownloadAnalyticsDialog> {
|
|
bool _initialized = false;
|
|
bool _downloaded = false;
|
|
bool _downloading = false;
|
|
|
|
bool get _loading => _downloading || !_initialized;
|
|
|
|
Object? _error;
|
|
|
|
Map<String, int> _downloadStatuses = {};
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
_initialize();
|
|
}
|
|
|
|
Future<void> _initialize() async {
|
|
_downloadStatuses = Map.fromEntries(
|
|
widget.analyticsRooms.map((room) => MapEntry(room.creatorId!, 0)),
|
|
);
|
|
if (mounted) setState(() => _initialized = true);
|
|
}
|
|
|
|
DownloadType _downloadType = DownloadType.csv;
|
|
|
|
void _setDownloadType(DownloadType type) {
|
|
_clean();
|
|
if (mounted) setState(() => _downloadType = type);
|
|
}
|
|
|
|
void _clean() {
|
|
_error = null;
|
|
_downloading = false;
|
|
_downloaded = false;
|
|
_downloadStatuses = Map.fromEntries(
|
|
widget.analyticsRooms.map((room) => MapEntry(room.creatorId!, 0)),
|
|
);
|
|
}
|
|
|
|
Color _downloadStatusColor(String userID) {
|
|
final status = _downloadStatuses[userID];
|
|
if (status == 1) return Colors.yellow;
|
|
if (status == 2) return Colors.green;
|
|
if ((status ?? 0) < 0) return Colors.red;
|
|
return Colors.grey;
|
|
}
|
|
|
|
String? get _statusText {
|
|
if (_downloading) return L10n.of(context).downloading;
|
|
if (_downloaded) return L10n.of(context).downloadInitiated;
|
|
return null;
|
|
}
|
|
|
|
String? get userL2 =>
|
|
MatrixState.pangeaController.userController.userL2?.langCode;
|
|
|
|
Future<void> _runDownload() async {
|
|
try {
|
|
if (!mounted || userL2 == null) return;
|
|
setState(() {
|
|
_error = null;
|
|
_downloading = true;
|
|
});
|
|
|
|
final List<SpaceAnalyticsSummaryModel> summaries = [];
|
|
for (final room in widget.analyticsRooms) {
|
|
final summary = await _getAnalyticsModel(room);
|
|
if (summary != null) {
|
|
summaries.add(summary);
|
|
}
|
|
}
|
|
|
|
for (final userID in _downloadStatuses.keys) {
|
|
if (_downloadStatuses[userID] == 0) {
|
|
_downloadStatuses[userID] = -1;
|
|
summaries.add(SpaceAnalyticsSummaryModel.emptyModel(userID));
|
|
}
|
|
}
|
|
|
|
await _downloadSpaceAnalytics(summaries);
|
|
|
|
if (mounted) {
|
|
setState(() {
|
|
_downloading = false;
|
|
_downloaded = true;
|
|
});
|
|
}
|
|
} catch (e, s) {
|
|
ErrorHandler.logError(e: e, s: s, data: {});
|
|
|
|
_clean();
|
|
_error = e;
|
|
if (mounted) setState(() {});
|
|
}
|
|
}
|
|
|
|
Future<void> _downloadSpaceAnalytics(
|
|
List<SpaceAnalyticsSummaryModel> summaries,
|
|
) async {
|
|
final content = _downloadType == DownloadType.xlsx
|
|
? _getExcelFileContent(summaries)
|
|
: _getCSVFileContent(summaries);
|
|
|
|
final fileName =
|
|
"analytics_${widget.space.name}_${DateTime.now().toIso8601String()}.${_downloadType == DownloadType.xlsx ? 'xlsx' : 'csv'}";
|
|
|
|
await DownloadUtil.downloadFile(content, fileName, DownloadType.csv);
|
|
}
|
|
|
|
Future<SpaceAnalyticsSummaryModel?> _getAnalyticsModel(
|
|
Room analyticsRoom,
|
|
) async {
|
|
final String? userID = analyticsRoom.creatorId;
|
|
if (userID == null) return null;
|
|
|
|
SpaceAnalyticsSummaryModel? summary;
|
|
try {
|
|
_downloadStatuses[userID] = 1;
|
|
if (mounted) setState(() {});
|
|
|
|
final constructEvents = await analyticsRoom.getAnalyticsEvents(
|
|
userId: userID,
|
|
);
|
|
|
|
if (constructEvents == null) {
|
|
if (mounted) setState(() => _downloadStatuses[userID] = -1);
|
|
return SpaceAnalyticsSummaryModel.emptyModel(userID);
|
|
}
|
|
summary = SpaceAnalyticsSummaryModel.fromEvents(
|
|
userID,
|
|
constructEvents,
|
|
analyticsRoom.blockedConstructs,
|
|
analyticsRoom.archivedActivitiesCount,
|
|
);
|
|
if (mounted) setState(() => _downloadStatuses[userID] = 2);
|
|
} catch (e, s) {
|
|
ErrorHandler.logError(e: e, s: s, data: {"userID": userID});
|
|
if (mounted) setState(() => _downloadStatuses[userID] = -2);
|
|
}
|
|
|
|
return summary;
|
|
}
|
|
|
|
List<CellValue> _formatExcelRow(SpaceAnalyticsSummaryModel summary) {
|
|
final List<CellValue> row = [];
|
|
for (int i = 0; i < SpaceAnalyticsSummaryEnum.values.length; i++) {
|
|
final key = SpaceAnalyticsSummaryEnum.values[i];
|
|
final value = summary.getValue(key, context);
|
|
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.join(", ")));
|
|
}
|
|
}
|
|
return row;
|
|
}
|
|
|
|
List<int> _getExcelFileContent(List<SpaceAnalyticsSummaryModel> summaries) {
|
|
final excel = Excel.createExcel();
|
|
final sheet = excel['Sheet1'];
|
|
|
|
for (final key in SpaceAnalyticsSummaryEnum.values) {
|
|
sheet
|
|
.cell(CellIndex.indexByColumnRow(rowIndex: 0, columnIndex: key.index))
|
|
.value = TextCellValue(
|
|
key.header(L10n.of(context)),
|
|
);
|
|
}
|
|
|
|
final rows = summaries.map((summary) => _formatExcelRow(summary)).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;
|
|
}
|
|
}
|
|
return excel.encode() ?? [];
|
|
}
|
|
|
|
String _getCSVFileContent(List<SpaceAnalyticsSummaryModel> summaries) {
|
|
final List<List<dynamic>> rows = [];
|
|
final headerRow = [];
|
|
for (final key in SpaceAnalyticsSummaryEnum.values) {
|
|
headerRow.add(key.header(L10n.of(context)));
|
|
}
|
|
rows.add(headerRow);
|
|
|
|
for (final summary in summaries) {
|
|
final row = [];
|
|
for (int i = 0; i < SpaceAnalyticsSummaryEnum.values.length; i++) {
|
|
final key = SpaceAnalyticsSummaryEnum.values[i];
|
|
final value = summary.getValue(key, context);
|
|
if (value == null) continue;
|
|
value is List<String> ? row.add(value.join(", ")) : row.add(value);
|
|
}
|
|
rows.add(row);
|
|
}
|
|
|
|
final String fileString = const ListToCsvConverter().convert(rows);
|
|
return fileString;
|
|
}
|
|
|
|
String getCopy(ConstructUses use) {
|
|
return getGrammarCopy(
|
|
category: use.category,
|
|
lemma: use.lemma,
|
|
context: context,
|
|
) ??
|
|
use.lemma;
|
|
}
|
|
|
|
@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: (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),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
Padding(
|
|
padding: const EdgeInsets.all(8.0),
|
|
child: ConstrainedBox(
|
|
constraints: const BoxConstraints(maxHeight: 300, minHeight: 0),
|
|
child: ListView.builder(
|
|
shrinkWrap: true,
|
|
itemCount: widget.analyticsRooms.length,
|
|
itemBuilder: (context, index) {
|
|
final userId = widget.analyticsRooms[index].creatorId;
|
|
|
|
String tooltip = "";
|
|
if (_downloadStatuses[userId] == -1) {
|
|
tooltip = L10n.of(context).analyticsNotAvailable;
|
|
} else if (_downloadStatuses[userId] == -2) {
|
|
tooltip = L10n.of(context).failedFetchUserAnalytics;
|
|
}
|
|
|
|
return Padding(
|
|
padding: const EdgeInsets.all(4.0),
|
|
child: AnimatedOpacity(
|
|
duration: FluffyThemes.animationDuration,
|
|
opacity: (_downloadStatuses[userId] ?? 0) > 0 ? 1 : 0.5,
|
|
child: Row(
|
|
children: [
|
|
SizedBox(
|
|
width: 40,
|
|
height: 30,
|
|
child: (_downloadStatuses[userId] ?? 0) < 0
|
|
? const Icon(Icons.error_outline, size: 16)
|
|
: Center(
|
|
child: AnimatedContainer(
|
|
duration:
|
|
FluffyThemes.animationDuration,
|
|
height: 12,
|
|
width: 12,
|
|
decoration: BoxDecoration(
|
|
color: _downloadStatusColor(userId!),
|
|
borderRadius: BorderRadius.circular(
|
|
100,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
Flexible(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
Text(userId!),
|
|
if (tooltip.isNotEmpty)
|
|
Text(
|
|
tooltip,
|
|
style: const TextStyle(fontSize: 10),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
},
|
|
),
|
|
),
|
|
),
|
|
if (!_downloaded)
|
|
Padding(
|
|
padding: const EdgeInsets.fromLTRB(8.0, 16.0, 8.0, 8.0),
|
|
child: OutlinedButton(
|
|
onPressed: _loading || !_initialized ? null : _runDownload,
|
|
child: _initialized && !_loading
|
|
? Text(
|
|
_loading
|
|
? L10n.of(context).downloading
|
|
: L10n.of(context).download,
|
|
)
|
|
: const SizedBox(
|
|
height: 10,
|
|
width: 100,
|
|
child: LinearProgressIndicator(),
|
|
),
|
|
),
|
|
),
|
|
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: L10n.of(context).errorDownloading,
|
|
),
|
|
)
|
|
: const SizedBox(),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|