reduce(value:nextValue:)
I've found that PreferenceKey's reduce(value:nextValue) will be called several times, even if there are no other Views that have a preference() with the same PreferenceKey in the View tree. And sometimes nextValue() in those calls will return a nil value. Because of this, nextValue() should do more than just
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
static func reduce(value: inout CGSize, nextValue: () -> CGSize) { | |
value = nextValue() | |
} |
One option is to reduce based on a timestamp:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
struct ColorPreference: Equatable { | |
static func == (lhs: ColorPreference, rhs: ColorPreference) -> Bool { | |
lhs.date == rhs.date | |
} | |
let color: Color | |
let date = Date() | |
} | |
struct ColorPreferenceKey: PreferenceKey { | |
static var defaultValue: ColorPreference? = nil | |
static func reduce(value: inout ColorPreference?, nextValue: () -> ColorPreference?) { | |
let nextValue = nextValue() | |
if let notNilValue = value, | |
let nextValue = nextValue { | |
if nextValue.date > notNilValue.date { | |
value = nextValue | |
} | |
} else if value == nil && nextValue != nil { | |
value = nextValue | |
} | |
// Do nothing for these conditions | |
// value == nil && nextValue == nil | |
// value != nil && nextValue == nil | |
} | |
} |
PreferenceKey Propagation Blocked by preference(key:value:)
Another issue I've found is a PreferenceKey will not get passed up through the View tree past any parent View that uses the same PreferenceKey in a preference(key:value:)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//: A Cocoa based Playground to present user interface | |
import AppKit | |
import SwiftUI | |
import PlaygroundSupport | |
struct ColorPreference: Equatable { | |
static func == (lhs: ColorPreference, rhs: ColorPreference) -> Bool { | |
lhs.date == rhs.date | |
} | |
let color: Color | |
let date = Date() | |
} | |
struct ColorPreferenceKey: PreferenceKey { | |
static var defaultValue: ColorPreference? = nil | |
static func reduce(value: inout ColorPreference?, nextValue: () -> ColorPreference?) { | |
let nextValue = nextValue() | |
if let notNilValue = value, | |
let nextValue = nextValue { | |
if nextValue.date > notNilValue.date { | |
value = nextValue | |
} | |
} else if value == nil && nextValue != nil { | |
value = nextValue | |
} | |
// Do nothing for these conditions | |
// value == nil && nextValue == nil | |
// value != nil && nextValue == nil | |
} | |
} | |
struct TestView: View { | |
var body: some View { | |
ZStack { | |
color | |
WrapperView() | |
} | |
.onPreferenceChange(ColorPreferenceKey.self) { colorPreference in | |
color = colorPreference?.color | |
} | |
} | |
@State private var color: Color? = .white | |
} | |
struct WrapperView: View { | |
var body: some View { | |
VStack { | |
ButtonsView() | |
Button("Blue") { | |
colorPreference = ColorPreference(color: .blue) | |
} | |
Button("Yellow") { | |
colorPreference = ColorPreference(color: .yellow) | |
} | |
} | |
.onPreferenceChange(ColorPreferenceKey.self) { colorPreference in | |
let color: String | |
if let colorPreferenceColor = colorPreference?.color { | |
color = "\(colorPreferenceColor)" | |
} else { | |
color = "nil" | |
} | |
print("\(color) before preference") | |
} | |
.preference(key: ColorPreferenceKey.self, value: colorPreference) | |
.onPreferenceChange(ColorPreferenceKey.self) { colorPreference in | |
let color: String | |
if let colorPreferenceColor = colorPreference?.color { | |
color = "\(colorPreferenceColor)" | |
} else { | |
color = "nil" | |
} | |
print("\(color) after preference") | |
} | |
} | |
@State private var colorPreference: ColorPreference? = nil | |
} | |
struct ButtonsView: View { | |
var body: some View { | |
VStack { | |
Button("Red") { | |
colorPreference = ColorPreference(color: .red) | |
} | |
Button("Green") { | |
colorPreference = ColorPreference(color: .green) | |
} | |
} | |
.preference(key: ColorPreferenceKey.self, value: colorPreference) | |
} | |
@State private var colorPreference: ColorPreference? = nil | |
} | |
// Present the view in Playground | |
PlaygroundPage.current.liveView = NSHostingView(rootView: TestView()) |
The "Red" and "Green" buttons should generate "before" and "after" messages while the "Blue" and "Yellow" buttons should only create "after" messages and all buttons should change the color. But because of the issue the "Red" and "Green" buttons don't work as expected.
A way around this is to capture the preference before using it again:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
struct ColorModifier: ViewModifier { | |
func body(content: Content) -> some View { | |
content | |
.onPreferenceChange(ColorPreferenceKey.self) { colorPreference in | |
self.colorPreference = colorPreference | |
} | |
.preference(key: ColorPreferenceKey.self, value: colorPreference) | |
} | |
@Binding var colorPreference: ColorPreference? | |
} | |
extension View { | |
func colorPreference(value: Binding<ColorPreference?>) -> some View { | |
self.modifier(ColorModifier(colorPreference: value)) | |
} | |
} | |
struct WrapperView: View { | |
var body: some View { | |
VStack { | |
ButtonsView() | |
Button("Blue") { | |
colorPreference = ColorPreference(color: .blue) | |
} | |
Button("Yellow") { | |
colorPreference = ColorPreference(color: .yellow) | |
} | |
} | |
.onPreferenceChange(ColorPreferenceKey.self) { colorPreference in | |
let color: String | |
if let colorPreferenceColor = colorPreference?.color { | |
color = "\(colorPreferenceColor)" | |
} else { | |
color = "nil" | |
} | |
print("\(color) before preference") | |
} | |
.colorPreference(value: $colorPreference) | |
.onPreferenceChange(ColorPreferenceKey.self) { colorPreference in | |
let color: String | |
if let colorPreferenceColor = colorPreference?.color { | |
color = "\(colorPreferenceColor)" | |
} else { | |
color = "nil" | |
} | |
print("\(color) after preference") | |
} | |
} | |
@State private var colorPreference: ColorPreference? = nil | |
} |
Now the "Red" and "Green" buttons work as expected.