2022-09-06 10:25:28 +00:00
|
|
|
import 'package:easy_localization/easy_localization.dart';
|
2022-08-24 05:35:49 +00:00
|
|
|
import 'package:flutter/material.dart';
|
2022-08-29 20:35:06 +00:00
|
|
|
import 'package:flutter_svg/svg.dart';
|
2022-09-06 10:25:28 +00:00
|
|
|
import 'package:selfprivacy/logic/common_enum/common_enum.dart';
|
|
|
|
import 'package:selfprivacy/logic/cubit/client_jobs/client_jobs_cubit.dart';
|
2022-08-29 20:35:06 +00:00
|
|
|
import 'package:selfprivacy/logic/cubit/services/services_cubit.dart';
|
2022-09-06 10:25:28 +00:00
|
|
|
import 'package:selfprivacy/logic/models/job.dart';
|
2022-08-29 20:35:06 +00:00
|
|
|
import 'package:selfprivacy/logic/models/service.dart';
|
2022-08-24 05:35:49 +00:00
|
|
|
import 'package:selfprivacy/ui/components/brand_cards/brand_cards.dart';
|
|
|
|
import 'package:selfprivacy/ui/components/brand_hero_screen/brand_hero_screen.dart';
|
2022-08-29 20:35:06 +00:00
|
|
|
import 'package:url_launcher/url_launcher.dart';
|
2022-08-24 05:35:49 +00:00
|
|
|
|
|
|
|
class ServicePage extends StatefulWidget {
|
2022-08-29 20:35:06 +00:00
|
|
|
const ServicePage({required this.serviceId, final super.key});
|
|
|
|
|
|
|
|
final String serviceId;
|
2022-08-24 05:35:49 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
State<ServicePage> createState() => _ServicePageState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _ServicePageState extends State<ServicePage> {
|
|
|
|
@override
|
2022-08-29 20:35:06 +00:00
|
|
|
Widget build(final BuildContext context) {
|
|
|
|
final Service? service =
|
|
|
|
context.watch<ServicesCubit>().state.getServiceById(widget.serviceId);
|
|
|
|
|
|
|
|
if (service == null) {
|
|
|
|
return const BrandHeroScreen(
|
2022-08-29 18:18:07 +00:00
|
|
|
hasBackButton: true,
|
|
|
|
children: [
|
2022-08-29 20:35:06 +00:00
|
|
|
Center(
|
|
|
|
child: CircularProgressIndicator(),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
2022-09-06 10:25:28 +00:00
|
|
|
|
|
|
|
final bool serviceDisabled = service.status == ServiceStatus.inactive ||
|
|
|
|
service.status == ServiceStatus.off;
|
|
|
|
|
2022-08-29 20:35:06 +00:00
|
|
|
return BrandHeroScreen(
|
|
|
|
hasBackButton: true,
|
|
|
|
children: [
|
|
|
|
Container(
|
|
|
|
alignment: Alignment.center,
|
|
|
|
child: SvgPicture.string(
|
|
|
|
service.svgIcon,
|
|
|
|
width: 48.0,
|
|
|
|
height: 48.0,
|
2022-09-14 16:45:50 +00:00
|
|
|
color: Theme.of(context).colorScheme.onBackground,
|
2022-08-29 20:35:06 +00:00
|
|
|
),
|
|
|
|
),
|
|
|
|
const SizedBox(height: 16),
|
|
|
|
Text(
|
|
|
|
service.displayName,
|
|
|
|
textAlign: TextAlign.center,
|
|
|
|
style: Theme.of(context).textTheme.headlineMedium!.copyWith(
|
|
|
|
color: Theme.of(context).colorScheme.onBackground,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
const SizedBox(height: 16),
|
|
|
|
ServiceStatusCard(status: service.status),
|
|
|
|
const SizedBox(height: 16),
|
|
|
|
if (service.url != null)
|
|
|
|
ListTile(
|
|
|
|
iconColor: Theme.of(context).colorScheme.onBackground,
|
|
|
|
onTap: () => _launchURL(service.url),
|
|
|
|
leading: const Icon(Icons.open_in_browser),
|
|
|
|
title: Text(
|
|
|
|
'Open in browser',
|
|
|
|
style: Theme.of(context).textTheme.titleMedium,
|
|
|
|
),
|
|
|
|
subtitle: Text(
|
|
|
|
service.url!.replaceAll('https://', ''),
|
|
|
|
style: Theme.of(context).textTheme.bodyMedium,
|
2022-08-29 18:18:07 +00:00
|
|
|
),
|
2022-08-24 05:35:49 +00:00
|
|
|
),
|
2022-08-29 20:35:06 +00:00
|
|
|
const SizedBox(height: 8),
|
|
|
|
const Divider(),
|
|
|
|
const SizedBox(height: 8),
|
|
|
|
ListTile(
|
|
|
|
iconColor: Theme.of(context).colorScheme.onBackground,
|
2022-09-06 10:25:28 +00:00
|
|
|
onTap: () => {
|
|
|
|
context.read<ServicesCubit>().restart(service.id),
|
|
|
|
},
|
2022-08-29 20:35:06 +00:00
|
|
|
leading: const Icon(Icons.restart_alt_outlined),
|
|
|
|
title: Text(
|
2022-09-06 10:25:28 +00:00
|
|
|
'services.service_page.restart'.tr(),
|
2022-08-29 20:35:06 +00:00
|
|
|
style: Theme.of(context).textTheme.titleMedium,
|
2022-08-24 05:35:49 +00:00
|
|
|
),
|
2022-08-29 20:35:06 +00:00
|
|
|
),
|
|
|
|
ListTile(
|
|
|
|
iconColor: Theme.of(context).colorScheme.onBackground,
|
2022-09-06 10:25:28 +00:00
|
|
|
onTap: () => {
|
|
|
|
context.read<JobsCubit>().createOrRemoveServiceToggleJob(
|
|
|
|
ServiceToggleJob(
|
|
|
|
type: _idToLegacyType(service.id),
|
|
|
|
needToTurnOn: serviceDisabled,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
},
|
2022-08-29 20:35:06 +00:00
|
|
|
leading: const Icon(Icons.power_settings_new),
|
|
|
|
title: Text(
|
2022-09-06 10:25:28 +00:00
|
|
|
serviceDisabled
|
|
|
|
? 'services.service_page.enable'.tr()
|
|
|
|
: 'services.service_page.disable'.tr(),
|
2022-08-29 20:35:06 +00:00
|
|
|
style: Theme.of(context).textTheme.titleMedium,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
if (service.isMovable)
|
|
|
|
ListTile(
|
|
|
|
iconColor: Theme.of(context).colorScheme.onBackground,
|
|
|
|
onTap: () => {},
|
|
|
|
leading: const Icon(Icons.drive_file_move_outlined),
|
|
|
|
title: Text(
|
2022-09-06 10:25:28 +00:00
|
|
|
'services.service_page.move'.tr(),
|
2022-08-29 20:35:06 +00:00
|
|
|
style: Theme.of(context).textTheme.titleMedium,
|
2022-08-29 18:18:07 +00:00
|
|
|
),
|
|
|
|
),
|
2022-08-29 20:35:06 +00:00
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
2022-09-06 10:25:28 +00:00
|
|
|
|
|
|
|
// TODO: Get rid as soon as possible
|
|
|
|
ServiceTypes _idToLegacyType(final String serviceId) {
|
|
|
|
switch (serviceId) {
|
|
|
|
case 'mailserver':
|
|
|
|
return ServiceTypes.mailserver;
|
|
|
|
case 'jitsi':
|
|
|
|
return ServiceTypes.jitsi;
|
|
|
|
case 'bitwarden':
|
|
|
|
return ServiceTypes.bitwarden;
|
|
|
|
case 'nextcloud':
|
|
|
|
return ServiceTypes.nextcloud;
|
|
|
|
case 'pleroma':
|
|
|
|
return ServiceTypes.pleroma;
|
|
|
|
case 'gitea':
|
|
|
|
return ServiceTypes.gitea;
|
|
|
|
case 'ocserv':
|
|
|
|
return ServiceTypes.ocserv;
|
|
|
|
default:
|
|
|
|
throw Exception('wrong state');
|
|
|
|
}
|
|
|
|
}
|
2022-08-29 20:35:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class ServiceStatusCard extends StatelessWidget {
|
|
|
|
const ServiceStatusCard({
|
|
|
|
required this.status,
|
|
|
|
final super.key,
|
|
|
|
});
|
|
|
|
final ServiceStatus status;
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(final BuildContext context) {
|
|
|
|
switch (status) {
|
|
|
|
case ServiceStatus.active:
|
|
|
|
return BrandCards.filled(
|
|
|
|
child: const ListTile(
|
|
|
|
leading: Icon(
|
|
|
|
Icons.check_circle_outline,
|
|
|
|
size: 24,
|
2022-08-29 18:18:07 +00:00
|
|
|
),
|
2022-08-29 20:35:06 +00:00
|
|
|
title: Text('Up and running'),
|
2022-08-24 05:35:49 +00:00
|
|
|
),
|
2022-08-29 20:35:06 +00:00
|
|
|
);
|
|
|
|
case ServiceStatus.inactive:
|
|
|
|
return BrandCards.filled(
|
|
|
|
child: const ListTile(
|
|
|
|
leading: Icon(
|
|
|
|
Icons.stop_circle_outlined,
|
|
|
|
size: 24,
|
|
|
|
),
|
|
|
|
title: Text('Stopped'),
|
|
|
|
),
|
|
|
|
tertiary: true,
|
|
|
|
);
|
|
|
|
case ServiceStatus.failed:
|
|
|
|
return BrandCards.filled(
|
|
|
|
child: const ListTile(
|
|
|
|
leading: Icon(
|
|
|
|
Icons.error_outline,
|
|
|
|
size: 24,
|
|
|
|
),
|
|
|
|
title: Text('Failed to start'),
|
|
|
|
),
|
|
|
|
error: true,
|
|
|
|
);
|
|
|
|
case ServiceStatus.off:
|
|
|
|
return BrandCards.filled(
|
|
|
|
child: const ListTile(
|
|
|
|
leading: Icon(
|
|
|
|
Icons.power_settings_new,
|
|
|
|
size: 24,
|
|
|
|
),
|
|
|
|
title: Text('Disabled'),
|
|
|
|
),
|
|
|
|
tertiary: true,
|
|
|
|
);
|
|
|
|
case ServiceStatus.activating:
|
|
|
|
return BrandCards.filled(
|
|
|
|
child: const ListTile(
|
|
|
|
leading: Icon(
|
|
|
|
Icons.restart_alt_outlined,
|
|
|
|
size: 24,
|
|
|
|
),
|
|
|
|
title: Text('Activating'),
|
|
|
|
),
|
|
|
|
tertiary: true,
|
|
|
|
);
|
|
|
|
case ServiceStatus.deactivating:
|
|
|
|
return BrandCards.filled(
|
|
|
|
child: const ListTile(
|
|
|
|
leading: Icon(
|
|
|
|
Icons.restart_alt_outlined,
|
|
|
|
size: 24,
|
|
|
|
),
|
|
|
|
title: Text('Deactivating'),
|
|
|
|
),
|
|
|
|
tertiary: true,
|
|
|
|
);
|
|
|
|
case ServiceStatus.reloading:
|
|
|
|
return BrandCards.filled(
|
|
|
|
child: const ListTile(
|
|
|
|
leading: Icon(
|
|
|
|
Icons.restart_alt_outlined,
|
|
|
|
size: 24,
|
|
|
|
),
|
|
|
|
title: Text('Restarting'),
|
|
|
|
),
|
|
|
|
tertiary: true,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _launchURL(final url) async {
|
|
|
|
try {
|
|
|
|
final Uri uri = Uri.parse(url);
|
|
|
|
await launchUrl(
|
|
|
|
uri,
|
|
|
|
mode: LaunchMode.externalApplication,
|
|
|
|
);
|
|
|
|
} catch (e) {
|
|
|
|
print(e);
|
|
|
|
}
|
2022-08-24 05:35:49 +00:00
|
|
|
}
|