adguard-home-manager/lib/screens/settings/encryption/encryption.dart

729 lines
30 KiB
Dart
Raw Normal View History

// ignore_for_file: use_build_context_synchronously
2022-10-22 23:06:27 +02:00
import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
2024-02-07 23:35:20 +01:00
import 'package:adguard_home_manager/widgets/custom_checkbox_list_tile.dart';
2022-10-25 21:12:00 +02:00
import 'package:adguard_home_manager/widgets/section_label.dart';
2022-10-27 22:29:23 +02:00
import 'package:adguard_home_manager/widgets/custom_switch_list_tile.dart';
import 'package:adguard_home_manager/screens/settings/encryption/status.dart';
2023-12-11 13:54:47 +01:00
import 'package:adguard_home_manager/screens/settings/encryption/reset_settings_modal.dart';
2022-10-23 19:27:23 +02:00
import 'package:adguard_home_manager/screens/settings/encryption/custom_text_field.dart';
import 'package:adguard_home_manager/screens/settings/encryption/master_switch.dart';
import 'package:adguard_home_manager/screens/settings/encryption/encryption_functions.dart';
2023-03-24 21:48:34 +01:00
import 'package:adguard_home_manager/screens/settings/encryption/error_message.dart';
2022-10-22 23:06:27 +02:00
2023-11-19 22:52:40 +01:00
import 'package:adguard_home_manager/constants/enums.dart';
import 'package:adguard_home_manager/models/encryption.dart';
import 'package:adguard_home_manager/classes/process_modal.dart';
2023-10-29 02:47:14 +01:00
import 'package:adguard_home_manager/functions/desktop_mode.dart';
2023-03-24 21:48:34 +01:00
import 'package:adguard_home_manager/functions/base64.dart';
import 'package:adguard_home_manager/functions/snackbar.dart';
2022-10-22 23:06:27 +02:00
import 'package:adguard_home_manager/providers/app_config_provider.dart';
import 'package:adguard_home_manager/providers/servers_provider.dart';
2023-11-19 22:52:40 +01:00
class EncryptionSettings extends StatefulWidget {
const EncryptionSettings({super.key});
2022-10-22 23:06:27 +02:00
@override
2023-11-19 22:52:40 +01:00
State<EncryptionSettings> createState() => _EncryptionSettingsState();
2022-10-22 23:06:27 +02:00
}
2023-11-19 22:52:40 +01:00
class _EncryptionSettingsState extends State<EncryptionSettings> {
LoadStatus loadStatus = LoadStatus.loading;
2022-10-23 19:27:23 +02:00
2022-10-22 23:06:27 +02:00
bool enabled = false;
2024-02-07 23:35:20 +01:00
bool? _plainDns;
2022-10-22 23:06:27 +02:00
final TextEditingController domainNameController = TextEditingController();
String? domainError;
bool redirectHttps = false;
final TextEditingController httpsPortController = TextEditingController();
String? httpsPortError;
final TextEditingController tlsPortController = TextEditingController();
String? tlsPortError;
final TextEditingController dnsOverQuicPortController = TextEditingController();
String? dnsOverQuicPortError;
int certificateOption = 0;
final TextEditingController certificatePathController = TextEditingController();
String? certificatePathError;
final TextEditingController certificateContentController = TextEditingController();
String? certificateContentError;
int privateKeyOption = 0;
2022-10-29 01:20:01 +02:00
bool existsPreviousKey = false;
2022-10-22 23:06:27 +02:00
bool usePreviouslySavedKey = false;
final TextEditingController privateKeyPathController = TextEditingController();
String? privateKeyPathError;
final TextEditingController pastePrivateKeyController = TextEditingController();
String? pastePrivateKeyError;
2022-10-23 19:27:23 +02:00
bool localValidationValid = false;
String? validDataError;
2024-02-07 23:44:32 +01:00
bool _dataValidApi = true;
2022-10-22 23:06:27 +02:00
2023-11-24 20:44:56 +01:00
EncryptionValidation? certKeyValid;
String? encryptionResultMessage;
2022-10-28 01:32:48 +02:00
bool formEdited = false;
2022-10-27 22:29:23 +02:00
2022-10-22 23:06:27 +02:00
void fetchData({bool? showRefreshIndicator}) async {
2023-11-19 22:52:40 +01:00
setState(() => loadStatus = LoadStatus.loading);
final result = await Provider.of<ServersProvider>(context, listen: false).apiClient2!.getEncryptionSettings();
if (!mounted) return;
if (result.successful == true) {
2023-12-09 04:06:35 +01:00
final data = result.content as EncryptionData;
2023-11-19 22:52:40 +01:00
await checkValidDataApi(data: data.toJson());
if (!mounted) return;
setState(() {
enabled = data.enabled;
domainNameController.text = data.serverName ?? '';
redirectHttps = data.forceHttps ?? false;
httpsPortController.text = data.portHttps != null ? data.portHttps.toString() : '';
tlsPortController.text = data.portDnsOverTls != null ? data.portDnsOverTls.toString() : '';
dnsOverQuicPortController.text = data.portDnsOverQuic != null ? data.portDnsOverQuic.toString() : '';
if (data.certificateChain != '') {
certificateOption = 1;
certificateContentController.text = decodeBase64(data.certificateChain);
2023-07-09 22:29:41 +02:00
}
2023-11-19 22:52:40 +01:00
else {
certificateOption = 0;
certificatePathController.text = data.certificatePath;
}
if (data.privateKey != '' || data.privateKeySaved == true) {
privateKeyOption = 1;
}
else {
privateKeyOption = 0;
privateKeyPathController.text = data.privateKeyPath;
}
usePreviouslySavedKey = data.privateKeySaved;
2024-02-07 23:35:20 +01:00
_plainDns = data.servePlainDns;
2023-11-19 22:52:40 +01:00
loadStatus = LoadStatus.loaded;
});
}
else {
setState(() => loadStatus = LoadStatus.error);
}
}
2022-10-23 15:41:25 +02:00
Future checkValidDataApi({Map<String, dynamic>? data}) async {
2023-11-19 22:52:40 +01:00
final result = await Provider.of<ServersProvider>(context, listen: false).apiClient2!.checkEncryptionSettings(
data: data ?? {
"enabled": enabled,
"server_name": domainNameController.text,
"force_https": redirectHttps,
"port_https": httpsPortController.text != '' ? int.parse(httpsPortController.text) : null,
"port_dns_over_tls": tlsPortController.text != '' ? int.parse(tlsPortController.text) : null,
"port_dns_over_quic": dnsOverQuicPortController.text != '' ? int.parse(dnsOverQuicPortController.text) : null,
if (certificateOption == 1) "certificate_chain": encodeBase64(certificateContentController.text),
if (privateKeyOption == 1 && usePreviouslySavedKey == false) "private_key": encodeBase64(pastePrivateKeyController.text),
"private_key_saved": usePreviouslySavedKey,
if (certificateOption == 0) "certificate_path": certificatePathController.text,
if (privateKeyOption == 0) "private_key_path": privateKeyPathController.text,
}
);
2023-11-24 01:26:44 +01:00
if (!mounted) return;
if (result.successful == true) {
2023-11-24 20:44:56 +01:00
final data = result.content as EncryptionValidationResult;
if (data.isObject == true) {
final object = data.encryptionValidation!;
setState(() {
if (object.warningValidation != null && object.warningValidation != '') {
2024-02-07 23:44:32 +01:00
_dataValidApi = false;
2023-11-24 20:44:56 +01:00
validDataError = object.warningValidation;
}
else {
2024-02-07 23:44:32 +01:00
_dataValidApi = true;
2023-11-24 20:44:56 +01:00
validDataError = null;
}
certKeyValid = object;
});
}
else {
setState(() {
encryptionResultMessage = data.message;
2024-02-07 23:44:32 +01:00
_dataValidApi = false;
2023-11-24 20:44:56 +01:00
});
}
}
}
2022-10-28 01:32:48 +02:00
bool checkcertKeyValid() {
if (
domainNameController.text != '' &&
domainError == null &&
httpsPortController.text != '' &&
httpsPortError == null &&
tlsPortController.text != '' &&
tlsPortError == null &&
dnsOverQuicPortController.text != '' &&
2023-05-20 01:16:33 +02:00
dnsOverQuicPortError == null
) {
2022-10-23 19:27:23 +02:00
setState(() => localValidationValid = true);
2022-10-28 01:32:48 +02:00
return true;
}
else {
2022-10-23 19:27:23 +02:00
setState(() => localValidationValid = false);
2022-10-28 01:32:48 +02:00
return false;
}
}
void onEditValidate() {
setState(() => formEdited = true);
final res = checkcertKeyValid();
if (res == true) {
checkValidDataApi();
}
}
2022-10-22 23:06:27 +02:00
@override
void initState() {
fetchData();
2022-10-22 23:06:27 +02:00
super.initState();
}
@override
Widget build(BuildContext context) {
final serversProvider = Provider.of<ServersProvider>(context);
final appConfigProvider = Provider.of<AppConfigProvider>(context);
final width = MediaQuery.of(context).size.width;
void saveData() async {
2023-11-20 15:16:20 +01:00
ProcessModal processModal = ProcessModal();
processModal.open(AppLocalizations.of(context)!.savingConfig);
2023-11-19 22:52:40 +01:00
final result = await serversProvider.apiClient2!.saveEncryptionSettings(
data: {
"enabled": enabled,
"server_name": domainNameController.text,
"force_https": redirectHttps,
"port_https": int.tryParse(httpsPortController.text),
"port_dns_over_tls": int.tryParse(tlsPortController.text),
"port_dns_over_quic": int.tryParse(dnsOverQuicPortController.text),
"certificate_chain": encodeBase64(certificateContentController.text),
"private_key": encodeBase64(pastePrivateKeyController.text),
"private_key_saved": usePreviouslySavedKey,
"certificate_path": certificatePathController.text,
"private_key_path": privateKeyPathController.text,
2024-02-07 23:35:20 +01:00
"serve_plain_dns": _plainDns
}
);
processModal.close();
2023-11-19 22:52:40 +01:00
if (result.successful == true) {
showSnacbkar(
appConfigProvider: appConfigProvider,
label: AppLocalizations.of(context)!.encryptionConfigSaved,
color: Colors.green
);
}
else {
showSnacbkar(
appConfigProvider: appConfigProvider,
label: AppLocalizations.of(context)!.encryptionConfigNotSaved,
color: Colors.red
);
2023-03-24 21:48:34 +01:00
2023-11-19 22:52:40 +01:00
if (result.content != null) {
2023-03-24 21:48:34 +01:00
showDialog(
context: context,
builder: (context) => ErrorMessageEncryption(
2023-11-19 22:52:40 +01:00
errorMessage: result.content
2023-03-24 21:48:34 +01:00
)
);
}
}
}
2023-12-11 13:54:47 +01:00
void resetSettings() async {
ProcessModal processModal = ProcessModal();
processModal.open(AppLocalizations.of(context)!.resettingConfig);
final result = await serversProvider.apiClient2!.saveEncryptionSettings(
data: {
"enabled": false,
"server_name": "",
"force_https": false,
"port_https": 443,
"port_dns_over_tls": 853,
"port_dns_over_quic": 853,
"certificate_chain": "",
"private_key": "",
"private_key_saved": false,
"certificate_path": "",
"private_key_path": "",
2024-02-07 23:35:20 +01:00
"serve_plain_dns": true
2023-12-11 13:54:47 +01:00
}
);
if (!mounted) return;
processModal.close();
if (result.successful == true) {
showSnacbkar(
appConfigProvider: appConfigProvider,
label: AppLocalizations.of(context)!.configurationResetSuccessfully,
color: Colors.green
);
}
else {
showSnacbkar(
appConfigProvider: appConfigProvider,
label: AppLocalizations.of(context)!.configurationResetError,
color: Colors.red
);
}
}
2022-10-22 23:06:27 +02:00
return Scaffold(
appBar: AppBar(
title: Text(AppLocalizations.of(context)!.encryptionSettings),
2023-10-29 02:47:14 +01:00
surfaceTintColor: isDesktop(width) ? Colors.transparent : null,
2023-03-16 23:29:18 +01:00
centerTitle: false,
actions: [
IconButton(
2023-12-11 13:54:47 +01:00
onPressed: () => showDialog(
context: context,
builder: (ctx) => ResetSettingsModal(onConfirm: resetSettings)
),
icon: const Icon(Icons.restore_rounded),
tooltip: AppLocalizations.of(context)!.resetSettings,
),
IconButton(
2023-05-20 01:16:33 +02:00
onPressed: localValidationValid ?
() => saveData()
: null,
icon: const Icon(Icons.save),
tooltip: AppLocalizations.of(context)!.save,
),
const SizedBox(width: 10),
],
2022-10-22 23:06:27 +02:00
),
2023-12-09 04:04:14 +01:00
body: SafeArea(
child: Builder(
builder: (context) {
switch (loadStatus) {
case LoadStatus.loading:
return SizedBox(
width: double.maxFinite,
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
2023-10-29 02:47:14 +01:00
children: [
2023-12-09 04:04:14 +01:00
const CircularProgressIndicator(),
const SizedBox(height: 30),
Text(
AppLocalizations.of(context)!.loadingEncryptionSettings,
textAlign: TextAlign.center,
style: TextStyle(
fontSize: 22,
color: Theme.of(context).colorScheme.onSurfaceVariant,
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
)
],
)
);
case LoadStatus.loaded:
return ListView(
children: [
2024-02-07 23:44:32 +01:00
if (_dataValidApi == false && (validDataError != null || encryptionResultMessage != null)) Card(
2023-12-11 13:54:47 +01:00
margin: const EdgeInsets.all(16),
color: Colors.red.withOpacity(0.2),
elevation: 0,
child: Padding(
padding: const EdgeInsets.all(16),
child: Row(
children: [
Icon(
Icons.error_rounded,
color: Theme.of(context).colorScheme.onSurfaceVariant,
),
const SizedBox(width: 16),
Expanded(
child: Text(validDataError ?? encryptionResultMessage ?? AppLocalizations.of(context)!.unknownError)
)
],
),
),
),
2023-12-09 04:04:14 +01:00
EncryptionMasterSwitch(
value: enabled,
onChange: (value) {
setState(() => enabled = value);
onEditValidate();
}
),
2024-02-07 23:35:20 +01:00
if (_plainDns != null) ...[
const SizedBox(height: 8),
CustomCheckboxListTile(
value: _plainDns!,
onChanged: (v) => setState(() => _plainDns = v),
title: AppLocalizations.of(context)!.enablePlainDns,
subtitle: AppLocalizations.of(context)!.enablePlainDnsDescription,
2024-02-09 01:48:32 +01:00
disabled: enabled == false,
2024-02-07 23:35:20 +01:00
),
],
2023-12-09 04:04:14 +01:00
SectionLabel(
label: AppLocalizations.of(context)!.serverConfiguration,
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 24),
),
EncryptionTextField(
enabled: enabled,
controller: domainNameController,
icon: Icons.link_rounded,
onChanged: (value) {
setState(() => domainError = validateDomain(context, value));
onEditValidate();
},
errorText: domainError,
label: AppLocalizations.of(context)!.domainName,
helperText: AppLocalizations.of(context)!.domainNameDescription,
),
const SizedBox(height: 10),
CustomSwitchListTile(
value: redirectHttps,
onChanged: (value) {
setState(() => redirectHttps = value);
onEditValidate();
},
title: AppLocalizations.of(context)!.redirectHttps,
disabled: !enabled,
),
const SizedBox(height: 10),
Wrap(
children: [
FractionallySizedBox(
2023-10-29 02:47:14 +01:00
widthFactor: width > 900 ? 0.33 : 1,
child: EncryptionTextField(
enabled: enabled,
2023-12-09 04:04:14 +01:00
controller: httpsPortController,
2023-10-29 02:47:14 +01:00
icon: Icons.numbers_rounded,
onChanged: (value) {
2023-12-09 04:04:14 +01:00
setState(() => httpsPortError = validatePort(context, value));
2023-10-29 02:47:14 +01:00
onEditValidate();
},
2023-12-09 04:04:14 +01:00
errorText: httpsPortError,
label: AppLocalizations.of(context)!.httpsPort,
2023-10-29 02:47:14 +01:00
keyboardType: TextInputType.number,
),
),
2023-12-09 04:04:14 +01:00
Padding(
padding: width <= 900
? const EdgeInsets.symmetric(vertical: 24)
: const EdgeInsets.all(0),
child: FractionallySizedBox(
widthFactor: width > 900 ? 0.33 : 1,
child: EncryptionTextField(
enabled: enabled,
controller: tlsPortController,
icon: Icons.numbers_rounded,
onChanged: (value) {
setState(() => tlsPortError = validatePort(context, value));
onEditValidate();
},
errorText: tlsPortError,
label: AppLocalizations.of(context)!.tlsPort,
keyboardType: TextInputType.number,
),
),
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
FractionallySizedBox(
widthFactor: width > 900 ? 0.33 : 1,
child: EncryptionTextField(
enabled: enabled,
controller: dnsOverQuicPortController,
icon: Icons.numbers_rounded,
onChanged: (value) {
setState(() => dnsOverQuicPortError = validatePort(context, value));
onEditValidate();
},
errorText: dnsOverQuicPortError,
label: AppLocalizations.of(context)!.dnsOverQuicPort,
keyboardType: TextInputType.number,
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
),
],
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
SectionLabel(
label: AppLocalizations.of(context)!.certificates,
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 24),
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
Card(
margin: const EdgeInsets.symmetric(horizontal: 16),
child: Padding(
padding: const EdgeInsets.all(20),
child: Row(
children: [
Icon(
Icons.info_rounded,
color: Theme.of(context).listTileTheme.iconColor,
),
const SizedBox(width: 20),
Flexible(
child: Text(
AppLocalizations.of(context)!.certificatesDescription,
style: TextStyle(
color: Theme.of(context).colorScheme.onSurface,
),
)
)
],
),
2023-10-29 02:47:14 +01:00
),
),
const SizedBox(height: 20),
2023-12-09 04:04:14 +01:00
RadioListTile(
value: 0,
groupValue: certificateOption,
onChanged: enabled == true
? (value) {
setState(() => certificateOption = int.parse(value.toString()));
onEditValidate();
}
: null,
title: Text(
AppLocalizations.of(context)!.certificateFilePath,
style: const TextStyle(
fontWeight: FontWeight.normal
),
2023-10-29 02:47:14 +01:00
),
),
2023-12-09 04:04:14 +01:00
RadioListTile(
value: 1,
groupValue: certificateOption,
onChanged: enabled == true
? (value) {
setState(() => certificateOption = int.parse(value.toString()));
onEditValidate();
}
: null,
title: Text(
AppLocalizations.of(context)!.pasteCertificateContent,
style: const TextStyle(
fontWeight: FontWeight.normal
),
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
),
const SizedBox(height: 10),
if (certificateOption == 0) EncryptionTextField(
enabled: enabled,
controller: certificatePathController,
icon: Icons.description_rounded,
onChanged: (value) {
setState(() => certificatePathError = validatePath(context, value));
onEditValidate();
},
label: AppLocalizations.of(context)!.certificatePath,
errorText: certificatePathError,
),
if (certificateOption == 1) EncryptionTextField(
enabled: enabled,
controller: certificateContentController,
icon: Icons.description_rounded,
onChanged: (value) {
setState(() => certificateContentError = validateCertificate(context, value));
onEditValidate();
},
label: AppLocalizations.of(context)!.certificateContent,
errorText: certificateContentError,
multiline: true,
keyboardType: TextInputType.multiline,
),
if (certKeyValid != null && (certificateContentController.text != '' || certificatePathController.text != '')) ...[
const SizedBox(height: 20),
if (certKeyValid!.validChain != null) ...[
Status(
valid: certKeyValid!.validChain ?? false,
label: certKeyValid!.validChain == true
? AppLocalizations.of(context)!.validCertificateChain
: AppLocalizations.of(context)!.invalidCertificateChain,
),
const SizedBox(height: 10),
],
if (certKeyValid!.subject != null) ...[
Status(
valid: true,
label: "${AppLocalizations.of(context)!.subject}: ${certKeyValid?.subject}"
),
const SizedBox(height: 10),
],
if (certKeyValid!.issuer != null) ...[
Status(
2023-10-29 02:47:14 +01:00
valid: true,
2023-12-09 04:04:14 +01:00
label: "${AppLocalizations.of(context)!.issuer}: ${certKeyValid?.issuer}"
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
const SizedBox(height: 10),
],
if (certKeyValid!.notAfter != null) ...[
Status(
valid: true,
label: "${AppLocalizations.of(context)!.expirationDate}: ${certKeyValid?.notAfter}"
),
const SizedBox(height: 10),
],
if (certKeyValid!.dnsNames != null) ...[
Status(
valid: true,
label: "${AppLocalizations.of(context)!.hostNames}: ${certKeyValid?.dnsNames?.join(', ')}"
),
const SizedBox(height: 10),
],
2023-10-29 02:47:14 +01:00
],
2023-12-09 04:04:14 +01:00
SectionLabel(
label: AppLocalizations.of(context)!.privateKey,
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 24),
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
RadioListTile(
value: 0,
groupValue: privateKeyOption,
onChanged: enabled == true
? (value) {
setState(() => privateKeyOption = int.parse(value.toString()));
onEditValidate();
}
: null,
title: Text(
AppLocalizations.of(context)!.privateKeyFile,
style: const TextStyle(
fontWeight: FontWeight.normal
),
2023-10-29 02:47:14 +01:00
),
),
2023-12-09 04:04:14 +01:00
RadioListTile(
value: 1,
groupValue: privateKeyOption,
onChanged: enabled == true
? (value) {
setState(() => privateKeyOption = int.parse(value.toString()));
onEditValidate();
}
: null,
title: Text(
AppLocalizations.of(context)!.pastePrivateKey,
style: const TextStyle(
fontWeight: FontWeight.normal
),
),
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
if (privateKeyOption == 0) const SizedBox(height: 10),
if (privateKeyOption == 1) ...[
CustomSwitchListTile(
value: usePreviouslySavedKey,
onChanged: (value) => setState(() => usePreviouslySavedKey = value),
title: AppLocalizations.of(context)!.usePreviousKey,
2023-10-29 02:47:14 +01:00
),
const SizedBox(height: 10)
],
2023-12-09 04:04:14 +01:00
if (privateKeyOption == 0) EncryptionTextField(
enabled: enabled,
controller: privateKeyPathController,
icon: Icons.description_rounded,
onChanged: (value) {
setState(() => privateKeyPathError = validatePath(context, value));
onEditValidate();
},
label: AppLocalizations.of(context)!.privateKeyPath,
errorText: privateKeyPathError,
),
if (privateKeyOption == 1) EncryptionTextField(
enabled: enabled == true
? !usePreviouslySavedKey
: false,
controller: pastePrivateKeyController,
icon: Icons.description_rounded,
onChanged: (value) {
setState(() => pastePrivateKeyError = validatePrivateKey(context, value));
onEditValidate();
},
label: AppLocalizations.of(context)!.pastePrivateKey,
errorText: pastePrivateKeyError,
keyboardType: TextInputType.multiline,
multiline: true,
),
const SizedBox(height: 20),
if (certKeyValid != null && (privateKeyPathController.text != '' || pastePrivateKeyController.text != '' || usePreviouslySavedKey == true)) ...[
if (certKeyValid!.validKey != null) ...[
Status(
valid: certKeyValid!.validKey ?? false,
label: certKeyValid!.validKey == true
? AppLocalizations.of(context)!.validPrivateKey
: AppLocalizations.of(context)!.invalidPrivateKey,
),
const SizedBox(height: 10)
],
if (certKeyValid!.validPair != null && certKeyValid!.validPair == false) ...[
Status(
valid: false,
label: AppLocalizations.of(context)!.keysNotMatch,
),
const SizedBox(height: 10)
],
if (certKeyValid!.keyType != null) ...[
Status(
valid: true,
label: "${AppLocalizations.of(context)!.keyType}: ${certKeyValid!.keyType}"
),
const SizedBox(height: 10),
],
2023-10-29 02:47:14 +01:00
const SizedBox(height: 10)
2023-12-09 04:04:14 +01:00
]
],
);
case LoadStatus.error:
return SizedBox(
width: double.maxFinite,
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
const Icon(
Icons.error,
color: Colors.red,
size: 50,
2023-10-29 02:47:14 +01:00
),
2023-12-09 04:04:14 +01:00
const SizedBox(height: 30),
Text(
AppLocalizations.of(context)!.encryptionSettingsNotLoaded,
textAlign: TextAlign.center,
style: TextStyle(
fontSize: 22,
color: Theme.of(context).colorScheme.onSurfaceVariant,
),
)
2023-10-29 02:47:14 +01:00
],
2023-12-09 04:04:14 +01:00
),
);
default:
return const SizedBox();
}
},
),
2023-10-29 02:47:14 +01:00
)
2022-10-22 23:06:27 +02:00
);
}
}