web-dev-qa-db-ja.com

フラグメント内部フラグメント

フラグメント内のフラグメントの操作に関するヘルプが必要です。実際、戻るボタンを押すと問題が発生します。アプリケーションのメイン画面にはボタンがあり、各ボタンビューを押すと新しいフラグメントに置き換えられ(そのフラグメントは別のフラグメント内に含まれます)、フラグメントを動的に追加/置換することはうまく機能しますもう一度ボタンを押すと、例外が発生します。

"Duplicate id 0x7f05000a, tag null, or parent id 0x7f050009 with
another fragment for com........ fragmentname"

フラグメントまたは内部フラグメントがすでに追加されていることを意味し、私はそれらを再度追加しようとしています。誰もがフラグメントの内部でフラグメントを操作し、サポートのおかげで問題なく前後に移動する方法を知っています。

MainActivity。フラグメントが動的に追加および置換されます。

public class FragmentInsideFragmentTestActivity extends Activity {

    private Button button1;
    private Button button2;
    private Button button3;
    private Button button4;


    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        button1 =(Button) this.findViewById(R.id.button1);
        button1.setOnClickListener(new View.OnClickListener() {
           public void onClick(View view) {
               onButtonClick(view);
            }
        });

        button2 =(Button) this.findViewById(R.id.button2);
        button2.setOnClickListener(new View.OnClickListener() {
            public void onClick(View view) {
                onButtonClick(view);
            }
        });

        button3 =(Button) this.findViewById(R.id.button3);
        button3.setOnClickListener(new View.OnClickListener() {
            public void onClick(View view) {
               onButtonClick(view);
            }
        });

        button4 =(Button) this.findViewById(R.id.button4);
        button4.setOnClickListener(new View.OnClickListener() {
           public void onClick(View view) {
               onButtonClick(view);
           }
        });
    }

    public void onButtonClick(View v) {
        Fragment fg;

        switch (v.getId()) {
           case R.id.button1:
                   fg=FirstFragment.newInstance();
                   replaceFragment(fg);
                   break;
           case R.id.button2:
                   fg=SecondFragment.newInstance();
                   replaceFragment(fg);
                   break;
           case R.id.button3:
                   fg=FirstFragment.newInstance();
                   replaceFragment(fg);
                   break;
           case R.id.button4:
                   fg=SecondFragment.newInstance();
                   replaceFragment(fg);
                   break;
        }
    }

    private void replaceFragment(Fragment newFragment) {
       FragmentTransaction trasection = getFragmentManager().beginTransaction();

        if(!newFragment.isAdded()) {
            try {
                //FragmentTransaction trasection =
                getFragmentManager().beginTransaction();
                trasection.replace(R.id.linearLayout2, newFragment);
                trasection.addToBackStack(null);
                trasection.commit();
            } catch (Exception e) {
                // TODO: handle exception
                // AppConstants.printLog(e.getMessage());
            } else {
                trasection.show(newFragment);
            }
        }
    }

レイアウトは次のとおりです。main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:layout_width="fill_parent"
    Android:layout_height="fill_parent"
    Android:orientation="vertical">

    <LinearLayout
        Android:id="@+id/linearLayout1"
        Android:layout_width="match_parent"
        Android:layout_height="wrap_content"
        Android:orientation="horizontal">

        <Button
            Android:id="@+id/button1"
            Android:layout_width="wrap_content"
            Android:layout_height="wrap_content"
            Android:text="Button1" />

        <Button
            Android:id="@+id/button2"
            Android:text="Button2"
            Android:layout_width="wrap_content"
            Android:layout_height="wrap_content" />

        <Button
            Android:id="@+id/button3"
            Android:text="Button3"
            Android:layout_width="wrap_content"
            Android:layout_height="wrap_content" />

        <Button
            Android:id="@+id/button4"
            Android:text="Button4"
            Android:layout_width="wrap_content"
            Android:layout_height="wrap_content" />
    </LinearLayout>

    <LinearLayout
        Android:id="@+id/linearLayout2"
        Android:layout_width="match_parent"
        Android:layout_height="wrap_content"
        Android:orientation="horizontal" />
</LinearLayout>

問題を解決しようとしたことを願っています。

116
Ijaz Ahmed

知る限り、フラグメントは他のフラグメントを保持できません。


UPDATE

Androidサポートパッケージの現在のバージョン(またはAPIレベル17以上のネイティブフラグメント)では、getChildFragmentManager()を使用してフラグメントをネストできます。これは、APIレベル11〜16でフラグメントのAndroidサポートパッケージバージョンを使用する必要があることを意味することに注意してください。これらのデバイスにはネイティブバージョンのフラグメントがありますが、そのバージョンにはgetChildFragmentManager()がないためです。

