Merge pull request 'refactor(ui): Reorganize placeholders for empty pages' (#359) from plug-backgrounds into master
continuous-integration/drone/push Build is passing Details

Reviewed-on: #359
Reviewed-by: Inex Code <inex.code@selfprivacy.org>
pull/361/head
NaiJi ✨ 2023-09-26 20:26:15 +03:00
commit e49b5db4b6
7 changed files with 124 additions and 128 deletions

View File

@ -35,7 +35,8 @@
"done": "Done", "done": "Done",
"continue": "Continue", "continue": "Continue",
"alert": "Alert", "alert": "Alert",
"copied_to_clipboard": "Copied to clipboard!" "copied_to_clipboard": "Copied to clipboard!",
"please_connect": "Please connect your server, domain and DNS provider to dive in!"
}, },
"more_page": { "more_page": {
"configuration_wizard": "Setup wizard", "configuration_wizard": "Setup wizard",
@ -315,6 +316,7 @@
"in_menu": "Server is not set up yet. Please finish setup using setup wizard for further work." "in_menu": "Server is not set up yet. Please finish setup using setup wizard for further work."
}, },
"service_page": { "service_page": {
"nothing_here": "Nothing here",
"open_in_browser": "Open in browser", "open_in_browser": "Open in browser",
"restart": "Restart service", "restart": "Restart service",
"disable": "Disable service", "disable": "Disable service",
@ -371,7 +373,6 @@
"add_new_user": "Add a first user", "add_new_user": "Add a first user",
"new_user": "New user", "new_user": "New user",
"delete_user": "Delete user", "delete_user": "Delete user",
"not_ready": "Please connect server, domain and DNS in the Providers tab, to be able to add a first user",
"nobody_here": "Nobody here", "nobody_here": "Nobody here",
"login": "Login", "login": "Login",
"new_user_info_note": "New user will automatically be granted an access to all of the services", "new_user_info_note": "New user will automatically be granted an access to all of the services",

View File

@ -75,7 +75,7 @@ class DnsRecordsCubit
@override @override
Future<void> clear() async { Future<void> clear() async {
emit(const DnsRecordsState(dnsState: DnsRecordsStatus.error)); emit(const DnsRecordsState(dnsState: DnsRecordsStatus.uninitialized));
} }
Future<void> refresh() async { Future<void> refresh() async {

View File

@ -0,0 +1,67 @@
import 'package:flutter/material.dart';
import 'package:selfprivacy/ui/components/not_ready_card/not_ready_card.dart';
class EmptyPagePlaceholder extends StatelessWidget {
const EmptyPagePlaceholder({
required this.title,
required this.iconData,
required this.description,
this.showReadyCard = false,
super.key,
});
final String title;
final String description;
final IconData iconData;
final bool showReadyCard;
@override
Widget build(final BuildContext context) => !showReadyCard
? _expandedContent(context)
: Column(
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
const Padding(
padding: EdgeInsets.symmetric(horizontal: 15),
child: NotReadyCard(),
),
Expanded(
child: Container(
padding: const EdgeInsets.symmetric(horizontal: 15),
child: Center(
child: _expandedContent(context),
),
),
)
],
);
Widget _expandedContent(final BuildContext context) => Padding(
padding: const EdgeInsets.symmetric(horizontal: 20),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
iconData,
size: 50,
color: Theme.of(context).colorScheme.onBackground,
),
const SizedBox(height: 16),
Text(
title,
style: Theme.of(context).textTheme.headlineMedium?.copyWith(
color: Theme.of(context).colorScheme.onBackground,
),
),
const SizedBox(height: 8),
Text(
description,
textAlign: TextAlign.center,
style: Theme.of(context).textTheme.titleSmall?.copyWith(
color: Theme.of(context).colorScheme.onBackground,
),
),
],
),
);
}

View File

