当前位置: 首页 > news >正文

flutter与原生Android通信方式之MethodChannel

闲来无事,flutter好久没看了,上次折腾flutter与Android通信没折腾完,有些事情耽搁了,这次继续

演示效果:

flutter与Android原生通信

flutter端

import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
class LikePages extends StatefulWidget {const LikePages({super.key});@overrideState<LikePages> createState() => _LikePagesState();
}class _LikePagesState extends State<LikePages> {final MethodChannel _channel = MethodChannel('abc');
// Flutter 调用 Android 方法Future<void> callNativeMethod() async {try {final String result = await _channel.invokeMethod('getPlatformVersion');print('Result from Android: $result');} catch (e) {print('Error calling native method: $e');}}Future<void> callNativeOpenCameraMethod() async {try {final String result = await _channel.invokeMethod('openCamera');print('Result from Android: $result');} catch (e) {print('Error calling native method: $e');}}@overrideWidget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text('like'),),body: Column(children: [Center(child: GestureDetector(child: Text('like'),onTap: (){// 调用 Flutter 调用 Android 方法callNativeMethod();},),),Padding(padding: const EdgeInsets.all(80.0),child: Center(child: GestureDetector(child: Text('openCamare'),onTap: (){// 调用 Flutter 调用 Android 方法callNativeOpenCameraMethod();},),),),],),);}
}
Android端
package com.example.f4import android.content.Context
import android.content.ContextWrapper
import android.content.Intent
import android.graphics.Bitmap
import android.net.Uri
import android.os.Build.VERSION
import android.os.Bundle
import android.provider.MediaStore
import android.util.Log
import android.widget.Toast
import androidx.annotation.NonNull
import androidx.annotation.Nullable
import com.hjq.permissions.OnPermissionCallback
import com.hjq.permissions.Permission
import com.hjq.permissions.XXPermissions
import io.flutter.embedding.android.FlutterActivity
import io.flutter.embedding.engine.FlutterEngine
import io.flutter.plugin.common.MethodChannel
import java.io.File
import java.io.FileOutputStream
import java.io.IOException
import java.util.*class MainActivity: FlutterActivity() {private val CHANNEL = "abc"private val REQUEST_IMAGE_CAPTURE = 1private val PERMISSION_REQUEST_CODE = 200var resultOut:MethodChannel.Result?=nulloverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)XXPermissions.with(this) // 申请单个权限.permission(Permission.CAMERA) // 申请多个权限.permission(Permission.Group.CALENDAR) // 设置权限请求拦截器(局部设置)//.interceptor(new PermissionInterceptor())// 设置不触发错误检测机制(局部设置)//.unchecked().request(object : OnPermissionCallback {override fun onGranted(permissions: List<String>, allGranted: Boolean) {if (!allGranted) {
//                        Toast(this@MainActivity,"获取部分权限成功,但部分权限未正常授予",Toast.LENGTH_SHORT).show()return}
//                    toast("获取录音和日历权限成功")}override fun onDenied(permissions: List<String>, doNotAskAgain: Boolean) {if (doNotAskAgain) {
//                        toast("被永久拒绝授权,请手动授予录音和日历权限")// 如果是被永久拒绝就跳转到应用权限系统设置页面XXPermissions.startPermissionActivity(context, permissions)} else {
//                        toast("获取录音和日历权限失败")}}})}override fun configureFlutterEngine(@NonNull flutterEngine: FlutterEngine) {super.configureFlutterEngine(flutterEngine)// 创建 MethodChannelMethodChannel(flutterEngine.dartExecutor.binaryMessenger, CHANNEL).setMethodCallHandler { call, result ->// 处理 Flutter 调用 Android 方法if (call.method == "getPlatformVersion") {result.success("Android ${VERSION.RELEASE}")}else if (call.method=="openCamera"){Log.e("TAG","call.method:="+call.method);resultOut = resultdispatchTakePictureIntent()} else {result.notImplemented()}}}// 获取真实路径的函数private fun getRealPathFromUri(uri: Uri): String? {val projection = arrayOf(MediaStore.Images.Media.DATA)val cursor = contentResolver.query(uri, projection, null, null, null)return cursor?.use {val columnIndex = it.getColumnIndexOrThrow(MediaStore.Images.Media.DATA)it.moveToFirst()it.getString(columnIndex)}}fun saveBitmapToFile(context: Context, bitmap: Bitmap): String {val cw = ContextWrapper(context)val directory: File = cw.getDir("images", Context.MODE_PRIVATE)// 为文件生成唯一的名称val imageName = "img_${UUID.randomUUID()}.jpg"// 创建文件对象val file = File(directory, imageName)var fos: FileOutputStream? = nulltry {fos = FileOutputStream(file)bitmap.compress(Bitmap.CompressFormat.JPEG, 100, fos) // 将 Bitmap 压缩为 JPEG 格式} catch (e: IOException) {e.printStackTrace()} finally {try {fos?.close()} catch (e: IOException) {e.printStackTrace()}}// 返回文件的路径return file.absolutePath}override fun onActivityResult(requestCode: Int, resultCode: Int,data: Intent) {super.onActivityResult(requestCode, resultCode, data)Log.e("TAG","==-->${data.extras}"+"  resultCode:=$resultCode "+" requestCode:$requestCode ")if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == RESULT_OK) {// 图片从相机返回val extras = data.extrasif (extras != null) {// 获取拍摄的照片val imageBitmap = extras.get("data") as Bitmap?// 保存 Bitmap 到文件并获取文件路径val imagePath = imageBitmap?.let { saveBitmapToFile(applicationContext, it) }// 在这里使用文件路径resultOut?.success(imagePath)}} else {Toast.makeText(this, "Failed to capture image", Toast.LENGTH_SHORT).show()}}private fun dispatchTakePictureIntent() {val takePictureIntent = Intent(MediaStore.ACTION_IMAGE_CAPTURE)if (takePictureIntent.resolveActivity(packageManager) != null) {startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE)}}
}