269
CommonsWare

enter image description here

もう少しコンテキストが必要だったので、これがどのように行われるかを示す例を作成しました。準備中に読んだ中で最も役立つものは次のとおりです。

アクティビティ

activity_main.xml

アクティビティにFrameLayoutを追加して、親フラグメントを保持します。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
              Android:orientation="vertical"
              Android:layout_width="match_parent"
              Android:layout_height="match_parent">

    <TextView
        Android:layout_width="wrap_content"
        Android:layout_height="wrap_content"
        Android:text="Activity"/>

    <FrameLayout
        Android:id="@+id/parent_fragment_container"
        Android:layout_width="match_parent"
        Android:layout_height="200dp"/>

 </LinearLayout>

MainActivity.Java

親フラグメントをロードし、フラグメントリスナーを実装します。 ( フラグメント通信 を参照してください。)

import Android.support.v4.app.FragmentTransaction;
import Android.support.v7.app.AppCompatActivity;

public class MainActivity extends AppCompatActivity implements ParentFragment.OnFragmentInteractionListener, ChildFragment.OnFragmentInteractionListener {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // Begin the transaction
        FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
        ft.replace(R.id.parent_fragment_container, new ParentFragment());
        ft.commit();
    }

    @Override
    public void messageFromParentFragment(Uri uri) {
        Log.i("TAG", "received communication from parent fragment");
    }

    @Override
    public void messageFromChildFragment(Uri uri) {
        Log.i("TAG", "received communication from child fragment");
    }
}

親フラグメント

fragment_parent.xml

子フラグメント用に別のFrameLayoutコンテナーを追加します。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
              Android:orientation="vertical"
              Android:layout_width="match_parent"
              Android:layout_height="match_parent"
              Android:layout_margin="20dp"
              Android:background="#91d0c2">

    <TextView
        Android:layout_width="wrap_content"
        Android:layout_height="wrap_content"
        Android:text="Parent fragment"/>

    <FrameLayout
        Android:id="@+id/child_fragment_container"
        Android:layout_width="match_parent"
        Android:layout_height="match_parent">

    </FrameLayout>

</LinearLayout>

ParentFragment.Java

getChildFragmentManageronViewCreatedを使用して、子フラグメントをセットアップします。

import Android.support.v4.app.Fragment;
import Android.support.v4.app.FragmentTransaction;    

public class ParentFragment extends Fragment {

    private OnFragmentInteractionListener mListener;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
        return inflater.inflate(R.layout.fragment_parent, container, false);
    }

    @Override
    public void onViewCreated(View view, Bundle savedInstanceState) {
        Fragment childFragment = new ChildFragment();
        FragmentTransaction transaction = getChildFragmentManager().beginTransaction();
        transaction.replace(R.id.child_fragment_container, childFragment).commit();
    }


    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        if (context instanceof OnFragmentInteractionListener) {
            mListener = (OnFragmentInteractionListener) context;
        } else {
            throw new RuntimeException(context.toString()
                    + " must implement OnFragmentInteractionListener");
        }
    }

    @Override
    public void onDetach() {
        super.onDetach();
        mListener = null;
    }

    public interface OnFragmentInteractionListener {
        // TODO: Update argument type and name
        void messageFromParentFragment(Uri uri);
    }
}

子の断片

fragment_child.xml

ここには特別なものはありません。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
              Android:orientation="vertical"
              Android:layout_width="match_parent"
              Android:layout_height="match_parent"
              Android:layout_margin="20dp"
              Android:background="#f1ff91">

    <TextView
        Android:layout_width="wrap_content"
        Android:layout_height="wrap_content"
        Android:text="Child fragment"/>
</LinearLayout>

ChildFragment.Java

ここでも特別なことは何もありません。

import Android.support.v4.app.Fragment;

public class ChildFragment extends Fragment {

    private OnFragmentInteractionListener mListener;


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        return inflater.inflate(R.layout.fragment_child, container, false);
    }

    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        if (context instanceof OnFragmentInteractionListener) {
            mListener = (OnFragmentInteractionListener) context;
        } else {
            throw new RuntimeException(context.toString()
                    + " must implement OnFragmentInteractionListener");
        }
    }

    @Override
    public void onDetach() {
        super.onDetach();
        mListener = null;
    }


    public interface OnFragmentInteractionListener {
        // TODO: Update argument type and name
        void messageFromChildFragment(Uri uri);
    }
}

