Change router for different screen size

This commit is contained in:
problematicconsumer
2023-11-01 20:36:16 +03:30
parent d5dc2da60c
commit a21004761d
23 changed files with 446 additions and 366 deletions

View File

@@ -1,7 +1,8 @@
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:go_router/go_router.dart'; import 'package:go_router/go_router.dart';
import 'package:hiddify/core/prefs/prefs.dart'; import 'package:hiddify/core/prefs/prefs.dart';
import 'package:hiddify/core/router/routes/routes.dart'; import 'package:hiddify/core/router/routes.dart';
import 'package:hiddify/services/deep_link_service.dart'; import 'package:hiddify/services/deep_link_service.dart';
import 'package:hiddify/utils/utils.dart'; import 'package:hiddify/utils/utils.dart';
import 'package:riverpod_annotation/riverpod_annotation.dart'; import 'package:riverpod_annotation/riverpod_annotation.dart';
@@ -9,6 +10,12 @@ import 'package:sentry_flutter/sentry_flutter.dart';
part 'app_router.g.dart'; part 'app_router.g.dart';
bool _debugMobileRouter = false;
final useMobileRouter =
!PlatformUtils.isDesktop || (kDebugMode && _debugMobileRouter);
final GlobalKey<NavigatorState> rootNavigatorKey = GlobalKey<NavigatorState>();
// TODO: test and improve handling of deep link // TODO: test and improve handling of deep link
@riverpod @riverpod
GoRouter router(RouterRef ref) { GoRouter router(RouterRef ref) {
@@ -31,7 +38,7 @@ GoRouter router(RouterRef ref) {
navigatorKey: rootNavigatorKey, navigatorKey: rootNavigatorKey,
initialLocation: initialLocation, initialLocation: initialLocation,
debugLogDiagnostics: true, debugLogDiagnostics: true,
routes: $routes, routes: useMobileRouter ? mobileRoutes : desktopRoutes,
refreshListenable: notifier, refreshListenable: notifier,
redirect: notifier.redirect, redirect: notifier.redirect,
observers: [ observers: [

View File

@@ -1,2 +1,2 @@
export 'app_router.dart'; export 'app_router.dart';
export 'routes/routes.dart'; export 'routes.dart';

View File

@@ -0,0 +1,16 @@
import 'package:hiddify/core/router/routes/desktop_routes.dart' as desktop;
import 'package:hiddify/core/router/routes/mobile_routes.dart' as mobile;
import 'package:hiddify/core/router/routes/shared_routes.dart' as shared;
export 'routes/mobile_routes.dart';
export 'routes/shared_routes.dart' hide $appRoutes;
final mobileRoutes = [
...shared.$appRoutes,
...mobile.$appRoutes,
];
final desktopRoutes = [
...shared.$appRoutes,
...desktop.$appRoutes,
];

View File

@@ -2,9 +2,9 @@ import 'package:flutter/material.dart';
import 'package:go_router/go_router.dart'; import 'package:go_router/go_router.dart';
import 'package:hiddify/core/router/routes/shared_routes.dart'; import 'package:hiddify/core/router/routes/shared_routes.dart';
import 'package:hiddify/features/about/view/view.dart'; import 'package:hiddify/features/about/view/view.dart';
import 'package:hiddify/features/common/adaptive_root_scaffold.dart';
import 'package:hiddify/features/logs/view/view.dart'; import 'package:hiddify/features/logs/view/view.dart';
import 'package:hiddify/features/settings/view/view.dart'; import 'package:hiddify/features/settings/view/view.dart';
import 'package:hiddify/features/wrapper/wrapper.dart';
part 'desktop_routes.g.dart'; part 'desktop_routes.g.dart';
@@ -61,7 +61,7 @@ class DesktopWrapperRoute extends ShellRouteData {
@override @override
Widget builder(BuildContext context, GoRouterState state, Widget navigator) { Widget builder(BuildContext context, GoRouterState state, Widget navigator) {
return DesktopWrapper(navigator); return AdaptiveRootScaffold(navigator);
} }
} }

View File

@@ -1,10 +1,11 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:go_router/go_router.dart'; import 'package:go_router/go_router.dart';
import 'package:hiddify/core/router/app_router.dart';
import 'package:hiddify/core/router/routes/shared_routes.dart'; import 'package:hiddify/core/router/routes/shared_routes.dart';
import 'package:hiddify/features/about/view/view.dart'; import 'package:hiddify/features/about/view/view.dart';
import 'package:hiddify/features/common/adaptive_root_scaffold.dart';
import 'package:hiddify/features/logs/view/view.dart'; import 'package:hiddify/features/logs/view/view.dart';
import 'package:hiddify/features/settings/view/view.dart'; import 'package:hiddify/features/settings/view/view.dart';
import 'package:hiddify/features/wrapper/wrapper.dart';
part 'mobile_routes.g.dart'; part 'mobile_routes.g.dart';
@@ -65,7 +66,7 @@ class MobileWrapperRoute extends ShellRouteData {
@override @override
Widget builder(BuildContext context, GoRouterState state, Widget navigator) { Widget builder(BuildContext context, GoRouterState state, Widget navigator) {
return MobileWrapper(navigator); return AdaptiveRootScaffold(navigator);
} }
} }

View File

@@ -1,16 +0,0 @@
import 'package:go_router/go_router.dart';
import 'package:hiddify/core/router/routes/desktop_routes.dart' as desktop;
import 'package:hiddify/core/router/routes/mobile_routes.dart' as mobile;
import 'package:hiddify/core/router/routes/shared_routes.dart' as shared;
import 'package:hiddify/utils/utils.dart';
export 'mobile_routes.dart';
export 'shared_routes.dart' hide $appRoutes;
List<RouteBase> get $routes => [
...shared.$appRoutes,
if (PlatformUtils.isDesktop)
...desktop.$appRoutes
else
...mobile.$appRoutes,
];

View File

@@ -1,5 +1,6 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:go_router/go_router.dart'; import 'package:go_router/go_router.dart';
import 'package:hiddify/core/router/app_router.dart';
import 'package:hiddify/features/home/view/view.dart'; import 'package:hiddify/features/home/view/view.dart';
import 'package:hiddify/features/intro/view/view.dart'; import 'package:hiddify/features/intro/view/view.dart';
import 'package:hiddify/features/profile_detail/view/view.dart'; import 'package:hiddify/features/profile_detail/view/view.dart';
@@ -9,8 +10,6 @@ import 'package:hiddify/utils/utils.dart';
part 'shared_routes.g.dart'; part 'shared_routes.g.dart';
final GlobalKey<NavigatorState> rootNavigatorKey = GlobalKey<NavigatorState>();
class HomeRoute extends GoRouteData { class HomeRoute extends GoRouteData {
const HomeRoute(); const HomeRoute();
static const path = '/'; static const path = '/';

View File

@@ -4,6 +4,7 @@ import 'package:gap/gap.dart';
import 'package:hiddify/core/core_providers.dart'; import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/domain/constants.dart'; import 'package:hiddify/domain/constants.dart';
import 'package:hiddify/domain/failures.dart'; import 'package:hiddify/domain/failures.dart';
import 'package:hiddify/features/common/nested_app_bar.dart';
import 'package:hiddify/features/common/common.dart'; import 'package:hiddify/features/common/common.dart';
import 'package:hiddify/features/common/new_version_dialog.dart'; import 'package:hiddify/features/common/new_version_dialog.dart';
import 'package:hiddify/gen/assets.gen.dart'; import 'package:hiddify/gen/assets.gen.dart';
@@ -71,7 +72,7 @@ class AboutPage extends HookConsumerWidget {
return Scaffold( return Scaffold(
body: CustomScrollView( body: CustomScrollView(
slivers: [ slivers: [
SliverAppBar( NestedAppBar(
title: Text(t.about.pageTitle), title: Text(t.about.pageTitle),
actions: [ actions: [
PopupMenuButton( PopupMenuButton(

View File

@@ -0,0 +1,171 @@
import 'package:flutter/material.dart';
import 'package:flutter_adaptive_scaffold/flutter_adaptive_scaffold.dart';
import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/core/router/router.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
abstract interface class RootScaffold {
static final stateKey = GlobalKey<ScaffoldState>();
static bool canShowDrawer(BuildContext context) =>
Breakpoints.small.isActive(context);
}
class AdaptiveRootScaffold extends HookConsumerWidget {
const AdaptiveRootScaffold(this.navigator, {super.key});
final Widget navigator;
@override
Widget build(BuildContext context, WidgetRef ref) {
final t = ref.watch(translationsProvider);
final selectedIndex = getCurrentIndex(context);
final destinations = [
NavigationDestination(
icon: const Icon(Icons.power_settings_new),
label: t.home.pageTitle,
),
NavigationDestination(
icon: const Icon(Icons.filter_list),
label: t.proxies.pageTitle,
),
NavigationDestination(
icon: const Icon(Icons.article),
label: t.logs.pageTitle,
),
NavigationDestination(
icon: const Icon(Icons.settings),
label: t.settings.pageTitle,
),
NavigationDestination(
icon: const Icon(Icons.info),
label: t.about.pageTitle,
),
];
return _CustomAdaptiveScaffold(
selectedIndex: selectedIndex,
onSelectedIndexChange: (index) {
RootScaffold.stateKey.currentState?.closeDrawer();
switchTab(index, context);
},
destinations: destinations,
drawerDestinationRange: useMobileRouter ? (2, null) : (0, null),
bottomDestinationRange: (0, 2),
useBottomSheet: useMobileRouter,
body: navigator,
);
}
}
class _CustomAdaptiveScaffold extends HookConsumerWidget {
const _CustomAdaptiveScaffold({
required this.selectedIndex,
required this.onSelectedIndexChange,
required this.destinations,
required this.drawerDestinationRange,
required this.bottomDestinationRange,
this.useBottomSheet = false,
required this.body,
});
final int selectedIndex;
final Function(int) onSelectedIndexChange;
final List<NavigationDestination> destinations;
final (int, int?) drawerDestinationRange;
final (int, int?) bottomDestinationRange;
final bool useBottomSheet;
final Widget body;
List<NavigationDestination> destinationsSlice((int, int?) range) =>
destinations.sublist(range.$1, range.$2);
int? selectedWithOffset((int, int?) range) {
final index = selectedIndex - range.$1;
return index < 0 || (range.$2 != null && index > (range.$2! - 1))
? null
: index;
}
void selectWithOffset(int index, (int, int?) range) =>
onSelectedIndexChange(index + range.$1);
@override
Widget build(BuildContext context, WidgetRef ref) {
return Scaffold(
key: RootScaffold.stateKey,
drawer: Breakpoints.small.isActive(context)
? SafeArea(
child: Drawer(
width: (MediaQuery.sizeOf(context).width * 0.88).clamp(0, 304),
child: NavigationRail(
extended: true,
selectedIndex: selectedWithOffset(drawerDestinationRange),
destinations: destinationsSlice(drawerDestinationRange)
.map((_) => AdaptiveScaffold.toRailDestination(_))
.toList(),
onDestinationSelected: (index) =>
selectWithOffset(index, drawerDestinationRange),
),
),
)
: null,
body: AdaptiveLayout(
primaryNavigation: SlotLayout(
config: <Breakpoint, SlotLayoutConfig>{
Breakpoints.medium: SlotLayout.from(
key: const Key('primaryNavigation'),
builder: (_) => AdaptiveScaffold.standardNavigationRail(
selectedIndex: selectedIndex,
destinations: destinations
.map((_) => AdaptiveScaffold.toRailDestination(_))
.toList(),
onDestinationSelected: onSelectedIndexChange,
),
),
Breakpoints.large: SlotLayout.from(
key: const Key('primaryNavigation1'),
builder: (_) => AdaptiveScaffold.standardNavigationRail(
extended: true,
selectedIndex: selectedIndex,
destinations: destinations
.map((_) => AdaptiveScaffold.toRailDestination(_))
.toList(),
onDestinationSelected: onSelectedIndexChange,
),
),
},
),
bottomNavigation: useBottomSheet ||
Breakpoints.smallMobile.isActive(context)
? SlotLayout(
config: <Breakpoint, SlotLayoutConfig>{
Breakpoints.small: SlotLayout.from(
key: const Key('bottomNavigation'),
builder: (_) =>
AdaptiveScaffold.standardBottomNavigationBar(
currentIndex: selectedWithOffset(bottomDestinationRange),
destinations: destinationsSlice(bottomDestinationRange),
onDestinationSelected: (index) =>
selectWithOffset(index, bottomDestinationRange),
),
),
},
)
: null,
body: SlotLayout(
config: <Breakpoint, SlotLayoutConfig?>{
Breakpoints.standard: SlotLayout.from(
key: const Key('body'),
inAnimation: AdaptiveScaffold.fadeIn,
outAnimation: AdaptiveScaffold.fadeOut,
builder: (context) => body,
),
},
),
),
);
}
}

View File

@@ -1,7 +1,7 @@
import 'package:freezed_annotation/freezed_annotation.dart'; import 'package:freezed_annotation/freezed_annotation.dart';
import 'package:hiddify/core/core_providers.dart'; import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/core/prefs/prefs.dart'; import 'package:hiddify/core/prefs/prefs.dart';
import 'package:hiddify/core/router/routes/routes.dart'; import 'package:hiddify/core/router/router.dart';
import 'package:hiddify/data/data_providers.dart'; import 'package:hiddify/data/data_providers.dart';
import 'package:hiddify/domain/app/app.dart'; import 'package:hiddify/domain/app/app.dart';
import 'package:hiddify/features/common/new_version_dialog.dart'; import 'package:hiddify/features/common/new_version_dialog.dart';

View File

@@ -1,6 +1,5 @@
export 'app_update_notifier.dart'; export 'app_update_notifier.dart';
export 'confirmation_dialogs.dart'; export 'confirmation_dialogs.dart';
export 'custom_app_bar.dart';
export 'general_pref_tiles.dart'; export 'general_pref_tiles.dart';
export 'profile_tile.dart'; export 'profile_tile.dart';
export 'qr_code_scanner_screen.dart'; export 'qr_code_scanner_screen.dart';

View File

@@ -1,24 +0,0 @@
import 'package:flutter/material.dart';
abstract class RootScaffold {
static final stateKey = GlobalKey<ScaffoldState>();
}
class NestedTabAppBar extends SliverAppBar {
NestedTabAppBar({
super.key,
super.title,
super.actions,
super.pinned = true,
super.forceElevated,
super.bottom,
}) : super(
leading: RootScaffold.stateKey.currentState?.hasDrawer ?? false
? DrawerButton(
onPressed: () {
RootScaffold.stateKey.currentState?.openDrawer();
},
)
: null,
);
}

View File

@@ -0,0 +1,50 @@
import 'package:flutter/material.dart';
import 'package:go_router/go_router.dart';
import 'package:hiddify/core/router/router.dart';
import 'package:hiddify/features/common/adaptive_root_scaffold.dart';
bool showDrawerButton(BuildContext context) {
if (!useMobileRouter) return true;
final String location = GoRouterState.of(context).uri.path;
if (location == const HomeRoute().location) return true;
if (location.startsWith(const ProxiesRoute().location)) return true;
return false;
}
class NestedAppBar extends StatelessWidget {
const NestedAppBar({
super.key,
this.title,
this.actions,
this.pinned = true,
this.forceElevated = false,
this.bottom,
});
final Widget? title;
final List<Widget>? actions;
final bool pinned;
final bool forceElevated;
final PreferredSizeWidget? bottom;
@override
Widget build(BuildContext context) {
RootScaffold.canShowDrawer(context);
return SliverAppBar(
leading: (RootScaffold.stateKey.currentState?.hasDrawer ?? false) &&
showDrawerButton(context)
? DrawerButton(
onPressed: () {
RootScaffold.stateKey.currentState?.openDrawer();
},
)
: null,
title: title,
actions: actions,
pinned: pinned,
forceElevated: forceElevated,
bottom: bottom,
);
}
}

View File

@@ -4,7 +4,7 @@ import 'package:gap/gap.dart';
import 'package:go_router/go_router.dart'; import 'package:go_router/go_router.dart';
import 'package:hiddify/core/core_providers.dart'; import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/core/prefs/prefs.dart'; import 'package:hiddify/core/prefs/prefs.dart';
import 'package:hiddify/core/router/routes/routes.dart'; import 'package:hiddify/core/router/router.dart';
import 'package:hiddify/domain/failures.dart'; import 'package:hiddify/domain/failures.dart';
import 'package:hiddify/domain/profiles/profiles.dart'; import 'package:hiddify/domain/profiles/profiles.dart';
import 'package:hiddify/features/common/confirmation_dialogs.dart'; import 'package:hiddify/features/common/confirmation_dialogs.dart';

View File

@@ -14,9 +14,10 @@ class WindowController extends _$WindowController
Future<bool> build() async { Future<bool> build() async {
await windowManager.ensureInitialized(); await windowManager.ensureInitialized();
const size = Size(868, 668); const size = Size(868, 668);
const minumumSize = Size(368, 568);
const windowOptions = WindowOptions( const windowOptions = WindowOptions(
size: size, size: size,
minimumSize: size, minimumSize: minumumSize,
center: true, center: true,
); );
await windowManager.setPreventClose(true); await windowManager.setPreventClose(true);

View File

@@ -6,7 +6,8 @@ import 'package:hiddify/core/router/router.dart';
import 'package:hiddify/domain/failures.dart'; import 'package:hiddify/domain/failures.dart';
import 'package:hiddify/features/common/active_profile/active_profile_notifier.dart'; import 'package:hiddify/features/common/active_profile/active_profile_notifier.dart';
import 'package:hiddify/features/common/active_profile/has_any_profile_notifier.dart'; import 'package:hiddify/features/common/active_profile/has_any_profile_notifier.dart';
import 'package:hiddify/features/common/common.dart'; import 'package:hiddify/features/common/nested_app_bar.dart';
import 'package:hiddify/features/common/profile_tile.dart';
import 'package:hiddify/features/home/widgets/widgets.dart'; import 'package:hiddify/features/home/widgets/widgets.dart';
import 'package:hiddify/utils/utils.dart'; import 'package:hiddify/utils/utils.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
@@ -28,7 +29,7 @@ class HomePage extends HookConsumerWidget {
children: [ children: [
CustomScrollView( CustomScrollView(
slivers: [ slivers: [
NestedTabAppBar( NestedAppBar(
title: Row( title: Row(
children: [ children: [
Text(t.general.appTitle), Text(t.general.appTitle),

View File

@@ -6,10 +6,12 @@ import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/core/prefs/prefs.dart'; import 'package:hiddify/core/prefs/prefs.dart';
import 'package:hiddify/domain/failures.dart'; import 'package:hiddify/domain/failures.dart';
import 'package:hiddify/domain/singbox/singbox.dart'; import 'package:hiddify/domain/singbox/singbox.dart';
import 'package:hiddify/features/common/nested_app_bar.dart';
import 'package:hiddify/features/logs/notifier/notifier.dart'; import 'package:hiddify/features/logs/notifier/notifier.dart';
import 'package:hiddify/services/service_providers.dart'; import 'package:hiddify/services/service_providers.dart';
import 'package:hiddify/utils/utils.dart'; import 'package:hiddify/utils/utils.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:sliver_tools/sliver_tools.dart';
class LogsPage extends HookConsumerWidget with PresLogger { class LogsPage extends HookConsumerWidget with PresLogger {
const LogsPage({super.key}); const LogsPage({super.key});
@@ -49,9 +51,15 @@ class LogsPage extends HookConsumerWidget with PresLogger {
: []; : [];
return Scaffold( return Scaffold(
appBar: AppBar( body: NestedScrollView(
// TODO: fix height headerSliverBuilder: (context, innerBoxIsScrolled) {
toolbarHeight: 90, return <Widget>[
SliverOverlapAbsorber(
handle: NestedScrollView.sliverOverlapAbsorberHandleFor(context),
sliver: MultiSliver(
children: [
NestedAppBar(
forceElevated: innerBoxIsScrolled,
title: Text(t.logs.pageTitle), title: Text(t.logs.pageTitle),
actions: [ actions: [
if (state.paused) if (state.paused)
@@ -78,10 +86,17 @@ class LogsPage extends HookConsumerWidget with PresLogger {
}, },
), ),
], ],
bottom: PreferredSize( ),
preferredSize: const Size.fromHeight(36), SliverPinnedHeader(
child: DecoratedBox(
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.background,
),
child: Padding( child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), padding: const EdgeInsets.symmetric(
horizontal: 16,
vertical: 8,
),
child: Row( child: Row(
children: [ children: [
Flexible( Flexible(
@@ -101,7 +116,8 @@ class LogsPage extends HookConsumerWidget with PresLogger {
if (v == null) return; if (v == null) return;
notifier.filterLevel(v.toNullable()); notifier.filterLevel(v.toNullable());
}, },
padding: const EdgeInsets.symmetric(horizontal: 8), padding:
const EdgeInsets.symmetric(horizontal: 8),
borderRadius: BorderRadius.circular(4), borderRadius: BorderRadius.circular(4),
items: [ items: [
DropdownMenuItem( DropdownMenuItem(
@@ -121,11 +137,19 @@ class LogsPage extends HookConsumerWidget with PresLogger {
), ),
), ),
), ),
body: switch (state.logs) { ],
AsyncData(value: final logs) => SelectionArea( ),
child: ListView.builder( ),
itemCount: logs.length, ];
},
body: Builder(
builder: (context) {
return CustomScrollView(
reverse: true, reverse: true,
slivers: <Widget>[
switch (state.logs) {
AsyncData(value: final logs) => SliverList.builder(
itemCount: logs.length,
itemBuilder: (context, index) { itemBuilder: (context, index) {
final log = logs[index]; final log = logs[index];
return Column( return Column(
@@ -142,26 +166,30 @@ class LogsPage extends HookConsumerWidget with PresLogger {
children: [ children: [
if (log.level != null) if (log.level != null)
Row( Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment:
MainAxisAlignment.spaceBetween,
children: [ children: [
Text( Text(
log.level!.name.toUpperCase(), log.level!.name.toUpperCase(),
style: Theme.of(context) style: Theme.of(context)
.textTheme .textTheme
.labelMedium .labelMedium
?.copyWith(color: log.level!.color), ?.copyWith(
color: log.level!.color),
), ),
if (log.time != null) if (log.time != null)
Text( Text(
log.time!.toString(), log.time!.toString(),
style: style: Theme.of(context)
Theme.of(context).textTheme.labelSmall, .textTheme
.labelSmall,
), ),
], ],
), ),
Text( Text(
log.message, log.message,
style: Theme.of(context).textTheme.bodySmall, style:
Theme.of(context).textTheme.bodySmall,
), ),
], ],
), ),
@@ -176,18 +204,21 @@ class LogsPage extends HookConsumerWidget with PresLogger {
); );
}, },
), ),
AsyncError(:final error) => SliverErrorBodyPlaceholder(
t.presentShortError(error),
), ),
AsyncError(:final error) => CustomScrollView( _ => const SliverLoadingBodyPlaceholder(),
slivers: [
SliverErrorBodyPlaceholder(t.presentShortError(error)),
],
),
_ => const CustomScrollView(
slivers: [
SliverLoadingBodyPlaceholder(),
],
),
}, },
SliverOverlapInjector(
handle: NestedScrollView.sliverOverlapAbsorberHandleFor(
context,
),
),
],
);
},
),
),
); );
} }
} }

View File

@@ -1,7 +1,7 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:hiddify/core/core_providers.dart'; import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/domain/failures.dart'; import 'package:hiddify/domain/failures.dart';
import 'package:hiddify/features/common/common.dart'; import 'package:hiddify/features/common/nested_app_bar.dart';
import 'package:hiddify/features/proxies/notifier/notifier.dart'; import 'package:hiddify/features/proxies/notifier/notifier.dart';
import 'package:hiddify/features/proxies/widgets/widgets.dart'; import 'package:hiddify/features/proxies/widgets/widgets.dart';
import 'package:hiddify/utils/utils.dart'; import 'package:hiddify/utils/utils.dart';
@@ -29,7 +29,7 @@ class ProxiesPage extends HookConsumerWidget with PresLogger {
return Scaffold( return Scaffold(
body: CustomScrollView( body: CustomScrollView(
slivers: [ slivers: [
NestedTabAppBar( NestedAppBar(
title: Text(t.proxies.pageTitle), title: Text(t.proxies.pageTitle),
), ),
SliverFillRemaining( SliverFillRemaining(
@@ -50,7 +50,7 @@ class ProxiesPage extends HookConsumerWidget with PresLogger {
return Scaffold( return Scaffold(
body: CustomScrollView( body: CustomScrollView(
slivers: [ slivers: [
NestedTabAppBar( NestedAppBar(
title: Text(t.proxies.pageTitle), title: Text(t.proxies.pageTitle),
actions: [ actions: [
PopupMenuButton<ProxiesSort>( PopupMenuButton<ProxiesSort>(
@@ -140,7 +140,7 @@ class ProxiesPage extends HookConsumerWidget with PresLogger {
return Scaffold( return Scaffold(
body: CustomScrollView( body: CustomScrollView(
slivers: [ slivers: [
NestedTabAppBar( NestedAppBar(
title: Text(t.proxies.pageTitle), title: Text(t.proxies.pageTitle),
), ),
SliverErrorBodyPlaceholder( SliverErrorBodyPlaceholder(
@@ -155,7 +155,7 @@ class ProxiesPage extends HookConsumerWidget with PresLogger {
return Scaffold( return Scaffold(
body: CustomScrollView( body: CustomScrollView(
slivers: [ slivers: [
NestedTabAppBar( NestedAppBar(
title: Text(t.proxies.pageTitle), title: Text(t.proxies.pageTitle),
), ),
const SliverLoadingBodyPlaceholder(), const SliverLoadingBodyPlaceholder(),

View File

@@ -1,6 +1,7 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:gap/gap.dart'; import 'package:gap/gap.dart';
import 'package:hiddify/core/core_providers.dart'; import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/features/common/nested_app_bar.dart';
import 'package:hiddify/features/settings/widgets/widgets.dart'; import 'package:hiddify/features/settings/widgets/widgets.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
@@ -12,10 +13,12 @@ class SettingsPage extends HookConsumerWidget {
final t = ref.watch(translationsProvider); final t = ref.watch(translationsProvider);
return Scaffold( return Scaffold(
appBar: AppBar( body: CustomScrollView(
slivers: [
NestedAppBar(
title: Text(t.settings.pageTitle), title: Text(t.settings.pageTitle),
), ),
body: ListView( SliverList.list(
children: [ children: [
SettingsSection(t.settings.general.sectionTitle), SettingsSection(t.settings.general.sectionTitle),
const GeneralSettingTiles(), const GeneralSettingTiles(),
@@ -26,6 +29,8 @@ class SettingsPage extends HookConsumerWidget {
const Gap(16), const Gap(16),
], ],
), ),
],
),
); );
} }
} }

View File

@@ -4,7 +4,7 @@ import 'package:flutter/material.dart';
import 'package:go_router/go_router.dart'; import 'package:go_router/go_router.dart';
import 'package:hiddify/core/core_providers.dart'; import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/core/prefs/prefs.dart'; import 'package:hiddify/core/prefs/prefs.dart';
import 'package:hiddify/core/router/routes/routes.dart'; import 'package:hiddify/core/router/router.dart';
import 'package:hiddify/domain/singbox/singbox.dart'; import 'package:hiddify/domain/singbox/singbox.dart';
import 'package:hiddify/features/common/common.dart'; import 'package:hiddify/features/common/common.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';

View File

@@ -1,65 +0,0 @@
import 'package:flutter/material.dart';
import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/core/router/router.dart';
import 'package:hiddify/features/common/stats/stats_overview.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
class DesktopWrapper extends HookConsumerWidget {
const DesktopWrapper(this.navigator, {super.key});
final Widget navigator;
@override
Widget build(BuildContext context, WidgetRef ref) {
final t = ref.watch(translationsProvider);
final currentIndex = getCurrentIndex(context);
final destinations = [
NavigationRailDestination(
icon: const Icon(Icons.power_settings_new),
label: Text(t.home.pageTitle),
),
NavigationRailDestination(
icon: const Icon(Icons.filter_list),
label: Text(t.proxies.pageTitle),
),
NavigationRailDestination(
icon: const Icon(Icons.article),
label: Text(t.logs.pageTitle),
),
NavigationRailDestination(
icon: const Icon(Icons.settings),
label: Text(t.settings.pageTitle),
),
NavigationRailDestination(
icon: const Icon(Icons.info),
label: Text(t.about.pageTitle),
),
];
return Scaffold(
body: Row(
children: [
SizedBox(
width: 192,
child: NavigationRail(
extended: true,
minExtendedWidth: 192,
destinations: destinations,
selectedIndex: currentIndex,
onDestinationSelected: (index) => switchTab(index, context),
trailing: const Expanded(
child: Align(
alignment: Alignment.bottomCenter,
child: StatsOverview(),
),
),
),
),
Expanded(child: navigator),
],
),
);
}
}

View File

@@ -1,95 +0,0 @@
import 'package:flutter/material.dart';
import 'package:gap/gap.dart';
import 'package:go_router/go_router.dart';
import 'package:hiddify/core/core_providers.dart';
import 'package:hiddify/core/router/router.dart';
import 'package:hiddify/features/common/common.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
class MobileWrapper extends HookConsumerWidget {
const MobileWrapper(this.navigator, {super.key});
final Widget navigator;
@override
Widget build(BuildContext context, WidgetRef ref) {
final t = ref.watch(translationsProvider);
final currentIndex = getCurrentIndex(context);
final location = GoRouterState.of(context).uri.path;
return Scaffold(
key: RootScaffold.stateKey,
body: navigator,
drawer: SafeArea(
child: Drawer(
width: (MediaQuery.of(context).size.width * 0.88).clamp(0, 304),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
const Gap(16),
DrawerTile(
label: t.settings.pageTitle,
icon: Icons.settings,
selected: location == SettingsRoute.path,
onSelect: () => const SettingsRoute().push(context),
),
DrawerTile(
label: t.logs.pageTitle,
icon: Icons.article,
selected: location == LogsRoute.path,
onSelect: () => const LogsRoute().push(context),
),
DrawerTile(
label: t.about.pageTitle,
icon: Icons.info,
selected: location == AboutRoute.path,
onSelect: () => const AboutRoute().push(context),
),
const Gap(16),
],
),
),
),
bottomNavigationBar: NavigationBar(
destinations: [
NavigationDestination(
icon: const Icon(Icons.power_settings_new),
label: t.home.pageTitle,
),
NavigationDestination(
icon: const Icon(Icons.filter_list),
label: t.proxies.pageTitle,
),
],
selectedIndex: currentIndex > 1 ? 0 : currentIndex,
onDestinationSelected: (index) => switchTab(index, context),
),
);
}
}
class DrawerTile extends StatelessWidget {
const DrawerTile({
super.key,
required this.label,
required this.icon,
required this.selected,
required this.onSelect,
});
final String label;
final IconData icon;
final bool selected;
final VoidCallback onSelect;
@override
Widget build(BuildContext context) {
return ListTile(
title: Text(label),
leading: Icon(icon),
selected: selected,
onTap: selected ? () {} : onSelect,
);
}
}

View File

@@ -1,2 +0,0 @@
export 'view/desktop_wrapper.dart';
export 'view/mobile_wrapper.dart';