link_expiry_picker_page.dart 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. import 'package:flutter/material.dart';
  2. import 'package:flutter_datetime_picker/flutter_datetime_picker.dart';
  3. import 'package:photos/ente_theme_data.dart';
  4. import 'package:photos/models/collection.dart';
  5. import 'package:photos/services/collections_service.dart';
  6. import 'package:photos/theme/ente_theme.dart';
  7. import 'package:photos/ui/components/captioned_text_widget.dart';
  8. import 'package:photos/ui/components/divider_widget.dart';
  9. import 'package:photos/ui/components/menu_item_widget/menu_item_widget.dart';
  10. import 'package:photos/ui/components/title_bar_title_widget.dart';
  11. import 'package:photos/ui/components/title_bar_widget.dart';
  12. import 'package:photos/utils/dialog_util.dart';
  13. import 'package:photos/utils/separators_util.dart';
  14. import 'package:tuple/tuple.dart';
  15. class LinkExpiryPickerPage extends StatelessWidget {
  16. final Collection collection;
  17. const LinkExpiryPickerPage(this.collection, {super.key});
  18. @override
  19. Widget build(BuildContext context) {
  20. return Scaffold(
  21. body: CustomScrollView(
  22. primary: false,
  23. slivers: <Widget>[
  24. const TitleBarWidget(
  25. flexibleSpaceTitle: TitleBarTitleWidget(
  26. title: "Link expiry",
  27. ),
  28. ),
  29. SliverList(
  30. delegate: SliverChildBuilderDelegate(
  31. (context, index) {
  32. return Padding(
  33. padding: const EdgeInsets.symmetric(
  34. horizontal: 16,
  35. vertical: 20,
  36. ),
  37. child: Column(
  38. mainAxisSize: MainAxisSize.min,
  39. children: [
  40. ClipRRect(
  41. borderRadius:
  42. const BorderRadius.all(Radius.circular(8)),
  43. child: ItemsWidget(collection),
  44. ),
  45. ],
  46. ),
  47. );
  48. },
  49. childCount: 1,
  50. ),
  51. ),
  52. const SliverPadding(padding: EdgeInsets.symmetric(vertical: 12)),
  53. ],
  54. ),
  55. );
  56. }
  57. }
  58. class ItemsWidget extends StatelessWidget {
  59. final Collection collection;
  60. ItemsWidget(this.collection, {super.key});
  61. // index, title, milliseconds in future post which link should expire (when >0)
  62. final List<Tuple2<String, int>> _expiryOptions = [
  63. const Tuple2("Never", 0),
  64. Tuple2("After 1 hour", const Duration(hours: 1).inMicroseconds),
  65. Tuple2("After 1 day", const Duration(days: 1).inMicroseconds),
  66. Tuple2("After 1 week", const Duration(days: 7).inMicroseconds),
  67. // todo: make this time calculation perfect
  68. Tuple2("After 1 month", const Duration(days: 30).inMicroseconds),
  69. Tuple2("After 1 year", const Duration(days: 365).inMicroseconds),
  70. const Tuple2("Custom", -1),
  71. ];
  72. @override
  73. Widget build(BuildContext context) {
  74. List<Widget> items = [];
  75. for (Tuple2<String, int> expiryOpiton in _expiryOptions) {
  76. items.add(
  77. _menuItemForPicker(context, expiryOpiton),
  78. );
  79. }
  80. items = addSeparators(
  81. items,
  82. DividerWidget(
  83. dividerType: DividerType.menuNoIcon,
  84. bgColor: getEnteColorScheme(context).fillFaint,
  85. ),
  86. );
  87. return Column(
  88. mainAxisSize: MainAxisSize.min,
  89. children: items,
  90. );
  91. }
  92. Widget _menuItemForPicker(
  93. BuildContext context,
  94. Tuple2<String, int> expiryOpiton,
  95. ) {
  96. return MenuItemWidget(
  97. menuItemColor: getEnteColorScheme(context).fillFaint,
  98. captionedTextWidget: CaptionedTextWidget(
  99. title: expiryOpiton.item1,
  100. ),
  101. alignCaptionedTextToLeft: true,
  102. isTopBorderRadiusRemoved: true,
  103. isBottomBorderRadiusRemoved: true,
  104. alwaysShowSuccessState: true,
  105. surfaceExecutionStates: expiryOpiton.item2 == -1 ? false : true,
  106. onTap: () async {
  107. int newValidTill = -1;
  108. final int expireAfterInMicroseconds = expiryOpiton.item2;
  109. // need to manually select time
  110. if (expireAfterInMicroseconds < 0) {
  111. final timeInMicrosecondsFromEpoch =
  112. await _showDateTimePicker(context);
  113. if (timeInMicrosecondsFromEpoch != null) {
  114. newValidTill = timeInMicrosecondsFromEpoch;
  115. }
  116. } else if (expireAfterInMicroseconds == 0) {
  117. // no expiry
  118. newValidTill = 0;
  119. } else {
  120. newValidTill =
  121. DateTime.now().microsecondsSinceEpoch + expireAfterInMicroseconds;
  122. }
  123. if (newValidTill >= 0) {
  124. debugPrint("Setting expirty $newValidTill");
  125. await updateTime(newValidTill, context);
  126. }
  127. },
  128. );
  129. }
  130. // _showDateTimePicker return null if user doesn't select date-time
  131. Future<int?> _showDateTimePicker(BuildContext context) async {
  132. final dateResult = await DatePicker.showDatePicker(
  133. context,
  134. minTime: DateTime.now(),
  135. currentTime: DateTime.now(),
  136. locale: LocaleType.en,
  137. theme: Theme.of(context).colorScheme.dateTimePickertheme,
  138. );
  139. if (dateResult == null) {
  140. return null;
  141. }
  142. final dateWithTimeResult = await DatePicker.showTime12hPicker(
  143. context,
  144. showTitleActions: true,
  145. currentTime: dateResult,
  146. locale: LocaleType.en,
  147. theme: Theme.of(context).colorScheme.dateTimePickertheme,
  148. );
  149. if (dateWithTimeResult == null) {
  150. return null;
  151. } else {
  152. return dateWithTimeResult.microsecondsSinceEpoch;
  153. }
  154. }
  155. Future<void> updateTime(int newValidTill, BuildContext context) async {
  156. await _updateUrlSettings(
  157. context,
  158. {'validTill': newValidTill},
  159. );
  160. }
  161. Future<void> _updateUrlSettings(
  162. BuildContext context,
  163. Map<String, dynamic> prop,
  164. ) async {
  165. try {
  166. await CollectionsService.instance.updateShareUrl(collection, prop);
  167. } catch (e) {
  168. showGenericErrorDialog(context: context);
  169. rethrow;
  170. }
  171. }
  172. }