Flutter 平台频道的频道名称应该是什么格式?

What format should the channel name be for Flutter platform channels?

我正在添加创建我自己的平台频道(具体来说,Android 的方法频道)并关注 https://flutter.io/docs/development/platform-integration/platform-channels#step-1-create-a-new-app-project

我明白了:

All channel names used in a single app must be unique; we recommend prefixing the channel name with a unique ‘domain prefix’, e.g. samples.flutter.io/battery.

名称是否需要特定格式?例如。我可以将我的频道命名为简单的 ol' battery 吗?或者它有一些围绕格式 my_company.flutter.io/battery 的东西吗?或者是其他东西?我已经尝试了各种字符串,但每次我得到一个:

MissingPluginException(No implementation found for method getBattery on channel battery)

所以我想知道是不是我的频道名称格式不正确。 我认为我的方法名称 getBattery 是正确的。但澄清一下,我想做的是:

final result = await platform.invokeMethod('getBattery');

没有插件前缀,对吧?:

final result = await platform.invokeMethod('battery.getBattery'); 

我有一个活生生的例子here我以前做过。

Does the name need to be a specific format? E.g. Can I name my channel just plain ol' battery? Or does it have be something around the format my_company.flutter.io/battery? Or something else?

由于频道名称必须是唯一的,因此最好在其前面加上您的 bundle(iOS)/package(Android) id

例如:

static const platform =
      const MethodChannel('it.versionestabile.flutterapp000001/pdfViewer');

无论如何,您可以随意调用它,但它应该是唯一的,并且在您的 Dart 和 Android/iOS 端相等。

private static final String CHANNEL = "it.versionestabile.flutterapp000001/pdfViewer";

但是"calling"这就像在频道里放东西一样。

因为通道是一个通道(想象一个管道),你放在里面的东西从一边(飞镖)穿过它到另一个(Android/iOS) 完全是另一个故事^_^。

你放在里面的东西,应该是从另一边抓到的。

platform.invokeMethod('viewPdf', args);

所以你通过频道发送的只是消息

你在管子里放了一个有字母的瓶子^_^

现在你必须像 'viewPdf'.

这样在你的本机代码端捕获这条消息
new MethodChannel(getFlutterView(), CHANNEL).setMethodCallHandler(
            new MethodChannel.MethodCallHandler() {
              @Override
              public void onMethodCall(MethodCall call, MethodChannel.Result result) {
                if (call.method.equals("viewPdf")) {
                  if (call.hasArgument("url")) {
                    String url = call.argument("url");
                    File file = new File(url);
                    //*
                    Uri photoURI = FileProvider.getUriForFile(MainActivity.this,
                            BuildConfig.APPLICATION_ID + ".provider",
                            file);
                            //*/
                    Intent target = new Intent(Intent.ACTION_VIEW);
                    target.setDataAndType(photoURI,"application/pdf");
                    target.setFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
                    target.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                    startActivity(target);
                    result.success(null);
                  }
                } else {
                  result.notImplemented();
                }
              }
            });
  }

否则你会陷入

} else {
   result.notImplemented();
}

如果您按照示例进行操作,那么在您的 Android 本机代码端应该有这个:

@Override
public void onMethodCall(MethodCall call, 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();
    }
}

所以你应该打电话给:

platform.invokeMethod('getBatteryLevel');

因为该示例的代码期望消息 'getBatteryLevel' 通过通道传递。

加强观念

因此,为了强化所有这些概念,我将进一步告诉您,您可以决定使用通道来服务于单个操作或服务于多个操作。选择权在你。

所以你可以

省道方:

  static const singleChannel =
      const MethodChannel('it.versionestabile.flutterapp000001/single');
  static const multiChannel =
      const MethodChannel('it.versionestabile.flutterapp000001/multi');

本机端(Android):

  private static final String SINGLE_CHANNEL = "it.versionestabile.flutterapp000001/single";
  private static final String MULTI_CHANNEL = "it.versionestabile.flutterapp000001/multi";

还有一些玩具搬运工:

new MethodChannel(getFlutterView(), MULTI_CHANNEL).setMethodCallHandler(
        new MethodChannel.MethodCallHandler() {
          @Override
          public void onMethodCall(MethodCall call, MethodChannel.Result result) {
            if (call.method.equals("op1")) {
              new AlertDialog.Builder(MainActivity.this)
                      .setTitle(call.method)
                      .setMessage("I'm the " + call.method + " of the by design multi operation channel!")
                      .create()
                      .show();
            } else if (call.method.equals("op2")) {
                new AlertDialog.Builder(MainActivity.this)
                        .setTitle(call.method)
                        .setMessage("I'm the " + call.method + " of the by design multi operation channel!")
                        .create()
                        .show();
            } else {
              result.notImplemented();
            }
          }
        });

new MethodChannel(getFlutterView(), SINGLE_CHANNEL).setMethodCallHandler(
        new MethodChannel.MethodCallHandler() {
          @Override
          public void onMethodCall(MethodCall call, MethodChannel.Result result) {
            if (call.method.equals("hello")) {
              new AlertDialog.Builder(MainActivity.this)
                      .setTitle("hello!")
                      .setMessage("I'm the by design single operation channel!")
                      .create()
                      .show();
            } else {
              result.notImplemented();
            }
          }
        });

  void _invokeMultiChannelOp2() {
    multiChannel.invokeMethod('op2');
  }

  void _invokeMultiChannelOp1() {
    multiChannel.invokeMethod('op1');
  }

  void _invokeSingleChannel() {
    singleChannel.invokeMethod('hello');
  }

floatingActionButton: SafeArea(
    child: Padding(
        padding: EdgeInsets.only(left: 8.0),
        child: Row(
          children: <Widget>[
            new IconButton(
                icon: new Icon(Icons.library_music),
                onPressed: _invokeMultiChannelOp1),
            new IconButton(
                icon: new Icon(Icons.note),
                onPressed: _invokeMultiChannelOp2),
            new IconButton(
                icon: new Icon(Icons.plus_one),
                onPressed: _invokeSingleChannel),
          ],
        )),
  ),