导航架构组件 - 对话框片段

时间:2018-05-13 00:02:41

标签: android android-architecture-components android-jetpack android-architecture-navigation

是否可以使用带有DialogFragment的新导航架构组件?我是否必须创建自定义导航器?

我很乐意将它们与导航图中的新功能结合使用。

10 个答案:

答案 0 :(得分:16)

不,从1.0.0-alpha01构建开始,作为导航图的一部分,不支持对话框。您应该继续使用show()来显示DialogFragment

答案 1 :(得分:8)

2019年5月更新

  从Navigation 2.1.0-alpha03开始完全支持

DialogFragment,您可以阅读更多herehere

导航的旧答案<= 2.1.0-alpha02:

我以这种方式进行:

1)至少将Navigation库更新为版本2.1.0-alpha01,并在您的项目中复制此modified gist的两个文件。

3),然后在导航主机片段中,将name参数更改为自定义NavHostFragment

<fragment
    android:id="@+id/nav_host_fragment"
    android:name="com.example.app.navigation.MyNavHostFragment"
    app:defaultNavHost="true"
    app:navGraph="@navigation/nav_graph"
    android:layout_width="0dp"
    android:layout_height="0dp"
    app:layout_constraintBottom_toBottomOf="parent"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toBottomOf="@id/toolbar" />

4)创建您的DialogFragment子类,并将它们添加到您的nav_graph.xml中,

<dialog-fragment
    android:id="@+id/my_dialog"
    android:name="com.example.ui.MyDialogFragment"
    tools:layout="@layout/my_dialog" />

5)现在从片段或活动开始使用它们

findNavController().navigate(R.id.my_dialog)

或类似方法。

答案 2 :(得分:3)

是的,您可以通过调用getParentFragment()。getView()从对话框片段访问父片段的视图。并使用该视图进行导航。

这里是例子

Navigation.findNavController(getParentFragment().getView()).navigate(R.id.nextfragment);

答案 3 :(得分:2)

我为DialogFragment创建了自定义导航器。

样本为here
(这只是示例,因此可能有任何问题。)

@Navigator.Name("dialog_fragment")
class DialogNavigator(
    private val fragmentManager: FragmentManager
) : Navigator<DialogNavigator.Destination>() {

    companion object {
        private const val TAG = "dialog"
    }

    override fun navigate(destination: Destination, args: Bundle?, 
            navOptions: NavOptions?, navigatorExtras: Extras?) {
        val fragment = destination.createFragment(args)
       fragment.setTargetFragment(fragmentManager.primaryNavigationFragment, 
               SimpleDialogArgs.fromBundle(args).requestCode)
        fragment.show(fragmentManager, TAG)
        dispatchOnNavigatorNavigated(destination.id, BACK_STACK_UNCHANGED)
    }

    override fun createDestination(): Destination {
        return Destination(this)
    }

    override fun popBackStack(): Boolean {
        return true
    }

    class Destination(
            navigator: Navigator<out NavDestination>
    ) : NavDestination(navigator) {

        private var fragmentClass: Class<out DialogFragment>? = null

        override fun onInflate(context: Context, attrs: AttributeSet) {
            super.onInflate(context, attrs)
            val a = context.resources.obtainAttributes(attrs,
                    R.styleable.FragmentNavigator)
            a.getString(R.styleable.FragmentNavigator_android_name)
                    ?.let { className ->
                fragmentClass = parseClassFromName(context, className, 
                        DialogFragment::class.java)
            }
            a.recycle()
        }

        fun createFragment(args: Bundle?): DialogFragment {
            val fragment = fragmentClass?.newInstance()
                ?: throw IllegalStateException("fragment class not set")
            args?.let {
                fragment.arguments = it
            }
            return fragment
        }
    }
}

答案 4 :(得分:1)

。该框架的制作方式是,您可以为未开箱即用的视图创建一个扩展Navigator抽象类的类,并使用方法{将其添加到您的NavController中{1}}

如果您使用的是getNavigatorProvider().addNavigator(Navigator navigator),则还需要对其进行扩展以添加自定义导航器,或者仅创建自己的实现NavHostFragment的{​​{1}}即可。它非常灵活,您可以像在创建自定义视图中一样,使用MyFragment中定义的自定义属性来创建自己的xml参数。这样的东西(未经测试):

NavHost

用法

my_navigation.xml

values

将导航的片段。

@Navigator.Name("dialog-fragment")
class DialogFragmentNavigator(
        val context: Context,
        private val fragmentManager: FragmentManager
) : Navigator<DialogFragmentNavigator.Destination>() {

    override fun navigate(destination: Destination, args: Bundle?,
                          navOptions: NavOptions?, navigatorExtras: Extras?
    ): NavDestination {
        val fragment = Class.forName(destination.name).newInstance() as DialogFragment
        fragment.show(fragmentManager, destination.id.toString())
        return destination
    }

    override fun createDestination(): Destination = Destination(this)

    override fun popBackStack() = fragmentManager.popBackStackImmediate()

    class Destination(navigator: DialogFragmentNavigator) : NavDestination(navigator) {

        // The value of <dialog-fragment app:name="com.example.MyFragmentDialog"/>
        lateinit var name: String

        override fun onInflate(context: Context, attrs: AttributeSet) {
            super.onInflate(context, attrs)
            val a = context.resources.obtainAttributes(
                    attrs, R.styleable.FragmentNavigator
            )
            name = a.getString(R.styleable.FragmentNavigator_android_name)
                    ?: throw RuntimeException("Error while inflating XML. " +
                            "`name` attribute is required")
            a.recycle()
        }
    }
}

答案 5 :(得分:0)

一种选择是仅使用常规片段,并使它看起来类似于对话框。我发现这不值得麻烦,所以我使用了show()的标准方法。如果您坚持See here做某事。

答案 6 :(得分:0)

Version 2.1.0-alpha03已发布,因此我们终于可以使用DialogFragments。对我来说不幸的是,在使用可取消对话框时,我在backstack上遇到了一些问题。可能是我的对话框的执行方式有误。

[LATER-EDIT]我的实现很好,该问题与issue tracker中所述的DialogFragmentNavigator的错误对话框计数有关 作为解决方法,您可以看看on my recommendation

答案 7 :(得分:0)

是的,现在有可能。在最初的版本中这是不可能的,但是现在  来自“ androidx.navigation:navigation-fragment:2.1.0-alpha03”的此导航版本,可以在导航组件中使用对话框片段。

检查一下:-Naviagtion dialog fragment support

答案 8 :(得分:0)

更新为:

implementation "androidx.navigation:navigation-ui-ktx:2.2.0-rc04"

并在my_nav_graph.xml中使用

<dialog
android:id="@+id/my_dialog"
android:name="com.example.ui.MyDialogFragment"
tools:layout="@layout/my_dialog" />

答案 9 :(得分:0)

是的。导航组件的最新更新中有可能。您可以检查此链接以了解清楚的概念。 raywenderlich.com