flutter与android原生通信调用原生的拍照和相册选择

文章目录

概要

flutter想要调用拍照选择相册或者视屏播放等等,那么就需要与android原生通信,可以使用插件进行调用,但本文讲的是flutter调用原生的功能以及需要记录容易出问题的点

flutter调用android原生的时候,需要先在android/app/main文件夹下的AndroidManifest.xml文件中的manifest标签下添加相应权限

<uses-permission android:name="android.permission.CAMERA" /> 拍照权限
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> 读取权限
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> 写入权限
<uses-permission android:name="android.permission.INTERNET" /> 网络权限

调用拍照需要注意的细节

需要添加动态获取权限,当进入当前Activity的时候就判断用户是否添加了拍照权限,如果没有那就弹出权限选择给用户选择,

  private void ouJurisdiction() {
        if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.M){
            int check = this.checkSelfPermission(Manifest.permission.CAMERA);
            if(check != PackageManager.PERMISSION_GRANTED){
                this.requestPermissions(new String[]{Manifest.permission.CAMERA},0);
            }else{
                openTakePhoto();
            }
        }
    }

//用来接收用户选择权限的结果,如果添加则进入权限判断然后调用拍照功能
@Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if(requestCode == 0){
            for (int i = 0; i < permissions.length; i++) {
                if(grantResults[i] == PackageManager.PERMISSION_GRANTED){
                    ouJurisdiction();
                }else{
                    backResult("0");
                }
            }
        }
    }

//如果权限通过,那么就调用拍照功能
private void openTakePhoto() {
        File file = null;
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

        try {
            file = createImageFile();
            if(file != null){
                Uri photoUr = FileProvider.getUriForFile(this,"com.example.flutter_android.fileProvider",file);
                intent.putExtra(MediaStore.EXTRA_OUTPUT,photoUr);
                startActivityForResult(intent, 1);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
//这是一个临时的图像,用来展示拍照动态,当点击拍照保存的时候,会进入到onActivityResult方法,然后将临时图片转为永久图片
private File createImageFile() throws IOException {
        File storage = getExternalFilesDir(Environment.DIRECTORY_PICTURES);
        String fileName = "IMG_" + UUID.randomUUID().toString() + ".jpg";
        File imageFile = File.createTempFile(fileName,".jpg",storage);
        imageFile.setReadable(true,false);
        currentPath = imageFile.getAbsolutePath();
        return imageFile;
    }
调用拍照方法后拍照结果的返回,进入到这个方法
  @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if(requestCode == 1 && resultCode == RESULT_OK){
            File teamFile = new File(currentPath);
            saveImageFile(teamFile);
            backResult(currentPath);
        }else{
            System.out.println("拍照失败");
            backResult("0");
        }
    }



//这个方法用来存储永久图片
 private void saveImageFile(File teamFile) {
        File storageDir = getExternalFilesDir(Environment.DIRECTORY_PICTURES);
        if(!storageDir.exists()){
            storageDir.mkdirs();
        }
        String time = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()).format(new Date());
        String fileName = "IMG_" + time + ".jpg";
        File imageFile = new File(storageDir,fileName);
//这个变量是用来返回真实路径给flutter的
        currentPath = imageFile.getAbsolutePath();

        try {
            InputStream in = new FileInputStream(teamFile);
            OutputStream out = new FileOutputStream(imageFile);
            int length;
            byte[] buffer = new byte[1024];
            while ((length = in.read(buffer)) > 0){
                out.write(buffer,0,length);
            }
            in.close();
            out.close();
        }catch (IOException e) {
            e.printStackTrace();
        }
    }



//这个方法用来返回结果
private void backResult(String path) {
        Intent intent = new Intent();
        intent.putExtra("path",path);
        setResult(RESULT_OK,intent);
        finish();
    }

使用上面代码的准备工作在res文件夹下添加一个xml文件夹 再添加一个file_paths资源文件,这段代码是Android的路径配置文件,用于定义应用程序可以访问的文件路径

<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-files-path
        name="my_images"
        path="Pictures"/>
</paths>

然后再在AndroidManifest.xml.xml中将这个文件添加到应用程序目录下,在application下添加

 <provider
            android:name="androidx.core.content.FileProvider"
            android:authorities="${applicationId}.fileProvider"
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/file_paths" />
        </provider>

这里是拍照功能的完整代码,注释就没写了,上面都讲解过了 

package com.example.flutter_android;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;
import androidx.core.content.FileProvider;

import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;
import java.util.UUID;

import io.flutter.embedding.android.FlutterActivity;

public class TakePhotoActivity extends FlutterActivity {

    private String currentPath;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        ouJurisdiction();
    }

    private void ouJurisdiction() {
        if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.M){
            int check = this.checkSelfPermission(Manifest.permission.CAMERA);
            if(check != PackageManager.PERMISSION_GRANTED){
                this.requestPermissions(new String[]{Manifest.permission.CAMERA},0);
            }else{
                openTakePhoto();
            }
        }
    }

    @Override
    protected void onDestroy() {
        backResult("0");
        finish();
        super.onDestroy();
    }

    private void openTakePhoto() {
        File file = null;
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

        try {
            file = createImageFile();
            if(file != null){
                Uri photoUr = FileProvider.getUriForFile(this,"com.example.flutter_android.fileProvider",file);
                intent.putExtra(MediaStore.EXTRA_OUTPUT,photoUr);
                startActivityForResult(intent, 1);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if(requestCode == 1 && resultCode == RESULT_OK){
            File teamFile = new File(currentPath);
            saveImageFile(teamFile);
            backResult(currentPath);
        }else{
            System.out.println("拍照失败");
            backResult("0");
        }
    }

    private void saveImageFile(File teamFile) {
        File storageDir = getExternalFilesDir(Environment.DIRECTORY_PICTURES);
        if(!storageDir.exists()){
            storageDir.mkdirs();
        }
        String time = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()).format(new Date());
        String fileName = "IMG_" + time + ".jpg";
        File imageFile = new File(storageDir,fileName);
        currentPath = imageFile.getAbsolutePath();

        try {
            InputStream in = new FileInputStream(teamFile);
            OutputStream out = new FileOutputStream(imageFile);
            int length;
            byte[] buffer = new byte[1024];
            while ((length = in.read(buffer)) > 0){
                out.write(buffer,0,length);
            }
            in.close();
            out.close();
        }catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if(requestCode == 0){
            for (int i = 0; i < permissions.length; i++) {
                if(grantResults[i] == PackageManager.PERMISSION_GRANTED){
                    ouJurisdiction();
                }else{
                    backResult("0");
                }
            }
        }
    }

    private void backResult(String path) {
        Intent intent = new Intent();
        intent.putExtra("path",path);
        setResult(RESULT_OK,intent);
        finish();
    }

    private File createImageFile() throws IOException {
        File storage = getExternalFilesDir(Environment.DIRECTORY_PICTURES);
        String fileName = "IMG_" + UUID.randomUUID().toString() + ".jpg";
        File imageFile = File.createTempFile(fileName,".jpg",storage);
        imageFile.setReadable(true,false);
        currentPath = imageFile.getAbsolutePath();
        return imageFile;
    }
}

