由群发短信功能学习:CheckBox和ListView组合使用的方法和注意事项.
2014-07-25 14:26
176 查看
转载请注明出处和作者,谢谢!!!
作为一个Android开发的小菜鸟,假期想通过做些想做的东西来锻炼自己的能力。想到小伙伴曾经说过想要一个群发短信的App,就开始做了。
我先是思考了群发短信和单独发短信需要的主要就是:避免重复操作。所以我想要一个能够看到联系人电话号码的列表,并且能在这个列表里面选择联系人。然后输入短信内容直接发送就OK了。然后就开始做了,经历了许多麻烦后终于成功,下面是正文:
下面我先放一个虚拟机(Nexus One)上能用,但是手机(三星、小米等等)上不能正常运行的源码。这是《疯狂Android讲义》的源码。和大家共同学习和思考问题所在。
先放他的Xml文件:
main.xml:
list.xml:
下面是代码:
这个具体运行效果的截图:
在手机上运行是会报错的,是NullPointerException,报错的位置是:
CheckBox checkBox = (CheckBox) listView.getChildAt(i);
在网上搜了会,解释比较少。靠谱的解释是:
在ListView中,使用getChildAt(index)的取值,只能是当前可见区域(列表可滚动)的子项! 所以如果想获取前部的将会出现返回Null值空指针问题;。
那么我们要怎么解释这个问题呢?我的解决方案是(明明是百度参考了别人的思路啊喂!),重写BaseAdapter,改变记录CheckBox勾选状态的方式。
下面是具体实现代码和思路:(在此先感谢 wangjia55---博客地址:http://blog.csdn.net/wangjia55/article/details/7905491
这篇文章的启发和指导)
xml文件:
刚刚的main.xml和list.xml我们还是可以用的,就不写了……不过我们现在需要一个xml,装我们ListView里的Item,XML里用LinearLayout装一个TextView和一个CheckBox就行了。
但是这里CheckBox我们应该设置成下边这样,因为我打算用ListView的OnItemClickListener去改变CheckBox的勾选状态。
android:clickable="false"
android:focusable="false"
android:focusableInTouchMode="false"
下面是代码:
一、重写BaseAdapter:
二、重写实现群发的MainActivity:
有哪里说的不好的,或者代码哪里可以改进的欢迎各位指出。很愿意和大家讨论,共同进步。谢谢~
作为一个Android开发的小菜鸟,假期想通过做些想做的东西来锻炼自己的能力。想到小伙伴曾经说过想要一个群发短信的App,就开始做了。
我先是思考了群发短信和单独发短信需要的主要就是:避免重复操作。所以我想要一个能够看到联系人电话号码的列表,并且能在这个列表里面选择联系人。然后输入短信内容直接发送就OK了。然后就开始做了,经历了许多麻烦后终于成功,下面是正文:
下面我先放一个虚拟机(Nexus One)上能用,但是手机(三星、小米等等)上不能正常运行的源码。这是《疯狂Android讲义》的源码。和大家共同学习和思考问题所在。
先放他的Xml文件:
main.xml:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <Button android:id="@+id/select" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/select" /> <EditText android:id="@+id/numbers" android:layout_width="fill_parent" android:layout_height="wrap_content" android:cursorVisible="false" android:ellipsize="end" android:lines="3" /> <TextView android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="@string/content" android:textSize="16dp" /> <EditText android:id="@+id/content" android:layout_width="fill_parent" android:layout_height="wrap_content" android:lines="6" /> <LinearLayout android:orientation="horizontal" android:layout_width="fill_parent" android:layout_height="fill_parent" android:gravity="center_horizontal" > <Button android:id="@+id/send" android:layout_width="wrap_content" android:layout_height="wrap_content" android:gravity="bottom" android:text="@string/send" android:textSize="35dp" /> </LinearLayout> </LinearLayout>
list.xml:
<?xml version="1.0" encoding="UTF-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <ListView android:id="@+id/list" android:layout_width="fill_parent" android:layout_height="wrap_content" /> </LinearLayout>
下面是代码:
package org.crazyit.manager; import java.util.ArrayList; import java.util.List; import android.app.Activity; import android.app.AlertDialog; import android.app.PendingIntent; import android.content.DialogInterface; import android.content.Intent; import android.database.Cursor; import android.os.Bundle; import android.provider.ContactsContract; import android.provider.ContactsContract.PhoneLookup; import android.telephony.SmsManager; import android.view.View; import android.view.ViewGroup; import android.view.View.OnClickListener; import android.widget.BaseAdapter; import android.widget.Button; import android.widget.CheckBox; import android.widget.EditText; import android.widget.ListView; import android.widget.TextView; import android.widget.Toast; public class GroupSend extends Activity { EditText content; TextView numbers; Button select, send; SmsManager sManager; // 记录需要群发的号码列表 ArrayList<String> sendList = new ArrayList<String>(); @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); sManager = SmsManager.getDefault(); numbers = (TextView) findViewById(R.id.numbers); content = (EditText) findViewById(R.id.content); select = (Button) findViewById(R.id.select); send = (Button) findViewById(R.id.send); send.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { for (String number : sendList) { String groupContent = content.getText().toString(); PendingIntent pi = PendingIntent.getActivity( GroupSend.this, 0, new Intent(), 0); if(groupContent.length()>70){ List<String> ms = sManager.divideMessage(groupContent); for(String str : ms){ sManager.sendTextMessage(number, null, str, pi, null); } } else{ sManager.sendTextMessage(number, null, groupContent, pi, null); } } Toast.makeText(GroupSend.this, "短信群发完成" , Toast.LENGTH_SHORT).show(); } }); select.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { final Cursor cursor = getContentResolver().query( ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, null, null, null); BaseAdapter adapter = new BaseAdapter() { @Override public int getCount() { return cursor.getCount(); } @Override public Object getItem(int position) { return position; } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { cursor.moveToPosition(position); CheckBox rb = new CheckBox(GroupSend.this); String number = cursor .getString(cursor.getColumnIndex(ContactsContract .CommonDataKinds.Phone.NUMBER)) .replace("-", "") .replace(" " , ""); String name = cursor.getString(cursor.getColumnIndex(PhoneLookup.DISPLAY_NAME)); rb.setText(name + " " + number); if (isChecked(number)) { rb.setChecked(true); } return rb; } }; View selectView = getLayoutInflater().inflate( R.layout.list, null); final ListView listView = (ListView) selectView .findViewById(R.id.list); listView.setAdapter(adapter); adapter.notifyDataSetChanged(); AlertDialog.Builder ad = new AlertDialog.Builder(GroupSend.this); ad.setView(selectView) .setPositiveButton("确定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { sendList.clear(); for (int i = 0; i < listView.getCount(); i++) { CheckBox checkBox = (CheckBox) listView .getChildAt(i); if (checkBox.isChecked()) { sendList.add(checkBox.getText() .toString()); } } numbers.setText(sendList.toString()); } }).show(); } }); } public boolean isChecked(String phone) { for (String s1 : sendList) { if (s1.equals(phone)) { return true; } } return false; } }
这个具体运行效果的截图:
在手机上运行是会报错的,是NullPointerException,报错的位置是:
CheckBox checkBox = (CheckBox) listView.getChildAt(i);
在网上搜了会,解释比较少。靠谱的解释是:
在ListView中,使用getChildAt(index)的取值,只能是当前可见区域(列表可滚动)的子项! 所以如果想获取前部的将会出现返回Null值空指针问题;。
那么我们要怎么解释这个问题呢?我的解决方案是(明明是百度参考了别人的思路啊喂!),重写BaseAdapter,改变记录CheckBox勾选状态的方式。
下面是具体实现代码和思路:(在此先感谢 wangjia55---博客地址:http://blog.csdn.net/wangjia55/article/details/7905491
这篇文章的启发和指导)
xml文件:
刚刚的main.xml和list.xml我们还是可以用的,就不写了……不过我们现在需要一个xml,装我们ListView里的Item,XML里用LinearLayout装一个TextView和一个CheckBox就行了。
但是这里CheckBox我们应该设置成下边这样,因为我打算用ListView的OnItemClickListener去改变CheckBox的勾选状态。
android:clickable="false"
android:focusable="false"
android:focusableInTouchMode="false"
下面是代码:
一、重写BaseAdapter:
package com.example.groupsendfinal; import java.util.HashMap; import android.content.Context; import android.database.Cursor; import android.provider.ContactsContract; import android.provider.ContactsContract.PhoneLookup; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.CheckBox; import android.widget.TextView; public class MyBaseAdapter extends BaseAdapter{ //用inflater装TextView和CheckBox的lisr_item.xml private LayoutInflater inflater; private Context context; //isSelected用于记录CheckBox的勾选状态 private static HashMap<Integer,Boolean> isSelected; //cursor用于获取通讯录里的联系人 private Cursor cursor; //初始化数据 public void myAdapter(Cursor cursor, Context context){ this.cursor = cursor; this.context = context; inflater = LayoutInflater.from(context); isSelected = new HashMap<Integer, Boolean>(); initData(); } public void initData(){ //默认所有人刚开始都是未被选中的 for(int i = 0;i < cursor.getCount();i++){ getIsSelected().put(i,false); } } public HashMap<Integer, Boolean> getIsSelected(){ return isSelected; } @Override public int getCount() { //获得联系人数量 return cursor.getCount(); } @Override public Object getItem(int position) { //获得联系人在List的位置 return position; } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { //ViewHolder是一个只有TextView和CheckBox对象的内部类 //主要是为了避免一次又一次的findViewById,从而提高效率 ViewHolder viewHolder = null; //获得对应位置的联系人信息 cursor.moveToPosition(position); if(convertView == null){ viewHolder = new ViewHolder(); convertView = inflater.inflate(R.layout.listview_item, null); viewHolder.checkBox = (CheckBox)convertView.findViewById(R.id.checkBox); viewHolder.textView = (TextView)convertView.findViewById(R.id.textView); convertView.setTag(viewHolder); } else{ viewHolder = (ViewHolder)convertView.getTag(); } //number和name存储当前位置联系人的名字和手机号码 //在Activity里只要setAdapter后,通讯录里每个联系人的名字和手机号码都会存进number和name里了 //这里要记得把手机号码的字符给替换掉,不然发短信会失败哦 String number = cursor .getString(cursor.getColumnIndex(ContactsContract .CommonDataKinds.Phone.NUMBER)) .replace("-", "") .replace(" " , ""); String name = cursor.getString(cursor.getColumnIndex(PhoneLookup.DISPLAY_NAME)); //让ListView每一个Item都获得对应的联系人名字和手机号码 viewHolder.textView.setText(name + " " + number); //初始化CheckBox的勾选状态,默认为未被勾选 viewHolder.checkBox.setChecked(getIsSelected().get(position)); return convertView; } //这个方法可以设置CheckBox的勾选状态,如果需要的话可以自己加上全选、全不选的功能哦 public static void setIsSelected(HashMap<Integer, Boolean> isSelected){ MyBaseAdapter.isSelected = isSelected; } public class ViewHolder{ public CheckBox checkBox; public TextView textView; } }
二、重写实现群发的MainActivity:
package com.example.groupsendfinal; import java.util.ArrayList; import java.util.List; import com.example.groupsendfinal.MyBaseAdapter.ViewHolder; import android.app.Activity; import android.app.AlertDialog; import android.app.PendingIntent; import android.content.DialogInterface; import android.content.Intent; import android.database.Cursor; import android.os.Bundle; import android.provider.ContactsContract; import android.provider.ContactsContract.PhoneLookup; import android.telephony.SmsManager; import android.view.View; import android.view.View.OnClickListener; import android.widget.AdapterView; import android.widget.AdapterView.OnItemClickListener; import android.widget.Button; import android.widget.EditText; import android.widget.ListView; import android.widget.TextView; import android.widget.Toast; public class MainActivity extends Activity{ private ListView listview; private MyBaseAdapter madapter; //sendList用于存储收群发短信的人的名字和手机号码 private ArrayList<String> sendlist = new ArrayList<>(); private SmsManager manager; //显示收取短信的人的信息的TextView private TextView numbers; //输入短信内容的EditText private EditText contents; //选择联系人的Button private Button select; //发送短信的Button private Button send; //当短信长度>70,就把短信拆分 private List<String> dividedMs; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main_activity); manager = SmsManager.getDefault(); numbers = (TextView)findViewById(R.id.numbers); contents = (EditText)findViewById(R.id.groupContent); select = (Button)findViewById(R.id.select); send = (Button)findViewById(R.id.send); send.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { //用groupContent存储我们输入的短信内容 String groupContent = contents.getText().toString(); PendingIntent pi = PendingIntent.getActivity( MainActivity.this, 0, new Intent(), 0); //如果短信长度>70,就拆分短信 if(groupContent.length()>70){ dividedMs = manager.divideMessage(groupContent); } //这里就是调用API群发短信咯 for (String num : sendlist) { if(groupContent.length()>70){ for(String str : dividedMs){ manager.sendTextMessage(num, null, str, pi, null); } } else{ manager.sendTextMessage(num, null, groupContent, pi, null); } } //短信完成后把TextView和EditText内容清空 AlertDialog.Builder alert = new AlertDialog.Builder(MainActivity.this); alert.setTitle("温馨小提示~") .setMessage("短信群发完成~") .setPositiveButton("确定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { contents.setText(""); numbers.setText(""); } }).show(); } }); select.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { //用final的cursor存储通讯录所有联系人的信息 final Cursor cursor = getContentResolver().query(ContactsContract.CommonDataKinds .Phone.CONTENT_URI, null, null, null, null); //selectView是我们选择联系人的时候弹出来的View View selectView = getLayoutInflater().inflate(R.layout.item, null); //清空sendlist sendlist.clear(); //初始化MyBaseAdapter的实例madapter madapter = new MyBaseAdapter(); madapter.myAdapter(cursor, getApplicationContext()); listview = (ListView)selectView.findViewById(R.id.list); listview.setAdapter(madapter); //通过listview的OnItemClickListener方法,改变CheckBox的勾选状态 listview.setOnItemClickListener(new OnItemClickListener() { @Override public void onItemClick(AdapterView<?> adapterView, View view, int position, long id) { // 取得ViewHolder对象,省去多次findViewById实例化CheckBox ViewHolder holder = (ViewHolder)view.getTag(); // 改变CheckBox的状态 holder.checkBox.toggle(); // 将CheckBox的选中状况记录下来 madapter.getIsSelected().put(position, holder.checkBox.isChecked()); //如果某个position的CheckBox被勾选,我们就把这个position对应的联系人信息取出来 //然后存入sendlist里 if(holder.checkBox.isChecked()){ cursor.moveToPosition(position); sendlist.add(cursor.getString(cursor.getColumnIndex(PhoneLookup.DISPLAY_NAME)) + " " + cursor.getString(cursor.getColumnIndex(ContactsContract .CommonDataKinds.Phone.NUMBER)) .replace(" ", "") .replace("-", "")); } } }); //用AlertDialog显示选择联系人的View //点击“确定”后就把被选中的联系人的信息在numbers里显示出来 AlertDialog.Builder ad = new AlertDialog.Builder(MainActivity.this); ad.setView(selectView) .setPositiveButton("确定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dataChanged(); numbers.setText(sendlist.toString()); } }).show(); } }); } private void dataChanged(){ madapter.notifyDataSetChanged(); } }
有哪里说的不好的,或者代码哪里可以改进的欢迎各位指出。很愿意和大家讨论,共同进步。谢谢~
相关文章推荐
- web前台工作笔记(时间戳、js中clone的使用、js中动态填充数据注意事项、前台查错方法的学习)
- ExpandableListView+CheckBox使用注意事项
- cygwin跨平台移植 gcc+vc联合使用的方法和注意事项
- ScrolledComposite使用方法及注意事项
- 使用ResultSet的geRow()方法的注意事项
- [学习笔记]寄存器使用注意事项
- C++中typename关键字的使用方法和注意事项
- 分页解决方案 之 分页算法——Pager_SQL的详细使用方法和注意事项
- new与delete函数使用方法以及注意事项
- C++ Primer学习笔记:引用的使用注意事项
- C++中typename关键字的使用方法和注意事项
- 如何寻找使用案例及其注意事项,学习笔记
- C++ typename 关键字的使用方法和注意事项
- C++中typename关键字的使用方法和注意事项
- ruby中字符串的转码 使用方法以及注意事项
- VB StrConv方法使用注意事项
- STL中heap的使用方法及注意事项(msdn2003中未提及)
- jQuery-使用选择器注意事项 学习笔记十
- cvFloodFill的使用方法和注意事项。
- Dom4J使用XPath功能注意事项