it-swarm.com.ru

Фрагмент onResume () и onPause () не вызывается в backstack

У меня есть несколько фрагментов внутри деятельности. По нажатию кнопки я начинаю новый фрагмент, добавляя его в backstack. Естественно, я ожидал вызова метода onPause() текущего фрагмента и onResume() нового фрагмента. Ну, это не происходит.

LoginFragment.Java

public class LoginFragment extends Fragment{
  @Override
  public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
      final View view  =   inflater.inflate(R.layout.login_fragment, container, false);
      final FragmentManager mFragmentmanager =  getFragmentManager();

      Button btnHome  = (Button)view.findViewById(R.id.home_btn);
      btnHome.setOnClickListener(new View.OnClickListener() {
        public void onClick(View view){
           HomeFragment fragment    = new HomeFragment();
           FragmentTransaction ft2   =  mFragmentmanager.beginTransaction();
           ft2.setCustomAnimations(R.anim.slide_right, R.anim.slide_out_left
                    , R.anim.slide_left, R.anim.slide_out_right);
           ft2.replace(R.id.middle_fragment, fragment);
           ft2.addToBackStack(""); 
           ft2.commit();    
         }
      });
  }

  @Override
  public void onResume() {
     Log.e("DEBUG", "onResume of LoginFragment");
     super.onResume();
  }

  @Override
  public void onPause() {
    Log.e("DEBUG", "OnPause of loginFragment");
    super.onPause();
  }
}

HomeFragment.Java

public class HomeFragment extends Fragment{
  @Override
  public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
     final View view  =   inflater.inflate(R.layout.login_fragment, container, false);
  }

  @Override
  public void onResume() {
     Log.e("DEBUG", "onResume of HomeFragment");
     super.onResume();
  }

  @Override
  public void onPause() {
     Log.e("DEBUG", "OnPause of HomeFragment");
     super.onPause();
  }
}

То, что я ожидал, было,

  1. При нажатии кнопки LoginFragment заменяется на HomeFragment , onPause() из LoginFragment и onResume() of HomeFragment вызывается
  2. Когда нажата кнопка «Назад», HomeFragment выскакивает и LoginFragment .... Видел, и onPause() из HomeFragment и onResume() из LoginFragment Вызывается.

Что я получаю,

  1. Когда кнопка нажата, HomeFragment корректно заменяет LoginFragment , вызывается onResume () of HomeFragment , но onPause () Of LoginFragment никогда не вызывается ,.
  2. При обратном нажатии HomeFragment правильно появляется, чтобы показать LoginFragment , onPause () of HomeFragment вызывается, но onResume () Of LoginFragment никогда не получает называется.

Это нормальное поведение? Почему onResume() of LoginFragment не вызывается, когда я нажимаю кнопку возврата.

167
Krishnabhadra

Фрагменты onResume() или onPause() будут вызываться только при вызове Activity onResume() или onPause() . Они тесно связаны с Activity.

Прочитайте раздел Обработка жизненного цикла фрагмента этой статьи .

157
Sagar Waghmare
  • Поскольку вы использовали ft2.replace(), вызывается метод FragmentTransaction.remove() И Loginfragment будет удален. Обратитесь к это . Поэтому вместо onStop() будет вызываться onPause() из LoginFragment. (Поскольку новый Фрагмент полностью заменяет старый).
  • Но так как вы также Использовали ft2.addtobackstack(), состояние Loginfragment будет сохранено в виде Как пакет, и при нажатии кнопки «Назад» в HomeFragment, onViewStateRestored() будет вызываться onStart() из LoginFragment. Так что в конечном итоге onResume() не будет вызван.
18
Gopal

Если вы действительно хотите заменить фрагмент внутри другого фрагмента, вы должны использовать Nested Fragments .

В вашем коде вы должны заменить

final FragmentManager mFragmentmanager =  getFragmentManager();

с

final FragmentManager mFragmentmanager =  getChildFragmentManager();
8
Bersh

