Android切近实战(八)

天冷了,老夫要把伙食搞上去,这不最近在软件园二楼吃,伙食15块,杠杠的。
java

wKiom1Rbd2eggnzrAAMtkr5vUJY784.jpg

wKioL1Rbd8yz1plfAAFubshtix0664.jpg


美包包,不说了,进入正题。今天老夫要讲的是读取联系人,其实我作这个的初衷是想作一个短信拦截,电话拦截的功能。android

咱们先看一下界面,仍是不错的,挺绚丽的。app

wKioL1RbeTHAM8hyAAIo4XZJDwQ669.jpg

OK,咱们一看就知道,这又是一个ListView。目前的功能是当你在复选框打钩,点击后面的拨号就会将电话打出去。若是不打钩,电话则不会拨出去。OK,咱们先看看页面布局ide

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
	android:layout_width="match_parent" android:layout_height="match_parent"
	android:orientation="vertical">
	<ListView android:id="@+id/contactListView" 
		android:descendantFocusability="blocksDescendants"
		android:layout_width="fill_parent"
		android:layout_height="fill_parent"
		android:divider="@color/teal"
		android:dividerHeight="1dp">
	</ListView>
	<LinearLayout 
		android:orientation="horizontal"
		android:layout_width="fill_parent"
		android:layout_height="wrap_content">
		<Button android:id="@+id/btnSelAll"
			android:text="@string/btnSelAll"
			android:textColor="@color/teal"
			android:textSize="14dp"
			android:textStyle="bold"
			android:layout_weight="1"
			android:layout_width="fill_parent"
			android:layout_height="fill_parent"></Button>
		<Button android:id="@+id/btnInverseSel"
			android:text="@string/btnSelInverse"
			android:textColor="@color/teal"
			android:layout_marginLeft="1dp"
			android:layout_weight="1"
			android:textSize="14dp"
			android:textStyle="bold"
			android:layout_width="fill_parent"
			android:layout_height="fill_parent"></Button>
		<Button android:id="@+id/btnSet"
			android:text="@string/btnSet"
			android:layout_weight="1"
			android:textColor="@color/teal"
			android:layout_marginLeft="1dp"
			android:textSize="14dp"
			android:textStyle="bold"
			android:layout_width="fill_parent"
			android:layout_height="fill_parent"></Button>
	</LinearLayout>
</LinearLayout>

咱们再看看ListView要加载的模版布局

<?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:id="@+id/contactTemplate">
	<TableLayout android:id="@+id/tabContatMain"
		android:layout_width="fill_parent" android:layout_height="wrap_content"
		android:stretchColumns="2" android:shrinkColumns="2" android:padding="3dip">
		<TableRow>
			<CheckBox android:id="@+id/chkContactUser" 
				android:layout_gravity="center_vertical"></CheckBox>
			<ImageView android:id="@+id/imgContactPhoto"
				android:layout_gravity="center_vertical" 
				android:scaleType="fitCenter"></ImageView>
			<TextView android:id="@+id/txtContactName"
				android:layout_marginLeft="10dp" 
				android:layout_gravity="center_vertical"
				android:textColor="@color/teal1"></TextView>
			<TextView android:id="@+id/txtContactTelNumber"
				android:layout_marginLeft="4dp" 
				android:gravity="right"
				android:layout_gravity="center_vertical" 
				android:textColor="@color/yellow"></TextView>
			<Button android:id="@+id/btnDail" 
				android:text="@string/btnDail"
				android:textSize="10dp"
				android:layout_marginLeft="10dp" 
				android:width="60dp"
				android:drawableRight="@drawable/dail"
				android:layout_gravity="center_vertical" 
				android:textColor="@color/purplered"></Button>
		</TableRow>
	</TableLayout>
</LinearLayout>

依然是TableLayout布局,咱们设置它的收缩列为第二列,伸展列也是第二列。这样若是第二列不够显示则会收缩。OK,咱们看一下后台代码测试

public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.punchinalarm);
		
		owner = this;
		btnSelAll = (Button) this.findViewById(R.id.btnSelAll);
		btnSelInverse = (Button) this.findViewById(R.id.btnInverseSel);
		btnSet = (Button) this.findViewById(R.id.btnSet);
		contactUserListView = (ListView) this
				.findViewById(R.id.contactListView);
		
		dataList = new ArrayList<Map<String, Object>>();
		this.InitData();
	}

在OnCreate方法中,咱们初始化数据。ui

