database_editor.rs 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. use crate::grid::mock_data::*;
  2. use bytes::Bytes;
  3. use flowy_database::entities::*;
  4. use flowy_database::services::cell::ToCellChangesetString;
  5. use flowy_database::services::field::SelectOptionPB;
  6. use flowy_database::services::field::*;
  7. use flowy_database::services::grid_editor::DatabaseRevisionEditor;
  8. use flowy_test::helper::ViewTest;
  9. use flowy_test::FlowySDKTest;
  10. use grid_model::*;
  11. use std::collections::HashMap;
  12. use std::sync::Arc;
  13. use strum::EnumCount;
  14. pub struct DatabaseEditorTest {
  15. pub sdk: FlowySDKTest,
  16. pub view_id: String,
  17. pub editor: Arc<DatabaseRevisionEditor>,
  18. pub field_revs: Vec<Arc<FieldRevision>>,
  19. pub block_meta_revs: Vec<Arc<GridBlockMetaRevision>>,
  20. pub row_revs: Vec<Arc<RowRevision>>,
  21. pub field_count: usize,
  22. pub row_by_row_id: HashMap<String, RowPB>,
  23. }
  24. impl DatabaseEditorTest {
  25. pub async fn new_table() -> Self {
  26. Self::new(DatabaseViewLayout::Grid).await
  27. }
  28. pub async fn new_board() -> Self {
  29. Self::new(DatabaseViewLayout::Board).await
  30. }
  31. pub async fn new(layout: DatabaseViewLayout) -> Self {
  32. let sdk = FlowySDKTest::default();
  33. let _ = sdk.init_user().await;
  34. let test = match layout {
  35. DatabaseViewLayout::Grid => {
  36. let build_context = make_test_grid();
  37. let view_data: Bytes = build_context.into();
  38. ViewTest::new_grid_view(&sdk, view_data.to_vec()).await
  39. }
  40. DatabaseViewLayout::Board => {
  41. let build_context = make_test_board();
  42. let view_data: Bytes = build_context.into();
  43. ViewTest::new_board_view(&sdk, view_data.to_vec()).await
  44. }
  45. DatabaseViewLayout::Calendar => {
  46. let build_context = make_test_calendar();
  47. let view_data: Bytes = build_context.into();
  48. ViewTest::new_calendar_view(&sdk, view_data.to_vec()).await
  49. }
  50. };
  51. let editor = sdk.grid_manager.open_database(&test.view.id).await.unwrap();
  52. let field_revs = editor.get_field_revs(None).await.unwrap();
  53. let block_meta_revs = editor.get_block_meta_revs().await.unwrap();
  54. let row_pbs = editor.get_all_row_revs(&test.view.id).await.unwrap();
  55. assert_eq!(block_meta_revs.len(), 1);
  56. // It seems like you should add the field in the make_test_grid() function.
  57. // Because we assert the initialize count of the fields is equal to FieldType::COUNT.
  58. assert_eq!(field_revs.len(), FieldType::COUNT);
  59. let grid_id = test.view.id;
  60. Self {
  61. sdk,
  62. view_id: grid_id,
  63. editor,
  64. field_revs,
  65. block_meta_revs,
  66. row_revs: row_pbs,
  67. field_count: FieldType::COUNT,
  68. row_by_row_id: HashMap::default(),
  69. }
  70. }
  71. pub async fn get_row_revs(&self) -> Vec<Arc<RowRevision>> {
  72. self.editor.get_all_row_revs(&self.view_id).await.unwrap()
  73. }
  74. pub async fn grid_filters(&self) -> Vec<FilterPB> {
  75. self.editor.get_all_filters().await.unwrap()
  76. }
  77. pub fn get_field_rev(&self, field_id: &str, field_type: FieldType) -> &Arc<FieldRevision> {
  78. self.field_revs
  79. .iter()
  80. .filter(|field_rev| {
  81. let t_field_type: FieldType = field_rev.ty.into();
  82. field_rev.id == field_id && t_field_type == field_type
  83. })
  84. .collect::<Vec<_>>()
  85. .pop()
  86. .unwrap()
  87. }
  88. /// returns the first `FieldRevision` in the build-in test grid.
  89. /// Not support duplicate `FieldType` in test grid yet.
  90. pub fn get_first_field_rev(&self, field_type: FieldType) -> &Arc<FieldRevision> {
  91. self.field_revs
  92. .iter()
  93. .filter(|field_rev| {
  94. let t_field_type: FieldType = field_rev.ty.into();
  95. t_field_type == field_type
  96. })
  97. .collect::<Vec<_>>()
  98. .pop()
  99. .unwrap()
  100. }
  101. pub fn get_multi_select_type_option(&self, field_id: &str) -> Vec<SelectOptionPB> {
  102. let field_type = FieldType::MultiSelect;
  103. let field_rev = self.get_field_rev(field_id, field_type.clone());
  104. let type_option = field_rev
  105. .get_type_option::<MultiSelectTypeOptionPB>(field_type.into())
  106. .unwrap();
  107. type_option.options
  108. }
  109. pub fn get_single_select_type_option(&self, field_id: &str) -> SingleSelectTypeOptionPB {
  110. let field_type = FieldType::SingleSelect;
  111. let field_rev = self.get_field_rev(field_id, field_type.clone());
  112. field_rev
  113. .get_type_option::<SingleSelectTypeOptionPB>(field_type.into())
  114. .unwrap()
  115. }
  116. #[allow(dead_code)]
  117. pub fn get_checklist_type_option(&self, field_id: &str) -> ChecklistTypeOptionPB {
  118. let field_type = FieldType::Checklist;
  119. let field_rev = self.get_field_rev(field_id, field_type.clone());
  120. field_rev
  121. .get_type_option::<ChecklistTypeOptionPB>(field_type.into())
  122. .unwrap()
  123. }
  124. #[allow(dead_code)]
  125. pub fn get_checkbox_type_option(&self, field_id: &str) -> CheckboxTypeOptionPB {
  126. let field_type = FieldType::Checkbox;
  127. let field_rev = self.get_field_rev(field_id, field_type.clone());
  128. field_rev
  129. .get_type_option::<CheckboxTypeOptionPB>(field_type.into())
  130. .unwrap()
  131. }
  132. pub fn block_id(&self) -> &str {
  133. &self.block_meta_revs.last().unwrap().block_id
  134. }
  135. pub async fn update_cell<T: ToCellChangesetString>(&mut self, field_id: &str, row_id: String, cell_changeset: T) {
  136. let field_rev = self
  137. .field_revs
  138. .iter()
  139. .find(|field_rev| field_rev.id == field_id)
  140. .unwrap();
  141. self.editor
  142. .update_cell_with_changeset(&row_id, &field_rev.id, cell_changeset)
  143. .await
  144. .unwrap();
  145. }
  146. pub(crate) async fn update_text_cell(&mut self, row_id: String, content: &str) {
  147. let field_rev = self
  148. .field_revs
  149. .iter()
  150. .find(|field_rev| {
  151. let field_type: FieldType = field_rev.ty.into();
  152. field_type == FieldType::RichText
  153. })
  154. .unwrap()
  155. .clone();
  156. self.update_cell(&field_rev.id, row_id, content.to_string()).await;
  157. }
  158. pub(crate) async fn update_single_select_cell(&mut self, row_id: String, option_id: &str) {
  159. let field_rev = self
  160. .field_revs
  161. .iter()
  162. .find(|field_rev| {
  163. let field_type: FieldType = field_rev.ty.into();
  164. field_type == FieldType::SingleSelect
  165. })
  166. .unwrap()
  167. .clone();
  168. let cell_changeset = SelectOptionCellChangeset::from_insert_option_id(option_id);
  169. self.update_cell(&field_rev.id, row_id, cell_changeset).await;
  170. }
  171. }