Merge remote-tracking branch 'origin/master' into api-connection-refactor

This commit is contained in:
Inex Code 2024-02-20 20:13:19 +03:00
commit 4eb8f34e37
29 changed files with 387 additions and 367 deletions

View file

@ -1,6 +1,9 @@
name: Windows Builder name: Windows Builder
on: tag on:
push:
tags:
- '*.*.*'
jobs: jobs:
build-windows: build-windows:
@ -14,7 +17,7 @@ jobs:
# Install Flutter # Install Flutter
- uses: subosito/flutter-action@v2 - uses: subosito/flutter-action@v2
with: with:
flutter-version: '3.3.10' flutter-version: '3.16.1'
channel: 'stable' channel: 'stable'
# Build Windows artifact # Build Windows artifact

View file

@ -10,7 +10,7 @@ AppDir:
id: org.selfprivacy.app id: org.selfprivacy.app
name: SelfPrivacy name: SelfPrivacy
icon: org.selfprivacy.app icon: org.selfprivacy.app
version: 0.10.0 version: 0.10.1
exec: selfprivacy exec: selfprivacy
exec_args: $@ exec_args: $@
apt: apt:

View file

@ -0,0 +1,15 @@
### Features
- Enabled the following languages:
- Azerbaijani
- Belarusian
- Hebrew
- Latvian
- Macedonian
- Slovak
- Slovenian
### Bug Fixes
- **Hetzner**: Fixed an issue where could not resize a volume on Hetzner ([#456](https://git.selfprivacy.org/SelfPrivacy/selfprivacy.org.app/issues/456), resolves [#455](https://git.selfprivacy.org/SelfPrivacy/selfprivacy.org.app/issues/455))
- **DNS**: Make sure that we notice domain ownership lost ([#441](https://git.selfprivacy.org/SelfPrivacy/selfprivacy.org.app/issues/441), resolves [#390](https://git.selfprivacy.org/SelfPrivacy/selfprivacy.org.app/issues/390))

View file

@ -1 +0,0 @@

View file

@ -547,7 +547,7 @@ class HetznerApi extends RestApiMap {
resizeVolumeResponse = await client.post( resizeVolumeResponse = await client.post(
'/volumes/${volume.id}/actions/resize', '/volumes/${volume.id}/actions/resize',
data: { data: {
'size': size.gibibyte, 'size': size.gibibyte.floor(),
}, },
); );
success = success =

View file

@ -1,8 +1,7 @@
import 'package:graphql/client.dart'; import 'package:graphql/client.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
final DateFormat formatter = DateFormat('hh:mm'); /// TODO(misterfourtytwo): add equality override
class Message { class Message {
Message({this.text, this.severity = MessageSeverity.normal}) Message({this.text, this.severity = MessageSeverity.normal})
: time = DateTime.now(); : time = DateTime.now();
@ -13,7 +12,9 @@ class Message {
final String? text; final String? text;
final DateTime time; final DateTime time;
final MessageSeverity severity; final MessageSeverity severity;
String get timeString => formatter.format(time);
static final DateFormat _formatter = DateFormat('hh:mm');
String get timeString => _formatter.format(time);
} }
enum MessageSeverity { enum MessageSeverity {

View file

@ -92,13 +92,15 @@ class SelfprivacyApp extends StatelessWidget {
? ThemeMode.dark ? ThemeMode.dark
: ThemeMode.light, : ThemeMode.light,
builder: (final BuildContext context, final Widget? widget) { builder: (final BuildContext context, final Widget? widget) {
Widget error = const Text('...rendering error...'); Widget error =
const Center(child: Text('...rendering error...'));
if (widget is Scaffold || widget is Navigator) { if (widget is Scaffold || widget is Navigator) {
error = Scaffold(body: Center(child: error)); error = Scaffold(body: error);
} }
ErrorWidget.builder = ErrorWidget.builder =
(final FlutterErrorDetails errorDetails) => error; (final FlutterErrorDetails errorDetails) => error;
return widget!;
return widget ?? error;
}, },
); );
}, },

View file

@ -14,6 +14,7 @@ class BrandHeader extends StatelessWidget {
@override @override
Widget build(final BuildContext context) => AppBar( Widget build(final BuildContext context) => AppBar(
centerTitle: true,
title: Padding( title: Padding(
padding: const EdgeInsets.only(top: 4.0), padding: const EdgeInsets.only(top: 4.0),
child: Text(title), child: Text(title),
@ -25,8 +26,5 @@ class BrandHeader extends StatelessWidget {
onBackButtonPressed ?? () => Navigator.of(context).pop(), onBackButtonPressed ?? () => Navigator.of(context).pop(),
) )
: null, : null,
actions: const [
SizedBox.shrink(),
],
); );
} }

View file

@ -7,8 +7,9 @@ class BrandButton {
final String? text, final String? text,
final Widget? child, final Widget? child,
}) { }) {
assert(text == null || child == null, 'required title or child'); assert((text ?? child) != null, 'either title or child must not be empty');
assert(text != null || child != null, 'required title or child'); assert(text != null || child != null, 'title or child must be provided');
return ConstrainedBox( return ConstrainedBox(
constraints: const BoxConstraints( constraints: const BoxConstraints(
minHeight: 48, minHeight: 48,
@ -28,8 +29,9 @@ class BrandButton {
final String? text, final String? text,
final Widget? child, final Widget? child,
}) { }) {
assert(text == null || child == null, 'required title or child'); assert((text ?? child) != null, 'either title or child must not be empty');
assert(text != null || child != null, 'required title or child'); assert(text != null || child != null, 'title or child must be provided');
return ConstrainedBox( return ConstrainedBox(
constraints: const BoxConstraints( constraints: const BoxConstraints(
minWidth: double.infinity, minWidth: double.infinity,

View file

@ -0,0 +1,2 @@
export 'brand_button.dart';
export 'sp_brand_button.dart';

View file

@ -46,20 +46,40 @@ class SegmentedButtons extends StatelessWidget {
color: Theme.of(context).colorScheme.onSurface, color: Theme.of(context).colorScheme.onSurface,
isSelected: isSelected, isSelected: isSelected,
onPressed: onPressed, onPressed: onPressed,
children: titles.asMap().entries.map((final entry) { children: [
final index = entry.key; for (int i = 0; i < titles.length; i++)
final title = entry.value; _ButtonSegment(
return Stack( key: ValueKey(i),
isSelected: isSelected[i],
title: titles[i],
),
],
),
);
}
class _ButtonSegment extends StatelessWidget {
const _ButtonSegment({
required this.isSelected,
required this.title,
super.key,
});
final bool isSelected;
final String title;
@override
Widget build(final BuildContext context) => Stack(
alignment: Alignment.centerLeft, alignment: Alignment.centerLeft,
children: [ children: [
AnimatedOpacity( AnimatedOpacity(
duration: const Duration(milliseconds: 200), duration: const Duration(milliseconds: 200),
opacity: isSelected[index] ? 1 : 0, opacity: isSelected ? 1 : 0,
child: AnimatedScale( child: AnimatedScale(
duration: const Duration(milliseconds: 200), duration: const Duration(milliseconds: 200),
curve: Curves.easeInOutCubicEmphasized, curve: Curves.easeInOutCubicEmphasized,
alignment: Alignment.centerLeft, alignment: Alignment.centerLeft,
scale: isSelected[index] ? 1 : 0, scale: isSelected ? 1 : 0,
child: Icon( child: Icon(
Icons.check, Icons.check,
size: 18, size: 18,
@ -68,9 +88,8 @@ class SegmentedButtons extends StatelessWidget {
), ),
), ),
AnimatedPadding( AnimatedPadding(
padding: isSelected[index] padding:
? const EdgeInsets.only(left: 24) isSelected ? const EdgeInsets.only(left: 24) : EdgeInsets.zero,
: EdgeInsets.zero,
duration: const Duration(milliseconds: 200), duration: const Duration(milliseconds: 200),
curve: Curves.easeInOutCubicEmphasized, curve: Curves.easeInOutCubicEmphasized,
child: Text( child: Text(
@ -80,7 +99,4 @@ class SegmentedButtons extends StatelessWidget {
), ),
], ],
); );
}).toList(),
),
);
} }

View file

@ -0,0 +1,28 @@
import 'package:flutter/material.dart';
class SPBrandButton extends StatelessWidget {
const SPBrandButton({
required this.child,
required this.onPressed,
super.key,
});
SPBrandButton.text({
required final String title,
required this.onPressed,
super.key,
}) : child = Text(title);
final Widget child;
final VoidCallback onPressed;
@override
Widget build(final BuildContext context) => FilledButton(
// TODO(misterfourtytwo): move button styles to theme configuration
style: const ButtonStyle(
minimumSize: MaterialStatePropertyAll(Size.fromHeight(48)),
),
onPressed: onPressed,
child: child,
);
}

View file

@ -1,5 +1,4 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:selfprivacy/logic/cubit/app_settings/app_settings_cubit.dart';
class ProgressBar extends StatefulWidget { class ProgressBar extends StatefulWidget {
const ProgressBar({ const ProgressBar({
@ -21,41 +20,6 @@ class _ProgressBarState extends State<ProgressBar> {
Widget build(final BuildContext context) { Widget build(final BuildContext context) {
final double progress = final double progress =
1 / widget.steps.length * (widget.activeIndex + 0.3); 1 / widget.steps.length * (widget.activeIndex + 0.3);
final bool isDark = context.watch<AppSettingsCubit>().state.isDarkModeOn;
final TextStyle style =
isDark ? progressTextStyleDark : progressTextStyleLight;
final Iterable<Container> allSteps = widget.steps.asMap().map(
(final i, final step) {
final Container value = _stepTitle(index: i, style: style, step: step);
return MapEntry(i, value);
},
).values;
final List<Widget> odd = [];
final List<Widget> even = [];
int i = 0;
for (final Container step in allSteps) {
if (i.isEven) {
even.add(step);
} else {
odd.add(step);
}
i++;
}
odd.insert(
0,
const SizedBox(
width: 10,
),
);
odd.add(
const SizedBox(
width: 20,
),
);
return Column( return Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
@ -91,39 +55,4 @@ class _ProgressBarState extends State<ProgressBar> {
], ],
); );
} }
Container _stepTitle({
required final int index,
TextStyle? style,
final String? step,
}) {
final bool isActive = index == widget.activeIndex;
style = isActive ? style!.copyWith(fontWeight: FontWeight.w700) : style;
return Container(
padding: const EdgeInsets.only(left: 10),
height: 20,
alignment: Alignment.center,
child: RichText(
textAlign: TextAlign.justify,
text: TextSpan(
style: progressTextStyleLight,
children: [
TextSpan(text: '${index + 1}.', style: style),
TextSpan(text: step, style: style),
],
),
),
);
}
} }
const TextStyle progressTextStyleLight = TextStyle(
fontSize: 11,
color: Colors.black,
height: 1.7,
);
final TextStyle progressTextStyleDark = progressTextStyleLight.copyWith(
color: Colors.white,
);

View file

@ -201,7 +201,7 @@ class _BottomBar extends StatelessWidget {
), ),
child: NavigationBar( child: NavigationBar(
selectedIndex: prevActiveIndex == -1 ? 0 : prevActiveIndex, selectedIndex: prevActiveIndex == -1 ? 0 : prevActiveIndex,
labelBehavior: NavigationDestinationLabelBehavior.onlyShowSelected, labelBehavior: NavigationDestinationLabelBehavior.alwaysShow,
onDestinationSelected: (final index) { onDestinationSelected: (final index) {
context.router.replaceAll([destinations[index].route]); context.router.replaceAll([destinations[index].route]);
}, },

View file

@ -17,7 +17,6 @@ class SnapshotIdListTile extends StatelessWidget {
PlatformAdapter.setClipboard(snapshotId); PlatformAdapter.setClipboard(snapshotId);
getIt<NavigationService>().showSnackBar( getIt<NavigationService>().showSnackBar(
'basis.copied_to_clipboard'.tr(), 'basis.copied_to_clipboard'.tr(),
behavior: SnackBarBehavior.floating,
); );
}, },
leading: Icon( leading: Icon(

View file

@ -1,8 +1,7 @@
import 'dart:collection';
import 'package:auto_route/auto_route.dart'; import 'package:auto_route/auto_route.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:gap/gap.dart';
import 'package:selfprivacy/config/get_it_config.dart'; import 'package:selfprivacy/config/get_it_config.dart';
import 'package:selfprivacy/logic/models/message.dart'; import 'package:selfprivacy/logic/models/message.dart';
import 'package:selfprivacy/ui/components/list_tiles/log_list_tile.dart'; import 'package:selfprivacy/ui/components/list_tiles/log_list_tile.dart';
@ -16,25 +15,36 @@ class ConsolePage extends StatefulWidget {
} }
class _ConsolePageState extends State<ConsolePage> { class _ConsolePageState extends State<ConsolePage> {
bool paused = false;
@override @override
void initState() { void initState() {
getIt.get<ConsoleModel>().addListener(update);
super.initState(); super.initState();
getIt<ConsoleModel>().addListener(update);
} }
@override @override
void dispose() { void dispose() {
getIt<ConsoleModel>().removeListener(update); getIt<ConsoleModel>().removeListener(update);
super.dispose(); super.dispose();
} }
bool paused = false;
void update() { void update() {
if (!paused) { /// listener update could come at any time, like when widget is already
/// unmounted or during frame build, adding as postframe callback ensures
/// that element is marked for rebuild
WidgetsBinding.instance.addPostFrameCallback((final _) {
if (!paused && mounted) {
setState(() => {}); setState(() => {});
} }
});
}
void togglePause() {
paused ^= true;
setState(() {});
} }
@override @override
@ -51,7 +61,7 @@ class _ConsolePageState extends State<ConsolePage> {
icon: Icon( icon: Icon(
paused ? Icons.play_arrow_outlined : Icons.pause_outlined, paused ? Icons.play_arrow_outlined : Icons.pause_outlined,
), ),
onPressed: () => setState(() => paused = !paused), onPressed: togglePause,
), ),
], ],
), ),
@ -69,12 +79,12 @@ class _ConsolePageState extends State<ConsolePage> {
reverse: true, reverse: true,
shrinkWrap: true, shrinkWrap: true,
children: [ children: [
const SizedBox(height: 20), const Gap(20),
...UnmodifiableListView( ...messages.reversed.map(
messages (final message) => LogListItem(
.map((final message) => LogListItem(message: message)) key: ValueKey(message),
.toList() message: message,
.reversed, ),
), ),
], ],
); );
@ -82,11 +92,10 @@ class _ConsolePageState extends State<ConsolePage> {
return Column( return Column(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [ children: [
Text('console_page.waiting'.tr()), Text('console_page.waiting'.tr()),
const SizedBox( const Gap(16),
height: 16,
),
const CircularProgressIndicator(), const CircularProgressIndicator(),
], ],
); );

View file

@ -1,8 +1,7 @@
import 'package:auto_route/auto_route.dart'; import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:selfprivacy/logic/cubit/app_settings/app_settings_cubit.dart'; import 'package:selfprivacy/logic/cubit/app_settings/app_settings_cubit.dart';
import 'package:selfprivacy/ui/components/buttons/brand_button.dart'; import 'package:selfprivacy/ui/pages/onboarding/views/views.dart';
import 'package:selfprivacy/ui/router/router.dart'; import 'package:selfprivacy/ui/router/router.dart';
@RoutePage() @RoutePage()
@ -17,152 +16,35 @@ class _OnboardingPageState extends State<OnboardingPage> {
PageController pageController = PageController(); PageController pageController = PageController();
@override @override
void initState() { void dispose() {
super.initState(); pageController.dispose();
super.dispose();
} }
@override Future<void> scrollTo(final int targetView) => pageController.animateToPage(
Widget build(final BuildContext context) => Scaffold( targetView,
body: PageView(
controller: pageController,
children: [
_withPadding(firstPage()),
_withPadding(secondPage()),
],
),
);
Widget _withPadding(final Widget child) => Padding(
padding: const EdgeInsets.symmetric(
horizontal: 15,
),
child: child,
);
Widget firstPage() => ConstrainedBox(
constraints: BoxConstraints(
maxHeight: MediaQuery.of(context).size.height,
),
child: Column(
children: [
Expanded(
child: ListView(
children: [
const SizedBox(height: 30),
Text(
'onboarding.page1_title'.tr(),
style: Theme.of(context).textTheme.headlineSmall,
),
const SizedBox(height: 16),
Text(
'onboarding.page1_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const SizedBox(height: 32),
Center(
child: Image.asset(
_fileName(
context: context,
path: 'assets/images/onboarding',
fileExtention: 'png',
fileName: 'onboarding1',
),
),
),
],
),
),
BrandButton.rised(
onPressed: () {
pageController.animateToPage(
1,
duration: const Duration(milliseconds: 300), duration: const Duration(milliseconds: 300),
curve: Curves.easeInOutCubicEmphasized, curve: Curves.easeInOutCubicEmphasized,
); );
},
text: 'basis.next'.tr(),
),
const SizedBox(height: 30),
],
),
);
Widget secondPage() => ConstrainedBox( @override
constraints: BoxConstraints( Widget build(final BuildContext context) => Material(
maxHeight: MediaQuery.of(context).size.height, child: PageView(
), controller: pageController,
child: Column(
children: [ children: [
Expanded( OnboardingFirstView(
child: ListView( onProceed: () => scrollTo(1),
children: [
const SizedBox(height: 30),
Text(
'onboarding.page2_title'.tr(),
style: Theme.of(context).textTheme.headlineSmall,
), ),
const SizedBox(height: 16), OnboardingSecondView(
Text( onProceed: () {
'onboarding.page2_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const SizedBox(height: 16),
Text(
'onboarding.page2_server_provider_title'.tr(),
style: Theme.of(context).textTheme.titleLarge,
),
const SizedBox(height: 16),
Text(
'onboarding.page2_server_provider_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const SizedBox(height: 16),
Text(
'onboarding.page2_dns_provider_title'.tr(),
style: Theme.of(context).textTheme.titleLarge,
),
const SizedBox(height: 16),
Text(
'onboarding.page2_dns_provider_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const SizedBox(height: 16),
Text(
'onboarding.page2_backup_provider_title'.tr(),
style: Theme.of(context).textTheme.titleLarge,
),
const SizedBox(height: 16),
Text(
'onboarding.page2_backup_provider_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const SizedBox(height: 16),
],
),
),
BrandButton.rised(
onPressed: () {
context.read<AppSettingsCubit>().turnOffOnboarding(); context.read<AppSettingsCubit>().turnOffOnboarding();
context.router.replaceAll([ context.router.replaceAll([
const RootRoute(), const RootRoute(),
const InitializingRoute(), const InitializingRoute(),
]); ]);
}, },
text: 'basis.got_it'.tr(),
), ),
const SizedBox(height: 30),
], ],
), ),
); );
} }
String _fileName({
required final BuildContext context,
required final String path,
required final String fileName,
required final String fileExtention,
}) {
final ThemeData theme = Theme.of(context);
final bool isDark = theme.brightness == Brightness.dark;
return '$path/$fileName${isDark ? '-dark' : '-light'}.$fileExtention';
}

View file

@ -0,0 +1,50 @@
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:gap/gap.dart';
import 'package:selfprivacy/ui/pages/onboarding/views/onboarding_view.dart';
class OnboardingFirstView extends StatelessWidget {
const OnboardingFirstView({
required this.onProceed,
super.key,
});
final VoidCallback onProceed;
String assetName({
required final BuildContext context,
required final String path,
required final String fileName,
required final String fileExtension,
}) {
final String suffix =
Theme.of(context).brightness == Brightness.dark ? '-dark' : '-light';
return '$path/$fileName$suffix.$fileExtension';
}
@override
Widget build(final BuildContext context) => OnboardingView(
onProceed: onProceed,
children: [
Text(
'onboarding.page1_title'.tr(),
style: Theme.of(context).textTheme.headlineSmall,
),
const Gap(15),
Text(
'onboarding.page1_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const Gap(30),
Image.asset(
assetName(
context: context,
path: 'assets/images/onboarding',
fileName: 'onboarding1',
fileExtension: 'png',
),
fit: BoxFit.fitWidth,
),
],
);
}

View file

@ -0,0 +1,60 @@
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:gap/gap.dart';
import 'package:selfprivacy/ui/pages/onboarding/views/onboarding_view.dart';
class OnboardingSecondView extends StatelessWidget {
const OnboardingSecondView({
required this.onProceed,
super.key,
});
final VoidCallback onProceed;
@override
Widget build(final BuildContext context) => OnboardingView(
buttonTitle: 'basis.got_it',
onProceed: onProceed,
children: [
Text(
'onboarding.page2_title'.tr(),
style: Theme.of(context).textTheme.headlineSmall,
),
const Gap(16),
Text(
'onboarding.page2_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const Gap(16),
Text(
'onboarding.page2_server_provider_title'.tr(),
style: Theme.of(context).textTheme.titleLarge,
),
const Gap(16),
Text(
'onboarding.page2_server_provider_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const Gap(16),
Text(
'onboarding.page2_dns_provider_title'.tr(),
style: Theme.of(context).textTheme.titleLarge,
),
const Gap(16),
Text(
'onboarding.page2_dns_provider_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
const Gap(16),
Text(
'onboarding.page2_backup_provider_title'.tr(),
style: Theme.of(context).textTheme.titleLarge,
),
const Gap(16),
Text(
'onboarding.page2_backup_provider_text'.tr(),
style: Theme.of(context).textTheme.bodyMedium,
),
],
);
}

View file

@ -0,0 +1,53 @@
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:selfprivacy/ui/components/buttons/buttons.dart';
// base widget for onboarding view
class OnboardingView extends StatelessWidget {
const OnboardingView({
required this.onProceed,
required this.children,
this.buttonTitle = 'basis.next',
super.key,
});
/// Proceed button title
final String buttonTitle;
/// Proceed button callback
final VoidCallback onProceed;
/// Current view content
final List<Widget> children;
@override
Widget build(final BuildContext context) => Scaffold(
body: Align(
child: ConstrainedBox(
constraints: const BoxConstraints(maxWidth: 480),
child: Column(
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
Expanded(
child: ListView(
primary: true,
shrinkWrap: true,
padding: const EdgeInsets.all(15) +
const EdgeInsets.only(top: 15),
children: children,
),
),
Padding(
padding: const EdgeInsets.symmetric(horizontal: 15) +
const EdgeInsets.only(bottom: 30),
child: SPBrandButton.text(
title: buttonTitle.tr(),
onPressed: onProceed,
),
),
],
),
),
),
);
}

View file

@ -0,0 +1,2 @@
export 'onboarding_first_view.dart';
export 'onboarding_second_view.dart';

View file

@ -5,7 +5,6 @@ import 'package:selfprivacy/logic/cubit/app_settings/app_settings_cubit.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/layouts/root_scaffold_with_navigation.dart'; import 'package:selfprivacy/ui/layouts/root_scaffold_with_navigation.dart';
import 'package:selfprivacy/ui/router/root_destinations.dart'; import 'package:selfprivacy/ui/router/root_destinations.dart';
import 'package:selfprivacy/ui/router/router.dart'; import 'package:selfprivacy/ui/router/router.dart';
@RoutePage() @RoutePage()
@ -150,8 +149,3 @@ class MainScreenNavigationDrawer extends StatelessWidget {
); );
} }
} }
class ChangeTab {
ChangeTab(this.onPress);
final ValueChanged<int> onPress;
}

View file

@ -56,5 +56,3 @@ class _TempMessage extends StatelessWidget {
), ),
); );
} }
final DateFormat formatter = DateFormat('HH:mm:ss');

View file

@ -165,83 +165,54 @@ class ServiceStatusCard extends StatelessWidget {
@override @override
Widget build(final BuildContext context) { Widget build(final BuildContext context) {
late IconData icon;
late String buttonTitle;
switch (status) { switch (status) {
case ServiceStatus.active: case ServiceStatus.active:
return FilledCard( icon = Icons.check_circle_outline;
child: ListTile( buttonTitle = 'service_page.status.active';
leading: const Icon( break;
Icons.check_circle_outline,
size: 24,
),
title: Text('service_page.status.active'.tr()),
),
);
case ServiceStatus.inactive: case ServiceStatus.inactive:
return FilledCard( icon = Icons.stop_circle_outlined;
tertiary: true, buttonTitle = 'service_page.status.inactive';
child: ListTile( break;
leading: const Icon(
Icons.stop_circle_outlined,
size: 24,
),
title: Text('service_page.status.inactive'.tr()),
),
);
case ServiceStatus.failed: case ServiceStatus.failed:
return FilledCard( icon = Icons.error_outline;
error: true, buttonTitle = 'service_page.status.failed';
child: ListTile( break;
leading: const Icon(
Icons.error_outline,
size: 24,
),
title: Text('service_page.status.failed'.tr()),
),
);
case ServiceStatus.off: case ServiceStatus.off:
return FilledCard( icon = Icons.power_settings_new;
tertiary: true, buttonTitle = 'service_page.status.off';
child: ListTile( break;
leading: const Icon(
Icons.power_settings_new,
size: 24,
),
title: Text('service_page.status.off'.tr()),
),
);
case ServiceStatus.activating: case ServiceStatus.activating:
return FilledCard( icon = Icons.restart_alt_outlined;
tertiary: true, buttonTitle = 'service_page.status.activating';
child: ListTile( break;
leading: const Icon(
Icons.restart_alt_outlined,
size: 24,
),
title: Text('service_page.status.activating'.tr()),
),
);
case ServiceStatus.deactivating: case ServiceStatus.deactivating:
return FilledCard( icon = Icons.restart_alt_outlined;
tertiary: true, buttonTitle = 'service_page.status.deactivating';
child: ListTile( break;
leading: const Icon(
Icons.restart_alt_outlined,
size: 24,
),
title: Text('service_page.status.deactivating'.tr()),
),
);
case ServiceStatus.reloading: case ServiceStatus.reloading:
icon = Icons.restart_alt_outlined;
buttonTitle = 'service_page.status.reloading';
}
return FilledCard( return FilledCard(
tertiary: true, tertiary: true,
child: ListTile( child: ListTile(
leading: const Icon( leading: Icon(
Icons.restart_alt_outlined, icon,
size: 24, size: 24,
), ),
title: Text('service_page.status.reloading'.tr()), title: Text(buttonTitle.tr()),
), ),
); );
} }
}
} }

View file

@ -89,7 +89,6 @@ class NewUserPage extends StatelessWidget {
PlatformAdapter.setClipboard(currentPassword); PlatformAdapter.setClipboard(currentPassword);
getIt<NavigationService>().showSnackBar( getIt<NavigationService>().showSnackBar(
'basis.copied_to_clipboard'.tr(), 'basis.copied_to_clipboard'.tr(),
behavior: SnackBarBehavior.floating,
); );
}, },
), ),

View file

@ -149,7 +149,6 @@ class _UserLogins extends StatelessWidget {
PlatformAdapter.setClipboard(email); PlatformAdapter.setClipboard(email);
getIt<NavigationService>().showSnackBar( getIt<NavigationService>().showSnackBar(
'basis.copied_to_clipboard'.tr(), 'basis.copied_to_clipboard'.tr(),
behavior: SnackBarBehavior.floating,
); );
}, },
title: email, title: email,

View file

@ -7,13 +7,13 @@ class UiHelpers {
static String getDomainName(final ServerInstallationState config) => static String getDomainName(final ServerInstallationState config) =>
config.isDomainSelected ? config.serverDomain!.domainName : 'example.com'; config.isDomainSelected ? config.serverDomain!.domainName : 'example.com';
static final _formatter = NumberFormat()..minimumFractionDigits = 0;
static String formatWithPrecision( static String formatWithPrecision(
final double value, { final double value, {
final int fraction = 2, final int fraction = 2,
}) { }) {
final NumberFormat formatter = NumberFormat(); _formatter.maximumFractionDigits = fraction;
formatter.minimumFractionDigits = 0; return _formatter.format(value);
formatter.maximumFractionDigits = fraction;
return formatter.format(value);
} }
} }

View file

@ -533,14 +533,22 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "3.2.0" version: "3.2.0"
gap:
dependency: "direct main"
description:
name: gap
sha256: f19387d4e32f849394758b91377f9153a1b41d79513ef7668c088c77dbc6955d
url: "https://pub.dev"
source: hosted
version: "3.0.1"
get_it: get_it:
dependency: "direct main" dependency: "direct main"
description: description:
name: get_it name: get_it
sha256: f79870884de16d689cf9a7d15eedf31ed61d750e813c538a6efb92660fea83c3 sha256: e6017ce7fdeaf218dc51a100344d8cb70134b80e28b760f8bb23c242437bafd7
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "7.6.4" version: "7.6.7"
glob: glob:
dependency: transitive dependency: transitive
description: description:
@ -681,10 +689,10 @@ packages:
dependency: "direct main" dependency: "direct main"
description: description:
name: http name: http
sha256: d4872660c46d929f6b8a9ef4e7a7eff7e49bbf0c4ec3f385ee32df5119175139 sha256: a2bbf9d017fcced29139daa8ed2bba4ece450ab222871df93ca9eec6f80c34ba
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.1.2" version: "1.2.0"
http_multi_server: http_multi_server:
dependency: transitive dependency: transitive
description: description:

View file

@ -1,7 +1,7 @@
name: selfprivacy name: selfprivacy
description: selfprivacy.org description: selfprivacy.org
publish_to: 'none' publish_to: 'none'
version: 0.10.0+20 version: 0.10.1+21
environment: environment:
sdk: '>=3.2.1 <4.0.0' sdk: '>=3.2.1 <4.0.0'
@ -29,6 +29,7 @@ dependencies:
flutter_markdown: ^0.6.18+2 flutter_markdown: ^0.6.18+2
flutter_secure_storage: ^9.0.0 flutter_secure_storage: ^9.0.0
flutter_svg: ^2.0.9 flutter_svg: ^2.0.9
gap: ^3.0.1
get_it: ^7.6.4 get_it: ^7.6.4
gql: ^1.0.0 gql: ^1.0.0
graphql: ^5.1.3 graphql: ^5.1.3