Вот моя более надежная версия ответа Гора (использование fragments.size () ненадежно из-за того, что размер не уменьшается после извлечения фрагмента)

getFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
        @Override
        public void onBackStackChanged() {
            if (getFragmentManager() != null) {

                Fragment topFrag = NavigationHelper.getCurrentTopFragment(getFragmentManager());

                if (topFrag != null) {
                    if (topFrag instanceof YourFragment) {
                        //This fragment is being shown. 
                    } else {
                        //Navigating away from this fragment. 
                    }
                }
            }
        }
    });

И метод getCurrentTopFragment:

public static Fragment getCurrentTopFragment(FragmentManager fm) {
    int stackCount = fm.getBackStackEntryCount();

    if (stackCount > 0) {
        FragmentManager.BackStackEntry backEntry = fm.getBackStackEntryAt(stackCount-1);
        return  fm.findFragmentByTag(backEntry.getName());
    } else {
        List<Fragment> fragments = fm.getFragments();
        if (fragments != null && fragments.size()>0) {
            for (Fragment f: fragments) {
                if (f != null && !f.isHidden()) {
                    return f;
                }
            }
        }
    }
    return null;
}
6
aaronmarino
getFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
        @Override
        public void onBackStackChanged() {
            List<Fragment> fragments = getFragmentManager().getFragments();
            if (fragments.size() > 0 && fragments.get(fragments.size() - 1) instanceof YoureFragment){
                //todo if fragment visible
            } else {
                //todo if fragment invisible
            }

        }
    });

но будьте осторожны, если видно более одного фрагмента

2
Gor

У меня есть код, очень похожий на ваш, и если он работает onPause () и onResume (). При изменении фрагмента эти функции активируются соответственно.

Код в фрагменте:

 @Override
public void onResume() {
    super.onResume();
    sensorManager.registerListener(this, proximidad, SensorManager.SENSOR_DELAY_NORMAL);
    sensorManager.registerListener(this, brillo, SensorManager.SENSOR_DELAY_NORMAL);
    Log.e("Frontales","resume");
}

@Override
public void onPause() {
    super.onPause();
    sensorManager.unregisterListener(this);
    Log.e("Frontales","Pause");

}

Журнал при изменении фрагмента:

05-19 22:28:54.284 2371-2371/madi.cajaherramientas E/Frontales: resume
05-19 22:28:57.002 2371-2371/madi.cajaherramientas E/Frontales: Pause
05-19 22:28:58.697 2371-2371/madi.cajaherramientas E/Frontales: resume
05-19 22:29:00.840 2371-2371/madi.cajaherramientas E/Frontales: Pause
05-19 22:29:02.248 2371-2371/madi.cajaherramientas E/Frontales: resume
05-19 22:29:03.718 2371-2371/madi.cajaherramientas E/Frontales: Pause

Фрагмент onCreateView:

View rootView;
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

    rootView = inflater.inflate(R.layout.activity_proximidad, container, false);
    ButterKnife.bind(this,rootView);
    inflar();
    setTextos();
    return rootView;
}

Действие при возврате импульса (в упражнении, в котором я загружаю фрагмент):

@Override
public void onBackPressed() {

    int count = getFragmentManager().getBackStackEntryCount();

    if (count == 0) {
        super.onBackPressed();

    } else {
        getFragmentManager().popBackStack();
    }

 }
2
PepitoGrillo

Вы можете попробовать это,

Шаг 1: переопределите метод Tabselected в вашей деятельности

@Override
public void onTabSelected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
    // When the given tab is selected, switch to the corresponding page in
    // the ViewPager.
    try {
    if(MyEventsFragment!=null && tab.getPosition()==3)
    {
        MyEvents.fragmentChanged();
    }
    }
    catch (Exception e)
    {

    }
    mViewPager.setCurrentItem(tab.getPosition());
}

Шаг 2: Используя статический метод, делайте что хотите в своем фрагменте,

