Android7.0 調用系統相機,相冊,剪裁上傳


由於在Android 7.0 采用了StrictMode API政策禁,其中有一條限制就是對目錄訪問的限制。

這項變更意味着我們無法通過File API訪問手機存儲上的數據,也就是說,給其他應用傳遞 file:// URI 類型的Uri,可能會導致接受者無法訪問該路徑,並且會會觸發 FileUriExposedException異常。

StrictMode API政策禁中的應用間共享文件就是對上述限制的應對方法,它指明了我們在在應用間共享文件可以發送 content:// URI類型的Uri,並授予 URI 臨時訪問權限,即使用FileProvider

接下來,我們使用FileProvider實現調用系統相機、相冊、剪裁圖片的功能兼容Android 7.0

第一步:FileProvider相關准備工作

  • 在AndroidManifest.xml中增加provider節點,如下:
     <provider
                android:name="android.support.v4.content.FileProvider"
                android:authorities="com.hansion.chosehead"
                android:grantUriPermissions="true"
                android:exported="false">
                <meta-data
                    android:name="android.support.FILE_PROVIDER_PATHS"
                    android:resource="@xml/filepaths" />
            </provider>

     

    其中: android:authorities 表示授權列表,填寫你的應用包名,當有多個授權時,用分號隔開 android:exported 表示該內容提供器(ContentProvider)是否能被第三方程序組件使用,必須為false,否則會報異常:ava.lang.RuntimeException: Unable to get provider android.support.v4.content.FileProvider: java.lang.SecurityException: Provider must not be exportedandroid:grantUriPermissions="true" 表示授予 URI 臨時訪問權限 android:resource 屬性指向我們自及創建的xml文件的路徑,文件名隨便起

  • 接下來,我們需要在資源(res)目錄下創建一個xml目錄,並建立一個以上面名字為文件名的xml文件,內容如下:
    <?xml version="1.0" encoding="utf-8"?>
    <paths>
        <external-path path="" name="image" />
    </paths>

     

    其中: external-path 代表根目錄為: Environment.getExternalStorageDirectory() ,也可以寫其他的,如: files-path 代表根目錄為:Context.getFilesDir() cache-path 代表根目錄為:getCacheDir() 其path屬性的值代表路徑后層級名稱,為空則代表就是根目錄,假如為“pictures”,就代表對應根目錄下的pictures目錄

     

