上次從一個路徑插件看來一下Flutter中如何調用iOS和Android中的方法以及平臺如何返回值給Flutter框架。今天就來詳細講講MethodChannel是如何連同另外一個世界的。c#
想要達成的效果是這樣使用能夠彈出一個時間較長的吐司windows
這個示例要講述的是Flutter中如何向平臺傳遞參數api
var show = RaisedButton(
onPressed: () {
IaToast.show(msg: "hello",type: Toast.LENGTH_LONG);
},
child: Text("點擊彈吐司"),
);
複製代碼
定義一個
IaToast
的吐司類,根據枚舉類型使用MethodChannel調用原生方法微信
import 'package:flutter/services.dart';
///吐司類型 [LENGTH_SHORT]短期,[LENGTH_LONG]長時間
enum Toast {
LENGTH_SHORT,
LENGTH_LONG
}
///吐司類
class IaToast {
static const MethodChannel _channel =//方法渠道名
const MethodChannel('www.toly1994.com.flutter_journey.toast');
static show(//靜態方法顯示吐司
{String msg, Toast type = Toast.LENGTH_SHORT}) {
if (type == Toast.LENGTH_SHORT) {
_channel.invokeMethod('showToast', {//渠道對象調用方法
"msg": msg,
"type": 0,
});
} else {
_channel.invokeMethod('showToast', {
"msg": msg,
"type": 1,
});
}
}
}
複製代碼
經過FlutterView和渠道名能夠獲取MethodChannel對象,對其進行方法調用監聽markdown
其中的兩個回調參數分別儲存着方法信息和返回信息。app
public class MainActivity extends FlutterActivity {
private static final String CHANNEL = "www.toly1994.com.flutter_journey.toast";//渠道名
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
GeneratedPluginRegistrant.registerWith(this);
MethodChannel channel = new MethodChannel(getFlutterView(), CHANNEL);//獲取渠道
channel.setMethodCallHandler(this::handleMethod);//設置方法監聽
}
/**
* 處理方法回調監聽
* @param methodCall 方法的參數相關
* @param result 方法的返回值相關
*/
private void handleMethod(MethodCall methodCall, MethodChannel.Result result) {
switch (methodCall.method){//根據方法名進行處理
case "showToast":
handleToast(this,methodCall);//具體處理
break;
default:
result.notImplemented();
}
}
public static void handleToast(Context context,MethodCall methodCall) {
String msg=methodCall.argument("msg");
int type=methodCall.argument("type");
Toast.makeText(context, msg, type).show();
}
}
複製代碼
這樣對應Android端,在Flutter中就能夠開心的彈吐司了框架
var show = RaisedButton(
onPressed: () {
IaToast.show(msg: "hello Flutter", type: Toast.LENGTH_LONG);//使用吐司
},
child: Text("點擊彈吐司"),
);
var app = MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: Scaffold(
appBar: AppBar(
title: Text('Flutter之旅'),
),
body: show,
),
);
void main() => runApp(app);
複製代碼
也簡單的畫了一幅Flutter和iOS溝通的圖異步
如今來看iOS端如何接受Flutter中的參數,和Android中基本一致,首先要得到渠道async
在iOS裏FlutterMethodChannel經過渠道標識和FlutterViewController來獲取。
有了渠道方法以後,剩下的就幾乎一致了,只是語法問題。
經過FlutterMethodCall回調中的call中的arguments值來獲取參數,強轉成NSDictionary
不過iOS系統並無直接彈吐司的方法,因此須要自定義吐司。ide
import UIKit
import Flutter
@UIApplicationMain
@objc class AppDelegate: FlutterAppDelegate {
public static let channelId="www.toly1994.com.flutter_journey.toast"
override func application(
_ application: UIApplication,
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
) -> Bool {
let controller:FlutterViewController = window.rootViewController as! FlutterViewController
let messageChannel = FlutterMethodChannel.init(//獲取方法渠道
name: AppDelegate.channelId,
binaryMessenger:controller)
messageChannel.setMethodCallHandler{(call, result) in
self.handle(call,result)
}
return super.application(application, didFinishLaunchingWithOptions: launchOptions)
}
public func handle(_ call: FlutterMethodCall,_ result: @escaping FlutterResult) {
let args: NSDictionary = call.arguments as! NSDictionary
switch call.method {
case "showToast":
let msg:String = args["msg"] as! String
let type:Int = args["type"] as! Int
handleToast(msg:msg,type:type)
default:
result(FlutterMethodNotImplemented)
}
}
public func handleToast(msg: String, type: Int) {
Toast.toast(text: msg,type:type)
}
}
複製代碼
使用UILabel和UIButton進行模擬一個吐司框
import UIKit
let toastDispalyDuration: CGFloat = 2.0
let toastBackgroundColor = UIColor(red: 0, green: 0, blue: 0, alpha: 0.6)
class Toast: NSObject {
var duration: CGFloat = toastDispalyDuration
var contentView: UIButton//內容框
init(text: String) {
let rect = text.boundingRect(
with: CGSize(width: 250, height: CGFloat.greatestFiniteMagnitude),
attributes: [NSAttributedString.Key.font: UIFont.systemFont(ofSize: 20)],//該值可調節邊距
context: nil)
let textLabel = UILabel(//標籤
frame: CGRect(x: 0, y: 0, width: rect.size.width + 40, height: rect.size.height + 20))
textLabel.backgroundColor = UIColor.clear
textLabel.textColor = UIColor.white
textLabel.textAlignment = .center
textLabel.font = UIFont.systemFont(ofSize: 16)
textLabel.text = text
textLabel.numberOfLines = 0
contentView = UIButton(type: .roundedRect)
contentView.frame=CGRect(x: 0, y: 0,
width: textLabel.frame.size.width,
height: textLabel.frame.size.height)
contentView.layer.cornerRadius = 15
contentView.backgroundColor = toastBackgroundColor
contentView.addSubview(textLabel)
contentView.autoresizingMask = UIView.AutoresizingMask.flexibleWidth
super.init()
contentView.addTarget(self, action: #selector(toastTaped), for: .touchDown)
NotificationCenter.default.addObserver(self, selector: #selector(toastTaped), name: UIDevice.orientationDidChangeNotification, object: UIDevice.current)
}
@objc func toastTaped() {
self.hideAnimation()
}
func deviceOrientationDidChanged(notify: Notification) {
self.hideAnimation()
}
@objc func dismissToast() {
contentView.removeFromSuperview()
}
func setDuration(duration: CGFloat) {
self.duration = duration
}
func showAnimation() {
UIView.beginAnimations("show", context: nil)
UIView.setAnimationCurve(UIView.AnimationCurve.easeIn)
UIView.setAnimationDuration(0.3)
contentView.alpha = 1.0
UIView.commitAnimations()
}
@objc func hideAnimation() {
UIView.beginAnimations("hide", context: nil)
UIView.setAnimationCurve(UIView.AnimationCurve.easeOut)
UIView.setAnimationDelegate(self)
UIView.setAnimationDidStop(#selector(dismissToast))
UIView.setAnimationDuration(0.3)
contentView.alpha = 0.0
UIView.commitAnimations()
}
func showFromBottomOffset(bottom: CGFloat) {
let window: UIWindow = UIApplication.shared.windows.last!
contentView.center = CGPoint(x: window.center.x, y: window.frame.size.height - (bottom + contentView.frame.size.height/2))
window.addSubview(contentView)
self.showAnimation()
self.perform(#selector(hideAnimation), with: nil, afterDelay: TimeInterval(duration))
}
class func toast(text: String,type: Int) {
let toast = Toast(text: text)
var duration=0
if type==0 {duration=1}else{duration=3}
toast.setDuration(duration: CGFloat(duration))
toast.showFromBottomOffset(bottom: 60)
}
}
複製代碼
如今應該對MethodChannel有了一個感性的認知了,它能夠連通Flutter框架和平臺。
在Flutter中MethodChannel是一個Dart類,
處於flutter/lib/src/services/platform_channel.dart
文件中
其中有三個成員變量,咱們在使用時只是傳來一個字符串而已,其實還有兩個是默認的
codec是消息的編解碼器,類型MethodCodec,默認是StandardMethodCodec
binaryMessenger是二進制信使,類型BinaryMessenger,默認是defaultBinaryMessenger
class MethodChannel {
const MethodChannel(this.name, [this.codec = const StandardMethodCodec(), this.binaryMessenger = defaultBinaryMessenger ])
: assert(name != null),
assert(binaryMessenger != null),
assert(codec != null);
final String name;
final MethodCodec codec;
final BinaryMessenger binaryMessenger;
複製代碼
首先它是一個異步方法,傳遞方法名和參數,能夠看出首先由codec編碼MethodCall對象
而後經過binaryMessenger去發送信息,獲取的結構是一個字節數據,
若是結果非空,經過codec去解碼,而後進行返回,可見這個泛型即是指望的結果類型
Future<T> invokeMethod<T>(String method, [ dynamic arguments ]) async {
assert(method != null);
final ByteData result = await binaryMessenger.send(
name,
codec.encodeMethodCall(MethodCall(method, arguments)),
);
if (result == null) {
throw MissingPluginException('No implementation found for method $method on channel $name');
}
final T typedResult = codec.decodeEnvelope(result);
return typedResult;
}
複製代碼
MethodCodec是一個抽象接口,定義了編解碼的方法,因此具體邏輯還要看它的實現類
MethodCodec有兩個實現類StandardMethodCodec和JSONMethodCodec
abstract class MethodCodec {
ByteData encodeMethodCall(MethodCall methodCall);
MethodCall decodeMethodCall(ByteData methodCall);
dynamic decodeEnvelope(ByteData envelope);
ByteData encodeSuccessEnvelope(dynamic result);
ByteData encodeErrorEnvelope({ @required String code, String message, dynamic details });
}
複製代碼
StandardMethodCodec的編碼方法
能夠看出StandardMethodCodec對MethodCall的編碼是經過messageCodec實現的
messageCodec是StandardMessageCodec對象,其中的writeValue是編碼的核心方法
將方法名和參數根據類型放入buffer中,從而將這些方法信息存儲其中。
class StandardMethodCodec implements MethodCodec {
const StandardMethodCodec([this.messageCodec = const StandardMessageCodec()]);
@override
ByteData encodeMethodCall(MethodCall call) {
final WriteBuffer buffer = WriteBuffer();
messageCodec.writeValue(buffer, call.method);
messageCodec.writeValue(buffer, call.arguments);
return buffer.done();
}
//略...
}
---->[StandardMessageCodec#writeValue]----
void writeValue(WriteBuffer buffer, dynamic value) {
if (value == null) {
buffer.putUint8(_valueNull);
} else if (value is bool) {
buffer.putUint8(value ? _valueTrue : _valueFalse);
} else if (value is int) {
if (-0x7fffffff - 1 <= value && value <= 0x7fffffff) {
buffer.putUint8(_valueInt32);
buffer.putInt32(value);
} else {
buffer.putUint8(_valueInt64);
buffer.putInt64(value);
}
//略...
} else if (value is List) {
buffer.putUint8(_valueList);
writeSize(buffer, value.length);
for (final dynamic item in value) {
writeValue(buffer, item);
}
} else if (value is Map) {
buffer.putUint8(_valueMap);
writeSize(buffer, value.length);
value.forEach((dynamic key, dynamic value) {
writeValue(buffer, key);
writeValue(buffer, value);
});
} else {
throw ArgumentError.value(value);
}
}
複製代碼
BinaryMessenger是一個抽象接口,默認使用的實現了是
defaultBinaryMessenger
_sendPlatformMessage方法進行對平臺發送信息
const BinaryMessenger defaultBinaryMessenger = _DefaultBinaryMessenger._();
---->[BinaryMessenger]----
abstract class BinaryMessenger {
const BinaryMessenger();
Future<void> handlePlatformMessage(String channel, ByteData data, ui.PlatformMessageResponseCallback callback);
Future<ByteData> send(String channel, ByteData message);
void setMessageHandler(String channel, Future<ByteData> handler(ByteData message));
void setMockMessageHandler(String channel, Future<ByteData> handler(ByteData message));
}
---->[_DefaultBinaryMessenger]----
@override
Future<ByteData> send(String channel, ByteData message) {
final MessageHandler handler = _mockHandlers[channel];
if (handler != null)
return handler(message);
return _sendPlatformMessage(channel, message);
}
複製代碼
_sendPlatformMessage
這裏使用Window對象進行信息發送,最終調用的是
Window_sendPlatformMessage
的native方法
final Window window = Window._();
Future<ByteData> _sendPlatformMessage(String channel, ByteData message) {
final Completer<ByteData> completer = Completer<ByteData>();
ui.window.sendPlatformMessage(channel, message, (ByteData reply) {
try {
completer.complete(reply);
} catch (exception, stack) {
FlutterError.reportError(FlutterErrorDetails(
exception: exception,
stack: stack,
library: 'services library',
context: ErrorDescription('during a platform message response callback'),
));
}
});
return completer.future;
}
---->[Window#sendPlatformMessage]----
void sendPlatformMessage(String name,
ByteData data,
PlatformMessageResponseCallback callback) {
final String error =
_sendPlatformMessage(name, _zonedPlatformMessageResponseCallback(callback), data);
if (error != null)
throw Exception(error);
}
String _sendPlatformMessage(String name,
PlatformMessageResponseCallback callback,
ByteData data) native 'Window_sendPlatformMessage';
複製代碼
在Android中MethodChannel是一個Java類,處於
io.flutter.plugin.common
包
主要的成員變量也是三位messenger,name和codec,在構造方法中須要傳入BinaryMessenger
默認的MethodCodec是StandardMethodCodec.INSTANCE
public final class MethodChannel {
private static final String TAG = "MethodChannel#";
private final BinaryMessenger messenger;
private final String name;
private final MethodCodec codec;
public MethodChannel(BinaryMessenger messenger, String name) {
this(messenger, name, StandardMethodCodec.INSTANCE);
}
複製代碼
監聽器是設置在了messenger的身上,若是監聽器非空會使用
IncomingMethodCallHandler
messenger須要的監聽器的類型是BinaryMessenger.BinaryMessageHandler
,因此關係以下
public void setMethodCallHandler(@Nullable MethodChannel.MethodCallHandler handler) {
this.messenger.setMessageHandler(this.name,
handler == null ?
null : new MethodChannel.IncomingMethodCallHandler(handler));
}
---->[BinaryMessenger]----
public interface BinaryMessenger {
@UiThread
void setMessageHandler(@NonNull String var1,
@Nullable BinaryMessenger.BinaryMessageHandler var2);
複製代碼
IncomingMethodCallHandler實現了BinaryMessageHandler接口,必然實現其接口方法
onMessage中須要回調了ByteBuffer的方法字節信息以及BinaryReply對象
回調中的MethodCall對象是經過codec將字節信息解碼生成的
MethodChannel.Result是一個接口,有三個接口方法,這裏直接new對象並實現三個方法
經過codec編碼success傳入的對象,後經過reply對象的reply將返回值傳給Flutter端
private final class IncomingMethodCallHandler implements BinaryMessageHandler {
private final MethodChannel.MethodCallHandler handler;
IncomingMethodCallHandler(MethodChannel.MethodCallHandler handler) {
this.handler = handler;
}
@UiThread
public void onMessage(ByteBuffer message, final BinaryReply reply) {
MethodCall call = MethodChannel.this.codec.decodeMethodCall(message);
try {
this.handler.onMethodCall(call, new MethodChannel.Result() {
public void success(Object result) {
reply.reply(MethodChannel.this.codec.encodeSuccessEnvelope(result));
}
public void error(String errorCode, String errorMessage, Object errorDetails) {
reply.reply(MethodChannel.this.codec.encodeErrorEnvelope(errorCode, errorMessage, errorDetails));
}
public void notImplemented() {
reply.reply((ByteBuffer)null);
}
});
} catch (RuntimeException var5) {
Log.e("MethodChannel#" + MethodChannel.this.name, "Failed to handle method call", var5);
reply.reply(MethodChannel.this.codec.encodeErrorEnvelope("error", var5.getMessage(), (Object)null));
}
}
複製代碼
到這裏一切矛頭指向BinaryMessenger,它是一個接口,定義了發生信息的三個方法。
和信息發送相關的類有四個:
public interface BinaryMessenger {
@UiThread
void send(@NonNull String var1, @Nullable ByteBuffer var2);
@UiThread
void send(@NonNull String var1, @Nullable ByteBuffer var2, @Nullable BinaryMessenger.BinaryReply var3);
@UiThread
void setMessageHandler(@NonNull String var1, @Nullable BinaryMessenger.BinaryMessageHandler var2);
public interface BinaryReply {
@UiThread
void reply(@Nullable ByteBuffer var1);
}
public interface BinaryMessageHandler {
@UiThread
void onMessage(@Nullable ByteBuffer var1, @NonNull BinaryMessenger.BinaryReply var2);
}
}
複製代碼
咱們在建立MethodChannel的時候傳入的是getFlutterView()
追蹤一下能夠看到返回的是一個FlutterView,這也就說明FlutterView實現了BinaryMessenger
因此能夠從實現的方法入手,最終發現是調用mNativeView的方法,其爲FlutterNativeView類型
MethodChannel channel = new MethodChannel(getFlutterView(), CHANNEL);//獲取渠道
---->[FlutterActivity]----
public FlutterView getFlutterView() {
return this.viewProvider.getFlutterView();
}
---->[FlutterView]----
public interface Provider {
FlutterView getFlutterView();
}
---->[FlutterView]----
@UiThread
public void send(String channel, ByteBuffer message) {
this.send(channel, message, (BinaryReply)null);
}
@UiThread
public void send(String channel, ByteBuffer message, BinaryReply callback) {
if (!this.isAttached()) {
Log.d("FlutterView", "FlutterView.send called on a detached view, channel=" + channel);
} else {
this.mNativeView.send(channel, message, callback);
}
}
複製代碼
FlutterNativeView調用dartExecutor的方法,其爲DartExecutor類型
在構造方法中建立了FlutterJNI對象來建立DartExecutor,
DartExecutor中經過DartMessenger對象messenger發送,這些DartMessenger跑不掉了
public class FlutterNativeView implements BinaryMessenger {
private final DartExecutor dartExecutor;
private final FlutterJNI mFlutterJNI;
public FlutterNativeView(@NonNull Context context) {
this(context, false);
}
public FlutterNativeView(@NonNull Context context, boolean isBackgroundView) {
this.mContext = context;
this.mPluginRegistry = new FlutterPluginRegistry(this, context);
this.mFlutterJNI = new FlutterJNI();
this.mFlutterJNI.setRenderSurface(new FlutterNativeView.RenderSurfaceImpl());
this.dartExecutor = new DartExecutor(this.mFlutterJNI);
---->[FlutterNativeView]----
@UiThread
public void send(String channel, ByteBuffer message) {
this.dartExecutor.send(channel, message);
}
@UiThread
public void send(String channel, ByteBuffer message, BinaryReply callback) {
if (!this.isAttached()) {
Log.d("FlutterNativeView", "FlutterView.send called on a detached view, channel=" + channel);
} else {
this.dartExecutor.send(channel, message, callback);
}
}
---->[DartExecutor]----
@UiThread
public void send(@NonNull String channel, @Nullable ByteBuffer message) {
this.messenger.send(channel, message, (BinaryReply)null);
}
@UiThread
public void send(@NonNull String channel, @Nullable ByteBuffer message, @Nullable BinaryReply callback) {
this.messenger.send(channel, message, callback);
}
複製代碼
DartMessenger經過flutterJNI.dispatchPlatformMessage發送信息
最終到nativeDispatchPlatformMessage一個native方法,
而後那些C++裏見不得人的勾當這裏就不說了,有機會再細細道來。
@UiThread
public void send(@NonNull String channel, @NonNull ByteBuffer message) {
Log.v("DartMessenger", "Sending message over channel '" + channel + "'");
this.send(channel, message, (BinaryReply)null);
}
public void send(@NonNull String channel, @Nullable ByteBuffer message, @Nullable BinaryReply callback) {
Log.v("DartMessenger", "Sending message with callback over channel '" + channel + "'");
int replyId = 0;
if (callback != null) {
replyId = this.nextReplyId++;
this.pendingReplies.put(replyId, callback);
}
if (message == null) {
this.flutterJNI.dispatchEmptyPlatformMessage(channel, replyId);
} else {
this.flutterJNI.dispatchPlatformMessage(channel, message, message.position(), replyId);
}
}
@UiThread
public void dispatchPlatformMessage(@NonNull String channel, @Nullable ByteBuffer message, int position, int responseId) {
this.ensureRunningOnMainThread();
if (this.isAttached()) {
this.nativeDispatchPlatformMessage(this.nativePlatformViewId, channel, message, position, responseId);
} else {
Log.w("FlutterJNI", "Tried to send a platform message to Flutter, but FlutterJNI was detached from native C++. Could not send. Channel: " + channel + ". Response ID: " + responseId);
}
}
複製代碼
源碼貼的有點多,整個關係看起來也不是很是複雜。雖然沒啥大用,邏輯捋一捋對Flutter的總體認知也有所提高。
本文到此接近尾聲了,若是想快速嚐鮮Flutter,《Flutter七日》會是你的必備佳品;若是想細細探究它,那就跟隨個人腳步,完成一次Flutter之旅。
另外本人有一個Flutter微信交流羣,歡迎小夥伴加入,共同探討Flutter的問題,本人微信號:zdl1994328
,期待與你的交流與切磋。