editor.rs 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. use crate::web_socket::{make_block_ws_manager, EditorCommandSender};
  2. use crate::{
  3. errors::FlowyError,
  4. queue::{EditBlockQueue, EditorCommand},
  5. TextBlockUser,
  6. };
  7. use bytes::Bytes;
  8. use flowy_error::{internal_error, FlowyResult};
  9. use flowy_revision::{
  10. RevisionCloudService, RevisionManager, RevisionObjectBuilder, RevisionWebSocket, RevisionWebSocketManager,
  11. };
  12. use flowy_sync::entities::ws_data::ServerRevisionWSData;
  13. use flowy_sync::{
  14. entities::{revision::Revision, text_block_info::TextBlockInfo},
  15. errors::CollaborateResult,
  16. util::make_delta_from_revisions,
  17. };
  18. use lib_ot::{
  19. core::{Interval, Operation},
  20. rich_text::{RichTextAttribute, RichTextDelta},
  21. };
  22. use lib_ws::WSConnectState;
  23. use std::sync::Arc;
  24. use tokio::sync::{mpsc, oneshot};
  25. pub struct ClientTextBlockEditor {
  26. pub doc_id: String,
  27. #[allow(dead_code)]
  28. rev_manager: Arc<RevisionManager>,
  29. ws_manager: Arc<RevisionWebSocketManager>,
  30. edit_cmd_tx: EditorCommandSender,
  31. }
  32. impl ClientTextBlockEditor {
  33. pub(crate) async fn new(
  34. doc_id: &str,
  35. user: Arc<dyn TextBlockUser>,
  36. mut rev_manager: RevisionManager,
  37. rev_web_socket: Arc<dyn RevisionWebSocket>,
  38. cloud_service: Arc<dyn RevisionCloudService>,
  39. ) -> FlowyResult<Arc<Self>> {
  40. let document_info = rev_manager.load::<TextBlockInfoBuilder>(Some(cloud_service)).await?;
  41. let delta = document_info.delta()?;
  42. let rev_manager = Arc::new(rev_manager);
  43. let doc_id = doc_id.to_string();
  44. let user_id = user.user_id()?;
  45. let edit_cmd_tx = spawn_edit_queue(user, rev_manager.clone(), delta);
  46. let ws_manager = make_block_ws_manager(
  47. doc_id.clone(),
  48. user_id.clone(),
  49. edit_cmd_tx.clone(),
  50. rev_manager.clone(),
  51. rev_web_socket,
  52. )
  53. .await;
  54. let editor = Arc::new(Self {
  55. doc_id,
  56. rev_manager,
  57. ws_manager,
  58. edit_cmd_tx,
  59. });
  60. Ok(editor)
  61. }
  62. pub async fn insert<T: ToString>(&self, index: usize, data: T) -> Result<(), FlowyError> {
  63. let (ret, rx) = oneshot::channel::<CollaborateResult<()>>();
  64. let msg = EditorCommand::Insert {
  65. index,
  66. data: data.to_string(),
  67. ret,
  68. };
  69. let _ = self.edit_cmd_tx.send(msg).await;
  70. let _ = rx.await.map_err(internal_error)??;
  71. Ok(())
  72. }
  73. pub async fn delete(&self, interval: Interval) -> Result<(), FlowyError> {
  74. let (ret, rx) = oneshot::channel::<CollaborateResult<()>>();
  75. let msg = EditorCommand::Delete { interval, ret };
  76. let _ = self.edit_cmd_tx.send(msg).await;
  77. let _ = rx.await.map_err(internal_error)??;
  78. Ok(())
  79. }
  80. pub async fn format(&self, interval: Interval, attribute: RichTextAttribute) -> Result<(), FlowyError> {
  81. let (ret, rx) = oneshot::channel::<CollaborateResult<()>>();
  82. let msg = EditorCommand::Format {
  83. interval,
  84. attribute,
  85. ret,
  86. };
  87. let _ = self.edit_cmd_tx.send(msg).await;
  88. let _ = rx.await.map_err(internal_error)??;
  89. Ok(())
  90. }
  91. pub async fn replace<T: ToString>(&self, interval: Interval, data: T) -> Result<(), FlowyError> {
  92. let (ret, rx) = oneshot::channel::<CollaborateResult<()>>();
  93. let msg = EditorCommand::Replace {
  94. interval,
  95. data: data.to_string(),
  96. ret,
  97. };
  98. let _ = self.edit_cmd_tx.send(msg).await;
  99. let _ = rx.await.map_err(internal_error)??;
  100. Ok(())
  101. }
  102. pub async fn can_undo(&self) -> bool {
  103. let (ret, rx) = oneshot::channel::<bool>();
  104. let msg = EditorCommand::CanUndo { ret };
  105. let _ = self.edit_cmd_tx.send(msg).await;
  106. rx.await.unwrap_or(false)
  107. }
  108. pub async fn can_redo(&self) -> bool {
  109. let (ret, rx) = oneshot::channel::<bool>();
  110. let msg = EditorCommand::CanRedo { ret };
  111. let _ = self.edit_cmd_tx.send(msg).await;
  112. rx.await.unwrap_or(false)
  113. }
  114. pub async fn undo(&self) -> Result<(), FlowyError> {
  115. let (ret, rx) = oneshot::channel();
  116. let msg = EditorCommand::Undo { ret };
  117. let _ = self.edit_cmd_tx.send(msg).await;
  118. let _ = rx.await.map_err(internal_error)??;
  119. Ok(())
  120. }
  121. pub async fn redo(&self) -> Result<(), FlowyError> {
  122. let (ret, rx) = oneshot::channel();
  123. let msg = EditorCommand::Redo { ret };
  124. let _ = self.edit_cmd_tx.send(msg).await;
  125. let _ = rx.await.map_err(internal_error)??;
  126. Ok(())
  127. }
  128. pub async fn delta_str(&self) -> FlowyResult<String> {
  129. let (ret, rx) = oneshot::channel::<CollaborateResult<String>>();
  130. let msg = EditorCommand::ReadDeltaStr { ret };
  131. let _ = self.edit_cmd_tx.send(msg).await;
  132. let json = rx.await.map_err(internal_error)??;
  133. Ok(json)
  134. }
  135. #[tracing::instrument(level = "trace", skip(self, data), err)]
  136. pub(crate) async fn compose_local_delta(&self, data: Bytes) -> Result<(), FlowyError> {
  137. let delta = RichTextDelta::from_bytes(&data)?;
  138. let (ret, rx) = oneshot::channel::<CollaborateResult<()>>();
  139. let msg = EditorCommand::ComposeLocalDelta {
  140. delta: delta.clone(),
  141. ret,
  142. };
  143. let _ = self.edit_cmd_tx.send(msg).await;
  144. let _ = rx.await.map_err(internal_error)??;
  145. Ok(())
  146. }
  147. pub fn stop(&self) {
  148. self.ws_manager.stop();
  149. }
  150. pub(crate) async fn receive_ws_data(&self, data: ServerRevisionWSData) -> Result<(), FlowyError> {
  151. self.ws_manager.receive_ws_data(data).await
  152. }
  153. pub(crate) fn receive_ws_state(&self, state: &WSConnectState) {
  154. self.ws_manager.connect_state_changed(state.clone());
  155. }
  156. }
  157. impl std::ops::Drop for ClientTextBlockEditor {
  158. fn drop(&mut self) {
  159. tracing::trace!("{} ClientBlockEditor was dropped", self.doc_id)
  160. }
  161. }
  162. // The edit queue will exit after the EditorCommandSender was dropped.
  163. fn spawn_edit_queue(
  164. user: Arc<dyn TextBlockUser>,
  165. rev_manager: Arc<RevisionManager>,
  166. delta: RichTextDelta,
  167. ) -> EditorCommandSender {
  168. let (sender, receiver) = mpsc::channel(1000);
  169. let edit_queue = EditBlockQueue::new(user, rev_manager, delta, receiver);
  170. tokio::spawn(edit_queue.run());
  171. sender
  172. }
  173. #[cfg(feature = "flowy_unit_test")]
  174. impl ClientTextBlockEditor {
  175. pub async fn text_block_delta(&self) -> FlowyResult<RichTextDelta> {
  176. let (ret, rx) = oneshot::channel::<CollaborateResult<RichTextDelta>>();
  177. let msg = EditorCommand::ReadDelta { ret };
  178. let _ = self.edit_cmd_tx.send(msg).await;
  179. let delta = rx.await.map_err(internal_error)??;
  180. Ok(delta)
  181. }
  182. pub fn rev_manager(&self) -> Arc<RevisionManager> {
  183. self.rev_manager.clone()
  184. }
  185. }
  186. struct TextBlockInfoBuilder();
  187. impl RevisionObjectBuilder for TextBlockInfoBuilder {
  188. type Output = TextBlockInfo;
  189. fn build_object(object_id: &str, revisions: Vec<Revision>) -> FlowyResult<Self::Output> {
  190. let (base_rev_id, rev_id) = revisions.last().unwrap().pair_rev_id();
  191. let mut delta = make_delta_from_revisions(revisions)?;
  192. correct_delta(&mut delta);
  193. Result::<TextBlockInfo, FlowyError>::Ok(TextBlockInfo {
  194. block_id: object_id.to_owned(),
  195. text: delta.to_delta_str(),
  196. rev_id,
  197. base_rev_id,
  198. })
  199. }
  200. }
  201. // quill-editor requires the delta should end with '\n' and only contains the
  202. // insert operation. The function, correct_delta maybe be removed in the future.
  203. fn correct_delta(delta: &mut RichTextDelta) {
  204. if let Some(op) = delta.ops.last() {
  205. let op_data = op.get_data();
  206. if !op_data.ends_with('\n') {
  207. tracing::warn!("The document must end with newline. Correcting it by inserting newline op");
  208. delta.ops.push(Operation::Insert("\n".into()));
  209. }
  210. }
  211. if let Some(op) = delta.ops.iter().find(|op| !op.is_insert()) {
  212. tracing::warn!("The document can only contains insert operations, but found {:?}", op);
  213. delta.ops.retain(|op| op.is_insert());
  214. }
  215. }