device_limit_picker_page.dart 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. import 'package:flutter/material.dart';
  2. import 'package:photos/core/constants.dart';
  3. import 'package:photos/models/collection.dart';
  4. import 'package:photos/services/collections_service.dart';
  5. import 'package:photos/theme/ente_theme.dart';
  6. import 'package:photos/ui/components/captioned_text_widget.dart';
  7. import 'package:photos/ui/components/divider_widget.dart';
  8. import 'package:photos/ui/components/menu_item_widget/menu_item_widget.dart';
  9. import 'package:photos/ui/components/menu_section_description_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. class DeviceLimitPickerPage extends StatelessWidget {
  15. final Collection collection;
  16. const DeviceLimitPickerPage(this.collection, {super.key});
  17. @override
  18. Widget build(BuildContext context) {
  19. return Scaffold(
  20. body: CustomScrollView(
  21. primary: false,
  22. slivers: <Widget>[
  23. const TitleBarWidget(
  24. flexibleSpaceTitle: TitleBarTitleWidget(
  25. title: "Device Limit",
  26. ),
  27. ),
  28. SliverList(
  29. delegate: SliverChildBuilderDelegate(
  30. (context, index) {
  31. return Padding(
  32. padding: const EdgeInsets.symmetric(
  33. horizontal: 16,
  34. vertical: 20,
  35. ),
  36. child: Column(
  37. mainAxisSize: MainAxisSize.min,
  38. children: [
  39. ClipRRect(
  40. borderRadius:
  41. const BorderRadius.all(Radius.circular(8)),
  42. child: ItemsWidget(collection),
  43. ),
  44. const MenuSectionDescriptionWidget(
  45. content:
  46. "When set to the maximum (50), the device limit will be relaxed"
  47. " to allow for temporary spikes of large number of viewers.",
  48. )
  49. ],
  50. ),
  51. );
  52. },
  53. childCount: 1,
  54. ),
  55. ),
  56. const SliverPadding(padding: EdgeInsets.symmetric(vertical: 12)),
  57. ],
  58. ),
  59. );
  60. }
  61. }
  62. class ItemsWidget extends StatefulWidget {
  63. final Collection collection;
  64. const ItemsWidget(this.collection, {super.key});
  65. @override
  66. State<ItemsWidget> createState() => _ItemsWidgetState();
  67. }
  68. class _ItemsWidgetState extends State<ItemsWidget> {
  69. late int currentDeviceLimit;
  70. late int initialDeviceLimit;
  71. List<Widget> items = [];
  72. bool isCustomLimit = false;
  73. @override
  74. void initState() {
  75. currentDeviceLimit = widget.collection.publicURLs!.first!.deviceLimit;
  76. initialDeviceLimit = currentDeviceLimit;
  77. if (!publicLinkDeviceLimits.contains(currentDeviceLimit)) {
  78. isCustomLimit = true;
  79. }
  80. super.initState();
  81. }
  82. @override
  83. Widget build(BuildContext context) {
  84. items.clear();
  85. if (isCustomLimit) {
  86. items.add(
  87. _menuItemForPicker(initialDeviceLimit),
  88. );
  89. }
  90. for (int deviceLimit in publicLinkDeviceLimits) {
  91. items.add(
  92. _menuItemForPicker(deviceLimit),
  93. );
  94. }
  95. items = addSeparators(
  96. items,
  97. DividerWidget(
  98. dividerType: DividerType.menuNoIcon,
  99. bgColor: getEnteColorScheme(context).fillFaint,
  100. ),
  101. );
  102. return Column(
  103. mainAxisSize: MainAxisSize.min,
  104. children: items,
  105. );
  106. }
  107. Widget _menuItemForPicker(int deviceLimit) {
  108. return MenuItemWidget(
  109. key: ValueKey(deviceLimit),
  110. menuItemColor: getEnteColorScheme(context).fillFaint,
  111. captionedTextWidget: CaptionedTextWidget(
  112. title: "$deviceLimit",
  113. ),
  114. trailingIcon: currentDeviceLimit == deviceLimit ? Icons.check : null,
  115. alignCaptionedTextToLeft: true,
  116. isTopBorderRadiusRemoved: true,
  117. isBottomBorderRadiusRemoved: true,
  118. showOnlyLoadingState: true,
  119. onTap: () async {
  120. await _updateUrlSettings(context, {
  121. 'deviceLimit': deviceLimit,
  122. }).then(
  123. (value) => setState(() {
  124. currentDeviceLimit = deviceLimit;
  125. }),
  126. );
  127. },
  128. );
  129. }
  130. Future<void> _updateUrlSettings(
  131. BuildContext context,
  132. Map<String, dynamic> prop,
  133. ) async {
  134. try {
  135. await CollectionsService.instance.updateShareUrl(widget.collection, prop);
  136. } catch (e) {
  137. showGenericErrorDialog(context: context);
  138. rethrow;
  139. }
  140. }
  141. }