event_handler.rs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. /*
  2. * The following code defines functions that handle creating, opening, and closing documents,
  3. * as well as performing actions on documents. These functions make use of a DocumentManager,
  4. * which you can think of as a higher-level interface to interact with documents.
  5. */
  6. use std::sync::Arc;
  7. use collab_document::blocks::{
  8. json_str_to_hashmap, Block, BlockAction, BlockActionPayload, BlockActionType, BlockEvent,
  9. BlockEventPayload, DeltaType,
  10. };
  11. use flowy_error::{FlowyError, FlowyResult};
  12. use lib_dispatch::prelude::{data_result_ok, AFPluginData, AFPluginState, DataResult};
  13. use crate::{
  14. document_data::DocumentDataWrapper,
  15. entities::{
  16. ApplyActionPayloadPB, BlockActionPB, BlockActionPayloadPB, BlockActionTypePB, BlockEventPB,
  17. BlockEventPayloadPB, BlockPB, CloseDocumentPayloadPB, CreateDocumentPayloadPB, DeltaTypePB,
  18. DocEventPB, DocumentDataPB, OpenDocumentPayloadPB,
  19. },
  20. manager::DocumentManager,
  21. };
  22. // Handler for creating a new document
  23. pub(crate) async fn create_document_handler(
  24. data: AFPluginData<CreateDocumentPayloadPB>,
  25. manager: AFPluginState<Arc<DocumentManager>>,
  26. ) -> FlowyResult<()> {
  27. let context = data.into_inner();
  28. let initial_data: DocumentDataWrapper = context
  29. .initial_data
  30. .map(|data| data.into())
  31. .unwrap_or_default();
  32. manager.create_document(context.document_id, initial_data)?;
  33. Ok(())
  34. }
  35. // Handler for opening an existing document
  36. pub(crate) async fn open_document_handler(
  37. data: AFPluginData<OpenDocumentPayloadPB>,
  38. manager: AFPluginState<Arc<DocumentManager>>,
  39. ) -> DataResult<DocumentDataPB, FlowyError> {
  40. let context = data.into_inner();
  41. let document = manager.open_document(context.document_id)?;
  42. let document_data = document.lock().get_document()?;
  43. data_result_ok(DocumentDataPB::from(DocumentDataWrapper(document_data)))
  44. }
  45. pub(crate) async fn close_document_handler(
  46. data: AFPluginData<CloseDocumentPayloadPB>,
  47. manager: AFPluginState<Arc<DocumentManager>>,
  48. ) -> FlowyResult<()> {
  49. let context = data.into_inner();
  50. manager.close_document(context.document_id)?;
  51. Ok(())
  52. }
  53. // Get the content of the existing document,
  54. // if the document does not exist, return an error.
  55. pub(crate) async fn get_document_data_handler(
  56. data: AFPluginData<OpenDocumentPayloadPB>,
  57. manager: AFPluginState<Arc<DocumentManager>>,
  58. ) -> DataResult<DocumentDataPB, FlowyError> {
  59. let context = data.into_inner();
  60. let document = manager.get_document(context.document_id)?;
  61. let document_data = document.lock().get_document()?;
  62. data_result_ok(DocumentDataPB::from(DocumentDataWrapper(document_data)))
  63. }
  64. // Handler for applying an action to a document
  65. pub(crate) async fn apply_action_handler(
  66. data: AFPluginData<ApplyActionPayloadPB>,
  67. manager: AFPluginState<Arc<DocumentManager>>,
  68. ) -> FlowyResult<()> {
  69. let context = data.into_inner();
  70. let doc_id = context.document_id;
  71. let document = manager.open_document(doc_id)?;
  72. let actions = context.actions.into_iter().map(BlockAction::from).collect();
  73. document.lock().apply_action(actions);
  74. Ok(())
  75. }
  76. impl From<BlockActionPB> for BlockAction {
  77. fn from(pb: BlockActionPB) -> Self {
  78. Self {
  79. action: pb.action.into(),
  80. payload: pb.payload.into(),
  81. }
  82. }
  83. }
  84. impl From<BlockActionTypePB> for BlockActionType {
  85. fn from(pb: BlockActionTypePB) -> Self {
  86. match pb {
  87. BlockActionTypePB::Insert => Self::Insert,
  88. BlockActionTypePB::Update => Self::Update,
  89. BlockActionTypePB::Delete => Self::Delete,
  90. BlockActionTypePB::Move => Self::Move,
  91. }
  92. }
  93. }
  94. impl From<BlockActionPayloadPB> for BlockActionPayload {
  95. fn from(pb: BlockActionPayloadPB) -> Self {
  96. Self {
  97. block: pb.block.into(),
  98. parent_id: pb.parent_id,
  99. prev_id: pb.prev_id,
  100. }
  101. }
  102. }
  103. impl From<BlockPB> for Block {
  104. fn from(pb: BlockPB) -> Self {
  105. // Use `json_str_to_hashmap()` from the `collab_document` crate to convert the JSON data to a hashmap
  106. let data = json_str_to_hashmap(&pb.data).unwrap_or_default();
  107. // Convert the protobuf `BlockPB` to our internal `Block` struct
  108. Self {
  109. id: pb.id,
  110. ty: pb.ty,
  111. children: pb.children_id,
  112. parent: pb.parent_id,
  113. data,
  114. external_id: None,
  115. external_type: None,
  116. }
  117. }
  118. }
  119. impl From<BlockEvent> for BlockEventPB {
  120. fn from(payload: BlockEvent) -> Self {
  121. // Convert each individual `BlockEvent` to a protobuf `BlockEventPB`, and collect the results into a `Vec`
  122. Self {
  123. event: payload.iter().map(|e| e.to_owned().into()).collect(),
  124. }
  125. }
  126. }
  127. impl From<BlockEventPayload> for BlockEventPayloadPB {
  128. fn from(payload: BlockEventPayload) -> Self {
  129. Self {
  130. command: payload.command.into(),
  131. path: payload.path,
  132. id: payload.id,
  133. value: payload.value,
  134. }
  135. }
  136. }
  137. impl From<DeltaType> for DeltaTypePB {
  138. fn from(action: DeltaType) -> Self {
  139. match action {
  140. DeltaType::Inserted => Self::Inserted,
  141. DeltaType::Updated => Self::Updated,
  142. DeltaType::Removed => Self::Removed,
  143. }
  144. }
  145. }
  146. impl From<(&Vec<BlockEvent>, bool)> for DocEventPB {
  147. fn from((events, is_remote): (&Vec<BlockEvent>, bool)) -> Self {
  148. // Convert each individual `BlockEvent` to a protobuf `BlockEventPB`, and collect the results into a `Vec`
  149. Self {
  150. events: events.iter().map(|e| e.to_owned().into()).collect(),
  151. is_remote,
  152. }
  153. }
  154. }