10

我有一个浏览器,当然有一些片段。这些片段中的每一个都将 CoordinatorLayout 作为父级。

我正在展示一个小吃店。问题是,如果 Fragment A 显示一个snackbar 很好,但如果它是相邻的 Fragment B 也显示一个snackbar,那么 Fragment A 中的snackbar 会自动隐藏。

由于 viewpager 相邻片段是预加载的,因此这是一个可见的问题。任何解决方法?还是我做错了?

4

2 回答 2

3

ViewPager 中的多个 Fragments 中的snackbar 存在一些问题

1) 如果在 Fragment A(可见)和 Fragment B(不可见)上显示了snackbar,则两个snackbar 都不可见。

2)如果我使用 @Override setUserVisibleHint 和 getUserVisibleHint() 的组合来根据片段可见性巧妙地显示和隐藏小吃栏,它只能在第一次工作。之后,调用snackbar.show() 不再显示snackbar(除非我重新创建snackbar)。

以下是我提出的解决方案(使用 v23.1.1 测试):

public class SnackbarManager {
    private static final String TAG = SnackbarManager.class.getName();

    private Snackbar snackbar;
    private Create instance;
    // private boolean isMultiSnackbar;

    public interface Create {
        Snackbar create();
    }

    public SnackbarManager(Create instance) {
        // why not pass in snackbar? coz snackbar.show will fail after 1st show (it multiple snackbar), thus need to recreate it
        snackbar = instance.create();
        this.instance = instance;
    }

    public void show(Fragment fragment) {
        if (fragment.getUserVisibleHint()) {
            snackbar.show();
        }
    }

    public void onSetUserVisibleHint(boolean isVisible) {
        if (isVisible) {
            if (snackbar == null) {
                snackbar = instance.create();
            }
            snackbar.show();
            Log.d(TAG, "showSnackbar="+snackbar.isShown());
            // if snackbar.isShown()=false, if means multiple snackbar exist (might or might not be in same fragment)
            /*
            boolean isMultiSnackbar = !snackbar.isShown();
            // the following is inaccurate when I manually dismiss one of the snackbar
            // even when isShown()=true, the snackbar is not shown
            if (isMultiSnackbar) {
                snackbar = null;
                snackbar = instance.create();
                snackbar.show();
            }
             */
            }
            else {
                Log.d(TAG, "dismissSnackbar");
                snackbar.dismiss();
                // subsequent show will fail, make sure to recreate next
                snackbar = null;
            }
        }
    }
}

public class TestFragment extends Fragment {
    private SnackbarManager snackbarManager;

    @Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        if (snackbarManager != null)
            snackbarManager.onSetUserVisibleHint(isVisibleToUser);
    }

    public void showSnackbar() {
        snackbarManager = new SnackbarManager(new SnackbarManager.Create() {
            @Override
            public Snackbar create() {
                Snackbar snackbar = Snackbar.make(snackbarLayout, "Create your first moment ;)", Snackbar.LENGTH_INDEFINITE);
                snackbar.setAction("Create", new View.OnClickListener() {
                    @Override
                    public void onClick(View view) {
                        snackbarManager = null;
                    }
                });
                // enable following if not Snackbar.LENGTH_INDEFINITE
                /*
                snackbar.setCallback(new Snackbar.Callback() {
                    @Override
                    public void onDismissed(Snackbar snackbar, int event) {
                        super.onDismissed(snackbar, event);
                        snackbarManager = null;
                    }
                })
                */
                return snackbar;
            }
        });
        snackbarManager.show(this);
    }
}
于 2016-02-20T07:02:07.337 回答
2

您可以使用setUserVisibleHint(boolean isVisible)更新片段可见性(例如在 ViewPager onPageSelected 上),然后在片段上使用 getUserVisibleHint()以仅在片段可见时显示 Snackbar。

假设您有片段 A、B 和 C

Fragment A 可见,Snackbar 也是可见的 如果 Fragment B 或 C 尝试打开 Snackbar,getUserVisibleHint() 将返回 false,它们不会与 Fragment A Snackbar 混淆。

观察。使用 FragmentStatePagerAdapter 执行此操作时要小心,因为您可以获得 NPE 保留对 Fragments 的引用并在此引用上调用 setUserVisibleHint()。

编辑:您不需要 setUserVisibleHint() 因为它已经被系统调用。

于 2016-01-12T18:11:17.200 回答