How can I implement PageView in SwiftUI?
我是SwiftUI的新手。 我有三个视图,我希望它们在PageView中。 我想像滑动浏览器一样滑动每个"视图",并希望这些小点表示我所在的视图。
页面控制
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 | struct PageControl: UIViewRepresentable { var numberOfPages: Int @Binding var currentPage: Int func makeCoordinator() -> Coordinator { Coordinator(self) } func makeUIView(context: Context) -> UIPageControl { let control = UIPageControl() control.numberOfPages = numberOfPages control.pageIndicatorTintColor = UIColor.lightGray control.currentPageIndicatorTintColor = UIColor.darkGray control.addTarget( context.coordinator, action: #selector(Coordinator.updateCurrentPage(sender:)), for: .valueChanged) return control } func updateUIView(_ uiView: UIPageControl, context: Context) { uiView.currentPage = currentPage } class Coordinator: NSObject { var control: PageControl init(_ control: PageControl) { self.control = control } @objc func updateCurrentPage(sender: UIPageControl) { control.currentPage = sender.currentPage } } } |
您的页面浏览量
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | struct PageView<Page: View>: View { var viewControllers: [UIHostingController<Page>] @State var currentPage = 0 init(_ views: [Page]) { self.viewControllers = views.map { UIHostingController(rootView: $0) } } var body: some View { ZStack(alignment: .bottom) { PageViewController(controllers: viewControllers, currentPage: $currentPage) PageControl(numberOfPages: viewControllers.count, currentPage: $currentPage) } } } |
您的页面视图控制器
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 | 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) { guard !controllers.isEmpty else { return } context.coordinator.parent = self 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 } } } } |
假设您有一个类似的观点
1 2 3 4 5 6 7 8 | struct CardView: View { var album: Album var body: some View { URLImage(URL(string: album.albumArtWork)!) .resizable() .aspectRatio(3 / 2, contentMode: .fit) } } |
您可以像这样在主SwiftUI视图中使用此组件。
1 | PageView(vM.Albums.map { CardView(album: $0) }).frame(height: 250) |
SwiftUI 2
SwiftUI中现在等效于
1 2 3 4 5 6 7 8 9 10 11 12 13 | @main struct TestApp: App { var body: some Scene { WindowGroup { TabView { FirstView() SecondView() ThirdView() } .tabViewStyle(PageTabViewStyle()) } } } |
您还可以控制分页点的显示方式:
1 2 | // hide paging dots .tabViewStyle(PageTabViewStyle(indexDisplayMode: .never)) |
您可以在此链接中找到更详细的说明:
- 如何使用tabViewStyle()创建内容的滚动页面
垂直变体
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | @main struct TestApp: App { var body: some Scene { WindowGroup { TabView { Group { FirstView() SecondView() ThirdView() } .rotationEffect(Angle(degrees: -90)) } .tabViewStyle(PageTabViewStyle(indexDisplayMode: .never)) .rotationEffect(Angle(degrees: 90)) } } } |
定制组件
如果您不想每次都通过
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | struct PageView<Content>: View where Content: View { let indexDisplayMode: PageTabViewStyle.IndexDisplayMode let indexBackgroundDisplayMode: PageIndexViewStyle.BackgroundDisplayMode let content: () -> Content init( indexDisplayMode: PageTabViewStyle.IndexDisplayMode = .automatic, indexBackgroundDisplayMode: PageIndexViewStyle.BackgroundDisplayMode = .automatic, @ViewBuilder content: @escaping () -> Content ) { self.indexDisplayMode = indexDisplayMode self.indexBackgroundDisplayMode = indexBackgroundDisplayMode self.content = content } var body: some View { TabView { content() } .tabViewStyle(PageTabViewStyle(indexDisplayMode: indexDisplayMode)) .indexViewStyle(PageIndexViewStyle(backgroundDisplayMode: indexBackgroundDisplayMode)) } } |
现在您有了默认的
1 2 3 4 5 | PageView { FirstView() SecondView() ThirdView() } |
也可以自定义:
1 | PageView(indexDisplayMode: .always, indexBackgroundDisplayMode: .always) { ... } |
对于以iOS 14及更高版本为目标的应用程序,@ pawello2222建议的答案应被视为正确的答案。我现在已经在两个应用程序中对其进行了尝试,并且只需很少的代码即可很好地工作。
我将提出的概念包装在一个结构中,该结构可以同时提供视图,项目列表和视图构建器。在这里能找到它。代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 | @available(iOS 14.0, *) public struct MultiPageView: View { public init<PageType: View>( pages: [PageType], indexDisplayMode: PageTabViewStyle.IndexDisplayMode = .automatic, currentPageIndex: Binding<Int>) { self.pages = pages.map { $0.any() } self.indexDisplayMode = indexDisplayMode self.currentPageIndex = currentPageIndex } public init<Model, ViewType: View>( items: [Model], indexDisplayMode: PageTabViewStyle.IndexDisplayMode = .automatic, currentPageIndex: Binding<Int>, pageBuilder: (Model) -> ViewType) { self.pages = items.map { pageBuilder($0).any() } self.indexDisplayMode = indexDisplayMode self.currentPageIndex = currentPageIndex } private let pages: [AnyView] private let indexDisplayMode: PageTabViewStyle.IndexDisplayMode private var currentPageIndex: Binding<Int> public var body: some View { TabView(selection: currentPageIndex) { ForEach(Array(pages.enumerated()), id: \\.offset) { $0.element.tag($0.offset) } } .tabViewStyle(PageTabViewStyle(indexDisplayMode: indexDisplayMode)) } } |