app_service.dart 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. import 'dart:async';
  2. import 'dart:convert';
  3. import 'package:appflowy_backend/protobuf/flowy-folder/workspace.pb.dart';
  4. import 'package:dartz/dartz.dart';
  5. import 'package:appflowy_backend/dispatch/dispatch.dart';
  6. import 'package:appflowy_backend/protobuf/flowy-error/errors.pb.dart';
  7. import 'package:appflowy_backend/protobuf/flowy-folder/app.pb.dart';
  8. import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
  9. class AppBackendService {
  10. Future<Either<AppPB, FlowyError>> readApp({required String appId}) {
  11. final payload = AppIdPB.create()..value = appId;
  12. return FolderEventReadApp(payload).send();
  13. }
  14. Future<Either<ViewPB, FlowyError>> createView({
  15. required String appId,
  16. required String name,
  17. String? desc,
  18. required ViewLayoutTypePB layoutType,
  19. /// The initial data should be the JSON of the document.
  20. /// Currently, only support create document with initial data.
  21. ///
  22. /// The initial data must be follow this format as shown below.
  23. /// {"document":{"type":"editor","children":[]}}
  24. String? initialData,
  25. /// The [ext] is used to pass through the custom configuration
  26. /// to the backend.
  27. /// Linking the view to the existing database, it needs to pass
  28. /// the database id. For example: "database_id": "xxx"
  29. ///
  30. Map<String, String> ext = const {},
  31. }) {
  32. final payload = CreateViewPayloadPB.create()
  33. ..belongToId = appId
  34. ..name = name
  35. ..desc = desc ?? ""
  36. ..layout = layoutType
  37. ..initialData = utf8.encode(
  38. initialData ?? "",
  39. );
  40. if (ext.isNotEmpty) {
  41. payload.ext.addAll(ext);
  42. }
  43. return FolderEventCreateView(payload).send();
  44. }
  45. Future<Either<List<ViewPB>, FlowyError>> getViews({required String appId}) {
  46. final payload = AppIdPB.create()..value = appId;
  47. return FolderEventReadApp(payload).send().then((result) {
  48. return result.fold(
  49. (app) => left(app.belongings.items),
  50. (error) => right(error),
  51. );
  52. });
  53. }
  54. Future<Either<Unit, FlowyError>> delete({required String appId}) {
  55. final request = AppIdPB.create()..value = appId;
  56. return FolderEventDeleteApp(request).send();
  57. }
  58. Future<Either<Unit, FlowyError>> deleteView({required String viewId}) {
  59. final request = RepeatedViewIdPB.create()..items.add(viewId);
  60. return FolderEventDeleteView(request).send();
  61. }
  62. Future<Either<Unit, FlowyError>> updateApp(
  63. {required String appId, String? name}) {
  64. UpdateAppPayloadPB payload = UpdateAppPayloadPB.create()..appId = appId;
  65. if (name != null) {
  66. payload.name = name;
  67. }
  68. return FolderEventUpdateApp(payload).send();
  69. }
  70. Future<Either<Unit, FlowyError>> moveView({
  71. required String viewId,
  72. required int fromIndex,
  73. required int toIndex,
  74. }) {
  75. final payload = MoveFolderItemPayloadPB.create()
  76. ..itemId = viewId
  77. ..from = fromIndex
  78. ..to = toIndex
  79. ..ty = MoveFolderItemType.MoveView;
  80. return FolderEventMoveItem(payload).send();
  81. }
  82. Future<List<Tuple2<AppPB, List<ViewPB>>>> fetchViews(
  83. ViewLayoutTypePB layoutType) async {
  84. final result = <Tuple2<AppPB, List<ViewPB>>>[];
  85. return FolderEventReadCurrentWorkspace().send().then((value) async {
  86. final workspaces = value.getLeftOrNull<WorkspaceSettingPB>();
  87. if (workspaces != null) {
  88. final apps = workspaces.workspace.apps.items;
  89. for (var app in apps) {
  90. final views = await getViews(appId: app.id).then(
  91. (value) => value
  92. .getLeftOrNull<List<ViewPB>>()
  93. ?.where((e) => e.layout == layoutType)
  94. .toList(),
  95. );
  96. if (views != null && views.isNotEmpty) {
  97. result.add(Tuple2(app, views));
  98. }
  99. }
  100. }
  101. return result;
  102. });
  103. }
  104. Future<Either<ViewPB, FlowyError>> getView(
  105. String appID,
  106. String viewID,
  107. ) async {
  108. final payload = AppIdPB.create()..value = appID;
  109. return FolderEventReadApp(payload).send().then((result) {
  110. return result.fold(
  111. (app) => left(
  112. app.belongings.items.firstWhere((e) => e.id == viewID),
  113. ),
  114. (error) => right(error),
  115. );
  116. });
  117. }
  118. }
  119. extension AppFlowy on Either {
  120. T? getLeftOrNull<T>() {
  121. if (isLeft()) {
  122. final result = fold<T?>((l) => l, (r) => null);
  123. return result;
  124. }
  125. return null;
  126. }
  127. }