我用mid编写了这个Extension方法。
func mid(fromIndex: Int, toIndex: Int) -> String {
var _fromIndex = 0
if( fromIndex >= self.count ) {
return ""
}
else if( fromIndex > 0 ){
_fromIndex = fromIndex
}
var _toIndex = 0
if( toIndex >= self.count ) {
_toIndex = self.count - 1
}
else if( toIndex > 0 ){
_toIndex = toIndex
}
if( _fromIndex > _toIndex ){
return ""
}
return String(self[self.index(self.startIndex, offsetBy: _fromIndex)...self.index(self.startIndex, offsetBy: _toIndex)])
}并使用此方法检索已删除的用户字符串。
func textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool {
defer{
if text.isEmpty {
let deletedText = textView.attributedText.string.mid(fromIndex: range.lowerBound, toIndex: range.upperBound)
}
}
}但有时它会抛出这个异常。
Fatal Exception: NSRangeException *** -[NSBigMutableString characterAtIndex:]: Index 672 out of bounds; string length 672 specialized String.subscript.getter
在使用String.subscript.getter之前,我已经检查了字符串的大小。
if( toIndex >= self.count ) {
_toIndex = self.count - 1
}文本操作应该在主线程上。为什么我还能得到这个例外?我是否误解了Swift中的字符串索引?
UPDATE1
根据@Wattholm的回答。我正在使用这种方式从NSRange从UITextView获取文本。
let contents: String = textView.text
if let _range = Range(range, in: contents) {
let deletedText = String(contents[_range])
}
else {
logHelper.w("Failed to get deletedText from NSRange, try UITextRange.")
if let startPosition = textView.position(from: textView.beginningOfDocument, offset: range.location),
let endPosition = textView.position(from: startPosition, offset: range.length),
let textRange = textView.textRange(from: startPosition, to: endPosition),
let deletedText = textView.text(in: textRange) {
// do something with deletedText
}
else {
logHelper.w("Failed to get deletedText from UITextRange, try NSString.")
let deletedText = (contents as NSString).mid(fromIndex: range.lowerBound, toIndex: range.upperBound)
}
}但它仍然属于第二个else和抛出NSRangeException。这意味着长度和索引仍然不匹配。
UPDATE2
尝试使用NSString.substring,但仍然继续获得此异常。
let deletedText = (contents as NSString).substring(with: range)也许UITextView中有一个bug?
发布于 2019-10-12 13:17:23
我可能是错的,但我认为这里的问题涉及字符串和NSString之间的内在区别,或者Range与NSRange之间的内在区别。字符串的计数并不总是与NSString的长度相同。
可能有一种更简单的方法,但我尝试的是为NSString创建一个扩展,就像您为字符串创建一个扩展一样,然后将'contents‘(我假设是一个字符串;这里似乎缺少一些代码,因为“contents”没有在任何地方声明)转换为NSString,这样它就可以正常工作了,因为textview为我们提供了NSRange:
extension NSString {
func mid(fromIndex: Int, toIndex: Int) -> NSString {
var _fromIndex = 0
if( fromIndex >= self.length ) {
return ""
}
else if( fromIndex > 0 ){
_fromIndex = fromIndex
}
var _toIndex = 0
if( toIndex >= self.length ) {
_toIndex = self.length - 1
}
else if( toIndex > 0 ){
_toIndex = toIndex
}
if( _fromIndex > _toIndex ){
return ""
}
return NSString(string: self.substring(with: NSRange(location: _fromIndex, length: _toIndex - _fromIndex + 1)))
}
}对于文本视图:
func textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool {
defer{
if text.isEmpty {
let deletedText = (contents as NSString).mid(fromIndex: range.lowerBound, toIndex: range.upperBound - 1)
}
}
}另一种方法是将NSRange转换为一个范围,这可能是更好的方法,特别是当您要求字符计数与字符串在Swift中的字符数完全相同时。textview函数中的range: NSRange可以转换为如下所示的范围:
设swiftRange = Range(range,in: rangeText)
其中,应将rangeText替换为范围所引用的text属性。
发布于 2019-10-26 09:59:54
这是一个更快捷的版本,它利用了从NSRange到Range<String.Index>的可靠转换,反之亦然。
extension String {
func mid(fromIndex: Int, toIndex: Int) -> String {
guard toIndex >= fromIndex else { return "" }
let nsRange = NSRange(location: fromIndex, length: toIndex + 1 - fromIndex)
guard let range = Range(nsRange, in: self) else { return "" }
return String(self[range])
}
}实际上,您不需要扩展,因为您已经有了一个NSRange,只需编写
func textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool {
defer{
if text.isEmpty {
let textViewString = textView.attributedText.string
guard let swiftRange = Range(range, in: textViewString) else { return }
let deletedText = String(textViewString[swiftRange])
}
}
}https://stackoverflow.com/questions/58353002
复制相似问题