database_test_op.dart 37 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202
  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. await tapButton(finder);
  268. }
  269. Future<void> toggleIncludeTime() async {
  270. final findDateEditor = find.byType(DateCellEditor);
  271. final findToggle = find.byType(Toggle);
  272. final finder = find.descendant(
  273. of: findDateEditor,
  274. matching: findToggle,
  275. );
  276. await tapButton(finder);
  277. }
  278. Future<void> changeDateFormat() async {
  279. final findDateEditor = find.byType(DateCellEditor);
  280. final findDateTimeOptionButton = find.byType(DateTypeOptionButton);
  281. final finder = find.descendant(
  282. of: findDateEditor,
  283. matching: findDateTimeOptionButton,
  284. );
  285. await tapButton(finder);
  286. final findDateFormatButton = find.byType(DateFormatButton);
  287. await tapButton(findDateFormatButton);
  288. final findNewDateFormat = find.text("Day/Month/Year");
  289. await tapButton(findNewDateFormat);
  290. }
  291. Future<void> changeTimeFormat() async {
  292. final findDateEditor = find.byType(DateCellEditor);
  293. final findDateTimeOptionButton = find.byType(DateTypeOptionButton);
  294. final finder = find.descendant(
  295. of: findDateEditor,
  296. matching: findDateTimeOptionButton,
  297. );
  298. await tapButton(finder);
  299. final findDateFormatButton = find.byType(TimeFormatButton);
  300. await tapButton(findDateFormatButton);
  301. final findNewDateFormat = find.text("12 hour");
  302. await tapButton(findNewDateFormat);
  303. }
  304. Future<void> tapSelectOptionCellInGrid({
  305. required int rowIndex,
  306. required FieldType fieldType,
  307. }) async {
  308. assert(
  309. fieldType == FieldType.SingleSelect || fieldType == FieldType.MultiSelect,
  310. );
  311. final findRow = find.byType(GridRow);
  312. final findCell = finderForFieldType(fieldType);
  313. final cell = find.descendant(
  314. of: findRow.at(rowIndex),
  315. matching: findCell,
  316. );
  317. await tapButton(cell);
  318. }
  319. /// The [SelectOptionCellEditor] must be opened first.
  320. Future<void> createOption({
  321. required String name,
  322. }) async {
  323. final findEditor = find.byType(SelectOptionCellEditor);
  324. expect(findEditor, findsOneWidget);
  325. final findTextField = find.byType(SelectOptionTextField);
  326. expect(findTextField, findsOneWidget);
  327. await enterText(findTextField, name);
  328. await pump();
  329. await testTextInput.receiveAction(TextInputAction.done);
  330. await pumpAndSettle();
  331. }
  332. Future<void> selectOption({
  333. required String name,
  334. }) async {
  335. final option = find.byWidgetPredicate(
  336. (widget) => widget is SelectOptionTagCell && widget.option.name == name,
  337. );
  338. await tapButton(option);
  339. }
  340. Future<void> findSelectOptionWithNameInGrid({
  341. required int rowIndex,
  342. required String name,
  343. }) async {
  344. final findRow = find.byType(GridRow);
  345. final option = find.byWidgetPredicate(
  346. (widget) => widget is SelectOptionTag && widget.name == name,
  347. );
  348. final cell = find.descendant(
  349. of: findRow.at(rowIndex),
  350. matching: option,
  351. );
  352. expect(cell, findsOneWidget);
  353. }
  354. Future<void> assertNumberOfSelectedOptionsInGrid({
  355. required int rowIndex,
  356. required Matcher matcher,
  357. }) async {
  358. final findRow = find.byType(GridRow);
  359. final options = find.byWidgetPredicate(
  360. (widget) => widget is SelectOptionTag,
  361. );
  362. final cell = find.descendant(
  363. of: findRow.at(rowIndex),
  364. matching: options,
  365. );
  366. expect(cell, matcher);
  367. }
  368. Future<void> openFirstRowDetailPage() async {
  369. await hoverOnFirstRowOfGrid();
  370. final expandButton = find.byType(PrimaryCellAccessory);
  371. expect(expandButton, findsOneWidget);
  372. await tapButton(expandButton);
  373. }
  374. void assertRowDetailPageOpened() async {
  375. final findRowDetailPage = find.byType(RowDetailPage);
  376. expect(findRowDetailPage, findsOneWidget);
  377. }
  378. Future<void> dismissRowDetailPage() async {
  379. await sendKeyEvent(LogicalKeyboardKey.escape);
  380. await pumpAndSettle();
  381. }
  382. Future<void> editTitleInRowDetailPage(String title) async {
  383. final titleField = find.byType(GridTextCell);
  384. await enterText(titleField, title);
  385. await pumpAndSettle();
  386. }
  387. Future<void> hoverRowBanner() async {
  388. final banner = find.byType(RowBanner);
  389. expect(banner, findsOneWidget);
  390. await startGesture(
  391. getTopLeft(banner),
  392. kind: PointerDeviceKind.mouse,
  393. );
  394. await pumpAndSettle();
  395. }
  396. Future<void> openEmojiPicker() async {
  397. await tapButton(find.byType(EmojiPickerButton));
  398. await tapButton(find.byType(EmojiSelectionMenu));
  399. }
  400. Future<void> tapDateCellInRowDetailPage() async {
  401. final findDateCell = find.byType(GridDateCell);
  402. await tapButton(findDateCell);
  403. }
  404. Future<void> duplicateRowInRowDetailPage() async {
  405. final duplicateButton = find.byType(RowDetailPageDuplicateButton);
  406. await tapButton(duplicateButton);
  407. }
  408. Future<void> deleteRowInRowDetailPage() async {
  409. final deleteButton = find.byType(RowDetailPageDeleteButton);
  410. await tapButton(deleteButton);
  411. }
  412. Future<void> scrollGridByOffset(Offset offset) async {
  413. await drag(find.byType(GridPage), offset);
  414. await pumpAndSettle();
  415. }
  416. Future<void> scrollRowDetailByOffset(Offset offset) async {
  417. await drag(find.byType(RowDetailPage), offset);
  418. await pumpAndSettle();
  419. }
  420. Future<void> scrollToRight(Finder find) async {
  421. final size = getSize(find);
  422. await drag(find, Offset(-size.width, 0));
  423. await pumpAndSettle(const Duration(milliseconds: 500));
  424. }
  425. Future<void> tapNewPropertyButton() async {
  426. await tapButtonWithName(LocaleKeys.grid_field_newProperty.tr());
  427. await pumpAndSettle();
  428. }
  429. Future<void> tapGridFieldWithName(String name) async {
  430. final field = find.byWidgetPredicate(
  431. (widget) => widget is FieldCellButton && widget.field.name == name,
  432. );
  433. await tapButton(field);
  434. await pumpAndSettle();
  435. }
  436. /// Should call [tapGridFieldWithName] first.
  437. Future<void> tapEditPropertyButton() async {
  438. await tapButtonWithName(LocaleKeys.grid_field_editProperty.tr());
  439. await pumpAndSettle(const Duration(milliseconds: 200));
  440. }
  441. /// Should call [tapGridFieldWithName] first.
  442. Future<void> tapDeletePropertyButton() async {
  443. final field = find.byWidgetPredicate(
  444. (widget) =>
  445. widget is FieldActionCell && widget.action == FieldAction.delete,
  446. );
  447. await tapButton(field);
  448. }
  449. /// Should call [tapGridFieldWithName] first.
  450. Future<void> tapDialogOkButton() async {
  451. final field = find.byWidgetPredicate(
  452. (widget) =>
  453. widget is PrimaryTextButton &&
  454. widget.label == LocaleKeys.button_OK.tr(),
  455. );
  456. await tapButton(field);
  457. }
  458. /// Should call [tapGridFieldWithName] first.
  459. Future<void> tapDuplicatePropertyButton() async {
  460. final field = find.byWidgetPredicate(
  461. (widget) =>
  462. widget is FieldActionCell && widget.action == FieldAction.duplicate,
  463. );
  464. await tapButton(field);
  465. }
  466. /// Should call [tapGridFieldWithName] first.
  467. Future<void> tapHidePropertyButton() async {
  468. final field = find.byWidgetPredicate(
  469. (widget) =>
  470. widget is FieldActionCell && widget.action == FieldAction.hide,
  471. );
  472. await tapButton(field);
  473. }
  474. Future<void> tapRowDetailPageCreatePropertyButton() async {
  475. await tapButton(find.byType(CreateRowFieldButton));
  476. }
  477. Future<void> tapRowDetailPageDeleteRowButton() async {
  478. await tapButton(find.byType(RowDetailPageDeleteButton));
  479. }
  480. Future<void> tapRowDetailPageDuplicateRowButton() async {
  481. await tapButton(find.byType(RowDetailPageDuplicateButton));
  482. }
  483. Future<void> tapTypeOptionButton() async {
  484. await tapButton(find.byType(SwitchFieldButton));
  485. }
  486. Future<void> tapEscButton() async {
  487. await sendKeyEvent(LogicalKeyboardKey.escape);
  488. }
  489. /// Must call [tapTypeOptionButton] first.
  490. Future<void> selectFieldType(FieldType fieldType) async {
  491. final fieldTypeCell = find.byType(FieldTypeCell);
  492. final fieldTypeButton = find.descendant(
  493. of: fieldTypeCell,
  494. matching: find.byWidgetPredicate(
  495. (widget) => widget is FlowyText && widget.text == fieldType.title(),
  496. ),
  497. );
  498. await tapButton(fieldTypeButton);
  499. }
  500. /// Each field has its own cell, so we can find the corresponding cell by
  501. /// the field type after create a new field.
  502. Future<void> findCellByFieldType(FieldType fieldType) async {
  503. final finder = finderForFieldType(fieldType);
  504. expect(finder, findsWidgets);
  505. }
  506. Future<void> assertNumberOfFieldsInGridPage(int num) async {
  507. expect(find.byType(GridFieldCell), findsNWidgets(num));
  508. }
  509. Future<void> assertNumberOfRowsInGridPage(int num) async {
  510. expect(
  511. find.byType(GridRow, skipOffstage: false),
  512. findsNWidgets(num),
  513. );
  514. }
  515. Future<void> assertDocumentExistInRowDetailPage() async {
  516. expect(find.byType(RowDocument), findsOneWidget);
  517. }
  518. /// Check the field type of the [FieldCellButton] is the same as the name.
  519. Future<void> assertFieldTypeWithFieldName(
  520. String name,
  521. FieldType fieldType,
  522. ) async {
  523. final field = find.byWidgetPredicate(
  524. (widget) =>
  525. widget is FieldCellButton &&
  526. widget.field.fieldType == fieldType &&
  527. widget.field.name == name,
  528. );
  529. expect(field, findsOneWidget);
  530. }
  531. Future<void> findFieldWithName(String name) async {
  532. final field = find.byWidgetPredicate(
  533. (widget) => widget is FieldCellButton && widget.field.name == name,
  534. );
  535. expect(field, findsOneWidget);
  536. }
  537. Future<void> noFieldWithName(String name) async {
  538. final field = find.byWidgetPredicate(
  539. (widget) => widget is FieldCellButton && widget.field.name == name,
  540. );
  541. expect(field, findsNothing);
  542. }
  543. Future<void> renameField(String newName) async {
  544. final textField = find.byType(FieldNameTextField);
  545. expect(textField, findsOneWidget);
  546. await enterText(textField, newName);
  547. await pumpAndSettle();
  548. }
  549. Future<void> dismissFieldEditor() async {
  550. await sendKeyEvent(LogicalKeyboardKey.escape);
  551. await pumpAndSettle(const Duration(milliseconds: 200));
  552. }
  553. Future<void> findFieldEditor(dynamic matcher) async {
  554. final finder = find.byType(FieldEditor);
  555. expect(finder, matcher);
  556. }
  557. Future<void> findDateEditor(dynamic matcher) async {
  558. final finder = find.byType(DateCellEditor);
  559. expect(finder, matcher);
  560. }
  561. Future<void> findSelectOptionEditor(dynamic matcher) async {
  562. final finder = find.byType(SelectOptionCellEditor);
  563. expect(finder, matcher);
  564. }
  565. Future<void> dismissCellEditor() async {
  566. await sendKeyEvent(LogicalKeyboardKey.escape);
  567. await pumpAndSettle();
  568. }
  569. Future<void> tapCreateRowButtonInGrid() async {
  570. await tapButton(find.byType(GridAddRowButton));
  571. }
  572. Future<void> tapCreateRowButtonInRowMenuOfGrid() async {
  573. await tapButton(find.byType(InsertRowButton));
  574. }
  575. Future<void> tapRowMenuButtonInGrid() async {
  576. await tapButton(find.byType(RowMenuButton));
  577. }
  578. /// Should call [tapRowMenuButtonInGrid] first.
  579. Future<void> tapDeleteOnRowMenu() async {
  580. await tapButtonWithName(LocaleKeys.grid_row_delete.tr());
  581. }
  582. Future<void> assertRowCountInGridPage(int num) async {
  583. final text = find.byWidgetPredicate(
  584. (widget) => widget is FlowyText && widget.text == rowCountString(num),
  585. );
  586. expect(text, findsOneWidget);
  587. }
  588. Future<void> createField(FieldType fieldType, String name) async {
  589. await scrollToRight(find.byType(GridPage));
  590. await tapNewPropertyButton();
  591. await renameField(name);
  592. await tapTypeOptionButton();
  593. await selectFieldType(fieldType);
  594. await dismissFieldEditor();
  595. }
  596. Future<void> tapDatabaseSettingButton() async {
  597. await tapButton(find.byType(SettingButton));
  598. }
  599. Future<void> tapDatabaseFilterButton() async {
  600. await tapButton(find.byType(FilterButton));
  601. }
  602. Future<void> tapDatabaseSortButton() async {
  603. await tapButton(find.byType(SortButton));
  604. }
  605. Future<void> tapCreateFilterByFieldType(
  606. FieldType fieldType,
  607. String title,
  608. ) async {
  609. final findFilter = find.byWidgetPredicate(
  610. (widget) =>
  611. widget is GridFilterPropertyCell &&
  612. widget.fieldInfo.fieldType == fieldType &&
  613. widget.fieldInfo.name == title,
  614. );
  615. await tapButton(findFilter);
  616. }
  617. Future<void> tapFilterButtonInGrid(String filterName) async {
  618. final findFilter = find.byType(FilterMenuItem);
  619. final button = find.descendant(
  620. of: findFilter,
  621. matching: find.text(filterName),
  622. );
  623. await tapButton(button);
  624. }
  625. Future<void> tapCreateSortByFieldType(
  626. FieldType fieldType,
  627. String title,
  628. ) async {
  629. final findSort = find.byWidgetPredicate(
  630. (widget) =>
  631. widget is GridSortPropertyCell &&
  632. widget.fieldInfo.fieldType == fieldType &&
  633. widget.fieldInfo.name == title,
  634. );
  635. await tapButton(findSort);
  636. }
  637. // Must call [tapSortMenuInSettingBar] first.
  638. Future<void> tapCreateSortByFieldTypeInSortMenu(
  639. FieldType fieldType,
  640. String title,
  641. ) async {
  642. await tapButton(find.byType(DatabaseAddSortButton));
  643. final findSort = find.byWidgetPredicate(
  644. (widget) =>
  645. widget is GridSortPropertyCell &&
  646. widget.fieldInfo.fieldType == fieldType &&
  647. widget.fieldInfo.name == title,
  648. );
  649. await tapButton(findSort);
  650. await pumpAndSettle();
  651. }
  652. Future<void> tapSortMenuInSettingBar() async {
  653. await tapButton(find.byType(SortMenu));
  654. await pumpAndSettle();
  655. }
  656. /// Must call [tapSortMenuInSettingBar] first.
  657. Future<void> tapSortButtonByName(String name) async {
  658. final findSortItem = find.byWidgetPredicate(
  659. (widget) =>
  660. widget is DatabaseSortItem && widget.sortInfo.fieldInfo.name == name,
  661. );
  662. await tapButton(findSortItem);
  663. }
  664. /// Must call [tapSortButtonByName] first.
  665. Future<void> tapSortByDescending() async {
  666. await tapButton(
  667. find.descendant(
  668. of: find.byType(OrderPannelItem),
  669. matching: find.byWidgetPredicate(
  670. (widget) =>
  671. widget is FlowyText &&
  672. widget.text == LocaleKeys.grid_sort_descending.tr(),
  673. ),
  674. ),
  675. );
  676. await sendKeyEvent(LogicalKeyboardKey.escape);
  677. await pumpAndSettle();
  678. }
  679. /// Must call [tapSortMenuInSettingBar] first.
  680. Future<void> tapAllSortButton() async {
  681. await tapButton(find.byType(DatabaseDeleteSortButton));
  682. }
  683. Future<void> scrollOptionFilterListByOffset(Offset offset) async {
  684. await drag(find.byType(SelectOptionFilterEditor), offset);
  685. await pumpAndSettle();
  686. }
  687. Future<void> enterTextInTextFilter(String text) async {
  688. final findEditor = find.byType(TextFilterEditor);
  689. final findTextField = find.descendant(
  690. of: findEditor,
  691. matching: find.byType(FlowyTextField),
  692. );
  693. await enterText(findTextField, text);
  694. await pumpAndSettle(const Duration(milliseconds: 300));
  695. }
  696. Future<void> tapDisclosureButtonInFinder(Finder finder) async {
  697. final findDisclosure = find.descendant(
  698. of: finder,
  699. matching: find.byType(DisclosureButton),
  700. );
  701. await tapButton(findDisclosure);
  702. }
  703. /// must call [tapDisclosureButtonInFinder] first.
  704. Future<void> tapDeleteFilterButtonInGrid() async {
  705. await tapButton(find.text(LocaleKeys.grid_settings_deleteFilter.tr()));
  706. }
  707. Future<void> tapCheckboxFilterButtonInGrid() async {
  708. await tapButton(find.byType(CheckboxFilterConditionList));
  709. }
  710. Future<void> tapChecklistFilterButtonInGrid() async {
  711. await tapButton(find.byType(ChecklistFilterConditionList));
  712. }
  713. /// The [SelectOptionFilterList] must show up first.
  714. Future<void> tapOptionFilterWithName(String name) async {
  715. final findCell = find.descendant(
  716. of: find.byType(SelectOptionFilterList),
  717. matching: find.byWidgetPredicate(
  718. (widget) =>
  719. widget is SelectOptionFilterCell && widget.option.name == name,
  720. skipOffstage: false,
  721. ),
  722. skipOffstage: false,
  723. );
  724. expect(findCell, findsOneWidget);
  725. await tapButton(findCell, warnIfMissed: false);
  726. }
  727. Future<void> tapCheckedButtonOnCheckboxFilter() async {
  728. final button = find.descendant(
  729. of: find.byType(HoverButton),
  730. matching: find.text(LocaleKeys.grid_checkboxFilter_isChecked.tr()),
  731. );
  732. await tapButton(button);
  733. }
  734. Future<void> tapUnCheckedButtonOnCheckboxFilter() async {
  735. final button = find.descendant(
  736. of: find.byType(HoverButton),
  737. matching: find.text(LocaleKeys.grid_checkboxFilter_isUnchecked.tr()),
  738. );
  739. await tapButton(button);
  740. }
  741. Future<void> tapCompletedButtonOnChecklistFilter() async {
  742. final button = find.descendant(
  743. of: find.byType(HoverButton),
  744. matching: find.text(LocaleKeys.grid_checklistFilter_isComplete.tr()),
  745. );
  746. await tapButton(button);
  747. }
  748. Future<void> tapUnCompletedButtonOnChecklistFilter() async {
  749. final button = find.descendant(
  750. of: find.byType(HoverButton),
  751. matching: find.text(LocaleKeys.grid_checklistFilter_isIncomplted.tr()),
  752. );
  753. await tapButton(button);
  754. }
  755. /// Should call [tapDatabaseSettingButton] first.
  756. Future<void> tapDatabaseLayoutButton() async {
  757. final findSettingItem = find.byType(DatabaseSettingItem);
  758. final findLayoutButton = find.byWidgetPredicate(
  759. (widget) =>
  760. widget is FlowyText &&
  761. widget.text == DatabaseSettingAction.showLayout.title(),
  762. );
  763. final button = find.descendant(
  764. of: findSettingItem,
  765. matching: findLayoutButton,
  766. );
  767. await tapButton(button);
  768. }
  769. Future<void> tapCalendarLayoutSettingButton() async {
  770. final findSettingItem = find.byType(DatabaseSettingItem);
  771. final findLayoutButton = find.byWidgetPredicate(
  772. (widget) =>
  773. widget is FlowyText &&
  774. widget.text == DatabaseSettingAction.showCalendarLayout.title(),
  775. );
  776. final button = find.descendant(
  777. of: findSettingItem,
  778. matching: findLayoutButton,
  779. );
  780. await tapButton(button);
  781. }
  782. Future<void> tapFirstDayOfWeek() async {
  783. await tapButton(find.byType(FirstDayOfWeek));
  784. }
  785. Future<void> tapFirstDayOfWeekStartFromSunday() async {
  786. final finder = find.byWidgetPredicate(
  787. (widget) => widget is StartFromButton && widget.dayIndex == 0,
  788. );
  789. await tapButton(finder);
  790. }
  791. Future<void> tapFirstDayOfWeekStartFromMonday() async {
  792. final finder = find.byWidgetPredicate(
  793. (widget) => widget is StartFromButton && widget.dayIndex == 1,
  794. );
  795. await tapButton(finder);
  796. // Dismiss the popover overlay in cause of obscure the tapButton
  797. // in the next test case.
  798. await sendKeyEvent(LogicalKeyboardKey.escape);
  799. await pumpAndSettle(const Duration(milliseconds: 200));
  800. }
  801. void assertFirstDayOfWeekStartFromMonday() {
  802. final finder = find.byWidgetPredicate(
  803. (widget) =>
  804. widget is StartFromButton &&
  805. widget.dayIndex == 1 &&
  806. widget.isSelected == true,
  807. );
  808. expect(finder, findsOneWidget);
  809. }
  810. void assertFirstDayOfWeekStartFromSunday() {
  811. final finder = find.byWidgetPredicate(
  812. (widget) =>
  813. widget is StartFromButton &&
  814. widget.dayIndex == 0 &&
  815. widget.isSelected == true,
  816. );
  817. expect(finder, findsOneWidget);
  818. }
  819. Future<void> scrollToToday() async {
  820. final todayCell = find.byWidgetPredicate(
  821. (widget) => widget is CalendarDayCard && widget.isToday,
  822. skipOffstage: false,
  823. );
  824. await ensureVisible(todayCell);
  825. await pumpAndSettle(const Duration(milliseconds: 300));
  826. }
  827. Future<void> hoverOnTodayCalendarCell() async {
  828. final todayCell = find.byWidgetPredicate(
  829. (widget) => widget is CalendarDayCard && widget.isToday,
  830. );
  831. await hoverOnWidget(todayCell);
  832. }
  833. Future<void> tapAddCalendarEventButton() async {
  834. final findFlowyButton = find.byType(FlowyIconButton);
  835. final findNewEventButton = find.byType(NewEventButton);
  836. final button = find.descendant(
  837. of: findNewEventButton,
  838. matching: findFlowyButton,
  839. );
  840. await tapButton(button);
  841. }
  842. /// Checks for a certain number of events. Parameters [date] and [title] can
  843. /// also be provided to restrict the scope of the search
  844. void assertNumberOfEventsInCalendar(int number, {String? title}) {
  845. Finder findEvents = find.byType(EventCard);
  846. if (title != null) {
  847. findEvents = find.descendant(of: findEvents, matching: find.text(title));
  848. }
  849. expect(findEvents, findsNWidgets(number));
  850. }
  851. void assertNumberofEventsOnSpecificDay(
  852. int number,
  853. DateTime date, {
  854. String? title,
  855. }) {
  856. final findDayCell = find.byWidgetPredicate(
  857. (widget) =>
  858. widget is CalendarDayCard &&
  859. isSameDay(
  860. widget.date,
  861. date,
  862. ),
  863. );
  864. Finder findEvents = find.descendant(
  865. of: findDayCell,
  866. matching: find.byType(EventCard),
  867. );
  868. if (title != null) {
  869. findEvents = find.descendant(of: findEvents, matching: find.text(title));
  870. }
  871. expect(findEvents, findsNWidgets(number));
  872. }
  873. Future<void> doubleClickCalendarCell(DateTime date) async {
  874. final todayCell = find.byWidgetPredicate(
  875. (widget) => widget is CalendarDayCard && isSameDay(date, widget.date),
  876. );
  877. await doubleTapButton(todayCell);
  878. }
  879. Future<void> openCalendarEvent({required index, DateTime? date}) async {
  880. final findDayCell = find.byWidgetPredicate(
  881. (widget) =>
  882. widget is CalendarDayCard &&
  883. isSameDay(widget.date, date ?? DateTime.now()),
  884. );
  885. final cards = find.descendant(
  886. of: findDayCell,
  887. matching: find.byType(EventCard),
  888. );
  889. await tapButton(cards.at(index));
  890. }
  891. Future<void> dragDropRescheduleCalendarEvent(DateTime startDate) async {
  892. final findEventCard = find.byType(EventCard);
  893. await drag(findEventCard.first, const Offset(0, 300));
  894. await pumpAndSettle();
  895. }
  896. Future<void> tapCreateLinkedDatabaseViewButton(AddButtonAction action) async {
  897. final findAddButton = find.byType(AddDatabaseViewButton);
  898. await tapButton(findAddButton);
  899. final findCreateButton = find.byWidgetPredicate(
  900. (widget) =>
  901. widget is TarBarAddButtonActionCell && widget.action == action,
  902. );
  903. await tapButton(findCreateButton);
  904. }
  905. Finder findTabBarLinkViewByViewLayout(ViewLayoutPB layout) {
  906. return find.byWidgetPredicate(
  907. (widget) => widget is TabBarItemButton && widget.view.layout == layout,
  908. );
  909. }
  910. Finder findTabBarLinkViewByViewName(String name) {
  911. return find.byWidgetPredicate(
  912. (widget) => widget is TabBarItemButton && widget.view.name == name,
  913. );
  914. }
  915. Future<void> renameLinkedView(Finder linkedView, String name) async {
  916. await tap(linkedView, buttons: kSecondaryButton);
  917. await pumpAndSettle();
  918. await tapButton(
  919. find.byWidgetPredicate(
  920. (widget) =>
  921. widget is ActionCellWidget &&
  922. widget.action == TabBarViewAction.rename,
  923. ),
  924. );
  925. await enterText(
  926. find.descendant(
  927. of: find.byType(FlowyFormTextInput),
  928. matching: find.byType(TextFormField),
  929. ),
  930. name,
  931. );
  932. final field = find.byWidgetPredicate(
  933. (widget) =>
  934. widget is PrimaryTextButton &&
  935. widget.label == LocaleKeys.button_OK.tr(),
  936. );
  937. await tapButton(field);
  938. }
  939. Future<void> deleteDatebaseView(Finder linkedView) async {
  940. await tap(linkedView, buttons: kSecondaryButton);
  941. await pumpAndSettle();
  942. await tapButton(
  943. find.byWidgetPredicate(
  944. (widget) =>
  945. widget is ActionCellWidget &&
  946. widget.action == TabBarViewAction.delete,
  947. ),
  948. );
  949. final okButton = find.byWidgetPredicate(
  950. (widget) =>
  951. widget is PrimaryTextButton &&
  952. widget.label == LocaleKeys.button_OK.tr(),
  953. );
  954. await tapButton(okButton);
  955. }
  956. Future<void> assertCurrentDatabaseTagIs(DatabaseLayoutPB layout) async {
  957. switch (layout) {
  958. case DatabaseLayoutPB.Board:
  959. expect(find.byType(BoardPage), findsOneWidget);
  960. break;
  961. case DatabaseLayoutPB.Calendar:
  962. expect(find.byType(CalendarPage), findsOneWidget);
  963. break;
  964. case DatabaseLayoutPB.Grid:
  965. expect(find.byType(GridPage), findsOneWidget);
  966. break;
  967. default:
  968. throw Exception('Unknown database layout type: $layout');
  969. }
  970. }
  971. Future<void> selectDatabaseLayoutType(DatabaseLayoutPB layout) async {
  972. final findLayoutCell = find.byType(DatabaseViewLayoutCell);
  973. final findText = find.byWidgetPredicate(
  974. (widget) => widget is FlowyText && widget.text == layout.layoutName(),
  975. );
  976. final button = find.descendant(
  977. of: findLayoutCell,
  978. matching: findText,
  979. );
  980. await tapButton(button);
  981. }
  982. Future<void> assertCurrentDatabaseLayoutType(DatabaseLayoutPB layout) async {
  983. expect(finderForDatabaseLayoutType(layout), findsOneWidget);
  984. }
  985. Future<void> tapDatabaseRawDataButton() async {
  986. await tapButtonWithName(LocaleKeys.importPanel_database.tr());
  987. }
  988. }
  989. Finder finderForDatabaseLayoutType(DatabaseLayoutPB layout) {
  990. switch (layout) {
  991. case DatabaseLayoutPB.Board:
  992. return find.byType(BoardPage);
  993. case DatabaseLayoutPB.Calendar:
  994. return find.byType(CalendarPage);
  995. case DatabaseLayoutPB.Grid:
  996. return find.byType(GridPage);
  997. default:
  998. throw Exception('Unknown database layout type: $layout');
  999. }
  1000. }
  1001. Finder finderForFieldType(FieldType fieldType) {
  1002. switch (fieldType) {
  1003. case FieldType.Checkbox:
  1004. return find.byType(GridCheckboxCell, skipOffstage: false);
  1005. case FieldType.DateTime:
  1006. return find.byType(GridDateCell, skipOffstage: false);
  1007. case FieldType.LastEditedTime:
  1008. case FieldType.CreatedTime:
  1009. return find.byType(GridDateCell, skipOffstage: false);
  1010. case FieldType.SingleSelect:
  1011. return find.byType(GridSingleSelectCell, skipOffstage: false);
  1012. case FieldType.MultiSelect:
  1013. return find.byType(GridMultiSelectCell, skipOffstage: false);
  1014. case FieldType.Checklist:
  1015. return find.byType(GridChecklistCell, skipOffstage: false);
  1016. case FieldType.Number:
  1017. return find.byType(GridNumberCell, skipOffstage: false);
  1018. case FieldType.RichText:
  1019. return find.byType(GridTextCell, skipOffstage: false);
  1020. case FieldType.URL:
  1021. return find.byType(GridURLCell, skipOffstage: false);
  1022. default:
  1023. throw Exception('Unknown field type: $fieldType');
  1024. }
  1025. }