Почему ListView остается над TextView в диалоговом окне ListPreference?

Мне нужно создать настраиваемый диалог ListPreference, чтобы добавить текст заголовка (TextView) над списком (ListView). Я создал класс MyListPreference, который расширяет ListPreference и переопределяет onCreateDialogView ():

@Override protected View onCreateDialogView() { LayoutInflater inflater = (LayoutInflater) getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE); View v = (View) inflater.inflate(R.layout.dialog_preference_list, null); return v; } 

Мой XML-макет dialog_preference_list.xml содержит:

 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="vertical" > <TextView android:id="@+id/textView1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="TextView" /> <ListView android:id="@android:id/list" android:layout_width="wrap_content" android:layout_height="wrap_content" android:drawSelectorOnTop="false" android:scrollbarAlwaysDrawVerticalTrack="true" /> </LinearLayout> 

Проблема: TextView отображается ниже ListView, а не выше. Мне нужно, чтобы TextView был выше. Я пробовал оба с LinearLayout и RelativeLayout (используя атрибуты «ниже» или «выше») без успеха: я не могу найти способ разместить TextView над ListView … Макет довольно прост, и я не вижу Почему список остается выше … Также обратите внимание, что проблема возникает как на реальном устройстве (Nexus 4, Android 4.2.2), так и на эмуляторе. Однако при просмотре макета, представленного в графическом макете Eclipse, макет правильный! Смотрите оба прикрепленных изображения.

Любая идея о том, как это решить?

Макет, отображаемый на устройстве (неверно):

Макет, сделанный на устройстве (неверно)

Макет, сделанный на Eclipse (правильный):

Макет, сделанный на Eclipse (правильный)

Редактировать с помощью решения 10.07.2013

Как было предложено принятым ответом, проблема возникает из-за использования builder.setSingleChoiceItems () в ListPreference onPrepareDialogBuilder (). Я исправил это, расширив ListPreference и переопределив onCreateDialogView (), чтобы построить Диалог без строителя, чтобы я мог создать собственный вид, отображающий текст заголовка над элементами списка.

