database_test_op.dart 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212
  1. import 'dart:io';
  2. import 'package:appflowy/generated/locale_keys.g.dart';
  3. import 'package:appflowy/plugins/database_view/board/presentation/board_page.dart';
  4. import 'package:appflowy/plugins/database_view/calendar/presentation/calendar_day.dart';
  5. import 'package:appflowy/plugins/database_view/calendar/presentation/calendar_page.dart';
  6. import 'package:appflowy/plugins/database_view/calendar/presentation/toolbar/calendar_layout_setting.dart';
  7. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/filter/choicechip/checkbox.dart';
  8. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/filter/choicechip/checklist/checklist.dart';
  9. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/filter/choicechip/select_option/option_list.dart';
  10. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/filter/choicechip/select_option/select_option.dart';
  11. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/filter/choicechip/text.dart';
  12. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/filter/create_filter_list.dart';
  13. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/filter/disclosure_button.dart';
  14. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/filter/filter_menu_item.dart';
  15. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/header/field_cell_action_sheet.dart';
  16. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/header/field_type_extension.dart';
  17. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/header/field_type_list.dart';
  18. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/header/field_type_option_editor.dart';
  19. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/sort/create_sort_list.dart';
  20. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/sort/order_panel.dart';
  21. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/sort/sort_editor.dart';
  22. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/sort/sort_menu.dart';
  23. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/header/type_option/date.dart';
  24. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/toolbar/filter_button.dart';
  25. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/toolbar/grid_layout.dart';
  26. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/toolbar/sort_button.dart';
  27. import 'package:appflowy/plugins/database_view/tar_bar/tab_bar_view.dart';
  28. import 'package:appflowy/plugins/database_view/tar_bar/tar_bar_add_button.dart';
  29. import 'package:appflowy/plugins/database_view/widgets/database_layout_ext.dart';
  30. import 'package:appflowy/plugins/database_view/widgets/row/cells/select_option_cell/extension.dart';
  31. import 'package:appflowy/plugins/database_view/widgets/row/cells/select_option_cell/select_option_editor.dart';
  32. import 'package:appflowy/plugins/database_view/widgets/row/cells/select_option_cell/text_field.dart';
  33. import 'package:appflowy/plugins/database_view/widgets/row/cells/checklist_cell/checklist_progress_bar.dart';
  34. import 'package:appflowy/plugins/database_view/widgets/row/row_document.dart';
  35. import 'package:appflowy/plugins/database_view/widgets/row/cells/date_cell/date_editor.dart';
  36. import 'package:appflowy/plugins/database_view/widgets/setting/database_setting.dart';
  37. import 'package:appflowy/plugins/database_view/widgets/setting/setting_button.dart';
  38. import 'package:appflowy/workspace/presentation/widgets/pop_up_action.dart';
  39. import 'package:appflowy/workspace/presentation/widgets/toggle/toggle.dart';
  40. import 'package:appflowy_backend/protobuf/flowy-database2/setting_entities.pbenum.dart';
  41. import 'package:appflowy_backend/protobuf/flowy-folder2/view.pb.dart';
  42. import 'package:easy_localization/easy_localization.dart';
  43. import 'package:flowy_infra_ui/flowy_infra_ui.dart';
  44. import 'package:flowy_infra_ui/style_widget/text_input.dart';
  45. import 'package:flowy_infra_ui/widget/buttons/primary_button.dart';
  46. import 'package:flutter/gestures.dart';
  47. import 'package:flutter/material.dart';
  48. import 'package:flutter/services.dart';
  49. import 'package:flutter_test/flutter_test.dart';
  50. import 'package:appflowy/plugins/database_view/grid/presentation/grid_page.dart';
  51. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/footer/grid_footer.dart';
  52. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/header/field_cell.dart';
  53. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/header/field_editor.dart';
  54. import 'package:appflowy/plugins/database_view/grid/presentation/widgets/row/row.dart';
  55. import 'package:appflowy/plugins/database_view/widgets/row/accessory/cell_accessory.dart';
  56. import 'package:appflowy/plugins/database_view/widgets/row/cells/cells.dart';
  57. import 'package:appflowy/plugins/database_view/widgets/row/row_action.dart';
  58. import 'package:appflowy/plugins/database_view/widgets/row/row_banner.dart';
  59. import 'package:appflowy/plugins/database_view/widgets/row/row_detail.dart';
  60. import 'package:appflowy/plugins/document/presentation/editor_plugins/emoji_picker/emoji_menu_item.dart';
  61. import 'package:appflowy_backend/protobuf/flowy-database2/field_entities.pbenum.dart';
  62. import 'package:table_calendar/table_calendar.dart';
  63. import 'base.dart';
  64. import 'common_operations.dart';
  65. import 'expectation.dart';
  66. import 'package:path/path.dart' as p;
  67. import 'mock/mock_file_picker.dart';
  68. extension AppFlowyDatabaseTest on WidgetTester {
  69. Future<void> openV020database() async {
  70. await initializeAppFlowy();
  71. await tapGoButton();
  72. // expect to see a readme page
  73. expectToSeePageName(readme);
  74. await tapAddButton();
  75. await tapImportButton();
  76. final testFileNames = ['v020.afdb'];
  77. final fileLocation = await currentFileLocation();
  78. for (final fileName in testFileNames) {
  79. final str = await rootBundle.loadString(
  80. p.join(
  81. 'assets/test/workspaces/database',
  82. fileName,
  83. ),
  84. );
  85. File(p.join(fileLocation, fileName)).writeAsStringSync(str);
  86. }
  87. // mock get files
  88. await mockPickFilePaths(testFileNames, name: 'import_files');
  89. await tapDatabaseRawDataButton();
  90. await openPage('v020');
  91. }
  92. Future<void> hoverOnFirstRowOfGrid() async {
  93. final findRow = find.byType(GridRow);
  94. expect(findRow, findsWidgets);
  95. final firstRow = findRow.first;
  96. await hoverOnWidget(firstRow);
  97. }
  98. Future<void> editCell({
  99. required int rowIndex,
  100. required FieldType fieldType,
  101. required String input,
  102. int cellIndex = 0,
  103. }) async {
  104. final cell = cellFinder(rowIndex, fieldType, cellIndex: cellIndex);
  105. expect(cell, findsOneWidget);
  106. await enterText(cell, input);
  107. await pumpAndSettle();
  108. }
  109. ///
  110. Finder cellFinder(int rowIndex, FieldType fieldType, {int cellIndex = 0}) {
  111. final findRow = find.byType(GridRow, skipOffstage: false);
  112. final findCell = finderForFieldType(fieldType);
  113. return find
  114. .descendant(
  115. of: findRow.at(rowIndex),
  116. matching: findCell,
  117. skipOffstage: false,
  118. )
  119. .at(cellIndex);
  120. }
  121. Future<void> tapCheckboxCellInGrid({
  122. required int rowIndex,
  123. }) async {
  124. final cell = cellFinder(rowIndex, FieldType.Checkbox);
  125. final button = find.descendant(
  126. of: cell,
  127. matching: find.byType(FlowyIconButton),
  128. );
  129. expect(cell, findsOneWidget);
  130. await tapButton(button);
  131. }
  132. Future<void> assertCheckboxCell({
  133. required int rowIndex,
  134. required bool isSelected,
  135. }) async {
  136. final cell = cellFinder(rowIndex, FieldType.Checkbox);
  137. var finder = find.byType(CheckboxCellUncheck);
  138. if (isSelected) {
  139. finder = find.byType(CheckboxCellCheck);
  140. }
  141. expect(
  142. find.descendant(
  143. of: cell,
  144. matching: finder,
  145. ),
  146. findsOneWidget,
  147. );
  148. }
  149. Future<void> tapCellInGrid({
  150. required int rowIndex,
  151. required FieldType fieldType,
  152. }) async {
  153. final cell = cellFinder(rowIndex, fieldType);
  154. expect(cell, findsOneWidget);
  155. await tapButton(cell, warnIfMissed: false);
  156. }
  157. /// The [fieldName] must be unique in the grid.
  158. Future<void> assertCellContent({
  159. required int rowIndex,
  160. required FieldType fieldType,
  161. required String content,
  162. int cellIndex = 0,
  163. }) async {
  164. final findCell = cellFinder(rowIndex, fieldType, cellIndex: cellIndex);
  165. final findContent = find.descendant(
  166. of: findCell,
  167. matching: find.text(content),
  168. skipOffstage: false,
  169. );
  170. final text = find.descendant(
  171. of: find.byType(TextField),
  172. matching: findContent,
  173. skipOffstage: false,
  174. );
  175. expect(text, findsOneWidget);
  176. }
  177. Future<void> assertSingleSelectOption({
  178. required int rowIndex,
  179. required String content,
  180. }) async {
  181. final findCell = cellFinder(rowIndex, FieldType.SingleSelect);
  182. if (content.isNotEmpty) {
  183. final finder = find.descendant(
  184. of: findCell,
  185. matching: find.byWidgetPredicate(
  186. (widget) => widget is SelectOptionTag && widget.name == content,
  187. ),
  188. );
  189. expect(finder, findsOneWidget);
  190. }
  191. }
  192. Future<void> assertMultiSelectOption({
  193. required int rowIndex,
  194. required List<String> contents,
  195. }) async {
  196. final findCell = cellFinder(rowIndex, FieldType.MultiSelect);
  197. for (final content in contents) {
  198. if (content.isNotEmpty) {
  199. final finder = find.descendant(
  200. of: findCell,
  201. matching: find.byWidgetPredicate(
  202. (widget) => widget is SelectOptionTag && widget.name == content,
  203. ),
  204. );
  205. expect(finder, findsOneWidget);
  206. }
  207. }
  208. }
  209. Future<void> assertChecklistCellInGrid({
  210. required int rowIndex,
  211. required double percent,
  212. }) async {
  213. final findCell = cellFinder(rowIndex, FieldType.Checklist);
  214. final finder = find.descendant(
  215. of: findCell,
  216. matching: find.byWidgetPredicate(
  217. (widget) {
  218. if (widget is ChecklistProgressBar) {
  219. return widget.percent == percent;
  220. }
  221. return false;
  222. },
  223. ),
  224. );
  225. expect(finder, findsOneWidget);
  226. }
  227. Future<void> assertDateCellInGrid({
  228. required int rowIndex,
  229. required FieldType fieldType,
  230. required String content,
  231. }) async {
  232. final findRow = find.byType(GridRow, skipOffstage: false);
  233. final findCell = find.descendant(
  234. of: findRow.at(rowIndex),
  235. matching: find.byWidgetPredicate(
  236. (widget) => widget is GridDateCell && widget.fieldType == fieldType,
  237. ),
  238. skipOffstage: false,
  239. );
  240. final dateCellText = find.descendant(
  241. of: findCell,
  242. matching: find.byType(GridDateCellText),
  243. );
  244. final text = find.descendant(
  245. of: dateCellText,
  246. matching: find.byWidgetPredicate(
  247. (widget) {
  248. if (widget is FlowyText) {
  249. return widget.text == content;
  250. }
  251. return false;
  252. },
  253. ),
  254. skipOffstage: false,
  255. );
  256. expect(text, findsOneWidget);
  257. }
  258. Future<void> selectDay({
  259. required int content,
  260. }) async {
  261. final findCalendar = find.byType(TableCalendar);
  262. final findDay = find.text(content.toString());
  263. final finder = find.descendant(
  264. of: findCalendar,
  265. matching: findDay,
  266. );
  267. // if the day is very near the beginning or the end of the month,
  268. // it may overlap with the same day in the next or previous month,
  269. // respectively because it was spilling over. This will lead to 2
  270. // widgets being found and thus cannot be tapped correctly.
  271. if (content < 15) {
  272. // e.g., Jan 2 instead of Feb 2
  273. await tapButton(finder.first);
  274. } else {
  275. // e.g. Jun 28 instead of May 28
  276. await tapButton(finder.last);
  277. }
  278. }
  279. Future<void> toggleIncludeTime() async {
  280. final findDateEditor = find.byType(DateCellEditor);
  281. final findToggle = find.byType(Toggle);
  282. final finder = find.descendant(
  283. of: findDateEditor,
  284. matching: findToggle,
  285. );
  286. await tapButton(finder);
  287. }
  288. Future<void> changeDateFormat() async {
  289. final findDateEditor = find.byType(DateCellEditor);
  290. final findDateTimeOptionButton = find.byType(DateTypeOptionButton);
  291. final finder = find.descendant(
  292. of: findDateEditor,
  293. matching: findDateTimeOptionButton,
  294. );
  295. await tapButton(finder);
  296. final findDateFormatButton = find.byType(DateFormatButton);
  297. await tapButton(findDateFormatButton);
  298. final findNewDateFormat = find.text("Day/Month/Year");
  299. await tapButton(findNewDateFormat);
  300. }
  301. Future<void> changeTimeFormat() async {
  302. final findDateEditor = find.byType(DateCellEditor);
  303. final findDateTimeOptionButton = find.byType(DateTypeOptionButton);
  304. final finder = find.descendant(
  305. of: findDateEditor,
  306. matching: findDateTimeOptionButton,
  307. );
  308. await tapButton(finder);
  309. final findDateFormatButton = find.byType(TimeFormatButton);
  310. await tapButton(findDateFormatButton);
  311. final findNewDateFormat = find.text("12 hour");
  312. await tapButton(findNewDateFormat);
  313. }
  314. Future<void> tapSelectOptionCellInGrid({
  315. required int rowIndex,
  316. required FieldType fieldType,
  317. }) async {
  318. assert(
  319. fieldType == FieldType.SingleSelect || fieldType == FieldType.MultiSelect,
  320. );
  321. final findRow = find.byType(GridRow);
  322. final findCell = finderForFieldType(fieldType);
  323. final cell = find.descendant(
  324. of: findRow.at(rowIndex),
  325. matching: findCell,
  326. );
  327. await tapButton(cell);
  328. }
  329. /// The [SelectOptionCellEditor] must be opened first.
  330. Future<void> createOption({
  331. required String name,
  332. }) async {
  333. final findEditor = find.byType(SelectOptionCellEditor);
  334. expect(findEditor, findsOneWidget);
  335. final findTextField = find.byType(SelectOptionTextField);
  336. expect(findTextField, findsOneWidget);
  337. await enterText(findTextField, name);
  338. await pump();
  339. await testTextInput.receiveAction(TextInputAction.done);
  340. await pumpAndSettle();
  341. }
  342. Future<void> selectOption({
  343. required String name,
  344. }) async {
  345. final option = find.byWidgetPredicate(
  346. (widget) => widget is SelectOptionTagCell && widget.option.name == name,
  347. );
  348. await tapButton(option);
  349. }
  350. Future<void> findSelectOptionWithNameInGrid({
  351. required int rowIndex,
  352. required String name,
  353. }) async {
  354. final findRow = find.byType(GridRow);
  355. final option = find.byWidgetPredicate(
  356. (widget) => widget is SelectOptionTag && widget.name == name,
  357. );
  358. final cell = find.descendant(
  359. of: findRow.at(rowIndex),
  360. matching: option,
  361. );
  362. expect(cell, findsOneWidget);
  363. }
  364. Future<void> assertNumberOfSelectedOptionsInGrid({
  365. required int rowIndex,
  366. required Matcher matcher,
  367. }) async {
  368. final findRow = find.byType(GridRow);
  369. final options = find.byWidgetPredicate(
  370. (widget) => widget is SelectOptionTag,
  371. );
  372. final cell = find.descendant(
  373. of: findRow.at(rowIndex),
  374. matching: options,
  375. );
  376. expect(cell, matcher);
  377. }
  378. Future<void> openFirstRowDetailPage() async {
  379. await hoverOnFirstRowOfGrid();
  380. final expandButton = find.byType(PrimaryCellAccessory);
  381. expect(expandButton, findsOneWidget);
  382. await tapButton(expandButton);
  383. }
  384. void assertRowDetailPageOpened() async {
  385. final findRowDetailPage = find.byType(RowDetailPage);
  386. expect(findRowDetailPage, findsOneWidget);
  387. }
  388. Future<void> dismissRowDetailPage() async {
  389. await sendKeyEvent(LogicalKeyboardKey.escape);
  390. await pumpAndSettle();
  391. }
  392. Future<void> editTitleInRowDetailPage(String title) async {
  393. final titleField = find.byType(GridTextCell);
  394. await enterText(titleField, title);
  395. await pumpAndSettle();
  396. }
  397. Future<void> hoverRowBanner() async {
  398. final banner = find.byType(RowBanner);
  399. expect(banner, findsOneWidget);
  400. await startGesture(
  401. getTopLeft(banner),
  402. kind: PointerDeviceKind.mouse,
  403. );
  404. await pumpAndSettle();
  405. }
  406. Future<void> openEmojiPicker() async {
  407. await tapButton(find.byType(EmojiPickerButton));
  408. await tapButton(find.byType(EmojiSelectionMenu));
  409. }
  410. Future<void> tapDateCellInRowDetailPage() async {
  411. final findDateCell = find.byType(GridDateCell);
  412. await tapButton(findDateCell);
  413. }
  414. Future<void> duplicateRowInRowDetailPage() async {
  415. final duplicateButton = find.byType(RowDetailPageDuplicateButton);
  416. await tapButton(duplicateButton);
  417. }
  418. Future<void> deleteRowInRowDetailPage() async {
  419. final deleteButton = find.byType(RowDetailPageDeleteButton);
  420. await tapButton(deleteButton);
  421. }
  422. Future<void> scrollGridByOffset(Offset offset) async {
  423. await drag(find.byType(GridPage), offset);
  424. await pumpAndSettle();
  425. }
  426. Future<void> scrollRowDetailByOffset(Offset offset) async {
  427. await drag(find.byType(RowDetailPage), offset);
  428. await pumpAndSettle();
  429. }
  430. Future<void> scrollToRight(Finder find) async {
  431. final size = getSize(find);
  432. await drag(find, Offset(-size.width, 0));
  433. await pumpAndSettle(const Duration(milliseconds: 500));
  434. }
  435. Future<void> tapNewPropertyButton() async {
  436. await tapButtonWithName(LocaleKeys.grid_field_newProperty.tr());
  437. await pumpAndSettle();
  438. }
  439. Future<void> tapGridFieldWithName(String name) async {
  440. final field = find.byWidgetPredicate(
  441. (widget) => widget is FieldCellButton && widget.field.name == name,
  442. );
  443. await tapButton(field);
  444. await pumpAndSettle();
  445. }
  446. /// Should call [tapGridFieldWithName] first.
  447. Future<void> tapEditPropertyButton() async {
  448. await tapButtonWithName(LocaleKeys.grid_field_editProperty.tr());
  449. await pumpAndSettle(const Duration(milliseconds: 200));
  450. }
  451. /// Should call [tapGridFieldWithName] first.
  452. Future<void> tapDeletePropertyButton() async {
  453. final field = find.byWidgetPredicate(
  454. (widget) =>
  455. widget is FieldActionCell && widget.action == FieldAction.delete,
  456. );
  457. await tapButton(field);
  458. }
  459. /// Should call [tapGridFieldWithName] first.
  460. Future<void> tapDialogOkButton() async {
  461. final field = find.byWidgetPredicate(
  462. (widget) =>
  463. widget is PrimaryTextButton &&
  464. widget.label == LocaleKeys.button_OK.tr(),
  465. );
  466. await tapButton(field);
  467. }
  468. /// Should call [tapGridFieldWithName] first.
  469. Future<void> tapDuplicatePropertyButton() async {
  470. final field = find.byWidgetPredicate(
  471. (widget) =>
  472. widget is FieldActionCell && widget.action == FieldAction.duplicate,
  473. );
  474. await tapButton(field);
  475. }
  476. /// Should call [tapGridFieldWithName] first.
  477. Future<void> tapHidePropertyButton() async {
  478. final field = find.byWidgetPredicate(
  479. (widget) =>
  480. widget is FieldActionCell && widget.action == FieldAction.hide,
  481. );
  482. await tapButton(field);
  483. }
  484. Future<void> tapRowDetailPageCreatePropertyButton() async {
  485. await tapButton(find.byType(CreateRowFieldButton));
  486. }
  487. Future<void> tapRowDetailPageDeleteRowButton() async {
  488. await tapButton(find.byType(RowDetailPageDeleteButton));
  489. }
  490. Future<void> tapRowDetailPageDuplicateRowButton() async {
  491. await tapButton(find.byType(RowDetailPageDuplicateButton));
  492. }
  493. Future<void> tapTypeOptionButton() async {
  494. await tapButton(find.byType(SwitchFieldButton));
  495. }
  496. Future<void> tapEscButton() async {
  497. await sendKeyEvent(LogicalKeyboardKey.escape);
  498. }
  499. /// Must call [tapTypeOptionButton] first.
  500. Future<void> selectFieldType(FieldType fieldType) async {
  501. final fieldTypeCell = find.byType(FieldTypeCell);
  502. final fieldTypeButton = find.descendant(
  503. of: fieldTypeCell,
  504. matching: find.byWidgetPredicate(
  505. (widget) => widget is FlowyText && widget.text == fieldType.title(),
  506. ),
  507. );
  508. await tapButton(fieldTypeButton);
  509. }
  510. /// Each field has its own cell, so we can find the corresponding cell by
  511. /// the field type after create a new field.
  512. Future<void> findCellByFieldType(FieldType fieldType) async {
  513. final finder = finderForFieldType(fieldType);
  514. expect(finder, findsWidgets);
  515. }
  516. Future<void> assertNumberOfFieldsInGridPage(int num) async {
  517. expect(find.byType(GridFieldCell), findsNWidgets(num));
  518. }
  519. Future<void> assertNumberOfRowsInGridPage(int num) async {
  520. expect(
  521. find.byType(GridRow, skipOffstage: false),
  522. findsNWidgets(num),
  523. );
  524. }
  525. Future<void> assertDocumentExistInRowDetailPage() async {
  526. expect(find.byType(RowDocument), findsOneWidget);
  527. }
  528. /// Check the field type of the [FieldCellButton] is the same as the name.
  529. Future<void> assertFieldTypeWithFieldName(
  530. String name,
  531. FieldType fieldType,
  532. ) async {
  533. final field = find.byWidgetPredicate(
  534. (widget) =>
  535. widget is FieldCellButton &&
  536. widget.field.fieldType == fieldType &&
  537. widget.field.name == name,
  538. );
  539. expect(field, findsOneWidget);
  540. }
  541. Future<void> findFieldWithName(String name) async {
  542. final field = find.byWidgetPredicate(
  543. (widget) => widget is FieldCellButton && widget.field.name == name,
  544. );
  545. expect(field, findsOneWidget);
  546. }
  547. Future<void> noFieldWithName(String name) async {
  548. final field = find.byWidgetPredicate(
  549. (widget) => widget is FieldCellButton && widget.field.name == name,
  550. );
  551. expect(field, findsNothing);
  552. }
  553. Future<void> renameField(String newName) async {
  554. final textField = find.byType(FieldNameTextField);
  555. expect(textField, findsOneWidget);
  556. await enterText(textField, newName);
  557. await pumpAndSettle();
  558. }
  559. Future<void> dismissFieldEditor() async {
  560. await sendKeyEvent(LogicalKeyboardKey.escape);
  561. await pumpAndSettle(const Duration(milliseconds: 200));
  562. }
  563. Future<void> findFieldEditor(dynamic matcher) async {
  564. final finder = find.byType(FieldEditor);
  565. expect(finder, matcher);
  566. }
  567. Future<void> findDateEditor(dynamic matcher) async {
  568. final finder = find.byType(DateCellEditor);
  569. expect(finder, matcher);
  570. }
  571. Future<void> findSelectOptionEditor(dynamic matcher) async {
  572. final finder = find.byType(SelectOptionCellEditor);
  573. expect(finder, matcher);
  574. }
  575. Future<void> dismissCellEditor() async {
  576. await sendKeyEvent(LogicalKeyboardKey.escape);
  577. await pumpAndSettle();
  578. }
  579. Future<void> tapCreateRowButtonInGrid() async {
  580. await tapButton(find.byType(GridAddRowButton));
  581. }
  582. Future<void> tapCreateRowButtonInRowMenuOfGrid() async {
  583. await tapButton(find.byType(InsertRowButton));
  584. }
  585. Future<void> tapRowMenuButtonInGrid() async {
  586. await tapButton(find.byType(RowMenuButton));
  587. }
  588. /// Should call [tapRowMenuButtonInGrid] first.
  589. Future<void> tapDeleteOnRowMenu() async {
  590. await tapButtonWithName(LocaleKeys.grid_row_delete.tr());
  591. }
  592. Future<void> assertRowCountInGridPage(int num) async {
  593. final text = find.byWidgetPredicate(
  594. (widget) => widget is FlowyText && widget.text == rowCountString(num),
  595. );
  596. expect(text, findsOneWidget);
  597. }
  598. Future<void> createField(FieldType fieldType, String name) async {
  599. await scrollToRight(find.byType(GridPage));
  600. await tapNewPropertyButton();
  601. await renameField(name);
  602. await tapTypeOptionButton();
  603. await selectFieldType(fieldType);
  604. await dismissFieldEditor();
  605. }
  606. Future<void> tapDatabaseSettingButton() async {
  607. await tapButton(find.byType(SettingButton));
  608. }
  609. Future<void> tapDatabaseFilterButton() async {
  610. await tapButton(find.byType(FilterButton));
  611. }
  612. Future<void> tapDatabaseSortButton() async {
  613. await tapButton(find.byType(SortButton));
  614. }
  615. Future<void> tapCreateFilterByFieldType(
  616. FieldType fieldType,
  617. String title,
  618. ) async {
  619. final findFilter = find.byWidgetPredicate(
  620. (widget) =>
  621. widget is GridFilterPropertyCell &&
  622. widget.fieldInfo.fieldType == fieldType &&
  623. widget.fieldInfo.name == title,
  624. );
  625. await tapButton(findFilter);
  626. }
  627. Future<void> tapFilterButtonInGrid(String filterName) async {
  628. final findFilter = find.byType(FilterMenuItem);
  629. final button = find.descendant(
  630. of: findFilter,
  631. matching: find.text(filterName),
  632. );
  633. await tapButton(button);
  634. }
  635. Future<void> tapCreateSortByFieldType(
  636. FieldType fieldType,
  637. String title,
  638. ) async {
  639. final findSort = find.byWidgetPredicate(
  640. (widget) =>
  641. widget is GridSortPropertyCell &&
  642. widget.fieldInfo.fieldType == fieldType &&
  643. widget.fieldInfo.name == title,
  644. );
  645. await tapButton(findSort);
  646. }
  647. // Must call [tapSortMenuInSettingBar] first.
  648. Future<void> tapCreateSortByFieldTypeInSortMenu(
  649. FieldType fieldType,
  650. String title,
  651. ) async {
  652. await tapButton(find.byType(DatabaseAddSortButton));
  653. final findSort = find.byWidgetPredicate(
  654. (widget) =>
  655. widget is GridSortPropertyCell &&
  656. widget.fieldInfo.fieldType == fieldType &&
  657. widget.fieldInfo.name == title,
  658. );
  659. await tapButton(findSort);
  660. await pumpAndSettle();
  661. }
  662. Future<void> tapSortMenuInSettingBar() async {
  663. await tapButton(find.byType(SortMenu));
  664. await pumpAndSettle();
  665. }
  666. /// Must call [tapSortMenuInSettingBar] first.
  667. Future<void> tapSortButtonByName(String name) async {
  668. final findSortItem = find.byWidgetPredicate(
  669. (widget) =>
  670. widget is DatabaseSortItem && widget.sortInfo.fieldInfo.name == name,
  671. );
  672. await tapButton(findSortItem);
  673. }
  674. /// Must call [tapSortButtonByName] first.
  675. Future<void> tapSortByDescending() async {
  676. await tapButton(
  677. find.descendant(
  678. of: find.byType(OrderPannelItem),
  679. matching: find.byWidgetPredicate(
  680. (widget) =>
  681. widget is FlowyText &&
  682. widget.text == LocaleKeys.grid_sort_descending.tr(),
  683. ),
  684. ),
  685. );
  686. await sendKeyEvent(LogicalKeyboardKey.escape);
  687. await pumpAndSettle();
  688. }
  689. /// Must call [tapSortMenuInSettingBar] first.
  690. Future<void> tapAllSortButton() async {
  691. await tapButton(find.byType(DatabaseDeleteSortButton));
  692. }
  693. Future<void> scrollOptionFilterListByOffset(Offset offset) async {
  694. await drag(find.byType(SelectOptionFilterEditor), offset);
  695. await pumpAndSettle();
  696. }
  697. Future<void> enterTextInTextFilter(String text) async {
  698. final findEditor = find.byType(TextFilterEditor);
  699. final findTextField = find.descendant(
  700. of: findEditor,
  701. matching: find.byType(FlowyTextField),
  702. );
  703. await enterText(findTextField, text);
  704. await pumpAndSettle(const Duration(milliseconds: 300));
  705. }
  706. Future<void> tapDisclosureButtonInFinder(Finder finder) async {
  707. final findDisclosure = find.descendant(
  708. of: finder,
  709. matching: find.byType(DisclosureButton),
  710. );
  711. await tapButton(findDisclosure);
  712. }
  713. /// must call [tapDisclosureButtonInFinder] first.
  714. Future<void> tapDeleteFilterButtonInGrid() async {
  715. await tapButton(find.text(LocaleKeys.grid_settings_deleteFilter.tr()));
  716. }
  717. Future<void> tapCheckboxFilterButtonInGrid() async {
  718. await tapButton(find.byType(CheckboxFilterConditionList));
  719. }
  720. Future<void> tapChecklistFilterButtonInGrid() async {
  721. await tapButton(find.byType(ChecklistFilterConditionList));
  722. }
  723. /// The [SelectOptionFilterList] must show up first.
  724. Future<void> tapOptionFilterWithName(String name) async {
  725. final findCell = find.descendant(
  726. of: find.byType(SelectOptionFilterList),
  727. matching: find.byWidgetPredicate(
  728. (widget) =>
  729. widget is SelectOptionFilterCell && widget.option.name == name,
  730. skipOffstage: false,
  731. ),
  732. skipOffstage: false,
  733. );
  734. expect(findCell, findsOneWidget);
  735. await tapButton(findCell, warnIfMissed: false);
  736. }
  737. Future<void> tapCheckedButtonOnCheckboxFilter() async {
  738. final button = find.descendant(
  739. of: find.byType(HoverButton),
  740. matching: find.text(LocaleKeys.grid_checkboxFilter_isChecked.tr()),
  741. );
  742. await tapButton(button);
  743. }
  744. Future<void> tapUnCheckedButtonOnCheckboxFilter() async {
  745. final button = find.descendant(
  746. of: find.byType(HoverButton),
  747. matching: find.text(LocaleKeys.grid_checkboxFilter_isUnchecked.tr()),
  748. );
  749. await tapButton(button);
  750. }
  751. Future<void> tapCompletedButtonOnChecklistFilter() async {
  752. final button = find.descendant(
  753. of: find.byType(HoverButton),
  754. matching: find.text(LocaleKeys.grid_checklistFilter_isComplete.tr()),
  755. );
  756. await tapButton(button);
  757. }
  758. Future<void> tapUnCompletedButtonOnChecklistFilter() async {
  759. final button = find.descendant(
  760. of: find.byType(HoverButton),
  761. matching: find.text(LocaleKeys.grid_checklistFilter_isIncomplted.tr()),
  762. );
  763. await tapButton(button);
  764. }
  765. /// Should call [tapDatabaseSettingButton] first.
  766. Future<void> tapDatabaseLayoutButton() async {
  767. final findSettingItem = find.byType(DatabaseSettingItem);
  768. final findLayoutButton = find.byWidgetPredicate(
  769. (widget) =>
  770. widget is FlowyText &&
  771. widget.text == DatabaseSettingAction.showLayout.title(),
  772. );
  773. final button = find.descendant(
  774. of: findSettingItem,
  775. matching: findLayoutButton,
  776. );
  777. await tapButton(button);
  778. }
  779. Future<void> tapCalendarLayoutSettingButton() async {
  780. final findSettingItem = find.byType(DatabaseSettingItem);
  781. final findLayoutButton = find.byWidgetPredicate(
  782. (widget) =>
  783. widget is FlowyText &&
  784. widget.text == DatabaseSettingAction.showCalendarLayout.title(),
  785. );
  786. final button = find.descendant(
  787. of: findSettingItem,
  788. matching: findLayoutButton,
  789. );
  790. await tapButton(button);
  791. }
  792. Future<void> tapFirstDayOfWeek() async {
  793. await tapButton(find.byType(FirstDayOfWeek));
  794. }
  795. Future<void> tapFirstDayOfWeekStartFromSunday() async {
  796. final finder = find.byWidgetPredicate(
  797. (widget) => widget is StartFromButton && widget.dayIndex == 0,
  798. );
  799. await tapButton(finder);
  800. }
  801. Future<void> tapFirstDayOfWeekStartFromMonday() async {
  802. final finder = find.byWidgetPredicate(
  803. (widget) => widget is StartFromButton && widget.dayIndex == 1,
  804. );
  805. await tapButton(finder);
  806. // Dismiss the popover overlay in cause of obscure the tapButton
  807. // in the next test case.
  808. await sendKeyEvent(LogicalKeyboardKey.escape);
  809. await pumpAndSettle(const Duration(milliseconds: 200));
  810. }
  811. void assertFirstDayOfWeekStartFromMonday() {
  812. final finder = find.byWidgetPredicate(
  813. (widget) =>
  814. widget is StartFromButton &&
  815. widget.dayIndex == 1 &&
  816. widget.isSelected == true,
  817. );
  818. expect(finder, findsOneWidget);
  819. }
  820. void assertFirstDayOfWeekStartFromSunday() {
  821. final finder = find.byWidgetPredicate(
  822. (widget) =>
  823. widget is StartFromButton &&
  824. widget.dayIndex == 0 &&
  825. widget.isSelected == true,
  826. );
  827. expect(finder, findsOneWidget);
  828. }
  829. Future<void> scrollToToday() async {
  830. final todayCell = find.byWidgetPredicate(
  831. (widget) => widget is CalendarDayCard && widget.isToday,
  832. skipOffstage: false,
  833. );
  834. await ensureVisible(todayCell);
  835. await pumpAndSettle(const Duration(milliseconds: 300));
  836. }
  837. Future<void> hoverOnTodayCalendarCell() async {
  838. final todayCell = find.byWidgetPredicate(
  839. (widget) => widget is CalendarDayCard && widget.isToday,
  840. );
  841. await hoverOnWidget(todayCell);
  842. }
  843. Future<void> tapAddCalendarEventButton() async {
  844. final findFlowyButton = find.byType(FlowyIconButton);
  845. final findNewEventButton = find.byType(NewEventButton);
  846. final button = find.descendant(
  847. of: findNewEventButton,
  848. matching: findFlowyButton,
  849. );
  850. await tapButton(button);
  851. }
  852. /// Checks for a certain number of events. Parameters [date] and [title] can
  853. /// also be provided to restrict the scope of the search
  854. void assertNumberOfEventsInCalendar(int number, {String? title}) {
  855. Finder findEvents = find.byType(EventCard);
  856. if (title != null) {
  857. findEvents = find.descendant(of: findEvents, matching: find.text(title));
  858. }
  859. expect(findEvents, findsNWidgets(number));
  860. }
  861. void assertNumberofEventsOnSpecificDay(
  862. int number,
  863. DateTime date, {
  864. String? title,
  865. }) {
  866. final findDayCell = find.byWidgetPredicate(
  867. (widget) =>
  868. widget is CalendarDayCard &&
  869. isSameDay(
  870. widget.date,
  871. date,
  872. ),
  873. );
  874. Finder findEvents = find.descendant(
  875. of: findDayCell,
  876. matching: find.byType(EventCard),
  877. );
  878. if (title != null) {
  879. findEvents = find.descendant(of: findEvents, matching: find.text(title));
  880. }
  881. expect(findEvents, findsNWidgets(number));
  882. }
  883. Future<void> doubleClickCalendarCell(DateTime date) async {
  884. final todayCell = find.byWidgetPredicate(
  885. (widget) => widget is CalendarDayCard && isSameDay(date, widget.date),
  886. );
  887. await doubleTapButton(todayCell);
  888. }
  889. Future<void> openCalendarEvent({required index, DateTime? date}) async {
  890. final findDayCell = find.byWidgetPredicate(
  891. (widget) =>
  892. widget is CalendarDayCard &&
  893. isSameDay(widget.date, date ?? DateTime.now()),
  894. );
  895. final cards = find.descendant(
  896. of: findDayCell,
  897. matching: find.byType(EventCard),
  898. );
  899. await tapButton(cards.at(index));
  900. }
  901. Future<void> dragDropRescheduleCalendarEvent(DateTime startDate) async {
  902. final findEventCard = find.byType(EventCard);
  903. await drag(findEventCard.first, const Offset(0, 300));
  904. await pumpAndSettle();
  905. }
  906. Future<void> tapCreateLinkedDatabaseViewButton(AddButtonAction action) async {
  907. final findAddButton = find.byType(AddDatabaseViewButton);
  908. await tapButton(findAddButton);
  909. final findCreateButton = find.byWidgetPredicate(
  910. (widget) =>
  911. widget is TarBarAddButtonActionCell && widget.action == action,
  912. );
  913. await tapButton(findCreateButton);
  914. }
  915. Finder findTabBarLinkViewByViewLayout(ViewLayoutPB layout) {
  916. return find.byWidgetPredicate(
  917. (widget) => widget is TabBarItemButton && widget.view.layout == layout,
  918. );
  919. }
  920. Finder findTabBarLinkViewByViewName(String name) {
  921. return find.byWidgetPredicate(
  922. (widget) => widget is TabBarItemButton && widget.view.name == name,
  923. );
  924. }
  925. Future<void> renameLinkedView(Finder linkedView, String name) async {
  926. await tap(linkedView, buttons: kSecondaryButton);
  927. await pumpAndSettle();
  928. await tapButton(
  929. find.byWidgetPredicate(
  930. (widget) =>
  931. widget is ActionCellWidget &&
  932. widget.action == TabBarViewAction.rename,
  933. ),
  934. );
  935. await enterText(
  936. find.descendant(
  937. of: find.byType(FlowyFormTextInput),
  938. matching: find.byType(TextFormField),
  939. ),
  940. name,
  941. );
  942. final field = find.byWidgetPredicate(
  943. (widget) =>
  944. widget is PrimaryTextButton &&
  945. widget.label == LocaleKeys.button_OK.tr(),
  946. );
  947. await tapButton(field);
  948. }
  949. Future<void> deleteDatebaseView(Finder linkedView) async {
  950. await tap(linkedView, buttons: kSecondaryButton);
  951. await pumpAndSettle();
  952. await tapButton(
  953. find.byWidgetPredicate(
  954. (widget) =>
  955. widget is ActionCellWidget &&
  956. widget.action == TabBarViewAction.delete,
  957. ),
  958. );
  959. final okButton = find.byWidgetPredicate(
  960. (widget) =>
  961. widget is PrimaryTextButton &&
  962. widget.label == LocaleKeys.button_OK.tr(),
  963. );
  964. await tapButton(okButton);
  965. }
  966. Future<void> assertCurrentDatabaseTagIs(DatabaseLayoutPB layout) async {
  967. switch (layout) {
  968. case DatabaseLayoutPB.Board:
  969. expect(find.byType(BoardPage), findsOneWidget);
  970. break;
  971. case DatabaseLayoutPB.Calendar:
  972. expect(find.byType(CalendarPage), findsOneWidget);
  973. break;
  974. case DatabaseLayoutPB.Grid:
  975. expect(find.byType(GridPage), findsOneWidget);
  976. break;
  977. default:
  978. throw Exception('Unknown database layout type: $layout');
  979. }
  980. }
  981. Future<void> selectDatabaseLayoutType(DatabaseLayoutPB layout) async {
  982. final findLayoutCell = find.byType(DatabaseViewLayoutCell);
  983. final findText = find.byWidgetPredicate(
  984. (widget) => widget is FlowyText && widget.text == layout.layoutName(),
  985. );
  986. final button = find.descendant(
  987. of: findLayoutCell,
  988. matching: findText,
  989. );
  990. await tapButton(button);
  991. }
  992. Future<void> assertCurrentDatabaseLayoutType(DatabaseLayoutPB layout) async {
  993. expect(finderForDatabaseLayoutType(layout), findsOneWidget);
  994. }
  995. Future<void> tapDatabaseRawDataButton() async {
  996. await tapButtonWithName(LocaleKeys.importPanel_database.tr());
  997. }
  998. }
  999. Finder finderForDatabaseLayoutType(DatabaseLayoutPB layout) {
  1000. switch (layout) {
  1001. case DatabaseLayoutPB.Board:
  1002. return find.byType(BoardPage);
  1003. case DatabaseLayoutPB.Calendar:
  1004. return find.byType(CalendarPage);
  1005. case DatabaseLayoutPB.Grid:
  1006. return find.byType(GridPage);
  1007. default:
  1008. throw Exception('Unknown database layout type: $layout');
  1009. }
  1010. }
  1011. Finder finderForFieldType(FieldType fieldType) {
  1012. switch (fieldType) {
  1013. case FieldType.Checkbox:
  1014. return find.byType(GridCheckboxCell, skipOffstage: false);
  1015. case FieldType.DateTime:
  1016. return find.byType(GridDateCell, skipOffstage: false);
  1017. case FieldType.LastEditedTime:
  1018. case FieldType.CreatedTime:
  1019. return find.byType(GridDateCell, skipOffstage: false);
  1020. case FieldType.SingleSelect:
  1021. return find.byType(GridSingleSelectCell, skipOffstage: false);
  1022. case FieldType.MultiSelect:
  1023. return find.byType(GridMultiSelectCell, skipOffstage: false);
  1024. case FieldType.Checklist:
  1025. return find.byType(GridChecklistCell, skipOffstage: false);
  1026. case FieldType.Number:
  1027. return find.byType(GridNumberCell, skipOffstage: false);
  1028. case FieldType.RichText:
  1029. return find.byType(GridTextCell, skipOffstage: false);
  1030. case FieldType.URL:
  1031. return find.byType(GridURLCell, skipOffstage: false);
  1032. default:
  1033. throw Exception('Unknown field type: $fieldType');
  1034. }
  1035. }