En-têtes Android ListView

122

J'ai ListView qui contient des événements. Les événements sont triés par jour, et j'aimerais avoir un en-tête avec la date pour chaque jour, puis les événements écoutent ci-dessous.

Voici comment je remplis cette liste:

ArrayList<TwoText> crs = new ArrayList<TwoText>();

crs.add(new TwoText("This will be header", event.getDate()));

for (Event event : events) {
    crs.add(new TwoText(event.getStartString() + "-" + event.getEndString(), event.getSubject()));
}

arrayAdapter = new TwoTextArrayAdapter(this, R.layout.my_list_item, crs);
lv1.setAdapter(arrayAdapter);

et voici à quoi ressemble ma classe TwoText:

public class TwoText {
    public String classID;
    public String state;

    public TwoText(String classID, String state) {
        this.classID = classID;
        this.state = state;
    }
}

et voici à quoi ressemble ma classe TwoTextArrayAdapter:

import java.util.ArrayList;
import android.app.Activity;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.TextView;

public class TwoTextArrayAdapter extends ArrayAdapter<TwoText> {

    private ArrayList<TwoText> classes;
    private Activity con;
    TextView seperator;

    public TwoTextArrayAdapter(Activity context, int textViewResourceId, ArrayList<TwoText> classes) {
        super(context, textViewResourceId, classes);
        this.con = context;
        this.classes = classes;

    }

    @Override

    public View getView(int position, View convertView, ViewGroup parent) {

        View v = convertView;

        if (v == null) {

            LayoutInflater vi = (LayoutInflater) con.getSystemService(Context.LAYOUT_INFLATER_SERVICE);

            v = vi.inflate(R.layout.my_list_item, null);

        }

        TwoText user = classes.get(position);

        if (user != null) {

            TextView content1 = (TextView) v.findViewById(R.id.list_content1);

            TextView content2 = (TextView) v.findViewById(R.id.list_content2);

            if (content1 != null) {

                content1.setText(user.classID);
            }   
            if(content2 != null) {

                content2.setText(user.state);
            }
        }
        return v;
    }
}

et c'est my_list_item.xml

<?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:orientation="vertical" >

    <TextView
        style="?android:attr/listSeparatorTextViewStyle"
        android:id="@+id/separator"
        android:text="Header"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:background="#757678"
        android:textColor="#f5c227" />

    <LinearLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="horizontal" >

        <TextView
            android:id="@+id/list_content1"
            android:layout_width="wrap_content"
            android:layout_height="match_parent"
            android:layout_margin="5dip"
            android:clickable="false"
            android:gravity="center"
            android:longClickable="false"
            android:paddingBottom="1dip"
            android:paddingTop="1dip"
            android:text="sample"
            android:textColor="#ff7f1d"
            android:textSize="17dip"
            android:textStyle="bold" />

        <TextView
            android:id="@+id/list_content2"
            android:layout_width="wrap_content"
            android:layout_height="match_parent"
            android:layout_margin="5dip"
            android:clickable="false"
            android:gravity="center"
            android:linksClickable="false"
            android:longClickable="false"
            android:paddingBottom="1dip"
            android:paddingTop="1dip"
            android:text="sample"
            android:textColor="#6d6d6d"
            android:textSize="17dip" />
    </LinearLayout>

</LinearLayout>

ce que je fais pour le moment, c'est que j'ajoute un en-tête comme un objet de liste normal, mais j'aimerais que ce soit comme en-tête et dans mon cas avoir une date dessus.

J'ai ce code dans mon xml pour l'en-tête:

<TextView
        style="?android:attr/listSeparatorTextViewStyle"
        android:id="@+id/separator"
        android:text="Header"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:background="#757678"
        android:textColor="#f5c227" />

et j'ai essayé de le cacher quand c'était inutile et de le montrer quand nécessaire mais j'ai juste foiré le reste de mon code. J'ai essayé quelques tutoriels supplémentaires mais ils ont également eu le même effet.

Quelqu'un pourrait-il me guider sur la façon de procéder facilement?

Rohit Malish
la source

Réponses:

334

Voici comment je le fais, les clés sont getItemViewType et getViewTypeCount dans la Adapterclasse. getViewTypeCountrenvoie le nombre de types d'éléments que nous avons dans la liste, dans ce cas, nous avons un élément d'en-tête et un élément d'événement, donc deux. getItemViewTypedevrait retourner quel type de Viewnous avons à l'entrée position.

Android va alors prendre soin de vous faire passer le bon type de Viewdans convertViewautomatiquement.

