Skip to content

Instantly share code, notes, and snippets.

@PimCoumans
Last active May 13, 2022 09:51
Show Gist options
  • Save PimCoumans/ba558c0916879b259e679ab98b266f1c to your computer and use it in GitHub Desktop.
Save PimCoumans/ba558c0916879b259e679ab98b266f1c to your computer and use it in GitHub Desktop.
Extended tap area for UIButton
// Methods to extend tap area to a minimal size
extension UIView {
// Golden standard of minimal tap sizes [citation needed]
static let minimalTapSize: CGSize = CGSize(width: 44, height: 44)
/// Extends the rect to be at least as big as ``minimalTapSize``
class func extendedTapArea(from rect: CGRect) -> CGRect {
rect.insetBy(
dx: min(0, (rect.width - minimalTapSize.width) / 2),
dy: min(0, (rect.height - minimalTapSize.height) / 2)
)
}
/// Extended `bounds` to be at least as big as ``minimalTapSize``
var extendedTapBounds: CGRect {
UIView.extendedTapArea(from: bounds)
}
}
class TapAreaButton: UIButton {
// Allow for a bigger tap area, if needed
override func point(inside point: CGPoint, with event: UIEvent?) -> Bool {
if super.point(inside: point, with: event) {
return true
}
return extendedTapBounds.contains(point)
}
}
class ButtonTapAreaContainer: UIView {
// Manually search for subviews with bigger tap area
override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? {
guard let defaultResult = super.hitTest(point, with: event) else {
return nil
}
guard defaultResult == self else {
// When the default hitTest already finds a suitable view
// there‘s no need for extended bounds checking
return defaultResult
}
// Find the first subview where the extended bounds contain the point
let foundSubview = subviews.first { subview in
let convertedPoint = self.convert(point, to: subview)
guard subview.extendedTapBounds.contains(convertedPoint) else {
return false
}
return true
}
// Return the found subview or otherwise just the original result
return foundSubview ?? defaultResult
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment