fluffychat/lib/pages/key_verification/key_verification_dialog.dart

417 lines
14 KiB
Dart
Raw Permalink Normal View History

import 'dart:convert';
2021-10-26 18:50:34 +02:00
import 'dart:ui';
2020-11-22 22:48:10 +01:00
import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
2021-10-26 18:50:34 +02:00
import 'package:adaptive_dialog/adaptive_dialog.dart';
import 'package:flutter_gen/gen_l10n/l10n.dart';
2020-12-25 09:58:34 +01:00
import 'package:future_loading_dialog/future_loading_dialog.dart';
2021-10-26 18:50:34 +02:00
import 'package:matrix/encryption.dart';
import 'package:matrix/matrix.dart';
import 'package:fluffychat/utils/platform_infos.dart';
import 'package:fluffychat/widgets/avatar.dart';
2021-11-09 21:32:16 +01:00
import '../../utils/beautify_string_extension.dart';
import '../../widgets/adaptive_flat_button.dart';
2020-11-22 22:48:10 +01:00
class KeyVerificationDialog extends StatefulWidget {
Future<void> show(BuildContext context) => PlatformInfos.isCupertinoStyle
2021-02-24 12:17:23 +01:00
? showCupertinoDialog(
context: context,
2021-04-09 15:53:26 +02:00
barrierDismissible: true,
2021-02-24 12:17:23 +01:00
builder: (context) => this,
2021-05-23 15:02:36 +02:00
useRootNavigator: false,
2021-02-24 12:17:23 +01:00
)
: showDialog(
context: context,
2021-04-09 15:53:26 +02:00
barrierDismissible: true,
2021-02-24 12:17:23 +01:00
builder: (context) => this,
2021-05-23 15:02:36 +02:00
useRootNavigator: false,
2021-02-24 12:17:23 +01:00
);
2020-06-25 16:29:06 +02:00
final KeyVerification request;
2021-10-14 18:09:30 +02:00
const KeyVerificationDialog({
2022-01-29 12:35:03 +01:00
Key? key,
required this.request,
2021-10-14 18:09:30 +02:00
}) : super(key: key);
2020-06-25 16:29:06 +02:00
@override
_KeyVerificationPageState createState() => _KeyVerificationPageState();
}
2020-11-22 22:48:10 +01:00
class _KeyVerificationPageState extends State<KeyVerificationDialog> {
2022-01-29 12:35:03 +01:00
void Function()? originalOnUpdate;
late final List<dynamic> sasEmoji;
2020-06-25 16:29:06 +02:00
@override
void initState() {
originalOnUpdate = widget.request.onUpdate;
widget.request.onUpdate = () {
2022-01-29 12:35:03 +01:00
originalOnUpdate?.call();
setState(() {});
2020-06-25 16:29:06 +02:00
};
widget.request.client.getProfileFromUserId(widget.request.userId).then((p) {
profile = p;
2022-01-29 12:35:03 +01:00
setState(() {});
2020-06-25 16:29:06 +02:00
});
rootBundle.loadString('assets/sas-emoji.json').then((e) {
sasEmoji = json.decode(e);
2022-01-29 12:35:03 +01:00
setState(() {});
});
2020-06-25 16:29:06 +02:00
super.initState();
}
@override
void dispose() {
widget.request.onUpdate =
originalOnUpdate; // don't want to get updates anymore
if (![KeyVerificationState.error, KeyVerificationState.done]
.contains(widget.request.state)) {
widget.request.cancel('m.user');
}
super.dispose();
}
2022-01-29 12:35:03 +01:00
Profile? profile;
2020-06-25 16:29:06 +02:00
2021-10-14 18:09:30 +02:00
Future<void> checkInput(String input) async {
2022-01-29 12:35:03 +01:00
if (input.isEmpty) return;
2021-10-14 18:09:30 +02:00
final valid = await showFutureLoadingDialog(
context: context,
future: () async {
// make sure the loading spinner shows before we test the keys
await Future.delayed(const Duration(milliseconds: 100));
var valid = false;
try {
await widget.request.openSSSS(keyOrPassphrase: input);
valid = true;
} catch (_) {
valid = false;
}
return valid;
});
if (valid.error != null) {
await showOkAlertDialog(
useRootNavigator: false,
context: context,
2022-01-29 12:35:03 +01:00
message: L10n.of(context)!.incorrectPassphraseOrKey,
2021-10-14 18:09:30 +02:00
);
}
}
2020-06-25 16:29:06 +02:00
@override
Widget build(BuildContext context) {
2022-01-29 12:35:03 +01:00
User? user;
2021-10-10 12:11:39 +02:00
final directChatId =
widget.request.client.getDirectChatFromUserId(widget.request.userId);
if (directChatId != null) {
user = widget.request.client
2022-01-29 12:35:03 +01:00
.getRoomById(directChatId)!
.unsafeGetUserFromMemoryOrFallback(widget.request.userId);
2021-10-10 12:11:39 +02:00
}
final displayName =
2022-01-29 12:35:03 +01:00
user?.calcDisplayname() ?? widget.request.userId.localpart!;
var title = Text(L10n.of(context)!.verifyTitle);
2020-06-25 16:29:06 +02:00
Widget body;
final buttons = <Widget>[];
switch (widget.request.state) {
case KeyVerificationState.askSSSS:
// prompt the user for their ssss passphrase / key
final textEditingController = TextEditingController();
String input;
body = Container(
2021-10-14 18:09:30 +02:00
margin: const EdgeInsets.only(left: 8.0, right: 8.0),
2020-06-25 16:29:06 +02:00
child: Column(
2021-03-04 12:28:06 +01:00
mainAxisSize: MainAxisSize.min,
2020-06-25 16:29:06 +02:00
children: <Widget>[
2022-01-29 12:35:03 +01:00
Text(L10n.of(context)!.askSSSSSign,
2021-10-14 18:09:30 +02:00
style: const TextStyle(fontSize: 20)),
2020-06-25 16:29:06 +02:00
Container(height: 10),
TextField(
controller: textEditingController,
autofocus: false,
autocorrect: false,
onSubmitted: (s) {
input = s;
2021-10-14 18:09:30 +02:00
checkInput(input);
2020-06-25 16:29:06 +02:00
},
minLines: 1,
maxLines: 1,
obscureText: true,
decoration: InputDecoration(
2022-01-29 12:35:03 +01:00
hintText: L10n.of(context)!.passphraseOrKey,
2020-06-25 16:29:06 +02:00
prefixStyle: TextStyle(color: Theme.of(context).primaryColor),
suffixStyle: TextStyle(color: Theme.of(context).primaryColor),
2021-10-14 18:09:30 +02:00
border: const OutlineInputBorder(),
2020-06-25 16:29:06 +02:00
),
),
],
),
);
2020-12-05 13:03:57 +01:00
buttons.add(AdaptiveFlatButton(
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.submit,
2021-10-26 18:47:05 +02:00
onPressed: () => checkInput(textEditingController.text),
2020-06-25 16:29:06 +02:00
));
2020-12-05 13:03:57 +01:00
buttons.add(AdaptiveFlatButton(
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.skip,
2020-06-25 16:29:06 +02:00
onPressed: () => widget.request.openSSSS(skip: true),
));
break;
case KeyVerificationState.askAccept:
2022-01-29 12:35:03 +01:00
title = Text(L10n.of(context)!.newVerificationRequest);
2021-10-10 12:11:39 +02:00
body = Column(
mainAxisSize: MainAxisSize.min,
children: [
Row(children: [
if (!PlatformInfos.isCupertinoStyle)
2021-11-20 10:42:23 +01:00
Avatar(mxContent: user?.avatarUrl, name: displayName),
2021-10-14 18:09:30 +02:00
const SizedBox(width: 12),
2021-10-10 12:11:39 +02:00
Expanded(
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: PlatformInfos.isCupertinoStyle
? CrossAxisAlignment.center
: CrossAxisAlignment.start,
children: [
Text(
displayName,
2021-10-14 18:09:30 +02:00
style: const TextStyle(fontSize: 16),
2021-10-10 12:11:39 +02:00
),
Text(
'${widget.request.userId} - ${widget.request.deviceId}',
2021-10-14 18:09:30 +02:00
style: const TextStyle(
2021-10-10 12:11:39 +02:00
fontWeight: FontWeight.w300,
fontSize: 14,
),
),
],
),
),
]),
const SizedBox(height: 16),
Image.asset('assets/verification.png', fit: BoxFit.contain),
const SizedBox(height: 16),
Text(
2022-01-29 12:35:03 +01:00
L10n.of(context)!.askVerificationRequest(displayName),
2021-10-10 12:11:39 +02:00
)
],
2020-06-25 16:29:06 +02:00
);
2020-12-05 13:03:57 +01:00
buttons.add(AdaptiveFlatButton(
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.reject,
2021-10-10 12:11:39 +02:00
textColor: Colors.red,
2021-10-26 18:47:05 +02:00
onPressed: () => widget.request
.rejectVerification()
.then((_) => Navigator.of(context, rootNavigator: false).pop()),
2020-06-25 16:29:06 +02:00
));
2021-10-10 12:11:39 +02:00
buttons.add(AdaptiveFlatButton(
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.accept,
2021-10-10 12:11:39 +02:00
onPressed: () => widget.request.acceptVerification(),
));
2020-06-25 16:29:06 +02:00
break;
case KeyVerificationState.waitingAccept:
body = Column(
2021-03-04 12:28:06 +01:00
mainAxisSize: MainAxisSize.min,
2020-06-25 16:29:06 +02:00
children: <Widget>[
2021-10-10 12:11:39 +02:00
Image.asset('assets/verification.png', fit: BoxFit.contain),
const SizedBox(height: 16),
2021-10-14 18:09:30 +02:00
const CircularProgressIndicator.adaptive(strokeWidth: 2),
2021-10-10 12:11:39 +02:00
const SizedBox(height: 16),
2020-11-22 14:44:56 +01:00
Text(
2022-01-29 12:35:03 +01:00
L10n.of(context)!.waitingPartnerAcceptRequest,
2020-11-22 14:44:56 +01:00
textAlign: TextAlign.center,
),
2020-06-25 16:29:06 +02:00
],
);
2021-02-20 10:28:04 +01:00
final key = widget.request.client.userDeviceKeys[widget.request.userId]
2022-01-29 12:35:03 +01:00
?.deviceKeys[widget.request.deviceId];
2021-02-20 10:28:04 +01:00
if (key != null) {
buttons.add(AdaptiveFlatButton(
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.verifyManual,
2021-02-20 10:28:04 +01:00
onPressed: () async {
final result = await showOkCancelAlertDialog(
2021-05-23 15:02:36 +02:00
useRootNavigator: false,
2021-02-20 10:28:04 +01:00
context: context,
2022-01-29 12:35:03 +01:00
title: L10n.of(context)!.verifyManual,
message: key.ed25519Key?.beautified ?? 'Key not found',
2021-02-20 10:28:04 +01:00
);
if (result == OkCancelResult.ok) {
await key.setVerified(true);
}
await widget.request.cancel();
2021-02-24 12:17:23 +01:00
Navigator.of(context, rootNavigator: false).pop();
2021-02-20 10:28:04 +01:00
},
));
}
2020-06-25 16:29:06 +02:00
break;
case KeyVerificationState.askSas:
2020-07-18 16:05:33 +02:00
TextSpan compareWidget;
2020-06-25 16:29:06 +02:00
// maybe add a button to switch between the two and only determine default
// view for if "emoji" is a present sasType or not?
String compareText;
if (widget.request.sasTypes.contains('emoji')) {
2022-01-29 12:35:03 +01:00
compareText = L10n.of(context)!.compareEmojiMatch;
2020-07-18 16:05:33 +02:00
compareWidget = TextSpan(
children: widget.request.sasEmojis
.map((e) => WidgetSpan(child: _Emoji(e, sasEmoji)))
2020-07-18 16:05:33 +02:00
.toList(),
);
2020-06-25 16:29:06 +02:00
} else {
2022-01-29 12:35:03 +01:00
compareText = L10n.of(context)!.compareNumbersMatch;
2020-06-25 16:29:06 +02:00
final numbers = widget.request.sasNumbers;
2020-07-18 16:05:33 +02:00
final numbstr = '${numbers[0]}-${numbers[1]}-${numbers[2]}';
compareWidget =
2021-10-14 18:09:30 +02:00
TextSpan(text: numbstr, style: const TextStyle(fontSize: 40));
2020-06-25 16:29:06 +02:00
}
body = Column(
2021-03-04 12:28:06 +01:00
mainAxisSize: MainAxisSize.min,
2020-06-25 16:29:06 +02:00
children: <Widget>[
2020-11-22 14:44:56 +01:00
Center(
2020-11-22 11:46:31 +01:00
child: Text(
compareText,
2021-10-14 18:09:30 +02:00
style: const TextStyle(fontSize: 16),
2020-11-22 11:46:31 +01:00
textAlign: TextAlign.center,
),
2020-06-25 16:29:06 +02:00
),
2021-10-14 18:09:30 +02:00
const SizedBox(height: 10),
2020-07-18 16:05:33 +02:00
Text.rich(
compareWidget,
2020-06-25 16:29:06 +02:00
textAlign: TextAlign.center,
),
],
);
2020-12-05 13:03:57 +01:00
buttons.add(AdaptiveFlatButton(
2020-11-22 22:48:10 +01:00
textColor: Colors.red,
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.theyDontMatch,
2020-06-25 16:29:06 +02:00
onPressed: () => widget.request.rejectSas(),
));
buttons.add(AdaptiveFlatButton(
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.theyMatch,
onPressed: () => widget.request.acceptSas(),
));
2020-06-25 16:29:06 +02:00
break;
case KeyVerificationState.waitingSas:
2021-04-14 10:37:15 +02:00
final acceptText = widget.request.sasTypes.contains('emoji')
2022-01-29 12:35:03 +01:00
? L10n.of(context)!.waitingPartnerEmoji
: L10n.of(context)!.waitingPartnerNumbers;
2020-06-25 16:29:06 +02:00
body = Column(
2021-03-04 12:28:06 +01:00
mainAxisSize: MainAxisSize.min,
2020-06-25 16:29:06 +02:00
children: <Widget>[
2021-10-14 18:09:30 +02:00
const CircularProgressIndicator.adaptive(strokeWidth: 2),
const SizedBox(height: 10),
2020-11-22 14:44:56 +01:00
Text(
acceptText,
textAlign: TextAlign.center,
),
2020-06-25 16:29:06 +02:00
],
);
break;
case KeyVerificationState.done:
body = Column(
2021-03-04 12:28:06 +01:00
mainAxisSize: MainAxisSize.min,
2020-06-25 16:29:06 +02:00
children: <Widget>[
2021-10-14 18:09:30 +02:00
const Icon(Icons.check_circle_outlined,
color: Colors.green, size: 200.0),
const SizedBox(height: 10),
2020-11-22 14:44:56 +01:00
Text(
2022-01-29 12:35:03 +01:00
L10n.of(context)!.verifySuccess,
2020-11-22 14:44:56 +01:00
textAlign: TextAlign.center,
),
2020-06-25 16:29:06 +02:00
],
);
2020-12-05 13:03:57 +01:00
buttons.add(AdaptiveFlatButton(
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.close,
2021-02-24 12:17:23 +01:00
onPressed: () => Navigator.of(context, rootNavigator: false).pop(),
2020-06-25 16:29:06 +02:00
));
break;
case KeyVerificationState.error:
body = Column(
2021-03-04 12:28:06 +01:00
mainAxisSize: MainAxisSize.min,
2020-06-25 16:29:06 +02:00
children: <Widget>[
2021-10-14 18:09:30 +02:00
const Icon(Icons.cancel, color: Colors.red, size: 200.0),
const SizedBox(height: 10),
2020-06-25 16:29:06 +02:00
Text(
2020-11-22 14:44:56 +01:00
'Error ${widget.request.canceledCode}: ${widget.request.canceledReason}',
textAlign: TextAlign.center,
),
2020-06-25 16:29:06 +02:00
],
);
2021-02-27 07:53:34 +01:00
buttons.add(AdaptiveFlatButton(
2022-01-29 12:35:03 +01:00
label: L10n.of(context)!.close,
2021-02-24 12:17:23 +01:00
onPressed: () => Navigator.of(context, rootNavigator: false).pop(),
2020-06-25 16:29:06 +02:00
));
break;
}
final content = SingleChildScrollView(
scrollDirection: Axis.vertical,
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
2021-10-14 18:09:30 +02:00
const SizedBox(height: 16),
body,
],
),
2020-11-22 22:48:10 +01:00
);
if (PlatformInfos.isCupertinoStyle) {
return CupertinoAlertDialog(
title: title,
content: content,
actions: buttons,
);
}
return AlertDialog(
title: title,
content: content,
actions: buttons,
2020-06-25 16:29:06 +02:00
);
}
}
class _Emoji extends StatelessWidget {
final KeyVerificationEmoji emoji;
2022-01-29 12:35:03 +01:00
final List<dynamic>? sasEmoji;
2020-06-25 16:29:06 +02:00
2021-10-14 18:09:30 +02:00
const _Emoji(this.emoji, this.sasEmoji);
String getLocalizedName() {
2022-01-29 12:35:03 +01:00
final sasEmoji = this.sasEmoji;
if (sasEmoji == null) {
// asset is still being loaded
return emoji.name;
}
2022-01-29 12:35:03 +01:00
final translations = Map<String, String?>.from(
sasEmoji[emoji.number]['translated_descriptions']);
translations['en'] = emoji.name;
for (final locale in window.locales) {
final wantLocaleParts = locale.toString().split('_');
final wantLanguage = wantLocaleParts.removeAt(0);
for (final haveLocale in translations.keys) {
final haveLocaleParts = haveLocale.split('_');
final haveLanguage = haveLocaleParts.removeAt(0);
if (haveLanguage == wantLanguage &&
(Set.from(haveLocaleParts)..removeAll(wantLocaleParts)).isEmpty &&
(translations[haveLocale]?.isNotEmpty ?? false)) {
2022-01-29 12:35:03 +01:00
return translations[haveLocale]!;
}
}
}
return emoji.name;
}
2020-06-25 16:29:06 +02:00
@override
Widget build(BuildContext context) {
return Column(
mainAxisSize: MainAxisSize.min,
children: <Widget>[
2021-10-14 18:09:30 +02:00
Text(emoji.emoji, style: const TextStyle(fontSize: 50)),
Text(getLocalizedName()),
2021-10-14 18:09:30 +02:00
const SizedBox(height: 10, width: 5),
2020-06-25 16:29:06 +02:00
],
);
}
}