Dismiss keyboard on touch anywhere outside UITextField

前端 未结 8 1813
梦毁少年i
梦毁少年i 2020-11-27 17:22

I am developing an iPad app that has a large number of UIViewControllers, UITableViews (with cells with accessoryViews of UIText

相关标签:
8条回答
  • 2020-11-27 17:26

    I actually really like Omar's technique - it works beautifully - but I've added a couple of lines in my case.

    - (void)keyboardDidShow:(NSNotification*)notification
    {
        UIButton *button = [[UIButton alloc] init];
    
        CGRect rect = self.view.bounds;
    
        button.frame = rect;
        button.backgroundColor = [UIColor blackColor];
    
        [button setAlpha:0.5f];
        button.tag = 111;
        UIView *currentResponder = [self.view findFirstResponder];
    
        [self.view bringSubviewToFront:currentResponder];
    
        if (currentResponder == _descriptionTextView) [_descriptionTextView setTextColor:[UIColor whiteColor]];
    
        [button addTarget:currentResponder action:@selector(resignFirstResponder) forControlEvents:UIControlEventTouchUpInside];
    
        [self.view insertSubview:button belowSubview:currentResponder];
    }
    

    I've added three lines to his solution (see above):

    1. Set the alpha of the button to 0.6, so that I can see some of my viewController behind it.
    2. Bring the currentResponder to the front, so that none of my other views are in front of the button.
    3. Switch the colour of my UITextView so that the text is clearer (my UITextView has a clear background, so the text wasn't showing clearly).

    Anyway, only minor changes, but hopefully these help. Thanks Omar.

    0 讨论(0)
  • 2020-11-27 17:31

    In Swift 3

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
    
        self.view.endEditing(true)
    }
    
    0 讨论(0)
  • 2020-11-27 17:33

    Here is a much easier and efficient way of dealing with that. This is gonna work for any UITextField in your view controller. You can even add it to your base view controller (if you have got one) and it will work like a charm.

    - (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
    
        UITouch *touch = [[event allTouches] anyObject];
    
        if (![[touch view] isKindOfClass:[UITextField class]]) {
            [self.view endEditing:YES];
        }
        [super touchesBegan:touches withEvent:event];
    }
    
    0 讨论(0)
  • 2020-11-27 17:37

    Add a "textFieldDidChange" notification method to the text field control.

    [textField addTarget:self action:@selector(textFieldDidChange:) forControlEvents:UIControlEventEditingDidEnd];

    works for me

    0 讨论(0)
  • 2020-11-27 17:40

    I normally use the following

    First:

    Include the following extension in your implementation file, findFirstResponder will help you find the first resonder

    @implementation UIView (FindFirstResponder)
    - (UIView*)findFirstResponder
    {
        if (self.isFirstResponder) {
            return self;     
        }
        for (UIView *subView in self.subviews) {
            UIView *responder = [subView findFirstResponder];
            if (responder)
                return responder;
        }
    
        return nil;
    }
    @end
    

    Then in your view controller viewDidLoad add the following

    - (void)viewDidLoad
    {
        [super viewDidLoad];
        NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
        [center addObserver:self selector:@selector(keyboardDidShow:) 
                       name:UIKeyboardDidShowNotification 
                     object:nil];
    
        [center addObserver:self selector:@selector(keyboardDidHide:) 
                       name:UIKeyboardWillHideNotification 
                     object:nil];
        // Do any additional setup after loading the view, typically from a nib.
    }
    

    The notification functions will be like this

    - (void) keyboardDidShow:(NSNotification*)notification
    {    
        UIButton *button = [[UIButton alloc] init];
    
        CGRect rect = self.view.bounds;
    
    
        button.frame = rect;
        button.backgroundColor = [UIColor blackColor];
        button.tag = 111;
        UIView *currentResponer = [self.view findFirstResponder];
        [button addTarget:currentResponer action:@selector(resignFirstResponder) forControlEvents:UIControlEventTouchUpInside];
    
        [self.view insertSubview:button belowSubview:currentResponer];
    }
    
    - (void) keyboardDidHide:(NSNotification*)notification
    {
        [[self.view viewWithTag:111] removeFromSuperview];
    }
    

    When the keyboard shows, i add a UIButton beneath the current first responder, this button action will hide the keyboard,

    A limitation here is that the UITextField has to be in self.view however you could adapt this technique for your need with some modifications, hope it helps you

    0 讨论(0)
  • 2020-11-27 17:49

    Your view hierarchy lives inside a UIWindow. The UIWindow is responsible for forwarding touch events to the correct view in its sendEvent: method. Let's make a subclass of UIWindow to override sendEvent:.

    @interface MyWindow : UIWindow
    @end
    

    The window will need a reference to the current first responder, if there is one. You might decide to also use UITextView, so we'll observe notifications from both text fields and text views.

    @implementation MyWindow {
        UIView *currentFirstResponder_;
    }
    
    - (void)startObservingFirstResponder {
        NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
        [center addObserver:self selector:@selector(observeBeginEditing:) name:UITextFieldTextDidBeginEditingNotification object:nil];
        [center addObserver:self selector:@selector(observeEndEditing:) name:UITextFieldTextDidEndEditingNotification object:nil];
        [center addObserver:self selector:@selector(observeBeginEditing:) name:UITextViewTextDidBeginEditingNotification object:nil];
        [center addObserver:self selector:@selector(observeEndEditing:) name:UITextViewTextDidEndEditingNotification object:nil];
    }
    
    - (void)stopObservingFirstResponder {
        NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
        [center removeObserver:self name:UITextFieldTextDidBeginEditingNotification object:nil];
        [center removeObserver:self name:UITextFieldTextDidEndEditingNotification object:nil];
        [center removeObserver:self name:UITextViewTextDidBeginEditingNotification object:nil];
        [center removeObserver:self name:UITextViewTextDidEndEditingNotification object:nil];
    }
    
    - (void)observeBeginEditing:(NSNotification *)note {
        currentFirstResponder_ = note.object;
    }
    
    - (void)observeEndEditing:(NSNotification *)note {
        if (currentFirstResponder_ == note.object) {
            currentFirstResponder_ = nil;
        }
    }
    

    The window will start observing the notifications when it's initialized, and stop when it's deallocated:

    - (id)initWithCoder:(NSCoder *)aDecoder {
        if ((self = [super initWithCoder:aDecoder])) {
            [self commonInit];
        }
        return self;
    }
    
    - (id)initWithFrame:(CGRect)frame {
        if ((self = [super initWithFrame:frame])) {
            [self commonInit];
        }
        return self;
    }
    
    - (void)commonInit {
        [self startObservingFirstResponder];
    }
    
    - (void)dealloc {
        [self stopObservingFirstResponder];
    }
    

    We'll override sendEvent: to “adjust” the first responder based on the event, and then call super's sendEvent: to send the event normally.

    - (void)sendEvent:(UIEvent *)event {
        [self adjustFirstResponderForEvent:event];
        [super sendEvent:event];
    }
    

    We don't need to do anything about the first responder if there is no first responder. If there is a first responder, and it contains a touch, we don't want to force it to resign. (Remember, there can be multiple touches simultaneously!) If there is a first responder, and a new touch appears in another view that can become the first responder, the system will handle that correctly automatically, so we also want to ignore that case. But if there is a first responder, and it doesn't contain any touches, and a new touch appears in a view that can't become first responder, we want to make the first responder resign.

    - (void)adjustFirstResponderForEvent:(UIEvent *)event {
        if (currentFirstResponder_
            && ![self eventContainsTouchInFirstResponder:event]
            && [self eventContainsNewTouchInNonresponder:event]) {
            [currentFirstResponder_ resignFirstResponder];
        }
    }
    

    Reporting whether an event contains a touch in the first responder is easy:

    - (BOOL)eventContainsTouchInFirstResponder:(UIEvent *)event {
        for (UITouch *touch in [event touchesForWindow:self]) {
            if (touch.view == currentFirstResponder_)
                return YES;
        }
        return NO;
    }
    

    Reporting whether an event contains a new touch in a view that can't become first responder is almost as easy:

    - (BOOL)eventContainsNewTouchInNonresponder:(UIEvent *)event {
        for (UITouch *touch in [event touchesForWindow:self]) {
            if (touch.phase == UITouchPhaseBegan && ![touch.view canBecomeFirstResponder])
                return YES;
        }
        return NO;
    }
    
    @end
    

    Once you've implemented this class, you need to change your app to use it instead of UIWindow.

    If you're creating your UIWindow in application:didFinishLaunchingWithOptions:, you need to #import "MyWindow.h" at the top of your AppDelegate.m, and then change application:didFinishLaunchingWithOptions: to create a MyWindow instead of a UIWindow.

    If you're creating your UIWindow in a nib, you need to set the custom class of the window to MyWindow in the nib.

    0 讨论(0)
提交回复
热议问题