天冷了,老夫要把伙食搞上去,這不最近在軟件園二樓吃,伙食15塊,杠杠的。
創(chuàng)新互聯(lián)公司是一家專注于成都做網(wǎng)站、網(wǎng)站建設(shè)與策劃設(shè)計(jì),橋西網(wǎng)站建設(shè)哪家好?創(chuàng)新互聯(lián)公司做網(wǎng)站,專注于網(wǎng)站建設(shè)十余年,網(wǎng)設(shè)計(jì)領(lǐng)域的專業(yè)建站公司;建站業(yè)務(wù)涵蓋:橋西等地區(qū)。橋西做網(wǎng)站價(jià)格咨詢:18982081108
美包包,不說了,進(jìn)入正題。今天老夫要講的是讀取聯(lián)系人,其實(shí)我做這個(gè)的初衷是想做一個(gè)短信攔截,電話攔截的功能。
我們先看一下界面,還是不錯(cuò)的,挺絢麗的。
OK,我們一看就知道,這又是一個(gè)ListView。目前的功能是當(dāng)你在復(fù)選框打鉤,點(diǎn)擊后面的撥號就會將電話打出去。如果不打鉤,電話則不會撥出去。OK,我們先看看頁面布局
我們再看看ListView要加載的模版
依然是TableLayout布局,我們設(shè)置它的收縮列為第二列,伸展列也是第二列。這樣如果第二列不夠顯示則會收縮。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
在OnCreate方法中,我們初始化數(shù)據(jù)。
private void InitData() { String phoneUserName = null; String phoneNumber = null; Long contactId = null; Long photoId = null; MapdataMap = null; ContentResolver resolver = this.getApplicationContext() .getContentResolver(); /* * 獲取Sim卡聯(lián)系人 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 (); 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提供的數(shù)據(jù),在Activity當(dāng)中通過getContentResolver()可以得到當(dāng)前應(yīng)用的 ContentResolver實(shí)例。因?yàn)橥ㄓ嶄浐投滔⒍伎梢酝ㄟ^接口訪問,所以我們就可以通過getContentResolver訪問SIM卡和手機(jī)中的聯(lián)系人信息。
我們主要到下面的這句
Cursor phoneCursor = resolver.query(Phone.CONTENT_URI, PHONE_PROJECTION, null, null, null);
通過接口查詢,我們會得到一個(gè)Cursor。通過查看Cursor的定義,我們發(fā)現(xiàn)它是個(gè)抽象類
public abstract interface android.database.Cursor
我們發(fā)現(xiàn)它提供了一些方法,如下
由此可見,它是一個(gè)既可以前進(jìn)又可以后退的無向游標(biāo),類似于SqlServer中的游標(biāo)。這樣的話我們不論是讀取聯(lián)系人信息,還是讀取短消息,都可以隨時(shí)定位游標(biāo)。
在上面我們看到Query的幾個(gè)參數(shù),Phone.Content_URI,獲取聯(lián)系人的時(shí)候需要去這個(gè)URI去取數(shù)據(jù)。其實(shí)這里的Phone.Content_URI的值是content://com.android.contacts/contacts。它的定義如下
public static final android.net.Uri CONTENT_URI;
ok,我們拿到聯(lián)系人之后,我們進(jìn)行循環(huán),游標(biāo)下移,拿出所有的有電話號碼的聯(lián)系人的數(shù)據(jù)。因?yàn)槲覀儌魅氲腜HONE_PROJECTION的順序是姓名,電話號碼,照片ID,以及一個(gè)ContactID。所以我們看到取數(shù)據(jù)的順序如下
phoneUserName = phoneCursor.getString(0); phoneNumber = phoneCursor.getString(1); photoId = phoneCursor.getLong(2); contactId = phoneCursor.getLong(3);
拿到PhotoID之后,我們判斷是否大于0,如果大于0,我們會獲取用戶圖像。
獲取用戶圖像的時(shí)候,先通過下面的代碼將URI和參數(shù)連接起來
ContentUris.withAppendedId( ContactsContract.Contacts.CONTENT_URI, contactId)
其實(shí)這個(gè)類似于Get方式的API,比如ContactUser/100,意思是獲取編號為100的人的信息。
OK,URI構(gòu)造好之后,我們獲取圖像
InputStream input = ContactsContract.Contacts .openContactPhotoInputStream(resolver, uri); contactPhoto = BitmapFactory.decodeStream(input);
最后加入List
class customAdapter extends BaseAdapter { private List
這里,其實(shí)很簡單,我們就是拿到控件,然后根據(jù)Position,拿到List中的某行數(shù)據(jù),然后賦值。
最后我們看看按鈕的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,最后我們希望在按回退鍵的時(shí)候,彈出是否退出的提示
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("確認(rèn)", 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(); }
最后,哥們的博客是貨真價(jià)實(shí),小米3測試機(jī)。