mirror of
https://github.com/flutter/samples.git
synced 2025-11-10 14:58:34 +00:00
Bumps [go_router](https://github.com/flutter/packages/tree/main/packages) from 6.5.9 to 7.0.0. <details> <summary>Commits</summary> <ul> <li><a href="b01736711c"><code>b017367</code></a> [go_router] Cleans up route match API and introduces dart fix (<a href="https://github.com/flutter/packages/tree/main/packages/issues/3819">#3819</a>)</li> <li><a href="2f95ecda81"><code>2f95ecd</code></a> [camerax] Add <code>LifecycleOwner</code> Proxy (<a href="https://github.com/flutter/packages/tree/main/packages/issues/3837">#3837</a>)</li> <li><a href="815cfca847"><code>815cfca</code></a> [file_selector] Add getDirectoryPaths implementation for Windows (<a href="https://github.com/flutter/packages/tree/main/packages/issues/3704">#3704</a>)</li> <li><a href="6dc28c9c3f"><code>6dc28c9</code></a> [various] Update Android example min SDKs (<a href="https://github.com/flutter/packages/tree/main/packages/issues/3847">#3847</a>)</li> <li><a href="1a90429ac1"><code>1a90429</code></a> [camerax] Implement Image Streaming (<a href="https://github.com/flutter/packages/tree/main/packages/issues/3454">#3454</a>)</li> <li><a href="124e330a2b"><code>124e330</code></a> [various] update agp and gradle for all examples in packages (<a href="https://github.com/flutter/packages/tree/main/packages/issues/3822">#3822</a>)</li> <li><a href="c71747df2c"><code>c71747d</code></a> Update xcode to 14c18 (<a href="https://github.com/flutter/packages/tree/main/packages/issues/3774">#3774</a>)</li> <li><a href="3dc59f914d"><code>3dc59f9</code></a> [camera_android] Add NV21 as an image stream format <a href="https://github.com/flutter/packages/tree/main/packages/issues/3277">#3277</a> (<a href="https://github.com/flutter/packages/tree/main/packages/issues/3639">#3639</a>)</li> <li>See full diff in <a href="https://github.com/flutter/packages/commits/go_router-v7.0.0/packages">compare view</a></li> </ul> </details> <br /> [](https://docs.github.com/en/github/managing-security-vulnerabilities/about-dependabot-security-updates#about-compatibility-scores) Dependabot will resolve any conflicts with this PR as long as you don't alter it yourself. You can also trigger a rebase manually by commenting `@dependabot rebase`. [//]: # (dependabot-automerge-start) [//]: # (dependabot-automerge-end) --- <details> <summary>Dependabot commands and options</summary> <br /> You can trigger Dependabot actions by commenting on this PR: - `@dependabot rebase` will rebase this PR - `@dependabot recreate` will recreate this PR, overwriting any edits that have been made to it - `@dependabot merge` will merge this PR after your CI passes on it - `@dependabot squash and merge` will squash and merge this PR after your CI passes on it - `@dependabot cancel merge` will cancel a previously requested merge and block automerging - `@dependabot reopen` will reopen this PR if it is closed - `@dependabot close` will close this PR and stop Dependabot recreating it. You can achieve the same result by closing it manually - `@dependabot ignore this major version` will close this PR and stop Dependabot creating any more for this major version (unless you reopen the PR or upgrade to it yourself) - `@dependabot ignore this minor version` will close this PR and stop Dependabot creating any more for this minor version (unless you reopen the PR or upgrade to it yourself) - `@dependabot ignore this dependency` will close this PR and stop Dependabot creating any more for this dependency (unless you reopen the PR or upgrade to it yourself) </details> --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Brett Morgan <brettmorgan@google.com>
252 lines
7.6 KiB
Dart
252 lines
7.6 KiB
Dart
// Copyright 2018 The Flutter team. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
import 'dart:io';
|
|
|
|
import 'package:flutter/cupertino.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/services.dart' show DeviceOrientation, SystemChrome;
|
|
import 'package:go_router/go_router.dart';
|
|
import 'package:provider/provider.dart';
|
|
import 'package:veggieseasons/data/app_state.dart';
|
|
import 'package:veggieseasons/data/preferences.dart';
|
|
import 'package:veggieseasons/screens/home.dart';
|
|
import 'package:veggieseasons/styles.dart';
|
|
import 'package:veggieseasons/widgets/fade_transition_page.dart';
|
|
import 'package:window_size/window_size.dart';
|
|
|
|
import 'screens/details.dart';
|
|
import 'screens/favorites.dart';
|
|
import 'screens/list.dart';
|
|
import 'screens/search.dart';
|
|
import 'screens/settings.dart';
|
|
|
|
void main() {
|
|
WidgetsFlutterBinding.ensureInitialized();
|
|
SystemChrome.setPreferredOrientations([
|
|
DeviceOrientation.portraitUp,
|
|
DeviceOrientation.portraitDown,
|
|
]);
|
|
setupWindow();
|
|
|
|
runApp(
|
|
const RootRestorationScope(
|
|
restorationId: 'root',
|
|
child: VeggieApp(),
|
|
),
|
|
);
|
|
}
|
|
|
|
const double windowWidth = 480;
|
|
const double windowHeight = 854;
|
|
|
|
void setupWindow() {
|
|
if (!kIsWeb && (Platform.isWindows || Platform.isLinux || Platform.isMacOS)) {
|
|
setWindowTitle('Veggie Seasons');
|
|
setWindowMinSize(const Size(windowWidth, windowHeight));
|
|
setWindowMaxSize(const Size(windowWidth, windowHeight));
|
|
getCurrentScreen().then((screen) {
|
|
setWindowFrame(Rect.fromCenter(
|
|
center: screen!.frame.center,
|
|
width: windowWidth,
|
|
height: windowHeight,
|
|
));
|
|
});
|
|
}
|
|
}
|
|
|
|
final _rootNavigatorKey = GlobalKey<NavigatorState>();
|
|
final _shellNavigatorKey = GlobalKey<NavigatorState>();
|
|
|
|
class VeggieApp extends StatefulWidget {
|
|
const VeggieApp({super.key});
|
|
|
|
@override
|
|
State<StatefulWidget> createState() => _VeggieAppState();
|
|
}
|
|
|
|
class _VeggieAppState extends State<VeggieApp> with RestorationMixin {
|
|
final _RestorableAppState _appState = _RestorableAppState();
|
|
|
|
@override
|
|
String get restorationId => 'wrapper';
|
|
|
|
@override
|
|
void restoreState(RestorationBucket? oldBucket, bool initialRestore) {
|
|
registerForRestoration(_appState, 'state');
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_appState.dispose();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return MultiProvider(
|
|
providers: [
|
|
ChangeNotifierProvider.value(
|
|
value: _appState.value,
|
|
),
|
|
ChangeNotifierProvider(
|
|
create: (_) => Preferences()..load(),
|
|
),
|
|
],
|
|
child: CupertinoApp.router(
|
|
theme: Styles.veggieThemeData,
|
|
debugShowCheckedModeBanner: false,
|
|
restorationScopeId: 'app',
|
|
routerConfig: GoRouter(
|
|
navigatorKey: _rootNavigatorKey,
|
|
restorationScopeId: 'router',
|
|
initialLocation: '/list',
|
|
redirect: (context, state) {
|
|
if (state.path == '/') {
|
|
return '/list';
|
|
}
|
|
return null;
|
|
},
|
|
debugLogDiagnostics: true,
|
|
routes: [
|
|
ShellRoute(
|
|
navigatorKey: _shellNavigatorKey,
|
|
pageBuilder: (context, state, child) {
|
|
return CupertinoPage(
|
|
restorationId: 'router.shell',
|
|
child: HomeScreen(
|
|
restorationId: 'home',
|
|
child: child,
|
|
onTap: (index) {
|
|
if (index == 0) {
|
|
context.go('/list');
|
|
} else if (index == 1) {
|
|
context.go('/favorites');
|
|
} else if (index == 2) {
|
|
context.go('/search');
|
|
} else {
|
|
context.go('/settings');
|
|
}
|
|
},
|
|
),
|
|
);
|
|
},
|
|
routes: [
|
|
GoRoute(
|
|
path: '/list',
|
|
pageBuilder: (context, state) {
|
|
return FadeTransitionPage(
|
|
key: state.pageKey,
|
|
restorationId: 'route.list',
|
|
child: const ListScreen(restorationId: 'list'),
|
|
);
|
|
},
|
|
routes: [
|
|
_buildDetailsRoute(),
|
|
],
|
|
),
|
|
GoRoute(
|
|
path: '/favorites',
|
|
pageBuilder: (context, state) {
|
|
return FadeTransitionPage(
|
|
key: state.pageKey,
|
|
restorationId: 'route.favorites',
|
|
child: const FavoritesScreen(restorationId: 'favorites'),
|
|
);
|
|
},
|
|
routes: [
|
|
_buildDetailsRoute(),
|
|
],
|
|
),
|
|
GoRoute(
|
|
path: '/search',
|
|
pageBuilder: (context, state) {
|
|
return FadeTransitionPage(
|
|
key: state.pageKey,
|
|
restorationId: 'route.search',
|
|
child: const SearchScreen(restorationId: 'search'),
|
|
);
|
|
},
|
|
routes: [
|
|
_buildDetailsRoute(),
|
|
],
|
|
),
|
|
GoRoute(
|
|
path: '/settings',
|
|
pageBuilder: (context, state) {
|
|
return FadeTransitionPage(
|
|
key: state.pageKey,
|
|
restorationId: 'route.settings',
|
|
child: const SettingsScreen(restorationId: 'settings'),
|
|
);
|
|
},
|
|
routes: [
|
|
GoRoute(
|
|
parentNavigatorKey: _rootNavigatorKey,
|
|
path: 'categories',
|
|
pageBuilder: (context, state) {
|
|
return VeggieCategorySettingsScreen.pageBuilder(
|
|
context);
|
|
},
|
|
),
|
|
GoRoute(
|
|
parentNavigatorKey: _rootNavigatorKey,
|
|
path: 'calories',
|
|
pageBuilder: (context, state) {
|
|
return CalorieSettingsScreen.pageBuilder(context);
|
|
},
|
|
),
|
|
],
|
|
),
|
|
],
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
// GoRouter does not support relative routes,
|
|
// see https://github.com/flutter/flutter/issues/108177
|
|
GoRoute _buildDetailsRoute() {
|
|
return GoRoute(
|
|
parentNavigatorKey: _rootNavigatorKey,
|
|
path: 'details/:id',
|
|
pageBuilder: (context, state) {
|
|
final veggieId = int.parse(state.pathParameters['id']!);
|
|
return CupertinoPage(
|
|
restorationId: 'route.details',
|
|
fullscreenDialog: true,
|
|
child: DetailsScreen(
|
|
id: veggieId,
|
|
restorationId: 'details',
|
|
),
|
|
);
|
|
},
|
|
);
|
|
}
|
|
}
|
|
|
|
class _RestorableAppState extends RestorableListenable<AppState> {
|
|
@override
|
|
AppState createDefaultValue() {
|
|
return AppState();
|
|
}
|
|
|
|
@override
|
|
AppState fromPrimitives(Object? data) {
|
|
final appState = AppState();
|
|
final favorites = (data as List<dynamic>).cast<int>();
|
|
for (var id in favorites) {
|
|
appState.setFavorite(id, true);
|
|
}
|
|
return appState;
|
|
}
|
|
|
|
@override
|
|
Object toPrimitives() {
|
|
return value.favoriteVeggies.map((veggie) => veggie.id).toList();
|
|
}
|
|
}
|