app_bloc.dart 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. import 'package:app_flowy/workspace/domain/i_app.dart';
  2. import 'package:flowy_log/flowy_log.dart';
  3. import 'package:flowy_sdk/protobuf/flowy-core-infra/app_create.pb.dart';
  4. import 'package:flowy_sdk/protobuf/flowy-core-infra/view_create.pb.dart';
  5. import 'package:flowy_sdk/protobuf/flowy-core/errors.pb.dart';
  6. import 'package:freezed_annotation/freezed_annotation.dart';
  7. import 'package:flutter_bloc/flutter_bloc.dart';
  8. import 'package:dartz/dartz.dart';
  9. part 'app_bloc.freezed.dart';
  10. class AppBloc extends Bloc<AppEvent, AppState> {
  11. final IApp appManager;
  12. final IAppListenr listener;
  13. AppBloc({required App app, required this.appManager, required this.listener}) : super(AppState.initial(app));
  14. @override
  15. Stream<AppState> mapEventToState(
  16. AppEvent event,
  17. ) async* {
  18. yield* event.map(initial: (e) async* {
  19. listener.start(
  20. viewsChangeCallback: _handleViewsChanged,
  21. updatedCallback: (app) => add(AppEvent.appDidUpdate(app)),
  22. );
  23. yield* _fetchViews();
  24. }, createView: (CreateView value) async* {
  25. final viewOrFailed = await appManager.createView(name: value.name, desc: value.desc, viewType: value.viewType);
  26. yield viewOrFailed.fold(
  27. (view) => state.copyWith(
  28. latestCreatedView: view,
  29. successOrFailure: left(unit),
  30. ),
  31. (error) {
  32. Log.error(error);
  33. return state.copyWith(successOrFailure: right(error));
  34. },
  35. );
  36. }, didReceiveViews: (e) async* {
  37. yield* handleDidReceiveViews(e.views);
  38. }, delete: (e) async* {
  39. final result = await appManager.delete();
  40. yield result.fold(
  41. (unit) => state.copyWith(successOrFailure: left(unit)),
  42. (error) => state.copyWith(successOrFailure: right(error)),
  43. );
  44. }, rename: (e) async* {
  45. final result = await appManager.rename(e.newName);
  46. yield result.fold(
  47. (l) => state.copyWith(successOrFailure: left(unit)),
  48. (error) => state.copyWith(successOrFailure: right(error)),
  49. );
  50. }, appDidUpdate: (e) async* {
  51. yield state.copyWith(app: e.app);
  52. });
  53. }
  54. @override
  55. Future<void> close() async {
  56. await listener.stop();
  57. return super.close();
  58. }
  59. void _handleViewsChanged(Either<List<View>, WorkspaceError> result) {
  60. result.fold(
  61. (views) => add(AppEvent.didReceiveViews(views)),
  62. (error) {
  63. Log.error(error);
  64. },
  65. );
  66. }
  67. Stream<AppState> handleDidReceiveViews(List<View> views) async* {
  68. final latestCreatedView = state.latestCreatedView;
  69. AppState newState = state.copyWith(views: views);
  70. if (latestCreatedView != null) {
  71. final index = views.indexWhere((element) => element.id == latestCreatedView.id);
  72. if (index == -1) {
  73. newState = newState.copyWith(latestCreatedView: null);
  74. }
  75. }
  76. yield newState;
  77. }
  78. Stream<AppState> _fetchViews() async* {
  79. final viewsOrFailed = await appManager.getViews();
  80. yield viewsOrFailed.fold(
  81. (apps) => state.copyWith(views: apps),
  82. (error) {
  83. Log.error(error);
  84. return state.copyWith(successOrFailure: right(error));
  85. },
  86. );
  87. }
  88. }
  89. @freezed
  90. class AppEvent with _$AppEvent {
  91. const factory AppEvent.initial() = Initial;
  92. const factory AppEvent.createView(String name, String desc, ViewType viewType) = CreateView;
  93. const factory AppEvent.delete() = Delete;
  94. const factory AppEvent.rename(String newName) = Rename;
  95. const factory AppEvent.didReceiveViews(List<View> views) = ReceiveViews;
  96. const factory AppEvent.appDidUpdate(App app) = AppDidUpdate;
  97. }
  98. @freezed
  99. class AppState with _$AppState {
  100. const factory AppState({
  101. required App app,
  102. required bool isLoading,
  103. required List<View>? views,
  104. View? latestCreatedView,
  105. required Either<Unit, WorkspaceError> successOrFailure,
  106. }) = _AppState;
  107. factory AppState.initial(App app) => AppState(
  108. app: app,
  109. isLoading: false,
  110. views: null,
  111. latestCreatedView: null,
  112. successOrFailure: left(unit),
  113. );
  114. }