Android: использование SimpleCursorAdapter - PullRequest
1 голос
/ 29 января 2011

У меня ОДНА раздражающая проблема с SimpleCursorAdapter. Моя программа имеет вид списка и ListActivity. Каждый ряд имеет свой собственный макет:

<TableLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_height="wrap_content" android:layout_width="fill_parent"
android:orientation="horizontal" android:weightSum="1.0">
<TableRow>
    <TextView android:id="@+id/task_time"
        android:layout_width="wrap_content" android:layout_height="wrap_content"
        android:textSize="24sp" android:text="Time">
    </TextView>
    <LinearLayout android:orientation="vertical"
        android:layout_width="wrap_content"
        android:layout_height="fill_parent">
        <TextView android:id="@+id/task_name"
            android:layout_width="wrap_content" android:layout_height="wrap_content"
            android:textSize="20sp" android:text="Name">
        </TextView>
        <TextView android:id="@+id/task_categoty"
            android:layout_width="wrap_content" android:layout_height="wrap_content"
            android:text="Category" android:textSize="12sp">
        </TextView>
    </LinearLayout>
    <TextView android:id="@+id/task_state"
        android:layout_width="wrap_content" android:layout_height="wrap_content"
        android:text="State" android:textSize="12sp">
    </TextView>
    <CheckBox android:id="@+id/task_enabled"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" android:focusable="false">
    </CheckBox>

</TableRow>

Задачи хранятся в базе данных SQLite. У меня есть объект DAO (синглтон) для доступа к базе данных. TaskDao:

    public void updateEnabled(int id, boolean enabled){
    SQLiteDatabase db = dbHelper.getWritableDatabase();
    ContentValues cv = new ContentValues();
    cv.put(ENABLED_COLUMN, enabled==true?1:0);
    Log.i(TAG, "update to " + cv.get(ENABLED_COLUMN) );
    try{
        db.beginTransaction();
        db.update(TASK_TABLE, cv, ID_COLUMN+"=?", new String[]{id+""});
        db.setTransactionSuccessful();
    } catch (SQLException e) {
        Log.i(TAG, "edit task failed!");
    } finally {
        db.endTransaction();
        if (db != null)
            db.close();
    }
}

и метод Cursor для ListActivity:

    public Cursor getTasks(){
    SQLiteDatabase db = dbHelper.getReadableDatabase();
    return db.query(TASK_TABLE, COLUMNS, null, null, null, null, NAME_COLUMN);
}

Я расширил SimpleCursorAdapter (TaskDbAdapter) следующим образом:

    @Override
public View getView(int position, View convertView, ViewGroup parent) {
    if(convertView==null){
        convertView = inflater.inflate(R.layout.task_list_row, null);
    }
    Cursor c = getCursor();
    c.moveToPosition(position);
    Log.i(TAG, "getView " + position + " = " + c.getInt(enabledIdx));
    enabled.setTag(c.getInt(c.getColumnIndex(BaseColumns._ID)));
    enabled.setChecked(c.getInt(enabledIdx)>0?true:false);
    enabled.setOnClickListener(this);
    return convertView;
}
@Override
public void onClick(View v) {
    CheckBox box = (CheckBox) v;
    Integer id = (Integer)box.getTag();
    TaskDao.getInstance(context).updateEnabled(id.intValue(), box.isChecked());
}

И, наконец, я использую все вышеперечисленное в своей основной ListActivity

    private void refreshList(){
    c = TaskDao.getInstance(this).getTasks();
    startManagingCursor(c);
    adapter = new TaskDbAdapter(this, R.layout.task_list_row, c, new String[]{TaskDao.ENABLED_COLUMN}, new int[]{R.id.task_enabled});
    setListAdapter(adapter);
}
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.task);
    getListView().setItemsCanFocus(false);
    getListView().setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);
    getListView().setVerticalScrollBarEnabled(true);
    registerForContextMenu(getListView());
    getListView().setOnCreateContextMenuListener(this);
    refreshList();
}


@Override
protected void onResume() {
    super.onResume();
    refreshList();
}
@Override
protected void onPause() {
    super.onPause();

}

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

getView 0 = 0
getView 2 = 0
getView 3 = 0

1021 * тогда *

uptate to 1

и затем (когда я прокручиваю до первого элемента)

getView 0 = 0
getView 2 = 0
getView 3 = 0

Я попытался сделать getCursor () .query (); в моем методе TaskDbAdapter onClick. Но тогда я не увидел никаких пунктов в списке! И исключение из-за управления курсором (соединение было закрыто с помощью Android). Когда я пишу startManagingCursor (c); в методе refreshList (), методы проверки и снятия флажка не работают. Пожалуйста, помогите!

Ответы [ 2 ]

1 голос
/ 09 апреля 2011

Я тоже с этим боролся.Я закончил тем, что сохранил все отмеченные поля в базе данных как 0 или 1. Затем я проверяю их состояние из базы данных, чтобы определить, отмечены они или нет.

