blob: 9c036cf98e15a60cb66a68783e4cbfbeb98e7bd0 [file] [log] [blame]
// Copyright 2017 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'dart:async';
import 'package:flutter/services.dart';
import 'package:meta/meta.dart';
import 'package:platform/platform.dart';
const String kChannelName = 'plugins.flutter.io/android_intent';
/// Flutter plugin for launching arbitrary Android Intents.
class AndroidIntent {
/// Builds an Android intent with the following parameters
/// [action] refers to the action parameter of the intent.
/// [flags] is the list of int that will be converted to native flags.
/// [category] refers to the category of the intent, can be null.
/// [data] refers to the string format of the URI that will be passed to
/// intent.
/// [arguments] is the map that will be converted into an extras bundle and
/// passed to the intent.
/// [package] refers to the package parameter of the intent, can be null.
/// [componentName] refers to the component name of the intent, can be null.
/// If not null, then [package] but also be provided.
const AndroidIntent({
@required this.action,
this.flags,
this.category,
this.data,
this.arguments,
this.package,
this.componentName,
Platform platform,
}) : assert(action != null),
_channel = const MethodChannel(kChannelName),
_platform = platform ?? const LocalPlatform();
@visibleForTesting
AndroidIntent.private({
@required this.action,
@required Platform platform,
@required MethodChannel channel,
this.flags,
this.category,
this.data,
this.arguments,
this.package,
this.componentName,
}) : _channel = channel,
_platform = platform;
final String action;
final List<int> flags;
final String category;
final String data;
final Map<String, dynamic> arguments;
final String package;
final String componentName;
final MethodChannel _channel;
final Platform _platform;
bool _isPowerOfTwo(int x) {
/* First x in the below expression is for the case when x is 0 */
return x != 0 && ((x & (x - 1)) == 0);
}
@visibleForTesting
int convertFlags(List<int> flags) {
int finalValue = 0;
for (int i = 0; i < flags.length; i++) {
if (!_isPowerOfTwo(flags[i])) {
throw ArgumentError.value(flags[i], 'flag\'s value must be power of 2');
}
finalValue |= flags[i];
}
return finalValue;
}
/// Launch the intent.
///
/// This works only on Android platforms.
Future<void> launch() async {
if (!_platform.isAndroid) {
return;
}
final Map<String, dynamic> args = <String, dynamic>{'action': action};
if (flags != null) {
args['flags'] = convertFlags(flags);
}
if (category != null) {
args['category'] = category;
}
if (data != null) {
args['data'] = data;
}
if (arguments != null) {
args['arguments'] = arguments;
}
if (package != null) {
args['package'] = package;
if (componentName != null) {
args['componentName'] = componentName;
}
}
await _channel.invokeMethod<void>('launch', args);
}
}