表情鍵盤及文字表情識別簡單demo

来源:http://www.cnblogs.com/howie-ch/archive/2016/08/18/5785395.html
-Advertisement-
Play Games

1、前言 瞭解了簡單圖文混排 (屬性字元串的使用)及 正則表達式的部分知識,為了加深印象,寫了個簡單表情鍵盤demo 展示文字用的是 UITextView 由於時間匆忙,存在一些bug,以及不完善的地方,僅作為小demo 練習一下 圖文混排可以用 TextKit ,下次有時間學習下 環境 xcode ...


1、前言

  • 瞭解了簡單圖文混排 (屬性字元串的使用)及 正則表達式的部分知識,為了加深印象,寫了個簡單表情鍵盤demo

  • 展示文字用的是 UITextView

  • 由於時間匆忙,存在一些bug,以及不完善的地方,僅作為小demo 練習一下

  • 圖文混排可以用 TextKit ,下次有時間學習下

  • 環境 xcode 7.3 , swift 2.3

2、效果

3、表情鍵盤相關

  • Emoticons.bundle 資源包來源於網路

  • 設計好模型 如(EmoticonPackage、EmoticonItem)


class EmoticonPackage: NSObject {
    // MARK: **** 屬性 ****
    var id: String?
    lazy var emoticons: [EmoticonItem] = [EmoticonItem]()
    // MARK: **** kvc ****
    override init() { 
    }
    init(dict: [String: AnyObject]) {
        super.init()
        setValuesForKeysWithDictionary(dict)
    }
    override func setValue(value: AnyObject?, forUndefinedKey key: String) {   
    }
}

class EmoticonItem: NSObject {
    /// 文件夾名
    var id: String? {
        didSet {
            guard let pn = png else {
                return
            }
            guard let iD = id else {
                return
            }
            pngPath = iD  + "/" + pn
        }
    }
    /// emoji
    var code: String? {
        didSet {
            guard let codeStr = code else {
                return
            }
            let scanner = NSScanner(string: codeStr)
            var result: UInt32 = 0
            scanner.scanHexInt(&result)
            let ch = Character(UnicodeScalar(result))
            emojiCode = "\(ch)"
        }
    }
    var emojiCode: String?
    /// 文字
    var chs: String?
    /// 圖片
    var png: String?
    /// 使用次數
    var count: Int = 0
    /// 圖片路徑
    var pngPath: String?

    // MARK: **** kvc ****
    override init() {   
    }
    init(dict: [String: AnyObject]) {
        super.init()
        setValuesForKeysWithDictionary(dict)
    }
    override func setValue(value: AnyObject?, forUndefinedKey key: String) {        
    }
}
  • 表情鍵盤 EmoticonKeyboardView 為一個自定義view ,裡面包含一個collectionView 來展示每個 EmoticonItem

    • collectionView 的cell 裡面是一個button,因為表情能夠顯示圖片也能夠點擊

    • 總共為4組數據, 最近這組數據為21個,包含最後的刪除按鈕,並且預設為空的,當點擊其他組的表情時,會根據該表情的點擊次數才決定在最近這組的顯示前後

    • 預設為圖片表情,浪小花為圖片表情,emoji為系統emoji表情

  • 設計了一個工具類 EmoticonTool 來處理數據模型

    • 每隔20個數據添加一個刪除按鈕 (刪除按鈕也是表情的模型,只是顯示的圖片是刪除的圖片)

    • 一頁是21個,每組的數據不足整頁的話需要補空白模型

  • 最近這組表情數據的添加



// MARK: **** 最近這組 添加表情 ****
extension EmoticonPackage {
    // 最近  第一組添加表情
    func addEmoticon(emoticon: EmoticonItem) {
        // 先移除刪除按鈕
        self.emoticons.removeAtIndex(20)
        // 判斷是否已經包含該表情
        let isContain = self.emoticons.contains(emoticon)
        if  !isContain {
            self.emoticons.append(emoticon)
        }
        // 數組中模型根據某一條件排序 生成新的數組
        let sortEmoArray = self.emoticons.sort { (e1, e2) -> Bool in
            return e1.count >= e2.count
        }
        self.emoticons = sortEmoArray
        // 添加刪除按鈕
        let deleEmo = EmoticonItem()
        deleEmo.png = "compose_emotion_delete"
        self.emoticons.insert(deleEmo, atIndex: 20)
        // 超過20 的 (空白)表情刪除 , 永遠只保留21個表情
        for i in 21..<self.emoticons.count {
            self.emoticons.removeAtIndex(i)
        }

    }
}
  • 插入表情是 在 UITextView 里做的