ノート

62
Suragch

Android 4.2(API 17)ネストされたフラグメントが利用可能になるため http://developer.Android.com/about/versions/Android-4.2.html#NestedFragments

フラグメントを他のフラグメント内に配置するには、getChildFragmentManager()を使用します

サポートライブラリでも利用可能です!

62
Nik

フラグメントは他のフラグメント内に追加できますが、親フラグメントのonDestroyView()メソッドが呼び出されるたびに、フラグメントを親フラグメントから削除する必要があります。そして再度、Parent FragmentのonCreateView()メソッドに追加します。

次のようにします:

@Override
    public void onDestroyView()
    {
                FragmentManager mFragmentMgr= getFragmentManager();
        FragmentTransaction mTransaction = mFragmentMgr.beginTransaction();
                Fragment childFragment =mFragmentMgr.findFragmentByTag("qa_fragment")
        mTransaction.remove(childFragment);
        mTransaction.commit();
        super.onDestroyView();
    }
11
Napolean

この問題を解決しました。サポートライブラリとViewPagerを使用できます。ジェスチャーによるスワイプが不要な場合は、スワイプを無効にできます。だからここに私のソリューションを改善するためのいくつかのコードがあります:

public class TestFragment extends Fragment{
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View v = inflater.inflate(R.layout.frag, container, false);
    final ArrayList<Fragment> list = new ArrayList<Fragment>();

    list.add(new TrFrag());
    list.add(new TrFrag());
    list.add(new TrFrag());

    ViewPager pager = (ViewPager) v.findViewById(R.id.pager);
    pager.setAdapter(new FragmentPagerAdapter(getChildFragmentManager()) {
        @Override
        public Fragment getItem(int i) {
            return list.get(i);
        }

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

追記:テスト用のいコードですが、可能性が向上しています。

P.P.S フラグメントChildFragmentManagerの内部をViewPagerAdapterに渡す必要があります

8
Vetalll

getChildFragmentManager()関数を使用できます。

例:

親フラグメント:

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {

    rootView = inflater.inflate(R.layout.parent_fragment, container,
            false);


    }

    //child fragment 
    FragmentManager childFragMan = getChildFragmentManager();
    FragmentTransaction childFragTrans = childFragMan.beginTransaction();
    ChildFragment fragB = new ChildFragment ();
    childFragTrans.add(R.id.FRAGMENT_PLACEHOLDER, fragB);
    childFragTrans.addToBackStack("B");
    childFragTrans.commit();        


    return rootView;
}

親レイアウト(parent_fragment.xml):

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:orientation="vertical" Android:layout_width="match_parent"
    Android:layout_height="match_parent"
    Android:background="@Android:color/white">



    <FrameLayout
        Android:id="@+id/FRAGMENT_PLACEHOLDER"
        Android:layout_width="match_parent"
        Android:layout_height="match_parent"/>




</LinearLayout>

子フラグメント:

public class ChildFragment extends Fragment implements View.OnClickListener{

    View v ;
    @Override
    public View onCreateView(LayoutInflater inflater,
                             @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        View rootView = inflater.inflate(R.layout.child_fragment, container, false);


        v = rootView;


        return rootView;
    }



    @Override
    public void onClick(View view) {


    }


} 
7
S.M_Emamian

FrameLayoutをフラグメントに追加し、初期化時に別のフラグメントに置き換えることができます。

この方法では、他のフラグメントが最初のフラグメント内にあると考えることができます。

3
LoveBigPizza

GetChildFragmentManager()を使用して、リンクをたどってください: Nested Fragment

3

複雑なことは何もありません。ここではgetFragmentManager()を使用できません。フラグメントフラグメント内を使用するには、getChildFragmentManager()を使用します。残りは同じになります。

2
Wijay Sharma

これは、Kotlinで作業する人に役立つかもしれません。拡張機能を使用して、kotlinファイルを作成できます。 "util。 kt "このコードを追加します

fun Fragment.addChildFragment(fragment: Fragment, frameId: Int) {

    val transaction = childFragmentManager.beginTransaction()
    transaction.replace(frameId, fragment).commit()
}

これがchildのクラスだとしましょう

class InputFieldPresentation: Fragment()
{
    var views: View? = null
    override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?,
                              savedInstanceState: Bundle?): View? {
        views = inflater!!.inflate(R.layout.input_field_frag, container, false)
        return views
    }

    override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        ...
    }
    ...
}

