Unverified Commit a4f79066 authored by Caffeinix's avatar Caffeinix Committed by GitHub

Reorders menu item button shortcuts on Mac-like platforms (#129309)

The Apple Human Interface Guidelines give a specific ordering of the symbols
used as modifier keys in menu shortcuts.  These guidelines are encoded into
the native Cocoa or UIKit menu classes, and are intended to ensure that symbols
are always aligned into columns of like symbols and do not move around in the
case of dynamic menu items (for example, holding Option will transform certain
menu items into different versions that take the Option key, so the Option key
symbol appears to the left of most other symbols to avoid reordering).

The Material spec says to use symbols for modifier keys on Mac and iOS, as this
is what users there are familiar with.  It stands to reason that we should
follow the platform guidelines fully, so this changes the MenuItemButton class
to honor the HIG-compliant symbol ordering on Mac and iOS.

Fixed: #129308
parent 76fe7e86
...@@ -1072,7 +1072,7 @@ class _MenuItemButtonState extends State<MenuItemButton> { ...@@ -1072,7 +1072,7 @@ class _MenuItemButtonState extends State<MenuItemButton> {
), ),
); );
if (_platformSupportsAccelerators() && widget.enabled) { if (_platformSupportsAccelerators && widget.enabled) {
child = MenuAcceleratorCallbackBinding( child = MenuAcceleratorCallbackBinding(
onInvoke: _handleSelect, onInvoke: _handleSelect,
child: child, child: child,
...@@ -1920,7 +1920,7 @@ class _SubmenuButtonState extends State<SubmenuButton> { ...@@ -1920,7 +1920,7 @@ class _SubmenuButtonState extends State<SubmenuButton> {
), ),
); );
if (_enabled && _platformSupportsAccelerators()) { if (_enabled && _platformSupportsAccelerators) {
return MenuAcceleratorCallbackBinding( return MenuAcceleratorCallbackBinding(
onInvoke: () => toggleShowMenu(context), onInvoke: () => toggleShowMenu(context),
hasSubmenu: true, hasSubmenu: true,
...@@ -2049,33 +2049,44 @@ class _LocalizedShortcutLabeler { ...@@ -2049,33 +2049,44 @@ class _LocalizedShortcutLabeler {
String getShortcutLabel(MenuSerializableShortcut shortcut, MaterialLocalizations localizations) { String getShortcutLabel(MenuSerializableShortcut shortcut, MaterialLocalizations localizations) {
final ShortcutSerialization serialized = shortcut.serializeForMenu(); final ShortcutSerialization serialized = shortcut.serializeForMenu();
final String keySeparator; final String keySeparator;
switch (defaultTargetPlatform) { if (_usesSymbolicModifiers) {
case TargetPlatform.iOS:
case TargetPlatform.macOS:
// Use "⌃ ⇧ A" style on macOS and iOS. // Use "⌃ ⇧ A" style on macOS and iOS.
keySeparator = ' '; keySeparator = ' ';
case TargetPlatform.android: } else {
case TargetPlatform.fuchsia:
case TargetPlatform.linux:
case TargetPlatform.windows:
// Use "Ctrl+Shift+A" style. // Use "Ctrl+Shift+A" style.
keySeparator = '+'; keySeparator = '+';
} }
if (serialized.trigger != null) { if (serialized.trigger != null) {
final List<String> modifiers = <String>[]; final List<String> modifiers = <String>[];
final LogicalKeyboardKey trigger = serialized.trigger!; final LogicalKeyboardKey trigger = serialized.trigger!;
// These should be in this order, to match the LogicalKeySet version. if (_usesSymbolicModifiers) {
if (serialized.alt!) { // macOS/iOS platform convention uses this ordering, with ⌘ always last.
modifiers.add(_getModifierLabel(LogicalKeyboardKey.alt, localizations)); if (serialized.control!) {
} modifiers.add(_getModifierLabel(LogicalKeyboardKey.control, localizations));
if (serialized.control!) { }
modifiers.add(_getModifierLabel(LogicalKeyboardKey.control, localizations)); if (serialized.alt!) {
} modifiers.add(_getModifierLabel(LogicalKeyboardKey.alt, localizations));
if (serialized.meta!) { }
modifiers.add(_getModifierLabel(LogicalKeyboardKey.meta, localizations)); if (serialized.shift!) {
} modifiers.add(_getModifierLabel(LogicalKeyboardKey.shift, localizations));
if (serialized.shift!) { }
modifiers.add(_getModifierLabel(LogicalKeyboardKey.shift, localizations)); if (serialized.meta!) {
modifiers.add(_getModifierLabel(LogicalKeyboardKey.meta, localizations));
}
} else {
// These should be in this order, to match the LogicalKeySet version.
if (serialized.alt!) {
modifiers.add(_getModifierLabel(LogicalKeyboardKey.alt, localizations));
}
if (serialized.control!) {
modifiers.add(_getModifierLabel(LogicalKeyboardKey.control, localizations));
}
if (serialized.meta!) {
modifiers.add(_getModifierLabel(LogicalKeyboardKey.meta, localizations));
}
if (serialized.shift!) {
modifiers.add(_getModifierLabel(LogicalKeyboardKey.shift, localizations));
}
} }
String? shortcutTrigger; String? shortcutTrigger;
final int logicalKeyId = trigger.keyId; final int logicalKeyId = trigger.keyId;
...@@ -2848,7 +2859,7 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> { ...@@ -2848,7 +2859,7 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> {
@override @override
void initState() { void initState() {
super.initState(); super.initState();
if (_platformSupportsAccelerators()) { if (_platformSupportsAccelerators) {
_showAccelerators = _altIsPressed(); _showAccelerators = _altIsPressed();
HardwareKeyboard.instance.addHandler(_handleKeyEvent); HardwareKeyboard.instance.addHandler(_handleKeyEvent);
} }
...@@ -2857,9 +2868,9 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> { ...@@ -2857,9 +2868,9 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> {
@override @override
void dispose() { void dispose() {
assert(_platformSupportsAccelerators() || _shortcutRegistryEntry == null); assert(_platformSupportsAccelerators || _shortcutRegistryEntry == null);
_displayLabel = ''; _displayLabel = '';
if (_platformSupportsAccelerators()) { if (_platformSupportsAccelerators) {
_shortcutRegistryEntry?.dispose(); _shortcutRegistryEntry?.dispose();
_shortcutRegistryEntry = null; _shortcutRegistryEntry = null;
_shortcutRegistry = null; _shortcutRegistry = null;
...@@ -2872,7 +2883,7 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> { ...@@ -2872,7 +2883,7 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> {
@override @override
void didChangeDependencies() { void didChangeDependencies() {
super.didChangeDependencies(); super.didChangeDependencies();
if (!_platformSupportsAccelerators()) { if (!_platformSupportsAccelerators) {
return; return;
} }
_binding = MenuAcceleratorCallbackBinding.maybeOf(context); _binding = MenuAcceleratorCallbackBinding.maybeOf(context);
...@@ -2900,7 +2911,7 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> { ...@@ -2900,7 +2911,7 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> {
} }
bool _handleKeyEvent(KeyEvent event) { bool _handleKeyEvent(KeyEvent event) {
assert(_platformSupportsAccelerators()); assert(_platformSupportsAccelerators);
final bool altIsPressed = _altIsPressed(); final bool altIsPressed = _altIsPressed();
if (altIsPressed != _showAccelerators) { if (altIsPressed != _showAccelerators) {
setState(() { setState(() {
...@@ -2913,7 +2924,7 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> { ...@@ -2913,7 +2924,7 @@ class _MenuAcceleratorLabelState extends State<MenuAcceleratorLabel> {
} }
void _updateAcceleratorShortcut() { void _updateAcceleratorShortcut() {
assert(_platformSupportsAccelerators()); assert(_platformSupportsAccelerators);
_shortcutRegistryEntry?.dispose(); _shortcutRegistryEntry?.dispose();
_shortcutRegistryEntry = null; _shortcutRegistryEntry = null;
// Before registering an accelerator as a shortcut it should meet these // Before registering an accelerator as a shortcut it should meet these
...@@ -3566,23 +3577,38 @@ bool _debugMenuInfo(String message, [Iterable<String>? details]) { ...@@ -3566,23 +3577,38 @@ bool _debugMenuInfo(String message, [Iterable<String>? details]) {
return true; return true;
} }
bool _platformSupportsAccelerators() { /// Whether [defaultTargetPlatform] is an Apple platform (Mac or iOS).
bool get _isApple {
switch (defaultTargetPlatform) { switch (defaultTargetPlatform) {
case TargetPlatform.iOS:
case TargetPlatform.macOS:
return true;
case TargetPlatform.android: case TargetPlatform.android:
case TargetPlatform.fuchsia: case TargetPlatform.fuchsia:
case TargetPlatform.linux: case TargetPlatform.linux:
case TargetPlatform.windows: case TargetPlatform.windows:
return true;
case TargetPlatform.iOS:
case TargetPlatform.macOS:
// On iOS and macOS, pressing the Option key (a.k.a. the Alt key) causes a
// different set of characters to be generated, and the native menus don't
// support accelerators anyhow, so we just disable accelerators on these
// platforms.
return false; return false;
} }
} }
/// Whether [defaultTargetPlatform] is one that uses symbolic shortcuts.
///
/// Mac and iOS use special symbols for modifier keys instead of their names,
/// render them in a particular order defined by Apple's human interface
/// guidelines, and format them so that the modifier keys always align.
bool get _usesSymbolicModifiers {
return _isApple;
}
bool get _platformSupportsAccelerators {
// On iOS and macOS, pressing the Option key (a.k.a. the Alt key) causes a
// different set of characters to be generated, and the native menus don't
// support accelerators anyhow, so we just disable accelerators on these
// platforms.
return !_isApple;
}
// BEGIN GENERATED TOKEN PROPERTIES - Menu // BEGIN GENERATED TOKEN PROPERTIES - Menu
// Do not edit by hand. The code between the "BEGIN GENERATED" and // Do not edit by hand. The code between the "BEGIN GENERATED" and
......
...@@ -2944,7 +2944,17 @@ void main() { ...@@ -2944,7 +2944,17 @@ void main() {
shift: true, shift: true,
alt: true, alt: true,
); );
final String allExpected = <String>[expectedAlt, expectedCtrl, expectedMeta, expectedShift, 'A'].join(expectedSeparator); late String allExpected;
switch (defaultTargetPlatform) {
case TargetPlatform.android:
case TargetPlatform.fuchsia:
case TargetPlatform.linux:
case TargetPlatform.windows:
allExpected = <String>[expectedAlt, expectedCtrl, expectedMeta, expectedShift, 'A'].join(expectedSeparator);
case TargetPlatform.iOS:
case TargetPlatform.macOS:
allExpected = <String>[expectedCtrl, expectedAlt, expectedShift, expectedMeta, 'A'].join(expectedSeparator);
}
const CharacterActivator charShortcuts = CharacterActivator('ñ'); const CharacterActivator charShortcuts = CharacterActivator('ñ');
const String charExpected = 'ñ'; const String charExpected = 'ñ';
await tester.pumpWidget( await tester.pumpWidget(
......
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