Flutterのナビゲーションの詳細な内訳

画像



Flutterは開発者の間で人気を集めています。アプリケーションを構築するためのアプローチのほとんどはすでに十分に確立されており、Eコマースアプリケーションの開発に毎日使用されています。ナビゲーションのトピックは、背景または3番目の計画に下げられます。フレームワークはどのナビゲーションAPIを提供しますか?どのようなアプローチが開発されましたか?これらのアプローチはどのように使用され、何に役立ちますか?



前書き



ナビゲーションとは何ですか?ナビゲーションは、指定されたパラメーターを使用してユーザーインターフェイス間を移動できるようにする方法です。

たとえば、IOSの世界では、UIViewControllerがナビゲーションを整理し、Androidではナビゲーションコンポーネントを整理します。Flutterは何を提供しますか?





Flutterの画面はルートと呼ばれます。ルート間を移動するために、さまざまなタイプのナビゲーションを実装するための広範なAPIを持つNavigatorクラスがあります。





簡単に始めましょう。新しい画面(ルート)へのナビゲーションは、push()メソッドによって呼び出されます。これは1つの引数を取ります-これはルートです。



Navigator.push(MaterialPageRoute(builder: (BuildContext context) => MyPage()));


pushメソッド呼び出しを詳しく見てみましょう。



Navigator — , .

Navigator.push() — route   .

MaterialPageRoute() — route,   .

builder — MaterialPageRoute, .

MyPage — Stateful/Stateless Widget



route



pop().



Navigator.pop();




builder . Navigator, build.



Navigator.push(context, MaterialPageRoute(builder: (context) => MyPage(someData: data)));


MyPage ( ).



pop() .



Navigator.pop(data);




Navigator, MaterialApp/CupertinoApp/WidgetsApp. State API .

Stack. "call flow" NavigatorState.



https://habrastorage.org/webt/5w/dg/nb/5wdgnb-tjlngub4c8y4rlpqkeqi.png



vs



. route, .



:



,?

: , . .



, — ?

: , .



Imperative vs Declarative Programming





. . Route. Flutter route, MaterialPageRoute CupertinoPageRoute. CupertinoPageRoute title, settings.



:



Navigator.push(
    context,
    CupertinoPageRoute<T>(
        title: "Setting",
        builder: (BuildContext context) => MyPage(),
        settings: RouteSettings(name:"my_page"),
    ),
);


route   ViewModel/Controller/BLoC/… .



MyPage CupertinoPageRoute. push .



:



, route .




. .



:



Navigator.pushNamed(context, '/my_page');


. " " .

MaterialApp/CupertinoApp/WidgetsApp. 2 onGenerateRoute onUnknownRoute .



:



MaterialApp(
    onUnknownRoute: (settings) => CupertinoPageRoute(
      builder: (context) {
                return UndefinedView(name: settings.name);
            }
    ),
  onGenerateRoute: (settings) {
    if (settings.name == '\my_page') {
      return CupertinoPageRoute(
                title: "MyPage",
                settings: settings,
        builder: (context) => MyPage(),
      );
    }
        //     
  },
);


:

onGenerateRoute — Navigator.pushNamed(). route.

onUnknownRouteonGenerateRoute null. route, web — 404 page.



. , .




. .



:



  • showAboutDialog
  • showBottomSheet
  • showDatePicker
  • showGeneralDialog
  • showMenu
  • showModalBottomSheet
  • showSearch
  • showTimePicker
  • showCupertinoDialog
  • showDialog
  • showLicensePage
  • showCupertinoModalPopup


. API, .



?



, showGeneralDialog.



:



Future<T> showGeneralDialog<T>({
  @required BuildContext context,
  @required RoutePageBuilder pageBuilder,
  bool barrierDismissible,
  String barrierLabel,
  Color barrierColor,
  Duration transitionDuration,
  RouteTransitionsBuilder transitionBuilder,
  bool useRootNavigator = true,
  RouteSettings routeSettings,
}) {
  assert(pageBuilder != null);
  assert(useRootNavigator != null);
  assert(!barrierDismissible || barrierLabel != null);
  return Navigator.of(context, rootNavigator: useRootNavigator).push<T>(_DialogRoute<T>(
    pageBuilder: pageBuilder,
    barrierDismissible: barrierDismissible,
    barrierLabel: barrierLabel,
    barrierColor: barrierColor,
    transitionDuration: transitionDuration,
    transitionBuilder: transitionBuilder,
    settings: routeSettings,
  ));
}


. showGeneralDialog push NavigatorState _DialogRoute(). , .



— route .


route



 "every thins is a route", . , route .



route Flutter — PageRoute PopupRoute.



PageRoute — route, .

PopupRoute — route, route.



