From 6568ab0d19efacc4eef2a0138f8e20979c2439d5 Mon Sep 17 00:00:00 2001 From: Ryan Zhao Date: Mon, 17 Jul 2023 16:44:52 +1000 Subject: [PATCH] carousel view with swiftui --- Session.xcodeproj/project.pbxproj | 8 +- .../SessionCarouselView+SwiftUI.swift | 181 ++++++++++++++++++ 2 files changed, 187 insertions(+), 2 deletions(-) create mode 100644 SessionUIKit/Components/SessionCarouselView+SwiftUI.swift diff --git a/Session.xcodeproj/project.pbxproj b/Session.xcodeproj/project.pbxproj index 5f3fe66c0..52426533c 100644 --- a/Session.xcodeproj/project.pbxproj +++ b/Session.xcodeproj/project.pbxproj @@ -161,6 +161,7 @@ 7BD687D42A5E852600D8E455 /* ProfilePictureView+SwiftUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BD687D32A5E852600D8E455 /* ProfilePictureView+SwiftUI.swift */; }; 7BDCFC08242186E700641C39 /* NotificationServiceExtensionContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BDCFC07242186E700641C39 /* NotificationServiceExtensionContext.swift */; }; 7BDCFC0B2421EB7600641C39 /* Localizable.strings in Resources */ = {isa = PBXBuildFile; fileRef = B6F509951AA53F760068F56A /* Localizable.strings */; }; + 7BE2701E2A64C11500CEB71A /* SessionCarouselView+SwiftUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BE2701D2A64C11500CEB71A /* SessionCarouselView+SwiftUI.swift */; }; 7BFA8AE32831D0D4001876F3 /* ContextMenuVC+EmojiReactsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BFA8AE22831D0D4001876F3 /* ContextMenuVC+EmojiReactsView.swift */; }; 7BFD1A8A2745C4F000FB91B9 /* Permissions.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BFD1A892745C4F000FB91B9 /* Permissions.swift */; }; 7BFD1A8C2747150E00FB91B9 /* TurnServerInfo.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BFD1A8B2747150E00FB91B9 /* TurnServerInfo.swift */; }; @@ -1266,6 +1267,7 @@ 7BD687D32A5E852600D8E455 /* ProfilePictureView+SwiftUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ProfilePictureView+SwiftUI.swift"; sourceTree = ""; }; 7BDCFC0424206E7300641C39 /* SessionNotificationServiceExtension.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = SessionNotificationServiceExtension.entitlements; sourceTree = ""; }; 7BDCFC07242186E700641C39 /* NotificationServiceExtensionContext.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = NotificationServiceExtensionContext.swift; sourceTree = ""; }; + 7BE2701D2A64C11500CEB71A /* SessionCarouselView+SwiftUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "SessionCarouselView+SwiftUI.swift"; sourceTree = ""; }; 7BFA8AE22831D0D4001876F3 /* ContextMenuVC+EmojiReactsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ContextMenuVC+EmojiReactsView.swift"; sourceTree = ""; }; 7BFD1A892745C4F000FB91B9 /* Permissions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Permissions.swift; sourceTree = ""; }; 7BFD1A8B2747150E00FB91B9 /* TurnServerInfo.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TurnServerInfo.swift; sourceTree = ""; }; @@ -2930,6 +2932,7 @@ 7BD687D32A5E852600D8E455 /* ProfilePictureView+SwiftUI.swift */, FD71165A28E6DDBC00B47552 /* StyledNavigationController.swift */, FD0B77AF29B69A65009169BA /* TopBannerController.swift */, + 7BE2701D2A64C11500CEB71A /* SessionCarouselView+SwiftUI.swift */, ); path = Components; sourceTree = ""; @@ -5355,8 +5358,8 @@ inputFileListPaths = ( ); inputPaths = ( - "$BUILT_PRODUCTS_DIR/$INFOPLIST_PATH", - "$TARGET_BUILD_DIR/$INFOPLIST_PATH", + $BUILT_PRODUCTS_DIR/$INFOPLIST_PATH, + $TARGET_BUILD_DIR/$INFOPLIST_PATH, ); name = "Add Commit Hash To Build Info Plist"; outputFileListPaths = ( @@ -5423,6 +5426,7 @@ FD37E9D328A1FCDB003AE748 /* Theme+OceanDark.swift in Sources */, FD71165928E436E800B47552 /* ConfirmationModal.swift in Sources */, 7BBBDC44286EAD2D00747E59 /* TappableLabel.swift in Sources */, + 7BE2701E2A64C11500CEB71A /* SessionCarouselView+SwiftUI.swift in Sources */, FD09B7E328865FDA00ED0B66 /* HighlightMentionBackgroundView.swift in Sources */, C331FFB82558FA8D00070591 /* DeviceUtilities.swift in Sources */, C331FFE72558FB0000070591 /* TextField.swift in Sources */, diff --git a/SessionUIKit/Components/SessionCarouselView+SwiftUI.swift b/SessionUIKit/Components/SessionCarouselView+SwiftUI.swift new file mode 100644 index 000000000..79a564887 --- /dev/null +++ b/SessionUIKit/Components/SessionCarouselView+SwiftUI.swift @@ -0,0 +1,181 @@ +// Copyright © 2023 Rangeproof Pty Ltd. All rights reserved. + +import SwiftUI + +struct SessionCarouselView_SwiftUI: View { + @State var index = 0 + + var colors: [Color] = [.red, .orange, .blue] + var body: some View { + HStack { + ArrowView(value: $index.animation(.easeInOut), range: 0...(colors.count - 1), type: .decrement) + .zIndex(1) + + PageView(index: $index.animation(), maxIndex: colors.count - 1) { + ForEach(self.colors, id: \.self) { color in + Rectangle() + .foregroundColor(color) + } + } + .aspectRatio(1, contentMode: .fit) + + ArrowView(value: $index.animation(.easeInOut), range: 0...(colors.count - 1), type: .increment) + .zIndex(1) + } + } +} + +struct ArrowView: View { + @Binding var value: Int + let range: ClosedRange + let type: ArrowType + + enum ArrowType { + case increment + case decrement + } + + init(value: Binding, range: ClosedRange, type: ArrowType) { + self._value = value + self.range = range + self.type = type + } + + var body: some View { + let imageName = self.type == .decrement ? "chevron.left" : "chevron.right" + Button { + print("Tap") + if self.type == .decrement { + decrement() + } else { + increment() + } + } label: { + Image(systemName: imageName) + .foregroundColor(.white) + .frame(width: 30, height: 30) + } + } + + func decrement() { + if value > range.lowerBound { + value -= 1 + } + if value < range.lowerBound { + value = range.lowerBound + } + } + + func increment() { + if value < range.upperBound { + value += 1 + } + if value > range.upperBound { + value = range.upperBound + } + } +} + +struct PageView: View where Content: View { + @Binding var index: Int + let maxIndex: Int + let content: () -> Content + + @State private var offset = CGFloat.zero + @State private var dragging = false + + init(index: Binding, maxIndex: Int, @ViewBuilder content: @escaping () -> Content) { + self._index = index + self.maxIndex = maxIndex + self.content = content + } + + var body: some View { + ZStack(alignment: .bottom) { + GeometryReader { geometry in + ScrollView(.horizontal, showsIndicators: false) { + HStack(spacing: 0) { + self.content() + .frame(width: geometry.size.width, height: geometry.size.height) + } + } + .content.offset(x: self.offset(in: geometry), y: 0) + .frame(width: geometry.size.width, alignment: .leading) + .clipShape(RoundedRectangle(cornerRadius: 15)) + .gesture( + DragGesture(coordinateSpace: .local) + .onChanged { value in + self.dragging = true + self.offset = -CGFloat(self.index) * geometry.size.width + value.translation.width + } + .onEnded { value in + let predictedEndOffset = -CGFloat(self.index) * geometry.size.width + value.predictedEndTranslation.width + let predictedIndex = Int(round(predictedEndOffset / -geometry.size.width)) + self.index = self.clampedIndex(from: predictedIndex) + withAnimation(.easeOut) { + self.dragging = false + } + } + ) + } + .clipped() + + PageControl(index: $index, maxIndex: maxIndex) + .padding(EdgeInsets(top: 0, leading: 0, bottom: 8, trailing: 0)) + } + } + + func offset(in geometry: GeometryProxy) -> CGFloat { + if self.dragging { + return max(min(self.offset, 0), -CGFloat(self.maxIndex) * geometry.size.width) + } else { + return -CGFloat(self.index) * geometry.size.width + } + } + + func clampedIndex(from predictedIndex: Int) -> Int { + let newIndex = min(max(predictedIndex, self.index - 1), self.index + 1) + guard newIndex >= 0 else { return 0 } + guard newIndex <= maxIndex else { return maxIndex } + return newIndex + } +} + +struct PageControl: View { + @Binding var index: Int + let maxIndex: Int + + var body: some View { + ZStack { + Capsule() + .foregroundColor(.init(white: 0, opacity: 0.4)) + HStack(spacing: 4) { + ForEach(0...maxIndex, id: \.self) { index in + Circle() + .fill(index == self.index ? Color.white : Color.gray) + .frame(width: 6.62, height: 6.62) + } + } + .padding(6) + } + .fixedSize(horizontal: true, vertical: true) + .frame( + maxWidth: .infinity, + maxHeight: 19 + ) + } +} + +struct SessionCarouselView_SwiftUI_Previews: PreviewProvider { + static var previews: some View { + ZStack { + if #available(iOS 14.0, *) { + Color.black.ignoresSafeArea() + } else { + Color.black + } + + SessionCarouselView_SwiftUI() + } + } +}