onBackStackChanged () вызывается дважды при добавлении нового фрагмента в backstack

Итак, у меня есть интересная проблема. Когда я добавляю новый фрагмент программно, метод onBackStackChanged из OnBackStackChangedListener вызывается дважды, но он должен вызываться только один раз. Вот мой код деятельности:

import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentManager;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Button;

public class MainActivity extends AppCompatActivity implements FragmentManager.OnBackStackChangedListener{

    private FragmentManager fragmentManager;
    private Button button1;
    private Button button2;
    private Button button3;

    private Fragment defaultFragment;
    private Fragment previousFragment;
    private Fragment currentFragment;
    private String currentFragmentTag;

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

        button1 = (Button) findViewById(R.id.button);
        button2 = (Button) findViewById(R.id.button2);
        button3 = (Button) findViewById(R.id.button3);

        button1.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                showFragment(new FragmentOne(), FragmentOne.TAG);
            }
        });

        button2.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                showFragment(new FragmentTwo(), FragmentTwo.TAG);
            }
        });

        button3.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                showFragment(new FragmentThree(), FragmentThree.TAG);
            }
        });

        setupFragmentManager();

    }

    private void setupFragmentManager() {
        fragmentManager = getSupportFragmentManager();
        fragmentManager.addOnBackStackChangedListener(this);


        FragmentOne fragmentOne = new FragmentOne();
        defaultFragment = fragmentOne;  // fragment for default

        currentFragmentTag = FragmentOne.TAG;
        currentFragment = fragmentOne;

        fragmentManager.beginTransaction()
                .add(R.id.fragments_container, currentFragment, currentFragmentTag)
                .addToBackStack(currentFragmentTag)
                .commit();
    }

    public void showFragment(Fragment fragment, String fragmentTag) {
        previousFragment = currentFragment;
        currentFragment = fragment;
        fragmentManager.beginTransaction()
                .hide(previousFragment)
                .add(R.id.fragments_container, fragment, fragmentTag)
                .addToBackStack(fragmentTag)
                .commit();
    }

    @Override
    public void onBackStackChanged() {
        Log.d("MY_TAG" , "onBackStackChangedListener - " + fragmentManager.getBackStackEntryCount() );

        if (fragmentManager.getBackStackEntryCount() > 0) {    
            String fragmentTag = fragmentManager.getBackStackEntryAt(fragmentManager.getBackStackEntryCount() - 1).getName();
            currentFragment = fragmentManager.findFragmentByTag(fragmentTag);
        } else {
            currentFragment = defaultFragment;
        }

// some code...
    }

    @Override
    public void onBackPressed() {
        if (fragmentManager.getBackStackEntryCount() == 1) { 
            finish();
        } else {
            super.onBackPressed();
        }
    }
}

и класс фрагмент:

public class FragmentOne extends Fragment {

    public static final String TAG = FragmentOne.class.getSimpleName();

    private Context context;

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


        context = getActivity();
        View topLevelView = LayoutInflater.from(context).inflate(R.layout.fragment_layout, container, false);

        TextView textView = (TextView) topLevelView.findViewById(R.id.fragment_Text);

        textView.setText("This is FRAGMENT ONE");

        return topLevelView;
    }

}

FragmentTwo и FragmentThree идентичны FragmentOne.

и когда я запускаю мое приложение, в журнале я вижу:

onBackStackChangedListener - 1
onBackStackChangedListener - 1

после нажатия кнопки (Показать следующий фрагмент) i см.:

onBackStackChangedListener - 2
onBackStackChangedListener - 2

и еще один:

onBackStackChangedListener - 3
onBackStackChangedListener - 3

и изменение

.add(R.id.fragments_container, currentFragment, currentFragmentTag)

to

.replace(R.id.fragments_container, currentFragment, currentFragmentTag)

не решит проблему.

но это очень странно. Почему мой OnBackStackChangedListener вызывает более одного собственного метода после одного добавления фрагмента?

2 ответов


не мог поверить, что это может случиться. Создал аналогичный проект и нырнул в отладчик. Оказывается, это ошибка в новой библиотеке поддержки. Есть ошибка открыта в трекере.

переключение обратно в 25.0.0 будет работать, как ожидалось.

compile 'com.android.support:appcompat-v7:25.0.0'
compile 'com.android.support:support-v4:25.0.0'

хотя не упоминается в заметки, но в баг-трекере, что проблема исправлена в 25.4.0, это хорошо .


просто обновление. Google, наконец, выпустила версию 25.4.0 с этой проблемой исправлена.

только, теперь материал поддержки предназначен для извлечения из репозитория Maven Google:

maven { url 'https://maven.google.com' }

таким образом, вы можете начать использовать новые версии всего и удалить локальный репозиторий поддержки из диспетчера SDK.