mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
WIP Commits separated as follows: - Update lints in analysis_options files - Run `dart fix --apply` - Clean up leftover analysis issues - Run `dart format .` in the right places. Local analysis and testing passes. Checking CI now. Part of https://github.com/flutter/flutter/issues/178827 - Adoption of flutter_lints in examples/api coming in a separate change (cc @loic-sharma) ## Pre-launch Checklist - [ ] I read the [Contributor Guide] and followed the process outlined there for submitting PRs. - [ ] I read the [Tree Hygiene] wiki page, which explains my responsibilities. - [ ] I read and followed the [Flutter Style Guide], including [Features we expect every widget to implement]. - [ ] I signed the [CLA]. - [ ] I listed at least one issue that this PR fixes in the description above. - [ ] I updated/added relevant documentation (doc comments with `///`). - [ ] I added new tests to check the change I am making, or this PR is [test-exempt]. - [ ] I followed the [breaking change policy] and added [Data Driven Fixes] where supported. - [ ] All existing and new tests are passing. If you need help, consider asking for advice on the #hackers-new channel on [Discord]. **Note**: The Flutter team is currently trialing the use of [Gemini Code Assist for GitHub](https://developers.google.com/gemini-code-assist/docs/review-github-code). Comments from the `gemini-code-assist` bot should not be taken as authoritative feedback from the Flutter team. If you find its comments useful you can update your code accordingly, but if you are unsure or disagree with the feedback, please feel free to wait for a Flutter team member's review for guidance on which automated comments should be addressed. <!-- Links --> [Contributor Guide]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#overview [Tree Hygiene]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md [test-exempt]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#tests [Flutter Style Guide]: https://github.com/flutter/flutter/blob/main/docs/contributing/Style-guide-for-Flutter-repo.md [Features we expect every widget to implement]: https://github.com/flutter/flutter/blob/main/docs/contributing/Style-guide-for-Flutter-repo.md#features-we-expect-every-widget-to-implement [CLA]: https://cla.developers.google.com/ [flutter/tests]: https://github.com/flutter/tests [breaking change policy]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#handling-breaking-changes [Discord]: https://github.com/flutter/flutter/blob/main/docs/contributing/Chat.md [Data Driven Fixes]: https://github.com/flutter/flutter/blob/main/docs/contributing/Data-driven-Fixes.md
163 lines
5.8 KiB
Dart
163 lines
5.8 KiB
Dart
// Copyright 2014 The Flutter Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
// ATTENTION!
|
|
//
|
|
// This file is not named "*_test.dart", and as such will not run when you run
|
|
// "flutter test". It is only intended to be run as part of the
|
|
// flutter_gallery_instrumentation_test devicelab test.
|
|
|
|
import 'dart:async';
|
|
|
|
import 'package:flutter/cupertino.dart';
|
|
import 'package:flutter/gestures.dart' show PointerDeviceKind, kPrimaryButton;
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/scheduler.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter_gallery/gallery/app.dart' show GalleryApp;
|
|
import 'package:flutter_gallery/gallery/demos.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
|
|
// Reports success or failure to the native code.
|
|
const MethodChannel _kTestChannel = MethodChannel('io.flutter.demo.gallery/TestLifecycleListener');
|
|
|
|
// We don't want to wait for animations to complete before tapping the
|
|
// back button in the demos with these titles.
|
|
const List<String> _kUnsynchronizedDemoTitles = <String>[
|
|
'Progress indicators',
|
|
'Activity Indicator',
|
|
'Video',
|
|
];
|
|
|
|
// These demos can't be backed out of by tapping a button whose
|
|
// tooltip is 'Back'.
|
|
const List<String> _kSkippedDemoTitles = <String>[
|
|
'Progress indicators',
|
|
'Activity Indicator',
|
|
'Video',
|
|
];
|
|
|
|
// There are 3 places where the Gallery demos are traversed.
|
|
// 1- In widget tests such as dev/integration_tests/flutter_gallery/test/smoke_test.dart
|
|
// 2- In driver tests such as dev/integration_tests/flutter_gallery/test_driver/transitions_perf_test.dart
|
|
// 3- In on-device instrumentation tests such as dev/integration_tests/flutter_gallery/test/live_smoketest.dart
|
|
//
|
|
// If you change navigation behavior in the Gallery or in the framework, make
|
|
// sure all 3 are covered.
|
|
|
|
Future<void> main() async {
|
|
try {
|
|
// Verify that _kUnsynchronizedDemos and _kSkippedDemos identify
|
|
// demos that actually exist.
|
|
final List<String> allDemoTitles = kAllGalleryDemos
|
|
.map((GalleryDemo demo) => demo.title)
|
|
.toList();
|
|
if (!Set<String>.from(allDemoTitles).containsAll(_kUnsynchronizedDemoTitles)) {
|
|
fail('Unrecognized demo titles in _kUnsynchronizedDemosTitles: $_kUnsynchronizedDemoTitles');
|
|
}
|
|
if (!Set<String>.from(allDemoTitles).containsAll(_kSkippedDemoTitles)) {
|
|
fail('Unrecognized demo names in _kSkippedDemoTitles: $_kSkippedDemoTitles');
|
|
}
|
|
|
|
print('Starting app...');
|
|
runApp(const GalleryApp(testMode: true));
|
|
final controller = _LiveWidgetController(WidgetsBinding.instance);
|
|
for (final GalleryDemoCategory category in kAllGalleryDemoCategories) {
|
|
print('Tapping "${category.name}" section...');
|
|
await controller.tap(find.text(category.name));
|
|
for (final GalleryDemo demo in kGalleryCategoryToDemos[category]!) {
|
|
final Finder demoItem = find.text(demo.title);
|
|
print('Scrolling to "${demo.title}"...');
|
|
await controller.scrollIntoView(demoItem, alignment: 0.5);
|
|
if (_kSkippedDemoTitles.contains(demo.title)) {
|
|
continue;
|
|
}
|
|
for (var i = 0; i < 2; i += 1) {
|
|
print('Tapping "${demo.title}"...');
|
|
await controller.tap(demoItem); // Launch the demo
|
|
controller.frameSync = !_kUnsynchronizedDemoTitles.contains(demo.title);
|
|
print('Going back to demo list...');
|
|
await controller.tap(backFinder);
|
|
controller.frameSync = true;
|
|
}
|
|
}
|
|
print('Going back to home screen...');
|
|
await controller.tap(find.byTooltip('Back'));
|
|
}
|
|
print('Finished successfully!');
|
|
_kTestChannel.invokeMethod<void>('success');
|
|
} catch (error, stack) {
|
|
print('Caught error: $error\n$stack');
|
|
_kTestChannel.invokeMethod<void>('failure');
|
|
}
|
|
}
|
|
|
|
final Finder backFinder = find.byElementPredicate(
|
|
(Element element) => switch (element.widget) {
|
|
Tooltip(message: 'Back') => true,
|
|
CupertinoNavigationBarBackButton() => true,
|
|
_ => false,
|
|
},
|
|
description: 'Material or Cupertino back button',
|
|
);
|
|
|
|
class _LiveWidgetController extends LiveWidgetController {
|
|
_LiveWidgetController(super.binding);
|
|
|
|
/// With [frameSync] enabled, Flutter Driver will wait to perform an action
|
|
/// until there are no pending frames in the app under test.
|
|
bool frameSync = true;
|
|
|
|
/// Waits until at the end of a frame the provided [condition] is [true].
|
|
Future<void> _waitUntilFrame(bool Function() condition, [Completer<void>? completer]) {
|
|
completer ??= Completer<void>();
|
|
if (!condition()) {
|
|
SchedulerBinding.instance.addPostFrameCallback((Duration timestamp) {
|
|
_waitUntilFrame(condition, completer);
|
|
});
|
|
} else {
|
|
completer.complete();
|
|
}
|
|
return completer.future;
|
|
}
|
|
|
|
/// Runs `finder` repeatedly until it finds one or more [Element]s.
|
|
Future<FinderBase<Element>> _waitForElement(FinderBase<Element> finder) async {
|
|
if (frameSync) {
|
|
await _waitUntilFrame(() => binding.transientCallbackCount == 0);
|
|
}
|
|
await _waitUntilFrame(() => finder.tryEvaluate());
|
|
if (frameSync) {
|
|
await _waitUntilFrame(() => binding.transientCallbackCount == 0);
|
|
}
|
|
return finder;
|
|
}
|
|
|
|
Future<void> scrollIntoView(FinderBase<Element> finder, {required double alignment}) async {
|
|
final FinderBase<Element> target = await _waitForElement(finder);
|
|
await Scrollable.ensureVisible(
|
|
target.evaluate().single,
|
|
duration: const Duration(milliseconds: 100),
|
|
alignment: alignment,
|
|
);
|
|
}
|
|
|
|
@override
|
|
Future<void> tap(
|
|
FinderBase<Element> finder, {
|
|
int? pointer,
|
|
int buttons = kPrimaryButton,
|
|
bool warnIfMissed = true,
|
|
PointerDeviceKind kind = PointerDeviceKind.touch,
|
|
}) async {
|
|
await super.tap(
|
|
await _waitForElement(finder),
|
|
pointer: pointer,
|
|
buttons: buttons,
|
|
warnIfMissed: warnIfMissed,
|
|
kind: kind,
|
|
);
|
|
}
|
|
}
|