8

I have an inputAccessoryView for a chat app that always remains visible and docked at the bottom of the screen for text input similar to most messaging apps.

When I present an alertController with actionSheet style, the inputAccessoryView animates down off screen as the alert is presented and then back up again when the alert is dismissed. This in turn scrolls my tableView and is undesirable.

This is happening because the chat viewController is giving up firstResponder when the alert is presented.

Is there anyway to present an alertController and not give up firstResponder, or anyway to keep an inputAccessoryView docked at the bottom of the screen when it's view resignsFirstResponder?

alionthego
  • 8,508
  • 9
  • 52
  • 125
  • It's been several years since I implemented this, but IIRC I just used a text view and moved it based on the keyboard notification, rather than making it an accessory. – Kevin Oct 29 '17 at 03:24
  • I need to use an inputAccessoryView so that's not an option for me – alionthego Oct 29 '17 at 03:28

4 Answers4

13

The InputAccessoryView sits outside of your ViewController's hierarchy - it's contained in the UITextEffectsWindow whose rootViewController is a UIInputWindowController. Similarly the keyboard is contained in UIRemoteKeyboardWindow and its own UIInputWindowController.

So, if we present the alert from the topmost window or higher (UITextEffectsWindow or UIRemoteKeyboardWindow), it won't resign first responder.

The simplest solution I've found is:

let topViewController = UIApplication.shared.windows.last!.rootViewController! topViewController.present(alert, animated: true, completion: nil)

Ideally you would safely handle those optionals. A potentially better solution (I've seen some console errors from the previous solution) would be to create a new UIWindow with a higher WindowLevel, make it the key window and visible, and present the alert from there.

Corey W.
  • 319
  • 3
  • 9
  • 1
    This logs a warning: `Keyboard cannot present view controllers (attempted to present )` – Jordan H Jun 29 '19 at 16:05
  • @JordanH Yea, these days I'm a fan of creating new windows that get presented over top - this is also useful for "toasts", drop-down banners, etc. – Corey W. Jul 19 '19 at 16:27
  • This doesn't work for full page modal View Controller. For example, when you open Camera app to pick an image, when you come back, the input toolbar would have disappeared. – coolcool1994 Mar 16 '20 at 08:35
2

Thanks to @Corey W. (give votes to him) we have a solution for this issue. Safer way in Swift 5:

// Instantiate AlertController
let actionSheet = UIAlertController(title: title,
                                    message: "Comment options",
                                    preferredStyle: .actionSheet)

// Add actions
actionSheet.addAction(UIAlertAction(title: "Edit",
                                    style: .default,
                                    handler: { (_: UIAlertAction) in
                                        self.showEditingView(withCommentId: commentId, withCommentText: commentText)
}))

// Present AlertController
if let lastApplicationWindow = UIApplication.shared.windows.last,
    let rootViewController = lastApplicationWindow.rootViewController {
    rootViewController.present(actionSheet, animated: true, completion: nil)
}
Egzon P.
  • 4,498
  • 3
  • 32
  • 31
1

For those looking the Objective-C equivalent of Corey's answer:

UIViewController *objViewController = [UIApplication sharedApplication].windows.lastObject.rootViewController;
[objViewController presentViewController:view animated:YES completion:nil];
Laurent Crivello
  • 3,809
  • 6
  • 45
  • 89
1
let rootViewController = UIApplication.shared.windows.filter {$0.isKeyWindow}.first?.windowScene?.windows.last?.rootViewController
rootViewController?.present(createImagePickAlertController(vc: self, pickerController: pickerController), animated: false)

This one works fix "Keyboard cannot present view controllers (attempted to present <UIAlertController: 0x7fd63004d600>)" error.