Flutter应用安装插件easy_app_installer的使用
Flutter应用安装插件easy_app_installer的使用
easy_app_installer简介
easy_app_installer
为Flutter提供简便的应用内升级功能,主要支持以下功能:
- 下载并安装apk(仅支持 Android)
- 取消下载中的任务(仅支持 Android)
- 仅安装apk(仅支持 Android)
- 跳转到应用市场-指定应用详情页(支持Android与iOS)
- 跳转到设置-应用详情页(支持Android与iOS)
配置 - Android
FileProvider配置
- 在
android/app/src/main/res
下新建xml
文件夹,并在其中创建file_provider_path.xml
文件,内容如下:
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<root-path name="root" path="."/>
<files-path name="files" path="."/>
<cache-path name="cache" path="."/>
<external-path name="external" path="."/>
<external-cache-path name="external_cache" path="."/>
<external-files-path name="external_file" path="."/>
</paths>
- 打开
android/app/src/main/AndroidManifest.xml
文件,在application
标签下添加:
<provider
android:authorities="${applicationId}.fileprovider"
android:exported="false"
android:grantUriPermissions="true"
android:name="androidx.core.content.FileProvider">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_provider_path" />
</provider>
示例文件可参考这里。
使用
安装
安装插件请参考官方文档。
功能使用说明
1. 下载并安装apk
此功能适用于Android平台。首次同意’允许安装应用’权限可能会导致应用进程关闭,这是系统行为,Android 12已经修复了这个问题。
参数名称 | 参数意义 | 是否必传 |
---|---|---|
fileUrl | 要下载apk的url地址 | 是 |
fileDirectory | 文件夹路径(无须拼接反斜杠) | 是 |
fileName | 文件名称(无需拼接反斜杠) | 是 |
explainContent | 权限弹窗的提示内容 | 否 |
positiveText | 权限弹窗的确认文字 | 否 |
negativeText | 权限弹窗的取消文字 | 否 |
isDeleteOriginalFile | 如已存在相同文件,是否要删除(默认为true) | 否 |
downloadListener | 下载进度回调,double类型,值为 0~100 | 否 |
cancelTagListener | 回调用于取消下载中任务的tag | 否 |
stateListener | 下载状态变化时回调,请参考下载状态 | 否 |
示例代码:
String _cancelTag = "";
EasyAppInstaller.instance.downloadAndInstallApk(
fileUrl: "https://xxxx.apk",
fileDirectory: "updateApk",
fileName: "new.apk",
downloadListener: (progress) {
if (progress < 100) {
EasyLoading.showProgress(progress / 100, status: "下载中");
} else {
EasyLoading.showSuccess("下载成功");
}
},
cancelTagListener: (cancelTag) {
_cancelTag = cancelTag;
},
stateListener: (newState, attachParam) {
_handleDownloadStateChanged(newState, attachParam);
}
);
2. 取消下载中的任务
参数名称 | 参数意义 | 是否必传 |
---|---|---|
cancelTag | 要取消任务的tag | 是 |
示例代码:
EasyAppInstaller.instance.cancelDownload(_cancelTag);
3. 仅安装apk
参数名称 | 参数意义 | 是否必传 |
---|---|---|
filePath | apk文件的绝对路径 | 是 |
explainContent | 权限弹窗的提示内容 | 否 |
positiveText | 权限弹窗的确认文字 | 否 |
negativeText | 权限弹窗的取消文字 | 否 |
示例代码:
await EasyAppInstaller.instance.installApk(filePath);
4. 跳转到应用市场-指定应用页面
仅支持Android平台。
参数名称 | 参数意义 | 是否必传 |
---|---|---|
applicationPackageName | 指定应用包名(默认为空) | 否 |
targetMarketPackageName | 指定应用市场包名(默认为空) | 否 |
isOpenSystemMarket | 是否打开厂商应用市场(默认为true) | 否 |
示例代码:
await EasyAppInstaller.instance.openAppMarket();
5. 跳转到AppStore-指定应用页面
仅支持iOS平台。
参数名称 | 参数意义 | 是否必传 |
---|---|---|
appId | Apple为应用生成的AppId | 是 |
示例代码:
EasyAppInstaller.instance.openAppStore(appId: "${appId}");
6. 跳转到设置-应用详情页
支持Android和iOS平台。
参数名称 | 参数意义 | 是否必传 |
---|---|---|
applicationPackageName | 指定应用包名(默认为空) | 否 |
示例代码:
await EasyAppInstaller.instance.openAppSettingDetails(applicationPackageName: "$targetAppPackage");
类说明
下载状态
enum EasyAppInstallerState {
//准备下载 (该状态回调在获取到权限之后)
onPrepared,
//下载中
onDownloading,
//下载成功
onSuccess,
//下载失败
onFailed,
//取消下载
onCanceled
}
示例Demo
import 'package:flutter/material.dart';
import 'dart:async';
import 'package:flutter/services.dart';
import 'package:easy_app_installer/easy_app_installer.dart';
import 'package:flutter_dotenv/flutter_dotenv.dart';
import 'package:flutter_easyloading/flutter_easyloading.dart';
void main() {
runApp(const MyApp());
}
class MyApp extends StatefulWidget {
const MyApp({Key? key}) : super(key: key);
@override
State<MyApp> createState() => _MyAppState();
}
class _MyAppState extends State<MyApp> {
String _platformVersion = 'Unknown';
String _cancelTag = "";
String _apkFilePath = "";
String _currentDownloadStateCH = "当前下载状态:还未开始";
@override
void initState() {
super.initState();
initData();
}
Future<void> initData() async {
String platformVersion;
try {
platformVersion = await EasyAppInstaller.instance.platformVersion ??
'Unknown platform version';
} on PlatformException {
platformVersion = 'Failed to get platform version.';
}
if (!mounted) return;
setState(() {
_platformVersion = platformVersion;
});
await dotenv.load(fileName: "assets/testdata.env");
}
@override
Widget build(BuildContext context) {
return MaterialApp(
builder: EasyLoading.init(),
home: Scaffold(
appBar: AppBar(
title: const Text('Plugin example app'),
),
body: Center(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
Text(_currentDownloadStateCH),
_buildButton("打开AppStore", () async {
final appId = dotenv.get("IOS_APP_ID", fallback: "");
final openAppStoreResult =
await EasyAppInstaller.instance.openAppStore(appId);
print("gfs openAppStoreResult: $openAppStoreResult");
}),
_buildButton('下载并安装apk', () {
downloadAndInstalApk();
}),
_buildButton('取消下载任务', () {
if (_cancelTag.isNotEmpty) {
EasyLoading.dismiss();
EasyAppInstaller.instance.cancelDownload(_cancelTag);
} else {
EasyLoading.showError("没有下载中的任务");
}
}),
_buildButton('仅安装', () async {
final path =
await EasyAppInstaller.instance.installApk(_apkFilePath);
print("gfs installApk: $path");
}),
_buildButton('打开应用市场', () {
EasyAppInstaller.instance.openAppMarket();
}),
_buildButton('打开设置详情页', () async {
final openResult =
await EasyAppInstaller.instance.openAppSettingDetails();
print("gfs openResult: $openResult");
}),
],
),
),
),
);
}
void downloadAndInstalApk() async {
//fileUrl需替换为指定apk地址
await EasyAppInstaller.instance.downloadAndInstallApk(
fileUrl: dotenv.get("APK_URL", fallback: ""),
fileDirectory: "updateApk",
fileName: "newApk.apk",
explainContent: "快去开启权限!!!",
onDownloadingListener: (progress) {
if (progress < 100) {
EasyLoading.showProgress(progress / 100, status: "下载中");
} else {
EasyLoading.showSuccess("下载成功");
}
},
onCancelTagListener: (cancelTag) {
_cancelTag = cancelTag;
},
onStateListener: (newState, attachParam) {
_handleDownloadStateChanged(newState, attachParam);
});
}
/// 处理下载状态更改
void _handleDownloadStateChanged(
EasyAppInstallerState newState, String? attachParam) {
switch (newState) {
case EasyAppInstallerState.onPrepared:
_currentDownloadStateCH = "当前下载状态:开始下载";
break;
case EasyAppInstallerState.onDownloading:
_currentDownloadStateCH = "当前下载状态:下载中";
break;
case EasyAppInstallerState.onSuccess:
if (attachParam != null) {
_currentDownloadStateCH = "当前下载状态:下载成功, $attachParam";
_apkFilePath = attachParam;
}
break;
case EasyAppInstallerState.onFailed:
_currentDownloadStateCH = "当前下载状态:下载失败, $attachParam";
break;
case EasyAppInstallerState.onCanceled:
_currentDownloadStateCH = "当前下载状态:取消下载";
break;
}
setState(() {});
}
Widget _buildButton(String text, Function function) {
return MaterialButton(
color: Colors.blue,
child: Text(
text,
style: const TextStyle(color: Colors.white),
),
onPressed: () {
function();
},
);
}
@override
void dispose() {
EasyAppInstaller.instance.dispose();
super.dispose();
}
}
通过上述内容,您可以更好地理解和使用easy_app_installer
插件来实现Flutter应用内的升级功能。希望这些信息对您有所帮助!
更多关于Flutter应用安装插件easy_app_installer的使用的实战系列教程也可以访问 https://www.itying.com/category-92-b0.html
更多关于Flutter应用安装插件easy_app_installer的使用的实战系列教程也可以访问 https://www.itying.com/category-92-b0.html
当然,下面是一个关于如何在Flutter应用中使用easy_app_installer
插件的示例代码。easy_app_installer
插件允许你在Flutter应用中方便地安装APK文件。
首先,确保你已经在pubspec.yaml
文件中添加了easy_app_installer
依赖:
dependencies:
flutter:
sdk: flutter
easy_app_installer: ^x.y.z # 替换为最新版本号
然后运行flutter pub get
来安装依赖。
接下来是具体的代码实现:
1. 导入插件
在你的Dart文件中导入easy_app_installer
插件:
import 'package:easy_app_installer/easy_app_installer.dart';
import 'package:flutter/material.dart';
2. 请求安装权限(Android)
由于安装APK文件需要特定的权限,因此在Android上你需要请求相关权限。你可以在AndroidManifest.xml
中添加以下权限:
<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/>
注意:从Android 8.0(API级别26)开始,REQUEST_INSTALL_PACKAGES
权限仅对系统应用或具有特权的应用程序(如默认的设备管理员或具有签名权限的应用)授予。对于普通应用,通常需要通过用户手动授予或使用其他机制来安装APK。
3. 使用EasyAppInstaller
下面是一个完整的示例,展示如何使用EasyAppInstaller
来安装APK文件:
void main() {
runApp(MyApp());
}
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
home: Scaffold(
appBar: AppBar(
title: Text('Easy App Installer Example'),
),
body: Center(
child: ElevatedButton(
onPressed: () async {
// APK文件的路径,可以是本地路径或者从网络下载的路径
String apkPath = "/path/to/your/app.apk";
// 检查并请求安装权限(仅适用于Android 7.0及以下)
if (Platform.isAndroid) {
bool hasPermission = await EasyAppInstaller.checkInstallPermission();
if (!hasPermission) {
bool granted = await EasyAppInstaller.requestInstallPermission();
if (!granted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text('Install permission denied')),
);
return;
}
}
}
// 安装APK
try {
bool isInstalled = await EasyAppInstaller.installApk(apkPath);
if (isInstalled) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text('APK installed successfully')),
);
} else {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text('APK installation failed')),
);
}
} catch (e) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text('Error: ${e.message}')),
);
}
},
child: Text('Install APK'),
),
),
),
);
}
}
注意事项
- APK路径:确保APK文件的路径是正确的。如果APK文件存储在设备的存储中,你需要确保你的应用有访问该路径的权限。
- 权限处理:从Android 8.0开始,安装APK的权限处理变得更加复杂。如果
REQUEST_INSTALL_PACKAGES
权限请求失败,你可能需要引导用户通过其他方式(如ADB安装)来安装APK。 - 错误处理:在实际应用中,应该添加更多的错误处理逻辑,以处理各种可能的异常情况。
希望这个示例能帮助你在Flutter应用中使用easy_app_installer
插件来安装APK文件。