Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/cookednick/watchable
A generic ObservableObject for every property!
https://github.com/cookednick/watchable
binding combine generic observableobject observedobject publisher swift swiftui view
Last synced: about 2 months ago
JSON representation
A generic ObservableObject for every property!
- Host: GitHub
- URL: https://github.com/cookednick/watchable
- Owner: cookednick
- License: mit
- Created: 2021-03-16T23:10:54.000Z (almost 4 years ago)
- Default Branch: main
- Last Pushed: 2024-06-20T21:09:17.000Z (7 months ago)
- Last Synced: 2024-06-21T15:57:31.853Z (7 months ago)
- Topics: binding, combine, generic, observableobject, observedobject, publisher, swift, swiftui, view
- Language: Swift
- Homepage:
- Size: 78.1 KB
- Stars: 44
- Watchers: 1
- Forks: 2
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Watchable
A property wrapper type that creates an `ObservableObject` for just a single property.
This can have **enormous performance benefits** due to how SwiftUI recomputes views.
## SwiftUI + Performance
Views get called to update each time **any** of an object's properties change. Even if it's irrelevant to a given view.
Consider the following common pattern of a typical custom view with one `@State` property.
```swift
struct BedroomLabel: View {
@State var areTheLightsOn = false
var numberOfPeople = 0
var body: some View {
VStack {
Text("This bedroom has \(numberOfPeople) people in it.")Toggle("Lights On/Off", isOn: $areTheLightsOn)
}
}
}
```This is a very common pattern in SwiftUI, and it works alright, but the `BedroomLabel` will update in its entirety whenever the lights get toggled. This means the CPU spends *unnecessary* time checking the entire body whenever this happens.
**How can we do better?**
## One `ObservableObject` per Property
Here is that same example written using `Watchable`.
```swift
struct BedroomLabel: View {
@Watchable var areTheLightsOn = false
var numberOfPeople = 0
var body: some View {
VStack {
Text("This bedroom has \(numberOfPeople) people in it.")
Watching(bindingOf: $areTheLightsOn) { isOn in
Toggle("Lights On/Off", isOn: isOn)
}
}
}
}
````@Watchable` writes just like `@State`, except it stores the value in an implicit generic `ObservableObject` type. `Watching` takes care of unwrapping that and recomputing a view closure whenever its changed.
This means `BedroomLabel` won't be computed on changes. Only the `Toggle` will be.
## Installation
Use Swift Package Manager with this GitHub URL.
## License
````
MIT LicenseCopyright © 2021-2024 Nicolas Cook Leon
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
````