GPListPreference.java:

 public class GPListPreference extends ListPreference { ... @Override protected void onPrepareDialogBuilder(AlertDialog.Builder builder) { builder.setNegativeButton(null, null); builder.setPositiveButton(null, null); } private int getValueIndex() { return findIndexOfValue(getValue()); } @Override protected View onCreateDialogView() { LayoutInflater inflater = (LayoutInflater) getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE); ListView lv = (ListView) inflater.inflate(R.layout.dialog_preference_list, null); TextView header = (TextView) inflater.inflate(R.layout.dialog_preference_list_header, null); header.setText(getDialogMessage()); // you should set the header text as android:dialogMessage in the preference XML lv.addHeaderView(header); ArrayAdapter<CharSequence> adapter = new ArrayAdapter<CharSequence>(getContext(), R.layout.dialog_preference_list_singlechoice, getEntries()); lv.setAdapter(adapter); lv.setClickable(true); lv.setEnabled(true); lv.setChoiceMode(ListView.CHOICE_MODE_SINGLE); lv.setItemChecked(getValueIndex() + 1, true); lv.setOnItemClickListener(new AdapterView.OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { setValueIndex(position - 1); getDialog().dismiss(); } }); return lv; } } 

dialog_preference_list.xml:

 <?xml version="1.0" encoding="utf-8"?> <ListView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@android:id/list" android:layout_width="wrap_content" android:layout_height="wrap_content" android:drawSelectorOnTop="false" android:scrollbarAlwaysDrawVerticalTrack="true" /> 

dialog_preference_list_singlechoice.xml

 <?xml version="1.0" encoding="utf-8"?> <CheckedTextView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@android:id/text1" android:layout_width="match_parent" android:layout_height="wrap_content" android:checkMark="?android:attr/listChoiceIndicatorSingle" android:ellipsize="marquee" android:gravity="center_vertical" android:minHeight="?android:attr/listPreferredItemHeight" android:paddingBottom="2dip" android:paddingLeft="10dip" android:paddingRight="10dip" android:paddingTop="2dip" android:textAppearance="?android:attr/textAppearanceMedium" /> 

dialog_preference_list_header.xml

 <?xml version="1.0" encoding="utf-8"?> <TextView xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="wrap_content" android:layout_height="wrap_content" android:padding="10dip" android:textAppearance="?android:attr/textAppearanceSmall"> </TextView> 

Я думаю, проблема связана с тем, как работает ListPreference. ListPreference использует Builder.setSingleChoiceItems () для создания строк с помощью RadioButtons и имеет предпочтение над настраиваемым макетом, который вы пытаетесь добавить (в вашем случае TextView и ListView внутри LinearLayout). Вместо этого решение расширяет DialogPreference. Ссылку на GitHub, где я создал пользовательский DialogPreference, который делает то, что вам нужно. Я не закодировал логику RadioButton.

Думаю, это тема. Попробуйте изменить тему своего диалога внутри конструктора, сделайте что-то вроде setStyle(STYLE_NO_TITLE, R.style.AppTheme) . Тема вашего базового приложения с типом no_title.

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

Можете ли вы воспроизвести тот же результат, играя с темами в графическом представлении XML-графики?

Другой вариант, который вы можете попробовать, – добавить TextView в качестве заголовка в ListView следующим образом:

 TextView textView = new TextView(getActivity()); ListView listView = new ListView(getActivity()); listView.addHeaderView(textView); 

addHeaderView принимает View поэтому у вас теоретически есть все, что вы хотите быть заголовком, но я использовал TextView .

Вышеуказанная ссылка нарушена. В этом решении идея переопределяет ListPreference и раздувает ваш собственный список, с данными, определенными в ListPreference.

 @Override protected View onCreateDialogView() { LayoutInflater inflater = (LayoutInflater) getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE); ListView lv = new ListView(getContext()); // Inflate the view into the header only if a message was set if (getDialogMessage() != null && ! getDialogMessage().equals("") ) { TextView header = (TextView) inflater.inflate(R.layout.dialog_preference_list_header, null); header.setText(getDialogMessage()); lv.addHeaderView(header, null, false); } // Create a new adapter and a list view and feed it with the ListPreference entries ArrayAdapter<CharSequence> adapter = new ArrayAdapter<CharSequence>(getContext(), R.layout.custom_dialog_single_choice_list_adapter, getEntries()); lv.setAdapter(adapter); lv.setClickable(true); lv.setEnabled(true); lv.setChoiceMode(ListView.CHOICE_MODE_SINGLE); lv.setItemChecked(getValueIndex() + 1, true); lv.setOnItemClickListener(new AdapterView.OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { setValueIndex(position - 1); getDialog().dismiss(); } }); return lv; } 

Еще одна важная вещь – вызвать onPrepareDialogBuilder и не называть супер. Это позволит избежать появления списка дважды.

 @Override protected void onPrepareDialogBuilder(AlertDialog.Builder builder) { // Not calling super, to avoid having 2 listviews // Set the positive button as null builder.setPositiveButton(null, null); } private int getValueIndex() { return findIndexOfValue(getValue()); } 

Где dialog_preference_list_header в моем случае только TestView, но это может быть более сложное представление, а custom_dialog_single_choice_list_adapter может быть примерно так:

 <?xml version="1.0" encoding="utf-8"?> <CheckedTextView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@android:id/text1" android:layout_width="match_parent" android:layout_height="wrap_content" android:checkMark="?android:attr/listChoiceIndicatorSingle" android:ellipsize="marquee" android:gravity="center_vertical" android:minHeight="?android:attr/listPreferredItemHeight" android:paddingBottom="2dip" android:paddingLeft="10dip" android:paddingRight="10dip" android:paddingTop="2dip" android:textAppearance="?android:attr/textAppearanceMedium" />