这是对此问题的跟进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"]
,如何在我的代码中使用它而不是正则表达式匹配?
答案 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
子类。在那里,您可以更好地控制已更改的文本范围,并仅将突出显示应用于该部分。