如何在 Flutter 插件中添加 OpenCV

Posted

技术标签:

【中文标题】如何在 Flutter 插件中添加 OpenCV【英文标题】:How to add OpenCV in Flutter Plugin 【发布时间】:2019-04-05 14:01:18 【问题描述】:

我正在做一个颤振项目,我必须检测物体的边缘。我已经搜索了现有的插件,但找不到任何插件。我打算使用 OpenCV 进行边缘检测。

我从 android Studio 创建了一个颤振插件。这是插件结构。

我从这里下载了适用于 Android 的 OpenCV - opencv-3.4.3-android-sdk 并解压缩。

我已将 OpenCV SDK 中的 java 文件夹作为模块导入到 Flutter 插件的 Android 项目中。

这就是我的 android 项目(flutter 插件)build.gradle 的样子

group 'com.xxxx.xxxxx'
version '1.0-SNAPSHOT'

buildscript 
    ext.kotlin_version = '1.2.30'
    repositories 
        google()
        jcenter()
    

    dependencies 
        classpath 'com.android.tools.build:gradle:3.1.2'
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
    


rootProject.allprojects 
    repositories 
        google()
        jcenter()
    


apply plugin: 'com.android.library'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'

android 
    compileSdkVersion 27

    sourceSets 
        main.java.srcDirs += 'src/main/kotlin'
    
    defaultConfig 
        minSdkVersion 16
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    
    lintOptions 
        disable 'InvalidPackage'
    


dependencies 
    implementation "org.jetbrains.kotlin:kotlin-stdlib-jre7:$kotlin_version"
    implementation 'com.android.support:appcompat-v7:27.1.1'
    implementation 'io.reactivex.rxjava2:rxandroid:2.1.0'

    implementation project(':openCVLibrary343')
    compileOnly files('templibs/flutter.jar')

这就是我的 opencv build.gradle 的样子

apply plugin: 'com.android.library'

android 
    compileSdkVersion 27
    buildToolsVersion "28.0.3"

    defaultConfig 
        minSdkVersion 8
        targetSdkVersion 21
    

    buildTypes 
        release 
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.txt'
        
    

这就是 Android 项目(flutter 插件)的 settings.gradle 的样子

rootProject.name = 'flutter_plugin'
include ':openCVLibrary343'

当我尝试运行由颤振插件模板创建的示例应用程序时,出现以下错误。

在调试模式下在 Nexus 6P 上启动 lib\main.dart... 正在初始化梯度... 解决依赖关系... * 运行 Gradle 时出错: 退出代码 1 来自:C:\Users\xxxxx\Documents\flutter_plugin\example\android\gradlew.bat app:properties:

FAILURE:构建失败并出现异常。

在哪里: 构建文件 'C:\Users\xxxx\Documents\flutter_plugin\android\build.gradle' 行:48

出了什么问题: 评估项目 ':flutter_plugin' 时出现问题。

在项目“:flutter_plugin”中找不到路径为“:openCVLibrary343”的项目。

尝试: 使用 --stacktrace 选项运行以获取堆栈跟踪。使用 --info 或 --debug 选项运行以获得更多日志输出。运行 --scan 以获得完整的见解。

通过https://help.gradle.org获得更多帮助

1 秒内构建失败

以错误结束:请检查 android/ 文件夹中的 Gradle 项目设置。

【问题讨论】:

How to use OpenCV in the Flutter SDK?的可能重复 @SnakeyHips 这怎么可能是重复的?我的问题是关于 Gradle 构建错误。 不能通过这种方式将 Android 插件添加到 Flutter 中。您必须使用 Flutter 插件,这就是为什么您会收到错误“在项目 ':flutter_plugin' 中找不到带有路径 ':openCVLibrary343' 的项目”。我也链接了这个帖子不仅说 OpenCV 目前在 Flutter 上不可用,而且还解释了如何provide your own plugin following this.。请看一下。 @SnakeyHips 如果您阅读了我的整个问题,您就会明白我创建了一个颤振插件并将 openCV 添加到插件的 android 部分。 【参考方案1】:

对于我正在进行的一个项目,我设法创建了一个颤振插件,它依赖于 OpenCV 来处理来自相机的图像。请注意,当我说“来自相机”时,我指的不是来自视频捕获的实时帧,而是之前保存在设备内存中的图片。

