Unverified Commit 6b06943c authored by Greg Spencer's avatar Greg Spencer Committed by GitHub

Handle null primary focus when receiving key events (#80302)

This allows the key handling code in the shortcut manager to function when there is a null primaryFocus. The primary focus can be null only in some special circumstances (e.g. when an app hasn't set any focus at all), but that shouldn't cause a crash.
parent a0abf410
...@@ -356,17 +356,18 @@ class ShortcutManager extends ChangeNotifier with Diagnosticable { ...@@ -356,17 +356,18 @@ class ShortcutManager extends ChangeNotifier with Diagnosticable {
); );
final Intent? matchedIntent = _find(keysPressed: keysPressed); final Intent? matchedIntent = _find(keysPressed: keysPressed);
if (matchedIntent != null) { if (matchedIntent != null) {
final BuildContext primaryContext = primaryFocus!.context!; final BuildContext? primaryContext = primaryFocus?.context;
assert (primaryContext != null); if (primaryContext != null) {
final Action<Intent>? action = Actions.maybeFind<Intent>( final Action<Intent>? action = Actions.maybeFind<Intent>(
primaryContext, primaryContext,
intent: matchedIntent, intent: matchedIntent,
); );
if (action != null && action.isEnabled(matchedIntent)) { if (action != null && action.isEnabled(matchedIntent)) {
Actions.of(primaryContext).invokeAction(action, matchedIntent, primaryContext); Actions.of(primaryContext).invokeAction(action, matchedIntent, primaryContext);
return action.consumesKey(matchedIntent) return action.consumesKey(matchedIntent)
? KeyEventResult.handled ? KeyEventResult.handled
: KeyEventResult.skipRemainingHandlers; : KeyEventResult.skipRemainingHandlers;
}
} }
} }
return modal ? KeyEventResult.skipRemainingHandlers : KeyEventResult.ignored; return modal ? KeyEventResult.skipRemainingHandlers : KeyEventResult.ignored;
......
...@@ -290,6 +290,37 @@ void main() { ...@@ -290,6 +290,37 @@ void main() {
expect(invoked, isTrue); expect(invoked, isTrue);
expect(pressedKeys, equals(<LogicalKeyboardKey>[LogicalKeyboardKey.shiftLeft])); expect(pressedKeys, equals(<LogicalKeyboardKey>[LogicalKeyboardKey.shiftLeft]));
}); });
testWidgets('ShortcutManager ignores keypresses with no primary focus', (WidgetTester tester) async {
final GlobalKey containerKey = GlobalKey();
final List<LogicalKeyboardKey> pressedKeys = <LogicalKeyboardKey>[];
final TestShortcutManager testManager = TestShortcutManager(pressedKeys);
bool invoked = false;
await tester.pumpWidget(
Actions(
actions: <Type, Action<Intent>>{
TestIntent: TestAction(
onInvoke: (Intent intent) {
invoked = true;
return true;
},
),
},
child: Shortcuts(
manager: testManager,
shortcuts: <LogicalKeySet, Intent>{
LogicalKeySet(LogicalKeyboardKey.shift): const TestIntent(),
},
child: SizedBox(key: containerKey, width: 100, height: 100),
),
),
);
await tester.pump();
expect(primaryFocus, isNull);
expect(Shortcuts.of(containerKey.currentContext!), isNotNull);
await tester.sendKeyDownEvent(LogicalKeyboardKey.shiftLeft);
expect(invoked, isFalse);
expect(pressedKeys, isEmpty);
});
testWidgets("Shortcuts passes to the next Shortcuts widget if it doesn't map the key", (WidgetTester tester) async { testWidgets("Shortcuts passes to the next Shortcuts widget if it doesn't map the key", (WidgetTester tester) async {
final GlobalKey containerKey = GlobalKey(); final GlobalKey containerKey = GlobalKey();
final List<LogicalKeyboardKey> pressedKeys = <LogicalKeyboardKey>[]; final List<LogicalKeyboardKey> pressedKeys = <LogicalKeyboardKey>[];
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment