拍照和调用系统相册(一)

作者: 淡日临窗 | 来源:发表于2018-06-19 09:45 被阅读69次

    拍照(调用相机)

    调用系统相机要注意的几个问题:

    1. Android 6.0+ 权限问题

      • Android版本6.0以上(含6.0)需要申请权限
        这里就贴下代码:
        首先加入权限
      <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
      <uses-permission android:name="android.permission.CAMERA"/>
      

      申请权限(这里除了相机权限,调用相册还需要文件读写权限)

      //检查权限
      int ret = ActivityCompat.checkSelfPermission(ActivateActivity.this, Manifest.permission .CAMERA);
      //申请权限
      if (ret != PackageManager.PERMISSION_GRANTED) {
           ActivityCompat.requestPermissions(this,new String[] {Manifest.permission.android.Manifest.permission.CAMERA)},
                       RESULT_CODE_CAMERA);
               return;
           }
       }
      
      //回调
      @Override
      public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
       switch (requestCode) {
           case RESULT_CODE_CAMERA: 
                 //TODO
                 break;
      
      
    2. Camera API使用问题

      • 在API21中Google发布了Camera2类来取代Camera类,所以当API版本在21以上可使用Camera2,API版本在21以下必须使用Camera;
    3. Android7.0调用相机问题

      • android升级到7.0后对权限又做了一个更新即应用间传递文件时不允许出现以file://的形式 调用隐式APP,需要用共享文件的形式:content:// URI,如果给出一个file://格式的URI的话,应用会抛出FileUriExposedException。这是由于谷歌认为目标app可能不具有文件权限,会造成潜在的问题,这里有两种解决方式

        1. 使用共享文件的形式(FileProvider方式)
          这是谷歌官方推荐的解决方案。即使用FileProvider来生成一个content://格式的URI。具体实现方式如下:

          • 在manifest文件中声明

            //android:name值是固定的
            //android:authorities随便写但是必须得保证唯一性,这边用的是包名
            //android:grantUriPermission和android:exported 是固定值
            <manifest>
                 ... 
                <application>
                  ...
                  <provider
                      android:name="android.support.v4.content.FileProvider"
                      android:authorities="com.test.fileprovider"
                      android:exported="false"
                      android:grantUriPermissions="true">
                      <meta-data
                         android:name="android.support.FILE_PROVIDER_PATHS"
                         android:resource="@xml/file_paths" />
                 </provider>
                 ...
               </application>
            </manifest>
            
          • 在res文件夹下新建一个xml文件夹,在xml文件夹下新建file_paths.xml文件

            //paths标签下的子元素一共有以下几种:
            //files-path 对应  Context.getFilesDir()
            //cache-path 对应  Context.getCacheDir()
            //external-path 对应 Environment.getExternalStorageDirectory()
            //external-files-path 对应  Context.getExternalFilesDir()
            //external-cache-path 对应  Context.getExternalCacheDir()
            <paths xmlns:android="http://schemas.android.com/apk/res/android">
                 //name表示生成URI时的别名,path是指相对路径。
                <files-path name="my_images" path="images/"/>
                 ...
            </paths>
            
          • 使用

            String filePath = Environment.getExternalStorageDirectory() + "/images/"+System.currentTimeMillis()+".jpg";
            File outputFile = new File(filePath);
            if (!outputFile.getParentFile().exists()) {
               outputFile.getParentFile().mkdir();
            }
            //创建content URI
            Uri imgUri = FileProvider.getUriForFile(this,"com.test.fileprovider", outputFile);
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, imgUri);
            intent.addFlags(FLAG_GRANT_READ_URI_PERMISSION);
            intent.addFlags(FLAG_GRANT_WRITE_URI_PERMISSION);
            startActivityForResult(intent, REQUEST_TAKE_PICTURE);
            
            //在回调onActivityForResult中获取图片URI
            @Override
            public  void onActivityResult(int requestCode, int resultCode, Intent data) {
               super.onActivityResult(requestCode, resultCode, data);
               switch (requestCode){
                   case REQUEST_TAKE_PICTURE:
                         Uri imgUri = FileProvider.getUriForFile(getActivity(), "com.test.fileprovider", mCameraFile);
                       break;
               }
              
            
            
            
        2. 开启严苛模式

          Android 2.3提供一个称为严苛模式(StrictMode)的调试特性,
          它将报告与线程及虚拟机相关的策略违例。一旦检测到策略违例(policy violation),你将获得警告,其包含了一个栈trace显示你的应用在何处发生违例。你可以强制用警告代替崩溃(crash),也可以仅将警告计入日志,让你的应用继续执行;

          严格模式的开启可以放在Application或者Activity以及其他组件的onCreate方法。为了更好地分析应用中的问题,建议放在Application的onCreate方法中。

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            StrictMode.VmPolicy.Builder builder = new  StrictMode.VmPolicy.Builder().detectFileUriExposure();
            StrictMode.setVmPolicy(builder.build());
         }
        
        

    相关文章

      网友评论

        本文标题:拍照和调用系统相册(一)

        本文链接:https://www.haomeiwen.com/subject/iesoqxtx.html