* chore: Nicer invite selection view
* chore: Do not request thousands of users on invite page
* build(deps): bump rexml from 3.3.6 to 3.3.9 in /ios
Bumps [rexml](https://github.com/ruby/rexml) from 3.3.6 to 3.3.9.
- [Release notes](https://github.com/ruby/rexml/releases)
- [Changelog](https://github.com/ruby/rexml/blob/master/NEWS.md)
- [Commits](https://github.com/ruby/rexml/compare/v3.3.6...v3.3.9)
---
updated-dependencies:
- dependency-name: rexml
dependency-type: indirect
...
Signed-off-by: dependabot[bot] <support@github.com>
* design: Highlight emoji only messages
* chore: Follow up emoji only messages
* Translated using Weblate (Galician)
Currently translated at 100.0% (672 of 672 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/gl/
* Translated using Weblate (Russian)
Currently translated at 99.7% (670 of 672 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/
* design: New login design
* chore: Improve spaces design
* chore: Improve spaces design
* chore: Improved UX for creating groups and spaces
* Translated using Weblate (German)
Currently translated at 100.0% (672 of 672 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/
* feat: Better wallpapers with blur and opacity sliders and improved styles page
* chore: Follow up wallpaper configs
* chore: Add max length to state messages
* chore: Follow up wallpaper design
* feat: Open account manage url when using MAS
* chore: follow up wellknown fetch
* Translated using Weblate (Arabic)
Currently translated at 100.0% (674 of 674 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ar/
* Translated using Weblate (Estonian)
Currently translated at 100.0% (674 of 674 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/
* Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 100.0% (674 of 674 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/
* Translated using Weblate (Indonesian)
Currently translated at 100.0% (674 of 674 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/id/
* Translated using Weblate (Finnish)
Currently translated at 79.0% (533 of 674 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/fi/
* Translated using Weblate (Latvian)
Currently translated at 100.0% (674 of 674 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/
* build: Add links to snapcraft.yaml file
* chore: Nicer empty page
* chore: Polish chat bubble colors
* chore: Follow up chat bubble design
* refactor: Remove unnecessary builder widget
* chore: Design adjustments
* chore: Follow up design
* refactor: Display two lines on new messages
* chore: Design follow up
* Translated using Weblate (Arabic)
Currently translated at 100.0% (678 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ar/
* Translated using Weblate (German)
Currently translated at 100.0% (678 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/
* Translated using Weblate (Estonian)
Currently translated at 99.7% (676 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/
* Translated using Weblate (Basque)
Currently translated at 100.0% (678 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/
* Translated using Weblate (Ukrainian)
Currently translated at 100.0% (678 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/
* Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 100.0% (678 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/
* chore: Follow up message bubbles
* chore: Follow up design
* chore: Follow up design
* chore: Follow up colors
* chore: Follow up homeserverpicker UX
* chore: Design follow up
* feat: Add about server page
* chore: Follow up update snackbar
* chore: Polish login design
* chore: Follow up login page
* chore: Follow up homeserver picker
* chore: Follow up appbar shadow
* refactor: Performance boost for avatar widget
* Revert "refactor: Performance boost for avatar widget"
This reverts commit 58577bb9e8.
* Translated using Weblate (Estonian)
Currently translated at 100.0% (678 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/
* Translated using Weblate (Ukrainian)
Currently translated at 100.0% (678 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/
* Translated using Weblate (Latvian)
Currently translated at 100.0% (678 of 678 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/
* Translated using Weblate (Arabic)
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ar/
* Translated using Weblate (Estonian)
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/
* Translated using Weblate (Basque)
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/
* Translated using Weblate (Galician)
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/gl/
* Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/
* Translated using Weblate (Indonesian)
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/id/
* Translated using Weblate (Latvian)
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/
* Translated using Weblate (Ukrainian)
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/
* Translated using Weblate (Korean)
Currently translated at 100.0% (687 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ko/
* chore: Follow up homeserver input field
* refactor: Move to upstream geolocator
* chore: Follow up send file dialog
* Translated using Weblate (Spanish)
Currently translated at 74.6% (513 of 687 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/
* refactor: Migrate to newer keyboard shortcuts package
* refactor: Remove keyboard shortcuts
This package right now
makes the web app
nearly unusable as it
throws multiple errors on
every key press. The
package seems to be
unmaintained. I tried
two other packages
and none of them worked.
* build: Update matrix dart sdk to 0.35.0
* chore: Better FluffyChat Logo for PWA
* build: (deps): bump samuelmeuli/action-snapcraft from 2 to 3
Bumps [samuelmeuli/action-snapcraft](https://github.com/samuelmeuli/action-snapcraft) from 2 to 3.
- [Release notes](https://github.com/samuelmeuli/action-snapcraft/releases)
- [Commits](https://github.com/samuelmeuli/action-snapcraft/compare/v2...v3)
---
updated-dependencies:
- dependency-name: samuelmeuli/action-snapcraft
dependency-type: direct:production
update-type: version-update:semver-major
...
Signed-off-by: dependabot[bot] <support@github.com>
* chore: Follow up send file dialog
* feat: Add markdown context actions for text input
* build: Update flutter to 3.24.5
* build: Remove snapcraft build workaround
* chore: Better error message when join room failed
* chore: Follow up join room
* chore: Make error dialog show full error
* chore: Follow up loading dialog
* chore: Follow up loading dialog
* build: Snapcraft from local build file
* chore: Follow up build snap
* chore: Follow up snapcraft in ci
* build: Revert build snapcraft changes
* build: Try downgrading flutter web auth
* chore: add hint in pubspec.yaml regarding flutter_web_auth_2
* Translated using Weblate (Estonian)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/
* Translated using Weblate (Galician)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/gl/
* Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/
* Translated using Weblate (Indonesian)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/id/
* Translated using Weblate (Irish)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ga/
* Translated using Weblate (Arabic)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ar/
* Translated using Weblate (Basque)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/
* Translated using Weblate (Ukrainian)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/
* Translated using Weblate (Latvian)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/
* Translated using Weblate (Italian)
Currently translated at 100.0% (688 of 688 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/it/
* Translated using Weblate (Estonian)
Currently translated at 100.0% (694 of 694 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/
* Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 100.0% (694 of 694 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/
* Translated using Weblate (Arabic)
Currently translated at 100.0% (694 of 694 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ar/
* Translated using Weblate (Basque)
Currently translated at 100.0% (694 of 694 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/
* Translated using Weblate (Irish)
Currently translated at 100.0% (694 of 694 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ga/
* Translated using Weblate (Indonesian)
Currently translated at 100.0% (694 of 694 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/id/
* Translated using Weblate (Latvian)
Currently translated at 100.0% (694 of 694 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/lv/
* Translated using Weblate (Arabic)
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ar/
* Translated using Weblate (Estonian)
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/
* Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/zh_Hans/
* Translated using Weblate (Irish)
Currently translated at 99.8% (694 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ga/
* Translated using Weblate (German)
Currently translated at 99.5% (692 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/
* fix: dont use thumbnails for emoticons
* chore: Improve presence performance
* Translated using Weblate (Basque)
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/
* Translated using Weblate (Galician)
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/gl/
* Translated using Weblate (Italian)
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/it/
* Translated using Weblate (Irish)
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ga/
* Translated using Weblate (Russian)
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/
* Translated using Weblate (Ukrainian)
Currently translated at 100.0% (695 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/
* Translated using Weblate (Catalan)
Currently translated at 95.1% (661 of 695 strings)
Translation: FluffyChat/Translations
Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ca/
* build: Bump version
* chore: Follow up send file dialog for images
* chore: Follow up send multiple images
* build: Add android build workaround for new flutter version
* build: Use file selector to save files
* chore: Follow up save file on desktop
* chore: Adjust default linux window height
* refactor: Update to new receive sharing intent package
* fluffychat merge
* fluffychat merge
* fluffychat merge
* fix android build
* fluffychat merge
* fluffychat merge
---------
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: Krille <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: josé m <correoxm@disroot.org>
Co-authored-by: v1s7 <v1s7@users.noreply.hosted.weblate.org>
Co-authored-by: Christian <christian-pauly@posteo.de>
Co-authored-by: Rex_sa <rex.sa@pm.me>
Co-authored-by: Priit Jõerüüt <hwlate@joeruut.com>
Co-authored-by: 大王叫我来巡山 <hamburger2048@users.noreply.hosted.weblate.org>
Co-authored-by: Linerly <linerly@proton.me>
Co-authored-by: Edgars Andersons <Edgars+Weblate@gaitenis.id.lv>
Co-authored-by: xabirequejo <xabi.rn@gmail.com>
Co-authored-by: Bezruchenko Simon <worcposj44@gmail.com>
Co-authored-by: Ihor Hordiichuk <igor_ck@outlook.com>
Co-authored-by: Bruno Roh <kane.roh429@gmail.com>
Co-authored-by: Kimby <kimisaes@naver.com>
Co-authored-by: Aindriú Mac Giolla Eoin <aindriu80@gmail.com>
Co-authored-by: Angelo Schirinzi <muten619@hotmail.it>
Co-authored-by: Marek Vospěl <marek@vospel.cz>
Co-authored-by: Александр (Alexandr1995) <stupino19951406@gmail.com>
505 lines
20 KiB
Dart
505 lines
20 KiB
Dart
import 'package:flutter/material.dart';
|
|
|
|
import 'package:adaptive_dialog/adaptive_dialog.dart';
|
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
|
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
|
import 'package:matrix/encryption.dart';
|
|
import 'package:matrix/matrix.dart';
|
|
|
|
import 'package:fluffychat/config/themes.dart';
|
|
import 'package:fluffychat/utils/error_reporter.dart';
|
|
import 'package:fluffychat/utils/fluffy_share.dart';
|
|
import 'package:fluffychat/utils/localized_exception_extension.dart';
|
|
import 'package:fluffychat/utils/platform_infos.dart';
|
|
import 'package:fluffychat/widgets/future_loading_dialog.dart';
|
|
import '../../utils/adaptive_bottom_sheet.dart';
|
|
import '../key_verification/key_verification_dialog.dart';
|
|
|
|
class BootstrapDialog extends StatefulWidget {
|
|
final bool wipe;
|
|
final Client client;
|
|
const BootstrapDialog({
|
|
super.key,
|
|
this.wipe = false,
|
|
required this.client,
|
|
});
|
|
|
|
Future<bool?> show(BuildContext context) => showAdaptiveBottomSheet(
|
|
context: context,
|
|
builder: (context) => this,
|
|
maxHeight: 600,
|
|
);
|
|
|
|
@override
|
|
BootstrapDialogState createState() => BootstrapDialogState();
|
|
}
|
|
|
|
class BootstrapDialogState extends State<BootstrapDialog> {
|
|
final TextEditingController _recoveryKeyTextEditingController =
|
|
TextEditingController();
|
|
|
|
late Bootstrap bootstrap;
|
|
|
|
String? _recoveryKeyInputError;
|
|
|
|
bool _recoveryKeyInputLoading = false;
|
|
|
|
String? titleText;
|
|
|
|
bool _recoveryKeyStored = false;
|
|
bool _recoveryKeyCopied = false;
|
|
|
|
bool? _storeInSecureStorage = false;
|
|
|
|
bool? _wipe;
|
|
|
|
String get _secureStorageKey =>
|
|
'ssss_recovery_key_${bootstrap.client.userID}';
|
|
|
|
bool get _supportsSecureStorage =>
|
|
PlatformInfos.isMobile || PlatformInfos.isDesktop;
|
|
|
|
String _getSecureStorageLocalizedName() {
|
|
if (PlatformInfos.isAndroid) {
|
|
return L10n.of(context).storeInAndroidKeystore;
|
|
}
|
|
if (PlatformInfos.isIOS || PlatformInfos.isMacOS) {
|
|
return L10n.of(context).storeInAppleKeyChain;
|
|
}
|
|
return L10n.of(context).storeSecurlyOnThisDevice;
|
|
}
|
|
|
|
@override
|
|
void initState() {
|
|
_createBootstrap(widget.wipe);
|
|
super.initState();
|
|
}
|
|
|
|
void _createBootstrap(bool wipe) async {
|
|
_wipe = wipe;
|
|
titleText = null;
|
|
_recoveryKeyStored = false;
|
|
bootstrap =
|
|
widget.client.encryption!.bootstrap(onUpdate: (_) => setState(() {}));
|
|
final key = await const FlutterSecureStorage().read(key: _secureStorageKey);
|
|
if (key == null) return;
|
|
_recoveryKeyTextEditingController.text = key;
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final theme = Theme.of(context);
|
|
_wipe ??= widget.wipe;
|
|
final buttons = <Widget>[];
|
|
Widget body = const CircularProgressIndicator.adaptive();
|
|
titleText = L10n.of(context).loadingPleaseWait;
|
|
|
|
if (bootstrap.newSsssKey?.recoveryKey != null &&
|
|
_recoveryKeyStored == false) {
|
|
final key = bootstrap.newSsssKey!.recoveryKey;
|
|
titleText = L10n.of(context).recoveryKey;
|
|
return Scaffold(
|
|
appBar: AppBar(
|
|
centerTitle: true,
|
|
leading: IconButton(
|
|
icon: const Icon(Icons.close),
|
|
onPressed: Navigator.of(context).pop,
|
|
),
|
|
title: Text(L10n.of(context).recoveryKey),
|
|
),
|
|
body: Center(
|
|
child: ConstrainedBox(
|
|
constraints:
|
|
const BoxConstraints(maxWidth: FluffyThemes.columnWidth * 1.5),
|
|
child: ListView(
|
|
padding: const EdgeInsets.all(16.0),
|
|
children: [
|
|
ListTile(
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 8.0),
|
|
trailing: CircleAvatar(
|
|
backgroundColor: Colors.transparent,
|
|
child: Icon(
|
|
Icons.info_outlined,
|
|
color: theme.colorScheme.primary,
|
|
),
|
|
),
|
|
subtitle: Text(L10n.of(context).chatBackupDescription),
|
|
),
|
|
const Divider(
|
|
height: 32,
|
|
thickness: 1,
|
|
),
|
|
TextField(
|
|
minLines: 2,
|
|
maxLines: 4,
|
|
readOnly: true,
|
|
style: const TextStyle(fontFamily: 'RobotoMono'),
|
|
controller: TextEditingController(text: key),
|
|
decoration: const InputDecoration(
|
|
contentPadding: EdgeInsets.all(16),
|
|
suffixIcon: Icon(Icons.key_outlined),
|
|
),
|
|
),
|
|
const SizedBox(height: 16),
|
|
if (_supportsSecureStorage)
|
|
CheckboxListTile.adaptive(
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 8.0),
|
|
value: _storeInSecureStorage,
|
|
activeColor: theme.colorScheme.primary,
|
|
onChanged: (b) {
|
|
setState(() {
|
|
_storeInSecureStorage = b;
|
|
});
|
|
},
|
|
title: Text(_getSecureStorageLocalizedName()),
|
|
subtitle:
|
|
Text(L10n.of(context).storeInSecureStorageDescription),
|
|
),
|
|
const SizedBox(height: 16),
|
|
CheckboxListTile.adaptive(
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 8.0),
|
|
value: _recoveryKeyCopied,
|
|
activeColor: theme.colorScheme.primary,
|
|
onChanged: (b) {
|
|
FluffyShare.share(key!, context);
|
|
setState(() => _recoveryKeyCopied = true);
|
|
},
|
|
title: Text(L10n.of(context).copyToClipboard),
|
|
subtitle: Text(L10n.of(context).saveKeyManuallyDescription),
|
|
),
|
|
const SizedBox(height: 16),
|
|
ElevatedButton.icon(
|
|
icon: const Icon(Icons.check_outlined),
|
|
label: Text(L10n.of(context).next),
|
|
onPressed:
|
|
(_recoveryKeyCopied || _storeInSecureStorage == true)
|
|
? () {
|
|
if (_storeInSecureStorage == true) {
|
|
const FlutterSecureStorage().write(
|
|
key: _secureStorageKey,
|
|
value: key,
|
|
);
|
|
}
|
|
setState(() => _recoveryKeyStored = true);
|
|
}
|
|
: null,
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
);
|
|
} else {
|
|
switch (bootstrap.state) {
|
|
case BootstrapState.loading:
|
|
break;
|
|
case BootstrapState.askWipeSsss:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.wipeSsss(_wipe!),
|
|
);
|
|
break;
|
|
case BootstrapState.askBadSsss:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.ignoreBadSecrets(true),
|
|
);
|
|
break;
|
|
case BootstrapState.askUseExistingSsss:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.useExistingSsss(!_wipe!),
|
|
);
|
|
break;
|
|
case BootstrapState.askUnlockSsss:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.unlockedSsss(),
|
|
);
|
|
break;
|
|
case BootstrapState.askNewSsss:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.newSsss(),
|
|
);
|
|
break;
|
|
case BootstrapState.openExistingSsss:
|
|
_recoveryKeyStored = true;
|
|
return Scaffold(
|
|
appBar: AppBar(
|
|
centerTitle: true,
|
|
leading: IconButton(
|
|
icon: const Icon(Icons.close),
|
|
onPressed: Navigator.of(context).pop,
|
|
),
|
|
title: Text(L10n.of(context).chatBackup),
|
|
),
|
|
body: Center(
|
|
child: ConstrainedBox(
|
|
constraints: const BoxConstraints(
|
|
maxWidth: FluffyThemes.columnWidth * 1.5,
|
|
),
|
|
child: ListView(
|
|
padding: const EdgeInsets.all(16.0),
|
|
children: [
|
|
ListTile(
|
|
contentPadding:
|
|
const EdgeInsets.symmetric(horizontal: 8.0),
|
|
trailing: Icon(
|
|
Icons.info_outlined,
|
|
color: theme.colorScheme.primary,
|
|
),
|
|
subtitle: Text(
|
|
L10n.of(context).pleaseEnterRecoveryKeyDescription,
|
|
),
|
|
),
|
|
const Divider(height: 32),
|
|
TextField(
|
|
minLines: 1,
|
|
maxLines: 2,
|
|
autocorrect: false,
|
|
readOnly: _recoveryKeyInputLoading,
|
|
autofillHints: _recoveryKeyInputLoading
|
|
? null
|
|
: [AutofillHints.password],
|
|
controller: _recoveryKeyTextEditingController,
|
|
style: const TextStyle(fontFamily: 'RobotoMono'),
|
|
decoration: InputDecoration(
|
|
contentPadding: const EdgeInsets.all(16),
|
|
hintStyle: TextStyle(
|
|
fontFamily: theme.textTheme.bodyLarge?.fontFamily,
|
|
),
|
|
prefixIcon: const Icon(Icons.key_outlined),
|
|
labelText: L10n.of(context).recoveryKey,
|
|
hintText: 'Es** **** **** ****',
|
|
errorText: _recoveryKeyInputError,
|
|
errorMaxLines: 2,
|
|
),
|
|
),
|
|
const SizedBox(height: 16),
|
|
ElevatedButton.icon(
|
|
style: ElevatedButton.styleFrom(
|
|
foregroundColor: theme.colorScheme.onPrimary,
|
|
backgroundColor: theme.colorScheme.primary,
|
|
),
|
|
icon: _recoveryKeyInputLoading
|
|
? const CircularProgressIndicator.adaptive()
|
|
: const Icon(Icons.lock_open_outlined),
|
|
label: Text(L10n.of(context).unlockOldMessages),
|
|
onPressed: _recoveryKeyInputLoading
|
|
? null
|
|
: () async {
|
|
setState(() {
|
|
_recoveryKeyInputError = null;
|
|
_recoveryKeyInputLoading = true;
|
|
});
|
|
try {
|
|
final key = _recoveryKeyTextEditingController
|
|
.text
|
|
.trim();
|
|
if (key.isEmpty) return;
|
|
await bootstrap.newSsssKey!.unlock(
|
|
keyOrPassphrase: key,
|
|
);
|
|
await bootstrap.openExistingSsss();
|
|
Logs().d('SSSS unlocked');
|
|
if (bootstrap.encryption.crossSigning.enabled) {
|
|
Logs().v(
|
|
'Cross signing is already enabled. Try to self-sign',
|
|
);
|
|
try {
|
|
await bootstrap
|
|
.client.encryption!.crossSigning
|
|
.selfSign(recoveryKey: key);
|
|
Logs().d('Successful selfsigned');
|
|
} catch (e, s) {
|
|
Logs().e(
|
|
'Unable to self sign with recovery key after successfully open existing SSSS',
|
|
e,
|
|
s,
|
|
);
|
|
}
|
|
}
|
|
} on InvalidPassphraseException catch (e) {
|
|
setState(
|
|
() => _recoveryKeyInputError =
|
|
e.toLocalizedString(context),
|
|
);
|
|
} on FormatException catch (_) {
|
|
setState(
|
|
() => _recoveryKeyInputError =
|
|
L10n.of(context).wrongRecoveryKey,
|
|
);
|
|
} catch (e, s) {
|
|
ErrorReporter(
|
|
context,
|
|
'Unable to open SSSS with recovery key',
|
|
).onErrorCallback(e, s);
|
|
setState(
|
|
() => _recoveryKeyInputError =
|
|
e.toLocalizedString(context),
|
|
);
|
|
} finally {
|
|
setState(
|
|
() => _recoveryKeyInputLoading = false,
|
|
);
|
|
}
|
|
},
|
|
),
|
|
const SizedBox(height: 16),
|
|
Row(
|
|
children: [
|
|
const Expanded(child: Divider()),
|
|
Padding(
|
|
padding: const EdgeInsets.all(12.0),
|
|
child: Text(L10n.of(context).or),
|
|
),
|
|
const Expanded(child: Divider()),
|
|
],
|
|
),
|
|
const SizedBox(height: 16),
|
|
ElevatedButton.icon(
|
|
icon: const Icon(Icons.cast_connected_outlined),
|
|
label: Text(L10n.of(context).transferFromAnotherDevice),
|
|
onPressed: _recoveryKeyInputLoading
|
|
? null
|
|
: () async {
|
|
final consent = await showOkCancelAlertDialog(
|
|
context: context,
|
|
title: L10n.of(context).verifyOtherDevice,
|
|
message: L10n.of(context)
|
|
.verifyOtherDeviceDescription,
|
|
okLabel: L10n.of(context).ok,
|
|
cancelLabel: L10n.of(context).cancel,
|
|
fullyCapitalizedForMaterial: false,
|
|
);
|
|
if (consent != OkCancelResult.ok) return;
|
|
final req = await showFutureLoadingDialog(
|
|
context: context,
|
|
delay: false,
|
|
future: () async {
|
|
await widget.client.updateUserDeviceKeys();
|
|
return widget.client
|
|
.userDeviceKeys[widget.client.userID!]!
|
|
.startVerification();
|
|
},
|
|
);
|
|
if (req.error != null) return;
|
|
await KeyVerificationDialog(request: req.result!)
|
|
.show(context);
|
|
Navigator.of(context, rootNavigator: false).pop();
|
|
},
|
|
),
|
|
const SizedBox(height: 16),
|
|
ElevatedButton.icon(
|
|
style: ElevatedButton.styleFrom(
|
|
backgroundColor: theme.colorScheme.errorContainer,
|
|
foregroundColor: theme.colorScheme.onErrorContainer,
|
|
),
|
|
icon: const Icon(Icons.delete_outlined),
|
|
label: Text(L10n.of(context).recoveryKeyLost),
|
|
onPressed: _recoveryKeyInputLoading
|
|
? null
|
|
: () async {
|
|
if (OkCancelResult.ok ==
|
|
await showOkCancelAlertDialog(
|
|
useRootNavigator: false,
|
|
context: context,
|
|
title: L10n.of(context).recoveryKeyLost,
|
|
message: L10n.of(context).wipeChatBackup,
|
|
okLabel: L10n.of(context).ok,
|
|
cancelLabel: L10n.of(context).cancel,
|
|
isDestructiveAction: true,
|
|
)) {
|
|
setState(() => _createBootstrap(true));
|
|
}
|
|
},
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
);
|
|
case BootstrapState.askWipeCrossSigning:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.wipeCrossSigning(_wipe!),
|
|
);
|
|
break;
|
|
case BootstrapState.askSetupCrossSigning:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.askSetupCrossSigning(
|
|
setupMasterKey: true,
|
|
setupSelfSigningKey: true,
|
|
setupUserSigningKey: true,
|
|
),
|
|
);
|
|
break;
|
|
case BootstrapState.askWipeOnlineKeyBackup:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.wipeOnlineKeyBackup(_wipe!),
|
|
);
|
|
|
|
break;
|
|
case BootstrapState.askSetupOnlineKeyBackup:
|
|
WidgetsBinding.instance.addPostFrameCallback(
|
|
(_) => bootstrap.askSetupOnlineKeyBackup(true),
|
|
);
|
|
break;
|
|
case BootstrapState.error:
|
|
titleText = L10n.of(context).oopsSomethingWentWrong;
|
|
body = const Icon(Icons.error_outline, color: Colors.red, size: 80);
|
|
buttons.add(
|
|
OutlinedButton(
|
|
onPressed: () =>
|
|
Navigator.of(context, rootNavigator: false).pop<bool>(false),
|
|
child: Text(L10n.of(context).close),
|
|
),
|
|
);
|
|
break;
|
|
case BootstrapState.done:
|
|
titleText = L10n.of(context).everythingReady;
|
|
body = Column(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
const Icon(
|
|
Icons.check_circle_rounded,
|
|
size: 120,
|
|
color: Colors.green,
|
|
),
|
|
const SizedBox(height: 16),
|
|
Text(
|
|
L10n.of(context).yourChatBackupHasBeenSetUp,
|
|
style: const TextStyle(fontSize: 20),
|
|
),
|
|
const SizedBox(height: 16),
|
|
],
|
|
);
|
|
buttons.add(
|
|
OutlinedButton(
|
|
onPressed: () =>
|
|
Navigator.of(context, rootNavigator: false).pop<bool>(false),
|
|
child: Text(L10n.of(context).close),
|
|
),
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return Scaffold(
|
|
appBar: AppBar(
|
|
leading: Center(
|
|
child: CloseButton(
|
|
onPressed: () =>
|
|
Navigator.of(context, rootNavigator: false).pop<bool>(true),
|
|
),
|
|
),
|
|
title: Text(titleText ?? L10n.of(context).loadingPleaseWait),
|
|
),
|
|
body: Center(
|
|
child: Column(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
body,
|
|
const SizedBox(height: 8),
|
|
...buttons,
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|