app_bloc.dart 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  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-workspace/app_create.pb.dart';
  4. import 'package:flowy_sdk/protobuf/flowy-workspace/errors.pb.dart';
  5. import 'package:flowy_sdk/protobuf/flowy-workspace/view_create.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(
  19. initial: (e) async* {
  20. listener.start(viewsChangeCallback: _handleViewsOrFail);
  21. yield* _fetchViews();
  22. },
  23. createView: (CreateView value) async* {
  24. final viewOrFailed = await appManager.createView(name: value.name, desc: value.desc, viewType: value.viewType);
  25. yield viewOrFailed.fold(
  26. (view) => state.copyWith(
  27. selectedView: view,
  28. successOrFailure: left(unit),
  29. ),
  30. (error) {
  31. Log.error(error);
  32. return state.copyWith(successOrFailure: right(error));
  33. },
  34. );
  35. },
  36. didReceiveViews: (e) async* {
  37. yield state.copyWith(views: e.views);
  38. },
  39. delete: (e) async* {
  40. final result = await appManager.delete();
  41. yield result.fold(
  42. (l) => state.copyWith(successOrFailure: left(unit)),
  43. (error) => state.copyWith(successOrFailure: right(error)),
  44. );
  45. },
  46. rename: (e) async* {
  47. final result = await appManager.rename(e.newName);
  48. yield result.fold(
  49. (l) => state.copyWith(successOrFailure: left(unit)),
  50. (error) => state.copyWith(successOrFailure: right(error)),
  51. );
  52. },
  53. );
  54. }
  55. @override
  56. Future<void> close() async {
  57. await listener.stop();
  58. return super.close();
  59. }
  60. void _handleViewsOrFail(Either<List<View>, WorkspaceError> viewsOrFail) {
  61. viewsOrFail.fold(
  62. (views) => add(AppEvent.didReceiveViews(views)),
  63. (error) {
  64. Log.error(error);
  65. },
  66. );
  67. }
  68. Stream<AppState> _fetchViews() async* {
  69. final viewsOrFailed = await appManager.getViews();
  70. yield viewsOrFailed.fold(
  71. (apps) => state.copyWith(views: apps),
  72. (error) {
  73. Log.error(error);
  74. return state.copyWith(successOrFailure: right(error));
  75. },
  76. );
  77. }
  78. }
  79. @freezed
  80. class AppEvent with _$AppEvent {
  81. const factory AppEvent.initial() = Initial;
  82. const factory AppEvent.createView(String name, String desc, ViewType viewType) = CreateView;
  83. const factory AppEvent.delete() = Delete;
  84. const factory AppEvent.rename(String newName) = Rename;
  85. const factory AppEvent.didReceiveViews(List<View> views) = ReceiveViews;
  86. }
  87. @freezed
  88. class AppState with _$AppState {
  89. const factory AppState({
  90. required App app,
  91. required bool isLoading,
  92. required List<View>? views,
  93. View? selectedView,
  94. required Either<Unit, WorkspaceError> successOrFailure,
  95. }) = _AppState;
  96. factory AppState.initial(App app) => AppState(
  97. app: app,
  98. isLoading: false,
  99. views: null,
  100. selectedView: null,
  101. successOrFailure: left(unit),
  102. );
  103. }