app_service.dart 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  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 AppService {
  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 doucment
  20. /// For example: {"document":{"type":"editor","children":[]}}
  21. String? initialData,
  22. Map<String, String> ext = const {},
  23. }) {
  24. final payload = CreateViewPayloadPB.create()
  25. ..belongToId = appId
  26. ..name = name
  27. ..desc = desc ?? ""
  28. ..layout = layoutType
  29. ..initialData = utf8.encode(
  30. initialData ?? "",
  31. );
  32. if (ext.isNotEmpty) {
  33. payload.ext.addAll(ext);
  34. }
  35. return FolderEventCreateView(payload).send();
  36. }
  37. Future<Either<List<ViewPB>, FlowyError>> getViews({required String appId}) {
  38. final payload = AppIdPB.create()..value = appId;
  39. return FolderEventReadApp(payload).send().then((result) {
  40. return result.fold(
  41. (app) => left(app.belongings.items),
  42. (error) => right(error),
  43. );
  44. });
  45. }
  46. Future<Either<Unit, FlowyError>> delete({required String appId}) {
  47. final request = AppIdPB.create()..value = appId;
  48. return FolderEventDeleteApp(request).send();
  49. }
  50. Future<Either<Unit, FlowyError>> deleteView({required String viewId}) {
  51. final request = RepeatedViewIdPB.create()..items.add(viewId);
  52. return FolderEventDeleteView(request).send();
  53. }
  54. Future<Either<Unit, FlowyError>> updateApp(
  55. {required String appId, String? name}) {
  56. UpdateAppPayloadPB payload = UpdateAppPayloadPB.create()..appId = appId;
  57. if (name != null) {
  58. payload.name = name;
  59. }
  60. return FolderEventUpdateApp(payload).send();
  61. }
  62. Future<Either<Unit, FlowyError>> moveView({
  63. required String viewId,
  64. required int fromIndex,
  65. required int toIndex,
  66. }) {
  67. final payload = MoveFolderItemPayloadPB.create()
  68. ..itemId = viewId
  69. ..from = fromIndex
  70. ..to = toIndex
  71. ..ty = MoveFolderItemType.MoveView;
  72. return FolderEventMoveItem(payload).send();
  73. }
  74. Future<List<Tuple2<AppPB, List<ViewPB>>>> fetchViews(
  75. ViewLayoutTypePB layoutType) async {
  76. final result = <Tuple2<AppPB, List<ViewPB>>>[];
  77. return FolderEventReadCurrentWorkspace().send().then((value) async {
  78. final workspaces = value.getLeftOrNull<WorkspaceSettingPB>();
  79. if (workspaces != null) {
  80. final apps = workspaces.workspace.apps.items;
  81. for (var app in apps) {
  82. final views = await getViews(appId: app.id).then(
  83. (value) => value
  84. .getLeftOrNull<List<ViewPB>>()
  85. ?.where((e) => e.layout == layoutType)
  86. .toList(),
  87. );
  88. if (views != null && views.isNotEmpty) {
  89. result.add(Tuple2(app, views));
  90. }
  91. }
  92. }
  93. return result;
  94. });
  95. }
  96. Future<Either<ViewPB, FlowyError>> getView(
  97. String appID,
  98. String viewID,
  99. ) async {
  100. final payload = AppIdPB.create()..value = appID;
  101. return FolderEventReadApp(payload).send().then((result) {
  102. return result.fold(
  103. (app) => left(
  104. app.belongings.items.firstWhere((e) => e.id == viewID),
  105. ),
  106. (error) => right(error),
  107. );
  108. });
  109. }
  110. }
  111. extension AppFlowy on Either {
  112. T? getLeftOrNull<T>() {
  113. if (isLeft()) {
  114. final result = fold<T?>((l) => l, (r) => null);
  115. return result;
  116. }
  117. return null;
  118. }
  119. }