This package has changed since the documentation was updated to add support for newer language features (async/await). To support older versions of operating systems, see the deprecated branch.
Loadability is an advanced networking and caching library for Swift that allows you to effortlessly implement a custom networking and/or caching stack in your app, with native support for Combine
and SwiftUI
.
Loadability uses Apple's Combine
framework to publish loaded objects and errors, and has built-in SwiftUI
support to create views that load content and display placeholders while loading.
Loadability supports iOS 14+, macOS 11+, and watchOS 7+. It does not have any dependencies.
You can use Loadability as a Swift Package, or add it manually to your project.
Swift Package Manager is a way to add dependencies to your app, and is natively integrated with Xcode.
To add Loadability with SPM, click File
â–º Swift Packages
â–º Add Package Dependency...
, then type in the URL to this Github repo. Xcode will then add the package to your project and perform all the necessary work to build it.
https://github.com/julianschiavo/Loadability
Alternatively, add the package to your Package.swift
file.
let package = Package(
// ...
dependencies: [
.package(url: "https://github.com/julianschiavo/Loadability.git", from: "1.0.0")
],
// ...
)
See SPM documentation to learn more.
If you prefer not to use SPM, you can also add Loadability as a normal framework by building the library from this repository. (See other sources for instructions on doing this.)
Loadability declares basic protocols and classes that you extend in your app to build loaders and caches. It also has an (optional) SwiftUI
integration (see below) for views to load their data and show placeholders while loading.
The library has extensive documentation. If you are looking to gain an advanced understanding of the library, or to implement something not discussed below, please consult the inline documentation before filing an issue.
Note that the code snippets in this section have some code omitted for brevity; see the Examples section for complete code examples.
Loadability networking in centered around the concept of loaders. A Loader
loads data from some source and publishes it as an ObservableObject
, which you can observe manually or integrate through the SwiftUI
support.
To create a loader, you can create a class that conforms to the abstract Loader
protocol, or conform to one of the sub-protocols that implement default behaviour, such as SimpleNetworkLoader
and CachedLoader
.
Loaders have some minimal shared requirements, including published properties for the object and a possible error while loading, and methods that load the data. You can extend loaders to conform to your own custom requirements.
This is an example of a loader. Each loader has an associated type for the object type that is loaded, and the key that is used by the loader to identify the object to load.
class YourLoader: Loader {
@Published var object: YourObject?
@Published var error: IdentifiableError?
func createRequest(for key: YourKey) -> URLRequest {
URLRequest(url: /* some URL */)
}
func createPublisher(key: YourKey) -> AnyPublisher<YourObject, Error>? {
let request = createRequest(for: key)
return URLSession.shared
.dataTaskPublisher(for: request)
.retry(3)
.tryMap { data, response in
try self.decode(data, key: key)
}
.eraseToAnyPublisher()
}
private func decode(_ data: Data, key: YourKey) throws -> YourObject {
/* decode the data, for example, using a JSONDecoder */
}
}
As noted previously, loaders automatically conform to ObservableObject
; you can observe the published properties yourself or use the SwiftUI integration.
Loadability has a few specific built-in loaders that implement common requirements, reducing the code required for some implementations. You can subclass a specific built-in loader instead of the general Loader
protocol if it meets your needs.
SimpleNetworkLoader
implements a basic networking loader, with optional predefined Codable
decoding support. You can use this loader for instances where you want a default network loader.
An example of this loader is shown here; as you can see, it is simpler to implement than a full Loader
subclass, handling the networking code for you using only a URLRequest
you create. You can create a custom decoding implementation to decode the Data
received from the network request, or, if your object type conforms to Codable
, take advantage of the predefined decoding implementation which uses a JSONDecoder
.
class YourLoader: SimpleNetworkLoader {
@Published var object: YourObject?
@Published var error: IdentifiableError?
func createRequest(for key: YourKey) -> URLRequest {
URLRequest(url: /* some URL */)
}
func decode(_ data: Data, key: YourKey) throws -> YourObject {
/* optional custom decoding implementation */
}
}
CachedLoader
uses the caching system discussed below to implement a loader that caches loaded data. You are encouraged to use this loader when you want to cache your data instead of implementing your own caching loader, as it handles tasks such as pre-loading cached data and updating the cache asynchronously after loads.
This is an example of a CachedLoader
subclass (see the Caching section to learn how to create a Cache
).
class YourLoader: CachedLoader {
@Published var object: YourObject?
@Published var error: IdentifiableError?
var cache = YourCache.self /* a SharedCache or SharedSerializableCache type */
func createRequest(for key: YourKey) -> URLRequest {
URLRequest(url: /* some URL */)
}
func createPublisher(key: YourKey) -> AnyPublisher<YourObject, Error>? {
/* see the Loader example above */
}
private func decode(_ data: Data, key: YourKey) throws -> YourObject {
/* see the Loader example above */
}
}
The code is similar to the Loader
snippet, as the caching is handled for you internally by the library; you simply need to provide a Cache
which will be used.
In addition to networking, Loadability has support for caching as a Swift implementation built on top of Apple's NSCache
with a number of additional features including support for serialization (saving caches to disk for reuse).
The basic Cache
class is a basic wrapper for NSCache
, supporting the same features as the Objective-C version. Create an instance of Cache
, and use subscripts to access, modify, or delete values, as such:
let cache = Cache<YourKey, YourObject>()
cache[key] = YourObject() // Add value
cache[key] = YourObject(someParameter: true) // Modify value
let object = cache[key] // YourObject?, Access cached value
cache[key] = nil // Remove value
SerializableCache
subclasses the base Cache
class to add support for saving caches to disk, which allows you to reload an old cache on app restart or similar events. Each SerializableCache
has a name, which must be unique per-app and the same on each initialization to identify the cache on disk. Note that the key and object types of serializable caches must conform to Codable
.
Instead of creating an instance directly, you use the class method load(name:)
, which attempts to load an existing cache from disk, falling back to creating a new cache if an existing one is not valid or does not exist, such as on the first initialization. Future calls of the method (such as on subsequent app restarts) load and decode the previously saved cache from disk.
let sCache = SerializableCache<YourKey, YourObject>.load(name: "Name")
The cache values are accessed, modified, and deleted in the same way as with the normal Cache
. The save()
method encodes and writes the cache to disk; this is called automatically on writes and modifications, so it is not necessary to call this yourself in normal implementations.
Creating new cache instances, especially of SerializableCache
, is costly and energy intensive. Additionally, creating multiple cache instances will result in cached data not being shared around your app, causing unnecessary network requests.
Loadability has support for "shared" caches, which are a singleton-like pattern that allows you to use the same cache throughout your app for each type of cache you have.
You create a shared cache by subclassing either SharedCache
or SharedSerializableCache
, depending on what features you want.
struct YourCache: SharedSerializableCache {
typealias Key = YourKey
typealias Value = YourValue
static let shared = SerializableCache<YourKey, YourObject>.load(name: "Name")
}
Instead of creating multiple cache instances, use the shared
instance every time you need that cache.
As discussed above in the CachedLoader section under Loaders, Loadability has an integration between caches and loaders. Although you can use cache instances directly with custom loaders, it is strongly recommended to use a shared cache and CachedLoader
, as loaders are, by definition, created for each object that is loaded.
Create a shared cache following the code in the previous section on shared caches, and set the cache
variable in your loader to the class's type (e.g. YourCache.self
).
Loadability has native support for SwiftUI; loading data in views is made extremely simple. After creating any type of Loader, make your views loadable by conforming to LoadableView
. The protocol handles loading the data and displays your placeholder content while loading, then passes a non-nil
object to a new body(with:)
method you implement to create your body content.
You only need to implement some basic requirements; creating a loader type, passing the key, and implementing the placeholder and body methods.
The following shows an example of creating a loadable view. Note a few key points:
- You must ensure that your loader variable is annotated by
@StateObject
or the loader's publisher will not update the view as expected - Because of Swift's behaviour with protocol conformance, your views must implement the
body
variable directly, passing in theloaderView
generated byLoadableView
struct YourView: View, LoadableView {
@StateObject var loader = YourLoader() // must be annotated by @StateObject
var body: some View {
loaderView // required
}
func body(with object: YourObject) -> some View {
/* create a body with the object */
}
func placeholder() -> some View {
ProgressView("Loading...")
}
}
Loadability also exposes Load
, which is a view type used internally by LoadableView
. If you require a custom implementation, use this view to gain more control over the loading behaviour. However, try LoadableView
first.
The Loadability Examples repository contains real-world app examples demonstrating how to use the library. The COVID-19
app project shows an implementation of Loadability with support for caching, and uses the cached and network loaders discussed above.
Contributions and pull requests are welcomed by anyone! If you find an issue with Loadability, file a Github Issue, or, if you know how to fix it, submit a pull request.
Please review our Code of Conduct and Contribution Guidelines before making a contribution.
Loadability was originally created by Julian Schiavo in his spare time, and made available under the MIT License. If you find the library useful, please consider sponsoring me on Github, which contributes to development and learning resources, and allows me to keep making cool stuff like this!
Loadability is inspired by (and uses code by permission) from John Sundell's in depth blog posts on Handling loading states within SwiftUI views and Caching in Swift, so thank you to John for the inspiration and examples. If you want to gain a better understanding of loading or caching in Swift, I strongly recommend you check out the articles linked above, in addition to his many other articles!
Available under the MIT License. See the License for more info.