apps目录的contacts应用(有读取通话记录功能),是访问provider目录的provider.contacts应用(有暴露通话记录),所以要阅读Android操作系统源码-->packages/providers/ContactsProvider通话记录的(内容提供者)

android stidio 操作记录 安卓系统操作记录_ide

 


 

阅读 com.android.providers.contacts 数据库与表

Android操作系统的文件系统目录/data/data/com.android.contacts,是访问/data/data/com.android.providers.contacts(内容提供者应用)

所以需要阅读/data/data/com.android.providers.contacts(内容提供者应用)的数据库:

android stidio 操作记录 安卓系统操作记录_android_02

 

首选要有一条通话记录 

android stidio 操作记录 安卓系统操作记录_android stidio 操作记录_03

 

打开 contacts2.db

 打开后:表非常多,视图非常多,等等,但只需关心,calls(通话记录表)

android stidio 操作记录 安卓系统操作记录_android stidio 操作记录_04

 

 


 

阅读 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应用效果图:

android stidio 操作记录 安卓系统操作记录_android stidio 操作记录_05

 


 

真实开发中,必须要用常量,才靠谱,万一字段变来怎么办,是吧

/**
         * 通话记录通常是有常量的
         */
        Calls.Date;
        CallLog.Calls._ID
        ....
        CallLog.AUTHORITY
        CallLog.Calls.NUMBER;
        CallLog.CONTENT_URI
        ....

android stidio 操作记录 安卓系统操作记录_android stidio 操作记录_06