Нема описа

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. library photo;
  2. import 'dart:async';
  3. import 'package:flutter/material.dart';
  4. import 'package:photo/src/entity/_theme.dart';
  5. import 'package:photo/src/ui/page/photo_list_page.dart';
  6. import 'package:photo/src/ui/page/photo_main_page.dart';
  7. import 'package:photo_manager/photo_manager.dart';
  8. import 'package:photo/src/delegate/badge_delegate.dart';
  9. import 'package:photo/src/delegate/checkbox_builder_delegate.dart';
  10. import 'package:photo/src/delegate/loading_delegate.dart';
  11. import 'package:photo/src/delegate/sort_delegate.dart';
  12. import 'package:photo/src/entity/options.dart';
  13. import 'package:photo/src/provider/i18n_provider.dart';
  14. import 'package:photo/src/ui/dialog/not_permission_dialog.dart';
  15. import 'package:provider/provider.dart';
  16. import 'src/provider/config_provider.dart';
  17. export 'package:photo/src/delegate/checkbox_builder_delegate.dart';
  18. export 'package:photo/src/delegate/loading_delegate.dart';
  19. export 'package:photo/src/delegate/sort_delegate.dart';
  20. export 'package:photo/src/provider/i18n_provider.dart'
  21. show I18NCustomProvider, I18nProvider, CNProvider, ENProvider;
  22. export 'package:photo/src/entity/options.dart' show PickType, PickedEntity;
  23. export 'package:photo/src/delegate/badge_delegate.dart';
  24. export 'package:photo/src/entity/_theme.dart';
  25. class PhotoPicker {
  26. static PhotoPicker _instance;
  27. PhotoPicker._();
  28. factory PhotoPicker() {
  29. _instance ??= PhotoPicker._();
  30. return _instance;
  31. }
  32. static const String rootRouteName = "photo_picker_image";
  33. /// 没有授予权限的时候,会开启一个dialog去帮助用户去应用设置页面开启权限
  34. /// 确定开启设置页面,取消关闭弹窗,无论选择什么,返回值都是null
  35. ///
  36. ///
  37. /// 当用户给予权限后
  38. ///
  39. /// 当用户确定时,返回一个图片[AssetEntity]列表
  40. ///
  41. /// 当用户取消时返回一个空数组
  42. ///
  43. /// [photoPathList] 一旦设置 则 [pickType]参数无效
  44. ///
  45. /// [pickedAssetList] 已选择的asset
  46. ///
  47. /// 关于参数可以查看readme文档介绍
  48. ///
  49. /// if user not grand permission, then return null and show a dialog to help user open setting.
  50. /// sure is open setting cancel ,cancel to dismiss dialog, return null
  51. ///
  52. /// when user give permission.
  53. ///
  54. /// when user sure , return a [AssetEntity] of [List]
  55. ///
  56. /// when user cancel selected,result is empty list
  57. ///
  58. /// when [photoPathList] is not null , [pickType] invalid
  59. ///
  60. /// [pickedAssetList]: The results of the last selection can be passed in for easy secondary selection.
  61. ///
  62. /// params see readme.md
  63. static Future<PickedEntity> pickAsset({
  64. @required BuildContext context,
  65. int rowCount = 4,
  66. int maxSelected = 9,
  67. double padding = 1,
  68. double itemRadio = 1.0,
  69. int thumbSize = 180,
  70. PhotoTheme theme,
  71. I18nProvider provider = I18nProvider.chinese,
  72. SortDelegate sortDelegate,
  73. LoadingDelegate loadingDelegate,
  74. PickType pickType = PickType.all,
  75. List<AssetPathEntity> photoPathList,
  76. List<AssetEntity> pickedAssetList,
  77. }) {
  78. assert(provider != null, "provider must be not null");
  79. assert(context != null, "context must be not null");
  80. assert(pickType != null, "pickType must be not null");
  81. sortDelegate ??= SortDelegate.common;
  82. loadingDelegate ??= DefaultLoadingDelegate();
  83. var options = Options(
  84. rowCount: rowCount,
  85. maxSelected: maxSelected,
  86. itemRadio: itemRadio,
  87. padding: padding,
  88. theme: theme ?? PhotoTheme.fallback(context),
  89. thumbSize: thumbSize,
  90. sortDelegate: sortDelegate,
  91. checkBoxBuilderDelegate: DefaultCheckBoxBuilderDelegate(),
  92. loadingDelegate: loadingDelegate,
  93. badgeDelegate: DefaultBadgeDelegate(),
  94. pickType: pickType,
  95. );
  96. return PhotoPicker()._pickAsset(
  97. context,
  98. options,
  99. provider,
  100. photoPathList,
  101. pickedAssetList,
  102. );
  103. }
  104. Future<PickedEntity> _pickAsset(
  105. BuildContext context,
  106. Options options,
  107. I18nProvider provider,
  108. List<AssetPathEntity> photoList,
  109. List<AssetEntity> pickedAssetList,
  110. ) async {
  111. await requestPermission(context, provider: provider);
  112. return _openGalleryContentPage(
  113. context,
  114. options,
  115. provider,
  116. photoList,
  117. pickedAssetList,
  118. );
  119. }
  120. static Widget buildGallery({
  121. @required BuildContext context,
  122. double padding = 1,
  123. int maxSelected = 9,
  124. int thumbSize = 180,
  125. double itemRadio = 1.0,
  126. PhotoTheme theme,
  127. I18nProvider provider = I18nProvider.chinese,
  128. SortDelegate sortDelegate,
  129. LoadingDelegate loadingDelegate,
  130. PickType pickType = PickType.all,
  131. List<AssetPathEntity> photoPathList,
  132. List<AssetEntity> pickedAssetList,
  133. bool isFullImage = false,
  134. void Function(PickedEntity value) onSelected,
  135. }) {
  136. sortDelegate ??= SortDelegate.common;
  137. loadingDelegate ??= DefaultLoadingDelegate();
  138. var options = ListOptions(
  139. padding: padding,
  140. maxSelected: maxSelected,
  141. itemRadio: itemRadio,
  142. theme: theme ?? PhotoTheme.fallback(context),
  143. thumbSize: thumbSize,
  144. sortDelegate: sortDelegate,
  145. checkBoxBuilderDelegate: DefaultCheckBoxBuilderDelegate(),
  146. loadingDelegate: loadingDelegate,
  147. badgeDelegate: DefaultBadgeDelegate(),
  148. pickType: pickType,
  149. );
  150. assert(provider != null, "provider must be not null");
  151. assert(context != null, "context must be not null");
  152. assert(pickType != null, "pickType must be not null");
  153. return ChangeNotifierProvider(
  154. create: (_) => PhotoProvider(
  155. provider: provider,
  156. options: options,
  157. pickedAssetList: pickedAssetList,
  158. ),
  159. child: PhotoListPage(
  160. onSelected: onSelected,
  161. options: options,
  162. isFullImage: isFullImage,
  163. ),
  164. );
  165. }
  166. Future<void> requestPermission(BuildContext context, {@required I18nProvider provider}) async {
  167. var requestPermission = await PhotoManager.requestPermission();
  168. if (requestPermission != true) {
  169. var result = await showDialog(
  170. context: context,
  171. builder: (ctx) => NotPermissionDialog(
  172. provider.getNotPermissionText(),
  173. ),
  174. );
  175. if (result == true) {
  176. PhotoManager.openSetting();
  177. }
  178. return null;
  179. }
  180. }
  181. Future<PickedEntity> _openGalleryContentPage(
  182. BuildContext context,
  183. Options options,
  184. I18nProvider provider,
  185. List<AssetPathEntity> photoList,
  186. List<AssetEntity> pickedAssetList,
  187. ) async {
  188. return Navigator.of(context, rootNavigator: true).push(
  189. MaterialPageRoute(
  190. builder: (ctx) => ChangeNotifierProvider(
  191. create: (_) => PhotoProvider(
  192. provider: provider,
  193. options: options,
  194. pickedAssetList: pickedAssetList,
  195. ),
  196. child: PhotoMainPage(
  197. onClose: (PickedEntity value) {
  198. Navigator.pop(ctx, value);
  199. },
  200. options: options,
  201. photoList: photoList,
  202. ),
  203. ),
  204. ),
  205. );
  206. }
  207. }