解決Android拍照6.0以上動態獲取許可權問題
概述
在Android開發過程中,拍照或者從相簿中選擇圖片是很常見的功能。下面要說得這個案例比較簡單,使用者點選按鈕選擇拍照或者開啟相簿選擇圖片,然後將選中的圖片顯示在手機上。android6.0後,推出了動態許可權管理。以往我們將涉及到的許可權全部寫在清單檔案中,只要使用者安裝了該程式,程式在執行過程中都會獲得相應許可權。android6.0後,對於一些特別敏感的許可權,開發者必須在程式中進行宣告。拍照和從相簿選擇圖片都是涉及到使用者隱私的敏感許可權,必須在程式中進行宣告。
大概的流程
- 建立佈局檔案,這裡不多說了
- 拍照的實現
2.1建立存放圖片的資料夾
2.2將資料夾路徑轉換為uri
2.3隱式啟動相機的Activity,uri作為intent的一個引數.
2.4拍照結束後,執行onActivityResult(…)獲得圖片 - 相簿選取圖片
3.1啟動相簿Activity
3.2選擇結束後,執行onActivityResult(…)獲得圖片 - 動態許可權管理
1,拍照的實現
拍照之前,我們需要建立一個資料夾來存放我們拍好的照片。建立好資料夾後,就可以進入拍照的Activity了,程式碼如下:
void takePhoto(){
/**
* 最後一個引數是資料夾的名稱,可以隨便起
*/
File file=new File(Environment.getExternalStorageDirectory(),"拍照");
if(!file.exists()){
file.mkdir();
}
/**
* 這裡將時間作為不同照片的名稱
*/
output=new File(file,System.currentTimeMillis()+".jpg");
/**
* 如果該資料夾已經存在,則刪除它,否則建立一個
*/
try {
if (output.exists()) {
output.delete();
}
output.createNewFile();
} catch (Exception e) {
e.printStackTrace();
}
/**
* 隱式開啟拍照的Activity,並且傳入CROP_PHOTO常量作為拍照結束後回撥的標誌
* 將檔案轉化為uri
*/
imageUri = Uri.fromFile(output);
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, CROP_PHOTO);
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
我們通過startActivityForResult(intent, CROP_PHOTO)方法進入拍照程式,拍照結束後,會執行onActivityResult(…)方法,所以我們需要重寫該方法獲得拍照Activity給我們返回的資料進而得到照片的bitmap物件。
public void onActivityResult(int req, int res, Intent data) {
switch (req) {
/**
* 拍照的請求標誌
*/
case CROP_PHOTO:
if (res==RESULT_OK) {
try {
/**
* 該uri就是照片資料夾對應的uri
*/
Bitmap bit = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));
imageView.setImageBitmap(bit);
} catch (Exception e) {
Toast.makeText(this,"程式崩潰",Toast.LENGTH_SHORT).show();
}
}
else{
Log.i("tag", "失敗");
}
break;
/**
* 從相簿中選取圖片的請求標誌
*/
case REQUEST_CODE_PICK_IMAGE:
if (res == RESULT_OK) {
try {
/**
* 該uri是上一個Activity返回的
*/
Uri uri = data.getData();
Bitmap bit = BitmapFactory.decodeStream(getContentResolver().openInputStream(uri));
imageView.setImageBitmap(bit);
} catch (Exception e) {
e.printStackTrace();
Log.d("tag",e.getMessage());
Toast.makeText(this,"程式崩潰",Toast.LENGTH_SHORT).show();
}
}
else{
Log.i("liang", "失敗");
}
break;
default:
break;
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
至此,拍照就結束了
2,從相簿獲取圖片的實現
和拍照非常相似,唯一不同的是開啟該Activity的intent不同,開啟相簿的程式碼如下:
void choosePhoto(){
/**
* 開啟選擇圖片的介面
*/
Intent intent = new Intent(Intent.ACTION_PICK);
intent.setType("image/*");//相片型別
startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
選擇好照片後,會執行onActivityResult(…)方法,在主Activity中需要重寫此方法,並且在此方法中得到相簿Activity傳回的資料進而得到照片的bitmap物件。這一過程和拍照的onActivityResult(…)方法是相同的。
3,動態許可權管理
無論是拍照還是從相簿中選擇圖片都涉及到使用者的隱私,所以我們需要宣告許可權,所以我們需要在清單檔案中加入這句:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
- 1
- 1
在android6.0中,這樣做還是不夠的。我們還需要在程式碼中動態宣告許可權。啟動拍照或者相簿的Activity時,會彈出一個dialog詢問使用者是否同意授權。
只有使用者同意授權後,才能順利完成接下來的操作。需要說明的是,就算使用者拒絕,我們仍然可以開啟相機和相簿的Activity,但是操作結束後,並不會返回什麼值,就是說系統拒絕我們讀取資料。
申請動態許可權的簡略步驟如下:
- 首先檢查許可權是否已經被授予,如果已經授予,那我們就可以直接執行相關的方法,否則,需要申請許可權。
- 如果許可權沒有授予,我們需要申請許可權。申請後app會想使用者彈出一個dialog詢問使用者是否授予該許可權。
-
使用者無論是同意還是拒絕授予該許可權,Activity都會執行onRequestPermissionsResult(…)方法,我們需要在該方法中判斷使用者是否同意該許可權。如果同意,執行相應的方法,如果拒絕,最好向使用者解釋下為什麼需要這個許可權。
檢查許可權是否被授予:
//第二個引數是需要申請的許可權
if (ContextCompat.checkSelfPermission(this,
Manifest.permission.WRITE_EXTERNAL_STORAGE)
!= PackageManager.PERMISSION_GRANTED)
{
//許可權還沒有授予,需要在這裡寫申請許可權的程式碼
}else {
//許可權已經被授予,在這裡直接寫要執行的相應方法即可
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
申請許可權:
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
MY_PERMISSIONS_REQUEST_CALL_PHONE2);
- 1
- 2
- 3
- 1
- 2
- 3
第二個引數是一個字串陣列,裡面是你需要申請的許可權。既然是一個數組,那麼就說明你一次可以申請多個許可權。由於拍照和選擇相簿圖片只涉及一個許可權,所以上面的字串陣列中就只寫了一個許可權。最後一個引數是一個整型常量,用於標誌你這次申請的許可權,該常量在onRequestPermissionsResult(…)方法中會用到。
判斷使用者是否授予該許可權
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults)
{
if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE)
{
if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
{
takePhoto();
} else
{
// Permission Denied
Toast.makeText(MainActivity.this, "Permission Denied", Toast.LENGTH_SHORT).show();
}
}
if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE2)
{
if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
{
choosePhoto();
} else
{
// Permission Denied
Toast.makeText(MainActivity.this, "Permission Denied", Toast.LENGTH_SHORT).show();
}
}
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
如果使用者同意,那麼就執行相應的方法,如果拒絕,可以向用戶解釋一下申請該許可權的原因。
4,程式碼
- 佈局檔案
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.example.takephoto.MainActivity">
<Button
android:text="拍照"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
android:layout_margin="10dp"
android:id="@+id/button"
android:onClick="takePhone"
/>
<Button
android:text="相簿選擇"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
android:layout_margin="10dp"
android:onClick="choosePhone"
/>
<ImageView
android:layout_width="match_parent"
android:layout_height="0dp"
android:id="@+id/image"
android:src="@color/colorAccent"
android:scaleType="centerCrop"
android:layout_weight="7"/>
</LinearLayout>
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- MainActivity程式碼
public class MainActivity extends AppCompatActivity {
private ImageView imageView;
private static final int CROP_PHOTO = 2;
private static final int REQUEST_CODE_PICK_IMAGE=3;
private static final int MY_PERMISSIONS_REQUEST_CALL_PHONE = 6;
private static final int MY_PERMISSIONS_REQUEST_CALL_PHONE2 = 7;
private File output;
private Uri imageUri;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
void initView(){
imageView=(ImageView)findViewById(R.id.image);
}
public void takePhone(View view){
if (ContextCompat.checkSelfPermission(this,
Manifest.permission.WRITE_EXTERNAL_STORAGE)
!= PackageManager.PERMISSION_GRANTED)
{
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
MY_PERMISSIONS_REQUEST_CALL_PHONE2);
}else {
takePhoto();
}
}
public void choosePhone(View view){
if (ContextCompat.checkSelfPermission(this,
Manifest.permission.WRITE_EXTERNAL_STORAGE)
!= PackageManager.PERMISSION_GRANTED)
{
ActivityCompat.requestPermissions(this,
new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
MY_PERMISSIONS_REQUEST_CALL_PHONE2);
}else {
choosePhoto();
}
}
/**
* 拍照
*/
void takePhoto(){
/**
* 最後一個引數是資料夾的名稱,可以隨便起
*/
File file=new File(Environment.getExternalStorageDirectory(),"拍照");
if(!file.exists()){
file.mkdir();
}
/**
* 這裡將時間作為不同照片的名稱
*/
output=new File(file,System.currentTimeMillis()+".jpg");
/**
* 如果該資料夾已經存在,則刪除它,否則建立一個
*/
try {
if (output.exists()) {
output.delete();
}
output.createNewFile();
} catch (Exception e) {
e.printStackTrace();
}
/**
* 隱式開啟拍照的Activity,並且傳入CROP_PHOTO常量作為拍照結束後回撥的標誌
*/
imageUri = Uri.fromFile(output);
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, CROP_PHOTO);
}
/**
* 從相簿選取圖片
*/
void choosePhoto(){
/**
* 開啟選擇圖片的介面
*/
Intent intent = new Intent(Intent.ACTION_PICK);
intent.setType("image/*");//相片型別
startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);
}
public void onActivityResult(int req, int res, Intent data) {
switch (req) {
/**
* 拍照的請求標誌
*/
case CROP_PHOTO:
if (res==RESULT_OK) {
try {
/**
* 該uri就是照片資料夾對應的uri
*/
Bitmap bit = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));
imageView.setImageBitmap(bit);
} catch (Exception e) {
Toast.makeText(this,"程式崩潰",Toast.LENGTH_SHORT).show();
}
}
else{
Log.i("tag", "失敗");
}
break;
/**
* 從相簿中選取圖片的請求標誌
*/
case REQUEST_CODE_PICK_IMAGE:
if (res == RESULT_OK) {
try {
/**
* 該uri是上一個Activity返回的
*/
Uri uri = data.getData();
Bitmap bit = BitmapFactory.decodeStream(getContentResolver().openInputStream(uri));
imageView.setImageBitmap(bit);
} catch (Exception e) {
e.printStackTrace();
Log.d("tag",e.getMessage());
Toast.makeText(this,"程式崩潰",Toast.LENGTH_SHORT).show();
}
}
else{
Log.i("liang", "失敗");
}
break;
default:
break;
}
}
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults)
{
if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE)
{
if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
{
takePhoto();
} else
{
// Permission Denied
Toast.makeText(MainActivity.this, "Permission Denied", Toast.LENGTH_SHORT).show();
}
}
if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE2)
{
if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
{
choosePhoto();
} else
{
// Permission Denied
Toast.makeText(MainActivity.this, "Permission Denied", Toast.LENGTH_SHORT).show();
}
}
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}
}