使用 @Namespace 为 .matchedGeometryEffect 自定义初始化
Custom init with @Namespace for .matchedGeometryEffect
我正在尝试将带有 @Namespace
属性 的自定义视图结构传递给 .matchedGeometryEffect
到 parent 视图。
由于 parent 将提供 Namespace
我正在使用自定义 init
。
当我使用类似于 @Binding
自定义初始化的语法时,Xcode 强制我在初始化自定义视图时使用包装器。这反过来杀死了我的 .matchedGeometryEffect
.
struct MyView<Content: View>: View {
@Binding var matched: Bool
@Namespace var nspace
let content: Content
init(matched: Binding<Bool>,
nspace: Namespace,
@ViewBuilder content: @escaping () -> Content
) {
self._matched = matched
self._nspace = nspace
self.content = content()
}
var body: some View {
...
}
}
似乎有效的是使用 var nspace: Namespace.ID
而不是 @Namespace var nspace
然后:
struct MyView<Content: View>: View {
@Binding var matched: Bool
var nspace: Namespace.ID
let content: Content
init(matched: Binding<Bool>,
nspace: Namespace.ID,
@ViewBuilder content: @escaping () -> Content
) {
self._matched = matched
self.nspace = nspace
self.content = content()
}
var body: some View {
...
}
}
这会不会在其他地方引起麻烦?有没有更好的方法?
Can this cause trouble somewhere else? Is there a better way?
它不是worse/better它是唯一正确的方法。让我们看看 API:
Namespace.ID
是用来标识匹配效果命名空间的值
/// A dynamic property type that allows access to a namespace defined
/// by the persistent identity of the object containing the property
/// (e.g. a view).
@available(iOS 14.0, macOS 11.0, tvOS 14.0, watchOS 7.0, *)
@frozen @propertyWrapper public struct Namespace : DynamicProperty {
@inlinable public init()
public var wrappedValue: Namespace.ID { get } // << here !!
正如所见
/// - namespace: The namespace in which defines the `id`. New
/// namespaces are created by adding an `@Namespace()` variable
/// to a ``View`` type and reading its value in the view's body
/// method.
/// - properties: The properties to copy from the source view.
/// - anchor: The relative location in the view used to produce
/// its shared position value.
/// - isSource: True if the view should be used as the source of
/// geometry for other views in the group.
///
/// - Returns: A new view that defines an entry in the global
/// database of views synchronizing their geometry.
///
@inlinable public func matchedGeometryEffect<ID>(id: ID,
in namespace: Namespace.ID, // << here !!
properties: MatchedGeometryProperties = .frame, anchor: UnitPoint = .center, isSource: Bool = true) -> some View where ID : Hashable
我正在尝试将带有 @Namespace
属性 的自定义视图结构传递给 .matchedGeometryEffect
到 parent 视图。
由于 parent 将提供 Namespace
我正在使用自定义 init
。
当我使用类似于 @Binding
自定义初始化的语法时,Xcode 强制我在初始化自定义视图时使用包装器。这反过来杀死了我的 .matchedGeometryEffect
.
struct MyView<Content: View>: View {
@Binding var matched: Bool
@Namespace var nspace
let content: Content
init(matched: Binding<Bool>,
nspace: Namespace,
@ViewBuilder content: @escaping () -> Content
) {
self._matched = matched
self._nspace = nspace
self.content = content()
}
var body: some View {
...
}
}
似乎有效的是使用 var nspace: Namespace.ID
而不是 @Namespace var nspace
然后:
struct MyView<Content: View>: View {
@Binding var matched: Bool
var nspace: Namespace.ID
let content: Content
init(matched: Binding<Bool>,
nspace: Namespace.ID,
@ViewBuilder content: @escaping () -> Content
) {
self._matched = matched
self.nspace = nspace
self.content = content()
}
var body: some View {
...
}
}
这会不会在其他地方引起麻烦?有没有更好的方法?
Can this cause trouble somewhere else? Is there a better way?
它不是worse/better它是唯一正确的方法。让我们看看 API:
Namespace.ID
是用来标识匹配效果命名空间的值
/// A dynamic property type that allows access to a namespace defined
/// by the persistent identity of the object containing the property
/// (e.g. a view).
@available(iOS 14.0, macOS 11.0, tvOS 14.0, watchOS 7.0, *)
@frozen @propertyWrapper public struct Namespace : DynamicProperty {
@inlinable public init()
public var wrappedValue: Namespace.ID { get } // << here !!
正如所见
/// - namespace: The namespace in which defines the `id`. New
/// namespaces are created by adding an `@Namespace()` variable
/// to a ``View`` type and reading its value in the view's body
/// method.
/// - properties: The properties to copy from the source view.
/// - anchor: The relative location in the view used to produce
/// its shared position value.
/// - isSource: True if the view should be used as the source of
/// geometry for other views in the group.
///
/// - Returns: A new view that defines an entry in the global
/// database of views synchronizing their geometry.
///
@inlinable public func matchedGeometryEffect<ID>(id: ID,
in namespace: Namespace.ID, // << here !!
properties: MatchedGeometryProperties = .frame, anchor: UnitPoint = .center, isSource: Bool = true) -> some View where ID : Hashable