在ListView中滚动的日期分隔符

问题描述 投票:3回答:1

在ListView(使用自定义CursorAdapter)中,我试图仅根据两个ListView项之间的日期是否发生更改来显示分隔符,例如:as so

[不幸的是,我设置代码的方式是,当我向下滚动,暂停一秒钟然后再次开始滚动时,我将分隔符调到了错误的位置(例如在这里,看到分隔符的消失了在ListView的第二项中):see the disappearance of the second item

或者,它可能出现在第三项中。

我遇到了两个主要问题:

  1. 日期分隔符在各个时间的出现和消失,取决于滚动的暂停,并且
    1. 日期分隔符的外观,因为它在错误的位置连续向上滚动。

我从原则上理解,出现这些问题是因为适配器中的lastDate的行为实际上仅考虑了单向向下滚动,此外还由ListView回收视图和容器的方式引起。

我的问题,本质上是:如何在MyAdapter.java中修改自定义的CursorAdapter或如何在MainFragment.java中存储数据,以便无论滚动方向,瞬时停顿和然后滚动?

非常感谢!

下面是MainFragment的代码,适配器的代码以及ListView项布局的XML。

MyAdapter.java

package com.cpsashta.seplistview;

import android.content.Context;
import android.database.Cursor;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.CursorAdapter;
import android.widget.ImageView;
import android.widget.TextView;

import com.cpsashta.seplistview.DbContract.DbEntry;

public class MyAdapter extends CursorAdapter {

    private LayoutInflater cursorInflater;
    private Context mContext;
    private Cursor mCursor;

    private String lastDate;

    TextView tvSep;
    TextView tvName;
    TextView tvTitle;
    TextView tvDate;
    TextView tvExp;

    public MyAdapter(Context context, Cursor c, int flags) {
        super(context,c,flags);
        this.mContext = context;
        this.mCursor = c;
        cursorInflater = LayoutInflater.from(context);
    }

    @Override
    public View newView(Context context, Cursor cursor, ViewGroup parent) {
        return cursorInflater.inflate(R.layout.list_layout_1, parent, false);

    }

    @Override
    public void bindView(View view, Context context, Cursor c) {
        tvTitle = (TextView) view.findViewById(R.id.tv_title);
        tvName = (TextView) view.findViewById(R.id.tv_name);
        tvSep = (TextView) view.findViewById(R.id.separator);
        tvDate = (TextView) view.findViewById(R.id.tv_date);
        tvExp = (TextView) view.findViewById(R.id.is_expanded_view);

        String title = c.getString(c.getColumnIndex(DbEntry.COLNAME_TITLE));
        String name = c.getString(c.getColumnIndex(DbEntry.COLNAME_NAME));
        String date = c.getString(c.getColumnIndex(DbEntry.COLNAME_DATE));

        tvTitle.setText(title);
        tvName.setText(name);
        tvDate.setText(date);
        if (!date.equals(lastDate) || c.getPosition() == 0) {
            tvSep.setVisibility(View.VISIBLE);
            tvSep.setText(date);
            tvExp.setText("true");
            lastDate = date;
        } else {
            tvSep.setVisibility(View.GONE);
            tvExp.setText("false");
        }
    }
}

list_layout_1.xml

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

    <TextView
        android:id="@+id/separator"
        style="?android:attr/listSeparatorTextViewStyle"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:textColor="@android:color/white"
        android:background="#FF0000"
        android:visibility="gone"/>

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:visibility="gone"
        android:id="@+id/is_expanded_view"/>

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textAppearance="?android:attr/textAppearanceMedium"
        android:text="Medium Text"
        android:id="@+id/tv_title"
        android:layout_gravity="center_horizontal" />

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/tv_date"/>

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textAppearance="?android:attr/textAppearanceSmall"
        android:text="Small Text"
        android:id="@+id/tv_name"
        android:layout_gravity="right" />
</LinearLayout>

MainFragment.java

package com.cpsashta.seplistview;

import android.content.ContentValues;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.support.v4.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.LinearLayout;
import android.widget.ListView;

import com.cpsashta.seplistview.DbContract.*;




public class MainFragment extends Fragment {

    String[] names = {"Dick", "Sammy", "Manuel", "Jenny", "Kirk", "Shimmy"};
    public static String titleSmile = "Damned crazy folks, Smile, you're on camera!";
    String[] titles = {"Here we go again!", titleSmile};
    String dateMonth = "Aug";

    DbHelper mDbHelper;
    SQLiteDatabase db;
    Button button;

    public MainFragment() {
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View rootView = inflater.inflate(R.layout.fragment_main, container, false);

        button = (Button) rootView.findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                LinearLayout ll = (LinearLayout) v.getParent();
                addOneItem(ll);
            }
        });

        return rootView;

    }


    @Override
    public void onViewCreated(View view, Bundle savedInstanceState) {
        ListView lv = (ListView) view.findViewById(R.id.listview);
        Cursor c = getAll();
        MyAdapter adapter = new MyAdapter(getActivity(),c,0);
        lv.setAdapter(adapter);

    }

    //Occurs on button press, the view pressed is intended to be the rootView (a.k.a. the LinearLayout housing the ListView and button)
    public void addOneItem(View view) {
        newItemToDb();
        ListView lv = (ListView) view.findViewById(R.id.listview);
        MyAdapter adapter = (MyAdapter) lv.getAdapter();
        adapter.notifyDataSetChanged();
    }

    //Adds random name, title, and date to the database
    public void newItemToDb() {
        if (db == null || !db.isOpen()) {
            if (mDbHelper == null) {
                mDbHelper = new DbHelper(getActivity());
            }
            db = mDbHelper.getWritableDatabase();
        }

        ContentValues values = new ContentValues();
        int nameVal = (int)(Math.random() * names.length);
        values.put(DbEntry.COLNAME_NAME, names[nameVal]);

        int titleVal = (int) (Math.random() * titles.length);
        values.put(DbEntry.COLNAME_TITLE, titles[titleVal]);

        //Chooses a date from August 1 to August 10
        //int dateDay = (int)(Math.random() * 10 + 1);
        //FOR NOW choosing the date of Aug 1 or 2 only
        int dateDay = (int)(Math.random() * 2 + 1);
        String date = dateMonth + " " + dateDay;
        values.put(DbEntry.COLNAME_DATE, date);


        db.insert(DbEntry.TABLE_NAME, null, values);
        db.close();
    }

    public Cursor getAll() {
        if (db == null || !db.isOpen()) {
            if (mDbHelper == null) {
                mDbHelper = new DbHelper(getActivity());
            }
            db = mDbHelper.getWritableDatabase();
        }

       String[] projection = { DbEntry._ID,
               DbEntry.COLNAME_TITLE,
               DbEntry.COLNAME_NAME,
               DbEntry.COLNAME_DATE
       };

        String sortOrder = null;

        //TODO delete the sortOrder below
        //String sortOrder = RecordingEntry.COLNAME_TAG_FULL
        //        + " DESC";
        String[] selectionArgs = null;
        String selection = null; // Setting this to null

        Cursor c = db.query(DbEntry.TABLE_NAME,
                projection,
                selection,
                selectionArgs,
                null, // don't group the rows
                null, // don't filter by row groups
                sortOrder
        );

        return c;
    }

}

EDIT:基于@ N.T。的建议,我使用的解决方案将MainFragment的bindView部分的结尾替换为以下内容:

        String lastDate = "";
        int position = c.getPosition();
        if (position != 0 ) {
            c.moveToPosition(position - 1);
            lastDate = c.getString(c.getColumnIndex(DbEntry.COLNAME_DATE));
            c.moveToPosition(position);
        }
        if (position == 0 || !date.equals(lastDate)) {
            separator.setVisibility(View.VISIBLE);
            separator.setText(date);
        } else {
            separator.setVisibility(View.GONE);
        }
android android-listview android-cursoradapter
1个回答
2
投票

您无法控制newView / bindView的调用顺序。假设您要在屏幕上显示项目10-20,最后绘制的是一个表格位置20,因此您的lastDate将指向该位置。现在您向上滚动,系统将要求您绘制项目9。此时,您需要在项目#9和lastDate之间进行比较,以指向项目#20,这不是您想要的。

要解决问题,您需要将光标移至上一个位置(mCursor.moveToPosition(position - 1)),然后将当前项目与列表中的上一个项目进行检查,而不是与先前请求的绘制项目进行比较。

© www.soinside.com 2019 - 2024. All rights reserved.