background

We’ve covered a lot about Flutter, but we haven’t covered how to implement Flutter to communicate with the native.

For example, if I click a button on the Flutter UI and I want the native to do something, how does the native know?

For example, if I have some changes that NEED to be notified to Flutter, how does Flutter know?

Let’s tackle the first problem first. That is, Flutter-> native communication.

Routing review

We’ve been talking about Flutter in the past, mostly in the main.dart file, in case many people think we’re overdoing it.

Therefore, we are here to add some information about the introduction of Flutter in the previous project of the third blog in the learn to Use Flutter series, 03.

Dart file of the Flutter Module can be used to write the following template code for multiple pages:

import 'dart:ui';
import 'package:flutter/material.dart';

void main() => runApp(_widgetForRoute(window.defaultRouteName));

Widget _widgetForRoute(String route) {
  switch (route) {
    case 'route1':
      returnSomeWidget(...) ;case 'route2':
      returnSomeOtherWidget(...) ; default:return Center(
        child: Text('Unknown route: $route', textDirection: TextDirection.ltr), ); }}Copy the code

In this code, we can focus on the switch code. Different pages will be returned depending on the route.

And we’re going to write it this way.

The actual case

Let’s take a practical example to illustrate how to implement Flutter to send messages to native.

For our example, let’s say I want to get the current battery on my Android device, and I want the battery to show up when I click the button.

Of course the button here and the text showing the power are from the Flutter interface.

So what are the steps?

1. Build the Flutter interface

We write the interface as a separate battery_widget.dart file:

import 'package:flutter/material.dart';

class BatteryWidget extends StatefulWidget {
  @override
  _BatteryWidgetState createState() => _BatteryWidgetState();
}

class _BatteryWidgetState extends State<BatteryWidget> {
  String _batteryLevel = 'Battery level: unknown.';

  void _getBatteryLevel() {}

  @override
  Widget build(BuildContext context) {
    return Center(
      child: Column(
        mainAxisAlignment: MainAxisAlignment.center,
        children: <Widget>[
          Text(_batteryLevel),
          RaisedButton(
            child: const Text('Refresh'), onPressed: _getBatteryLevel, ), ], ), ); }}Copy the code

Very simple interface, is a text and a button, lined up.

Then we modify main.dart as follows:

import 'dart:ui';

import 'package:flutter/material.dart';
import 'package:my_flutter/battery_widget.dart';

void main() => runApp(_widgetForRoute(window.defaultRouteName));

Widget _widgetForRoute(String route) {
  switch (route) {
    case 'battery':
      return MaterialApp(
        home: Scaffold(
          body: BatteryWidget(),
        ),
      );
    default:
      returnMaterialApp( home: Scaffold( body: Container(), ), ); }}Copy the code

The key point here is to return to our newly created battery_widget screen when you specify route name as battery.

2. Call the Flutter interface natively

In mainactivity.java, we write the following code:

package com.nesger.flutterdemo;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.view.ViewGroup;
import android.widget.FrameLayout;

import io.flutter.facade.Flutter;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        View flutterView = Flutter.createView(
                MainActivity.this,
                getLifecycle(),
                "battery"); FrameLayout.LayoutParams layout = new FrameLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT); addContentView(flutterView, layout); }}Copy the code

You can see that the battery specifies the Flutter interface to load.

After running, the effect is as follows:

Next comes the key to capturing native device power at the click of a button.

From the code above, we know that clicking the button executes the _getBatteryLevel method. So we’re going to make some changes here.

3. Flutter defines MethodChannel

We add the following variables to _BatteryWidgetState:

static const MethodChannel methodChannel = MethodChannel('samples.flutter.io/battery');
Copy the code

Samples. The flutter. IO/’ can you specify, guarantee the only commonly, so the samples actually used can be replaced with the package name. Basically be to want to correspond with native can.

4. The Flutter calls the methodChannel API invokeMethod to call a native method and get the corresponding value.
final int result = await methodChannel.invokeMethod('getBatteryLevel');
Copy the code

For example, we will use the native getBatteryLevel method to get the corresponding battery and store the return value as result.

We say await here because the operation is asynchronous. _getBatteryLevel is also changed to the corresponding asynchronous method, so the final method code is as follows:

Future<void> _getBatteryLevel() async {
    String batteryLevel;
    try {
      final int result = await methodChannel.invokeMethod('getBatteryLevel');
      batteryLevel = 'Battery level: $result%.';
    } on PlatformException {
      batteryLevel = 'Failed to get battery level.';
    }
    setState(() {
      _batteryLevel = batteryLevel;
    });
  }
Copy the code

You can see that the setState method is used to update the interface after obtaining the power using the asynchronous method.

5. Define MethodChannel natively
private static final String BATTERY_CHANNEL = "samples.flutter.io/battery";
Copy the code

Note that there needs to be a one-to-one correspondence with Flutter.

6. The native call creates a MethodChannel and receives a method call to the Flutter via the MethodCallHandler
new MethodChannel((FlutterView)flutterView, BATTERY_CHANNEL).setMethodCallHandler(
        new MethodChannel.MethodCallHandler() {
            @Override
            public void onMethodCall(MethodCall call, MethodChannel.Result result) {
                if (call.method.equals("getBatteryLevel")) {
                    int batteryLevel = getBatteryLevel();

                    if(batteryLevel ! = -1) { result.success(batteryLevel); }else {
                        result.error("UNAVAILABLE"."Battery level not available.", null); }}else{ result.notImplemented(); }}});Copy the code

We can see that we distinguish between different method calls of Flutter by call.method.

Here result.success returns a successful callback. Result. error returns an error callback. Result. notImplemented indicates that there is no corresponding implementation.

Finally, we implement the native getBatteryLevel method.

As follows:

private int getBatteryLevel() {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        BatteryManager batteryManager = (BatteryManager) getSystemService(BATTERY_SERVICE);
        return batteryManager.getIntProperty(BatteryManager.BATTERY_PROPERTY_CAPACITY);
    } else {
        Intent intent = new ContextWrapper(getApplicationContext()).
                registerReceiver(null, new IntentFilter(Intent.ACTION_BATTERY_CHANGED));
        return(intent.getIntExtra(BatteryManager.EXTRA_LEVEL, -1) * 100) / intent.getIntExtra(BatteryManager.EXTRA_SCALE, -1); }}Copy the code

Run the click button and the effect is as follows:

That concludes our discussion of the methods that Flutter calls the native and gets the return value.

Here we summarize as follows:

Preparation of Flutter:

  1. Define MethodChannel
  2. InvokeMethod, which calls methodChannel through an asynchronous method, specifies the name of the method to be called by this methodChannel

Primary preparation:

  1. Define a CHANNEL (corresponding to a Flutter)
  2. Create a MethodChannel and use the setMethodCallHandler method to distinguish between different Flutter calling method names and return corresponding callbacks

Source location: github.com/nesger/Flut…

Reference links:

Flutter. Dev/docs/develo… Github.com/flutter/flu…

Read more: Learn to Play with Flutter series blog – 01 Environment build with Flutter series blog – 02 A pure Flutter Demo explains the Flutter with Flutter series blog – 03 introduced in the old project StatelessWidget vs StatefulWidget Flutter Is ready to Use series blog — 06 RenderFlex Overlearning new Words & Expressions

How to elegantly avoid the air with a Flutter map… use