database_test_op.dart 33 KB

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