上述代码演示了直接获取Android版本号,以及调用Android原生拍照并发挥图片路径到Flutter的方式,至于为什么会想要搞搞这个,原因还是在与Flutter插件的稳定性问题,假如要实现一个功能,但是受限于Flutter插件自身的问题,就只能自己来实现了,这时候通过原生代码返回需要的数据就不失为一种解决方案,

相关文章:

flutter与原生Android通信方式之MethodChannel

闲来无事&#xff0c;flutter好久没看了&#xff0c;上次折腾flutter与Android通信没折腾完&#xff0c;有些事情耽搁了&#xff0c;这次继续 演示效果&#xff1a; flutter与Android原生通信 flutter端 import package:flutter/cupertino.dart; import package:flutter/mater…...

[PyTorch][chapter 66][强化学习-值函数近似]

前言 现实强化学习任务面临的状态空间往往是连续的,无穷多个。 这里主要针对这种连续的状态空间处理。后面DQN 也是这种处理思路。 目录&#xff1a; 1&#xff1a; 原理 2&#xff1a; 梯度更新 3&#xff1a; target 和 预测值 4 流程 一 原理 强化学习最重要的是得到 …...

hdlbits系列verilog解答(Exams/m2014 q4e)-46

文章目录 一、问题描述二、verilog源码三、仿真结果 一、问题描述 实现以下电路&#xff1a; 二、verilog源码 module top_module (input in1,input in2,output out);assign out ~(in1 | in2);endmodule三、仿真结果 转载请注明出处&#xff01;...

小程序如何实现下拉刷新?

一、全局下拉刷新 在app.json的window节点中&#xff0c;将enablePullDownRefresh设置为true&#xff1b; onPullDownRefresh: function () {console.log(下拉刷新);// 在这里编写数据更新的逻辑wx.stopPullDownRefresh(); // 数据更新完成后&#xff0c;调用该方法停止刷新}二…...

二进制数据转换成十六进制表示 binascii.hexlify()

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 二进制数据转换成十六进制表示 binascii.hexlify() 选择题 binascii.hexlify()参数的数据类型可以是&#xff1f; import binascii number 11 byte_data number.to_bytes() hex_data bin…...

苍穹外卖--店铺营业状态设置

需求分析和设计 1.1.1 产品原型 进到苍穹外卖后台&#xff0c;显示餐厅的营业状态&#xff0c;营业状态分为营业中和打烊中&#xff0c;若当前餐厅处于营业状态&#xff0c;自动接收任何订单&#xff0c;客户可在小程序进行下单操作&#xff1b;若当前餐厅处于打烊状态&#…...

2023金盾杯线上赛-AGRT战队-WP

目录 WEB ApeCoin get_source ezupload easyphp MISC 来都来了 芙宁娜 Honor Crypto 我看看谁还不会RSA hakiehs babyrsa PWN sign-format RE Re1 WEB ApeCoin 扫描发现有源码泄露&#xff0c;访问www.tar.gz得到源码。 在源码中发现了冰蝎马。 Md5解码&am…...