Voici à quoi ressemble le résultat du code ci-dessous:

Nous avons d'abord une interface que nos deux types d'éléments de liste implémenteront

public interface Item {
    public int getViewType();
    public View getView(LayoutInflater inflater, View convertView);
}

Ensuite, nous avons un adaptateur qui prend une liste de Item

public class TwoTextArrayAdapter extends ArrayAdapter<Item> {
    private LayoutInflater mInflater;

    public enum RowType {
        LIST_ITEM, HEADER_ITEM
    }

    public TwoTextArrayAdapter(Context context, List<Item> items) {
        super(context, 0, items);
        mInflater = LayoutInflater.from(context);
    }

    @Override
    public int getViewTypeCount() {
        return RowType.values().length;

    }

    @Override
    public int getItemViewType(int position) {
        return getItem(position).getViewType();
    }
@Override
public View getView(int position, View convertView, ViewGroup parent) {
   return getItem(position).getView(mInflater, convertView);
}

EDIT Better For Performance .. peut être remarqué lors du défilement

private static final int TYPE_ITEM = 0; 
private static final int TYPE_SEPARATOR = 1; 

public View getView(int position, View convertView, ViewGroup parent)  {
    ViewHolder holder = null;
    int rowType = getItemViewType(position);
    View View;
    if (convertView == null) {
        holder = new ViewHolder();
        switch (rowType) {
            case TYPE_ITEM:
                convertView = mInflater.inflate(R.layout.task_details_row, null);
                holder.View=getItem(position).getView(mInflater, convertView);
                break;
            case TYPE_SEPARATOR:
                convertView = mInflater.inflate(R.layout.task_detail_header, null);
                holder.View=getItem(position).getView(mInflater, convertView);
                break;
        }
        convertView.setTag(holder);
    }
    else
    {
        holder = (ViewHolder) convertView.getTag();
    }
    return convertView; 
} 

public static class ViewHolder {
    public  View View; } 
}

Ensuite, nous avons des classes pour implémenter Itemet gonfler les bonnes dispositions. Dans votre cas, vous aurez quelque chose comme une Headerclasse et une ListItemclasse.

   public class Header implements Item {
    private final String         name;

    public Header(String name) {
        this.name = name;
    }

    @Override
    public int getViewType() {
        return RowType.HEADER_ITEM.ordinal();
    }

    @Override
    public View getView(LayoutInflater inflater, View convertView) {
        View view;
        if (convertView == null) {
            view = (View) inflater.inflate(R.layout.header, null);
            // Do some initialization
        } else {
            view = convertView;
        }

        TextView text = (TextView) view.findViewById(R.id.separator);
        text.setText(name);

        return view;
    }

}

Et puis la ListItemclasse

    public class ListItem implements Item {
    private final String         str1;
    private final String         str2;

    public ListItem(String text1, String text2) {
        this.str1 = text1;
        this.str2 = text2;
    }

    @Override
    public int getViewType() {
        return RowType.LIST_ITEM.ordinal();
    }

    @Override
    public View getView(LayoutInflater inflater, View convertView) {
        View view;
        if (convertView == null) {
            view = (View) inflater.inflate(R.layout.my_list_item, null);
            // Do some initialization
        } else {
            view = convertView;
        }

        TextView text1 = (TextView) view.findViewById(R.id.list_content1);
        TextView text2 = (TextView) view.findViewById(R.id.list_content2);
        text1.setText(str1);
        text2.setText(str2);

        return view;
    }

}

Et un simple Activitypour l'afficher

public class MainActivity extends ListActivity {

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

        List<Item> items = new ArrayList<Item>();
        items.add(new Header("Header 1"));
        items.add(new ListItem("Text 1", "Rabble rabble"));
        items.add(new ListItem("Text 2", "Rabble rabble"));
        items.add(new ListItem("Text 3", "Rabble rabble"));
        items.add(new ListItem("Text 4", "Rabble rabble"));
        items.add(new Header("Header 2"));
        items.add(new ListItem("Text 5", "Rabble rabble"));
        items.add(new ListItem("Text 6", "Rabble rabble"));
        items.add(new ListItem("Text 7", "Rabble rabble"));
        items.add(new ListItem("Text 8", "Rabble rabble"));

        TwoTextArrayAdapter adapter = new TwoTextArrayAdapter(this, items);
        setListAdapter(adapter);
    }

}

Disposition pour R.layout.header

<?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:orientation="horizontal" >

    <TextView
        style="?android:attr/listSeparatorTextViewStyle"
        android:id="@+id/separator"
        android:text="Header"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:background="#757678"
        android:textColor="#f5c227" />

