1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129
|
//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2023 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
extension BidirectionalCollection where Index == String.Index {
internal func _alignIndex(roundingDown i: Index) -> Index {
index(i, offsetBy: 0)
}
internal func _alignIndex(roundingUp i: Index) -> Index {
let truncated = _alignIndex(roundingDown: i)
guard i > truncated && truncated < endIndex else { return truncated }
return index(after: truncated)
}
internal func _boundaryAlignedRange(_ r: some RangeExpression<Index>) -> Range<Index> {
let range = r.relative(to: self)
return _alignIndex(roundingDown: range.lowerBound)..<_alignIndex(roundingUp: range.upperBound)
}
internal func _checkRange(_ r: Range<Index>) -> Range<Index>? {
guard r.lowerBound >= startIndex, r.upperBound <= endIndex else {
return nil
}
return r
}
}
extension BidirectionalCollection {
func _trimmingCharacters(while predicate: (Element) -> Bool) -> SubSequence {
var idx = startIndex
while idx < endIndex && predicate(self[idx]) {
formIndex(after: &idx)
}
let startOfNonTrimmedRange = idx // Points at the first char not in the set
guard startOfNonTrimmedRange != endIndex else {
return self[endIndex...]
}
let beforeEnd = index(before: endIndex)
guard startOfNonTrimmedRange < beforeEnd else {
return self[startOfNonTrimmedRange ..< endIndex]
}
var backIdx = beforeEnd
// No need to bound-check because we've already trimmed from the beginning, so we'd definitely break off of this loop before `backIdx` rewinds before `startIndex`
while predicate(self[backIdx]) {
formIndex(before: &backIdx)
}
return self[startOfNonTrimmedRange ... backIdx]
}
// Equal to calling `index(&idx, offsetBy: -other.count)` with just one loop
func _index<S: BidirectionalCollection>(_ index: Index, backwardsOffsetByCountOf other: S) -> Index? {
var idx = index
var otherIdx = other.endIndex
while otherIdx > other.startIndex {
guard idx > startIndex else {
// other.count > self.count: bail
return nil
}
other.formIndex(before: &otherIdx)
formIndex(before: &idx)
}
return idx
}
package func _range<S: BidirectionalCollection>(of other: S, anchored: Bool, backwards: Bool) -> Range<Index>? where S.Element == Element, Element : Equatable {
var result: Range<Index>? = nil
var fromLoc: Index
var toLoc: Index
if backwards {
guard let idx = _index(endIndex, backwardsOffsetByCountOf: other) else {
// other.count > string.count: bail
return nil
}
fromLoc = idx
toLoc = anchored ? fromLoc : startIndex
} else {
fromLoc = startIndex
if anchored {
toLoc = fromLoc
} else {
guard let idx = _index(endIndex, backwardsOffsetByCountOf: other) else {
return nil
}
toLoc = idx
}
}
let delta = fromLoc <= toLoc ? 1 : -1
while true {
var str1Index = fromLoc
var str2Index = other.startIndex
while str2Index < other.endIndex && str1Index < endIndex {
if self[str1Index] != other[str2Index] {
break
}
formIndex(after: &str1Index)
other.formIndex(after: &str2Index)
}
if str2Index == other.endIndex {
result = fromLoc..<str1Index
break
}
if fromLoc == toLoc {
break
}
formIndex(&fromLoc, offsetBy: delta)
}
return result
}
}
|