Scrolling a UITableView When Displaying the Keyboard

Scrolling a UITableView When Displaying the Keyboard

OCT 15TH, 2012

The problem

I have a few apps that allow you to edit text in a table cell of a UITableView. One thing that can happen when doing that (especially on the iPhone) is that when the keyboard is shown it can actually cover the text field being edited.

Scrolling a UITableView When Displaying the Keyboard_第1张图片 The initial screen before starting to edit Scrolling a UITableView When Displaying the Keyboard_第2张图片 Attempting to edit Player 2

Since that’s not very usable you need to scroll the edited cell into view when the keyboard is shown. My first attempt was to use the scrollToRowAtIndexPath:atScrollPosition:animated:method on UITableView. Unfortunately this is only part of the solution. While the table view will scroll the supplied row into view, it doesn’t take into account the fact that the keyboard is visible. In other words the table view scrolls the row into view according to the table view’s frame, ignoring the fact that the keyboard is covering part of this frame. So the row may still not be visible.

The route I took is to change the table view’s frame when the keyboard is shown. Combining this with a call to scrollToRowAtIndexPath:atScrollPosition:animated: results in the desired row being displayed when editing begins.

Responding to the keyboard appearing

The first step is to be notified when the keyboard is shown or hidden. This is fairly straightforward to do by registering for the UIKeyboardDidShowNotification/UIKeyboardDidHideNotification.

123456789101112
            
            
            
            
-( id ) initWithNibName: ( NSString * ) nibNameOrNil bundle: nibBundleOrNil {
if ( self = [ super initWithNibName : nibNameOrNil bundle : nibBundleOrNil ]) {
[[ NSNotificationCenter defaultCenter ] addObserver : self selector : @selector ( keyboardShown :) name : UIKeyboardDidShowNotification object : nil ];
[[ NSNotificationCenter defaultCenter ] addObserver : self selector : @selector ( keyboardHidden :) name : UIKeyboardDidHideNotification object : nil ];
}
return self ;
}
 
- ( void ) dealloc {
[[ NSNotificationCenter defaultCenter ] removeObserver : self name : UIKeyboardDidShowNotification object : nil ];
[[ NSNotificationCenter defaultCenter ] removeObserver : self name : UIKeyboardDidHideNotification object : nil ];
}
view raw ViewController_RegisterNotifications.m hosted with ❤ by  GitHub

When the notification arrives it’s time to do the adjustment of the frame:

123456789101112131415161718
            
            
            
            
-( void ) keyboardShown: ( NSNotification * ) notification {
_initialTVHeight = _tableView . frame . size . height ;
 
CGRect initialFrame = [[[ notification userInfo ] objectForKey : UIKeyboardFrameEndUserInfoKey ] CGRectValue ];
CGRect convertedFrame = [ self . view convertRect : initialFrame fromView : nil ];
CGRect tvFrame = _tableView . frame ;
tvFrame . size . height = convertedFrame . origin . y ;
_tableView . frame = tvFrame ;
}
 
-( void ) keyboardHidden: ( NSNotification * ) notification {
CGRect tvFrame = _tableView . frame ;
tvFrame . size . height = _initialTVHeight ;
[ UIView beginAnimations : @"TableViewDown" context : NULL ];
[ UIView setAnimationDuration : 0.3f ];
_tableView . frame = tvFrame ;
[ UIView commitAnimations ];
}
view raw ViewController_Keyboard.m hosted with ❤ by  GitHub

First I save the initial height of the view. This is a handy shortcut that makes it easier to restore the initial size when the keyboard is hidden. On line 4 we grab the keyboard frame from the notification. This frame hasn’t been adjusted for any transforms applied to the view (like rotation) so it needs to be converted to view coordinates (line 5). Now all that’s left is to adjust the height of the main view. I just make the height of the main view equal to the y origin of the keyboard frame - this makes the view “touch” the top of the keyboard.

Restoring the view’s height when the keyboard is hidden is a straightforward affair. I just restore the view’s initial height to what it was before shrinking it (which is why saving it in keyboardShown: was useful). The only additional step here is animating the view size change. While not strictly necessary, this makes the transition look a little nicer on the device.

Finishing up

So now the table view’s frame is being adjusted when the keyboard is shown. The last step is to scroll the newly resized table view to make the selected row visible. This is accomplished in the UITextFieldDelegate’s textFieldDidBeginEditing method. The only thing to be aware of here is that the delegate will be called before the keyboard is actually shown. To deal with that we delay the scrolling using delayed performance:

12345678
            
            
            
            
-( void ) scrollToCell: ( NSIndexPath * ) path {
[ _tableView scrollToRowAtIndexPath : path atScrollPosition : UITableViewScrollPositionNone animated : YES ];
}
 
-( void ) textFieldDidBeginEditing: ( UITextField * ) textField {
NSIndexPath * path = [ NSIndexPath indexPathForRow : row inSection : section ];
[ self performSelector : @selector ( scrollToCell :) withObject : path afterDelay : 0.5f ];
}
view raw ViewController_Scroll.m hosted with ❤ by  GitHub

On line 6 you’ll need the row and section to scroll to - these need to be tracked/figured out based on your needs (they are the same values you would need if you weren’t dealing with the view resizing).

After implementing these changes, the selected row nicely scrolls into view when the keyboard is shown. Much better!

你可能感兴趣的:(Scrolling a UITableView When Displaying the Keyboard)