apps目录的contacts应用(有读取通话记录功能),是访问provider目录的provider.contacts应用(有暴露通话记录),所以要阅读Android操作系统源码-->packages/providers/ContactsProvider通话记录的(内容提供者)
阅读 com.android.providers.contacts 数据库与表
Android操作系统的文件系统目录/data/data/com.android.contacts,是访问/data/data/com.android.providers.contacts(内容提供者应用)
所以需要阅读/data/data/com.android.providers.contacts(内容提供者应用)的数据库:
首选要有一条通话记录
打开 contacts2.db
打开后:表非常多,视图非常多,等等,但只需关心,calls(通话记录表)
阅读 com.android.providers.contacts AndroidManifest.xml
android:name="CallLogProvider" 通话记录的内容提供者
android:authorities="call_log" 授权唯一标识
android:exported="true" 允许对外输出
android:readPermission="android.permission.READ_CALL_LOG" 访问者必须要配置的权限
android:writePermission="android.permission.WRITE_CALL_LOG" 访问者必须要配置的权限
<provider android:name="CallLogProvider" android:authorities="call_log" android:syncable="false" android:multiprocess="false" android:exported="true" android:readPermission="android.permission.READ_CALL_LOG" android:writePermission="android.permission.WRITE_CALL_LOG"> </provider>
阅读 com.android.providers.contacts CallLogProvider.java
首先要找到的就是Uri,所以搜索UriMatcher的matcher.addURI
有规律,通常情况下,matcher.addURI(授权, path, code),第二个参数 path 和数据库表名对应的 calls
private static final UriMatcher sURIMatcher = new UriMatcher(UriMatcher.NO_MATCH); static { sURIMatcher.addURI(CallLog.AUTHORITY, "calls", CALLS); sURIMatcher.addURI(CallLog.AUTHORITY, "calls/#", CALLS_ID); sURIMatcher.addURI(CallLog.AUTHORITY, "calls/filter/*", CALLS_FILTER); }
C应用 AndroidManifest.xml 权限的配置:
<!-- 访问操作系统短信通话记录提供者应用,需要加入的权限 android:readPermission="android.permission.READ_CALL_LOG" android:writePermission="android.permission.WRITE_CALL_LOG" --> <uses-permission android:name="android.permission.READ_CALL_LOG" /> <uses-permission android:name="android.permission.WRITE_CALL_LOG" />
<!-- C应用配置拨打电话的权限 拨打电话的权限 --> <uses-permission android:name="android.permission.CALL_PHONE" />
C应用 读取操作系统通话记录并/拨打电话/发送短信/复制号码到拨号盘 Java代码:
package liudeli.cp.client; import android.app.Activity; import android.app.AlertDialog; import android.content.DialogInterface; import android.content.Intent; import android.database.Cursor; import android.net.Uri; import android.os.Bundle; import android.provider.CallLog; import android.util.Log; import android.view.View; import android.widget.AdapterView; import android.widget.ListView; import android.widget.SimpleCursorAdapter; public class CallLogActivity extends Activity { /** * 找寻到了Android操作系统的通话记录内容提供者的授权和Uri * android:authorities="call_log" * sURIMatcher.addURI(CallLog.AUTHORITY, "calls", CALLS); */ private final String AUTHORITY = "call_log"; private Uri callLogUri = Uri.parse("content://" + AUTHORITY + "/calls"); /** * 系统也提供了常量的方式来获取Uir,为了学习,还是直接看源码复制出来的比较理解些 */ /*private final String AUTHORITY = CallLog.AUTHORITY; private Uri callLogUri = Uri.parse(CallLog.CONTENT_URI + "/calls");*/ @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_call_log); ListView listView = findViewById(R.id.listview); String[] porjecting = new String[]{"_id" ,"number", "date", "type"}; final Cursor cursor = getContentResolver().query(callLogUri, porjecting, null, // 不要查询条件 null, // 不要查询条件值 null); // 不排序 while (cursor.moveToNext()) { Log.d("cccc", "" + cursor.getString(0) + " "+ cursor.getString(1) + " " + cursor.getString(2) + " "+ cursor.getString(3)); } final SimpleCursorAdapter cursorAdapter = new SimpleCursorAdapter(this, // 上下文 R.layout.layou_calllog_item, // Item显示的布局 cursor, // 游标数据 porjecting, // 数据从哪里来(从Cursor中获取对应的字段) new int[]{R.id.tv_id, R.id.tv_number, R.id.tv_date, R.id.tv_type} // 数据到哪里去,到Item布局里面的控件显示 ); listView.setAdapter(cursorAdapter); // 千万不能关闭 cursor.close();,否则数据展示不出来 listView.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() { @Override public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id) { // 获取Item的值 (转换规则,传入什么类型,就转换成什么类型) final Cursor itemCursor = (Cursor) cursorAdapter.getItem(position); /** * 把Cursor移动到指定的行数,然后在取值 itemCursor.getString(0~9) */ itemCursor.moveToPosition(position); new AlertDialog.Builder(CallLogActivity.this) .setTitle("请选择") /*.setMessage("请选择,下面列表的功能")*/ //列表对话框不能设置这个,否则显示不出来 .setItems(new String[]{"拨打电话", "复制号码到拨号盘", "复制号码到短信编辑界面"}, new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { // 把Cursor移动到指定的行数,然后在取值 itemCursor.getString(0~9) String number = itemCursor.getString(itemCursor.getColumnIndex("number")); /** * 下面这三个功能,都需要隐式意图的方式去激活系统暴露的组件 * 匹配规则:只要匹配一组,就可以来 */ switch (which) { case 0: // 拨打电话 Intent intentCall = new Intent(); intentCall.setAction(Intent.ACTION_CALL); intentCall.setData(Uri.parse("tel:" + number)); startActivity(intentCall); break; case 1: // 复制号码到拨号盘 Intent intentCopyCall = new Intent(); intentCopyCall.setAction(Intent.ACTION_DIAL); intentCopyCall.setData(Uri.parse("tel:" + number)); startActivity(intentCopyCall); break; case 2: // 复制号码到短信编辑界面 Intent intentSmsEdit = new Intent(); intentSmsEdit.setAction(Intent.ACTION_VIEW); intentSmsEdit.setData(Uri.parse("sms:" + number)); startActivity(intentSmsEdit); break; default: break; } } }) .show(); return false; } }); } }
C应用显示的Layout:
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"> <ListView android:id="@+id/listview" android:layout_width="match_parent" android:layout_height="match_parent" /> </RelativeLayout>
C应用显示的Layout --> ListVIew-->Item布局:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="horizontal" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_marginBottom="8dp"> <!-- 默认比重为0 我先填充 --> <LinearLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="vertical"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:textSize="20sp" android:text="id" android:textColor="@android:color/black" /> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:textSize="20sp" android:text="number" android:textColor="@android:color/black" /> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:textSize="20sp" android:text="date" android:textColor="@android:color/black" android:layout_marginTop="5dp" /> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:textSize="20sp" android:text="type" android:textColor="@android:color/black" android:layout_marginTop="5dp" /> </LinearLayout> <!-- 哥们,你已经填充完来吧,剩下的空间我全部使用 --> <LinearLayout android:layout_width="0dp" android:layout_height="wrap_content" android:orientation="vertical" android:layout_weight="1" android:layout_marginLeft="20dp" > <TextView android:id="@+id/tv_id" android:layout_width="wrap_content" android:layout_height="wrap_content" android:textSize="20sp" android:text="id" android:textColor="@android:color/black" /> <TextView android:id="@+id/tv_number" android:layout_width="wrap_content" android:layout_height="wrap_content" android:textSize="20sp" android:text="number" android:textColor="@android:color/black" /> <TextView android:id="@+id/tv_date" android:layout_width="wrap_content" android:layout_height="wrap_content" android:textSize="20sp" android:text="date" android:textColor="@android:color/black" android:layout_marginTop="5dp" /> <TextView android:id="@+id/tv_type" android:layout_width="wrap_content" android:layout_height="wrap_content" android:textSize="20sp" android:text="type" android:textColor="@android:color/black" android:layout_marginTop="5dp" /> </LinearLayout> </LinearLayout>
C应用效果图:
真实开发中,必须要用常量,才靠谱,万一字段变来怎么办,是吧
/** * 通话记录通常是有常量的 */ Calls.Date; CallLog.Calls._ID .... CallLog.AUTHORITY CallLog.Calls.NUMBER; CallLog.CONTENT_URI ....