yubioath-flutter/lib/about_page.dart

227 lines
7.9 KiB
Dart
Raw Normal View History

import 'dart:convert';
import 'dart:io';
2021-11-19 17:05:57 +03:00
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
2022-09-07 14:59:44 +03:00
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:logging/logging.dart';
2022-06-05 13:26:40 +03:00
import 'package:url_launcher/url_launcher.dart';
import 'app/logging.dart';
2022-03-25 17:43:32 +03:00
import 'app/message.dart';
2022-08-04 09:23:07 +03:00
import 'app/state.dart';
import 'core/state.dart';
import 'desktop/state.dart';
2022-08-04 09:23:07 +03:00
import 'version.dart';
import 'widgets/responsive_dialog.dart';
import 'widgets/choice_filter_chip.dart';
final _log = Logger('about');
2021-11-19 17:05:57 +03:00
class AboutPage extends ConsumerWidget {
2022-05-12 10:56:55 +03:00
const AboutPage({super.key});
2021-11-19 17:05:57 +03:00
@override
Widget build(BuildContext context, WidgetRef ref) {
2022-03-15 20:04:26 +03:00
return ResponsiveDialog(
2022-06-02 15:53:29 +03:00
title: const Text('About'),
2022-03-15 20:04:26 +03:00
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
2022-06-02 15:53:29 +03:00
Image.asset('assets/graphics/app-icon.png', scale: 1 / 0.75),
Padding(
padding: const EdgeInsets.only(top: 24.0),
2022-06-02 15:53:29 +03:00
child: Text(
2022-08-04 09:23:07 +03:00
Platform.isAndroid
? 'Yubico Authenticator Preview'
: 'Yubico Authenticator',
2022-06-02 15:53:29 +03:00
style: Theme.of(context).textTheme.titleMedium,
),
),
const Text(version),
const Text(''),
2022-06-02 15:53:29 +03:00
const Text('Copyright © 2022 Yubico'),
const Text('All rights reserved'),
const Text(''),
Row(
mainAxisSize: MainAxisSize.min,
2022-06-05 13:26:40 +03:00
children: [
TextButton(
2022-09-07 14:59:44 +03:00
child: Text(
AppLocalizations.of(context)!.general_terms_of_use,
style: const TextStyle(decoration: TextDecoration.underline),
),
onPressed: () {
launchUrl(
Uri.parse(
'https://www.yubico.com/support/terms-conditions/yubico-license-agreement/'),
mode: LaunchMode.externalApplication,
);
},
),
TextButton(
2022-09-07 14:59:44 +03:00
child: Text(
AppLocalizations.of(context)!.general_privacy_policy,
style: const TextStyle(decoration: TextDecoration.underline),
),
onPressed: () {
launchUrl(
Uri.parse(
'https://www.yubico.com/support/terms-conditions/privacy-notice/'),
mode: LaunchMode.externalApplication,
);
},
),
2022-06-02 15:53:29 +03:00
],
),
TextButton(
2022-09-07 14:59:44 +03:00
child: Text(
AppLocalizations.of(context)!.general_open_src_licenses,
style: const TextStyle(decoration: TextDecoration.underline),
),
onPressed: () {
Navigator.of(context).push(MaterialPageRoute<void>(
builder: (BuildContext context) => const LicensePage(
applicationVersion: version,
),
settings: const RouteSettings(name: 'licenses'),
));
},
),
2022-06-02 15:53:29 +03:00
const Padding(
padding: EdgeInsets.only(top: 24.0, bottom: 8.0),
child: Divider(),
),
Padding(
padding: const EdgeInsets.symmetric(vertical: 16.0),
child: Text(
2022-09-07 14:59:44 +03:00
AppLocalizations.of(context)!.general_help_and_feedback,
2022-06-02 15:53:29 +03:00
style: Theme.of(context).textTheme.titleMedium,
),
),
Row(
mainAxisSize: MainAxisSize.min,
2022-06-05 13:26:40 +03:00
children: [
TextButton(
2022-09-07 14:59:44 +03:00
child: Text(
AppLocalizations.of(context)!.general_send_feedback,
style: const TextStyle(decoration: TextDecoration.underline),
),
onPressed: () {
launchUrl(
2022-06-10 15:38:41 +03:00
Uri.parse('https://forms.gle/nYPVWcFnqoprZX1S9'),
mode: LaunchMode.externalApplication,
);
},
),
TextButton(
2022-09-07 14:59:44 +03:00
child: Text(
AppLocalizations.of(context)!.general_i_need_help,
style: const TextStyle(decoration: TextDecoration.underline),
),
onPressed: () {
launchUrl(
Uri.parse('https://support.yubico.com/support/home'),
mode: LaunchMode.externalApplication,
);
},
2022-06-05 13:26:40 +03:00
),
2022-06-02 15:53:29 +03:00
],
),
const Padding(
padding: EdgeInsets.only(top: 24.0, bottom: 8.0),
child: Divider(),
),
Padding(
padding: const EdgeInsets.symmetric(vertical: 16.0),
child: Text(
2022-09-08 12:42:40 +03:00
AppLocalizations.of(context)!.general_troubleshooting,
2022-06-02 15:53:29 +03:00
style: Theme.of(context).textTheme.titleMedium,
),
),
const LoggingPanel(),
if (isDesktop) ...[
2022-06-02 15:53:29 +03:00
const SizedBox(height: 12.0),
ActionChip(
avatar: const Icon(Icons.bug_report_outlined),
2022-09-08 12:42:40 +03:00
label:
Text(AppLocalizations.of(context)!.general_run_diagnostics),
2022-03-15 20:04:26 +03:00
onPressed: () async {
_log.info('Running diagnostics...');
final response =
await ref.read(rpcProvider).command('diagnose', []);
final data = response['diagnostics'] as List;
data.insert(0, {
'app_version': version,
'dart': Platform.version,
});
final text = const JsonEncoder.withIndent(' ').convert(data);
await Clipboard.setData(ClipboardData(text: text));
2022-05-12 09:34:51 +03:00
await ref.read(withContextProvider)(
(context) async {
2022-09-08 12:42:40 +03:00
showMessage(
context,
AppLocalizations.of(context)!
.general_diagnostics_copied);
2022-05-12 09:34:51 +03:00
},
);
2022-03-15 20:04:26 +03:00
},
),
]
2022-03-15 20:04:26 +03:00
],
2021-11-19 17:05:57 +03:00
),
);
}
}
class LoggingPanel extends ConsumerWidget {
2022-05-12 10:56:55 +03:00
const LoggingPanel({super.key});
@override
Widget build(BuildContext context, WidgetRef ref) {
final logLevel = ref.watch(logLevelProvider);
return Wrap(
alignment: WrapAlignment.center,
spacing: 4.0,
runSpacing: 8.0,
children: [
ChoiceFilterChip<Level>(
avatar: Icon(
Icons.insights,
color: Theme.of(context).colorScheme.primary,
2022-06-02 15:53:29 +03:00
),
value: logLevel,
items: Levels.LEVELS,
selected: logLevel != Level.INFO,
labelBuilder: (value) => Text(
2022-09-08 12:42:40 +03:00
'${AppLocalizations.of(context)!.general_log_level}: ${value.name[0]}${value.name.substring(1).toLowerCase()}'),
itemBuilder: (value) =>
Text('${value.name[0]}${value.name.substring(1).toLowerCase()}'),
onChanged: (level) {
ref.read(logLevelProvider.notifier).setLogLevel(level);
2022-05-03 12:24:25 +03:00
_log.debug('Log level set to $level');
2022-09-08 12:42:40 +03:00
showMessage(context,
'${AppLocalizations.of(context)!.general_log_level_set_to} $level');
},
),
ActionChip(
avatar: const Icon(Icons.copy),
2022-09-08 12:42:40 +03:00
label: Text(AppLocalizations.of(context)!.general_copy_log),
onPressed: () async {
_log.info('Copying log to clipboard ($version)...');
2022-05-11 16:47:35 +03:00
final logs = await ref.read(logLevelProvider.notifier).getLogs();
await Clipboard.setData(ClipboardData(text: logs.join('\n')));
2022-05-12 09:34:51 +03:00
await ref.read(withContextProvider)(
(context) async {
2022-09-08 12:42:40 +03:00
showMessage(
context, AppLocalizations.of(context)!.general_log_copied);
2022-05-12 09:34:51 +03:00
},
);
},
),
],
);
}
}