Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> Android拍照,相冊選擇圖片以及Android6.0權限管理

Android拍照,相冊選擇圖片以及Android6.0權限管理

編輯:關於Android編程

概述

在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);

    }

我們通過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;
        }
    }

至此,拍照就結束了

2,從相冊獲取圖片的實現

和拍照非常相似,唯一不同的是打開該Activity的intent不同,打開相冊的代碼如下:

   void choosePhoto(){
        /**
         * 打開選擇圖片的界面
         */
        Intent intent = new Intent(Intent.ACTION_PICK);
        intent.setType("image/*");//相片類型
        startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);

    }

選擇好照片後,會執行onActivityResult(…)方法,在主Activity中需要重寫此方法,並且在此方法中得到相冊Activity傳回的數據進而得到照片的bitmap對象。這一過程和拍照的onActivityResult(…)方法是相同的。

3,動態權限管理

無論是拍照還是從相冊中選擇圖片都涉及到用戶的隱私,所以我們需要聲明權限,所以我們需要在清單文件中加入這句:

在android6.0中,這樣做還是不夠的。我們還需要在代碼中動態聲明權限。啟動拍照或者相冊的Activity時,會彈出一個dialog詢問用戶是否同意授權。

這裡寫圖片描述

只有用戶同意授權後,才能順利完成接下來的操作。需要說明的是,就算用戶拒絕,我們仍然可以打開相機和相冊的Activity,但是操作結束後,並不會返回什麼值,就是說系統拒絕我們讀取數據。
申請動態權限的簡略步驟如下:

首先檢查權限是否已經被授予,如果已經授予,那我們就可以直接執行相關的方法,否則,需要申請權限。 如果權限沒有授予,我們需要申請權限。申請後app會想用戶彈出一個dialog詢問用戶是否授予該權限。

用戶無論是同意還是拒絕授予該權限,Activity都會執行onRequestPermissionsResult(…)方法,我們需要在該方法中判斷用戶是否同意該權限。如果同意,執行相應的方法,如果拒絕,最好向用戶解釋下為什麼需要這個權限。

檢查權限是否被授予:<喎?/kf/ware/vc/" target="_blank" class="keylink">vc3Ryb25nPjwvcD4NCjxwcmUgY2xhc3M9"brush:java;"> //第二個參數是需要申請的權限 if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { //權限還沒有授予,需要在這裡寫申請權限的代碼 }else { //權限已經被授予,在這裡直接寫要執行的相應方法即可 }

申請權限:

ActivityCompat.requestPermissions(this,
                    new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                    MY_PERMISSIONS_REQUEST_CALL_PHONE2);

第二個參數是一個字符串數組,裡面是你需要申請的權限。既然是一個數組,那麼就說明你一次可以申請多個權限。由於拍照和選擇相冊圖片只涉及一個權限,所以上面的字符串數組中就只寫了一個權限。最後一個參數是一個整型常量,用於標志你這次申請的權限,該常量在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);
    }

如果用戶同意,那麼就執行相應的方法,如果拒絕,可以向用戶解釋一下申請該權限的原因。

4,代碼

布局文件

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);
    }
}
  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved