English | 中文说明 (🚧 WIP)
A Flutter plugin that provides assets abstraction management APIs without UI integration, you can get assets (image/video/audio) on Android, iOS and macOS.
name | pub | github |
---|---|---|
wechat_assets_picker | ||
wechat_camera_picker |
For versions upgrade across major versions, see the migration guide for detailed info.
- Common issues
- Prepare for use
- Usage
- Cache mechanism
- Native extra configs
Please search common issues in GitHub issues for build errors, runtime exceptions, etc.
Two ways to add the plugin to your pubspec:
- (Recommend) Run
flutter pub add photo_manager
. - Add the plugin reference in your
pubspec.yaml
'sdependencies
section:
dependencies:
photo_manager: $latest_version
import 'package:photo_manager/photo_manager.dart';
Minumum platform versions: Android 16, iOS 9.0, macOS 10.15.
- Android: Android config preparation.
- iOS: iOS config preparation.
- macOS: Pretty much the same with iOS.
Starting from 1.2.7, We ship this plugin with
Kotlin 1.5.21
and Android Gradle Plugin 4.1.0
.
If your projects use a lower version of Kotlin/Gradle/AGP,
please upgrade them to a newer version.
More specifically:
- Upgrade your Gradle version (
gradle-wrapper.properties
) to6.8.3
or the latest version but lower than7.0.0
. - Upgrade your Kotlin version (
ext.kotlin_version
) to1.4.32
or the latest version.
If you're compiling or targeting with an Android version that belows 29, you can skip this section.
On Android 10, Scoped Storage was introduced, which causes the origin resource file inaccessible.
If your compileSdkVersion
is above 29, you must add
android:requestLegacyExternalStorage="true"
to your
AndroidManifest.xml
in order to obtain resources:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="top.kikt.photo_manager_example">
<application
android:label="photo_manager_example"
android:icon="@mipmap/ic_launcher"
android:requestLegacyExternalStorage="true">
</application>
</manifest>
The plugin use Glide to create thumbnail bytes for Android.
If you found some warning logs with Glide appearing,
it means the main project needs an implementation of AppGlideModule
.
See Generated API for the implementation.
Define the NSPhotoLibraryUsageDescription
key-value in the ios/Runner/Info.plist
:
<key>NSPhotoLibraryUsageDescription</key>
<string>In order to access your photo library</string>
If you want to grant only write-access to the photo library on iOS 11 and above,
define the NSPhotoLibraryAddUsageDescription
key-value in the ios/Runner/Info.plist
.
It's pretty much the same as the NSPhotoLibraryUsageDescription
.
Most of APIs can only use with granted permission.
final PermissionState _ps = await PhotoManager.requestPermissionExtend();
if (_ps.isAuth) {
// Granted.
} else {
// Limited(iOS) or Rejected, use `==` for more precise judgements.
// You can call `PhotoManager.openSetting()` to open settings for further steps.
}
But if you're pretty sure your callers will be only called after the permission is granted, you can ignore permission checks:
PhotoManager.setIgnorePermissionCheck(true);
With iOS 14 released,
Apple broughts a "Limited Photos Library" to iOS.
So use the PhotoManager.requestPermissionExtend()
to request permissions.
The method will return PermissionState
.
See PHAuthorizationStatus for more detail.
To reselect accessible entites for the app,
use PhotoManager.presentLimited()
to call the modal of
accessible entities management.
This method only available for iOS 14+ and when the permission state
is limited (PermissionState.limited
),
other platform won't make a valid call.
Albums or folders are abstracted as the AssetPathEntity
class.
It represent a bucket in the MediaStore
on Android,
and the PHAssetCollection
object on iOS/macOS.
To get all of them:
final List<AssetPathEntity> paths = await PhotoManager.getAssetPathList();
See getAssetPathList
for more detail.
Assets (images/videos/audios) are abstracted as the AssetEntity
class.
It represents a series of fields with MediaStore
on Android,
and the PHAsset
object on iOS/macOS.
You can use the pagination method:
final List<AssetEntity> entities = await path.getAssetListPaged(page: 0, size: 80);
Or use the range method:
final List<AssetEntity> entities = await path.getAssetListRange(start: 0, end: 80);
The ID concept represents:
- The ID field of the
MediaStore
on Android. - The
localIdentifier
field of thePHAsset
on iOS.
You can store the ID if you want to implement features
that's related to presistent selections.
Use AssetEntity.fromId
to retrieve the entity
once you persist an ID.
final AssetEntity? asset = await AssetEntity.fromId(id);
Be aware that the created asset might have limited access or got deleted in anytime, so the result might be null.
You can create your own entity from raw data, such as downloaded images, recorded videos, etc. The created entity will shown as a corresponing resource on your device's gallery app.
final Uint8List rawData = yourRawData;
// Save an image to an entity from `Uint8List`.
final AssetEntity? entity = await PhotoManager.editor.saveImage(
rawData,
title: 'write_your_own_title.jpg', // Affects EXIF reading.
);
// Save an existed image to an entity from it's path.
final AssetEntity? imageEntityWithPath = await PhotoManager.editor.saveImageWithPath(
path, // Use the absolute path of your source file, it's more like a copy method.
title: 'same_as_above.jpg',
);
// Save a video entity from `File`.
final File videoFile = File('path/to/your/video.mp4');
final AssetEntity? videoEntity = await PhotoManager.editor.saveVideo(
videoFile, // You can check whether the file is exist for better test coverage.
title: 'write_your_own_title.mp4',
);
Be aware that the created asset might have limited access or got deleted in anytime, so the result might be null.
Resources might be saved only on iCloud to save disk space.
When retrieving file from iCloud, the speed is depend on the network condition,
which might be very slow that makes users feel anxious.
To provide a responsive user interface, you can use PMProgressHandler
to retrieve the progress when load a file.
The preferred implementation would be the LocallyAvailableBuilder
in the wechat_asset_picker
package, which provides a progress indicator
when the file is downloading.
The plugin provided the AssetEntityImage
widget and
the AssetEntityImageProvider
to display assets:
final Widget image = AssetEntityImage(
yourAssetEntity,
isOriginal: false, // Defaults to `true`.
thumbnailSize: const ThumbnailSize.square(200), // Preferred value.
thumbnailFormat: ThumbnailFormat.jpeg, // Defaults to `jpeg`.
);
final Widget imageFromProvider = Image(
image: AssetEntityImageProvider(
yourAssetEntity,
isOriginal: false,
thumbnailSize: const ThumbnailSize.square(200),
thumbnailFormat: ThumbnailFormat.jpeg,
),
);
This plugin supports obtain live photos and filtering them:
This is supported when filtering only image.
final List<AssetPathEntity> paths = await PhotoManager.getAssetPathList(
type: RequestType.image,
filterOption: FilterOptionGroup(onlyLivePhotos: true),
);
final AssetEntity entity = livePhotoEntity;
final String? mediaUrl = await entity.getMediaUrl();
final File? imageFile = await entity.file;
final File? videoFile = await entity.fileWithSubtype;
final File? originImageFile = await entity.originFile;
final File? originVideoFile = await entity.originFileWithSubtype;
Due to the privacy policy issues on Android 10, it is necessary to grant the location permission to obtain the original data with the location info and the EXIF metadata.
If you want to use the location permission,
add the ACCESS_MEDIA_LOCATION
permission to your manifest.
The originFile
and originBytes
getter
will return the original data of an entity.
However, there are some cases that the original data is invalid in Flutter.
Here are some common cases:
- HEIC files are not fully supported across platforms. We suggest you to upload the JPEG file (99% quality compressed thumbnail) in order to keep a consistent behavior between multiple platforms. See flutter/flutter#20522 for more detail.
- Videos will only be obtained in the original format, not the exported/composited format, which might cause some behavior difference when playing videos.
There are several I/O methods in this library targeting AssetEntity
,
typically they are:
- All methods named with
file
. AssetEntity.originBytes
.
File retrieving and caches are limited by the sandbox mechanisim on iOS.
An existing PHAsset
doesn't mean the file located on the device.
In generall, a PHAsset
will have three status:
isLocallyAvailable
equalstrue
, also cached: Available for obtain.isLocallyAvailable
equalstrue
, but not cached: When you call I/O methods, the resource will first cached into the sandbox, then available for obtain.isLocallyAvailable
equalsfalse
: Typically this means the asset exists, but it's saved only on iCloud, or some videos that not exported yet. In this case, the best practise is to use thePMProgressHandler
to provide a responsive user interface.
Plugin will post entities change events from native,
but they will include different contents.
See the logs
folder for more recorded logs.
To register a callback for these events, use
[PhotoManager.addChangeCallback
] to add a callback,
and use [PhotoManager.removeChangeCallback
] to remove the callback,
just like addListener
and removeListener
methods.
After you added/removed callbacks, you can call
[PhotoManager.startChangeNotify
] method to enable to notify,
and [PhotoManager.stopChangeNotify
] method to stop notify.
import 'package:flutter/services.dart';
void changeNotify(MethodCall call) {
// Your custom callback.
}
/// Register your callback.
PhotoManager.addChangeCallback(changeNotify);
/// Enable change notify.
PhotoManager.startChangeNotify();
/// Remove your callback.
PhotoManager.removeChangeCallback(changeNotify);
/// Disable change notify.
PhotoManager.stopChangeNotify();
Because Android 10 restricts the ability to access the resource path directly,
some large image caches will be generated during I/O processes.
More specifically, when the file
, originFile
and any other I/O getters are called,
the plugin will save a file in the cache folder for further use.
Fortunately, in Android 11, the resource path can be obtained directly again,
but for Android 10, we can only use
requestLegacyExternalStorage
as a workaround.
See Android 10 extra configs
for how to add the attribute.
iOS does not directly provide APIs to access the original files of the album.
So a cached file will be generated locally
into the container of the current application
when you called file
, originFile
and any other I/O getters.
If occupied disk spaces are sensitive in your use case, you can delete it after your usage has done (iOS only).
import 'dart:io';
Future<void> useEntity(AssetEntity entity) async {
File? file;
try {
file = await entity.file;
handleFile(file!); // Custom method to handle the obtained file.
} finally {
if (Platform.isIOS) {
file?.deleteSync(); // Delete it once the process has done.
}
}
}
You can use the PhotoManager.clearFileCache()
method
to clear all caches that generated by the plugin.
Here are caches generatation on different'
platforms, types and resolutions.
Platform | Thumbnail | File / Origin File |
---|---|---|
Android | Yes | No |
iOS | No | Yes |
If your found any conflicting issues against Glide,
then you'll need to edit the android/build.gradle
file:
rootProject.allprojects {
subprojects {
project.configurations.all {
resolutionStrategy.eachDependency { details ->
if (details.requested.group == 'com.github.bumptech.glide'
&& details.requested.name.contains('glide')) {
details.useVersion '4.11.0'
}
}
}
}
}
See ProGuard for Glide if you want to know more about using ProGuard and Glide together.
By default, iOS will retrieve system album names only in English no matter what language has been set to devices. To change the default language, see the following steps:
-
Select the project "Runner" and in the localizations table, click on the + icon.
-
Select the adequate language(s) you want to retrieve localized strings.
-
Validate the popup screen without any modification.
-
Rebuild your flutter project.
Now system albums label should display accordingly.
Warning: Features here aren't guaranteed to be fully usable since they involved with data modification. They can be modified/removed in any time, without following a proper version semantic.
Some APIs will make irreversible modification/deletion to datas. Please be careful and implement your own test mechanism when using them.
You can preload thumbnails for entites with specified thumbnail options
using PhotoCachingManager.requestCacheAssets
or PhotoCachingManager.requestCacheAssetsWithIds
.
PhotoCachingManager().requestCacheAssets(assets: assets, option: option);
And you can stop in anytime by calling
PhotoCachingManager().cancelCacheRequest()
.
Usually, when we're previewing assets, thumbnails will be use. But sometimes we want to preload assets to make them display faster.
The PhotoCachingManager
uses the PHCachingImageManager on iOS,
and Glide's file cache on Android.
This method will delete the asset completely from your device. Use it with extra cautious.
// Deleted IDs will returned, if it fails, the result will be an empty list.
final List<String> result = await PhotoManager.editor.deleteWithIds(
<String>[entity.id],
);
After the delection, you can call the refreshPathProperties
method
to refresh the corresponding AssetPathEntity
in order to get latest fields.
You can use copyAssetToPath
method to "Copy" an entity
from its current position to the targeting AssetPathEntity
:
// Make sure your path entity is accessible.
final AssetPathEntity anotherPathEntity = anotherAccessiblePath;
final AssetEntity entity = yourEntity;
final AssetEntity? newEntity = await PhotoManager.editor.copyAssetToPath(
asset: entity,
pathEntity: anotherPathEntity,
); // The result could be null when the path is not accessible.
The "Copy" means differently here on Android and iOS:
- For Android, it inserts a copy of the source entity:
- On platforms <=28, the method will copy most of the origin info.
- On platforms >=29, some fields cannot be modified during the insertion, e.g. MediaColumns.RELATIVE_PATH.
- For iOS, it makes a shortcut thing rather than create a new physical entity.
- Some albums are smart albums, their content is automatically managed by the system and cannot inserted entities manually.
(For Android 30+, this feature is blocked by system limitations currently.)
// Make sure your path entity is accessible.
final AssetPathEntity pathEntity = accessiblePath;
final AssetEntity entity = yourEntity;
await PhotoManager.editor.android.moveAssetToAnother(
entity: entity,
target: pathEntity,
);
(For Android 30+, this feature is blocked by system limitations currently.)
This will remove all items (records) that's not existed locally.
A record in Android MediaStore
could have the corresponding file deleted.
Those abnormal behaviors usually caused by operations from
file manager, helper tools or adb tool.
This operation is resource-consuming,
Please use the await
keyword to call the cleaning process
before you call another one.
await PhotoManager.editor.android.removeAllNoExistsAsset();
Some operating systems will prompt confirmation dialogs for each entities' deletion, we have no way to avoid them. Make sure your customers accept repeatly confirmations.
PhotoManager.editor.iOS.createFolder(
name,
parent: parent, // Null, the root path or accessible folders.
);
PhotoManager.editor.iOS.createAlbum(
name,
parent: parent, // Null, the root path or accessible folders.
);
Remove the entry of the asset from the specific album. The asset won't be deleted from the device, only removed from the album.
// Make sure your path entity is accessible.
final AssetPathEntity pathEntity = accessiblePath;
final AssetEntity entity = yourEntity;
final List<AssetEntity> entities = <AssetEntity>[yourEntity, anotherEntity];
// Remove single asset from the album.
// It'll call the list method as the implementation.
await PhotoManager.editor.iOS.removeInAlbum(
yourEntity,
accessiblePath,
);
// Remove assets from the album in batches.
await PhotoManager.editor.iOS.removeAssetsInAlbum(
entities,
accessiblePath,
);
Smart albums can't be deleted.
PhotoManager.editor.iOS.deletePath();