ActionBar选项卡内容重叠

时间:2013-10-28 18:24:27

标签: java android android-fragments tabs android-actionbar

我在StackOverflow中发现了大量这些消息。像其他许多人一样,切换标签时,标签内容重叠也存在同样的问题。我找到的所有建议都没有解决我的问题。

当我的应用启动时,它会正确显示第一个标签的内容。当我单击其他选项卡时,旧内容将保留在屏幕上,另一个选项卡的内容也会添加到屏幕上。第二次切换标签时,所有内容都会消失。切换标签将不再适用。

我关注了Google的开发人员文档here

我的应用程序使用此onCreate方法。该类从支持库ActionBarActivity扩展android.support.v7.app

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    ActionBar actionBar = getSupportActionBar();
    actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
    actionBar.setDisplayShowTitleEnabled(false);

    Tab tab = actionBar.newTab().setText("TAB1").setTabListener(new TabListener<Tab1Class>(this, "tab1", Tab1Class.class));
    actionBar.addTab(tab);

    tab = actionBar.newTab().setText("TAB2").setTabListener(new TabListener<Tab2Class>(this, "tab2", Tab2Class.class));
    actionBar.addTab(tab);
}

我的TabListener课程是从我链接的页面复制的:

public class TabListener<T extends Fragment> implements ActionBar.TabListener {
    private Fragment mFragment;
    private final Activity mActivity;
    private final String mTag;
    private final Class<T> mClass;

    public TabListener(Activity activity, String tag, Class<T> clz) {
        mActivity = activity;
        mTag = tag;
        mClass = clz;
    }

    @Override
    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        if(mFragment == null) {
            mFragment = Fragment.instantiate(mActivity, mClass.getName());
            ft.add(android.R.id.content, mFragment, mTag);
        } else {
            ft.attach(mFragment);
        }
    }

    @Override
    public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        if(mFragment != null) {
            ft.detach(mFragment);
        }
    }

    @Override
    public void onTabReselected(Tab tab, FragmentTransaction ft) {} 
}

我用作选项卡内容的两个类都扩展了android.support.v4.app.Fragment的片段。他们用onCreateView方法夸大了他们的布局。

怎么了?

1 个答案:

答案 0 :(得分:7)

  

怎么了?

快速查看ActionBarActivity的代码后,ICS的{​​{1}}以及ActionBar实施的上方似乎存在错误(代码应该适用于预ICS设备),它也负责标签。

在代表ActionBarImplICS设备实现的ICS类中,传递给FragmentTransaction回调的onTabUnselected()似乎完全没用,因为它在任何地方都没有提交侦听器的回调返回(事务是为TabListener的其他两个回调提交的)。因此,一个已提交的片段永远不会从选项卡选项上的布局中分离出来,并且它将保持获取重叠的内容(由于FrameLayout包含两个片段)。

我已经编写了TabListener接口的另一个实现,它只从一个不受上述错误(onTabSelected())影响的回调中完成所有工作:

public class TabListenerImpl implements ActionBar.TabListener {

    private List<TabInfo> mTabs = new ArrayList<TabInfo>();
    private Context mContext;

    public TabListenerImpl(Context context) {
        mContext = context;
    }

    @Override
    public void onTabReselected(Tab tab, FragmentTransaction ft) {

    }

    @Override
    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        // iterate over all of the tabs, match the tag we have and see if
        // we also have a fragment instance for it. If we don't, create one
        // and add it to the container, if we have an instance simply attach
        // it. Detach every other tag which doesn't match the tag.
        for (TabInfo t : mTabs) {
            if (tab.getTag() == t.tag) {
                if (t.pageFragment == null) {
                    t.pageFragment = Fragment.instantiate(mContext,
                            t.clazz.getName());
                    ft.add(android.R.id.content, t.pageFragment, t.tag);
                } else {
                    ft.attach(t.pageFragment);
                }
            } else {
                if (t.pageFragment != null && !t.pageFragment.isDetached()) {
                    ft.detach(t.pageFragment);
                }
            }
        }
    }

    @Override
    public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        // faulty method
    }

    /**
     * Call this method BEFORE you call the actionBar.addTab() method!
     * 
     * @param tag
     *            a String representing the tag that was set on the tab to
     *            identify itself
     * @param clazz
     *            the class of the Fragment
     */
    public void addTab(String tag, Class<? extends Fragment> clazz) {
        TabInfo ti = new TabInfo();
        ti.clazz = clazz;
        ti.tag = tag;
        mTabs.add(ti);
    }

    // wrapper class
    private class TabInfo {
        Class<? extends Fragment> clazz;
        Fragment pageFragment;
        String tag;
    }

}

然后您可以将其用作:

TabListenerImpl listener = new TabListenerImpl(this);
Tab tab = actionBar.newTab().setText("TAB1").setTag("TAB1").setTabListener(listener);
listener.addTab("TAB1", Tab1Class.class);
actionBar.addTab(tab);

tab = actionBar.newTab().setText("TAB2").setTag("TAB2").setTabListener(listener);
listener.addTab("TAB2", Tab2Class.class);
actionBar.addTab(tab);

我建议您将容器设置为内容视图(以及选项卡内容),而不是使用android.R.id.content容器。请记住,我的实现不会考虑配置更改。