打电话Demo及Android6.0的运行时权限问题
来源:互联网 发布:国云大数据魔镜 编辑:程序博客网 时间:2024/05/18 03:46
新手开局,查看一些旧资料,从打电话、发短信的小应用开始。代码很简单,主要是学习了:
- 用StartActivity()激活一个Activity组件。这里是激活了系统原生的打电话和发短信Activity。
- Intent意图对象的使用,包括设置其动作和数据。
- 在Manifest.xml清单文件中添加所需的权限。
做个笔记,主要代码如下:
public class MainActivity extends Activity { private Button btn_dial; private EditText et_number; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 找到控件 et_number = (EditText) findViewById(R.id.et_number); btn_dial = (Button) findViewById(R.id.btn_dial); // 给按钮设置监听(点击事件) btn_dial.setOnClickListener(new View.OnClickListener() { // 匿名内部类 // 按钮点击时回调 @Override public void onClick(View view) { // 获取号码 String number = et_number.getText().toString(); if (TextUtils.isEmpty(number)){ // 提醒用户 // 注意:在这个匿名内部类中如果用this则表示是View.OnClickListener类的对象, // 所以必须用MainActivity.this来指定上下文环境。 Toast.makeText(MainActivity.this, "号码不能为空!", Toast.LENGTH_SHORT).show(); }else{ // 拨号:激活系统的拨号组件 Intent intent = new Intent(); // 意图对象:动作 + 数据 intent.setAction(Intent.ACTION_CALL); // 设置动作 Uri data = Uri.parse("tel:" + number); // 设置数据 intent.setData(data); startActivity(intent); // 激活Activity组件 } } }); }}
在Manifest清单中也配置了所需的拨号权限
<?xml version="1.0" encoding="utf-8"?><manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.demo.guxin.a01_helloworld"> <!-- 拨号权限 --> <uses-permission android:name="android.permission.CALL_PHONE"></uses-permission> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application></manifest>
然后遇到的问题是,在Android6.0目标平台下,即便已经添加了打电话的权限,运行时依然会报错安全异常:权限被拒绝。
在Stack Overflow上搜到了一个很靠谱的答案:
添加了权限依然报错权限拒绝
根据老外的提示,原来从Android6.0开始使用了新的运行时权限,权限种类被分为了【普通权限】和【危险权限】,顺藤摸瓜在官方文档学到了很有价值的东西,罗列一下以后在更新:
https://developer.android.com/training/permissions/index.html
学习关于系统权限
https://developer.android.com/training/permissions/declaring.html
如何申明权限
https://developer.android.com/guide/topics/security/permissions.html#normal-dangerous
权限被分为了普通和危险两种
https://developer.android.com/training/permissions/requesting.html#perm-check
【重要】如何在运行时检查并申请权限!!!!
推荐博文:
http://droidyue.com/blog/2016/01/17/understanding-marshmallow-runtime-permission/
http://blog.csdn.net/lmj623565791/article/details/50709663
http://www.cnblogs.com/mengdd/p/4892856.html
http://www.cnblogs.com/mengdd/p/4892856.html
http://www.2cto.com/kf/201512/455888.html
重要参考:http://stackoverflow.com/questions/30719047/android-m-check-runtime-permission-how-to-determine-if-the-user-checked-nev
根据以上资料,再次整理出打电话的Demo如下:
package com.demo.guxin.a01_helloworld;import android.Manifest;import android.app.Activity;import android.content.Intent;import android.content.pm.PackageManager;import android.net.Uri;import android.os.Bundle;import android.provider.Settings;import android.support.v4.app.ActivityCompat;import android.support.v4.content.ContextCompat;import android.text.TextUtils;import android.view.View;import android.widget.Button;import android.widget.EditText;import android.widget.Toast;public class MainActivity extends Activity { private static final int MY_PERMISSIONS_REQUEST_CALL_PHONE = 1; private Button btn_dial; private EditText et_number; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 找到控件 et_number = (EditText) findViewById(R.id.et_number); btn_dial = (Button) findViewById(R.id.btn_dial); // 给按钮设置监听(点击事件) btn_dial.setOnClickListener(new View.OnClickListener() { // 匿名内部类 // 按钮点击时回调 @Override public void onClick(View view) { // 检查是否获得了权限(Android6.0运行时权限) if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED){ // 没有获得授权,申请授权 if (ActivityCompat.shouldShowRequestPermissionRationale(MainActivity.this, Manifest.permission.CALL_PHONE)) { // 返回值:// 如果app之前请求过该权限,被用户拒绝, 这个方法就会返回true.// 如果用户之前拒绝权限的时候勾选了对话框中”Don’t ask again”的选项,那么这个方法会返回false.// 如果设备策略禁止应用拥有这条权限, 这个方法也返回false. // 弹窗需要解释为何需要该权限,再次请求授权 Toast.makeText(MainActivity.this, "请授权!", Toast.LENGTH_LONG).show(); // 帮跳转到该应用的设置界面,让用户手动授权 Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS); Uri uri = Uri.fromParts("package", getPackageName(), null); intent.setData(uri); startActivity(intent); }else{ // 不需要解释为何需要该权限,直接请求授权 ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.CALL_PHONE}, MY_PERMISSIONS_REQUEST_CALL_PHONE); } }else { // 已经获得授权,可以打电话 CallPhone(); } } }); } private void CallPhone() { String number = et_number.getText().toString(); if (TextUtils.isEmpty(number)) { // 提醒用户 // 注意:在这个匿名内部类中如果用this则表示是View.OnClickListener类的对象, // 所以必须用MainActivity.this来指定上下文环境。 Toast.makeText(MainActivity.this, "号码不能为空!", Toast.LENGTH_SHORT).show(); } else { // 拨号:激活系统的拨号组件 Intent intent = new Intent(); // 意图对象:动作 + 数据 intent.setAction(Intent.ACTION_CALL); // 设置动作 Uri data = Uri.parse("tel:" + number); // 设置数据 intent.setData(data); startActivity(intent); // 激活Activity组件 } } // 处理权限申请的回调 @Override public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) { switch (requestCode){ case MY_PERMISSIONS_REQUEST_CALL_PHONE: { if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { // 授权成功,继续打电话 CallPhone(); } else { // 授权失败! Toast.makeText(this, "授权失败!", Toast.LENGTH_LONG).show(); } break; } } }}
- 打电话Demo及Android6.0的运行时权限问题
- 打电话Demo及Android6.0的运行时权限问题
- Android6.0运行时权限的处理及解决办法
- Android6.0运行时权限的处理及解决办法
- 关于Android6.0以上动态获取运行时权限、及无法正常运行百度地图API的问题
- Android6.0运行时权限的封装
- Android6.0运行时权限的处理
- Android6.0的运行时权限
- Android6.0+运行时权限的处理
- Android6.0 运行时权限
- Android6.0 运行时权限
- Android6.0运行时权限
- 【Android6.0】运行时权限
- Android6.0 运行时权限
- Android6.0 运行时权限
- Android6.0 运行时权限
- Android6.0运行时权限
- Android6.0运行时权限
- 九度+简单模拟题+主要是那个整数翻转的处理方法好
- leetcode-003 Longest Substring Without Repeating Characters
- Light OJ 1203 凸包+寻找凸多边形最小角度
- Lua中的元表与元方法
- stl map find使用不当导致的低概率core dump问题的定位
- 打电话Demo及Android6.0的运行时权限问题
- 【CODEVS 1183】泥泞的道路 SPFA+二分
- 遗传算法 一个模拟自然进化过程的启发式搜索算法
- 邮件发送和接收原理
- 计算机三座高峰:
- Leetcode 95. Unique Binary Search Trees II (Medium) (cpp)
- 【Easy】88. Merge Sorted Array
- BZOJ 1024 SCOI2009 生日快乐 暴搜
- Liberty版本Neutron LBaas学习