转载请注明出处和作者,谢谢!!!
作为一个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();
}
}
有哪里说的不好的,或者代码哪里可以改进的欢迎各位指出。很愿意和大家讨论,共同进步。谢谢~