Python面向对象编程——类方法、实例方法和静态方法总结

在Python面向对象编程中&#xff0c;类方法&#xff08;class methods&#xff09;、实例方法&#xff08;instance methods&#xff09;和静态方法&#xff08;static methods&#xff09;是不同类型的方法&#xff0c;它们有一些联系&#xff0c;但也存在一些明显的区别。 类…...

HarmonyOS开发(五):常用基础组件

1、组件介绍 组件&#xff08;Component&#xff09;,是界面搭建及显示的最小单元。 组件根据功能可以分为五大类&#xff1a;基础组件、容器组件、媒体组件、绘制组件、画布组件 2、基础组件 基础组件是视图层的基本组成单元&#xff0c;它包含&#xff1a;Text、Image、T…...

Hive中常出现的错误(不定时更新)

1.加载数据失败 hive> load data local inpath /home/user/hive.txt into table studentl> ; FAILED: SemanticException [Error 10001]: Line 1:56 Table not found studentl hive> load data local inpath /home/user/hive.txt into table student; Loading data to…...

c++ 重写 多态

1 重写(继承后(拼接基类后)) 1.1 非虚函数 同名成员函数 (各自有一个xFunction() 内存 ) #include <iostream> #include <String> class BaseClass { public:void xFunction() {std::cout << "BaseClass::xFunction()\n"; } };class Subclass1 …...

Git如何修改提交(commit)用户名称(user.name)和邮箱(user.email)

Git用户名 Git查看用户名 git config user.name修改Git提交用户名 修改全局Git用户名 git config --global user.name "xx" 修改当前服务/项目Git用户名 git config user.name "xx"如果出现以下错误&#xff0c;解决方案如下&#xff1a; 错误案例&am…...

知行之桥EDI系统HTTP签名验证

本文简要概述如何在知行之桥EDI系统中使用 HTTP 签名身份验证&#xff0c;并将使用 CyberSource 作为该集成的示例。 API 概述 首字母缩略词 API 代表“应用程序编程接口”。这听起来可能很复杂&#xff0c;但真正归结为 API 是一种允许两个不同实体相互通信的软件。自开发以…...

C++ DAY08 异常

概念 异常事件&#xff08;如&#xff1a;除 0 溢出&#xff0c;数组下标越界&#xff0c;所要读取的文件不存在 , 空指针&#xff0c;内存不足 等等&#xff09; 在 C 语言对错误的处理是两种方法&#xff1a; 一是使用整型的返回值标识错误&#xff1b; 二是使用 errn…...

vscode导入STM32CubeIDE工程文件夹未定义警告清除方法

0 前言 在我们使用vscode去编辑STM32CubeIDE的工程文件时&#xff0c;经常会出现一些类型未定义、头文件路径无效的问题&#xff0c;无法正常使用且非常影响观感。本文介绍如何设置vscode导入的STM32CubeIDE配置文件&#xff0c;解决这一问题。 1 vscode导入STM32CubeIDE工程…...

SparkSQL之Optimized LogicalPlan生成过程

经过Analyzer的处理&#xff0c;Unresolved LogicalPlan已经解析成为Analyzed LogicalPlan。Analyzed LogicalPlan中自底向上节点分别对应Relation、Subquery、Filter和Project算子。   Analyzed LogicalPlan基本上是根据Unresolved LogicalPlan一对一转换过来的&#xff0c;…...

Java中有几种基本数据类型以及转换方式【Java面经(1)】

问&#xff1a;Java中有几种基本数据类型呢&#xff1f;以及它们之间的转换方式。详细介绍下 总共有8种基本数据类型 byte 、short 、long 、float 、double 、boolean 、char 详细类型以及字节数&#xff1a; 基本数据类型的转换方式 自动类型转换&#xff1a;小–>大 byt…...

JVM虚拟机:JVM调优第一步,了解JVM常用命令行参数

本文重点 从本文课程开始&#xff0c;我们将用几篇文章来介绍JVM中常用的命令行的参数&#xff0c;这个非常重要&#xff0c;第一我们可以通过参数了解JVM的配置&#xff0c;第二我们可以通过参数完成对JVM的调参。以及后面的JVM的调优也需要用到这些参数&#xff0c;所以我们…...

CSS特效019:图标图片悬浮旋转一周

CSS常用示例100专栏目录 本专栏记录的是经常使用的CSS示例与技巧&#xff0c;主要包含CSS布局&#xff0c;CSS特效&#xff0c;CSS花边信息三部分内容。其中CSS布局主要是列出一些常用的CSS布局信息点&#xff0c;CSS特效主要是一些动画示例&#xff0c;CSS花边是描述了一些CSS…...

