• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            面對現實,超越自己
            逆水行舟,不進則退
            posts - 269,comments - 32,trackbacks - 0

            在iOS應用開發中,有三類視圖對象會打開虛擬鍵盤,進行輸入操作,但如何關閉虛擬鍵盤,卻沒有提供自動化的方法。這個需要我們自己去實現。這三類視圖對象分別是UITextField,UITextView和UISearchBar。
            這里介紹一下UITextField中關閉虛擬鍵盤的幾種方法。

             

            (原文鏈接: http://mikixiyou.iteye.com/blog/1753330 )


            第一種方法,使用它的委托UITextFieldDelegate中的方法textFieldShouldReturn:來關閉虛擬鍵盤。
            在UITextField視圖對象如birdNameInput所在的類中實現這個方法。

            - (BOOL)textFieldShouldReturn:(UITextField *)textField {  
                if ((textField == self.birdNameInput) || (textField == self.locationInput)) {  
                    [textField resignFirstResponder];  
                }  
                return YES;  

             這樣,在輸入框birdNameInput中打開虛擬鍵盤后,輕擊鍵盤的return鍵就會自動關閉掉虛擬鍵盤。


            第二種方法,將birdNameInput的屬性中Return Key修改為done,再定義一個方法和Done鍵的Did End On Exit連接。通過輕擊done鍵觸發這個事件來關閉虛擬鍵盤。
            定義的方法如下:

            - (IBAction) textFieldDoneEditing:(id)sender  
            {  
                    [sender resignFirstResponder];  
             

            這兩個方法都是輕擊虛擬鍵盤上一個鍵來關閉它。這屬于精確操作,而手指不像鼠標,做這種操作不容易。因此就UI層面而言,這兩個方法都不是最好的方法。
            在iphone或ipad屏幕上,虛擬鍵盤占用的面積大小是有限的。通過輕擊虛擬鍵盤之外的區域而關閉虛擬鍵盤。

             

            第三種方法,通過輕擊鍵盤之外的空白區域關閉虛擬鍵盤。
            在birdNameInput所屬的視圖控制器類的viewDidLoad方法中定義一個UITapGestureRecognizer的對象,然后將它賦值為它的視圖。

            UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc]   initWithTarget:self action:@selector(dismissKeyboard)];  
            [self.view addGestureRecognizer:tap];  
            [tap release]; 


            再定義一下選擇器調用的方法dismissKeyboard。

            -(void)dismissKeyboard {  
                   [birdNameInput resignFirstResponder];  


            如果屏幕上有多個textField的話,一個一個地列出來就有些麻煩。那么將方法修改一下,如下:

            -(void)dismissKeyboard {  
                NSArray *subviews = [self.view subviews];  
                for (id objInput in subviews) {  
                    if ([objInput isKindOfClass:[UITextField class]]) {  
                        UITextField *theTextField = objInput;  
                        if ([objInput isFirstResponder]) {  
                            [theTextField resignFirstResponder];  
                        }  
                    }  
                }  
            }

            如果這個屏幕上的視圖對象很復雜的話,另當別論。
            這個方法是編碼新建一個手勢對象。也可以直接使用interface builder圖形化開發工具,在storyboard中拉入一個手勢對象到視圖控制器類中,再將此手勢對象建立一個IBACTION,名稱可以是dismissKeyboard。

            第四種方法,通過輕擊鍵盤之外的空白區域關閉虛擬鍵盤。
            將屏幕上的view也就是textField的父視圖拖一個touch down事件出來,和一個能關閉虛擬鍵盤的方法連接。如果視圖沒有touch down事件,可將view的父類從UIView修改為UIButton。
            首先定義并實現一個方法backgroundTap:。

             - (IBAction) backgroundTap:(id)sender  
            {  
                    NSArray *subviews = [self.view subviews];  
                for (id objInput in subviews) {  
                    if ([objInput isKindOfClass:[UITextField class]]) {  
                        UITextField *theTextField = objInput;  
                        if ([objInput isFirstResponder]) {  
                            [theTextField resignFirstResponder];  
                        }  
                    }  
                }  
            }

            然后選擇背景視圖的Touch Down事件,連接 backgroundTap:即可。這樣只要輕擊一下虛擬鍵盤之外的區域,就能關閉虛擬鍵盤。這些方法都是使用resignFirstResponder方法來關閉虛擬鍵盤,還有其他的方法。

             

            第五種方法,使用endEditing:方法
            在所在的視圖控制器類中,覆蓋這個方法。

            - (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {  
                  [[self view] endEditing:YES];  
            }

             

            This method looks at the current view and its subview hierarchy for the text field that is currently the first responder. If it finds one, it asks that text field to resign as first responder. If the force parameter is set to YES, the text field is never even asked; it is forced to resign.
            但是,如果這個屏幕很復雜,虛擬鍵盤之外的區域中有很多按鈕。輕擊這些區域時可能會輕擊到這些按鈕,這樣虛擬鍵盤就不能關閉。
            要是找到一個沒有按鈕的空白區域都不容易且還有隱藏的視圖對象時,通過輕擊虛擬鍵盤之外的區域關閉虛擬鍵盤的方法實現起來就難了。

             

            第六種方法,覆蓋hitTest:withEvent:方法關閉虛擬鍵盤

             

            在stackoverflow.com上,有人這樣總結。說使用hitTest:withEvent:方法是最好的,也是最容易的解決方法。

             

            I think the easiest (and best) way to do this is to subclass your global view and use hitTest:withEvent method to listen to any touch. 
            Touches on keyboard aren't registered, so hitTest:withEvent is only called when you touch/scroll/swipe/pinch... somewhere else, then call [self endEditing:YES].
            This is better than using touchesBegan because touchesBegan are not called if you click on a button on top of the view. 
            It is better than UITapGestureRecognizer which can't recognize a scrolling gesture for example. It is also better than using a dim screen because in a complexe and dynamic user interface, you can't put dim screen every where. Moreover, it doesn't block other actions, you don't need to tap twice to select a button outside (like in the case of a UIPopover).
            Also, it's better than calling [textField resignFirstResponder], because you may have many text fields on screen, so this works for all of them.

             

            因此,我再建立一個繼承UIView的視圖類。在這個視圖類中,覆蓋hitTest:withEvent:方法,增加[self endEditing:YES]方法。

             - (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {  
            UIView *result = [super hitTest:point withEvent:event];  
            [self endEditing:YES]  
            return result;  

            我將視圖控制器的主視圖所屬類修改為這個新建視圖類。這樣在屏幕上輕擊任何位置都會關閉虛擬鍵盤。
            這個方法是最簡單,也是最好的關閉虛擬鍵盤的方法。
            使用好hitTest:withEvent:這個方法,還可以實現很多很復雜的功能。
            The implementation of hitTest:withEvent: in UIResponder does the following:

            •     It calls pointInside:withEvent: of self
            •     If the return is NO, hitTest:withEvent: returns nil. the end of the story.
            •     If the return is YES, it sends hitTest:withEvent: messages to its subviews. it starts from the top-level subview, and continues to other views until a subview returns a non-nil object, or all subviews receive the message.
            •     If a subview returns a non-nil object in the first time, the first hitTest:withEvent: returns that object. the end of the story.
            •     If no subview returns a non-nil object, the first hitTest:withEvent: returns self

            This process repeats recursively, so normally the leaf view of the view hierarchy is returned eventually.
            However, you might override hitTest:withEvent to do something differently. In many cases, overriding pointInside:withEvent: is simpler and still provides enough options to tweak event handling in your application.

            posted on 2014-06-23 13:54 王海光 閱讀(316) 評論(0)  編輯 收藏 引用 所屬分類: IOS
            少妇人妻综合久久中文字幕| 亚洲AV无码成人网站久久精品大| 欧美日韩成人精品久久久免费看 | 99久久无色码中文字幕人妻| 久久综合综合久久97色| 色妞色综合久久夜夜| 久久狠狠爱亚洲综合影院 | 四虎影视久久久免费观看| 久久久久亚洲精品男人的天堂| 国産精品久久久久久久| 久久久久亚洲AV综合波多野结衣| 精品久久久久中文字| 性做久久久久久免费观看| 一级a性色生活片久久无| 综合网日日天干夜夜久久| 亚洲va久久久噜噜噜久久狠狠| 亚洲日韩中文无码久久| 久久精品夜夜夜夜夜久久| 老司机国内精品久久久久| 久久综合九色综合精品| 四虎影视久久久免费| 无码专区久久综合久中文字幕| 欧美黑人又粗又大久久久| 亚洲国产精品久久66| 久久综合给合综合久久| 无码人妻久久久一区二区三区| 日韩人妻无码一区二区三区久久 | 狠狠色婷婷久久综合频道日韩| 麻豆成人久久精品二区三区免费 | 国内精品久久久久久野外| 久久精品国产精品亚洲| 久久精品国产99久久久古代| 91精品国产色综合久久| 亚洲国产成人久久综合一区77| 色综合久久综合中文综合网| 91亚洲国产成人久久精品网址| 久久综合九色综合网站| 国产精品欧美久久久久无广告 | 久久精品国产免费| 麻豆av久久av盛宴av| 久久精品免费大片国产大片|