@ -49,6 +49,8 @@ class _ProvidersPageState extends State<ProvidersPage> {
return StateType.stable; return StateType.stable;
} }
bool isClickable() => getServerStatus() != StateType.uninitialized;
StateType getDnsStatus() { StateType getDnsStatus() {
if (dnsStatus == DnsRecordsStatus.uninitialized || if (dnsStatus == DnsRecordsStatus.uninitialized ||
dnsStatus == DnsRecordsStatus.refreshing) { dnsStatus == DnsRecordsStatus.refreshing) {
@ -83,7 +85,9 @@ class _ProvidersPageState extends State<ProvidersPage> {
subtitle: diskStatus.isDiskOkay subtitle: diskStatus.isDiskOkay
? 'storage.status_ok'.tr() ? 'storage.status_ok'.tr()
: 'storage.status_error'.tr(), : 'storage.status_error'.tr(),
onTap: () => context.pushRoute(const ServerDetailsRoute()), onTap: isClickable()
? () => context.pushRoute(const ServerDetailsRoute())
: null,
), ),
const SizedBox(height: 16), const SizedBox(height: 16),
_Card( _Card(
@ -93,7 +97,9 @@ class _ProvidersPageState extends State<ProvidersPage> {
subtitle: appConfig.isDomainSelected subtitle: appConfig.isDomainSelected
? appConfig.serverDomain!.domainName ? appConfig.serverDomain!.domainName
: '', : '',
onTap: () => context.pushRoute(const DnsDetailsRoute()), onTap: isClickable()
? () => context.pushRoute(const DnsDetailsRoute())
: null,
), ),
const SizedBox(height: 16), const SizedBox(height: 16),
_Card( _Card(
@ -103,7 +109,9 @@ class _ProvidersPageState extends State<ProvidersPage> {
icon: BrandIcons.save, icon: BrandIcons.save,
title: 'backup.card_title'.tr(), title: 'backup.card_title'.tr(),
subtitle: isBackupInitialized ? 'backup.card_subtitle'.tr() : '', subtitle: isBackupInitialized ? 'backup.card_subtitle'.tr() : '',
onTap: () => context.pushRoute(const BackupDetailsRoute()), onTap: isClickable()
? () => context.pushRoute(const BackupDetailsRoute())
: null,
), ),
], ],
), ),

View File

@ -7,9 +7,10 @@ import 'package:selfprivacy/logic/cubit/services/services_cubit.dart';
import 'package:selfprivacy/logic/models/service.dart'; import 'package:selfprivacy/logic/models/service.dart';
import 'package:selfprivacy/logic/models/state_types.dart'; import 'package:selfprivacy/logic/models/state_types.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/icon_status_mask/icon_status_mask.dart'; import 'package:selfprivacy/ui/components/icon_status_mask/icon_status_mask.dart';
import 'package:selfprivacy/ui/components/not_ready_card/not_ready_card.dart';
import 'package:easy_localization/easy_localization.dart'; import 'package:easy_localization/easy_localization.dart';
import 'package:selfprivacy/ui/helpers/empty_page_placeholder.dart';
import 'package:selfprivacy/ui/router/router.dart'; import 'package:selfprivacy/ui/router/router.dart';
import 'package:selfprivacy/utils/breakpoints.dart'; import 'package:selfprivacy/utils/breakpoints.dart';
import 'package:selfprivacy/utils/launch_url.dart'; import 'package:selfprivacy/utils/launch_url.dart';
@ -42,30 +43,36 @@ class _ServicesPageState extends State<ServicesPage> {
), ),
) )
: null, : null,
body: RefreshIndicator( body: !isReady
onRefresh: () async { ? EmptyPagePlaceholder(
await context.read<ServicesCubit>().reload(); showReadyCard: true,
}, title: 'service_page.nothing_here'.tr(),
child: ListView( description: 'basis.please_connect'.tr(),
padding: paddingH15V0, iconData: BrandIcons.box,
children: [
Text(
'basis.services_title'.tr(),
style: Theme.of(context).textTheme.bodyLarge,
),
const SizedBox(height: 24),
if (!isReady) ...[const NotReadyCard(), const SizedBox(height: 24)],
...services.map(
(final service) => Padding(
padding: const EdgeInsets.only(
bottom: 30,
),
child: _Card(service: service),
),
) )
], : RefreshIndicator(
), onRefresh: () async {
), await context.read<ServicesCubit>().reload();
},
child: ListView(
padding: paddingH15V0,
children: [
Text(
'basis.services_title'.tr(),
style: Theme.of(context).textTheme.bodyLarge,
),
const SizedBox(height: 24),
...services.map(
(final service) => Padding(
padding: const EdgeInsets.only(
bottom: 30,
),
child: _Card(service: service),
),
)
],
),
),
); );
} }
} }

View File