private void InitData() {
		String phoneUserName = null;
		String phoneNumber = null;
		Long contactId = null;
		Long photoId = null;
		Map<String, Object> dataMap = null;

		ContentResolver resolver = this.getApplicationContext()
				.getContentResolver();

		/*
		 * 获取Sim卡联系人 Uri uri = Uri.parse("content://icc/adn");
		 */
		Cursor phoneCursor = resolver.query(Phone.CONTENT_URI,
				PHONE_PROJECTION, null, null, null);

		if (phoneCursor != null) {
			while (phoneCursor.moveToNext()) {
				dataMap = new HashMap<String, Object>();
				phoneUserName = phoneCursor.getString(0);
				phoneNumber = phoneCursor.getString(1);
				photoId = phoneCursor.getLong(2);
				contactId = phoneCursor.getLong(3);

				if (phoneNumber == null || phoneNumber.trim().length() < 11) {
					continue;
				}

				Bitmap contactPhoto = null;
				if (photoId > 0) {
					Uri uri = ContentUris.withAppendedId(
							ContactsContract.Contacts.CONTENT_URI, contactId);
					InputStream input = ContactsContract.Contacts
							.openContactPhotoInputStream(resolver, uri);
					contactPhoto = BitmapFactory.decodeStream(input);
				} else {
					contactPhoto = BitmapFactory.decodeResource(getResources(),
							R.drawable.usersmall);
				}

				dataMap.put("UserName", phoneUserName);
				dataMap.put("UserPhoneNumber", phoneNumber);
				dataMap.put("UserPhoto", contactPhoto);
				dataList.add(dataMap);
			}

			if (dataList != null && dataList.size() > 0) {
				customAdapter simpleAdapter = new customAdapter(this, dataList,
						R.layout.contactdetailtemplate, new String[] {
								"UserPhoto", "UserName", "UserPhoneNumber" },
						new int[] { R.id.chkContactUser, R.id.imgContactPhoto,
								R.id.txtContactName, R.id.txtContactTelNumber,
								R.id.btnDail });

				this.contactUserListView.setAdapter(simpleAdapter);
			}
		}
	}

咱们知道外界的程序经过ContentResolver接口能够访问ContentProvider提供的数据,在Activity当中经过getContentResolver()能够获得当前应用的 ContentResolver实例。由于通信录和短消息均可以通过接口访问,因此咱们就能够经过getContentResolver访问SIM卡和手机中的联系人信息。this

咱们主要到下面的这句spa

Cursor phoneCursor = resolver.query(Phone.CONTENT_URI,
				PHONE_PROJECTION, null, null, null);

经过接口查询,咱们会获得一个Cursor。经过查看Cursor的定义,咱们发现它是个抽象类.net

public abstract interface android.database.Cursor

咱们发现它提供了一些方法,以下

wKioL1RbgASRasFnAAEq2WtayDw502.jpg

因而可知,它是一个既能够前进又能够后退的无向游标,相似于SqlServer中的游标。这样的话咱们不管是读取联系人信息,仍是读取短消息,均可以随时定位游标。

在上面咱们看到Query的几个参数,Phone.Content_URI,获取联系人的时候须要去这个URI去取数据。其实这里的Phone.Content_URI的值是content://com.android.contacts/contacts。它的定义以下

 public static final android.net.Uri CONTENT_URI;


ok,咱们拿到联系人以后,咱们进行循环,游标下移,拿出全部的有电话号码的联系人的数据。由于咱们传入的PHONE_PROJECTION的顺序是姓名,电话号码,照片ID,以及一个ContactID。因此咱们看到取数据的顺序以下

phoneUserName = phoneCursor.getString(0);
				phoneNumber = phoneCursor.getString(1);
				photoId = phoneCursor.getLong(2);
				contactId = phoneCursor.getLong(3);

拿到PhotoID以后,咱们判断是否大于0,若是大于0,咱们会获取用户图像。

获取用户图像的时候,先经过下面的代码将URI和参数链接起来

ContentUris.withAppendedId(
							ContactsContract.Contacts.CONTENT_URI, contactId)

其实这个相似于Get方式的API,好比ContactUser/100,意思是获取编号为100的人的信息。

OK,URI构造好以后,咱们获取图像

InputStream input = ContactsContract.Contacts
							.openContactPhotoInputStream(resolver, uri);
					contactPhoto = BitmapFactory.decodeStream(input);

最后加入List<Map<String,Object>>,对ListView运用适配器。

class customAdapter extends BaseAdapter {
		private List<Map<String, Object>> dataList;
		private LayoutInflater mInflater;
		private Context context;
		private String[] keyString;
		private int[] valueViewID;
		Holder holder;

