フラグメント内のフラグメントの操作に関するヘルプが必要です。実際、戻るボタンを押すと問題が発生します。アプリケーションのメイン画面にはボタンがあり、各ボタンビューを押すと新しいフラグメントに置き換えられ(そのフラグメントは別のフラグメント内に含まれます)、フラグメントを動的に追加/置換することはうまく機能しますもう一度ボタンを押すと、例外が発生します。
"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>
問題を解決しようとしたことを願っています。
知る限り、フラグメントは他のフラグメントを保持できません。
UPDATE
Androidサポートパッケージの現在のバージョン(またはAPIレベル17以上のネイティブフラグメント)では、getChildFragmentManager()
を使用してフラグメントをネストできます。これは、APIレベル11〜16でフラグメントのAndroidサポートパッケージバージョンを使用する必要があることを意味することに注意してください。これらのデバイスにはネイティブバージョンのフラグメントがありますが、そのバージョンにはgetChildFragmentManager()
がないためです。
もう少しコンテキストが必要だったので、これがどのように行われるかを示す例を作成しました。準備中に読んだ中で最も役立つものは次のとおりです。
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
getChildFragmentManager
でonViewCreated
を使用して、子フラグメントをセットアップします。
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);
}
}
Android 4.2(API 17)ネストされたフラグメントが利用可能になるため http://developer.Android.com/about/versions/Android-4.2.html#NestedFragments
フラグメントを他のフラグメント内に配置するには、getChildFragmentManager()を使用します
サポートライブラリでも利用可能です!
フラグメントは他のフラグメント内に追加できますが、親フラグメントの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();
}
この問題を解決しました。サポートライブラリと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
に渡す必要があります
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) {
}
}
FrameLayout
をフラグメントに追加し、初期化時に別のフラグメントに置き換えることができます。
この方法では、他のフラグメントが最初のフラグメント内にあると考えることができます。
GetChildFragmentManager()を使用して、リンクをたどってください: Nested Fragment
複雑なことは何もありません。ここではgetFragmentManager()
を使用できません。フラグメントフラグメント内を使用するには、getChildFragmentManager()
を使用します。残りは同じになります。
これは、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>
現在、ネストされたフラグメントでは、ネストされたフラグメントは、プログラムで生成された場合にのみサポートされます!そのため、現時点では、XMLレイアウトスキームではネストされたフラグメントレイアウトはサポートされていません。
MapFragment
はサポートされていません。Android 3.0以降、Androidチームは作業中です。 こちら 問題に関する詳細情報しかし、MapActivity
を返すFragmentを作成することで何ができますか。 ここ はコード例です。 inazarukに感謝します。
仕組み:
FragmentActivity
を拡張し、2つのMapFragmentsをホストするアクティビティです。MapView.
を持っていますLocalActivityManagerFragment
はLocalActivityManagerをホストします。LocalActivityManagerFragment
を拡張し、TabHost
の助けを借りてMyMapActivityの内部インスタンスを作成します。疑問がある場合はお知らせください
こんにちは、フラグメントごとに個別のレイアウトを配置することでこの問題を解決しました。関連するレイアウトだけを表示し、他の表示を消しました。
というのは:
<?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は非表示であり、レイアウトのためにスペースをとらないため、このアプローチはスペースの問題を引き起こさないと思います。
追伸:ソリューションコードに構文ミスや未完成の構造がある可能性があることを説明しようとしました。