diff --git a/.swiftpm/xcode/xcshareddata/xcschemes/WrappingStack.xcscheme b/.swiftpm/xcode/xcshareddata/xcschemes/WrappingStack.xcscheme
new file mode 100644
index 0000000..b693472
--- /dev/null
+++ b/.swiftpm/xcode/xcshareddata/xcschemes/WrappingStack.xcscheme
@@ -0,0 +1,91 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/Docs/Resources/wrapping-hstack-macos.png b/Docs/Resources/wrapping-hstack-macos.png
new file mode 100644
index 0000000..85eadcd
Binary files /dev/null and b/Docs/Resources/wrapping-hstack-macos.png differ
diff --git a/Readme.md b/Readme.md
index 2958531..102b354 100644
--- a/Readme.md
+++ b/Readme.md
@@ -1,7 +1,67 @@
-# Swiftui WrappingStack
+# SwiftUI WrappingStack
-data:image/s3,"s3://crabby-images/89427/8942777fa5386e0591a3a24e3076ba6140c96501" alt="Swift 5.3" data:image/s3,"s3://crabby-images/a4c74/a4c747fdcc2d11117148aa82535e5f8a9df95869" alt="Xcode 12.5" data:image/s3,"s3://crabby-images/43d46/43d461faacc18f0ffc14f5026d8d1184e17fcab0" alt="iOS 9.0" data:image/s3,"s3://crabby-images/98ac8/98ac8eb2c5e0092e99584058067d11c5e0d6b18b" alt="iPadOS 9.0" data:image/s3,"s3://crabby-images/2f714/2f714dd6f3e09efcffea84a7bc39dabb7ff42260" alt="MacOS 10.10" [data:image/s3,"s3://crabby-images/431ad/431adc63c0f4053741ede0dd454773f9966d508a" alt="Build & Test"](https://github.com/diniska/swiftui-wrapping-stack/actions/workflows/test.yml)
+data:image/s3,"s3://crabby-images/89427/8942777fa5386e0591a3a24e3076ba6140c96501" alt="Swift 5.3" data:image/s3,"s3://crabby-images/a4de6/a4de6b73ca992eb6eaef6c410da78014a2dd7930" alt="Xcode 12.5" data:image/s3,"s3://crabby-images/1cd64/1cd6447e037fdccb70a08ea502052cc0552568aa" alt="iOS 9.0" data:image/s3,"s3://crabby-images/0a2a6/0a2a6a413ce1e5160fb1c9ed96aea1254ce64423" alt="iPadOS 9.0" data:image/s3,"s3://crabby-images/2f714/2f714dd6f3e09efcffea84a7bc39dabb7ff42260" alt="MacOS 10.10" [data:image/s3,"s3://crabby-images/431ad/431adc63c0f4053741ede0dd454773f9966d508a" alt="Build & Test"](https://github.com/diniska/swiftui-wrapping-stack/actions/workflows/test.yml)
A SwiftUI Views for wrapping HStack elements into multiple lines.
-`WrappingHStack` - provides `HStack` that supports line wrapping
+## List of supported views
+
+* `WrappingHStack` - provides `HStack` that supports line wrapping
+
+## How to use
+### Step 1
+Add a dependency using Swift Package Manager to your project: [https://github.com/diniska/swiftui-wrapping-stack](https://github.com/diniska/swiftui-wrapping-stack)
+
+### Step 2
+Import the dependency
+
+```swift
+import WrappingStack
+```
+
+### Step 3
+Replace `HStack` with `WrappingHStack` in your view structure. It is compatible with `ForEach`.
+
+```swift
+struct MyView: View {
+
+ let elements = ["Cat 🐱", "Dog 🐶", "Sun 🌞", "Moon 🌕", "Tree 🌳"]
+
+ var body: some View {
+ WrappingHStack(id: \.self) { // use the same id is in the `ForEach` below
+ ForEach(elements, id: \.self) { element in
+ Text(element)
+ .padding()
+ .background(Color.gray)
+ .cornerRadius(6)
+ }
+ }
+ .frame(width: 300) // limiting the width for demo purpose. This line is not needed in real code
+ }
+
+}
+```
+
+The result of the code above:
+
+data:image/s3,"s3://crabby-images/5f106/5f106a220e86b5ff0c55dafc42f1638be687b1cc" alt="WrappingHStack for macOS"
+
+
+## Customization
+
+Customize appearance using the next parameters. All the default SwiftUI modifiers can be applied as well.
+
+### `WrappingHStack` parameters
+
+Parameter name | Description
+---------------|--------------
+`alignment` | horizontal and vertical alignment. `.center` is used by default. Vertical alignment is applied to every row
+`horizontalSpacing` | horizontal spacing between elements
+`verticalSpacing` | vertical spacing between the lines
+
+## Performance considerations
+
+The code written in a way to cache the elements representing views sizes, it doesn't re-calculate the size for different views with the same id.
+
+* huge numbers of elements are not recommended, although the same applies to `HStack` where `LazyHStack` is a better alternative for the long rows. If you have a large number of elements - double-check the memory and performance on a real device
+* it is pretty good in terms of CPU consumption as every element calculates its size only once.
diff --git a/Sources/WrappingStack/WrappingHStack.swift b/Sources/WrappingStack/WrappingHStack.swift
index 6c5416b..a3690a5 100644
--- a/Sources/WrappingStack/WrappingHStack.swift
+++ b/Sources/WrappingStack/WrappingHStack.swift
@@ -32,13 +32,21 @@ public struct WrappingHStack,
alignment: Alignment = .center,
horizontalSpacing: CGFloat = 0,
verticalSpacing: CGFloat = 0,
@ViewBuilder content create: () -> ForEach
- ){
+ ) {
let forEach = create()
data = forEach.data
content = forEach.content
@@ -111,8 +119,24 @@ public struct WrappingHStack ForEach) {
- self.init(id: \.id, content: create)
+ /// Creates a new WrappingHStack
+ ///
+ /// - Parameters:
+ /// - alignment: horizontal and vertical alignment. Vertical alignment is applied to every row
+ /// - horizontalSpacing: horizontal spacing between elements
+ /// - verticalSpacing: vertical spacing between the lines
+ /// - create: a method that creates an array of elements
+ public init(
+ alignment: Alignment = .center,
+ horizontalSpacing: CGFloat = 0,
+ verticalSpacing: CGFloat = 0,
+ @ViewBuilder content create: () -> ForEach
+ ) {
+ self.init(id: \.id,
+ alignment: alignment,
+ horizontalSpacing: horizontalSpacing,
+ verticalSpacing: verticalSpacing,
+ content: create)
}
}
@@ -121,15 +145,21 @@ extension WrappingHStack where ID == Data.Element.ID, Data.Element: Identifiable
@available(iOS 14, macOS 11, *)
struct WrappingHStack_Previews: PreviewProvider {
static var previews: some View {
- WrappingHStack(id: \.self, alignment: .topLeading) {
- ForEach(["Hello1", "world1", "Hello2", "world2", "Hello3", "world3", "Hello4", "world4 ", "Hello1"], id: \.self) { item in
- Text(item)
+ WrappingHStack(
+ id: \.self,
+ horizontalSpacing: 8,
+ verticalSpacing: 8
+ ) {
+ ForEach(["Cat 🐱", "Dog 🐶", "Sun 🌞", "Moon 🌕", "Tree 🌳"], id: \.self) { element in
+ Text(element)
.padding()
- .background(Color(.systemGray))
- .cornerRadius(3)
+ .background(Color.gray.opacity(0.1))
+ .cornerRadius(6)
}
}
+ .padding()
.frame(width: 300)
+ .background(Color.white)
}
}