如何在textView中更改预先选择的单词的样式?

时间:2016-11-10 15:52:18

标签: ios swift uitextview swift3 nsmutableattributedstring

这是对此问题的跟进How can I change style of some words in my UITextView one by one in Swift?

感谢@Josh的帮助,我能够编写一段代码,突出显示以#开头的每个单词 - 并逐一进行。我的最终代码是:

func highlight (to index: Int) {

    let regex = try? NSRegularExpression(pattern: "#(\\w+)", options: [])
    let matches = regex!.matches(in: hashtagExplanationTextView.text, options: [], range: NSMakeRange(0, (hashtagExplanationTextView.text.characters.count)))
    let titleDict: NSDictionary = [NSForegroundColorAttributeName: orangeColor]
    let titleDict2: NSDictionary = [NSForegroundColorAttributeName: UIColor.red]
    let storedAttributedString = NSMutableAttributedString(string: hashtagExplanationTextView.text!, attributes: titleDict as! [String : AnyObject])


    let attributedString = NSMutableAttributedString(attributedString: storedAttributedString)
    guard index < matches.count else {
        return
    }

    for i in 0..<index{
        let matchRange = matches[i].rangeAt(0)
        attributedString.addAttributes(titleDict2 as! [String : AnyObject], range: matchRange)
    }
    hashtagExplanationTextView.attributedText = attributedString
    if #available(iOS 10.0, *) {
        let _ = Timer.scheduledTimer(withTimeInterval: 1, repeats: false) { _ in
            self.highlight(to: index + 1)
        }
    } else {
        DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
            self.highlight(to: index + 1)
        }
    }
}

这样可以正常工作,但我想更改逻辑,使其不突出显示#个单词,而是从这些单词的预选数组中突出显示(逐个)单词。

所以我有这个数组var myArray:[String] = ["those","words","are","highlighted"],如何在我的代码中使用它而不是正则表达式匹配?

3 个答案:

答案 0 :(得分:3)

我相信您正在使用正则表达式来获取NSRange的数组。在这里,您需要一个稍微不同的数据结构,如[String : [NSRange]]。然后,您可以使用rangeOfString函数来检测单词所在的NSRange。您可以按照以下示例进行操作:

let wordMatchArray:[String] = ["those", "words", "are", "highlighted"]
let labelText:NSString = NSString(string: "those words, those ldsnvldnvsdnds, are, highlighted,words are highlighted")
let textLength:Int = labelText.length

var dictionaryForEachWord:[String : [NSRange]] = [:]

for eachWord:String in wordMatchArray {

   var prevRange:NSRange = NSMakeRange(0, 0)
   var rangeArray:[NSRange] = []

   while ((prevRange.location + prevRange.length) < textLength) {

      let start:Int = (prevRange.location + prevRange.length)
      let rangeEach:NSRange = labelText.range(of: eachWord, options: NSString.CompareOptions.literal, range: NSMakeRange(start, textLength-start))
      if rangeEach.length == 0 {
         break
      }
      rangeArray.append(rangeEach)
      prevRange = rangeEach
   }

   dictionaryForEachWord[eachWord] = rangeArray
}

现在,对于存储在字典中的每个单词,您有一个NSRange数组,[NSRange],您可以在UITextView中相应地突出显示每个单词。

如果您对实施有任何疑问,请随意发表评论:)

答案 1 :(得分:0)

对于这个新要求,您不需要正则表达式,您可以迭代您的单词数组并使用rangeOfString来查明该字符串是否存在并设置所定位范围的属性。

要匹配原始功能,找到匹配的范围后,您需要再次搜索,从该范围的结尾开始,以查看源文本中是否还有其他匹配。

答案 2 :(得分:0)

到目前为止,建议的解决方案建议您浏览每个单词,然后在文本视图中搜索它们。这样做有效,但是你正在遍历文本的次数太多了。

我建议的是列举文本中的所有单词,看看它们是否与要强调的任何单词匹配:

class ViewController: UIViewController {

    @IBOutlet var textView: UITextView!

    override func viewDidLoad() {
        super.viewDidLoad()
        textView.delegate = self
        highlight()
    }

    func highlight() {
        guard let attributedText = textView.attributedText else {
            return
        }
        let wordsToHighlight = ["those", "words", "are", "highlighted"]
        let text = NSMutableAttributedString(attributedString: attributedText)
        let textRange = NSRange(location: 0, length: text.length)
        text.removeAttribute(NSForegroundColorAttributeName, range: textRange)

        (text.string as NSString).enumerateSubstrings(in: textRange, options: [.byWords]) { [weak textView] (word, range, _, _) in
            guard let word = word else { return }
            if wordsToHighlight.contains(word) {
                textView?.textStorage.setAttributes([NSForegroundColorAttributeName: UIColor.red], range: range)
            } else {
                textView?.textStorage.removeAttribute(NSForegroundColorAttributeName, range: range)
            }
        }
        textView.typingAttributes.removeValue(forKey: NSForegroundColorAttributeName)
    }
}

extension ViewController: UITextViewDelegate {
    func textViewDidChange(_ textView: UITextView) {
        highlight()
    }
}

对于小文本,这应该没问题。对于长文本,在每次更改中查看所有内容都会严重影响性能。在这种情况下,我建议使用自定义NSTextStorage子类。在那里,您可以更好地控制已更改的文本范围,并仅将突出显示应用于该部分。