Flutter中的信鸽 Pigeon使用

Pigeon是Flutter官方为了解决Flutter和Native互相调用过于繁琐,所发布的一款插件。使用Pigeon开发插件,过程会变得非常简单。

本文档编写时间为2022年4月24日,使用Flutter版本为 stable 2.10.4 Pigeon版本为3.0.3

新建plugin并引入pigeon

flutter create --template=plugin --platforms android,ios flutter_pigeon_plugin

pubspec.yaml中加入以下代码

dev_dependencies:
  flutter_test:
    sdk: flutter
  flutter_lints: ^1.0.0
  
  # 将下方代码加入对应位置
  pigeon: 

执行flutter pub get

编写pigeon文件

项目内新建文件夹pigeons,并在其中新建message.dart(此为示例,实际开发中文件名及文件夹随意)

编写message.dart

// pigeons/message.dart

import 'package:pigeon/pigeon.dart';

class F2NMessage{
  String? msg;
}

class N2FMessage{
  String? msg2;
}

@HostApi()
abstract class FlutterMessage{
  void flutterSendMessage(F2NMessage msg);
}

@FlutterApi()
abstract class NativeMessage{
  void nativeSendMessage(N2FMessage msg);
}

在demo中,我新建了一个flutter发往native的方法flutterSendMessage,参数为F2NMessage。新建了一个native发往flutter的方法nativeSendMessage,参数为N2FMessage

新建一个sh脚本(作者为Mac系统,windows系统请自行编写)run_pigeon.sh

# run_pigeon.sh


mkdir -p android/src/main/java/com/example/flutter_pigeon_plugin

flutter pub run pigeon \
  --input pigeons/message.dart \
  --dart_out lib/message.dart \
  --objc_header_out ios/Classes/Message.h \
  --objc_source_out ios/Classes/Message.m \
  --java_out android/src/main/java/com/example/flutter_pigeon_plugin/Message.java \
  --java_package "com.example.flutter_pigeon_plugin" 

由于我新建的是kt/swift插件,所以需要新建android/src/main/java/com/example/flutter_pigeon_plugin文件夹用于存放自动生成的Message.java文件,pigeon目前仅支持java和objc类型,不过由于kt/swift与java/objc均可互相调用,所以没有任何影响。

编写插件

上一步sh执行完成后,插件自动在lib下生成了对应的message.dart文件,我们这时候可以在lib/flutter_pigeon_plugin.dart中开始编写插件。

// lib/flutter_pigeon_plugin.dart


import 'package:flutter_pigeon_plugin/message.dart';

class FlutterPigeonPlugin extends NativeMessage {
  static final FlutterMessage _flutterMessage = FlutterMessage();

  factory FlutterPigeonPlugin() => _getInstance();

  static FlutterPigeonPlugin get instance => _getInstance();

  static FlutterPigeonPlugin? _instance = null;

  static FlutterPigeonPlugin _getInstance() {
    _instance ??= FlutterPigeonPlugin._internal();
    NativeMessage.setup(_instance);
    return _instance!;
  }

  FlutterPigeonPlugin._internal();

  void sendMessage() {
    F2NMessage message = F2NMessage();
    message.msg = "这是一条从Flutter发往Native的消息";
    _flutterMessage.flutterSendMessage(message);
  }

  @override
  void nativeSendMessage(N2FMessage msg) {
    print("收到了Native发来的消息:${msg.msg2}");
  }
}

从以上代码中,我们可以了解到如何向Native发送消息,及如何接收Native发来的消息。

编写iOS端代码

由于插件使用swift开发,故仅提供swift类型插件的相关说明,Objective-C可以参阅swift代码自行修改

run_pigeon.sh文件执行完成后,ios/Classes文件夹中已经自行生成大量代码。首先我们需要新建flutter_pigeon_plugin.h文件,方便让swift文件自动引入我们新建的代码

// ios/Classes/flutter_pigeon_plugin.h

#import "Message.h"

编写SwiftFlutterPigeonPlugin.swift文件

// ios/Classes/SwiftFlutterPigeonPlugin.swift


import Flutter
import UIKit

public class SwiftFlutterPigeonPlugin: NSObject, FlutterPlugin,FlutterMessage {
    public func flutterSendMsg(_ msg: F2NMessage, error: AutoreleasingUnsafeMutablePointer<FlutterError?>) {
        print("收到来自于Flutter的消息"+msg.msg!)
        let message:N2FMessage = N2FMessage.make(withMsg2: "iOS返回的消息")
        nativeMessage.nativeSendMsg(message, completion: {e in})
    }
    public var nativeMessage:NativeMessage!
    public static func register(with registrar: FlutterPluginRegistrar) {
        let messenger: FlutterBinaryMessenger = registrar.messenger()
        let api = SwiftFlutterPigeonPlugin.init()
        FlutterMessageSetup(messenger, api)
        api.nativeMessage = NativeMessage(binaryMessenger: messenger)
    }
}

编写example中的main.dart看一下效果

import 'package:flutter/material.dart';

import 'package:flutter_pigeon_plugin/flutter_pigeon_plugin.dart';

void main() {
  runApp(const MyApp());
}

class MyApp extends StatefulWidget {
  const MyApp({Key? key}) : super(key: key);

  @override
  State<MyApp> createState() => _MyAppState();
}

class _MyAppState extends State<MyApp> {
  late FlutterPigeonPlugin _flutterPigeonPlugin;


  @override
  void initState() {
    super.initState();
    _flutterPigeonPlugin = FlutterPigeonPlugin();
  }

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      home: Scaffold(
        appBar: AppBar(
          title: const Text('Plugin example app'),
        ),
        body: Center(
          child: TextButton(
              child: Text("点击这里向native发送消息"),
              onPressed: () {
                _flutterPigeonPlugin.sendMessage();
              }),
        ),
      ),
    );
  }
}

点击屏幕中的按钮,控制台打印返回消息

截屏2022-04-24 10.17.42.png

编写Android端代码

由于插件使用kt开发,故仅提供kt类型插件的相关说明,java可以参阅kt代码自行修改

run_pigeon.sh文件执行完成后,src/main/java/com/example/flutter_pigeon_plugin文件夹中已经自行生成大量代码。

直接开始编辑FlutterPigeonPlugin.kt

// src/main/kotlin/com/example/flutter_pigeon_plugin/FlutterPigeonPlugin.kt

package com.example.flutter_pigeon_plugin

import android.util.Log
import androidx.annotation.NonNull

import io.flutter.embedding.engine.plugins.FlutterPlugin

/** FlutterPigeonPlugin */
class FlutterPigeonPlugin : FlutterPlugin, Message.FlutterMessage {
    lateinit var nativeMessage: Message.NativeMessage

    override fun onAttachedToEngine(@NonNull flutterPluginBinding: FlutterPlugin.FlutterPluginBinding) {
        Message.FlutterMessage.setup(flutterPluginBinding.binaryMessenger, this)
        nativeMessage = Message.NativeMessage(flutterPluginBinding.binaryMessenger)
    }

    override fun onDetachedFromEngine(binding: FlutterPlugin.FlutterPluginBinding) {

    }

    override fun flutterSendMessage(msg: Message.F2NMessage) {
        Log.d("FlutterPigeonPlugin", msg.msg.toString())
        var respMsg: Message.N2FMessage = Message.N2FMessage()
        respMsg.msg2 = "Android的返回值"
        nativeMessage.nativeSendMessage(respMsg, {})

    }
}

点击屏幕中的按钮,控制台打印消息

截屏2022-04-24 10.18.29.png

GitHub源码
Gitee源码

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容