selfprivacy.org.app/lib/ui/pages/setup/initializing.dart

599 lines
23 KiB
Dart
Raw Normal View History

2020-12-30 14:13:25 +00:00
import 'package:cubit_form/cubit_form.dart';
2022-02-08 06:59:19 +00:00
import 'package:easy_localization/easy_localization.dart';
2020-12-10 20:33:19 +00:00
import 'package:flutter/material.dart';
import 'package:selfprivacy/config/brand_theme.dart';
import 'package:selfprivacy/logic/cubit/server_installation/server_installation_cubit.dart';
import 'package:selfprivacy/logic/cubit/forms/factories/field_cubit_factory.dart';
import 'package:selfprivacy/logic/cubit/forms/setup/initializing/backblaze_form_cubit.dart';
import 'package:selfprivacy/logic/cubit/forms/setup/initializing/dns_provider_form_cubit.dart';
import 'package:selfprivacy/logic/cubit/forms/setup/initializing/domain_setup_cubit.dart';
2022-07-13 11:58:23 +00:00
import 'package:selfprivacy/logic/cubit/forms/setup/initializing/provider_form_cubit.dart';
import 'package:selfprivacy/logic/cubit/forms/setup/initializing/root_user_form_cubit.dart';
2021-06-20 21:08:52 +00:00
import 'package:selfprivacy/ui/components/brand_bottom_sheet/brand_bottom_sheet.dart';
2020-12-10 20:33:19 +00:00
import 'package:selfprivacy/ui/components/brand_button/brand_button.dart';
2021-05-25 21:53:54 +00:00
import 'package:selfprivacy/ui/components/brand_cards/brand_cards.dart';
2021-03-18 07:26:54 +00:00
import 'package:selfprivacy/ui/components/brand_md/brand_md.dart';
2020-12-10 20:33:19 +00:00
import 'package:selfprivacy/ui/components/brand_text/brand_text.dart';
2021-02-16 18:48:15 +00:00
import 'package:selfprivacy/ui/components/brand_timer/brand_timer.dart';
2020-12-30 14:13:25 +00:00
import 'package:selfprivacy/ui/components/progress_bar/progress_bar.dart';
2022-05-24 18:55:39 +00:00
import 'package:selfprivacy/ui/pages/root_route.dart';
import 'package:selfprivacy/ui/pages/setup/recovering/recovery_routing.dart';
2020-12-30 14:13:25 +00:00
import 'package:selfprivacy/utils/route_transitions/basic.dart';
2020-12-10 20:33:19 +00:00
2021-01-06 17:35:57 +00:00
class InitializingPage extends StatelessWidget {
const InitializingPage({super.key});
2020-12-10 20:33:19 +00:00
@override
Widget build(final BuildContext context) {
final cubit = context.watch<ServerInstallationCubit>();
if (cubit.state is ServerInstallationRecovery) {
2022-05-24 18:55:39 +00:00
return const RecoveryRouting();
} else {
2022-08-29 19:54:06 +00:00
Widget? actualInitializingPage;
if (cubit.state is! ServerInstallationFinished) {
actualInitializingPage = [
() => _stepHetzner(cubit),
() => _stepCloudflare(cubit),
() => _stepBackblaze(cubit),
() => _stepDomain(cubit),
() => _stepUser(cubit),
() => _stepServer(cubit),
() => _stepCheck(cubit),
() => _stepCheck(cubit),
() => _stepCheck(cubit),
() => _stepCheck(cubit)
][cubit.state.progress.index]();
}
return BlocListener<ServerInstallationCubit, ServerInstallationState>(
listener: (final context, final state) {
if (cubit.state is ServerInstallationFinished) {
2022-05-24 18:55:39 +00:00
Navigator.of(context)
.pushReplacement(materialRoute(const RootPage()));
}
},
child: SafeArea(
child: Scaffold(
body: SingleChildScrollView(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Padding(
padding: paddingH15V0.copyWith(top: 10, bottom: 10),
child: cubit.state is ServerInstallationFinished
2022-05-24 18:55:39 +00:00
? const SizedBox(
height: 80,
)
: ProgressBar(
2022-05-24 18:55:39 +00:00
steps: const [
'Hetzner',
'CloudFlare',
'Backblaze',
'Domain',
'User',
'Server',
'Check',
],
activeIndex: cubit.state.porgressBar,
),
2021-04-22 18:04:24 +00:00
),
_addCard(
AnimatedSwitcher(
2022-05-24 18:55:39 +00:00
duration: const Duration(milliseconds: 300),
child: actualInitializingPage,
2021-04-22 18:04:24 +00:00
),
),
ConstrainedBox(
constraints: BoxConstraints(
minHeight: MediaQuery.of(context).size.height -
MediaQuery.of(context).padding.top -
MediaQuery.of(context).padding.bottom -
566,
),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Container(
alignment: Alignment.center,
child: BrandButton.text(
title: cubit.state is ServerInstallationFinished
? 'basis.close'.tr()
: 'basis.later'.tr(),
onPressed: () {
Navigator.of(context).pushAndRemoveUntil(
materialRoute(const RootPage()),
(final predicate) => false,
);
},
),
),
if (cubit.state is ServerInstallationFinished)
Container()
else
Container(
alignment: Alignment.center,
child: BrandButton.text(
title: 'basis.connect_to_existing'.tr(),
onPressed: () {
Navigator.of(context).push(
materialRoute(
const RecoveryRouting(),
),
);
},
),
)
],
),
),
],
),
2021-04-22 18:04:24 +00:00
),
2021-01-06 17:35:57 +00:00
),
2020-12-30 14:13:25 +00:00
),
);
}
2020-12-10 20:33:19 +00:00
}
Widget _stepHetzner(final ServerInstallationCubit serverInstallationCubit) =>
BlocProvider(
2022-07-12 12:54:16 +00:00
create: (final context) => ProviderFormCubit(
serverInstallationCubit,
),
child: Builder(
builder: (final context) {
2022-07-12 12:54:16 +00:00
final formCubitState = context.watch<ProviderFormCubit>().state;
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Image.asset(
'assets/images/logos/hetzner.png',
width: 150,
),
const SizedBox(height: 10),
BrandText.h2('initializing.connect_to_server'.tr()),
const SizedBox(height: 10),
BrandText.body2('initializing.place_where_data'.tr()),
const Spacer(),
CubitFormTextField(
2022-07-12 12:54:16 +00:00
formFieldCubit: context.read<ProviderFormCubit>().apiKey,
textAlign: TextAlign.center,
scrollPadding: const EdgeInsets.only(bottom: 70),
decoration: const InputDecoration(
hintText: 'Hetzner API Token',
),
),
const Spacer(),
BrandButton.rised(
onPressed: formCubitState.isSubmitting
? null
2022-07-12 12:54:16 +00:00
: () => context.read<ProviderFormCubit>().trySubmit(),
text: 'basis.connect'.tr(),
),
const SizedBox(height: 10),
BrandButton.text(
onPressed: () => _showModal(
context,
const _HowTo(fileName: 'how_hetzner'),
),
title: 'initializing.how'.tr(),
),
],
);
},
),
);
2020-12-30 14:13:25 +00:00
void _showModal(final BuildContext context, final Widget widget) {
2020-12-10 20:33:19 +00:00
showModalBottomSheet<void>(
context: context,
isScrollControlled: true,
backgroundColor: Colors.transparent,
builder: (final BuildContext context) => widget,
2020-12-10 20:33:19 +00:00
);
}
Widget _stepCloudflare(final ServerInstallationCubit initializingCubit) =>
BlocProvider(
create: (final context) => DnsProviderFormCubit(initializingCubit),
child: Builder(
builder: (final context) {
final formCubitState = context.watch<DnsProviderFormCubit>().state;
2021-01-06 17:35:57 +00:00
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Image.asset(
'assets/images/logos/cloudflare.png',
width: 150,
),
const SizedBox(height: 10),
BrandText.h2('initializing.connect_cloudflare'.tr()),
const SizedBox(height: 10),
BrandText.body2('initializing.manage_domain_dns'.tr()),
const Spacer(),
CubitFormTextField(
formFieldCubit: context.read<DnsProviderFormCubit>().apiKey,
textAlign: TextAlign.center,
scrollPadding: const EdgeInsets.only(bottom: 70),
decoration: InputDecoration(
hintText: 'initializing.cloudflare_api_token'.tr(),
),
),
const Spacer(),
BrandButton.rised(
onPressed: formCubitState.isSubmitting
? null
: () => context.read<DnsProviderFormCubit>().trySubmit(),
text: 'basis.connect'.tr(),
),
const SizedBox(height: 10),
BrandButton.text(
onPressed: () => _showModal(
context,
const _HowTo(
fileName: 'how_cloudflare',
),
),
title: 'initializing.how'.tr(),
),
],
);
},
),
);
2021-02-03 19:51:07 +00:00
Widget _stepBackblaze(final ServerInstallationCubit initializingCubit) =>
BlocProvider(
create: (final context) => BackblazeFormCubit(initializingCubit),
child: Builder(
builder: (final context) {
final formCubitState = context.watch<BackblazeFormCubit>().state;
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Image.asset(
'assets/images/logos/backblaze.png',
height: 50,
),
const SizedBox(height: 10),
BrandText.h2('initializing.connect_backblaze_storage'.tr()),
const SizedBox(height: 10),
const Spacer(),
CubitFormTextField(
formFieldCubit: context.read<BackblazeFormCubit>().keyId,
textAlign: TextAlign.center,
scrollPadding: const EdgeInsets.only(bottom: 70),
decoration: const InputDecoration(
hintText: 'KeyID',
),
),
const Spacer(),
CubitFormTextField(
formFieldCubit:
context.read<BackblazeFormCubit>().applicationKey,
textAlign: TextAlign.center,
scrollPadding: const EdgeInsets.only(bottom: 70),
decoration: const InputDecoration(
hintText: 'Master Application Key',
),
),
const Spacer(),
BrandButton.rised(
onPressed: formCubitState.isSubmitting
? null
: () => context.read<BackblazeFormCubit>().trySubmit(),
text: 'basis.connect'.tr(),
),
const SizedBox(height: 10),
BrandButton.text(
onPressed: () => _showModal(
context,
const _HowTo(
fileName: 'how_backblaze',
2021-02-15 18:58:29 +00:00
),
),
title: 'initializing.how'.tr(),
),
],
);
},
),
);
Widget _stepDomain(final ServerInstallationCubit initializingCubit) =>
BlocProvider(
create: (final context) => DomainSetupCubit(initializingCubit)..load(),
child: Builder(
builder: (final context) {
final DomainSetupState state =
context.watch<DomainSetupCubit>().state;
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Image.asset(
'assets/images/logos/cloudflare.png',
width: 150,
),
const SizedBox(height: 30),
BrandText.h2('basis.domain'.tr()),
const SizedBox(height: 10),
if (state is Empty)
BrandText.body2('initializing.no_connected_domains'.tr()),
if (state is Loading)
BrandText.body2(
state.type == LoadingTypes.loadingDomain
? 'initializing.loading_domain_list'.tr()
: 'basis.saving'.tr(),
),
if (state is MoreThenOne)
BrandText.body2(
'initializing.found_more_domains'.tr(),
),
if (state is Loaded) ...[
const SizedBox(height: 10),
Row(
mainAxisSize: MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.end,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Expanded(
child: BrandText.h3(
state.domain,
textAlign: TextAlign.center,
),
),
SizedBox(
width: 56,
child: BrandButton.rised(
onPressed: () =>
context.read<DomainSetupCubit>().load(),
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
mainAxisSize: MainAxisSize.min,
children: const [
Icon(
Icons.refresh,
color: Colors.white,
),
],
2021-02-15 18:58:29 +00:00
),
),
2021-02-15 18:58:29 +00:00
),
],
)
],
if (state is Empty) ...[
const SizedBox(height: 30),
BrandButton.rised(
onPressed: () => context.read<DomainSetupCubit>().load(),
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
const Icon(
Icons.refresh,
color: Colors.white,
),
const SizedBox(width: 10),
BrandText.buttonTitleText('domain.update_list'.tr()),
],
2021-02-15 18:58:29 +00:00
),
),
],
if (state is Loaded) ...[
const SizedBox(height: 30),
BrandButton.rised(
onPressed: () =>
context.read<DomainSetupCubit>().saveDomain(),
text: 'initializing.save_domain'.tr(),
),
],
const SizedBox(
height: 10,
width: double.infinity,
2021-02-15 18:58:29 +00:00
),
],
);
},
),
);
2020-12-30 14:13:25 +00:00
Widget _stepUser(final ServerInstallationCubit initializingCubit) =>
BlocProvider(
create: (final context) =>
RootUserFormCubit(initializingCubit, FieldCubitFactory(context)),
child: Builder(
builder: (final context) {
final formCubitState = context.watch<RootUserFormCubit>().state;
2021-01-06 17:35:57 +00:00
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
BrandText.h2('initializing.create_master_account'.tr()),
const SizedBox(height: 10),
BrandText.body2(
'initializing.enter_username_and_password'.tr(),
),
const Spacer(),
if (formCubitState.isErrorShown)
Text(
'users.username_rule'.tr(),
style: TextStyle(
color: Theme.of(context).colorScheme.error,
),
),
const SizedBox(height: 10),
CubitFormTextField(
formFieldCubit: context.read<RootUserFormCubit>().userName,
2021-02-15 18:58:29 +00:00
textAlign: TextAlign.center,
2022-05-24 18:55:39 +00:00
scrollPadding: const EdgeInsets.only(bottom: 70),
2021-02-15 18:58:29 +00:00
decoration: InputDecoration(
hintText: 'basis.username'.tr(),
2021-02-15 18:58:29 +00:00
),
),
const SizedBox(height: 10),
BlocBuilder<FieldCubit<bool>, FieldCubitState<bool>>(
bloc: context.read<RootUserFormCubit>().isVisible,
builder: (final context, final state) {
final bool isVisible = state.value;
return CubitFormTextField(
obscureText: !isVisible,
formFieldCubit:
context.read<RootUserFormCubit>().password,
textAlign: TextAlign.center,
scrollPadding: const EdgeInsets.only(bottom: 70),
decoration: InputDecoration(
hintText: 'basis.password'.tr(),
suffixIcon: IconButton(
icon: Icon(
isVisible ? Icons.visibility : Icons.visibility_off,
),
onPressed: () => context
.read<RootUserFormCubit>()
.isVisible
.setValue(!isVisible),
),
suffixIconConstraints:
const BoxConstraints(minWidth: 60),
prefixIconConstraints:
const BoxConstraints(maxWidth: 60),
prefixIcon: Container(),
),
);
},
),
const Spacer(),
BrandButton.rised(
onPressed: formCubitState.isSubmitting
? null
: () => context.read<RootUserFormCubit>().trySubmit(),
text: 'basis.connect'.tr(),
),
],
);
},
),
);
2020-12-10 20:33:19 +00:00
Widget _stepServer(final ServerInstallationCubit appConfigCubit) {
final bool isLoading =
(appConfigCubit.state as ServerInstallationNotFinished).isLoading;
return Builder(
builder: (final context) => Column(
2021-01-06 17:35:57 +00:00
crossAxisAlignment: CrossAxisAlignment.start,
children: [
2022-05-24 18:55:39 +00:00
const Spacer(flex: 2),
2021-03-24 13:12:09 +00:00
BrandText.h2('initializing.final'.tr()),
2022-05-24 18:55:39 +00:00
const SizedBox(height: 10),
BrandText.body2('initializing.create_server'.tr()),
2022-05-24 18:55:39 +00:00
const Spacer(),
2021-01-06 17:35:57 +00:00
BrandButton.rised(
onPressed:
isLoading ? null : appConfigCubit.createServerAndSetDnsRecords,
text: isLoading
? 'basis.loading'.tr()
: 'initializing.create_server'.tr(),
2021-01-06 17:35:57 +00:00
),
],
),
);
2021-01-06 17:35:57 +00:00
}
Widget _stepCheck(final ServerInstallationCubit appConfigCubit) {
assert(
appConfigCubit.state is ServerInstallationNotFinished,
'wrong state',
);
final state = appConfigCubit.state as TimerState;
2021-06-20 21:08:52 +00:00
late int doneCount;
2021-03-23 19:50:11 +00:00
late String? text;
2021-03-31 11:37:39 +00:00
if (state.isServerResetedSecondTime) {
text = 'initializing.server_rebooted'.tr();
2021-06-20 21:08:52 +00:00
doneCount = 3;
2021-03-31 11:37:39 +00:00
} else if (state.isServerResetedFirstTime) {
text = 'initializing.one_more_restart'.tr();
2021-06-20 21:08:52 +00:00
doneCount = 2;
2021-03-23 19:50:11 +00:00
} else if (state.isServerStarted) {
text = 'initializing.server_started'.tr();
2021-06-20 21:08:52 +00:00
doneCount = 1;
2021-02-16 18:48:15 +00:00
} else if (state.isServerCreated) {
text = 'initializing.server_created'.tr();
2021-06-20 21:08:52 +00:00
doneCount = 0;
2021-02-16 18:48:15 +00:00
}
return Builder(
builder: (final context) => Column(
2021-02-16 18:48:15 +00:00
crossAxisAlignment: CrossAxisAlignment.start,
children: [
2022-05-24 18:55:39 +00:00
const SizedBox(height: 15),
2021-06-20 21:08:52 +00:00
BrandText.h4(
2022-05-24 18:55:39 +00:00
'initializing.checks'.tr(args: [doneCount.toString(), '4']),
2021-06-20 21:08:52 +00:00
),
2022-05-24 18:55:39 +00:00
const Spacer(flex: 2),
const SizedBox(height: 10),
2021-02-16 18:48:15 +00:00
BrandText.body2(text),
2022-05-24 18:55:39 +00:00
const SizedBox(height: 10),
2022-02-08 06:59:19 +00:00
if (doneCount == 0 && state.dnsMatches != null)
Column(
children: state.dnsMatches!.entries.map((final entry) {
final String domain = entry.key;
final bool isCorrect = entry.value;
2022-02-08 06:59:19 +00:00
return Row(
children: [
2022-05-24 18:55:39 +00:00
if (isCorrect) const Icon(Icons.check, color: Colors.green),
if (!isCorrect)
const Icon(Icons.schedule, color: Colors.amber),
const SizedBox(width: 10),
2022-02-08 06:59:19 +00:00
Text(domain),
],
);
}).toList(),
),
2022-05-24 18:55:39 +00:00
const SizedBox(height: 10),
2021-04-22 18:04:24 +00:00
if (!state.isLoading)
2021-02-16 18:48:15 +00:00
Row(
children: [
BrandText.body2('initializing.until_the_next_check'.tr()),
2021-02-16 18:48:15 +00:00
BrandTimer(
2021-03-23 19:50:11 +00:00
startDateTime: state.timerStart!,
duration: state.duration!,
2021-02-16 18:48:15 +00:00
)
],
),
if (state.isLoading) BrandText.body2('initializing.check'.tr()),
2021-02-16 18:48:15 +00:00
],
),
2020-12-30 14:13:25 +00:00
);
2020-12-10 20:33:19 +00:00
}
Widget _addCard(final Widget child) => Container(
height: 450,
padding: paddingH15V0,
child: BrandCards.big(child: child),
);
2020-12-10 20:33:19 +00:00
}
2021-12-23 13:52:12 +00:00
class _HowTo extends StatelessWidget {
const _HowTo({
required this.fileName,
});
2020-12-10 20:33:19 +00:00
2021-12-23 13:52:12 +00:00
final String fileName;
2020-12-10 20:33:19 +00:00
@override
Widget build(final BuildContext context) => BrandBottomSheet(
isExpended: true,
child: Padding(
padding: paddingH15V0,
2022-06-09 21:13:06 +00:00
child: ListView(
padding: const EdgeInsets.symmetric(vertical: 16),
children: [
BrandMarkdown(
fileName: fileName,
),
],
),
2021-06-20 21:08:52 +00:00
),
);
2020-12-10 20:33:19 +00:00
}