PageRoute:



  • MaterialPageRoute
  • CupertinoPageRoute
  • _SearchPageRoute
  • PageRouteBuilder


PopupRoute:



  • _ContexMenuRoute
  • _DialogRoute
  • _ModalBottomSheetRoute
  • _CupertinoModalPopupRoute
  • _PopupMenuRoute


PopupRoute , .



:



, .


Best practices



, " ?". , API , . .



:



  • .
  • BuildContext ( , BuildContext).
  • . route, CupertinoPageRoute, BottomSheetRoute, DialogRoute ..


:



abstract class IRouter {
  Future<T> routeTo<T extends Object>(RouteBundle bundle);
  Future<bool> back<T extends Object>({T data, bool rootNavigator});
  GlobalKey<NavigatorState> rootNavigatorKey;
}


:

routeTo - .

back — .

rootNavigatorKey — GlobalKey NavigatorState.

, .



class Router implements IRouter {
    @override
  GlobalKey<NavigatorState> rootNavigatorKey = GlobalKey<NavigatorState>();

    @override
  Future<T> routeTo<T>(RouteBundle bundle) async {
   // Push logic here
  }

    @override
  Future<bool> back<T>({T data, bool rootNavigator = false}) async {
        // Back logic here
    }
}


 , routeTo().



@override
  Future<T> routeTo<T>(RouteBundle bundle) async {
        assert(bundle != null, "The bundle [RouteBundle.bundle] is null");
    NavigatorState rootState = rootNavigatorKey.currentState;
    assert(rootState != null, 'rootState [NavigatorState] is null');

    switch (bundle.route) {
      case "/routeExample":
        return await rootState.push(
          _buildRoute<T>(
            bundle: bundle,
            child: RouteExample(),
          ),
        );

      case "/my_page":
        return await rootState.push(
          _buildRoute<T>(
            bundle: bundle,
            child: MyPage(),
          ),
        );
      default:
        throw Exception('Route is not found');
    }
  }


root NavigatorState ( WidgetsApp) push RouteBundle .



RouteBundle. , .



enum ContainerType {
  /// The parent type is [Scaffold].
  ///
  /// In IOS route with an iOS transition [CupertinoPageRoute].
  /// In Android route with an Android transition [MaterialPageRoute].
  ///
  scaffold,

  /// Used for show child in dialog.
  ///
  /// Route with [DialogRoute].
  dialog,

  /// Used for show child in [BottomSheet].
  ///
  /// Route with [ModalBottomSheetRoute].
  bottomSheet,

  /// Used for show child only.
  /// [AppBar] and other features is not implemented.
  window,
}
class RouteBundle {
  /// Creates a bundle that can be used for [Router].
  RouteBundle({
    this.route,
    this.containerType,
  });

  /// The route for current navigation.
  ///
  /// See [Routes] for details.
  final String route;

  /// The current status of this animation.
  final ContainerType containerType;
}


enum ContainerType — , .

RouteBundle — - route.



_buildRoute. , route .



Route<T> _buildRoute<T>({@required RouteBundle bundle, @required Widget child}) {
    assert(bundle.containerType != null, "The bundle.containerType [RouteBundle.containerType] is null");

    switch (bundle.containerType) {
      case ContainerType.scaffold:
        return CupertinoPageRoute<T>(
          title: bundle.title,
          builder: (BuildContext context) => child,
          settings: RouteSettings(name: bundle.route),
        );
      case ContainerType.dialog:
        return DialogRoute<T>(
          title: '123',
          settings: RouteSettings(name: bundle.route),
          pageBuilder: (BuildContext context, Animation<double> animation, Animation<double> secondaryAnimation) {
                        return child;
                    },
        );
      case ContainerType.bottomSheet:
        return ModalBottomSheetRoute<T>(
          settings: RouteSettings(name: bundle.route),
          isScrollControlled: true,
          builder: (BuildContext context) => child,
        );
      case ContainerType.window:
        return CupertinoPageRoute<T>(
          settings: RouteSettings(name: bundle.route),
          builder: (BuildContext context) => child,
        );
      default:
        throw Exception('ContainerType is not found');
    }
  }


ModalBottomSheetRoute DialogRoute, . route Material Flutter.



back.



@override
Future<bool> back<T>({T data, bool rootNavigator = false}) async {
    NavigatorState rootState = rootNavigatorKey.currentState;
  return await (rootState).maybePop<T>(data);
}


rootNavigatorKey App :



MaterialApp(
    navigatorKey: widget.router.rootNavigatorKey,
    home: Home()
);


, route. - "" , , Dependency Injection.



画像



router.routeTo(RouteBundle(route: '/my_page', containerType: ContainerType.window));


, :



  • BuildContext GlobalKey
  • route View




Flutter , .



:



Flutter Dev Podcast

Flutter




All Articles