открытый класс DetailCursorAdapter extends SimpleCursorAdapter {

private Cursor c;
private Context context;    

public DetailCursorAdapter(Context context, int layout, Cursor c,
        String[] from, int[] to) {
    super(context, layout, c, from, to);
    this.c = c;
        this.context = context;

}

public View getView(int pos, View inView, ViewGroup parent) {
    View v = inView;
    if (v == null) {
        LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        v = inflater.inflate(R.layout.check_list, null);
    }
    Log.i("pos = ..................", "pos = "+pos);
    this.c.moveToPosition(pos); 
    //this.c.moveToPosition(this.c.getInt(this.c.getColumnIndex("_id")));   
    CheckBox cBox = (CheckBox) v.findViewById(R.id.bcheck);
    cBox.setTag(this.c.getInt(this.c.getColumnIndex("_id")));

    /*
     * when reloading the list, check for chkd status, this is broken.  Need to query db directly.
     */
    EventDbAdapter mDbHelper = new EventDbAdapter(context);
    mDbHelper.open(); 

    int idTag = (Integer) cBox.getTag();                
    int checked = mDbHelper.selectChk(idTag);
    mDbHelper.close();
    Log.i("results from selectChk.....................", ""+checked);
    if (checked == 1) {
        cBox.setChecked(true);          
    } else {
        cBox.setChecked(false);
    }

    /*
     * Populate the list
     */     
    TextView txtdateTime = (TextView)v.findViewById(R.id.time);
    txtdateTime.setText(this.c.getString(this.c.getColumnIndex("time")));   
    TextView txtdateEvent = (TextView)v.findViewById(R.id.event);
    txtdateEvent.setText(this.c.getString(this.c.getColumnIndex("event")));
    TextView txtdateLocation = (TextView)v.findViewById(R.id.location);
    txtdateLocation.setText(this.c.getString(this.c.getColumnIndex("location")));
    ImageView arrow = (ImageView) v.findViewById(R.id.arrowId);
    arrow.setImageResource(R.drawable.rightarrow);


    Log.i("if chk in db is = 1 then set checked.........",this.c.getString(this.c.getColumnIndex("checked")) +" " +this.c.getString(this.c.getColumnIndex("time")));        


    /*
     * Controls action based on clicked list item (background)
     */
    View lv = v.getRootView(); 
    lv.setOnClickListener(new OnClickListener() {

        @Override
        public void onClick(View lv) {
            CheckBox cBox = (CheckBox) lv.findViewById(R.id.bcheck);

            // id holds the rowid of each event.  pass this to a new activity to query for description

            // Call Event Detail
            String id = cBox.getTag().toString();
            Intent i = new Intent(context, EventDetail.class);
            //i.putExtra("description", c.getString(c.getColumnIndex("description")));
            i.putExtra("_id", id);
            context.startActivity(i);

        }

    });

    /*
     * Begin - Controls action based on clicked Text only

    txtdateEvent.setOnClickListener(new OnClickListener() {

        @Override
        public void onClick(View v) {
            // TODO Auto-generated method stub
            CharSequence charseq = "Darth Vader is alive";
            Toast.makeText(context, charseq, Toast.LENGTH_SHORT).show();
        }

    });

    * End - Controls action based on clicked Text only
    */


    /*
     * Controls action based on clicked checkbox 
     */
    cBox.setOnClickListener(new OnClickListener() {  
        @Override
        public void onClick(View v) {
            EventDbAdapter mDbHelper = new EventDbAdapter(context);
            mDbHelper.open(); 

            CheckBox cBox = (CheckBox) v.findViewById(R.id.bcheck);
            if (cBox.isChecked()) {
                //cBox.setChecked(false);
                CharSequence charseq = "Added to My Schedule";
                Toast.makeText(context, charseq, Toast.LENGTH_SHORT).show();

                // Update the database for each checked item
                mDbHelper.updateChecked(cBox.getTag().toString(), "1");     
                c.requery();

                // Verify that the db was updated for debugging purposes
                String event = c.getString(c.getColumnIndex("event"));                  
                int id = (Integer) cBox.getTag();

                Log.i("checked _id...........", "id= " + id + " " +c.getString(c.getColumnIndex("_id"))); 
                Log.i("checked checked...........", ""+c.getString(c.getColumnIndex("checked")));

            } else if (!cBox.isChecked()) {
                //cBox.setChecked(true);
                CharSequence charseq = "Removed from My Schedule";
                Toast.makeText(context, charseq, Toast.LENGTH_SHORT).show();
                // checkList.remove(cBox.getTag());
                //checkList.add((Integer) cBox.getTag());
                String event = c.getString(c.getColumnIndex("event"));
                //int id = c.getInt(c.getColumnIndex("_id"));
                int id = (Integer) cBox.getTag();
                mDbHelper.updateChecked(cBox.getTag().toString(), "0"); 
                c.requery();
                //int sqlresult = mDbHelper.selectChk(id, event);                   
                //Log.i("sqlresult checked value after update...........", ""+ sqlresult);                  
                //Log.i("unchecked _id...........", ""+c.getString(c.getColumnIndex("_id"))); 
                //Log.i("unchecked checked...........", ""+c.getString(c.getColumnIndex("checked")));
            }
            //mDbHelper.close();
        }
    });

    return(v);
}

}

1 голос
/ 29 января 2011

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

Ознакомьтесь с документацией BaseAdapter класса.

public void notifyDataSetChanged () 

может делать работу.

Вы также можете зарегистрировать Observer для этого ...

public void registerDataSetObserver (DataSetObserver observer)
...