From fff086dcf4c0a7dfa9d4e7f7288870e9f0cb5165 Mon Sep 17 00:00:00 2001 From: Emin Date: Wed, 2 Oct 2024 11:45:33 +0500 Subject: [PATCH] ios: make fullscreen images viewer with zoom --- iphone/Maps/Maps.xcodeproj/project.pbxproj | 4 + .../Profile/PlaceDetails/AllPicsScreen.swift | 10 +- .../PlaceDetails/FullscreenImageViewer.swift | 140 ++++++++++++++++++ .../PlaceDetails/Gallery/GalleryScreen.swift | 61 ++++---- .../Reviews/Components/ReviewView.swift | 4 +- iphone/Maps/UI/Storyboard/Main.storyboard | 16 +- 6 files changed, 195 insertions(+), 40 deletions(-) create mode 100644 iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/FullscreenImageViewer.swift diff --git a/iphone/Maps/Maps.xcodeproj/project.pbxproj b/iphone/Maps/Maps.xcodeproj/project.pbxproj index 07530623e5..e4d427d26f 100644 --- a/iphone/Maps/Maps.xcodeproj/project.pbxproj +++ b/iphone/Maps/Maps.xcodeproj/project.pbxproj @@ -584,6 +584,7 @@ CDCA278622451F5000167D87 /* RouteInfo.swift in Sources */ = {isa = PBXBuildFile; fileRef = CDCA278522451F5000167D87 /* RouteInfo.swift */; }; CDCA278E2248F34C00167D87 /* MWMRoutingManager.mm in Sources */ = {isa = PBXBuildFile; fileRef = CDCA278B2248F34C00167D87 /* MWMRoutingManager.mm */; }; CE2D27F82CA2C49F00094565 /* BackButtonWithText.m in Sources */ = {isa = PBXBuildFile; fileRef = CE2D27F72CA2C49F00094565 /* BackButtonWithText.m */; }; + CE60A8C52CAD15C20055F49C /* FullscreenImageViewer.swift in Sources */ = {isa = PBXBuildFile; fileRef = CE60A8C42CAD15C20055F49C /* FullscreenImageViewer.swift */; }; CE64501B2C93F5840075A59B /* PlacePersistenceControllerTesterBro.swift in Sources */ = {isa = PBXBuildFile; fileRef = CE64501A2C93F5840075A59B /* PlacePersistenceControllerTesterBro.swift */; }; CE64501D2C93F8350075A59B /* ReviewsPersistenceController.swift in Sources */ = {isa = PBXBuildFile; fileRef = CE64501C2C93F8350075A59B /* ReviewsPersistenceController.swift */; }; CE6450202C9402EC0075A59B /* ReviewsPersistenceControllerTesterBro.swift in Sources */ = {isa = PBXBuildFile; fileRef = CE64501F2C9402EC0075A59B /* ReviewsPersistenceControllerTesterBro.swift */; }; @@ -1638,6 +1639,7 @@ CDE0F3AD225B8D45008BA5C3 /* MWMSpeedCameraManagerMode.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = MWMSpeedCameraManagerMode.h; sourceTree = ""; }; CE2D27F72CA2C49F00094565 /* BackButtonWithText.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = BackButtonWithText.m; sourceTree = ""; }; CE2D27FB2CA2C64700094565 /* BackButtonWithText.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = BackButtonWithText.h; sourceTree = ""; }; + CE60A8C42CAD15C20055F49C /* FullscreenImageViewer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FullscreenImageViewer.swift; sourceTree = ""; }; CE64501A2C93F5840075A59B /* PlacePersistenceControllerTesterBro.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlacePersistenceControllerTesterBro.swift; sourceTree = ""; }; CE64501C2C93F8350075A59B /* ReviewsPersistenceController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ReviewsPersistenceController.swift; sourceTree = ""; }; CE64501F2C9402EC0075A59B /* ReviewsPersistenceControllerTesterBro.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ReviewsPersistenceControllerTesterBro.swift; sourceTree = ""; }; @@ -3365,6 +3367,7 @@ 52EF1B612C8989F1003046A4 /* PlaceViewController.swift */, 52EF1B652C8989F9003046A4 /* PlaceViewModel.swift */, CED0E01A2C8B048C008C61CA /* AllPicsScreen.swift */, + CE60A8C42CAD15C20055F49C /* FullscreenImageViewer.swift */, ); name = PlaceDetails; path = Profile/PlaceDetails; @@ -5572,6 +5575,7 @@ 47B9065321C7FA400079C85E /* MWMImageCache.m in Sources */, F6FEA82E1C58F108007223CC /* MWMButton.m in Sources */, 34B924431DC8A29C0008D971 /* MWMMailViewController.m in Sources */, + CE60A8C52CAD15C20055F49C /* FullscreenImageViewer.swift in Sources */, 340475651E081A4600C92850 /* MWMRouter.mm in Sources */, 47E3C72F2111F472008B3B27 /* CoverVerticalModalTransitioning.swift in Sources */, 52522F4C2C6E10FD0015709C /* LoadImg.swift in Sources */, diff --git a/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/AllPicsScreen.swift b/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/AllPicsScreen.swift index 052917ab14..397320a8ae 100644 --- a/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/AllPicsScreen.swift +++ b/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/AllPicsScreen.swift @@ -22,10 +22,12 @@ struct AllPicsScreen: View { spacing: 16 ) { ForEach(urls, id: \.self) { url in - LoadImageView(url: url) - .frame(maxWidth: minWidth, maxHeight: maxHeight) - .clipShape(RoundedRectangle(cornerRadius: 8)) - .scaledToFill() + NavigationLink(destination: FullscreenImageViewer(selectedImageUrl: url, imageUrls: urls)) { + LoadImageView(url: url) + .frame(maxWidth: minWidth, maxHeight: maxHeight) + .clipShape(RoundedRectangle(cornerRadius: 8)) + .scaledToFill() + } } } } diff --git a/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/FullscreenImageViewer.swift b/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/FullscreenImageViewer.swift new file mode 100644 index 0000000000..f008602ee2 --- /dev/null +++ b/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/FullscreenImageViewer.swift @@ -0,0 +1,140 @@ +import SwiftUI +import Combine + +struct FullscreenImageViewer: View { + let imageUrls: [String] + @State private var currentPage: Int + @State private var scale: CGFloat = 1.0 + @State private var lastScale: CGFloat = 1.0 + @State private var offset: CGSize = .zero + @State private var lastOffset: CGSize = .zero + @GestureState private var magnifyBy = 1.0 + + @Environment(\.presentationMode) var presentationMode + + init(selectedImageUrl: String, imageUrls: [String]) { + self.imageUrls = imageUrls + let initialIndex = imageUrls.firstIndex(of: selectedImageUrl) ?? 0 + _currentPage = State(initialValue: initialIndex) + } + + var body: some View { + GeometryReader { geometry in + ZStack { + // images + SwiftUI.TabView(selection: $currentPage) { + ForEach(imageUrls.indices, id: \.self) { index in + ZoomableImageView( + imageUrl: imageUrls[index], + scale: $scale, + offset: $offset + ) + .tag(index) + } + } + .tabViewStyle(PageTabViewStyle(indexDisplayMode: .never)) + .gesture( + MagnificationGesture() + .updating($magnifyBy) { currentState, gestureState, _ in + gestureState = currentState + } + .onEnded { value in + scale = min(max(1, scale * value), 3) + } + ) + .simultaneousGesture( + DragGesture() + .onChanged { value in + if scale > 1 { + offset = CGSize( + width: lastOffset.width + value.translation.width, + height: lastOffset.height + value.translation.height + ) + } + } + .onEnded { _ in + if scale > 1 { + lastOffset = offset + } else { + offset = .zero + lastOffset = .zero + } + } + ) + + // back button + VStack { + HStack { + BackButtonWithText { + presentationMode.wrappedValue.dismiss() + }.padding(.leading, 16) + Spacer() + } + Spacer() + } + + //page indicator + VStack { + Spacer() + HStack(spacing: 8) { + ForEach(imageUrls.indices, id: \.self) { index in + Circle() + .fill(currentPage == index ? Color.primary : Color.primary.opacity(0.25)) + .frame(width: 8, height: 8) + } + } + } + .padding(.bottom, 16) + } + } + } +} + +struct ZoomableImageView: View { + let imageUrl: String + @Binding var scale: CGFloat + @Binding var offset: CGSize + @StateObject private var imageLoader = ImageLoader() + + var body: some View { + Group { + if let image = imageLoader.image { + Image(uiImage: image) + .resizable() + .aspectRatio(contentMode: .fit) + .scaleEffect(scale) + .offset(offset) + } else if imageLoader.isLoading { + ProgressView() + } else { + Text(L("error")) + } + } + .onAppear { + imageLoader.load(fromURLString: imageUrl) + } + } +} + +class ImageLoader: ObservableObject { + @Published var image: UIImage? + @Published var isLoading = false + private var cancellable: AnyCancellable? + + func load(fromURLString urlString: String) { + guard let url = URL(string: urlString) else { return } + + cancellable?.cancel() + self.image = nil + self.isLoading = true + + cancellable = URLSession.shared.dataTaskPublisher(for: url) + .map { UIImage(data: $0.data) } + .replaceError(with: nil) + .receive(on: DispatchQueue.main) + .sink { [weak self] in + self?.image = $0 + self?.isLoading = false + } + } +} diff --git a/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/Gallery/GalleryScreen.swift b/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/Gallery/GalleryScreen.swift index efbf59a1cd..6f56582144 100644 --- a/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/Gallery/GalleryScreen.swift +++ b/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/Gallery/GalleryScreen.swift @@ -3,47 +3,54 @@ import SwiftUI struct GalleryScreen: View { let urls: [String]? - let secondRowHeight = 100.0 + let secondRowHeight = 130.0 let shape = RoundedRectangle(cornerRadius: 8) @State var goToAllGalleryScreen = false var body: some View { if let urls = urls, !urls.isEmpty { VStack { - LoadImageView(url: urls.first) - .frame(maxWidth: UIScreen.main.bounds.width - 32, minHeight: 200, maxHeight: 200) - .clipShape(shape) + NavigationLink(destination: FullscreenImageViewer(selectedImageUrl: urls[0], imageUrls: urls)) { + LoadImageView(url: urls.first) + .frame(maxWidth: UIScreen.main.bounds.width - 32, minHeight: 200, maxHeight: 200) + .clipShape(shape) + } VerticalSpace(height: 16) - HStack(spacing: 16) { - if urls.count > 1 { - LoadImageView(url: urls[1]) - .frame(height: secondRowHeight) - .clipShape(shape) - .aspectRatio(1, contentMode: .fit) - - if urls.count > 2 { - NavigationLink(destination: AllPicsScreen(urls: urls)) { - ZStack { - LoadImageView(url: urls[2]) - .frame(height: secondRowHeight) - - if urls.count > 3 { - SwiftUI.Color.black.opacity(0.5) - .frame(height: secondRowHeight) - .clipShape(shape) + GeometryReader { geometry in + let secondColumnWidth = geometry.size.width / 2 - 8 + + HStack(spacing: 16) { + if urls.count > 1 { + NavigationLink(destination: FullscreenImageViewer(selectedImageUrl: urls[1], imageUrls: urls)) { + LoadImageView(url: urls[1]) + .frame(width: secondColumnWidth, height: secondRowHeight) + .clipShape(shape) + } + if (urls.count == 2) { Spacer() } + + if urls.count > 2 { + NavigationLink(destination: AllPicsScreen(urls: urls)) { + ZStack { + LoadImageView(url: urls[2]) - Text("+\(urls.count - 3)") - .font(.headline) - .foregroundColor(.white) + if urls.count > 3 { + SwiftUI.Color.black.opacity(0.5) + .frame(height: secondRowHeight) + .clipShape(shape) + + Text("+\(urls.count - 3)") + .font(.headline) + .foregroundColor(.white) + } } + .frame(width: secondColumnWidth, height: secondRowHeight) + .clipShape(shape) } - .clipShape(shape) - .aspectRatio(1, contentMode: .fit) } } - } + }.frame(width: geometry.size.width) } Spacer() }.padding(.horizontal, 16) diff --git a/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/Reviews/Components/ReviewView.swift b/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/Reviews/Components/ReviewView.swift index f24f91c551..f63efa6cfe 100644 --- a/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/Reviews/Components/ReviewView.swift +++ b/iphone/Maps/Tourism/Presentation/Home/Screens/Profile/PlaceDetails/Reviews/Components/ReviewView.swift @@ -36,7 +36,9 @@ struct ReviewView: View { ShowMoreView(url: url, remaining: review.picsUrls.count - 3) } } else { - ReviewPicView(url: url) + NavigationLink(destination: FullscreenImageViewer(selectedImageUrl: url, imageUrls: review.picsUrls)) { + ReviewPicView(url: url) + } } } } diff --git a/iphone/Maps/UI/Storyboard/Main.storyboard b/iphone/Maps/UI/Storyboard/Main.storyboard index 77d159d49f..dffc69cb09 100644 --- a/iphone/Maps/UI/Storyboard/Main.storyboard +++ b/iphone/Maps/UI/Storyboard/Main.storyboard @@ -240,22 +240,22 @@ - - + + - + @@ -270,22 +270,22 @@ - + - + - - + + @@ -312,7 +312,7 @@ - +