ตั้งแต่Swift 4 (Xcode 9) ไลบรารีมาตรฐาน Swift มีวิธีการแปลงระหว่างช่วงสตริงของ Swift ( Range<String.Index>
) และNSString
ช่วง ( NSRange
) ตัวอย่าง:
let str = "a👿b🇩🇪c"
let r1 = str.range(of: "🇩🇪")!
// String range to NSRange:
let n1 = NSRange(r1, in: str)
print((str as NSString).substring(with: n1)) // 🇩🇪
// NSRange back to String range:
let r2 = Range(n1, in: str)!
print(str[r2]) // 🇩🇪
ดังนั้นการแทนที่ข้อความในวิธีการมอบหมายเขตข้อมูลข้อความสามารถทำได้ดังนี้
func textField(_ textField: UITextField,
shouldChangeCharactersIn range: NSRange,
replacementString string: String) -> Bool {
if let oldString = textField.text {
let newString = oldString.replacingCharacters(in: Range(range, in: oldString)!,
with: string)
// ...
}
// ...
}
(คำตอบที่เก่ากว่าสำหรับ Swift 3 และรุ่นก่อนหน้านี้ :)
ตั้งแต่ Swift 1.2 String.Index
มี initializer
init?(_ utf16Index: UTF16Index, within characters: String)
ซึ่งสามารถใช้ในการแปลงNSRange
ให้Range<String.Index>
ถูกต้อง (รวมถึงทุกกรณีของ Emojis ตัวชี้วัดระดับภูมิภาคหรือกลุ่มกราฟขยายอื่น ๆ ) โดยไม่มีการแปลงระดับกลางเป็นNSString
:
extension String {
func rangeFromNSRange(nsRange : NSRange) -> Range<String.Index>? {
let from16 = advance(utf16.startIndex, nsRange.location, utf16.endIndex)
let to16 = advance(from16, nsRange.length, utf16.endIndex)
if let from = String.Index(from16, within: self),
let to = String.Index(to16, within: self) {
return from ..< to
}
return nil
}
}
เมธอดนี้ส่งคืนช่วงสตริงที่เป็นทางเลือกเนื่องจากไม่ใช่NSRange
s ทั้งหมดที่ใช้ได้สำหรับสตริง Swift ที่กำหนด
UITextFieldDelegate
วิธีการของผู้ร่วมประชุมจากนั้นสามารถเขียนเป็น
func textField(textField: UITextField, shouldChangeCharactersInRange range: NSRange, replacementString string: String) -> Bool {
if let swRange = textField.text.rangeFromNSRange(range) {
let newString = textField.text.stringByReplacingCharactersInRange(swRange, withString: string)
// ...
}
return true
}
การแปลงผกผันคือ
extension String {
func NSRangeFromRange(range : Range<String.Index>) -> NSRange {
let utf16view = self.utf16
let from = String.UTF16View.Index(range.startIndex, within: utf16view)
let to = String.UTF16View.Index(range.endIndex, within: utf16view)
return NSMakeRange(from - utf16view.startIndex, to - from)
}
}
การทดสอบอย่างง่าย:
let str = "a👿b🇩🇪c"
let r1 = str.rangeOfString("🇩🇪")!
// String range to NSRange:
let n1 = str.NSRangeFromRange(r1)
println((str as NSString).substringWithRange(n1)) // 🇩🇪
// NSRange back to String range:
let r2 = str.rangeFromNSRange(n1)!
println(str.substringWithRange(r2)) // 🇩🇪
อัปเดตสำหรับ Swift 2:
rangeFromNSRange()
Serhii Yakovenko เวอร์ชั่น Swift 2 ได้ให้ไว้แล้วในคำตอบนี้ฉันรวมมันไว้ที่นี่เพื่อความสมบูรณ์:
extension String {
func rangeFromNSRange(nsRange : NSRange) -> Range<String.Index>? {
let from16 = utf16.startIndex.advancedBy(nsRange.location, limit: utf16.endIndex)
let to16 = from16.advancedBy(nsRange.length, limit: utf16.endIndex)
if let from = String.Index(from16, within: self),
let to = String.Index(to16, within: self) {
return from ..< to
}
return nil
}
}
เวอร์ชั่นของ Swift 2 NSRangeFromRange()
คือ
extension String {
func NSRangeFromRange(range : Range<String.Index>) -> NSRange {
let utf16view = self.utf16
let from = String.UTF16View.Index(range.startIndex, within: utf16view)
let to = String.UTF16View.Index(range.endIndex, within: utf16view)
return NSMakeRange(utf16view.startIndex.distanceTo(from), from.distanceTo(to))
}
}
อัปเดตสำหรับ Swift 3 (Xcode 8):
extension String {
func nsRange(from range: Range<String.Index>) -> NSRange {
let from = range.lowerBound.samePosition(in: utf16)
let to = range.upperBound.samePosition(in: utf16)
return NSRange(location: utf16.distance(from: utf16.startIndex, to: from),
length: utf16.distance(from: from, to: to))
}
}
extension String {
func range(from nsRange: NSRange) -> Range<String.Index>? {
guard
let from16 = utf16.index(utf16.startIndex, offsetBy: nsRange.location, limitedBy: utf16.endIndex),
let to16 = utf16.index(utf16.startIndex, offsetBy: nsRange.location + nsRange.length, limitedBy: utf16.endIndex),
let from = from16.samePosition(in: self),
let to = to16.samePosition(in: self)
else { return nil }
return from ..< to
}
}
ตัวอย่าง:
let str = "a👿b🇩🇪c"
let r1 = str.range(of: "🇩🇪")!
// String range to NSRange:
let n1 = str.nsRange(from: r1)
print((str as NSString).substring(with: n1)) // 🇩🇪
// NSRange back to String range:
let r2 = str.range(from: n1)!
print(str.substring(with: r2)) // 🇩🇪