common_operations.dart 13 KB

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