gpt4 book ai didi

ios - 在 onAppear 中更改状态会破坏 PageViewController

转载 作者:行者123 更新时间:2023-12-01 22:00:11 26 4
gpt4 key购买 nike

我很难理解为什么在 .onAppear 期间 @Published EnvironmentObject 变量发生更改时 PageViewController 无法按预期工作在其父 View 中。

如果我注释掉 .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())
}
}

最佳答案

这建立在@Mac3n 的有用答案之上,他正确地指出,将 ViewControllers 传递给状态的 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
)
}
}
}

关于ios - 在 onAppear 中更改状态会破坏 PageViewController,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60241723/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com