app_bloc.dart 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. import 'dart:collection';
  2. import 'package:appflowy/startup/plugin/plugin.dart';
  3. import 'package:appflowy/startup/startup.dart';
  4. import 'package:appflowy/workspace/application/app/app_listener.dart';
  5. import 'package:appflowy/workspace/application/view/view_service.dart';
  6. import 'package:appflowy/workspace/presentation/home/menu/menu.dart';
  7. import 'package:expandable/expandable.dart';
  8. import 'package:appflowy_backend/log.dart';
  9. import 'package:appflowy_backend/protobuf/flowy-folder2/view.pb.dart';
  10. import 'package:appflowy_backend/protobuf/flowy-error/errors.pb.dart';
  11. import 'package:flutter/foundation.dart';
  12. import 'package:freezed_annotation/freezed_annotation.dart';
  13. import 'package:flutter_bloc/flutter_bloc.dart';
  14. import 'package:dartz/dartz.dart';
  15. part 'app_bloc.freezed.dart';
  16. class AppBloc extends Bloc<AppEvent, AppState> {
  17. final ViewBackendService appService;
  18. final AppListener appListener;
  19. AppBloc({required ViewPB view})
  20. : appService = ViewBackendService(),
  21. appListener = AppListener(viewId: view.id),
  22. super(AppState.initial(view)) {
  23. on<AppEvent>((event, emit) async {
  24. await event.map(
  25. initial: (e) async {
  26. _startListening();
  27. await _loadViews(emit);
  28. },
  29. createView: (CreateView value) async {
  30. await _createView(value, emit);
  31. },
  32. loadViews: (_) async {
  33. await _loadViews(emit);
  34. },
  35. delete: (e) async {
  36. await _deleteApp(emit);
  37. },
  38. deleteView: (deletedView) async {
  39. await _deleteView(emit, deletedView.viewId);
  40. },
  41. rename: (e) async {
  42. await _renameView(e, emit);
  43. },
  44. appDidUpdate: (e) async {
  45. final latestCreatedView = state.latestCreatedView;
  46. final views = e.app.childViews;
  47. AppState newState = state.copyWith(
  48. views: views,
  49. view: e.app,
  50. );
  51. if (latestCreatedView != null) {
  52. final index = views
  53. .indexWhere((element) => element.id == latestCreatedView.id);
  54. if (index == -1) {
  55. newState = newState.copyWith(latestCreatedView: null);
  56. }
  57. emit(newState);
  58. }
  59. emit(newState);
  60. },
  61. );
  62. });
  63. }
  64. void _startListening() {
  65. appListener.start(
  66. onAppUpdated: (app) {
  67. if (!isClosed) {
  68. add(AppEvent.appDidUpdate(app));
  69. }
  70. },
  71. );
  72. }
  73. Future<void> _renameView(Rename e, Emitter<AppState> emit) async {
  74. final result = await ViewBackendService.updateView(
  75. viewId: state.view.id,
  76. name: e.newName,
  77. );
  78. result.fold(
  79. (l) => emit(state.copyWith(successOrFailure: left(unit))),
  80. (error) => emit(state.copyWith(successOrFailure: right(error))),
  81. );
  82. }
  83. // Delete the current app
  84. Future<void> _deleteApp(Emitter<AppState> emit) async {
  85. final result = await ViewBackendService.delete(viewId: state.view.id);
  86. result.fold(
  87. (unit) => emit(state.copyWith(successOrFailure: left(unit))),
  88. (error) => emit(state.copyWith(successOrFailure: right(error))),
  89. );
  90. }
  91. Future<void> _deleteView(Emitter<AppState> emit, String viewId) async {
  92. final result = await ViewBackendService.deleteView(viewId: viewId);
  93. result.fold(
  94. (unit) => emit(state.copyWith(successOrFailure: left(unit))),
  95. (error) => emit(state.copyWith(successOrFailure: right(error))),
  96. );
  97. }
  98. Future<void> _createView(CreateView value, Emitter<AppState> emit) async {
  99. // create a child view for the current view
  100. final result = await ViewBackendService.createView(
  101. parentViewId: state.view.id,
  102. name: value.name,
  103. desc: value.desc ?? "",
  104. layoutType: value.pluginBuilder.layoutType!,
  105. initialDataBytes: value.initialDataBytes,
  106. ext: value.ext ?? {},
  107. );
  108. result.fold(
  109. (view) => emit(
  110. state.copyWith(
  111. latestCreatedView: value.openAfterCreated ? view : null,
  112. successOrFailure: left(unit),
  113. ),
  114. ),
  115. (error) {
  116. Log.error(error);
  117. emit(state.copyWith(successOrFailure: right(error)));
  118. },
  119. );
  120. }
  121. @override
  122. Future<void> close() async {
  123. await appListener.stop();
  124. return super.close();
  125. }
  126. Future<void> _loadViews(Emitter<AppState> emit) async {
  127. final viewsOrFailed =
  128. await ViewBackendService.getViews(viewId: state.view.id);
  129. viewsOrFailed.fold(
  130. (views) => emit(state.copyWith(views: views)),
  131. (error) {
  132. Log.error(error);
  133. emit(state.copyWith(successOrFailure: right(error)));
  134. },
  135. );
  136. }
  137. }
  138. @freezed
  139. class AppEvent with _$AppEvent {
  140. const factory AppEvent.initial() = Initial;
  141. const factory AppEvent.createView(
  142. String name,
  143. PluginBuilder pluginBuilder, {
  144. String? desc,
  145. /// ~~The initial data should be the JSON of the document~~
  146. /// ~~For example: {"document":{"type":"editor","children":[]}}~~
  147. ///
  148. /// - Document:
  149. /// the initial data should be the string that can be converted into [DocumentDataPB]
  150. ///
  151. List<int>? initialDataBytes,
  152. Map<String, String>? ext,
  153. /// open the view after created
  154. @Default(true) bool openAfterCreated,
  155. }) = CreateView;
  156. const factory AppEvent.loadViews() = LoadApp;
  157. const factory AppEvent.delete() = DeleteApp;
  158. const factory AppEvent.deleteView(String viewId) = DeleteView;
  159. const factory AppEvent.rename(String newName) = Rename;
  160. const factory AppEvent.appDidUpdate(ViewPB app) = AppDidUpdate;
  161. }
  162. @freezed
  163. class AppState with _$AppState {
  164. const factory AppState({
  165. required ViewPB view,
  166. required List<ViewPB> views,
  167. ViewPB? latestCreatedView,
  168. required Either<Unit, FlowyError> successOrFailure,
  169. }) = _AppState;
  170. factory AppState.initial(ViewPB view) => AppState(
  171. view: view,
  172. views: view.childViews,
  173. successOrFailure: left(unit),
  174. );
  175. }
  176. class AppViewDataContext extends ChangeNotifier {
  177. final String viewId;
  178. final ValueNotifier<List<ViewPB>> _viewsNotifier = ValueNotifier([]);
  179. final ValueNotifier<ViewPB?> _selectedViewNotifier = ValueNotifier(null);
  180. VoidCallback? _menuSharedStateListener;
  181. ExpandableController expandController =
  182. ExpandableController(initialExpanded: false);
  183. AppViewDataContext({required this.viewId}) {
  184. _setLatestView(getIt<MenuSharedState>().latestOpenView);
  185. _menuSharedStateListener =
  186. getIt<MenuSharedState>().addLatestViewListener((view) {
  187. _setLatestView(view);
  188. });
  189. }
  190. VoidCallback addSelectedViewChangeListener(void Function(ViewPB?) callback) {
  191. listener() {
  192. callback(_selectedViewNotifier.value);
  193. }
  194. _selectedViewNotifier.addListener(listener);
  195. return listener;
  196. }
  197. void removeSelectedViewListener(VoidCallback listener) {
  198. _selectedViewNotifier.removeListener(listener);
  199. }
  200. void _setLatestView(ViewPB? view) {
  201. view?.freeze();
  202. if (_selectedViewNotifier.value != view) {
  203. _selectedViewNotifier.value = view;
  204. _expandIfNeed();
  205. notifyListeners();
  206. }
  207. }
  208. ViewPB? get selectedView => _selectedViewNotifier.value;
  209. set views(List<ViewPB> views) {
  210. if (_viewsNotifier.value != views) {
  211. _viewsNotifier.value = views;
  212. _expandIfNeed();
  213. notifyListeners();
  214. }
  215. }
  216. UnmodifiableListView<ViewPB> get views =>
  217. UnmodifiableListView(_viewsNotifier.value);
  218. VoidCallback addViewsChangeListener(
  219. void Function(UnmodifiableListView<ViewPB>) callback,
  220. ) {
  221. listener() {
  222. callback(views);
  223. }
  224. _viewsNotifier.addListener(listener);
  225. return listener;
  226. }
  227. void removeViewsListener(VoidCallback listener) {
  228. _viewsNotifier.removeListener(listener);
  229. }
  230. void _expandIfNeed() {
  231. if (_selectedViewNotifier.value == null) {
  232. return;
  233. }
  234. if (!_viewsNotifier.value.contains(_selectedViewNotifier.value)) {
  235. return;
  236. }
  237. if (expandController.expanded == false) {
  238. // Workaround: Delay 150 milliseconds to make the smooth animation while expanding
  239. Future.delayed(const Duration(milliseconds: 150), () {
  240. expandController.expanded = true;
  241. });
  242. }
  243. }
  244. @override
  245. void dispose() {
  246. if (_menuSharedStateListener != null) {
  247. getIt<MenuSharedState>()
  248. .removeLatestViewListener(_menuSharedStateListener!);
  249. }
  250. super.dispose();
  251. }
  252. }