backup_folder_selection_page.dart 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. import 'package:flutter/material.dart';
  2. import 'package:photos/core/configuration.dart';
  3. import 'package:photos/core/event_bus.dart';
  4. import 'package:photos/db/files_db.dart';
  5. import 'package:photos/events/backup_folders_updated_event.dart';
  6. import 'package:photos/models/file.dart';
  7. import 'package:photos/ui/common_elements.dart';
  8. import 'package:photos/ui/loading_widget.dart';
  9. import 'package:photos/ui/thumbnail_widget.dart';
  10. class BackupFolderSelectionPage extends StatefulWidget {
  11. final bool shouldSelectAll;
  12. final String buttonText;
  13. const BackupFolderSelectionPage({
  14. @required this.buttonText,
  15. this.shouldSelectAll = false,
  16. Key key,
  17. }) : super(key: key);
  18. @override
  19. _BackupFolderSelectionPageState createState() =>
  20. _BackupFolderSelectionPageState();
  21. }
  22. class _BackupFolderSelectionPageState extends State<BackupFolderSelectionPage> {
  23. final Set<String> _allFolders = Set<String>();
  24. Set<String> _backedupFolders = Set<String>();
  25. List<File> _latestFiles;
  26. bool _isSelectAll = false;
  27. @override
  28. void initState() {
  29. _backedupFolders = Configuration.instance.getPathsToBackUp();
  30. FilesDB.instance.getLatestLocalFiles().then((files) {
  31. setState(() {
  32. _latestFiles = files;
  33. for (final file in _latestFiles) {
  34. _allFolders.add(file.deviceFolder);
  35. }
  36. if (widget.shouldSelectAll) {
  37. _backedupFolders.addAll(_allFolders);
  38. }
  39. _isSelectAll = _backedupFolders.length == _allFolders.length;
  40. });
  41. });
  42. super.initState();
  43. }
  44. @override
  45. Widget build(BuildContext context) {
  46. return Scaffold(
  47. appBar: AppBar(title: Text("select folders to backup")),
  48. body: Column(
  49. mainAxisAlignment: MainAxisAlignment.center,
  50. children: [
  51. Padding(
  52. padding: EdgeInsets.all(12),
  53. ),
  54. Padding(
  55. padding: const EdgeInsets.only(left: 32, right: 32),
  56. child: Text(
  57. "the selected folders will be end-to-end encrypted and backed up",
  58. style: TextStyle(
  59. color: Colors.white.withOpacity(0.36),
  60. fontSize: 14,
  61. height: 1.3,
  62. ),
  63. textAlign: TextAlign.center,
  64. ),
  65. ),
  66. Padding(
  67. padding: EdgeInsets.all(6),
  68. ),
  69. GestureDetector(
  70. behavior: HitTestBehavior.translucent,
  71. child: Padding(
  72. padding: const EdgeInsets.fromLTRB(6, 6, 64, 6),
  73. child: Align(
  74. alignment: Alignment.centerRight,
  75. child: Text(
  76. _isSelectAll ? "unselect all" : "select all",
  77. textAlign: TextAlign.right,
  78. style: TextStyle(
  79. fontSize: 12,
  80. color: Colors.white.withOpacity(0.8),
  81. ),
  82. ),
  83. ),
  84. ),
  85. onTap: () {
  86. _isSelectAll = !_isSelectAll;
  87. if (_isSelectAll) {
  88. _backedupFolders.addAll(_allFolders);
  89. } else {
  90. _backedupFolders.clear();
  91. }
  92. setState(() {});
  93. }),
  94. Expanded(child: _getFolderList()),
  95. Padding(
  96. padding: EdgeInsets.all(20),
  97. ),
  98. Hero(
  99. tag: "select_folders",
  100. child: Container(
  101. padding: EdgeInsets.only(left: 60, right: 60, bottom: 32),
  102. child: button(
  103. widget.buttonText,
  104. fontSize: 18,
  105. onPressed: _backedupFolders.length == 0
  106. ? null
  107. : () async {
  108. await Configuration.instance
  109. .setPathsToBackUp(_backedupFolders);
  110. Bus.instance.fire(BackupFoldersUpdatedEvent());
  111. Navigator.of(context).pop();
  112. },
  113. padding: EdgeInsets.fromLTRB(60, 20, 60, 20),
  114. ),
  115. ),
  116. ),
  117. ],
  118. ),
  119. );
  120. }
  121. Widget _getFolderList() {
  122. Widget child;
  123. if (_latestFiles != null) {
  124. _latestFiles.sort((first, second) {
  125. return first.deviceFolder
  126. .toLowerCase()
  127. .compareTo(second.deviceFolder.toLowerCase());
  128. });
  129. final List<Widget> foldersWidget = [];
  130. for (final file in _latestFiles) {
  131. final isSelected = _backedupFolders.contains(file.deviceFolder);
  132. foldersWidget.add(
  133. Padding(
  134. padding: const EdgeInsets.only(bottom: 1, right: 4),
  135. child: Container(
  136. decoration: BoxDecoration(
  137. border: Border.all(
  138. color: Colors.black,
  139. ),
  140. borderRadius: BorderRadius.all(
  141. Radius.circular(10),
  142. ),
  143. color: isSelected
  144. ? Color.fromRGBO(16, 32, 32, 1)
  145. : Color.fromRGBO(8, 18, 18, 0.4),
  146. ),
  147. padding: EdgeInsets.fromLTRB(20, 16, 20, 16),
  148. child: InkWell(
  149. child: Row(
  150. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  151. children: [
  152. Container(
  153. child: Expanded(
  154. child: Row(
  155. children: [
  156. _getThumbnail(file),
  157. Padding(padding: EdgeInsets.all(10)),
  158. Expanded(
  159. child: Text(
  160. file.deviceFolder,
  161. style: TextStyle(fontSize: 16, height: 1.5),
  162. overflow: TextOverflow.clip,
  163. ),
  164. ),
  165. ],
  166. ),
  167. ),
  168. ),
  169. Checkbox(
  170. value: isSelected,
  171. onChanged: (value) {
  172. if (value) {
  173. _backedupFolders.add(file.deviceFolder);
  174. } else {
  175. _backedupFolders.remove(file.deviceFolder);
  176. }
  177. setState(() {});
  178. },
  179. ),
  180. ],
  181. ),
  182. onTap: () {
  183. final value = !_backedupFolders.contains(file.deviceFolder);
  184. if (value) {
  185. _backedupFolders.add(file.deviceFolder);
  186. } else {
  187. _backedupFolders.remove(file.deviceFolder);
  188. }
  189. setState(() {});
  190. },
  191. ),
  192. ),
  193. ),
  194. );
  195. }
  196. final scrollController = ScrollController();
  197. child = Scrollbar(
  198. isAlwaysShown: true,
  199. controller: scrollController,
  200. child: SingleChildScrollView(
  201. controller: scrollController,
  202. child: Container(
  203. padding: EdgeInsets.only(right: 4),
  204. child: Column(
  205. crossAxisAlignment: CrossAxisAlignment.stretch,
  206. children: foldersWidget,
  207. ),
  208. ),
  209. ),
  210. );
  211. } else {
  212. child = loadWidget;
  213. }
  214. return Container(
  215. padding: EdgeInsets.only(left: 40, right: 40),
  216. child: child,
  217. );
  218. }
  219. Widget _getThumbnail(File file) {
  220. return ClipRRect(
  221. borderRadius: BorderRadius.circular(4.0),
  222. child: Container(
  223. child: ThumbnailWidget(
  224. file,
  225. shouldShowSyncStatus: false,
  226. key: Key("backup_selection_widget" + file.tag()),
  227. ),
  228. height: 60,
  229. width: 60,
  230. ),
  231. );
  232. }
  233. }