我的应用程序中有一个UITextView,它会在用户输入时更新。但是当我开始输入UITextView时,光标开始闪烁并快速上下移动。我该如何修复这个错误?
这是UITextView的更新方式 -
一个名为Highlighter.swift的Cocoa Touch类文件 -
import Foundation
import UIKit
// text hightlighter
class SyntaxGroup {
var wordCollection : [String]
var type : String
var color : UIColor
init(wordCollection_I : [String], type_I : String, color_I: UIColor) {
wordCollection = wordCollection_I
type = type_I
color = color_I
}
}
class SyntaxDictionairy {
var collections : [SyntaxGroup] = []
}
class SyntaxRange {
var range : NSRange
var color : UIColor
init (color_I : UIColor, range_I : NSRange) {
color = color_I
range = range_I
}
}
class HighLighter {
private var ranges : [SyntaxRange] = []
var highlightedString : NSMutableAttributedString = NSMutableAttributedString()
var syntaxDictionairy : SyntaxDictionairy
init (syntaxDictionairy_I : SyntaxDictionairy) {
syntaxDictionairy = syntaxDictionairy_I
}
func run(string : String?, completion: (finished: Bool) -> Void) {
ranges = []
highlightedString = NSMutableAttributedString()
var baseString = NSMutableString()
let qualityOfServiceClass = QOS_CLASS_DEFAULT
let backgroundQueue = dispatch_get_global_queue(qualityOfServiceClass, 0)
dispatch_async(backgroundQueue) { () -> Void in
if string != nil && string != "" {
self.highlightedString = NSMutableAttributedString(string: string!)
for i in 0..<self.syntaxDictionairy.collections.count {
for iB in 0..<self.syntaxDictionairy.collections[i].wordCollection.count {
let currentWordToCheck = self.syntaxDictionairy.collections[i].wordCollection[iB]
baseString = NSMutableString(string: string!)
while baseString.containsString(self.syntaxDictionairy.collections[i].wordCollection[iB]) {
let nsRange = (baseString as NSString).rangeOfString(currentWordToCheck)
let newSyntaxRange = SyntaxRange(color_I: self.syntaxDictionairy.collections[i].color, range_I: nsRange)
self.ranges.append(newSyntaxRange)
var replaceString = ""
for _ in 0..<nsRange.length {
replaceString += "§" // secret unallowed character
}
baseString.replaceCharactersInRange(nsRange, withString: replaceString)
}
}
}
for i in 0..<self.ranges.count {
self.highlightedString.addAttribute(NSForegroundColorAttributeName, value: self.ranges[i].color, range: self.ranges[i].range)
}
}
dispatch_sync(dispatch_get_main_queue()) { () -> Void in
completion(finished: true)
}
}
}
}
这是ViewController.swift文件中的代码 -
import UIKit
class ViewController: UIViewController, UITextViewDelegate {
@IBOutlet weak var myTextView: UITextView!
var syntaxHighLighter : HighLighter!
override func viewDidLoad() {
super.viewDidLoad()
setUpHighLighter()
myTextView.delegate = self
}
func setUpHighLighter() {
// build a dict of words to highlight
let redColor = UIColor(red: 0.5, green: 0.0, blue: 0.0, alpha: 1.0)
let blueColor = UIColor(red: 0.0, green: 0.0, blue: 0.5, alpha: 1.0)
let greenColor = UIColor(red: 0.0, green: 0.5, blue: 0.0, alpha: 1.0)
let redGroup = SyntaxGroup(wordCollection_I: ["red","bordeaux"], type_I: "Color", color_I: redColor)
let blueGroup = SyntaxGroup(wordCollection_I: ["coralblue","blue","skyblue","azur"], type_I: "Color", color_I: blueColor)
let greenGroup = SyntaxGroup(wordCollection_I: ["green"], type_I: "Color", color_I: greenColor)
let dictionairy : SyntaxDictionairy = SyntaxDictionairy()
dictionairy.collections.append(blueGroup)
dictionairy.collections.append(greenGroup)
dictionairy.collections.append(redGroup)
syntaxHighLighter = HighLighter(syntaxDictionairy_I: dictionairy)
}
func textViewDidChange(textView: UITextView) {
let currentRange = myTextView.selectedRange
syntaxHighLighter.run(myTextView.text) { (finished) -> Void in
self.myTextView.attributedText = self.syntaxHighLighter.highlightedString
self.myTextView.selectedRange = currentRange
}
}
}
答案 0 :(得分:1)
检查您是否正在运行最新的操作系统和Xcode。
尝试这一点来弄清楚它何时跳跃:
您可以再次使用bool
检查选择更改是否是由于突出显示后的重置而导致的。如果不是,您可以使用断点来查看正在发生的事情。
func textViewDidChangeSelection(textView: UITextView) {
print("selection changed: \(myTextView.selectedTextRange!)")
}
这可能会有所帮助:检查textview的长度是否仍然等于突出显示的字符串的长度。
func textViewDidChange(textView: UITextView) {
let currentRange = myTextView.selectedRange
syntaxHighLighter.run(myTextView.text) { (finished) -> Void in
// if the highlighter was slower than typing, ABORT
guard let textInUITextView = self.myTextfield.attributedText where textInUITextView.length == self.syntaxHighLighter.highlightedString.length else {
return
}
self.myTextView.attributedText = self.syntaxHighLighter.highlightedString
self.myTextView.selectedRange = currentRange
}
}
这可能会有所帮助:它会阻止荧光笔多次运行。
class HighLighter {
private var ranges : [SyntaxRange] = []
var highlightedString : NSMutableAttributedString = NSMutableAttributedString()
var syntaxDictionairy : SyntaxDictionairy
private var running : Bool = false
init (syntaxDictionairy_I : SyntaxDictionairy) {
syntaxDictionairy = syntaxDictionairy_I
}
func run(string : String?, completion: (finished: Bool) -> Void) {
if running == true {
print("double action")
return
}
running = true
ranges = []
highlightedString = NSMutableAttributedString()
var baseString = NSMutableString()
let qualityOfServiceClass = QOS_CLASS_DEFAULT
let backgroundQueue = dispatch_get_global_queue(qualityOfServiceClass, 0)
dispatch_async(backgroundQueue) { () -> Void in
if string != nil && string != "" {
self.highlightedString = NSMutableAttributedString(string: string!)
for i in 0..<self.syntaxDictionairy.collections.count {
for iB in 0..<self.syntaxDictionairy.collections[i].wordCollection.count {
let currentWordToCheck = self.syntaxDictionairy.collections[i].wordCollection[iB]
baseString = NSMutableString(string: string!)
while baseString.containsString(self.syntaxDictionairy.collections[i].wordCollection[iB]) {
let nsRange = (baseString as NSString).rangeOfString(currentWordToCheck)
let newSyntaxRange = SyntaxRange(color_I: self.syntaxDictionairy.collections[i].color, range_I: nsRange)
self.ranges.append(newSyntaxRange)
var replaceString = ""
for _ in 0..<nsRange.length {
replaceString += "§" // secret unallowed character
}
baseString.replaceCharactersInRange(nsRange, withString: replaceString)
}
}
}
for i in 0..<self.ranges.count {
self.highlightedString.addAttribute(NSForegroundColorAttributeName, value: self.ranges[i].color, range: self.ranges[i].range)
}
}
dispatch_sync(dispatch_get_main_queue()) { () -> Void in
self.running = false
completion(finished: true)
}
}
}
}