Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/fluttercommunity/flutter_downloader
Flutter Downloader - A plugin for creating and managing download tasks.
https://github.com/fluttercommunity/flutter_downloader
android background-worker download-manager downloader flutter flutter-plugin hacktoberfest ios
Last synced: 1 day ago
JSON representation
Flutter Downloader - A plugin for creating and managing download tasks.
- Host: GitHub
- URL: https://github.com/fluttercommunity/flutter_downloader
- Owner: fluttercommunity
- License: bsd-3-clause
- Created: 2018-06-07T08:38:12.000Z (over 6 years ago)
- Default Branch: master
- Last Pushed: 2024-05-07T18:39:19.000Z (9 months ago)
- Last Synced: 2024-05-20T22:49:15.940Z (8 months ago)
- Topics: android, background-worker, download-manager, downloader, flutter, flutter-plugin, hacktoberfest, ios
- Language: Kotlin
- Homepage: https://pub.dev/packages/flutter_downloader
- Size: 2.27 MB
- Stars: 889
- Watchers: 24
- Forks: 505
- Open Issues: 304
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
- Codeowners: .github/CODEOWNERS
Awesome Lists containing this project
README
[![flutter_community][fluttercommunity_badge]][fluttercommunity_link]
# Flutter Downloader
[![flutter_downloader on pub.dev][pub_badge]][pub_link]
A plugin for creating and managing download tasks. Supports iOS and Android.
This plugin is using [`WorkManager`][work_manager] on Android and
[`NSURLSessionDownloadTask`][url_session_download_task] on iOS to run download
tasks in background.### _Development note_:
_The changes of external storage APIs in Android 11 cause some problems with the
current implementation. I decide to re-design this plugin with new strategy to
manage download file location. It is still in triage and discussion in this
[PR](https://github.com/fluttercommunity/flutter_downloader/pull/550). It is
very appreciated to have contribution and feedback from Flutter developer to get
better design for the plugin._# Past Versions and SQL Injection Vulnerabilities
In previous versions of this package, there were known vulnerabilities related to SQL injection. SQL injection is a type of security vulnerability that can allow malicious users to manipulate SQL queries executed by an application, potentially leading to unauthorized access or manipulation of the database.
It is strongly recommended to upgrade to the latest version of this package to ensure that your application is not exposed to SQL injection vulnerabilities. The latest version contains the necessary security improvements and patches to mitigate such risks.
## iOS integration
### Required configuration:
The following steps require to open your `ios` project in Xcode.
1. Enable background mode.
2. Add `sqlite` library.
3. Configure `AppDelegate`:
Objective-C:
```objective-c
/// AppDelegate.h
#import
#import@interface AppDelegate : FlutterAppDelegate
@end
``````objective-c
// AppDelegate.m
#include "AppDelegate.h"
#include "GeneratedPluginRegistrant.h"
#include "FlutterDownloaderPlugin.h"@implementation AppDelegate
void registerPlugins(NSObject* registry) {
if (![registry hasPlugin:@"FlutterDownloaderPlugin"]) {
[FlutterDownloaderPlugin registerWithRegistrar:[registry registrarForPlugin:@"FlutterDownloaderPlugin"]];
}
}- (BOOL)application:(UIApplication *)application
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
[GeneratedPluginRegistrant registerWithRegistry:self];
[FlutterDownloaderPlugin setPluginRegistrantCallback:registerPlugins];
// Override point for customization after application launch.
return [super application:application didFinishLaunchingWithOptions:launchOptions];
}@end
```
Or Swift:
```swift
import UIKit
import Flutter
import flutter_downloader@UIApplicationMain
@objc class AppDelegate: FlutterAppDelegate {
override func application(
_ application: UIApplication,
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
) -> Bool {
GeneratedPluginRegistrant.register(with: self)
FlutterDownloaderPlugin.setPluginRegistrantCallback(registerPlugins)
return super.application(application, didFinishLaunchingWithOptions: launchOptions)
}
}private func registerPlugins(registry: FlutterPluginRegistry) {
if (!registry.hasPlugin("FlutterDownloaderPlugin")) {
FlutterDownloaderPlugin.register(with: registry.registrar(forPlugin: "FlutterDownloaderPlugin")!)
}
}```
### Optional configuration:
- **Support HTTP request:** if you want to download file with HTTP request, you
need to disable Apple Transport Security (ATS) feature. There're two options:1. Disable ATS for a specific domain only: (add the following code to your
`Info.plist` file)```xml
NSAppTransportSecurityNSExceptionDomains
www.yourserver.com
NSIncludesSubdomains
NSTemporaryExceptionAllowsInsecureHTTPLoads
NSTemporaryExceptionMinimumTLSVersion
TLSv1.1
```
2. Completely disable ATS. Add the following to your `Info.plist` file)
```xml
NSAppTransportSecurityNSAllowsArbitraryLoads
```
- **Configure maximum number of concurrent tasks:** the plugin allows 3 download
tasks running at a moment by default (if you enqueue more than 3 tasks,
there're only 3 tasks running, other tasks are put in pending state). You can
change this number by adding the following code to your `Info.plist` file.```xml
FDMaximumConcurrentTasks
5
```- **Localize notification messages:** the plugin will send a notification
message to notify user in case all files are downloaded while your application
is not running in foreground. This message is English by default. You can
localize this message by adding and localizing following message in
`Info.plist` file. (you can find the detail of `Info.plist` localization in
this [link][3])```xml
FDAllFilesDownloadedMessage
All files have been downloaded
```**Note:**
- This plugin only supports save files in `NSDocumentDirectory`
## Android integration
You don't have to do anything extra to make the plugin work on Android.
There are although a few optional settings you might want to configure.
### Open downloaded file from notification
To make tapping on notification open the downloaded file on Android, add the
following code to `AndroidManifest.xml`:```xml
```
**Notes**
- You have to save your downloaded files in external storage (where the other
applications have permission to read your files)
- The downloaded files are only able to be opened if your device has at least
one application that can read these file types (mp3, pdf, etc.)### Configure maximum number of concurrent download tasks
The plugin depends on `WorkManager` library and `WorkManager` depends on the
number of available processor to configure the maximum number of tasks running
at a moment. You can setup a fixed number for this configuration by adding the
following code to your `AndroidManifest.xml`:```xml
```
### Localize strings in notifications
You can localize texts in download progress notifications by localizing
following messages.```xml
Download started
Download in progress
Download canceled
Download failed
Download complete
Download paused
```You can learn more about localization on Android [here][4].
### Install .apk files
To open and install `.apk` files, your application needs
`REQUEST_INSTALL_PACKAGES` permission. Add the following in your
`AndroidManifest.xml`:```xml
```
See also:
- [Fix Cleartext Traffic error on Android 9 Pie][android_9_cleartext_traffic]
## Usage
### Import and initialize
```dart
import 'package:flutter_downloader/flutter_downloader.dart';void main() {
WidgetsFlutterBinding.ensureInitialized();// Plugin must be initialized before using
await FlutterDownloader.initialize(
debug: true, // optional: set to false to disable printing logs to console (default: true)
ignoreSsl: true // option: set to false to disable working with http links (default: false)
);runApp(/*...*/)
}
```### Create new download task
The directory must be created in advance.
After that, you need to provide the path of the directory in the `savedDir` parameter.```dart
final taskId = await FlutterDownloader.enqueue(
url: 'your download link',
headers: {}, // optional: header send with url (auth token etc)
savedDir: 'the path of directory where you want to save downloaded files',
showNotification: true, // show download progress in status bar (for Android)
openFileFromNotification: true, // click on notification to open downloaded file (for Android)
);
```### Update download progress
```dart
await FlutterDownloader.registerCallback(callback); // callback is a top-level or static function
```**Important**
UI is rendered on the main isolate, while download events come from the
background isolate (in other words, code in `callback` is run in the background
isolate), so you have to handle the communication between two isolates. For
example:```dart
ReceivePort _port = ReceivePort();@override
void initState() {
super.initState();IsolateNameServer.registerPortWithName(_port.sendPort, 'downloader_send_port');
_port.listen((dynamic data) {
String id = data[0];
DownloadTaskStatus status = DownloadTaskStatus.fromInt(data[1]);
int progress = data[2];
setState((){ });
});FlutterDownloader.registerCallback(downloadCallback);
}@override
void dispose() {
IsolateNameServer.removePortNameMapping('downloader_send_port');
super.dispose();
}@pragma('vm:entry-point')
static void downloadCallback(String id, int status, int progress) {
final SendPort? send = IsolateNameServer.lookupPortByName('downloader_send_port');
send?.send([id, status, progress]);
}```
`@pragma('vm:entry-point')` must be placed above the `callback` function to
avoid tree shaking in release mode for Android.### Load all download tasks
```dart
final List? tasks = await FlutterDownloader.loadTasks();
```### Load download tasks using a raw SQL query
```dart
final List? tasks = await FlutterDownloader.loadTasksWithRawQuery(query: query);
```In order to parse data into `DownloadTask` object successfully, you should load
data with all fields from the database (in the other words, use `SELECT *` ).
For example:```SQL
SELECT * FROM task WHERE status=3
```Below is the schema of the `task` table where `flutter_downloader` plugin stores
information about download tasks```SQL
CREATE TABLE `task` (
`id` INTEGER PRIMARY KEY AUTOINCREMENT,
`task_id` VARCHAR ( 256 ),
`url` TEXT,
`status` INTEGER DEFAULT 0,
`progress` INTEGER DEFAULT 0,
`file_name` TEXT,
`saved_dir` TEXT,
`resumable` TINYINT DEFAULT 0,
`headers` TEXT,
`show_notification` TINYINT DEFAULT 0,
`open_file_from_notification` TINYINT DEFAULT 0,
`time_created` INTEGER DEFAULT 0
);
```### Cancel a task
```dart
FlutterDownloader.cancel(taskId: taskId);
```### Cancel all tasks
```dart
FlutterDownloader.cancelAll();
```### Pause a task
```dart
FlutterDownloader.pause(taskId: taskId);
```### Resume a task
```dart
FlutterDownloader.resume(taskId: taskId);
````resume()` will return a new `taskId` corresponding to a new background task
that is created to continue the download process. You should replace the old
`taskId` (that has `paused` status) by the new `taskId` to continue tracking the
download progress.### Retry a failed task
```dart
FlutterDownloader.retry(taskId: taskId);
````retry()` will return a new `taskId` (just like `resume()`)
### Remove a task
```dart
FlutterDownloader.remove(taskId: taskId, shouldDeleteContent:false);
```### Open and preview a downloaded file
```dart
FlutterDownloader.open(taskId: taskId);
```On Android, you can only open a downloaded file if it is placed in the external
storage and there's at least one application that can read that file type on
your device.## Bugs/Requests
Feel free to open an issue if you encounter any problems or think that the
plugin is missing some feature.Pull request are also very welcome!
[fluttercommunity_badge]: https://fluttercommunity.dev/_github/header/flutter_downloader
[fluttercommunity_link]: https://github.com/fluttercommunity/community
[pub_badge]: https://img.shields.io/pub/v/flutter_downloader.svg
[pub_link]: https://pub.dartlang.org/packages/flutter_downloader
[work_manager]: https://developer.android.com/topic/libraries/architecture/workmanager
[url_session_download_task]: https://developer.apple.com/documentation/foundation/nsurlsessiondownloadtask?language=objc
[android_9_cleartext_traffic]: https://medium.com/@son.rommer/fix-cleartext-traffic-error-in-android-9-pie-2f4e9e2235e6
[3]: https://medium.com/@guerrix/info-plist-localization-ad5daaea732a
[4]: https://developer.android.com/training/basics/supporting-devices/languages