From dda8c0f37e83b5bc12d5393a446474b835c2d24b Mon Sep 17 00:00:00 2001 From: Marie Denis Date: Tue, 4 Jun 2024 10:23:48 +0200 Subject: [PATCH 1/4] [SR] Add background color support for TabBar --- .../ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift index 14a7903d80..a2cbfa5295 100644 --- a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift +++ b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift @@ -36,6 +36,10 @@ internal struct UITabBarRecorder: NodeRecorder { private func inferColor(of tabBar: UITabBar) -> CGColor { // TODO: RUMM-2791 Enhance appearance of `UITabBar` and `UINavigationBar` in SR + if let color = tabBar.backgroundColor { + return color.cgColor + } + if #available(iOS 13.0, *) { switch UITraitCollection.current.userInterfaceStyle { case .light: From 26e31526178097ed414a19ca037ac4c730aae076 Mon Sep 17 00:00:00 2001 From: Marie Denis Date: Mon, 10 Jun 2024 12:18:57 +0200 Subject: [PATCH 2/4] Add snapshot tests --- .../Sources/SRFixtures/Fixtures.swift | 10 + .../Resources/Storyboards/Tabbars.storyboard | 334 ++++++++++++++++++ .../ViewControllers/TabbarControllers.swift | 32 ++ .../SRHost/MenuViewController.swift | 10 + .../SRSnapshotTests/SRSnapshotTests.swift | 43 ++- .../NodeRecorders/UITabBarRecorder.swift | 2 - 6 files changed, 428 insertions(+), 3 deletions(-) create mode 100644 DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/Resources/Storyboards/Tabbars.storyboard create mode 100644 DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/ViewControllers/TabbarControllers.swift diff --git a/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/Fixtures.swift b/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/Fixtures.swift index decdaebf42..a2b7ead57d 100644 --- a/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/Fixtures.swift +++ b/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/Fixtures.swift @@ -42,6 +42,9 @@ public enum Fixture: CaseIterable { case navigationBarDefaultNonTranslucentBarTint case navigationBarDefaultTranslucentBackground case navigationBarDefaultNonTranslucentBackground + case tabbar + case embeddedTabbar + case embeddedTabbarUnselectedTintColor public func instantiateViewController() -> UIViewController { switch self { @@ -108,6 +111,12 @@ public enum Fixture: CaseIterable { return vc case .navigationBarDefaultNonTranslucentBackground: return UIStoryboard.navigationBars.instantiateViewController(withIdentifier: "Default_Non_Translucent_Background_Navbar") + case .tabbar: + return UIStoryboard.tabbars.instantiateViewController(withIdentifier: "Tabbars") + case .embeddedTabbar: + return UIStoryboard.tabbars.instantiateViewController(withIdentifier: "EmbeddedTabbar") + case .embeddedTabbarUnselectedTintColor: + return UIStoryboard.tabbars.instantiateViewController(withIdentifier: "EmbeddedTabbarUnselectedTintColor") } } } @@ -119,4 +128,5 @@ internal extension UIStoryboard { static var images: UIStoryboard { UIStoryboard(name: "Images", bundle: .module) } static var unsupportedViews: UIStoryboard { UIStoryboard(name: "UnsupportedViews", bundle: .module) } static var navigationBars: UIStoryboard { UIStoryboard(name: "NavigationBars", bundle: .module) } + static var tabbars: UIStoryboard { UIStoryboard(name: "Tabbars", bundle: .module) } } diff --git a/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/Resources/Storyboards/Tabbars.storyboard b/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/Resources/Storyboards/Tabbars.storyboard new file mode 100644 index 0000000000..3ae05572e0 --- /dev/null +++ b/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/Resources/Storyboards/Tabbars.storyboard @@ -0,0 +1,334 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/ViewControllers/TabbarControllers.swift b/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/ViewControllers/TabbarControllers.swift new file mode 100644 index 0000000000..caaade9074 --- /dev/null +++ b/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/ViewControllers/TabbarControllers.swift @@ -0,0 +1,32 @@ +/* + * Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. + * This product includes software developed at Datadog (https://www.datadoghq.com/). + * Copyright 2019-Present Datadog, Inc. + */ + +import UIKit + +public class TabbarViewControllers: UIViewController { + + @IBOutlet var tabbars: [UITabBar]! + + public override func viewDidLoad() { + for tabbar in tabbars { + // Select the first tabbar's item + // so we can see both a selected and unselected item + tabbar.selectedItem = tabbar.items?[0] + } + } +} + +public class EmbeddedTabbarController: UITabBarController { + public override func viewDidLoad() { + tabBar.unselectedItemTintColor = nil + } +} + +public class EmbeddedTabbarUnselectedTintColorController: UITabBarController { + public override func viewDidLoad() { + tabBar.unselectedItemTintColor = UIColor.green + } +} diff --git a/DatadogSessionReplay/SRSnapshotTests/SRHost/MenuViewController.swift b/DatadogSessionReplay/SRSnapshotTests/SRHost/MenuViewController.swift index ee9ab26870..9b2b947dff 100644 --- a/DatadogSessionReplay/SRSnapshotTests/SRHost/MenuViewController.swift +++ b/DatadogSessionReplay/SRSnapshotTests/SRHost/MenuViewController.swift @@ -64,8 +64,18 @@ internal extension Fixture { return "Embedded Navigation Bar Default + Translucent + Background color" case .navigationBarDefaultNonTranslucentBackground: return "Embedded Navigation Bar Default + Non Translucent + Background color" + case .tabbar: + return "Tab Bars" + case .embeddedTabbar: + return "Embedded Tab Bar" + case .embeddedTabbarUnselectedTintColor: + return "Embedded Tab Bar Unselected Tint Color" } } + + var slug: String { + self.menuItemTitle.lowercased().replacingOccurrences(of: " ", with: "") + } } internal class MenuViewController: UITableViewController { diff --git a/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/SRSnapshotTests.swift b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/SRSnapshotTests.swift index 4efb9f5b0a..f70310ffbe 100644 --- a/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/SRSnapshotTests.swift +++ b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/SRSnapshotTests.swift @@ -296,7 +296,7 @@ final class SRSnapshotTests: SnapshotTestCase { try forPrivacyModes([.allow, .mask]) { privacyMode in let image = try takeSnapshot(with: privacyMode) - let fileNamePrefix = fixture.menuItemTitle.lowercased().replacingOccurrences(of: " ", with: "") + let fileNamePrefix = fixture.slug DDAssertSnapshotTest( newImage: image, snapshotLocation: .folder(named: snapshotsFolderPath, fileNameSuffix: "-\(fileNamePrefix)-\(privacyMode)-privacy"), @@ -306,4 +306,45 @@ final class SRSnapshotTests: SnapshotTestCase { } } + func testTabBars() throws { + + // - Static Tab Bars + show(fixture: .tabbar) + + try forPrivacyModes([.allow, .mask]) { privacyMode in + let image = try takeSnapshot(with: privacyMode) + DDAssertSnapshotTest( + newImage: image, + snapshotLocation: .folder(named: snapshotsFolderPath, fileNameSuffix: "-\(privacyMode)-privacy"), + record: recordingMode + ) + } + + // - Embedded Tab Bar + show(fixture: .embeddedTabbar) + + try forPrivacyModes([.allow, .mask]) { privacyMode in + let image = try takeSnapshot(with: privacyMode) + let fileNamePrefix = Fixture.embeddedTabbar.slug + DDAssertSnapshotTest( + newImage: image, + snapshotLocation: .folder(named: snapshotsFolderPath, fileNameSuffix: "-\(fileNamePrefix)-\(privacyMode)-privacy"), + record: recordingMode + ) + } + + // - Embedded Tab Bar, with unselected item tint color + show(fixture: .embeddedTabbarUnselectedTintColor) + + try forPrivacyModes([.allow, .mask]) { privacyMode in + let image = try takeSnapshot(with: privacyMode) + let fileNamePrefix = Fixture.embeddedTabbarUnselectedTintColor.slug + DDAssertSnapshotTest( + newImage: image, + snapshotLocation: .folder(named: snapshotsFolderPath, fileNameSuffix: "-\(fileNamePrefix)-\(privacyMode)-privacy"), + record: recordingMode + ) + } + } + } diff --git a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift index a2cbfa5295..1d49a33ba6 100644 --- a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift +++ b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UITabBarRecorder.swift @@ -25,7 +25,6 @@ internal struct UITabBarRecorder: NodeRecorder { } private func inferOccupiedFrame(of tabBar: UITabBar, in context: ViewTreeRecordingContext) -> CGRect { - // TODO: RUMM-2791 Enhance appearance of `UITabBar` and `UINavigationBar` in SR var occupiedFrame = tabBar.frame for subview in tabBar.subviews { let subviewFrame = subview.convert(subview.bounds, to: context.coordinateSpace) @@ -35,7 +34,6 @@ internal struct UITabBarRecorder: NodeRecorder { } private func inferColor(of tabBar: UITabBar) -> CGColor { - // TODO: RUMM-2791 Enhance appearance of `UITabBar` and `UINavigationBar` in SR if let color = tabBar.backgroundColor { return color.cgColor } From 690aaa663aa70b954f9a56430fc246ae471a2e12 Mon Sep 17 00:00:00 2001 From: Marie Denis Date: Mon, 10 Jun 2024 12:23:05 +0200 Subject: [PATCH 3/4] Add new snapshots for Tab Bars --- .../_snapshots_/pointers/testTabBars()-allow-privacy.png.json | 1 + .../pointers/testTabBars()-embeddedtabbar-allow-privacy.png.json | 1 + .../pointers/testTabBars()-embeddedtabbar-mask-privacy.png.json | 1 + ...rs()-embeddedtabbarunselectedtintcolor-allow-privacy.png.json | 1 + ...ars()-embeddedtabbarunselectedtintcolor-mask-privacy.png.json | 1 + .../_snapshots_/pointers/testTabBars()-mask-privacy.png.json | 1 + 6 files changed, 6 insertions(+) create mode 100644 DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-allow-privacy.png.json create mode 100644 DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbar-allow-privacy.png.json create mode 100644 DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbar-mask-privacy.png.json create mode 100644 DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbarunselectedtintcolor-allow-privacy.png.json create mode 100644 DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbarunselectedtintcolor-mask-privacy.png.json create mode 100644 DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-mask-privacy.png.json diff --git a/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-allow-privacy.png.json b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-allow-privacy.png.json new file mode 100644 index 0000000000..67f186dd17 --- /dev/null +++ b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-allow-privacy.png.json @@ -0,0 +1 @@ +{"hash":"c8db9a89ac040c3733ec239bf92a2bfd6253dee4"} \ No newline at end of file diff --git a/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbar-allow-privacy.png.json b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbar-allow-privacy.png.json new file mode 100644 index 0000000000..ee875c9a23 --- /dev/null +++ b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbar-allow-privacy.png.json @@ -0,0 +1 @@ +{"hash":"72d50780fb4df2e058dad0e8953e439097fd3eaa"} \ No newline at end of file diff --git a/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbar-mask-privacy.png.json b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbar-mask-privacy.png.json new file mode 100644 index 0000000000..14834eb0ce --- /dev/null +++ b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbar-mask-privacy.png.json @@ -0,0 +1 @@ +{"hash":"a15b968196c3fd0a268bd3e82a37fef10b31009f"} \ No newline at end of file diff --git a/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbarunselectedtintcolor-allow-privacy.png.json b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbarunselectedtintcolor-allow-privacy.png.json new file mode 100644 index 0000000000..7705782f2e --- /dev/null +++ b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbarunselectedtintcolor-allow-privacy.png.json @@ -0,0 +1 @@ +{"hash":"a0cf5709cf5cb508728fd1d2ad74f6afd2153a27"} \ No newline at end of file diff --git a/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbarunselectedtintcolor-mask-privacy.png.json b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbarunselectedtintcolor-mask-privacy.png.json new file mode 100644 index 0000000000..50a1a552f5 --- /dev/null +++ b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-embeddedtabbarunselectedtintcolor-mask-privacy.png.json @@ -0,0 +1 @@ +{"hash":"dccf7efc3a8379f8cf0d1ee4d63e00a30206bc04"} \ No newline at end of file diff --git a/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-mask-privacy.png.json b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-mask-privacy.png.json new file mode 100644 index 0000000000..5028df5782 --- /dev/null +++ b/DatadogSessionReplay/SRSnapshotTests/SRSnapshotTests/_snapshots_/pointers/testTabBars()-mask-privacy.png.json @@ -0,0 +1 @@ +{"hash":"ace6b7b1f9db81e24d7d19d3fa3652e7b6f290dc"} \ No newline at end of file From 5778c80835b633f8ce7a49c3c29602ba79cf2a56 Mon Sep 17 00:00:00 2001 From: Marie Denis Date: Tue, 11 Jun 2024 11:53:10 +0200 Subject: [PATCH 4/4] RUM-4151 Address nit --- .../Sources/SRFixtures/ViewControllers/TabbarControllers.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/ViewControllers/TabbarControllers.swift b/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/ViewControllers/TabbarControllers.swift index caaade9074..103ac32b29 100644 --- a/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/ViewControllers/TabbarControllers.swift +++ b/DatadogSessionReplay/SRSnapshotTests/SRFixtures/Sources/SRFixtures/ViewControllers/TabbarControllers.swift @@ -14,7 +14,7 @@ public class TabbarViewControllers: UIViewController { for tabbar in tabbars { // Select the first tabbar's item // so we can see both a selected and unselected item - tabbar.selectedItem = tabbar.items?[0] + tabbar.selectedItem = tabbar.items?.first } } }