Change router for different screen size
This commit is contained in:
@@ -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: [
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
export 'app_router.dart';
|
export 'app_router.dart';
|
||||||
export 'routes/routes.dart';
|
export 'routes.dart';
|
||||||
|
|||||||
16
lib/core/router/routes.dart
Normal file
16
lib/core/router/routes.dart
Normal 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,
|
||||||
|
];
|
||||||
@@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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,
|
|
||||||
];
|
|
||||||
@@ -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 = '/';
|
||||||
|
|||||||
@@ -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(
|
||||||
|
|||||||
171
lib/features/common/adaptive_root_scaffold.dart
Normal file
171
lib/features/common/adaptive_root_scaffold.dart
Normal 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,
|
||||||
|
),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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';
|
||||||
|
|||||||
@@ -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';
|
||||||
|
|||||||
@@ -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,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
50
lib/features/common/nested_app_bar.dart
Normal file
50
lib/features/common/nested_app_bar.dart
Normal 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,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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';
|
||||||
|
|||||||
@@ -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);
|
||||||
|
|||||||
@@ -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),
|
||||||
|
|||||||
@@ -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,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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(),
|
||||||
|
|||||||
@@ -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),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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';
|
||||||
|
|||||||
@@ -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),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -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,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
export 'view/desktop_wrapper.dart';
|
|
||||||
export 'view/mobile_wrapper.dart';
|
|
||||||
Reference in New Issue
Block a user