database_test_op.dart 31 KB

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