share_button.dart 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. import 'package:appflowy/generated/locale_keys.g.dart';
  2. import 'package:appflowy/startup/startup.dart';
  3. import 'package:appflowy/plugins/document/application/share_bloc.dart';
  4. import 'package:appflowy/util/file_picker/file_picker_service.dart';
  5. import 'package:appflowy/workspace/application/view/view_listener.dart';
  6. import 'package:appflowy/workspace/presentation/home/toast.dart';
  7. import 'package:appflowy/workspace/presentation/widgets/pop_up_action.dart';
  8. import 'package:appflowy_backend/protobuf/flowy-document2/entities.pb.dart';
  9. import 'package:appflowy_popover/appflowy_popover.dart';
  10. import 'package:easy_localization/easy_localization.dart';
  11. import 'package:flowy_infra_ui/widget/rounded_button.dart';
  12. import 'package:appflowy_backend/protobuf/flowy-error/errors.pb.dart';
  13. import 'package:appflowy_backend/protobuf/flowy-folder2/view.pb.dart';
  14. import 'package:flutter/material.dart';
  15. import 'package:flutter_bloc/flutter_bloc.dart';
  16. class DocumentShareButton extends StatelessWidget {
  17. const DocumentShareButton({
  18. super.key,
  19. required this.view,
  20. });
  21. final ViewPB view;
  22. @override
  23. Widget build(BuildContext context) {
  24. return BlocProvider(
  25. create: (context) => getIt<DocShareBloc>(param1: view),
  26. child: BlocListener<DocShareBloc, DocShareState>(
  27. listener: (context, state) {
  28. state.mapOrNull(
  29. finish: (state) {
  30. state.successOrFail.fold(
  31. (data) => _handleExportData(context, data),
  32. _handleExportError,
  33. );
  34. },
  35. );
  36. },
  37. child: BlocBuilder<DocShareBloc, DocShareState>(
  38. builder: (context, state) => ConstrainedBox(
  39. constraints: const BoxConstraints.expand(
  40. height: 30,
  41. width: 100,
  42. ),
  43. child: ShareActionList(view: view),
  44. ),
  45. ),
  46. ),
  47. );
  48. }
  49. void _handleExportData(BuildContext context, ExportDataPB exportData) {
  50. switch (exportData.exportType) {
  51. case ExportType.Markdown:
  52. showSnackBarMessage(
  53. context,
  54. LocaleKeys.settings_files_exportFileSuccess.tr(),
  55. );
  56. break;
  57. case ExportType.Link:
  58. case ExportType.Text:
  59. break;
  60. }
  61. }
  62. void _handleExportError(FlowyError error) {
  63. showMessageToast(error.msg);
  64. }
  65. }
  66. class ShareActionList extends StatefulWidget {
  67. const ShareActionList({
  68. super.key,
  69. required this.view,
  70. });
  71. final ViewPB view;
  72. @override
  73. State<ShareActionList> createState() => ShareActionListState();
  74. }
  75. @visibleForTesting
  76. class ShareActionListState extends State<ShareActionList> {
  77. late String name;
  78. late final ViewListener viewListener = ViewListener(viewId: widget.view.id);
  79. @override
  80. void initState() {
  81. super.initState();
  82. listenOnViewUpdated();
  83. }
  84. @override
  85. void dispose() {
  86. viewListener.stop();
  87. super.dispose();
  88. }
  89. @override
  90. Widget build(BuildContext context) {
  91. final docShareBloc = context.read<DocShareBloc>();
  92. return PopoverActionList<ShareActionWrapper>(
  93. direction: PopoverDirection.bottomWithCenterAligned,
  94. offset: const Offset(0, 8),
  95. actions: ShareAction.values
  96. .map((action) => ShareActionWrapper(action))
  97. .toList(),
  98. buildChild: (controller) {
  99. return RoundedTextButton(
  100. title: LocaleKeys.shareAction_buttonText.tr(),
  101. onPressed: () => controller.show(),
  102. );
  103. },
  104. onSelected: (action, controller) async {
  105. switch (action.inner) {
  106. case ShareAction.markdown:
  107. final exportPath = await getIt<FilePickerService>().saveFile(
  108. dialogTitle: '',
  109. fileName: '$name.md',
  110. );
  111. if (exportPath != null) {
  112. docShareBloc.add(DocShareEvent.shareMarkdown(exportPath));
  113. }
  114. break;
  115. }
  116. controller.close();
  117. },
  118. );
  119. }
  120. void listenOnViewUpdated() {
  121. name = widget.view.name;
  122. viewListener.start(
  123. onViewUpdated: (view) {
  124. name = view.name;
  125. },
  126. );
  127. }
  128. }
  129. enum ShareAction {
  130. markdown,
  131. }
  132. class ShareActionWrapper extends ActionCell {
  133. final ShareAction inner;
  134. ShareActionWrapper(this.inner);
  135. Widget? icon(Color iconColor) => null;
  136. @override
  137. String get name {
  138. switch (inner) {
  139. case ShareAction.markdown:
  140. return LocaleKeys.shareAction_markdown.tr();
  141. }
  142. }
  143. }