您的位置:首页 > 其它

由群发短信功能学习:CheckBox和ListView组合使用的方法和注意事项.

2014-07-25 14:26 176 查看
转载请注明出处和作者,谢谢!!!

作为一个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();
	}
}


有哪里说的不好的,或者代码哪里可以改进的欢迎各位指出。很愿意和大家讨论,共同进步。谢谢~
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: