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),
],
)),
),
我正在添加创建我自己的平台频道(具体来说,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),
],
)),
),