Заголовок пользовательского раздела Android Listview - PullRequest
6 голосов
/ 28 февраля 2012

Как я могу реализовать пользовательский раздел или заголовок ListView, как приложение Instagram в Android.

http://prsarahevans.com/wp-content/uploads/2011/06/photo.PNG

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

Спасибо.

Ответы [ 5 ]

9 голосов
/ 20 апреля 2012

Мне удалось решить эту проблему с помощью прослушивателя прокрутки в просмотре списка. (проверено на 2.1)

Допустим, для каждой строки списка у меня есть макет, подобный приведенному ниже. Есть часть контента и часть заголовка. Неважно, какой тип представления вы используете для заголовка или содержимого.

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" android:background="#FFFFFF">

    <ImageView
        android:id="@+id/content"
        android:layout_width="fill_parent"
        android:layout_height="300dp"
        android:scaleType="centerCrop" 
        android:src="@drawable/pic"
        android:background="#aaaaff" 
        android:layout_marginTop="40dp"/>

    <TextView
        android:id="@+id/header"
        android:layout_width="fill_parent"
        android:layout_height="40dp"
        android:padding="12dp"
        android:text="Deneme Row"
        android:textColor="#000000" 
        android:background="#99ffffff"/>
</RelativeLayout>

Макет теста для действия следующий:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

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

Наконец, код для деятельности приведен ниже. Здесь я должен был иметь адаптер, который использует ViewHolder для хранения представления заголовка, а также переменную для отслеживания изменений в прокрутке для каждого последующего события прокрутки (previousTop). Это связано с тем, что offsetTopAndBottom () изменяет смещение представления, связанного с его предыдущим местоположением.

public class TestActivity extends Activity implements AbsListView.OnScrollListener{

    ListView list; 
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        list = (ListView) findViewById(R.id.list);
        list.setAdapter(new Adapter(this));        
        list.setOnScrollListener(this); 
    }

    @Override
    public void onScroll(AbsListView view, int firstVisibleItem,
            int visibleItemCount, int totalItemCount) {     
        //the listview has only few children (of course according to the height of each child) who are visible
        for(int i=0; i < list.getChildCount(); i++){
            View child = list.getChildAt(i);
            ViewHolder holder = (ViewHolder) child.getTag();

            //if the view is the first item at the top we will do some processing
            if(i == 0){             
                boolean isAtBottom = child.getHeight() <= holder.header.getBottom();
                int offset = holder.previousTop - child.getTop();
                if(!(isAtBottom && offset > 0)){                    
                    holder.previousTop = child.getTop();
                    holder.header.offsetTopAndBottom(offset);                   
                    holder.header.invalidate();
                }
            } //if the view is not the first item it "may" need some correction because of view re-use
            else if (holder.header.getTop() != 0) {
                int offset = -1 * holder.header.getTop(); 
                holder.header.offsetTopAndBottom(offset);
                holder.previousTop = 0;
                holder.header.invalidate();
            }
        }
    }

    @Override
    public void onScrollStateChanged(AbsListView view, int scrollState) {}

    private static class Adapter extends ArrayAdapter<String> {
        public Adapter(Context context) {
            super(context, R.layout.row, R.id.header);
            for(int i=0; i < 50; i++){
                add(Integer.toString(i));
            }
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            if(convertView == null){
                convertView = LayoutInflater.from(getContext()).inflate(R.layout.row, parent, false);
                ViewHolder holder = new ViewHolder();
                holder.header = (TextView) convertView.findViewById(R.id.header);
                convertView.setTag(holder);             
            }
            ViewHolder holder = (ViewHolder) convertView.getTag();
            holder.header.setText(getItem(position));
            return convertView;
        }
    }

    private static class ViewHolder {
        TextView header;
        int previousTop = 0;
    }
}
1 голос
/ 15 января 2013

Я немного улучшил вопрос Сиямеда, чтобы заголовок не пропадал при перерисовке представления, например, при изменении растрового изображения в элементе списка.

Вместо того, чтобы использовать координаты относительно последней позиции, я использую координаты относительно верхней части вида и использую отступы вместо смещений.

@Override
public void onScroll(AbsListView absListView, int firstVisibleItem, int visibleItemCount, int totalItemCount) {

    for(int i=0; i < list.getChildCount(); i++){
        View child = list.getChildAt(i);
        ViewHolder holder = (ViewHolder) child.getTag();

        if(i == 0){
            try {
                boolean isAtBottom = child.getHeight() <= holder.movingHeader.getBottom();
                if(!(isAtBottom)){
                    if (child.getTop() >= 0){}
                    else if (child.getHeight() - movingHeader.getHeight() - 1 > -child.getTop())
                    {
                        holder.movingHeader.setPadding(0, -child.getTop(), 0, 0);
                        holder.movingHeader.invalidate();
                    }
                    else {
                        holder.movingHeader.setPadding(0, child.getHeight() - movingHeader.getHeight(), 0, 0);
                        holder.movingHeader.invalidate();
                    }
                }
            }
            catch (Exception e)
            {
                e.printStackTrace();
            }
        }
        else if (holder.movingHeader.getPaddingTop() != 0)
        {
            holder.movingHeader.setPadding(0, 0, 0, 0);
            holder.movingHeader.invalidate();
        }
    }
}
0 голосов
/ 13 ноября 2015

Я знаю, что этот вопрос немного стар, но ему удалось найти библиотеку StickyListHeaders, которая довольно неплохо справляется с абстракцией создания заголовков разделов.

https://github.com/emilsjolander/StickyListHeaders

0 голосов
/ 28 февраля 2012

После установки setcontentview сделайте что-то вроде приведенного ниже кода.

    ListView list = getListView();
    View footer = getLayoutInflater().inflate(R.layout.footerlayout, list, false);
    View header = getLayoutInflater().inflate(R.layout.headerlayout, list, false);
    list.addHeaderView(header);
    list.addFooterView(footer); 
0 голосов
/ 28 февраля 2012

Если вы используете вид списка, то у вас есть 2 вида внутри элемента списка.Один - заголовок, а другой - элемент.

Первоначально скрыть вид заголовка, а когда состояние прокрутки изменяется или пользователь касается элемента списка, видимость заголовка изменяется.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...