requests请求django接口跨域问题处理

参考&#xff1a; https://zhuanlan.zhihu.com/p/416978320 https://blog.csdn.net/SweetHeartHuaZai/article/details/130983179 使用httpx代替requests import httpxheaders {"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.3…...

124.二叉树中的最大路径和

package org.example;class Solution {/*** 最大路径和*/private int maxPathSum Integer.MIN_VALUE;public int maxPathSum(TreeNode root) {maxPath(root);return maxPathSum;}/*** 计算经过 treeNode 结点的路径的最大路径和* * param treeNode 结点* return 经过 treeNode…...

PX4飞控固件编译调试避坑实录:从GCC版本冲突到Python模块缺失的完整解决流程

PX4飞控固件编译调试避坑实录&#xff1a;从GCC版本冲突到Python模块缺失的完整解决流程 当你在深夜的办公室里&#xff0c;面对着满屏红色错误提示的终端窗口&#xff0c;PX4固件编译又一次失败了——这可能是每个无人机开发者都经历过的噩梦时刻。不同于简单的"复制粘贴…...

长沙心理科医院暖心指南+真实案例分享

行业痛点分析在长沙&#xff0c;心理疾病就诊仍面临多重挑战。据《2023年湖南省心理健康白皮书》显示&#xff0c;约62%的轻度心理障碍患者因“怕被歧视”而延迟就诊&#xff0c;其中39%将情绪波动误认为“性格问题”&#xff0c;导致轻症恶化为中重度。三甲医院普遍面临心理科…...

Dify 社区版本地部署实战:从零到一的Docker Compose避坑指南

1. 为什么选择Docker Compose部署Dify社区版 第一次接触Dify社区版时&#xff0c;我被它"开箱即用"的特性吸引。作为一个长期在AI应用开发领域摸爬滚打的开发者&#xff0c;我深知搭建一个完整的LLM应用开发生态需要多少工作量。Dify把模型接入、Prompt工程、RAG流程…...

为什么Java的try块里定义的变量在finally块中不可见?

为什么Java的try块里定义的变量在finally块中不可见&#xff1f; 在Java编程中&#xff0c;try-catch-finally结构是处理异常的核心机制。许多开发者可能会遇到一个奇怪的现象&#xff1a;在try块中定义的变量&#xff0c;在finally块中无法直接访问。这看似违反直觉的设计背后…...

如何通过插件化架构解决Java字节码编辑工具的扩展性难题

如何通过插件化架构解决Java字节码编辑工具的扩展性难题 【免费下载链接】Recaf The modern Java bytecode editor 项目地址: https://gitcode.com/gh_mirrors/re/Recaf Recaf作为一款现代化的Java字节码编辑器&#xff0c;为开发者提供了强大的Java反编译和分析功能。然…...

【SITS2026官方认证指南】:AI音乐生成应用落地的5大技术门槛与3步合规部署法

第一章&#xff1a;SITS2026官方认证体系与AI音乐生成应用的战略定位 2026奇点智能技术大会(https://ml-summit.org) SITS2026&#xff08;Singularity Intelligence Technology Standard 2026&#xff09;是由国际人工智能标准联盟&#xff08;IAISA&#xff09;联合全球五大…...

移动端安全测试

移动端安全测试&#xff1a;守护指尖上的数字防线 在智能手机普及的今天&#xff0c;移动应用已成为生活与工作的核心工具。随着应用数量的爆炸式增长&#xff0c;安全漏洞、数据泄露和恶意攻击等问题也日益突出。移动端安全测试作为保障用户隐私与数据安全的关键环节&#xf…...

地质雷达电磁波仿真终极指南:gprMax开源软件完全解析

地质雷达电磁波仿真终极指南&#xff1a;gprMax开源软件完全解析 【免费下载链接】gprMax gprMax is open source software that simulates electromagnetic wave propagation using the Finite-Difference Time-Domain (FDTD) method for numerical modelling of Ground Penet…...

六要素自动气象站 自动气象站六要素

六要素自动气象站设备搭载低功耗采集器&#xff0c;静态功耗小于1mA&#xff0c;大幅降低电能消耗&#xff0c;搭配太阳能充电管理系统&#xff0c;可实现长期稳定运行&#xff0c;无需频繁更换电源或充电。即使在光照不足的阴雨天&#xff0c;也能凭借低功耗特性延长续航时间&…...