@ -1,73 +0,0 @@
part of 'users.dart';
class _NoUsers extends StatelessWidget {
const _NoUsers({required this.text});
final String text;
@override
Widget build(final BuildContext context) => Padding(
padding: const EdgeInsets.symmetric(horizontal: 20),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
BrandIcons.users,
size: 50,
color: Theme.of(context).colorScheme.onBackground,
),
const SizedBox(height: 20),
Text(
'users.nobody_here'.tr(),
style: Theme.of(context).textTheme.headlineMedium?.copyWith(
color: Theme.of(context).colorScheme.onBackground,
),
),
const SizedBox(height: 10),
Text(
text,
textAlign: TextAlign.center,
style: Theme.of(context).textTheme.titleSmall?.copyWith(
color: Theme.of(context).colorScheme.onBackground,
),
),
],
),
);
}
class _CouldNotLoadUsers extends StatelessWidget {
const _CouldNotLoadUsers({required this.text});
final String text;
@override
Widget build(final BuildContext context) => Padding(
padding: const EdgeInsets.symmetric(horizontal: 20),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
BrandIcons.users,
size: 50,
color: Theme.of(context).colorScheme.onBackground,
),
const SizedBox(height: 20),
Text(
'users.could_not_fetch_users'.tr(),
style: Theme.of(context).textTheme.headlineMedium?.copyWith(
color: Theme.of(context).colorScheme.onBackground,
),
),
const SizedBox(height: 10),
Text(
text,
textAlign: TextAlign.center,
style: Theme.of(context).textTheme.titleSmall?.copyWith(
color: Theme.of(context).colorScheme.onBackground,
),
),
],
),
);
}

View File

@ -16,17 +16,16 @@ import 'package:selfprivacy/ui/components/buttons/brand_button.dart';
import 'package:selfprivacy/ui/components/buttons/outlined_button.dart'; import 'package:selfprivacy/ui/components/buttons/outlined_button.dart';
import 'package:selfprivacy/ui/components/cards/filled_card.dart'; import 'package:selfprivacy/ui/components/cards/filled_card.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/helpers/empty_page_placeholder.dart';
import 'package:selfprivacy/ui/layouts/brand_hero_screen.dart'; import 'package:selfprivacy/ui/layouts/brand_hero_screen.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/info_box/info_box.dart'; import 'package:selfprivacy/ui/components/info_box/info_box.dart';
import 'package:selfprivacy/ui/components/list_tiles/list_tile_on_surface_variant.dart'; import 'package:selfprivacy/ui/components/list_tiles/list_tile_on_surface_variant.dart';
import 'package:selfprivacy/ui/components/not_ready_card/not_ready_card.dart';
import 'package:selfprivacy/ui/router/router.dart'; import 'package:selfprivacy/ui/router/router.dart';
import 'package:selfprivacy/utils/breakpoints.dart'; import 'package:selfprivacy/utils/breakpoints.dart';
import 'package:selfprivacy/utils/platform_adapter.dart'; import 'package:selfprivacy/utils/platform_adapter.dart';
import 'package:selfprivacy/utils/ui_helpers.dart'; import 'package:selfprivacy/utils/ui_helpers.dart';
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';
@ -43,7 +42,12 @@ class UsersPage extends StatelessWidget {
Widget child; Widget child;
if (!isReady) { if (!isReady) {
child = isNotReady(); child = EmptyPagePlaceholder(
showReadyCard: true,
title: 'users.nobody_here'.tr(),
description: 'basis.please_connect'.tr(),
iconData: BrandIcons.users,
);
} else { } else {
child = BlocBuilder<UsersCubit, UsersState>( child = BlocBuilder<UsersCubit, UsersState>(
builder: (final BuildContext context, final UsersState state) { builder: (final BuildContext context, final UsersState state) {
@ -64,8 +68,10 @@ class UsersPage extends StatelessWidget {
child: Column( child: Column(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
_CouldNotLoadUsers( EmptyPagePlaceholder(
text: 'users.could_not_fetch_description'.tr(), title: 'users.could_not_fetch_users'.tr(),
description: 'users.could_not_fetch_description'.tr(),
iconData: BrandIcons.users,
), ),
const SizedBox(height: 18), const SizedBox(height: 18),
BrandOutlinedButton( BrandOutlinedButton(
@ -132,24 +138,4 @@ class UsersPage extends StatelessWidget {
body: child, body: child,
); );
} }
Widget isNotReady() => Column(
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
const Padding(
padding: EdgeInsets.symmetric(horizontal: 15),
child: NotReadyCard(),
),
Expanded(
child: Container(
padding: const EdgeInsets.symmetric(horizontal: 15),
child: Center(
child: _NoUsers(
text: 'users.not_ready'.tr(),
),
),
),
)
],
);
} }