返回介绍

Android 开发:setContentView 切换界面,自定义带 CheckBox 的 ListView 显示 SQlite 条目 - 实现

发布于 2025-02-26 12:46:19 字数 11415 浏览 0 评论 0 收藏 0

问题背景:

我在其他 Activity 里有一个数据库,里面有若干条目,数据库里存的是最简单的“名字”string 类型的信息。我在另外一个 Activity 里,通过按键 Button,显示出一个带 checkbox 的列表,显示出数据库里的姓名,然后可以选中多个。类似于文件夹删除的功能。

下面是实现:

第一部分,在布局文件夹下新建一个 my_checkbox.xml.

这个布局是用来控制将来 listview 里,每一行怎么显示。在这里我是左边显示名字,右边显示复选框 CheckBox。

<?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="wrap_content"
android:orientation="horizontal" >
<TextView
android:id="@+id/item_text"
android:textSize="25dip"
android:layout_weight="1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
<CheckBox
android:id="@+id/item_check"
android:textSize="25dip"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:clickable="false"
android:focusable="false"
android:checkMark="?android:attr/listChoiceIndicatorMultiple"/>

注意:

1,上面的 TextView 里的 layout_weight=1 用来实现最左边显示名字,最右边显示 CheckBox。

2,由于 CheckBox 的响应优先级高于 ListView,这里需要把 CheckBox 的 clickable 和 focuseable 属性都关闭。将来只通过 listview 的 item 是否点击来判断。

3,CheckBox 的 checkMark 用来设置当选中之后,是个什么效果。

第二部分:新建一个布局 list_check.xml,用来显示数据库的条目。当在主 Activity 里点击后就会切换到这个界面。

<?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="wrap_content"
android:orientation="vertical" >
<Button
android:id="@+id/confirmBtn"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="确定" />
<ListView
android:id="@+id/checkList"
android:layout_width="fill_parent"
android:layout_height="wrap_content" >

注: 上面的 button 是选中若干条目后的确定按键,同时也是返回按键,返回到主界面。

第三部分 :在主 Activity 里设置利用 setContentView 的方法切换页面。这里主 Activity 的布局文件就不提供了。下面是主 Activity 的代码:

//为了实现新的布局
  Button mChoseBtn = null;
  Button mConfirmBtn = null;

  boolean firstFlag = true;
  ListView list2 = null;
  View checkListView = null;
  View mainView = null;
  /**Called when the activity is first created. */
  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    LayoutInflater inflater = this.getLayoutInflater();
    checkListView = inflater.inflate(R.layout.list_check, null);
    mainView = inflater.inflate(R.layout.main, null);

    setContentView(mainView);
    //切换布局监听
    mChoseBtn = (Button)mainView.findViewById(R.id.choseBtn);
    mChoseBtn.setOnClickListener(new ButtonListener());
    setUpViews();

  }

  class ButtonListener implements OnClickListener{

    public void onClick(View v) {
      // TODO Auto-generated method stub
      switch (v.getId()){
      case R.id.choseBtn:

        Jump2CheckList();
        break;
      case R.id.confirmBtn:
        String s = getCheckInfo();
        showToast("您选中的姓名有:"+ s);
        Jump2Main();
        break;
      default:
        break;
      }
    }

  }
  /*切换到主布局*/
  public void Jump2Main(){
    setContentView(mainView);     
    setUpViews();
  }
  /*切换到选中布局*/
  public void Jump2CheckList(){
    setContentView(checkListView);
    if(firstFlag){
      mConfirmBtn = (Button)checkListView.findViewById(R.id.confirmBtn);
      mConfirmBtn.setOnClickListener(new ButtonListener());
      firstFlag = false;
    }
    initCheckList();

  }

第四部分:给 ListView 写适配器 ,其实很简单 也就是上面的 initCheckList 函数,包含初始化 ListView 和适配器两个部分。先看源码:

  public void initCheckList(){
    list2 = (ListView)(checkListView).findViewById(R.id.checkList);
    list2.setItemsCanFocus(false);
    list2.setAdapter(new CheckListAdapter(this, cursor));
    list2.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);
    list2.setOnItemClickListener(new OnItemClickListener() {

      public void onItemClick(AdapterView<?> arg0, View view, int positon,
          long id) {
        // TODO Auto-generated method stub
        ViewHolder vHolder = (ViewHolder) view.getTag();
        vHolder.check.toggle();
        isSelected.put(positon, vHolder.check.isChecked());

      }
    });
  }

下面是适配器:

/*给 CheckList 设置适配器*/
  public static  Map<Integer, Boolean> isSelected;
  public  class CheckListAdapter extends BaseAdapter{

    private Context mContext;
    private Cursor mCursor;

    //构造函数
    public CheckListAdapter(Context context, Cursor cursor){
      mContext = context;
      mCursor = cursor;

      isSelected = new HashMap<Integer, Boolean>();
      for(int i=0; i<mCursor.getCount(); i++){
        isSelected.put(i, false);
      }
    }

    public int getCount() {
      // TODO Auto-generated method stub
      return cursor.getCount();
    }

    public Object getItem(int arg0) {
      // TODO Auto-generated method stub
      return null;
    }

    public long getItemId(int arg0) {
      // TODO Auto-generated method stub
      return 0;
    }

    public View getView(int position, View convertView, ViewGroup arg2) {
      // TODO Auto-generated method stub
      ViewHolder holder = null;
      if(convertView == null){
        holder = new ViewHolder();
        LayoutInflater inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        convertView = inflater.inflate(R.layout.my_checkbox, null);
        holder.text = (TextView) convertView.findViewById(R.id.item_text);
        holder.check = (CheckBox)convertView.findViewById(R.id.item_check);
        convertView.setTag(holder);
      }
      else
      {
        holder = (ViewHolder)convertView.getTag();
      }

      mCursor.moveToPosition(position);
      holder.text.setText(Integer.toString(mCursor.getInt(0)));
      holder.text.append(mCursor.getString(1));
      holder.check.setChecked(isSelected.get(position));
      return convertView;

    }

    public final class ViewHolder{
      public TextView text;
      public CheckBox check;
    }

  }

