当我从弹出的视图控制器退出时,UIScrollView向下移动

时间:2013-11-03 14:41:05

标签: ios iphone objective-c uiscrollview uinavigationcontroller

导航控制器和2个视图控制器出现问题。当我弹出到其他视图控制器时一切正常,但当我返回到我的第一个视图时,它会在导航栏下显示大白色

在第二个控制器上,当我编辑UITextField时,我会自动显示和隐藏键盘。 当我从显示键盘的第二个视图控制器返回时,此问题就开始了。 其他方式(没有编辑UITextField)我对第一个控制器没有这个问题。

我需要做什么?

首先.m文件

#import "OTPTRecordViewController.h"
#import "OTPTAppDelegate.h"
#import "OTPTPractice.h"


@implementation OTPTRecordViewController
@synthesize scrollView, activeField;

- (void)viewDidLoad
{
    [super viewDidLoad];
    [self registerForKeyboardNotifications];

    viewBackgroundTap.cancelsTouchesInView = NO;
    [self.view addGestureRecognizer:viewBackgroundTap];
}

- (void)viewWillAppear:(BOOL)animated
{
    [super viewWillAppear:animated];

    UIEdgeInsets contentInsets = UIEdgeInsetsMake(-20.0, 0.0, 0.0, 0.0);
    //I need it for correct view insets

    scrollView.contentInset = contentInsets;
    scrollView.scrollIndicatorInsets = contentInsets;    
}

#pragma mark Textfields&Keyboard management

- (BOOL)textFieldShouldReturn:(UITextField *)textField {
    [textField resignFirstResponder];
    return NO;
}

- (IBAction)backgroundTap:(id)sender
{
    [nameTextField resignFirstResponder];
    [lengthTextField resignFirstResponder];    
}

- (void)registerForKeyboardNotifications
{
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(keyboardWasShown:)
                                                 name:UIKeyboardDidShowNotification object:nil];

    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(keyboardWillBeHidden:)
                                                 name:UIKeyboardWillHideNotification object:nil];    
}

- (void)keyboardWasShown:(NSNotification*)aNotification
{    
    NSDictionary* info = [aNotification userInfo];
    CGSize kbSize = [[info objectForKey:UIKeyboardFrameBeginUserInfoKey] CGRectValue].size;

    if (([holdOutTextField isEditing]) || ([cyclesTextField isEditing])) {
        UIEdgeInsets contentInsets = UIEdgeInsetsMake(0.0, 0.0, kbSize.height+50.0, 0.0);
        scrollView.contentInset = contentInsets;
        scrollView.scrollIndicatorInsets = contentInsets;
    }       

    CGRect aRect = self.view.frame;
    //For scroll view moves
    aRect.size.height -= kbSize.height + self.navigationController.navigationBar.frame.size.height + [UIApplication sharedApplication].statusBarFrame.size.height;
    CGPoint origin = activeField.frame.origin;
    origin.y -= scrollView.contentOffset.y;
    if (!CGRectContainsPoint(aRect, origin) ) {
        CGPoint scrollPoint = CGPointMake(0.0, activeField.frame.origin.y-(aRect.size.height));
        [scrollView setContentOffset:scrollPoint animated:YES];
    }
}

- (void)keyboardWillBeHidden:(NSNotification*)aNotification
{
    UIEdgeInsets contentInsets = UIEdgeInsetsMake(-20.0, 0.0, 0.0, 0.0);

    scrollView.contentInset = contentInsets;
    scrollView.scrollIndicatorInsets = contentInsets;
}

- (void)textFieldDidBeginEditing:(UITextField *)textField
{
    activeField = textField;
    [activeField setSelectedTextRange:[activeField textRangeFromPosition:activeField.beginningOfDocument toPosition:activeField.endOfDocument]];
}

- (void)textFieldDidEndEditing:(UITextField *)textField
{
    activeField = nil;    
}

@end

第二个.m文件

#import "OTPTSaveViewController.h"
#import "OTPTAppDelegate.h"
#import "OTPTRecordViewController.h"

@interface OTPTSaveViewController () {
    UIViewController *parentViewController;
}

@end

@implementation OTPTSaveViewController

@synthesize textField;

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    UITapGestureRecognizer *viewBackgroundTap = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(backgroundTap:)];
    viewBackgroundTap.cancelsTouchesInView = NO;
    [self.view addGestureRecognizer:viewBackgroundTap];
    parentViewController = [self presentingViewController];
}

- (IBAction)cancel:(id)sender {
    [[self navigationController] popViewControllerAnimated:YES];
}

- (IBAction)save:(id)sender {
    //some saving
}


- (BOOL)textFieldShouldReturn:(UITextField *)tf {
    [tf resignFirstResponder];
    return NO;
}

- (IBAction)backgroundTap:(id)sender
{
    [textField resignFirstResponder];
}

@end

我认为这个问题是因为来自弹出控制器的键盘通知。 但是当我添加

[self unregisterForKeyboardNotifications];

viewWillDisappear和make方法

- (void) unregisterForKeyboardNotifications
{
    [[NSNotificationCenter defaultCenter] removeObserver:self];    
}

它没有给我带来任何影响

2 个答案:

答案 0 :(得分:0)

从iOS 7开始,UIViewController添加了一个新属性...

@property(nonatomic, assign) BOOL automaticallyAdjustsScrollViewInsets

根据Apple Docs:

  

默认值为YES,允许视图控制器调整其滚动视图插入以响应状态栏,导航栏和工具栏或标签栏所消耗的屏幕区域。如果要自己管理滚动视图插入调整,则设置为NO,例如视图层次结构中有多个滚动视图时。

如果您手动管理插入内容,可能需要将此属性设置为NO,因为这可能会干扰您尝试执行的操作。

答案 1 :(得分:0)

我以某种狡猾的方式解决了这个问题

我在AppDelegate中创建了一个名为 isAfterSafe 的BOOL,在第二个文件中将其设为YES:

- (void)textFieldDidBeginEditing:(UITextField *)textField
{
    OTPTAppDelegate *appDelegate = [UIApplication sharedApplication].delegate;
    appDelegate.isAfterSave = YES;
}

并在第一个文件中使用简单的 if

- (void)keyboardWasShown:(NSNotification*)aNotification
{    
    OTPTAppDelegate *appDelegate = [UIApplication sharedApplication].delegate;
    if (!appDelegate.isAfterSave) {
        NSDictionary* info = [aNotification userInfo];
        CGSize kbSize = [[info objectForKey:UIKeyboardFrameBeginUserInfoKey] CGRectValue].size;

        if (([holdOutTextField isEditing]) || ([cyclesTextField isEditing])) {
            UIEdgeInsets contentInsets = UIEdgeInsetsMake(0.0, 0.0, kbSize.height+50.0, 0.0);
            scrollView.contentInset = contentInsets;
            scrollView.scrollIndicatorInsets = contentInsets;
        }

        CGRect aRect = self.view.frame;

        aRect.size.height -= kbSize.height + self.navigationController.navigationBar.frame.size.height + [UIApplication sharedApplication].statusBarFrame.size.height;
        CGPoint origin = activeField.frame.origin;
        origin.y -= scrollView.contentOffset.y;
        if (!CGRectContainsPoint(aRect, origin) ) {
            CGPoint scrollPoint = CGPointMake(0.0, activeField.frame.origin.y-(aRect.size.height));
            [scrollView setContentOffset:scrollPoint animated:YES];
        }
    }
    else appDelegate.isAfterSave = NO;    
}

当然,这不是一种美妙的方式,但它解决了这个问题