grid_view_editor.rs 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  1. use crate::dart_notification::{send_dart_notification, GridNotification};
  2. use crate::entities::{
  3. CreateRowParams, GridFilterConfiguration, GridSettingPB, GroupPB, GroupRowsChangesetPB, InsertedRowPB, RowPB,
  4. };
  5. use crate::services::grid_editor_task::GridServiceTaskScheduler;
  6. use crate::services::grid_view_manager::{GridViewFieldDelegate, GridViewRowDelegate};
  7. use crate::services::group::{default_group_configuration, GroupConfigurationDelegate, GroupService};
  8. use crate::services::setting::make_grid_setting;
  9. use flowy_error::{FlowyError, FlowyResult};
  10. use flowy_grid_data_model::revision::{FieldRevision, GroupConfigurationRevision, RowChangeset, RowRevision};
  11. use flowy_revision::{RevisionCloudService, RevisionManager, RevisionObjectBuilder};
  12. use flowy_sync::client_grid::{GridViewRevisionChangeset, GridViewRevisionPad};
  13. use flowy_sync::entities::grid::GridSettingChangesetParams;
  14. use flowy_sync::entities::revision::Revision;
  15. use lib_infra::future::{wrap_future, AFFuture, FutureResult};
  16. use std::sync::Arc;
  17. use tokio::sync::RwLock;
  18. #[allow(dead_code)]
  19. pub struct GridViewRevisionEditor {
  20. user_id: String,
  21. view_id: String,
  22. pad: Arc<RwLock<GridViewRevisionPad>>,
  23. rev_manager: Arc<RevisionManager>,
  24. field_delegate: Arc<dyn GridViewFieldDelegate>,
  25. row_delegate: Arc<dyn GridViewRowDelegate>,
  26. group_service: Arc<RwLock<GroupService>>,
  27. scheduler: Arc<dyn GridServiceTaskScheduler>,
  28. }
  29. impl GridViewRevisionEditor {
  30. pub(crate) async fn new(
  31. user_id: &str,
  32. token: &str,
  33. view_id: String,
  34. field_delegate: Arc<dyn GridViewFieldDelegate>,
  35. row_delegate: Arc<dyn GridViewRowDelegate>,
  36. scheduler: Arc<dyn GridServiceTaskScheduler>,
  37. mut rev_manager: RevisionManager,
  38. ) -> FlowyResult<Self> {
  39. let cloud = Arc::new(GridViewRevisionCloudService {
  40. token: token.to_owned(),
  41. });
  42. let view_revision_pad = rev_manager.load::<GridViewRevisionPadBuilder>(Some(cloud)).await?;
  43. let pad = Arc::new(RwLock::new(view_revision_pad));
  44. let rev_manager = Arc::new(rev_manager);
  45. let group_service = GroupService::new(Box::new(pad.clone())).await;
  46. let user_id = user_id.to_owned();
  47. Ok(Self {
  48. pad,
  49. user_id,
  50. view_id,
  51. rev_manager,
  52. scheduler,
  53. field_delegate,
  54. row_delegate,
  55. group_service: Arc::new(RwLock::new(group_service)),
  56. })
  57. }
  58. pub(crate) async fn will_create_row(&self, row_rev: &mut RowRevision, params: &CreateRowParams) {
  59. match params.group_id.as_ref() {
  60. None => {}
  61. Some(group_id) => {
  62. self.group_service
  63. .read()
  64. .await
  65. .will_create_row(row_rev, group_id, |field_id| {
  66. self.field_delegate.get_field_rev(&field_id)
  67. })
  68. .await;
  69. }
  70. }
  71. }
  72. pub(crate) async fn did_create_row(&self, row_pb: &RowPB, params: &CreateRowParams) {
  73. // Send the group notification if the current view has groups
  74. match params.group_id.as_ref() {
  75. None => {}
  76. Some(group_id) => {
  77. let inserted_row = InsertedRowPB {
  78. row: row_pb.clone(),
  79. index: None,
  80. };
  81. let changeset = GroupRowsChangesetPB::insert(group_id.clone(), vec![inserted_row]);
  82. self.notify_did_update_group(changeset).await;
  83. }
  84. }
  85. }
  86. pub(crate) async fn did_delete_row(&self, row_rev: &RowRevision) {
  87. // Send the group notification if the current view has groups;
  88. if let Some(changesets) = self
  89. .group_service
  90. .write()
  91. .await
  92. .did_delete_row(row_rev, |field_id| self.field_delegate.get_field_rev(&field_id))
  93. .await
  94. {
  95. for changeset in changesets {
  96. self.notify_did_update_group(changeset).await;
  97. }
  98. }
  99. }
  100. pub(crate) async fn did_update_row(&self, row_rev: &RowRevision) {
  101. if let Some(changesets) = self
  102. .group_service
  103. .write()
  104. .await
  105. .did_update_row(row_rev, |field_id| self.field_delegate.get_field_rev(&field_id))
  106. .await
  107. {
  108. for changeset in changesets {
  109. self.notify_did_update_group(changeset).await;
  110. }
  111. }
  112. }
  113. pub(crate) async fn did_move_row(
  114. &self,
  115. row_rev: &RowRevision,
  116. row_changeset: &mut RowChangeset,
  117. upper_row_id: &str,
  118. ) {
  119. if let Some(changesets) = self
  120. .group_service
  121. .write()
  122. .await
  123. .did_move_row(row_rev, row_changeset, upper_row_id, |field_id| {
  124. self.field_delegate.get_field_rev(&field_id)
  125. })
  126. .await
  127. {
  128. for changeset in changesets {
  129. tracing::trace!("Group: {} changeset: {}", changeset.group_id, changeset);
  130. self.notify_did_update_group(changeset).await;
  131. }
  132. }
  133. }
  134. pub(crate) async fn load_groups(&self) -> FlowyResult<Vec<GroupPB>> {
  135. let field_revs = self.field_delegate.get_field_revs().await;
  136. let row_revs = self.row_delegate.gv_row_revs().await;
  137. match self
  138. .group_service
  139. .write()
  140. .await
  141. .load_groups(&field_revs, row_revs)
  142. .await
  143. {
  144. None => Ok(vec![]),
  145. Some(groups) => Ok(groups.into_iter().map(GroupPB::from).collect()),
  146. }
  147. }
  148. pub(crate) async fn get_setting(&self) -> GridSettingPB {
  149. let field_revs = self.field_delegate.get_field_revs().await;
  150. let grid_setting = make_grid_setting(self.pad.read().await.get_setting_rev(), &field_revs);
  151. grid_setting
  152. }
  153. pub(crate) async fn update_setting(&self, changeset: GridSettingChangesetParams) -> FlowyResult<()> {
  154. let _ = self.modify(|pad| Ok(pad.update_setting(changeset)?)).await;
  155. Ok(())
  156. }
  157. pub(crate) async fn get_filters(&self) -> Vec<GridFilterConfiguration> {
  158. let field_revs = self.field_delegate.get_field_revs().await;
  159. match self.pad.read().await.get_setting_rev().get_all_filters(&field_revs) {
  160. None => vec![],
  161. Some(filters) => filters
  162. .into_values()
  163. .flatten()
  164. .map(|filter| GridFilterConfiguration::from(filter.as_ref()))
  165. .collect(),
  166. }
  167. }
  168. async fn notify_did_update_group(&self, changeset: GroupRowsChangesetPB) {
  169. send_dart_notification(&changeset.group_id, GridNotification::DidUpdateGroup)
  170. .payload(changeset)
  171. .send();
  172. }
  173. async fn modify<F>(&self, f: F) -> FlowyResult<()>
  174. where
  175. F: for<'a> FnOnce(&'a mut GridViewRevisionPad) -> FlowyResult<Option<GridViewRevisionChangeset>>,
  176. {
  177. let mut write_guard = self.pad.write().await;
  178. match f(&mut *write_guard)? {
  179. None => {}
  180. Some(change) => {
  181. let _ = self.apply_change(change).await?;
  182. }
  183. }
  184. Ok(())
  185. }
  186. async fn apply_change(&self, change: GridViewRevisionChangeset) -> FlowyResult<()> {
  187. let GridViewRevisionChangeset { delta, md5 } = change;
  188. let user_id = self.user_id.clone();
  189. let (base_rev_id, rev_id) = self.rev_manager.next_rev_id_pair();
  190. let delta_data = delta.json_bytes();
  191. let revision = Revision::new(
  192. &self.rev_manager.object_id,
  193. base_rev_id,
  194. rev_id,
  195. delta_data,
  196. &user_id,
  197. md5,
  198. );
  199. let _ = self.rev_manager.add_local_revision(&revision).await?;
  200. Ok(())
  201. }
  202. }
  203. struct GridViewRevisionCloudService {
  204. #[allow(dead_code)]
  205. token: String,
  206. }
  207. impl RevisionCloudService for GridViewRevisionCloudService {
  208. #[tracing::instrument(level = "trace", skip(self))]
  209. fn fetch_object(&self, _user_id: &str, _object_id: &str) -> FutureResult<Vec<Revision>, FlowyError> {
  210. FutureResult::new(async move { Ok(vec![]) })
  211. }
  212. }
  213. struct GridViewRevisionPadBuilder();
  214. impl RevisionObjectBuilder for GridViewRevisionPadBuilder {
  215. type Output = GridViewRevisionPad;
  216. fn build_object(object_id: &str, revisions: Vec<Revision>) -> FlowyResult<Self::Output> {
  217. let pad = GridViewRevisionPad::from_revisions(object_id, revisions)?;
  218. Ok(pad)
  219. }
  220. }
  221. impl GroupConfigurationDelegate for Arc<RwLock<GridViewRevisionPad>> {
  222. fn get_group_configuration(&self, field_rev: Arc<FieldRevision>) -> AFFuture<GroupConfigurationRevision> {
  223. let view_pad = self.clone();
  224. wrap_future(async move {
  225. let grid_pad = view_pad.read().await;
  226. let configurations = grid_pad.get_groups(&field_rev.id, &field_rev.field_type_rev);
  227. match configurations {
  228. None => default_group_configuration(&field_rev),
  229. Some(mut configurations) => {
  230. assert_eq!(configurations.len(), 1);
  231. (&*configurations.pop().unwrap()).clone()
  232. }
  233. }
  234. })
  235. }
  236. }