Actionbarsherlock + tabs +多片段?

时间:2012-04-10 01:54:19

标签: android tabs android-fragments actionbarsherlock

我努力让actionbarsherlock + tabs +片段正常工作。

我只能将这个设置作为静态工作,我想创建这个像Android市场应用程序(滑动运动)。

当你需要在内部使用多个片段来扩充布局时,我会陷入困境。

在Support4demos中,我以FragmentsTabsPager为例进行了跟踪。

3 个答案:

答案 0 :(得分:38)

我实际上设法让这个工作没有比ABS库和支持库更多的工作。这是我的代码:

public class ActionBarTabs extends SherlockFragmentActivity {
CustomViewPager mViewPager;
TabsAdapter mTabsAdapter;
TextView tabCenter;
TextView tabText;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
            WindowManager.LayoutParams.FLAG_FULLSCREEN);

    mViewPager = new CustomViewPager(this);
    mViewPager.setId(R.id.pager);

    setContentView(mViewPager);
    ActionBar bar = getSupportActionBar();
    bar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
    bar.setDisplayOptions(0, ActionBar.DISPLAY_SHOW_TITLE);

    mTabsAdapter = new TabsAdapter(this, mViewPager);

    mTabsAdapter.addTab(bar.newTab().setText("Home"),
            ToolKitFragment.class, null);
    mTabsAdapter.addTab(bar.newTab().setText("FujiSan"),
            FujiFragment.class, null);
}

public static class TabsAdapter extends FragmentPagerAdapter implements
        ActionBar.TabListener, ViewPager.OnPageChangeListener {
    private final Context mContext;
    private final ActionBar mActionBar;
    private final ViewPager mViewPager;
    private final ArrayList<TabInfo> mTabs = new ArrayList<TabInfo>();

    static final class TabInfo {
        private final Class<?> clss;
        private final Bundle args;

        TabInfo(Class<?> _class, Bundle _args) {
            clss = _class;
            args = _args;
        }
    }

    public TabsAdapter(SherlockFragmentActivity activity, ViewPager pager) {
        super(activity.getSupportFragmentManager());
        mContext = activity;
        mActionBar = activity.getSupportActionBar();
        mViewPager = pager;
        mViewPager.setAdapter(this);
        mViewPager.setOnPageChangeListener(this);
    }

    public void addTab(ActionBar.Tab tab, Class<?> clss, Bundle args) {
        TabInfo info = new TabInfo(clss, args);
        tab.setTag(info);
        tab.setTabListener(this);
        mTabs.add(info);
        mActionBar.addTab(tab);
        notifyDataSetChanged();
    }

    @Override
    public int getCount() {
        return mTabs.size();
    }

    @Override
    public Fragment getItem(int position) {
        TabInfo info = mTabs.get(position);
        return Fragment.instantiate(mContext, info.clss.getName(),
                info.args);
    }

    @Override
    public void onPageScrolled(int position, float positionOffset,
            int positionOffsetPixels) {
    }

    @Override
    public void onPageSelected(int position) {
        mActionBar.setSelectedNavigationItem(position);
    }

    @Override
    public void onPageScrollStateChanged(int state) {
    }

    @Override
    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        Object tag = tab.getTag();
        for (int i = 0; i < mTabs.size(); i++) {
            if (mTabs.get(i) == tag) {
                mViewPager.setCurrentItem(i);
            }
        }
    }

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

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

您可以根据以下内容加载标签内容。

mTabsAdapter.addTab(bar.newTab().setText("Home"),
            YOURFRAGMENTHERE.class, null);  

这将为您提供可爱的“swipey tab”效果,正如您所指的那样,使用ABS,支持库和片段。 ABS真的使它与使用本机库几乎相同。我实际上直接从Googles分页标签示例中复制了这段代码,并对ABS进行了稍微修改。

希望这有帮助!

答案 1 :(得分:10)

您需要正确的库来实现您想要的目标。

基本上,ViewPager库是您所缺少的。我的建议:

<强> 1。 ActionbarSherlock

这很容易使用,我不会解释它。

<强> 2。 ViewPagerExtensions

你可以找到它here。下载ZIP文件并从中创建一个新项目。

  

我只能让这个设置作为静态工作,我想创建这个像Android市场应用程序(swype运动)。

从此项目实施com.astuetz.viewpager.extensions.SwipeyTabsView。有一些易于理解的例子。它完全符合你的要求。甚至还有其他选项卡样式可供选择(包括ICS附带的新人员选项卡)。此外,它很容易设计风格以匹配您的应用程序标识。

第3。 FragmentPagerAdapter

最后,来自支持库的那个类(v4)。

祝你好运,如果你需要更多帮助,可以随意问我。


