flutter/dev/devicelab/lib/tasks/keyboard_hot_restart_test.dart
Loïc Sharma 08a59c2802
[iOS] Hide keyboard on hot restart (#167013)
This hides the keyboard and text input context menu if you hot restart
your iOS app.

Before | After
-- | --
<video
src="https://github.com/user-attachments/assets/7ca5dbfe-a809-478c-9b36-4c168527b176"
/> | <video
src="https://github.com/user-attachments/assets/d1a48c16-f171-4d22-baa4-5c40488d055b"
/>

Part of https://github.com/flutter/flutter/issues/10713

## Pre-launch Checklist

- [x] I read the [Contributor Guide] and followed the process outlined
there for submitting PRs.
- [x] I read the [Tree Hygiene] wiki page, which explains my
responsibilities.
- [x] I read and followed the [Flutter Style Guide], including [Features
we expect every widget to implement].
- [x] I signed the [CLA].
- [x] I listed at least one issue that this PR fixes in the description
above.
- [x] I updated/added relevant documentation (doc comments with `///`).
- [x] I added new tests to check the change I am making, or this PR is
[test-exempt].
- [x] I followed the [breaking change policy] and added [Data Driven
Fixes] where supported.
- [x] All existing and new tests are passing.

If you need help, consider asking for advice on the #hackers-new channel
on [Discord].

<!-- 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
2025-04-22 23:33:02 +00:00

153 lines
5.1 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.
import 'dart:async';
import 'dart:convert';
import 'dart:io';
import 'package:path/path.dart' as path;
import '../framework/devices.dart';
import '../framework/framework.dart';
import '../framework/task_result.dart';
import '../framework/utils.dart';
// This test verifies that hot restart hides the keyboard if it is visible.
//
// Steps:
//
// 1. Launch an app that focuses a text field at startup.
// This makes the keyboard visible.
// 2. Wait until the keyboard is visible.
// 3. Update the app's source code to no longer focus a text field at startup.
// 4. Hot restart the app
// 5. Wait until the keyboard is no longer visible.
//
// App under test: //dev/integration_tests/keyboard_hot_restart/lib/main.dart
//
// Since this test must hot restart the app under test, this test cannot use
// testing frameworks like XCUITest or Flutter's integration_test as they don't
// support hot restart. Instead, this test uses the Flutter tool to run the app,
// hot restart it, and verify its log output.
TaskFunction createKeyboardHotRestartTest({
String? deviceIdOverride,
bool checkAppRunningOnLocalDevice = false,
List<String>? additionalOptions,
}) {
final Directory appDir = dir(
path.join(flutterDirectory.path, 'dev/integration_tests/keyboard_hot_restart'),
);
// This file is modified during the test and needs to be restored at the end.
final File mainFile = file(path.join(appDir.path, 'lib/main.dart'));
final String oldContents = mainFile.readAsStringSync();
// When the test starts, the app forces the keyboard to be visible.
// The test turns off this behavior by mutating the app's source code from
// `forceKeyboardOn` to `forceKeyboardOff`.
// See: //dev/integration_tests/keyboard_hot_restart/lib/main.dart
const String forceKeyboardOn = 'const bool forceKeyboard = true;';
const String forceKeyboardOff = 'const bool forceKeyboard = false;';
return () async {
if (deviceIdOverride == null) {
final Device device = await devices.workingDevice;
await device.unlock();
deviceIdOverride = device.deviceId;
}
return inDirectory<TaskResult>(appDir, () async {
try {
section('Create app');
await createAppProject();
// Ensure the app forces the keyboard to be visible.
final String newContents = oldContents.replaceFirst(forceKeyboardOff, forceKeyboardOn);
mainFile.writeAsStringSync(newContents);
section('Launch app and wait for keyboard to be visible');
TestState state = TestState.waitUntilKeyboardOpen;
final int exitCode = await runApp(
options: <String>['-d', deviceIdOverride!],
onLine: (String line, Process process) {
if (state == TestState.waitUntilKeyboardOpen) {
if (!line.contains('flutter: Keyboard is open')) {
return;
}
section('Update the app to no longer force the keyboard to be visible');
final String newContents = oldContents.replaceFirst(
forceKeyboardOn,
forceKeyboardOff,
);
mainFile.writeAsStringSync(newContents);
section('Hot restart the app');
process.stdin.writeln('R');
section('Wait until the keyboard is no longer visible');
state = TestState.waitUntilKeyboardClosed;
} else if (state == TestState.waitUntilKeyboardClosed) {
if (!line.contains('flutter: Keyboard is closed')) {
return;
}
// Quit the app. This makes the 'flutter run' process exit.
process.stdin.writeln('q');
}
},
);
if (exitCode != 0) {
return TaskResult.failure('flutter run exited with non-zero exit code: $exitCode');
}
} finally {
mainFile.writeAsStringSync(oldContents);
}
return TaskResult.success(null);
});
};
}
enum TestState { waitUntilKeyboardOpen, waitUntilKeyboardClosed }
Future<void> createAppProject() async {
await exec(path.join(flutterDirectory.path, 'bin', 'flutter'), <String>[
'create',
'--platforms=android,ios',
'.',
]);
}
Future<int> runApp({
required List<String> options,
required void Function(String, Process) onLine,
}) async {
final Process process = await startFlutter('run', options: options);
final Completer<void> stdoutDone = Completer<void>();
final Completer<void> stderrDone = Completer<void>();
void onStdout(String line) {
onLine(line, process);
print('stdout: $line');
}
process.stdout
.transform<String>(utf8.decoder)
.transform<String>(const LineSplitter())
.listen(onStdout, onDone: stdoutDone.complete);
process.stderr
.transform<String>(utf8.decoder)
.transform<String>(const LineSplitter())
.listen((String line) => print('stderr: $line'), onDone: stderrDone.complete);
await Future.wait<void>(<Future<void>>[stdoutDone.future, stderrDone.future]);
return process.exitCode;
}