common_operations.dart 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. import 'package:appflowy/generated/flowy_svgs.g.dart';
  2. import 'package:appflowy/generated/locale_keys.g.dart';
  3. import 'package:appflowy/plugins/document/presentation/share/share_button.dart';
  4. import 'package:appflowy/user/presentation/screens/screens.dart';
  5. import 'package:appflowy/workspace/presentation/home/menu/sidebar/sidebar_new_page_button.dart';
  6. import 'package:appflowy/workspace/presentation/home/menu/view/draggable_view_item.dart';
  7. import 'package:appflowy/workspace/presentation/home/menu/view/view_action_type.dart';
  8. import 'package:appflowy/workspace/presentation/home/menu/view/view_add_button.dart';
  9. import 'package:appflowy/workspace/presentation/home/menu/view/view_more_action_button.dart';
  10. import 'package:appflowy/workspace/presentation/settings/widgets/settings_language_view.dart';
  11. import 'package:appflowy_backend/log.dart';
  12. import 'package:appflowy_backend/protobuf/flowy-folder2/view.pb.dart';
  13. import 'package:easy_localization/easy_localization.dart';
  14. import 'package:flowy_infra_ui/widget/buttons/primary_button.dart';
  15. import 'package:flutter/gestures.dart';
  16. import 'package:flutter/material.dart';
  17. import 'package:flutter/services.dart';
  18. import 'package:flutter_test/flutter_test.dart';
  19. import 'util.dart';
  20. extension CommonOperations on WidgetTester {
  21. /// Tap the GetStart button on the launch page.
  22. Future<void> tapGoButton() async {
  23. final goButton = find.byType(GoButton);
  24. await tapButton(goButton);
  25. }
  26. /// Tap the + button on the home page.
  27. Future<void> tapAddViewButton({
  28. String name = gettingStarted,
  29. }) async {
  30. await hoverOnPageName(
  31. name,
  32. onHover: () async {
  33. final addButton = find.byType(ViewAddButton);
  34. await tapButton(addButton);
  35. },
  36. );
  37. }
  38. /// Tap the 'New Page' Button on the sidebar.
  39. Future<void> tapNewPageButton() async {
  40. final newPageButton = find.byType(SidebarNewPageButton);
  41. await tapButton(newPageButton);
  42. }
  43. /// Tap the create document button.
  44. ///
  45. /// Must call [tapAddViewButton] first.
  46. Future<void> tapCreateDocumentButton() async {
  47. await tapButtonWithName(LocaleKeys.document_menuName.tr());
  48. }
  49. /// Tap the create grid button.
  50. ///
  51. /// Must call [tapAddViewButton] first.
  52. Future<void> tapCreateGridButton() async {
  53. await tapButtonWithName(LocaleKeys.grid_menuName.tr());
  54. }
  55. /// Tap the create grid button.
  56. ///
  57. /// Must call [tapAddViewButton] first.
  58. Future<void> tapCreateCalendarButton() async {
  59. await tapButtonWithName(LocaleKeys.calendar_menuName.tr());
  60. }
  61. /// Tap the import button.
  62. ///
  63. /// Must call [tapAddViewButton] first.
  64. Future<void> tapImportButton() async {
  65. await tapButtonWithName(LocaleKeys.moreAction_import.tr());
  66. }
  67. /// Tap the import from text & markdown button.
  68. ///
  69. /// Must call [tapImportButton] first.
  70. Future<void> tapTextAndMarkdownButton() async {
  71. await tapButtonWithName(LocaleKeys.importPanel_textAndMarkdown.tr());
  72. }
  73. /// Tap the LanguageSelectorOnWelcomePage widget on the launch page.
  74. Future<void> tapLanguageSelectorOnWelcomePage() async {
  75. final languageSelector = find.byType(LanguageSelectorOnWelcomePage);
  76. await tapButton(languageSelector);
  77. }
  78. /// Tap languageItem on LanguageItemsListView.
  79. ///
  80. /// [scrollDelta] is the distance to scroll the ListView.
  81. /// Default value is 100
  82. ///
  83. /// If it is positive -> scroll down.
  84. ///
  85. /// If it is negative -> scroll up.
  86. Future<void> tapLanguageItem({
  87. required String languageCode,
  88. String? countryCode,
  89. double? scrollDelta,
  90. }) async {
  91. final languageItemsListView = find.descendant(
  92. of: find.byType(ListView),
  93. matching: find.byType(Scrollable),
  94. );
  95. final languageItem = find.byWidgetPredicate(
  96. (widget) =>
  97. widget is LanguageItem &&
  98. widget.locale.languageCode == languageCode &&
  99. widget.locale.countryCode == countryCode,
  100. );
  101. // scroll the ListView until zHCNLanguageItem shows on the screen.
  102. await scrollUntilVisible(
  103. languageItem,
  104. scrollDelta ?? 100,
  105. scrollable: languageItemsListView,
  106. // maxHeight of LanguageItemsListView
  107. maxScrolls: 400,
  108. );
  109. try {
  110. await tapButton(languageItem);
  111. } on FlutterError catch (e) {
  112. Log.warn('tapLanguageItem error: $e');
  113. }
  114. }
  115. /// Hover on the widget.
  116. Future<void> hoverOnWidget(
  117. Finder finder, {
  118. Offset? offset,
  119. Future<void> Function()? onHover,
  120. bool removePointer = true,
  121. }) async {
  122. try {
  123. final gesture = await createGesture(kind: PointerDeviceKind.mouse);
  124. await gesture.addPointer(location: Offset.zero);
  125. await pump();
  126. await gesture.moveTo(offset ?? getCenter(finder));
  127. await pumpAndSettle();
  128. await onHover?.call();
  129. await gesture.removePointer();
  130. } catch (err) {
  131. Log.error('hoverOnWidget error: $err');
  132. }
  133. }
  134. /// Hover on the page name.
  135. Future<void> hoverOnPageName(
  136. String name, {
  137. ViewLayoutPB layout = ViewLayoutPB.Document,
  138. Future<void> Function()? onHover,
  139. bool useLast = true,
  140. }) async {
  141. final pageNames = findPageName(name, layout: layout);
  142. if (useLast) {
  143. await hoverOnWidget(
  144. pageNames.last,
  145. onHover: onHover,
  146. );
  147. } else {
  148. await hoverOnWidget(
  149. pageNames.first,
  150. onHover: onHover,
  151. );
  152. }
  153. }
  154. /// open the page with given name.
  155. Future<void> openPage(
  156. String name, {
  157. ViewLayoutPB layout = ViewLayoutPB.Document,
  158. }) async {
  159. final finder = findPageName(name, layout: layout);
  160. expect(finder, findsOneWidget);
  161. await tapButton(finder);
  162. }
  163. /// Tap the ... button beside the page name.
  164. ///
  165. /// Must call [hoverOnPageName] first.
  166. Future<void> tapPageOptionButton() async {
  167. final optionButton = find.byType(ViewMoreActionButton);
  168. await tapButton(optionButton);
  169. }
  170. /// Tap the delete page button.
  171. Future<void> tapDeletePageButton() async {
  172. await tapPageOptionButton();
  173. await tapButtonWithName(ViewMoreActionType.delete.name);
  174. }
  175. /// Tap the rename page button.
  176. Future<void> tapRenamePageButton() async {
  177. await tapPageOptionButton();
  178. await tapButtonWithName(ViewMoreActionType.rename.name);
  179. }
  180. /// Tap the favorite page button
  181. Future<void> tapFavoritePageButton() async {
  182. await tapPageOptionButton();
  183. await tapButtonWithName(ViewMoreActionType.favorite.name);
  184. }
  185. /// Tap the unfavorite page button
  186. Future<void> tapUnfavoritePageButton() async {
  187. await tapPageOptionButton();
  188. await tapButtonWithName(ViewMoreActionType.unFavorite.name);
  189. }
  190. /// Tap the Open in a new tab button
  191. Future<void> tapOpenInTabButton() async {
  192. await tapPageOptionButton();
  193. await tapButtonWithName(ViewMoreActionType.openInNewTab.name);
  194. }
  195. /// Rename the page.
  196. Future<void> renamePage(String name) async {
  197. await tapRenamePageButton();
  198. await enterText(find.byType(TextFormField), name);
  199. await tapOKButton();
  200. }
  201. Future<void> tapOKButton() async {
  202. final okButton = find.byWidgetPredicate(
  203. (widget) =>
  204. widget is PrimaryTextButton &&
  205. widget.label == LocaleKeys.button_OK.tr(),
  206. );
  207. await tapButton(okButton);
  208. }
  209. /// Tap the restore button.
  210. ///
  211. /// the restore button will show after the current page is deleted.
  212. Future<void> tapRestoreButton() async {
  213. final restoreButton = find.textContaining(
  214. LocaleKeys.deletePagePrompt_restore.tr(),
  215. );
  216. await tapButton(restoreButton);
  217. }
  218. /// Tap the delete permanently button.
  219. ///
  220. /// the restore button will show after the current page is deleted.
  221. Future<void> tapDeletePermanentlyButton() async {
  222. final restoreButton = find.textContaining(
  223. LocaleKeys.deletePagePrompt_deletePermanent.tr(),
  224. );
  225. await tapButton(restoreButton);
  226. }
  227. /// Tap the share button above the document page.
  228. Future<void> tapShareButton() async {
  229. final shareButton = find.byWidgetPredicate(
  230. (widget) => widget is DocumentShareButton,
  231. );
  232. await tapButton(shareButton);
  233. }
  234. /// Tap the export markdown button
  235. ///
  236. /// Must call [tapShareButton] first.
  237. Future<void> tapMarkdownButton() async {
  238. final markdownButton = find.textContaining(
  239. LocaleKeys.shareAction_markdown.tr(),
  240. );
  241. await tapButton(markdownButton);
  242. }
  243. Future<void> createNewPageWithName({
  244. String? name,
  245. ViewLayoutPB layout = ViewLayoutPB.Document,
  246. String? parentName,
  247. bool openAfterCreated = true,
  248. }) async {
  249. // create a new page
  250. await tapAddViewButton(name: parentName ?? gettingStarted);
  251. await tapButtonWithName(layout.menuName);
  252. await tapOKButton();
  253. await pumpAndSettle();
  254. // hover on it and change it's name
  255. if (name != null) {
  256. await hoverOnPageName(
  257. LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
  258. layout: layout,
  259. onHover: () async {
  260. await renamePage(name);
  261. await pumpAndSettle();
  262. },
  263. );
  264. await pumpAndSettle();
  265. }
  266. // open the page after created
  267. if (openAfterCreated) {
  268. await openPage(
  269. // if the name is null, use the default name
  270. name ?? LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
  271. layout: layout,
  272. );
  273. await pumpAndSettle();
  274. }
  275. }
  276. Future<void> simulateKeyEvent(
  277. LogicalKeyboardKey key, {
  278. bool isControlPressed = false,
  279. bool isShiftPressed = false,
  280. bool isAltPressed = false,
  281. bool isMetaPressed = false,
  282. }) async {
  283. if (isControlPressed) {
  284. await simulateKeyDownEvent(LogicalKeyboardKey.control);
  285. }
  286. if (isShiftPressed) {
  287. await simulateKeyDownEvent(LogicalKeyboardKey.shift);
  288. }
  289. if (isAltPressed) {
  290. await simulateKeyDownEvent(LogicalKeyboardKey.alt);
  291. }
  292. if (isMetaPressed) {
  293. await simulateKeyDownEvent(LogicalKeyboardKey.meta);
  294. }
  295. await simulateKeyDownEvent(key);
  296. await simulateKeyUpEvent(key);
  297. if (isControlPressed) {
  298. await simulateKeyUpEvent(LogicalKeyboardKey.control);
  299. }
  300. if (isShiftPressed) {
  301. await simulateKeyUpEvent(LogicalKeyboardKey.shift);
  302. }
  303. if (isAltPressed) {
  304. await simulateKeyUpEvent(LogicalKeyboardKey.alt);
  305. }
  306. if (isMetaPressed) {
  307. await simulateKeyUpEvent(LogicalKeyboardKey.meta);
  308. }
  309. await pumpAndSettle();
  310. }
  311. Future<void> openAppInNewTab(String name, ViewLayoutPB layout) async {
  312. await hoverOnPageName(
  313. name,
  314. onHover: () async {
  315. await tapOpenInTabButton();
  316. await pumpAndSettle();
  317. },
  318. );
  319. await pumpAndSettle();
  320. }
  321. Future<void> favoriteViewByName(
  322. String name, {
  323. ViewLayoutPB layout = ViewLayoutPB.Document,
  324. }) async {
  325. await hoverOnPageName(
  326. name,
  327. layout: layout,
  328. useLast: true,
  329. onHover: () async {
  330. await tapFavoritePageButton();
  331. await pumpAndSettle();
  332. },
  333. );
  334. }
  335. Future<void> unfavoriteViewByName(
  336. String name, {
  337. ViewLayoutPB layout = ViewLayoutPB.Document,
  338. }) async {
  339. await hoverOnPageName(
  340. name,
  341. layout: layout,
  342. useLast: true,
  343. onHover: () async {
  344. await tapUnfavoritePageButton();
  345. await pumpAndSettle();
  346. },
  347. );
  348. }
  349. Future<void> movePageToOtherPage({
  350. required String name,
  351. required String parentName,
  352. required ViewLayoutPB layout,
  353. required ViewLayoutPB parentLayout,
  354. DraggableHoverPosition position = DraggableHoverPosition.center,
  355. }) async {
  356. final from = findPageName(name, layout: layout);
  357. final to = findPageName(parentName, layout: parentLayout);
  358. final gesture = await startGesture(getCenter(from));
  359. Offset offset = Offset.zero;
  360. switch (position) {
  361. case DraggableHoverPosition.center:
  362. offset = getCenter(to);
  363. break;
  364. case DraggableHoverPosition.top:
  365. offset = getTopLeft(to);
  366. break;
  367. case DraggableHoverPosition.bottom:
  368. offset = getBottomLeft(to);
  369. break;
  370. default:
  371. }
  372. await gesture.moveTo(offset, timeStamp: const Duration(milliseconds: 400));
  373. await gesture.up();
  374. await pumpAndSettle();
  375. }
  376. // tap the button with [FlowySvgData]
  377. Future<void> tapButtonWithFlowySvgData(FlowySvgData svg) async {
  378. final button = find.byWidgetPredicate(
  379. (widget) => widget is FlowySvg && widget.svg.path == svg.path,
  380. );
  381. await tapButton(button);
  382. }
  383. }
  384. extension ViewLayoutPBTest on ViewLayoutPB {
  385. String get menuName {
  386. switch (this) {
  387. case ViewLayoutPB.Grid:
  388. return LocaleKeys.grid_menuName.tr();
  389. case ViewLayoutPB.Board:
  390. return LocaleKeys.board_menuName.tr();
  391. case ViewLayoutPB.Document:
  392. return LocaleKeys.document_menuName.tr();
  393. case ViewLayoutPB.Calendar:
  394. return LocaleKeys.calendar_menuName.tr();
  395. default:
  396. throw UnsupportedError('Unsupported layout: $this');
  397. }
  398. }
  399. String get referencedMenuName {
  400. switch (this) {
  401. case ViewLayoutPB.Grid:
  402. return LocaleKeys.document_plugins_referencedGrid.tr();
  403. case ViewLayoutPB.Board:
  404. return LocaleKeys.document_plugins_referencedBoard.tr();
  405. case ViewLayoutPB.Calendar:
  406. return LocaleKeys.document_plugins_referencedCalendar.tr();
  407. default:
  408. throw UnsupportedError('Unsupported layout: $this');
  409. }
  410. }
  411. }