首先,我遵循了有关创建 plugins here 的 Flutter 文档

当flutter创建插件文件夹和文件时,我最终得到了一个类似这样的结构(第一级的“opencv”文件夹是插件的根目录):

请注意我添加的 opencv-341.jar 文件,其中包含用于控制本机库的 Java 方法,OpenCVPlugin.javaopencv.dart 是从 Flutter 访问 Java 方法的入口点,它们是从 Flutter 创建的。

.java 和 .dart 文件的文件名可能不同,并反映了您在上述插件创建阶段作为参数传递的名称。

.jar 文件可以在 OpenCV 包中作为预编译库找到。在同一个包中,您会在这个小伪装的末尾找到所需的本机库。

[plugin_root]/android 中的build.gradle 文件应该像这样引用jar 文件

dependencies 
    implementation files('opencv/opencv-341.jar')

OpencvPlugin.java 应该承载从 dart 调用的方法。就我而言,为了更好地管理不同的方法,我创建了这样的代码:

if (call.method.equals("processReceiptImage")) 
    if (!call.hasArgument("input")) 
        result.error("NOTFOUND", "File not found", null);
        return;
    
    try 
        String input = call.argument("input");
        int maxWidth = (int) call.argument("maxWidth");
        int quality = (int) call.argument("quality");
        String output = call.argument("output");
        result.success(ProcessReceiptImage.run(input, maxWidth, quality, output));
        return;
     catch (Exception e) 
        e.printStackTrace();
        result.error("UNKNOWN", e.getMessage(), null);
        return;
    

静态方法ProcessReceiptImage包含在相关文件ProcessReceiptImage.java中:

package com.yourplugin.opencv;

import org.opencv.core.Mat;
import org.opencv.core.MatOfInt;
import org.opencv.core.Size;
import org.opencv.imgcodecs.Imgcodecs;
import org.opencv.imgproc.Imgproc;

import java.util.ArrayList;


public class ProcessReceiptImage 
    static String run(String input, int maxWidth, int quality, String output) throws Exception 
        Mat src = Imgcodecs.imread(input, Imgcodecs.CV_LOAD_IMAGE_GRAYSCALE);
        Size size = src.size();

        int height = (int) (maxWidth * size.height / size.width);
        Imgproc.resize(src, src, new Size(maxWidth, height));

        ArrayList<Integer> parameters;
        parameters = new ArrayList();

        parameters.add(Imgcodecs.CV_IMWRITE_JPEG_QUALITY);
        parameters.add(quality);

        MatOfInt par = new MatOfInt();
        par.fromList(parameters);
        Imgcodecs.imwrite(output, src, par);
        return output;
    

从现在开始,如果您运行您的应用程序,它可能会由于缺少库 opencv_java3 而失败。 为了解决这部分,我现在创建的唯一方法是将原生库复制到您的最终项目中(这是您的应用程序的根目录,而不是插件):

我已经为每个我想要定位的平台包含了最终的 .so 本机库,因此无论用户使用什么平台,代码始终可以获取正确的 .so 并正确加载 OpenCV。

提示:由于包含所有 .so 文件会导致 APK 很大,我建议您配置 build.gradle 以便将包拆分为 abis,这样您将拥有一个较小的 apk,它将自动通过Play 商店。

【讨论】:

我已经按照您的步骤创建了这个插件。 github.com/sawankumarbundelkhandi/edge_detection谢谢 嗨 Sawan,我在 pub.dev 存储库中找不到您的插件。什么时候发布插件? 您可以直接从 GitHub 安装该插件。您必须添加 pubspec.yam 文件(在依赖项部分):edge_detection: git: https://github.com/sawankumarbundelkhandi/edge_detection.git @SawanKumarBundelkhandi 能否请您提供一种从画廊图像中检测对象的方法?

以上是关于如何在 Flutter 插件中添加 OpenCV的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Flutter 中添加文件选择器插件?

如何在 Flutter 包/插件开发中添加资产?

如何手动将 .xcframework 添加到 Flutter iOS 插件?

如何将新标头添加到 Flutter 插件 iOS 项目?

如何添加不支持 Null Safety 的 Flutter 插件

如何为 Flutter 添加额外的 iOS 模拟器?