Finish recovery key screen

This commit is contained in:
Inex Code 2022-05-31 02:06:08 +03:00
parent 1db8e9556e
commit 8ec3b8c3e3
19 changed files with 285 additions and 213 deletions

View file

@ -291,7 +291,7 @@
"method_select_other_device": "I have access on another device", "method_select_other_device": "I have access on another device",
"method_select_recovery_key": "I have a recovery key", "method_select_recovery_key": "I have a recovery key",
"method_select_nothing": "I don't have any of that", "method_select_nothing": "I don't have any of that",
"method_device_description": "Open the application on another device, then go to the device page. Press \"Add device\" to receive your token.", "method_device_description": "Open the application on another device, then go to the devices page. Press \"Add device\" to receive your token.",
"method_device_button": "I have received my token", "method_device_button": "I have received my token",
"method_device_input_description": "Enter your authorization token", "method_device_input_description": "Enter your authorization token",
"method_device_input_placeholder": "Token", "method_device_input_placeholder": "Token",
@ -342,7 +342,8 @@
"key_replace_button": "Generate new key", "key_replace_button": "Generate new key",
"key_receiving_description": "Write down this key and put to a safe place. It is used to restore full access to your server:", "key_receiving_description": "Write down this key and put to a safe place. It is used to restore full access to your server:",
"key_receiving_info": "The key will never ever be shown again, but you will be able to replace it with another one.", "key_receiving_info": "The key will never ever be shown again, but you will be able to replace it with another one.",
"key_receiving_done": "Done!" "key_receiving_done": "Done!",
"generation_error": "Couldn't generate a recovery key. {}"
}, },
"modals": { "modals": {
"_comment": "messages in modals", "_comment": "messages in modals",

View file

@ -323,13 +323,25 @@
"confirm_backblaze_description": "Введите токен Backblaze, который имеет права на хранилище резервных копий:" "confirm_backblaze_description": "Введите токен Backblaze, который имеет права на хранилище резервных копий:"
}, },
"recovery_key": { "recovery_key": {
"key_connection_error": "Не удалось соединиться с сервером",
"key_synchronizing": "Синхронизация...",
"key_main_header": "Ключ восстановления", "key_main_header": "Ключ восстановления",
"key_main_description": "Требуется для авторизации SelfPrivacy, когда авторизованные устройства недоступны.", "key_main_description": "Требуется для авторизации SelfPrivacy, когда авторизованные устройства недоступны.",
"key_amount_toggle": "Ограничить использования", "key_amount_toggle": "Ограничить использования",
"key_amount_field_title": "Макс. кол-во использований", "key_amount_field_title": "Макс. кол-во использований",
"key_duedate_toggle": "Ограничить срок использования", "key_duedate_toggle": "Ограничить срок использования",
"key_duedate_field_title": "Дата окончания срока", "key_duedate_field_title": "Дата окончания срока",
"key_receive_button": "Получить ключ" "key_receive_button": "Получить ключ",
"key_valid": "Ваш ключ действителен",
"key_invalid": "Ваш ключ больше не действителен",
"key_valid_until": "Действителен до {}",
"key_valid_for": "Можно использовать ещё {} раз",
"key_creation_date": "Создан {}",
"key_replace_button": "Сгенерировать новый ключ",
"key_receiving_description": "Запишите этот ключ в безопасном месте. Он предоставляет полный доступ к вашему серверу:",
"key_receiving_info": "Этот ключ больше не будет показан, но вы сможете заменить его новым.",
"key_receiving_done": "Готово!",
"generation_error": "Не удалось сгенерировать ключ. {}"
}, },
"modals": { "modals": {
"_comment": "messages in modals", "_comment": "messages in modals",

View file

@ -109,6 +109,9 @@ class BNames {
/// A boolean field of [serverInstallationBox] box. /// A boolean field of [serverInstallationBox] box.
static String isServerResetedSecondTime = 'isServerResetedSecondTime'; static String isServerResetedSecondTime = 'isServerResetedSecondTime';
/// A boolean field of [serverInstallationBox] box.
static String isRecoveringServer = 'isRecoveringServer';
/// Deprecated users box as it is unencrypted /// Deprecated users box as it is unencrypted
static String usersDeprecated = 'users'; static String usersDeprecated = 'users';

View file

@ -664,7 +664,8 @@ class ServerApi extends ApiMap {
var client = await getClient(); var client = await getClient();
var data = {}; var data = {};
if (expiration != null) { if (expiration != null) {
data['expiration'] = expiration.toIso8601String(); data['expiration'] = '${expiration.toIso8601String()}Z';
print(data['expiration']);
} }
if (uses != null) { if (uses != null) {
data['uses'] = uses; data['uses'] = uses;

View file

@ -14,7 +14,16 @@ class RecoveryDeviceFormCubit extends FormCubit {
@override @override
FutureOr<void> onSubmit() async { FutureOr<void> onSubmit() async {
installationCubit.tryToRecover(tokenField.state.value, recoveryMethod); late final String token;
// Trim spaces and make lowercase
if (recoveryMethod == ServerRecoveryMethods.recoveryKey ||
recoveryMethod == ServerRecoveryMethods.newDeviceKey) {
token = tokenField.state.value.trim().toLowerCase();
} else {
token = tokenField.state.value.trim();
}
installationCubit.tryToRecover(token, recoveryMethod);
} }
final ServerInstallationCubit installationCubit; final ServerInstallationCubit installationCubit;

View file

@ -287,6 +287,7 @@ class ServerInstallationCubit extends Cubit<ServerInstallationState> {
await repository.getRecoveryCapabilities(serverDomain); await repository.getRecoveryCapabilities(serverDomain);
await repository.saveDomain(serverDomain); await repository.saveDomain(serverDomain);
await repository.saveIsRecoveringServer(true);
emit(ServerInstallationRecovery( emit(ServerInstallationRecovery(
serverDomain: serverDomain, serverDomain: serverDomain,
@ -458,6 +459,7 @@ class ServerInstallationCubit extends Cubit<ServerInstallationState> {
await repository.saveIsServerResetedFirstTime(true); await repository.saveIsServerResetedFirstTime(true);
await repository.saveIsServerResetedSecondTime(true); await repository.saveIsServerResetedSecondTime(true);
await repository.saveHasFinalChecked(true); await repository.saveHasFinalChecked(true);
await repository.saveIsRecoveringServer(false);
final mainUser = await repository.getMainUser(); final mainUser = await repository.getMainUser();
final updatedState = (state as ServerInstallationRecovery).copyWith( final updatedState = (state as ServerInstallationRecovery).copyWith(
backblazeCredential: backblazeCredential, backblazeCredential: backblazeCredential,

View file

@ -62,7 +62,8 @@ class ServerInstallationRepository {
); );
} }
if (serverDomain != null && serverDomain.provider == DnsProvider.unknown) { if (box.get(BNames.isRecoveringServer, defaultValue: false) &&
serverDomain != null) {
return ServerInstallationRecovery( return ServerInstallationRecovery(
hetznerKey: hetznerToken, hetznerKey: hetznerToken,
cloudFlareKey: cloudflareToken, cloudFlareKey: cloudflareToken,
@ -601,6 +602,10 @@ class ServerInstallationRepository {
await box.put(BNames.rootUser, rootUser); await box.put(BNames.rootUser, rootUser);
} }
Future<void> saveIsRecoveringServer(bool value) async {
await box.put(BNames.isRecoveringServer, value);
}
Future<void> saveHasFinalChecked(bool value) async { Future<void> saveHasFinalChecked(bool value) async {
await box.put(BNames.hasFinalChecked, value); await box.put(BNames.hasFinalChecked, value);
} }

View file

@ -16,12 +16,12 @@ class FilledButton extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final ButtonStyle _enabledStyle = ElevatedButton.styleFrom( final ButtonStyle enabledStyle = ElevatedButton.styleFrom(
onPrimary: Theme.of(context).colorScheme.onPrimary, onPrimary: Theme.of(context).colorScheme.onPrimary,
primary: Theme.of(context).colorScheme.primary, primary: Theme.of(context).colorScheme.primary,
).copyWith(elevation: ButtonStyleButton.allOrNull(0.0)); ).copyWith(elevation: ButtonStyleButton.allOrNull(0.0));
final ButtonStyle _disabledStyle = ElevatedButton.styleFrom( final ButtonStyle disabledStyle = ElevatedButton.styleFrom(
onPrimary: Theme.of(context).colorScheme.onSurface.withAlpha(30), onPrimary: Theme.of(context).colorScheme.onSurface.withAlpha(30),
primary: Theme.of(context).colorScheme.onSurface.withAlpha(98), primary: Theme.of(context).colorScheme.onSurface.withAlpha(98),
).copyWith(elevation: ButtonStyleButton.allOrNull(0.0)); ).copyWith(elevation: ButtonStyleButton.allOrNull(0.0));
@ -33,7 +33,7 @@ class FilledButton extends StatelessWidget {
), ),
child: ElevatedButton( child: ElevatedButton(
onPressed: onPressed, onPressed: onPressed,
style: disabled ? _disabledStyle : _enabledStyle, style: disabled ? disabledStyle : enabledStyle,
child: child ?? Text(title ?? ''), child: child ?? Text(title ?? ''),
), ),
); );

View file

@ -1,5 +1,4 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:selfprivacy/config/brand_colors.dart';
class BrandCards { class BrandCards {
static Widget big({required Widget child}) => _BrandCard( static Widget big({required Widget child}) => _BrandCard(
@ -23,7 +22,9 @@ class BrandCards {
static Widget outlined({required Widget child}) => _OutlinedCard( static Widget outlined({required Widget child}) => _OutlinedCard(
child: child, child: child,
); );
static Widget filled({required Widget child}) => _FilledCard( static Widget filled({required Widget child, bool tertiary = false}) =>
_FilledCard(
tertiary: tertiary,
child: child, child: child,
); );
} }
@ -80,12 +81,11 @@ class _OutlinedCard extends StatelessWidget {
} }
class _FilledCard extends StatelessWidget { class _FilledCard extends StatelessWidget {
const _FilledCard({ const _FilledCard({Key? key, required this.child, required this.tertiary})
Key? key, : super(key: key);
required this.child,
}) : super(key: key);
final Widget child; final Widget child;
final bool tertiary;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Card( return Card(
@ -94,7 +94,9 @@ class _FilledCard extends StatelessWidget {
borderRadius: BorderRadius.all(Radius.circular(12)), borderRadius: BorderRadius.all(Radius.circular(12)),
), ),
clipBehavior: Clip.antiAlias, clipBehavior: Clip.antiAlias,
color: Theme.of(context).colorScheme.surfaceVariant, color: tertiary
? Theme.of(context).colorScheme.tertiaryContainer
: Theme.of(context).colorScheme.surfaceVariant,
child: child, child: child,
); );
} }

View file

@ -1,13 +1,11 @@
import 'package:easy_localization/easy_localization.dart'; import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:ionicons/ionicons.dart'; import 'package:ionicons/ionicons.dart';
import 'package:selfprivacy/config/brand_colors.dart';
import 'package:selfprivacy/config/brand_theme.dart'; import 'package:selfprivacy/config/brand_theme.dart';
import 'package:selfprivacy/logic/cubit/server_installation/server_installation_cubit.dart'; import 'package:selfprivacy/logic/cubit/server_installation/server_installation_cubit.dart';
import 'package:selfprivacy/ui/components/brand_divider/brand_divider.dart'; import 'package:selfprivacy/ui/components/brand_cards/brand_cards.dart';
import 'package:selfprivacy/ui/components/brand_header/brand_header.dart'; import 'package:selfprivacy/ui/components/brand_header/brand_header.dart';
import 'package:selfprivacy/ui/components/brand_icons/brand_icons.dart'; import 'package:selfprivacy/ui/components/brand_icons/brand_icons.dart';
import 'package:selfprivacy/ui/components/brand_text/brand_text.dart';
import 'package:selfprivacy/ui/pages/recovery_key/recovery_key.dart'; import 'package:selfprivacy/ui/pages/recovery_key/recovery_key.dart';
import 'package:selfprivacy/ui/pages/setup/initializing.dart'; import 'package:selfprivacy/ui/pages/setup/initializing.dart';
import 'package:selfprivacy/ui/pages/onboarding/onboarding.dart'; import 'package:selfprivacy/ui/pages/onboarding/onboarding.dart';
@ -26,6 +24,9 @@ class MorePage extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
var isReady = context.watch<ServerInstallationCubit>().state
is ServerInstallationFinished;
return Scaffold( return Scaffold(
appBar: PreferredSize( appBar: PreferredSize(
preferredSize: const Size.fromHeight(52), preferredSize: const Size.fromHeight(52),
@ -39,52 +40,53 @@ class MorePage extends StatelessWidget {
padding: paddingH15V0, padding: paddingH15V0,
child: Column( child: Column(
children: [ children: [
const BrandDivider(), if (!isReady)
_NavItem( _MoreMenuItem(
title: 'more.configuration_wizard'.tr(), title: 'more.configuration_wizard'.tr(),
iconData: BrandIcons.triangle, iconData: Icons.change_history_outlined,
goTo: const InitializingPage(), goTo: const InitializingPage(),
subtitle: 'not_ready_card.in_menu'.tr(),
accent: true,
), ),
_NavItem( if (isReady)
title: 'more.settings.title'.tr(), _MoreMenuItem(
iconData: BrandIcons.settings,
goTo: const AppSettingsPage(),
),
_NavItem(
title: 'more.about_project'.tr(),
iconData: BrandIcons.engineer,
goTo: const AboutPage(),
),
_NavItem(
title: 'more.about_app'.tr(),
iconData: BrandIcons.fire,
goTo: const InfoPage(),
),
_NavItem(
title: 'more.onboarding'.tr(),
iconData: BrandIcons.start,
goTo: const OnboardingPage(nextPage: RootPage()),
),
_NavItem(
title: 'more.console'.tr(),
iconData: BrandIcons.terminal,
goTo: const Console(),
),
_NavItem(
isEnabled: context.read<ServerInstallationCubit>().state
is ServerInstallationFinished,
title: 'more.create_ssh_key'.tr(), title: 'more.create_ssh_key'.tr(),
iconData: Ionicons.key_outline, iconData: Ionicons.key_outline,
goTo: SshKeysPage( goTo: SshKeysPage(
user: context.read<UsersCubit>().state.rootUser, user: context.read<UsersCubit>().state.rootUser,
)), )),
_NavItem( if (isReady)
isEnabled: context.read<ServerInstallationCubit>().state _MoreMenuItem(
is ServerInstallationFinished,
iconData: Icons.password_outlined, iconData: Icons.password_outlined,
goTo: const RecoveryKey(), goTo: const RecoveryKey(),
title: 'recovery_key.key_main_header'.tr(), title: 'recovery_key.key_main_header'.tr(),
) ),
_MoreMenuItem(
title: 'more.settings.title'.tr(),
iconData: Icons.settings_outlined,
goTo: const AppSettingsPage(),
),
_MoreMenuItem(
title: 'more.about_project'.tr(),
iconData: BrandIcons.engineer,
goTo: const AboutPage(),
),
_MoreMenuItem(
title: 'more.about_app'.tr(),
iconData: BrandIcons.fire,
goTo: const InfoPage(),
),
if (!isReady)
_MoreMenuItem(
title: 'more.onboarding'.tr(),
iconData: BrandIcons.start,
goTo: const OnboardingPage(nextPage: RootPage()),
),
_MoreMenuItem(
title: 'more.console'.tr(),
iconData: BrandIcons.terminal,
goTo: const Console(),
),
], ],
), ),
) )
@ -94,77 +96,53 @@ class MorePage extends StatelessWidget {
} }
} }
class _NavItem extends StatelessWidget {
const _NavItem({
Key? key,
this.isEnabled = true,
required this.iconData,
required this.goTo,
required this.title,
}) : super(key: key);
final IconData iconData;
final Widget goTo;
final String title;
final bool isEnabled;
@override
Widget build(BuildContext context) {
return GestureDetector(
onTap: isEnabled
? () => Navigator.of(context).push(materialRoute(goTo))
: null,
child: _MoreMenuItem(
iconData: iconData,
title: title,
isActive: isEnabled,
),
);
}
}
class _MoreMenuItem extends StatelessWidget { class _MoreMenuItem extends StatelessWidget {
const _MoreMenuItem({ const _MoreMenuItem({
Key? key, Key? key,
required this.iconData, required this.iconData,
required this.title, required this.title,
required this.isActive, this.subtitle,
this.goTo,
this.accent = false,
}) : super(key: key); }) : super(key: key);
final IconData iconData; final IconData iconData;
final String title; final String title;
final bool isActive; final Widget? goTo;
final String? subtitle;
final bool accent;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Container( final color = accent
padding: const EdgeInsets.symmetric(vertical: 24), ? Theme.of(context).colorScheme.onTertiaryContainer
decoration: const BoxDecoration( : Theme.of(context).colorScheme.onSurface;
border: Border( return BrandCards.filled(
bottom: BorderSide( tertiary: accent,
width: 1.0, child: ListTile(
color: BrandColors.dividerColor, contentPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
), onTap: goTo != null
), ? () => Navigator.of(context).push(materialRoute(goTo!))
), : null,
child: Row( leading: Icon(
children: [
BrandText.body1(
title,
style: TextStyle(
color: isActive ? null : Colors.grey,
),
),
const Spacer(),
SizedBox(
width: 56,
child: Icon(
iconData, iconData,
size: 20, size: 24,
color: isActive ? null : Colors.grey, color: color,
),
title: Text(
title,
style: Theme.of(context).textTheme.titleMedium?.copyWith(
color: color,
), ),
), ),
], subtitle: subtitle != null
? Text(
subtitle!,
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
color: color,
),
)
: null,
), ),
); );
} }

View file

@ -2,6 +2,7 @@ import 'package:cubit_form/cubit_form.dart';
import 'package:easy_localization/easy_localization.dart'; import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:selfprivacy/config/get_it_config.dart';
import 'package:selfprivacy/logic/common_enum/common_enum.dart'; import 'package:selfprivacy/logic/common_enum/common_enum.dart';
import 'package:selfprivacy/logic/cubit/recovery_key/recovery_key_cubit.dart'; import 'package:selfprivacy/logic/cubit/recovery_key/recovery_key_cubit.dart';
import 'package:selfprivacy/logic/cubit/server_installation/server_installation_cubit.dart'; import 'package:selfprivacy/logic/cubit/server_installation/server_installation_cubit.dart';
@ -88,7 +89,7 @@ class _RecoveryKeyContentState extends State<RecoveryKeyContent> {
if (keyStatus.exists && !_isConfigurationVisible) if (keyStatus.exists && !_isConfigurationVisible)
RecoveryKeyInformation(state: keyStatus), RecoveryKeyInformation(state: keyStatus),
if (_isConfigurationVisible || !keyStatus.exists) if (_isConfigurationVisible || !keyStatus.exists)
RecoveryKeyConfiguration(), const RecoveryKeyConfiguration(),
const SizedBox(height: 16), const SizedBox(height: 16),
if (!_isConfigurationVisible && keyStatus.isValid) if (!_isConfigurationVisible && keyStatus.isValid)
BrandButton.text( BrandButton.text(
@ -161,8 +162,10 @@ class RecoveryKeyInformation extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
const padding = EdgeInsets.symmetric(vertical: 8.0); const padding = EdgeInsets.symmetric(vertical: 8.0, horizontal: 16.0);
return Column( return SizedBox(
width: double.infinity,
child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
if (state.expiresAt != null) if (state.expiresAt != null)
@ -170,7 +173,7 @@ class RecoveryKeyInformation extends StatelessWidget {
padding: padding, padding: padding,
child: Text( child: Text(
'recovery_key.key_valid_until'.tr( 'recovery_key.key_valid_until'.tr(
args: [state.expiresAt!.toIso8601String()], args: [DateFormat.yMMMMd().format(state.expiresAt!)],
), ),
), ),
), ),
@ -188,12 +191,13 @@ class RecoveryKeyInformation extends StatelessWidget {
padding: padding, padding: padding,
child: Text( child: Text(
'recovery_key.key_creation_date'.tr( 'recovery_key.key_creation_date'.tr(
args: [state.generatedAt!.toIso8601String()], args: [DateFormat.yMMMMd().format(state.generatedAt!)],
), ),
textAlign: TextAlign.start, textAlign: TextAlign.start,
), ),
), ),
], ],
),
); );
} }
} }
@ -218,6 +222,38 @@ class _RecoveryKeyConfigurationState extends State<RecoveryKeyConfiguration> {
DateTime _selectedDate = DateTime.now(); DateTime _selectedDate = DateTime.now();
bool _isDateSelected = false; bool _isDateSelected = false;
bool _isLoading = false;
Future<void> _generateRecoveryToken() async {
setState(() {
_isLoading = true;
});
try {
final token = await context.read<RecoveryKeyCubit>().generateRecoveryKey(
numberOfUses:
_isAmountToggled ? int.tryParse(_amountController.text) : null,
expirationDate: _isExpirationToggled ? _selectedDate : null,
);
if (!mounted) return;
setState(() {
_isLoading = false;
});
Navigator.of(context).push(
materialRoute(
RecoveryKeyReceiving(recoveryKey: token), // TO DO
),
);
} on GenerationError catch (e) {
setState(() {
_isLoading = false;
});
getIt<NavigationService>().showSnackBar(
'recovery_key.generation_error'.tr(args: [e.message]),
);
return;
}
}
void _updateErrorStatuses() { void _updateErrorStatuses() {
final amount = _amountController.text; final amount = _amountController.text;
final expiration = _expirationController.text; final expiration = _expirationController.text;
@ -241,8 +277,7 @@ class _RecoveryKeyConfigurationState extends State<RecoveryKeyConfiguration> {
} else if (expiration.isEmpty) { } else if (expiration.isEmpty) {
_isExpirationError = true; _isExpirationError = true;
} else { } else {
_isExpirationError = _isExpirationError = _selectedDate.isBefore(DateTime.now());
_selectedDate == null || _selectedDate.isBefore(DateTime.now());
} }
}); });
@ -266,10 +301,10 @@ class _RecoveryKeyConfigurationState extends State<RecoveryKeyConfiguration> {
value: _isAmountToggled, value: _isAmountToggled,
title: Text('recovery_key.key_amount_toggle'.tr()), title: Text('recovery_key.key_amount_toggle'.tr()),
activeColor: Theme.of(context).colorScheme.primary, activeColor: Theme.of(context).colorScheme.primary,
onChanged: (bool toogled) { onChanged: (bool toggled) {
setState( setState(
() { () {
_isAmountToggled = toogled; _isAmountToggled = toggled;
}, },
); );
_updateErrorStatuses(); _updateErrorStatuses();
@ -287,7 +322,7 @@ class _RecoveryKeyConfigurationState extends State<RecoveryKeyConfiguration> {
enabled: _isAmountToggled, enabled: _isAmountToggled,
controller: _amountController, controller: _amountController,
decoration: InputDecoration( decoration: InputDecoration(
border: OutlineInputBorder(), border: const OutlineInputBorder(),
errorText: _isAmountError ? ' ' : null, errorText: _isAmountError ? ' ' : null,
labelText: 'recovery_key.key_amount_field_title'.tr()), labelText: 'recovery_key.key_amount_field_title'.tr()),
keyboardType: TextInputType.number, keyboardType: TextInputType.number,
@ -304,10 +339,10 @@ class _RecoveryKeyConfigurationState extends State<RecoveryKeyConfiguration> {
value: _isExpirationToggled, value: _isExpirationToggled,
title: Text('recovery_key.key_duedate_toggle'.tr()), title: Text('recovery_key.key_duedate_toggle'.tr()),
activeColor: Theme.of(context).colorScheme.primary, activeColor: Theme.of(context).colorScheme.primary,
onChanged: (bool toogled) { onChanged: (bool toggled) {
setState( setState(
() { () {
_isExpirationToggled = toogled; _isExpirationToggled = toggled;
}, },
); );
_updateErrorStatuses(); _updateErrorStatuses();
@ -329,7 +364,7 @@ class _RecoveryKeyConfigurationState extends State<RecoveryKeyConfiguration> {
}, },
readOnly: true, readOnly: true,
decoration: InputDecoration( decoration: InputDecoration(
border: OutlineInputBorder(), border: const OutlineInputBorder(),
errorText: _isExpirationError ? ' ' : null, errorText: _isExpirationError ? ' ' : null,
labelText: 'recovery_key.key_duedate_field_title'.tr()), labelText: 'recovery_key.key_duedate_field_title'.tr()),
keyboardType: TextInputType.number, keyboardType: TextInputType.number,
@ -344,15 +379,9 @@ class _RecoveryKeyConfigurationState extends State<RecoveryKeyConfiguration> {
const SizedBox(height: 16), const SizedBox(height: 16),
FilledButton( FilledButton(
title: 'recovery_key.key_receive_button'.tr(), title: 'recovery_key.key_receive_button'.tr(),
disabled: _isAmountError || _isExpirationError, disabled: _isAmountError || _isExpirationError || _isLoading,
onPressed: !_isAmountError && !_isExpirationError onPressed: !_isAmountError && !_isExpirationError
? () { ? _generateRecoveryToken
Navigator.of(context).push(
materialRoute(
const RecoveryKeyReceiving(recoveryKey: ''), // TO DO
),
);
}
: null, : null,
), ),
], ],

View file

@ -15,14 +15,31 @@ class RecoveryKeyReceiving extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return BrandHeroScreen( return BrandHeroScreen(
heroTitle: 'recovery_key.key_main_header'.tr(), heroTitle: 'recovery_key.key_main_header'.tr(),
heroSubtitle: 'recovering.method_select_description'.tr(), heroSubtitle: 'recovery_key.key_receiving_description'.tr(),
hasBackButton: true, hasBackButton: true,
hasFlashButton: false, hasFlashButton: false,
children: [ children: [
Text(recoveryKey, style: Theme.of(context).textTheme.bodyLarge), const Divider(),
const SizedBox(height: 16),
Text(
recoveryKey,
style: Theme.of(context).textTheme.bodyLarge!.copyWith(
fontSize: 24,
fontFamily: 'RobotoMono',
),
textAlign: TextAlign.center,
),
const SizedBox(height: 16),
const Divider(),
const SizedBox(height: 16),
Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
const Icon(Icons.info_outlined, size: 24),
const SizedBox(height: 16), const SizedBox(height: 16),
const Icon(Icons.info_outlined, size: 14),
Text('recovery_key.key_receiving_info'.tr()), Text('recovery_key.key_receiving_info'.tr()),
],
),
const SizedBox(height: 16), const SizedBox(height: 16),
FilledButton( FilledButton(
title: 'recovery_key.key_receiving_done'.tr(), title: 'recovery_key.key_receiving_done'.tr(),

View file

@ -1,6 +1,5 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:provider/provider.dart'; import 'package:selfprivacy/logic/cubit/server_installation/server_installation_cubit.dart';
import 'package:selfprivacy/logic/api_maps/server.dart';
import 'package:selfprivacy/ui/components/brand_tab_bar/brand_tab_bar.dart'; import 'package:selfprivacy/ui/components/brand_tab_bar/brand_tab_bar.dart';
import 'package:selfprivacy/ui/pages/more/more.dart'; import 'package:selfprivacy/ui/pages/more/more.dart';
import 'package:selfprivacy/ui/pages/providers/providers.dart'; import 'package:selfprivacy/ui/pages/providers/providers.dart';
@ -48,10 +47,11 @@ class _RootPageState extends State<RootPage> with TickerProviderStateMixin {
super.dispose(); super.dispose();
} }
var selfprivacyServer = ServerApi();
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
var isReady = context.watch<ServerInstallationCubit>().state
is ServerInstallationFinished;
return SafeArea( return SafeArea(
child: Provider<ChangeTab>( child: Provider<ChangeTab>(
create: (_) => ChangeTab(tabController.animateTo), create: (_) => ChangeTab(tabController.animateTo),
@ -68,7 +68,8 @@ class _RootPageState extends State<RootPage> with TickerProviderStateMixin {
bottomNavigationBar: BrandTabBar( bottomNavigationBar: BrandTabBar(
controller: tabController, controller: tabController,
), ),
floatingActionButton: SizedBox( floatingActionButton: isReady
? SizedBox(
height: 104 + 16, height: 104 + 16,
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.end, crossAxisAlignment: CrossAxisAlignment.end,
@ -76,28 +77,14 @@ class _RootPageState extends State<RootPage> with TickerProviderStateMixin {
children: [ children: [
ScaleTransition( ScaleTransition(
scale: _animation, scale: _animation,
child: FloatingActionButton.small( child: const AddUserFab(),
heroTag: 'new_user_fab',
onPressed: () {
showModalBottomSheet<void>(
context: context,
isScrollControlled: true,
backgroundColor: Colors.transparent,
builder: (BuildContext context) {
return Padding(
padding: MediaQuery.of(context).viewInsets,
child: NewUser());
},
);
},
child: const Icon(Icons.person_add_outlined),
),
), ),
const SizedBox(height: 16), const SizedBox(height: 16),
const BrandFab(), const BrandFab(),
], ],
), ),
), )
: null,
), ),
), ),
); );

View file

@ -29,21 +29,17 @@ class RecoveryConfirmBackblaze extends StatelessWidget {
children: [ children: [
CubitFormTextField( CubitFormTextField(
formFieldCubit: context.read<BackblazeFormCubit>().keyId, formFieldCubit: context.read<BackblazeFormCubit>().keyId,
textAlign: TextAlign.center,
scrollPadding: const EdgeInsets.only(bottom: 70),
decoration: const InputDecoration( decoration: const InputDecoration(
border: OutlineInputBorder(), border: OutlineInputBorder(),
hintText: 'KeyID', labelText: 'KeyID',
), ),
), ),
const SizedBox(height: 16), const SizedBox(height: 16),
CubitFormTextField( CubitFormTextField(
formFieldCubit: context.read<BackblazeFormCubit>().applicationKey, formFieldCubit: context.read<BackblazeFormCubit>().applicationKey,
textAlign: TextAlign.center,
scrollPadding: const EdgeInsets.only(bottom: 70),
decoration: const InputDecoration( decoration: const InputDecoration(
border: OutlineInputBorder(), border: OutlineInputBorder(),
hintText: 'Master Application Key', labelText: 'Master Application Key',
), ),
), ),
const SizedBox(height: 16), const SizedBox(height: 16),

View file

@ -31,11 +31,9 @@ class RecoveryConfirmCloudflare extends StatelessWidget {
children: [ children: [
CubitFormTextField( CubitFormTextField(
formFieldCubit: context.read<CloudFlareFormCubit>().apiKey, formFieldCubit: context.read<CloudFlareFormCubit>().apiKey,
textAlign: TextAlign.center,
scrollPadding: const EdgeInsets.only(bottom: 70),
decoration: InputDecoration( decoration: InputDecoration(
border: const OutlineInputBorder(), border: const OutlineInputBorder(),
hintText: 'initializing.5'.tr(), labelText: 'initializing.5'.tr(),
), ),
), ),
const SizedBox(height: 16), const SizedBox(height: 16),

View file

@ -143,7 +143,7 @@ class _RecoveryConfirmServerState extends State<RecoveryConfirmServer> {
), ),
), ),
leading: Icon( leading: Icon(
Icons.dns, Icons.dns_outlined,
color: Theme.of(context).colorScheme.onSurface, color: Theme.of(context).colorScheme.onSurface,
), ),
subtitle: Column( subtitle: Column(
@ -199,10 +199,11 @@ class _RecoveryConfirmServerState extends State<RecoveryConfirmServer> {
crossAxisAlignment: CrossAxisAlignment.center, crossAxisAlignment: CrossAxisAlignment.center,
children: [ children: [
const Icon(Icons.warning_amber_outlined), const Icon(Icons.warning_amber_outlined),
const SizedBox(height: 8), const SizedBox(height: 16),
Text( Text(
'recovering.modal_confirmation_title'.tr(), 'recovering.modal_confirmation_title'.tr(),
style: Theme.of(context).textTheme.headlineSmall, style: Theme.of(context).textTheme.headlineSmall,
textAlign: TextAlign.center,
), ),
], ],
), ),
@ -212,7 +213,9 @@ class _RecoveryConfirmServerState extends State<RecoveryConfirmServer> {
children: <Widget>[ children: <Widget>[
Text('recovering.modal_confirmation_description'.tr(), Text('recovering.modal_confirmation_description'.tr(),
style: Theme.of(context).textTheme.bodyMedium), style: Theme.of(context).textTheme.bodyMedium),
const SizedBox(height: 12),
const Divider(), const Divider(),
const SizedBox(height: 12),
Text( Text(
server.name, server.name,
style: Theme.of(context).textTheme.titleMedium, style: Theme.of(context).textTheme.titleMedium,
@ -275,7 +278,8 @@ class IsValidStringDisplay extends StatelessWidget {
? Icon(Icons.check, color: Theme.of(context).colorScheme.onSurface) ? Icon(Icons.check, color: Theme.of(context).colorScheme.onSurface)
: Icon(Icons.close, color: Theme.of(context).colorScheme.error), : Icon(Icons.close, color: Theme.of(context).colorScheme.error),
const SizedBox(width: 8), const SizedBox(width: 8),
isValid Expanded(
child: isValid
? Text( ? Text(
textIfValid, textIfValid,
style: Theme.of(context).textTheme.bodyMedium!.copyWith( style: Theme.of(context).textTheme.bodyMedium!.copyWith(
@ -287,7 +291,7 @@ class IsValidStringDisplay extends StatelessWidget {
style: Theme.of(context).textTheme.bodyMedium!.copyWith( style: Theme.of(context).textTheme.bodyMedium!.copyWith(
color: Theme.of(context).colorScheme.error, color: Theme.of(context).colorScheme.error,
), ),
) )),
], ],
); );
} }

View file

@ -0,0 +1,25 @@
part of 'users.dart';
class AddUserFab extends StatelessWidget {
const AddUserFab({Key? key}) : super(key: key);
@override
Widget build(BuildContext context) {
return FloatingActionButton.small(
heroTag: 'new_user_fab',
onPressed: () {
showModalBottomSheet<void>(
context: context,
isScrollControlled: true,
backgroundColor: Colors.transparent,
builder: (BuildContext context) {
return Padding(
padding: MediaQuery.of(context).viewInsets,
child: const NewUser());
},
);
},
child: const Icon(Icons.person_add_outlined),
);
}
}

View file

@ -1,6 +1,8 @@
part of 'users.dart'; part of 'users.dart';
class NewUser extends StatelessWidget { class NewUser extends StatelessWidget {
const NewUser({Key? key}) : super(key: key);
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
var config = context.watch<ServerInstallationCubit>().state; var config = context.watch<ServerInstallationCubit>().state;

View file

@ -30,6 +30,7 @@ part 'empty.dart';
part 'new_user.dart'; part 'new_user.dart';
part 'user.dart'; part 'user.dart';
part 'user_details.dart'; part 'user_details.dart';
part 'add_user_fab.dart';
class UsersPage extends StatelessWidget { class UsersPage extends StatelessWidget {
const UsersPage({Key? key}) : super(key: key); const UsersPage({Key? key}) : super(key: key);