app_service.dart 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  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,
  64. String? name,
  65. }) {
  66. UpdateAppPayloadPB payload = UpdateAppPayloadPB.create()..appId = appId;
  67. if (name != null) {
  68. payload.name = name;
  69. }
  70. return FolderEventUpdateApp(payload).send();
  71. }
  72. Future<Either<Unit, FlowyError>> moveView({
  73. required String viewId,
  74. required int fromIndex,
  75. required int toIndex,
  76. }) {
  77. final payload = MoveFolderItemPayloadPB.create()
  78. ..itemId = viewId
  79. ..from = fromIndex
  80. ..to = toIndex
  81. ..ty = MoveFolderItemType.MoveView;
  82. return FolderEventMoveItem(payload).send();
  83. }
  84. Future<List<Tuple2<AppPB, List<ViewPB>>>> fetchViews(
  85. ViewLayoutTypePB layoutType,
  86. ) async {
  87. final result = <Tuple2<AppPB, List<ViewPB>>>[];
  88. return FolderEventReadCurrentWorkspace().send().then((value) async {
  89. final workspaces = value.getLeftOrNull<WorkspaceSettingPB>();
  90. if (workspaces != null) {
  91. final apps = workspaces.workspace.apps.items;
  92. for (var app in apps) {
  93. final views = await getViews(appId: app.id).then(
  94. (value) => value
  95. .getLeftOrNull<List<ViewPB>>()
  96. ?.where((e) => e.layout == layoutType)
  97. .toList(),
  98. );
  99. if (views != null && views.isNotEmpty) {
  100. result.add(Tuple2(app, views));
  101. }
  102. }
  103. }
  104. return result;
  105. });
  106. }
  107. Future<Either<ViewPB, FlowyError>> getView(
  108. String appID,
  109. String viewID,
  110. ) async {
  111. final payload = AppIdPB.create()..value = appID;
  112. return FolderEventReadApp(payload).send().then((result) {
  113. return result.fold(
  114. (app) => left(
  115. app.belongings.items.firstWhere((e) => e.id == viewID),
  116. ),
  117. (error) => right(error),
  118. );
  119. });
  120. }
  121. }
  122. extension AppFlowy on Either {
  123. T? getLeftOrNull<T>() {
  124. if (isLeft()) {
  125. final result = fold<T?>((l) => l, (r) => null);
  126. return result;
  127. }
  128. return null;
  129. }
  130. }