common_operations.dart 13 KB

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