		public customAdapter(Context context,
				List<Map<String, Object>> dataList, int resource,
				String[] from, int[] to) {
			this.dataList = dataList;
			this.context = context;
			mInflater = (LayoutInflater) this.context
					.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
			keyString = new String[from.length];
			valueViewID = new int[to.length];
			System.arraycopy(from, 0, keyString, 0, from.length);
			System.arraycopy(to, 0, valueViewID, 0, to.length);
		}

		@Override
		public int getCount() {
			return dataList.size();
		}

		@Override
		public Object getItem(int position) {
			return dataList.get(position);
		}

		@Override
		public long getItemId(int position) {
			return position;
		}

		public void removeItem(int position) {
			dataList.remove(position);
			this.notifyDataSetChanged();
		}

		public View getView(int position, View convertView, ViewGroup parent) {

			if (convertView != null) {
				holder = (Holder) convertView.getTag();
			} else {
				convertView = mInflater.inflate(R.layout.contactdetailtemplate,
						null);
				holder = new Holder();
				holder.chkContactUser = (CheckBox) convertView
						.findViewById(valueViewID[0]);
				holder.imgUserPhoto = (ImageView) convertView
						.findViewById(valueViewID[1]);
				holder.labUserName = (TextView) convertView
						.findViewById(valueViewID[2]);
				holder.labPhoneNumber = (TextView) convertView
						.findViewById(valueViewID[3]);
				holder.btnDail = (Button) convertView
						.findViewById(valueViewID[4]);

				convertView.setTag(holder);
			}

			Map<String, Object> appInfo = dataList.get(position);
			if (appInfo != null) {
				String userName = appInfo.get(keyString[1]).toString();
				String userPhoneNumber = appInfo.get(keyString[2]) == null ? ""
						: appInfo.get(keyString[2]).toString();
				Bitmap userPhoto = (Bitmap) appInfo.get(keyString[0]);

				holder.labUserName.setText(userName);
				holder.labPhoneNumber.setText(userPhoneNumber);
				holder.imgUserPhoto.setImageBitmap(userPhoto);
				holder.btnDail.setOnClickListener(new ViewButtonListener(
						position, holder.chkContactUser));
			}
			return convertView;
		}
	}

	class Holder {
		public TextView labUserName;
		public TextView labPhoneNumber;
		public ImageView imgUserPhoto;
		public Button btnDail;
		public CheckBox chkContactUser;
	}

这里,其实很简单,咱们就是拿到控件,而后根据Position,拿到List中的某行数据,而后赋值。

最后咱们看看按钮的Click事件,咱们传入了Position和每行的CheckBox。

class ViewButtonListener implements OnClickListener {
		private int position;
		Object phoneNumber;
		CheckBox chkContactUser;

		ViewButtonListener(int position,CheckBox chkContactUser) {
			this.position = position;
			this.phoneNumber = dataList.get(position).get("UserPhoneNumber");
			this.chkContactUser= chkContactUser;
		}

		@Override
		public void onClick(View view) {
			int vid = view.getId();
			if (vid == R.id.btnDail&&chkContactUser.isChecked()) {
				Intent dialIntent = new Intent(Intent.ACTION_CALL, Uri
						.parse("tel:" + phoneNumber));
				startActivity(dialIntent);
			}
		}
	}

在OnClick中,咱们判断若是是拨号按钮而且列头的CheckBox是勾选的,则会拨号,不然不会拨号。OK,最后咱们但愿在按回退键的时候,弹出是否退出的提示

wKiom1RbjdHDxnOlAAGKTIuOxw4762.jpg

ok,代码以下

public boolean onKeyDown(int keyCode, KeyEvent event) {
		if ((keyCode == KeyEvent.KEYCODE_BACK && event.getRepeatCount() == 0)) {
			dialog();
			return true;
		}
		return true;
	}

	protected void dialog() {
		AlertDialog.Builder builder = new Builder(punchinalarm.this);
		builder.setMessage("肯定要退出吗?");
		builder.setTitle("提示");
		builder.setPositiveButton("确认", new DialogInterface.OnClickListener() {
			@Override
			public void onClick(DialogInterface dialog, int which) {
				dialog.dismiss();
				android.os.Process.killProcess(android.os.Process.myPid());
			}
		});
		builder.setNegativeButton("取消",
				new android.content.DialogInterface.OnClickListener() {
					@Override
					public void onClick(DialogInterface dialog, int which) {
						dialog.dismiss();
					}
				});
		builder.create().show();
	}


最后,哥们的博客是货真价实,小米3测试机。

wKiom1Rbj22DJ0SbAAHqhJl9btY782.jpgwKiom1Rbj6DRdUVHAATC_rI71sI887.jpg

相关文章
相关标签/搜索