9

ビューページャーとフラグメントを使用して 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 も発生しました。新しいページをリストの最後に追加すると、正常に機能します。fragment.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)と、再びnullpointerになりました。使用しただけiでは、フラグメントのタグを変更できないため、 illegalstateexception が発生しました。beginTransaction().add(pager.getId,fragments.get(i).toFragment())それはまだnullpointerです...

私の質問は次のとおりです。何が間違っているのか、どうすれば適切に行うことができますか? (そしておそらく:notyfyDataSetChanged()nullpointerexceptionが発生したときにどこからデータを取得しますか?)

4

3 に答える 3

7

これは、インスタンス化されたときにフラグメントがタグ付けされる問題と、ページを追加または削除しようとしたときに、以前にタグ付けされたフラグメントがViewPager内の位置を変更できないという事実を回避するために使用したものです。つまり、例外が発生します。

java.lang.IllegalStateException:フラグメントMyFragmentのタグを変更できません{4055f558id = 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);
    }
}
于 2013-02-21T21:41:26.097 に答える
1

非フラグメント ベースのアダプターで動作するソリューションがあります。私の投稿を見て、それが役立つかどうかを確認してください。

于 2012-12-02T20:56:46.803 に答える
0

0 以外の任意のフラグメントを削除できます。

私にとっては、removeItem の for ループ内で >= 条件を使用するだけで機能します。

     if (i >= pos){
          fm.beginTransaction().remove(fragments.get(i).getFragment()).commit();
      }
于 2013-02-08T23:01:06.977 に答える