How is the Dart layer compatible with Android and iOS platform features

Why Android and ISO compatibility

  • Since Flutter only takes over the application rendering layer, these underlying system capabilities cannot be supported within the Flutter framework;
  • Flutter is still a relatively young ecosystem, so some of the Java, C++, or Objective-C libraries that are relatively mature in native development, such as image processing, audio and video codec, may not be implemented in Flutter.

Flutter provides us with a MethodChannel to do the native invocation of flutter.

The entire call process is similar to the network request:

  1. Flutter, as a client, initiates method invocation requests through method channels.
  2. Being native to the server, the request is received through the method channel, and the native API is called to handle the request
  3. Through the method channel, the processed file is sent back to the FLUTTER client
  4. Flutter takes over the result of processing the request, determines whether it was successful, and then performs any subsequent logical processing

Here is a small example of opening the wechat app:

Flutter server-side code

Class FlutterCallNativeDemo extends StatelessWidget {// declare MethodChannel var platform = MethodChannel("samples. Chenhang /utils"); OpenWeChat () async {int result; Try {/ / through method calls, call native methods result = (await platform. InvokeListMethod (" openWeChat ")) as an int; } catch (e) { print("result error=$e"); result = -1; } print("result=$result"); } @override Widget build(BuildContext context) {return Scaffold(appBar: appBar (title: Text(" native ")), body: Center(child: Column(children: [MaterialButton(onPressed: () {openWeChat();}, child: Text(" openWeChat "),),),); }}Copy the code

Android: Find the entry to the Flutter application, which is implemented in The FlutterView in MainActivity. Open the Android directory under your project, find the MainActivity file, and open the onCreate(savedInstanceState: Bundle?) Notice that this is in the onCreate method with only one parameter.

class MainActivity : FlutterActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) flutterEngine? .let { GeneratedPluginRegistrant.registerWith(it) } MethodChannel(flutterEngine? .dartExecutor? .binaryMessenger, "samples.chenhang/utils").setMethodCallHandler { call, result -> if (call.method == "openWeChat") { try { val weChatPackageName = "com.tencent.mm" if (isApplicationAvilible(weChatPackageName)) { val intent = Intent(Intent.ACTION_MAIN) intent.addCategory(Intent.CATEGORY_LAUNCHER) intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) val cn = ComponentName(weChatPackageName, "Com. Tencent. Mm. UI. LauncherUI") intent.com ponent = cn startActivity (intent)} else {Toast. MakeText (this, "you don't have to install WeChat, LENGTH_SHORT). Show ()} result.success(0)} catch (e: Exception) {result.error("UNAVAILABLE", "You have not installed wechat ", null) } } else { result.notImplemented() } } } private fun isApplicationAvilible(packageName: String): Boolean { val installedPackages = packageManager.getInstalledPackages(0) if (installedPackages ! = null) { for (packageInfo in installedPackages) { if (packageInfo.packageName == packageName) { return true } } } return false } }Copy the code

This flutter calls Android, which seems pretty simple.

Since there is no real iPhone, wechat is not installed on the emulator. Let’s take opening a browser as an example:

@objc class AppDelegate: FlutterAppDelegate { override func application( _ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? ) -> Bool { GeneratedPluginRegistrant.register(with: self) let controller : FlutterViewController = window? .rootViewController as! FlutterViewController; / / 1. The declarative approach channel let batteryChannel = FlutterMethodChannel. Init (name: "samples. Chenhang/utils binaryMessenger: controller.binaryMessenger); batteryChannel.setMethodCallHandler({ //2. Handle flutter requests (Call: FlutterMethodCall, result: FlutterResult) -> Void in if(call.method=="openWeChat"){self.openwechat () result(0)}else{ result(FlutterMethodNotImplemented) } // Handle battery messages. }); return super.application(application, didFinishLaunchingWithOptions: LaunchOptions)} / / call native open a browser func openWeChat () {let urlString = "https://flutterchina.club/platform-channels/" let url URL = (string: urlString) # if available (iOS 10.0. *) {UIApplication. Shared. Open (URL! , options: [:], completionHandler: nil) } else { // Fallback on earlier versions } } }Copy the code

Find the entry to find the Flutter application, i.e. in the AppDelegate, and then perform the Flutter request processing in the Application method.

How does Flutter use native views

Through the method channel, the underlying capabilities provided by the native operating system and some relatively mature solutions in the existing native development can be quickly solved in Dart layer in the form of interface encapsulation, thus solving the reuse problem of native code on Flutter. We can then take advantage of the rich controls that Flutter itself provides for UI rendering.

Is everything done in APP development just by going through the method channel? Obviously not, so let’s take a look at what it takes to build a complex APP.

With Flutter and method channels, we can solve application-layer capabilities, application-layer rendering, and low-level capabilities. However, for scenes that involve low-level rendering, such as browsers, cameras, maps, and native custom views, it is not practical to develop one on Flutter yourself.

Flutter provides the concept of a Platform View. It provides a way for developers to embed native systems (Android and iOS) views into Flutter and add them to the render tree of Flutter to achieve an interactive experience consistent with Flutter.

The process for creating a platform view:

  • First, the client of Flutter initiates the creation request of the native view by passing a view identifier to the native view wrapper class of Flutter (UIKitView and AndroidView on iOS and Android platforms respectively).
  • The native code side then hands off the creation of the corresponding native view to the PlatformViewFactory.
  • Finally, the view identifier is associated with the platform view factory on the native code side, so that view creation requests initiated by Flutter can directly find the corresponding view creation factory.

Here is an example using Android TextView and iOS UIText:

The flutter end

// Load native view, Android use: AndroidView; Ios uses UiKitView Class NativeWidget extends StatelessWidget {@override Widget Build (BuildContext Context) {if (defaultTargetPlatform == TargetPlatform.android) { return AndroidView(viewType: "com.example.flutter_base_app/textView"); } else { return UiKitView( viewType: "com.example.flutter_base_app/textView", ); } } } class NativeWidgetDemo extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( AppBar: appBar (title: Text(" display native control "),), Body: Center(Child: NativeWidget(),); }}Copy the code

It’s as simple as creating different widgets for different platforms. Android is AndroidView; Ios UiKitView.

Android:

  1. Build the returned view inherits PlatformView
class NativeTextView(val context: Context? , val message: BinaryMessenger) : PlatformView, MethodChannel MethodCallHandler {/ / incoming BinaryMessenger just to create MethodChannel use private var textView: TextView? = null private val updateText = "update_text" init { textView = TextView(context) textView? .text = "I'm a textView from Android" val methodChannel = methodChannel (message, "") methodChannel.setMethodCallHandler(this) } override fun getView(): View { return textView!! } override fun dispose() { } override fun onMethodCall(call: MethodCall, result: MethodChannel.Result) { when (call.method) { updateText -> { } } } }Copy the code
  1. Create the view factory implementation
class NativeViewFactory(private val message: BinaryMessenger) : PlatformViewFactory(StandardMessageCodec.INSTANCE) { override fun create(context: Context? , viewId: Int, args: Any?) : PlatformView { return NativeTextView(context = context, message = message) } }Copy the code
  1. Custom Plugin class to register a view factory
object NativeViewPlugin { private val viewTypeId = "com.example.flutter_base_app/textView" fun registerWith(flutterEngine: FlutterEngine?) { flutterEngine? .let { val key = NativeViewPlugin::class.java.canonicalName val shimPluginRegistry = ShimPluginRegistry(flutterEngine); if (shimPluginRegistry.hasPlugin(key)) return val registry = shimPluginRegistry.registrarFor(key) registry.platformViewRegistry().registerViewFactory(viewTypeId, NativeViewFactory(registry.messenger())) } } }Copy the code

iOS

  1. In the info. Plist addedio.flutter.embedded_views_preview= true

2. Create the returned view class

import Foundation import Foundation import Flutter class PlatformTextView: NSObject,FlutterPlatformView { let frame: CGRect; let viewId: Int64; var text:String = "" init(_ frame: CGRect,viewID: Int64,args :Any?) { self.frame = frame self.viewId = viewID // if(args is NSDictionary){ // let dict = args as! NSDictionary // self.text = dict.value(forKey: "text") as! String //}} func view() -> UIView {let label = UILabel() label.text = "I am text from iOS" label.textColor = uicolor.red label.frame = self.frame return label } }Copy the code
  1. Create the view factory class
import Foundation import Flutter class PlatformTextViewFactory: NSObject,FlutterPlatformViewFactory { func create(withFrame frame: CGRect, viewIdentifier viewId: Int64, arguments args: Any?) -> FlutterPlatformView {return PlatformTextView(frame,viewID: viewID,args: args) } func createArgsCodec() -> FlutterMessageCodec & NSObjectProtocol { return FlutterStandardMessageCodec.sharedInstance() } }Copy the code
  1. Register the factory view in the Application of the AppDelegate
// From platform view, Return the native control to the flutter layer func testPlatformView() {let Factory = PlatformTextViewFactory() let registrar = Self. registrar(forPlugin: "platform_text_view_plugin") .register(factory, withId: "com.example.flutter_base_app/textView")Copy the code

conclusion

The dark level calls to native methods are done using the MethodChannel. In the platform view, get the native view and add it to the Widget tree to complete the rendering. Through these two ways, the dark layer can complete the call to the native, so that the APP has the underlying ability and the underlying rendering function, and complete the construction of the complex APP.

Refer to the link

Blog.csdn.net/sinat_17775… Blog.csdn.net/weixin_3615… www.jianshu.com/p/8d74a7318…