Skip to content
This repository was archived by the owner on May 31, 2024. It is now read-only.

Commit f90ceaa

Browse files
committed
Initial commit
0 parents  commit f90ceaa

File tree

8 files changed

+250
-0
lines changed

8 files changed

+250
-0
lines changed

.gitignore

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
.DS_Store
2+
/.build
3+
/Packages
4+
/*.xcodeproj
5+
xcuserdata/

.swiftpm/xcode/package.xcworkspace/contents.xcworkspacedata

+7
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
<?xml version="1.0" encoding="UTF-8"?>
2+
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
3+
<plist version="1.0">
4+
<dict>
5+
<key>IDEDidComputeMac32BitWarning</key>
6+
<true/>
7+
</dict>
8+
</plist>

Package.swift

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
// swift-tools-version:5.2
2+
// The swift-tools-version declares the minimum version of Swift required to build this package.
3+
4+
import PackageDescription
5+
6+
let package = Package(
7+
name: "PageView",
8+
platforms: [
9+
.iOS(.v13),
10+
.tvOS(.v13),
11+
.watchOS(.v6),
12+
.macOS(.v10_15)
13+
],
14+
products: [
15+
.library(
16+
name: "PageView",
17+
targets: ["PageView"]),
18+
],
19+
targets: [
20+
.target(
21+
name: "PageView",
22+
dependencies: [])
23+
]
24+
)

README.md

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# PageView
2+
3+
This package contains a PageView purely written in SwiftUI.
+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
//
2+
// IndicatorPosition.swift
3+
// ViewCTestApp
4+
//
5+
// Created by Marlo Kessler on 20.07.20.
6+
// Copyright © 2020 Marlo Kessler. All rights reserved.
7+
//
8+
9+
import Foundation
10+
11+
@available(iOS 13, *)
12+
public enum IndicatorPosition {
13+
case topOutBounds
14+
case topInBounds
15+
case bottomOutBounds
16+
case bottomInBounds
17+
}

Sources/PageView/PageIndicator.swift

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
//
2+
// PageIndicator.swift
3+
// ViewCTestApp
4+
//
5+
// Created by Marlo Kessler on 20.07.20.
6+
// Copyright © 2020 Marlo Kessler. All rights reserved.
7+
//
8+
9+
import SwiftUI
10+
11+
@available(iOS 13, *)
12+
struct PageIndicator: UIViewRepresentable {
13+
14+
@Binding var currentIndex: Int
15+
var pageNumber: Int
16+
var pageIndicatorColor: UIColor?
17+
var currentPageIndicatorColor: UIColor?
18+
19+
private let pageControl = UIPageControl()
20+
21+
func makeUIView(context: Context) -> UIPageControl {
22+
pageControl.numberOfPages = pageNumber
23+
if let color = pageIndicatorColor{ pageControl.pageIndicatorTintColor = color }
24+
if let color = currentPageIndicatorColor{ pageControl.currentPageIndicatorTintColor = color }
25+
pageControl.addTarget(context.coordinator, action: #selector(context.coordinator.change), for: .valueChanged)
26+
return pageControl
27+
}
28+
29+
func updateUIView(_ uiView: UIPageControl, context: Context) {
30+
uiView.currentPage = currentIndex
31+
}
32+
33+
func makeCoordinator() -> Coordinator {
34+
return Coordinator(parent: self)
35+
}
36+
37+
class Coordinator {
38+
39+
init(parent: PageIndicator) {
40+
self.parent = parent
41+
}
42+
43+
private let parent: PageIndicator
44+
45+
@objc func change() {
46+
parent.currentIndex = parent.pageControl.currentPage
47+
}
48+
}
49+
}

Sources/PageView/PageView.swift

+137
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,137 @@
1+
//
2+
// PageView.swift
3+
// ViewCTestApp
4+
//
5+
// Created by Marlo Kessler on 18.07.20.
6+
// Copyright © 2020 Marlo Kessler. All rights reserved.
7+
//
8+
9+
import SwiftUI
10+
11+
@available(iOS 13, *)
12+
public struct PageView<Content: View>: View {
13+
14+
public init(pageCount: Int,
15+
currentIndex: Binding<Int> = Binding<Int>(get: {return 0}, set: {_ in}),
16+
@ViewBuilder content: @escaping () -> Content)
17+
{
18+
self.pageCount = pageCount
19+
self._currentIndex = currentIndex
20+
self.content = content
21+
}
22+
23+
private init(_ pageCount: Int,
24+
_ currentIndex: Binding<Int>,
25+
_ hideIndicator: Bool,
26+
_ indicatorPosition: IndicatorPosition,
27+
_ currentPageIndicatorColor: UIColor?,
28+
_ indicatorColor: UIColor?,
29+
_ indicatorBackgroundColor:Color?,
30+
@ViewBuilder _ content: @escaping () -> Content)
31+
{
32+
self.pageCount = pageCount
33+
self._currentIndex = currentIndex
34+
35+
self.hideIndicator = hideIndicator
36+
self.indicatorPosition = indicatorPosition
37+
self.currentPageIndicatorColor = currentPageIndicatorColor
38+
self.indicatorColor = indicatorColor
39+
self.indicatorBackgroundColor = indicatorBackgroundColor
40+
41+
self.content = content
42+
}
43+
44+
45+
46+
// MARK: - Variables
47+
private let pageCount: Int
48+
@Binding private var currentIndex: Int
49+
private let content: () -> Content
50+
51+
@GestureState private var translation: CGFloat = 0
52+
53+
private var hideIndicator: Bool = false
54+
private var indicatorPosition: IndicatorPosition = .bottomInBounds
55+
private var indicatorColor: UIColor?
56+
private var currentPageIndicatorColor: UIColor?
57+
private var indicatorBackgroundColor: Color?
58+
59+
60+
61+
// MARK: - Methods
62+
public func indicator(position: IndicatorPosition = .bottomInBounds, current: UIColor? = nil, other: UIColor? = nil, background: Color? = nil) -> PageView {
63+
return PageView(pageCount, $currentIndex, hideIndicator, position, current, other, background, content)
64+
}
65+
66+
public func hideIndicator(_ hide: Bool = true) -> PageView {
67+
return PageView(pageCount, $currentIndex, hide, indicatorPosition, currentPageIndicatorColor, indicatorColor, indicatorBackgroundColor, content)
68+
}
69+
70+
71+
72+
// MARK: - View
73+
private var indicator: some View {
74+
PageIndicator(currentIndex: self.$currentIndex,
75+
pageNumber: self.pageCount,
76+
pageIndicatorColor: self.indicatorColor,
77+
currentPageIndicatorColor: self.currentPageIndicatorColor)
78+
.frame(height: 15)
79+
.padding(.horizontal, 8)
80+
.background(self.indicatorBackgroundColor ?? Color.clear.opacity(0.5))
81+
.clipped()
82+
.cornerRadius(7.5)
83+
}
84+
85+
public var body: some View {
86+
GeometryReader { geometry in
87+
88+
ZStack {
89+
90+
if self.indicatorPosition == .topInBounds {
91+
VStack {
92+
self.indicator
93+
.offset(y: 16)
94+
Spacer()
95+
}
96+
.zIndex(1)
97+
}
98+
99+
VStack {
100+
101+
if self.indicatorPosition == .topOutBounds { self.indicator }
102+
103+
HStack(spacing: 0) {
104+
self.content().frame(width: geometry.size.width)
105+
}
106+
.frame(width: geometry.size.width, alignment: .leading)
107+
.offset(x: -CGFloat(self.currentIndex) * geometry.size.width)
108+
.offset(x: self.translation)
109+
.clipped()
110+
.animation(.interactiveSpring())
111+
.gesture(
112+
DragGesture()
113+
.updating(self.$translation) { value, state, _ in
114+
state = value.translation.width
115+
}
116+
.onEnded { value in
117+
let offset = value.translation.width / geometry.size.width
118+
let newIndex = (CGFloat(self.currentIndex) - offset).rounded()
119+
self.currentIndex = min(max(Int(newIndex), 0), self.pageCount - 1)
120+
}
121+
)
122+
123+
if self.indicatorPosition == .bottomOutBounds { self.indicator }
124+
}
125+
126+
if self.indicatorPosition == .bottomInBounds {
127+
VStack {
128+
Spacer()
129+
self.indicator
130+
.offset(y: -16)
131+
}
132+
}
133+
134+
}
135+
}
136+
}
137+
}

0 commit comments

Comments
 (0)