doc_bloc.dart 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. import 'dart:convert';
  2. import 'package:app_flowy/workspace/infrastructure/repos/doc_repo.dart';
  3. import 'package:app_flowy/workspace/infrastructure/repos/trash_repo.dart';
  4. import 'package:app_flowy/workspace/infrastructure/repos/view_repo.dart';
  5. import 'package:flowy_sdk/protobuf/flowy-folder-data-model/trash.pb.dart';
  6. import 'package:flowy_sdk/protobuf/flowy-folder-data-model/view.pb.dart';
  7. import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
  8. import 'package:flutter_quill/flutter_quill.dart';
  9. import 'package:flowy_sdk/log.dart';
  10. import 'package:flutter_bloc/flutter_bloc.dart';
  11. import 'package:freezed_annotation/freezed_annotation.dart';
  12. import 'package:dartz/dartz.dart';
  13. import 'dart:async';
  14. part 'doc_bloc.freezed.dart';
  15. class DocBloc extends Bloc<DocEvent, DocState> {
  16. final View view;
  17. final DocRepository repo;
  18. final ViewListener listener;
  19. final TrashRepo trashRepo;
  20. late Document document;
  21. StreamSubscription? _subscription;
  22. DocBloc({
  23. required this.view,
  24. required this.repo,
  25. required this.listener,
  26. required this.trashRepo,
  27. }) : super(DocState.initial()) {
  28. on<DocEvent>((event, emit) async {
  29. await event.map(
  30. initial: (Initial value) async {
  31. await _initial(value, emit);
  32. },
  33. deleted: (Deleted value) async {
  34. emit(state.copyWith(isDeleted: true));
  35. },
  36. restore: (Restore value) async {
  37. emit(state.copyWith(isDeleted: false));
  38. },
  39. deletePermanently: (DeletePermanently value) async {
  40. final result = await trashRepo.deleteViews([Tuple2(view.id, TrashType.TrashView)]);
  41. final newState = result.fold((l) => state.copyWith(forceClose: true), (r) => state);
  42. emit(newState);
  43. },
  44. restorePage: (RestorePage value) async {
  45. final result = await trashRepo.putback(view.id);
  46. final newState = result.fold((l) => state.copyWith(isDeleted: false), (r) => state);
  47. emit(newState);
  48. },
  49. );
  50. });
  51. }
  52. @override
  53. Future<void> close() async {
  54. await listener.close();
  55. if (_subscription != null) {
  56. await _subscription?.cancel();
  57. }
  58. repo.closeDoc();
  59. return super.close();
  60. }
  61. Future<void> _initial(Initial value, Emitter<DocState> emit) async {
  62. listener.deletedNotifier.addPublishListener((result) {
  63. result.fold(
  64. (view) => add(const DocEvent.deleted()),
  65. (error) {},
  66. );
  67. });
  68. listener.restoredNotifier.addPublishListener((result) {
  69. result.fold(
  70. (view) => add(const DocEvent.restore()),
  71. (error) {},
  72. );
  73. });
  74. listener.start();
  75. final result = await repo.readDoc();
  76. result.fold(
  77. (doc) {
  78. document = _decodeJsonToDocument(doc.deltaJson);
  79. _subscription = document.changes.listen((event) {
  80. final delta = event.item2;
  81. final documentDelta = document.toDelta();
  82. _composeDelta(delta, documentDelta);
  83. });
  84. emit(state.copyWith(loadState: DocLoadState.finish(left(unit))));
  85. },
  86. (err) {
  87. emit(state.copyWith(loadState: DocLoadState.finish(right(err))));
  88. },
  89. );
  90. }
  91. // Document _decodeListToDocument(Uint8List data) {
  92. // final json = jsonDecode(utf8.decode(data));
  93. // final document = Document.fromJson(json);
  94. // return document;
  95. // }
  96. void _composeDelta(Delta composedDelta, Delta documentDelta) async {
  97. final json = jsonEncode(composedDelta.toJson());
  98. Log.debug("doc_id: $view.id - Send json: $json");
  99. final result = await repo.composeDelta(data: json);
  100. result.fold((rustDoc) {
  101. // final json = utf8.decode(doc.data);
  102. final rustDelta = Delta.fromJson(jsonDecode(rustDoc.deltaJson));
  103. if (documentDelta != rustDelta) {
  104. Log.error("Receive : $rustDelta");
  105. Log.error("Expected : $documentDelta");
  106. }
  107. }, (r) => null);
  108. }
  109. Document _decodeJsonToDocument(String data) {
  110. final json = jsonDecode(data);
  111. final document = Document.fromJson(json);
  112. return document;
  113. }
  114. }
  115. @freezed
  116. class DocEvent with _$DocEvent {
  117. const factory DocEvent.initial() = Initial;
  118. const factory DocEvent.deleted() = Deleted;
  119. const factory DocEvent.restore() = Restore;
  120. const factory DocEvent.restorePage() = RestorePage;
  121. const factory DocEvent.deletePermanently() = DeletePermanently;
  122. }
  123. @freezed
  124. class DocState with _$DocState {
  125. const factory DocState({
  126. required DocLoadState loadState,
  127. required bool isDeleted,
  128. required bool forceClose,
  129. }) = _DocState;
  130. factory DocState.initial() => const DocState(
  131. loadState: _Loading(),
  132. isDeleted: false,
  133. forceClose: false,
  134. );
  135. }
  136. @freezed
  137. class DocLoadState with _$DocLoadState {
  138. const factory DocLoadState.loading() = _Loading;
  139. const factory DocLoadState.finish(Either<Unit, FlowyError> successOrFail) = _Finish;
  140. }