Андроид виджет TextView содержимое теряется, хотя андроид:freezesText имеет значение true
У меня есть одно действие с двумя фрагментами. У действия есть разные макеты для портрета и ландшафта, но фрагменты остаются теми же (созданные с тегами и повторно используемые). В одном из фрагментов я определил XML textview, и у него есть строка android:freezesText="true", но он все равно теряет текст после поворота экрана. Почему это могло случиться?
Расположение фрагмента:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center_horizontal"
android:orientation="vertical" >
<TextView
android:id="@+id/tvDisplay"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="@dimen/dip10"
android:freezesText="true"
android:gravity="center"
android:text="@string/hello_world"
android:textAppearance="?android:attr/textAppearanceLarge" />
</LinearLayout>
Класс фрагмента:
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
public class DisplayFrag extends Fragment {
@Override
public View
onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.display_frag, container, false);
TextView tvDisplay = (TextView) view.findViewById(R.id.tvDisplay);
tvDisplay.setText(getArguments().getString(Keys.NAME)
+ ", how are you? your phone number is " + getArguments().getString(Keys.PHONE));
return view;
}
}
Деятельность:
import android.os.Bundle;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentManager;
import android.view.View;
import android.widget.EditText;
public class MainActivity extends FragmentActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
FragmentManager fragMan = getSupportFragmentManager();
InputFrag inputFrag = (InputFrag) fragMan.findFragmentByTag(Keys.TAG_INPUT_FRAG);
if (inputFrag == null) {
fragMan.beginTransaction().add(R.id.hook1, new InputFrag(), Keys.TAG_INPUT_FRAG)
.commit();
} else {
fragMan.beginTransaction().replace(R.id.hook1, inputFrag, Keys.TAG_INPUT_FRAG).commit();
}
}
public void submit(View view) {
FragmentManager fragMan = getSupportFragmentManager();
DisplayFrag displayFrag = (DisplayFrag) fragMan.findFragmentByTag(Keys.TAG_DISPLAY_FRAG);
if (displayFrag == null) {
displayFrag = new DisplayFrag();
displayFrag = putBundle(displayFrag);
fragMan.beginTransaction().add(R.id.hook2, displayFrag, Keys.TAG_DISPLAY_FRAG).commit();
} else {
fragMan.beginTransaction().replace(R.id.hook2, displayFrag, Keys.TAG_DISPLAY_FRAG)
.commit();
}
}
private DisplayFrag putBundle(DisplayFrag displayFrag) {
Bundle bundle = new Bundle();
bundle.putString(Keys.NAME, ((EditText) findViewById(R.id.etName)).getText().toString());
bundle.putString(Keys.PHONE, ((EditText) findViewById(R.id.etPhone)).getText().toString());
displayFrag.setArguments(bundle);
return displayFrag;
}
}
(представить реагирует на кнопку на другом фрагменте, который ведет себя корректно).
3 ответа:
Нашел решение: не нужно .замените() фрагменты вообще. вот новый код для MainActivity и DisplayFrag:
import android.os.Bundle; import android.support.v4.app.FragmentActivity; import android.support.v4.app.FragmentManager; import android.view.View; import android.widget.EditText; public class MainActivity extends FragmentActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); FragmentManager fragMan = getSupportFragmentManager(); InputFrag inputFrag = (InputFrag) fragMan.findFragmentByTag(Keys.TAG_INPUT_FRAG); if (inputFrag == null) { fragMan.beginTransaction().add(R.id.hook1, new InputFrag(), Keys.TAG_INPUT_FRAG) .commit(); } DisplayFrag displayFrag = (DisplayFrag) fragMan.findFragmentByTag(Keys.TAG_DISPLAY_FRAG); if (displayFrag == null) { fragMan.beginTransaction().add(R.id.hook2, new DisplayFrag(), Keys.TAG_DISPLAY_FRAG) .commit(); } } public void submit(View v) { String name = ((EditText) findViewById(R.id.etName)).getText().toString(); String phone = ((EditText) findViewById(R.id.etPhone)).getText().toString(); if (!"".matches(name) && !"".matches(phone)) { FragmentManager fragMan = getSupportFragmentManager(); DisplayFrag displayFrag = (DisplayFrag) fragMan.findFragmentByTag(Keys.TAG_DISPLAY_FRAG); if (displayFrag != null) { displayFrag.display(name, phone); } } } } import android.os.Bundle; import android.support.v4.app.Fragment; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.TextView; public class DisplayFrag extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = inflater.inflate(R.layout.display_frag, container, false); return view; } public void display(String name, String phone) { TextView tvDisplay = (TextView) getActivity().findViewById(R.id.tvDisplay); tvDisplay.setText(name + ", how are you? your phone number is " + phone); } }
Принимая мой ответ на данный момент. Если у кого-то есть другой ответ, совет или знание, чтобы поделиться, я с радостью приму другие ответы.
Это потому, что вы каждый раз создаете новый фрагмент. вы можете сделать это, просто проверив savedInstanceState. надеюсь, мой пример поможет вам почувствовать себя лучше.
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); if (savedInstanceState == null){ fragMan.beginTransaction().add(R.id.hook1, new InputFrag()) .commit(); fragMan.beginTransaction().add(R.id.hook2, new DisplayFrag()) .commit(); } }
И поставить этот какой-то баг случится на определенной версии.
@Override protected void onSaveInstanceState(Bundle outState) { // first saving my state, so the bundle wont be empty. // http://code.google.com/p/android/issues/detail?id=19917 outState.putString("WORKAROUND_FOR_BUG_19917_KEY", "WORKAROUND_FOR_BUG_19917_VALUE"); super.onSaveInstanceState(outState); }
Фрагмент имеет разный жизненный цикл от действия, задаваемого
android:configChanges="orientation
На самом деле не решит проблему.
Измените свою деятельность в AndroidManifest.xml для переопределения поведения изменения ориентации, добавив это к вашей деятельности / действиям:
android:configChanges="orientation|keyboardHidden"
Это должно выглядеть примерно так:
<activity android:name=".activity.LoginActivity" android:configChanges="orientation|keyboardHidden" android:label="@string/app_name"> </activity>