【问题标题】:insert and remove fragments into viewpager properly正确地将片段插入和删除到 viewpager
【发布时间】:2012-04-24 06:12:15
【问题描述】:

我正在制作一个带有 viewpager 和 Fragment 的 android 应用程序。我想选择动态地向寻呼机添加或删除片段页面。我有一个自定义 FragmentPagerAdapter:

public class MyAdapter extends FragmentPagerAdapter implements TitleProvider{

    protected final List<PageFragment> fragments;

    /**
    * @param fm
    * @param fragments
    */
    public MyAdapter(FragmentManager fm, List<PageFragment> fragments) {

        super(fm);
        this.fragments = fragments;
    }

    public void addItem(PageFragment f){
        fragments.add(f);
        notifyDataSetChanged();
    }

    public void addItem(int pos, PageFragment f){
        for(int i=0;i<fragments.size();i++){
            if(i>=pos){
                getSupportFragmentManager().beginTransaction().remove(getSupportFragmentManager().findFragmentByTag(getFragmentTag(i))).commit();
            }
        }
        fragments.add(pos,f);
        notifyDataSetChanged();
        pager.setAdapter(this);
        pager.setCurrentItem(pos);

    }

    public void removeItem(int pos){
        getSupportFragmentManager().beginTransaction().remove(getSupportFragmentManager().findFragmentByTag(getFragmentTag(pos))).commit();
        for(int i=0;i<fragments.size();i++){
            if(i>pos){
                getSupportFragmentManager().beginTransaction().remove(getSupportFragmentManager().findFragmentByTag(getFragmentTag(i))).commit();
            }
        }
        fragments.remove(pos);

        notifyDataSetChanged();
        pager.setAdapter(this);
        if(pos<fragments.size()){
            pager.setCurrentItem(pos);
        }else{
            pager.setCurrentItem(pos-1);
        }
    }



    @Override
    public Fragment getItem(int position) {
        return fragments.get(position).toFragment();
    }

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

    @Override
    public int getItemPosition(Object object) {
        return POSITION_NONE;
    }

    private String getFragmentTag(int pos){
        return "android:switcher:"+R.id.pager+":"+pos;
    }

    public String getTitle(int position) {
        String name =  fragments.get(position).getName();
        if(name.equals(""))
            return "- "+position+" -";
        return name;
    }
}

我可以删除除 0 以外的任何片段。当我尝试删除它时,我在 notifyDataSetChanged(); 上收到 NullPointerException。或在 pager.setAdapter(this);如果我注释掉通知。

当我尝试插入新页面时,我也收到了 NullPointerException。当我将新页面添加到列表末尾时,它工作正常。 我什至尝试在 Fragments.add(pos,f) 之后用这个来读取插入中的片段

    for(int i=0;i<fragments.size();i++){
            if(i>=pos){
                getSupportFragmentManager().beginTransaction().add(fragments.get(i).toFragment(),getFragmentTag(i-1)).commit();
            }
        }

如果我使用getFragmentTag(i-1),我又得到了空指针。仅使用i 我得到了非法状态异常,因为无法修改片段的标签。使用beginTransaction().add(pager.getId,fragments.get(i).toFragment()),它仍然是空指针...

我的问题是:我做错了什么,如何才能正确完成? (也许:当导致 nullpointerexception 时,notyfyDataSetChanged() 从哪里获取数据?)

【问题讨论】:

    标签: android fragment android-viewpager


    【解决方案1】:

    这就是我用来解决在实例化时标记片段的问题以及以前标记的片段在尝试添加或删除页面时无法在 ViewPager 中更改位置的事实,即获取异常:

    java.lang.IllegalStateException:无法更改片段的标签 MyFragment{4055f558 id=0x7f050034 android:switcher:2131034164:3}:是 android:switcher:2131034164:3 现在 android:switcher:2131034164:4

    此适配器是预先创建的所有页面列表,然后您可以使用 setEnabled(int position, boolean enabled) 禁用或启用某些页面,这些页面在 ViewPager 中隐藏或显示这些页面。

    它通过维护所有片段的内部列表但暴露给 ViewPager 并仅映射启用片段的位置来工作。

    public class DynamicFragmentPagerAdapter extends FragmentPagerAdapter {
    
        public final ArrayList<Fragment> screens = new ArrayList<Fragment>();
    
        private Context context;
        private List<AtomicBoolean> flags = new ArrayList<AtomicBoolean>();
    
        public DynamicFragmentPagerAdapter(FragmentManager fm, Context context, List<Class<?>> screens) {
            super(fm);
            this.context = context;
    
            for(Class<?> screen : screens)
                addScreen(screen, null);
    
            notifyDataSetChanged();
        }
    
        public DynamicFragmentPagerAdapter(FragmentManager fm, Context context, Map<Class<?>, Bundle> screens) {
            super(fm);
            this.context = context;
    
            for(Class<?> screen : screens.keySet())
                addScreen(screen, screens.get(screen));
    
            notifyDataSetChanged();
        }
    
        private void addScreen(Class<?> clazz, Bundle args) {
            screens.add(Fragment.instantiate(context, clazz.getName(), args));
            flags.add(new AtomicBoolean(true));
        }
    
        public boolean isEnabled(int position) {
            return flags.get(position).get();
        }
    
        public void setEnabled(int position, boolean enabled) {
            AtomicBoolean flag = flags.get(position);
            if(flag.get() != enabled) {
                flag.set(enabled);
                notifyDataSetChanged();
            }
        }
    
        @Override
        public int getCount() {
            int n = 0;
            for(AtomicBoolean flag : flags) {
                if(flag.get())
                    n++;
            }
            return n;
        }
    
        @Override
        public Fragment getItem(int position) {
            return screens.get(position);
        }
    
        @Override
        public int getItemPosition(Object object) {
            return POSITION_NONE; // To make notifyDataSetChanged() do something
        }
    
        @Override
        public void notifyDataSetChanged() {
            try {
                super.notifyDataSetChanged();
            } catch (Exception e) {
                Log.w("", e);
            }
        }
    
        private List<Fragment> getEnabledScreens() {
            List<Fragment> res = new ArrayList<Fragment>();
            for(int n = 0; n < screens.size(); n++) {
                if(flags.get(n).get())
                    res.add(screens.get(n));
            }
            return res;
        }
    
        @Override
        public Object instantiateItem(ViewGroup container, int position) {
            // We map the requested position to the position as per our screens array list
            Fragment fragment = getEnabledScreens().get(position);
            int internalPosition = screens.indexOf(fragment);
            return super.instantiateItem(container, internalPosition);
        }
    
        @Override
        public CharSequence getPageTitle(int position) {
            Fragment fragment = getEnabledScreens().get(position);
            if(fragment instanceof TitledFragment)
                return ((TitledFragment)fragment).getTitle(context);
            return super.getPageTitle(position);
        }
    
        public static interface TitledFragment {
            public String getTitle(Context context);
        }
    }
    

    【讨论】:

    • 它工作正常,但是当notifyDatasetChanged 被调用时,它会为每个Fragment 创建新视图,所以我失去了 UI 状态。我怎样才能防止这种情况发生?
    【解决方案2】:

    我有一个适用于基于非片段的适配器的解决方案。看看my post 看看是否有帮助。

    【讨论】:

    • 不推荐仅链接答案。
    【解决方案3】:

    我可以删除除 0 以外的任何片段。

    对我来说,如果在 removeItem 的 for 循环中使用 >= 条件,它就可以工作:

         if (i >= pos){
              fm.beginTransaction().remove(fragments.get(i).getFragment()).commit();
          }
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2021-09-14
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多