</LinearLayout>

Disposition pour R.layout.my_list_item

<?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:orientation="horizontal" >

    <TextView
        android:id="@+id/list_content1"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:layout_margin="5dip"
        android:clickable="false"
        android:gravity="center"
        android:longClickable="false"
        android:paddingBottom="1dip"
        android:paddingTop="1dip"
        android:text="sample"
        android:textColor="#ff7f1d"
        android:textSize="17dip"
        android:textStyle="bold" />

    <TextView
        android:id="@+id/list_content2"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:layout_margin="5dip"
        android:clickable="false"
        android:gravity="center"
        android:linksClickable="false"
        android:longClickable="false"
        android:paddingBottom="1dip"
        android:paddingTop="1dip"
        android:text="sample"
        android:textColor="#6d6d6d"
        android:textSize="17dip" />

</LinearLayout>

Disposition pour R.layout.activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity" >

    <ListView
        android:id="@android:id/list"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent" />

</RelativeLayout>

Vous pouvez également devenir plus sophistiqué et utiliser ViewHolders, charger des choses de manière asynchrone, ou ce que vous voulez.

antew
la source
5
J'ai aimé votre solution ici, mais puisque vous étendez un ArrayAdapter, vous ne devriez pas suivre votre propre liste d'éléments. Utilisez simplement celui suivi en interne dans ArrayAdapter. Sinon, vous doublez la quantité de mémoire pour stocker vos objets
Jay Soyer
3
Excellente solution. Je n'ajouterais cela qu'à l'adaptateur, si les en-têtes ne sont pas cliquables (ce qui pourrait être défini dans un constructeur) @Override public boolean isEnabled (int position) {return (getItem (position) .getViewType () == RowType.LIST_ITEM .ordinal()); }
dwbrito
2
les lignes deviennent aléatoires lorsque la vue de liste défile? Quelqu'un pourrait-il s'il vous plaît guide
i_raqz
1
Pourquoi Google ne peut-il pas simplement y arriver avec seulement 3 lignes de code?
Ojonugwa Jude Ochalifu
4
Cette réponse est l'une des meilleures réponses que j'ai trouvées sur SO - claire, concise et bien expliquée. Cependant, j'ai eu un problème avec les résultats de ListView dans un ordre semi-aléatoire (le premier en-tête et les éléments étaient corrects, les suivants étaient foirés), j'ai réussi à découvrir quel était le problème. Le bloc de code sous 'EDIT Better For Performance .. peut être remarqué lorsque le défilement' l'a gâché pour moi - la suppression de cela de la classe Custom ArrayAdapter a résolu le problème pour moi. Je conseille à tous ceux qui obtiennent des résultats aléatoires d'essayer ceci. Merci pour une excellente réponse. M'a vraiment aidé!
blueprintchris
9

Vous recherchez probablement un ExpandableListView qui a des en-têtes (groupes) pour séparer les éléments (enfants).

Joli tutoriel sur le sujet: ici .

Saito Mea
la source
Je ne veux pas qu'ils soient extensibles
Rohit Malish
Si c'est le seul problème, vous pouvez remplacer la méthode onItemClick pour empêcher l'expansion / la réduction des vues.
Saito Mea
Mais j'ai toujours besoin qu'ils soient cliquables à d'autres fins
Rohit Malish
Euh ... Je voulais en fait dire onGroupClickqui ne gère que les clics sur les "en-têtes" et vous n'avez pas besoin de désactiver le clic ou quelque chose comme ça, annulez simplement toute action de réduction et définissez tous les groupes en expansion depuis le début.
Saito Mea
Je conviens que ExpandableListViewc'est mieux dans la plupart des cas. Cependant, j'ai une situation où je veux parfois afficher une liste plate et une liste avec des en-têtes à d'autres moments de mon activité. Malheureusement, l' ExpandableListAdapterinterface n'étend pas l' ListAdapterinterface, donc pour le polymorphisme, je suis obligé d'utiliser la solution de @ antew.
tytk
3

Au lieu de cela, il existe une belle bibliothèque tierce conçue spécialement pour ce cas d'utilisation. Dans lequel vous devez générer des en-têtes en fonction des données stockées dans l'adaptateur. Ils sont appelés adaptateurs Rolodex et sont utilisés avec ExpandableListViews. Ils peuvent facilement être personnalisés pour se comporter comme une liste normale avec des en-têtes.

