我在努力理解为什么在父视图.onAppear
期间对@Published EnvironmentObject变量进行更改时,PageViewController无法按预期工作的原因。
如果我注释掉.onAppear
中的DetailView
修饰符,则一切正常-幻灯片可以按PageViewController的预期滑动。但是,一旦重新启用该修饰符,页面就无法正确滑动。
由于状态变化,这与重新渲染有一定关系,但我无法弄清楚。我知道状态更改会调用updateUIViewController
,这似乎会更改viewControllers的数组。
如果有人对如何解决这个问题有指导,我将不胜感激!
如果运行下面的代码,请确保更改SceneDelegate:
let contentView = ContentView()
到
let contentView = ContentView().environmentObject(Global())
这是问题的一个简单示例:
import SwiftUI
import Foundation
import UIKit
struct Item: Hashable {
var text: String
}
let testItems: [Item] = [
Item(text: "I am item 1"),
Item(text: "I am item 2"),
Item(text: "I am item 3")
]
final class Global : ObservableObject {
@Published var hideText: Bool = false
}
struct ContentView: View {
@EnvironmentObject var global: Global
var body: some View {
NavigationView {
VStack {
List(testItems, id: \.self) { item in
NavigationLink(destination: DetailView(item: item)) {
Text(item.text)
}
}
Text("I should vanish when detail view loads")
.opacity(self.global.hideText ? 0 : 1)
}
.navigationBarTitle("Test")
}
}
}
struct DetailView: View {
@EnvironmentObject var global: Global
var item: Item
var testViews: [Text] = [
Text("Slide 1").font(.title),
Text("Slide 2").font(.title),
Text("Slide 3").font(.title),
]
var body: some View {
// The .onAppear modifier breaks the PageViewController
// when it is commented out, the slides work as expected
PageView(testViews)
.onAppear {
self.global.hideText = true
}
}
}
struct PageView<Page: View>: View {
@State var currentPage = 0
var viewControllers: [UIHostingController<Page>]
init(_ views: [Page]) {
self.viewControllers = views.map{ UIHostingController(rootView: $0) }
}
var body: some View {
VStack(alignment: .center) {
PageViewController(
viewControllers: viewControllers,
currentPageIndex: $currentPage
)
}
}
}
struct PageViewController: UIViewControllerRepresentable {
var viewControllers: [UIViewController]
@Binding var currentPageIndex: Int
func makeCoordinator() -> Coordinator {
Coordinator(self)
}
func makeUIViewController(context: Context) -> UIPageViewController {
let pageViewController = UIPageViewController(
transitionStyle: .scroll,
navigationOrientation: .horizontal
)
pageViewController.view.backgroundColor = UIColor.clear
pageViewController.dataSource = context.coordinator
pageViewController.delegate = context.coordinator
return pageViewController
}
func updateUIViewController(_ pageViewController: UIPageViewController, context: Context) {
pageViewController.setViewControllers([viewControllers[currentPageIndex]], direction: .forward, animated: true)
}
class Coordinator: NSObject, UIPageViewControllerDataSource, UIPageViewControllerDelegate {
var parent: PageViewController
init(_ pageViewController: PageViewController) {
self.parent = pageViewController
}
func pageViewController(_ pageViewController: UIPageViewController, viewControllerBefore viewController: UIViewController) -> UIViewController? {
// retrieves the index of the currently displayed view controller
guard let index = parent.viewControllers.firstIndex(of: viewController) else {
return nil
}
// shows the last view controller when the user swipes back from the first view controller
if index == 0 {
return parent.viewControllers.last
}
//show the view controller before the currently displayed view controller
return parent.viewControllers[index - 1]
}
func pageViewController(_ pageViewController: UIPageViewController, viewControllerAfter viewController: UIViewController) -> UIViewController? {
// retrieves the index of the currently displayed view controller
guard let index = parent.viewControllers.firstIndex(of: viewController) else {
print("View controller not found!!")
return nil
}
// shows the first view controller when the user swipes further from the last view controller
if index + 1 == parent.viewControllers.count {
return parent.viewControllers.first
}
// show the view controller after the currently displayed view controller
return parent.viewControllers[index + 1]
}
func pageViewController(_ pageViewController: UIPageViewController,
didFinishAnimating finished: Bool,
previousViewControllers: [UIViewController],
transitionCompleted completed: Bool
) {
if completed,
let visibleViewController = pageViewController.viewControllers?.first,
let index = parent.viewControllers.firstIndex(of: visibleViewController) {
parent.currentPageIndex = index
}
}
}
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
ContentView()
.environmentObject(Global())
}
}
答案 0 :(得分:1)
这是基于@ Mac3n的有用答案的,他正确地指出,将ViewController传递给状态的UIViewControllerRepresentable部分可以解决此问题。所缺少的只是如何通过init
方法来做到这一点。
这是PageView
struct PageView<Page: View>: View {
@State var currentPage = 0
@State var viewControllers: [UIHostingController<Page>]
init(_ views: [Page]) {
_viewControllers = State(initialValue: views.map{ UIHostingController(rootView: $0) })
}
var body: some View {
VStack(alignment: .center) {
PageViewController(
viewControllers: viewControllers,
currentPageIndex: $currentPage
)
}
}
}
答案 1 :(得分:0)
在EnvironmentObject
的父视图中更改.onAppear
变量时,它会再次呈现该视图,并且我猜值会随着新对象而更新。
如果您在func pageViewController(_ pageViewController: UIPageViewController, viewControllerAfter viewController: UIViewController) -> UIViewController?
方法中设置了一个断点并在控制台中获取po,则可以看到视图的内存地址不同,因此找不到下一个视图控制器。
parent:
(lldb) po parent
▿ PageViewController
▿ viewControllers : 3 elements
▿ 0 : <_TtGC7SwiftUI19UIHostingControllerVS_4Text_: 0x7fc881e04b10>
▿ 1 : <_TtGC7SwiftUI19UIHostingControllerVS_4Text_: 0x7fc881f13f70>
▿ 2 : <_TtGC7SwiftUI19UIHostingControllerVS_4Text_: 0x7fc881f14dd0>
(lldb) po viewController
<_TtGC7SwiftUI19UIHostingControllerVS_4Text_: 0x7fc881f186c0>
如您所见,在父视图控制器中找不到视图控制器
对于解决方案,如果将PageView
更改为此,则可以解决问题,但也可以是其他解决方案。
struct PageView: View {
@State var currentPage = 0
@State var viewControllers: [UIViewController]
var body: some View {
VStack(alignment: .center) {
PageViewController(
viewControllers: viewControllers,
currentPageIndex: $currentPage
)
}
}
}
将您的viewControllers
更改为@State并将其通过父视图传递