Фрагмент внутри фрагмента
Мне нужна помощь по работе с фрагментом внутри фрагмента, на самом деле я столкнулся с проблемой при нажатии кнопки назад. На главном экране приложения есть кнопки, и нажатие на каждом представлении кнопок заменяет новым фрагментом (и этот фрагмент содержится внутри другого фрагмента), динамическое добавление/замена фрагмента работает нормально, при нажатии замененного фрагмента button1 происходит то же самое при нажатии кнопки, но если я нажимаю кнопка снова получила исключение:
"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>
Надеюсь, я попытался очистить мою проблему.
Ответы
Ответ 1
AFAIK, фрагменты не могут содержать другие фрагменты.
UPDATE
С текущими версиями пакета поддержки Android - или родными фрагментами на уровне API 17 и выше - вы можете вставлять фрагменты с помощью getChildFragmentManager()
. Обратите внимание, что это означает, что вам необходимо использовать версию пакета поддержки Android Support на уровнях API 11-16, поскольку, несмотря на то, что на этих устройствах имеется родная версия фрагментов, эта версия не имеет getChildFragmentManager()
.
Ответ 2
Начиная с Android 4.2 (API 17) становятся доступными вложенные фрагменты http://developer.android.com/about/versions/android-4.2.html#NestedFragments
Чтобы разместить фрагмент внутри другого фрагмента, используйте getChildFragmentManager()
Это также доступно в библиотеке поддержки!
Ответ 3
![введите описание изображения здесь]()
Мне нужен еще какой-то контекст, поэтому я сделал пример, чтобы показать, как это делается. Самое полезное, что я прочитал во время подготовки, было следующим:
Деятельность
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.
Ответ 4
Фрагменты могут быть добавлены в другие фрагменты, но тогда вам придется удалять его из родительского фрагмента каждый раз, когда onDestroyView()
метод родительского фрагмента onDestroyView()
. И снова добавьте его в метод родительского фрагмента 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();
}
Ответ 5
Я решил эту проблему. Вы можете использовать библиотеку поддержки и 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.S. Это уродливый код для теста, но он улучшает его возможность.
P.P.S Внутренний фрагмент ChildFragmentManager
должен быть передан в ViewPagerAdapter
Ответ 6
вы можете использовать функцию 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
Вы можете добавить FrameLayout
к фрагменту и заменить его другим фрагментом при его инициализации.
Таким образом, вы можете считать, что другой фрагмент находится внутри первого фрагмента.
Ответ 8
Используйте getChildFragmentManager(), перейдите по ссылке:
Вложенные фрагменты
Ответ 9
Ничего сложного. Мы не можем использовать getFragmentManager()
здесь. Для использования фрагментов внутри фрагмента мы используем getChildFragmentManager()
. Отдых будет таким же.
Ответ 10
По MapFragment
команды Android, поддержка MapFragment
отсутствует с MapFragment
Android 3.0. Здесь больше информации о проблеме. Но что вы можете сделать, создав Фрагмент, который возвращает MapActivity
. Вот пример кода. Благодаря Иназарук:
Как это устроено :
- MainFragmentActivity - это действие, которое расширяет
FragmentActivity
и содержит два MapFragment. - MyMapActivity расширяет MapActivity и имеет
MapView.
-
LocalActivityManagerFragment
содержит LocalActivityManager. - MyMapFragment расширяет
LocalActivityManagerFragment
и с помощью TabHost
создает внутренний экземпляр MyMapActivity.
Если у вас есть какие-либо сомнения, пожалуйста, дайте мне знать
Ответ 11
Вкратце во вложенном фрагменте вложенные элементы поддерживаются только в том случае, если они сгенерированы программно! Таким образом, в схеме компоновки xml не поддерживается макет вложенных фрагментов!
Ответ 12
Это может помочь тем, кто работает на Kotlin, вы можете использовать функцию расширения, поэтому создайте файл kotlin, скажем "util.kt" и добавьте этот фрагмент кода
fun Fragment.addChildFragment(fragment: Fragment, frameId: Int) {
val transaction = childFragmentManager.beginTransaction()
transaction.replace(frameId, fragment).commit()
}
Допустим, это класс ребенка
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)
...
}
...
}
Теперь вы можете добавить детей к фрагменту отца, как это
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 - это идентификатор макета, который будет содержать фрагмент. Конечно, этот lyout находится внутри родительского фрагмента. Вот пример
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>
Ответ 13
Привет, я решил эту проблему, поставив каждый фрагмент в отдельный макет. И я сделал только что связанный макет видимым и сделал видимость остальных.
Я имею в виду:
<?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. Вы можете контролировать видимость вашего фрагмента при щелчке. Вы можете сделать связанный макет видимым, а остальные ушли, а Fragment Manager - взять фрагмент. Этот подход сработал у меня. И поскольку вид, который имеет видимость: ушел, невидим, и для пространства макета не требуется места. Я думаю, что этот подход не вызывает проблем с пространством.
P.S: Я просто попытался объяснить, что мой код решения может иметь синтаксические ошибки или незавершенную структуру.