grid_test.rs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381
  1. use crate::grid::script::EditorScript::*;
  2. use crate::grid::script::*;
  3. use chrono::NaiveDateTime;
  4. use flowy_grid::services::field::{
  5. DateCellContentChangeset, DateCellData, MultiSelectTypeOptionPB, SelectOption, SelectOptionCellContentChangeset,
  6. SingleSelectTypeOption, SELECTION_IDS_SEPARATOR,
  7. };
  8. use flowy_grid::services::row::{decode_cell_data_from_type_option_cell_data, CreateRowMetaBuilder};
  9. use flowy_grid_data_model::entities::{
  10. CellChangeset, FieldChangesetParams, FieldType, GridBlockInfoChangeset, GridBlockMetaSnapshot, RowMetaChangeset,
  11. TypeOptionDataEntry,
  12. };
  13. #[tokio::test]
  14. async fn grid_create_field() {
  15. let mut test = GridEditorTest::new().await;
  16. let (text_field_params, text_field_meta) = create_text_field(&test.grid_id);
  17. let (single_select_params, single_select_field) = create_single_select_field(&test.grid_id);
  18. let scripts = vec![
  19. CreateField {
  20. params: text_field_params,
  21. },
  22. AssertFieldEqual {
  23. field_index: test.field_count,
  24. field_meta: text_field_meta,
  25. },
  26. ];
  27. test.run_scripts(scripts).await;
  28. let scripts = vec![
  29. CreateField {
  30. params: single_select_params,
  31. },
  32. AssertFieldEqual {
  33. field_index: test.field_count,
  34. field_meta: single_select_field,
  35. },
  36. ];
  37. test.run_scripts(scripts).await;
  38. }
  39. #[tokio::test]
  40. async fn grid_create_duplicate_field() {
  41. let mut test = GridEditorTest::new().await;
  42. let (params, _) = create_text_field(&test.grid_id);
  43. let field_count = test.field_count;
  44. let expected_field_count = field_count + 1;
  45. let scripts = vec![
  46. CreateField { params: params.clone() },
  47. CreateField { params },
  48. AssertFieldCount(expected_field_count),
  49. ];
  50. test.run_scripts(scripts).await;
  51. }
  52. #[tokio::test]
  53. async fn grid_update_field_with_empty_change() {
  54. let mut test = GridEditorTest::new().await;
  55. let (params, field_meta) = create_single_select_field(&test.grid_id);
  56. let changeset = FieldChangesetParams {
  57. field_id: field_meta.id.clone(),
  58. grid_id: test.grid_id.clone(),
  59. ..Default::default()
  60. };
  61. let scripts = vec![
  62. CreateField { params },
  63. UpdateField { changeset },
  64. AssertFieldEqual {
  65. field_index: test.field_count,
  66. field_meta,
  67. },
  68. ];
  69. test.run_scripts(scripts).await;
  70. }
  71. #[tokio::test]
  72. async fn grid_update_field() {
  73. let mut test = GridEditorTest::new().await;
  74. let (single_select_params, single_select_field) = create_single_select_field(&test.grid_id);
  75. let mut cloned_field = single_select_field.clone();
  76. let mut single_select_type_option = SingleSelectTypeOption::from(&single_select_field);
  77. single_select_type_option.options.push(SelectOption::new("Unknown"));
  78. let changeset = FieldChangesetParams {
  79. field_id: single_select_field.id.clone(),
  80. grid_id: test.grid_id.clone(),
  81. frozen: Some(true),
  82. width: Some(1000),
  83. type_option_data: Some(single_select_type_option.protobuf_bytes().to_vec()),
  84. ..Default::default()
  85. };
  86. cloned_field.frozen = true;
  87. cloned_field.width = 1000;
  88. cloned_field.insert_type_option_entry(&single_select_type_option);
  89. let scripts = vec![
  90. CreateField {
  91. params: single_select_params,
  92. },
  93. UpdateField { changeset },
  94. AssertFieldEqual {
  95. field_index: test.field_count,
  96. field_meta: cloned_field,
  97. },
  98. ];
  99. test.run_scripts(scripts).await;
  100. }
  101. #[tokio::test]
  102. async fn grid_delete_field() {
  103. let mut test = GridEditorTest::new().await;
  104. let expected_field_count = test.field_count;
  105. let (text_params, text_field) = create_text_field(&test.grid_id);
  106. let scripts = vec![
  107. CreateField { params: text_params },
  108. DeleteField { field_meta: text_field },
  109. AssertFieldCount(expected_field_count),
  110. ];
  111. test.run_scripts(scripts).await;
  112. }
  113. #[tokio::test]
  114. async fn grid_create_block() {
  115. let grid_block = GridBlockMetaSnapshot::new();
  116. let scripts = vec![
  117. AssertBlockCount(1),
  118. CreateBlock { block: grid_block },
  119. AssertBlockCount(2),
  120. ];
  121. GridEditorTest::new().await.run_scripts(scripts).await;
  122. }
  123. #[tokio::test]
  124. async fn grid_update_block() {
  125. let grid_block = GridBlockMetaSnapshot::new();
  126. let mut cloned_grid_block = grid_block.clone();
  127. let changeset = GridBlockInfoChangeset {
  128. block_id: grid_block.block_id.clone(),
  129. start_row_index: Some(2),
  130. row_count: Some(10),
  131. };
  132. cloned_grid_block.start_row_index = 2;
  133. cloned_grid_block.row_count = 10;
  134. let scripts = vec![
  135. AssertBlockCount(1),
  136. CreateBlock { block: grid_block },
  137. UpdateBlock { changeset },
  138. AssertBlockCount(2),
  139. AssertBlockEqual {
  140. block_index: 1,
  141. block: cloned_grid_block,
  142. },
  143. ];
  144. GridEditorTest::new().await.run_scripts(scripts).await;
  145. }
  146. #[tokio::test]
  147. async fn grid_create_row() {
  148. let scripts = vec![AssertRowCount(3), CreateEmptyRow, CreateEmptyRow, AssertRowCount(5)];
  149. GridEditorTest::new().await.run_scripts(scripts).await;
  150. }
  151. #[tokio::test]
  152. async fn grid_create_row2() {
  153. let mut test = GridEditorTest::new().await;
  154. let create_row_context = CreateRowMetaBuilder::new(&test.field_metas).build();
  155. let scripts = vec![
  156. AssertRowCount(3),
  157. CreateRow {
  158. context: create_row_context,
  159. },
  160. AssertRowCount(4),
  161. ];
  162. test.run_scripts(scripts).await;
  163. }
  164. #[tokio::test]
  165. async fn grid_update_row() {
  166. let mut test = GridEditorTest::new().await;
  167. let context = CreateRowMetaBuilder::new(&test.field_metas).build();
  168. let changeset = RowMetaChangeset {
  169. row_id: context.row_id.clone(),
  170. height: None,
  171. visibility: None,
  172. cell_by_field_id: Default::default(),
  173. };
  174. let scripts = vec![
  175. AssertRowCount(3),
  176. CreateRow { context },
  177. UpdateRow {
  178. changeset: changeset.clone(),
  179. },
  180. AssertRow { changeset },
  181. AssertRowCount(4),
  182. ];
  183. test.run_scripts(scripts).await;
  184. }
  185. #[tokio::test]
  186. async fn grid_delete_row() {
  187. let mut test = GridEditorTest::new().await;
  188. let context_1 = CreateRowMetaBuilder::new(&test.field_metas).build();
  189. let context_2 = CreateRowMetaBuilder::new(&test.field_metas).build();
  190. let row_ids = vec![context_1.row_id.clone(), context_2.row_id.clone()];
  191. let scripts = vec![
  192. AssertRowCount(3),
  193. CreateRow { context: context_1 },
  194. CreateRow { context: context_2 },
  195. AssertBlockCount(1),
  196. AssertBlock {
  197. block_index: 0,
  198. row_count: 5,
  199. start_row_index: 0,
  200. },
  201. DeleteRow { row_ids },
  202. AssertBlock {
  203. block_index: 0,
  204. row_count: 3,
  205. start_row_index: 0,
  206. },
  207. ];
  208. test.run_scripts(scripts).await;
  209. }
  210. #[tokio::test]
  211. async fn grid_row_add_cells_test() {
  212. let mut test = GridEditorTest::new().await;
  213. let mut builder = CreateRowMetaBuilder::new(&test.field_metas);
  214. for field in &test.field_metas {
  215. match field.field_type {
  216. FieldType::RichText => {
  217. builder.add_cell(&field.id, "hello world".to_owned()).unwrap();
  218. }
  219. FieldType::Number => {
  220. builder.add_cell(&field.id, "18,443".to_owned()).unwrap();
  221. }
  222. FieldType::DateTime => {
  223. builder
  224. .add_cell(&field.id, make_date_cell_string("1647251762"))
  225. .unwrap();
  226. }
  227. FieldType::SingleSelect => {
  228. let type_option = SingleSelectTypeOption::from(field);
  229. let option = type_option.options.first().unwrap();
  230. builder.add_select_option_cell(&field.id, option.id.clone()).unwrap();
  231. }
  232. FieldType::MultiSelect => {
  233. let type_option = MultiSelectTypeOptionPB::from(field);
  234. let ops_ids = type_option
  235. .options
  236. .iter()
  237. .map(|option| option.id.clone())
  238. .collect::<Vec<_>>()
  239. .join(SELECTION_IDS_SEPARATOR);
  240. builder.add_select_option_cell(&field.id, ops_ids).unwrap();
  241. }
  242. FieldType::Checkbox => {
  243. builder.add_cell(&field.id, "false".to_string()).unwrap();
  244. }
  245. FieldType::URL => {
  246. builder.add_cell(&field.id, "1".to_string()).unwrap();
  247. }
  248. }
  249. }
  250. let context = builder.build();
  251. let scripts = vec![CreateRow { context }, AssertGridMetaPad];
  252. test.run_scripts(scripts).await;
  253. }
  254. #[tokio::test]
  255. async fn grid_row_add_date_cell_test() {
  256. let mut test = GridEditorTest::new().await;
  257. let mut builder = CreateRowMetaBuilder::new(&test.field_metas);
  258. let mut date_field = None;
  259. let timestamp = 1647390674;
  260. for field in &test.field_metas {
  261. if field.field_type == FieldType::DateTime {
  262. date_field = Some(field.clone());
  263. NaiveDateTime::from_timestamp(123, 0);
  264. // The data should not be empty
  265. assert!(builder.add_cell(&field.id, "".to_string()).is_err());
  266. assert!(builder.add_cell(&field.id, make_date_cell_string("123")).is_ok());
  267. assert!(builder
  268. .add_cell(&field.id, make_date_cell_string(&timestamp.to_string()))
  269. .is_ok());
  270. }
  271. }
  272. let context = builder.build();
  273. let date_field = date_field.unwrap();
  274. let cell_data = context.cell_by_field_id.get(&date_field.id).unwrap().clone();
  275. assert_eq!(
  276. decode_cell_data_from_type_option_cell_data(cell_data.data.clone(), &date_field, &date_field.field_type)
  277. .parse::<DateCellData>()
  278. .unwrap()
  279. .date,
  280. "2022/03/16",
  281. );
  282. let scripts = vec![CreateRow { context }];
  283. test.run_scripts(scripts).await;
  284. }
  285. #[tokio::test]
  286. async fn grid_cell_update() {
  287. let mut test = GridEditorTest::new().await;
  288. let field_metas = &test.field_metas;
  289. let row_metas = &test.row_metas;
  290. let grid_blocks = &test.grid_blocks;
  291. assert_eq!(row_metas.len(), 3);
  292. assert_eq!(grid_blocks.len(), 1);
  293. let block_id = &grid_blocks.first().unwrap().block_id;
  294. let mut scripts = vec![];
  295. for (index, row_meta) in row_metas.iter().enumerate() {
  296. for field_meta in field_metas {
  297. if index == 0 {
  298. let data = match field_meta.field_type {
  299. FieldType::RichText => "".to_string(),
  300. FieldType::Number => "123".to_string(),
  301. FieldType::DateTime => make_date_cell_string("123"),
  302. FieldType::SingleSelect => {
  303. let type_option = SingleSelectTypeOption::from(field_meta);
  304. SelectOptionCellContentChangeset::from_insert(&type_option.options.first().unwrap().id).to_str()
  305. }
  306. FieldType::MultiSelect => {
  307. let type_option = MultiSelectTypeOptionPB::from(field_meta);
  308. SelectOptionCellContentChangeset::from_insert(&type_option.options.first().unwrap().id).to_str()
  309. }
  310. FieldType::Checkbox => "1".to_string(),
  311. FieldType::URL => "1".to_string(),
  312. };
  313. scripts.push(UpdateCell {
  314. changeset: CellChangeset {
  315. grid_id: block_id.to_string(),
  316. row_id: row_meta.id.clone(),
  317. field_id: field_meta.id.clone(),
  318. cell_content_changeset: Some(data),
  319. },
  320. is_err: false,
  321. });
  322. }
  323. if index == 1 {
  324. let (data, is_err) = match field_meta.field_type {
  325. FieldType::RichText => ("1".to_string().repeat(10001), true),
  326. FieldType::Number => ("abc".to_string(), true),
  327. FieldType::DateTime => ("abc".to_string(), true),
  328. FieldType::SingleSelect => (SelectOptionCellContentChangeset::from_insert("abc").to_str(), false),
  329. FieldType::MultiSelect => (SelectOptionCellContentChangeset::from_insert("abc").to_str(), false),
  330. FieldType::Checkbox => ("2".to_string(), false),
  331. FieldType::URL => ("2".to_string(), false),
  332. };
  333. scripts.push(UpdateCell {
  334. changeset: CellChangeset {
  335. grid_id: block_id.to_string(),
  336. row_id: row_meta.id.clone(),
  337. field_id: field_meta.id.clone(),
  338. cell_content_changeset: Some(data),
  339. },
  340. is_err,
  341. });
  342. }
  343. }
  344. }
  345. test.run_scripts(scripts).await;
  346. }
  347. fn make_date_cell_string(s: &str) -> String {
  348. serde_json::to_string(&DateCellContentChangeset {
  349. date: Some(s.to_string()),
  350. time: None,
  351. })
  352. .unwrap()
  353. }