En utilisant les Eventobjets de l'OP et en sachant que les en-têtes sont basés sur ceux qui lui sont Dateassociés ... le code ressemblerait à ceci:

L'activité

    //There's no need to pre-compute what the headers are. Just pass in your List of objects. 
    EventDateAdapter adapter = new EventDateAdapter(this, mEvents);
    mExpandableListView.setAdapter(adapter);

L'adaptateur

private class EventDateAdapter extends NFRolodexArrayAdapter<Date, Event> {

    public EventDateAdapter(Context activity, Collection<Event> items) {
        super(activity, items);
    }

    @Override
    public Date createGroupFor(Event childItem) {
        //This is how the adapter determines what the headers are and what child items belong to it
        return (Date) childItem.getDate().clone();
    }

    @Override
    public View getChildView(LayoutInflater inflater, int groupPosition, int childPosition,
                             boolean isLastChild, View convertView, ViewGroup parent) {
        //Inflate your view

        //Gets the Event data for this view
        Event event = getChild(groupPosition, childPosition);

        //Fill view with event data
    }

    @Override
    public View getGroupView(LayoutInflater inflater, int groupPosition, boolean isExpanded,
                             View convertView, ViewGroup parent) {
        //Inflate your header view

        //Gets the Date for this view
        Date date = getGroup(groupPosition);

        //Fill view with date data
    }

    @Override
    public boolean hasAutoExpandingGroups() {
        //This forces our group views (headers) to always render expanded.
        //Even attempting to programmatically collapse a group will not work.
        return true;
    }

    @Override
    public boolean isGroupSelectable(int groupPosition) {
        //This prevents a user from seeing any touch feedback when a group (header) is clicked.
        return false;
    }
}
Jay Soyer
la source
1

Ce que j'ai fait pour faire de la date (par exemple le 01 décembre 2016) comme en-tête. J'ai utilisé la bibliothèque StickyHeaderListView

https://github.com/emilsjolander/StickyListHeaders

Convertissez la date en millisecondes [n'incluez pas l'heure] et faites-en l'ID d'en-tête.

@Override
public long getHeaderId(int position) {
    return <date in millis>;
}
Ban Daculan
la source
1

Voici un exemple de projet , basé sur la réponse détaillée et utile d'Antew, qui implémente un ListViewavec plusieurs en-têtes qui incorpore des détenteurs de vue pour améliorer les performances de défilement.

Dans ce projet, les objets représentés dans ListViewsont des instances de la classe HeaderItemou de la classe RowItem, qui sont toutes deux des sous-classes de la classe abstraite Item. Chaque sous - classe de Itemcorrespond à un type d'affichage différent dans l'adaptateur personnalisé, ItemAdapter. La méthode getView()sur ItemAdapterdélègue la création de la vue pour chaque élément de liste à une getView()méthode individualisée sur HeaderItemou RowItem, selon la Itemsous - classe utilisée à la position transmise à la getView()méthode sur l'adaptateur. Chaque Itemsous-classe fournit son propre support de vue.

Les détenteurs de vue sont implémentés comme suit. Les getView()méthodes des Itemsous - classes vérifient si l' Viewobjet qui a été passé à la getView()méthode sur ItemAdapterest nul. Si tel est le cas, la disposition appropriée est gonflée et un objet de support de vue est instancié et associé à la vue gonflée via View.setTag(). Si l' Viewobjet n'est pas nul, alors un objet de support de vue était déjà associé à la vue et le support de vue est récupéré via View.getTag(). La façon dont les détenteurs de vue sont utilisés peut être vue dans l'extrait de code suivant de HeaderItem:

@Override
View getView(LayoutInflater i, View v) {
    ViewHolder h;
    if (v == null) {
        v = i.inflate(R.layout.header, null);
        h = new ViewHolder(v);
        v.setTag(h);
    } else {
        h = (ViewHolder) v.getTag();
    }
    h.category.setText(text());
    return v;
}

private class ViewHolder {
    final TextView category;

    ViewHolder(View v) {
        category = v.findViewById(R.id.category);
    }
}

La mise en œuvre complète de ListView suit. Voici le code Java:

import android.app.ListActivity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.TextView;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