これで、father fragmentに子を追加できます

 FatherPresentation:Fragment()
{
  ...

    override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        val fieldFragment= InputFieldPresentation()
        addChildFragment(fieldFragment,R.id.fragmet_field)
    }
...
}

ここで、R.id.fragmet_fieldは、フラグメントを含むレイアウトのIDです。このLyoutはもちろん、fatherフラグメント内にあります。ここに例があります

father_fragment.xml

<LinearLayout Android:layout_width="match_parent"
    Android:layout_height="match_parent"
    Android:orientation="vertical"
    xmlns:Android="http://schemas.Android.com/apk/res/Android"
    >

    ...

        <LinearLayout
            Android:layout_width="match_parent"
            Android:layout_height="wrap_content"
            Android:gravity="center"
            Android:id="@+id/fragmet_field"
            Android:orientation="vertical"
            >
        </LinearLayout>
    ...

    </LinearLayout>
1
DINA TAKLIT

現在、ネストされたフラグメントでは、ネストされたフラグメントは、プログラムで生成された場合にのみサポートされます!そのため、現時点では、XMLレイアウトスキームではネストされたフラグメントレイアウトはサポートされていません。

1
Andrea Bellitto

MapFragmentはサポートされていません。Android 3.0以降、Androidチームは作業中です。 こちら 問題に関する詳細情報しかし、MapActivityを返すFragmentを作成することで何ができますか。 ここ はコード例です。 inazarukに感謝します。

仕組み:

  • MainFragmentActivityは、FragmentActivityを拡張し、2つのMapFragmentsをホストするアクティビティです。
  • MyMapActivityはMapActivityを拡張し、MapView.を持っています
  • LocalActivityManagerFragmentはLocalActivityManagerをホストします。
  • MyMapFragmentはLocalActivityManagerFragmentを拡張し、TabHostの助けを借りてMyMapActivityの内部インスタンスを作成します。

疑問がある場合はお知らせください

1
rallat

こんにちは、フラグメントごとに個別のレイアウトを配置することでこの問題を解決しました。関連するレイアウトだけを表示し、他の表示を消しました。

というのは:

<?xml version="1.0" encoding="utf-8"?>
 <LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
 Android:orientation="vertical"
 Android:layout_width="fill_parent"
 Android:layout_height="fill_parent">

     <LinearLayout Android:id="@+id/linearLayout1"
           Android:layout_width="match_parent"
           Android:layout_height="wrap_content"
           Android:orientation="horizontal">
           <Button Android:layout_width="wrap_content"
                   Android:id="@+id/button1"
                   Android:layout_height="wrap_content"
                   Android:text="Button1"></Button>
           <Button Android:text="Button2"
                   Android:id="@+id/button2"
                   Android:layout_width="wrap_content"
                   Android:layout_height="wrap_content"></Button>
           <Button Android:text="Button3"
                   Android:id="@+id/button3"
                   Android:layout_width="wrap_content"
                   Android:layout_height="wrap_content"></Button>
           <Button Android:text="Button4"
                   Android:id="@+id/button4"
                   Android:layout_width="wrap_content"
                   Android:layout_height="wrap_content"></Button>
   </LinearLayout>
   <LinearLayout Android:layout_width="full_screen"
              Android:layout_height="0dp"
              Android:layout_weight="1"
              Android:id="action_For_Button1"
              Android:visibility="visible">
                    <Fragment Android:layout_width="full_screen"
                              Android:layout_height="full_screen"
                              Android:id="fragment1"
                                        .
                                        .
                                        .
             / >
     </LinearLayout>

     <LinearLayout Android:layout_width="full_screen"
              Android:layout_height="0dp"
              Android:id="action_For_Button1"
              Android:layout_weight="1"
              Android:visibility="gone">
                       <Fragment Android:layout_width="full_screen"
                                 Android:layout_height="full_screen"
                                 Android:id="fragment2"
                                           .
                                           .
                                           .
             / >
        </LinearLayout>
                     .
                     .
                     .
        </LinearLayout>

ボタン1がクリックされたときにページを開くと仮定しました。クリックアクションでフラグメントの可視性を制御できます。関連するレイアウトを表示し、他のレイアウトを削除し、フラグメントマネージャによってフラグメントを取得できます。また、visibility:goneを持つviewは非表示であり、レイアウトのためにスペースをとらないため、このアプローチはスペースの問題を引き起こさないと思います。

追伸:ソリューションコードに構文ミスや未完成の構造がある可能性があることを説明しようとしました。

0
Sevil