【问题标题】:SwiftUI: How to connect currentPage of PageView with firing view's passing indexSwiftUI:如何将 PageView 的 currentPage 与触发视图的传递索引连接起来
【发布时间】:2020-04-20 17:19:21
【问题描述】:

我按照 Apple 的示例 interfacing-with-uikit 在 SwiftUI 中使用 PageView。我想在我的应用程序中实现它。场景是MovieView中,当用户点击图像列表中的图像时,会触发以视图和选定索引为参数对PageView的调用。

但是当我点击图像列表的任何项目时,它总是显示第一个元素的页面视图。我尝试在PageViewinit() 中更改它,但是它不起作用。

在控制台中,我尝试打印一些东西,我看到 currentPage 没有改变。

ReceivedIdx: 0
Current page: 1
ReceivedIdx: 3
Current page: 1
ReceivedIdx: 5
Current page: 1

目前currentPagePageViewPageViewCollection之间是连接的,但是如何连接selectedIdx呢?因为它是选择图像的触发点。

电影视图

...
private struct ImageList: View {
    var images: [ImageViewModel]
    @State private var showSheet = false
    @State private var selectedIdx = 0

    var body: some View {

        VStack(alignment: .leading) {
            Text("Images")
                .font(.headline)
            ScrollView(.horizontal) {
                HStack(alignment: .top, spacing: 6) {
                    ForEach(0..<images.count, id: \.self) { i in
                        KFImage(source: .network(self.images[i].fileURL))
                            .resizable()
                            .frame(width: 200)
                            .aspectRatio(1.77, contentMode: .fit)
                            .onTapGesture {
                                self.selectedIdx = i
                                self.showSheet.toggle()
                        }
                    }
                }.sheet(isPresented: $showSheet) {
                    PageView(self.images.map { PresentedImageView(image: $0) }, selectedIdx: self.selectedIdx)
                }
            }.frame(height: 120)
        }
        .padding(.horizontal).padding(.bottom)
    }
}

private struct PresentedImageView: View {
    var image: ImageViewModel

    var body: some View {
        KFImage(source: .network(image.fileURL))
        .resizable()
        //.frame(width: gr.size.width - 6, alignment: .center)
        .aspectRatio(1.77, contentMode: .fit)
    }
}
...

页面浏览

import SwiftUI

struct PageView<Page: View>: View {

    var viewControllers: [UIHostingController<Page>]
    var selectedIdx = 0
    @State var currentPage = 1

    init(_ views: [Page], selectedIdx: Int) {
        self.viewControllers = views.map { UIHostingController(rootView: $0) }
        self.currentPage = selectedIdx
        print("ReceivedIdx: \(selectedIdx)")
        print("Current page: \(currentPage)")
    }

    var body: some View {
        PageViewController(controllers: viewControllers, currentPage: $currentPage)
    }
}

//struct PageView_Previews: PreviewProvider {
//    static var previews: some View {
//        PageView(features.map { FeatureCard(landmark: $0) })
//            .aspectRatio(3/2, contentMode: .fit)
//    }
//}

页面视图控制器

import SwiftUI
import UIKit

struct PageViewController: UIViewControllerRepresentable {

    var controllers: [UIViewController]
    @Binding var currentPage: Int

    func makeCoordinator() -> Coordinator {
        Coordinator(self)
    }

    func makeUIViewController(context: Context) -> UIPageViewController {
        let pageViewController = UIPageViewController(transitionStyle: .scroll,
                                                      navigationOrientation: .horizontal)

        pageViewController.dataSource = context.coordinator
        pageViewController.delegate = context.coordinator
        return pageViewController
    }

    func updateUIViewController(_ pageViewController: UIPageViewController, context: Context) {
        pageViewController.setViewControllers([controllers[currentPage]], 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? {
            guard let index = parent.controllers.firstIndex(of: viewController) else {
                return nil
            }

            if index == 0 {
                return parent.controllers.last
            }

            return parent.controllers[index - 1]
        }

        func pageViewController(_ pageViewController: UIPageViewController, viewControllerAfter viewController: UIViewController) -> UIViewController? {
            guard let index = parent.controllers.firstIndex(of: viewController) else {
                return nil
            }

            if index + 1 == parent.controllers.count {
                return parent.controllers.first
            }

            return parent.controllers[index + 1]
        }

        func pageViewController(_ pageViewController: UIPageViewController, didFinishAnimating finished: Bool, previousViewControllers: [UIViewController], transitionCompleted completed: Bool) {
            if completed,
                let visibleViewController = pageViewController.viewControllers?.first,
            let index = parent.controllers.firstIndex(of: visibleViewController)
            {
                parent.currentPage = index
            }
        }
    }
}

【问题讨论】:

    标签: uikit swiftui uipageviewcontroller


    【解决方案1】:

    终于,我从this post in stackoverflow找到了答案。今天早上!

    一句话:

    SwiftUI 不允许您在初始化程序中更改 @State,但您可以对其进行初始化。

    因此,在我的情况下,删除@State currentPage 上的初始值,并在您的init() 中使用State(initialValue:) 直接设置它。

    记得在currentPage之前使用_

    import SwiftUI
    
    struct PageView<Page: View>: View {
    
        var viewControllers: [UIHostingController<Page>]
        @State var currentPage: Int
    
        init(_ views: [Page], selectedIdx: Int) {
            self.viewControllers = views.map { UIHostingController(rootView: $0) }
            _currentPage = State(initialValue: selectedIdx)
        }
    
        var body: some View {
            PageViewController(controllers: viewControllers, currentPage: $currentPage)
        }
    }
    

    【讨论】:

      猜你喜欢
      • 2020-02-17
      • 2011-02-05
      • 1970-01-01
      • 1970-01-01
      • 2017-06-06
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2018-10-03
      相关资源
      最近更新 更多