如果您使用我建议的内容,则无需覆盖instantiateItem中的FragmentPagerAdapter。你只需要

  1. 为构造函数提供FragmentManager并调用超级实现;
  2. 覆盖getCount以返回寻呼机中的片段数量,
  3. getItem您将用于返回片段进行创作。
  4. 这是我在这里的代码示例(一个完整的工作,生产示例)。它是实现寻呼机的活动的内部类:

    static class MyFragmentPagerAdapter extends FragmentPagerAdapter {
    
        public MyFragmentPagerAdapter(FragmentManager fm) {
            super(fm);
        }
    
        @Override
        public int getCount() {
            return 2;
        }
    
        @Override
        public Fragment getItem(int position) {
            Fragment f;
            switch(position) {
            case 0:
                f= new ItemSalesDataFragment();
                break;
            case 1:
                f= new DepartmentChooserFragment();
                break;
            default:
                throw new IllegalArgumentException("not this many fragments: " + position);
            }
            return f;
        }
    }
    

    如您所见,此寻呼机处理两个“页面”,左侧显示销售数据。正确的允许用户选择哪个部门。这些是以getItem方法返回的。适当的片段(您可以使用已有的任何片段,无需修改)。

    正如预期的那样,您可以通过1)创建一个实例化MyFragmentPagerAdapter的对象,以及2)将适配器设置为您刚刚实例化的类ViewPager对象来加入所有这些。如您所见,其getItem方法将为寻呼机“提供”您需要的所有片段。例如:

    mFragmentPagerAdapter = new MyFragmentPagerAdapter(getSupportFragmentManager());
    mViewPager.setAdapter(mFragmentPagerAdapter);
    

    当然,还有其他事情要做,比如自己创建标签按钮并加入所有这些以进行交叉通信。我建议查看从GitHub下载的ViewPagerExtensions ZIP文件中提供的示例。这只是您评论中问题的答案。正如您所看到的,使用此库的代码并不多。

答案 2 :(得分:3)

当切换回之前选择的标签时,很多例子通常会抛出异常(IllegalStateException: The specified child already has a parent. You must call removeView() on the child's parent first)。

我喜欢使用FragmentStatePageAdapter而不是普通的FragmentPageAdapter来调整Google的示例,它将为您替换片段并解决此错误。通常这会破坏它认为可以移除的碎片以节省空间;如果您想要始终保留片段​​,请使用空块覆盖destroyItem(ViewGroup, int, Object)

以下是一个例子:

public class ActionBarTabs extends SherlockFragmentActivity {
    CustomViewPager mViewPager;
    TabsAdapter mTabsAdapter;
    TextView tabCenter;
    TextView tabText;
    ActionBar mActionBar;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
            WindowManager.LayoutParams.FLAG_FULLSCREEN);

        mViewPager = new CustomViewPager(this);
        mViewPager.setId(R.id.pager);

        setContentView(mViewPager);
        mActionBar = getSupportActionBar();
        mActionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
        mActionBar.setDisplayOptions(0, ActionBar.DISPLAY_SHOW_TITLE);

        mTabsAdapter = new TabsAdapter(this, mViewPager);

        mTabsAdapter.addTab(mActionBar.newTab().setText("Page 1"), 
            YOURFRAGMENT_A.class, null);
        mTabsAdapter.addTab(mActionBar.newTab().setText("Page 2"), 
            YOURFRAGMENT_B.class, null);
        mTabsAdapter.addTab(mActionBar.newTab().setText("Page 3"), 
            YOURFRAGMENT_C.class, null);
    }

    public static class TabsAdapter extends FragmentPagerAdapter
            implements ActionBar.TabListener, ViewPager.OnPageChangeListener {
        private final Context mContext;
        private final ActionBar mActionBar;
        private final ViewPager mViewPager;
        private final ArrayList<TabInfo> mTabs = new ArrayList<TabInfo>();

        static final class TabInfo {
            private final Class<?> clss;
            private final Bundle args;

            TabInfo(Class<?> _class, Bundle _args) {
                clss = _class;
                args = _args;
            }
        }

        public TabsAdapter(SherlockFragmentActivity activity, ViewPager pager) {
            super(activity.getSupportFragmentManager());
            mContext = activity;
            mActionBar = activity.getSupportActionBar();
            mViewPager = pager;
            mViewPager.setAdapter(this);
            mViewPager.setOnPageChangeListener(this);
        }

        public void addTab(ActionBar.Tab tab, Class<?> clss, Bundle args) {
            TabInfo info = new TabInfo(clss, args);
            tab.setTag(info);
            tab.setTabListener(this);
            mTabs.add(info);
            mActionBar.addTab(tab);
            notifyDataSetChanged();
        }

        @Override
        public int getCount() {
            return mTabs.size();
        }

        @Override
        public Fragment getItem(int position) {
            TabInfo info = mTabs.get(position);
            return Fragment.instantiate(mContext,
                info.clss.getName(), info.args);
        }

        @Override
        public void onPageScrolled(int position, float positionOffset,
            int positionOffsetPixels) {
        }

        @Override
        public void onPageSelected(int position) {
            mActionBar.setSelectedNavigationItem(position);
        }

        @Override
        public void onPageScrollStateChanged(int state) {
        }

        @Override
        public void onTabSelected(Tab tab, FragmentTransaction ft) {
            Object tag = tab.getTag();
            for (int i = 0; i < mTabs.size(); i++) {
                if (mTabs.get(i) == tag) {
                    mViewPager.setCurrentItem(i);
                }
            }
        }

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

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

与ABS配合良好,实施起来相对简单。