第二步:使用FileProvider

  • 在這之前,我們需要在AndroidManifest.xml中增加必要的讀寫權限:
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
        <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

    1. 通過相機獲取圖片

    在通過Intent跳轉系統相機前,我們需要對版本進行判斷,如果在Android7.0以上,使用FileProvider獲取Uri,代碼如下:

    /**
         * 從相機獲取圖片
         */
        private void getPicFromCamera() {
            //用於保存調用相機拍照后所生成的文件
            tempFile = new File(Environment.getExternalStorageDirectory().getPath(), System.currentTimeMillis() + ".jpg");
            //跳轉到調用系統相機
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            //判斷版本
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {   //如果在Android7.0以上,使用FileProvider獲取Uri
                intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
                Uri contentUri = FileProvider.getUriForFile(MainActivity.this, "com.hansion.chosehead", tempFile);
                intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);
            } else {    //否則使用Uri.fromFile(file)方法獲取Uri
                intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(tempFile));
            }
            startActivityForResult(intent, CAMERA_REQUEST_CODE);
        }

    如果你好奇通過FileProvider獲取的Uri是什么樣的,可以打印出來看一看,例如:

    content://com.hansion.chosehead/image/1509356493642.jpg
    

      其中: com.hansion.chosehead 是我的包名 image 是上文中xml文件中的name屬性的值 1509356493642.jpg 是我創建的圖片的名字 也就是說,content://com.hansion.chosehead/image/ 代表的就是根目錄

    2.通過相冊獲取圖片

    /**
         * 從相冊獲取圖片
         */
        private void getPicFromAlbm() {
            Intent photoPickerIntent = new Intent(Intent.ACTION_PICK);
            photoPickerIntent.setType("image/*");
            startActivityForResult(photoPickerIntent, ALBUM_REQUEST_CODE);
        }

    3.剪裁圖片

    /**
         * 裁剪圖片
         */
        private void cropPhoto(Uri uri) {
            Intent intent = new Intent("com.android.camera.action.CROP");
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
            intent.setDataAndType(uri, "image/*");
            intent.putExtra("crop", "true");
            intent.putExtra("aspectX", 1);
            intent.putExtra("aspectY", 1);
            intent.putExtra("outputX", 300);
            intent.putExtra("outputY", 300);
            intent.putExtra("return-data", true);
            startActivityForResult(intent, CROP_REQUEST_CODE);
        }

    第三步:接收圖片信息

  • 我們在onActivityResult方法中獲得返回的圖片信息,在這里我們會先調用剪裁去剪裁圖片,然后對剪裁返回的圖片進行設置、保存、上傳等操作
     @Override
        protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
            switch (requestCode) {
                case CAMERA_REQUEST_CODE:   //調用相機后返回
                    if (resultCode == RESULT_OK) {
                        //用相機返回的照片去調用剪裁也需要對Uri進行處理
                        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                            Uri contentUri = FileProvider.getUriForFile(MainActivity.this, "com.hansion.chosehead", tempFile);
                            cropPhoto(contentUri);
                        } else {
                            cropPhoto(Uri.fromFile(tempFile));
                        }
                    }
                    break;
                case ALBUM_REQUEST_CODE:    //調用相冊后返回
                    if (resultCode == RESULT_OK) {
                        Uri uri = intent.getData();
                        cropPhoto(uri);
                    }
                    break;
                case CROP_REQUEST_CODE:     //調用剪裁后返回
                    Bundle bundle = intent.getExtras();
                    if (bundle != null) {
                        //在這里獲得了剪裁后的Bitmap對象,可以用於上傳
                        Bitmap image = bundle.getParcelable("data");
                        //設置到ImageView上
                        mHeader_iv.setImageBitmap(image);
                        //也可以進行一些保存、壓縮等操作后上傳
    //                    String path = saveImage("crop", image);
                    }
                    break;
            }
        }
  • 保存Bitmap到本地的方法:
     public String saveImage(String name, Bitmap bmp) {
            File appDir = new File(Environment.getExternalStorageDirectory().getPath());
            if (!appDir.exists()) {
                appDir.mkdir();
            }
            String fileName = name + ".jpg";
            File file = new File(appDir, fileName);
            try {
                FileOutputStream fos = new FileOutputStream(file);
                bmp.compress(Bitmap.CompressFormat.PNG, 100, fos);
                fos.flush();
                fos.close();
                return file.getAbsolutePath();
            } catch (IOException e) {
                e.printStackTrace();
            }
            return null;
        }

    至此,對Android7.0的兼容就結束了 總結一下,在調用相機和剪裁時,傳入的Uri需要使用FileProvider來獲取

    public class MainActivity extends AppCompatActivity implements View.OnClickListener {
        private ImageView mHeader_iv;
        //相冊請求碼
        private static final int ALBUM_REQUEST_CODE = 1;
        //相機請求碼
        private static final int CAMERA_REQUEST_CODE = 2;
        //剪裁請求碼
        private static final int CROP_REQUEST_CODE = 3;
        //調用照相機返回圖片文件
        private File tempFile;
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            initView();
        }
        private void initView() {
            mHeader_iv = (ImageView) findViewById(R.id.mHeader_iv);
            Button mGoCamera_btn = (Button) findViewById(R.id.mGoCamera_btn);
            Button mGoAlbm_btn = (Button) findViewById(R.id.mGoAlbm_btn);
            mGoCamera_btn.setOnClickListener(this);
            mGoAlbm_btn.setOnClickListener(this);
        }
        @Override
        public void onClick(View view) {
            switch (view.getId()) {
                case R.id.mGoCamera_btn:
                    getPicFromCamera();
                    break;
                case R.id.mGoAlbm_btn:
                    getPicFromAlbm();
                    break;
                default:
                    break;
            }
        }
        /**
         * 從相機獲取圖片
         */
        private void getPicFromCamera() {
            //用於保存調用相機拍照后所生成的文件
            tempFile = new File(Environment.getExternalStorageDirectory().getPath(), System.currentTimeMillis() + ".jpg");
            //跳轉到調用系統相機
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            //判斷版本
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {   //如果在Android7.0以上,使用FileProvider獲取Uri
                intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
                Uri contentUri = FileProvider.getUriForFile(MainActivity.this, "com.hansion.chosehead", tempFile);
                intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);
                Log.e("dasd", contentUri.toString());
            } else {    //否則使用Uri.fromFile(file)方法獲取Uri
                intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(tempFile));
            }
            startActivityForResult(intent, CAMERA_REQUEST_CODE);
        }
        /**
         * 從相冊獲取圖片
         */
        private void getPicFromAlbm() {
            Intent photoPickerIntent = new Intent(Intent.ACTION_PICK);
            photoPickerIntent.setType("image/*");
            startActivityForResult(photoPickerIntent, ALBUM_REQUEST_CODE);
        }
        /**
         * 裁剪圖片
         */
        private void cropPhoto(Uri uri) {
            Intent intent = new Intent("com.android.camera.action.CROP");
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
            intent.setDataAndType(uri, "image/*");
            intent.putExtra("crop", "true");
            intent.putExtra("aspectX", 1);
            intent.putExtra("aspectY", 1);
            intent.putExtra("outputX", 300);
            intent.putExtra("outputY", 300);
            intent.putExtra("return-data", true);
            startActivityForResult(intent, CROP_REQUEST_CODE);
        }
        @Override
        protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
            switch (requestCode) {
                case CAMERA_REQUEST_CODE:   //調用相機后返回
                    if (resultCode == RESULT_OK) {
                        //用相機返回的照片去調用剪裁也需要對Uri進行處理
                        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                            Uri contentUri = FileProvider.getUriForFile(MainActivity.this, "com.hansion.chosehead", tempFile);
                            cropPhoto(contentUri);
                        } else {
                            cropPhoto(Uri.fromFile(tempFile));
                        }
                    }
                    break;
                case ALBUM_REQUEST_CODE:    //調用相冊后返回
                    if (resultCode == RESULT_OK) {
                        Uri uri = intent.getData();
                        cropPhoto(uri);
                    }
                    break;
                case CROP_REQUEST_CODE:     //調用剪裁后返回
                    Bundle bundle = intent.getExtras();
                    if (bundle != null) {
                        //在這里獲得了剪裁后的Bitmap對象,可以用於上傳
                        Bitmap image = bundle.getParcelable("data");
                        //設置到ImageView上
                        mHeader_iv.setImageBitmap(image);
                        //也可以進行一些保存、壓縮等操作后上傳
    //                    String path = saveImage("crop", image);
                    }
                    break;
            }
        }
        public String saveImage(String name, Bitmap bmp) {
            File appDir = new File(Environment.getExternalStorageDirectory().getPath());
            if (!appDir.exists()) {
                appDir.mkdir();
            }
            String fileName = name + ".jpg";
            File file = new File(appDir, fileName);
            try {
                FileOutputStream fos = new FileOutputStream(file);
                bmp.compress(Bitmap.CompressFormat.PNG, 100, fos);
                fos.flush();
                fos.close();
                return file.getAbsolutePath();
            } catch (IOException e) {
                e.printStackTrace();
            }
            return null;
        }
    }
    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:padding="5dp"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center_horizontal"
            android:layout_marginTop="30dp"
            android:text="選擇頭像"
            android:textSize="18sp" />
        <ImageView
            android:id="@+id/mHeader_iv"
            android:layout_width="100dp"
            android:layout_height="100dp"
            android:layout_gravity="center_horizontal"
            android:layout_marginTop="50dp"
            android:src="@mipmap/ic_launcher" />
        <android.support.v4.widget.Space
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:layout_weight="1"/>
        <Button
            android:id="@+id/mGoCamera_btn"
            android:text="拍照選擇"
            android:layout_marginBottom="5dp"
            android:layout_width="match_parent"
            android:layout_height="wrap_content" />
        <Button
            android:id="@+id/mGoAlbm_btn"
            android:text="本地相冊選擇"
            android:layout_marginBottom="10dp"
            android:layout_width="match_parent"
            android:layout_height="wrap_content" />
    </LinearLayout>

     

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM