Files
umbrix/lib/features/log/overview/logs_overview_page.dart

233 lines
9.6 KiB
Dart
Raw Normal View History

2024-02-15 15:23:02 +03:30
import 'package:fluentui_system_icons/fluentui_system_icons.dart';
2023-07-06 17:18:41 +03:30
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
2023-07-06 17:18:41 +03:30
import 'package:fpdart/fpdart.dart';
import 'package:gap/gap.dart';
2023-12-01 12:56:24 +03:30
import 'package:hiddify/core/localization/translations.dart';
import 'package:hiddify/core/model/failures.dart';
import 'package:hiddify/core/preferences/general_preferences.dart';
2024-02-15 15:23:02 +03:30
import 'package:hiddify/core/widget/adaptive_icon.dart';
import 'package:hiddify/features/common/nested_app_bar.dart';
2023-11-28 18:24:31 +03:30
import 'package:hiddify/features/log/data/log_data_providers.dart';
import 'package:hiddify/features/log/model/log_level.dart';
import 'package:hiddify/features/log/overview/logs_overview_notifier.dart';
2023-07-06 17:18:41 +03:30
import 'package:hiddify/utils/utils.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:sliver_tools/sliver_tools.dart';
2023-09-07 01:56:59 +03:30
2023-11-28 18:24:31 +03:30
class LogsOverviewPage extends HookConsumerWidget with PresLogger {
const LogsOverviewPage({super.key});
2023-07-06 17:18:41 +03:30
@override
Widget build(BuildContext context, WidgetRef ref) {
final t = ref.watch(translationsProvider);
2023-11-28 18:24:31 +03:30
final state = ref.watch(logsOverviewNotifierProvider);
final notifier = ref.watch(logsOverviewNotifierProvider.notifier);
2023-07-06 17:18:41 +03:30
2023-09-07 13:43:46 +03:30
final debug = ref.watch(debugModeNotifierProvider);
2023-11-28 18:24:31 +03:30
final pathResolver = ref.watch(logPathResolverProvider);
2023-08-24 16:18:05 +03:30
final filterController = useTextEditingController(text: state.filter);
2023-08-24 16:18:05 +03:30
final List<PopupMenuEntry> popupButtons = debug || PlatformUtils.isDesktop
? [
PopupMenuItem(
2023-09-07 01:56:59 +03:30
child: Text(t.logs.shareCoreLogs),
2023-08-24 16:18:05 +03:30
onTap: () async {
2023-08-25 17:58:04 +03:30
await UriUtils.tryShareOrLaunchFile(
2023-11-28 18:24:31 +03:30
Uri.parse(pathResolver.coreFile().path),
fileOrDir: pathResolver.directory.uri,
2023-08-24 22:19:22 +03:30
);
2023-08-24 16:18:05 +03:30
},
),
PopupMenuItem(
2023-09-07 01:56:59 +03:30
child: Text(t.logs.shareAppLogs),
2023-08-24 16:18:05 +03:30
onTap: () async {
2023-08-25 17:58:04 +03:30
await UriUtils.tryShareOrLaunchFile(
2023-11-28 18:24:31 +03:30
Uri.parse(pathResolver.appFile().path),
fileOrDir: pathResolver.directory.uri,
2023-08-24 22:19:22 +03:30
);
2023-08-24 16:18:05 +03:30
},
),
]
: [];
return Scaffold(
body: NestedScrollView(
headerSliverBuilder: (context, innerBoxIsScrolled) {
return <Widget>[
SliverOverlapAbsorber(
handle: NestedScrollView.sliverOverlapAbsorberHandleFor(context),
sliver: MultiSliver(
children: [
NestedAppBar(
forceElevated: innerBoxIsScrolled,
title: Text(t.logs.pageTitle),
actions: [
if (state.paused)
IconButton(
onPressed: notifier.resume,
2024-02-15 15:23:02 +03:30
icon: const Icon(FluentIcons.play_20_regular),
tooltip: t.logs.resumeTooltip,
2024-02-15 15:23:02 +03:30
iconSize: 20,
)
else
IconButton(
onPressed: notifier.pause,
2024-02-15 15:23:02 +03:30
icon: const Icon(FluentIcons.pause_20_regular),
tooltip: t.logs.pauseTooltip,
2024-02-15 15:23:02 +03:30
iconSize: 20,
),
IconButton(
onPressed: notifier.clear,
2024-02-15 15:23:02 +03:30
icon: const Icon(FluentIcons.delete_lines_20_regular),
tooltip: t.logs.clearTooltip,
2024-02-15 15:23:02 +03:30
iconSize: 20,
),
if (popupButtons.isNotEmpty)
PopupMenuButton(
2024-02-15 15:23:02 +03:30
icon: Icon(AdaptiveIcon(context).more),
itemBuilder: (context) {
return popupButtons;
},
),
],
),
SliverPinnedHeader(
child: DecoratedBox(
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.background,
),
child: Padding(
padding: const EdgeInsets.symmetric(
horizontal: 16,
vertical: 8,
),
child: Row(
children: [
Flexible(
child: TextFormField(
controller: filterController,
onChanged: notifier.filterMessage,
decoration: InputDecoration(
isDense: true,
hintText: t.logs.filterHint,
),
),
),
const Gap(16),
DropdownButton<Option<LogLevel>>(
value: optionOf(state.levelFilter),
onChanged: (v) {
if (v == null) return;
notifier.filterLevel(v.toNullable());
},
padding:
const EdgeInsets.symmetric(horizontal: 8),
borderRadius: BorderRadius.circular(4),
items: [
DropdownMenuItem(
value: none(),
child: Text(t.logs.allLevelsFilter),
),
...LogLevel.choices.map(
(e) => DropdownMenuItem(
value: some(e),
child: Text(e.name),
),
),
],
),
],
),
),
2023-07-06 17:18:41 +03:30
),
),
],
),
2023-07-06 17:18:41 +03:30
),
];
},
body: Builder(
builder: (context) {
return CustomScrollView(
2023-11-10 12:04:13 +03:30
primary: false,
reverse: true,
slivers: <Widget>[
switch (state.logs) {
AsyncData(value: final logs) => SliverList.builder(
itemCount: logs.length,
itemBuilder: (context, index) {
final log = logs[index];
return Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Padding(
padding: const EdgeInsets.symmetric(
horizontal: 16,
vertical: 4,
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (log.level != null)
Row(
mainAxisAlignment:
MainAxisAlignment.spaceBetween,
children: [
Text(
log.level!.name.toUpperCase(),
style: Theme.of(context)
.textTheme
.labelMedium
?.copyWith(
2023-11-09 15:23:48 +03:30
color: log.level!.color,
),
),
if (log.time != null)
Text(
log.time!.toString(),
style: Theme.of(context)
.textTheme
.labelSmall,
),
],
),
Text(
log.message,
style:
Theme.of(context).textTheme.bodySmall,
),
],
),
),
if (index != 0)
const Divider(
indent: 16,
endIndent: 16,
height: 4,
),
],
);
},
2023-07-06 17:18:41 +03:30
),
AsyncError(:final error) => SliverErrorBodyPlaceholder(
t.presentShortError(error),
),
_ => const SliverLoadingBodyPlaceholder(),
},
SliverOverlapInjector(
handle: NestedScrollView.sliverOverlapAbsorberHandleFor(
context,
),
),
],
);
},
),
),
);
2023-07-06 17:18:41 +03:30
}
}