• <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
            国产精品久久久久…| 91久久成人免费| 性欧美丰满熟妇XXXX性久久久 | 久久青草国产精品一区| 天天爽天天爽天天片a久久网| 久久精品天天中文字幕人妻 | 91麻精品国产91久久久久| 中文字幕久久欲求不满| 久久久这里只有精品加勒比| 亚洲国产精品一区二区久久hs| 国产Av激情久久无码天堂| 久久久久亚洲精品天堂| 国产精品久久久久久久久久免费| 欧美亚洲国产精品久久高清 | 久久久无码精品亚洲日韩蜜臀浪潮 | 99久久人妻无码精品系列| 青青草原综合久久大伊人导航 | 久久精品无码专区免费| 中文字幕久久波多野结衣av| 久久99精品久久久久久动态图| 精品久久久久久久国产潘金莲| 1000部精品久久久久久久久| 国内精品久久久久国产盗摄| 蜜臀av性久久久久蜜臀aⅴ麻豆 | 亚洲精品乱码久久久久久蜜桃| 日产精品久久久一区二区| AA级片免费看视频久久| 国产成人久久AV免费| 久久久久AV综合网成人| 中文字幕乱码久久午夜| 亚洲综合久久夜AV | 亚洲国产精品综合久久网络| 久久激情亚洲精品无码?V| 韩国三级大全久久网站| 欧美大香线蕉线伊人久久| 日本人妻丰满熟妇久久久久久| 亚洲伊人久久精品影院| 久久精品国产久精国产一老狼| 久久伊人精品青青草原日本| 久久99精品国产麻豆不卡| 国产毛片久久久久久国产毛片|