public class MainActivity extends ListActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setListAdapter(new ItemAdapter(getItems()));
    }

    class ItemAdapter extends ArrayAdapter<Item> {
        final private List<Class<?>> viewTypes;

        ItemAdapter(List<Item> items) {
            super(MainActivity.this, 0, items);
            if (items.contains(null))
                throw new IllegalArgumentException("null item");
            viewTypes = getViewTypes(items);
        }

        private List<Class<?>> getViewTypes(List<Item> items) {
            Set<Class<?>> set = new HashSet<>();
            for (Item i : items) 
                set.add(i.getClass());
            List<Class<?>> list = new ArrayList<>(set);
            return Collections.unmodifiableList(list);
        }

        @Override
        public int getViewTypeCount() {
            return viewTypes.size();
        }

        @Override
        public int getItemViewType(int position) {
            Item t = getItem(position);
            return viewTypes.indexOf(t.getClass());
        }

        @Override
        public View getView(int position, View v, ViewGroup unused) {
            return getItem(position).getView(getLayoutInflater(), v);
        }
    }

    abstract private class Item {
        final private String text;

        Item(String text) {
            this.text = text;
        }

        String text() { return text; }

        abstract View getView(LayoutInflater i, View v);
    }

    private class HeaderItem extends Item {
        HeaderItem(String text) {
            super(text);
        }

        @Override
        View getView(LayoutInflater i, View v) {
            ViewHolder h;
            if (v == null) {
                v = i.inflate(R.layout.header, null);
                h = new ViewHolder(v);
                v.setTag(h);
            } else {
                h = (ViewHolder) v.getTag();
            }
            h.category.setText(text());
            return v;
        }

        private class ViewHolder {
            final TextView category;

            ViewHolder(View v) {
                category = v.findViewById(R.id.category);
            }
        }
    }

    private class RowItem extends Item {
        RowItem(String text) {
            super(text);
        }

        @Override
        View getView(LayoutInflater i, View v) {
            ViewHolder h;
            if (v == null) {
                v = i.inflate(R.layout.row, null);
                h = new ViewHolder(v);
                v.setTag(h);
            } else {
                h = (ViewHolder) v.getTag();
            }
            h.option.setText(text());
            return v;
        }

        private class ViewHolder {
            final TextView option;

            ViewHolder(View v) {
                option = v.findViewById(R.id.option);
            }
        }
    }

    private List<Item> getItems() {
        List<Item> t = new ArrayList<>();
        t.add(new HeaderItem("Header 1"));
        t.add(new RowItem("Row 2"));
        t.add(new HeaderItem("Header 3"));
        t.add(new RowItem("Row 4"));

        t.add(new HeaderItem("Header 5"));
        t.add(new RowItem("Row 6"));
        t.add(new HeaderItem("Header 7"));
        t.add(new RowItem("Row 8"));

        t.add(new HeaderItem("Header 9"));
        t.add(new RowItem("Row 10"));
        t.add(new HeaderItem("Header 11"));
        t.add(new RowItem("Row 12"));

        t.add(new HeaderItem("Header 13"));
        t.add(new RowItem("Row 14"));
        t.add(new HeaderItem("Header 15"));
        t.add(new RowItem("Row 16"));

        t.add(new HeaderItem("Header 17"));
        t.add(new RowItem("Row 18"));
        t.add(new HeaderItem("Header 19"));
        t.add(new RowItem("Row 20"));

        t.add(new HeaderItem("Header 21"));
        t.add(new RowItem("Row 22"));
        t.add(new HeaderItem("Header 23"));
        t.add(new RowItem("Row 24"));

        t.add(new HeaderItem("Header 25"));
        t.add(new RowItem("Row 26"));
        t.add(new HeaderItem("Header 27"));
        t.add(new RowItem("Row 28"));
        t.add(new RowItem("Row 29"));
        t.add(new RowItem("Row 30"));

        t.add(new HeaderItem("Header 31"));
        t.add(new RowItem("Row 32"));
        t.add(new HeaderItem("Header 33"));
        t.add(new RowItem("Row 34"));
        t.add(new RowItem("Row 35"));
        t.add(new RowItem("Row 36"));

        return t;
    }

}

Il existe également deux présentations d'éléments de liste, une pour chaque sous-classe d'éléments. Voici la disposition header, utilisée par HeaderItem:

<?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="wrap_content"
    android:background="#FFAAAAAA"
    >
    <TextView
        android:id="@+id/category"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_margin="4dp"
        android:textColor="#FF000000"
        android:textSize="20sp"
        android:textStyle="bold"
        />
 </LinearLayout>

Et voici la disposition row, utilisée par RowItem:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:minHeight="?android:attr/listPreferredItemHeight"
    >
    <TextView
        android:id="@+id/option"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textSize="15sp"
        />
</LinearLayout>

Voici une image d'une partie de la ListView résultante:

ListView avec plusieurs en-têtes

stevehs17
la source