Skip to content

Commit 489289c

Browse files
author
Chris
authored
Merge pull request #10 from crelies/dev
Pagination is now implemented as a modifier 🚀
2 parents 147a211 + 8c768a0 commit 489289c

10 files changed

+310
-219
lines changed

README.md

+132-40
Original file line numberDiff line numberDiff line change
@@ -37,52 +37,39 @@ AdvancedList(yourData, content: { item in
3737

3838
### 📄 Pagination
3939

40-
The `Pagination` is implemented as a class (conforming to `ObservableObject`) so the `AdvancedList` can observe it.
41-
It has three different states: `error`, `idle` and `loading`. If the `state` of the `Pagination` changes the `AdvancedList` updates itself to show or hide the state related view (`ErrorView` for state `.error(Error)` or `LoadingView` for state `.loading`, `.idle` will display nothing). Update the `state` if you start loading (`.loading`), stop loading ( `.idle`) or if an error occurred (`.error(Error)`) so the `AdvancedList` can render the appropriate view.
40+
The `Pagination` functionality is now (>= `5.0.0`) implemented as a `modifier`.
41+
It has three different states: `error`, `idle` and `loading`. If the `state` of the `Pagination` changes the `AdvancedList` displays the view created by the view builder of the specified pagination object (`AdvancedListPagination`). Keep track of the current pagination state by creating a local state variable (`@State`) of type `AdvancedListPaginationState`. Use this state variable in the `content` `ViewBuilder` of your pagination configuration object to determine which view should be displayed in the list (see the example below).
4242

43-
If you want to use pagination you can choose between the `lastItemPagination` and the `thresholdItemPagination`. Both concepts are described [here](https://github.com/crelies/ListPagination). Just pass `.lastItemPagination` or `.thresholdItemPagination` including the required parameters to the `AdvancedList` initializer.
43+
If you want to use pagination you can choose between the `lastItemPagination` and the `thresholdItemPagination`. Both concepts are described [here](https://github.com/crelies/ListPagination). Just specify the type of the pagination when adding the `.pagination` modifier to your `AdvancedList`.
4444

45-
Both pagination types require
46-
47-
- an **ErrorView** and a **LoadingView** (**ViewBuilder**)
48-
- a block (**shouldLoadNextPage**) which is called if the `last or threshold item appeared` and
49-
- the initial state (**AdvancedListPaginationState**) of the pagination which determines the visibility of the pagination state related view.
50-
51-
The `thresholdItemPagination` expects an offset parameter (number of items before the last item) to determine the threshold item.
52-
53-
**The ErrorView or LoadingView will only be visible below the List if the last item of the List appeared! That way the user is only interrupted if needed.**
54-
55-
**Skip pagination setup by using `.noPagination`.**
45+
**The view created by the `content` `ViewBuilder` of your pagination configuration object will only be visible below the List if the last item of the List appeared! That way the user is only interrupted if needed.**
5646

5747
**Example:**
5848

5949
```swift
60-
private(set) lazy var pagination: AdvancedListPagination<AnyView, AnyView> = {
61-
.thresholdItemPagination(errorView: { error in
62-
AnyView(
63-
VStack {
64-
Text(error.localizedDescription)
65-
.lineLimit(nil)
66-
.multilineTextAlignment(.center)
67-
68-
Button(action: {
69-
// load current page again
70-
}) {
71-
Text("Retry")
72-
}.padding()
73-
}
74-
)
75-
}, loadingView: {
76-
AnyView(
77-
VStack {
78-
Divider()
79-
Text("Loading...")
80-
}
81-
)
82-
}, offset: 25, shouldLoadNextPage: {
83-
// load next page
84-
}, state: .idle)
85-
}()
50+
@State private var paginationState: AdvancedListPaginationState = .idle
51+
52+
AdvancedList(...)
53+
.pagination(.init(type: .lastItem, shouldLoadNextPage: {
54+
paginationState = .loading
55+
DispatchQueue.main.asyncAfter(deadline: .now() + 2) {
56+
items.append(contentsOf: moreItems)
57+
paginationState = .idle
58+
}
59+
}) {
60+
switch paginationState {
61+
case .idle:
62+
EmptyView()
63+
case .loading:
64+
if #available(iOS 14.0, *) {
65+
ProgressView()
66+
} else {
67+
Text("Loading ...")
68+
}
69+
case let .error(error):
70+
Text(error.localizedDescription)
71+
}
72+
})
8673
```
8774

8875
### 📁 Move and 🗑️ delete items
@@ -297,3 +284,108 @@ AdvancedList(yourData, content: { item in
297284
}
298285
```
299286
</details>
287+
288+
<details>
289+
<summary>Migration 4.0 -> 5.0</summary>
290+
291+
`Pagination` is now implemented as a `modifier` 💪 And last but not least the code documentation arrived 😀
292+
293+
**Before:**
294+
295+
```swift
296+
private lazy var pagination: AdvancedListPagination<AnyView, AnyView> = {
297+
.thresholdItemPagination(errorView: { error in
298+
AnyView(
299+
VStack {
300+
Text(error.localizedDescription)
301+
.lineLimit(nil)
302+
.multilineTextAlignment(.center)
303+
304+
Button(action: {
305+
// load current page again
306+
}) {
307+
Text("Retry")
308+
}.padding()
309+
}
310+
)
311+
}, loadingView: {
312+
AnyView(
313+
VStack {
314+
Divider()
315+
Text("Loading...")
316+
}
317+
)
318+
}, offset: 25, shouldLoadNextPage: {
319+
// load next page
320+
}, state: .idle)
321+
}()
322+
323+
@State private var listState: ListState = .items
324+
325+
AdvancedList(yourData, content: { item in
326+
Text("Item")
327+
}, listState: $listState, emptyStateView: {
328+
Text("No data")
329+
}, errorStateView: { error in
330+
VStack {
331+
Text(error.localizedDescription)
332+
.lineLimit(nil)
333+
334+
Button(action: {
335+
// do something
336+
}) {
337+
Text("Retry")
338+
}
339+
}
340+
}, loadingStateView: {
341+
Text("Loading ...")
342+
}, pagination: pagination)
343+
344+
```
345+
346+
**After:**
347+
348+
```swift
349+
@State private var listState: ListState = .items
350+
@State private var paginationState: AdvancedListPaginationState = .idle
351+
352+
AdvancedList(yourData, content: { item in
353+
Text("Item")
354+
}, listState: $listState, emptyStateView: {
355+
Text("No data")
356+
}, errorStateView: { error in
357+
VStack {
358+
Text(error.localizedDescription)
359+
.lineLimit(nil)
360+
361+
Button(action: {
362+
// do something
363+
}) {
364+
Text("Retry")
365+
}
366+
}
367+
}, loadingStateView: {
368+
Text("Loading ...")
369+
})
370+
.pagination(.init(type: .lastItem, shouldLoadNextPage: {
371+
paginationState = .loading
372+
DispatchQueue.main.asyncAfter(deadline: .now() + 2) {
373+
items.append(contentsOf: moreItems)
374+
paginationState = .idle
375+
}
376+
}) {
377+
switch paginationState {
378+
case .idle:
379+
EmptyView()
380+
case .loading:
381+
if #available(iOS 14.0, *) {
382+
ProgressView()
383+
} else {
384+
Text("Loading ...")
385+
}
386+
case let .error(error):
387+
Text(error.localizedDescription)
388+
}
389+
})
390+
```
391+
</details>

Sources/AdvancedList/private/Models/AdvancedListPaginationType.swift

-14
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
//
2+
// AnyAdvancedListPagination.swift
3+
//
4+
//
5+
// Created by Christian Elies on 31.07.20.
6+
//
7+
8+
import SwiftUI
9+
10+
struct AnyAdvancedListPagination {
11+
let type: AdvancedListPaginationType
12+
let shouldLoadNextPage: () -> Void
13+
let content: () -> AnyView
14+
15+
init<Content: View>(_ pagination: AdvancedListPagination<Content>) {
16+
type = pagination.type
17+
shouldLoadNextPage = pagination.shouldLoadNextPage
18+
content = { AnyView(pagination.content()) }
19+
}
20+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
//
2+
// ListState+error.swift
3+
//
4+
//
5+
// Created by Christian Elies on 02.08.20.
6+
//
7+
8+
extension ListState {
9+
var error: Swift.Error? {
10+
guard case let ListState.error(error) = self else {
11+
return nil
12+
}
13+
return error
14+
}
15+
}

Sources/AdvancedList/public/Models/AdvancedListPagination.swift

+16-45
Original file line numberDiff line numberDiff line change
@@ -5,56 +5,27 @@
55
// Created by Christian Elies on 15.08.19.
66
//
77

8-
import Combine
98
import SwiftUI
109

11-
public final class AdvancedListPagination<ErrorView: View, LoadingView: View>: NSObject, ObservableObject {
12-
let errorView: (Error) -> ErrorView
13-
let loadingView: () -> LoadingView
10+
/// Represents the pagination configuration.
11+
public struct AdvancedListPagination<Content: View> {
1412
let type: AdvancedListPaginationType
1513
let shouldLoadNextPage: () -> Void
16-
17-
public let objectWillChange = PassthroughSubject<Void, Never>()
18-
19-
public var state: AdvancedListPaginationState {
20-
didSet {
21-
objectWillChange.send()
22-
}
23-
}
24-
25-
init(@ViewBuilder errorView: @escaping (Error) -> ErrorView, @ViewBuilder loadingView: @escaping () -> LoadingView, type: AdvancedListPaginationType, shouldLoadNextPage: @escaping () -> Void, state: AdvancedListPaginationState) {
26-
self.errorView = errorView
27-
self.loadingView = loadingView
14+
let content: () -> Content
15+
16+
/// Initializes the pagination configuration with the given values.
17+
///
18+
/// - Parameters:
19+
/// - type: The type of the pagination, choose between `lastItem` or `thresholdItem`.
20+
/// - shouldLoadNextPage: A closure that is called everytime the end of a page is reached.
21+
/// - content: A closure providing a `View` which should be displayed at the end of a page.
22+
public init(
23+
type: AdvancedListPaginationType,
24+
shouldLoadNextPage: @escaping () -> Void,
25+
@ViewBuilder content: @escaping () -> Content
26+
) {
2827
self.type = type
2928
self.shouldLoadNextPage = shouldLoadNextPage
30-
self.state = state
31-
}
32-
}
33-
34-
extension AdvancedListPagination {
35-
public static func lastItemPagination(@ViewBuilder errorView: @escaping (Error) -> ErrorView, @ViewBuilder loadingView: @escaping () -> LoadingView, shouldLoadNextPage: @escaping () -> Void, state: AdvancedListPaginationState) -> AdvancedListPagination {
36-
AdvancedListPagination(errorView: errorView,
37-
loadingView: loadingView,
38-
type: .lastItem,
39-
shouldLoadNextPage: shouldLoadNextPage,
40-
state: state)
41-
}
42-
43-
public static func thresholdItemPagination(@ViewBuilder errorView: @escaping (Error) -> ErrorView, @ViewBuilder loadingView: @escaping () -> LoadingView, offset: Int, shouldLoadNextPage: @escaping () -> Void, state: AdvancedListPaginationState) -> AdvancedListPagination {
44-
AdvancedListPagination(errorView: errorView,
45-
loadingView: loadingView,
46-
type: .thresholdItem(offset: offset),
47-
shouldLoadNextPage: shouldLoadNextPage,
48-
state: state)
49-
}
50-
}
51-
52-
extension AdvancedListPagination where ErrorView == EmptyView, ErrorView == LoadingView {
53-
public static var noPagination: AdvancedListPagination {
54-
AdvancedListPagination(errorView: { _ in ErrorView() },
55-
loadingView: { LoadingView() },
56-
type: .noPagination,
57-
shouldLoadNextPage: {},
58-
state: .idle)
29+
self.content = content
5930
}
6031
}

Sources/AdvancedList/public/Models/AdvancedListPaginationState.swift

+6-18
Original file line numberDiff line numberDiff line change
@@ -7,24 +7,12 @@
77

88
import Foundation
99

10-
public enum AdvancedListPaginationState {
11-
case error(_ error: Error)
10+
/// Represents the different states of a pagination.
11+
public enum AdvancedListPaginationState: Equatable {
12+
/// The error state; use this state if an error occurs while loading a page.
13+
case error(_ error: NSError)
14+
/// The idle state; use this state if no page loading is in progress.
1215
case idle
16+
/// The loading state; use this state if a page is loaded.
1317
case loading
1418
}
15-
16-
extension AdvancedListPaginationState: Equatable {
17-
public static func ==(lhs: AdvancedListPaginationState,
18-
rhs: AdvancedListPaginationState) -> Bool {
19-
switch (lhs, rhs) {
20-
case (.error(let lhsError), .error(let rhsError)):
21-
return (lhsError as NSError) == (rhsError as NSError)
22-
case (.idle, .idle):
23-
return true
24-
case (.loading, .loading):
25-
return true
26-
default:
27-
return false
28-
}
29-
}
30-
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
//
2+
// AdvancedListPaginationType.swift
3+
//
4+
//
5+
// Created by Christian Elies on 16.08.19.
6+
//
7+
8+
/// Specifies the different pagination types.
9+
public enum AdvancedListPaginationType {
10+
/// Notifies the pagination configuration object when the last item in the list was reached.
11+
case lastItem
12+
/// Notifies the pagination configuration object when the given offset was passed.
13+
case thresholdItem(offset: Int)
14+
}

Sources/AdvancedList/public/Models/ListState.swift

+6-26
Original file line numberDiff line numberDiff line change
@@ -8,32 +8,12 @@
88

99
import Foundation
1010

11-
public enum ListState {
12-
case error(_ error: Error)
11+
/// Specifies the different states of an `AdvancedList`.
12+
public enum ListState: Equatable {
13+
/// The `error` state; displays the error view instead of the list to the user.
14+
case error(_ error: NSError)
15+
/// The `items` state (`default`); displays the items or the empty state view (if there are no items) to the user.
1316
case items
17+
/// The `loading` state; displays the loading state view instead of the list to the user.
1418
case loading
1519
}
16-
17-
extension ListState {
18-
var error: Error? {
19-
guard case let ListState.error(error) = self else {
20-
return nil
21-
}
22-
return error
23-
}
24-
}
25-
26-
extension ListState: Equatable {
27-
public static func ==(lhs: ListState, rhs: ListState) -> Bool {
28-
switch (lhs, rhs) {
29-
case (.error(let lhsError), .error(let rhsError)):
30-
return (lhsError as NSError) == (rhsError as NSError)
31-
case (.items, .items):
32-
return true
33-
case (.loading, .loading):
34-
return true
35-
default:
36-
return false
37-
}
38-
}
39-
}

0 commit comments

Comments
 (0)