From ca9e03c6516427871e53a4e758d9e5b6da6ddaec Mon Sep 17 00:00:00 2001 From: Maciej Burda Date: Mon, 10 Apr 2023 14:34:03 +0100 Subject: [PATCH 1/3] REPLAY-1510 Initial solution using image recorder custom predicate --- .../SRHost/Fixtures/Images.storyboard | 39 +++++++++++++++---- .../Fixtures/ImagesViewControllers.swift | 19 +++++++++ .../NodeRecorders/UIImageViewRecorder.swift | 4 +- 3 files changed, 54 insertions(+), 8 deletions(-) diff --git a/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/Images.storyboard b/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/Images.storyboard index b2a0b4052a..ba6f1f4501 100644 --- a/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/Images.storyboard +++ b/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/Images.storyboard @@ -4,6 +4,7 @@ + @@ -25,15 +26,15 @@ - + - - + + - + @@ -183,7 +184,7 @@ - + @@ -198,7 +199,7 @@ - + @@ -210,18 +211,35 @@ - + + + @@ -229,15 +247,18 @@ + + + @@ -253,6 +274,7 @@ + @@ -268,6 +290,9 @@ + + + diff --git a/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/ImagesViewControllers.swift b/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/ImagesViewControllers.swift index 8ee2e0f98b..6297a325a7 100644 --- a/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/ImagesViewControllers.swift +++ b/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/ImagesViewControllers.swift @@ -7,10 +7,29 @@ import UIKit internal class ImagesViewController: UIViewController { + @IBOutlet weak var customButton: UIButton! @IBOutlet weak var customImageView: UIImageView! override func viewDidLoad() { super.viewDidLoad() + + customButton.setBackgroundImage(UIImage(color: .lightGray), for: .normal) + customImageView.image = UIImage(named: "dd_logo")?.withRenderingMode(.alwaysTemplate) } } + +fileprivate extension UIImage { + convenience init?(color: UIColor, size: CGSize = CGSize(width: 1.0, height: 1.0)) { + let rect = CGRect(origin: .zero, size: size) + UIGraphicsBeginImageContextWithOptions(rect.size, false, 1.0) + defer { UIGraphicsEndImageContext() } + + guard let context = UIGraphicsGetCurrentContext() else { return nil } + context.setFillColor(color.cgColor) + context.fill(rect) + + guard let cgImage = context.makeImage() else { return nil } + self.init(cgImage: cgImage) + } +} diff --git a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift index 1e728f96fc..e7af0ff8df 100644 --- a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift +++ b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift @@ -28,6 +28,9 @@ internal struct UIImageViewRecorder: NodeRecorder { }, shouldRecordImagePredicate: @escaping (UIImageView) -> Bool = { imageView in if #available(iOS 13.0, *), let image = imageView.image { + if let button = imageView.superview as? UIButton, button.buttonType == .custom { + return true + } return image.isSymbolImage || image.isBundled || image.isAlwaysTemplate } else { return false @@ -144,7 +147,6 @@ internal struct UIImageViewWireframesBuilder: NodeWireframesBuilder { tintColor: tintColor ) } - if let contentFrame = contentFrame { wireframes.append( builder.createImageWireframe( From 58e5d9d2de996d00bc16d244bf8bc790c8821de5 Mon Sep 17 00:00:00 2001 From: Maciej Burda Date: Mon, 10 Apr 2023 15:55:48 +0100 Subject: [PATCH 2/3] REPLAY-1510 Support for image background in system bars --- .../SRHost/Fixtures/Images.storyboard | 2 + .../Fixtures/ImagesViewControllers.swift | 9 ++- .../NodeRecorders/UIImageViewRecorder.swift | 55 +++++++++++++++---- 3 files changed, 53 insertions(+), 13 deletions(-) diff --git a/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/Images.storyboard b/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/Images.storyboard index ba6f1f4501..06a778a7b3 100644 --- a/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/Images.storyboard +++ b/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/Images.storyboard @@ -260,6 +260,8 @@ + + diff --git a/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/ImagesViewControllers.swift b/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/ImagesViewControllers.swift index 6297a325a7..0db8e71456 100644 --- a/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/ImagesViewControllers.swift +++ b/DatadogSessionReplay/SRSnapshotTests/SRHost/Fixtures/ImagesViewControllers.swift @@ -9,11 +9,18 @@ import UIKit internal class ImagesViewController: UIViewController { @IBOutlet weak var customButton: UIButton! @IBOutlet weak var customImageView: UIImageView! + @IBOutlet weak var tabBar: UITabBar! + @IBOutlet weak var navigationBar: UINavigationBar! override func viewDidLoad() { super.viewDidLoad() - customButton.setBackgroundImage(UIImage(color: .lightGray), for: .normal) + let color = UIColor(white: 0, alpha: 0.05) + customButton.setBackgroundImage(UIImage(color: color), for: .normal) + + tabBar.backgroundImage = UIImage(color: color) + tabBar.selectedItem = tabBar.items?.first + navigationBar.setBackgroundImage(UIImage(color: color), for: .default) customImageView.image = UIImage(named: "dd_logo")?.withRenderingMode(.alwaysTemplate) } diff --git a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift index e7af0ff8df..75a660c55a 100644 --- a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift +++ b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift @@ -11,30 +11,24 @@ internal struct UIImageViewRecorder: NodeRecorder { private let shouldRecordImagePredicate: (UIImageView) -> Bool /// An option for overriding default semantics from parent recorder. var semanticsOverride: (UIImageView, ViewAttributes) -> NodeSemantics? = { imageView, _ in - let className = "\(type(of: imageView))" - // This gets effective on iOS 15.0+ which is the earliest version that displays some elements in popover views. - // Here we explicitly ignore the "shadow" effect applied to popover. - let isSystemShadow = className == "_UICutoutShadowView" - return isSystemShadow ? IgnoredElement(subtreeStrategy: .ignore) : nil + return imageView.isSystemShadow ? IgnoredElement(subtreeStrategy: .ignore) : nil } internal init( tintColorProvider: @escaping (UIImageView) -> UIColor? = { imageView in if #available(iOS 13.0, *), let image = imageView.image { - return image.isSymbolImage || image.isAlwaysTemplate ? imageView.tintColor : nil + return image.isTinted ? imageView.tintColor : nil } else { return nil } }, shouldRecordImagePredicate: @escaping (UIImageView) -> Bool = { imageView in if #available(iOS 13.0, *), let image = imageView.image { - if let button = imageView.superview as? UIButton, button.buttonType == .custom { - return true - } - return image.isSymbolImage || image.isBundled || image.isAlwaysTemplate + return image.isContextual || imageView.isSystemControlBackground } else { return false } + } ) { self.tintColorProvider = tintColorProvider @@ -162,11 +156,48 @@ internal struct UIImageViewWireframesBuilder: NodeWireframesBuilder { } fileprivate extension UIImage { - var isBundled: Bool { + @available(iOS 13.0, *) + var isContextual: Bool { + return isSymbolImage || isBundled || isAlwaysTemplate + } + + @available(iOS 13.0, *) + var isTinted: Bool { + return isSymbolImage || isAlwaysTemplate + } + + private var isBundled: Bool { return description.contains("named(") } - var isAlwaysTemplate: Bool { + private var isAlwaysTemplate: Bool { return renderingMode == .alwaysTemplate } } + +fileprivate extension UIImageView { + + var isSystemControlBackground: Bool { + return isButtonBackground || isBarBackground + } + + var isSystemShadow: Bool { + let className = "\(type(of: self))" + // This gets effective on iOS 15.0+ which is the earliest version that displays some elements in popover views. + // Here we explicitly ignore the "shadow" effect applied to popover. + return className == "_UICutoutShadowView" + } + + var isButtonBackground: Bool { + if let button = superview as? UIButton, button.buttonType == .custom { + return button.backgroundImage(for: button.state) == image + } + return false + } + + var isBarBackground: Bool { + guard let superview = superview else { return false } + let superViewType = "\(type(of: superview))" + return superViewType == "_UIBarBackground" + } +} From 05fe95992e0df4817ea0f019fd12d43da437eafc Mon Sep 17 00:00:00 2001 From: Maciej Burda Date: Tue, 11 Apr 2023 11:40:08 +0100 Subject: [PATCH 3/3] REPLAY-1510 Fix linter issues --- .../NodeRecorders/UIImageViewRecorder.swift | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift index 75a660c55a..fdca087905 100644 --- a/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift +++ b/DatadogSessionReplay/Sources/Recorder/ViewTreeSnapshotProducer/ViewTreeSnapshot/NodeRecorders/UIImageViewRecorder.swift @@ -28,7 +28,6 @@ internal struct UIImageViewRecorder: NodeRecorder { } else { return false } - } ) { self.tintColorProvider = tintColorProvider @@ -176,7 +175,6 @@ fileprivate extension UIImage { } fileprivate extension UIImageView { - var isSystemControlBackground: Bool { return isButtonBackground || isBarBackground } @@ -196,7 +194,9 @@ fileprivate extension UIImageView { } var isBarBackground: Bool { - guard let superview = superview else { return false } + guard let superview = superview else { + return false + } let superViewType = "\(type(of: superview))" return superViewType == "_UIBarBackground" }