In SwiftUI, Combine provides a variety of tools to handle asynchronous and reactive data, and CurrentValueSubject
is one of the most practical publishers for managing state. Unlike PassthroughSubject
, which only broadcasts values when you explicitly send them, CurrentValueSubject
maintains a current value and emits that value to new subscribers immediately. This makes it ideal for cases where you need to hold and update a piece of data that multiple parts of your app might observe.
What is CurrentValueSubject
?
CurrentValueSubject
is a publisher that holds a current value and emits it to any new subscribers. You can update this value at any time, and all subscribers will automatically receive the updated value. It’s commonly used to create state-like behavior in reactive programming, which fits perfectly with SwiftUI’s data-binding approach.
In this article, we’ll create a SwiftUI view that demonstrates how to use CurrentValueSubject
to manage and display a count that users can increment by pressing a button.
Example SwiftUI View with CurrentValueSubject
Below is a SwiftUI view that uses CurrentValueSubject
to hold and update a counter value, which is displayed in the UI.
import SwiftUI
import Combine
struct CurrentValueSubjectView: View {
// State variable to display the current count in the view
@State private var countText: String = "Current Count: 0"
// CurrentValueSubject to hold and update the count
private var countSubject = CurrentValueSubject<Int, Never>(0)
// AnyCancellable to store the subscription
@State private var cancellable: AnyCancellable?
var body: some View {
VStack(spacing: 20) {
Text("CurrentValueSubject Example")
.font(.headline)
.padding()
Text(countText) // Display the current count
.font(.title)
.padding()
Button("Increment Count") {
// Increment the count value in CurrentValueSubject
countSubject.value += 1
}
.padding()
.background(Color.green)
.foregroundColor(.white)
.cornerRadius(8)
}
.onAppear {
// Subscribe to the CurrentValueSubject and update the countText on change
cancellable = countSubject
.sink { newCount in
countText = "Current Count: \(newCount)"
}
}
.onDisappear {
// Cancel the subscription when the view disappears
cancellable?.cancel()
}
}
}
struct CurrentValueSubjectView_Previews: PreviewProvider {
static var previews: some View {
CurrentValueSubjectView()
}
}
Explanation of the Code
- State Variable:
@State private var countText
: Holds the string displayed in the view that shows the current count. This variable is updated whenever theCurrentValueSubject
emits a new value.
- CurrentValueSubject:
private var countSubject = CurrentValueSubject<Int, Never>(0)
: ThisCurrentValueSubject
holds an integer value (starting at0
) and has aNever
failure type, meaning it doesn’t produce errors.CurrentValueSubject
maintains the current count and allows us to emit updated values whenever the count changes.
- Button to Increment the Count:
- The Increment Count button increments the
countSubject
‘svalue
by 1. SinceCurrentValueSubject
emits its current value immediately to all subscribers, updatingvalue
will automatically broadcast the new count to the subscriber.
- The Increment Count button increments the
- Subscriber Setup in
onAppear
:- In
.onAppear
, we subscribe tocountSubject
with.sink
, updatingcountText
each time theCurrentValueSubject
emits a new count. This automatically refreshes the view to display the updated count.
- In
- Cleaning Up the Subscription:
- The
.onDisappear
modifier cancels the subscription when the view disappears, which is a good practice to prevent retaining subscriptions when they’re no longer needed.
- The
Summary
This example shows how CurrentValueSubject
can be used as a reactive variable that holds and updates a value over time. Each time the button is pressed, the count increases by one, and CurrentValueSubject
automatically notifies all subscribers of the new count. SwiftUI then observes the updated countText
, causing the view to re-render with the new count displayed.
Using CurrentValueSubject
allows you to manage state reactively, making it easy to handle dynamic data that updates frequently or based on user interactions. This pattern is especially valuable in SwiftUI for managing shared data across views or features that benefit from Combine’s reactive capabilities.
Why not just increment the value?
Here’s why CurrentValueSubject
might still be beneficial in certain cases:
- Shared State Across Multiple Views:
CurrentValueSubject
is useful when you need a single source of truth that can be observed by multiple views or components. For example, if you have a shared counter value across different parts of your app,CurrentValueSubject
lets each view automatically react to changes in the count without manually passing values around. - Combine-Friendly Architecture: If your app already uses Combine extensively (e.g., for networking, state management, or complex data flows),
CurrentValueSubject
allows you to handle data changes in a consistent reactive manner. This makes it easier to maintain a cohesive codebase where data is managed through Combine rather than using a mix of reactive and imperative patterns. - Event-Driven Programming:
CurrentValueSubject
enables a more event-driven approach, which is particularly helpful if you plan to add Combine operators (likemap
,filter
,throttle
, etc.) to your publisher. This allows you to process or transform the data in a reactive chain without modifying the original code logic. - External Data or Asynchronous Updates: When the data source might come from something external or asynchronous (e.g., a network call or external service),
CurrentValueSubject
can hold the current state and broadcast it whenever it updates. For example, aCurrentValueSubject
could be updated from a background task and then push new values to the UI.
In simple cases, a direct @State
variable may indeed be the best choice. However, as your app grows in complexity, CurrentValueSubject
offers flexibility for shared, reactive, and asynchronous data management that scales well with Combine and SwiftUI.