2013-06-25 10 views
7

List(ListView)の上にいくつかのヘッダーテキスト(TextView)を追加できるように、カスタムListPreferenceダイアログを作成する必要があります。 私はListPreferenceを拡張し、onCreateDialogViewを(上書きされますMyListPreferenceクラス)を作成しました:なぜListViewはListPreferenceダイアログのTextViewの上にとどまっていますか?

@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には、リストビューの下に表示される代わりに、上記。上記のようにTextViewが必要です。私はLinearLayoutとRelativeLayout( "下"または "上の"属性を使用して)成功なしで試しました:私はTextViewをListViewの上に置く方法を見つけることができません...レイアウトはかなりシンプルで、リストが上記のままである理由 また、問題は実デバイス(Nexus 4、Android 4.2.2)とエミュレータの両方で発生することに注意してください。しかし、Eclipseのグラフィカルレイアウトでレンダリングされたレイアウトを見ると、レイアウトは正しいです!添付の両方の写真を参照してください。

これを解決する方法はありますか?溶液2013年7月10日と

Layout rendered on Eclipse (correct)

編集:Eclipseの(正しい)上にレンダリング

Layout rendered on the device (incorrect)

レイアウト:(誤った)デバイス上でレンダリング

レイアウト

受理された回答で示唆されているように、問題はListPreferenceのonPrepareDialogBu​​ilder()でbuilder.setSingleChoiceItems()を使用することに起因します。 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> 
+0

あなたが値を初期化コードを追加することができますリストのために?または 'ListView'を参照するコード – Muzikant

+0

' android:layout_above = "@ + id/list" ' とオプションでアンドロイド:layout_alignParentTop =" true "'を試しました。元のdialog_preference_list.xmlのTextView、 が正しいですか? – JaKXz

+0

すてきな答え!ありがとう、あなたmuslidrikk。 – Tiago

答えて

4

問題はListPreferenceの仕組みにあると思います。 ListPreferenceはBuilder.setSingleChoiceItems()を使用してRadioButtonsで行を作成し、追加しようとしているカスタムレイアウト(LinearLayout内のTextViewとListView)よりも優先されます(Hereは解決されています)。 。。私は何が必要ないカスタムDialogPreferenceを作成したのGitHubへのリンクは、私はRadioButtonのロジックをコーディングしていない

+0

これはあなたを助けましたか? – Emmanuel

+0

あなたの提案をありがとう。実際、この問題はBuilder.setSingleChoiceItems()から発生します。私はListPreferenceを拡張し、ダイアログビルダを使用しないようにonCreateDialogView()をオーバーライドすることで修正しました。コードを別の答えに入れます。 – muslidrikk

+0

私は助けてくれました。 GitHubのリンクはあなたを助けましたか? – Emmanuel

2

私はそれがテーマの問題だと思います。コンストラクタ内のダイアログのテーマを変更して、setStyle(STYLE_NO_TITLE, R.style.AppTheme)のようにしてみてください。 no_titleスタイルの基本アプリテーマ。

これは問題ではない場合は、ListPreferenceクラス自体に関連する可能性があります。環境設定をテーマにして一貫性を持たせるため、レイアウトを上書きしているかもしれません。しかし、私は前にListPreferenceを使用していないので、そのちょうど推測。

テーマをXMLグラフィックレイアウトプレビューで再生しても同じ結果を再現できますか?

+0

ありがとうございます。しかし、テーマで遊んだ後、それは問題ではないように見えました。 – muslidrikk

1

あなたが試すことができます別のオプションは、このようなListViewにヘッダとしてTextViewを追加することです。

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

addHeaderViewにはViewが含まれているため、理論的にはヘッダーになりたいですが、私はTextViewしか使っていません。

+0

ありがとうございますが、addHeaderView()はListPreferenceのListViewで使用されても動作しないようです... – muslidrikk

0

上記のリンクは壊れています。このソリューションでは、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; 
} 

もう一つ重要なことは、それにスーパーを呼び出しonPrepareDialogBu​​ilder電話しないことです。これにより、リストビューが2回表示されなくなります。

dialog_preference_list_headerは、私の場合にのみTestViewにあるが、それは、より複雑なビュー可能性があり、かつcustom_dialog_single_choice_list_adapterはこのようなものかもしれない
@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()); 
} 

<?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" /> 
関連する問題