DrawerLayout的项目点击 - 何时是替换片段的正确时间?

dor*_*506 46 android android-fragments fragmenttransaction navigation-drawer drawerlayout

我正在开发一个使用导航抽屉图案的应用程序(使用DrawerLayout).

每次单击抽屉的项目,都会替换主容器中的片段.

但是,我不确定何时是进行片段交易的合适时机?抽屉开始关闭时?或者关闭后?

在google的documentaion示例中,您可以看到他们在项目单击后立即执行事务,然后关闭抽屉.
结果,抽屉看起来很迟钝而且不平滑,看起来非常糟糕(它也发生在我的应用中).

GmailGoogle云端硬盘应用程序中,另一方面,它似乎是在抽屉关闭后进行交易(我是对吗?).
结果,抽屉没有延迟且非常平滑,但至少需要大约1秒钟(抽屉关闭所需的时间)才能看到下一个片段.

在立即进行片段交易时,似乎抽屉没有办法顺畅.

你觉得怎么样?

提前致谢!

eve*_*otc 25

Yup,不能同意,执行一个片段(带有视图)事务导致布局传递,导致动画视图上的janky动画,引用DrawerLayout 文档:

DrawerLayout.DrawerListener可用于监视抽屉视图的状态和动作.避免在动画期间执行昂贵的操作,例如布局,因为它可能导致口吃; 尝试在STATE_IDLE状态期间执行昂贵的操作.

因此,请在抽屉关闭或有人修补支持库以某种方式修复:)之后执行您的片段事务

  • 它只是支持库遇到这个问题吗? (4认同)
  • 如何等到状态更改为STATE_IDLE? (2认同)

Mak*_*rio 25

另一种解决方案是在关闭抽屉后创建Handler并发布延迟Runnable,如下所示:https://stackoverflow.com/a/18483633/769501.这种方法的好处是你的碎片会比你等待时更快地被替换DrawerListener#onDrawerClosed(),但当然任意延迟并不能100%保证抽屉动画能够及时完成.

也就是说,我使用了200ms的延迟,它运行得非常好.

private class DrawerItemClickListener implements OnItemClickListener {
    @Override
    public void onItemClick(AdapterView<?> parent, View view, final int position, long id) {
        drawerLayout.closeDrawer(drawerList);
        new Handler().postDelayed(new Runnable() {
            @Override
            public void run() {
                switchFragments(position); // your fragment transactions go here
            }
        }, 200);
    }
}
Run Code Online (Sandbox Code Playgroud)


sag*_*nav 15

这就是我为实现类似于Gmail应用程序的流畅交易动画所做的工作:

activity_drawer.xml

<android.support.v4.widget.DrawerLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/drawer_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <!-- The main content view -->
    <FrameLayout
        android:id="@+id/content_frame"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

    <!-- The navigation drawer -->
    <ListView 
    android:id="@+id/left_drawer"
        android:layout_width="280dp"
        android:layout_height="match_parent"
        android:layout_gravity="left"
        android:choiceMode="singleChoice" />

</android.support.v4.widget.DrawerLayout>
Run Code Online (Sandbox Code Playgroud)

DrawerActivity.java

private Fragment mContentFragment;
private Fragment mNextContentFragment;
private boolean mChangeContentFragment = false;

private Handler mHandler = new Handler();

...

@Override
public void onCreate(Bundle savedInstanceState) {
    ...

    mDrawerLayout.setDrawerListener(new DrawerListener());

    mDrawerList.setOnItemClickListener(new DrawerItemClickListener());

    ...
}

....

private class DrawerItemClickListener implements ListView.OnItemClickListener {

    @Override
    public void onItemClick(AdapterView parent, View view, int position, long id) {
        getSupportFragmentManager().beginTransaction().remove(mContentFragment).commit();

        switch (position) {
            case 0:
                mNextContentFragment = new Fragment1();
                break;

            case 1:
                mNextContentFragment = new Fragment2();
                break;

            case 2:
                mNextContentFragment = new Fragment3();
                break;
        }

        mChangeContentFragment = true;

        mDrawerList.setItemChecked(position, true);

        mHandler.postDelayed(new Runnable() {

            @Override
            public void run() {
                mDrawerLayout.closeDrawer(mDrawerList);
            }           
        }, 150);
    }
}

private class DrawerListener implements android.support.v4.widget.DrawerLayout.DrawerListener {

    @Override
    public void onDrawerClosed(View view) {
        if (mChangeContentFragment) {
             getSupportFragmentManager().beginTransaction().setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN).replace(R.id.content_frame, mNextContentFragment).commit();

             mContentFragment = mNextContentFragment;           
             mNextContentFragment = null;

             mChangeContentFragment = false;
         }
     }
 }
Run Code Online (Sandbox Code Playgroud)

希望对你有所帮助!:-)

  • 后备堆栈不需要更改 - Google的应用程序都没有通过后退按钮切换碎片.这令人困惑. (2认同)

use*_*637 5

我知道这个问题已经过时了,但我遇到了同样的问题并且认为我会发布我的解决方案,因为我认为这是一个比添加硬编码延迟时间更好的实现.我所做的是onDrawerClosed在执行任务之前使用该函数来验证抽屉是否已关闭.

//on button click...
private void displayView(int position) {
    switch (position) {
    //if item 1 is selected, update a global variable `"int itemPosition"` to be 1
    case 1:
        itemPosition = 1;
        //();
        break;
    default:
        break;
    }

    // update selected item and title, then close the drawer
    mDrawerList.setItemChecked(position, true);
    mDrawerList.setSelection(position);
    mDrawerLayout.closeDrawer(mDrawerList); //close drawer
}
Run Code Online (Sandbox Code Playgroud)

然后在onDrawerClosed,打开相应的活动.

public void onDrawerClosed(View view) {
    getSupportActionBar().setTitle(mTitle);
    // calling onPrepareOptionsMenu() to show action bar icons
    supportInvalidateOptionsMenu();
    if (itemPosition == 1) {
        Intent intent = new Intent(BaseActivity.this, SecondActivity.class);
        startActivity(intent);
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 抽屉关闭和开始新活动之间仍然存在延迟。 (2认同)