【Flutter】创建Widget显示PlatformView

What is PlatformView


Flutter提供PlatformView,可以用来创建显示Native的控件的Widget,例如Android端的WebView、MapView等。
需注意显示PlatformView的Widget默认时无边界的,这要求其parent必须提供边界约束,在没有边界约束的容器中使用会出现如下情况
在这里插入图片描述
接下来通过例子演示如何基于PlatformView实现一个Widget:一个可以显示Android端TextView的Widget。
【Flutter】创建Widget显示PlatformView_第1张图片

Create Project


启动一个新项目,选择Flutter Plugin
【Flutter】创建Widget显示PlatformView_第2张图片【Flutter】创建Widget显示PlatformView_第3张图片

Create PlatformView Widget


创建显示PlatformView的Widget:./lib/text_view.dart

import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';

typedef void TextViewCreatedCallback(TextViewController controller);

class TextView extends StatefulWidget {
  const TextView({
    Key key,
    this.onTextViewCreated,
  }) : super(key: key);

  final TextViewCreatedCallback onTextViewCreated;

  @override
  State<StatefulWidget> createState() => _TextViewState();
}

class _TextViewState extends State<TextView> {
  @override
  Widget build(BuildContext context) {
    if (defaultTargetPlatform == TargetPlatform.android) {
      return AndroidView(
        viewType: 'plugins.my.example/textview',
        onPlatformViewCreated: _onPlatformViewCreated,
      );
    }
    return Text(
        '$defaultTargetPlatform is not yet supported by the text_view plugin');
  }

  void _onPlatformViewCreated(int id) {
    if (widget.onTextViewCreated == null) {
      return;
    }
    widget.onTextViewCreated(new TextViewController._(id));
  }
}

class TextViewController {
  TextViewController._(int id)
      : _channel = new MethodChannel('plugins.felix.angelov/textview_$id');

  final MethodChannel _channel;

  Future<void> setText(String text) async {
    assert(text != null);
    return _channel.invokeMethod('setText', text);
  }
}
  • build中返回了AndroidView的子Widget,并制定了ViewType。AndroidView就是在Android端对应的PlatformView
  • 当PlatformView创建完成时,会收到onPlatformViewCreated回调,在此时可以做一些初始化操作,我们在这里创建了一个TextViewController并传递了出去,TextViewController通过MethodChannel与Native的TextView建立通信。
  • AndroidView只是负责Native的控件显示,Native的通信还是要依靠MethodChannel

Create Native Plugin


Android端实现插件类./android/src/main/java/{organization_name}/TextViewPlugin.java

import io.flutter.plugin.common.PluginRegistry.Registrar;

public class TextViewPlugin {
  public static void registerWith(Registrar registrar) {
    registrar
            .platformViewRegistry()
            .registerViewFactory(
                    "plugins.my.example/textview", new TextViewFactory(registrar.messenger()));
  }
}

使用前面AndroidView中ViewType注册一个Factory,此Factory用于创建真正的Native控件:FlutterTextView

./android/src/main/java/{organization_name}/TextViewFactory.java创建上述Factory,继承自PlatformViewFactory

import android.content.Context;
import io.flutter.plugin.common.BinaryMessenger;
import io.flutter.plugin.common.StandardMessageCodec;
import io.flutter.plugin.platform.PlatformView;
import io.flutter.plugin.platform.PlatformViewFactory;

public class TextViewFactory extends PlatformViewFactory {
    private final BinaryMessenger messenger;

    public TextViewFactory(BinaryMessenger messenger) {
        super(StandardMessageCodec.INSTANCE);
        this.messenger = messenger;
    }

    @Override
    public PlatformView create(Context context, int id, Object o) {
        return new FlutterTextView(context, messenger, id);
    }
}

Factory会创建被显示的Native View

Create Native View


实现FlutterTextView,继承自PlatformView

import android.content.Context;
import android.view.View;
import android.widget.TextView;
import io.flutter.plugin.common.MethodCall;
import io.flutter.plugin.common.MethodChannel;
import static io.flutter.plugin.common.MethodChannel.MethodCallHandler;
import static io.flutter.plugin.common.MethodChannel.Result;
import io.flutter.plugin.common.BinaryMessenger;
import io.flutter.plugin.platform.PlatformView;

public class FlutterTextView implements PlatformView, MethodCallHandler  {
    private final TextView textView;
    private final MethodChannel methodChannel;

    FlutterTextView(Context context, BinaryMessenger messenger, int id) {
        textView = new TextView(context);
        methodChannel = new MethodChannel(messenger, "plugins.my.example/textview_" + id);
        methodChannel.setMethodCallHandler(this);
    }

    @Override
    public View getView() {
        return textView;
    }

    @Override
    public void onMethodCall(MethodCall methodCall, MethodChannel.Result result) {
        switch (methodCall.method) {
            case "setText":
                setText(methodCall, result);
                break;
            default:
                result.notImplemented();
        }

    }

    private void setText(MethodCall methodCall, Result result) {
        String text = (String) methodCall.arguments;
        textView.setText(text);
        result.success(null);
    }

    @Override
    public void dispose() {}
}

FlutterTextView同时继承MethodCallHandler,需要实现onMethodCall与Flutter通信

Use PlatformView Widget


./example/lib/main.dart使用Widget效果如下:

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

void main() => runApp(MaterialApp(home: TextViewExample()));

class TextViewExample extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
        appBar: AppBar(title: const Text('Flutter TextView example')),
        body: Column(children: [
          Center(
              child: Container(
                  padding: EdgeInsets.symmetric(vertical: 30.0),
                  width: 130.0,
                  height: 100.0,
                  child: TextView(
                    onTextViewCreated: _onTextViewCreated,
                  ))),
          Expanded(
              flex: 3,
              child: Container(
                  color: Colors.blue[100],
                  child: Center(child: Text("Hello from Flutter!"))))
        ]));
  }

  void _onTextViewCreated(TextViewController controller) {
    controller.setText('Hello from Android!');
  }
}

创建TextView的同时,通过onTextViewCreated回调获取TextViewController,上文已经知道TextViewController可以帮助我们调用native的setText方法,显示我们想显示的内容
【Flutter】创建Widget显示PlatformView_第4张图片

你可能感兴趣的:(Flutter)