iOS在UITextField 輸入string限制長度默辨,解決輸入中文拼音被算入計(jì)數(shù)問題

需求:

在文本輸入框中輸入文字時限制輸入字符數(shù)為50压储,可以輸入emoji頭像鲜漩。

問題:

  1. 輸入漢字時拼音也會被算入計(jì)數(shù),被限制集惋;
  2. 輸入emoji頭像時在最后截取時出現(xiàn)頭像被截取一半的情況孕似;

代碼:

 #define WordsNumber 50
  • 添加注冊UITextField事件監(jiān)聽:
 [self.textField addTarget:self action:@selector(textFieldEditChanged:) forControlEvents:UIControlEventEditingChanged];
  • 監(jiān)聽方法實(shí)現(xiàn)
- (void)textFieldEditChanged:(UITextField *)textField
{
    NSString *lang = [textField textInputMode].primaryLanguage;
    if (StringEqual(lang, @"zh-Hans")) {
        //輸入簡體中文內(nèi)容
        //獲取高亮部分,如拼音
        UITextRange *selectedRange = [textField markedTextRange];
        UITextPosition *position = [textField positionFromPosition:selectedRange.start offset:0];
        if (!position) {
            [self handleTextFieldCharLength:textField];
        }
    }
    else{
        //輸入簡體中文以外的內(nèi)容
        [self handleTextFieldCharLength:textField];
    }
}
  • 輸入字符超出限制處理
- (void)handleTextFieldCharLength:(UITextField *)textField
{
    NSString *toBeString = textField.text;
    if (textField.text.length > WordsNumber) {
        //獲取超過50最大字符數(shù)的多余字符range
        NSRange rangeIndex = [toBeString rangeOfComposedCharacterSequenceAtIndex:WordsNumber];
        if (rangeIndex.length == 1){
            //如果多余字符的length = 1刮刑,則直接截取最大字符數(shù)
            textField.text = [toBeString substringToIndex:WordsNumber];
        }
        else{
            //如果多余字符的length > 1喉祭,則截取位置為(0.50),按輸入內(nèi)容單位截取
            NSRange rangeRange = [toBeString rangeOfComposedCharacterSequencesForRange:NSMakeRange(0, WordsNumber)];
            textField.text = [toBeString substringWithRange:rangeRange];
        }
    }
}

解決輸入emoj被截取一半的問題:

在官方文檔中已經(jīng)提供了解決方法:

/* To avoid breaking up character sequences such as Emoji, you can do:
    [str substringFromIndex:[str rangeOfComposedCharacterSequenceAtIndex:index].location]
    [str substringToIndex:NSMaxRange([str rangeOfComposedCharacterSequenceAtIndex:index])]
    [str substringWithRange:[str rangeOfComposedCharacterSequencesForRange:range]
*/
- (NSString *)substringFromIndex:(NSUInteger)from;
- (NSString *)substringToIndex:(NSUInteger)to;
- (NSString *)substringWithRange:(NSRange)range;
- (NSRange)rangeOfComposedCharacterSequenceAtIndex:(NSUInteger)index;
- (NSRange)rangeOfComposedCharacterSequencesForRange:(NSRange)range 

最后記錄下判斷文本中是否存在emoji的方法:

//過濾所有表情
+ (BOOL)stringContainsEmoji:(NSString *)string {
    __block BOOL returnValue = NO;
    [string enumerateSubstringsInRange:NSMakeRange(0, [string length]) options:NSStringEnumerationByComposedCharacterSequences usingBlock:^(NSString *substring, NSRange substringRange, NSRange enclosingRange, BOOL *stop) {
        const unichar hs = [substring characterAtIndex:0];
        // surrogate pair
        if (0xd800 <= hs && hs <= 0xdbff) {
            if (substring.length > 1) {
                const unichar ls = [substring characterAtIndex:1];
                const int uc = ((hs - 0xd800) * 0x400) + (ls - 0xdc00) + 0x10000;
                if (0x1d000 <= uc && uc <= 0x1f9c0) {
                    returnValue = YES;
                }
            }
        } else if (substring.length > 1) {
            const unichar ls = [substring characterAtIndex:1];
            if (ls == 0x20e3||ls == 0xfe0f ||ls == 0xd83c) {
                returnValue = YES;
            }
        } else {
            // non surrogate
            if (0x2100 <= hs && hs <= 0x27ff) {
                returnValue = YES;
            }
            else if (0x2B05 <= hs && hs <= 0x2b07)
            {
                returnValue = YES;
            }
            else if (0x2934 <= hs && hs <= 0x2935) {
                returnValue = YES;
            }
            else if (0x3297 <= hs && hs <= 0x3299) {
                returnValue = YES;
            }
            else if (hs == 0xa9 || hs == 0xae || hs == 0x303d || hs == 0x3030 || hs == 0x2b55 || hs == 0x2b1c || hs == 0x2b1b || hs == 0x2b50) {
                returnValue = YES;
            }
        }
    }];
    return returnValue;
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末雷绢,一起剝皮案震驚了整個濱河市泛烙,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌翘紊,老刑警劉巖胶惰,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異霞溪,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)中捆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進(jìn)店門鸯匹,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人泄伪,你說我怎么就攤上這事殴蓬。” “怎么了蟋滴?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵染厅,是天一觀的道長。 經(jīng)常有香客問我津函,道長肖粮,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任尔苦,我火速辦了婚禮涩馆,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘允坚。我一直安慰自己魂那,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布稠项。 她就那樣靜靜地躺著涯雅,像睡著了一般。 火紅的嫁衣襯著肌膚如雪展运。 梳的紋絲不亂的頭發(fā)上活逆,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天精刷,我揣著相機(jī)與錄音,去河邊找鬼划乖。 笑死贬养,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的琴庵。 我是一名探鬼主播误算,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼迷殿!你這毒婦竟也來了儿礼?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤庆寺,失蹤者是張志新(化名)和其女友劉穎蚊夫,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體懦尝,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡知纷,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了陵霉。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片琅轧。...
    茶點(diǎn)故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖踊挠,靈堂內(nèi)的尸體忽然破棺而出乍桂,到底是詐尸還是另有隱情,我是刑警寧澤效床,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布睹酌,位于F島的核電站,受9級特大地震影響剩檀,放射性物質(zhì)發(fā)生泄漏憋沿。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一沪猴、第九天 我趴在偏房一處隱蔽的房頂上張望卤妒。 院中可真熱鬧,春花似錦字币、人聲如沸则披。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽士复。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間阱洪,已是汗流浹背便贵。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留冗荸,地道東北人承璃。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像蚌本,于是被迫代替她去往敵國和親盔粹。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評論 2 345

推薦閱讀更多精彩內(nèi)容