注:

1,initCheckList 里要设置相应的参数,如多选等。

2, public static Map<Integer, Boolean> isSelected; ​这是一个 全局变量 ,且是静态的,用来存储 checkbox 的选中状态。 http://mobile.51cto.com/android-254823.htm 这里将其设成适配器里的一个静态变量,但奇怪的是到我这就不中了,暂且弄成全局的吧。

3,在适配器的构造函数里初始化上面这个变量,并且所有都设成未选中。

isSelected = new HashMap<Integer, Boolean>();
for(int i=0; i<mCursor.getCount(); i++){
isSelected.put(i, false);

4,由于我跟数据库做了关联,所以在适配器的构造函数里传进去一个 cursor,关于 cursor 的理解可以参考 http://www.cnblogs.com/TerryBlog/archive/2010/07/05/1771459.html 说白了他就是一组信息,是个集合。通过 cursor.getCount 获得他有多少行的信息。cursor.MoveToposition(i) 定位到第 i 行。获得数据的方法跟数据时建的表的结构有关。

5,

public final class ViewHolder{
public TextView text;
public CheckBox check;
}

至于这个 viewholder,实际上不用也可以。他就是把每一行的元素集成了一下,跟布局相对应。但到后来莫名其妙的要用到 View.setTag() 和 View.getTag() 来传递数据,所以我又把他加上了。

6,适配器的两大关键。

第一个是

public int getCount() {
// TODO Auto-generated method stub
return cursor.getCount();
}

这里是返回 list 有多少行。调用 ListView.getCount() 实际上就是在调用这个函数。

第二个就是public View getView(int position, View convertView, ViewGroup arg2) ​ 这个函数。注意第二个参数 convertView 就是给每一行设置的布局。通过

LayoutInflater inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
convertView = inflater.inflate(R.layout.my_checkbox, null);

将第一个布局文件,和 view 关联起来。至于 convertView.setTag(holder); ​其实是给 view 贴标签,也就是传数据的一种方式。当这个布局不为空时通过 holder = (ViewHolder)convertView.getTag(); ​直接获得。mCursor 包含所有行的信息,

mCursor.moveToPosition(position); ​这句话将其定位到 position 的位置:

holder.text.setText(Integer.toString(mCursor.getInt(0)));
holder.text.append(mCursor.getString(1));

这两句话是获得每行的信息。我这个表的结构是第一列(对应索引为 0)是一个 int,第二列(索引为 1)是一个 string。

holder.check.setChecked(isSelected.get(position)); ​这句话是设置 checkbox 的状态。 也就是说通过 isSelected 来设定,他的初始态是全不选。所以每次跳转时,默认的是都不选。

最后程序 return convertView;返回这个 view,然后当下次调用时,他又当做参数传进来。

7,解释下 listview 设置监听:

list2.setOnItemClickListener(new OnItemClickListener() {
public void onItemClick(AdapterView<?> arg0, View view, int positon,long id) {
// TODO Auto-generated method stub
ViewHolder vHolder = (ViewHolder) view.getTag();
vHolder.check.toggle();
isSelected.put(positon, vHolder.check.isChecked());
}
});

当每个 item 被点击时,通过被点击的 view.getTag 获得具体的控件 view, CheckBox 的状态反转。 isSelected.put(positon, vHolder.check.isChecked()); ​这句话是通过访问 position 位置的 CheckBox 的状态来更新保存信息的 isSelected。

8, 最后就是我怎么获得选中的信息?

可以这么写:

OnClickListener bPop = new OnClickListener() {  
    @Override  
    public void onClick(View v) {  
      for(int i=0;i<list.getCount();i++){  
        if(MyAdapter.isSelected.get(i)){  
          ViewHolder vHollder = (ViewHolder) list.getChildAt(i).getTag();  
Log.i(TAG, "--onClick --"+vHollder.title.getText());  
        }  
      }  
    }  
  }; 

通过 list.getCount 进行遍历,通过 list.getChildAt(i).getTag ​得到被选中的 ViewHolder,然后得到信息。 因为这里我跟数据库挂了钩,所以我直接读 Cursor 里面的信息就可以了。

我的写法是:

  public String getCheckInfo()
  {
    String info = "";
    for(int i=0; i<list2.getCount(); i++){
      if(isSelected.get(i)){
        //ViewHolder holder = (ViewHolder)list2.getChildAt(i).getTag();
        cursor.moveToPosition(i);
        info+=cursor.getInt(0)+".";
      }
    }
    return info;
  }

上面的 list2.getCount 和 cursor.getCount 是一样的效果。

最后说下这个 Cursor,因为他包含了数据库里所有行的信息,所以我直接用他来填充到每个 item。如果这个填充的信息是其他的,就是用到其他数据结构了,如 List<Map<String, Object>> mData; ​来保存信息。具体可以参考:

我主要参考的第一篇。

另外,在数据库里怎么获得 Cursor 呢?

cursor = mPalmDB.select();

select() 函数的封装是:

  public Cursor select(){   
    SQLiteDatabase db = this.getReadableDatabase();
    Cursor cursor = db.query(TABLE_NAME, null, null, null, null, null, null);
    return cursor;

  }

这个可以封装在数据库类里,也可以写到函数里。源码连同数据库操作部分改日再提供哈!

源码下载: http://download.csdn.net/detail/yanzi1225627/5226894

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文