Xcode 16 breaks rich button gestures in scroll views in iOS 18

Xcode 16 causes a breaking gesture bug, where multi-gesture buttons stop working in scroll views in iOS 18. This affects the emoji keyboard. This post describes how KeyboardKit will fix it.

KeyboardKit logo

Background

Triggering many gestures with a single button is pretty complicated in SwiftUI. KeyboardKit thus has a custom GestureButton that can trigger multiple gestures with a single button.

The GestureButton supports triggering actions for press, release, long press, repeat, drag start, drag changed, drag end and ended, which is used to support many native keyboard operations.

How does it work?

The regular GestureButton uses a single DragGesture to derive all different gestures, but this doesn’t work within a ScrollView since the drag gesture blocks the scroll gesture.

The GestureButton therefore has a second underlying button called ScrollViewGestureButton, which implements the gestures in a MUCH more complicated way, in a way that works within a scroll view.

The ScrollViewGestureButton uses an intricate combination of styles and gestures to avoid blocking the scroll gesture, while trying to behave as regular keyboard buttons to the greatest extent.

This is how both the KeyboardView and KeyboardKit Pro’s EmojiKeyboard can use the same gestures, although the latter is in a scroll view and the former is not.

Xcode 16 breaking change

Xcode 16 breaks scroll view support in iOS 18 by making the ScrollViewGestureButton stop working. Any multi-gesture button that is added to a scroll view will now block the scroll gesture.

Note that it still works great in Xcode 15, or when building with Xcode 16 and running on iOS 17. It’s just the combination of Xcode 16 and iOS 18 that doesn’t work.

As such, the emoji keyboard stops working in this case, since the emoji keys block the scroll gesture.

KeyboardKit 8.8.6 to the rescue

KeyboardKit 8.8.6 rewrites the gesture engine from scratch to work with both Xcode 15, Xcode 16, iOS 17 (and earlier) and iOS 18, and without the need for complex scroll handling in iOS 18.

It will do so by using the same GestureButton for both KeyboardView and EmojiKeyboard in iOS 18, by changing the gesture to be simultaneous instead of exclusive.

To make this possible, the new gestures keep track of state in a different way than before, to ensure that only the actions that should trigger will actually trigger.

The new implementation also moves state to a separate class, which removes the need to annotate every state property with @State, which in turn should lead to fewer redraws.

By using the same gestures everywhere, the updated gesture engine will also actually improve the emoji keyboard behavior in iOS 18, where it will handle scrolling much better.

The GestureButton still uses the ScrollViewGestureButton in iOS 17, since it works without changes.

Please help by testing this

Since the gesture engine is at the heart of the keyboard, rewriting it is pretty scary. As such, it would be amazing if you could give it a try when the 8.9 Release Candidate comes out next week.

You are also most welcome to discuss this in this GitHub issue.

Discussions & More

If you found this interesting, please share your thoughts on Bluesky, Mastodon, and X. Also make sure to follow to be notified when new content is published.