5

我在 StackOverflow 中发现了大量此类消息。像其他许多人一样,我在切换选项卡时也遇到了选项卡内容重叠的问题。我发现的所有建议都不适用于我的问题。

当我的应用程序启动时,它会正确显示第一个选项卡的内容。当我单击另一个选项卡时,旧的内容会保留在屏幕上,并且另一个选项卡的内容也会添加到屏幕上。第二次切换标签时,所有内容都消失了。切换标签不会再做任何事情了。

我在这里关注了谷歌的开发者文档。

我的应用程序有这个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他们在方法中夸大了他们的布局。

怎么了?

4

1 回答 1

7

怎么了?

快速浏览 的代码后ActionBarActivity,似乎有一个错误,用于ICS实现的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容器。请记住,我的实现不负责配置更改。

于 2013-10-28T20:36:45.383 回答