Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Animate the keyboard in sync with the UIView while edge-swiping back in iOS7

I'd like to get the behavior similar to Messages app (also common in most texting apps) in iOS7, where in a conversation view swiping right from the left edge of the screen would behave like the back button in a UINavigationController.

I have managed to implement this behavior, however, if the keyboard is open in the presenting view, when I start swiping back, the keyboard gets stuck and does not animate with the view to the right as I move my finger. I'd like to animate keyboard and the presenting view as one unit, not as if keyboard is on top of the other views and they are animating behind it, which is what I get now (see the second screenshot):

(UPDATE: Note that the keyboard will eventually go away after the main view animation is finished; what I am focused on is the position of keyboard during the swipe process, and when you keep touching the device half of the way, which is not in sync with the actual view. The second screenshot clarifies this desired behavior. I also wonder why it is not the default.)

It is easy to replicate the issue by simply creating a new master-detail iPhone app in Xcode 5.0.2 and adding a Text Field to the detail view (preferably somewhere in the upper half) in the StoryBoard, running the app, adding an item, tapping on it to go to the detail view and clicking on the text field you added. Edge-swipe from the left side of the device while keeping your finger on it and you'll see the issue.

Desired behavior:

Desired behavior

Current behavior:

Keyboard tearing

like image 838
mmx Avatar asked Jan 07 '14 23:01

mmx


3 Answers

Unfortunately, there is no built-in method to do that. I really hope there will be something like UIScrollViewKeyboardDismissModeInteractive for UIViewControllers.

For now, to do any animations in-between viewControllers, you should use a transitionCoordinator:

- (BOOL)animateAlongsideTransition:(void (^)(id <UIViewControllerTransitionCoordinatorContext>context))animation
                        completion:(void (^)(id <UIViewControllerTransitionCoordinatorContext>context))completion;

- (BOOL)animateAlongsideTransitionInView:(UIView *)view
                               animation:(void (^)(id <UIViewControllerTransitionCoordinatorContext>context))animation
                              completion:(void (^)(id <UIViewControllerTransitionCoordinatorContext>context))completion;

For the keyboard you should do something like this:

[self.transitionCoordinator animateAlongsideTransitionInView:self.keyboardSuperview
                                                   animation:
^(id<UIViewControllerTransitionCoordinatorContext> context) {
    self.keyboardSuperview.x = self.view.width;
}
                                                  completion:nil];

As for keyboardSuperview - you can get that by creating a fake inputAccessoryView:

self.textField.inputAccessoryView = [[UIView alloc] init];

Then the superview will be self.textField.inputAccessoryView.superview

like image 154
monder Avatar answered Nov 18 '22 21:11

monder


It should just work automatically, but sometimes it doesn't because of some conditions.

If the current firstResponder is located inside of active UIViewController and it dismiss throughout UINavigationController mechanism, the expected keyboard animation (horizontal) will be performed automatically. Therefore, sometimes this default behaviour is broken by other strange factors and the keyboard starts to disappear with slide-down animation instead of horizontal animation.

I spent some days with debugging internal UIKit stuff (around methods needDeferredTransition, allowCustomTransition and other) to find one special factor that plays key role in my case.

I discovered that the logic inside UIPeripheralHost checks frame of current UIViewConroller's view, frame of UINavigationController's view (container) and screen size and, if it all doesn’t equal each other, UIPeripheralHost decides that this current situation seems like modal window and sets flag allowCustomTransition = NO. That turn-off UINavigationController-specific horizontal animation.

Fixing issue with frames completely solves my problem.

If you are experiencing same problems, you can try to debug internal UIKit stuff around these private methods and find your conditions that turn off horizontal animation:

https://github.com/JaviSoto/iOS8-Runtime-Headers/blob/master/Frameworks/UIKit.framework/UIPeripheralHost.h

https://github.com/JaviSoto/iOS8-Runtime-Headers/blob/master/Frameworks/UIKit.framework/_UIViewControllerKeyboardAnimationStyle.h

like image 40
Valentin Shergin Avatar answered Nov 18 '22 20:11

Valentin Shergin


You can use https://github.com/cotap/TAPKeyboardPop if you don't need anything special.

In my case I've got some logic connected with UIKeyboardWillShowNotification and UIKeyboardWillHideNotification that were fired on "swipe-to-back" gesture. I've combine this answer and TAPKeyboardPop and this is what I've got:

- (void)beginAppearanceTransition:(BOOL)isAppearing animated:(BOOL)animated {
    [super beginAppearanceTransition:isAppearing animated:animated];
    if (isAppearing || !animated || !_keyboardIsShown) {
        return;
    }
    if ([self respondsToSelector:@selector(transitionCoordinator)]) {
        UIView *keyboardView = self.searchBar.inputAccessoryView.superview;
        [self.searchBar becomeFirstResponder];
        [self.transitionCoordinator animateAlongsideTransitionInView:keyboardView
                                                           animation:^(id<UIViewControllerTransitionCoordinatorContext> context)
         {
             CGRect endFrame = CGRectOffset(keyboardView.frame, CGRectGetWidth(keyboardView.frame), 0);
             keyboardView.frame = endFrame;
         } completion:^(id<UIViewControllerTransitionCoordinatorContext> context)
         {
             if (![context isCancelled]) {
                 [self.searchBar resignFirstResponder];
             }
         }];
    }
}

EDIT:

I've added >iOS7 support and logic for knowing when keyboard is shown (_keyboardIsShown is set in UIKeyboardWillShowNotification/UIKeyboardWillHideNotification or in UIKeyboardDidHideNotification/UIKeyboardDidShowNotification).

like image 2
derpoliuk Avatar answered Nov 18 '22 21:11

derpoliuk