Почему saveStateInstance не работает с ориентацией экрана? - PullRequest
0 голосов
/ 03 октября 2019

Я делаю приложение для гидов, которое состоит из одного основного упражнения и двух фрагментов. Верхний фрагмент содержит список городов, а когда вы смотрите на город, нижний фрагмент отображает описание этого города. Проблема в том, что когда я меняю ориентацию, я получаю сообщение об ошибке «Попытка вызвать виртуальный метод void android.widget.TextView.setText (java.lang.CharSequence)» для нулевой ссылки на объект »

Вотфрагмент списка городов:

        package com.example.tourguide;


        import android.os.Bundle;
        import android.support.annotation.NonNull;
        import android.support.v4.app.Fragment;
        import android.view.LayoutInflater;
        import android.view.View;
        import android.view.ViewGroup;
        import android.widget.AdapterView;
        import android.widget.ArrayAdapter;
        import android.widget.ListView;


        /**
         * A simple {@link Fragment} subclass.
         */
        public class CityFragment extends Fragment {

            View view;
            String[] cities;
            String[] descriptions;
            ListView listView;
            DescriptionFragment text;
            int mPosition;

            public CityFragment() {
                // Required empty public constructor
            }


            @Override
            public View onCreateView(LayoutInflater inflater, ViewGroup container,
                                     Bundle savedInstanceState) {
                super.onCreate(savedInstanceState);
                // Check whether we're recreating a previously destroyed instance
                if (savedInstanceState != null) {
                    // Restore value of member from saved state
                    text = (DescriptionFragment) getFragmentManager().getFragment(savedInstanceState, getString(R.string.DESCRIPTION_FRAG));
                    mPosition = savedInstanceState.getInt(getString(R.string.POSITION));
                    cities = savedInstanceState.getStringArray(getString(R.string.CITY_ARRAY));
                    descriptions = savedInstanceState.getStringArray(getString(R.string.DESCRIPTION_ARRAY));
                    text.change(descriptions[mPosition], cities[mPosition]);
                }

                    // Get views
                    this.view = inflater.inflate(R.layout.fragment_city, container, false);
                    cities = getResources().getStringArray(R.array.cities);
                    descriptions = getResources().getStringArray(R.array.cities_description);
                    listView = this.view.findViewById(R.id.city_list);

                    // Create an ArrayAdapter
                    ArrayAdapter<String> listViewAdapter = new ArrayAdapter<String>(getActivity(),
                            android.R.layout.simple_list_item_1, cities);

                    // Set adapter on the listView
                    listView.setAdapter(listViewAdapter);

                // Set an item click listener for ListView
                listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {

                    @Override
                    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                        // Get the selected item text from ListView
                        text = (DescriptionFragment) getFragmentManager().findFragmentById(R.id.fragmentBottom);
                        text.change(descriptions[Integer.parseInt(String.valueOf(position))], cities[Integer.parseInt(String.valueOf(position))]);
                        mPosition = position;
                    }
                });

                // Inflate the layout for this fragment
                return this.view;

            }

            @Override
            public void onSaveInstanceState(@NonNull Bundle outState) {
                super.onSaveInstanceState(outState);

                // Save the fragments Instance
                getFragmentManager().putFragment(outState, getString(R.string.DESCRIPTION_FRAG), text);
                outState.putInt(getString(R.string.POSITION), mPosition);
                outState.putStringArray(getString(R.string.CITY_ARRAY), cities);
                outState.putStringArray(getString(R.string.DESCRIPTION_ARRAY), descriptions);
            }
        }

А вот фрагмент описания:

        package com.example.tourguide;


        import android.os.Bundle;
        import android.support.annotation.NonNull;
        import android.support.v4.app.Fragment;
        import android.view.LayoutInflater;
        import android.view.View;
        import android.view.ViewGroup;
        import android.widget.TextView;


        /**
         * A simple {@link Fragment} subclass.
         */
        public class DescriptionFragment extends Fragment {

            TextView cityName;
            TextView text;

            public DescriptionFragment() {
                // Required empty public constructor
            }


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

                if (savedInstanceState != null) {

                }

                // Inflate the layout for this fragment
                View view = inflater.inflate(R.layout.fragment_description, container, false);
                text = view.findViewById(R.id.city_description);
                cityName = view.findViewById(R.id.city_name);
                return view;
            }

            public void change(String description, String city) {
                text.setText(description);
                cityName.setText(city);
            }
        }

Кажется, что метод text.change заключается в том, в чем заключается проблема, но как еще я могу реализовать его всохранить состояние описания?

            if (savedInstanceState != null) {
                // Restore value of member from saved state
                text = (DescriptionFragment) getFragmentManager().getFragment(savedInstanceState, getString(R.string.DESCRIPTION_FRAG));
                mPosition = savedInstanceState.getInt(getString(R.string.POSITION));
                cities = savedInstanceState.getStringArray(getString(R.string.CITY_ARRAY));
                descriptions = savedInstanceState.getStringArray(getString(R.string.DESCRIPTION_ARRAY));
                text.change(descriptions[mPosition], cities[mPosition]);
            }

Ответы [ 2 ]

0 голосов
/ 03 октября 2019

Почему бы вам просто не сохранить состояние в пределах DescriptionFragment.

/**
 * A simple {@link Fragment} subclass.
 */
public class DescriptionFragment extends Fragment {

    private static final String DESCRIPTION = "DESCRIPTION";
    private static final String CITY = "CITY";

    TextView cityName;
    TextView text;

    public DescriptionFragment() {
        // Required empty public constructor
    }


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

    @Override
    public void onViewCreated(@NonNull View view, Bundle savedInstanceState) {
        text = view.findViewById(R.id.city_description);
        cityName = view.findViewById(R.id.city_name);

        if (savedInstanceState != null) {
            String description = savedInstanceState.getString(DESCRIPTION);
            String city = savedInstanceState.getString(CITY);
            change(description, city);
        }
    }

    @Override
    public void onSaveInstanceState(@NonNull Bundle outState) {
        super.onSaveInstanceState(outState);

        // Save the fragments Instance
        outState.putString(DESCRIPTION, text.getText().toString());
        outState.putString(CITY, cityName.getText().toString());
    }

    public void change(String description, String city) {
        text.setText(description);
        cityName.setText(city);
    }
}
0 голосов
/ 03 октября 2019

Может использовать свойство в вашем xml AndroidManifest.xml

 android:configChanges="keyboardHidden|orientation"

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

...