调用相册选择需要注意的细节

权限在上面已经讲解过了,就不做过多赘述了,这个方法是用来判断用户选择的权限

@Override    //接收权限选择的回调
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if(requestCode == 0){
            for (int i = 0; i < permissions.length; i++) {
                if(grantResults[i] == PackageManager.PERMISSION_GRANTED){
                    ouJurisidction();
                }else{
                    backResult("0");
                }
            }
        }
    }
 
//这个方法是用来判断权限如果没有权限那就添加权限,让用户选择是否添加权限
private void ouJurisdiction() {
        int check = ContextCompat.checkSelfPermission(this,Manifest.permission.READ_EXTERNAL_STORAGE);
        if(check != PackageManager.PERMISSION_GRANTED){
            ActivityCompat.requestPermissions(this,new String[]{
                    Manifest.permission.READ_EXTERNAL_STORAGE,
                    Manifest.permission.WRITE_EXTERNAL_STORAGE
            },0);
        }else{
            openPickImage();
        }
    }

打开相册选择,选取一张照片

 private void openPickImage() {
        Intent intent = new Intent(Intent.ACTION_PICK,null);
        intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,null);
        startActivityForResult(intent,21);
    }


   @Override    //相册结果的返回 如果选择了那么会产生一个Uri然后将uri处理成真实路径
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if(requestCode == 21 && resultCode == RESULT_OK){
            Uri uri = data.getData();
            backResult(realPathFromUri(uri));
        }else{
            backResult("0");
        }
    }

    //处理uri返回真实路径
    private String realPathFromUri(Uri uri) {
        String realPath = "0";
        String[] projection = {MediaStore.Images.Media.DATA};
        Cursor cursor = getContentResolver().query(uri,projection,null,null,null);
        if(cursor != null && cursor.moveToFirst()){
            int columnIndex = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
            realPath = cursor.getString(columnIndex);
        }
        cursor.close();
        return realPath;
    }

    //将结果返回
    private void backResult(String path) {
        Intent intent = new Intent();
        intent.putExtra("path",path);
        setResult(RESULT_OK,intent);
        finish();
    }

    //如果用户直接点击自带的返回按钮,那就直接返回0
    @Override
    protected void onDestroy() {
        backResult("0");
        super.onDestroy();
    }

小结

我待程序如初恋,它却虐我千百遍

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值