share_button.dart 4.5 KB

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