diff --git a/README.md b/README.md index 680e06c..0f75c08 100644 --- a/README.md +++ b/README.md @@ -36,12 +36,13 @@ For more examples, see `SnapToScrollDemo/ContentView.swift`. ## Configuration -`HStackSnap` comes with two customizable properties: +`HStackSnap` comes with some customizable properties: - `alignment`: The way you'd like your elements to be arranged. - `leading(CGFloat)`: Aligns your child views to the leading edge of `HStackSnap`. This configuration supports elements of various sizes, so long as they don't take up all available horizontal space (which would extend beyond the screen). Use the value to set the size of the left offset. - `center(CGFloat)`: Automatically aligns your child view to the center of the screen, using the offset value you've provided. This is accomplished with inside of the `.snapAlignmentHelper` which sets the frame width based on the available space. Note that setting your own width elsewhere may produce unexpected layouts. - `coordinateSpace`: Option to set custom name for the coordinate space, in the case you're using multiple `HStackSnap`s of various sizes. If you use this, set the same value in `.snapAlignmentHelper`. +- `selectedIndex`: (Optional) A @State variable binding e.g. $selectedIndex, which can be used programatically slide between Views. `.snapAlignmentHelper` comes with two options as well: diff --git a/Sources/HStackSnap.swift b/Sources/HStackSnap.swift index a2b5a14..991fd03 100644 --- a/Sources/HStackSnap.swift +++ b/Sources/HStackSnap.swift @@ -5,14 +5,18 @@ public struct HStackSnap: View { // MARK: Lifecycle + @Binding var selectedIndex: Int? + public init( alignment: SnapAlignment, + selectedIndex: Binding? = nil, spacing: CGFloat? = nil, coordinateSpace: String = "SnapToScroll", @ViewBuilder content: @escaping () -> Content, eventHandler: SnapToScrollEventHandler? = .none) { self.content = content + self._selectedIndex = selectedIndex != nil ? Binding(selectedIndex!) : .constant(nil) self.alignment = alignment self.leadingOffset = alignment.scrollOffset self.spacing = spacing @@ -34,6 +38,7 @@ public struct HStackSnap: View { HStackSnapCore( leadingOffset: leadingOffset, + selectedIndex: $selectedIndex, spacing: spacing, coordinateSpace: coordinateSpace, content: content, diff --git a/Sources/Views/HStackSnapCore.swift b/Sources/Views/HStackSnapCore.swift index 0d7876e..1688f0f 100644 --- a/Sources/Views/HStackSnapCore.swift +++ b/Sources/Views/HStackSnapCore.swift @@ -8,14 +8,18 @@ public typealias SnapToScrollEventHandler = ((SnapToScrollEvent) -> Void) public struct HStackSnapCore: View { // MARK: Lifecycle + @Binding var selectedIndex: Int? + public init( leadingOffset: CGFloat, + selectedIndex: Binding? = nil, spacing: CGFloat? = nil, coordinateSpace: String = "SnapToScroll", @ViewBuilder content: @escaping () -> Content, eventHandler: SnapToScrollEventHandler? = .none) { self.content = content + self._selectedIndex = selectedIndex ?? .constant(nil) self.targetOffset = leadingOffset self.spacing = spacing self.scrollOffset = leadingOffset @@ -90,6 +94,14 @@ public struct HStackSnapCore: View { eventHandler?(.didLayout(layoutInfo: itemScrollPositions)) } }) + .onChange(of: selectedIndex) { newValue in + // Update state + withAnimation(.easeOut(duration: 0.2)) { + scrollOffset = snapLocations[selectedIndex] ?? 0 + } + prevScrollOffset = scrollOffset + eventHandler?(.didLayout(layoutInfo: itemScrollPositions)) + } .contentShape(Rectangle()) .gesture(snapDrag) }