From c29b8a375eefd6ce527a1c68912dc106912818b0 Mon Sep 17 00:00:00 2001 From: Nikita Vasilev Date: Mon, 26 Aug 2024 09:49:50 +0400 Subject: [PATCH 1/2] Integrate header and footer views into the list view (#3) * Integrate header and footer views into the list view * Update `CHANGELOG.md` --- CHANGELOG.md | 11 +++- .../ViewModifiers/LoadingViewModifier.swift | 2 +- .../Views/PaginatorListView.swift | 61 ++++++++++++++++--- .../Presentation/Views/PaginatorView.swift | 33 ++++++++++ 4 files changed, 97 insertions(+), 10 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 63e5ce9..d369d25 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,11 +2,18 @@ All notable changes to this project will be documented in this file. #### 1.x Releases -- `1.0.x` Releases - [1.0.0](#100) +- `1.0.x` Releases - [1.0.0](#100) | [1.1.0](#110) + +## [1.1.0](https://github.com/space-code/blade/releases/tag/1.1.0) +Released on 2024-08-26. + +#### Added +- Integrate header and footer views into the list view. + - Added in Pull Request [#3](https://github.com/space-code/blade/pull/3). ## [1.0.0](https://github.com/space-code/blade/releases/tag/1.0.0) Released on 2024-02-05. #### Added - Initial release of Blade. - - Added by [Nikita Vasilev](https://github.com/nik3212). \ No newline at end of file + - Added by [Nikita Vasilev](https://github.com/nik3212). diff --git a/Sources/BladeTCA/Classes/Presentation/ViewModifiers/LoadingViewModifier.swift b/Sources/BladeTCA/Classes/Presentation/ViewModifiers/LoadingViewModifier.swift index ae29da8..c428494 100644 --- a/Sources/BladeTCA/Classes/Presentation/ViewModifiers/LoadingViewModifier.swift +++ b/Sources/BladeTCA/Classes/Presentation/ViewModifiers/LoadingViewModifier.swift @@ -13,7 +13,7 @@ struct LoadingViewModifier: ViewModifier { // MARK: ViewModifier func body(content: Content) -> some View { - VStack { + Group { content if isLoading { diff --git a/Sources/BladeTCA/Classes/Presentation/Views/PaginatorListView.swift b/Sources/BladeTCA/Classes/Presentation/Views/PaginatorListView.swift index 9ff9cca..63ac507 100644 --- a/Sources/BladeTCA/Classes/Presentation/Views/PaginatorListView.swift +++ b/Sources/BladeTCA/Classes/Presentation/Views/PaginatorListView.swift @@ -7,10 +7,14 @@ import BladeTCA import ComposableArchitecture import SwiftUI +// MARK: - PaginatorListView + public struct PaginatorListView< State: Equatable & Identifiable, Action: Equatable, + Header: View, Body: View, + Footer: View, PositionType: Equatable, Request: Equatable >: View { @@ -18,30 +22,73 @@ public struct PaginatorListView< private typealias StoreType = ViewStoreOf> + @SwiftUI.State private var isLoading = false + // MARK: Properties public let store: Store, PaginatorAction> + public let header: () -> Header public let content: (State) -> Body + public let footer: () -> Footer public init( store: Store, PaginatorAction>, - content: @escaping (State) -> Body + @ViewBuilder header: @escaping () -> Header, + @ViewBuilder content: @escaping (State) -> Body, + @ViewBuilder footer: @escaping () -> Footer ) { self.store = store + self.header = header self.content = content + self.footer = footer } // MARK: View public var body: some View { - WithViewStore(store, observe: { $0 }) { (viewStore: StoreType) in - List(viewStore.items) { item in - content(item) - .onAppear { - viewStore.send(.itemAppeared(item.id)) + List { + Section { + header() + } + + WithViewStore(store, observe: { $0 }) { (viewStore: StoreType) in + Section(content: { + ForEach(viewStore.items) { item in + content(item) + .onAppear { + viewStore.send(.itemAppeared(item.id)) + } } + }) + } + + Section { + footer() + } + + Section { + EmptyView() + .modifier(LoadingViewModifier(isLoading: isLoading)) } - .modifier(LoadingViewModifier(isLoading: viewStore.isLoading && !viewStore.items.isEmpty)) } } } + +public extension PaginatorListView where Header == EmptyView, Footer == EmptyView { + init( + store: Store, PaginatorAction>, + @ViewBuilder content: @escaping (State) -> Body + ) { + self.init(store: store, header: { EmptyView() }, content: content, footer: { EmptyView() }) + } +} + +public extension PaginatorListView where Header: View, Footer == EmptyView { + init( + store: Store, PaginatorAction>, + header: @escaping () -> Header, + @ViewBuilder content: @escaping (State) -> Body + ) { + self.init(store: store, header: header, content: content, footer: { EmptyView() }) + } +} diff --git a/Sources/BladeTCA/Classes/Presentation/Views/PaginatorView.swift b/Sources/BladeTCA/Classes/Presentation/Views/PaginatorView.swift index fadfde8..904f12d 100644 --- a/Sources/BladeTCA/Classes/Presentation/Views/PaginatorView.swift +++ b/Sources/BladeTCA/Classes/Presentation/Views/PaginatorView.swift @@ -14,7 +14,9 @@ public struct PaginatorView< Action: Equatable, PositionType: Equatable, Request: Equatable, + Header: View, Body: View, + Footer: View, RowContent: View >: View { // MARK: Types @@ -24,18 +26,24 @@ public struct PaginatorView< // MARK: Properties public let store: Store, PaginatorAction> + public let header: () -> Header public let content: ([State], @escaping (State) -> AnyView) -> Body + public let footer: () -> Footer public let rowContent: (State) -> RowContent // MARK: Initialization public init( store: Store, PaginatorAction>, + @ViewBuilder header: @escaping () -> Header, @ViewBuilder content: @escaping ([State], @escaping (State) -> AnyView) -> Body, + @ViewBuilder footer: @escaping () -> Footer, @ViewBuilder rowContent: @escaping (State) -> RowContent ) { self.store = store + self.header = header self.content = content + self.footer = footer self.rowContent = rowContent } @@ -43,16 +51,41 @@ public struct PaginatorView< public var body: some View { WithViewStore(store, observe: { $0 }) { (viewStore: StoreType) in + header() + content(viewStore.items.elements) { item in rowContent(item) .onAppear { viewStore.send(.itemAppeared(item.id)) } .any } .modifier(LoadingViewModifier(isLoading: viewStore.isLoading && !viewStore.items.isEmpty)) + + footer() } } } +public extension PaginatorView where Header == EmptyView, Footer == EmptyView { + init( + store: Store, PaginatorAction>, + @ViewBuilder content: @escaping ([State], @escaping (State) -> AnyView) -> Body, + @ViewBuilder rowContent: @escaping (State) -> RowContent + ) { + self.init(store: store, header: { EmptyView() }, content: content, footer: { EmptyView() }, rowContent: rowContent) + } +} + +public extension PaginatorView where Header: View, Footer == EmptyView { + init( + store: Store, PaginatorAction>, + header: @escaping () -> Header, + @ViewBuilder content: @escaping ([State], @escaping (State) -> AnyView) -> Body, + @ViewBuilder rowContent: @escaping (State) -> RowContent + ) { + self.init(store: store, header: header, content: content, footer: { EmptyView() }, rowContent: rowContent) + } +} + // MARK: Extension private extension View { From f8ff627ab3044d982f98e852d12afa1735dfe3fd Mon Sep 17 00:00:00 2001 From: Nikita Vasilev Date: Mon, 26 Aug 2024 11:05:26 +0400 Subject: [PATCH 2/2] Bump codecov version from `3.1.0` to `4.5.0` (#5) --- .github/workflows/ci.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index b09f602..8b526c2 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -54,7 +54,7 @@ jobs: - name: ${{ matrix.name }} run: xcodebuild test -scheme "Blade-Package" -destination "platform=macOS" clean -enableCodeCoverage YES ${{ matrix.params }} -resultBundlePath "test_output/${{ matrix.name }}.xcresult" || exit 1 - name: Upload coverage reports to Codecov - uses: codecov/codecov-action@v3.1.0 + uses: codecov/codecov-action@v4.5.0 with: token: ${{ secrets.CODECOV_TOKEN }} xcode: true @@ -116,7 +116,7 @@ jobs: - name: ${{ matrix.name }} run: xcodebuild test -scheme "Blade-Package" -destination "${{ matrix.destination }}" clean -enableCodeCoverage YES ${{ matrix.params }} -resultBundlePath "test_output/${{ matrix.name }}.xcresult" || exit 1 - name: Upload coverage reports to Codecov - uses: codecov/codecov-action@v3.1.0 + uses: codecov/codecov-action@v4.5.0 with: token: ${{ secrets.CODECOV_TOKEN }} xcode: true @@ -155,7 +155,7 @@ jobs: - name: ${{ matrix.name }} run: xcodebuild test -scheme "Blade-Package" -destination "${{ matrix.destination }}" clean -enableCodeCoverage YES ${{ matrix.params }} -resultBundlePath "test_output/${{ matrix.name }}.xcresult" || exit 1 - name: Upload coverage reports to Codecov - uses: codecov/codecov-action@v3.1.0 + uses: codecov/codecov-action@v4.5.0 with: token: ${{ secrets.CODECOV_TOKEN }} xcode: true