由FragmentStatePagerAdapter管理的选定ViewPager片段上的UserVisibleHint为false。

6
我在一个Android应用程序中遇到了一个很难诊断的问题。在当前选定的ViewPager片段上,getUserVisibleHint()返回false,但实际上它应该返回true(因为它可见且已选定)。
我将这种行为定义为以下几种情况:
  • 片段被选中并在ViewPager中显示
  • ViewPagerFragmentStatePagerAdapter管理
  • 该片段之前已经被选中过,并保存了其状态,稍后由PagerAdapter恢复
    • viewpager中要求至少3个选项卡
    • 用户导航到选项卡3,然后到选项卡1,再返回到选项卡3。
  • 应用程序使用支持库版本24.0.0或更高版本

2
Bug报告:[https://issuetracker.google.com/issues/38414371] - Jon
2个回答

8
调试揭示了 FragmentStatePagerAdaptersetPrimaryItem(ViewGroup container, int position, Object object) 中实际上正确地设置了所选标签的状态,但后来在 FragmentManager#moveToState(Fragment f, int newState, int transit, int transitionStyle, boolean keepActive) 中被设置为 false。
//from FragmentManager#moveToState(Fragment f, int newState, int transit, int transitionStyle, boolean keepActive)
f.mUserVisibleHint = f.mSavedFragmentState.getBoolean(
        FragmentManagerImpl.USER_VISIBLE_HINT_TAG, true);

f.mSavedFragmentState中的visible state被保存为false,因为该片段已不在屏幕上时进行了保存。

因此,这里存在状态丢失的问题;可见状态是在FragmentStatePagerAdapter#setPrimaryItem中设置的,但在该片段的onResume方法被调用之前就丢失了。

解决方案

在库中修复此错误之前,请在您的PagerAdapter中覆盖setPrimaryItem并强制执行任何未完成的事务。

public static class SectionsPagerAdapter extends FragmentStatePagerAdapter {
    public SectionsPagerAdapter(FragmentManager fm) {
        super(fm);
    }

    @Override
    public void setPrimaryItem(ViewGroup container, int position, Object object) {
        //Force any pending transactions to save before we set an item as primary
        finishUpdate(null);
        super.setPrimaryItem(container, position, object);
    }

    @Override
    public Fragment getItem(int position) {
        Fragment fragment = new DummySectionFragment();
        Bundle args = new Bundle();
        args.putInt(DummySectionFragment.ARG_SECTION_NUMBER, position + 1);
        fragment.setArguments(args);
        return fragment;
    }

    @Override
    public int getCount() {
        return 4;
    }

    @Override
    public CharSequence getPageTitle(int position) {
        return "Page " + (position + 1);
    }
}

为了解决这个问题,必须在设置用户可见提示之前,让FragmentStatePagerAdapter提交任何片段事务。 < h3 > < code > FragmentStatePagerAdapter < / code > < / h3 > 只是为了展示FragmentStatePagerAdapter内部发生的情况。
@Override
public Object instantiateItem(ViewGroup container, int position) {
    // If we already have this item instantiated, there is nothing
    // to do.  This can happen when we are restoring the entire pager
    // from its saved state, where the fragment manager has already
    // taken care of restoring the fragments we previously had instantiated.
    if (mFragments.size() > position) {
        Fragment f = mFragments.get(position);
        if (f != null) {
            return f;
        }
    }

    if (mCurTransaction == null) {
        mCurTransaction = mFragmentManager.beginTransaction();
    }

    Fragment fragment = getItem(position);
    if (DEBUG) Log.v(TAG, "Adding item #" + position + ": f=" + fragment);
    if (mSavedState.size() > position) {
        Fragment.SavedState fss = mSavedState.get(position);
        if (fss != null) {
            fragment.setInitialSavedState(fss);
        }
    }
    while (mFragments.size() <= position) {
        mFragments.add(null);
    }
    fragment.setMenuVisibility(false);
    fragment.setUserVisibleHint(false);
    mFragments.set(position, fragment);
    mCurTransaction.add(container.getId(), fragment);

    return fragment;
}

@Override
public void setPrimaryItem(ViewGroup container, int position, Object object) {
    Fragment fragment = (Fragment)object;
    if (fragment != mCurrentPrimaryItem) {
        if (mCurrentPrimaryItem != null) {
            mCurrentPrimaryItem.setMenuVisibility(false);
            mCurrentPrimaryItem.setUserVisibleHint(false);
        }
        if (fragment != null) {
            fragment.setMenuVisibility(true);
            fragment.setUserVisibleHint(true);
        }
        mCurrentPrimaryItem = fragment;
    }
}

@Override
public void finishUpdate(ViewGroup container) {
    if (mCurTransaction != null) {
        mCurTransaction.commitNowAllowingStateLoss();
        mCurTransaction = null;
    }
}

遇到了与appcompat-v7:25.3.1相同的bug。您的解决方案有效,谢谢。 - alders
我已经拥有了由Google团队标记为已修复的新版本(27.0.2),但问题仍然存在,我尝试了您提供的修复方法,但问题仍然存在..!有什么帮助吗? - Alaa AbuZarifa
@AlaaAbuZarifa 27.0.2于2017年11月发布。错误报告在2018年2月底标记为已修复。我预计它已经在27.1.1中得到修复,但我还没有测试过这个版本。 - Jon

1
如果与您的项目兼容,请尝试使用支持库的版本27.1.1(或更高版本)。相关的错误报告已于2018年2月底被标记为已修复,而版本27.1.1则在2018年4月发布。

我正在使用支持库版本27.1.1,但仍然遇到这个问题,将尝试您的解决方案并相应地更新。 - Abbas
你的解决方案修复了这个问题,但是在查看代码时我发现FragmentStatePagerAdapter V1327.1.0版本起已被弃用,建议使用FragmentStatePagerAdapter V4代替,但似乎只能与V4支持的Fragments一起使用。这是开发人员的解决方案吗?改用V4 Fragments。 - Abbas

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接