// MARK: **** 插入表情 ****
extension TextView {
    /// 插入表情
    func insertEmoticon(emoticon: EmoticonItem) {
        // 1、emoji表情
        if  emoticon.code != nil && emoticon.emojiCode != nil   {
            // 獲取游標的range
            
            guard let selectRange = self.selectedTextRange else {
                self.selectedRange = NSRange(location: 0, length: 0)
                self.replaceRange(self.selectedTextRange!, withText: emoticon.emojiCode!)
                return
            }
            self.replaceRange(selectRange, withText: emoticon.emojiCode!)
            return
        }
        // 2、圖片表情
        // 獲取當前的顯示的內容 生成一個屬性字元串
        let strM = NSMutableAttributedString(attributedString: self.attributedText)
        
        
        // strM 里 替換
        // textView 游標位置
        let range = self.selectedRange
        
        // 把圖片 變成屬性字元串 ,並設置 attachment 的高度, 設置圖片表情的bounds,沒有frame
        guard let attrStr = TextAttachment.createAttachmetn(emoticon, height: self.font!.lineHeight) else {
            return
        }
        strM.replaceCharactersInRange(range, withAttributedString: attrStr)
        // 設置 游標位置下一個的 strM 的字體大小為 textView的字體大小
        // 解決插入第一個表情圖片大小合適,第二個以後的所有表情都比較小的問題
        strM.addAttribute(NSFontAttributeName, value: self.font! , range: NSMakeRange(range.location, 1))
        // 設置 textView 的屬性文本
        self.attributedText = strM
        // 還原textView 的游標位置
        // 解決插入表情後游標始終在最後面而不是在當前位置
        self.selectedRange = NSRange(location: range.location + 1 , length: range.length)
    }
    
}

4、識別文字中的表情及高亮url、特殊欄位

  • 當UITextView 只是用來展示文字的時候需要 讓UITextViw 不能編輯

  • 根據表情文字找到對應的表情模型 ,在EmoticonTool工具類里


// MARK: **** 根據表情文字找到對應的表情模型 ****
extension EmoticonTool {
    /**
     根據表情文字找到對應的表情模型
     :param: str 表情文字
     :returns: 表情模型
     */
    class func emoticonWithStr(str: String) -> EmoticonItem?
    {
        var emoticon: EmoticonItem?
        for package in getDefaultEmoticon()!
        {
            // filter ,找數組中模型,條件是模型的某個屬性等於某個值
            emoticon = package.emoticons.filter({ (emo) -> Bool in
                return emo.chs == str
            }).last
            if  emoticon != nil {
                break
            }

        }
        return emoticon
    }
}
  • 識別、匹配表情

// MARK: **** 識別、匹配表情 ****
extension UITextView {
     func recognizeEmoticon() {
        // 表情符號的 range
        let pattern = "\\[\\w+\\]"
        guard let regex = try? NSRegularExpression(pattern: pattern, options: NSRegularExpressionOptions(rawValue: 0)) else {
            return
        }
        let result = regex.matchesInString(self.text, options: NSMatchingOptions(rawValue: 0), range: NSRange(location: 0, length: self.text.characters.count))
        // textView裡面最開始的文字
        let strM = NSMutableAttributedString(attributedString: self.attributedText)
        // 記錄textView 游標位置
        let cursorRange = self.selectedRange
        // 要從後往前遍歷 尋找表情文字
        for temp in result.reverse()  {
            let resultStr = (self.text as NSString).substringWithRange(temp.range)
            guard let emo = EmoticonTool.emoticonWithStr(resultStr) else {
                continue
            }
            
            guard let attachStr = TextAttachment.createAttachmetn(emo, height: self.font!.lineHeight) else {
                return
            }
            // 生成總的屬性字元串, 替換range位置 為  表情圖片的屬性字元串
            
            strM.replaceCharactersInRange(temp.range, withAttributedString: attachStr)
        }
        
        self.attributedText = strM
        self.selectedRange = cursorRange
        
    }
}
  • 高亮特殊欄位

// MARK: **** 高亮顯示特殊字元 ****
extension TextView {
    
