我想突出显示两个字符串之间的文本差异。第一个字符串是正确的字符串,第二个是用户输入的字符串。它显示了两者之间的拼写和语法错误。下面的例子
我有一个解决方案,先检查每个单词,然后检查每个字母,它在某种程度上确实起作用,但并非在每种情况下都起作用。有更好的方法吗?
另一个无法正常工作的示例。
“ The”应为完全红色,在士兵中,“ i”和“ s”应为红色。所有“放入”应为红色。
func colorTextDiff(correctStr:String, answerText:String) -> NSAttributedString {
var hintTextIndex = 0
let attribute = NSMutableAttributedString.init(string: correctStr)
attribute.addAttribute(NSAttributedStringKey.foregroundColor, value: UIColor.black , range: NSRange.init(location: 0, length: correctStr.count))
attribute.addAttribute(NSAttributedStringKey.font, value: UIFont.systemFont(ofSize:14.0), range: NSRange.init(location: 0, length: correctStr.count))
let correctWords = correctStr.split(separator: " ")
var answerWords = answerText.split(separator: " ")
var answerWordIndex = 0
//match on words, when a word doesnt match test the word's character
for correctWord in correctWords {
if answerWordIndex>=answerWords.count { break}
let answerWord = answerWords[answerWordIndex]
var wrongChar = 0, answerCharIndex = 0
print("words ", correctWord, " ", answerWord)
if correctWord.lowercased() != answerWord.lowercased() {
for c in correctWord {
if answerCharIndex>=answerWord.count {
let len = correctWord.count-answerCharIndex
attribute.addAttribute(NSAttributedStringKey.foregroundColor, value: UIColor.red , range: NSRange.init(location: hintTextIndex, length: len))
hintTextIndex += len+1
break
}
let correctChar = String(c)
let answerChar = String(answerWord)[answerCharIndex..<answerCharIndex+1]
print("chars ", correctChar, " ", answerChar)
if correctChar.lowercased() != answerChar.lowercased() {
print("hint index: ", hintTextIndex)
attribute.addAttribute(NSAttributedStringKey.foregroundColor, value: UIColor.red , range: NSRange.init(location: hintTextIndex+1, length: 1))
wrongChar+=1
}
answerCharIndex+=1
hintTextIndex+=1
}
} else {
hintTextIndex += correctWord.count+1
}
if(wrongChar<correctWord.count) {answerWordIndex+=1 } //probably a missed word not missed typed word
}
hintTextIndex+=1
return attribute
}
答案 0 :(得分:2)
这是一个很酷的问题。
这是我的解决方法:
extension NSMutableAttributedString {
func setCharacterColor(at location: Int) {
let range = NSRange(location: location, length: 1)
self.addAttribute(NSAttributedStringKey.foregroundColor, value: UIColor.red, range: range)
}
}
extension Array where Element == Character {
func index(of character: Character, greaterThan index: Int) -> Int? {
for i in index...self.count where self[i] == character {
return i
}
return nil
}
}
let correctString = "The soldiers stormed into the village just after sunrise"
let incorrectString = "solder stormed village just after sunrise"
let correctArray = Array(correctString)
let incorrectArray = Array(incorrectString)
var correctMutableString = NSMutableAttributedString(string: correctString)
var currentPosition = 0
for char in incorrectArray {
guard let position = correctArray.index(of: char, greaterThan: currentPosition) else {
continue
}
while currentPosition < position {
correctMutableString.setCharacterColor(at: currentPosition)
currentPosition = currentPosition + 1
}
currentPosition = position + 1
}
labelCorrect.attributedText = correctMutableString
labelIncorrect.attributedText = NSMutableAttributedString(string: incorrectString)
它仍然是n阶的平方,但是这个问题总是很复杂。
它通过循环进行工作,直到找到正确字符串中的字符,其中该字符与错误字符串中的字符匹配。然后突出显示在此过程中经过的所有字符。关键是它只会突出显示新字符,而不会突出显示以前已经循环过的字符。