public static void fragmentChanged()
{
    Toast.makeText(actvity, "Fragment Changed", Toast.LENGTH_SHORT).show();
}
1
Manoj Kumar

Вы просто не можете добавить фрагмент к фрагменту. Это должно произойти в FragmentActivity. Я предполагаю, что вы создаете LoginFragment в FragmentActivity, поэтому для того, чтобы это работало, вам нужно добавить HomeFragment через FragmentActivity при закрытии входа в систему.

В общем, вам нужен класс FragmentActivity, из которого вы добавляете каждый фрагмент в FragmentManager. Это невозможно сделать это внутри класса Fragment.

1
Nickolaus

Если вы добавите фрагмент в XML, вы не сможете поменять их динамически. То, что происходит, они чрезмерно, так что события не происходят, как можно было бы ожидать. Проблема задокументирована в этом вопросе. FragmenManager заменяет делает оверлей

Превратите middle_fragment в FrameLayout, и загрузите его, как показано ниже, и ваши события сработают.

getFragmentManager().beginTransation().
    add(R.id.middle_fragment, new MiddleFragment()).commit();
1
Dustin

Что я делаю в дочернем фрагменте:

@Override
public void onDetach() {
   super.onDetach();
   ParentFragment pf = (ParentFragment) this.getParentFragment();
   pf.onResume();
}

А затем переопределить onResume на ParentFragment

1
Guillermo De Juan

Фрагмент всегда должен быть встроен в действие, и жизненный цикл фрагмента напрямую зависит от жизненного цикла хоста. Например, когда действие приостановлено, все фрагменты в нем, а когда действие уничтожено, все фрагменты

0
Xar E Ahmer

Метод onPause() работает в классе деятельности, который вы можете использовать:

public void onDestroyView(){
super.onDestroyView    
}

для той же цели ..

0
Vikrant Kaloniya

Хотя с другим кодом я столкнулся с той же проблемой, что и OP, потому что изначально использовал 

fm.beginTransaction()
            .add(R.id.fragment_container_main, fragment)
            .addToBackStack(null)
            .commit();

вместо

fm.beginTransaction()
                .replace(R.id.fragment_container_main, fragment)
                .addToBackStack(null)
                .commit();

При «замене» первый фрагмент воссоздается, когда вы возвращаетесь из второго фрагмента, и поэтому также вызывается onResume (). 

0
Greg Holst

Вызов ft.replace должен вызвать onPause (замещенного фрагмента) и onResume (замещающего фрагмента).

Я заметил, что ваш код раздувает login_fragment на домашнем фрагменте, а также не возвращает представления в onCreateView. Если это опечатки, можете ли вы показать, как эти фрагменты вызываются из вашей деятельности?

0
David Refaeli

Исходя из ответа @Gor я написал подобное в Kotlin. Поместите этот код в onCreate() действия. Это работает для одного видимого фрагмента. Если у вас есть ViewPager с фрагментами, он вызовет фрагмент ViewPager, а не предыдущий.

supportFragmentManager.addOnBackStackChangedListener {
    supportFragmentManager.fragments.lastOrNull()?.onResume()
}

После прочтения https://medium.com/@elye.project/puzzle-fragment-stack-pop-cause-issue-on-toolbar-8b947c5c07c6 я понял, что во многих ситуациях было бы лучше присоединить новые фрагменты с replace, а не add. Поэтому необходимость в onResume в некоторых случаях исчезнет.

0
CoolMind

Выполните следующие шаги, и вы получите нужный ответ

1- Для обоих фрагментов создайте новый абстрактный родительский.
2- Добавьте пользовательский абстрактный метод, который должен быть реализован обоими.
3- Вызовите его из текущего экземпляра перед его заменой вторым.

0
Mostafa Magdy

Я использую в своей деятельности -KOTLIN

supportFragmentManager.addOnBackStackChangedListener {
                val f = supportFragmentManager.findFragmentById(R.id.fragment_container)

                if (f?.tag == "MyFragment")
                {
                    //doSomething
                }
            }
0
Álvaro Agüero