    func setupHighlight() {
        // 高亮顯示
        let pattern1 = "#\\w+#"
        let pattern2 = "@\\w+:"
        let pattern3 = "(https?|ftp|file)://[-A-Za-z0-9+&@#/%?=~_|!:,.;]+[-A-Za-z0-9+&@#/%=~_|]?"
        let pattern4 = "\\[\\w+\\]"
        let pattern = pattern1 + "|" + pattern2 + "|" + pattern3 + "|" + pattern4
        guard let regex = try? NSRegularExpression(pattern: pattern, options: NSRegularExpressionOptions(rawValue: 0)) else {
            return
        }
        let resultArray = regex.matchesInString(self.text, options: NSMatchingOptions(rawValue: 0), range: NSRange(location: 0, length: self.text.characters.count))
        // textView裡面最開始的文字
        let strM = NSMutableAttributedString(attributedString: self.attributedText)
        for temp in resultArray {
            // 拿到range
            strM.setAttributes([NSForegroundColorAttributeName: UIColor.redColor(),
                NSFontAttributeName: self.font!], range: temp.range)
        }
        self.attributedText = strM

    }
}
  • 監聽特殊欄位的點擊

// MARK: **** 監聽特殊字元的點擊 ****
extension TextView {
    override func touchesBegan(touches: Set<UITouch>, withEvent event: UIEvent?) {
        // 獲取點
        guard let touch = (touches as NSSet).anyObject() as? UITouch else {
            return
        }
        let point = touch.locationInView(self)
        
        //
        let pattern1 = "#\\w+#"
        let pattern2 = "@\\w+:"
        let pattern3 = "http(s)?://([\\w-]+\\.)+[\\w-]+(/[\\w- ./?%&=]*)?"
        let pattern4 = "\\[\\w+\\]"
        let pattern = pattern1 + "|" + pattern2 + "|" + pattern3 + "|" + pattern4
        guard let regex = try? NSRegularExpression(pattern: pattern, options: NSRegularExpressionOptions(rawValue: 0)) else {
            return
        }
        let resultArray = regex.matchesInString(text, options: NSMatchingOptions(rawValue: 0), range: NSRange(location: 0, length: text.characters.count))
        for temp in resultArray {
            // 拿到range
            selectedRange = temp.range
            let array =  self.selectionRectsForRange(selectedTextRange!)
            for tmp in array {
                
                if CGRectContainsPoint(tmp.rect, point)  {
                    // 點在 範圍里 (tmp.rect frame)
                    // 加入想做的事情 

                    print( (text as NSString).substringWithRange(temp.range))
                    
                }
            }
            // 最後取消選中的range ,設置 range 只是來轉換為 rect frame
            self.selectedRange = NSRange(location: 0, length: 0)
        }
    }

}

5、小demo地址: https://github.com/CH-HOWIE/emoticon


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 對apply、call、bind的認識,並且列出一些它們的妙用加深記憶。 apply、call 在 javascript 中,call 和 apply 都是為了改變某個函數運行時的上下文(context)而存在的,換句話說,就是為了改變函數體內部 this 的指向。 JavaScript 的一大特點 ...
  • 2016-08-18 text-decoration blink貌似在firefox里也不起作用? 下劃線 頂劃線 消除線 文字閃爍 ...
  • 我們常用的在a標簽中有點擊事件:1. a href="javascript:js_method();" 這是我們平臺上常用的方法,但是這種方法在傳遞this等參數的時候很容易出問題,而且javascript:協議作為a的href屬性的時候不僅會導致不必要的觸發window.onbeforeunloa ...
  • 1. 概述 jquery允許拓展自定義的方法, 綁定到$.fn對象上, 編寫一個jQuery插件的原則: 2. example 3. 使用過濾 針對特定元素的拓展 類似submit方法只能作用在form上,我們也可以自定義針對指定dom元素使用的方法。 jquery的filter這時派上了用場。 e ...
  • xcode7下的ios模擬器輸入內容無法系統鍵盤,只能用電腦鍵盤輸入內容,這樣可能會對調試帶來麻煩。 其實xcode7下的ios模擬器預設只能使用一種,要麼是模擬器系統鍵盤,要麼就是是電腦鍵盤。設置方法如下: 打開ios模擬器菜單欄:Hardware-->Keyboard,會分別看到三個選項:Use ...
  • Swift - 用UIScrollView實現視差動畫效果 效果 源碼 https://github.com/YouXianMing/Swift-Animations ...
  • 3 RenderScript運行時層與反射層 3.1 RenderScript運行時層 RenderScript運行時層是指.rs代碼運行時所在的層級。當對安卓項目進行編譯的時候,.rs或者.rsh中編寫的代碼都會被llvm編譯器編譯成位元組碼。當該安卓應用在設備上運行的時候,這些位元組碼將會被設備上另 ...
  • 簡單敘述了在開發過程中遇到的 autoresizing 及 autoresizingMask 相關的問題 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...